Explorar el Código

add skeleton routines for setting tuner gain

Dimitri Stolnikov hace 13 años
padre
commit
4ad93906a4
Se han modificado 3 ficheros con 70 adiciones y 28 borrados
  1. 6 2
      include/rtl-sdr.h
  2. 13 2
      src/main.c
  3. 51 24
      src/rtl-sdr.c

+ 6 - 2
include/rtl-sdr.h

@@ -47,9 +47,13 @@ int rtlsdr_set_center_freq(rtlsdr_dev_t *dev, uint32_t freq);
 
 int rtlsdr_get_center_freq(rtlsdr_dev_t *dev);
 
-int rtlsdr_set_freq_correction(rtlsdr_dev_t *dev, int32_t ppm);
+int rtlsdr_set_freq_correction(rtlsdr_dev_t *dev, int ppm);
 
-int32_t rtlsdr_get_freq_correction(rtlsdr_dev_t *dev);
+int rtlsdr_get_freq_correction(rtlsdr_dev_t *dev);
+
+int rtlsdr_set_tuner_gain(rtlsdr_dev_t *dev, int gain);
+
+int rtlsdr_get_tuner_gain(rtlsdr_dev_t *dev);
 
 /* this will select the baseband filters according to the requested sample rate */
 int rtlsdr_set_sample_rate(rtlsdr_dev_t *dev, uint32_t rate);

+ 13 - 2
src/main.c

@@ -35,6 +35,8 @@ void usage(void)
 	printf("rtl-sdr, an I/Q recorder for RTL2832 based DVB-T receivers\n\n"
 		"Usage:\t -f frequency to tune to [Hz]\n"
 		"\t[-s samplerate (default: 2048000 Hz)]\n"
+	        "\t[-d device index (default: 0)]\n"
+	        "\t[-g tuner gain (default: 0 dB)]\n"
 		"\toutput filename\n");
 	exit(1);
 }
@@ -54,9 +56,9 @@ int main(int argc, char **argv)
 	uint32_t n_read;
 	FILE *file;
 	rtlsdr_dev_t *dev = NULL;
-	uint32_t dev_index = 0;
+	uint32_t dev_index = 0, gain = 0;
 
-	while ((opt = getopt(argc, argv, "d:f:s:")) != -1) {
+	while ((opt = getopt(argc, argv, "d:f:g:s:")) != -1) {
 		switch (opt) {
 		case 'd':
 			dev_index = atoi(optarg);
@@ -64,6 +66,9 @@ int main(int argc, char **argv)
 		case 'f':
 			frequency = atoi(optarg);
 			break;
+		case 'g':
+			gain = atoi(optarg);
+			break;
 		case 's':
 			samp_rate = atoi(optarg);
 			break;
@@ -116,6 +121,12 @@ int main(int argc, char **argv)
 	else
 		fprintf(stderr, "Tuned to %i Hz.\n", frequency);
 
+	r = rtlsdr_set_tuner_gain(dev, gain);
+	if (r < 0)
+		fprintf(stderr, "WARNING: Failed to set tuner gain.\n");
+	else
+		fprintf(stderr, "Tuner gain set to %i dB.\n", gain);
+
 	file = fopen(filename, "wb");
 
 	if (!file) {

+ 51 - 24
src/rtl-sdr.c

@@ -29,32 +29,41 @@
 #include "tuner_e4000.h"
 #include "tuner_fc0013.h"
 
+typedef struct rtlsdr_tuner {
+	int(*init)(void *);
+	int(*exit)(void *);
+	int(*tune)(void *, int freq /* Hz */);
+	int(*set_bw)(void *, int bw /* Hz */);
+	int(*set_gain)(void *, int gain /* dB */);
+	int freq; /* Hz */
+	int corr; /* ppm */
+	int gain; /* dB */
+} rtlsdr_tuner_t;
+
 /* generic tuner interface functions, shall be moved to the tuner implementations */
 int e4k_init(void *dev) { return e4000_Initialize(dev); }
 int e4k_exit(void *dev) { return 0; }
 int e4k_tune(void *dev, int freq) { return e4000_SetRfFreqHz(dev, freq); }
 int e4k_set_bw(void *dev, int bw) { return e4000_SetBandwidthHz(dev, 8000000); }
+int e4k_set_gain(void *dev, int gain) { return 0; }
 
 int fc0012_init(void *dev) { return 0; }
 int fc0012_exit(void *dev) { return 0; }
 int fc0012_tune(void *dev, int freq) { return 0; }
 int fc0012_set_bw(void *dev, int bw) { return 0; }
+int fc0012_set_gain(void *dev, int gain) { return 0; }
 
 int fc0013_init(void *dev) { return FC0013_Open(dev); }
 int fc0013_exit(void *dev) { return 0; }
 int fc0013_tune(void *dev, int freq) {
-	/* read bandwidth mode to reapply it */
 	unsigned int bw = 6;
-	//fc0013_GetBandwidthMode(dev, &bw); // FIXME: missing
 	return FC0013_SetFrequency(dev, freq/1000, bw & 0xff);
 }
-
 int fc0013_set_bw(void *dev, int bw) {
-	/* read frequency to reapply it */
-	unsigned long freq = 0;
-	//fc0013_GetRfFreqHz(dev, &freq); // FIXME: missing
-	return FC0013_SetFrequency(dev, freq/1000, 8);
+	unsigned long freq = ((rtlsdr_tuner_t *)dev)->freq;
+	return FC0013_SetFrequency(dev, freq/1000, bw/1000000);
 }
+int fc0013_set_gain(void *dev, int gain) { return 0; }
 
 enum rtlsdr_tuners {
 	RTLSDR_TUNER_E4000,
@@ -62,20 +71,12 @@ enum rtlsdr_tuners {
 	RTLSDR_TUNER_FC0013,
 };
 
-typedef struct rtlsdr_tuner {
-	enum rtlsdr_tuners tuner;
-	int(*init)(void *);
-	int(*exit)(void *);
-	int(*tune)(void *, int freq /* Hz */);
-	int(*set_bw)(void *, int bw /* Hz */);
-	int freq; /* Hz */
-	int corr; /* ppm */
-} rtlsdr_tuner_t;
 
-rtlsdr_tuner_t tuners[] = {
-	{ RTLSDR_TUNER_E4000, e4k_init, e4k_exit, e4k_tune, e4k_set_bw, 0, 0 },
-	{ RTLSDR_TUNER_FC0012, fc0012_init, fc0012_exit, fc0012_tune, fc0012_set_bw, 0, 0 },
-	{ RTLSDR_TUNER_FC0013, fc0013_init, fc0013_exit, fc0013_tune, fc0013_set_bw, 0, 0 },
+
+static rtlsdr_tuner_t tuners[] = {
+	{ e4k_init, e4k_exit, e4k_tune, e4k_set_bw, e4k_set_gain, 0, 0, 0 },
+	{ fc0012_init, fc0012_exit, fc0012_tune, fc0012_set_bw, fc0012_set_gain, 0, 0, 0 },
+	{ fc0013_init, fc0013_exit, fc0013_tune, fc0013_set_bw, fc0013_set_gain, 0, 0, 0 },
 };
 
 typedef struct rtlsdr_device {
@@ -84,7 +85,7 @@ typedef struct rtlsdr_device {
 	const char *name;
 } rtlsdr_device_t;
 
-rtlsdr_device_t devices[] = {
+static rtlsdr_device_t devices[] = {
 	{ 0x0bda, 0x2832, "Generic RTL2832U (e.g. hama nano)" },
 	{ 0x0bda, 0x2838, "ezcap USB 2.0 DVB-T/DAB/FM dongle" },
 	{ 0x0ccd, 0x00b3, "Terratec NOXON DAB/DAB+ USB dongle (rev 1)" },
@@ -370,13 +371,16 @@ int rtlsdr_set_center_freq(rtlsdr_dev_t *dev, uint32_t freq)
 
 	rtlsdr_set_i2c_repeater(dev, 1);
 
-	dev->tuner->freq = freq;
+
 	double f = (double) freq;
 	f *= 1.0 + dev->tuner->corr / 1e6;
 	r = dev->tuner->tune((void *)dev, (int) f);
 
 	rtlsdr_set_i2c_repeater(dev, 0);
 
+	if (!r)
+		dev->tuner->freq = freq;
+
 	return r;
 }
 
@@ -388,7 +392,7 @@ int rtlsdr_get_center_freq(rtlsdr_dev_t *dev)
 	return dev->tuner->freq;
 }
 
-int rtlsdr_set_freq_correction(rtlsdr_dev_t *dev, int32_t ppm)
+int rtlsdr_set_freq_correction(rtlsdr_dev_t *dev, int ppm)
 {
 	int r;
 
@@ -406,7 +410,7 @@ int rtlsdr_set_freq_correction(rtlsdr_dev_t *dev, int32_t ppm)
 	return r;
 }
 
-int32_t rtlsdr_get_freq_correction(rtlsdr_dev_t *dev)
+int rtlsdr_get_freq_correction(rtlsdr_dev_t *dev)
 {
 	if (!dev || !dev->tuner)
 		return -1;
@@ -414,6 +418,29 @@ int32_t rtlsdr_get_freq_correction(rtlsdr_dev_t *dev)
 	return dev->tuner->corr;
 }
 
+int rtlsdr_set_tuner_gain(rtlsdr_dev_t *dev, int gain)
+{
+	int r;
+
+	if (!dev || !dev->tuner)
+		return -1;
+
+	r = dev->tuner->set_gain((void *)dev, gain);
+
+	if (!r)
+		dev->tuner->gain = gain;
+
+	return r;
+}
+
+int rtlsdr_get_tuner_gain(rtlsdr_dev_t *dev)
+{
+	if (!dev || !dev->tuner)
+		return -1;
+
+	return dev->tuner->gain;
+}
+
 int rtlsdr_set_sample_rate(rtlsdr_dev_t *dev, uint32_t samp_rate)
 {
 	uint16_t tmp;