Hamlib/rs/rs.c

307 wiersze
6.8 KiB
C

/*
* Hamlib R&S backend - main file
* Copyright (c) 2009 by Stéphane Fillod
*
* $Id: rs.c,v 1.2 2009/08/07 18:48:34 fillods Exp $
*
* 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 "register.h"
#include "rs.h"
#define BUFSZ 64
#define RESPSZ 64
#define LF "\x0a"
#define CR "\x0d"
#define BOM CR
#define EOM CR
/*
* R&S GB2 protocol ?
*/
/*
* rs_transaction
* We assume that rig!=NULL, rig->state!= NULL, data!=NULL, data_len!=NULL
*/
int rs_transaction(RIG *rig, const char *cmd, int cmd_len, char *data, int *data_len)
{
int retval;
struct rig_state *rs;
rs = &rig->state;
serial_flush(&rs->rigport);
retval = write_block(&rs->rigport, cmd, cmd_len);
if (retval != RIG_OK)
return retval;
/* no data expected */
if (!data || !data_len)
return RIG_OK;
retval = read_string(&rs->rigport, data, BUFSZ, CR, 1);
if (retval < 0)
return retval;
*data_len = retval;
return RIG_OK;
}
/*
* rs_set_freq
* Assumes rig!=NULL
*/
int rs_set_freq(RIG *rig, vfo_t vfo, freq_t freq)
{
char freqbuf[32];
int freq_len, retval;
freq_len = sprintf(freqbuf, BOM "FREQ %"PRIll EOM, (long long)freq);
retval = rs_transaction(rig, freqbuf, freq_len, NULL, NULL);
return retval;
}
/*
* rs_get_freq
* Assumes rig!=NULL
*/
int rs_get_freq(RIG *rig, vfo_t vfo, freq_t *freq)
{
char buf[RESPSZ];
int len, retval;
#define FREQ_QUERY BOM "FREQ?" EOM
retval = rs_transaction(rig, FREQ_QUERY, strlen(FREQ_QUERY), buf, &len);
if (retval < 0)
return retval;
retval = (sscanf(buf, "%"SCNfreq, freq)==1) ? RIG_OK : -RIG_EPROTO;
return retval;
}
/*
* rs_set_mode
* Assumes rig!=NULL
*/
int rs_set_mode(RIG *rig, vfo_t vfo, rmode_t mode, pbwidth_t width)
{
char buf[32], *smode;
int len, retval;
switch (mode) {
case RIG_MODE_AM: smode = "AM"; break;
case RIG_MODE_WFM:
case RIG_MODE_FM: smode = "FM"; break;
case RIG_MODE_CW: smode = "CW"; break;
case RIG_MODE_USB: smode = "USB"; break;
case RIG_MODE_LSB: smode = "LSB"; break;
default:
return -RIG_EINVAL;
}
len = sprintf(buf, BOM "DEM %s" EOM, smode);
retval = rs_transaction(rig, buf, len, NULL, NULL);
if (width == RIG_PASSBAND_NORMAL)
width = rig_passband_normal(rig, mode);
if (width > 0) {
len = sprintf(buf, BOM "BAND %d" EOM, (int) width);
retval = rs_transaction(rig, buf, len, NULL, NULL);
}
return retval;
}
/*
* rs_get_mode
* Assumes rig!=NULL
*/
int rs_get_mode(RIG *rig, vfo_t vfo, rmode_t *mode, pbwidth_t *width)
{
char buf[RESPSZ];
int buf_len, retval;
#define DEM_QUERY BOM "DEM?" EOM
retval = rs_transaction(rig, DEM_QUERY, strlen(DEM_QUERY), buf, &buf_len);
if (retval < 0)
return retval;
*mode = rig_parse_mode(buf);
#define BAND_QUERY BOM "BAND?" EOM
retval = rs_transaction(rig, BAND_QUERY, strlen(BAND_QUERY), buf, &buf_len);
if (retval < 0)
return retval;
*width = atoi(buf);
return retval;
}
int rs_set_func(RIG *rig, vfo_t vfo, setting_t func, int status)
{
char buf[32], *sfunc;
int len, retval;
switch (func) {
case RIG_FUNC_AFC: sfunc = "FREQ:AFC"; break;
case RIG_FUNC_SQL: sfunc = "OUTP:SQU"; break;
default:
return -RIG_EINVAL;
}
len = sprintf(buf, BOM "%s %s" EOM, sfunc, status ? "ON" : "OFF");
retval = rs_transaction(rig, buf, len, NULL, NULL);
return retval;
}
int rs_get_func(RIG *rig, vfo_t vfo, setting_t func, int *status)
{
char buf[RESPSZ], *sfunc;
int buf_len, retval;
switch (func) {
case RIG_FUNC_AFC: sfunc = "FREQ:AFC?"; break;
case RIG_FUNC_SQL: sfunc = "OUTP:SQU?"; break;
default:
return -RIG_EINVAL;
}
retval = rs_transaction(rig, sfunc, strlen(sfunc), buf, &buf_len);
if (retval < 0)
return retval;
*status = (memcmp(buf, "ON", 2) == 0) ? 1 : 0;
return retval;
}
int rs_set_level(RIG *rig, vfo_t vfo, setting_t level, value_t val)
{
char buf[32];
int len, retval;
switch (level) {
case RIG_LEVEL_ATT:
len = sprintf(buf, BOM "INP:ATT:STAT %s" EOM, val.i ? "ON" : "OFF");
break;
case RIG_LEVEL_SQL:
case RIG_LEVEL_AGC:
case RIG_LEVEL_RF:
return -RIG_ENIMPL;
default:
return -RIG_EINVAL;
}
retval = rs_transaction(rig, buf, len, NULL, NULL);
return retval;
}
int rs_get_level(RIG *rig, vfo_t vfo, setting_t level, value_t *val)
{
char buf[RESPSZ], *slevel;
int buf_len, retval;
switch (level) {
case RIG_LEVEL_STRENGTH: slevel = BOM "SENS:DATA? \"VOLT:AC\"" EOM; break;
case RIG_LEVEL_ATT: slevel = BOM "INP:ATT:STAT?" EOM; break;
break;
case RIG_LEVEL_SQL:
case RIG_LEVEL_AGC:
case RIG_LEVEL_RF:
return -RIG_ENIMPL;
default:
return -RIG_EINVAL;
}
retval = rs_transaction(rig, slevel, strlen(slevel), buf, &buf_len);
if (retval < 0)
return retval;
switch (level) {
case RIG_LEVEL_STRENGTH:
/* assumes FORMAat:DATA ASCii
* result in dBuV, keep only integer part
*/
sscanf(buf, "%d", &val->i);
val->i -= 34;
break;
case RIG_LEVEL_ATT:
val->i = (memcmp(buf, "ON", 2) == 0) ? rig->state.attenuator[0] : 0;
break;
default:
return -RIG_EINVAL;
}
return retval;
}
const char * rs_get_info(RIG *rig)
{
static char infobuf[128];
int info_len, retval;
#define ID_QUERY BOM "*IDN?" EOM
retval = rs_transaction(rig, ID_QUERY, strlen(ID_QUERY), infobuf, &info_len);
if (retval < 0)
return NULL;
return infobuf;
}
/*
* initrigs_rs is called by rig_backend_load
*/
DECLARE_INITRIG_BACKEND(rs)
{
rig_debug(RIG_DEBUG_VERBOSE, "rs: _init called\n");
rig_register(&esmc_caps);
return RIG_OK;
}