osmocom-analog/src/mpt1327/main.c

399 lines
12 KiB
C

/* MPT1327 main
*
* (C) 2021 by Andreas Eversberg <jolly@eversberg.eu>
* 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 <http://www.gnu.org/licenses/>.
*/
#include <stdio.h>
#include <stdint.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include <unistd.h>
#include <errno.h>
#include <math.h>
#include <sys/types.h>
#include <sys/stat.h>
#include "../libsample/sample.h"
#include "../libmobile/main_mobile.h"
#include "../libdebug/debug.h"
#include "../libtimer/timer.h"
#include "../anetz/freiton.h"
#include "../anetz/besetztton.h"
#include "../liboptions/options.h"
#include "mpt1327.h"
#include "dsp.h"
#include "message.h"
/* settings */
int num_freq = 0;
static int num_chan_type = 0;
static double squelch_db = -INFINITY;
static enum mpt1327_band band = BAND_REGIONET43_SUB1;
static enum mpt1327_chan_type chan_type[MAX_SENDER] = { CHAN_TYPE_CC_TC };
static int16_t sys = -1;
static int wt = 10;
static int per = 5;
static int pon = 1;
static int timeout = 30;
void print_image(void) {}
void print_help(const char *arg0)
{
main_mobile_print_help(arg0, "-O ... | -I ... ");
/* - - */
printf(" -B --band <name> | list\n");
printf(" Select frequency Band (default = '%s')\n", mpt1327_band_name(band));
printf(" -T --channel-type <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(" -O --operator <OPID> <NDD> <LAB>\n");
printf(" -> decimal, '0x' for hex or all binary digits\n");
printf(" Give System Identity Code of regional network (1st bit = 0)\n");
printf(" OPID: Operator Identity (7 binary digits)\n");
printf(" -> Check subscription data of mobile unit\n");
printf(" NDD: Network Dependent Data (4 binary digts)\n");
printf(" -> Check subscription data of mobile unit (must be '0001' or greater)\n");
printf(" -> Change it to force re-registering of mobile unit.\n");
printf(" LAB: Label for multiple control channels (3 binary digits)\n");
printf(" -> Use '001' to allow all categories\n");
printf(" -N --net <NET> <NDD> <LAB>\n");
printf(" -> decimal, '0x' for hex or all binary digits\n");
printf(" Give System Identity Code of national network (1st bit = 1)\n");
printf(" NET: Network Identity (2 binary digits)\n");
printf(" -> Check subscription data of mobile unit (must be '000000001' or greater)\n");
printf(" -> Change it to force re-registering of mobile unit.\n");
printf(" NDD: Network Dependent Data (9 binary digts)\n");
printf(" LAB: Label for multiple control channels (3 binary digits)\n");
printf(" -> Use '001' to allow all categories\n");
printf(" -S --sysdef wt=5 | wt=10 | wt=15\n");
printf(" Number of slots the Radio Unit waits for response. A slot lasts about\n");
printf(" 107 ms. (default = %d)\n", wt);
printf(" -S --sysdef per=<secs> | per=0\n");
printf(" Interval of periodic messages from the Radio Unit while transmitting\n");
printf(" speech. Use 1..31 to enable and 0 to disable. Also the 'timeout' value\n");
printf(" must be greater than value given here. (default = %d)\n", per);
printf(" -S --sysdef pon=1 | pon=0\n");
printf(" The Radio Unit must send 'Pressel On' message to unmute the uplink.\n");
printf(" If disabled, squelch must be enabled. (default = %d)\n", pon);
printf(" -S --sysdef timeout=<secs> | timeout=off\n");
printf(" The Traffic Channel is released, if no radio transmits for given amount of time.\n");
printf(" (default = %d)\n", timeout);
printf(" -Q --squelch <dB> | auto\n");
printf(" Use given RF level to detect transmission on Traffic Channel, if\n");
printf(" 'Pressel On' is disabled.\n");
printf(" and stays below this level, the connection is released.\n");
printf(" Use 'auto' to do automatic noise floor calibration to detect loss.\n");
printf(" Only works with SDR! (disabled by default)\n");
printf("\nstation-id: Give 7 digits of Radio Unit's prefix/ident, you don't need to\n");
printf(" enter it for every start of this program.\n");
main_mobile_print_hotkeys();
printf("Press 'i' key to dump list of seen Radio Units.\n");
}
static void add_options(void)
{
main_mobile_add_options();
option_add('B', "band", 1);
option_add('T', "channel-type", 1);
option_add('O', "operator", 3);
option_add('N', "net", 3);
option_add('S', "sysdef", 1);
option_add('Q', "squelch", 1);
}
static int read_sys(const char *param, const char *value, int digits)
{
int result = 0;
int i;
if ((int)strlen(value) < digits) {
result = strtoul(value, NULL, 0);
if (result >= (1 << digits)) {
fprintf(stderr, "Given '%s' value is out of range for %d binary digits, use '-h' for help!\n", param, digits);
return -EINVAL;
}
return result;
}
if ((int)strlen(value) > digits) {
fprintf(stderr, "Given '%s' value must have exactly %d binary digits, use '-h' for help!\n", param, digits);
return -EINVAL;
}
for (i = 0; i < (int)strlen(value); i++) {
if (value[i] < '0' || value[i] > '1') {
fprintf(stderr, "Given '%s' value must only have binary digits of '0' or '1', use '-h' for help!\n", param);
return -EINVAL;
}
result = (result << 1) | (value[i] - '0');
}
return result;
}
static int handle_options(int short_option, int argi, char **argv)
{
int rc;
const char *p;
switch (short_option) {
case 'B':
if (!strcmp(argv[argi], "list")) {
mpt1327_band_list();
return 0;
}
rc = mpt1327_band_by_short_name(argv[argi]);
if (rc < 0) {
fprintf(stderr, "Given band '%s' is illegal, use '-h' for help!\n", argv[argi]);
return -EINVAL;
}
band = rc;
break;
case 'T':
if (!strcmp(argv[argi], "list")) {
mpt1327_channel_list();
return 0;
}
rc = mpt1327_channel_by_short_name(argv[argi]);
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", argv[argi]);
return -EINVAL;
}
OPT_ARRAY(num_chan_type, chan_type, rc)
break;
case 'O':
sys = 0x0000;
rc = read_sys("OID", argv[argi + 0], 7);
if (rc < 0)
return rc;
sys = sys | (rc << 7);
rc = read_sys("NDD", argv[argi + 1], 4);
if (rc < 0)
return rc;
sys = sys | (rc << 3);
rc = read_sys("LAB", argv[argi + 2], 3);
if (rc < 0)
return rc;
sys = sys | rc;
break;
case 'N':
sys = 0x4000;
rc = read_sys("NET", argv[argi + 0], 2);
if (rc < 0)
return rc;
sys = sys | (rc << 12);
rc = read_sys("NDD", argv[argi + 1], 9);
if (rc < 0)
return rc;
sys = sys | (rc << 3);
rc = read_sys("LAB", argv[argi + 2], 3);
if (rc < 0)
return rc;
sys = sys | rc;
break;
case 'S':
p = strchr(argv[argi], '=');
if (!p) {
fprintf(stderr, "Given sysdef parameter '%s' requires '=' character to set value, use '-h' for help!\n", argv[argi]);
return -EINVAL;
}
p++;
if (!strncasecmp(argv[argi], "wt=", p - argv[argi])) {
wt = atoi(p);
if (wt != 5 && wt != 10 && wt != 15) {
sysdef_oor:
fprintf(stderr, "Given sysdef parameter '%s' out of range, use '-h' for help!\n", argv[argi]);
return -EINVAL;
}
} else
if (!strncasecmp(argv[argi], "per=", p - argv[argi])) {
per = atoi(p);
if (per < 0 || per >31)
goto sysdef_oor;
} else
if (!strncasecmp(argv[argi], "pon=", p - argv[argi])) {
pon = atoi(p);
if (pon != 0 && pon != 1)
goto sysdef_oor;
} else
if (!strncasecmp(argv[argi], "timeout=", p - argv[argi])) {
timeout = atoi(p);
} else
{
fprintf(stderr, "Given sysdef parameter '%s' unknown, use '-h' for help!\n", argv[argi]);
return -EINVAL;
}
break;
case 'Q':
if (!strcasecmp(argv[argi], "auto"))
squelch_db = 0.0;
else
squelch_db = atof(argv[argi]);
break;
default:
return main_mobile_handle_options(short_option, argi, argv);
}
return 1;
}
int main(int argc, char *argv[])
{
int rc, argi;
const char *station_id = "";
int mandatory = 0;
int i;
/* init tones */
init_freiton();
init_besetzton();
// init_ansage();
console_digits = "0123456789*#";
main_mobile_init();
/* handle options / config file */
add_options();
rc = options_config_file(argc, argv, "~/.osmocom/analog/mpt1327.conf", handle_options);
if (rc < 0)
return 0;
argi = options_command_line(argc, argv, handle_options);
if (argi <= 0)
return argi;
if (argi < argc) {
station_id = argv[argi];
if (strlen(station_id) != 7) {
printf("Given station ID '%s' does not have 4 digits\n", station_id);
return 0;
}
}
if (!num_kanal) {
printf("No channel (\"Kanal\") is specified, I suggest channel 1.\n\n");
mandatory = 1;
}
if (use_sdr) {
/* set audiodev */
for (i = 0; i < num_kanal; i++)
audiodev[i] = "sdr";
num_audiodev = num_kanal;
/* set channel types for more than 1 channel */
if (num_kanal > 1 && num_chan_type == 0) {
chan_type[0] = CHAN_TYPE_CC;
for (i = 1; i < num_kanal; i++)
chan_type[i] = CHAN_TYPE_TC;
num_chan_type = num_kanal;
}
}
if (num_kanal == 1 && num_audiodev == 0)
num_audiodev = 1; /* use default */
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 default */
if (num_kanal != num_chan_type) {
fprintf(stderr, "You need to specify as many channel types as you have channels.\n");
exit(0);
}
if (sys < 0) {
fprintf(stderr, "No System Identity Code is specified, make them match with your radio unit.\n\n");
mandatory = 1;
}
if (isinf(squelch_db) && pon == 0) {
fprintf(stderr, "'Pressel On' message (PON) and squelch are turned off. Enable one of them.\n\n");
mandatory = 1;
}
if (!isinf(squelch_db) && pon == 1) {
fprintf(stderr, "'Pressel On' message (PON) and squelch are turned on. Disable one of them.\n\n");
mandatory = 1;
}
if (pon && timeout <= per) {
fprintf(stderr, "The defined timeout value is lower than the Periodic message interval (PER). Define a greater timeout.\n\n");
mandatory = 1;
}
if (pon && (timeout && !per)) {
fprintf(stderr, "You must enable Periodic message interval (PER), if you use timeout (and have no squelch).\n\n");
mandatory = 1;
}
if (!pon && !timeout) {
fprintf(stderr, "Warning: 'Pressel On' message (PON) and timeout is both disabled. There will be no way to detect loss of Radio Unit.\n\n");
}
if (do_de_emphasis || do_pre_emphasis) {
printf("Don't use pre-/de-emphasis, it is not used for Speech, nor for signaling.\n\n");
mandatory = 1;
}
if (mandatory) {
print_help(argv[0]);
return 0;
}
/* no SDR, no squelch */
if (!use_sdr && !isinf(squelch_db)) {
fprintf(stderr, "Cannot use squelch without SDR! Analog receivers don't give use RSSI.\n");
goto fail;
}
printf("Using Sysdef 0x%04x:\n", sys);
if (!(sys & 0x4000)) {
printf("OID=%d NDD=%d LAB=%d\n", (sys >> 7) & 0x7f, (sys >> 3) & 0xf, sys & 0x7);
} else {
printf("NET=%d NDD=%d LAB=%d\n", (sys >> 12) & 0x3, (sys >> 3) & 0x1ff, sys & 0x7);
}
/* inits */
fm_init(fast_math);
dsp_init();
init_codeword();
init_sysdef(sys, wt, per, pon, timeout);
/* create transceiver instance */
for (i = 0; i < num_kanal; i++) {
rc = mpt1327_create(band, kanal[i], chan_type[i], audiodev[i], use_sdr, samplerate, rx_gain, tx_gain, write_rx_wave, write_tx_wave, read_rx_wave, read_tx_wave, loopback, squelch_db);
if (rc < 0) {
fprintf(stderr, "Failed to create transceiver instance. Quitting!\n");
goto fail;
}
printf("base station on channel %s ready, please tune transmitter to %.4f MHz and receiver to %.4f MHz. (%s %.3f MHz offset)\n", kanal[i], mpt1327_channel2freq(band, atoi(kanal[i]), 0) / 1e6, mpt1327_channel2freq(band, atoi(kanal[i]), 1) / 1e6, mpt1327_band_name(band), mpt1327_channel2freq(band, atoi(kanal[i]), 2) / 1e6);
}
mpt1327_check_channels();
main_mobile("mpt1327", &quit, latency, interval, NULL, station_id, 7);
fail:
/* destroy transceiver instance */
while (sender_head)
mpt1327_destroy(sender_head);
/* exits */
fm_exit();
flush_units();
options_free();
return 0;
}