2004-09-12 21:30:21 +00:00
|
|
|
/*
|
|
|
|
* Hamlib Watkins-Johnson backend - main file
|
|
|
|
* Copyright (c) 2004 by Stephane Fillod
|
|
|
|
*
|
2006-10-07 19:00:05 +00:00
|
|
|
* $Id: wj.c,v 1.3 2006-10-07 18:55:19 csete Exp $
|
2004-09-12 21:30:21 +00:00
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Library General Public License as
|
|
|
|
* published by the Free Software Foundation; either version 2 of
|
|
|
|
* the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU Library General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Library General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h> /* String function definitions */
|
|
|
|
#include <unistd.h> /* UNIX standard function definitions */
|
|
|
|
#include <math.h>
|
|
|
|
|
|
|
|
#include "hamlib/rig.h"
|
|
|
|
#include "serial.h"
|
|
|
|
#include "misc.h"
|
|
|
|
#include "cal.h"
|
|
|
|
#include "register.h"
|
|
|
|
#include "token.h"
|
|
|
|
|
|
|
|
#include "wj.h"
|
|
|
|
|
|
|
|
|
|
|
|
#define CMDSZ 10
|
|
|
|
|
|
|
|
const struct confparams wj_cfg_params[] = {
|
|
|
|
{ TOK_RIGID, "receiver_id", "receiver ID", "receiver ID",
|
|
|
|
"0", RIG_CONF_NUMERIC, { .n = { 0, 15, 1 } }
|
|
|
|
},
|
|
|
|
{ RIG_CONF_END, NULL, }
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* modes
|
|
|
|
*/
|
|
|
|
#define MD_AM 0
|
|
|
|
#define MD_FM 1
|
|
|
|
#define MD_CW 2
|
|
|
|
#define MD_VCW 3 /* BFO variable */
|
|
|
|
#define MD_ISB 4
|
|
|
|
#define MD_LSB 5
|
|
|
|
#define MD_USB 6
|
|
|
|
#define MD_AMNL 7
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* wj_transaction
|
|
|
|
*
|
|
|
|
* I'm not sure how monitor protocol works, whether you have
|
|
|
|
* to send the full frame, or just the modal byte. --SF
|
|
|
|
*
|
|
|
|
* TODO: decode the whole reply, and maybe do some caching
|
|
|
|
*/
|
|
|
|
static int wj_transaction(RIG *rig, int monitor)
|
|
|
|
{
|
|
|
|
struct wj_priv_data *priv = (struct wj_priv_data*)rig->state.priv;
|
|
|
|
|
|
|
|
unsigned char buf[CMDSZ] = { 0x8, 0, 0, 0, 0, 0, 0, 0, 0, 0 };
|
|
|
|
unsigned char rxbuf[CMDSZ];
|
|
|
|
unsigned char freqbuf[4];
|
|
|
|
unsigned wj_agc, wj_mode, wj_width, wj_bfo, wj_rfgain;
|
|
|
|
int retval;
|
|
|
|
|
|
|
|
if (monitor)
|
|
|
|
buf[1] |= 0x40; /* Monitor+AGC dump */
|
|
|
|
else
|
|
|
|
buf[0] |= 0x40; /* Command */
|
|
|
|
|
|
|
|
buf[0] |= priv->receiver_id & 0x0f;
|
|
|
|
|
|
|
|
/* tuned frequency */
|
|
|
|
to_bcd_be(freqbuf, priv->freq/10, 7);
|
|
|
|
buf[1] |= freqbuf[0] & 0x3f;
|
|
|
|
buf[2] |= freqbuf[1]>>1;
|
|
|
|
buf[3] |= ((freqbuf[1]&0x1)<<6) | (freqbuf[2]>>2);
|
|
|
|
buf[4] |= ((freqbuf[2]&0x2)<<5) | (freqbuf[3]>>3);
|
|
|
|
|
|
|
|
/* gain mode */
|
|
|
|
switch (priv->agc.i) {
|
|
|
|
case RIG_AGC_SLOW: wj_agc = 0; break; /* slow, 2s */
|
|
|
|
case RIG_AGC_OFF: wj_agc = 1; break; /* "not used" */
|
|
|
|
case RIG_AGC_FAST: wj_agc = 2; break; /* normal, 0.1s */
|
|
|
|
case RIG_AGC_USER: wj_agc = 3; break; /* manual */
|
|
|
|
default: return -RIG_EINVAL;
|
|
|
|
}
|
|
|
|
buf[4] |= wj_agc & 0x1;
|
|
|
|
buf[5] |= (wj_agc & 0x2)<<5;
|
|
|
|
|
|
|
|
/* IF BW */
|
|
|
|
switch (priv->width) {
|
|
|
|
case 200:
|
|
|
|
case 1000: wj_width = 0; break; /* spare */
|
|
|
|
|
|
|
|
case 500: wj_width = 1; break;
|
|
|
|
case 2000: wj_width = 2; break;
|
|
|
|
case 4000: wj_width = 3; break;
|
|
|
|
case 8000: wj_width = 4; break;
|
|
|
|
|
|
|
|
case 3000:
|
|
|
|
case 6000:
|
|
|
|
case 12000:
|
|
|
|
case 16000: wj_width = 5; break; /* spare */
|
|
|
|
default:
|
|
|
|
return -RIG_EINVAL;
|
|
|
|
}
|
|
|
|
buf[5] |= (wj_width & 0x2)<<3;
|
|
|
|
|
|
|
|
/* Detection mode */
|
|
|
|
switch (priv->mode) {
|
|
|
|
case RIG_MODE_CW: wj_mode = (priv->ifshift.i!=0) ? MD_VCW:MD_CW; break;
|
|
|
|
case RIG_MODE_USB: wj_mode = MD_USB; break;
|
|
|
|
case RIG_MODE_LSB: wj_mode = MD_LSB; break;
|
|
|
|
case RIG_MODE_FM: wj_mode = MD_FM; break;
|
|
|
|
case RIG_MODE_AM: wj_mode = MD_AM; break;
|
|
|
|
case RIG_MODE_AMS: wj_mode = MD_ISB; break;
|
|
|
|
default:
|
|
|
|
rig_debug(RIG_DEBUG_ERR, "%s: unsupported mode %d\n",
|
|
|
|
__FUNCTION__, priv->mode);
|
|
|
|
return -RIG_EINVAL;
|
|
|
|
}
|
|
|
|
buf[5] |= wj_mode & 0x3;
|
|
|
|
|
|
|
|
/* BFO frequency, not sure though */
|
|
|
|
wj_bfo = (priv->ifshift.i/10) + 0x400; /* LSBit is 10Hz, +455kHz */
|
|
|
|
buf[6] |= (wj_bfo >> 5) & 0x3f;
|
|
|
|
buf[7] |= (wj_bfo & 0x1f) << 2;
|
|
|
|
|
|
|
|
/* RF gain */
|
|
|
|
wj_rfgain = (unsigned)(priv->rfgain.f * 0x7f);
|
|
|
|
buf[7] |= (wj_rfgain >> 6) & 0x1;
|
|
|
|
buf[8] |= (wj_rfgain & 0x3f) << 1;
|
|
|
|
|
|
|
|
/* buf[9]: not used if command byte, but must be transmitted */
|
|
|
|
|
|
|
|
serial_flush(&rig->state.rigport);
|
|
|
|
|
2006-10-07 19:00:05 +00:00
|
|
|
retval = write_block(&rig->state.rigport, (char *) buf, CMDSZ);
|
2004-09-12 21:30:21 +00:00
|
|
|
if (retval != RIG_OK)
|
|
|
|
return retval;
|
|
|
|
|
|
|
|
if (monitor) {
|
|
|
|
/*
|
|
|
|
* Transceiver sends back ">"
|
|
|
|
*/
|
2006-10-07 19:00:05 +00:00
|
|
|
retval = read_block(&rig->state.rigport, (char *) rxbuf, CMDSZ);
|
2004-09-12 21:30:21 +00:00
|
|
|
if (retval < 0 || retval > CMDSZ)
|
2005-04-10 21:49:38 +00:00
|
|
|
return -RIG_ERJCTED;
|
2004-09-12 21:30:21 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* TODO: analyze back the reply, and fill in the priv struct
|
|
|
|
*/
|
|
|
|
priv->rawstr.i = rxbuf[9] & 0x7f;
|
|
|
|
}
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
int wj_init(RIG *rig)
|
|
|
|
{
|
|
|
|
struct wj_priv_data *priv;
|
|
|
|
|
|
|
|
if (!rig || !rig->caps)
|
|
|
|
return -RIG_EINVAL;
|
|
|
|
|
|
|
|
priv = (struct wj_priv_data*)malloc(sizeof(struct wj_priv_data));
|
|
|
|
if (!priv) {
|
|
|
|
/* whoops! memory shortage! */
|
|
|
|
return -RIG_ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
rig->state.priv = (void*)priv;
|
|
|
|
|
|
|
|
priv->receiver_id = 0;
|
|
|
|
priv->freq = kHz(500);
|
|
|
|
priv->mode = RIG_MODE_AM;
|
|
|
|
priv->width = kHz(8);
|
|
|
|
priv->agc.i = RIG_AGC_SLOW;
|
|
|
|
priv->rfgain.f = 1;
|
|
|
|
priv->ifshift.i = 0;
|
|
|
|
|
|
|
|
return RIG_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
*/
|
|
|
|
int wj_cleanup(RIG *rig)
|
|
|
|
{
|
|
|
|
if (!rig)
|
|
|
|
return -RIG_EINVAL;
|
|
|
|
|
|
|
|
if (rig->state.priv)
|
|
|
|
free(rig->state.priv);
|
|
|
|
rig->state.priv = NULL;
|
|
|
|
|
|
|
|
return RIG_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Assumes rig!=NULL, rig->state.priv!=NULL
|
|
|
|
*/
|
|
|
|
int wj_set_conf(RIG *rig, token_t token, const char *val)
|
|
|
|
{
|
|
|
|
struct wj_priv_data *priv = (struct wj_priv_data*)rig->state.priv;
|
|
|
|
|
|
|
|
switch (token) {
|
|
|
|
case TOK_RIGID:
|
|
|
|
priv->receiver_id = atoi(val);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return -RIG_EINVAL;
|
|
|
|
}
|
|
|
|
return RIG_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* assumes rig!=NULL,
|
|
|
|
* Assumes rig!=NULL, rig->state.priv!=NULL
|
|
|
|
* and val points to a buffer big enough to hold the conf value.
|
|
|
|
*/
|
|
|
|
int wj_get_conf(RIG *rig, token_t token, char *val)
|
|
|
|
{
|
|
|
|
struct wj_priv_data *priv = (struct wj_priv_data*)rig->state.priv;
|
|
|
|
|
|
|
|
switch(token) {
|
|
|
|
case TOK_RIGID:
|
|
|
|
sprintf(val, "%d", priv->receiver_id);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return -RIG_EINVAL;
|
|
|
|
}
|
|
|
|
return RIG_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* wj_set_freq
|
|
|
|
* Assumes rig!=NULL
|
|
|
|
*/
|
|
|
|
int wj_set_freq(RIG *rig, vfo_t vfo, freq_t freq)
|
|
|
|
{
|
|
|
|
struct wj_priv_data *priv = (struct wj_priv_data*)rig->state.priv;
|
|
|
|
|
|
|
|
priv->freq = freq;
|
|
|
|
|
|
|
|
return wj_transaction (rig, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* wj_get_freq
|
|
|
|
* Assumes rig!=NULL
|
|
|
|
*/
|
|
|
|
int wj_get_freq(RIG *rig, vfo_t vfo, freq_t *freq)
|
|
|
|
{
|
|
|
|
struct wj_priv_data *priv = (struct wj_priv_data*)rig->state.priv;
|
|
|
|
int retval;
|
|
|
|
|
|
|
|
retval = wj_transaction (rig, 1);
|
|
|
|
if (retval == RIG_OK)
|
|
|
|
return retval;
|
|
|
|
|
|
|
|
*freq = priv->freq;
|
|
|
|
|
|
|
|
return RIG_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* wj_set_mode
|
|
|
|
* Assumes rig!=NULL
|
|
|
|
*/
|
|
|
|
int wj_set_mode(RIG *rig, vfo_t vfo, rmode_t mode, pbwidth_t width)
|
|
|
|
{
|
|
|
|
struct wj_priv_data *priv = (struct wj_priv_data*)rig->state.priv;
|
|
|
|
|
|
|
|
priv->mode = mode;
|
|
|
|
|
|
|
|
if (width == RIG_PASSBAND_NORMAL)
|
|
|
|
width = rig_passband_normal(rig, mode);
|
|
|
|
|
|
|
|
priv->width = width;
|
|
|
|
|
|
|
|
return wj_transaction (rig, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* wj_get_mode
|
|
|
|
* Assumes rig!=NULL
|
|
|
|
*/
|
|
|
|
int wj_get_mode(RIG *rig, vfo_t vfo, rmode_t *mode, pbwidth_t *width)
|
|
|
|
{
|
|
|
|
struct wj_priv_data *priv = (struct wj_priv_data*)rig->state.priv;
|
|
|
|
int retval;
|
|
|
|
|
|
|
|
retval = wj_transaction (rig, 1);
|
|
|
|
if (retval == RIG_OK)
|
|
|
|
return retval;
|
|
|
|
|
|
|
|
*mode = priv->mode;
|
|
|
|
*width = priv->width;
|
|
|
|
|
|
|
|
return RIG_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* wj_set_level
|
|
|
|
* Assumes rig!=NULL
|
|
|
|
*/
|
|
|
|
int wj_set_level(RIG *rig, vfo_t vfo, setting_t level, value_t val)
|
|
|
|
{
|
|
|
|
struct wj_priv_data *priv = (struct wj_priv_data*)rig->state.priv;
|
|
|
|
|
|
|
|
switch (level) {
|
|
|
|
case RIG_LEVEL_IF:
|
|
|
|
priv->ifshift.i = val.i;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RIG_LEVEL_RF:
|
|
|
|
priv->rfgain.f = val.f;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RIG_LEVEL_AGC:
|
|
|
|
priv->agc.i = val.i;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
rig_debug(RIG_DEBUG_ERR,"%s: unsupported %d\n", __FUNCTION__, level);
|
|
|
|
return -RIG_EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return wj_transaction (rig, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* wj_get_level
|
|
|
|
* Assumes rig!=NULL
|
|
|
|
*/
|
|
|
|
int wj_get_level(RIG *rig, vfo_t vfo, setting_t level, value_t *val)
|
|
|
|
{
|
|
|
|
struct wj_priv_data *priv = (struct wj_priv_data*)rig->state.priv;
|
|
|
|
int retval = RIG_OK;
|
|
|
|
|
|
|
|
retval = wj_transaction (rig, 1);
|
|
|
|
if (retval == RIG_OK)
|
|
|
|
return retval;
|
|
|
|
|
|
|
|
switch (level) {
|
|
|
|
case RIG_LEVEL_RAWSTR:
|
|
|
|
val->i = priv->rawstr.i;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RIG_LEVEL_IF:
|
|
|
|
val->i = priv->ifshift.i;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RIG_LEVEL_RF:
|
|
|
|
val->f = priv->rfgain.f;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RIG_LEVEL_AGC:
|
|
|
|
val->i = priv->agc.i;
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
rig_debug(RIG_DEBUG_ERR,"%s: Unsupported %d\n", __FUNCTION__, level);
|
|
|
|
return -RIG_EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* initrigs_wj is called by rig_backend_load
|
|
|
|
*/
|
|
|
|
DECLARE_INITRIG_BACKEND(wj)
|
|
|
|
{
|
|
|
|
rig_debug(RIG_DEBUG_VERBOSE, "wj: _init called\n");
|
|
|
|
|
|
|
|
rig_register(&wj8888_caps);
|
|
|
|
|
|
|
|
return RIG_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|