/* C-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 "../common/main.h" #include "../common/debug.h" #include "../common/timer.h" #include "../common/call.h" #include "../common/mncc_sock.h" #include "../common/freiton.h" #include "../common/besetztton.h" #include "cnetz.h" #include "database.h" #include "sysinfo.h" #include "dsp.h" #include "telegramm.h" #include "image.h" #include "ansage.h" /* settings */ int num_chan_type = 0; enum cnetz_chan_type chan_type[MAX_SENDER] = { CHAN_TYPE_OGK_SPK }; int measure_speed = 0; double clock_speed[2] = { 0.0, 0.0 }; int set_clock_speed = 0; const char *flip_polarity = "auto"; double noise = 0.0; int ms_power = 0; /* 0..3 */ int auth = 0; int voice_deviation = 1; void print_help(const char *arg0) { print_help_common(arg0, "[-M] -S , -p -d "); /* - - */ printf(" -T --channel-type | list\n"); printf(" Give channel type, use 'list' to get a list. (default = '%s')\n", chan_type_short_name(chan_type[0])); printf(" -M --measure-speed\n"); printf(" Measures clock speed. THIS IS REQUIRED! See documentation!\n"); printf(" -S --clock-speed ,\n"); printf(" Correct speed of sound card's clock. Use '-M' to measure speed for\n"); printf(" some hours after temperature has settled. The use these results to\n"); printf(" correct signal processing speed. After adjustment, the clock must match\n"); printf(" +- 1ppm or better. CORRECTING CLOCK SPEED IS REQUIRED! See\n"); printf(" documentation on how to measure correct value.\n"); printf(" -F --flip-polarity no | yes | auto\n"); printf(" Flip polarity of transmitted FSK signal. If yes, the sound card\n"); printf(" generates a negative signal rather than a positive one. If auto, the\n"); printf(" base station generates two virtual base stations with both polarities.\n"); printf(" Once a mobile registers, the correct polarity is selected and used.\n"); printf(" (default = %s)\n", flip_polarity); printf(" -N --noise 0.0 .. 1.0 (default = %.1f)\n", noise); printf(" Between frames on OgK, send noise at given level. Use 0.0 for Silence.\n"); printf(" -P --ms-power \n"); printf(" Give power level of the mobile station 0..3. (default = '%d')\n", ms_power); printf(" 0 = 50-125 mW; 1 = 0.5-1 W; 2 = 4-8 W; 3 = 10-20 W\n"); printf(" -A --authentication\n"); printf(" Enable authentication on the base station. Since we cannot\n"); printf(" authenticate, because we don't know the secret key and the algorithm,\n"); printf(" we just accept any card. With this we get the vendor IDs of the phone.\n"); printf(" -V --voice-deviation\n"); printf(" For some unknown reason, Siemens C5 use double deviation for voice.\n"); printf(" This option raises audio level on TX and lowers on RX.\n"); printf("\nstation-id: Give 7 digit station-id, you don't need to enter it for every\n"); printf(" start of this program.\n"); print_hotkeys_common(); printf("Press 'i' key to dump list of currently attached subscribers.\n"); } static int handle_options(int argc, char **argv) { int skip_args = 0; int rc; const char *p; static struct option long_options_special[] = { {"channel-type", 1, 0, 'T'}, {"measure-speed", 0, 0, 'M'}, {"clock-speed", 1, 0, 'S'}, {"flip-polarity", 1, 0, 'F'}, {"noise", 1, 0, 'N'}, {"ms-power", 1, 0, 'P'}, {"authentication", 0, 0, 'A'}, {"voice-deviation", 0, 0, 'V'}, {0, 0, 0, 0} }; set_options_common("T:MS:F:N:P:AV", 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 'T': if (!strcmp(optarg, "list")) { cnetz_channel_list(); exit(0); } rc = cnetz_channel_by_short_name(optarg); if (rc < 0) { fprintf(stderr, "Error, channel type '%s' unknown. Please use '-t list' to get a list. I suggest to use the default.\n", optarg); exit(0); } OPT_ARRAY(num_chan_type, chan_type, rc) skip_args += 2; break; case 'M': measure_speed = 1; skip_args++; break; case 'S': p = strchr(optarg, ','); if (!p) { fprintf(stderr, "Illegal clock speed, use two values, seperated by comma and no spaces!\n"); exit(0); } clock_speed[0] = strtold(optarg, NULL); clock_speed[1] = strtold(p + 1, NULL); set_clock_speed = 1; skip_args += 2; break; case 'F': if (!strcasecmp(optarg, "no")) flip_polarity = "no"; else if (!strcasecmp(optarg, "yes")) flip_polarity = "yes"; else if (!strcasecmp(optarg, "auto")) flip_polarity = "auto"; else { fprintf(stderr, "Given polarity '%s' is illegal, see help!\n", optarg); exit(0); } skip_args += 2; break; case 'N': noise = strtold(optarg, NULL); skip_args += 2; break; case 'P': ms_power = atoi(optarg); if (ms_power > 3) ms_power = 3; if (ms_power < 0) ms_power = 0; skip_args += 2; break; case 'A': auth = 1; skip_args += 1; break; case 'V': voice_deviation = 2; skip_args += 1; break; default: opt_switch_common(c, argv[0], &skip_args); } } free(long_options); return skip_args; } int main(int argc, char *argv[]) { int rc; int skip_args; const char *station_id = ""; int mandatory = 0; int polarity; 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) != 7) { printf("Given station ID '%s' does not have 7 digits\n", station_id); return 0; } } if (!num_kanal) { printf("No channel (\"Kanal\") is specified, I suggest channel %d.\n\n", CNETZ_OGK_KANAL); mandatory = 1; } if (num_kanal == 1 && num_audiodev == 0) num_audiodev = 1; /* use defualt */ if (num_kanal != num_audiodev) { fprintf(stderr, "You need to specify as many sound devices as you have channels.\n"); exit(0); } if (num_kanal == 1 && num_chan_type == 0) num_chan_type = 1; /* use defualt */ if (num_kanal != num_chan_type) { fprintf(stderr, "You need to specify as many channel types as you have channels.\n"); exit(0); } if (!set_clock_speed && !measure_speed) { printf("No clock speed given. You need to measure clock using '-M' and later correct clock using '-S ,'. See documentation for help!\n\n"); mandatory = 1; } if (mandatory) { print_help(argv[-skip_args]); return 0; } if (!loopback) print_image(); /* init functions */ rc = init_common(station_id, 7); if (rc < 0) goto fail; scrambler_init(); init_sysinfo(); dsp_init(); rc = init_telegramm(); if (rc < 0) { fprintf(stderr, "Error in Telegramm structure. Quitting!\n"); goto fail; } init_coding(); cnetz_init(); /* check for mandatory OgK */ for (i = 0; i < num_kanal; i++) { if (chan_type[i] == CHAN_TYPE_OGK || chan_type[i] == CHAN_TYPE_OGK_SPK) break; } if (i == num_kanal) { fprintf(stderr, "You must define at least one OgK (control) or OgK/SPK (control/speech) channel type. Quitting!\n"); goto fail; } /* check for mandatory OgK */ for (i = 0; i < num_kanal; i++) { if (chan_type[i] == CHAN_TYPE_SPK || chan_type[i] == CHAN_TYPE_OGK_SPK) break; } if (i == num_kanal) fprintf(stderr, "You did not define any SpK (speech) channel. You will not be able to make any call.\n"); /* SDR always requires emphasis */ if (!strcmp(audiodev[0], "sdr")) { do_pre_emphasis = 1; do_de_emphasis = 1; } if (!do_pre_emphasis || !do_de_emphasis) { fprintf(stderr, "*******************************************************************************\n"); fprintf(stderr, "I strongly suggest to let me do pre- and de-emphasis (options -p -d)!\n"); fprintf(stderr, "Use a transmitter/receiver without emphasis and let me do that!\n"); fprintf(stderr, "Because carrier FSK signaling and scrambled voice (default) does not use\n"); fprintf(stderr, "emphasis, I like to control emphasis by myself for best results.\n"); fprintf(stderr, "*******************************************************************************\n"); } polarity = 0; /* auto */ if (!strcmp(flip_polarity, "no")) polarity = 1; /* positive */ if (!strcmp(flip_polarity, "yes")) polarity = -1; /* negative */ if (!strcmp(audiodev[0], "sdr") && polarity == 0) polarity = 1; /* SDR is always positive */ /* create transceiver instance */ for (i = 0; i < num_kanal; i++) { rc = cnetz_create(kanal[i], chan_type[i], audiodev[i], samplerate, rx_gain, auth, ms_power, (i == 0) ? measure_speed : 0, clock_speed, polarity, noise, do_pre_emphasis, do_de_emphasis, write_rx_wave, write_tx_wave, read_rx_wave, loopback); if (rc < 0) { fprintf(stderr, "Failed to create \"Sender\" instance. Quitting!\n"); goto fail; } printf("Base station on channel %d ready, please tune transmitter to %.3f MHz and receiver to %.3f MHz.\n", kanal[i], cnetz_kanal2freq(kanal[i], 0) / 1e6, cnetz_kanal2freq(kanal[i], 1) / 1e6); } main_common(&quit, latency, interval, NULL); fail: /* cleanup functions */ cleanup_common(); flush_db(); /* destroy transceiver instance */ while (sender_head) cnetz_destroy(sender_head); return 0; }