OpenRTX/openrtx/src/rtx/OpMode_M17.cpp

233 lines
6.9 KiB
C++

/***************************************************************************
* Copyright (C) 2021 by Federico Amedeo Izzo IU2NUO, *
* Niccolò Izzo IU2KIN *
* Frederik Saraci IU2NRO *
* Silvano Seva IU2KWO *
* *
* This program 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 3 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 General Public License for more details. *
* *
* You should have received a copy of the GNU General Public License *
* along with this program; if not, see <http://www.gnu.org/licenses/> *
***************************************************************************/
#include <interfaces/audio_stream.h>
#include <interfaces/audio_path.h>
#include <interfaces/platform.h>
#include <interfaces/delays.h>
#include <interfaces/audio.h>
#include <interfaces/radio.h>
#include <OpMode_M17.h>
#include <codec2.h>
#include <memory>
#include <vector>
#include <rtx.h>
#include <dsp.h>
using namespace std;
pthread_t codecThread; // Thread running CODEC2
pthread_mutex_t codecMtx; // Mutex for shared access between codec and rtx threads
pthread_cond_t codecCv; // Condition variable for data ready
bool runCodec; // Flag signalling that codec is running
bool newData; // Flag signalling that new data is available
array< uint8_t, 16 > encodedData; // Buffer for encoded data
void *threadFunc(void *arg)
{
struct CODEC2 *codec2 = codec2_create(CODEC2_MODE_3200);
unique_ptr< stream_sample_t > audioBuf(new stream_sample_t[320]);
streamId micId = inputStream_start(SOURCE_MIC, PRIO_TX,
audioBuf.get(), 320,
BUF_CIRC_DOUBLE, 8000);
size_t pos = 0;
while(runCodec)
{
dataBlock_t audio = inputStream_getData(micId);
if(audio.data != NULL)
{
#if defined(PLATFORM_MD3x0) || defined(PLATFORM_MDUV3x0)
// Pre-amplification stage
for(size_t i = 0; i < audio.len; i++) audio.data[i] <<= 3;
// DC removal
dsp_dcRemoval(audio.data, audio.len);
// Post-amplification stage
for(size_t i = 0; i < audio.len; i++) audio.data[i] *= 4;
#endif
// CODEC2 encodes 160ms of speech into 8 bytes: here we write the
// new encoded data into a buffer of 16 bytes writing the first
// half and then the second one, sequentially.
// Data ready flag is rised once all the 16 bytes contain new data.
uint8_t *curPos = encodedData.data() + 8*pos;
codec2_encode(codec2, curPos, audio.data);
pos++;
if(pos >= 2)
{
pthread_mutex_lock(&codecMtx);
newData = true;
pthread_cond_signal(&codecCv);
pthread_mutex_unlock(&codecMtx);
pos = 0;
}
}
}
// Unlock waiting thread(s)
pthread_mutex_lock(&codecMtx);
newData = true;
pthread_cond_signal(&codecCv);
pthread_mutex_unlock(&codecMtx);
// Tear down codec and input stream
codec2_destroy(codec2);
inputStream_stop(micId);
return NULL;
}
OpMode_M17::OpMode_M17() : enterRx(true), m17Tx(modulator)
{
}
OpMode_M17::~OpMode_M17()
{
}
void OpMode_M17::enable()
{
// Start CODEC2 thread
runCodec = true;
newData = false;
pthread_mutex_init(&codecMtx, NULL);
pthread_cond_init(&codecCv, NULL);
pthread_attr_t codecAttr;
pthread_attr_init(&codecAttr);
pthread_attr_setstacksize(&codecAttr, 16384);
pthread_create(&codecThread, &codecAttr, threadFunc, NULL);
modulator.init();
enterRx = true;
}
void OpMode_M17::disable()
{
// Shut down CODEC2 thread and wait until it effectively stops
runCodec = false;
pthread_join(codecThread, NULL);
pthread_mutex_destroy(&codecMtx);
pthread_cond_destroy(&codecCv);
enterRx = false;
modulator.terminate();
}
void OpMode_M17::update(rtxStatus_t *const status, const bool newCfg)
{
(void) newCfg;
// RX logic
if(status->opStatus == RX)
{
// TODO: Implement M17 Rx
sleepFor(0u, 30u);
}
else if((status->opStatus == OFF) && enterRx)
{
radio_disableRtx();
radio_enableRx();
status->opStatus = RX;
enterRx = false;
}
// TX logic
if(platform_getPttStatus() && (status->txDisable == 0))
{
// Enter Tx mode, setup transmission
if(status->opStatus != TX)
{
audio_disableAmp();
radio_disableRtx();
audio_enableMic();
radio_enableTx();
std::string source_address(status->source_address);
std::string destination_address(status->destination_address);
m17Tx.start(source_address, destination_address);
status->opStatus = TX;
}
else
{
// Transmission is ongoing, just modulate
sendData(false);
}
}
// PTT is off, transition to Rx state
if(!platform_getPttStatus() && (status->opStatus == TX))
{
// Send last audio frame
sendData(true);
audio_disableMic();
radio_disableRtx();
status->opStatus = OFF;
enterRx = true;
}
// Led control logic
switch(status->opStatus)
{
case RX:
// TODO: Implement Rx LEDs
break;
case TX:
platform_ledOff(GREEN);
platform_ledOn(RED);
break;
default:
platform_ledOff(GREEN);
platform_ledOff(RED);
break;
}
}
void OpMode_M17::sendData(bool lastFrame)
{
payload_t dataFrame;
// Wait until there are 16 bytes of compressed speech, then send them
pthread_mutex_lock(&codecMtx);
while(newData == false)
{
pthread_cond_wait(&codecCv, &codecMtx);
}
newData = false;
pthread_mutex_unlock(&codecMtx);
std::copy(encodedData.begin(), encodedData.end(), dataFrame.begin());
m17Tx.send(dataFrame, lastFrame);
}