rtl_tcp.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604
  1. /*
  2. * rtl-sdr, turns your Realtek RTL2832 based DVB dongle into a SDR receiver
  3. * Copyright (C) 2012 by Steve Markgraf <steve@steve-m.de>
  4. * Copyright (C) 2012-2013 by Hoernchen <la@tfc-server.de>
  5. *
  6. * This program is free software: you can redistribute it and/or modify
  7. * it under the terms of the GNU General Public License as published by
  8. * the Free Software Foundation, either version 2 of the License, or
  9. * (at your option) any later version.
  10. *
  11. * This program is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. * GNU General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU General Public License
  17. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  18. */
  19. #include <errno.h>
  20. #include <signal.h>
  21. #include <string.h>
  22. #include <stdio.h>
  23. #include <stdlib.h>
  24. #ifndef _WIN32
  25. #include <unistd.h>
  26. #include <arpa/inet.h>
  27. #include <sys/socket.h>
  28. #include <sys/types.h>
  29. #include <sys/socket.h>
  30. #include <sys/time.h>
  31. #include <netinet/in.h>
  32. #include <fcntl.h>
  33. #else
  34. #include <winsock2.h>
  35. #include "getopt/getopt.h"
  36. #endif
  37. #include <pthread.h>
  38. #include "rtl-sdr.h"
  39. #include "convenience/convenience.h"
  40. #ifdef _WIN32
  41. #pragma comment(lib, "ws2_32.lib")
  42. typedef int socklen_t;
  43. #else
  44. #define closesocket close
  45. #define SOCKADDR struct sockaddr
  46. #define SOCKET int
  47. #define SOCKET_ERROR -1
  48. #endif
  49. static SOCKET s;
  50. static pthread_t tcp_worker_thread;
  51. static pthread_t command_thread;
  52. static pthread_cond_t exit_cond;
  53. static pthread_mutex_t exit_cond_lock;
  54. static pthread_mutex_t ll_mutex;
  55. static pthread_cond_t cond;
  56. struct llist {
  57. char *data;
  58. size_t len;
  59. struct llist *next;
  60. };
  61. typedef struct { /* structure size must be multiple of 2 bytes */
  62. char magic[4];
  63. uint32_t tuner_type;
  64. uint32_t tuner_gain_count;
  65. } dongle_info_t;
  66. static rtlsdr_dev_t *dev = NULL;
  67. static int global_numq = 0;
  68. static struct llist *ll_buffers = 0;
  69. static int llbuf_num = 500;
  70. static volatile int do_exit = 0;
  71. void usage(void)
  72. {
  73. printf("rtl_tcp, an I/Q spectrum server for RTL2832 based DVB-T receivers\n\n"
  74. "Usage:\t[-a listen address]\n"
  75. "\t[-p listen port (default: 1234)]\n"
  76. "\t[-f frequency to tune to [Hz]]\n"
  77. "\t[-g gain (default: 0 for auto)]\n"
  78. "\t[-s samplerate in Hz (default: 2048000 Hz)]\n"
  79. "\t[-b number of buffers (default: 15, set by library)]\n"
  80. "\t[-n max number of linked list buffers to keep (default: 500)]\n"
  81. "\t[-d device index (default: 0)]\n"
  82. "\t[-P ppm_error (default: 0)]\n");
  83. exit(1);
  84. }
  85. #ifdef _WIN32
  86. int gettimeofday(struct timeval *tv, void* ignored)
  87. {
  88. FILETIME ft;
  89. unsigned __int64 tmp = 0;
  90. if (NULL != tv) {
  91. GetSystemTimeAsFileTime(&ft);
  92. tmp |= ft.dwHighDateTime;
  93. tmp <<= 32;
  94. tmp |= ft.dwLowDateTime;
  95. tmp /= 10;
  96. #ifdef _MSC_VER
  97. tmp -= 11644473600000000Ui64;
  98. #else
  99. tmp -= 11644473600000000ULL;
  100. #endif
  101. tv->tv_sec = (long)(tmp / 1000000UL);
  102. tv->tv_usec = (long)(tmp % 1000000UL);
  103. }
  104. return 0;
  105. }
  106. BOOL WINAPI
  107. sighandler(int signum)
  108. {
  109. if (CTRL_C_EVENT == signum) {
  110. fprintf(stderr, "Signal caught, exiting!\n");
  111. do_exit = 1;
  112. rtlsdr_cancel_async(dev);
  113. return TRUE;
  114. }
  115. return FALSE;
  116. }
  117. #else
  118. static void sighandler(int signum)
  119. {
  120. fprintf(stderr, "Signal caught, exiting!\n");
  121. rtlsdr_cancel_async(dev);
  122. do_exit = 1;
  123. }
  124. #endif
  125. void rtlsdr_callback(unsigned char *buf, uint32_t len, void *ctx)
  126. {
  127. if(!do_exit) {
  128. struct llist *rpt = (struct llist*)malloc(sizeof(struct llist));
  129. rpt->data = (char*)malloc(len);
  130. memcpy(rpt->data, buf, len);
  131. rpt->len = len;
  132. rpt->next = NULL;
  133. pthread_mutex_lock(&ll_mutex);
  134. if (ll_buffers == NULL) {
  135. ll_buffers = rpt;
  136. } else {
  137. struct llist *cur = ll_buffers;
  138. int num_queued = 0;
  139. while (cur->next != NULL) {
  140. cur = cur->next;
  141. num_queued++;
  142. }
  143. if(llbuf_num && llbuf_num == num_queued-2){
  144. struct llist *curelem;
  145. free(ll_buffers->data);
  146. curelem = ll_buffers->next;
  147. free(ll_buffers);
  148. ll_buffers = curelem;
  149. }
  150. cur->next = rpt;
  151. if (num_queued > global_numq)
  152. printf("ll+, now %d\n", num_queued);
  153. else if (num_queued < global_numq)
  154. printf("ll-, now %d\n", num_queued);
  155. global_numq = num_queued;
  156. }
  157. pthread_cond_signal(&cond);
  158. pthread_mutex_unlock(&ll_mutex);
  159. }
  160. }
  161. static void *tcp_worker(void *arg)
  162. {
  163. struct llist *curelem,*prev;
  164. int bytesleft,bytessent, index;
  165. struct timeval tv= {1,0};
  166. struct timespec ts;
  167. struct timeval tp;
  168. fd_set writefds;
  169. int r = 0;
  170. while(1) {
  171. if(do_exit)
  172. pthread_exit(0);
  173. pthread_mutex_lock(&ll_mutex);
  174. gettimeofday(&tp, NULL);
  175. ts.tv_sec = tp.tv_sec+5;
  176. ts.tv_nsec = tp.tv_usec * 1000;
  177. r = pthread_cond_timedwait(&cond, &ll_mutex, &ts);
  178. if(r == ETIMEDOUT) {
  179. pthread_mutex_unlock(&ll_mutex);
  180. printf("worker cond timeout\n");
  181. sighandler(0);
  182. pthread_exit(NULL);
  183. }
  184. curelem = ll_buffers;
  185. ll_buffers = 0;
  186. pthread_mutex_unlock(&ll_mutex);
  187. while(curelem != 0) {
  188. bytesleft = curelem->len;
  189. index = 0;
  190. bytessent = 0;
  191. while(bytesleft > 0) {
  192. FD_ZERO(&writefds);
  193. FD_SET(s, &writefds);
  194. tv.tv_sec = 1;
  195. tv.tv_usec = 0;
  196. r = select(s+1, NULL, &writefds, NULL, &tv);
  197. if(r) {
  198. bytessent = send(s, &curelem->data[index], bytesleft, 0);
  199. bytesleft -= bytessent;
  200. index += bytessent;
  201. }
  202. if(bytessent == SOCKET_ERROR || do_exit) {
  203. printf("worker socket bye\n");
  204. sighandler(0);
  205. pthread_exit(NULL);
  206. }
  207. }
  208. prev = curelem;
  209. curelem = curelem->next;
  210. free(prev->data);
  211. free(prev);
  212. }
  213. }
  214. }
  215. static int set_gain_by_index(rtlsdr_dev_t *_dev, unsigned int index)
  216. {
  217. int res = 0;
  218. int* gains;
  219. int count = rtlsdr_get_tuner_gains(_dev, NULL);
  220. if (count > 0 && (unsigned int)count > index) {
  221. gains = malloc(sizeof(int) * count);
  222. count = rtlsdr_get_tuner_gains(_dev, gains);
  223. res = rtlsdr_set_tuner_gain(_dev, gains[index]);
  224. free(gains);
  225. }
  226. return res;
  227. }
  228. #ifdef _WIN32
  229. #define __attribute__(x)
  230. #pragma pack(push, 1)
  231. #endif
  232. struct command{
  233. unsigned char cmd;
  234. unsigned int param;
  235. }__attribute__((packed));
  236. #ifdef _WIN32
  237. #pragma pack(pop)
  238. #endif
  239. static void *command_worker(void *arg)
  240. {
  241. int left, received = 0;
  242. fd_set readfds;
  243. struct command cmd={0, 0};
  244. struct timeval tv= {1, 0};
  245. int r = 0;
  246. uint32_t tmp;
  247. while(1) {
  248. left=sizeof(cmd);
  249. while(left >0) {
  250. FD_ZERO(&readfds);
  251. FD_SET(s, &readfds);
  252. tv.tv_sec = 1;
  253. tv.tv_usec = 0;
  254. r = select(s+1, &readfds, NULL, NULL, &tv);
  255. if(r) {
  256. received = recv(s, (char*)&cmd+(sizeof(cmd)-left), left, 0);
  257. left -= received;
  258. }
  259. if(received == SOCKET_ERROR || do_exit) {
  260. printf("comm recv bye\n");
  261. sighandler(0);
  262. pthread_exit(NULL);
  263. }
  264. }
  265. switch(cmd.cmd) {
  266. case 0x01:
  267. printf("set freq %d\n", ntohl(cmd.param));
  268. rtlsdr_set_center_freq(dev,ntohl(cmd.param));
  269. break;
  270. case 0x02:
  271. printf("set sample rate %d\n", ntohl(cmd.param));
  272. rtlsdr_set_sample_rate(dev, ntohl(cmd.param));
  273. break;
  274. case 0x03:
  275. printf("set gain mode %d\n", ntohl(cmd.param));
  276. rtlsdr_set_tuner_gain_mode(dev, ntohl(cmd.param));
  277. break;
  278. case 0x04:
  279. printf("set gain %d\n", ntohl(cmd.param));
  280. rtlsdr_set_tuner_gain(dev, ntohl(cmd.param));
  281. break;
  282. case 0x05:
  283. printf("set freq correction %d\n", ntohl(cmd.param));
  284. rtlsdr_set_freq_correction(dev, ntohl(cmd.param));
  285. break;
  286. case 0x06:
  287. tmp = ntohl(cmd.param);
  288. printf("set if stage %d gain %d\n", tmp >> 16, (short)(tmp & 0xffff));
  289. rtlsdr_set_tuner_if_gain(dev, tmp >> 16, (short)(tmp & 0xffff));
  290. break;
  291. case 0x07:
  292. printf("set test mode %d\n", ntohl(cmd.param));
  293. rtlsdr_set_testmode(dev, ntohl(cmd.param));
  294. break;
  295. case 0x08:
  296. printf("set agc mode %d\n", ntohl(cmd.param));
  297. rtlsdr_set_agc_mode(dev, ntohl(cmd.param));
  298. break;
  299. case 0x09:
  300. printf("set direct sampling %d\n", ntohl(cmd.param));
  301. rtlsdr_set_direct_sampling(dev, ntohl(cmd.param));
  302. break;
  303. case 0x0a:
  304. printf("set offset tuning %d\n", ntohl(cmd.param));
  305. rtlsdr_set_offset_tuning(dev, ntohl(cmd.param));
  306. break;
  307. case 0x0b:
  308. printf("set rtl xtal %d\n", ntohl(cmd.param));
  309. rtlsdr_set_xtal_freq(dev, ntohl(cmd.param), 0);
  310. break;
  311. case 0x0c:
  312. printf("set tuner xtal %d\n", ntohl(cmd.param));
  313. rtlsdr_set_xtal_freq(dev, 0, ntohl(cmd.param));
  314. break;
  315. case 0x0d:
  316. printf("set tuner gain by index %d\n", ntohl(cmd.param));
  317. set_gain_by_index(dev, ntohl(cmd.param));
  318. break;
  319. default:
  320. break;
  321. }
  322. cmd.cmd = 0xff;
  323. }
  324. }
  325. int main(int argc, char **argv)
  326. {
  327. int r, opt, i;
  328. char* addr = "127.0.0.1";
  329. int port = 1234;
  330. uint32_t frequency = 100000000, samp_rate = 2048000;
  331. struct sockaddr_in local, remote;
  332. uint32_t buf_num = 0;
  333. int dev_index = 0;
  334. int dev_given = 0;
  335. int gain = 0;
  336. int ppm_error = 0;
  337. struct llist *curelem,*prev;
  338. pthread_attr_t attr;
  339. void *status;
  340. struct timeval tv = {1,0};
  341. struct linger ling = {1,0};
  342. SOCKET listensocket;
  343. socklen_t rlen;
  344. fd_set readfds;
  345. u_long blockmode = 1;
  346. dongle_info_t dongle_info;
  347. #ifdef _WIN32
  348. WSADATA wsd;
  349. i = WSAStartup(MAKEWORD(2,2), &wsd);
  350. #else
  351. struct sigaction sigact, sigign;
  352. #endif
  353. while ((opt = getopt(argc, argv, "a:p:f:g:s:b:n:d:P:")) != -1) {
  354. switch (opt) {
  355. case 'd':
  356. dev_index = verbose_device_search(optarg);
  357. dev_given = 1;
  358. break;
  359. case 'f':
  360. frequency = (uint32_t)atofs(optarg);
  361. break;
  362. case 'g':
  363. gain = (int)(atof(optarg) * 10); /* tenths of a dB */
  364. break;
  365. case 's':
  366. samp_rate = (uint32_t)atofs(optarg);
  367. break;
  368. case 'a':
  369. addr = optarg;
  370. break;
  371. case 'p':
  372. port = atoi(optarg);
  373. break;
  374. case 'b':
  375. buf_num = atoi(optarg);
  376. break;
  377. case 'n':
  378. llbuf_num = atoi(optarg);
  379. break;
  380. case 'P':
  381. ppm_error = atoi(optarg);
  382. break;
  383. default:
  384. usage();
  385. break;
  386. }
  387. }
  388. if (argc < optind)
  389. usage();
  390. if (!dev_given) {
  391. dev_index = verbose_device_search("0");
  392. }
  393. if (dev_index < 0) {
  394. exit(1);
  395. }
  396. rtlsdr_open(&dev, (uint32_t)dev_index);
  397. if (NULL == dev) {
  398. fprintf(stderr, "Failed to open rtlsdr device #%d.\n", dev_index);
  399. exit(1);
  400. }
  401. #ifndef _WIN32
  402. sigact.sa_handler = sighandler;
  403. sigemptyset(&sigact.sa_mask);
  404. sigact.sa_flags = 0;
  405. sigign.sa_handler = SIG_IGN;
  406. sigaction(SIGINT, &sigact, NULL);
  407. sigaction(SIGTERM, &sigact, NULL);
  408. sigaction(SIGQUIT, &sigact, NULL);
  409. sigaction(SIGPIPE, &sigign, NULL);
  410. #else
  411. SetConsoleCtrlHandler( (PHANDLER_ROUTINE) sighandler, TRUE );
  412. #endif
  413. /* Set the tuner error */
  414. verbose_ppm_set(dev, ppm_error);
  415. /* Set the sample rate */
  416. r = rtlsdr_set_sample_rate(dev, samp_rate);
  417. if (r < 0)
  418. fprintf(stderr, "WARNING: Failed to set sample rate.\n");
  419. /* Set the frequency */
  420. r = rtlsdr_set_center_freq(dev, frequency);
  421. if (r < 0)
  422. fprintf(stderr, "WARNING: Failed to set center freq.\n");
  423. else
  424. fprintf(stderr, "Tuned to %i Hz.\n", frequency);
  425. if (0 == gain) {
  426. /* Enable automatic gain */
  427. r = rtlsdr_set_tuner_gain_mode(dev, 0);
  428. if (r < 0)
  429. fprintf(stderr, "WARNING: Failed to enable automatic gain.\n");
  430. } else {
  431. /* Enable manual gain */
  432. r = rtlsdr_set_tuner_gain_mode(dev, 1);
  433. if (r < 0)
  434. fprintf(stderr, "WARNING: Failed to enable manual gain.\n");
  435. /* Set the tuner gain */
  436. r = rtlsdr_set_tuner_gain(dev, gain);
  437. if (r < 0)
  438. fprintf(stderr, "WARNING: Failed to set tuner gain.\n");
  439. else
  440. fprintf(stderr, "Tuner gain set to %f dB.\n", gain/10.0);
  441. }
  442. /* Reset endpoint before we start reading from it (mandatory) */
  443. r = rtlsdr_reset_buffer(dev);
  444. if (r < 0)
  445. fprintf(stderr, "WARNING: Failed to reset buffers.\n");
  446. pthread_mutex_init(&exit_cond_lock, NULL);
  447. pthread_mutex_init(&ll_mutex, NULL);
  448. pthread_mutex_init(&exit_cond_lock, NULL);
  449. pthread_cond_init(&cond, NULL);
  450. pthread_cond_init(&exit_cond, NULL);
  451. memset(&local,0,sizeof(local));
  452. local.sin_family = AF_INET;
  453. local.sin_port = htons(port);
  454. local.sin_addr.s_addr = inet_addr(addr);
  455. listensocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
  456. r = 1;
  457. setsockopt(listensocket, SOL_SOCKET, SO_REUSEADDR, (char *)&r, sizeof(int));
  458. setsockopt(listensocket, SOL_SOCKET, SO_LINGER, (char *)&ling, sizeof(ling));
  459. bind(listensocket,(struct sockaddr *)&local,sizeof(local));
  460. #ifdef _WIN32
  461. ioctlsocket(listensocket, FIONBIO, &blockmode);
  462. #else
  463. r = fcntl(listensocket, F_GETFL, 0);
  464. r = fcntl(listensocket, F_SETFL, r | O_NONBLOCK);
  465. #endif
  466. while(1) {
  467. printf("listening...\n");
  468. printf("Use the device argument 'rtl_tcp=%s:%d' in OsmoSDR "
  469. "(gr-osmosdr) source\n"
  470. "to receive samples in GRC and control "
  471. "rtl_tcp parameters (frequency, gain, ...).\n",
  472. addr, port);
  473. listen(listensocket,1);
  474. while(1) {
  475. FD_ZERO(&readfds);
  476. FD_SET(listensocket, &readfds);
  477. tv.tv_sec = 1;
  478. tv.tv_usec = 0;
  479. r = select(listensocket+1, &readfds, NULL, NULL, &tv);
  480. if(do_exit) {
  481. goto out;
  482. } else if(r) {
  483. rlen = sizeof(remote);
  484. s = accept(listensocket,(struct sockaddr *)&remote, &rlen);
  485. break;
  486. }
  487. }
  488. setsockopt(s, SOL_SOCKET, SO_LINGER, (char *)&ling, sizeof(ling));
  489. printf("client accepted!\n");
  490. memset(&dongle_info, 0, sizeof(dongle_info));
  491. memcpy(&dongle_info.magic, "RTL0", 4);
  492. r = rtlsdr_get_tuner_type(dev);
  493. if (r >= 0)
  494. dongle_info.tuner_type = htonl(r);
  495. r = rtlsdr_get_tuner_gains(dev, NULL);
  496. if (r >= 0)
  497. dongle_info.tuner_gain_count = htonl(r);
  498. r = send(s, (const char *)&dongle_info, sizeof(dongle_info), 0);
  499. if (sizeof(dongle_info) != r)
  500. printf("failed to send dongle information\n");
  501. pthread_attr_init(&attr);
  502. pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_JOINABLE);
  503. r = pthread_create(&tcp_worker_thread, &attr, tcp_worker, NULL);
  504. r = pthread_create(&command_thread, &attr, command_worker, NULL);
  505. pthread_attr_destroy(&attr);
  506. r = rtlsdr_read_async(dev, rtlsdr_callback, NULL, buf_num, 0);
  507. pthread_join(tcp_worker_thread, &status);
  508. pthread_join(command_thread, &status);
  509. closesocket(s);
  510. printf("all threads dead..\n");
  511. curelem = ll_buffers;
  512. ll_buffers = 0;
  513. while(curelem != 0) {
  514. prev = curelem;
  515. curelem = curelem->next;
  516. free(prev->data);
  517. free(prev);
  518. }
  519. do_exit = 0;
  520. global_numq = 0;
  521. }
  522. out:
  523. rtlsdr_close(dev);
  524. closesocket(listensocket);
  525. closesocket(s);
  526. #ifdef _WIN32
  527. WSACleanup();
  528. #endif
  529. printf("bye!\n");
  530. return r >= 0 ? r : -r;
  531. }