simulator.c 26.2 KB
Newer Older
1
/*
laurent's avatar
laurent committed
2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
* Licensed to the OpenAirInterface (OAI) Software Alliance under one or more
* contributor license agreements.  See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The OpenAirInterface Software Alliance licenses this file to You under
* the OAI Public License, Version 1.1  (the "License"); you may not use this file
* except in compliance with the License.
* You may obtain a copy of the License at
*
*      http://www.openairinterface.org/?page_id=698
*
* Author and copyright: Laurent Thomas, open-cells.com
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*-------------------------------------------------------------------------------
* For more information about the OpenAirInterface (OAI) Software Alliance:
*      contact@openairinterface.org
22 23
*/

laurent's avatar
laurent committed
24

laurent's avatar
laurent committed
25 26 27 28 29 30
/*
 * Open issues and limitations
 * The read and write should be called in the same thread, that is not new USRP UHD design
 * When the opposite side switch from passive reading to active R+Write, the synchro is not fully deterministic
 */

31 32 33 34 35 36 37 38 39 40 41 42 43 44 45
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/tcp.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <stdbool.h>
#include <errno.h>
#include <sys/epoll.h>
#include <string.h>

#include <common/utils/assertions.h>
#include <common/utils/LOG/log.h>
46 47
#include <common/utils/load_module_shlib.h>
#include <common/config/config_userapi.h>
48 49 50
#include "common_lib.h"
#include <openair1/PHY/defs_eNB.h>
#include "openair1/PHY/defs_UE.h"
51
#define CHANNELMOD_DYNAMICLOAD
52
#include <openair1/SIMULATION/TOOLS/sim.h>
laurent's avatar
laurent committed
53
#include <targets/ARCH/rfsimulator/rfsimulator.h>
54

55
#define PORT 4043 //default TCP port for this simulator
56
#define CirSize 307200 // 100ms is enough
57 58 59
#define sampleToByte(a,b) ((a)*(b)*sizeof(sample_t))
#define byteToSample(a,b) ((a)/(sizeof(sample_t)*(b)))

60 61 62 63 64
#define MAX_SIMULATION_CONNECTED_NODES 5
#define GENERATE_CHANNEL 10 //each frame in DL

// Fixme: datamodel, external variables in .h files, ...
#include <common/ran_context.h>
65

66 67
extern RAN_CONTEXT_t RC;
//
68

69 70 71 72 73
#define RFSIMU_SECTION    "rfsimulator"
#define RFSIMU_OPTIONS_PARAMNAME "options"
# define RFSIM_CONFIG_HELP_OPTIONS     " list of comma separated options to enable rf simulator functionalities. Available options: \n"\
  "        chanmod:   enable channel modelisation\n"\
  "        saviq:     enable saving written iqs to a file\n"
74 75 76 77
/*-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------*/
/*                                            configuration parameters for the rfsimulator device                                                                              */
/*   optname                     helpstr                     paramflags           XXXptr                               defXXXval                          type         numelt  */
/*-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------*/
78
#define RFSIMULATOR_PARAMS_DESC {\
79 80
    {"serveraddr",             "<ip address to connect to>\n",          0,         strptr:&(rfsimulator->ip),              defstrval:"127.0.0.1",           TYPE_STRING,    0 },\
    {"serverport",             "<port to connect to>\n",                0,         u16ptr:&(rfsimulator->port),            defuintval:PORT,                 TYPE_UINT16,    0 },\
81 82 83
    {RFSIMU_OPTIONS_PARAMNAME, RFSIM_CONFIG_HELP_OPTIONS,               0,         strlistptr:NULL,                        defstrlistval:NULL,              TYPE_STRINGLIST,0},\
    {"IQfile",                 "<file path to use when saving IQs>\n",  0,         strptr:&(saveF),                        defstrval:"/tmp/rfsimulator.iqs",TYPE_STRING,    0 },\
    {"modelname",              "<channel model name>\n",                0,         strptr:&(modelname),                    defstrval:"AWGN",                TYPE_STRING,    0 }\
84
  };
85 86

pthread_mutex_t Sockmutex;
87

88
typedef struct complex16 sample_t; // 2*16 bits complex number
89

90 91
typedef struct buffer_s {
  int conn_sock;
laurent's avatar
laurent committed
92
  openair0_timestamp lastReceivedTS;
93
  bool headerMode;
94
  bool trashingPacket;
95
  samplesBlockHeader_t th;
96 97 98 99
  char *transferPtr;
  uint64_t remainToTransfer;
  char *circularBufEnd;
  sample_t *circularBuf;
100
  channel_desc_t *channel_model;
101 102 103 104
} buffer_t;

typedef struct {
  int listen_sock, epollfd;
laurent's avatar
laurent committed
105
  openair0_timestamp nextTimestamp;
106
  openair0_timestamp lastWroteTS;
107 108
  uint64_t typeStamp;
  char *ip;
109
  uint16_t port;
110
  int saveIQfile;
111
  buffer_t buf[FD_SETSIZE];
112 113 114 115
  int rx_num_channels;
  int tx_num_channels;
  double sample_rate;
  double tx_bw;
116
  int channelmod;
117 118
} rfsimulator_state_t;

119

120 121 122 123 124
void allocCirBuf(rfsimulator_state_t *bridge, int sock) {
  buffer_t *ptr=&bridge->buf[sock];
  AssertFatal ( (ptr->circularBuf=(sample_t *) malloc(sampleToByte(CirSize,1))) != NULL, "");
  ptr->circularBufEnd=((char *)ptr->circularBuf)+sampleToByte(CirSize,1);
  ptr->conn_sock=sock;
125
  ptr->lastReceivedTS=0;
126
  ptr->headerMode=true;
127
  ptr->trashingPacket=false;
128
  ptr->transferPtr=(char *)&ptr->th;
129
  ptr->remainToTransfer=sizeof(samplesBlockHeader_t);
130
  int sendbuff=1000*1000*100;
131 132 133 134 135
  AssertFatal ( setsockopt(sock, SOL_SOCKET, SO_SNDBUF, &sendbuff, sizeof(sendbuff)) == 0, "");
  struct epoll_event ev= {0};
  ev.events = EPOLLIN | EPOLLRDHUP;
  ev.data.fd = sock;
  AssertFatal(epoll_ctl(bridge->epollfd, EPOLL_CTL_ADD,  sock, &ev) != -1, "");
136

137
  if ( bridge->channelmod > 0) {
138 139 140 141 142 143 144 145 146 147 148 149
    // create channel simulation model for this mode reception
    // snr_dB is pure global, coming from configuration paramter "-s"
    // Fixme: referenceSignalPower should come from the right place
    // but the datamodel is inconsistant
    // legacy: RC.ru[ru_id]->frame_parms.pdsch_config_common.referenceSignalPower
    // (must not come from ru[]->frame_parms as it doesn't belong to ru !!!)
    // Legacy sets it as:
    // ptr->channel_model->path_loss_dB = -132.24 + snr_dB - RC.ru[0]->frame_parms->pdsch_config_common.referenceSignalPower;
    // we use directly the paramter passed on the command line ("-s")
    // the value channel_model->path_loss_dB seems only a storage place (new_channel_desc_scm() only copy the passed value)
    // Legacy changes directlty the variable channel_model->path_loss_dB place to place
    // while calling new_channel_desc_scm() with path losses = 0
laurent's avatar
laurent committed
150
    static bool init_done=false;
Laurent's avatar
Laurent committed
151

laurent's avatar
laurent committed
152
    if (!init_done) {
153 154
	    uint64_t rand;
	    FILE *h=fopen("/dev/random","r");
Laurent's avatar
Laurent committed
155 156
            if ( 1 != fread(&rand,sizeof(rand),1,h) )
              LOG_W(HW, "Simulator can't read /dev/random\n");
157 158 159
	    fclose(h);
      randominit(rand);
      tableNor(rand);
laurent's avatar
laurent committed
160 161
      init_done=true;
    }
Laurent's avatar
Laurent committed
162

163
    ptr->channel_model=new_channel_desc_scm(bridge->tx_num_channels,bridge->rx_num_channels,
164
                                            bridge->channelmod,
165 166 167 168 169 170 171
                                            bridge->sample_rate,
                                            bridge->tx_bw,
                                            0.0, // forgetting_factor
                                            0, // maybe used for TA
                                            0); // path_loss in dB
    random_channel(ptr->channel_model,false);
  }
172 173 174 175 176 177
}

void removeCirBuf(rfsimulator_state_t *bridge, int sock) {
  AssertFatal( epoll_ctl(bridge->epollfd, EPOLL_CTL_DEL,  sock, NULL) != -1, "");
  close(sock);
  free(bridge->buf[sock].circularBuf);
178 179 180
  // Fixme: no free_channel_desc_scm(bridge->buf[sock].channel_model) implemented
  // a lot of mem leaks
  free(bridge->buf[sock].channel_model);
181 182 183 184 185 186 187 188 189
  memset(&bridge->buf[sock], 0, sizeof(buffer_t));
  bridge->buf[sock].conn_sock=-1;
}

void socketError(rfsimulator_state_t *bridge, int sock) {
  if (bridge->buf[sock].conn_sock!=-1) {
    LOG_W(HW,"Lost socket \n");
    removeCirBuf(bridge, sock);

190
    if (bridge->typeStamp==UE_MAGICDL_FDD)
191 192 193 194 195 196 197 198 199
      exit(1);
  }
}

#define helpTxt "\
\x1b[31m\
rfsimulator: error: you have to run one UE and one eNB\n\
For this, export RFSIMULATOR=enb (eNB case) or \n\
                 RFSIMULATOR=<an ip address> (UE case)\n\
200
                 or use rfsimulator.serveraddr configuration option\n\
201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219
\x1b[m"

enum  blocking_t {
  notBlocking,
  blocking
};

void setblocking(int sock, enum blocking_t active) {
  int opts;
  AssertFatal( (opts = fcntl(sock, F_GETFL)) >= 0,"");

  if (active==blocking)
    opts = opts & ~O_NONBLOCK;
  else
    opts = opts | O_NONBLOCK;

  AssertFatal(fcntl(sock, F_SETFL, opts) >= 0, "");
}

220
static bool flushInput(rfsimulator_state_t *t, int timeout, int nsamps);
221 222 223 224 225 226

void fullwrite(int fd, void *_buf, ssize_t count, rfsimulator_state_t *t) {
  if (t->saveIQfile != -1) {
    if (write(t->saveIQfile, _buf, count) != count )
      LOG_E(HW,"write in save iq file failed (%s)\n",strerror(errno));
  }
227

228 229
  AssertFatal(fd>=0 && _buf && count >0 && t,
              "Bug: %d/%p/%zd/%p", fd, _buf, count, t);
230
  char *buf = _buf;
231
  ssize_t l;
232 233 234 235 236 237 238 239 240 241
  setblocking(fd, notBlocking);

  while (count) {
    l = write(fd, buf, count);

    if (l <= 0) {
      if (errno==EINTR)
        continue;

      if(errno==EAGAIN) {
242 243
        // The opposite side is saturated
        // we read incoming sockets meawhile waiting
244 245
        //flushInput(t, 5);
        usleep(500);
246 247 248 249 250 251 252 253 254
        continue;
      } else
        return;
    }

    count -= l;
    buf += l;
  }
}
255

256
void rfsimulator_readconfig(rfsimulator_state_t *rfsimulator) {
257
  char *saveF=NULL;
258
  char *modelname=NULL;
259
  paramdef_t rfsimu_params[] = RFSIMULATOR_PARAMS_DESC;
260 261 262
  int p = config_paramidx_fromname(rfsimu_params,sizeof(rfsimu_params)/sizeof(paramdef_t), RFSIMU_OPTIONS_PARAMNAME) ;
  int ret = config_get( rfsimu_params,sizeof(rfsimu_params)/sizeof(paramdef_t),RFSIMU_SECTION);
  AssertFatal(ret >= 0, "configuration couldn't be performed");
263 264
  rfsimulator->saveIQfile = -1;

265
  for(int i=0; i<rfsimu_params[p].numelt ; i++) {
266 267 268 269 270 271 272 273 274 275
    if (strcmp(rfsimu_params[p].strlistptr[i],"saviq") == 0) {
      rfsimulator->saveIQfile=open(saveF,O_APPEND| O_CREAT|O_TRUNC | O_WRONLY, 0666);

      if ( rfsimulator->saveIQfile != -1 )
        LOG_I(HW,"rfsimulator: will save written IQ samples  in %s\n", saveF);
      else
        LOG_E(HW, "can't open %s for IQ saving (%s)\n", saveF, strerror(errno));

      break;
    } else if (strcmp(rfsimu_params[p].strlistptr[i],"chanmod") == 0) {
276 277
      init_channelmod();
      rfsimulator->channelmod=modelid_fromname(modelname);
278 279 280 281
    } else {
      fprintf(stderr,"Unknown rfsimulator option: %s\n",rfsimu_params[p].strlistptr[i]);
      exit(-1);
    }
282
  }
283

284 285
  /* for compatibility keep environment variable usage */
  if ( getenv("RFSIMULATOR") != NULL ) {
286
    rfsimulator->ip=getenv("RFSIMULATOR");
287
  }
288

289
  if ( strncasecmp(rfsimulator->ip,"enb",3) == 0 ||
290
       strncasecmp(rfsimulator->ip,"server",3) == 0 )
291 292 293 294
    rfsimulator->typeStamp = ENB_MAGICDL_FDD;
  else
    rfsimulator->typeStamp = UE_MAGICDL_FDD;
}
295 296 297

int server_start(openair0_device *device) {
  rfsimulator_state_t *t = (rfsimulator_state_t *) device->priv;
298
  t->typeStamp=ENB_MAGICDL_FDD;
299 300 301 302 303 304 305
  AssertFatal((t->listen_sock = socket(AF_INET, SOCK_STREAM, 0)) >= 0, "");
  int enable = 1;
  AssertFatal(setsockopt(t->listen_sock, SOL_SOCKET, SO_REUSEADDR, &enable, sizeof(int)) == 0, "");
  struct sockaddr_in addr = {
sin_family:
    AF_INET,
sin_port:
306
    htons(t->port),
307 308 309 310 311
sin_addr:
    { s_addr: INADDR_ANY }
  };
  bind(t->listen_sock, (struct sockaddr *)&addr, sizeof(addr));
  AssertFatal(listen(t->listen_sock, 5) == 0, "");
laurent's avatar
laurent committed
312
  struct epoll_event ev={0};
313 314 315 316 317 318 319 320
  ev.events = EPOLLIN;
  ev.data.fd = t->listen_sock;
  AssertFatal(epoll_ctl(t->epollfd, EPOLL_CTL_ADD,  t->listen_sock, &ev) != -1, "");
  return 0;
}

int start_ue(openair0_device *device) {
  rfsimulator_state_t *t = device->priv;
321
  t->typeStamp=UE_MAGICDL_FDD;
322 323 324 325 326 327
  int sock;
  AssertFatal((sock = socket(AF_INET, SOCK_STREAM, 0)) >= 0, "");
  struct sockaddr_in addr = {
sin_family:
    AF_INET,
sin_port:
328
    htons(t->port),
329 330 331 332 333 334 335
sin_addr:
    { s_addr: INADDR_ANY }
  };
  addr.sin_addr.s_addr = inet_addr(t->ip);
  bool connected=false;

  while(!connected) {
336
    LOG_I(HW,"rfsimulator: trying to connect to %s:%d\n", t->ip, t->port);
337 338 339 340 341 342 343 344 345 346 347 348 349 350 351

    if (connect(sock, (struct sockaddr *)&addr, sizeof(addr)) == 0) {
      LOG_I(HW,"rfsimulator: connection established\n");
      connected=true;
    }

    perror("rfsimulator");
    sleep(1);
  }

  setblocking(sock, notBlocking);
  allocCirBuf(t, sock);
  return 0;
}

352 353 354 355
static int rfsimulator_write_internal(rfsimulator_state_t *t, openair0_timestamp timestamp, void **samplesVoid, int nsamps, int nbAnt, int flags, bool alreadyLocked) {
  if (!alreadyLocked)
    pthread_mutex_lock(&Sockmutex);

356
  LOG_D(HW,"sending %d samples at time: %ld\n", nsamps, timestamp);
357 358

  for (int i=0; i<FD_SETSIZE; i++) {
laurent's avatar
laurent committed
359
    buffer_t *b=&t->buf[i];
360

laurent's avatar
laurent committed
361
    if (b->conn_sock >= 0 ) {
362
      samplesBlockHeader_t header= {t->typeStamp, nsamps, nbAnt, timestamp};
laurent's avatar
laurent committed
363
      fullwrite(b->conn_sock,&header, sizeof(header), t);
364 365 366 367 368 369 370 371 372
      sample_t tmpSamples[nsamps][nbAnt];

      for(int a=0; a<nbAnt; a++) {
        sample_t *in=(sample_t *)samplesVoid[a];

        for(int s=0; s<nsamps; s++)
          tmpSamples[s][a]=in[s];
      }

laurent's avatar
laurent committed
373 374 375
      if (b->conn_sock >= 0 ) {
        fullwrite(b->conn_sock, (void *)tmpSamples, sampleToByte(nsamps,nbAnt), t);
      }
376 377 378
    }
  }

379 380 381
  if ( t->lastWroteTS != 0 && abs((double)t->lastWroteTS-timestamp) > (double)CirSize)
    LOG_E(HW,"Discontinuous TX gap too large Tx:%lu, %lu\n", t->lastWroteTS, timestamp);

laurent's avatar
laurent committed
382
  if (t->lastWroteTS > timestamp+nsamps)
383
    LOG_E(HW,"Not supported to send Tx out of order (same in USRP) %lu, %lu\n",
384
              t->lastWroteTS, timestamp);
Laurent's avatar
Laurent committed
385

386 387 388 389 390
  t->lastWroteTS=timestamp+nsamps;

  if (!alreadyLocked)
    pthread_mutex_unlock(&Sockmutex);

391 392 393 394 395
  LOG_D(HW,"sent %d samples at time: %ld->%ld, energy in first antenna: %d\n",
        nsamps, timestamp, timestamp+nsamps, signal_energy(samplesVoid[0], nsamps) );
  return nsamps;
}

396
int rfsimulator_write(openair0_device *device, openair0_timestamp timestamp, void **samplesVoid, int nsamps, int nbAnt, int flags) {
397
  return rfsimulator_write_internal(device->priv, timestamp, samplesVoid, nsamps, nbAnt, flags, false);
398 399 400
}

static bool flushInput(rfsimulator_state_t *t, int timeout, int nsamps_for_initial) {
401 402 403
  // Process all incoming events on sockets
  // store the data in lists
  struct epoll_event events[FD_SETSIZE]= {0};
404
  int nfds = epoll_wait(t->epollfd, events, FD_SETSIZE, timeout);
405 406

  if ( nfds==-1 ) {
407
    if ( errno==EINTR || errno==EAGAIN ) {
408
      return false;
409
    } else
410 411 412 413 414 415 416 417 418 419 420
      AssertFatal(false,"error in epoll_wait\n");
  }

  for (int nbEv = 0; nbEv < nfds; ++nbEv) {
    int fd=events[nbEv].data.fd;

    if (events[nbEv].events & EPOLLIN && fd == t->listen_sock) {
      int conn_sock;
      AssertFatal( (conn_sock = accept(t->listen_sock,NULL,NULL)) != -1, "");
      setblocking(conn_sock, notBlocking);
      allocCirBuf(t, conn_sock);
421 422 423 424 425 426 427
      LOG_I(HW,"A ue connected, sending the current time\n");
      struct complex16 v= {0};
      void *samplesVoid[t->tx_num_channels];

      for ( int i=0; i < t->tx_num_channels; i++)
        samplesVoid[i]=(void *)&v;

428
      rfsimulator_write_internal(t, t->lastWroteTS > 1 ? t->lastWroteTS-1 : 0,
429
                                 samplesVoid, 1,
430
                                 t->tx_num_channels, 1, false);
431 432 433 434 435 436 437 438 439 440 441 442 443
    } else {
      if ( events[nbEv].events & (EPOLLHUP | EPOLLERR | EPOLLRDHUP) ) {
        socketError(t,fd);
        continue;
      }

      buffer_t *b=&t->buf[fd];

      if ( b->circularBuf == NULL ) {
        LOG_E(HW, "received data on not connected socket %d\n", events[nbEv].data.fd);
        continue;
      }

444
      ssize_t blockSz;
445 446 447 448 449 450 451 452

      if ( b->headerMode)
        blockSz=b->remainToTransfer;
      else
        blockSz= b->transferPtr+b->remainToTransfer < b->circularBufEnd ?
                 b->remainToTransfer :
                 b->circularBufEnd - 1 - b->transferPtr ;

453
      ssize_t sz=recv(fd, b->transferPtr, blockSz, MSG_DONTWAIT);
454 455 456 457

      if ( sz < 0 ) {
        if ( errno != EAGAIN ) {
          LOG_E(HW,"socket failed %s\n", strerror(errno));
458
          //abort();
459 460 461 462
        }
      } else if ( sz == 0 )
        continue;

463
      LOG_D(HW, "Socket rcv %zd bytes\n", sz);
464 465 466 467 468 469 470 471
      AssertFatal((b->remainToTransfer-=sz) >= 0, "");
      b->transferPtr+=sz;

      if (b->transferPtr==b->circularBufEnd - 1)
        b->transferPtr=(char *)b->circularBuf;

      // check the header and start block transfer
      if ( b->headerMode==true && b->remainToTransfer==0) {
472 473
        AssertFatal( (t->typeStamp == UE_MAGICDL_FDD  && b->th.magic==ENB_MAGICDL_FDD) ||
                     (t->typeStamp == ENB_MAGICDL_FDD && b->th.magic==UE_MAGICDL_FDD), "Socket Error in protocol");
474 475
        b->headerMode=false;

476 477 478 479 480 481 482 483 484 485
        if ( t->nextTimestamp == 0 ) { // First block in UE, resync with the eNB current TS
	  t->nextTimestamp=b->th.timestamp> nsamps_for_initial ?
	    b->th.timestamp -  nsamps_for_initial :
	    0;
	  b->lastReceivedTS=b->th.timestamp> nsamps_for_initial ?
	    b->th.timestamp :
	    nsamps_for_initial;
	  LOG_W(HW,"UE got first timestamp: starting at %lu\n",  t->nextTimestamp);
	  b->trashingPacket=true;
	} else if ( b->lastReceivedTS < b->th.timestamp) {
486
          int nbAnt= b->th.nbAnt;
Laurent's avatar
Laurent committed
487

laurent's avatar
laurent committed
488
          if ( b->th.timestamp-b->lastReceivedTS < CirSize ) {
489 490
          for (uint64_t index=b->lastReceivedTS; index < b->th.timestamp; index++ ) {
            for (int a=0; a < nbAnt; a++) {
491 492
              b->circularBuf[(index*nbAnt+a)%CirSize].r = 0;
              b->circularBuf[(index*nbAnt+a)%CirSize].i = 0;
493 494
            }
          }
laurent's avatar
laurent committed
495 496 497
          } else {
	    memset(b->circularBuf, 0, sampleToByte(CirSize,1));
	  }
Laurent's avatar
Laurent committed
498

499
          if (b->lastReceivedTS != 0 && b->th.timestamp-b->lastReceivedTS > 50 )
500
            LOG_W(HW,"UEsock: %d gap of: %ld in reception\n", fd, b->th.timestamp-b->lastReceivedTS );
Laurent's avatar
Laurent committed
501

502
          b->lastReceivedTS=b->th.timestamp;
503 504
	  
        } else if ( b->lastReceivedTS > b->th.timestamp && b->th.size == 1 ) {
505 506 507 508 509
          LOG_W(HW,"Received Rx/Tx synchro out of order\n");
          b->trashingPacket=true;
        } else if ( b->lastReceivedTS == b->th.timestamp ) {
          // normal case
        } else {
laurent's avatar
laurent committed
510 511
          LOG_E(HW, "received data in past: current is %lu, new reception: %lu!\n", b->lastReceivedTS, b->th.timestamp);
	  b->trashingPacket=true;
laurent's avatar
laurent committed
512
        }
513

514 515
        pthread_mutex_lock(&Sockmutex);

516 517
        if (t->lastWroteTS != 0 && ( abs((double)t->lastWroteTS-b->lastReceivedTS) > (double)CirSize))
          LOG_E(HW,"UEsock: %d Tx/Rx shift too large Tx:%lu, Rx:%lu\n", fd, t->lastWroteTS, b->lastReceivedTS);
518 519

        pthread_mutex_unlock(&Sockmutex);
Laurent's avatar
Laurent committed
520
        b->transferPtr=(char *)&b->circularBuf[(b->lastReceivedTS*b->th.nbAnt)%CirSize];
521 522 523 524
        b->remainToTransfer=sampleToByte(b->th.size, b->th.nbAnt);
      }

      if ( b->headerMode==false ) {
525 526
        if ( ! b->trashingPacket ) {
          b->lastReceivedTS=b->th.timestamp+b->th.size-byteToSample(b->remainToTransfer,b->th.nbAnt);
527
          LOG_D(HW,"UEsock: %d Set b->lastReceivedTS %ld\n", fd, b->lastReceivedTS);
528
        }
529

laurent's avatar
laurent committed
530
        if ( b->remainToTransfer==0) {
531
          LOG_D(HW,"UEsock: %d Completed block reception: %ld\n", fd, b->lastReceivedTS);
laurent's avatar
laurent committed
532 533
          b->headerMode=true;
          b->transferPtr=(char *)&b->th;
534
          b->remainToTransfer=sizeof(samplesBlockHeader_t);
laurent's avatar
laurent committed
535
          b->th.magic=-1;
536
          b->trashingPacket=false;
537 538 539 540 541 542 543 544 545 546
        }
      }
    }
  }

  return nfds>0;
}

int rfsimulator_read(openair0_device *device, openair0_timestamp *ptimestamp, void **samplesVoid, int nsamps, int nbAnt) {
  if (nbAnt != 1) {
547
    LOG_W(HW, "rfsimulator: only 1 antenna tested\n");
548 549 550 551 552 553 554 555 556 557 558 559 560 561
  }

  rfsimulator_state_t *t = device->priv;
  LOG_D(HW, "Enter rfsimulator_read, expect %d samples, will release at TS: %ld\n", nsamps, t->nextTimestamp+nsamps);
  // deliver data from received data
  // check if a UE is connected
  int first_sock;

  for (first_sock=0; first_sock<FD_SETSIZE; first_sock++)
    if (t->buf[first_sock].circularBuf != NULL )
      break;

  if ( first_sock ==  FD_SETSIZE ) {
    // no connected device (we are eNB, no UE is connected)
562 563
    if ( t->nextTimestamp == 0)
      LOG_W(HW,"No connected device, generating void samples...\n");
564

565
    if (!flushInput(t, 10,  nsamps)) {
566 567 568 569
      for (int x=0; x < nbAnt; x++)
        memset(samplesVoid[x],0,sampleToByte(nsamps,1));

      t->nextTimestamp+=nsamps;
570

571
      if ( ((t->nextTimestamp/nsamps)%100) == 0)
572
        LOG_W(HW,"No UE, Generated void samples for Rx: %ld\n", t->nextTimestamp);
573

574 575 576 577
      *ptimestamp = t->nextTimestamp-nsamps;
      return nsamps;
    }
  } else {
578 579 580 581 582 583
    pthread_mutex_lock(&Sockmutex);

    if ( t->nextTimestamp > 0 && t->lastWroteTS < t->nextTimestamp) {
      pthread_mutex_unlock(&Sockmutex);
      usleep(10000);
      pthread_mutex_lock(&Sockmutex);
laurent's avatar
laurent committed
584

585 586 587 588 589 590 591 592 593 594 595 596 597
      if ( t->lastWroteTS < t->nextTimestamp ) {
        // Assuming Tx is not done fully in another thread
        // We can never write is the past from the received time
        // So, the node perform receive but will never write these symbols
        // let's tell this to the opposite node
	// We send timestamp for nb samples required
	// assuming this should have been done earlier if a Tx would exist
        pthread_mutex_unlock(&Sockmutex);
        struct complex16 v= {0};
        void *samplesVoid[t->tx_num_channels];

        for ( int i=0; i < t->tx_num_channels; i++)
          samplesVoid[i]=(void *)&v;
laurent's avatar
laurent committed
598

599 600 601 602 603 604 605 606 607 608 609 610
	LOG_I(HW, "No samples Tx occured, so we send 1 sample to notify it: Tx:%lu, Rx:%lu\n",
	      t->lastWroteTS, t->nextTimestamp);
        rfsimulator_write_internal(t, t->nextTimestamp,
                                   samplesVoid, 1,
                                   t->tx_num_channels, 1, true);
      } else {
	pthread_mutex_unlock(&Sockmutex);
        LOG_W(HW, "trx_write came from another thread\n");
      }
    } else
      pthread_mutex_unlock(&Sockmutex);
    
611 612 613 614 615
    bool have_to_wait;

    do {
      have_to_wait=false;

616
      for ( int sock=0; sock<FD_SETSIZE; sock++) {
617 618
        buffer_t *b=&t->buf[sock];

laurent's avatar
laurent committed
619 620
        if ( b->circularBuf )
          if ( t->nextTimestamp+nsamps > b->lastReceivedTS ) {
laurent's avatar
laurent committed
621 622
            have_to_wait=true;
            break;
623
          }
laurent's avatar
laurent committed
624
      }
625 626 627 628 629 630

      if (have_to_wait)
        /*printf("Waiting on socket, current last ts: %ld, expected at least : %ld\n",
          ptr->lastReceivedTS,
          t->nextTimestamp+nsamps);
        */
631
        flushInput(t, 3, nsamps);
632 633 634 635 636 637 638
    } while (have_to_wait);
  }

  // Clear the output buffer
  for (int a=0; a<nbAnt; a++)
    memset(samplesVoid[a],0,sampleToByte(nsamps,1));

laurent's avatar
laurent committed
639
  // Add all input nodes signal in the output buffer
640 641 642
  for (int sock=0; sock<FD_SETSIZE; sock++) {
    buffer_t *ptr=&t->buf[sock];

laurent's avatar
laurent committed
643
    if ( ptr->circularBuf ) {
644 645 646 647 648 649 650
      bool reGenerateChannel=false;

      //fixme: when do we regenerate
      // it seems legacy behavior is: never in UL, each frame in DL
      if (reGenerateChannel)
        random_channel(ptr->channel_model,0);

651
      for (int a=0; a<nbAnt; a++) {
652
        if ( ptr->channel_model != NULL ) // apply a channel model
653 654 655 656
          rxAddInput( ptr->circularBuf, (struct complex16 *) samplesVoid[a],
                      a,
                      ptr->channel_model,
                      nsamps,
657 658
                      t->nextTimestamp,
                      CirSize
659 660 661
                    );
        else { // no channel modeling
          sample_t *out=(sample_t *)samplesVoid[a];
laurent's avatar
laurent committed
662
          const int64_t base=t->nextTimestamp*nbAnt+a;
Laurent's avatar
Laurent committed
663

664
          for ( int i=0; i < nsamps; i++ ) {
laurent's avatar
laurent committed
665 666 667
	    const int idx=(i*nbAnt+base)%CirSize;
            out[i].r+=ptr->circularBuf[idx].r;
            out[i].i+=ptr->circularBuf[idx].i;
668 669 670
          }
        } // end of no channel modeling
      } // end for a...
671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705
    }
  }

  *ptimestamp = t->nextTimestamp; // return the time of the first sample
  t->nextTimestamp+=nsamps;
  LOG_D(HW,"Rx to upper layer: %d from %ld to %ld, energy in first antenna %d\n",
        nsamps,
        *ptimestamp, t->nextTimestamp,
        signal_energy(samplesVoid[0], nsamps));
  return nsamps;
}
int rfsimulator_request(openair0_device *device, void *msg, ssize_t msg_len) {
  abort();
  return 0;
}
int rfsimulator_reply(openair0_device *device, void *msg, ssize_t msg_len) {
  abort();
  return 0;
}
int rfsimulator_get_stats(openair0_device *device) {
  return 0;
}
int rfsimulator_reset_stats(openair0_device *device) {
  return 0;
}
void rfsimulator_end(openair0_device *device) {}
int rfsimulator_stop(openair0_device *device) {
  return 0;
}
int rfsimulator_set_freq(openair0_device *device, openair0_config_t *openair0_cfg,int exmimo_dump_config) {
  return 0;
}
int rfsimulator_set_gains(openair0_device *device, openair0_config_t *openair0_cfg) {
  return 0;
}
WANG Tsu-Han's avatar
WANG Tsu-Han committed
706 707 708
int rfsimulator_write_init(openair0_device *device){
  return 0;
}
709 710
__attribute__((__visibility__("default")))
int device_init(openair0_device *device, openair0_config_t *openair0_cfg) {
711 712
  // to change the log level, use this on command line
  // --log_config.hw_log_level debug
713
  rfsimulator_state_t *rfsimulator = (rfsimulator_state_t *)calloc(sizeof(rfsimulator_state_t),1);
714
  rfsimulator_readconfig(rfsimulator);
715 716 717
  pthread_mutex_init(&Sockmutex, NULL);
  LOG_I(HW,"rfsimulator: running as %s\n", rfsimulator-> typeStamp == ENB_MAGICDL_FDD ? "(eg)NB" : "UE");
  device->trx_start_func       = rfsimulator->typeStamp == ENB_MAGICDL_FDD ?
718 719 720 721 722 723 724 725 726 727 728 729 730 731
                                 server_start :
                                 start_ue;
  device->trx_get_stats_func   = rfsimulator_get_stats;
  device->trx_reset_stats_func = rfsimulator_reset_stats;
  device->trx_end_func         = rfsimulator_end;
  device->trx_stop_func        = rfsimulator_stop;
  device->trx_set_freq_func    = rfsimulator_set_freq;
  device->trx_set_gains_func   = rfsimulator_set_gains;
  device->trx_write_func       = rfsimulator_write;
  device->trx_read_func      = rfsimulator_read;
  /* let's pretend to be a b2x0 */
  device->type = USRP_B200_DEV;
  device->openair0_cfg=&openair0_cfg[0];
  device->priv = rfsimulator;
WANG Tsu-Han's avatar
WANG Tsu-Han committed
732
  device->trx_write_init = rfsimulator_write_init;
733 734 735 736 737

  for (int i=0; i<FD_SETSIZE; i++)
    rfsimulator->buf[i].conn_sock=-1;

  AssertFatal((rfsimulator->epollfd = epoll_create1(0)) != -1,"");
738 739 740 741 742
  // initialize channel simulation
  rfsimulator->tx_num_channels=openair0_cfg->tx_num_channels;
  rfsimulator->rx_num_channels=openair0_cfg->rx_num_channels;
  rfsimulator->sample_rate=openair0_cfg->sample_rate;
  rfsimulator->tx_bw=openair0_cfg->tx_bw;
laurent's avatar
laurent committed
743
  //randominit(0);
744
  set_taus_seed(0);
745 746
  return 0;
}