This commit addresses all warnings emitted from -Wunused-variable in the C++ src. Signed-off-by: Noel Eck <noel.eck@intel.com>
504 lines
11 KiB
C++
504 lines
11 KiB
C++
/*
|
|
* Author: Jon Trulson <jtrulson@ics.com>
|
|
* Copyright (c) 2015 Intel Corporation.
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining
|
|
* a copy of this software and associated documentation files (the
|
|
* "Software"), to deal in the Software without restriction, including
|
|
* without limitation the rights to use, copy, modify, merge, publish,
|
|
* distribute, sublicense, and/or sell copies of the Software, and to
|
|
* permit persons to whom the Software is furnished to do so, subject to
|
|
* the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be
|
|
* included in all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
* LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
* OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
* WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
*/
|
|
|
|
#include <iostream>
|
|
#include <string>
|
|
#include <stdexcept>
|
|
|
|
#include "hmtrp.hpp"
|
|
|
|
using namespace upm;
|
|
using namespace std;
|
|
|
|
static const int defaultDelay = 100; // max wait time for read
|
|
|
|
// protocol start code
|
|
const uint8_t HMTRP_START1 = 0xaa;
|
|
const uint8_t HMTRP_START2 = 0xfa;
|
|
|
|
HMTRP::HMTRP(int uart)
|
|
{
|
|
m_ttyFd = -1;
|
|
|
|
if ( !(m_uart = mraa_uart_init(uart)) )
|
|
{
|
|
throw std::invalid_argument(std::string(__FUNCTION__) +
|
|
": mraa_uart_init() failed");
|
|
return;
|
|
}
|
|
|
|
// This requires a recent MRAA (1/2015)
|
|
const char *devPath = mraa_uart_get_dev_path(m_uart);
|
|
|
|
if (!devPath)
|
|
{
|
|
throw std::runtime_error(std::string(__FUNCTION__) +
|
|
": mraa_uart_get_dev_path() failed");
|
|
return;
|
|
}
|
|
|
|
// now open the tty
|
|
if ( (m_ttyFd = open(devPath, O_RDWR)) == -1)
|
|
{
|
|
throw std::runtime_error(std::string(__FUNCTION__) +
|
|
": open of " +
|
|
string(devPath) + " failed: " +
|
|
string(strerror(errno)));
|
|
return;
|
|
}
|
|
}
|
|
|
|
HMTRP::~HMTRP()
|
|
{
|
|
if (m_ttyFd != -1)
|
|
close(m_ttyFd);
|
|
}
|
|
|
|
bool HMTRP::dataAvailable(unsigned int millis)
|
|
{
|
|
if (m_ttyFd == -1)
|
|
return false;
|
|
|
|
struct timeval timeout;
|
|
|
|
timeout.tv_sec = 0;
|
|
timeout.tv_usec = millis * 1000;
|
|
|
|
fd_set readfds;
|
|
|
|
FD_ZERO(&readfds);
|
|
|
|
FD_SET(m_ttyFd, &readfds);
|
|
|
|
if (select(m_ttyFd + 1, &readfds, NULL, NULL, &timeout) > 0)
|
|
return true; // data is ready
|
|
else
|
|
return false;
|
|
}
|
|
|
|
int HMTRP::readData(char *buffer, int len, int millis)
|
|
{
|
|
if (m_ttyFd == -1)
|
|
return(-1);
|
|
|
|
// if specified, wait to see if input shows up, otherwise block
|
|
if (millis >= 0)
|
|
{
|
|
if (!dataAvailable(millis))
|
|
return 0; // timed out
|
|
}
|
|
|
|
int rv = read(m_ttyFd, buffer, len);
|
|
|
|
if (rv < 0)
|
|
{
|
|
throw std::runtime_error(std::string(__FUNCTION__) +
|
|
": read() failed: " +
|
|
string(strerror(errno)));
|
|
return rv;
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
int HMTRP::writeData(char *buffer, int len)
|
|
{
|
|
if (m_ttyFd == -1)
|
|
return(-1);
|
|
|
|
int rv = write(m_ttyFd, buffer, len);
|
|
|
|
if (rv < 0)
|
|
{
|
|
throw std::runtime_error(std::string(__FUNCTION__) +
|
|
": write() failed: " +
|
|
string(strerror(errno)));
|
|
return rv;
|
|
}
|
|
|
|
tcdrain(m_ttyFd);
|
|
|
|
return rv;
|
|
}
|
|
|
|
bool HMTRP::setupTty(speed_t baud)
|
|
{
|
|
if (m_ttyFd == -1)
|
|
return(false);
|
|
|
|
struct termios termio;
|
|
|
|
// get current modes
|
|
tcgetattr(m_ttyFd, &termio);
|
|
|
|
// setup for a 'raw' mode. 81N, no echo or special character
|
|
// handling, such as flow control.
|
|
cfmakeraw(&termio);
|
|
|
|
// set our baud rates
|
|
cfsetispeed(&termio, baud);
|
|
cfsetospeed(&termio, baud);
|
|
|
|
// make it so
|
|
if (tcsetattr(m_ttyFd, TCSAFLUSH, &termio) < 0)
|
|
{
|
|
throw std::runtime_error(std::string(__FUNCTION__) +
|
|
": tcsetattr() failed: " +
|
|
string(strerror(errno)));
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool HMTRP::checkOK()
|
|
{
|
|
char buf[4];
|
|
|
|
int rv = readData(buf, 4, defaultDelay);
|
|
|
|
if (rv != 4)
|
|
{
|
|
cerr << __FUNCTION__ << ": failed to receive OK response, rv = "
|
|
<< rv << ", expected 4" << endl;
|
|
return false;
|
|
}
|
|
|
|
// looking for "OK\r\n"
|
|
if (buf[0] == 'O' && buf[1] == 'K' &&
|
|
buf[2] == '\r' && buf[3] == '\n')
|
|
return true;
|
|
else
|
|
return false;
|
|
}
|
|
|
|
bool HMTRP::reset()
|
|
{
|
|
char pkt[3];
|
|
|
|
pkt[0] = HMTRP_START1;
|
|
pkt[1] = HMTRP_START2;
|
|
pkt[2] = RESET;
|
|
|
|
writeData(pkt, 3);
|
|
|
|
return checkOK();
|
|
}
|
|
|
|
bool HMTRP::getConfig(uint32_t *freq, uint32_t *dataRate,
|
|
uint16_t *rxBandwidth, uint8_t *modulation,
|
|
uint8_t *txPower, uint32_t *uartBaud)
|
|
{
|
|
char pkt[3];
|
|
pkt[0] = HMTRP_START1;
|
|
pkt[1] = HMTRP_START2;
|
|
pkt[2] = GET_CONFIG;
|
|
|
|
writeData(pkt, 3);
|
|
usleep(100000);
|
|
|
|
// now read back a 16 byte response
|
|
char buf[16];
|
|
int rv = readData(buf, 16, defaultDelay);
|
|
|
|
if (rv != 16)
|
|
{
|
|
cerr << __FUNCTION__ << ": failed to receive correct response: rv = "
|
|
<< rv << ", expected 16" << endl;
|
|
return false;
|
|
}
|
|
|
|
// now decode
|
|
if (freq)
|
|
{
|
|
*freq = ( ((buf[0] & 0xff) << 24) |
|
|
((buf[1] & 0xff) << 16) |
|
|
((buf[2] & 0xff) << 8) |
|
|
(buf[3] & 0xff) );
|
|
}
|
|
|
|
if (dataRate)
|
|
{
|
|
*dataRate = ( ((buf[4] & 0xff) << 24) |
|
|
((buf[5] & 0xff) << 16) |
|
|
((buf[6] & 0xff) << 8) |
|
|
(buf[7] & 0xff) );
|
|
}
|
|
|
|
if (rxBandwidth)
|
|
{
|
|
*rxBandwidth = ( ((buf[8] & 0xff) << 8) |
|
|
(buf[9] & 0xff) );
|
|
}
|
|
|
|
if (modulation)
|
|
{
|
|
*modulation = buf[10] & 0xff;
|
|
}
|
|
|
|
if (txPower)
|
|
{
|
|
*txPower = buf[11] & 0xff;
|
|
}
|
|
|
|
if (uartBaud)
|
|
{
|
|
*uartBaud = ( ((buf[12] & 0xff) << 24) |
|
|
((buf[13] & 0xff) << 16) |
|
|
((buf[14] & 0xff) << 8) |
|
|
(buf[15] & 0xff) );
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool HMTRP::setFrequency(uint32_t freq)
|
|
{
|
|
char pkt[7];
|
|
|
|
pkt[0] = HMTRP_START1;
|
|
pkt[1] = HMTRP_START2;
|
|
pkt[2] = SET_FREQUENCY;
|
|
|
|
pkt[3] = ( ((freq & 0xff000000) >> 24) & 0xff );
|
|
pkt[4] = ( ((freq & 0x00ff0000) >> 16) & 0xff );
|
|
pkt[5] = ( ((freq & 0x0000ff00) >> 8) & 0xff );
|
|
pkt[6] = ( (freq & 0x000000ff) & 0xff );
|
|
|
|
writeData(pkt, 7);
|
|
|
|
return checkOK();
|
|
}
|
|
|
|
bool HMTRP::setRFDataRate(uint32_t rate)
|
|
{
|
|
// Valid values are between 1200-115200
|
|
|
|
if (rate < 1200 || rate > 115200)
|
|
{
|
|
throw std::out_of_range(std::string(__FUNCTION__) +
|
|
": Valid rate values are between 1200-115200");
|
|
return false;
|
|
}
|
|
|
|
char pkt[7];
|
|
|
|
pkt[0] = HMTRP_START1;
|
|
pkt[1] = HMTRP_START2;
|
|
pkt[2] = SET_RF_DATARATE;
|
|
|
|
pkt[3] = ( ((rate & 0xff000000) >> 24) & 0xff );
|
|
pkt[4] = ( ((rate & 0x00ff0000) >> 16) & 0xff );
|
|
pkt[5] = ( ((rate & 0x0000ff00) >> 8) & 0xff );
|
|
pkt[6] = ( (rate & 0x000000ff) & 0xff );
|
|
|
|
writeData(pkt, 7);
|
|
|
|
return checkOK();
|
|
}
|
|
|
|
bool HMTRP::setRXBandwidth(uint16_t rxBand)
|
|
{
|
|
// Valid values are between 30-620 (in Khz)
|
|
|
|
if (rxBand < 30 || rxBand > 620)
|
|
{
|
|
throw std::out_of_range(std::string(__FUNCTION__) +
|
|
": Valid rxBand values are between 30-620");
|
|
return false;
|
|
}
|
|
|
|
char pkt[5];
|
|
|
|
pkt[0] = HMTRP_START1;
|
|
pkt[1] = HMTRP_START2;
|
|
pkt[2] = SET_RX_BW;
|
|
|
|
pkt[3] = ( ((rxBand & 0xff00) >> 8) & 0xff );
|
|
pkt[4] = ( rxBand & 0xff );
|
|
|
|
writeData(pkt, 5);
|
|
|
|
return checkOK();
|
|
}
|
|
|
|
bool HMTRP::setFrequencyModulation(uint8_t modulation)
|
|
{
|
|
// Valid values are between 10-160 (in Khz)
|
|
|
|
if (modulation < 10 || modulation > 160)
|
|
{
|
|
throw std::out_of_range(std::string(__FUNCTION__) +
|
|
": Valid modulation values are between 10-160");
|
|
return false;
|
|
}
|
|
|
|
char pkt[4];
|
|
|
|
pkt[0] = HMTRP_START1;
|
|
pkt[1] = HMTRP_START2;
|
|
pkt[2] = SET_FREQ_MODULATION;
|
|
|
|
pkt[3] = modulation;
|
|
|
|
writeData(pkt, 4);
|
|
|
|
return checkOK();
|
|
}
|
|
|
|
bool HMTRP::setTransmitPower(uint8_t power)
|
|
{
|
|
// Valid values are between 0-7
|
|
|
|
if (power > 7)
|
|
{
|
|
throw std::out_of_range(std::string(__FUNCTION__) +
|
|
": Valid power values are between 0-7");
|
|
return false;
|
|
}
|
|
|
|
char pkt[4];
|
|
|
|
pkt[0] = HMTRP_START1;
|
|
pkt[1] = HMTRP_START2;
|
|
pkt[2] = SET_TX_POWER;
|
|
|
|
pkt[3] = power;
|
|
|
|
writeData(pkt, 4);
|
|
|
|
return checkOK();
|
|
}
|
|
|
|
bool HMTRP::setUARTSpeed(uint32_t speed)
|
|
{
|
|
// Valid values are between 1200-115200
|
|
|
|
if (speed < 1200 || speed > 115200)
|
|
{
|
|
throw std::out_of_range(std::string(__FUNCTION__) +
|
|
": Valid speed values are between 1200-115200");
|
|
return false;
|
|
}
|
|
|
|
char pkt[7];
|
|
|
|
pkt[0] = HMTRP_START1;
|
|
pkt[1] = HMTRP_START2;
|
|
pkt[2] = SET_UART_SPEED;
|
|
|
|
pkt[3] = ( ((speed & 0xff000000) >> 24) & 0xff );
|
|
pkt[4] = ( ((speed & 0x00ff0000) >> 16) & 0xff );
|
|
pkt[5] = ( ((speed & 0x0000ff00) >> 8) & 0xff );
|
|
pkt[6] = ( (speed & 0x000000ff) & 0xff );
|
|
|
|
writeData(pkt, 7);
|
|
|
|
return checkOK();
|
|
}
|
|
|
|
|
|
bool HMTRP::getRFSignalStrength(uint8_t *strength)
|
|
{
|
|
if (!strength)
|
|
return false;
|
|
|
|
*strength = 0;
|
|
|
|
char pkt[3];
|
|
pkt[0] = HMTRP_START1;
|
|
pkt[1] = HMTRP_START2;
|
|
pkt[2] = GET_RF_SIGNAL_STR;
|
|
|
|
writeData(pkt, 3);
|
|
usleep(100000);
|
|
|
|
// now read back a 1 byte response
|
|
char buf;
|
|
int rv = readData(&buf, 1, defaultDelay);
|
|
|
|
if (rv != 1)
|
|
{
|
|
cerr << __FUNCTION__ << ": failed to receive correct response: rv = "
|
|
<< rv << ", expected 1" << endl;
|
|
return false;
|
|
}
|
|
|
|
// now decode
|
|
*strength = (uint8_t)buf;
|
|
|
|
return true;
|
|
}
|
|
|
|
uint8_t HMTRP::getRFSignalStrength()
|
|
{
|
|
uint8_t strength = 0;
|
|
if (!getRFSignalStrength(&strength))
|
|
throw std::runtime_error(std::string(__PRETTY_FUNCTION__) +
|
|
": readData() failed");
|
|
return strength;
|
|
}
|
|
|
|
bool HMTRP::getModSignalStrength(uint8_t *strength)
|
|
{
|
|
if (!strength)
|
|
return false;
|
|
|
|
*strength = 0;
|
|
|
|
char pkt[3];
|
|
pkt[0] = HMTRP_START1;
|
|
pkt[1] = HMTRP_START2;
|
|
pkt[2] = GET_MOD_SIGNAL_STR;
|
|
|
|
writeData(pkt, 3);
|
|
usleep(100000);
|
|
|
|
// now read back a 1 byte response
|
|
char buf;
|
|
int rv = readData(&buf, 1, defaultDelay);
|
|
|
|
if (rv != 1)
|
|
{
|
|
cerr << __FUNCTION__ << ": failed to receive correct response: rv = "
|
|
<< rv << ", expected 1" << endl;
|
|
return false;
|
|
}
|
|
|
|
// now decode
|
|
*strength = (uint8_t)buf;
|
|
|
|
return true;
|
|
}
|
|
|
|
uint8_t HMTRP::getModSignalStrength()
|
|
{
|
|
uint8_t strength = 0;
|
|
if (!getModSignalStrength(&strength))
|
|
throw std::runtime_error(std::string(__PRETTY_FUNCTION__) +
|
|
": readData() failed");
|
|
return strength;
|
|
}
|
|
|