2003-01-29 23:06:30 +00:00
|
|
|
/* -*-C++-*-
|
|
|
|
*******************************************************************************
|
|
|
|
*
|
|
|
|
* File: microtune_eval_board.cc
|
|
|
|
* Description:
|
|
|
|
*
|
|
|
|
*******************************************************************************
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright 2001 Free Software Foundation, Inc.
|
|
|
|
*
|
|
|
|
* This file is part of GNU Radio
|
|
|
|
*
|
|
|
|
* GNU Radio is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2, or (at your option)
|
|
|
|
* any later version.
|
|
|
|
*
|
|
|
|
* GNU Radio 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 General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with GNU Radio; see the file COPYING. If not, write to
|
|
|
|
* the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
|
|
* Boston, MA 02111-1307, USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "microtune_eval_board.h"
|
|
|
|
#include "microtune_eval_board_defs.h"
|
2003-09-28 15:28:37 +00:00
|
|
|
#include "serial.h"
|
2003-01-29 23:06:30 +00:00
|
|
|
#include "i2cio_pp.h"
|
|
|
|
#include "i2c.h"
|
2003-09-28 15:28:37 +00:00
|
|
|
#include <cmath>
|
2003-01-29 23:06:30 +00:00
|
|
|
|
2003-09-28 15:28:37 +00:00
|
|
|
static int AGC_DAC_I2C_ADDR = 0x2C;
|
|
|
|
|
2005-04-03 12:27:17 +00:00
|
|
|
microtune_eval_board::microtune_eval_board (hamlib_port_t *port)
|
2003-01-29 23:06:30 +00:00
|
|
|
{
|
2003-09-28 15:28:37 +00:00
|
|
|
m_ppio = port;
|
2003-01-29 23:06:30 +00:00
|
|
|
m_i2cio = new i2cio_pp (m_ppio);
|
|
|
|
m_i2c = new i2c (m_i2cio);
|
|
|
|
|
|
|
|
// disable upstream amplifier
|
2003-09-28 15:28:37 +00:00
|
|
|
par_lock (m_ppio);
|
|
|
|
unsigned char t;
|
|
|
|
par_read_data (m_ppio, &t);
|
2003-01-29 23:06:30 +00:00
|
|
|
t &= ~(UT_DP_TX_ENABLE | UT_DP_TX_SDA | UT_DP_TX_SCL);
|
|
|
|
t |= UT_DP_TX_AS;
|
2003-09-28 15:28:37 +00:00
|
|
|
par_write_data (m_ppio, t);
|
|
|
|
par_unlock (m_ppio);
|
2003-01-29 23:06:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
microtune_eval_board::~microtune_eval_board ()
|
|
|
|
{
|
|
|
|
delete m_i2c;
|
|
|
|
delete m_i2cio;
|
|
|
|
delete m_i2c;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//! is the eval board present?
|
|
|
|
bool
|
|
|
|
microtune_eval_board::board_present_p ()
|
|
|
|
{
|
2003-09-28 15:28:37 +00:00
|
|
|
bool result = true;
|
|
|
|
par_lock (m_ppio);
|
|
|
|
|
|
|
|
unsigned char t;
|
|
|
|
par_read_status (m_ppio, &t);
|
2003-01-29 23:06:30 +00:00
|
|
|
if ((t & UT_SP_SHOULD_BE_ZERO) != 0
|
|
|
|
|| (t & UT_SP_SHOULD_BE_ONE) != UT_SP_SHOULD_BE_ONE)
|
2003-09-28 15:28:37 +00:00
|
|
|
result = false;
|
2003-01-29 23:06:30 +00:00
|
|
|
|
|
|
|
// could also see if SCL is looped back or not, but that seems like overkill
|
|
|
|
|
2003-09-28 15:28:37 +00:00
|
|
|
par_unlock (m_ppio);
|
|
|
|
return result;
|
2003-01-29 23:06:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// returns true iff successful
|
|
|
|
bool
|
|
|
|
microtune_eval_board::i2c_write (int addr, const unsigned char *buf, int nbytes)
|
|
|
|
{
|
|
|
|
return m_i2c->write (addr, buf, nbytes);
|
|
|
|
}
|
|
|
|
|
|
|
|
// returns number of bytes read or -1 if error
|
|
|
|
int
|
|
|
|
microtune_eval_board::i2c_read (int addr, unsigned char *buf, int max_bytes)
|
|
|
|
{
|
|
|
|
return m_i2c->read (addr, buf, max_bytes);
|
|
|
|
}
|
2003-09-28 15:28:37 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* ----------------------------------------------------------------
|
|
|
|
* AGC stuff
|
|
|
|
*
|
|
|
|
* We're using a MAX518 8-bit 5V dual dac for setting the AGC's
|
|
|
|
* ----------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
microtune_eval_board::write_dac (int which, int value)
|
|
|
|
{
|
|
|
|
unsigned char cmd[2];
|
|
|
|
cmd[0] = which & 1;
|
|
|
|
cmd[1] = value;
|
|
|
|
i2c_write (AGC_DAC_I2C_ADDR, cmd, sizeof (cmd));
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
microtune_eval_board::write_both_dacs (int value0, int value1)
|
|
|
|
{
|
|
|
|
unsigned char cmd[4];
|
|
|
|
cmd[0] = 0;
|
|
|
|
cmd[1] = value0;
|
|
|
|
cmd[2] = 1;
|
|
|
|
cmd[3] = value1;
|
|
|
|
i2c_write (AGC_DAC_I2C_ADDR, cmd, sizeof (cmd));
|
|
|
|
}
|
|
|
|
|
|
|
|
static int scale_volts (float volts)
|
|
|
|
{
|
|
|
|
int n;
|
|
|
|
n = (int) rint (volts * (256 / 5.0));
|
|
|
|
if (n < 0)
|
|
|
|
n = 0;
|
|
|
|
if (n > 255)
|
|
|
|
n = 255;
|
|
|
|
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
microtune_eval_board::set_RF_AGC_voltage (float volts)
|
|
|
|
{
|
|
|
|
write_dac (0, scale_volts (volts));
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
microtune_eval_board::set_IF_AGC_voltage (float volts)
|
|
|
|
{
|
|
|
|
write_dac (1, scale_volts (volts));
|
|
|
|
}
|
|
|
|
|
|
|
|
static const float RF_MIN_V = 1.5; // RF AGC control voltages
|
|
|
|
static const float RF_MAX_V = 4.0;
|
|
|
|
static const float IF_MIN_V = 2.0; // IF AGC control voltages
|
|
|
|
static const float IF_MAX_V = 4.0;
|
|
|
|
|
|
|
|
static const float MIN_AGC = 0; // bottom of synthetic range
|
|
|
|
static const float MAX_AGC = 1000; // top of synthetic range
|
|
|
|
|
|
|
|
static const float CUTOVER_POINT = 667;
|
|
|
|
|
|
|
|
|
|
|
|
// linear is in the range MIN_AGC to MAX_AGC
|
|
|
|
|
|
|
|
static float
|
|
|
|
linear_to_RF_AGC_voltage (float linear)
|
|
|
|
{
|
|
|
|
if (linear >= CUTOVER_POINT)
|
|
|
|
return RF_MAX_V;
|
|
|
|
|
|
|
|
float slope = (RF_MAX_V - RF_MIN_V) / CUTOVER_POINT;
|
|
|
|
return RF_MIN_V + linear * slope;
|
|
|
|
}
|
|
|
|
|
|
|
|
static float
|
|
|
|
linear_to_IF_AGC_voltage (float linear)
|
|
|
|
{
|
|
|
|
if (linear < CUTOVER_POINT)
|
|
|
|
return IF_MIN_V;
|
|
|
|
|
|
|
|
float slope = (IF_MAX_V - IF_MIN_V) / (MAX_AGC - CUTOVER_POINT);
|
|
|
|
return IF_MIN_V + (linear - CUTOVER_POINT) * slope;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
microtune_eval_board::set_AGC (float v)
|
|
|
|
{
|
|
|
|
if (v < MIN_AGC)
|
|
|
|
v = MIN_AGC;
|
|
|
|
|
|
|
|
if (v > MAX_AGC)
|
|
|
|
v = MAX_AGC;
|
|
|
|
|
|
|
|
float rf_agc_voltage = linear_to_RF_AGC_voltage (v);
|
|
|
|
float if_agc_voltage = linear_to_IF_AGC_voltage (v);
|
|
|
|
|
|
|
|
set_RF_AGC_voltage (rf_agc_voltage);
|
|
|
|
set_IF_AGC_voltage (if_agc_voltage);
|
|
|
|
}
|