/* Copyright (C) 2019-2020 Andrei Kopanchuk UZ7HO This file is part of QtSoundModem QtSoundModem 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. QtSoundModem 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 QtSoundModem. If not, see http://www.gnu.org/licenses */ //#define TXSILENCE // UZ7HO Soundmodem Port by John Wiseman G8BPQ // // Audio interface Routine // Passes audio samples to/from the sound interface // As this is platform specific it also has the main() routine, which does // platform specific initialisation before calling ardopmain() // This is ALSASound.c for Linux // Windows Version is Waveout.c #include #include #include #include #include "UZ7HOStuff.h" #define VOID void char * strlop(char * buf, char delim); int gethints(); struct timespec pttclk; extern int Closing; int SoundMode = 0; int stdinMode = 0; int onlyMixSnoop = 0; int txLatency; //#define SHARECAPTURE // if defined capture device is opened and closed for each transission #define HANDLE int void gpioSetMode(unsigned gpio, unsigned mode); void gpioWrite(unsigned gpio, unsigned level); int WriteLog(char * msg, int Log); int _memicmp(unsigned char *a, unsigned char *b, int n); int stricmp(const unsigned char * pStr1, const unsigned char *pStr2); int gpioInitialise(void); HANDLE OpenCOMPort(char * pPort, int speed, BOOL SetDTR, BOOL SetRTS, BOOL Quiet, int Stopbits); int CloseSoundCard(); int PackSamplesAndSend(short * input, int nSamples); void displayLevel(int max); BOOL WriteCOMBlock(HANDLE fd, char * Block, int BytesToWrite); VOID processargs(int argc, char * argv[]); void PollReceivedSamples(); void closeTraceLog(); HANDLE OpenCOMPort(char * Port, int speed, BOOL SetDTR, BOOL SetRTS, BOOL Quiet, int Stopbits); VOID COMSetDTR(HANDLE fd); VOID COMClearDTR(HANDLE fd); VOID COMSetRTS(HANDLE fd); VOID COMClearRTS(HANDLE fd); int oss_read(short * samples, int nSamples); int oss_write(short * ptr, int len); int oss_flush(); int oss_audio_open(char * adevice_in, char * adevice_out); void oss_audio_close(); int listpulse(); int pulse_read(short * ptr, int len); int pulse_write(short * ptr, int len); int pulse_flush(); int pulse_audio_open(char * adevice_in, char * adevice_out); void pulse_audio_close(); int initdisplay(); extern BOOL blnDISCRepeating; extern BOOL UseKISS; // Enable Packet (KISS) interface extern short * DMABuffer; #define MaxReceiveSize 2048 // Enough for 9600 #define MaxSendSize 4096 short buffer[2][MaxSendSize * 2]; // Two Transfer/DMA buffers of 0.1 Sec (x2 for Stereo) short inbuffer[MaxReceiveSize * 2]; // Input Transfer/ buffers of 0.1 Sec (x2 for Stereo) extern short * DMABuffer; extern int Number; int ReceiveSize = 512; int SendSize = 1024; int using48000 = 0; int SampleRate = 12000; BOOL UseLeft = TRUE; BOOL UseRight = TRUE; char LogDir[256] = ""; void WriteDebugLog(char * Msg); VOID Debugprintf(const char * format, ...) { char Mess[10000]; va_list(arglist); va_start(arglist, format); vsprintf(Mess, format, arglist); WriteDebugLog(Mess); return; } void Sleep(int mS) { usleep(mS * 1000); return; } // Windows and ALSA work with signed samples +- 32767 // STM32 and Teensy DAC uses unsigned 0 - 4095 BOOL Loopback = FALSE; //BOOL Loopback = TRUE; char CaptureDevice[80] = "plughw:0,0"; char PlaybackDevice[80] = "plughw:0,0"; char * CaptureDevices = CaptureDevice; char * PlaybackDevices = CaptureDevice; int CaptureIndex = 0; int PlayBackIndex = 0; int Ticks; int LastNow; extern int Number; // Number waiting to be sent snd_pcm_sframes_t MaxAvail; #include FILE *logfile[3] = {NULL, NULL, NULL}; char LogName[3][256] = {"ARDOPDebug", "ARDOPException", "ARDOPSession"}; #define DEBUGLOG 0 #define EXCEPTLOG 1 #define SESSIONLOG 2 FILE *statslogfile = NULL; void printtick(char * msg) { Debugprintf("%s %i", msg, Now - LastNow); LastNow = Now; } struct timespec time_start; unsigned int getTicks() { struct timespec tp; clock_gettime(CLOCK_MONOTONIC, &tp); return (tp.tv_sec - time_start.tv_sec) * 1000 + (tp.tv_nsec - time_start.tv_nsec) / 1000000; } void PlatformSleep(int mS) { Sleep(mS); } // PTT via GPIO code #ifdef __ARM_ARCH #define PI_INPUT 0 #define PI_OUTPUT 1 #define PI_ALT0 4 #define PI_ALT1 5 #define PI_ALT2 6 #define PI_ALT3 7 #define PI_ALT4 3 #define PI_ALT5 2 // Set GPIO pin as output and set low void SetupGPIOPTT() { if (pttGPIOPin == -1) { Debugprintf("GPIO PTT disabled"); useGPIO = FALSE; } else { if (pttGPIOPin < 0) { pttGPIOInvert = TRUE; pttGPIOPin = -pttGPIOPin; } gpioSetMode(pttGPIOPin, PI_OUTPUT); gpioWrite(pttGPIOPin, pttGPIOInvert ? 1 : 0); Debugprintf("Using GPIO pin %d for Left/Mono PTT", pttGPIOPin); if (pttGPIOPinR != -1) { gpioSetMode(pttGPIOPinR, PI_OUTPUT); gpioWrite(pttGPIOPinR, pttGPIOInvert ? 1 : 0); Debugprintf("Using GPIO pin %d for Right PTT", pttGPIOPin); } useGPIO = TRUE; } } #endif static void sigterm_handler(int n) { UNUSED(n); printf("terminating on SIGTERM\n"); Closing = TRUE; } static void sigint_handler(int n) { UNUSED(n); printf("terminating on SIGINT\n"); Closing = TRUE; } char * PortString = NULL; void platformInit() { struct sigaction act; // Sleep(1000); // Give LinBPQ time to complete init if exec'ed by linbpq // Get Time Reference clock_gettime(CLOCK_MONOTONIC, &time_start); LastNow = getTicks(); // Trap signals memset (&act, '\0', sizeof(act)); act.sa_handler = &sigint_handler; if (sigaction(SIGINT, &act, NULL) < 0) perror ("SIGINT"); act.sa_handler = &sigterm_handler; if (sigaction(SIGTERM, &act, NULL) < 0) perror ("SIGTERM"); act.sa_handler = SIG_IGN; if (sigaction(SIGHUP, &act, NULL) < 0) perror ("SIGHUP"); if (sigaction(SIGPIPE, &act, NULL) < 0) perror ("SIGPIPE"); } void txSleep(int mS) { // called while waiting for next TX buffer. Run background processes if (mS < 0) return; while (mS > 50) { PollReceivedSamples(); // discard any received samples Sleep(50); mS -= 50; } Sleep(mS); PollReceivedSamples(); // discard any received samples } // ALSA Code #define true 1 #define false 0 snd_pcm_t * playhandle = NULL; snd_pcm_t * rechandle = NULL; int m_playchannels = 2; int m_recchannels = 2; char SavedCaptureDevice[256]; // Saved so we can reopen char SavedPlaybackDevice[256]; int Savedplaychannels = 2; int SavedCaptureRate; int SavedPlaybackRate; char CaptureNames[16][256] = { "" }; char PlaybackNames[16][256] = { "" }; int PlaybackCount = 0; int CaptureCount = 0; // Routine to check that library is available int CheckifLoaded() { // Prevent CTRL/C from closing the TNC // (This causes problems if the TNC is started by LinBPQ) signal(SIGHUP, SIG_IGN); signal(SIGINT, SIG_IGN); signal(SIGPIPE, SIG_IGN); return TRUE; } int GetOutputDeviceCollection() { // Get all the suitable devices and put in a list for GetNext to return snd_ctl_t *handle= NULL; snd_pcm_t *pcm= NULL; snd_ctl_card_info_t *info; snd_pcm_info_t *pcminfo; snd_pcm_hw_params_t *pars; snd_pcm_format_mask_t *fmask; char NameString[256]; Debugprintf("Playback Devices\n"); CloseSoundCard(); // free old struct if called again // while (PlaybackCount) // { // PlaybackCount--; // free(PlaybackNames[PlaybackCount]); // } // if (PlaybackNames) // free(PlaybackNames); PlaybackCount = 0; // Get Device List from ALSA snd_ctl_card_info_alloca(&info); snd_pcm_info_alloca(&pcminfo); snd_pcm_hw_params_alloca(&pars); snd_pcm_format_mask_alloca(&fmask); char hwdev[80]; unsigned min, max, ratemin, ratemax; int card, err, dev, nsubd; snd_pcm_stream_t stream = SND_PCM_STREAM_PLAYBACK; card = -1; if (snd_card_next(&card) < 0) { Debugprintf("No Devices"); return 0; } if (playhandle) snd_pcm_close(playhandle); playhandle = NULL; while (card >= 0) { sprintf(hwdev, "hw:%d", card); err = snd_ctl_open(&handle, hwdev, 0); err = snd_ctl_card_info(handle, info); Debugprintf("Card %d, ID `%s', name `%s'", card, snd_ctl_card_info_get_id(info), snd_ctl_card_info_get_name(info)); dev = -1; if(snd_ctl_pcm_next_device(handle, &dev) < 0) { // Card has no devices snd_ctl_close(handle); goto nextcard; } while (dev >= 0) { snd_pcm_info_set_device(pcminfo, dev); snd_pcm_info_set_subdevice(pcminfo, 0); snd_pcm_info_set_stream(pcminfo, stream); err = snd_ctl_pcm_info(handle, pcminfo); if (err == -ENOENT) goto nextdevice; nsubd = snd_pcm_info_get_subdevices_count(pcminfo); Debugprintf(" Device hw:%d,%d ID `%s', name `%s', %d subdevices (%d available)", card, dev, snd_pcm_info_get_id(pcminfo), snd_pcm_info_get_name(pcminfo), nsubd, snd_pcm_info_get_subdevices_avail(pcminfo)); sprintf(hwdev, "hw:%d,%d", card, dev); err = snd_pcm_open(&pcm, hwdev, stream, SND_PCM_NONBLOCK); if (err) { Debugprintf("Error %d opening output device", err); goto nextdevice; } // Get parameters for this device err = snd_pcm_hw_params_any(pcm, pars); snd_pcm_hw_params_get_channels_min(pars, &min); snd_pcm_hw_params_get_channels_max(pars, &max); snd_pcm_hw_params_get_rate_min(pars, &ratemin, NULL); snd_pcm_hw_params_get_rate_max(pars, &ratemax, NULL); if( min == max ) if( min == 1 ) Debugprintf(" 1 channel, sampling rate %u..%u Hz", ratemin, ratemax); else Debugprintf(" %d channels, sampling rate %u..%u Hz", min, ratemin, ratemax); else Debugprintf(" %u..%u channels, sampling rate %u..%u Hz", min, max, ratemin, ratemax); // Add device to list sprintf(NameString, "hw:%d,%d %s(%s)", card, dev, snd_pcm_info_get_name(pcminfo), snd_ctl_card_info_get_name(info)); strcpy(PlaybackNames[PlaybackCount++], NameString); snd_pcm_close(pcm); pcm= NULL; nextdevice: if (snd_ctl_pcm_next_device(handle, &dev) < 0) break; } snd_ctl_close(handle); nextcard: Debugprintf(""); if (snd_card_next(&card) < 0) // No more cards break; } return PlaybackCount; } int GetInputDeviceCollection() { // Get all the suitable devices and put in a list for GetNext to return snd_ctl_t *handle= NULL; snd_pcm_t *pcm= NULL; snd_ctl_card_info_t *info; snd_pcm_info_t *pcminfo; snd_pcm_hw_params_t *pars; snd_pcm_format_mask_t *fmask; char NameString[256]; Debugprintf("Capture Devices\n"); CaptureCount = 0; // Get Device List from ALSA snd_ctl_card_info_alloca(&info); snd_pcm_info_alloca(&pcminfo); snd_pcm_hw_params_alloca(&pars); snd_pcm_format_mask_alloca(&fmask); char hwdev[80]; unsigned min, max, ratemin, ratemax; int card, err, dev, nsubd; snd_pcm_stream_t stream = SND_PCM_STREAM_CAPTURE; card = -1; if(snd_card_next(&card) < 0) { Debugprintf("No Devices"); return 0; } if (rechandle) snd_pcm_close(rechandle); rechandle = NULL; while(card >= 0) { sprintf(hwdev, "hw:%d", card); err = snd_ctl_open(&handle, hwdev, 0); err = snd_ctl_card_info(handle, info); Debugprintf("Card %d, ID `%s', name `%s'", card, snd_ctl_card_info_get_id(info), snd_ctl_card_info_get_name(info)); dev = -1; if (snd_ctl_pcm_next_device(handle, &dev) < 0) // No Devicdes { snd_ctl_close(handle); goto nextcard; } while(dev >= 0) { snd_pcm_info_set_device(pcminfo, dev); snd_pcm_info_set_subdevice(pcminfo, 0); snd_pcm_info_set_stream(pcminfo, stream); err= snd_ctl_pcm_info(handle, pcminfo); if (err == -ENOENT) goto nextdevice; nsubd= snd_pcm_info_get_subdevices_count(pcminfo); Debugprintf(" Device hw:%d,%d ID `%s', name `%s', %d subdevices (%d available)", card, dev, snd_pcm_info_get_id(pcminfo), snd_pcm_info_get_name(pcminfo), nsubd, snd_pcm_info_get_subdevices_avail(pcminfo)); sprintf(hwdev, "hw:%d,%d", card, dev); err = snd_pcm_open(&pcm, hwdev, stream, SND_PCM_NONBLOCK); if (err) { Debugprintf("Error %d opening input device", err); goto nextdevice; } err = snd_pcm_hw_params_any(pcm, pars); snd_pcm_hw_params_get_channels_min(pars, &min); snd_pcm_hw_params_get_channels_max(pars, &max); snd_pcm_hw_params_get_rate_min(pars, &ratemin, NULL); snd_pcm_hw_params_get_rate_max(pars, &ratemax, NULL); if( min == max ) if( min == 1 ) Debugprintf(" 1 channel, sampling rate %u..%u Hz", ratemin, ratemax); else Debugprintf(" %d channels, sampling rate %u..%u Hz", min, ratemin, ratemax); else Debugprintf(" %u..%u channels, sampling rate %u..%u Hz", min, max, ratemin, ratemax); sprintf(NameString, "hw:%d,%d %s(%s)", card, dev, snd_pcm_info_get_name(pcminfo), snd_ctl_card_info_get_name(info)); // Debugprintf("%s", NameString); strcpy(CaptureNames[CaptureCount++], NameString); snd_pcm_close(pcm); pcm= NULL; nextdevice: if (snd_ctl_pcm_next_device(handle, &dev) < 0) break; } snd_ctl_close(handle); nextcard: Debugprintf(""); if (snd_card_next(&card) < 0 ) break; } strcpy(CaptureNames[CaptureCount++], "stdin"); return CaptureCount; } int OpenSoundPlayback(char * PlaybackDevice, int m_sampleRate, int channels, int Report) { int err = 0; char buf1[256]; char buf2[256]; char * ptr; if (playhandle) { snd_pcm_close(playhandle); playhandle = NULL; } strcpy(SavedPlaybackDevice, PlaybackDevice); // Saved so we can reopen in error recovery SavedPlaybackRate = m_sampleRate; strcpy(buf2, PlaybackDevice); ptr = strchr(buf2, ' '); if (ptr) *ptr = 0; // Get Device part of name if (strstr(buf2, "plug") == 0 && strchr(buf2, ':')) sprintf(buf1, "plug%s", buf2); else strcpy(buf1, buf2); if (Report) Debugprintf("Real Device %s", buf1); snd_pcm_hw_params_t *hw_params; if ((err = snd_pcm_open(&playhandle, buf1, SND_PCM_STREAM_PLAYBACK, SND_PCM_NONBLOCK)) < 0) { Debugprintf("cannot open playback audio device %s (%s)", buf1, snd_strerror(err)); return false; } if ((err = snd_pcm_hw_params_malloc (&hw_params)) < 0) { Debugprintf("cannot allocate hardware parameter structure (%s)", snd_strerror(err)); return false; } if ((err = snd_pcm_hw_params_any (playhandle, hw_params)) < 0) { Debugprintf("cannot initialize hardware parameter structure (%s)", snd_strerror(err)); return false; } if ((err = snd_pcm_hw_params_set_access (playhandle, hw_params, SND_PCM_ACCESS_RW_INTERLEAVED)) < 0) { Debugprintf("cannot set playback access type (%s)", snd_strerror (err)); return false; } if ((err = snd_pcm_hw_params_set_format (playhandle, hw_params, SND_PCM_FORMAT_S16_LE)) < 0) { Debugprintf("cannot setplayback sample format (%s)", snd_strerror(err)); return false; } if ((err = snd_pcm_hw_params_set_rate (playhandle, hw_params, m_sampleRate, 0)) < 0) { Debugprintf("cannot set playback sample rate (%s)", snd_strerror(err)); return false; } // Initial call has channels set to 1. Subequent ones set to what worked last time channels = 2; if ((err = snd_pcm_hw_params_set_channels (playhandle, hw_params, channels)) < 0) { Debugprintf("cannot set play channel count to %d (%s)", channels, snd_strerror(err)); if (channels == 2) return false; // Shouldn't happen as should have worked before channels = 2; if ((err = snd_pcm_hw_params_set_channels (playhandle, hw_params, 2)) < 0) { Debugprintf("cannot play set channel count to 2 (%s)", snd_strerror(err)); return false; } } if (Report) Debugprintf("Play using %d channels", channels); if ((err = snd_pcm_hw_params (playhandle, hw_params)) < 0) { Debugprintf("cannot set parameters (%s)", snd_strerror(err)); return false; } snd_pcm_hw_params_free(hw_params); if ((err = snd_pcm_prepare (playhandle)) < 0) { Debugprintf("cannot prepare audio interface for use (%s)", snd_strerror(err)); return false; } Savedplaychannels = m_playchannels = channels; MaxAvail = snd_pcm_avail_update(playhandle); if (Report) Debugprintf("Playback Buffer Size %d", (int)MaxAvail); return true; } int OpenSoundCapture(char * CaptureDevice, int m_sampleRate, int Report) { int err = 0; char buf1[256]; char buf2[256]; char * ptr; snd_pcm_hw_params_t *hw_params; if (strcmp(CaptureDevice, "stdin") == 0) { stdinMode = 1; Debugprintf("Input from stdin"); return TRUE; } if (rechandle) { snd_pcm_close(rechandle); rechandle = NULL; } strcpy(SavedCaptureDevice, CaptureDevice); // Saved so we can reopen in error recovery SavedCaptureRate = m_sampleRate; strcpy(buf2, CaptureDevice); ptr = strchr(buf2, ' '); if (ptr) *ptr = 0; // Get Device part of name if (strstr(buf2, "plug") == 0 && strchr(buf2, ':')) sprintf(buf1, "plug%s", buf2); else strcpy(buf1, buf2); if (Report) Debugprintf("Real Device %s", buf1); if ((err = snd_pcm_open (&rechandle, buf1, SND_PCM_STREAM_CAPTURE, 0)) < 0) { Debugprintf("cannot open capture audio device %s (%s)", buf1, snd_strerror(err)); return false; } if ((err = snd_pcm_hw_params_malloc (&hw_params)) < 0) { Debugprintf("cannot allocate capture hardware parameter structure (%s)", snd_strerror(err)); return false; } if ((err = snd_pcm_hw_params_any (rechandle, hw_params)) < 0) { Debugprintf("cannot initialize capture hardware parameter structure (%s)", snd_strerror(err)); return false; } if ((err = snd_pcm_hw_params_set_access (rechandle, hw_params, SND_PCM_ACCESS_RW_INTERLEAVED)) < 0) { Debugprintf("cannot set capture access type (%s)", snd_strerror (err)); return false; } if ((err = snd_pcm_hw_params_set_format (rechandle, hw_params, SND_PCM_FORMAT_S16_LE)) < 0) { Debugprintf("cannot set capture sample format (%s)", snd_strerror(err)); return false; } if ((err = snd_pcm_hw_params_set_rate (rechandle, hw_params, m_sampleRate, 0)) < 0) { Debugprintf("cannot set capture sample rate (%s)", snd_strerror(err)); return false; } m_recchannels = 2; if ((err = snd_pcm_hw_params_set_channels(rechandle, hw_params, m_recchannels)) < 0) { if (Report) Debugprintf("cannot set rec channel count to 2 (%s)", snd_strerror(err)); m_recchannels = 1; if ((err = snd_pcm_hw_params_set_channels(rechandle, hw_params, 1)) < 0) { Debugprintf("cannot set rec channel count to 1 (%s)", snd_strerror(err)); return false; } if (Report) Debugprintf("Record channel count set to 1"); } else if (Report) Debugprintf("Record channel count set to 2"); /* { unsigned int val = 0; unsigned int dir = 0, frames = 0; snd_pcm_hw_params_get_channels(rechandle, &val); printf("channels = %d\n", val); snd_pcm_hw_params_get_rate(rechandle, &val, &dir); printf("rate = %d bps\n", val); snd_pcm_hw_params_get_period_time(rechandle, &val, &dir); printf("period time = %d us\n", val); snd_pcm_hw_params_get_period_size(rechandle, &frames, &dir); printf("period size = %d frames\n", (int)frames); snd_pcm_hw_params_get_buffer_time(rechandle, &val, &dir); printf("buffer time = %d us\n", val); snd_pcm_hw_params_get_buffer_size(rechandle, (snd_pcm_uframes_t *)&val); printf("buffer size = %d frames\n", val); snd_pcm_hw_params_get_periods(rechandle, &val, &dir); printf("periods per buffer = %d frames\n", val); } */ if ((err = snd_pcm_hw_params (rechandle, hw_params)) < 0) { // Try setting some more params Have to reinit params snd_pcm_hw_params_any(rechandle, hw_params); snd_pcm_hw_params_set_access(rechandle, hw_params, SND_PCM_ACCESS_RW_INTERLEAVED); snd_pcm_hw_params_set_format(rechandle, hw_params, SND_PCM_FORMAT_S16_LE); snd_pcm_hw_params_set_rate(rechandle, hw_params, m_sampleRate, 0); snd_pcm_hw_params_set_channels(rechandle, hw_params, m_recchannels); // err = snd_pcm_hw_params_set_buffer_size(rechandle, hw_params, 65536); // if (err) // Debugprintf("cannot set buffer size (%s)", snd_strerror(err)); err = snd_pcm_hw_params_set_period_size(rechandle, hw_params, (snd_pcm_uframes_t) { 1024 }, (int) { 0 }); if (err) Debugprintf("cannot set period size (%s)", snd_strerror(err)); if ((err = snd_pcm_hw_params(rechandle, hw_params)) < 0) { Debugprintf("cannot set parameters (%s)", snd_strerror(err)); return false; } } snd_pcm_hw_params_free(hw_params); if ((err = snd_pcm_prepare (rechandle)) < 0) { Debugprintf("cannot prepare audio interface for use (%s)", snd_strerror(err)); return FALSE; } if (Report) Debugprintf("Capture using %d channels", m_recchannels); short buf[256]; if ((err = snd_pcm_readi(rechandle, buf, 12)) != 12) { Debugprintf("read from audio interface failed (%s)", snd_strerror(err)); } // Debugprintf("Read got %d", err); return TRUE; } int OpenSoundCard(char * CaptureDevice, char * PlaybackDevice, int c_sampleRate, int p_sampleRate, int Report) { int Channels = 1; if (Report) Debugprintf("Opening Playback Device %s Rate %d", PlaybackDevice, p_sampleRate); // if (UseLeft == 0 || UseRight == 0) Channels = 2; // L or R implies stereo if (OpenSoundPlayback(PlaybackDevice, p_sampleRate, Channels, Report)) { #ifdef SHARECAPTURE // Close playback device so it can be shared if (playhandle) { snd_pcm_close(playhandle); playhandle = NULL; } #endif if (Report) Debugprintf("Opening Capture Device %s Rate %d", CaptureDevice, c_sampleRate); return OpenSoundCapture(CaptureDevice, c_sampleRate, Report); } else return false; } int CloseSoundCard() { if (rechandle) { snd_pcm_close(rechandle); rechandle = NULL; } if (playhandle) { snd_pcm_close(playhandle); playhandle = NULL; } return 0; } int SoundCardWrite(short * input, int nSamples) { unsigned int ret; snd_pcm_sframes_t avail; // , maxavail; if (playhandle == NULL) return 0; // Stop Capture // if (rechandle) // { // snd_pcm_close(rechandle); // rechandle = NULL; // } avail = snd_pcm_avail_update(playhandle); // Debugprintf("avail before play returned %d", (int)avail); if (avail < 0) { if (avail != -32) Debugprintf("Playback Avail Recovering from %d ..", (int)avail); snd_pcm_recover(playhandle, avail, 1); avail = snd_pcm_avail_update(playhandle); if (avail < 0) Debugprintf("avail play after recovery returned %d", (int)avail); } // maxavail = avail; // Debugprintf("Tosend %d Avail %d", nSamples, (int)avail); while (avail < nSamples || (MaxAvail - avail) > SampleRate) // Limit to 1 sec of audio { txSleep(10); avail = snd_pcm_avail_update(playhandle); // Debugprintf("After Sleep Tosend %d Avail %d", nSamples, (int)avail); } ret = PackSamplesAndSend(input, nSamples); return ret; } int PackSamplesAndSend(short * input, int nSamples) { unsigned short samples[256000]; int ret; ret = snd_pcm_writei(playhandle, input, nSamples); if (ret < 0) { // Debugprintf("Write Recovering from %d ..", ret); snd_pcm_recover(playhandle, ret, 1); ret = snd_pcm_writei(playhandle, samples, nSamples); // Debugprintf("Write after recovery returned %d", ret); } snd_pcm_avail_update(playhandle); return ret; } /* int xSoundCardClearInput() { short samples[65536]; int n; int ret; int avail; if (rechandle == NULL) return 0; // Clear queue avail = snd_pcm_avail_update(rechandle); if (avail < 0) { Debugprintf("Discard Recovering from %d ..", avail); if (rechandle) { snd_pcm_close(rechandle); rechandle = NULL; } OpenSoundCapture(SavedCaptureDevice, SavedCaptureRate, NULL); avail = snd_pcm_avail_update(rechandle); } while (avail) { if (avail > 65536) avail = 65536; ret = snd_pcm_readi(rechandle, samples, avail); // Debugprintf("Discarded %d samples from card", ret); avail = snd_pcm_avail_update(rechandle); // Debugprintf("Discarding %d samples from card", avail); } return 0; } */ int SoundCardRead(short * input, int nSamples) { short samples[65536]; int n; int ret; int avail; if (SoundMode == 1) // OSS { ret = oss_read(samples, nSamples); } else if (SoundMode == 2)// Pulse { ret = pulse_read(samples, nSamples); } else { if (rechandle == NULL) return 0; avail = snd_pcm_avail_update(rechandle); if (avail < 0) { Debugprintf("avail Recovering from %d ..", avail); if (rechandle) { snd_pcm_close(rechandle); rechandle = NULL; } OpenSoundCapture(SavedCaptureDevice, SavedCaptureRate, 0); // snd_pcm_recover(rechandle, avail, 0); avail = snd_pcm_avail_update(rechandle); Debugprintf("After avail recovery %d ..", avail); } if (avail < nSamples) return 0; // Debugprintf("ALSARead available %d", avail); ret = snd_pcm_readi(rechandle, samples, nSamples); if (ret < 0) { Debugprintf("RX Error %d", ret); // snd_pcm_recover(rechandle, avail, 0); if (rechandle) { snd_pcm_close(rechandle); rechandle = NULL; } OpenSoundCapture(SavedCaptureDevice, SavedCaptureRate, 0); // snd_pcm_recover(rechandle, avail, 0); avail = snd_pcm_avail_update(rechandle); Debugprintf("After Read recovery Avail %d ..", avail); return 0; } } if (ret < nSamples) return 0; if (m_recchannels == 1) { for (n = 0; n < ret; n++) { *(input++) = samples[n]; *(input++) = samples[n]; // Duplicate } } else { for (n = 0; n < ret * 2; n++) // return all { *(input++) = samples[n]; } } return ret; } int PriorSize = 0; int Index = 0; // DMA Buffer being used 0 or 1 int inIndex = 0; // DMA Buffer being used 0 or 1 BOOL DMARunning = FALSE; // Used to start DMA on first write void ProcessNewSamples(short * Samples, int nSamples); short * SendtoCard(short * buf, int n) { if (Loopback) { // Loop back to decode for testing ProcessNewSamples(buf, 1200); // signed } if (SoundMode == 1) // OSS oss_write(buf, n); else if (SoundMode == 2) // Pulse pulse_write(buf, n); else { if (playhandle) SoundCardWrite(buf, n); // txSleep(10); // Run buckground while waiting } Index = !Index; return &buffer[Index][0]; } short loopbuff[1200]; // Temp for testing - loop sent samples to decoder // // This generates a nice musical pattern for sound interface testing // for (t = 0; t < sizeof(buffer); ++t) // buffer[t] =((((t * (t >> 8 | t >> 9) & 46 & t >> 8)) ^ (t & t >> 13 | t >> 6)) & 0xFF); short * SoundInit(); void GetSoundDevices() { if (onlyMixSnoop) { gethints(); } else { if (SoundMode == 0) { GetInputDeviceCollection(); GetOutputDeviceCollection(); } else if (SoundMode == 1) { PlaybackCount = 3; strcpy(&PlaybackNames[0][0], "/dev/dsp0"); strcpy(&PlaybackNames[1][0], "/dev/dsp1"); strcpy(&PlaybackNames[2][0], "/dev/dsp2"); CaptureCount = 3; strcpy(&CaptureNames[0][0], "/dev/dsp0"); strcpy(&CaptureNames[1][0], "/dev/dsp1"); strcpy(&CaptureNames[2][0], "/dev/dsp2"); } else if (SoundMode == 2) { // Pulse listpulse(); } } } int InitSound(BOOL Quiet) { if (using48000) { ReceiveSize = 2048; SendSize = 4096; // 100 mS for now SampleRate = 48000; } else { ReceiveSize = 512; SendSize = 1024; SampleRate = 12000; } GetSoundDevices(); switch (SoundMode) { case 0: // ALSA if (!OpenSoundCard(CaptureDevice, PlaybackDevice, SampleRate, SampleRate, Quiet)) return FALSE; break; case 1: // OSS if (!oss_audio_open(CaptureDevice, PlaybackDevice)) return FALSE; break; case 2: // PulseAudio if (!pulse_audio_open(CaptureDevice, PlaybackDevice)) return FALSE; break; } printf("InitSound %s %s\n", CaptureDevice, PlaybackDevice); DMABuffer = SoundInit(); return TRUE; } int min = 0, max = 0, lastlevelreport = 0, lastlevelGUI = 0; UCHAR CurrentLevel = 0; // Peak from current samples UCHAR CurrentLevelR = 0; // Peak from current samples void PollReceivedSamples() { // Process any captured samples // Ideally call at least every 100 mS, more than 200 will loose data int bytes; #ifdef TXSILENCE SendSilence(); // send silence (attempt to fix CM delay issue) #endif if (stdinMode) { // will block if no input. May get less, in which case wait a bit then try to read rest // rtl_udp outputs mono samples short input[1200]; short * ptr1, *ptr2; int n = 20; // Max Wait bytes = read(STDIN_FILENO, input, ReceiveSize * 2); // 4 = Stereo 2 bytes per sample while (bytes < ReceiveSize * 2 && n--) { Sleep(50); //mS bytes += read(STDIN_FILENO, &input[bytes / 2], (ReceiveSize * 2) - bytes); } // if still not enough, too bad! if (bytes != ReceiveSize * 2) { // This seems to happen occasionally even when we shouldn't be in stdin mode. Exit Debugprintf("Short Read %d", bytes); closeTraceLog(); Closing = TRUE; sleep(1); exit(1); } // convert to stereo ptr1 = input; ptr2 = inbuffer; n = ReceiveSize; while (n--) { *ptr2++ = *ptr1; *ptr2++ = *ptr1++; } } else bytes = SoundCardRead(inbuffer, ReceiveSize); // returns ReceiveSize or none if (bytes > 0) { short * ptr = inbuffer; int i; for (i = 0; i < ReceiveSize; i++) { if (*(ptr) < min) min = *ptr; else if (*(ptr) > max) max = *ptr; ptr++; } CurrentLevel = ((max - min) * 75) /32768; // Scale to 150 max if ((Now - lastlevelGUI) > 2000) // 2 Secs { lastlevelGUI = Now; if ((Now - lastlevelreport) > 60000) // 60 Secs { char HostCmd[64]; lastlevelreport = Now; sprintf(HostCmd, "INPUTPEAKS %d %d", min, max); Debugprintf("Input peaks = %d, %d", min, max); } min = max = 0; // Every 2 secs } ProcessNewSamples(inbuffer, ReceiveSize); } } void StopCapture() { Capturing = FALSE; #ifdef SHARECAPTURE // Stopcapture is only called when we are about to transmit, so use it to open plaback device. We don't keep // it open all the time to facilitate sharing. OpenSoundPlayback(SavedPlaybackDevice, SavedPlaybackRate, Savedplaychannels, NULL); #endif } void StartCapture() { Capturing = TRUE; // Debugprintf("Start Capture"); } void CloseSound() { switch (SoundMode) { case 0: // ALSA CloseSoundCard(); return; case 1: // OSS oss_audio_close(); return; case 2: // PulseAudio pulse_audio_close(); return; } } short * SoundInit() { Index = 0; return &buffer[0][0]; } // Called at end of transmission int useTimedPTT = 1; extern int SampleNo; int pttOnTime(); void SoundFlush() { // Append Trailer then send remaining samples snd_pcm_status_t *status = NULL; int err, res; int lastavail = 0; if (Loopback) ProcessNewSamples(&buffer[Index][0], Number); SendtoCard(&buffer[Index][0], Number); // Wait for tx to complete // Debugprintf("Flush Soundmode = %d", SoundMode); if (SoundMode == 0) // ALSA { if (useTimedPTT) { // Calulate PTT Time from Number of samples and samplerate // samples sent is is in SampleNo, Time PTT was raised in timeval pttclk // txLatency is extra ptt time to compenstate for time soundcard takes to start outputting samples struct timespec pttnow; clock_gettime(CLOCK_MONOTONIC, &pttnow); time_t pttontimemS = (pttclk.tv_sec * 1000) + (pttclk.tv_nsec / 1000000); time_t nowtimemS = (pttnow.tv_sec * 1000) + (pttnow.tv_nsec / 1000000); // We have already added latency to tail, so don't add again int txlenMs = (1000 * SampleNo / TX_Samplerate); // 12000 samples per sec. Debugprintf("Tx Time %d Time till end = %d", txlenMs, (nowtimemS - pttontimemS)); txSleep(txlenMs - (nowtimemS - pttontimemS)); } else { usleep(100000); while (1 && playhandle) { snd_pcm_sframes_t avail = snd_pcm_avail_update(playhandle); // Debugprintf("Waiting for complete. Avail %d Max %d", avail, MaxAvail); snd_pcm_status_alloca(&status); // alloca allocates once per function, does not need a free // Debugprintf("Waiting for complete. Avail %d Max %d last %d", avail, MaxAvail, lastavail); if ((err = snd_pcm_status(playhandle, status)) != 0) { Debugprintf("snd_pcm_status() failed: %s", snd_strerror(err)); break; } res = snd_pcm_status_get_state(status); // Debugprintf("PCM Status = %d", res); if (res != SND_PCM_STATE_RUNNING || lastavail == avail) // If sound system is not running then it needs data // if (res != SND_PCM_STATE_RUNNING) // If sound system is not running then it needs data // if (MaxAvail - avail < 100) { // Send complete - Restart Capture OpenSoundCapture(SavedCaptureDevice, SavedCaptureRate, 0); break; } lastavail = avail; usleep(50000); } } // I think we should turn round the link here. I dont see the point in // waiting for MainPoll #ifdef SHARECAPTURE if (playhandle) { snd_pcm_close(playhandle); playhandle = NULL; } #endif } else if (SoundMode == 1) { oss_flush(); } else if (SoundMode == 2) { pulse_flush(); } SoundIsPlaying = FALSE; Number = 0; memset(buffer, 0, sizeof(buffer)); DMABuffer = &buffer[0][0]; #ifdef TXSILENCE SendtoCard(&buffer[0][0], 1200); // Start sending silence (attempt to fix CM delay issue) #endif StartCapture(); return; } #ifdef TXSILENCE // send silence (attempt to fix CM delay issue) void SendSilence() { short buffer[2400]; snd_pcm_sframes_t Avail = snd_pcm_avail_update(playhandle); if ((MaxAvail - Avail) < 1200) { // Keep at least 100 ms of audio in buffer // printtick("Silence"); memset(buffer, 0, sizeof(buffer)); SendtoCard(buffer, 1200); // Start sending silence (attempt to fix CM delay issue) } } #endif // GPIO access stuff for PTT on PI #ifdef __ARM_ARCH /* tiny_gpio.c 2016-04-30 Public Domain */ #include #include #include #include #include #include #include #include #define GPSET0 7 #define GPSET1 8 #define GPCLR0 10 #define GPCLR1 11 #define GPLEV0 13 #define GPLEV1 14 #define GPPUD 37 #define GPPUDCLK0 38 #define GPPUDCLK1 39 unsigned piModel; unsigned piRev; static volatile uint32_t *gpioReg = MAP_FAILED; #define PI_BANK (gpio>>5) #define PI_BIT (1<<(gpio&0x1F)) /* gpio modes. */ void gpioSetMode(unsigned gpio, unsigned mode) { int reg, shift; reg = gpio/10; shift = (gpio%10) * 3; gpioReg[reg] = (gpioReg[reg] & ~(7<> shift) & 7; } /* Values for pull-ups/downs off, pull-down and pull-up. */ #define PI_PUD_OFF 0 #define PI_PUD_DOWN 1 #define PI_PUD_UP 2 void gpioSetPullUpDown(unsigned gpio, unsigned pud) { *(gpioReg + GPPUD) = pud; usleep(20); *(gpioReg + GPPUDCLK0 + PI_BANK) = PI_BIT; usleep(20); *(gpioReg + GPPUD) = 0; *(gpioReg + GPPUDCLK0 + PI_BANK) = 0; } int gpioRead(unsigned gpio) { if ((*(gpioReg + GPLEV0 + PI_BANK) & PI_BIT) != 0) return 1; else return 0; } void gpioWrite(unsigned gpio, unsigned level) { if (level == 0) *(gpioReg + GPCLR0 + PI_BANK) = PI_BIT; else *(gpioReg + GPSET0 + PI_BANK) = PI_BIT; } void gpioTrigger(unsigned gpio, unsigned pulseLen, unsigned level) { if (level == 0) *(gpioReg + GPCLR0 + PI_BANK) = PI_BIT; else *(gpioReg + GPSET0 + PI_BANK) = PI_BIT; usleep(pulseLen); if (level != 0) *(gpioReg + GPCLR0 + PI_BANK) = PI_BIT; else *(gpioReg + GPSET0 + PI_BANK) = PI_BIT; } /* Bit (1<user_speed != -1; s++) if (s->user_speed == speed) break; if (s->user_speed == -1) { fprintf(stderr, "tty_speed: invalid speed %d", speed); return FALSE; } if (tcgetattr(fd, &term) == -1) { perror("tty_speed: tcgetattr"); return FALSE; } cfmakeraw(&term); cfsetispeed(&term, s->termios_speed); cfsetospeed(&term, s->termios_speed); if (tcsetattr(fd, TCSANOW, &term) == -1) { perror("tty_speed: tcsetattr"); return FALSE; } ioctl(fd, FIONBIO, ¶m); Debugprintf("Port %s fd %d", fulldev, fd); if (SetDTR) COMSetDTR(fd); else COMClearDTR(fd); if (SetRTS) COMSetRTS(fd); else COMClearRTS(fd); return fd; } BOOL WriteCOMBlock(HANDLE fd, char * Block, int BytesToWrite) { // Some systems seem to have a very small max write size int ToSend = BytesToWrite; int Sent = 0, ret; while (ToSend) { ret = write(fd, &Block[Sent], ToSend); if (ret >= ToSend) return TRUE; // perror("WriteCOM"); if (ret == -1) { if (errno != 11 && errno != 35) // Would Block return FALSE; usleep(10000); ret = 0; } Sent += ret; ToSend -= ret; } return TRUE; } VOID CloseCOMPort(HANDLE fd) { close(fd); } // "hints" processing for looking for SNOOP/MIX devices int gethints() { const char *iface = "pcm"; void **hints; char **n; int err; char hwdev[256]; snd_pcm_t *pcm = NULL; char NameString[256]; CloseSoundCard(); Debugprintf("Available Mix/Snoop Devices\n"); PlaybackCount = 0; CaptureCount = 0; err = snd_device_name_hint(-1, iface, &hints); if (err < 0) Debugprintf("snd_device_name_hint error: %s", snd_strerror(err)); n = (char **)hints; while (*n != NULL) { if (memcmp(*n, "NAMEmix", 7) == 0) //NAMEmix00|DESCQtSM Mix for hw0:0 { char Hint[256]; char * ptr; snd_pcm_stream_t stream = SND_PCM_STREAM_PLAYBACK; strcpy(Hint, *n); ptr = strchr(Hint, '|'); if (ptr) { *ptr++ = 0; } strcpy(hwdev, &Hint[4]); err = snd_pcm_open(&pcm, hwdev, stream, SND_PCM_NONBLOCK); if (err) { Debugprintf("Error %d opening output device %s ", err, hwdev); goto nextdevice; } // Add device to list if (ptr) sprintf(NameString, "%s %s", hwdev, &ptr[4]); else strcpy(NameString, hwdev); Debugprintf(NameString); strcpy(PlaybackNames[PlaybackCount++], NameString); snd_pcm_close(pcm); pcm = NULL; } else if (memcmp(*n, "NAMEsnoop", 9) == 0) { char Hint[256]; char * ptr; snd_pcm_stream_t stream = SND_PCM_STREAM_CAPTURE; strcpy(Hint, *n); ptr = strchr(Hint, '|'); if (ptr) { *ptr++ = 0; } strcpy(hwdev, &Hint[4]); err = snd_pcm_open(&pcm, hwdev, stream, SND_PCM_NONBLOCK); if (err) { Debugprintf("Error %d opening input device %s ", err, hwdev); goto nextdevice; } // Add device to list if (ptr) sprintf(NameString, "%s %s", hwdev, &ptr[4]); else strcpy(NameString, hwdev); Debugprintf(NameString); strcpy(CaptureNames[CaptureCount++], NameString); snd_pcm_close(pcm); pcm = NULL; } nextdevice: n++; } snd_device_name_free_hint(hints); return 0; } // Microsoft routines not available in gcc int memicmp(unsigned char *a, unsigned char *b, int n) { if (n) { while (n && (toupper(*a) == toupper(*b))) n--, a++, b++; if (n) return toupper(*a) - toupper(*b); } return 0; } int stricmp(const unsigned char * pStr1, const unsigned char *pStr2) { unsigned char c1, c2; int v; if (pStr1 == NULL) { if (pStr2) Debugprintf("stricmp called with NULL 1st param - 2nd %s ", pStr2); else Debugprintf("stricmp called with two NULL params"); return 1; } do { c1 = *pStr1++; c2 = *pStr2++; /* The casts are necessary when pStr1 is shorter & char is signed */ v = tolower(c1) - tolower(c2); } while ((v == 0) && (c1 != '\0') && (c2 != '\0')); return v; } char * strupr(char* s) { char* p = s; if (s == 0) return 0; while (*p = toupper(*p)) p++; return s; } char * strlwr(char* s) { char* p = s; while (*p = tolower(*p)) p++; return s; }