FiFi-SDR backend improvements

Patch from Rolf Meeser <rolfm_9dq@yahoo.de>

git-svn-id: https://hamlib.svn.sourceforge.net/svnroot/hamlib/trunk@3038 7ae35d74-ebe9-4afe-98af-79ac388436b8
Hamlib-1.2.13
Nate Bargmann, N0NB 2011-02-04 23:20:27 +00:00
rodzic c85bdb02f6
commit 8ecf823704
1 zmienionych plików z 109 dodań i 115 usunięć

Wyświetl plik

@ -49,22 +49,10 @@
/* Selected request codes of the original AVR USB Si570 firmware */ /* Selected request codes of the original AVR USB Si570 firmware */
#define REQUEST_READ_VERSION 0x00 #define REQUEST_SET_FREQ_BY_VALUE (0x32)
#define REQUEST_SET_DDRB 0x01 #define REQUEST_SET_XTALL_FREQ (0x33)
#define REQUEST_SET_PORTB 0x04 #define REQUEST_READ_MULTIPLY_LO (0x39)
#define REQUEST_READ_EEPROM 0x11 #define REQUEST_READ_FREQUENCY (0x3A)
#define REQUEST_FILTERS 0x17
#define REQUEST_SET_FREQ 0x30
#define REQUEST_SET_FREQ_BY_VALUE 0x32
#define REQUEST_SET_XTALL_FREQ 0x33
#define REQUEST_SET_STARTUP_FREQ 0x34
#define REQUEST_READ_MULTIPLY_LO 0x39
#define REQUEST_READ_FREQUENCY 0x3A
#define REQUEST_READ_SMOOTH_TUNE_PPM 0x3B
#define REQUEST_READ_STARTUP 0x3C
#define REQUEST_READ_XTALL 0x3D
#define REQUEST_READ_REGISTERS 0x3F
#define REQUEST_READ_KEYS 0x51
/* FiFi-SDR specific requests */ /* FiFi-SDR specific requests */
#define REQUEST_FIFISDR_READ (0xAB) #define REQUEST_FIFISDR_READ (0xAB)
@ -85,13 +73,12 @@
static int fifisdr_init(RIG *rig); static int fifisdr_init(RIG *rig);
static int fifisdr_cleanup(RIG *rig); static int fifisdr_cleanup(RIG *rig);
static int fifisdr_open(RIG *rig); static int fifisdr_open(RIG *rig);
static int fifisdr_set_freq_by_value(RIG *rig, vfo_t vfo, freq_t freq); static int fifisdr_set_freq(RIG *rig, vfo_t vfo, freq_t freq);
static int fifisdr_get_freq_by_value(RIG *rig, vfo_t vfo, freq_t *freq); static int fifisdr_get_freq(RIG *rig, vfo_t vfo, freq_t *freq);
static const char *fifisdr_get_info(RIG *rig); static const char *fifisdr_get_info(RIG *rig);
static int fifisdr_set_conf(RIG *rig, token_t token, const char *val);
static int fifisdr_get_conf(RIG *rig, token_t token, char *val);
static int fifisdr_set_mode(RIG *rig, vfo_t vfo, rmode_t mode, pbwidth_t width); static int fifisdr_set_mode(RIG *rig, vfo_t vfo, rmode_t mode, pbwidth_t width);
static int fifisdr_get_mode(RIG *rig, vfo_t vfo, rmode_t *mode, pbwidth_t * width); static int fifisdr_get_mode(RIG *rig, vfo_t vfo, rmode_t *mode, pbwidth_t * width);
static int fifisdr_set_level(RIG *rig, vfo_t vfo, setting_t level, value_t val);
static int fifisdr_get_level(RIG *rig, vfo_t vfo, setting_t level, value_t *val); static int fifisdr_get_level(RIG *rig, vfo_t vfo, setting_t level, value_t *val);
static int fifisdr_get_ext_level(RIG *rig, vfo_t vfo, token_t token, value_t *val); static int fifisdr_get_ext_level(RIG *rig, vfo_t vfo, token_t token, value_t *val);
@ -99,17 +86,7 @@ static int fifisdr_get_ext_level(RIG *rig, vfo_t vfo, token_t token, value_t *va
/* Private tokens. */ /* Private tokens. */
#define TOK_PARAM_MULTIPLIER TOKEN_BACKEND(1) /* Factor between VCO and RX frequency */ #define TOK_LVL_FMCENTER TOKEN_BACKEND(1) /* FM center frequency deviation */
#define TOK_LVL_FMCENTER TOKEN_BACKEND(20) /* FM center frequency deviation */
static const struct confparams fifisdr_cfg_params[] = {
{ TOK_PARAM_MULTIPLIER, "multiplier", "Freq Multiplier", "Frequency multiplier",
"4", RIG_CONF_NUMERIC, { .n = { 0.000001, 100 } }
},
{ RIG_CONF_END, NULL, }
};
/* Extra levels definitions */ /* Extra levels definitions */
@ -122,22 +99,14 @@ static const struct confparams fifisdr_ext_levels[] = {
/* /** Private instance data. */
* Common data struct struct fifisdr_priv_instance_data {
*/ double multiplier;
struct fifisdr_priv_data {
double osc_freq; /* MHz */
double multiplier; /* default to 4 for QSD/QSE */
}; };
/** FiFi-SDR receiver description. */
/* FiFi-SDR receiver description.
*
* Based on AVR USB description, but uses different vendor and product strings.
*/
const struct rig_caps fifisdr_caps = { const struct rig_caps fifisdr_caps = {
.rig_model = RIG_MODEL_FIFISDR, .rig_model = RIG_MODEL_FIFISDR,
.model_name = "FiFi-SDR", .model_name = "FiFi-SDR",
@ -158,8 +127,8 @@ const struct rig_caps fifisdr_caps = {
.has_get_func = RIG_FUNC_NONE, .has_get_func = RIG_FUNC_NONE,
.has_set_func = RIG_FUNC_NONE, .has_set_func = RIG_FUNC_NONE,
.has_get_level = RIG_LEVEL_STRENGTH, .has_get_level = RIG_LEVEL_PREAMP | RIG_LEVEL_STRENGTH,
.has_set_level = RIG_LEVEL_SET(RIG_LEVEL_NONE), .has_set_level = RIG_LEVEL_SET(RIG_LEVEL_PREAMP),
.has_get_parm = RIG_PARM_NONE, .has_get_parm = RIG_PARM_NONE,
.has_set_parm = RIG_PARM_SET(RIG_PARM_NONE), .has_set_parm = RIG_PARM_SET(RIG_PARM_NONE),
@ -169,7 +138,7 @@ const struct rig_caps fifisdr_caps = {
.extparms = NULL, .extparms = NULL,
.extlevels = fifisdr_ext_levels, .extlevels = fifisdr_ext_levels,
.preamp = { RIG_DBLST_END }, .preamp = { 6, RIG_DBLST_END },
.attenuator = { RIG_DBLST_END }, .attenuator = { RIG_DBLST_END },
.max_rit = Hz(0), .max_rit = Hz(0),
.max_xit = Hz(0), .max_xit = Hz(0),
@ -215,31 +184,29 @@ const struct rig_caps fifisdr_caps = {
RIG_FLT_END, RIG_FLT_END,
}, },
.cfgparams = fifisdr_cfg_params, .cfgparams = NULL,
.rig_init = fifisdr_init, .rig_init = fifisdr_init,
.rig_cleanup = fifisdr_cleanup, .rig_cleanup = fifisdr_cleanup,
.rig_open = fifisdr_open, .rig_open = fifisdr_open,
.rig_close = NULL, .rig_close = NULL,
.set_freq = fifisdr_set_freq_by_value, .set_freq = fifisdr_set_freq,
.get_freq = fifisdr_get_freq_by_value, .get_freq = fifisdr_get_freq,
.set_mode = fifisdr_set_mode, .set_mode = fifisdr_set_mode,
.get_mode = fifisdr_get_mode, .get_mode = fifisdr_get_mode,
.set_level = fifisdr_set_level,
.get_level = fifisdr_get_level, .get_level = fifisdr_get_level,
.get_ext_level = fifisdr_get_ext_level, .get_ext_level = fifisdr_get_ext_level,
.set_conf = fifisdr_set_conf,
.get_conf = fifisdr_get_conf,
.get_info = fifisdr_get_info, .get_info = fifisdr_get_info,
}; };
/* Conversion to/from little endian */ /** Convert from host endianness to FiFi-SDR little endian. */
static uint32_t fifisdr_tole32 (uint32_t x) static uint32_t fifisdr_tole32 (uint32_t x)
{ {
return return
@ -249,6 +216,9 @@ static uint32_t fifisdr_tole32 (uint32_t x)
((((x) / 16777216ul) % 256ul) << 24)); ((((x) / 16777216ul) % 256ul) << 24));
} }
/** Convert FiFi-SDR little endian to host endianness. */
static uint32_t fifisdr_fromle32 (uint32_t x) static uint32_t fifisdr_fromle32 (uint32_t x)
{ {
return return
@ -259,7 +229,8 @@ static uint32_t fifisdr_fromle32 (uint32_t x)
} }
/** fifisdr_usb_write. */
/** USB OUT transfer via vendor device command. */
static int fifisdr_usb_write (RIG *rig, static int fifisdr_usb_write (RIG *rig,
int request, int value, int index, int request, int value, int index,
char *bytes, int size) char *bytes, int size)
@ -284,7 +255,7 @@ static int fifisdr_usb_write (RIG *rig,
/** fifisdr_usb_read. */ /** USB IN transfer via vendor device command. */
static int fifisdr_usb_read (RIG *rig, static int fifisdr_usb_read (RIG *rig,
int request, int value, int index, int request, int value, int index,
char *bytes, int size) char *bytes, int size)
@ -308,12 +279,13 @@ static int fifisdr_usb_read (RIG *rig,
} }
int fifisdr_init(RIG *rig)
int fifisdr_init (RIG *rig)
{ {
hamlib_port_t *rp = &rig->state.rigport; hamlib_port_t *rp = &rig->state.rigport;
struct fifisdr_priv_data *priv; struct fifisdr_priv_instance_data *priv;
priv = (struct fifisdr_priv_data*)calloc(sizeof(struct fifisdr_priv_data), 1); priv = (struct fifisdr_priv_instance_data*)calloc(sizeof(struct fifisdr_priv_instance_data), 1);
if (!priv) { if (!priv) {
/* whoops! memory shortage! */ /* whoops! memory shortage! */
return -RIG_ENOMEM; return -RIG_ENOMEM;
@ -327,7 +299,7 @@ int fifisdr_init(RIG *rig)
/* no usb_set_configuration() and usb_claim_interface() */ /* no usb_set_configuration() and usb_claim_interface() */
rp->parm.usb.iface = -1; rp->parm.usb.iface = -1;
rp->parm.usb.conf = 1; rp->parm.usb.conf = 1;
rp->parm.usb.alt = 0; /* necessary ? */ rp->parm.usb.alt = 0;
rp->parm.usb.vendor_name = FIFISDR_VENDOR_NAME; rp->parm.usb.vendor_name = FIFISDR_VENDOR_NAME;
rp->parm.usb.product = FIFISDR_PRODUCT_NAME; rp->parm.usb.product = FIFISDR_PRODUCT_NAME;
@ -338,7 +310,8 @@ int fifisdr_init(RIG *rig)
} }
int fifisdr_cleanup(RIG *rig)
int fifisdr_cleanup (RIG *rig)
{ {
if (!rig) if (!rig)
return -RIG_EINVAL; return -RIG_EINVAL;
@ -350,52 +323,31 @@ int fifisdr_cleanup(RIG *rig)
return RIG_OK; return RIG_OK;
} }
int fifisdr_set_conf(RIG *rig, token_t token, const char *val)
int fifisdr_open (RIG *rig)
{ {
struct fifisdr_priv_data *priv; int ret;
double multiplier; uint32_t multiply;
struct fifisdr_priv_instance_data *priv;
priv = (struct fifisdr_priv_data*)rig->state.priv;
switch(token) { priv = (struct fifisdr_priv_instance_data*)rig->state.priv;
case TOK_PARAM_MULTIPLIER:
if (sscanf(val, "%lf", &multiplier) != 1) /* The VCO is a multiple of the RX frequency. Typically 4 */
return -RIG_EINVAL; ret = fifisdr_usb_read(rig, REQUEST_FIFISDR_READ, 0,
if (multiplier == 0.) 11, /* Read virtual VCO factor */
return -RIG_EINVAL; (char *)&multiply, sizeof(multiply));
priv->multiplier = multiplier; if (ret == RIG_OK) {
break; priv->multiplier = fifisdr_fromle32(multiply);
default:
return -RIG_EINVAL;
} }
return RIG_OK;
}
int fifisdr_get_conf(RIG *rig, token_t token, char *val)
{
struct fifisdr_priv_data *priv = (struct fifisdr_priv_data*)rig->state.priv;
switch(token) {
case TOK_PARAM_MULTIPLIER:
sprintf(val, "%f", priv->multiplier);
break;
default:
return -RIG_EINVAL;
}
return RIG_OK;
}
int fifisdr_open(RIG *rig)
{
rig_debug(RIG_DEBUG_TRACE,"%s called\n", __func__);
return RIG_OK; return RIG_OK;
} }
const char * fifisdr_get_info(RIG *rig)
const char * fifisdr_get_info (RIG *rig)
{ {
static char buf[64]; static char buf[64];
int ret; int ret;
@ -416,18 +368,20 @@ const char * fifisdr_get_info(RIG *rig)
int fifisdr_set_freq_by_value(RIG *rig, vfo_t vfo, freq_t freq) int fifisdr_set_freq (RIG *rig, vfo_t vfo, freq_t freq)
{ {
struct fifisdr_priv_data *priv = (struct fifisdr_priv_data *)rig->state.priv; struct fifisdr_priv_instance_data *priv = (struct fifisdr_priv_instance_data *)rig->state.priv;
int ret; int ret;
double f; double mhz;
uint32_t lfreq; uint32_t freq1121;
f = (freq * priv->multiplier)/1e6;
lfreq = fifisdr_tole32(round(f * 2097152.0)); /* Need frequency in 11.21 format */
mhz = (freq * priv->multiplier) / 1e6;
freq1121 = fifisdr_tole32(round(mhz * 2097152.0));
ret = fifisdr_usb_write(rig, REQUEST_SET_FREQ_BY_VALUE, 0, 0, ret = fifisdr_usb_write(rig, REQUEST_SET_FREQ_BY_VALUE, 0, 0,
(char *)&lfreq, sizeof(lfreq)); (char *)&freq1121, sizeof(freq1121));
if (ret != RIG_OK) { if (ret != RIG_OK) {
return -RIG_EIO; return -RIG_EIO;
} }
@ -437,17 +391,18 @@ int fifisdr_set_freq_by_value(RIG *rig, vfo_t vfo, freq_t freq)
int fifisdr_get_freq_by_value(RIG *rig, vfo_t vfo, freq_t *freq) int fifisdr_get_freq (RIG *rig, vfo_t vfo, freq_t *freq)
{ {
struct fifisdr_priv_data *priv = (struct fifisdr_priv_data *)rig->state.priv; struct fifisdr_priv_instance_data *priv = (struct fifisdr_priv_instance_data *)rig->state.priv;
int ret; int ret;
uint32_t iFreq; uint32_t freq1121;
ret = fifisdr_usb_read(rig, REQUEST_READ_FREQUENCY, 0, 0, (char *)&iFreq, sizeof(iFreq));
ret = fifisdr_usb_read(rig, REQUEST_READ_FREQUENCY, 0, 0, (char *)&freq1121, sizeof(freq1121));
if (ret == RIG_OK) { if (ret == RIG_OK) {
iFreq = fifisdr_fromle32(iFreq); freq1121 = fifisdr_fromle32(freq1121);
*freq = MHz(((double)iFreq / (1ul << 21)) / priv->multiplier); *freq = MHz(((double)freq1121 / (1ul << 21)) / priv->multiplier);
} }
return ret; return ret;
@ -455,8 +410,6 @@ int fifisdr_get_freq_by_value(RIG *rig, vfo_t vfo, freq_t *freq)
static int fifisdr_set_mode(RIG *rig, vfo_t vfo, rmode_t mode, pbwidth_t width) static int fifisdr_set_mode(RIG *rig, vfo_t vfo, rmode_t mode, pbwidth_t width)
{ {
int ret; int ret;
@ -550,18 +503,59 @@ static int fifisdr_get_mode(RIG *rig, vfo_t vfo, rmode_t *mode, pbwidth_t * widt
static int fifisdr_set_level (RIG * rig, vfo_t vfo, setting_t level, value_t val)
{
int ret = RIG_OK;
uint8_t fifi_preamp;
switch (level) {
/* Preamplifier (ADC 0/+6dB switch) */
case RIG_LEVEL_PREAMP:
/* Value can be 0 (0 dB) or 1 (+6 dB) */
fifi_preamp = 0;
if (val.i == 6) {
fifi_preamp = 1;
}
ret = fifisdr_usb_write(rig, REQUEST_FIFISDR_WRITE, 0,
19, /* Preamp */
(char *)&fifi_preamp, sizeof(fifi_preamp));
break;
/* Unsupported option */
default:
ret = -RIG_ENIMPL;
}
return ret;
}
static int fifisdr_get_level(RIG *rig, vfo_t vfo, setting_t level, value_t *val) static int fifisdr_get_level(RIG *rig, vfo_t vfo, setting_t level, value_t *val)
{ {
int ret = RIG_OK; int ret = RIG_OK;
uint32_t fifi_meter = 0; uint32_t fifi_meter = 0;
uint8_t fifi_preamp = 0;
switch (level) { switch (level) {
/* Preamplifier (ADC 0/+6dB switch) */
case RIG_LEVEL_PREAMP:
ret = fifisdr_usb_read(rig, REQUEST_FIFISDR_READ, 0,
19, /* Preamp */
(char *)&fifi_preamp, sizeof(fifi_preamp));
if (ret == RIG_OK) {
/* Value can be 0 (0 dB) or 1 (+6 dB) */
val->i = 0;
if (fifi_preamp != 0) {
val->i = 6;
}
}
break;
/* Signal strength */ /* Signal strength */
case RIG_LEVEL_STRENGTH: case RIG_LEVEL_STRENGTH:
/* Read current signal strength */
ret = fifisdr_usb_read(rig, REQUEST_FIFISDR_READ, 0, ret = fifisdr_usb_read(rig, REQUEST_FIFISDR_READ, 0,
17, /* S-Meter */ 17, /* S-Meter */
(char *)&fifi_meter, sizeof(fifi_meter)); (char *)&fifi_meter, sizeof(fifi_meter));
@ -579,6 +573,7 @@ static int fifisdr_get_level(RIG *rig, vfo_t vfo, setting_t level, value_t *val)
} }
static int fifisdr_get_ext_level(RIG *rig, vfo_t vfo, token_t token, value_t *val) static int fifisdr_get_ext_level(RIG *rig, vfo_t vfo, token_t token, value_t *val)
{ {
int ret = RIG_OK; int ret = RIG_OK;
@ -586,9 +581,8 @@ static int fifisdr_get_ext_level(RIG *rig, vfo_t vfo, token_t token, value_t *va
switch (token) { switch (token) {
/* Signal strength */ /* FM center frequency deviation */
case TOK_LVL_FMCENTER: case TOK_LVL_FMCENTER:
/* Read current signal strength */
ret = fifisdr_usb_read(rig, REQUEST_FIFISDR_READ, 0, ret = fifisdr_usb_read(rig, REQUEST_FIFISDR_READ, 0,
18, /* FM center frequency */ 18, /* FM center frequency */
(char *)&u32, sizeof(u32)); (char *)&u32, sizeof(u32));