-
Notifications
You must be signed in to change notification settings - Fork 46
/
Copy pathmain.c
983 lines (884 loc) · 35.3 KB
/
main.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
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
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
// Core of KA9Q radiod
// downconvert, filter, demodulate, multicast output
// Copyright 2017-2025, Phil Karn, KA9Q, [email protected]
#define _GNU_SOURCE 1
#include <assert.h>
#include <errno.h>
#include <limits.h>
#include <pthread.h>
#include <string.h>
#if defined(linux)
#include <bsd/string.h>
#endif
#include <math.h>
#include <complex.h>
#undef I
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <stdint.h>
#include <stdbool.h>
#include <locale.h>
#include <sys/time.h>
#include <sys/resource.h>
#include <signal.h>
#include <getopt.h>
#include <ctype.h>
#include <arpa/inet.h>
#include <iniparser/iniparser.h>
#include <net/if.h>
#include <sched.h>
#include <sysexits.h>
#include <fcntl.h>
#include <strings.h>
#include <dirent.h>
#include <sys/stat.h>
#include <dlfcn.h>
#include "conf.h"
#include "misc.h"
#include "multicast.h"
#include "rtp.h"
#include "radio.h"
#include "filter.h"
#include "status.h"
#include "config.h"
#include "avahi.h"
// Configuration constants & defaults
static char const DEFAULT_PRESET[] = "am";
static int const DEFAULT_FFTW_THREADS = 2;
static int const DEFAULT_IP_TOS = 48; // AF12 left shifted 2 bits
static int const DEFAULT_MCAST_TTL = 0; // Don't blast LANs with cheap Wifi!
static float const DEFAULT_BLOCKTIME = 20.0;
static int const DEFAULT_OVERLAP = 5;
static int const DEFAULT_UPDATE = 25; // 2 Hz for 20 ms blocktime (50 Hz frame rate)
static int const DEFAULT_LIFETIME = 20; // 20 sec for idle sessions tuned to 0 Hz
#define GLOBAL "global"
char const *Iface;
char const *Data;
char const *Preset = DEFAULT_PRESET;
char Preset_file[PATH_MAX];
char const *Config_file;
char Hostname[256]; // can't use sysconf(_SC_HOST_NAME_MAX) at file scope
int IP_tos = DEFAULT_IP_TOS;
int Mcast_ttl = DEFAULT_MCAST_TTL;
float Blocktime = DEFAULT_BLOCKTIME;
int Overlap = DEFAULT_OVERLAP;
static int Update = DEFAULT_UPDATE;
static int RTCP_enable = false;
static int SAP_enable = false;
// List of valid config keys in [global] section, for error checking
char const *Global_keys[] = {
"verbose",
"dns",
"fft-time-limit",
"fft-plan-level",
"iface",
"data",
"update",
"tos",
"ttl",
"blocktime",
"overlap",
"fft-threads",
"rtcp",
"sap",
"mode-file",
"presets-file",
"wisdom-file",
"hardware",
"status",
"preset",
"mode",
NULL
};
// Command line and environ params
const char *App_path;
int Verbose;
static char const *Locale = "en_US.UTF-8";
static dictionary *Configtable; // Configtable file descriptor for iniparser for main radiod config file
dictionary *Preset_table; // Table of presets, usually in /usr/local/share/ka9q-radio/modes.conf or presets.conf
volatile bool Stop_transfers = false; // Request to stop data transfers; how should this get set?
static int64_t Starttime; // System clock at timestamp 0, for RTCP
static pthread_t Status_thread;
struct sockaddr Metadata_dest_socket; // Dest of global metadata
static char const *Metadata_dest_string; // DNS name of default multicast group for status/commands
int Output_fd = -1; // Unconnected socket used for other hosts
struct channel Template;
// If a channel is tuned to 0 Hz and then not polled for this many seconds, destroy it
// Must be computed at run time because it depends on the block time
int Channel_idle_timeout; // = DEFAULT_LIFETIME * 1000 / Blocktime;
int Ctl_fd = -1; // File descriptor for receiving user commands
static char const *Name;
extern int N_worker_threads; // owned by filter.c
static void *Dl_handle;
static char Ttlmsg[100];
static bool Global_use_dns;
static int Nchans;
static void closedown(int);
static void verbosity(int);
static int loadconfig(char const *file);
static int setup_hardware(char const *sname);
static void *rtcp_send(void *);
void *process_section(void *p);
// In sdrplay.c (maybe someday)
int sdrplay_setup(struct frontend *,dictionary *,char const *);
int sdrplay_startup(struct frontend *);
double sdrplay_tune(struct frontend *,double);
// In rx888.c
int rx888_setup(struct frontend *,dictionary *,char const *);
int rx888_startup(struct frontend *);
double rx888_tune(struct frontend *,double);
float rx888_gain(struct frontend *, float);
float rx888_atten(struct frontend *,float);
// In airspy.c
int airspy_setup(struct frontend *,dictionary *,char const *);
int airspy_startup(struct frontend *);
double airspy_tune(struct frontend *,double);
// In airspyhf.c
int airspyhf_setup(struct frontend *,dictionary *,char const *);
int airspyhf_startup(struct frontend *);
double airspyhf_tune(struct frontend *,double);
// In funcube.c
int funcube_setup(struct frontend *,dictionary *,char const *);
int funcube_startup(struct frontend *);
double funcube_tune(struct frontend *,double);
// In rtlsdr.c:
int rtlsdr_setup(struct frontend *,dictionary *,char const *);
int rtlsdr_startup(struct frontend *);
double rtlsdr_tune(struct frontend *,double);
// In sig_gen.c:
int sig_gen_setup(struct frontend *,dictionary *,char const *);
int sig_gen_startup(struct frontend *);
double sig_gen_tune(struct frontend *,double);
// The main program sets up the demodulator parameter defaults,
// overwrites them with command-line arguments and/or state file settings,
// initializes the various local oscillators, pthread mutexes and conditions
// sets up multicast I/Q input and PCM audio output
// Sets up the input half of the pre-detection filter
// starts the RTP input and downconverter/filter threads
// sets the initial demodulation mode, which starts the demodulator thread
// catches signals and eventually becomes the user interface/display loop
int main(int argc,char *argv[]){
App_path = argv[0];
VERSION();
#ifndef NDEBUG
fprintf(stdout,"Assertion checking enabled, execution will be slower\n");
#endif
setlinebuf(stdout);
Starttime = gps_time_ns();
struct timespec start_realtime;
clock_gettime(CLOCK_MONOTONIC,&start_realtime);
// Set up program defaults
// Some can be overridden by command line args
{
// The display thread assumes en_US.UTF-8, or anything with a thousands grouping character
// Otherwise the cursor movements will be wrong
char const * const cp = getenv("LANG");
if(cp != NULL)
Locale = cp;
}
setlocale(LC_ALL,Locale); // Set either the hardwired default or the value of $LANG if it exists
int c;
while((c = getopt(argc,argv,"N:hvp:V")) != -1){
switch(c){
case 'V': // Already shown above
exit(EX_OK);
case 'p':
FFTW_plan_timelimit = strtod(optarg,NULL);
break;
case 'v':
Verbose++;
break;
case 'N':
Name = optarg;
break;
default: // including 'h'
fprintf(stdout,"Unknown command line option %c\n",c);
fprintf(stderr,"Usage: %s [-I] [-N name] [-h] [-p fftw_plan_time_limit] [-v [-v] ...] <CONFIG_FILE>\n", argv[0]);
exit(EX_USAGE);
}
}
// Graceful signal catch
signal(SIGPIPE,closedown);
signal(SIGINT,closedown);
signal(SIGKILL,closedown);
signal(SIGQUIT,closedown);
signal(SIGTERM,closedown);
signal(SIGPIPE,SIG_IGN);
signal(SIGUSR1,verbosity);
signal(SIGUSR2,verbosity);
if(argc <= optind){
fprintf(stdout,"Configtable file missing\n");
exit(EX_NOINPUT);
}
Config_file = argv[optind];
if(Name == NULL){
// Extract name from config file pathname
Name = argv[optind]; // Ah, just use whole thing
}
fprintf(stdout,"Loading config file %s\n",Config_file);
int const n = loadconfig(Config_file);
if(n < 0){
fprintf(stdout,"Can't load config file %s\n",Config_file);
exit(EX_NOINPUT);
}
fprintf(stdout,"%d total demodulators started\n",n);
// Measure CPU usage
int sleep_period = 60;
struct timespec last_realtime = start_realtime;
struct timespec last_cputime = {0};
while(true){
sleep(sleep_period);
if(Verbose){
struct timespec new_realtime;
clock_gettime(CLOCK_MONOTONIC,&new_realtime);
double total_real = new_realtime.tv_sec - start_realtime.tv_sec
+ 1e-9 * (new_realtime.tv_nsec - start_realtime.tv_nsec);
struct timespec new_cputime;
clock_gettime(CLOCK_PROCESS_CPUTIME_ID,&new_cputime);
double total_cpu = new_cputime.tv_sec + 1e-9 * (new_cputime.tv_nsec);
double total_percent = 100. * total_cpu / total_real;
double period_real = new_realtime.tv_sec - last_realtime.tv_sec
+ 1e-9 * (new_realtime.tv_nsec - last_realtime.tv_nsec);
double period_cpu = new_cputime.tv_sec - last_cputime.tv_sec
+ 1e-9 * (new_cputime.tv_nsec - last_cputime.tv_nsec);
double period_percent = 100. * period_cpu / period_real;
last_realtime = new_realtime;
last_cputime = new_cputime;
fprintf(stdout,"CPU usage: %.1lf%% since start, %.1lf%% in last %.1lf sec\n",
total_percent, period_percent,period_real);
}
}
exit(EX_OK); // Can't happen
}
// Load the radiod config file, e.g., /etc/radio/[email protected]
static int loadconfig(char const *file){
if(file == NULL || strlen(file) == 0)
return -1;
DIR *dirp = NULL;
struct stat statbuf;
if(stat(file,&statbuf) == 0 && (statbuf.st_mode & S_IFMT) == S_IFDIR){
// If the argument is a directory, read its contents
dirp = opendir(file);
} else {
// Otherwise append ".d" and see if that's a directory
char dname[PATH_MAX];
snprintf(dname,sizeof(dname),"%s.d",file);
if(stat(dname,&statbuf) == 0 && (statbuf.st_mode & S_IFMT) == S_IFDIR)
dirp = opendir(dname);
}
if(dirp != NULL){
// Read and sort list of foo.d/*.conf files, merge into temp file
int dfd = dirfd(dirp); // this gets used for openat() and fstatat() so don't close dirp right way
struct dirent *dp;
char *subfiles[100]; // List of subfiles
memset(subfiles,0,sizeof(subfiles));
int sf = 0;
while ((dp = readdir(dirp)) != NULL && sf < 100) {
// only consider regular files ending in .conf
if(strcmp(".conf",dp->d_name + strlen(dp->d_name) - 5) == 0
&& fstatat(dfd,dp->d_name,&statbuf,0) == 0
&& (statbuf.st_mode & S_IFMT) == S_IFREG)
subfiles[sf++] = strdup(dp->d_name);
}
// Don't close dirp just yet, would invalidate dfd
qsort(subfiles,sf,sizeof(subfiles[0]),(int (*)(void const *,void const *))strcmp);
// Create temporary copy of all files concatenated
char template[PATH_MAX];
strlcpy(template,"/tmp/radiod-configXXXXXXXX",sizeof(template));
int tfd = mkstemp(template);
FILE *tfp = fdopen(tfd,"rw+");
if(tfp != NULL){
// copy original file, if it exists, to temporary
if(stat(file,&statbuf) == 0 && (statbuf.st_mode & S_IFMT) == S_IFREG){
FILE *fp = fopen(file,"r");
if(fp != NULL){
fprintf(tfp,"# %s\n",file); // for debugging
int c;
while((c = getc(fp)) != EOF)
fputc(c,tfp);
fclose(fp);
fp = NULL;
}
}
// Concatenate the sub config files in order
for(int i=0; i < sf; i++){
int fd;
FILE *fp;
if((fd = openat(dfd,subfiles[i],O_RDONLY)) != -1 && (fp = fdopen(fd,"r")) != NULL){ // There's no "fopenat()"
fprintf(tfp,"# %s\n",subfiles[i]); // for debugging
int c;
while((c = getc(fp)) != EOF)
fputc(c,tfp);
fclose(fp);
fp = NULL;
}
}
fclose(tfp);
(void)closedir(dirp); // also invalidates dfd
Configtable = iniparser_load(template);
unlink(template);
} else {
fprintf(stdout,"Can't create temp config file %s: %s\n",template,strerror(errno));
Configtable = iniparser_load(file); // Just try to read the primary
}
} else {
Configtable = iniparser_load(file); // No subdirectory, just read the primary
}
if(Configtable == NULL)
return -1;
config_validate_section(stdout,Configtable,GLOBAL,Global_keys,Channel_keys);
// Set up template for all new channels
set_defaults(&Template);
Template.lifetime = DEFAULT_LIFETIME * 1000 / Blocktime; // If freq == 0, goes away 20 sec after last command
// Process [global] section applying to all demodulator blocks
Verbose = config_getint(Configtable,GLOBAL,"verbose",Verbose);
// Set up the hardware early, in case it fails
const char *hardware = config_getstring(Configtable,GLOBAL,"hardware",NULL);
if(hardware == NULL){
// 'hardware =' now required, no default
fprintf(stdout,"'hardware = [sectionname]' now required to specify front end configuration\n");
exit(EX_USAGE);
}
// Look for specified hardware section
{
int const nsect = iniparser_getnsec(Configtable);
int sect;
for(sect = 0; sect < nsect; sect++){
char const * const sname = iniparser_getsecname(Configtable,sect);
if(strcasecmp(sname,hardware) == 0){
if(setup_hardware(sname) != 0)
exit(EX_NOINPUT);
break;
}
}
if(sect == nsect){
fprintf(stdout,"no hardware section [%s] found, please create it\n",hardware);
exit(EX_USAGE);
}
}
FFTW_plan_timelimit = config_getdouble(Configtable,GLOBAL,"fft-time-limit",FFTW_plan_timelimit);
{
char const *cp = config_getstring(Configtable,GLOBAL,"fft-plan-level","patient");
if(strcasecmp(cp,"estimate") == 0){
FFTW_planning_level = FFTW_ESTIMATE;
} else if(strcasecmp(cp,"measure") == 0){
FFTW_planning_level = FFTW_MEASURE;
} else if(strcasecmp(cp,"patient") == 0){
FFTW_planning_level = FFTW_PATIENT;
} else if(strcasecmp(cp,"exhaustive") == 0){
FFTW_planning_level = FFTW_EXHAUSTIVE;
} else if(strcasecmp(cp,"wisdom-only") == 0){
FFTW_planning_level = FFTW_WISDOM_ONLY;
}
}
// Default multicast interface
{
// The area pointed to by returns from config_getstring() is freed and overwritten when the config dictionary is closed
char const *p = config_getstring(Configtable,GLOBAL,"iface",Iface);
if(p != NULL){
Iface = strdup(p);
Default_mcast_iface = Iface;
}
}
// Overrides in [global] of compiled-in defaults
{
char data_default[256];
snprintf(data_default,sizeof(data_default),"%s-pcm",Name);
Data = strdup(config_getstring(Configtable,GLOBAL,"data",data_default));
}
strlcpy(Template.output.dest_string,Data,sizeof(Template.output.dest_string));
Update = config_getint(Configtable,GLOBAL,"update",Update);
IP_tos = config_getint(Configtable,GLOBAL,"tos",IP_tos);
Mcast_ttl = config_getint(Configtable,GLOBAL,"ttl",Mcast_ttl);
// Set up default output stream file descriptor and socket
// There can be multiple senders to an output stream, so let avahi suppress the duplicate addresses
snprintf(Ttlmsg,sizeof(Ttlmsg),"TTL=%d",Mcast_ttl);
// Look quickly (2 tries max) to see if it's already in the DNS
Global_use_dns = config_getboolean(Configtable,GLOBAL,"dns",false);
{
uint32_t addr = 0;
if(!Global_use_dns || resolve_mcast(Data,&Template.output.dest_socket,DEFAULT_RTP_PORT,NULL,0,2) != 0)
addr = make_maddr(Data);
size_t slen = sizeof(Template.output.dest_socket);
avahi_start(Frontend.description != NULL ? Frontend.description : Name,
"_rtp._udp",
DEFAULT_RTP_PORT,
Data,
addr,
Ttlmsg,
addr != 0 ? &Template.output.dest_socket : NULL,
addr != 0 ? &slen : NULL);
// Status sent to same group, different port
memcpy(&Template.status.dest_socket,&Template.output.dest_socket,sizeof(Template.status.dest_socket));
struct sockaddr_in *sin = (struct sockaddr_in *)&Template.status.dest_socket;
sin->sin_port = htons(DEFAULT_STAT_PORT);
}
Output_fd = output_mcast(&Template.output.dest_socket,Iface,Mcast_ttl,IP_tos);
if(Output_fd < 0){
fprintf(stdout,"can't create output socket: %s\n",strerror(errno));
exit(EX_NOHOST); // let systemd restart us
}
join_group(Output_fd,&Template.output.dest_socket,Iface); // Work around snooping switch problem
Blocktime = fabs(config_getdouble(Configtable,GLOBAL,"blocktime",Blocktime));
Channel_idle_timeout = 20 * 1000 / Blocktime;
Overlap = abs(config_getint(Configtable,GLOBAL,"overlap",Overlap));
N_worker_threads = config_getint(Configtable,GLOBAL,"fft-threads",DEFAULT_FFTW_THREADS); // variable owned by filter.c
RTCP_enable = config_getboolean(Configtable,GLOBAL,"rtcp",RTCP_enable);
SAP_enable = config_getboolean(Configtable,GLOBAL,"sap",SAP_enable);
{
// Accept either keyword; "preset" is more descriptive than the old (but still accepted) "mode"
char const *p = config_getstring(Configtable,GLOBAL,"mode-file","presets.conf");
p = config_getstring(Configtable,GLOBAL,"presets-file",p);
dist_path(Preset_file,sizeof(Preset_file),p);
fprintf(stdout,"Loading presets file %s\n",Preset_file);
Preset_table = iniparser_load(Preset_file); // Kept open for duration of program
config_validate(stdout,Preset_table,Channel_keys,NULL);
if(Preset_table == NULL){
fprintf(stdout,"Can't load preset file %s\n",Preset_file);
exit(EX_UNAVAILABLE); // Can't really continue without fixing
}
}
{
char const *p = config_getstring(Configtable,GLOBAL,"wisdom-file",NULL);
if(p != NULL)
Wisdom_file = strdup(p);
}
// Set up status/command stream, global for all receiver channels
// Form default status dns name
gethostname(Hostname,sizeof(Hostname));
// Edit off .domain, .local, etc
{
char *cp = strchr(Hostname,'.');
if(cp != NULL)
*cp = '\0';
}
char default_status[strlen(Hostname) + strlen(Name) + 20]; // Enough room for snprintf
snprintf(default_status,sizeof(default_status),"%s-%s.local",Hostname,Name);
Metadata_dest_string = strdup(config_getstring(Configtable,GLOBAL,"status",default_status)); // Status/command target for all demodulators
if(0 == strcmp(Metadata_dest_string,Data)){
fprintf(stdout,"Duplicate status/data stream names: data=%s, status=%s\n",Data,Metadata_dest_string);
exit(EX_USAGE);
}
// Look quickly (2 tries max) to see if it's already in the DNS
{
uint32_t addr = 0;
if(!Global_use_dns || resolve_mcast(Metadata_dest_string,&Metadata_dest_socket,DEFAULT_STAT_PORT,NULL,0,2) != 0)
addr = make_maddr(Metadata_dest_string);
// If dns name already exists in the DNS, advertise the service record but not an address record
size_t slen = sizeof(Metadata_dest_socket);
avahi_start(Frontend.description != NULL ? Frontend.description : Name,"_ka9q-ctl._udp",DEFAULT_STAT_PORT,
Metadata_dest_string,addr,Ttlmsg,
addr != 0 ? &Metadata_dest_socket : NULL,
addr != 0 ? &slen : NULL);
}
// either resolve_mcast() or avahi_start() has resolved the target DNS name into Metadata_dest_socket and inserted the port number
join_group(Output_fd,&Metadata_dest_socket,Iface);
// Same remote socket as status
Ctl_fd = listen_mcast(&Metadata_dest_socket,Iface);
if(Ctl_fd < 0){
fprintf(stdout,"can't listen for commands from %s: %s\n",Metadata_dest_string,strerror(errno));
exit(EX_NOHOST);
}
ASSERT_ZEROED(&Status_thread,sizeof Status_thread);
if(Ctl_fd >= 3)
pthread_create(&Status_thread,NULL,radio_status,NULL);
// Preset/mode must be specified to create a dynamic channel
// (Trying to switch from term "mode" to term "preset" as more descriptive)
char const * p = config_getstring(Configtable,GLOBAL,"preset","am"); // Hopefully "am" is defined in presets.conf
char const * preset = config_getstring(Configtable,GLOBAL,"mode",p); // Must be specified to create a dynamic channel
if(preset != NULL){
if(loadpreset(&Template,Preset_table,preset) != 0)
fprintf(stdout,"warning: loadpreset(%s,%s) in [global]\n",Preset_file,preset);
strlcpy(Template.preset,preset,sizeof(Template.preset));
loadpreset(&Template,Configtable,GLOBAL); // Overwrite with other entries from this section, without overwriting those
} else {
fprintf(stdout,"No default mode for template\n");
}
// Process individual demodulator sections in parallel for speed
int const nsect = iniparser_getnsec(Configtable);
pthread_t startup_threads[nsect];
memset(startup_threads,0,sizeof startup_threads); // Apparently necessary to prevent segfaults on pthread_join()
for(int sect = 0; sect < nsect; sect++){
char const * const sname = iniparser_getsecname(Configtable,sect);
if(strcasecmp(sname,GLOBAL) == 0)
continue; // Already processed above
if(strcasecmp(sname,hardware) == 0)
continue; // Already processed as a hardware section (possibly without device=)
if(config_getstring(Configtable,sname,"device",NULL) != NULL)
continue; // It's a front end configuration, ignore
if(config_getboolean(Configtable,sname,"disable",false))
continue; // section is disabled
ASSERT_ZEROED(&startup_threads[sect],sizeof startup_threads[sect]);
pthread_create(&startup_threads[sect],NULL,process_section,(void *)sname);
}
// Wait for them all to start
for(int sect = 0; sect < nsect; sect++){
pthread_join(startup_threads[sect],NULL);
#if 0
printf("startup thread %s joined\n",iniparser_getsecname(Configtable,sect));
#endif
}
iniparser_freedict(Configtable);
Configtable = NULL;
return Nchans;
}
void *process_section(void *p){
char const *sname = (char *)p;
if(sname == NULL)
return NULL;
config_validate_section(stdout,Configtable,sname,Channel_keys,NULL);
// fall back to setting in [global] if parameter not specified in individual section
// Set parameters even when unused for the current demodulator in case the demod is changed later
char const * preset = config2_getstring(Configtable,Configtable,GLOBAL,sname,"mode",NULL);
preset = config2_getstring(Configtable,Configtable,GLOBAL,sname,"preset",preset);
if(preset == NULL || strlen(preset) == 0)
fprintf(stdout,"[%s] preset/mode not specified, all parameters must be explicitly set\n",sname);
// Override [global] settings with section settings
char const *data = config_getstring(Configtable,sname,"data",Data);
// Override global defaults
char const *iface = config_getstring(Configtable,sname,"iface",Iface);
// data stream is shared by all channels in this section
// Now also used for per-channel status/control, with different port number
struct sockaddr data_dest_socket;
struct sockaddr metadata_dest_socket;
memset(&data_dest_socket,0,sizeof(data_dest_socket));
memset(&metadata_dest_socket,0,sizeof(metadata_dest_socket));
// There can be multiple senders to an output stream, so let avahi suppress the duplicate addresses
{
// Look quickly (2 tries max) to see if it's already in the DNS. Otherwise make a multicast address.
uint32_t addr = 0;
bool use_dns = config_getboolean(Configtable,sname,"dns",Global_use_dns);
if(!use_dns || resolve_mcast(data,&data_dest_socket,DEFAULT_RTP_PORT,NULL,0,2) != 0)
// Hash name string to make IP multicast address in 239.x.x.x range
addr = make_maddr(data);
char const *cp = config_getstring(Configtable,sname,"encoding","s16be");
bool is_opus = strcasecmp(cp,"opus") == 0 ? true : false;
size_t slen = sizeof(data_dest_socket);
// there may be several hosts with the same section names
// prepend the host name to the service name
char service_name[512] = {0};
snprintf(service_name, sizeof service_name, "%s %s", Hostname, sname);
avahi_start(service_name,
is_opus ? "_opus._udp" : "_rtp._udp",
DEFAULT_RTP_PORT,
data,addr,Ttlmsg,
addr != 0 ? &data_dest_socket : NULL,
addr != 0 ? &slen : NULL);
// metadata for this stream is same except for port number
memcpy(&metadata_dest_socket,&data_dest_socket,sizeof(metadata_dest_socket));
struct sockaddr_in *sin = (struct sockaddr_in *)&metadata_dest_socket;
sin->sin_port = htons(DEFAULT_STAT_PORT);
}
join_group(Output_fd,&data_dest_socket,iface);
// No need to also join group for status socket, since the IP addresses are the same
// Process frequency/frequencies
// We need to do this first to ensure the resulting SSRCs are unique
// To work around iniparser's limited line length, we look for multiple keywords
// "freq", "freq0", "freq1", etc, up to "freq9"
int nchans = 0;
for(int ff = -1; ff < 10; ff++){
char fname[10];
if(ff == -1)
snprintf(fname,sizeof(fname),"freq");
else
snprintf(fname,sizeof(fname),"freq%d",ff);
char const * const frequencies = config_getstring(Configtable,sname,fname,NULL);
if(frequencies == NULL)
continue; // none with this prefix; look for more
// Parse the frequency list(s)
char *freq_list = strdup(frequencies); // Need writeable copy for strtok
char *saveptr = NULL;
for(char const *tok = strtok_r(freq_list," \t",&saveptr);
tok != NULL;
tok = strtok_r(NULL," \t",&saveptr)){
double const f = parse_frequency(tok,true);
if(f < 0){
fprintf(stdout,"[%s] can't parse frequency %s\n",sname,tok);
continue;
}
uint32_t ssrc = 0;
// Generate default ssrc from frequency string
for(char const *cp = tok ; cp != NULL && *cp != '\0' ; cp++){
if(isdigit(*cp)){
ssrc *= 10;
ssrc += *cp - '0';
}
}
ssrc = config_getint(Configtable,sname,"ssrc",ssrc); // Explicitly set?
if(ssrc == 0)
continue; // Reserved ssrc
struct channel *chan = NULL;
// Try to create it, incrementing in case of collision
int const max_collisions = 100;
for(int i=0; i < max_collisions; i++){
chan = create_chan(ssrc+i);
if(chan != NULL){
ssrc += i;
break;
}
}
if(chan == NULL){
fprintf(stdout,"Can't allocate requested ssrc %u-%u\n",ssrc,ssrc + max_collisions);
continue;
}
// Parameter priority, from high to low:
// 1. this section
// 2. the preset database entry, if specified
// 3. the [global] section
// 4. compiled-in defaults to keep things from blowing up
set_defaults(chan);
loadpreset(chan,Configtable,GLOBAL);
if(preset != NULL && loadpreset(chan,Preset_table,preset) != 0)
fprintf(stdout,"[%s] loadpreset(%s,%s) failed; compiled-in defaults and local settings used\n",sname,Preset_file,preset);
strlcpy(chan->preset,preset,sizeof(chan->preset));
loadpreset(chan,Configtable,sname);
// Set up output stream (data + status)
// Data multicast group has already been joined
memcpy(&chan->output.dest_socket,&data_dest_socket,sizeof(chan->output.dest_socket));
strlcpy(chan->output.dest_string,data,sizeof(chan->output.dest_string));
memcpy(&chan->status.dest_socket,&metadata_dest_socket,sizeof(chan->status.dest_socket));
chan->output.rtp.type = pt_from_info(chan->output.samprate,chan->output.channels,chan->output.encoding);
// Time to start it -- ssrc is stashed by create_chan()
set_freq(chan,f);
start_demod(chan);
nchans++;
Nchans++;
if(SAP_enable){
// Highly experimental, off by default
char sap_dest[] = "224.2.127.254:9875"; // sap.mcast.net
resolve_mcast(sap_dest,&chan->sap.dest_socket,0,NULL,0,0);
join_group(Output_fd,&chan->sap.dest_socket,iface);
ASSERT_ZEROED(&chan->sap.thread,sizeof chan->sap.thread);
pthread_create(&chan->sap.thread,NULL,sap_send,chan);
}
// RTCP Real Time Control Protocol daemon is optional
if(RTCP_enable){
// Set the dest socket to the RTCP port on the output group
// What messy code just to overwrite a structure field, eh?
memcpy(&chan->rtcp.dest_socket,&chan->output.dest_socket,sizeof(chan->rtcp.dest_socket));
switch(chan->rtcp.dest_socket.sa_family){
case AF_INET:
{
struct sockaddr_in *sock = (struct sockaddr_in *)&chan->rtcp.dest_socket;
sock->sin_port = htons(DEFAULT_RTCP_PORT);
}
break;
case AF_INET6:
{
struct sockaddr_in6 *sock = (struct sockaddr_in6 *)&chan->rtcp.dest_socket;
sock->sin6_port = htons(DEFAULT_RTCP_PORT);
}
break;
}
ASSERT_ZEROED(&chan->rtcp.thread,sizeof chan->rtcp.thread);
pthread_create(&chan->rtcp.thread,NULL,rtcp_send,chan);
}
}
// Done processing frequency list(s) and creating chans
FREE(freq_list);
}
fprintf(stdout,"[%s] %d channels started\n",sname,nchans);
return NULL;
}
// Set up a local front end device
static int setup_hardware(char const *sname){
if(sname == NULL)
return -1; // Possible?
char const *device = config_getstring(Configtable,sname,"device",sname);
// Do we support it?
// This should go into a table somewhere
#ifndef FORCE_DYNAMIC
if(strcasecmp(device,"rx888") == 0){
Frontend.setup = rx888_setup;
Frontend.start = rx888_startup;
Frontend.tune = rx888_tune;
Frontend.gain = rx888_gain;
Frontend.atten = rx888_atten;
} else if(strcasecmp(device,"airspy") == 0){
Frontend.setup = airspy_setup;
Frontend.start = airspy_startup;
Frontend.tune = airspy_tune;
} else if(strcasecmp(device,"airspyhf") == 0){
Frontend.setup = airspyhf_setup;
Frontend.start = airspyhf_startup;
Frontend.tune = airspyhf_tune;
} else if(strcasecmp(device,"funcube") == 0){
Frontend.setup = funcube_setup;
Frontend.start = funcube_startup;
Frontend.tune = funcube_tune;
} else if(strcasecmp(device,"rtlsdr") == 0){
Frontend.setup = rtlsdr_setup;
Frontend.start = rtlsdr_startup;
Frontend.tune = rtlsdr_tune;
} else if(strcasecmp(device,"sig_gen") == 0) {
Frontend.setup = sig_gen_setup;
Frontend.start = sig_gen_startup;
Frontend.tune = sig_gen_tune;
/* SDRPlay is now a dynamically loaded module.
1. Install the API package (I used https://github.com/srcejon/sdrplayapi)
Note this will also install a strange half-megabyte daemon in /etc/systemd/system/
I have no idea what it does, but it burns up much more CPU than all of ka9q-radio
2. run "make SDRPLAY=1" to build and install the sdrplay.so module so radiod can load it
It's not built by default because the compile will fail unless the API package is installed,
and it's just too much hassle for people who don't have an SDRPlay anyway
The sdrplay library is still proprietary and object-only, so I can't bundle it in ka9q-radio
Everything else either has a standard Debian package or I have information to program them directly.
To hell with vendors who deliberately make their products hard to use when they have plenty of competition.
*/
} else
#endif
{
// Try to find it dynamically
char defname[PATH_MAX];
snprintf(defname,sizeof(defname),"%s/%s.so",SODIR,device);
char const *dlname = config_getstring(Configtable,device,"library",defname);
if(dlname == NULL){
fprintf(stdout,"No dynamic library specified for device %s\n",device);
return -1;
}
fprintf(stdout,"Dynamically loading %s hardware driver from %s\n",device,dlname);
char *error;
Dl_handle = dlopen(dlname,RTLD_GLOBAL|RTLD_NOW);
if(Dl_handle == NULL){
error = dlerror();
fprintf(stdout,"Error loading %s to handle device %s: %s\n",dlname,device,error);
return -1;
}
char symname[128];
snprintf(symname,sizeof(symname),"%s_setup",device);
Frontend.setup = dlsym(Dl_handle,symname);
if((error = dlerror()) != NULL){
fprintf(stdout,"error: symbol %s not found in %s for %s: %s\n",symname,dlname,device,error);
dlclose(Dl_handle);
return -1;
}
snprintf(symname,sizeof(symname),"%s_startup",device);
Frontend.start = dlsym(Dl_handle,symname);
if((error = dlerror()) != NULL){
fprintf(stdout,"error: symbol %s not found in %s for %s: %s\n",symname,dlname,device,error);
dlclose(Dl_handle);
return -1;
}
snprintf(symname,sizeof(symname),"%s_tune",device);
Frontend.tune = dlsym(Dl_handle,symname);
if((error = dlerror()) != NULL){
// Not fatal, but no tuning possible
fprintf(stdout,"warning: symbol %s not found in %s for %s: %s\n",symname,dlname,device,error);
}
// No error checking on these, they're optional
snprintf(symname,sizeof(symname),"%s_gain",device);
Frontend.gain = dlsym(Dl_handle,symname);
snprintf(symname,sizeof(symname),"%s_atten",device);
Frontend.atten = dlsym(Dl_handle,symname);
}
int r = (*Frontend.setup)(&Frontend,Configtable,sname);
if(r != 0){
fprintf(stdout,"device setup returned %d\n",r);
return r;
}
// Create input filter now that we know the parameters
// FFT and filter sizes computed from specified block duration and sample rate
// L = input data block size
// M = filter impulse response duration
// N = FFT size = L + M - 1
// Note: no checking that N is an efficient FFT blocksize; choose your parameters wisely
assert(Frontend.samprate != 0);
double const eL = Frontend.samprate * Blocktime / 1000.0; // Blocktime is in milliseconds
Frontend.L = lround(eL);
if(Frontend.L != eL)
fprintf(stdout,"Warning: non-integral samples in %.3f ms block at sample rate %d Hz: remainder %g\n",
Blocktime,Frontend.samprate,eL-Frontend.L);
Frontend.M = Frontend.L / (Overlap - 1) + 1;
assert(Frontend.M != 0);
assert(Frontend.L != 0);
create_filter_input(&Frontend.in,Frontend.L,Frontend.M, Frontend.isreal ? REAL : COMPLEX);
pthread_mutex_init(&Frontend.status_mutex,NULL);
pthread_cond_init(&Frontend.status_cond,NULL);
if(Frontend.start){
int r = (*Frontend.start)(&Frontend);
if(r != 0)
fprintf(stdout,"Front end start returned %d\n",r);
return r;
} else {
fprintf(stdout,"No front end start routine?\n");
return -1;
}
}
// RTP control protocol sender task
static void *rtcp_send(void *arg){
struct channel *chan = (struct channel *)arg;
if(chan == NULL)
pthread_exit(NULL);
{
char name[100];
snprintf(name,sizeof(name),"rtcp %u",chan->output.rtp.ssrc);
pthread_setname(name);
}
while(true){
if(chan->output.rtp.ssrc == 0) // Wait until it's set by output RTP subsystem
goto done;
uint8_t buffer[PKTSIZE]; // much larger than necessary
memset(buffer,0,sizeof(buffer));
// Construct sender report
struct rtcp_sr sr;
memset(&sr,0,sizeof(sr));
sr.ssrc = chan->output.rtp.ssrc;
// Construct NTP timestamp (NTP uses UTC, ignores leap seconds)
{
struct timespec now;
clock_gettime(CLOCK_REALTIME,&now);
sr.ntp_timestamp = ((int64_t)now.tv_sec + NTP_EPOCH) << 32;
sr.ntp_timestamp += ((int64_t)now.tv_nsec << 32) / BILLION; // NTP timestamps are units of 2^-32 sec
}
// The zero is to remind me that I start timestamps at zero, but they could start anywhere
sr.rtp_timestamp = (0 + gps_time_ns() - Starttime) / BILLION;
sr.packet_count = chan->output.rtp.seq;
sr.byte_count = chan->output.rtp.bytes;
uint8_t *dp = gen_sr(buffer,sizeof(buffer),&sr,NULL,0);
// Construct SDES
struct rtcp_sdes sdes[4];
// CNAME
char *string = NULL;
int sl = asprintf(&string,"radio@%s",Hostname);
if(sl > 0 && sl <= 255){
sdes[0].type = CNAME;
strlcpy(sdes[0].message,string,sizeof(sdes[0].message));
sdes[0].mlen = strlen(sdes[0].message);
}
FREE(string);
sdes[1].type = NAME;
strlcpy(sdes[1].message,"KA9Q Radio Program",sizeof(sdes[1].message));
sdes[1].mlen = strlen(sdes[1].message);
sdes[2].type = EMAIL;
strlcpy(sdes[2].message,"[email protected]",sizeof(sdes[2].message));
sdes[2].mlen = strlen(sdes[2].message);
sdes[3].type = TOOL;
strlcpy(sdes[3].message,"KA9Q Radio Program",sizeof(sdes[3].message));
sdes[3].mlen = strlen(sdes[3].message);
dp = gen_sdes(dp,sizeof(buffer) - (dp-buffer),chan->output.rtp.ssrc,sdes,4);
if(sendto(Output_fd,buffer,dp-buffer,0,&chan->rtcp.dest_socket,sizeof(chan->rtcp.dest_socket)) < 0)
chan->output.errors++;
done:;
sleep(1);
}
}
static void closedown(int a){
fprintf(stdout,"Received signal %d, exiting\n",a);
Stop_transfers = true;
sleep(1); // pause for threads to see it
if(a == SIGTERM)
exit(EX_OK); // Return success when terminated by systemd
else
exit(EX_SOFTWARE);
}
// Increase or decrease logging level (thanks AI6VN for idea)
static void verbosity(int a){
if(a == SIGUSR1)
Verbose++;
else if(a == SIGUSR2)
Verbose = (Verbose <= 0) ? 0 : Verbose - 1;
else
return;
fprintf(stdout,"Verbose = %d\n",Verbose);
}