/* B-Netz main * * (C) 2016 by Andreas Eversberg * All Rights Reserved * * 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 . */ #include #include #include #include #include #include #include #include "../common/debug.h" #include "../common/timer.h" #include "../common/call.h" #include "../common/mncc_sock.h" #include "../common/main.h" #include "../common/freiton.h" #include "../common/besetztton.h" #include "bnetz.h" #include "dsp.h" #include "stations.h" #include "image.h" #include "ansage.h" int gfs = 2; int num_pilot = 0; const char *pilot[MAX_SENDER] = { "tone" }; double lossdetect = 0; void print_help(const char *arg0) { print_help_common(arg0, ""); /* - - */ printf(" -g --gfs | ,\n"); printf(" Gruppenfreisignal\" 1..9 | 19 | 10..18 (default = '%d')\n", gfs); printf(" Alternative give your coordinates of your location, to find closest\n"); printf(" base station. (e.g. '--gfs 54.487291,9.069993') Or use '--gfs list' to\n"); printf(" get a list of all base station locations.\n"); printf(" -g --gfs 19\n"); printf(" Set to 19 in order to make the phone transmit at 100 mW instead of\n"); printf(" full 15 Watts. If supported, the phone uses the channel with low power\n"); printf(" (Kanal kleiner Leistung).\n"); printf(" -P --pilot tone | positive | negative | =:\n"); printf(" Send a tone, give a signal or write to a file when switching to\n"); printf(" channel 19. (paging the phone).\n"); printf(" 'tone', 'positive', 'negative' is sent on second audio channel.\n"); printf(" 'tone' sends a tone whenever channel 19 is switchted.\n"); printf(" 'positive' sends a positive signal for channel 19, else negative.\n"); printf(" 'negative' sends a negative signal for channel 19, else positive.\n"); printf(" Example: /sys/class/gpio/gpio17/value=1:0 writes a '1' to\n"); printf(" /sys/class/gpio/gpio17/value to switching to channel 19 and a '0' to\n"); printf(" switch back. (default = %s)\n", pilot[0]); printf(" -0 --loss \n"); printf(" Detect loss of carrier by detecting steady noise above given volume in\n"); printf(" percent. (disabled by default)\n"); printf("\nstation-id: Give 5 digit station-id, you don't need to enter it for every\n"); printf(" start of this program.\n"); } static int handle_options(int argc, char **argv) { int skip_args = 0; char *p; static struct option long_options_special[] = { {"gfs", 1, 0, 'g'}, {"pilot", 1, 0, 'P'}, {"loss", 1, 0, '0'}, {0, 0, 0, 0}, }; set_options_common("g:P:0:", long_options_special); while (1) { int option_index = 0, c; c = getopt_long(argc, argv, optstring, long_options, &option_index); if (c == -1) break; switch (c) { case 'g': if (!strcasecmp(optarg, "list")) { station_list(); exit(0); } if ((p = strchr(optarg, ','))) { gfs = get_station_by_coordinates(atof(optarg), atof(p + 1)); if (gfs == 0) exit(0); } else gfs = atoi(optarg); skip_args += 2; break; case 'P': OPT_ARRAY(num_pilot, pilot, strdup(optarg)) skip_args += 2; break; case '0': lossdetect = atoi(optarg); skip_args += 2; break; default: opt_switch_common(c, argv[0], &skip_args); } } return skip_args; } int main(int argc, char *argv[]) { int rc; int skip_args; const char *station_id = ""; int i; /* init common tones */ init_freiton(); init_besetzton(); init_ansage(); skip_args = handle_options(argc, argv); argc -= skip_args; argv += skip_args; if (argc > 1) { station_id = argv[1]; if (strlen(station_id) != 5) { printf("Given station ID '%s' does not have 5 digits\n", station_id); return 0; } } if (!num_kanal) { printf("No channel (\"Kanal\") is specified, I suggest channel 1.\n\n"); print_help(argv[-skip_args]); return 0; } if (num_kanal == 1 && num_sounddev == 0) num_sounddev = 1; /* use defualt */ if (num_kanal != num_sounddev) { fprintf(stdout, "You need to specify as many sound devices as you have channels.\n"); exit(0); } if (num_kanal == 1 && num_pilot == 0) num_pilot = 1; /* use defualt */ if (num_kanal != num_pilot) { fprintf(stdout, "You need to specify as many pilot tone settings as you have channels.\n"); exit(0); } if (!loopback) print_image(); /* init functions */ if (use_mncc_sock) { rc = mncc_init("/tmp/bsc_mncc"); if (rc < 0) { fprintf(stderr, "Failed to setup MNCC socket. Quitting!\n"); return -1; } } dsp_init(); bnetz_init(); rc = call_init(station_id, call_sounddev, samplerate, latency, 5, loopback); if (rc < 0) { fprintf(stderr, "Failed to create call control instance. Quitting!\n"); goto fail; } /* create transceiver instance */ for (i = 0; i < num_kanal; i++) { rc = bnetz_create(kanal[i], sounddev[i], samplerate, cross_channels, rx_gain, gfs, do_pre_emphasis, do_de_emphasis, write_wave, read_wave, loopback, (double)lossdetect / 100.0, pilot[i]); if (rc < 0) { fprintf(stderr, "Failed to create \"Sender\" instance. Quitting!\n"); goto fail; } printf("Base station for channel %d ready, please tune transmitter to %.3f MHz and receiver " "to %.3f MHz.\n", kanal[i], bnetz_kanal2freq(kanal[i], 0), bnetz_kanal2freq(kanal[i], 1)); printf("To call phone, switch transmitter (using pilot signal) to %.3f MHz.\n", bnetz_kanal2freq(19, 0)); } signal(SIGINT,sighandler); signal(SIGHUP,sighandler); signal(SIGTERM,sighandler); signal(SIGPIPE,sighandler); if (rt_prio > 0) { struct sched_param schedp; int rc; memset(&schedp, 0, sizeof(schedp)); schedp.sched_priority = rt_prio; rc = sched_setscheduler(0, SCHED_RR, &schedp); if (rc) fprintf(stderr, "Error setting SCHED_RR with prio %d\n", rt_prio); } main_loop(&quit, latency, interval, NULL); if (rt_prio > 0) { struct sched_param schedp; memset(&schedp, 0, sizeof(schedp)); schedp.sched_priority = 0; sched_setscheduler(0, SCHED_OTHER, &schedp); } fail: /* cleanup functions */ call_cleanup(); if (use_mncc_sock) mncc_exit(); /* destroy transceiver instance */ while(sender_head) bnetz_destroy(sender_head); return 0; }