mirror of
				https://gitea.osmocom.org/cellular-infrastructure/osmo-trx.git
				synced 2025-11-03 21:53:18 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			1172 lines
		
	
	
		
			33 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1172 lines
		
	
	
		
			33 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						|
* Copyright 2008, 2009, 2010 Free Software Foundation, Inc.
 | 
						|
*
 | 
						|
* This software is distributed under the terms of the GNU Public License.
 | 
						|
* See the COPYING file in the main directory for details.
 | 
						|
*
 | 
						|
* This use of this software may be subject to additional restrictions.
 | 
						|
* See the LEGAL file in the main directory for details.
 | 
						|
 | 
						|
    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 <stdio.h>
 | 
						|
#include <iomanip>      // std::setprecision
 | 
						|
#include <fstream>
 | 
						|
#include "Transceiver.h"
 | 
						|
#include <Logger.h>
 | 
						|
 | 
						|
extern "C" {
 | 
						|
#include "osmo_signal.h"
 | 
						|
#include "proto_trxd.h"
 | 
						|
 | 
						|
#include <osmocom/core/utils.h>
 | 
						|
#include <osmocom/core/socket.h>
 | 
						|
}
 | 
						|
 | 
						|
#ifdef HAVE_CONFIG_H
 | 
						|
#include "config.h"
 | 
						|
#endif
 | 
						|
 | 
						|
using namespace GSM;
 | 
						|
 | 
						|
#define USB_LATENCY_INTRVL		10,0
 | 
						|
 | 
						|
/* Number of running values use in noise average */
 | 
						|
#define NOISE_CNT			20
 | 
						|
 | 
						|
TransceiverState::TransceiverState()
 | 
						|
  : mRetrans(false), mNoiseLev(0.0), mNoises(NOISE_CNT), mPower(0.0)
 | 
						|
{
 | 
						|
  for (int i = 0; i < 8; i++) {
 | 
						|
    chanType[i] = Transceiver::NONE;
 | 
						|
    fillerModulus[i] = 26;
 | 
						|
    chanResponse[i] = NULL;
 | 
						|
    DFEForward[i] = NULL;
 | 
						|
    DFEFeedback[i] = NULL;
 | 
						|
 | 
						|
    for (int n = 0; n < 102; n++)
 | 
						|
      fillerTable[n][i] = NULL;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
TransceiverState::~TransceiverState()
 | 
						|
{
 | 
						|
  for (int i = 0; i < 8; i++) {
 | 
						|
    delete chanResponse[i];
 | 
						|
    delete DFEForward[i];
 | 
						|
    delete DFEFeedback[i];
 | 
						|
 | 
						|
    for (int n = 0; n < 102; n++)
 | 
						|
      delete fillerTable[n][i];
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
bool TransceiverState::init(FillerType filler, size_t sps, float scale, size_t rtsc, unsigned rach_delay)
 | 
						|
{
 | 
						|
  signalVector *burst;
 | 
						|
 | 
						|
  if ((sps != 1) && (sps != 4))
 | 
						|
    return false;
 | 
						|
 | 
						|
  for (size_t n = 0; n < 8; n++) {
 | 
						|
    for (size_t i = 0; i < 102; i++) {
 | 
						|
      switch (filler) {
 | 
						|
      case FILLER_DUMMY:
 | 
						|
        burst = generateDummyBurst(sps, n);
 | 
						|
        break;
 | 
						|
      case FILLER_NORM_RAND:
 | 
						|
        burst = genRandNormalBurst(rtsc, sps, n);
 | 
						|
        break;
 | 
						|
      case FILLER_EDGE_RAND:
 | 
						|
        burst = generateEdgeBurst(rtsc);
 | 
						|
        break;
 | 
						|
      case FILLER_ACCESS_RAND:
 | 
						|
        burst = genRandAccessBurst(rach_delay, sps, n);
 | 
						|
        break;
 | 
						|
      case FILLER_ZERO:
 | 
						|
      default:
 | 
						|
        burst = generateEmptyBurst(sps, n);
 | 
						|
      }
 | 
						|
 | 
						|
      scaleVector(*burst, scale);
 | 
						|
      fillerTable[i][n] = burst;
 | 
						|
    }
 | 
						|
 | 
						|
    if ((filler == FILLER_NORM_RAND) ||
 | 
						|
        (filler == FILLER_EDGE_RAND)) {
 | 
						|
        chanType[n] = TSC;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  return false;
 | 
						|
}
 | 
						|
 | 
						|
Transceiver::Transceiver(int wBasePort,
 | 
						|
                         const char *TRXAddress,
 | 
						|
                         const char *GSMcoreAddress,
 | 
						|
                         size_t tx_sps, size_t rx_sps, size_t chans,
 | 
						|
                         GSM::Time wTransmitLatency,
 | 
						|
                         RadioInterface *wRadioInterface,
 | 
						|
                         double wRssiOffset, int wStackSize)
 | 
						|
  : mBasePort(wBasePort), mLocalAddr(TRXAddress), mRemoteAddr(GSMcoreAddress),
 | 
						|
    mClockSocket(-1), mTransmitLatency(wTransmitLatency), mRadioInterface(wRadioInterface),
 | 
						|
    rssiOffset(wRssiOffset), stackSize(wStackSize),
 | 
						|
    mSPSTx(tx_sps), mSPSRx(rx_sps), mChans(chans), mEdge(false), mOn(false), mForceClockInterface(false),
 | 
						|
    mTxFreq(0.0), mRxFreq(0.0), mTSC(0), mMaxExpectedDelayAB(0), mMaxExpectedDelayNB(0),
 | 
						|
    mWriteBurstToDiskMask(0), mVersionTRXD(0)
 | 
						|
{
 | 
						|
  txFullScale = mRadioInterface->fullScaleInputValue();
 | 
						|
  rxFullScale = mRadioInterface->fullScaleOutputValue();
 | 
						|
 | 
						|
  for (int i = 0; i < 8; i++) {
 | 
						|
    for (int j = 0; j < 8; j++)
 | 
						|
      mHandover[i][j] = false;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
Transceiver::~Transceiver()
 | 
						|
{
 | 
						|
  stop();
 | 
						|
 | 
						|
  sigProcLibDestroy();
 | 
						|
 | 
						|
  if (mClockSocket >= 0)
 | 
						|
    close(mClockSocket);
 | 
						|
 | 
						|
  for (size_t i = 0; i < mChans; i++) {
 | 
						|
    mControlServiceLoopThreads[i]->cancel();
 | 
						|
    mControlServiceLoopThreads[i]->join();
 | 
						|
    delete mControlServiceLoopThreads[i];
 | 
						|
 | 
						|
    mTxPriorityQueues[i].clear();
 | 
						|
    if (mCtrlSockets[i] >= 0)
 | 
						|
      close(mCtrlSockets[i]);
 | 
						|
    if (mDataSockets[i] >= 0)
 | 
						|
      close(mDataSockets[i]);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * Initialize transceiver
 | 
						|
 *
 | 
						|
 * Start or restart the control loop. Any further control is handled through the
 | 
						|
 * socket API. Randomize the central radio clock set the downlink burst
 | 
						|
 * counters. Note that the clock will not update until the radio starts, but we
 | 
						|
 * are still expected to report clock indications through control channel
 | 
						|
 * activity.
 | 
						|
 */
 | 
						|
bool Transceiver::init(FillerType filler, size_t rtsc, unsigned rach_delay,
 | 
						|
                       bool edge, bool ext_rach)
 | 
						|
{
 | 
						|
  int d_srcport, d_dstport, c_srcport, c_dstport;
 | 
						|
 | 
						|
  if (!mChans) {
 | 
						|
    LOG(ALERT) << "No channels assigned";
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  if (!sigProcLibSetup()) {
 | 
						|
    LOG(ALERT) << "Failed to initialize signal processing library";
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  mExtRACH = ext_rach;
 | 
						|
  mEdge = edge;
 | 
						|
 | 
						|
  mDataSockets.resize(mChans, -1);
 | 
						|
  mCtrlSockets.resize(mChans, -1);
 | 
						|
  mControlServiceLoopThreads.resize(mChans);
 | 
						|
  mTxPriorityQueueServiceLoopThreads.resize(mChans);
 | 
						|
  mRxServiceLoopThreads.resize(mChans);
 | 
						|
 | 
						|
  mTxPriorityQueues.resize(mChans);
 | 
						|
  mReceiveFIFO.resize(mChans);
 | 
						|
  mStates.resize(mChans);
 | 
						|
 | 
						|
  /* Filler table retransmissions - support only on channel 0 */
 | 
						|
  if (filler == FILLER_DUMMY)
 | 
						|
    mStates[0].mRetrans = true;
 | 
						|
 | 
						|
  /* Setup sockets */
 | 
						|
  mClockSocket = osmo_sock_init2(AF_UNSPEC, SOCK_DGRAM, IPPROTO_UDP,
 | 
						|
				    mLocalAddr.c_str(), mBasePort,
 | 
						|
				    mRemoteAddr.c_str(), mBasePort + 100,
 | 
						|
				    OSMO_SOCK_F_BIND | OSMO_SOCK_F_CONNECT);
 | 
						|
 | 
						|
  for (size_t i = 0; i < mChans; i++) {
 | 
						|
    c_srcport = mBasePort + 2 * i + 1;
 | 
						|
    c_dstport = mBasePort + 2 * i + 101;
 | 
						|
    d_srcport = mBasePort + 2 * i + 2;
 | 
						|
    d_dstport = mBasePort + 2 * i + 102;
 | 
						|
 | 
						|
    mCtrlSockets[i] = osmo_sock_init2(AF_UNSPEC, SOCK_DGRAM, IPPROTO_UDP,
 | 
						|
                                      mLocalAddr.c_str(), c_srcport,
 | 
						|
                                      mRemoteAddr.c_str(), c_dstport,
 | 
						|
				      OSMO_SOCK_F_BIND | OSMO_SOCK_F_CONNECT);
 | 
						|
    if (mCtrlSockets[i] < 0)
 | 
						|
      return false;
 | 
						|
 | 
						|
    mDataSockets[i] = osmo_sock_init2(AF_UNSPEC, SOCK_DGRAM, IPPROTO_UDP,
 | 
						|
                                      mLocalAddr.c_str(), d_srcport,
 | 
						|
                                      mRemoteAddr.c_str(), d_dstport,
 | 
						|
				      OSMO_SOCK_F_BIND | OSMO_SOCK_F_CONNECT);
 | 
						|
    if (mCtrlSockets[i] < 0)
 | 
						|
      return false;
 | 
						|
  }
 | 
						|
 | 
						|
  /* Randomize the central clock */
 | 
						|
  GSM::Time startTime(random() % gHyperframe, 0);
 | 
						|
  mRadioInterface->getClock()->set(startTime);
 | 
						|
  mTransmitDeadlineClock = startTime;
 | 
						|
  mLastClockUpdateTime = startTime;
 | 
						|
  mLatencyUpdateTime = startTime;
 | 
						|
 | 
						|
  /* Start control threads */
 | 
						|
  for (size_t i = 0; i < mChans; i++) {
 | 
						|
    TransceiverChannel *chan = new TransceiverChannel(this, i);
 | 
						|
    mControlServiceLoopThreads[i] = new Thread(stackSize);
 | 
						|
    mControlServiceLoopThreads[i]->start((void * (*)(void*))
 | 
						|
                                 ControlServiceLoopAdapter, (void*) chan);
 | 
						|
 | 
						|
    if (i && filler == FILLER_DUMMY)
 | 
						|
      filler = FILLER_ZERO;
 | 
						|
 | 
						|
    mStates[i].init(filler, mSPSTx, txFullScale, rtsc, rach_delay);
 | 
						|
  }
 | 
						|
 | 
						|
  return true;
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * Start the transceiver
 | 
						|
 *
 | 
						|
 * Submit command(s) to the radio device to commence streaming samples and
 | 
						|
 * launch threads to handle sample I/O. Re-synchronize the transmit burst
 | 
						|
 * counters to the central radio clock here as well.
 | 
						|
 */
 | 
						|
bool Transceiver::start()
 | 
						|
{
 | 
						|
  ScopedLock lock(mLock);
 | 
						|
 | 
						|
  if (mOn) {
 | 
						|
    LOG(ERR) << "Transceiver already running";
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  LOG(NOTICE) << "Starting the transceiver";
 | 
						|
 | 
						|
  GSM::Time time = mRadioInterface->getClock()->get();
 | 
						|
  mTransmitDeadlineClock = time;
 | 
						|
  mLastClockUpdateTime = time;
 | 
						|
  mLatencyUpdateTime = time;
 | 
						|
 | 
						|
  if (!mRadioInterface->start()) {
 | 
						|
    LOG(ALERT) << "Device failed to start";
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  /* Device is running - launch I/O threads */
 | 
						|
  mRxLowerLoopThread = new Thread(stackSize);
 | 
						|
  mTxLowerLoopThread = new Thread(stackSize);
 | 
						|
  mTxLowerLoopThread->start((void * (*)(void*))
 | 
						|
                            TxLowerLoopAdapter,(void*) this);
 | 
						|
  mRxLowerLoopThread->start((void * (*)(void*))
 | 
						|
                            RxLowerLoopAdapter,(void*) this);
 | 
						|
 | 
						|
  /* Launch uplink and downlink burst processing threads */
 | 
						|
  for (size_t i = 0; i < mChans; i++) {
 | 
						|
    TransceiverChannel *chan = new TransceiverChannel(this, i);
 | 
						|
    mRxServiceLoopThreads[i] = new Thread(stackSize);
 | 
						|
    mRxServiceLoopThreads[i]->start((void * (*)(void*))
 | 
						|
                            RxUpperLoopAdapter, (void*) chan);
 | 
						|
 | 
						|
    chan = new TransceiverChannel(this, i);
 | 
						|
    mTxPriorityQueueServiceLoopThreads[i] = new Thread(stackSize);
 | 
						|
    mTxPriorityQueueServiceLoopThreads[i]->start((void * (*)(void*))
 | 
						|
                            TxUpperLoopAdapter, (void*) chan);
 | 
						|
  }
 | 
						|
 | 
						|
  mForceClockInterface = true;
 | 
						|
  mOn = true;
 | 
						|
  return true;
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * Stop the transceiver
 | 
						|
 *
 | 
						|
 * Perform stopping by disabling receive streaming and issuing cancellation
 | 
						|
 * requests to running threads. Most threads will timeout and terminate once
 | 
						|
 * device is disabled, but the transmit loop may block waiting on the central
 | 
						|
 * UMTS clock. Explicitly signal the clock to make sure that the transmit loop
 | 
						|
 * makes it to the thread cancellation point.
 | 
						|
 */
 | 
						|
void Transceiver::stop()
 | 
						|
{
 | 
						|
  ScopedLock lock(mLock);
 | 
						|
 | 
						|
  if (!mOn)
 | 
						|
    return;
 | 
						|
 | 
						|
  LOG(NOTICE) << "Stopping the transceiver";
 | 
						|
  mTxLowerLoopThread->cancel();
 | 
						|
  mRxLowerLoopThread->cancel();
 | 
						|
  mTxLowerLoopThread->join();
 | 
						|
  mRxLowerLoopThread->join();
 | 
						|
  delete mTxLowerLoopThread;
 | 
						|
  delete mRxLowerLoopThread;
 | 
						|
 | 
						|
  for (size_t i = 0; i < mChans; i++) {
 | 
						|
    mRxServiceLoopThreads[i]->cancel();
 | 
						|
    mTxPriorityQueueServiceLoopThreads[i]->cancel();
 | 
						|
  }
 | 
						|
 | 
						|
  LOG(INFO) << "Stopping the device";
 | 
						|
  mRadioInterface->stop();
 | 
						|
 | 
						|
  for (size_t i = 0; i < mChans; i++) {
 | 
						|
    mRxServiceLoopThreads[i]->join();
 | 
						|
    mTxPriorityQueueServiceLoopThreads[i]->join();
 | 
						|
    delete mRxServiceLoopThreads[i];
 | 
						|
    delete mTxPriorityQueueServiceLoopThreads[i];
 | 
						|
 | 
						|
    mTxPriorityQueues[i].clear();
 | 
						|
  }
 | 
						|
 | 
						|
  mOn = false;
 | 
						|
  LOG(NOTICE) << "Transceiver stopped";
 | 
						|
}
 | 
						|
 | 
						|
void Transceiver::addRadioVector(size_t chan, BitVector &bits,
 | 
						|
                                 int RSSI, GSM::Time &wTime)
 | 
						|
{
 | 
						|
  signalVector *burst;
 | 
						|
  radioVector *radio_burst;
 | 
						|
 | 
						|
  if (chan >= mTxPriorityQueues.size()) {
 | 
						|
    LOG(ALERT) << "Invalid channel " << chan;
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (wTime.TN() > 7) {
 | 
						|
    LOG(ALERT) << "Received burst with invalid slot " << wTime.TN();
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  /* Use the number of bits as the EDGE burst indicator */
 | 
						|
  if (bits.size() == EDGE_BURST_NBITS)
 | 
						|
    burst = modulateEdgeBurst(bits, mSPSTx);
 | 
						|
  else
 | 
						|
    burst = modulateBurst(bits, 8 + (wTime.TN() % 4 == 0), mSPSTx);
 | 
						|
 | 
						|
  scaleVector(*burst, txFullScale * pow(10, -RSSI / 10));
 | 
						|
 | 
						|
  radio_burst = new radioVector(wTime, burst);
 | 
						|
 | 
						|
  mTxPriorityQueues[chan].write(radio_burst);
 | 
						|
}
 | 
						|
 | 
						|
void Transceiver::updateFillerTable(size_t chan, radioVector *burst)
 | 
						|
{
 | 
						|
  int TN, modFN;
 | 
						|
  TransceiverState *state = &mStates[chan];
 | 
						|
 | 
						|
  TN = burst->getTime().TN();
 | 
						|
  modFN = burst->getTime().FN() % state->fillerModulus[TN];
 | 
						|
 | 
						|
  delete state->fillerTable[modFN][TN];
 | 
						|
  state->fillerTable[modFN][TN] = burst->getVector();
 | 
						|
  burst->setVector(NULL);
 | 
						|
}
 | 
						|
 | 
						|
void Transceiver::pushRadioVector(GSM::Time &nowTime)
 | 
						|
{
 | 
						|
  int TN, modFN;
 | 
						|
  radioVector *burst;
 | 
						|
  TransceiverState *state;
 | 
						|
  std::vector<signalVector *> bursts(mChans);
 | 
						|
  std::vector<bool> zeros(mChans);
 | 
						|
  std::vector<bool> filler(mChans, true);
 | 
						|
 | 
						|
  for (size_t i = 0; i < mChans; i ++) {
 | 
						|
    state = &mStates[i];
 | 
						|
 | 
						|
    while ((burst = mTxPriorityQueues[i].getStaleBurst(nowTime))) {
 | 
						|
      LOGCHAN(i, DMAIN, NOTICE) << "dumping STALE burst in TRX->SDR interface ("
 | 
						|
                  << burst->getTime() <<" vs " << nowTime << "), retrans=" << state->mRetrans;
 | 
						|
      if (state->mRetrans)
 | 
						|
        updateFillerTable(i, burst);
 | 
						|
      delete burst;
 | 
						|
    }
 | 
						|
 | 
						|
    TN = nowTime.TN();
 | 
						|
    modFN = nowTime.FN() % state->fillerModulus[TN];
 | 
						|
 | 
						|
    bursts[i] = state->fillerTable[modFN][TN];
 | 
						|
    zeros[i] = state->chanType[TN] == NONE;
 | 
						|
 | 
						|
    if ((burst = mTxPriorityQueues[i].getCurrentBurst(nowTime))) {
 | 
						|
      bursts[i] = burst->getVector();
 | 
						|
 | 
						|
      if (state->mRetrans) {
 | 
						|
        updateFillerTable(i, burst);
 | 
						|
      } else {
 | 
						|
        burst->setVector(NULL);
 | 
						|
        filler[i] = false;
 | 
						|
      }
 | 
						|
 | 
						|
      delete burst;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  mRadioInterface->driveTransmitRadio(bursts, zeros);
 | 
						|
 | 
						|
  for (size_t i = 0; i < mChans; i++) {
 | 
						|
    if (!filler[i])
 | 
						|
      delete bursts[i];
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void Transceiver::setModulus(size_t timeslot, size_t chan)
 | 
						|
{
 | 
						|
  TransceiverState *state = &mStates[chan];
 | 
						|
 | 
						|
  switch (state->chanType[timeslot]) {
 | 
						|
  case NONE:
 | 
						|
  case I:
 | 
						|
  case II:
 | 
						|
  case III:
 | 
						|
  case FILL:
 | 
						|
    state->fillerModulus[timeslot] = 26;
 | 
						|
    break;
 | 
						|
  case IV:
 | 
						|
  case VI:
 | 
						|
  case V:
 | 
						|
    state->fillerModulus[timeslot] = 51;
 | 
						|
    break;
 | 
						|
    //case V:
 | 
						|
  case VII:
 | 
						|
    state->fillerModulus[timeslot] = 102;
 | 
						|
    break;
 | 
						|
  case XIII:
 | 
						|
    state->fillerModulus[timeslot] = 52;
 | 
						|
    break;
 | 
						|
  default:
 | 
						|
    break;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
CorrType Transceiver::expectedCorrType(GSM::Time currTime,
 | 
						|
                                       size_t chan)
 | 
						|
{
 | 
						|
  static int tchh_subslot[26] = { 0,1,0,1,0,1,0,1,0,1,0,1,0,0,1,0,1,0,1,0,1,0,1,0,1,1 };
 | 
						|
  static int sdcch4_subslot[102] = { 3,3,3,3,0,0,2,2,2,2,3,3,3,3,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,1,1,1,1,0,0,2,2,2,2,
 | 
						|
                                     3,3,3,3,0,0,0,0,0,0,1,1,1,1,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,1,1,1,1,0,0,2,2,2,2 };
 | 
						|
  static int sdcch8_subslot[102] = { 5,5,5,5,6,6,6,6,7,7,7,7,0,0,0,0,0,0,0,1,1,1,1,2,2,2,2,3,3,3,3,4,4,4,4,5,5,5,5,6,6,6,6,7,7,7,7,0,0,0,0,
 | 
						|
                                     1,1,1,1,2,2,2,2,3,3,3,3,0,0,0,0,0,0,0,1,1,1,1,2,2,2,2,3,3,3,3,4,4,4,4,5,5,5,5,6,6,6,6,7,7,7,7,4,4,4,4 };
 | 
						|
  TransceiverState *state = &mStates[chan];
 | 
						|
  unsigned burstTN = currTime.TN();
 | 
						|
  unsigned burstFN = currTime.FN();
 | 
						|
  int subch;
 | 
						|
 | 
						|
  switch (state->chanType[burstTN]) {
 | 
						|
  case NONE:
 | 
						|
    return OFF;
 | 
						|
    break;
 | 
						|
  case FILL:
 | 
						|
    return IDLE;
 | 
						|
    break;
 | 
						|
  case I:
 | 
						|
    // TODO: Are we expecting RACH on an IDLE frame?
 | 
						|
/*    if (burstFN % 26 == 25)
 | 
						|
      return IDLE;*/
 | 
						|
    if (mHandover[burstTN][0])
 | 
						|
      return RACH;
 | 
						|
    return TSC;
 | 
						|
    break;
 | 
						|
  case II:
 | 
						|
    subch = tchh_subslot[burstFN % 26];
 | 
						|
    if (subch == 1)
 | 
						|
      return IDLE;
 | 
						|
    if (mHandover[burstTN][0])
 | 
						|
      return RACH;
 | 
						|
    return TSC;
 | 
						|
    break;
 | 
						|
  case III:
 | 
						|
    subch = tchh_subslot[burstFN % 26];
 | 
						|
    if (mHandover[burstTN][subch])
 | 
						|
      return RACH;
 | 
						|
    return TSC;
 | 
						|
    break;
 | 
						|
  case IV:
 | 
						|
  case VI:
 | 
						|
    return mExtRACH ? EXT_RACH : RACH;
 | 
						|
    break;
 | 
						|
  case V: {
 | 
						|
    int mod51 = burstFN % 51;
 | 
						|
    if ((mod51 <= 36) && (mod51 >= 14))
 | 
						|
      return mExtRACH ? EXT_RACH : RACH;
 | 
						|
    else if ((mod51 == 4) || (mod51 == 5))
 | 
						|
      return mExtRACH ? EXT_RACH : RACH;
 | 
						|
    else if ((mod51 == 45) || (mod51 == 46))
 | 
						|
      return mExtRACH ? EXT_RACH : RACH;
 | 
						|
    else if (mHandover[burstTN][sdcch4_subslot[burstFN % 102]])
 | 
						|
      return RACH;
 | 
						|
    else
 | 
						|
      return TSC;
 | 
						|
    break;
 | 
						|
  }
 | 
						|
  case VII:
 | 
						|
    if ((burstFN % 51 <= 14) && (burstFN % 51 >= 12))
 | 
						|
      return IDLE;
 | 
						|
    else if (mHandover[burstTN][sdcch8_subslot[burstFN % 102]])
 | 
						|
      return RACH;
 | 
						|
    else
 | 
						|
      return TSC;
 | 
						|
    break;
 | 
						|
  case XIII: {
 | 
						|
    int mod52 = burstFN % 52;
 | 
						|
    if ((mod52 == 12) || (mod52 == 38))
 | 
						|
      return mExtRACH ? EXT_RACH : RACH;
 | 
						|
    else if ((mod52 == 25) || (mod52 == 51))
 | 
						|
      return IDLE;
 | 
						|
    else
 | 
						|
      return TSC;
 | 
						|
    break;
 | 
						|
  }
 | 
						|
  case LOOPBACK:
 | 
						|
    if ((burstFN % 51 <= 50) && (burstFN % 51 >=48))
 | 
						|
      return IDLE;
 | 
						|
    else
 | 
						|
      return TSC;
 | 
						|
    break;
 | 
						|
  default:
 | 
						|
    return OFF;
 | 
						|
    break;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void writeToFile(radioVector *radio_burst, size_t chan)
 | 
						|
{
 | 
						|
  GSM::Time time = radio_burst->getTime();
 | 
						|
  std::ostringstream fname;
 | 
						|
  fname << chan << "_" << time.FN() << "_" << time.TN() << ".fc";
 | 
						|
  std::ofstream outfile (fname.str().c_str(), std::ofstream::binary);
 | 
						|
  outfile.write((char*)radio_burst->getVector()->begin(), radio_burst->getVector()->size() * 2 * sizeof(float));
 | 
						|
  outfile.close();
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * Pull bursts from the FIFO and handle according to the slot
 | 
						|
 * and burst correlation type. Equalzation is currently disabled.
 | 
						|
 */
 | 
						|
bool Transceiver::pullRadioVector(size_t chan, struct trx_ul_burst_ind *bi)
 | 
						|
{
 | 
						|
  int rc;
 | 
						|
  struct estim_burst_params ebp;
 | 
						|
  float max = -1.0, avg = 0.0;
 | 
						|
  unsigned max_toa;
 | 
						|
  int max_i = -1;
 | 
						|
  signalVector *burst;
 | 
						|
  GSM::Time burstTime;
 | 
						|
  SoftVector *rxBurst;
 | 
						|
  TransceiverState *state = &mStates[chan];
 | 
						|
 | 
						|
  /* Blocking FIFO read */
 | 
						|
  radioVector *radio_burst = mReceiveFIFO[chan]->read();
 | 
						|
  if (!radio_burst)
 | 
						|
    return false;
 | 
						|
 | 
						|
  /* Set time and determine correlation type */
 | 
						|
  burstTime = radio_burst->getTime();
 | 
						|
  CorrType type = expectedCorrType(burstTime, chan);
 | 
						|
 | 
						|
  /* Enable 8-PSK burst detection if EDGE is enabled */
 | 
						|
  if (mEdge && (type == TSC))
 | 
						|
    type = EDGE;
 | 
						|
 | 
						|
  /* Debug: dump bursts to disk */
 | 
						|
  /* bits 0-7  - chan 0 timeslots
 | 
						|
   * bits 8-15 - chan 1 timeslots */
 | 
						|
  if (mWriteBurstToDiskMask & ((1<<bi->tn) << (8*chan)))
 | 
						|
    writeToFile(radio_burst, chan);
 | 
						|
 | 
						|
  /* No processing if the timeslot is off.
 | 
						|
   * Not even power level or noise calculation. */
 | 
						|
  if (type == OFF) {
 | 
						|
    delete radio_burst;
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  /* Initialize struct bi */
 | 
						|
  bi->nbits = 0;
 | 
						|
  bi->fn = burstTime.FN();
 | 
						|
  bi->tn = burstTime.TN();
 | 
						|
  bi->rssi = 0.0;
 | 
						|
  bi->toa = 0.0;
 | 
						|
  bi->noise = 0.0;
 | 
						|
  bi->idle = false;
 | 
						|
  bi->modulation = MODULATION_GMSK;
 | 
						|
  bi->tss = 0; /* TODO: we only support tss 0 right now */
 | 
						|
  bi->tsc = 0;
 | 
						|
  bi->ci = 0.0;
 | 
						|
 | 
						|
  /* Select the diversity channel with highest energy */
 | 
						|
  for (size_t i = 0; i < radio_burst->chans(); i++) {
 | 
						|
    float pow = energyDetect(*radio_burst->getVector(i), 20 * mSPSRx);
 | 
						|
    if (pow > max) {
 | 
						|
      max = pow;
 | 
						|
      max_i = i;
 | 
						|
    }
 | 
						|
    avg += pow;
 | 
						|
  }
 | 
						|
 | 
						|
  if (max_i < 0) {
 | 
						|
    LOG(ALERT) << "Received empty burst";
 | 
						|
    goto ret_idle;
 | 
						|
  }
 | 
						|
 | 
						|
  /* Average noise on diversity paths and update global levels */
 | 
						|
  burst = radio_burst->getVector(max_i);
 | 
						|
  avg = sqrt(avg / radio_burst->chans());
 | 
						|
 | 
						|
  if (type == IDLE) {
 | 
						|
    /* Update noise levels */
 | 
						|
    state->mNoises.insert(avg);
 | 
						|
    state->mNoiseLev = state->mNoises.avg();
 | 
						|
  }
 | 
						|
 | 
						|
  bi->rssi = 20.0 * log10(rxFullScale / avg) + rssiOffset;
 | 
						|
  bi->noise = 20.0 * log10(rxFullScale / state->mNoiseLev) + rssiOffset;
 | 
						|
 | 
						|
  if (type == IDLE)
 | 
						|
    goto ret_idle;
 | 
						|
 | 
						|
  max_toa = (type == RACH || type == EXT_RACH) ?
 | 
						|
            mMaxExpectedDelayAB : mMaxExpectedDelayNB;
 | 
						|
 | 
						|
  /* Detect normal or RACH bursts */
 | 
						|
  rc = detectAnyBurst(*burst, mTSC, BURST_THRESH, mSPSRx, type, max_toa, &ebp);
 | 
						|
  if (rc <= 0) {
 | 
						|
    if (rc == -SIGERR_CLIP)
 | 
						|
      LOG(WARNING) << "Clipping detected on received RACH or Normal Burst";
 | 
						|
    else if (rc != SIGERR_NONE)
 | 
						|
      LOG(WARNING) << "Unhandled RACH or Normal Burst detection error";
 | 
						|
    goto ret_idle;
 | 
						|
  }
 | 
						|
 | 
						|
  type = (CorrType) rc;
 | 
						|
  bi->toa = ebp.toa;
 | 
						|
  bi->tsc = ebp.tsc;
 | 
						|
  bi->ci = ebp.ci;
 | 
						|
  rxBurst = demodAnyBurst(*burst, mSPSRx, ebp.amp, ebp.toa, type);
 | 
						|
 | 
						|
  /* EDGE demodulator returns 444 (gSlotLen * 3) bits */
 | 
						|
  if (rxBurst->size() == EDGE_BURST_NBITS) {
 | 
						|
    bi->modulation = MODULATION_8PSK;
 | 
						|
    bi->nbits = EDGE_BURST_NBITS;
 | 
						|
  } else { /* size() here is actually gSlotLen + 8, due to guard periods */
 | 
						|
    bi->modulation = MODULATION_GMSK;
 | 
						|
    bi->nbits = gSlotLen;
 | 
						|
  }
 | 
						|
 | 
						|
  // Convert -1..+1 soft bits to 0..1 soft bits
 | 
						|
  vectorSlicer(bi->rx_burst, rxBurst->begin(), bi->nbits);
 | 
						|
 | 
						|
  delete rxBurst;
 | 
						|
  delete radio_burst;
 | 
						|
  return true;
 | 
						|
 | 
						|
ret_idle:
 | 
						|
  bi->idle = true;
 | 
						|
  delete radio_burst;
 | 
						|
  return false;
 | 
						|
}
 | 
						|
 | 
						|
void Transceiver::reset()
 | 
						|
{
 | 
						|
  for (size_t i = 0; i < mTxPriorityQueues.size(); i++)
 | 
						|
    mTxPriorityQueues[i].clear();
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
#define MAX_PACKET_LENGTH 100
 | 
						|
 | 
						|
/**
 | 
						|
 * Matches a buffer with a command.
 | 
						|
 * @param  buf    a buffer to look command in
 | 
						|
 * @param  cmd    a command to look in buffer
 | 
						|
 * @param  params pointer to arguments, or NULL
 | 
						|
 * @return        true if command matches, otherwise false
 | 
						|
 */
 | 
						|
static bool match_cmd(char *buf,
 | 
						|
  const char *cmd, char **params)
 | 
						|
{
 | 
						|
  size_t cmd_len = strlen(cmd);
 | 
						|
 | 
						|
  /* Check a command itself */
 | 
						|
  if (strncmp(buf, cmd, cmd_len))
 | 
						|
    return false;
 | 
						|
 | 
						|
  /* A command has arguments */
 | 
						|
  if (params != NULL) {
 | 
						|
    /* Make sure there is a space */
 | 
						|
    if (buf[cmd_len] != ' ')
 | 
						|
      return false;
 | 
						|
 | 
						|
    /* Update external pointer */
 | 
						|
    *params = buf + cmd_len + 1;
 | 
						|
  }
 | 
						|
 | 
						|
  return true;
 | 
						|
}
 | 
						|
 | 
						|
void Transceiver::driveControl(size_t chan)
 | 
						|
{
 | 
						|
  char buffer[MAX_PACKET_LENGTH + 1];
 | 
						|
  char response[MAX_PACKET_LENGTH + 1];
 | 
						|
  char *command, *params;
 | 
						|
  int msgLen;
 | 
						|
 | 
						|
  /* Attempt to read from control socket */
 | 
						|
  msgLen = read(mCtrlSockets[chan], buffer, MAX_PACKET_LENGTH);
 | 
						|
  if (msgLen <= 0) {
 | 
						|
    LOGCHAN(chan, DTRXCTRL, WARNING) << "mCtrlSockets read(" << mCtrlSockets[chan] << ") failed: " << msgLen;
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  /* Zero-terminate received string */
 | 
						|
  buffer[msgLen] = '\0';
 | 
						|
 | 
						|
  /* Verify a command signature */
 | 
						|
  if (strncmp(buffer, "CMD ", 4)) {
 | 
						|
    LOGC(DTRXCTRL, WARNING) << "bogus message on control interface";
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  /* Set command pointer */
 | 
						|
  command = buffer + 4;
 | 
						|
  LOGCHAN(chan, DTRXCTRL, INFO) << "command is '" << command << "'";
 | 
						|
 | 
						|
  if (match_cmd(command, "POWEROFF", NULL)) {
 | 
						|
    stop();
 | 
						|
    sprintf(response,"RSP POWEROFF 0");
 | 
						|
  } else if (match_cmd(command, "POWERON", NULL)) {
 | 
						|
    if (!start()) {
 | 
						|
      sprintf(response,"RSP POWERON 1");
 | 
						|
    } else {
 | 
						|
      sprintf(response,"RSP POWERON 0");
 | 
						|
      for (int i = 0; i < 8; i++) {
 | 
						|
        for (int j = 0; j < 8; j++)
 | 
						|
          mHandover[i][j] = false;
 | 
						|
      }
 | 
						|
    }
 | 
						|
  } else if (match_cmd(command, "HANDOVER", ¶ms)) {
 | 
						|
    unsigned ts = 0, ss = 0;
 | 
						|
    sscanf(params, "%u %u", &ts, &ss);
 | 
						|
    if (ts > 7 || ss > 7) {
 | 
						|
      sprintf(response, "RSP NOHANDOVER 1 %u %u", ts, ss);
 | 
						|
    } else {
 | 
						|
      mHandover[ts][ss] = true;
 | 
						|
      sprintf(response, "RSP HANDOVER 0 %u %u", ts, ss);
 | 
						|
    }
 | 
						|
  } else if (match_cmd(command, "NOHANDOVER", ¶ms)) {
 | 
						|
    unsigned ts = 0, ss = 0;
 | 
						|
    sscanf(params, "%u %u", &ts, &ss);
 | 
						|
    if (ts > 7 || ss > 7) {
 | 
						|
      sprintf(response, "RSP NOHANDOVER 1 %u %u", ts, ss);
 | 
						|
    } else {
 | 
						|
      mHandover[ts][ss] = false;
 | 
						|
      sprintf(response, "RSP NOHANDOVER 0 %u %u", ts, ss);
 | 
						|
    }
 | 
						|
  } else if (match_cmd(command, "SETMAXDLY", ¶ms)) {
 | 
						|
    //set expected maximum time-of-arrival
 | 
						|
    int maxDelay;
 | 
						|
    sscanf(params, "%d", &maxDelay);
 | 
						|
    mMaxExpectedDelayAB = maxDelay; // 1 GSM symbol is approx. 1 km
 | 
						|
    sprintf(response,"RSP SETMAXDLY 0 %d",maxDelay);
 | 
						|
  } else if (match_cmd(command, "SETMAXDLYNB", ¶ms)) {
 | 
						|
    //set expected maximum time-of-arrival
 | 
						|
    int maxDelay;
 | 
						|
    sscanf(params, "%d", &maxDelay);
 | 
						|
    mMaxExpectedDelayNB = maxDelay; // 1 GSM symbol is approx. 1 km
 | 
						|
    sprintf(response,"RSP SETMAXDLYNB 0 %d",maxDelay);
 | 
						|
  } else if (match_cmd(command, "SETRXGAIN", ¶ms)) {
 | 
						|
    //set expected maximum time-of-arrival
 | 
						|
    int newGain;
 | 
						|
    sscanf(params, "%d", &newGain);
 | 
						|
    newGain = mRadioInterface->setRxGain(newGain, chan);
 | 
						|
    sprintf(response,"RSP SETRXGAIN 0 %d",newGain);
 | 
						|
  } else if (match_cmd(command, "NOISELEV", NULL)) {
 | 
						|
    if (mOn) {
 | 
						|
      float lev = mStates[chan].mNoiseLev;
 | 
						|
      sprintf(response,"RSP NOISELEV 0 %d",
 | 
						|
              (int) round(20.0 * log10(rxFullScale / lev)));
 | 
						|
    }
 | 
						|
    else {
 | 
						|
      sprintf(response,"RSP NOISELEV 1  0");
 | 
						|
    }
 | 
						|
  } else if (match_cmd(command, "SETPOWER", ¶ms)) {
 | 
						|
    int power;
 | 
						|
    sscanf(params, "%d", &power);
 | 
						|
    power = mRadioInterface->setPowerAttenuation(power, chan);
 | 
						|
    mStates[chan].mPower = power;
 | 
						|
    sprintf(response, "RSP SETPOWER 0 %d", power);
 | 
						|
  } else if (match_cmd(command, "ADJPOWER", ¶ms)) {
 | 
						|
    int power, step;
 | 
						|
    sscanf(params, "%d", &step);
 | 
						|
    power = mStates[chan].mPower + step;
 | 
						|
    power = mRadioInterface->setPowerAttenuation(power, chan);
 | 
						|
    mStates[chan].mPower = power;
 | 
						|
    sprintf(response, "RSP ADJPOWER 0 %d", power);
 | 
						|
  } else if (match_cmd(command, "RXTUNE", ¶ms)) {
 | 
						|
    // tune receiver
 | 
						|
    int freqKhz;
 | 
						|
    sscanf(params, "%d", &freqKhz);
 | 
						|
    mRxFreq = freqKhz * 1e3;
 | 
						|
    if (!mRadioInterface->tuneRx(mRxFreq, chan)) {
 | 
						|
       LOGC(DTRXCTRL, ALERT) << "RX failed to tune";
 | 
						|
       sprintf(response,"RSP RXTUNE 1 %d",freqKhz);
 | 
						|
    }
 | 
						|
    else
 | 
						|
       sprintf(response,"RSP RXTUNE 0 %d",freqKhz);
 | 
						|
  } else if (match_cmd(command, "TXTUNE", ¶ms)) {
 | 
						|
    // tune txmtr
 | 
						|
    int freqKhz;
 | 
						|
    sscanf(params, "%d", &freqKhz);
 | 
						|
    mTxFreq = freqKhz * 1e3;
 | 
						|
    if (!mRadioInterface->tuneTx(mTxFreq, chan)) {
 | 
						|
       LOGC(DTRXCTRL, ALERT) << "TX failed to tune";
 | 
						|
       sprintf(response,"RSP TXTUNE 1 %d",freqKhz);
 | 
						|
    }
 | 
						|
    else
 | 
						|
       sprintf(response,"RSP TXTUNE 0 %d",freqKhz);
 | 
						|
  } else if (match_cmd(command, "SETTSC", ¶ms)) {
 | 
						|
    // set TSC
 | 
						|
    unsigned TSC;
 | 
						|
    sscanf(params, "%u", &TSC);
 | 
						|
    if (TSC > 7) {
 | 
						|
      sprintf(response, "RSP SETTSC 1 %d", TSC);
 | 
						|
    } else {
 | 
						|
      LOGC(DTRXCTRL, NOTICE) << "Changing TSC from " << mTSC << " to " << TSC;
 | 
						|
      mTSC = TSC;
 | 
						|
      sprintf(response,"RSP SETTSC 0 %d", TSC);
 | 
						|
    }
 | 
						|
  } else if (match_cmd(command, "SETSLOT", ¶ms)) {
 | 
						|
    // set slot type
 | 
						|
    int  corrCode;
 | 
						|
    int  timeslot;
 | 
						|
    sscanf(params, "%d %d", ×lot, &corrCode);
 | 
						|
    if ((timeslot < 0) || (timeslot > 7)) {
 | 
						|
      LOGC(DTRXCTRL, WARNING) << "bogus message on control interface";
 | 
						|
      sprintf(response,"RSP SETSLOT 1 %d %d",timeslot,corrCode);
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    mStates[chan].chanType[timeslot] = (ChannelCombination) corrCode;
 | 
						|
    setModulus(timeslot, chan);
 | 
						|
    sprintf(response,"RSP SETSLOT 0 %d %d",timeslot,corrCode);
 | 
						|
  } else if (match_cmd(command, "SETFORMAT", ¶ms)) {
 | 
						|
    // set TRXD protocol version
 | 
						|
    unsigned version_recv;
 | 
						|
    sscanf(params, "%u", &version_recv);
 | 
						|
    LOGC(DTRXCTRL, INFO) << "BTS requests TRXD version switch: " << version_recv;
 | 
						|
    if (version_recv > TRX_DATA_FORMAT_VER) {
 | 
						|
      LOGC(DTRXCTRL, INFO) << "rejecting TRXD version " << version_recv
 | 
						|
                           << "in favor of " <<  TRX_DATA_FORMAT_VER;
 | 
						|
      sprintf(response, "RSP SETFORMAT %u %u", TRX_DATA_FORMAT_VER, version_recv);
 | 
						|
    } else {
 | 
						|
      LOGC(DTRXCTRL, NOTICE) << "switching to TRXD version " << version_recv;
 | 
						|
      mVersionTRXD = version_recv;
 | 
						|
      sprintf(response, "RSP SETFORMAT %u %u", version_recv, version_recv);
 | 
						|
    }
 | 
						|
  } else if (match_cmd(command, "_SETBURSTTODISKMASK", ¶ms)) {
 | 
						|
    // debug command! may change or disapear without notice
 | 
						|
    // set a mask which bursts to dump to disk
 | 
						|
    int mask;
 | 
						|
    sscanf(params, "%d", &mask);
 | 
						|
    mWriteBurstToDiskMask = mask;
 | 
						|
    sprintf(response,"RSP _SETBURSTTODISKMASK 0 %d",mask);
 | 
						|
  } else {
 | 
						|
    LOGC(DTRXCTRL, WARNING) << "bogus command " << command << " on control interface.";
 | 
						|
    sprintf(response,"RSP ERR 1");
 | 
						|
  }
 | 
						|
 | 
						|
  LOGCHAN(chan, DTRXCTRL, INFO) << "response is '" << response << "'";
 | 
						|
  msgLen = write(mCtrlSockets[chan], response, strlen(response) + 1);
 | 
						|
  if (msgLen <= 0)
 | 
						|
    LOGCHAN(chan, DTRXCTRL, WARNING) << "mCtrlSockets write(" << mCtrlSockets[chan] << ") failed: " << msgLen;
 | 
						|
}
 | 
						|
 | 
						|
bool Transceiver::driveTxPriorityQueue(size_t chan)
 | 
						|
{
 | 
						|
  int msgLen;
 | 
						|
  int burstLen;
 | 
						|
  char buffer[EDGE_BURST_NBITS + 50];
 | 
						|
 | 
						|
  // check data socket
 | 
						|
  msgLen = read(mDataSockets[chan], buffer, sizeof(buffer));
 | 
						|
  if (msgLen <= 0) {
 | 
						|
    LOGCHAN(chan, DTRXCTRL, WARNING) << "mDataSockets read(" << mCtrlSockets[chan] << ") failed: " << msgLen;
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  if (msgLen == gSlotLen + 1 + 4 + 1) {
 | 
						|
    burstLen = gSlotLen;
 | 
						|
  } else if (msgLen == EDGE_BURST_NBITS + 1 + 4 + 1) {
 | 
						|
    if (mSPSTx != 4)
 | 
						|
      return false;
 | 
						|
 | 
						|
    burstLen = EDGE_BURST_NBITS;
 | 
						|
  } else {
 | 
						|
    LOG(ERR) << "badly formatted packet on GSM->TRX interface";
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  int timeSlot = (int) buffer[0];
 | 
						|
  uint32_t frameNum = 0;
 | 
						|
  for (int i = 0; i < 4; i++)
 | 
						|
    frameNum = (frameNum << 8) | (0x0ff & buffer[i+1]);
 | 
						|
 | 
						|
  LOG(DEBUG) << "rcvd. burst at: " << GSM::Time(frameNum,timeSlot);
 | 
						|
 | 
						|
  int RSSI = (int) buffer[5];
 | 
						|
  BitVector newBurst(burstLen);
 | 
						|
  BitVector::iterator itr = newBurst.begin();
 | 
						|
  char *bufferItr = buffer+6;
 | 
						|
  while (itr < newBurst.end())
 | 
						|
    *itr++ = *bufferItr++;
 | 
						|
 | 
						|
  GSM::Time currTime = GSM::Time(frameNum,timeSlot);
 | 
						|
 | 
						|
  addRadioVector(chan, newBurst, RSSI, currTime);
 | 
						|
 | 
						|
  return true;
 | 
						|
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
void Transceiver::driveReceiveRadio()
 | 
						|
{
 | 
						|
  int rc = mRadioInterface->driveReceiveRadio();
 | 
						|
  if (rc == 0) {
 | 
						|
    usleep(100000);
 | 
						|
  } else if (rc < 0) {
 | 
						|
    LOG(FATAL) << "radio Interface receive failed, requesting stop.";
 | 
						|
    osmo_signal_dispatch(SS_MAIN, S_MAIN_STOP_REQUIRED, NULL);
 | 
						|
  } else if (mForceClockInterface || mTransmitDeadlineClock > mLastClockUpdateTime + GSM::Time(216,0)) {
 | 
						|
    mForceClockInterface = false;
 | 
						|
    writeClockInterface();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void Transceiver::logRxBurst(size_t chan, const struct trx_ul_burst_ind *bi)
 | 
						|
{
 | 
						|
  std::ostringstream os;
 | 
						|
  for (size_t i=0; i < bi->nbits; i++) {
 | 
						|
    if (bi->rx_burst[i] > 0.5) os << "1";
 | 
						|
    else if (bi->rx_burst[i] > 0.25) os << "|";
 | 
						|
    else if (bi->rx_burst[i] > 0.0) os << "'";
 | 
						|
    else os << "-";
 | 
						|
  }
 | 
						|
 | 
						|
  LOG(DEBUG) << std::fixed << std::right
 | 
						|
    << " chan: "   << chan
 | 
						|
    << " time: "   << bi->tn << ":" << bi->fn
 | 
						|
    << " RSSI: "   << std::setw(5) << std::setprecision(1) << (bi->rssi - rssiOffset)
 | 
						|
                   << "dBFS/" << std::setw(6) << -bi->rssi << "dBm"
 | 
						|
    << " noise: "  << std::setw(5) << std::setprecision(1) << (bi->noise - rssiOffset)
 | 
						|
                   << "dBFS/" << std::setw(6) << -bi->noise << "dBm"
 | 
						|
    << " TOA: "    << std::setw(5) << std::setprecision(2) << bi->toa
 | 
						|
    << " C/I: "    << std::setw(5) << std::setprecision(2) << bi->ci << "dB"
 | 
						|
    << " bits: "   << os;
 | 
						|
}
 | 
						|
 | 
						|
void Transceiver::driveReceiveFIFO(size_t chan)
 | 
						|
{
 | 
						|
  struct trx_ul_burst_ind bi;
 | 
						|
 | 
						|
  if (!pullRadioVector(chan, &bi))
 | 
						|
        return;
 | 
						|
  if (!bi.idle)
 | 
						|
       logRxBurst(chan, &bi);
 | 
						|
 | 
						|
  switch (mVersionTRXD) {
 | 
						|
    case 0:
 | 
						|
      trxd_send_burst_ind_v0(chan, mDataSockets[chan], &bi);
 | 
						|
      break;
 | 
						|
    case 1:
 | 
						|
      trxd_send_burst_ind_v1(chan, mDataSockets[chan], &bi);
 | 
						|
      break;
 | 
						|
    default:
 | 
						|
      OSMO_ASSERT(false);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void Transceiver::driveTxFIFO()
 | 
						|
{
 | 
						|
 | 
						|
  /**
 | 
						|
      Features a carefully controlled latency mechanism, to
 | 
						|
      assure that transmit packets arrive at the radio/USRP
 | 
						|
      before they need to be transmitted.
 | 
						|
 | 
						|
      Deadline clock indicates the burst that needs to be
 | 
						|
      pushed into the FIFO right NOW.  If transmit queue does
 | 
						|
      not have a burst, stick in filler data.
 | 
						|
  */
 | 
						|
 | 
						|
 | 
						|
  RadioClock *radioClock = (mRadioInterface->getClock());
 | 
						|
 | 
						|
  if (mOn) {
 | 
						|
    //radioClock->wait(); // wait until clock updates
 | 
						|
    LOG(DEBUG) << "radio clock " << radioClock->get();
 | 
						|
    while (radioClock->get() + mTransmitLatency > mTransmitDeadlineClock) {
 | 
						|
      // if underrun, then we're not providing bursts to radio/USRP fast
 | 
						|
      //   enough.  Need to increase latency by one GSM frame.
 | 
						|
      if (mRadioInterface->getWindowType() == RadioDevice::TX_WINDOW_USRP1) {
 | 
						|
        if (mRadioInterface->isUnderrun()) {
 | 
						|
          // only update latency at the defined frame interval
 | 
						|
          if (radioClock->get() > mLatencyUpdateTime + GSM::Time(USB_LATENCY_INTRVL)) {
 | 
						|
            mTransmitLatency = mTransmitLatency + GSM::Time(1,0);
 | 
						|
            LOG(INFO) << "new latency: " << mTransmitLatency << " (underrun "
 | 
						|
                      << radioClock->get() << " vs " << mLatencyUpdateTime + GSM::Time(USB_LATENCY_INTRVL) << ")";
 | 
						|
            mLatencyUpdateTime = radioClock->get();
 | 
						|
          }
 | 
						|
        }
 | 
						|
        else {
 | 
						|
          // if underrun hasn't occurred in the last sec (216 frames) drop
 | 
						|
          //    transmit latency by a timeslot
 | 
						|
          if (mTransmitLatency > mRadioInterface->minLatency()) {
 | 
						|
              if (radioClock->get() > mLatencyUpdateTime + GSM::Time(216,0)) {
 | 
						|
              mTransmitLatency.decTN();
 | 
						|
              LOG(INFO) << "reduced latency: " << mTransmitLatency;
 | 
						|
              mLatencyUpdateTime = radioClock->get();
 | 
						|
            }
 | 
						|
          }
 | 
						|
        }
 | 
						|
      }
 | 
						|
      // time to push burst to transmit FIFO
 | 
						|
      pushRadioVector(mTransmitDeadlineClock);
 | 
						|
      mTransmitDeadlineClock.incTN();
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  radioClock->wait();
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
void Transceiver::writeClockInterface()
 | 
						|
{
 | 
						|
  int msgLen;
 | 
						|
  char command[50];
 | 
						|
  // FIXME -- This should be adaptive.
 | 
						|
  sprintf(command,"IND CLOCK %llu",(unsigned long long) (mTransmitDeadlineClock.FN()+2));
 | 
						|
 | 
						|
  LOG(INFO) << "ClockInterface: sending " << command;
 | 
						|
 | 
						|
  msgLen = write(mClockSocket, command, strlen(command) + 1);
 | 
						|
  if (msgLen <= 0)
 | 
						|
    LOG(WARNING) << "mClockSocket write(" << mClockSocket << ") failed: " << msgLen;
 | 
						|
 | 
						|
  mLastClockUpdateTime = mTransmitDeadlineClock;
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
void *RxUpperLoopAdapter(TransceiverChannel *chan)
 | 
						|
{
 | 
						|
  char thread_name[16];
 | 
						|
  Transceiver *trx = chan->trx;
 | 
						|
  size_t num = chan->num;
 | 
						|
 | 
						|
  delete chan;
 | 
						|
 | 
						|
  snprintf(thread_name, 16, "RxUpper%zu", num);
 | 
						|
  set_selfthread_name(thread_name);
 | 
						|
 | 
						|
  trx->setPriority(0.42);
 | 
						|
 | 
						|
  while (1) {
 | 
						|
    trx->driveReceiveFIFO(num);
 | 
						|
    pthread_testcancel();
 | 
						|
  }
 | 
						|
  return NULL;
 | 
						|
}
 | 
						|
 | 
						|
void *RxLowerLoopAdapter(Transceiver *transceiver)
 | 
						|
{
 | 
						|
  set_selfthread_name("RxLower");
 | 
						|
 | 
						|
  transceiver->setPriority(0.45);
 | 
						|
 | 
						|
  while (1) {
 | 
						|
    transceiver->driveReceiveRadio();
 | 
						|
    pthread_testcancel();
 | 
						|
  }
 | 
						|
  return NULL;
 | 
						|
}
 | 
						|
 | 
						|
void *TxLowerLoopAdapter(Transceiver *transceiver)
 | 
						|
{
 | 
						|
  set_selfthread_name("TxLower");
 | 
						|
 | 
						|
  transceiver->setPriority(0.44);
 | 
						|
 | 
						|
  while (1) {
 | 
						|
    transceiver->driveTxFIFO();
 | 
						|
    pthread_testcancel();
 | 
						|
  }
 | 
						|
  return NULL;
 | 
						|
}
 | 
						|
 | 
						|
void *ControlServiceLoopAdapter(TransceiverChannel *chan)
 | 
						|
{
 | 
						|
  char thread_name[16];
 | 
						|
  Transceiver *trx = chan->trx;
 | 
						|
  size_t num = chan->num;
 | 
						|
 | 
						|
  delete chan;
 | 
						|
 | 
						|
  snprintf(thread_name, 16, "CtrlService%zu", num);
 | 
						|
  set_selfthread_name(thread_name);
 | 
						|
 | 
						|
  while (1) {
 | 
						|
    trx->driveControl(num);
 | 
						|
    pthread_testcancel();
 | 
						|
  }
 | 
						|
  return NULL;
 | 
						|
}
 | 
						|
 | 
						|
void *TxUpperLoopAdapter(TransceiverChannel *chan)
 | 
						|
{
 | 
						|
  char thread_name[16];
 | 
						|
  Transceiver *trx = chan->trx;
 | 
						|
  size_t num = chan->num;
 | 
						|
 | 
						|
  delete chan;
 | 
						|
 | 
						|
  snprintf(thread_name, 16, "TxUpper%zu", num);
 | 
						|
  set_selfthread_name(thread_name);
 | 
						|
 | 
						|
  trx->setPriority(0.40);
 | 
						|
 | 
						|
  while (1) {
 | 
						|
    trx->driveTxPriorityQueue(num);
 | 
						|
    pthread_testcancel();
 | 
						|
  }
 | 
						|
  return NULL;
 | 
						|
}
 |