Mercurial > hg > beaglert
view core/RTAudioCommandLine.cpp @ 45:579c86316008 newapi
Major API overhaul. Moved to a single data structure for handling render functions. Functionally, generally similar except for scheduling within PRU loop function, which now uses interrupts from the PRU rather than polling. This requires an updated kernel.
author | andrewm |
---|---|
date | Thu, 28 May 2015 14:35:55 -0400 |
parents | 46d87f680da5 |
children | 643cbee74eda |
line wrap: on
line source
/* * RTAudioCommandLine.cpp * * Created on: Nov 8, 2014 * Author: parallels */ #include <iostream> #include <cstdlib> #include <cstring> #include <getopt.h> #include "../include/BeagleRT.h" #ifndef OPT_PRU_FILE #define OPT_PRU_FILE 176 // this is an extended ASCII code #endif // Default command-line options for RTAudio struct option gDefaultLongOptions[] = { {"period", 1, NULL, 'p'}, {"verbose", 0, NULL, 'v'}, {"use-analog", 1, NULL, 'm'}, {"use-digital-gpio", 1, NULL, 'g'}, {"analog-channels", 1, NULL, 'C'}, {"digital-channels", 1, NULL, 'G'}, {"mute-speaker", 1, NULL, 'M'}, {"dac-level", 1, NULL, 'D'}, {"adc-level", 1, NULL, 'A'}, {"hp-level", 1, NULL, 'H'}, {"receive-port", 1, NULL, 'r'}, {"transmit-port", 1, NULL, 't'}, {"server-name", 1, NULL, 's'}, {"pru-file", 1, NULL, OPT_PRU_FILE}, {NULL, 0, NULL, 0} }; const char gDefaultShortOptions[] = "p:vm:M:C:D:A:H:g:G:r:t:s:"; // This function sets the default settings for the BeagleRTInitSettings structure void BeagleRT_defaultSettings(BeagleRTInitSettings *settings) { // Set default values for settings settings->periodSize = 8; settings->useAnalog = 1; settings->useDigital = 1; settings->numAnalogChannels = 8; settings->numDigitalChannels = 16; settings->beginMuted = 0; settings->dacLevel = DEFAULT_DAC_LEVEL; settings->adcLevel = DEFAULT_ADC_LEVEL; settings->headphoneLevel = DEFAULT_HP_LEVEL; settings->verbose = 0; settings->pruFilename[0] = '\0'; // These two deliberately have no command-line flags by default. // A given program might prefer one mode or another, but it's unlikely // the user would want to switch at runtime settings->interleave = 1; settings->analogOutputsPersist = 1; settings->codecI2CAddress = CODEC_I2C_ADDRESS; settings->receivePort = 9998; settings->transmitPort = 9999; strcpy(settings->serverName, "127.0.0.1"); settings->ampMutePin = kAmplifierMutePin; } // This function drops in place of getopt() in the main() function // and handles the initialisation of the RTAudio settings using // standard command-line arguments. System default arguments will // be stored in settings, otherwise arguments will be returned // as getopt() normally does. int BeagleRT_getopt_long(int argc, char *argv[], const char *customShortOptions, const struct option *customLongOptions, BeagleRTInitSettings *settings) { static int firstRun = 1; static char totalShortOptions[256]; static struct option totalLongOptions[256]; int c; // Prep total option string the first time this is // run. As a getopt() substitute, it will be called repeatedly working its // way through argc and argv. if(firstRun) { firstRun = 0; // Copy short options into one string strcpy(totalShortOptions, gDefaultShortOptions); strncat(totalShortOptions, customShortOptions, 256 - strlen(gDefaultShortOptions) - 1); // Copy long options into one array int n = 0; while(1) { if(gDefaultLongOptions[n].name == NULL) break; totalLongOptions[n].name = gDefaultLongOptions[n].name; totalLongOptions[n].has_arg = gDefaultLongOptions[n].has_arg; totalLongOptions[n].flag = gDefaultLongOptions[n].flag; totalLongOptions[n].val = gDefaultLongOptions[n].val; n++; } // Copy custom options into the array, if present if(customLongOptions == 0) { // Terminate the array totalLongOptions[n].name = NULL; totalLongOptions[n].has_arg = 0; totalLongOptions[n].flag = NULL; totalLongOptions[n].val = 0; } else { int customIndex = 0; while(n < 256) { if(customLongOptions[customIndex].name == NULL) break; totalLongOptions[n].name = customLongOptions[customIndex].name; totalLongOptions[n].has_arg = customLongOptions[customIndex].has_arg; totalLongOptions[n].flag = customLongOptions[customIndex].flag; totalLongOptions[n].val = customLongOptions[customIndex].val; n++; customIndex++; } // Terminate the array totalLongOptions[n].name = NULL; totalLongOptions[n].has_arg = 0; totalLongOptions[n].flag = NULL; totalLongOptions[n].val = 0; } } while(1) { if ((c = getopt_long(argc, argv, totalShortOptions, totalLongOptions, NULL)) < 0) return c; switch (c) { case 'p': settings->periodSize = atoi(optarg); if(settings->periodSize < 1) settings->periodSize = 1; break; case 'v': settings->verbose = 1; break; case 'm': settings->useAnalog = atoi(optarg); break; case 'g': settings->useDigital = atoi(optarg); settings->numDigitalChannels = 0; break; case 'C': settings->numAnalogChannels = atoi(optarg); if(settings->numAnalogChannels >= 8) settings->numAnalogChannels = 8; else if(settings->numAnalogChannels >= 4) settings->numAnalogChannels = 4; else settings->numAnalogChannels = 2; break; case 'G': settings->numDigitalChannels = atoi(optarg); if(settings->numDigitalChannels >= 16) settings->numDigitalChannels = 16; else if (settings->numDigitalChannels < 1){ settings->numDigitalChannels = 0; settings->useDigital = 0; //TODO: this actually works only if -G 0 is specified after -g 1. //No worries, though: disabling numDigital will only prevent the pins from being exported. } break; case 'M': settings->beginMuted = atoi(optarg); break; case 'D': settings->dacLevel = atof(optarg); break; case 'A': settings->adcLevel = atof(optarg); break; case 'H': settings->headphoneLevel = atof(optarg); break; case 'r': settings->receivePort = atoi(optarg); break; case 't': settings->transmitPort = atoi(optarg); break; case 's': if(strlen(optarg)<MAX_SERVERNAME_LENGTH) strcpy(settings->serverName, optarg); else std::cerr << "Warning: server name is too long (>" << MAX_SERVERNAME_LENGTH << " characters)." " Using default severName Instead ( " << settings->serverName << " ).\n"; break; case OPT_PRU_FILE: if(strlen(optarg) < MAX_PRU_FILENAME_LENGTH) strcpy(settings->pruFilename, optarg); else std::cerr << "Warning: filename for the PRU code is too long (>" << MAX_PRU_FILENAME_LENGTH << " characters). Using embedded PRU code instead\n"; break; case '?': default: return c; } } } // This function prints standard usage information for default arguments // Call from within your own usage function void BeagleRT_usage() { std::cerr << " --period [-p] period: Set the hardware period (buffer) size in analog samples\n"; std::cerr << " --dac-level [-D] dBs: Set the DAC output level (0dB max; -63.5dB min)\n"; std::cerr << " --adc-level [-A] dBs: Set the ADC input level (0dB max; -12dB min)\n"; std::cerr << " --hp-level [-H] dBs: Set the headphone output level (0dB max; -63.5dB min)\n"; std::cerr << " --mute-speaker [-M] val: Set whether to mute the speaker initially (default: no)\n"; std::cerr << " --use-analog [-m] val: Set whether to use ADC/DAC analog (default: yes)\n"; std::cerr << " --use-digital [-g] val: Set whether to use digital GPIO channels (default: yes)\n"; std::cerr << " --analog-channels [-C] val: Set the number of ADC/DAC channels (default: 8)\n"; std::cerr << " --digital-channels [-G] val: Set the number of GPIO channels (default: 16)\n"; std::cerr << " --digital-channels [-G] val: Set the number of digital GPIO channels (default: 16)\n"; std::cerr << " --receive-port [-r] val: Set the receive port (default: 9998)\n"; std::cerr << " --transmit-port [-t] val: Set the transmit port (default: 9999)\n"; std::cerr << " --server-name [-s] val: Set the destination server name (default: '127.0.0.1')\n"; std::cerr << " --pru-file val: Set an optional external file to use for the PRU binary code\n"; std::cerr << " --verbose [-v]: Enable verbose logging information\n"; }