Parse scanlists.

This commit is contained in:
Serge Vakulenko 2018-08-30 20:31:23 -07:00
parent d1836b365f
commit 85f0df06d7
2 changed files with 374 additions and 40 deletions

203
md380.c
View File

@ -390,21 +390,45 @@ static int md380_is_compatible(radio_device_t *radio)
} }
// //
// Set the bitmask of zones for a given channel. // Add channel to a zone.
// Return 0 on failure. // Return 0 on failure.
// //
static void setup_zone(int zone_index, int chan_index) static void zone_append(int zone_index, int chan_index)
{ {
uint8_t *data = &radio_mem[OFFSET_ZONES + zone_index*0x80 + chan_index/8]; zone_t *z = (zone_t*) &radio_mem[OFFSET_ZONES + zone_index*64];
*data |= 1 << (chan_index & 7); //TODO: append channel to a zone
(void)z;
} }
static void erase_zone(int zone_index) static void erase_zone(int zone_index)
{ {
uint8_t *data = &radio_mem[OFFSET_ZONES + zone_index*0x80]; zone_t *z = (zone_t*) &radio_mem[OFFSET_ZONES + zone_index*64];
memset(data, 0, 0x80); memset(z, 0, 64);
}
//
// Set parameters for a given scan list.
//
static void setup_scanlist(int index, const char *name,
int prio1, int prio2, int txchan)
{
scanlist_t *sl = (scanlist_t*) &radio_mem[OFFSET_SCANL + index*104];
// Bytes 0-31
utf8_decode(sl->name, name, 16);
// Bytes 32-37
sl->priority_ch1 = prio1;
sl->priority_ch2 = prio2;
sl->tx_designated_ch = txchan;
// Bytes 38-41
sl->_unused1 = 0xf1;
sl->sign_hold_time = 500 / 25; // 500 msec
sl->prio_sample_time = 2000 / 250; // 2 sec
sl->_unused2 = 0xff;
} }
static void erase_scanlist(int index) static void erase_scanlist(int index)
@ -425,6 +449,24 @@ static void erase_scanlist(int index)
sl->_unused2 = 0xff; sl->_unused2 = 0xff;
} }
//
// Add channel to a zone.
// Return 0 on failure.
//
static int scanlist_append(int list_index, int cnum)
{
scanlist_t *sl = (scanlist_t*) &radio_mem[OFFSET_SCANL + list_index*104];
int i;
for (i=0; i<31; i++) {
if (sl->member[i] == 0) {
sl->member[i] = cnum;
return 1;
}
}
return 0;
}
// //
// Check that the radio does support this frequency. // Check that the radio does support this frequency.
// //
@ -1011,11 +1053,11 @@ static void md380_print_config(radio_device_t *radio, FILE *out, int verbose)
fprintf(out, "\n"); fprintf(out, "\n");
if (verbose) { if (verbose) {
fprintf(out, "# Table of scan lists.\n"); fprintf(out, "# Table of scan lists.\n");
fprintf(out, "# 1) Zone number: 1-%d\n", NSCANL); fprintf(out, "# 1) Scan list number: 1-%d\n", NSCANL);
fprintf(out, "# 2) Name: up to 16 characters, use '_' instead of space\n"); fprintf(out, "# 2) Name: up to 16 characters, use '_' instead of space\n");
fprintf(out, "# 3) Priority channel 1 (50%% of scans): -, Sel or index\n"); fprintf(out, "# 3) Priority channel 1 (50%% of scans): -, Sel or index\n");
fprintf(out, "# 4) Priority channel 2 (25%% of scans): -, Sel or index\n"); fprintf(out, "# 4) Priority channel 2 (25%% of scans): -, Sel or index\n");
fprintf(out, "# 5) Designated transmit channel: -, Last or index\n"); fprintf(out, "# 5) Designated transmit channel: Last, Sel or index\n");
fprintf(out, "# 6) List of channels: numbers and ranges (N-M) separated by comma\n"); fprintf(out, "# 6) List of channels: numbers and ranges (N-M) separated by comma\n");
fprintf(out, "#\n"); fprintf(out, "#\n");
} }
@ -1049,9 +1091,9 @@ static void md380_print_config(radio_device_t *radio, FILE *out, int verbose)
fprintf(out, "%-4d ", sl->priority_ch2); fprintf(out, "%-4d ", sl->priority_ch2);
} }
if (sl->tx_designated_ch == 0xffff) { if (sl->tx_designated_ch == 0xffff) {
fprintf(out, "- ");
} else if (sl->tx_designated_ch == 0) {
fprintf(out, "Last "); fprintf(out, "Last ");
} else if (sl->tx_designated_ch == 0) {
fprintf(out, "Sel ");
} else { } else {
fprintf(out, "%-4d ", sl->tx_designated_ch); fprintf(out, "%-4d ", sl->tx_designated_ch);
} }
@ -1199,7 +1241,7 @@ static void md380_save_image(radio_device_t *radio, FILE *img)
} }
// //
// Erase all channels, zones and scanlists. // Erase all channels.
// //
static void erase_channels() static void erase_channels()
{ {
@ -1208,9 +1250,27 @@ static void erase_channels()
for (i=0; i<NCHAN; i++) { for (i=0; i<NCHAN; i++) {
erase_channel(i); erase_channel(i);
} }
}
//
// Erase all zones.
//
static void erase_zones()
{
int i;
for (i=0; i<NZONES; i++) { for (i=0; i<NZONES; i++) {
erase_zone(i); erase_zone(i);
} }
}
//
// Erase all scanlists.
//
static void erase_scanlists()
{
int i;
for (i=0; i<NSCANL; i++) { for (i=0; i<NSCANL; i++) {
erase_scanlist(i); erase_scanlist(i);
} }
@ -1408,6 +1468,8 @@ badtx: fprintf(stderr, "Bad transmit frequency.\n");
if (first_row && radio->channel_count == 0) { if (first_row && radio->channel_count == 0) {
// On first entry, erase all channels, zones and scanlists. // On first entry, erase all channels, zones and scanlists.
erase_channels(); erase_channels();
erase_zones();
erase_scanlists();
} }
setup_channel(num-1, MODE_DIGITAL, name_str, rx_mhz, tx_mhz, setup_channel(num-1, MODE_DIGITAL, name_str, rx_mhz, tx_mhz,
@ -1576,7 +1638,7 @@ static int parse_zones(int first_row, char *line)
if (first_row) { if (first_row) {
// On first entry, erase the Zones table. // On first entry, erase the Zones table.
memset(&radio_mem[OFFSET_ZONES], 0, NZONES * 0x80); erase_zones();
} }
if (*chan_str == '-') if (*chan_str == '-')
@ -1605,12 +1667,12 @@ static int parse_zones(int first_row, char *line)
// Add range. // Add range.
int c; int c;
for (c=last; c<cnum; c++) { for (c=last; c<cnum; c++) {
setup_zone(bnum-1, c); zone_append(bnum-1, c);
nchan++; nchan++;
} }
} else { } else {
// Add single channel. // Add single channel.
setup_zone(bnum-1, cnum-1); zone_append(bnum-1, cnum-1);
nchan++; nchan++;
} }
@ -1634,12 +1696,115 @@ static int parse_zones(int first_row, char *line)
// //
static int parse_scanlist(int first_row, char *line) static int parse_scanlist(int first_row, char *line)
{ {
//TODO: parse scanlist Name char num_str[256], name_str[256], prio1_str[256], prio2_str[256];
//TODO: parse scanlist PCh1 char tx_str[256], chan_str[256];
//TODO: parse scanlist PCh2 int snum, prio1, prio2, txchan;
//TODO: parse scanlist TxCh
//TODO: parse scanlist Channels if (sscanf(line, "%s %s %s %s %s %s",
num_str, name_str, prio1_str, prio2_str, tx_str, chan_str) != 6)
return 0; return 0;
snum = atoi(num_str);
if (snum < 1 || snum > NSCANL) {
fprintf(stderr, "Bad scan list number.\n");
return 0;
}
if (first_row) {
// On first entry, erase the Scanlists table.
erase_scanlists();
}
if (*prio1_str == '-') {
prio1 = 0xffff;
} else if (strcasecmp("Sel", prio1_str) == 0) {
prio1 = 0;
} else {
prio1 = atoi(prio1_str);
if (prio1 < 1 || prio1 > NCHAN) {
fprintf(stderr, "Bad priority channel 1.\n");
return 0;
}
}
if (*prio2_str == '-') {
prio2 = 0xffff;
} else if (strcasecmp("Sel", prio2_str) == 0) {
prio2 = 0;
} else {
prio2 = atoi(prio2_str);
if (prio2 < 1 || prio2 > NCHAN) {
fprintf(stderr, "Bad priority channel 2.\n");
return 0;
}
}
if (strcasecmp("Last", tx_str) == 0) {
txchan = 0xffff;
} else if (strcasecmp("Sel", tx_str) == 0) {
txchan = 0;
} else {
txchan = atoi(tx_str);
if (txchan < 1 || txchan > NCHAN) {
fprintf(stderr, "Bad transmit channel.\n");
return 0;
}
}
setup_scanlist(snum-1, name_str, prio1, prio2, txchan);
if (*chan_str != '-') {
char *str = chan_str;
int nchan = 0;
int range = 0;
int last = 0;
// Parse channel list.
for (;;) {
char *eptr;
int cnum = strtoul(str, &eptr, 10);
if (eptr == str) {
fprintf(stderr, "Scan list %d: wrong channel list '%s'.\n", snum, str);
return 0;
}
if (cnum < 1 || cnum > NCHAN) {
fprintf(stderr, "Scan list %d: wrong channel number %d.\n", snum, cnum);
return 0;
}
if (range) {
// Add range.
int c;
for (c=last+1; c<=cnum; c++) {
if (!scanlist_append(snum-1, c)) {
fprintf(stderr, "Scan list %d: too many channels.\n", snum);
return 0;
}
nchan++;
}
} else {
// Add single channel.
if (!scanlist_append(snum-1, cnum)) {
fprintf(stderr, "Scan list %d: too many channels.\n", snum);
return 0;
}
nchan++;
}
if (*eptr == 0)
break;
if (*eptr != ',' && *eptr != '-') {
fprintf(stderr, "Scan list %d: wrong channel list '%s'.\n", snum, eptr);
return 0;
}
range = (*eptr == '-');
last = cnum;
str = eptr + 1;
}
}
return 1;
} }
// //

207
uv380.c
View File

@ -406,21 +406,49 @@ static int uv380_is_compatible(radio_device_t *radio)
} }
// //
// Set the bitmask of zones for a given channel. // Add channel to a zone.
// Return 0 on failure. // Return 0 on failure.
// //
static void setup_zone(int zone_index, int chan_index) static void zone_append(int zone_index, int chan_index)
{ {
uint8_t *data = &radio_mem[OFFSET_ZONES + zone_index*0x80 + chan_index/8]; zone_t *z = (zone_t*) &radio_mem[OFFSET_ZONES + zone_index*64];
zone_ext_t *zext = (zone_ext_t*) &radio_mem[OFFSET_ZONEXT + zone_index*224];
*data |= 1 << (chan_index & 7); //TODO: append channel to a zone
(void)z;
(void)zext;
} }
static void erase_zone(int zone_index) static void erase_zone(int zone_index)
{ {
uint8_t *data = &radio_mem[OFFSET_ZONES + zone_index*0x80]; zone_t *z = (zone_t*) &radio_mem[OFFSET_ZONES + zone_index*64];
zone_ext_t *zext = (zone_ext_t*) &radio_mem[OFFSET_ZONEXT + zone_index*224];
memset(data, 0, 0x80); memset(z, 0, 64);
memset(zext, 0, 224);
}
//
// Set parameters for a given scan list.
//
static void setup_scanlist(int index, const char *name,
int prio1, int prio2, int txchan)
{
scanlist_t *sl = (scanlist_t*) &radio_mem[OFFSET_SCANL + index*104];
// Bytes 0-31
utf8_decode(sl->name, name, 16);
// Bytes 32-37
sl->priority_ch1 = prio1;
sl->priority_ch2 = prio2;
sl->tx_designated_ch = txchan;
// Bytes 38-41
sl->_unused1 = 0xf1;
sl->sign_hold_time = 500 / 25; // 500 msec
sl->prio_sample_time = 2000 / 250; // 2 sec
sl->_unused2 = 0xff;
} }
static void erase_scanlist(int index) static void erase_scanlist(int index)
@ -441,6 +469,24 @@ static void erase_scanlist(int index)
sl->_unused2 = 0xff; sl->_unused2 = 0xff;
} }
//
// Add channel to a zone.
// Return 0 on failure.
//
static int scanlist_append(int list_index, int cnum)
{
scanlist_t *sl = (scanlist_t*) &radio_mem[OFFSET_SCANL + list_index*104];
int i;
for (i=0; i<31; i++) {
if (sl->member[i] == 0) {
sl->member[i] = cnum;
return 1;
}
}
return 0;
}
// //
// Check that the radio does support this frequency. // Check that the radio does support this frequency.
// //
@ -1055,11 +1101,11 @@ static void uv380_print_config(radio_device_t *radio, FILE *out, int verbose)
fprintf(out, "\n"); fprintf(out, "\n");
if (verbose) { if (verbose) {
fprintf(out, "# Table of scan lists.\n"); fprintf(out, "# Table of scan lists.\n");
fprintf(out, "# 1) Zone number: 1-%d\n", NSCANL); fprintf(out, "# 1) Scan list number: 1-%d\n", NSCANL);
fprintf(out, "# 2) Name: up to 16 characters, use '_' instead of space\n"); fprintf(out, "# 2) Name: up to 16 characters, use '_' instead of space\n");
fprintf(out, "# 3) Priority channel 1 (50%% of scans): -, Sel or index\n"); fprintf(out, "# 3) Priority channel 1 (50%% of scans): -, Sel or index\n");
fprintf(out, "# 4) Priority channel 2 (25%% of scans): -, Sel or index\n"); fprintf(out, "# 4) Priority channel 2 (25%% of scans): -, Sel or index\n");
fprintf(out, "# 5) Designated transmit channel: -, Last or index\n"); fprintf(out, "# 5) Designated transmit channel: Last, Sel or index\n");
fprintf(out, "# 6) List of channels: numbers and ranges (N-M) separated by comma\n"); fprintf(out, "# 6) List of channels: numbers and ranges (N-M) separated by comma\n");
fprintf(out, "#\n"); fprintf(out, "#\n");
} }
@ -1093,9 +1139,9 @@ static void uv380_print_config(radio_device_t *radio, FILE *out, int verbose)
fprintf(out, "%-4d ", sl->priority_ch2); fprintf(out, "%-4d ", sl->priority_ch2);
} }
if (sl->tx_designated_ch == 0xffff) { if (sl->tx_designated_ch == 0xffff) {
fprintf(out, "- ");
} else if (sl->tx_designated_ch == 0) {
fprintf(out, "Last "); fprintf(out, "Last ");
} else if (sl->tx_designated_ch == 0) {
fprintf(out, "Sel ");
} else { } else {
fprintf(out, "%-4d ", sl->tx_designated_ch); fprintf(out, "%-4d ", sl->tx_designated_ch);
} }
@ -1248,7 +1294,7 @@ static void uv380_save_image(radio_device_t *radio, FILE *img)
} }
// //
// Erase all channels, zones and scanlists. // Erase all channels.
// //
static void erase_channels() static void erase_channels()
{ {
@ -1257,9 +1303,27 @@ static void erase_channels()
for (i=0; i<NCHAN; i++) { for (i=0; i<NCHAN; i++) {
erase_channel(i); erase_channel(i);
} }
}
//
// Erase all zones.
//
static void erase_zones()
{
int i;
for (i=0; i<NZONES; i++) { for (i=0; i<NZONES; i++) {
erase_zone(i); erase_zone(i);
} }
}
//
// Erase all scanlists.
//
static void erase_scanlists()
{
int i;
for (i=0; i<NSCANL; i++) { for (i=0; i<NSCANL; i++) {
erase_scanlist(i); erase_scanlist(i);
} }
@ -1463,6 +1527,8 @@ badtx: fprintf(stderr, "Bad transmit frequency.\n");
if (first_row && radio->channel_count == 0) { if (first_row && radio->channel_count == 0) {
// On first entry, erase all channels, zones and scanlists. // On first entry, erase all channels, zones and scanlists.
erase_channels(); erase_channels();
erase_zones();
erase_scanlists();
} }
setup_channel(num-1, MODE_DIGITAL, name_str, rx_mhz, tx_mhz, setup_channel(num-1, MODE_DIGITAL, name_str, rx_mhz, tx_mhz,
@ -1633,7 +1699,7 @@ static int parse_zones(int first_row, char *line)
if (first_row) { if (first_row) {
// On first entry, erase the Zones table. // On first entry, erase the Zones table.
memset(&radio_mem[OFFSET_ZONES], 0, NZONES * 0x80); erase_zones();
} }
if (*chan_str == '-') if (*chan_str == '-')
@ -1662,12 +1728,12 @@ static int parse_zones(int first_row, char *line)
// Add range. // Add range.
int c; int c;
for (c=last; c<cnum; c++) { for (c=last; c<cnum; c++) {
setup_zone(bnum-1, c); zone_append(bnum-1, c);
nchan++; nchan++;
} }
} else { } else {
// Add single channel. // Add single channel.
setup_zone(bnum-1, cnum-1); zone_append(bnum-1, cnum-1);
nchan++; nchan++;
} }
@ -1691,12 +1757,115 @@ static int parse_zones(int first_row, char *line)
// //
static int parse_scanlist(int first_row, char *line) static int parse_scanlist(int first_row, char *line)
{ {
//TODO: parse scanlist Name char num_str[256], name_str[256], prio1_str[256], prio2_str[256];
//TODO: parse scanlist PCh1 char tx_str[256], chan_str[256];
//TODO: parse scanlist PCh2 int snum, prio1, prio2, txchan;
//TODO: parse scanlist TxCh
//TODO: parse scanlist Channels if (sscanf(line, "%s %s %s %s %s %s",
num_str, name_str, prio1_str, prio2_str, tx_str, chan_str) != 6)
return 0; return 0;
snum = atoi(num_str);
if (snum < 1 || snum > NSCANL) {
fprintf(stderr, "Bad scan list number.\n");
return 0;
}
if (first_row) {
// On first entry, erase the Scanlists table.
erase_scanlists();
}
if (*prio1_str == '-') {
prio1 = 0xffff;
} else if (strcasecmp("Sel", prio1_str) == 0) {
prio1 = 0;
} else {
prio1 = atoi(prio1_str);
if (prio1 < 1 || prio1 > NCHAN) {
fprintf(stderr, "Bad priority channel 1.\n");
return 0;
}
}
if (*prio2_str == '-') {
prio2 = 0xffff;
} else if (strcasecmp("Sel", prio2_str) == 0) {
prio2 = 0;
} else {
prio2 = atoi(prio2_str);
if (prio2 < 1 || prio2 > NCHAN) {
fprintf(stderr, "Bad priority channel 2.\n");
return 0;
}
}
if (strcasecmp("Last", tx_str) == 0) {
txchan = 0xffff;
} else if (strcasecmp("Sel", tx_str) == 0) {
txchan = 0;
} else {
txchan = atoi(tx_str);
if (txchan < 1 || txchan > NCHAN) {
fprintf(stderr, "Bad transmit channel.\n");
return 0;
}
}
setup_scanlist(snum-1, name_str, prio1, prio2, txchan);
if (*chan_str != '-') {
char *str = chan_str;
int nchan = 0;
int range = 0;
int last = 0;
// Parse channel list.
for (;;) {
char *eptr;
int cnum = strtoul(str, &eptr, 10);
if (eptr == str) {
fprintf(stderr, "Scan list %d: wrong channel list '%s'.\n", snum, str);
return 0;
}
if (cnum < 1 || cnum > NCHAN) {
fprintf(stderr, "Scan list %d: wrong channel number %d.\n", snum, cnum);
return 0;
}
if (range) {
// Add range.
int c;
for (c=last+1; c<=cnum; c++) {
if (!scanlist_append(snum-1, c)) {
fprintf(stderr, "Scan list %d: too many channels.\n", snum);
return 0;
}
nchan++;
}
} else {
// Add single channel.
if (!scanlist_append(snum-1, cnum)) {
fprintf(stderr, "Scan list %d: too many channels.\n", snum);
return 0;
}
nchan++;
}
if (*eptr == 0)
break;
if (*eptr != ',' && *eptr != '-') {
fprintf(stderr, "Scan list %d: wrong channel list '%s'.\n", snum, eptr);
return 0;
}
range = (*eptr == '-');
last = cnum;
str = eptr + 1;
}
}
return 1;
} }
// //