2210 lines
77 KiB
C
2210 lines
77 KiB
C
/*
|
||
* Copyright (C) 2010 DSD Author
|
||
* GPG Key ID: 0x3F1D7FD0 (74EF 430D F7F2 0A48 FCE6 F630 FAA2 635D 3F1D 7FD0)
|
||
*
|
||
* Permission to use, copy, modify, and/or distribute this software for any
|
||
* purpose with or without fee is hereby granted, provided that the above
|
||
* copyright notice and this permission notice appear in all copies.
|
||
*
|
||
* THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
|
||
* REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
|
||
* AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
|
||
* INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
|
||
* LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
|
||
* OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
||
* PERFORMANCE OF THIS SOFTWARE.
|
||
*/
|
||
|
||
#define _MAIN
|
||
|
||
#include "dsd.h"
|
||
#include "p25p1_const.h"
|
||
#include "x2tdma_const.h"
|
||
#include "dstar_const.h"
|
||
#include "nxdn_const.h"
|
||
#include "dmr_const.h"
|
||
#include "provoice_const.h"
|
||
#include "git_ver.h"
|
||
|
||
int pretty_colors()
|
||
{
|
||
fprintf (stderr,"%sred\n", KRED);
|
||
fprintf (stderr,"%sgreen\n", KGRN);
|
||
fprintf (stderr,"%syellow\n", KYEL);
|
||
fprintf (stderr,"%sblue\n", KBLU);
|
||
fprintf (stderr,"%smagenta\n", KMAG);
|
||
fprintf (stderr,"%scyan\n", KCYN);
|
||
fprintf (stderr,"%swhite\n", KWHT);
|
||
fprintf (stderr,"%snormal\n", KNRM);
|
||
|
||
return 0;
|
||
}
|
||
|
||
|
||
#include "p25p1_heuristics.h"
|
||
#include "pa_devs.h"
|
||
|
||
char * FM_banner[9] = {
|
||
" ",
|
||
" ██████╗ ██████╗██████╗ ███████╗███╗ ███╗███████╗",
|
||
" ██╔══██╗██╔════╝██╔══██╗ ██╔════╝████╗ ████║██╔════╝",
|
||
" ██║ ██║╚█████╗ ██║ ██║ █████╗ ██╔████╔██║█████╗ ",
|
||
" ██║ ██║ ╚═══██╗██║ ██║ ██╔══╝ ██║╚██╔╝██║██╔══╝ ",
|
||
" ██████╔╝██████╔╝██████╔╝ ██║ ██║ ╚═╝ ██║███████╗",
|
||
" ╚═════╝ ╚═════╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝╚══════╝",
|
||
" DEV BUILD - May contain changes and broken features "
|
||
};
|
||
|
||
int
|
||
comp (const void *a, const void *b)
|
||
{
|
||
if (*((const int *) a) == *((const int *) b))
|
||
return 0;
|
||
else if (*((const int *) a) < *((const int *) b))
|
||
return -1;
|
||
else
|
||
return 1;
|
||
}
|
||
|
||
//struct for checking existence of directory to write to
|
||
struct stat st = {0};
|
||
char wav_file_directory[1024] = {0};
|
||
unsigned long long int p2vars = 0;
|
||
|
||
char * pEnd; //bugfix
|
||
|
||
void
|
||
noCarrier (dsd_opts * opts, dsd_state * state)
|
||
{
|
||
|
||
//tune back to last known CC when using trunking after x second hangtime
|
||
if (opts->p25_trunk == 1 && opts->p25_is_tuned == 1 && ( (time(NULL) - state->last_cc_sync_time) > opts->trunk_hangtime) )
|
||
{
|
||
if (state->p25_cc_freq != 0)
|
||
{
|
||
|
||
//cap+ rest channel - redundant?
|
||
if (state->dmr_rest_channel != -1)
|
||
{
|
||
if (state->trunk_chan_map[state->dmr_rest_channel] != 0)
|
||
{
|
||
state->p25_cc_freq = state->trunk_chan_map[state->dmr_rest_channel];
|
||
}
|
||
}
|
||
|
||
if (opts->use_rigctl == 1) //rigctl tuning
|
||
{
|
||
if (opts->setmod_bw != 0 ) SetModulation(opts->rigctl_sockfd, opts->setmod_bw);
|
||
SetFreq(opts->rigctl_sockfd, state->p25_cc_freq);
|
||
state->dmr_rest_channel = -1; //maybe?
|
||
}
|
||
|
||
else if (opts->audio_in_type == 3) //rtl_fm tuning
|
||
{
|
||
//UDP command to tune the RTL dongle
|
||
rtl_udp_tune(opts, state, state->p25_cc_freq);
|
||
state->dmr_rest_channel = -1; //maybe?
|
||
}
|
||
|
||
opts->p25_is_tuned = 0;
|
||
state->edacs_tuned_lcn = -1;
|
||
|
||
//only for EDACS/PV
|
||
if (opts->frame_provoice == 1 && opts->wav_out_file != NULL)
|
||
{
|
||
closeWavOutFile(opts, state);
|
||
}
|
||
state->last_cc_sync_time = time(NULL);
|
||
//test to switch back to 10/4 P1 QPSK for P25 FDMA CC
|
||
if (opts->mod_qpsk == 1 && state->p25_cc_is_tdma == 0)
|
||
{
|
||
state->samplesPerSymbol = 10;
|
||
state->symbolCenter = 4;
|
||
}
|
||
}
|
||
//zero out vc frequencies?
|
||
state->p25_vc_freq[0] = 0;
|
||
state->p25_vc_freq[1] = 0;
|
||
}
|
||
|
||
state->dibit_buf_p = state->dibit_buf + 200;
|
||
memset (state->dibit_buf, 0, sizeof (int) * 200);
|
||
//dmr buffer
|
||
state->dmr_payload_p = state->dibit_buf + 200;
|
||
memset (state->dmr_payload_buf, 0, sizeof (int) * 200);
|
||
//dmr buffer end
|
||
if (opts->mbe_out_f != NULL)
|
||
{
|
||
closeMbeOutFile (opts, state);
|
||
}
|
||
state->jitter = -1;
|
||
state->lastsynctype = -1;
|
||
state->carrier = 0;
|
||
state->max = 15000;
|
||
state->min = -15000;
|
||
state->center = 0;
|
||
state->err_str[0] = 0;
|
||
state->err_strR[0] = 0;
|
||
sprintf (state->fsubtype, " ");
|
||
sprintf (state->ftype, " ");
|
||
state->errs = 0;
|
||
state->errs2 = 0;
|
||
|
||
//zero out right away if not trunking
|
||
if (opts->p25_trunk == 0)
|
||
{
|
||
state->lasttg = 0;
|
||
state->lastsrc = 0;
|
||
state->lasttgR = 0;
|
||
state->lastsrcR = 0;
|
||
|
||
//zero out vc frequencies?
|
||
state->p25_vc_freq[0] = 0;
|
||
state->p25_vc_freq[1] = 0;
|
||
|
||
//only reset cap+ rest channel if not trunking
|
||
state->dmr_rest_channel = -1;
|
||
|
||
//zero out nxdn site/srv/cch info if not trunking
|
||
state->nxdn_location_site_code = 0;
|
||
state->nxdn_location_sys_code = 0;
|
||
sprintf (state->nxdn_location_category, "%s", " ");
|
||
|
||
//dmr mfid branding and site parms
|
||
sprintf(state->dmr_branding_sub, "%s", "");
|
||
sprintf(state->dmr_branding, "%s", "");
|
||
sprintf (state->dmr_site_parms, "%s", "");
|
||
}
|
||
|
||
//zero out after x second hangtime when trunking to prevent premature zeroing on these variables
|
||
//mainly bugfix for ncurses and per call wavs (edacs) and also signal fade, etc
|
||
if (opts->p25_trunk == 1 && opts->p25_is_tuned == 1 && time(NULL) - state->last_cc_sync_time > opts->trunk_hangtime)
|
||
{
|
||
state->lasttg = 0;
|
||
state->lastsrc = 0;
|
||
state->lasttgR = 0;
|
||
state->lastsrcR = 0;
|
||
|
||
}
|
||
|
||
state->lastp25type = 0;
|
||
state->repeat = 0;
|
||
state->nac = 0;
|
||
state->numtdulc = 0;
|
||
sprintf (state->slot1light, "%s", "");
|
||
sprintf (state->slot2light, "%s", "");
|
||
state->firstframe = 0;
|
||
if (opts->audio_gain == (float) 0)
|
||
{
|
||
state->aout_gain = 25;
|
||
}
|
||
|
||
if (opts->audio_gainR == (float) 0)
|
||
{
|
||
state->aout_gainR = 25;
|
||
}
|
||
memset (state->aout_max_buf, 0, sizeof (float) * 200);
|
||
state->aout_max_buf_p = state->aout_max_buf;
|
||
state->aout_max_buf_idx = 0;
|
||
|
||
memset (state->aout_max_bufR, 0, sizeof (float) * 200);
|
||
state->aout_max_buf_pR = state->aout_max_bufR;
|
||
state->aout_max_buf_idxR = 0;
|
||
|
||
sprintf (state->algid, "________");
|
||
sprintf (state->keyid, "________________");
|
||
mbe_initMbeParms (state->cur_mp, state->prev_mp, state->prev_mp_enhanced);
|
||
mbe_initMbeParms (state->cur_mp2, state->prev_mp2, state->prev_mp_enhanced2);
|
||
|
||
state->dmr_ms_mode = 0;
|
||
|
||
//not sure if desirable here or not just yet, may need to disable a few of these
|
||
state->payload_mi = 0;
|
||
state->payload_miR = 0;
|
||
state->payload_mfid = 0;
|
||
state->payload_mfidR = 0;
|
||
state->payload_algid = 0;
|
||
state->payload_algidR = 0;
|
||
state->payload_keyid = 0;
|
||
state->payload_keyidR = 0;
|
||
|
||
state->HYTL = 0;
|
||
state->HYTR = 0;
|
||
state->DMRvcL = 0;
|
||
state->DMRvcR = 0;
|
||
state->dropL = 256;
|
||
state->dropR = 256;
|
||
|
||
state->payload_miN = 0;
|
||
state->p25vc = 0;
|
||
state->payload_miP = 0;
|
||
|
||
//initialize dmr data header source
|
||
state->dmr_lrrp_source[0] = 0;
|
||
state->dmr_lrrp_source[1] = 0;
|
||
|
||
|
||
//initialize data header bits
|
||
state->data_header_blocks[0] = 1; //initialize with 1, otherwise we may end up segfaulting when no/bad data header
|
||
state->data_header_blocks[1] = 1; //when trying to fill the superframe and 0-1 blocks give us an overflow
|
||
state->data_header_padding[0] = 0;
|
||
state->data_header_padding[1] = 0;
|
||
state->data_header_format[0] = 0;
|
||
state->data_header_format[1] = 0;
|
||
state->data_header_sap[0] = 0;
|
||
state->data_header_sap[1] = 0;
|
||
state->data_block_counter[0] = 0;
|
||
state->data_block_counter[1] = 0;
|
||
state->data_p_head[0] = 0;
|
||
state->data_p_head[1] = 0;
|
||
|
||
state->dmr_encL = 0;
|
||
state->dmr_encR = 0;
|
||
|
||
state->dmrburstL = 17;
|
||
state->dmrburstR = 17;
|
||
|
||
//reset P2 ESS_B fragments and 4V counter
|
||
for (short i = 0; i < 4; i++)
|
||
{
|
||
state->ess_b[0][i] = 0;
|
||
state->ess_b[1][i] = 0;
|
||
}
|
||
state->fourv_counter[0] = 0;
|
||
state->fourv_counter[1] = 0;
|
||
|
||
//values displayed in ncurses terminal
|
||
state->p25_vc_freq[0] = 0;
|
||
state->p25_vc_freq[1] = 0;
|
||
|
||
//new nxdn stuff
|
||
state->nxdn_part_of_frame = 0;
|
||
state->nxdn_ran = 0;
|
||
state->nxdn_sf = 0;
|
||
memset (state->nxdn_sacch_frame_segcrc, 1, sizeof(state->nxdn_sacch_frame_segcrc)); //init on 1, bad CRC all
|
||
state->nxdn_sacch_non_superframe = TRUE;
|
||
memset (state->nxdn_sacch_frame_segment, 0, sizeof(state->nxdn_sacch_frame_segment));
|
||
state->nxdn_alias_block_number = 0;
|
||
memset (state->nxdn_alias_block_segment, 0, sizeof(state->nxdn_alias_block_segment));
|
||
sprintf (state->nxdn_call_type, "%s", "");
|
||
|
||
//dmr overaching manufacturer in use for a particular system or radio
|
||
// state->dmr_mfid = -1;
|
||
|
||
//dmr slco stuff
|
||
memset(state->dmr_cach_fragment, 1, sizeof(state->dmr_cach_fragment));
|
||
state->dmr_cach_counter = 0;
|
||
|
||
//initialize unified dmr pdu 'superframe'
|
||
memset (state->dmr_pdu_sf, 0, sizeof (state->dmr_pdu_sf));
|
||
|
||
//init confirmed data individual block crc as invalid
|
||
memset (state->data_block_crc_valid, 0, sizeof(state->data_block_crc_valid));
|
||
|
||
//embedded signalling
|
||
memset(state->dmr_embedded_signalling, 0, sizeof(state->dmr_embedded_signalling));
|
||
|
||
//late entry mi fragments
|
||
memset (state->late_entry_mi_fragment, 0, sizeof (state->late_entry_mi_fragment));
|
||
|
||
//dmr talker alias new/fixed stuff
|
||
memset(state->dmr_alias_format, 0, sizeof(state->dmr_alias_format));
|
||
memset(state->dmr_alias_len, 0, sizeof(state->dmr_alias_len));
|
||
memset(state->dmr_alias_block_segment, 0, sizeof(state->dmr_alias_block_segment));
|
||
memset(state->dmr_embedded_gps, 0, sizeof(state->dmr_embedded_gps));
|
||
memset(state->dmr_lrrp_gps, 0, sizeof(state->dmr_lrrp_gps));
|
||
|
||
//multi-purpose call_string
|
||
sprintf (state->call_string[0], "%s", " "); //20 spaces
|
||
sprintf (state->call_string[1], "%s", " "); //20 spaces
|
||
|
||
if (time(NULL) - state->last_cc_sync_time > 30) //thirty seconds of no carrier
|
||
{
|
||
state->dmr_rest_channel = -1;
|
||
state->p25_vc_freq[0] = 0;
|
||
state->p25_vc_freq[1] = 0;
|
||
state->dmr_mfid = -1;
|
||
sprintf(state->dmr_branding_sub, "%s", "");
|
||
sprintf(state->dmr_branding, "%s", "");
|
||
sprintf (state->dmr_site_parms, "%s", "");
|
||
opts->p25_is_tuned = 0;
|
||
}
|
||
|
||
} //nocarrier
|
||
|
||
void
|
||
initOpts (dsd_opts * opts)
|
||
{
|
||
|
||
opts->onesymbol = 10;
|
||
opts->mbe_in_file[0] = 0;
|
||
opts->mbe_in_f = NULL;
|
||
opts->errorbars = 1;
|
||
opts->datascope = 0;
|
||
opts->symboltiming = 0;
|
||
opts->verbose = 2;
|
||
opts->p25enc = 0;
|
||
opts->p25lc = 0;
|
||
opts->p25status = 0;
|
||
opts->p25tg = 0;
|
||
opts->scoperate = 15;
|
||
sprintf (opts->audio_in_dev, "pulse");
|
||
opts->audio_in_fd = -1;
|
||
|
||
sprintf (opts->audio_out_dev, "pulse");
|
||
opts->audio_out_fd = -1;
|
||
|
||
opts->split = 0;
|
||
opts->playoffset = 0;
|
||
opts->playoffsetR = 0;
|
||
opts->mbe_out_dir[0] = 0;
|
||
opts->mbe_out_file[0] = 0;
|
||
opts->mbe_out_path[0] = 0;
|
||
opts->mbe_out_f = NULL;
|
||
opts->audio_gain = 0;
|
||
opts->audio_gainR = 0;
|
||
opts->audio_out = 1;
|
||
opts->wav_out_file[0] = 0;
|
||
opts->wav_out_fileR[0] = 0;
|
||
opts->wav_out_file_raw[0] = 0;
|
||
opts->symbol_out_file[0] = 0;
|
||
opts->lrrp_out_file[0] = 0;
|
||
//csv import filenames
|
||
opts->group_in_file[0] = 0;
|
||
opts->lcn_in_file[0] = 0;
|
||
opts->chan_in_file[0] = 0;
|
||
//end import filenames
|
||
opts->szNumbers[0] = 0;
|
||
opts->symbol_out_f = NULL;
|
||
opts->symbol_out = 0;
|
||
opts->mbe_out = 0;
|
||
opts->wav_out_f = NULL;
|
||
opts->wav_out_fR = NULL;
|
||
opts->wav_out_raw = NULL;
|
||
|
||
opts->dmr_stereo_wav = 0; //flag for per call dmr stereo wav recordings
|
||
//opts->wav_out_fd = -1;
|
||
opts->serial_baud = 115200;
|
||
sprintf (opts->serial_dev, "/dev/ttyUSB0");
|
||
opts->resume = 0;
|
||
opts->frame_dstar = 0;
|
||
opts->frame_x2tdma = 0;
|
||
opts->frame_p25p1 = 1;
|
||
opts->frame_p25p2 = 1;
|
||
opts->frame_nxdn48 = 0;
|
||
opts->frame_nxdn96 = 0;
|
||
opts->frame_dmr = 1;
|
||
opts->frame_dpmr = 0;
|
||
opts->frame_provoice = 0;
|
||
opts->frame_ysf = 0; //forgot to init this, and Cygwin treated as it was turned on.
|
||
opts->mod_c4fm = 1;
|
||
opts->mod_qpsk = 0;
|
||
opts->mod_gfsk = 0;
|
||
opts->uvquality = 3;
|
||
opts->inverted_x2tdma = 1; // most transmitter + scanner + sound card combinations show inverted signals for this
|
||
opts->inverted_dmr = 0; // most transmitter + scanner + sound card combinations show non-inverted signals for this
|
||
opts->mod_threshold = 26;
|
||
opts->ssize = 128; //36 default, max is 128, much cleaner data decodes on Phase 2 cqpsk at max
|
||
opts->msize = 1024; //15 default, max is 1024, much cleaner data decodes on Phase 2 cqpsk at max
|
||
opts->playfiles = 0;
|
||
opts->delay = 0;
|
||
opts->use_cosine_filter = 1;
|
||
opts->unmute_encrypted_p25 = 0;
|
||
opts->rtl_dev_index = 0; //choose which device we want by index number
|
||
opts->rtl_gain_value = 26; //mid value, 0 - AGC (not recommended) 49 highest value
|
||
opts->rtl_squelch_level = 0; //fully open by default, want to specify level for things like NXDN with false positives
|
||
opts->rtl_volume_multiplier = 1; //sample multiplier; 1 seems like a good value
|
||
opts->rtl_udp_port = 6020; //set UDP port for RTL remote
|
||
opts->rtl_bandwidth = 12; //changed recommended default to 12, 24 for ProVoice
|
||
opts->rtlsdr_ppm_error = 0; //initialize ppm with 0 value; bug reported by N.
|
||
opts->rtl_started = 0;
|
||
opts->pulse_raw_rate_in = 48000;
|
||
opts->pulse_raw_rate_out = 48000; //
|
||
opts->pulse_digi_rate_in = 48000;
|
||
opts->pulse_digi_rate_out = 24000; //new default for XDMA
|
||
opts->pulse_raw_in_channels = 1;
|
||
opts->pulse_raw_out_channels = 1;
|
||
opts->pulse_digi_in_channels = 1; //2
|
||
opts->pulse_digi_out_channels = 2; //new default for XDMA
|
||
|
||
opts->wav_sample_rate = 48000; //default value (DSDPlus uses 96000 on raw signal wav files)
|
||
opts->wav_interpolator = 1; //default factor of 1 on 48000; 2 on 96000; sample rate / decimator
|
||
opts->wav_decimator = 48000; //maybe for future use?
|
||
|
||
sprintf (opts->output_name, "XDMA");
|
||
opts->pulse_flush = 1; //set 0 to flush, 1 for flushed
|
||
opts->use_ncurses_terminal = 0;
|
||
opts->ncurses_compact = 0;
|
||
opts->ncurses_history = 1;
|
||
opts->payload = 0;
|
||
opts->inverted_dpmr = 0;
|
||
opts->dmr_mono = 0;
|
||
opts->dmr_stereo = 1;
|
||
opts->aggressive_framesync = 1; //more aggressive to combat wonk wonk voice decoding
|
||
//see if initializing these values causes issues elsewhere, if so, then disable.
|
||
opts->audio_in_type = 0; //this was never initialized, causes issues on rPI 64 (bullseye) if not initialized
|
||
opts->audio_out_type = 0; //this was never initialized, causes issues on rPI 64 (bullseye) if not initialized
|
||
|
||
opts->lrrp_file_output = 0;
|
||
|
||
opts->dmr_mute_encL = 1;
|
||
opts->dmr_mute_encR = 1;
|
||
|
||
opts->monitor_input_audio = 0;
|
||
|
||
opts->inverted_p2 = 0;
|
||
opts->p2counter = 0;
|
||
|
||
opts->call_alert = 0; //call alert beeper for ncurses
|
||
|
||
//rigctl options
|
||
opts->use_rigctl = 0;
|
||
opts->rigctl_sockfd = 0;
|
||
opts->rigctlportno = 7356; //TCP Port Number; GQRX - 7356; SDR++ - 4532
|
||
opts->rigctlhostname = "localhost";
|
||
|
||
//udp input options
|
||
opts->udp_sockfd = 0;
|
||
opts->udp_portno = 7355; //default favored by GQRX and SDR++
|
||
opts->udp_hostname = "localhost";
|
||
|
||
//tcp input options
|
||
opts->tcp_sockfd = 0;
|
||
opts->tcp_portno = 7355; //default favored by SDR++
|
||
sprintf (opts->tcp_hostname, "%s", "localhost");
|
||
|
||
opts->p25_trunk = 0; //0 disabled, 1 is enabled
|
||
opts->p25_is_tuned = 0; //set to 1 if currently on VC, set back to 0 on carrier drop
|
||
opts->trunk_hangtime = 1; //1 second hangtime by default before tuning back to CC
|
||
|
||
//reverse mute
|
||
opts->reverse_mute = 0;
|
||
|
||
//setmod bandwidth
|
||
opts->setmod_bw = 0; //default to 0 - off
|
||
|
||
//DMR Location Area - DMRLA B***S***
|
||
opts->dmr_dmrla_is_set = 0;
|
||
opts->dmr_dmrla_n = 0;
|
||
|
||
} //initopts
|
||
|
||
void
|
||
initState (dsd_state * state)
|
||
{
|
||
|
||
int i, j;
|
||
// state->testcounter = 0;
|
||
state->last_dibit = 0;
|
||
state->dibit_buf = malloc (sizeof (int) * 1000000);
|
||
state->dibit_buf_p = state->dibit_buf + 200;
|
||
memset (state->dibit_buf, 0, sizeof (int) * 200);
|
||
//dmr buffer -- double check this set up
|
||
state->dmr_payload_buf = malloc (sizeof (int) * 1000000);
|
||
state->dmr_payload_p = state->dmr_payload_buf + 200;
|
||
memset (state->dmr_payload_buf, 0, sizeof (int) * 200);
|
||
//dmr buffer end
|
||
state->repeat = 0;
|
||
state->audio_out_buf = malloc (sizeof (short) * 1000000);
|
||
state->audio_out_bufR = malloc (sizeof (short) * 1000000);
|
||
memset (state->audio_out_buf, 0, 100 * sizeof (short));
|
||
memset (state->audio_out_bufR, 0, 100 * sizeof (short));
|
||
state->audio_out_buf_p = state->audio_out_buf + 100;
|
||
state->audio_out_buf_pR = state->audio_out_bufR + 100;
|
||
state->audio_out_float_buf = malloc (sizeof (float) * 1000000);
|
||
state->audio_out_float_bufR = malloc (sizeof (float) * 1000000);
|
||
memset (state->audio_out_float_buf, 0, 100 * sizeof (float));
|
||
memset (state->audio_out_float_bufR, 0, 100 * sizeof (float));
|
||
state->audio_out_float_buf_p = state->audio_out_float_buf + 100;
|
||
state->audio_out_float_buf_pR = state->audio_out_float_bufR + 100;
|
||
state->audio_out_idx = 0;
|
||
state->audio_out_idx2 = 0;
|
||
state->audio_out_idxR = 0;
|
||
state->audio_out_idx2R = 0;
|
||
state->audio_out_temp_buf_p = state->audio_out_temp_buf;
|
||
state->audio_out_temp_buf_pR = state->audio_out_temp_bufR;
|
||
//state->wav_out_bytes = 0;
|
||
state->center = 0;
|
||
state->jitter = -1;
|
||
state->synctype = -1;
|
||
state->min = -15000;
|
||
state->max = 15000;
|
||
state->lmid = 0;
|
||
state->umid = 0;
|
||
state->minref = -12000;
|
||
state->maxref = 12000;
|
||
state->lastsample = 0;
|
||
for (i = 0; i < 128; i++)
|
||
{
|
||
state->sbuf[i] = 0;
|
||
}
|
||
state->sidx = 0;
|
||
for (i = 0; i < 1024; i++)
|
||
{
|
||
state->maxbuf[i] = 15000;
|
||
}
|
||
for (i = 0; i < 1024; i++)
|
||
{
|
||
state->minbuf[i] = -15000;
|
||
}
|
||
state->midx = 0;
|
||
state->err_str[0] = 0;
|
||
state->err_strR[0] = 0;
|
||
sprintf (state->fsubtype, " ");
|
||
sprintf (state->ftype, " ");
|
||
state->symbolcnt = 0;
|
||
state->symbolc = 0; //
|
||
state->rf_mod = 0;
|
||
state->numflips = 0;
|
||
state->lastsynctype = -1;
|
||
state->lastp25type = 0;
|
||
state->offset = 0;
|
||
state->carrier = 0;
|
||
for (i = 0; i < 25; i++)
|
||
{
|
||
for (j = 0; j < 16; j++)
|
||
{
|
||
state->tg[i][j] = 48;
|
||
}
|
||
}
|
||
state->tgcount = 0;
|
||
state->lasttg = 0;
|
||
state->lastsrc = 0;
|
||
state->lasttgR = 0;
|
||
state->lastsrcR = 0;
|
||
state->nac = 0;
|
||
state->errs = 0;
|
||
state->errs2 = 0;
|
||
state->mbe_file_type = -1;
|
||
state->optind = 0;
|
||
state->numtdulc = 0;
|
||
state->firstframe = 0;
|
||
sprintf (state->slot1light, "%s", "");
|
||
sprintf (state->slot2light, "%s", "");
|
||
state->aout_gain = 25;
|
||
memset (state->aout_max_buf, 0, sizeof (float) * 200);
|
||
state->aout_max_buf_p = state->aout_max_buf;
|
||
state->aout_max_buf_idx = 0;
|
||
|
||
memset (state->aout_max_bufR, 0, sizeof (float) * 200);
|
||
state->aout_max_buf_pR = state->aout_max_bufR;
|
||
state->aout_max_buf_idxR = 0;
|
||
|
||
state->samplesPerSymbol = 10;
|
||
state->symbolCenter = 4;
|
||
sprintf (state->algid, "________");
|
||
sprintf (state->keyid, "________________");
|
||
state->currentslot = 0;
|
||
state->cur_mp = malloc (sizeof (mbe_parms));
|
||
state->prev_mp = malloc (sizeof (mbe_parms));
|
||
state->prev_mp_enhanced = malloc (sizeof (mbe_parms));
|
||
|
||
state->cur_mp2 = malloc (sizeof (mbe_parms));
|
||
state->prev_mp2 = malloc (sizeof (mbe_parms));
|
||
state->prev_mp_enhanced2 = malloc (sizeof (mbe_parms));
|
||
|
||
mbe_initMbeParms (state->cur_mp, state->prev_mp, state->prev_mp_enhanced);
|
||
mbe_initMbeParms (state->cur_mp2, state->prev_mp2, state->prev_mp_enhanced2);
|
||
state->p25kid = 0;
|
||
|
||
state->debug_audio_errors = 0;
|
||
state->debug_audio_errorsR = 0;
|
||
state->debug_header_errors = 0;
|
||
state->debug_header_critical_errors = 0;
|
||
|
||
state->nxdn_last_ran = -1;
|
||
state->nxdn_last_rid = 0;
|
||
state->nxdn_last_tg = 0;
|
||
state->nxdn_cipher_type = 0;
|
||
state->nxdn_key = 0;
|
||
sprintf (state->nxdn_call_type, "%s", "");
|
||
state->payload_miN = 0;
|
||
|
||
state->dpmr_color_code = -1;
|
||
state->dpmr_caller_id = 0;
|
||
state->dpmr_target_id = 0;
|
||
|
||
state->payload_mi = 0;
|
||
state->payload_miR = 0;
|
||
state->payload_mfid = 0;
|
||
state->payload_mfidR = 0;
|
||
state->payload_algid = 0;
|
||
state->payload_algidR = 0;
|
||
state->payload_keyid = 0;
|
||
state->payload_keyidR = 0;
|
||
|
||
//init P2 ESS_B fragments and 4V counter
|
||
for (short i = 0; i < 4; i++)
|
||
{
|
||
state->ess_b[0][i] = 0;
|
||
state->ess_b[1][i] = 0;
|
||
}
|
||
state->fourv_counter[0] = 0;
|
||
state->fourv_counter[1] = 0;
|
||
|
||
state->K = 0;
|
||
state->R = 0;
|
||
state->RR = 0;
|
||
state->H = 0;
|
||
state->K1 = 0;
|
||
state->K2 = 0;
|
||
state->K3 = 0;
|
||
state->K4 = 0;
|
||
state->M = 0; //force key priority over settings from fid/so
|
||
|
||
state->dmr_stereo = 0; //1, or 0?
|
||
state->dmrburstL = 17; //initialize at higher value than possible
|
||
state->dmrburstR = 17; //17 in char array is set for ERR
|
||
state->dmr_so = 0;
|
||
state->dmr_soR = 0;
|
||
state->dmr_fid = 0;
|
||
state->dmr_fidR = 0;
|
||
state->dmr_ms_mode = 0;
|
||
|
||
state->HYTL = 0;
|
||
state->HYTR = 0;
|
||
state->DMRvcL = 0;
|
||
state->DMRvcR = 0;
|
||
state->dropL = 256;
|
||
state->dropR = 256;
|
||
|
||
state->p25vc = 0;
|
||
|
||
state->payload_miP = 0;
|
||
|
||
memset(state->dstarradioheader, 0, 41);
|
||
|
||
//initialize dmr data header source
|
||
state->dmr_lrrp_source[0] = 0;
|
||
state->dmr_lrrp_source[1] = 0;
|
||
|
||
|
||
//initialize data header bits
|
||
state->data_header_blocks[0] = 1; //initialize with 1, otherwise we may end up segfaulting when no/bad data header
|
||
state->data_header_blocks[1] = 1; //when trying to fill the superframe and 0-1 blocks give us an overflow
|
||
state->data_header_padding[0] = 0;
|
||
state->data_header_padding[1] = 0;
|
||
state->data_header_format[0] = 0;
|
||
state->data_header_format[1] = 0;
|
||
state->data_header_sap[0] = 0;
|
||
state->data_header_sap[1] = 0;
|
||
state->data_block_counter[0] = 0;
|
||
state->data_block_counter[1] = 0;
|
||
state->data_p_head[0] = 0;
|
||
state->data_p_head[1] = 0;
|
||
|
||
state->menuopen = 0; //is the ncurses menu open, if so, don't process frame sync
|
||
|
||
state->dmr_encL = 0;
|
||
state->dmr_encR = 0;
|
||
|
||
//P2 variables
|
||
state->p2_wacn = 0;
|
||
state->p2_sysid = 0;
|
||
state->p2_cc = 0;
|
||
state->p2_hardset = 0;
|
||
state->p2_is_lcch = 0;
|
||
state->p25_cc_is_tdma = 2; //init on 2, TSBK NET_STS will set 0, TDMA NET_STS will set 1. //used to determine if we need to change symbol rate when cc hunting
|
||
|
||
//experimental symbol file capture read throttle
|
||
state->symbol_throttle = 0; //throttle speed
|
||
state->use_throttle = 0; //only use throttle if set to 1
|
||
|
||
state->p2_scramble_offset = 0;
|
||
state->p2_vch_chan_num = 0;
|
||
|
||
//p25 iden_up values
|
||
state->p25_chan_iden = 0;
|
||
for (int i = 0; i < 16; i++)
|
||
{
|
||
state->p25_chan_type[i] = 0;
|
||
state->p25_trans_off[i] = 0;
|
||
state->p25_chan_spac[i] = 0;
|
||
state->p25_base_freq[i] = 0;
|
||
}
|
||
|
||
//values displayed in ncurses terminal
|
||
state->p25_cc_freq = 0;
|
||
state->p25_vc_freq[0] = 0;
|
||
state->p25_vc_freq[1] = 0;
|
||
|
||
//edacs - may need to make these user configurable instead for stability on non-ea systems
|
||
state->ea_mode = -1; //init on -1, 0 is standard, 1 is ea
|
||
state->esk_mode = -1; //same as above, but with esk or not
|
||
state->esk_mask = 0x0; //toggles from 0x0 to 0xA0 if esk mode enabled
|
||
state->edacs_site_id = 0;
|
||
state->edacs_lcn_count = 0;
|
||
state->edacs_cc_lcn = 0;
|
||
state->edacs_vc_lcn = 0;
|
||
state->edacs_tuned_lcn = -1;
|
||
|
||
//trunking
|
||
memset (state->trunk_lcn_freq, 0, sizeof(state->trunk_lcn_freq));
|
||
memset (state->trunk_chan_map, 0, sizeof(state->trunk_chan_map));
|
||
state->group_tally = 0;
|
||
state->lcn_freq_count = 0; //number of frequncies imported as an enumerated lcn list
|
||
state->lcn_freq_roll = 0; //needs reset if sync is found?
|
||
state->last_cc_sync_time = time(NULL);
|
||
state->last_vc_sync_time = time(NULL);
|
||
state->is_con_plus = 0;
|
||
|
||
//dmr trunking/ncurses stuff
|
||
state->dmr_rest_channel = -1; //init on -1
|
||
state->dmr_mfid = -1; //
|
||
state->dmr_tuned_lcn = -1; //logical slot, lcn * ts?
|
||
state->dmr_vc_lcn = -1; //
|
||
state->dmr_vc_lsn = -1;
|
||
|
||
//new nxdn stuff
|
||
state->nxdn_part_of_frame = 0;
|
||
state->nxdn_ran = 0;
|
||
state->nxdn_sf = 0;
|
||
memset (state->nxdn_sacch_frame_segcrc, 1, sizeof(state->nxdn_sacch_frame_segcrc)); //init on 1, bad CRC all
|
||
state->nxdn_sacch_non_superframe = TRUE;
|
||
memset (state->nxdn_sacch_frame_segment, 0, sizeof(state->nxdn_sacch_frame_segment));
|
||
state->nxdn_alias_block_number = 0;
|
||
memset (state->nxdn_alias_block_segment, 0, sizeof(state->nxdn_alias_block_segment));
|
||
//site/srv/cch info
|
||
state->nxdn_location_site_code = 0;
|
||
state->nxdn_location_sys_code = 0;
|
||
sprintf (state->nxdn_location_category, "%s", " ");
|
||
|
||
//Remus DMR End Call Alert Beep
|
||
state->dmr_end_alert[0] = 0;
|
||
state->dmr_end_alert[1] = 0;
|
||
|
||
sprintf (state->dmr_branding, "%s", "");
|
||
sprintf (state->dmr_branding_sub, "%s", "");
|
||
sprintf (state->dmr_site_parms, "%s", "");
|
||
|
||
//initialize unified dmr pdu 'superframe'
|
||
memset (state->dmr_pdu_sf, 0, sizeof (state->dmr_pdu_sf));
|
||
|
||
//init confirmed data individual block crc as invalid
|
||
memset (state->data_block_crc_valid, 0, sizeof(state->data_block_crc_valid));
|
||
|
||
//dmr slco stuff
|
||
memset(state->dmr_cach_fragment, 1, sizeof(state->dmr_cach_fragment));
|
||
state->dmr_cach_counter = 0;
|
||
|
||
//embedded signalling
|
||
memset(state->dmr_embedded_signalling, 0, sizeof(state->dmr_embedded_signalling));
|
||
|
||
//dmr talker alias new/fixed stuff
|
||
memset(state->dmr_alias_format, 0, sizeof(state->dmr_alias_format));
|
||
memset(state->dmr_alias_len, 0, sizeof(state->dmr_alias_len));
|
||
memset(state->dmr_alias_block_segment, 0, sizeof(state->dmr_alias_block_segment));
|
||
memset(state->dmr_embedded_gps, 0, sizeof(state->dmr_embedded_gps));
|
||
memset(state->dmr_lrrp_gps, 0, sizeof(state->dmr_lrrp_gps));
|
||
|
||
//multi-purpose call_string
|
||
sprintf (state->call_string[0], "%s", " "); //20 spaces
|
||
sprintf (state->call_string[1], "%s", " "); //20 spaces
|
||
|
||
//late entry mi fragments
|
||
memset (state->late_entry_mi_fragment, 0, sizeof (state->late_entry_mi_fragment));
|
||
|
||
initialize_p25_heuristics(&state->p25_heuristics);
|
||
|
||
} //init_state
|
||
|
||
void
|
||
usage ()
|
||
{
|
||
|
||
printf ("\n");
|
||
printf ("Usage: dsd-fme [options] Decoder/Trunking Mode\n");
|
||
printf (" or: dsd-fme [options] -r <files> Read/Play saved mbe data from file(s)\n");
|
||
printf (" or: dsd-fme -h Show help\n");
|
||
printf ("\n");
|
||
printf ("Display Options:\n");
|
||
printf (" -N Use NCurses Terminal\n");
|
||
printf (" dsd-fme -N 2> log.ans \n");
|
||
printf (" -Z Log MBE/PDU Payloads to console\n");
|
||
// printf (" -e Show Frame Info and errorbars (default)\n");
|
||
// printf (" -pe Show P25 encryption sync bits\n");
|
||
// printf (" -pl Show P25 link control bits\n");
|
||
// printf (" -ps Show P25 status bits and low speed data\n");
|
||
// printf (" -pt Show P25 talkgroup info\n");
|
||
// printf (" -q Don't show Frame Info/errorbars\n");
|
||
//printf (" -s Datascope (disables other display options)\n");
|
||
//printf (" -t Show symbol timing during sync\n");
|
||
// printf (" -v <num> Frame information Verbosity\n");
|
||
// printf (" -z <num> Frame rate for datascope\n");
|
||
printf ("\n");
|
||
printf ("Input/Output options:\n");
|
||
printf (" -i <device> Audio input device (default is pulse audio)\n");
|
||
printf (" - for piped stdin\n");
|
||
printf (" rtl:dev:freq:gain:ppm:bw:sq:udp for rtl dongle (see below)\n");
|
||
printf (" tcp for tcp client SDR++/GNURadio Companion/Other (Port 7355)\n");
|
||
printf (" tcp:192.168.7.5:7355 for custom address and port \n");
|
||
printf (" filename.bin for OP25/FME capture bin files\n");
|
||
printf (" filename.wav for 48K/1 wav files (SDR++, GQRX)\n");
|
||
printf (" filename.wav -s 96000 for 96K/1 wav files (DSDPlus)\n");
|
||
printf (" (Use single quotes '/directory/audio file.wav' when directories/spaces are present)\n");
|
||
printf (" -s <rate> Sample Rate of wav input files (usually 48000 or 96000) Mono only!\n");
|
||
printf (" -o <device> Audio output device (default is pulse audio)(null for no audio output)\n");
|
||
printf (" -d <dir> Create mbe data files, use this directory\n");
|
||
printf (" -r <files> Read/Play saved mbe data from file(s)\n");
|
||
printf (" -g <num> Audio output gain (default = 0 = auto, disable = -1)\n");
|
||
printf (" -w <file> Output synthesized speech to a .wav file, legacy auto modes only.\n");
|
||
printf (" -P Enable Per Call WAV file saving in XDMA and NXDN decoding classes\n");
|
||
printf (" (Per Call can only be used in Ncurses Terminal!)\n");
|
||
printf (" (Running in console will use static wav files)\n");
|
||
printf (" -a Enable Call Alert Beep (NCurses Terminal Only)\n");
|
||
printf (" (Warning! Might be annoying.)\n");
|
||
printf (" -L <file> Specify Filename for LRRP Data Output.\n");
|
||
printf (" -c <file> Output symbol capture to .bin file\n");
|
||
printf (" -n Throttle Symbol Capture Bin Input\n");
|
||
printf (" (useful when reading files still being written to by OP25)");
|
||
printf ("\n");
|
||
printf ("RTL-SDR options:\n");
|
||
printf (" WARNING! Old CLI Switch Handling has been depreciated in favor of rtl:<parms>\n");
|
||
printf (" Usage: rtl:dev:freq:gain:ppm:bw:sq:udp\n");
|
||
printf (" dev <num> RTL-SDR Device Index Number\n");
|
||
printf (" freq <num> RTL-SDR Frequency\n");
|
||
printf (" gain <num> RTL-SDR Device Gain (0-49) (default = 0 Auto Gain)\n");
|
||
printf (" ppm <num> RTL-SDR PPM Error (default = 0)\n");
|
||
printf (" bw <num> RTL-SDR VFO Bandwidth kHz (default = 26)(0 = Hardware AGC, not recommended) \n");
|
||
printf (" sq <num> RTL-SDR Squelch Level (0 - Open, 25 - Little, 50 - Higher)\n");
|
||
// printf (" -V <num> RTL-SDR Sample Gain Multiplier (default = 1)\n");
|
||
printf (" udp <num> RTL-SDR UDP Remote Port (default = 6020)\n");
|
||
printf (" Example: dsd-fme -fp -i rtl:0:851.375M:22:-2:12:0:6021\n");
|
||
printf ("\n");
|
||
// printf ("Scanner control options:\n");
|
||
// printf (" -B <num> Serial port baud rate (default=115200)\n");
|
||
// printf (" -C <device> Serial port for scanner control (default=/dev/ttyUSB0)\n");
|
||
// printf (" -R <num> Resume scan after <num> TDULC frames or any PDU or TSDU\n");
|
||
// printf ("\n");
|
||
printf ("Decoder options:\n");
|
||
printf (" -fa Legacy Auto Detection (old methods default)\n");
|
||
printf (" -ft XDMA P25 and DMR BS/MS frame types (new default)\n");
|
||
printf (" -fs DMR Stereo BS and MS Simplex only\n");
|
||
printf (" -f1 Decode only P25 Phase 1\n");
|
||
printf (" -fd Decode only D-STAR\n");
|
||
printf (" -fr Decode only DMR Mono - Single Slot Voice\n");
|
||
printf (" -fx Decode only X2-TDMA\n");
|
||
printf (" -fi Decode only NXDN48* (6.25 kHz) / IDAS*\n");
|
||
printf (" -fn Decode only NXDN96* (12.5 kHz)\n");
|
||
printf (" -fp Decode only EDACS/ProVoice*\n");
|
||
printf (" -fm Decode only dPMR*\n");
|
||
printf (" -l Disable DMR and NXDN input filtering\n");
|
||
printf (" -pu Unmute Encrypted P25\n");
|
||
printf (" -u <num> Unvoiced speech quality (default=3)\n");
|
||
printf (" -xx Expect non-inverted X2-TDMA signal\n");
|
||
printf (" -xr Expect inverted DMR signal\n");
|
||
printf (" -xd Expect inverted ICOM dPMR signal\n");
|
||
printf ("\n");
|
||
printf (" * denotes frame types that cannot be auto-detected.\n");
|
||
printf ("\n");
|
||
printf ("Advanced Decoder options:\n");
|
||
printf (" -X <hex> Manually Set P2 Parameters (WACN, SYSID, CC/NAC)\n");
|
||
printf (" (-X BEE00ABC123)\n");
|
||
printf (" -D <dec> Manually Set TIII DMR Location Area n bit len (0-10)(10 max)\n");
|
||
printf (" (Value defaults to max n bit value for site model size)\n");
|
||
printf (" (Setting 0 will show full Site ID, no area/subarea)\n");
|
||
printf ("\n");
|
||
printf (" -A <num> QPSK modulation auto detection threshold (default=26)\n");
|
||
printf (" -S <num> Symbol buffer size for QPSK decision point tracking\n");
|
||
printf (" (default=36)\n");
|
||
printf (" -M <num> Min/Max buffer size for QPSK decision point tracking\n");
|
||
printf (" (default=15)\n");
|
||
printf (" -ma Auto-select modulation optimizations (default)\n");
|
||
printf (" -mc Use only C4FM modulation optimizations\n");
|
||
printf (" -mg Use only GFSK modulation optimizations\n");
|
||
printf (" -mq Use only QPSK modulation optimizations\n");
|
||
printf (" -m2 Use P25p2 6000 sps CQPSK modulation optimizations\n");
|
||
//printf (" (4 Level, not 8 Level LSM) (this is honestly unknown since I can't verify what local systems are using)\n");
|
||
printf (" -F Relax P25 Phase 2 MAC_SIGNAL CRC Checksum Pass/Fail\n");
|
||
printf (" Use this feature to allow MAC_SIGNAL even if bad CRC errors.\n");
|
||
printf (" -F Relax DMR CACH/Burst FEC Pass/Fail - Passive Frame Sync\n");
|
||
printf (" Use if skipping occurs, but may cause wonky audio due to loss of good sync\n");
|
||
|
||
printf ("\n");
|
||
printf (" -K <dec> Manually Enter Basic Privacy Key (Decimal Value of Key Number)\n");
|
||
printf ("\n");
|
||
printf (" -H <hex> Manually Enter **tera 10/32/64 Char Privacy Hex Key (see example below)\n");
|
||
printf (" Encapulate in Single Quotation Marks; Space every 16 chars.\n");
|
||
printf (" -H 0B57935150 \n");
|
||
printf (" -H '736B9A9C5645288B 243AD5CB8701EF8A' \n");
|
||
printf (" -H '20029736A5D91042 C923EB0697484433 005EFC58A1905195 E28E9C7836AA2DB8' \n");
|
||
printf ("\n");
|
||
printf (" -R <dec> Manually Enter NXDN 4800/9600 EHR Scrambler Key Value (Decimal Value)\n");
|
||
printf (" \n");
|
||
printf (" -4 Force Privacy Key over FID and SVC bits \n");
|
||
printf ("\n");
|
||
printf (" New and Experimental Functions and Features---------------------------------------------------\n");
|
||
printf (" -C <file> Import Channel to Frequency Map (channum, freq) from csv file. (Capital C) \n");
|
||
printf (" (See channel_map.csv for example)\n");
|
||
printf (" -G <file> Import Group List Allow/Block and Label from csv file.\n");
|
||
printf (" (See group.csv for example)\n");
|
||
printf (" -T Enable Trunking Features (NXDN/P25/EDACS/DMR) with RIGCTL/TCP or RTL Input\n");
|
||
printf (" -U <port> Enable RIGCTL/TCP; Set TCP Port for RIGCTL. (4532 on SDR++)\n");
|
||
printf (" -B <Hertz> Set RIGCTL Setmod Bandwidth in Hertz (0 - default - OFF)\n");
|
||
printf (" P25 - 7000; NXDN48 - 4000; DMR - 7000; EDACS/PV - 12500; May vary based on system stregnth, etc.\n");
|
||
printf (" -t <secs> Set Trunking VC/sync loss hangtime in seconds. (default = 1 second)\n");
|
||
printf (" -q Reverse Mute - Mute Unencrypted Voice and Unmute Encrypted Voice\n");
|
||
printf ("\n");
|
||
exit (0);
|
||
}
|
||
|
||
void
|
||
liveScanner (dsd_opts * opts, dsd_state * state)
|
||
{
|
||
|
||
if (opts->audio_in_type == 1)
|
||
{
|
||
opts->pulse_digi_rate_out = 8000; //revert to 8K/1 for STDIN input, sometimes crackles when upsampling
|
||
opts->pulse_digi_out_channels = 1;
|
||
if (opts->dmr_stereo == 1)
|
||
{
|
||
opts->pulse_digi_rate_out = 24000; //stdin needs 24000 by 2 channel for DMR TDMA Stereo output
|
||
opts->pulse_digi_out_channels = 2;
|
||
fprintf (stderr, "STDIN Audio Rate Out set to 24000 Khz/2 Channel \n");
|
||
}
|
||
else fprintf (stderr, "STDIN Audio Rate Out set to 48000 Khz/1 Channel \n");
|
||
opts->pulse_raw_rate_out = 48000;
|
||
opts->pulse_raw_out_channels = 1;
|
||
|
||
}
|
||
|
||
#ifdef USE_RTLSDR
|
||
if(opts->audio_in_type == 3)
|
||
{
|
||
//still need this section mostly due the the crackling on the rtl dongle when upsampled
|
||
//probably need to dig a little deeper, maybe inlvl releated?
|
||
opts->pulse_digi_rate_out = 8000; //revert to 8K/1 for RTL input, random crackling otherwise
|
||
opts->pulse_digi_out_channels = 1;
|
||
if (opts->dmr_stereo == 1)
|
||
{
|
||
opts->pulse_digi_rate_out = 24000; //rtl needs 24000 by 2 channel for DMR TDMA Stereo output
|
||
opts->pulse_digi_out_channels = 2; //minimal crackling 'may' be observed, not sure, can't test to see on DMR with RTL
|
||
fprintf (stderr, "RTL Audio Rate Out set to 24000 Khz/2 Channel \n");
|
||
}
|
||
else fprintf (stderr, "RTL Audio Rate Out set to 48000 Khz/1 Channel \n");
|
||
opts->pulse_raw_rate_out = 48000;
|
||
opts->pulse_raw_out_channels = 1;
|
||
|
||
open_rtlsdr_stream(opts);
|
||
opts->rtl_started = 1; //set here so ncurses terminal doesn't attempt to open it again
|
||
}
|
||
#endif
|
||
|
||
if (opts->use_ncurses_terminal == 1)
|
||
{
|
||
ncursesOpen(opts, state);
|
||
}
|
||
|
||
if (opts->audio_in_type == 0)
|
||
{
|
||
openPulseInput(opts);
|
||
}
|
||
|
||
if (opts->audio_out_type == 0)
|
||
{
|
||
openPulseOutput(opts);
|
||
}
|
||
|
||
while (1)
|
||
{
|
||
|
||
noCarrier (opts, state);
|
||
if (state->menuopen == 0)
|
||
{
|
||
state->synctype = getFrameSync (opts, state);
|
||
// recalibrate center/umid/lmid
|
||
state->center = ((state->max) + (state->min)) / 2;
|
||
state->umid = (((state->max) - state->center) * 5 / 8) + state->center;
|
||
state->lmid = (((state->min) - state->center) * 5 / 8) + state->center;
|
||
}
|
||
|
||
|
||
|
||
while (state->synctype != -1)
|
||
{
|
||
processFrame (opts, state);
|
||
|
||
#ifdef TRACE_DSD
|
||
state->debug_prefix = 'S';
|
||
#endif
|
||
|
||
// state->synctype = getFrameSync (opts, state);
|
||
|
||
#ifdef TRACE_DSD
|
||
state->debug_prefix = '\0';
|
||
#endif
|
||
|
||
// // recalibrate center/umid/lmid
|
||
// state->center = ((state->max) + (state->min)) / 2;
|
||
// state->umid = (((state->max) - state->center) * 5 / 8) + state->center;
|
||
// state->lmid = (((state->min) - state->center) * 5 / 8) + state->center;
|
||
if (state->menuopen == 0)
|
||
{
|
||
state->synctype = getFrameSync (opts, state);
|
||
// recalibrate center/umid/lmid
|
||
state->center = ((state->max) + (state->min)) / 2;
|
||
state->umid = (((state->max) - state->center) * 5 / 8) + state->center;
|
||
state->lmid = (((state->min) - state->center) * 5 / 8) + state->center;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
void
|
||
cleanupAndExit (dsd_opts * opts, dsd_state * state)
|
||
{
|
||
noCarrier (opts, state);
|
||
if (opts->wav_out_f != NULL)
|
||
{
|
||
closeWavOutFile (opts, state);
|
||
}
|
||
if (opts->wav_out_raw != NULL)
|
||
{
|
||
closeWavOutFileRaw (opts, state);
|
||
}
|
||
if (opts->dmr_stereo_wav == 1) //cause of crash on exit, need to check if NULL first, may need to set NULL when turning off in nterm
|
||
{
|
||
closeWavOutFileL (opts, state);
|
||
closeWavOutFileR (opts, state);
|
||
}
|
||
if (opts->symbol_out == 1)
|
||
{
|
||
closeSymbolOutFile (opts, state);
|
||
}
|
||
if (opts->mbe_out_f != NULL)
|
||
{
|
||
closeMbeOutFile (opts, state);
|
||
}
|
||
|
||
fprintf (stderr,"\n");
|
||
fprintf (stderr,"Total audio errors: %i\n", state->debug_audio_errors);
|
||
fprintf (stderr,"Total header errors: %i\n", state->debug_header_errors);
|
||
fprintf (stderr,"Total irrecoverable header errors: %i\n", state->debug_header_critical_errors);
|
||
fprintf (stderr,"Exiting.\n");
|
||
exit (0);
|
||
}
|
||
|
||
// void
|
||
// sigfun (int sig) //not convinced this is truly required, maybe some environments require it?
|
||
// {
|
||
// exitflag = 1;
|
||
// #ifdef USE_RTLSDR
|
||
// // if (opts->audio_in_type == 3)
|
||
// //rtlsdr_sighandler();
|
||
// #endif
|
||
// signal (SIGINT, SIG_DFL);
|
||
// ncursesClose ();
|
||
|
||
// // #ifdef USE_RTLSDR
|
||
// // if (opts->audio_in_type == 3) rtlsdr_sighandler();
|
||
// // #endif
|
||
// }
|
||
|
||
double atofs(char *s)
|
||
{
|
||
char last;
|
||
int len;
|
||
double suff = 1.0;
|
||
len = strlen(s);
|
||
last = s[len-1];
|
||
s[len-1] = '\0';
|
||
switch (last) {
|
||
case 'g':
|
||
case 'G':
|
||
suff *= 1e3;
|
||
case 'm':
|
||
case 'M':
|
||
suff *= 1e3;
|
||
case 'k':
|
||
case 'K':
|
||
suff *= 1e3;
|
||
suff *= atof(s);
|
||
s[len-1] = last;
|
||
return suff;
|
||
}
|
||
s[len-1] = last;
|
||
return atof(s);
|
||
}
|
||
|
||
int
|
||
main (int argc, char **argv)
|
||
{
|
||
int c;
|
||
extern char *optarg;
|
||
extern int optind, opterr, optopt;
|
||
dsd_opts opts;
|
||
dsd_state state;
|
||
char versionstr[25];
|
||
mbe_printVersion (versionstr);
|
||
|
||
fprintf (stderr," Digital Speech Decoder: Florida Man Edition\n");
|
||
for (short int i = 1; i < 8; i++) {
|
||
//fprintf (stderr,"%s%s \n", FM_banner[i], KGRN); //cyan/blue text
|
||
fprintf (stderr,"%s\n", FM_banner[i]); //cyan/blue tex
|
||
}
|
||
//fprintf (stderr,"%s", KNRM); //change back to normal
|
||
|
||
fprintf (stderr, "Github Build Version: %s \n", GIT_TAG);
|
||
fprintf (stderr,"MBElib version %s\n", versionstr);
|
||
|
||
initOpts (&opts);
|
||
initState (&state);
|
||
|
||
InitAllFecFunction();
|
||
|
||
exitflag = 0;
|
||
// signal (SIGINT, sigfun);
|
||
|
||
while ((c = getopt (argc, argv, "haep:Pqs:t:v:z:i:o:d:c:g:nw:B:C:R:f:m:u:x:A:S:M:G:D:L:V:U:Y:K:H:X:NQWrlZTF1:2:345:6:7:89:")) != -1)
|
||
{
|
||
opterr = 0;
|
||
switch (c)
|
||
{
|
||
case 'h':
|
||
usage ();
|
||
exit (0);
|
||
case 'a':
|
||
opts.call_alert = 1;
|
||
break;
|
||
|
||
//Free'd up switches include e,p,v,Q,V,Y,z
|
||
//make sure to put a colon : after each if they need an argument
|
||
//or remove colon if no argument required
|
||
|
||
//Disabled the Serial Port Dev and Baud Rate, etc, If somebody uses that function, sorry...
|
||
|
||
case 'D': //user set DMRLA n value
|
||
sscanf (optarg, "%c", &opts.dmr_dmrla_n);
|
||
if (opts.dmr_dmrla_n > 10) opts.dmr_dmrla_n = 10; //max out at 10;
|
||
// if (opts.dmr_dmrla_n != 0) opts.dmr_dmrla_is_set = 1; //zero will fix a capmax site id value...I think
|
||
opts.dmr_dmrla_is_set = 1;
|
||
fprintf (stderr, "DMRLA n value set to %d. \n", opts.dmr_dmrla_n);
|
||
break;
|
||
|
||
case 'C': //new letter assignment for Channel import, flow down to allow temp numbers
|
||
case '1': //LCN to Frequency CSV Import <--> flow down to chanimport now, tell users to use a channel map for edacs
|
||
case '7': //Channel Map Frequency CSV Import
|
||
strncpy(opts.chan_in_file, optarg, 1023);
|
||
opts.chan_in_file[1023] = '\0';
|
||
csvChanImport (&opts, &state);
|
||
break;
|
||
|
||
case 'G': //new letter assignment for group import, flow down to allow temp numbers
|
||
case '2': //Group CSV Import
|
||
strncpy(opts.group_in_file, optarg, 1023);
|
||
opts.group_in_file[1023] = '\0';
|
||
csvGroupImport(&opts, &state);
|
||
break;
|
||
|
||
case 'T': //new letter assignment for trunking, flow down to allow temp numbers
|
||
case '3':
|
||
opts.p25_trunk = 1;
|
||
break;
|
||
|
||
case 'U': //New letter assignment for RIGCTL TCP port, flow down to allow temp numbers
|
||
case '5': //RIGCTL TCP port
|
||
sscanf (optarg, "%d", &opts.rigctlportno);
|
||
if (opts.rigctlportno != 0) opts.use_rigctl = 1;
|
||
break;
|
||
|
||
case 't': //New letter assignment for Trunk Hangtime, flow down to allow temp numbers
|
||
case '6': //hangtime in seconds, default is 1;
|
||
sscanf (optarg, "%d", &opts.trunk_hangtime);
|
||
break;
|
||
|
||
case 'q': //New letter assignment for Reverse Mute, flow down to allow temp numbers
|
||
case '8':
|
||
opts.reverse_mute = 1;
|
||
fprintf (stderr, "Reverse Mute\n");
|
||
break;
|
||
|
||
case 'B': //New letter assignment for RIGCTL SetMod BW, flow down to allow temp numbers
|
||
case '9': //rigctl setmod bandwidth;
|
||
sscanf (optarg, "%d", &opts.setmod_bw);
|
||
if (opts.setmod_bw > 25000) opts.setmod_bw = 25000; //not too high
|
||
break;
|
||
|
||
// case 'e':
|
||
// opts.errorbars = 1;
|
||
// opts.datascope = 0;
|
||
// break;
|
||
// case 'p':
|
||
// if (optarg[0] == 'e')
|
||
// {
|
||
// opts.p25enc = 1;
|
||
// }
|
||
// else if (optarg[0] == 'l')
|
||
// {
|
||
// opts.p25lc = 1;
|
||
// }
|
||
// else if (optarg[0] == 's')
|
||
// {
|
||
// opts.p25status = 1;
|
||
// }
|
||
// else if (optarg[0] == 't')
|
||
// {
|
||
// opts.p25tg = 1;
|
||
// }
|
||
// else if (optarg[0] == 'u')
|
||
// {
|
||
// opts.unmute_encrypted_p25 = 1;
|
||
// fprintf(stderr, "P25 Encrypted Audio Unmuted\n");
|
||
// }
|
||
// break;
|
||
// case 'P':
|
||
// sscanf (optarg, "%d", &opts.rtlsdr_ppm_error);
|
||
// break;
|
||
// case 'q':
|
||
// opts.errorbars = 0;
|
||
// opts.verbose = 0;
|
||
// break;
|
||
|
||
case 's':
|
||
sscanf (optarg, "%d", &opts.wav_sample_rate);
|
||
opts.wav_interpolator = opts.wav_sample_rate / opts.wav_decimator;
|
||
state.samplesPerSymbol = state.samplesPerSymbol * opts.wav_interpolator;
|
||
state.symbolCenter = state.symbolCenter * opts.wav_interpolator;
|
||
break;
|
||
|
||
// case 't':
|
||
// opts.symboltiming = 1;
|
||
// opts.errorbars = 1;
|
||
// opts.datascope = 0;
|
||
// break;
|
||
|
||
case 'v':
|
||
sscanf (optarg, "%d", &opts.verbose);
|
||
break;
|
||
|
||
case 'n': //disable or reclaim?
|
||
state.use_throttle = 1;
|
||
break;
|
||
|
||
case 'K':
|
||
sscanf (optarg, "%lld", &state.K);
|
||
if (state.K > 256)
|
||
{
|
||
state.K = 256;
|
||
}
|
||
opts.dmr_mute_encL = 0;
|
||
opts.dmr_mute_encR = 0;
|
||
if (state.K == 0)
|
||
{
|
||
opts.dmr_mute_encL = 1;
|
||
opts.dmr_mute_encR = 1;
|
||
}
|
||
break;
|
||
|
||
case 'R':
|
||
sscanf (optarg, "%lld", &state.R);
|
||
if (state.R > 0x7FFF)
|
||
{
|
||
state.R = 0x7FFF;
|
||
}
|
||
// opts.dmr_mute_encL = 0;
|
||
// opts.dmr_mute_encR = 0;
|
||
if (state.R == 0)
|
||
{
|
||
// opts.dmr_mute_encL = 1;
|
||
// opts.dmr_mute_encR = 1;
|
||
}
|
||
break;
|
||
|
||
case 'H':
|
||
//new handling for 10/32/64 Char Key
|
||
|
||
strncpy(opts.szNumbers, optarg, 1023);
|
||
opts.szNumbers[1023] = '\0';
|
||
state.K1 = strtoull (opts.szNumbers, &pEnd, 16);
|
||
state.K2 = strtoull (pEnd, &pEnd, 16);
|
||
state.K3 = strtoull (pEnd, &pEnd, 16);
|
||
state.K4 = strtoull (pEnd, &pEnd, 16);
|
||
fprintf (stderr, "**tera Key = %016llX %016llX %016llX %016llX\n", state.K1, state.K2, state.K3, state.K4);
|
||
opts.dmr_mute_encL = 0;
|
||
opts.dmr_mute_encR = 0;
|
||
if (state.K1 == 0 && state.K2 == 0 && state.K3 == 0 && state.K4 == 0)
|
||
{
|
||
opts.dmr_mute_encL = 1;
|
||
opts.dmr_mute_encR = 1;
|
||
}
|
||
state.H = state.K1; //shim still required?
|
||
break;
|
||
|
||
case '4':
|
||
state.M = 1;
|
||
fprintf (stderr,"Force Privacy Key Priority over FID and SVC bits.\n");
|
||
break;
|
||
|
||
//manually set Phase 2 TDMA WACN/SYSID/CC
|
||
case 'X':
|
||
sscanf (optarg, "%llX", &p2vars);
|
||
if (p2vars > 0)
|
||
{
|
||
state.p2_wacn = p2vars >> 24;
|
||
state.p2_sysid = (p2vars >> 12) & 0xFFF;
|
||
state.p2_cc = p2vars & 0xFFF;
|
||
}
|
||
if (state.p2_wacn != 0 && state.p2_sysid != 0 && state.p2_cc != 0)
|
||
{
|
||
state.p2_hardset = 1;
|
||
}
|
||
break;
|
||
|
||
// case 'G': //Set rtl device gain
|
||
// sscanf (optarg, "%d", &opts.rtl_gain_value); //multiple value by ten to make it consitent with the way rtl_fm really works
|
||
// break;
|
||
|
||
// case 'L': //Set rtl squelch level
|
||
// sscanf (optarg, "%d", &opts.rtl_squelch_level); //set squelch by user to prevent false positives on NXDN and others
|
||
// break;
|
||
|
||
// case 'V': //Set rtl voltage level
|
||
// sscanf (optarg, "%d", &opts.rtl_volume_multiplier); //set 'volume' multiplier for rtl-sdr samples
|
||
// break;
|
||
|
||
// case 'U': //Set rtl udp remote port
|
||
// sscanf (optarg, "%d", &opts.rtl_udp_port); //set udp port number for RTL remote
|
||
// break;
|
||
|
||
// case 'D': //Set rtl device index number
|
||
// sscanf (optarg, "%d", &opts.rtl_dev_index);
|
||
// break;
|
||
|
||
// case 'Y': //Set rtl VFO bandwidth --recommend 6, 12, 24, 36, 48, default 48? or 12?
|
||
// sscanf (optarg, "%d", &opts.rtl_bandwidth);
|
||
// break;
|
||
|
||
case 'N':
|
||
opts.use_ncurses_terminal = 1;
|
||
fprintf (stderr,"Enabling NCurses Terminal.\n");
|
||
break;
|
||
|
||
case 'Z':
|
||
opts.payload = 1;
|
||
fprintf (stderr,"Logging Frame Payload to console\n");
|
||
break;
|
||
|
||
case 'L': //LRRP output to file
|
||
strncpy(opts.lrrp_out_file, optarg, 1023);
|
||
opts.lrrp_out_file[1023] = '\0';
|
||
opts.lrrp_file_output = 1;
|
||
fprintf (stderr,"Writing + Appending LRRP data to file %s\n", opts.lrrp_out_file);
|
||
break;
|
||
|
||
case 'P': //TDMA/NXDN Per Call - was T, now is P
|
||
sprintf (wav_file_directory, "./WAV");
|
||
wav_file_directory[1023] = '\0';
|
||
if (stat(wav_file_directory, &st) == -1)
|
||
{
|
||
fprintf (stderr, "-T %s WAV file directory does not exist\n", wav_file_directory);
|
||
fprintf (stderr, "Creating directory %s to save decoded wav files\n", wav_file_directory);
|
||
mkdir(wav_file_directory, 0700); //user read write execute, needs execute for some reason or segfault
|
||
}
|
||
fprintf (stderr,"XDMA and NXDN Per Call Wav File Saving Enabled. (NCurses Terminal Only)\n");
|
||
sprintf (opts.wav_out_file, "./WAV/DSD-FME-X1.wav");
|
||
sprintf (opts.wav_out_fileR, "./WAV/DSD-FME-X2.wav");
|
||
opts.dmr_stereo_wav = 1;
|
||
openWavOutFileL (&opts, &state);
|
||
openWavOutFileR (&opts, &state);
|
||
break;
|
||
|
||
case 'F':
|
||
opts.aggressive_framesync = 0;
|
||
fprintf (stderr, "%s", KYEL);
|
||
//fprintf (stderr,"DMR Stereo Aggressive Resync Disabled!\n");
|
||
fprintf (stderr, "Relax P25 Phase 2 MAC_SIGNAL CRC Checksum Pass/Fail\n");
|
||
fprintf (stderr, "Relax DMR CACH/Burst FEC Pass/Fail\n");
|
||
fprintf (stderr, "%s", KNRM);
|
||
break;
|
||
|
||
// case 'z':
|
||
// sscanf (optarg, "%d", &opts.scoperate);
|
||
// opts.errorbars = 0;
|
||
// opts.p25enc = 0;
|
||
// opts.p25lc = 0;
|
||
// opts.p25status = 0;
|
||
// opts.p25tg = 0;
|
||
// opts.datascope = 1;
|
||
// opts.symboltiming = 0;
|
||
// fprintf (stderr,"Setting datascope frame rate to %i frame per second.\n", opts.scoperate);
|
||
// break;
|
||
|
||
case 'i':
|
||
strncpy(opts.audio_in_dev, optarg, 2047);
|
||
opts.audio_in_dev[2047] = '\0';
|
||
break;
|
||
case 'o':
|
||
strncpy(opts.audio_out_dev, optarg, 1023);
|
||
opts.audio_out_dev[1023] = '\0';
|
||
break;
|
||
case 'd':
|
||
strncpy(opts.mbe_out_dir, optarg, 1023);
|
||
opts.mbe_out_dir[1023] = '\0';
|
||
if (stat(opts.mbe_out_dir, &st) == -1)
|
||
{
|
||
fprintf (stderr, "-d %s directory does not exist\n", opts.mbe_out_dir);
|
||
fprintf (stderr, "Creating directory %s to save MBE files\n", opts.mbe_out_dir);
|
||
mkdir(opts.mbe_out_dir, 0700); //user read write execute, needs execute for some reason or segfault
|
||
}
|
||
else fprintf (stderr,"Writing mbe data files to directory %s\n", opts.mbe_out_dir);
|
||
// fprintf (stderr,"Writing mbe data temporarily disabled!\n");
|
||
break;
|
||
|
||
// case 'c':
|
||
// opts.rtlsdr_center_freq = (uint32_t)atofs(optarg);
|
||
// fprintf (stderr,"Tuning to frequency: %i Hz\n", opts.rtlsdr_center_freq);
|
||
// break;
|
||
|
||
case 'c': //now is symbol capture bin output like FME Lite
|
||
strncpy(opts.symbol_out_file, optarg, 1023);
|
||
opts.symbol_out_file[1023] = '\0';
|
||
fprintf (stderr,"Writing + Appending symbol capture to file %s\n", opts.symbol_out_file);
|
||
opts.symbol_out = 1;
|
||
openSymbolOutFile (&opts, &state);
|
||
break;
|
||
|
||
case 'g':
|
||
sscanf (optarg, "%f", &opts.audio_gain);
|
||
if (opts.audio_gain < (float) 0 )
|
||
{
|
||
fprintf (stderr,"Disabling audio out gain setting\n");
|
||
}
|
||
else if (opts.audio_gain == (float) 0)
|
||
{
|
||
opts.audio_gain = (float) 0;
|
||
fprintf (stderr,"Enabling audio out auto-gain\n");
|
||
}
|
||
else
|
||
{
|
||
fprintf (stderr,"Setting audio out gain to %f\n", opts.audio_gain);
|
||
state.aout_gain = opts.audio_gain;
|
||
state.aout_gainR = opts.audio_gain;
|
||
}
|
||
break;
|
||
|
||
case 'w':
|
||
strncpy(opts.wav_out_file, optarg, 1023);
|
||
opts.wav_out_file[1023] = '\0';
|
||
fprintf (stderr,"Writing + Appending decoded audio to file %s\n", opts.wav_out_file);
|
||
opts.dmr_stereo_wav = 0;
|
||
openWavOutFile (&opts, &state);
|
||
break;
|
||
|
||
// case 'B':
|
||
// sscanf (optarg, "%d", &opts.serial_baud);
|
||
// break;
|
||
|
||
// case 'C':
|
||
// strncpy(opts.serial_dev, optarg, 1023);
|
||
// opts.serial_dev[1023] = '\0';
|
||
// break;
|
||
|
||
case 'f':
|
||
if (optarg[0] == 'a')
|
||
{
|
||
opts.frame_dstar = 1;
|
||
opts.frame_x2tdma = 1;
|
||
opts.frame_p25p1 = 1;
|
||
opts.frame_p25p2 = 0;
|
||
opts.frame_nxdn48 = 0;
|
||
opts.frame_nxdn96 = 0;
|
||
opts.frame_dmr = 1;
|
||
opts.frame_dpmr = 0;
|
||
opts.frame_provoice = 0;
|
||
opts.frame_ysf = 0;
|
||
opts.pulse_digi_rate_out = 8000;
|
||
opts.pulse_digi_out_channels = 1;
|
||
opts.dmr_stereo = 0;
|
||
opts.dmr_mono = 1;
|
||
state.dmr_stereo = 0;
|
||
opts.setmod_bw = 7000;
|
||
sprintf (opts.output_name, "Legacy Auto");
|
||
}
|
||
else if (optarg[0] == 'd')
|
||
{
|
||
opts.frame_dstar = 1;
|
||
opts.frame_x2tdma = 0;
|
||
opts.frame_p25p1 = 0;
|
||
opts.frame_p25p2 = 0;
|
||
opts.frame_nxdn48 = 0;
|
||
opts.frame_nxdn96 = 0;
|
||
opts.frame_dmr = 0;
|
||
opts.frame_dpmr = 0;
|
||
opts.frame_provoice = 0;
|
||
opts.frame_ysf = 0;
|
||
opts.pulse_digi_rate_out = 8000;
|
||
opts.pulse_digi_out_channels = 1;
|
||
opts.dmr_stereo = 0;
|
||
state.dmr_stereo = 0;
|
||
opts.dmr_mono = 0;
|
||
state.rf_mod = 0;
|
||
sprintf (opts.output_name, "D-STAR");
|
||
fprintf (stderr,"Decoding only D-STAR frames.\n");
|
||
}
|
||
else if (optarg[0] == 'x')
|
||
{
|
||
opts.frame_dstar = 0;
|
||
opts.frame_x2tdma = 1;
|
||
opts.frame_p25p1 = 0;
|
||
opts.frame_p25p2 = 0;
|
||
opts.frame_nxdn48 = 0;
|
||
opts.frame_nxdn96 = 0;
|
||
opts.frame_dmr = 0;
|
||
opts.frame_dpmr = 0;
|
||
opts.frame_provoice = 0;
|
||
opts.frame_ysf = 0;
|
||
opts.pulse_digi_rate_out = 8000;
|
||
opts.pulse_digi_out_channels = 1;
|
||
opts.dmr_stereo = 0;
|
||
opts.dmr_mono = 0;
|
||
state.dmr_stereo = 0;
|
||
sprintf (opts.output_name, "X2-TDMA");
|
||
fprintf (stderr,"Decoding only X2-TDMA frames.\n");
|
||
}
|
||
else if (optarg[0] == 'p')
|
||
{
|
||
opts.frame_dstar = 0;
|
||
opts.frame_x2tdma = 0;
|
||
opts.frame_p25p1 = 0;
|
||
opts.frame_p25p2 = 0;
|
||
opts.frame_nxdn48 = 0;
|
||
opts.frame_nxdn96 = 0;
|
||
opts.frame_dmr = 0;
|
||
opts.frame_dpmr = 0;
|
||
opts.frame_provoice = 1;
|
||
opts.frame_ysf = 0;
|
||
state.samplesPerSymbol = 5;
|
||
state.symbolCenter = 2;
|
||
opts.mod_c4fm = 0;
|
||
opts.mod_qpsk = 0;
|
||
opts.mod_gfsk = 1;
|
||
state.rf_mod = 2;
|
||
opts.pulse_digi_rate_out = 8000;
|
||
opts.pulse_digi_out_channels = 1;
|
||
opts.dmr_stereo = 0;
|
||
opts.dmr_mono = 0;
|
||
state.dmr_stereo = 0;
|
||
opts.setmod_bw = 12500;
|
||
sprintf (opts.output_name, "EDACS/PV");
|
||
fprintf (stderr,"Setting symbol rate to 9600 / second\n");
|
||
fprintf (stderr,"Decoding only ProVoice frames.\n");
|
||
}
|
||
else if (optarg[0] == '1')
|
||
{
|
||
opts.frame_dstar = 0;
|
||
opts.frame_x2tdma = 0;
|
||
opts.frame_p25p1 = 1;
|
||
opts.frame_p25p2 = 0;
|
||
opts.frame_nxdn48 = 0;
|
||
opts.frame_nxdn96 = 0;
|
||
opts.frame_dmr = 0;
|
||
opts.frame_dpmr = 0;
|
||
opts.frame_provoice = 0;
|
||
opts.frame_ysf = 0;
|
||
opts.dmr_stereo = 0;
|
||
state.dmr_stereo = 0;
|
||
opts.mod_c4fm = 1;
|
||
opts.mod_qpsk = 0;
|
||
opts.mod_gfsk = 0;
|
||
state.rf_mod = 0; //
|
||
opts.dmr_stereo = 0;
|
||
opts.dmr_mono = 0;
|
||
opts.pulse_digi_rate_out = 8000;
|
||
opts.pulse_digi_out_channels = 1;
|
||
opts.setmod_bw = 7000;
|
||
sprintf (opts.output_name, "P25P1");
|
||
fprintf (stderr,"Decoding only P25 Phase 1 frames.\n");
|
||
}
|
||
else if (optarg[0] == 'i')
|
||
{
|
||
opts.frame_dstar = 0;
|
||
opts.frame_x2tdma = 0;
|
||
opts.frame_p25p1 = 0;
|
||
opts.frame_p25p2 = 0;
|
||
opts.frame_nxdn48 = 1;
|
||
opts.frame_nxdn96 = 0;
|
||
opts.frame_dmr = 0;
|
||
opts.frame_dpmr = 0;
|
||
opts.frame_provoice = 0;
|
||
opts.frame_ysf = 0;
|
||
state.samplesPerSymbol = 20;
|
||
state.symbolCenter = 10;
|
||
opts.mod_c4fm = 1;
|
||
opts.mod_qpsk = 0;
|
||
opts.mod_gfsk = 0;
|
||
state.rf_mod = 0;
|
||
opts.pulse_digi_rate_out = 8000;
|
||
opts.pulse_digi_out_channels = 1;
|
||
opts.dmr_stereo = 0;
|
||
state.dmr_stereo = 0;
|
||
opts.dmr_mono = 0;
|
||
opts.setmod_bw = 4000;
|
||
sprintf (opts.output_name, "NXDN48");
|
||
fprintf (stderr,"Setting symbol rate to 2400 / second\n");
|
||
fprintf (stderr,"Decoding only NXDN 4800 baud frames.\n");
|
||
}
|
||
else if (optarg[0] == 'y')
|
||
{
|
||
opts.frame_dstar = 0;
|
||
opts.frame_x2tdma = 0;
|
||
opts.frame_p25p1 = 0;
|
||
opts.frame_p25p2 = 0;
|
||
opts.frame_p25p2 = 0;
|
||
opts.frame_nxdn48 = 0;
|
||
opts.frame_nxdn96 = 0;
|
||
opts.frame_dmr = 0;
|
||
opts.frame_dpmr = 0;
|
||
opts.frame_provoice = 0;
|
||
opts.frame_ysf = 1;
|
||
state.samplesPerSymbol = 20; //10
|
||
state.symbolCenter = 10;
|
||
opts.mod_c4fm = 1;
|
||
opts.mod_qpsk = 0;
|
||
opts.mod_gfsk = 0;
|
||
state.rf_mod = 0;
|
||
opts.pulse_digi_rate_out = 8000;
|
||
opts.pulse_digi_out_channels = 1;
|
||
opts.dmr_stereo = 0;
|
||
state.dmr_stereo = 0;
|
||
opts.dmr_mono = 0;
|
||
sprintf (opts.output_name, "YSF");
|
||
fprintf (stderr,"Setting symbol rate to 2400 / second\n");
|
||
fprintf (stderr,"Decoding only YSF frames.\nNot working yet!\n");
|
||
}
|
||
else if (optarg[0] == '2')
|
||
{
|
||
opts.frame_dstar = 0;
|
||
opts.frame_x2tdma = 0;
|
||
opts.frame_p25p1 = 0;
|
||
opts.frame_p25p2 = 1;
|
||
opts.frame_nxdn48 = 0;
|
||
opts.frame_nxdn96 = 0;
|
||
opts.frame_dmr = 0;
|
||
opts.frame_dpmr = 0;
|
||
opts.frame_provoice = 0;
|
||
opts.frame_ysf = 0;
|
||
state.samplesPerSymbol = 10;
|
||
state.symbolCenter = 4;
|
||
opts.mod_c4fm = 1;
|
||
opts.mod_qpsk = 0;
|
||
opts.mod_gfsk = 0;
|
||
state.rf_mod = 0;
|
||
opts.dmr_stereo = 1;
|
||
state.dmr_stereo = 0;
|
||
opts.dmr_mono = 0;
|
||
opts.setmod_bw = 7000;
|
||
sprintf (opts.output_name, "P25P2");
|
||
fprintf (stderr,"Decoding P25-P2 frames C4FM or OP25 Symbol Captures!\n");
|
||
}
|
||
else if (optarg[0] == 's')
|
||
{
|
||
opts.frame_dstar = 0;
|
||
opts.frame_x2tdma = 0;
|
||
opts.frame_p25p1 = 0;
|
||
opts.frame_p25p2 = 0;
|
||
opts.inverted_p2 = 0;
|
||
opts.frame_nxdn48 = 0;
|
||
opts.frame_nxdn96 = 0;
|
||
opts.frame_dmr = 1;
|
||
opts.frame_dpmr = 0;
|
||
opts.frame_provoice = 0;
|
||
opts.frame_ysf = 0;
|
||
opts.mod_c4fm = 1;
|
||
opts.mod_qpsk = 0;
|
||
opts.mod_gfsk = 0;
|
||
state.rf_mod = 0;
|
||
opts.dmr_stereo = 1;
|
||
opts.dmr_mono = 0;
|
||
// opts.setmod_bw = 7000;
|
||
opts.setmod_bw = 7000;
|
||
opts.pulse_digi_rate_out = 24000;
|
||
opts.pulse_digi_out_channels = 2;
|
||
sprintf (opts.output_name, "DMR Stereo");
|
||
fprintf (stderr,"Decoding DMR Stereo BS/MS Simplex\n");
|
||
}
|
||
else if (optarg[0] == 't')
|
||
{
|
||
opts.frame_dstar = 0;
|
||
opts.frame_x2tdma = 1;
|
||
opts.frame_p25p1 = 1;
|
||
opts.frame_p25p2 = 1;
|
||
opts.inverted_p2 = 0;
|
||
opts.frame_nxdn48 = 0;
|
||
opts.frame_nxdn96 = 0;
|
||
opts.frame_dmr = 1;
|
||
opts.frame_dpmr = 0;
|
||
opts.frame_provoice = 0;
|
||
opts.frame_ysf = 0;
|
||
opts.mod_c4fm = 1;
|
||
opts.mod_qpsk = 0;
|
||
opts.mod_gfsk = 0;
|
||
//Need a new demodulator is needed to handle p2 cqpsk
|
||
//or consider using an external modulator (GNURadio?)
|
||
state.rf_mod = 0;
|
||
opts.dmr_stereo = 1;
|
||
opts.dmr_mono = 0;
|
||
opts.setmod_bw = 7000;
|
||
opts.pulse_digi_rate_out = 24000;
|
||
opts.pulse_digi_out_channels = 2;
|
||
sprintf (opts.output_name, "XDMA");
|
||
fprintf (stderr,"Decoding XDMA P25 and DMR\n");
|
||
}
|
||
else if (optarg[0] == 'n')
|
||
{
|
||
opts.frame_dstar = 0;
|
||
opts.frame_x2tdma = 0;
|
||
opts.frame_p25p1 = 0;
|
||
opts.frame_p25p2 = 0;
|
||
opts.frame_nxdn48 = 0;
|
||
opts.frame_nxdn96 = 1;
|
||
opts.frame_dmr = 0;
|
||
opts.frame_dpmr = 0;
|
||
opts.frame_provoice = 0;
|
||
opts.frame_ysf = 0;
|
||
opts.mod_c4fm = 1;
|
||
opts.mod_qpsk = 0;
|
||
opts.mod_gfsk = 0;
|
||
state.rf_mod = 0;
|
||
opts.pulse_digi_rate_out = 8000;
|
||
opts.pulse_digi_out_channels = 1;
|
||
opts.dmr_stereo = 0;
|
||
opts.dmr_mono = 0;
|
||
state.dmr_stereo = 0;
|
||
opts.setmod_bw = 7000;
|
||
sprintf (opts.output_name, "NXDN96");
|
||
fprintf (stderr,"Decoding only NXDN 9600 baud frames.\n");
|
||
}
|
||
else if (optarg[0] == 'r')
|
||
{
|
||
opts.frame_dstar = 0;
|
||
opts.frame_x2tdma = 0;
|
||
opts.frame_p25p1 = 0;
|
||
opts.frame_p25p2 = 0;
|
||
opts.frame_nxdn48 = 0;
|
||
opts.frame_nxdn96 = 0;
|
||
opts.frame_dmr = 1;
|
||
opts.frame_dpmr = 0;
|
||
opts.frame_provoice = 0;
|
||
opts.frame_ysf = 0;
|
||
opts.mod_c4fm = 1;
|
||
opts.mod_qpsk = 0;
|
||
opts.mod_gfsk = 0; //
|
||
state.rf_mod = 0; //
|
||
opts.pulse_digi_rate_out = 8000;
|
||
opts.pulse_digi_out_channels = 1;
|
||
opts.dmr_mono = 1;
|
||
opts.dmr_stereo = 0;
|
||
state.dmr_stereo = 0; //0
|
||
opts.setmod_bw = 7000;
|
||
sprintf (opts.output_name, "DMR Mono");
|
||
//fprintf(stderr, "Notice: DMR cannot autodetect polarity. \n Use -xr option if Inverted Signal expected.\n");
|
||
fprintf (stderr,"Decoding only DMR Mono. \nUsing DMR Stereo or XDMA is highly encouraged.\n");
|
||
//dmr stereo below
|
||
// opts.frame_dstar = 0;
|
||
// opts.frame_x2tdma = 0;
|
||
// opts.frame_p25p1 = 0;
|
||
// opts.frame_p25p2 = 0;
|
||
// opts.inverted_p2 = 0;
|
||
// opts.frame_nxdn48 = 0;
|
||
// opts.frame_nxdn96 = 0;
|
||
// opts.frame_dmr = 1;
|
||
// opts.frame_dpmr = 0;
|
||
// opts.frame_provoice = 0;
|
||
// opts.frame_ysf = 0;
|
||
// opts.mod_c4fm = 1;
|
||
// opts.mod_qpsk = 0;
|
||
// opts.mod_gfsk = 0;
|
||
// state.rf_mod = 0;
|
||
// opts.dmr_stereo = 1;
|
||
// state.dmr_stereo = 0;
|
||
// opts.pulse_digi_rate_out = 24000;
|
||
// opts.pulse_digi_out_channels = 2;
|
||
// sprintf (opts.output_name, "DMR Stereo");
|
||
// fprintf (stderr,"Decoding DMR Stereo BS/MS Simplex\n");
|
||
}
|
||
else if (optarg[0] == 'm')
|
||
{
|
||
opts.frame_dstar = 0;
|
||
opts.frame_x2tdma = 0;
|
||
opts.frame_p25p1 = 0;
|
||
opts.frame_p25p2 = 0;
|
||
opts.frame_nxdn48 = 0;
|
||
opts.frame_nxdn96 = 0;
|
||
opts.frame_dmr = 0;
|
||
opts.frame_provoice = 0;
|
||
opts.frame_dpmr = 1;
|
||
opts.frame_ysf = 0; //testing with NXDN48 parameters
|
||
state.samplesPerSymbol = 20; //same as NXDN48 - 20
|
||
state.symbolCenter = 10; //same as NXDN48 - 10
|
||
opts.mod_c4fm = 1;
|
||
opts.mod_qpsk = 0;
|
||
opts.mod_gfsk = 0;
|
||
state.rf_mod = 0;
|
||
opts.pulse_digi_rate_out = 8000;
|
||
opts.pulse_digi_out_channels = 1;
|
||
opts.dmr_stereo = 0;
|
||
opts.dmr_mono = 0;
|
||
state.dmr_stereo = 0;
|
||
sprintf (opts.output_name, "dPMR");
|
||
fprintf(stderr, "Notice: dPMR cannot autodetect polarity. \n Use -xd option if Inverted Signal expected.\n");
|
||
fprintf(stderr, "Decoding only dPMR frames.\n");
|
||
}
|
||
break;
|
||
//don't mess with the modulations, doesn't really do anything meaningful anyways
|
||
case 'm':
|
||
if (optarg[0] == 'a')
|
||
{
|
||
// opts.mod_c4fm = 1;
|
||
// opts.mod_qpsk = 1;
|
||
// opts.mod_gfsk = 1;
|
||
// state.rf_mod = 0;
|
||
fprintf (stderr,"Don't use the -ma switch.\n");
|
||
}
|
||
else if (optarg[0] == 'c')
|
||
{
|
||
opts.mod_c4fm = 1;
|
||
opts.mod_qpsk = 0;
|
||
opts.mod_gfsk = 0;
|
||
state.rf_mod = 0;
|
||
fprintf (stderr,"Enabling only C4FM modulation optimizations.\n");
|
||
}
|
||
else if (optarg[0] == 'g')
|
||
{
|
||
opts.mod_c4fm = 0;
|
||
opts.mod_qpsk = 0;
|
||
opts.mod_gfsk = 1;
|
||
state.rf_mod = 2;
|
||
fprintf (stderr,"Enabling only GFSK modulation optimizations.\n");
|
||
}
|
||
else if (optarg[0] == 'q')
|
||
{
|
||
opts.mod_c4fm = 0;
|
||
opts.mod_qpsk = 1;
|
||
opts.mod_gfsk = 0;
|
||
state.rf_mod = 1;
|
||
opts.setmod_bw = 12500;
|
||
fprintf (stderr,"Enabling only QPSK modulation optimizations.\n");
|
||
}
|
||
else if (optarg[0] == '2')
|
||
{
|
||
opts.mod_c4fm = 0;
|
||
opts.mod_qpsk = 1;
|
||
opts.mod_gfsk = 0;
|
||
state.rf_mod = 1;
|
||
state.samplesPerSymbol = 8;
|
||
state.symbolCenter = 3;
|
||
opts.setmod_bw = 12500;
|
||
fprintf (stderr,"Enabling 6000 sps P25p2 CQPSK.\n");
|
||
}
|
||
break;
|
||
case 'u':
|
||
sscanf (optarg, "%i", &opts.uvquality);
|
||
if (opts.uvquality < 1)
|
||
{
|
||
opts.uvquality = 1;
|
||
}
|
||
else if (opts.uvquality > 64)
|
||
{
|
||
opts.uvquality = 64;
|
||
}
|
||
fprintf (stderr,"Setting unvoice speech quality to %i waves per band.\n", opts.uvquality);
|
||
break;
|
||
case 'x':
|
||
if (optarg[0] == 'x')
|
||
{
|
||
opts.inverted_x2tdma = 0;
|
||
fprintf (stderr,"Expecting non-inverted X2-TDMA signals.\n");
|
||
}
|
||
else if (optarg[0] == 'r')
|
||
{
|
||
opts.inverted_dmr = 1;
|
||
fprintf (stderr,"Expecting inverted DMR signals.\n");
|
||
}
|
||
else if (optarg[0] == 'd')
|
||
{
|
||
opts.inverted_dpmr = 1;
|
||
fprintf (stderr, "Expecting inverted ICOM dPMR signals.\n");
|
||
}
|
||
break;
|
||
case 'A':
|
||
sscanf (optarg, "%i", &opts.mod_threshold);
|
||
fprintf (stderr,"Setting C4FM/QPSK auto detection threshold to %i\n", opts.mod_threshold);
|
||
case 'S':
|
||
sscanf (optarg, "%i", &opts.ssize);
|
||
if (opts.ssize > 128)
|
||
{
|
||
opts.ssize = 128;
|
||
}
|
||
else if (opts.ssize < 1)
|
||
{
|
||
opts.ssize = 1;
|
||
}
|
||
fprintf (stderr,"Setting QPSK symbol buffer to %i\n", opts.ssize);
|
||
break;
|
||
case 'M':
|
||
sscanf (optarg, "%i", &opts.msize);
|
||
if (opts.msize > 1024)
|
||
{
|
||
opts.msize = 1024;
|
||
}
|
||
else if (opts.msize < 1)
|
||
{
|
||
opts.msize = 1;
|
||
}
|
||
fprintf (stderr,"Setting QPSK Min/Max buffer to %i\n", opts.msize);
|
||
break;
|
||
case 'r':
|
||
opts.playfiles = 1;
|
||
opts.errorbars = 0;
|
||
opts.datascope = 0;
|
||
opts.pulse_digi_rate_out = 8000;
|
||
opts.pulse_digi_out_channels = 1;
|
||
opts.dmr_stereo = 0;
|
||
state.dmr_stereo = 0;
|
||
sprintf (opts.output_name, "MBE Playback");
|
||
state.optind = optind;
|
||
break;
|
||
case 'l':
|
||
opts.use_cosine_filter = 0;
|
||
break;
|
||
default:
|
||
usage ();
|
||
exit (0);
|
||
}
|
||
}
|
||
|
||
if (opts.resume > 0)
|
||
{
|
||
openSerial (&opts, &state);
|
||
}
|
||
|
||
//need error handling on opening rigctl so we don't exit or crash on disconnect
|
||
if (opts.use_rigctl == 1)
|
||
{
|
||
opts.rigctl_sockfd = Connect(opts.rigctlhostname, opts.rigctlportno);
|
||
}
|
||
|
||
if((strncmp(opts.audio_in_dev, "tcp", 3) == 0)) //tcp socket input from SDR++ and others
|
||
{
|
||
//use same handling as connect function from rigctl
|
||
//also still needs err handling
|
||
fprintf (stderr, "TCP Direct Link: ");
|
||
char * curr;
|
||
|
||
curr = strtok(opts.audio_in_dev, ":"); //should be 'tcp'
|
||
if (curr != NULL) ; //continue
|
||
else goto TCPEND; //end early with preset values
|
||
|
||
curr = strtok(NULL, ":"); //host address
|
||
if (curr != NULL) strncpy (opts.tcp_hostname, curr, 1023);
|
||
|
||
curr = strtok(NULL, ":"); //host port
|
||
if (curr != NULL) opts.tcp_portno = atoi (curr);
|
||
|
||
TCPEND:
|
||
fprintf (stderr, "%s:", opts.tcp_hostname);
|
||
fprintf (stderr, "%d \n", opts.tcp_portno);
|
||
opts.tcp_sockfd = Connect(opts.tcp_hostname, opts.tcp_portno);
|
||
opts.audio_in_type = 8;
|
||
}
|
||
|
||
if((strncmp(opts.audio_in_dev, "rtl", 3) == 0)) //rtl dongle input
|
||
{
|
||
uint8_t rtl_ok = 0;
|
||
#ifdef USE_RTLSDR
|
||
fprintf (stderr, "RTL Input: ");
|
||
char * curr;
|
||
|
||
curr = strtok(opts.audio_in_dev, ":"); //should be 'rtl'
|
||
if (curr != NULL) ; //continue
|
||
else goto RTLEND; //end early with preset values
|
||
|
||
curr = strtok(NULL, ":"); //rtl device number "-D"
|
||
if (curr != NULL) opts.rtl_dev_index = atoi (curr);
|
||
else goto RTLEND;
|
||
|
||
curr = strtok(NULL, ":"); //rtl freq "-c"
|
||
if (curr != NULL) opts.rtlsdr_center_freq = (uint32_t)atofs(curr);
|
||
else goto RTLEND;
|
||
|
||
curr = strtok(NULL, ":"); //rtl gain value "-G"
|
||
if (curr != NULL) opts.rtl_gain_value = atoi (curr);
|
||
else goto RTLEND;
|
||
|
||
curr = strtok(NULL, ":"); //rtl ppm err "-P"
|
||
if (curr != NULL) opts.rtlsdr_ppm_error = atoi (curr);
|
||
else goto RTLEND;
|
||
|
||
curr = strtok(NULL, ":"); //rtl bandwidth "-Y"
|
||
if (curr != NULL)
|
||
{
|
||
int bw = 0;
|
||
bw = atoi (curr);
|
||
//check for proper values (6,8,12,24)
|
||
if (bw == 6 || bw == 8 || bw == 12 || bw == 24)
|
||
{
|
||
opts.rtl_bandwidth = bw;
|
||
}
|
||
else
|
||
opts.rtl_bandwidth = 12; //safe default
|
||
}
|
||
else goto RTLEND;
|
||
|
||
curr = strtok(NULL, ":"); //rtl squelch level "-L"
|
||
if (curr != NULL) opts.rtl_squelch_level = atoi (curr);
|
||
else goto RTLEND;
|
||
|
||
curr = strtok(NULL, ":"); //rtl udp port "-U"
|
||
if (curr != NULL) opts.rtl_udp_port = atoi (curr);
|
||
else goto RTLEND;
|
||
|
||
RTLEND:
|
||
fprintf (stderr, "Dev %d ", opts.rtl_dev_index);
|
||
fprintf (stderr, "Freq %d ", opts.rtlsdr_center_freq);
|
||
fprintf (stderr, "Gain %d ", opts.rtl_gain_value);
|
||
fprintf (stderr, "PPM %d ", opts.rtlsdr_ppm_error);
|
||
fprintf (stderr, "BW %d ", opts.rtl_bandwidth);
|
||
fprintf (stderr, "SQ %d ", opts.rtl_squelch_level);
|
||
fprintf (stderr, "UDP %d \n", opts.rtl_udp_port);
|
||
opts.audio_in_type = 3;
|
||
rtl_ok = 1;
|
||
#endif
|
||
|
||
if (rtl_ok == 0) //not set, means rtl support isn't compiled/available
|
||
{
|
||
fprintf (stderr, "RTL Support not enabled/compiled, falling back to Pulse Audio Input.\n");
|
||
sprintf (opts.audio_in_dev, "%s", "pulse");
|
||
opts.audio_in_type = 0;
|
||
}
|
||
}
|
||
|
||
//still need to work out why I can't use this
|
||
//issues with samples received, may be using UDP DGRAMS incorrectly, incorrect procedure?
|
||
if((strncmp(opts.audio_in_dev, "udp", 3) == 0)) //udp socket input from SDR++, GQRX, and others
|
||
{
|
||
//also still needs err handling
|
||
opts.udp_sockfd = UDPBind(opts.udp_hostname, opts.udp_portno);
|
||
opts.audio_in_type = 6;
|
||
}
|
||
|
||
if((strncmp(opts.audio_in_dev, "/dev/dsp", 8) == 0))
|
||
{
|
||
sprintf (opts.audio_in_dev, "%s", "pulse");
|
||
fprintf (stderr, "OSS and PA Handling Disabled; Using Pulse Audio.\n");
|
||
opts.audio_in_type = 0;
|
||
}
|
||
|
||
if((strncmp(opts.audio_in_dev, "/dev/audio", 10) == 0))
|
||
{
|
||
sprintf (opts.audio_in_dev, "%s", "pulse");
|
||
fprintf (stderr, "OSS and PA Handling Disabled; Using Pulse Audio.\n");
|
||
opts.audio_in_type = 0;
|
||
}
|
||
|
||
if((strncmp(opts.audio_out_dev, "/dev/dsp", 8) == 0))
|
||
{
|
||
sprintf (opts.audio_out_dev, "%s", "pulse");
|
||
fprintf (stderr, "OSS and PA Handling Disabled; Using Pulse Audio.\n");
|
||
opts.audio_out_type = 0;
|
||
}
|
||
|
||
if((strncmp(opts.audio_out_dev, "/dev/audio", 10) == 0))
|
||
{
|
||
sprintf (opts.audio_out_dev, "%s", "pulse");
|
||
fprintf (stderr, "OSS and PA Handling Disabled; Using Pulse Audio.\n");
|
||
opts.audio_out_type = 0;
|
||
}
|
||
|
||
if((strncmp(opts.audio_out_dev, "pa", 2) == 0))
|
||
{
|
||
sprintf (opts.audio_out_dev, "%s", "pulse");
|
||
fprintf (stderr, "OSS and PA Handling Disabled; Using Pulse Audio.\n");
|
||
opts.audio_out_type = 0;
|
||
}
|
||
|
||
if((strncmp(opts.audio_in_dev, "pa", 2) == 0))
|
||
{
|
||
sprintf (opts.audio_in_dev, "%s", "pulse");
|
||
fprintf (stderr, "OSS and PA Handling Disabled; Using Pulse Audio.\n");
|
||
opts.audio_out_type = 0;
|
||
}
|
||
|
||
if((strncmp(opts.audio_in_dev, "pulse", 5) == 0))
|
||
{
|
||
opts.audio_in_type = 0;
|
||
}
|
||
|
||
if((strncmp(opts.audio_out_dev, "pulse", 5) == 0))
|
||
{
|
||
opts.audio_out_type = 0;
|
||
}
|
||
|
||
if((strncmp(opts.audio_out_dev, "null", 4) == 0))
|
||
{
|
||
opts.audio_out_type = 9; //9 for NULL, or mute output
|
||
opts.audio_out = 0; //turn off so we won't playSynthesized
|
||
}
|
||
|
||
if (opts.playfiles == 1)
|
||
{
|
||
opts.split = 1;
|
||
opts.playoffset = 0;
|
||
opts.playoffsetR = 0;
|
||
opts.delay = 0;
|
||
if (strlen(opts.wav_out_file) > 0 && opts.dmr_stereo_wav == 0)
|
||
{
|
||
openWavOutFile (&opts, &state);
|
||
}
|
||
else
|
||
{
|
||
openAudioOutDevice (&opts, SAMPLE_RATE_OUT);
|
||
}
|
||
}
|
||
else if (strcmp (opts.audio_in_dev, opts.audio_out_dev) != 0)
|
||
{
|
||
opts.split = 1;
|
||
opts.playoffset = 0;
|
||
opts.playoffsetR = 0;
|
||
opts.delay = 0;
|
||
if (strlen(opts.wav_out_file) > 0 && opts.dmr_stereo_wav == 0)
|
||
{
|
||
openWavOutFile (&opts, &state);
|
||
}
|
||
else
|
||
{
|
||
openAudioOutDevice (&opts, SAMPLE_RATE_OUT);
|
||
}
|
||
openAudioInDevice (&opts);
|
||
|
||
|
||
fprintf (stderr,"Press CTRL + C to close.\n");
|
||
}
|
||
|
||
else
|
||
{
|
||
opts.split = 0;
|
||
opts.playoffset = 25;
|
||
opts.playoffsetR = 25;
|
||
opts.delay = 0;
|
||
openAudioInDevice (&opts);
|
||
opts.audio_out_fd = opts.audio_in_fd;
|
||
}
|
||
|
||
if (opts.playfiles == 1)
|
||
{
|
||
opts.pulse_digi_rate_out = 8000;
|
||
opts.pulse_digi_out_channels = 1;
|
||
openPulseOutput(&opts); //need to open it up for output
|
||
playMbeFiles (&opts, &state, argc, argv);
|
||
}
|
||
|
||
else
|
||
{
|
||
liveScanner (&opts, &state);
|
||
}
|
||
|
||
cleanupAndExit (&opts, &state);
|
||
|
||
return (0);
|
||
}
|