andrewm@5: /* andrewm@5: * RTAudioCommandLine.cpp andrewm@5: * andrewm@5: * Created on: Nov 8, 2014 andrewm@5: * Author: parallels andrewm@5: */ andrewm@5: andrewm@5: #include andrewm@5: #include andrewm@5: #include andrewm@5: #include andrewm@5: andrewm@5: #include "../include/RTAudio.h" andrewm@5: andrewm@5: // Default command-line options for RTAudio andrewm@5: struct option gDefaultLongOptions[] = andrewm@5: { andrewm@5: {"period", 1, NULL, 'p'}, andrewm@5: {"verbose", 0, NULL, 'v'}, andrewm@5: {"use-matrix", 1, NULL, 'm'}, andrewm@5: {"mute-speaker", 1, NULL, 'M'}, andrewm@5: {"dac-level", 1, NULL, 'D'}, andrewm@5: {"adc-level", 1, NULL, 'A'}, andrewm@5: {"hp-level", 1, NULL, 'H'}, andrewm@5: {NULL, 0, NULL, 0} andrewm@5: }; andrewm@5: andrewm@5: const char gDefaultShortOptions[] = "p:vm:M:D:A:H:"; andrewm@5: andrewm@5: // This function sets the default settings for the RTAudioSettings structure andrewm@5: void BeagleRT_defaultSettings(RTAudioSettings *settings) andrewm@5: { andrewm@5: // Set default values for settings andrewm@5: settings->periodSize = 8; andrewm@5: settings->beginMuted = 0; andrewm@5: settings->dacLevel = DEFAULT_DAC_LEVEL; andrewm@5: settings->adcLevel = DEFAULT_ADC_LEVEL; andrewm@5: settings->headphoneLevel = DEFAULT_HP_LEVEL; andrewm@5: settings->useMatrix = 1; andrewm@5: settings->verbose = 0; andrewm@5: settings->codecI2CAddress = CODEC_I2C_ADDRESS; andrewm@5: settings->ampMutePin = kAmplifierMutePin; andrewm@5: } andrewm@5: andrewm@5: // This function drops in place of getopt() in the main() function andrewm@5: // and handles the initialisation of the RTAudio settings using andrewm@5: // standard command-line arguments. System default arguments will andrewm@5: // be stored in settings, otherwise arguments will be returned andrewm@5: // as getopt() normally does. andrewm@5: andrewm@5: int BeagleRT_getopt_long(int argc, char *argv[], const char *customShortOptions, const struct option *customLongOptions, RTAudioSettings *settings) andrewm@5: { andrewm@5: static int firstRun = 1; andrewm@5: static char totalShortOptions[256]; andrewm@5: static struct option totalLongOptions[256]; andrewm@5: andrewm@5: int c; andrewm@5: andrewm@5: // Prep total option string the first time this is andrewm@5: // run. As a getopt() substitute, it will be called repeatedly working its andrewm@5: // way through argc and argv. andrewm@5: if(firstRun) { andrewm@5: firstRun = 0; andrewm@5: andrewm@5: // Copy short options into one string andrewm@5: strcpy(totalShortOptions, gDefaultShortOptions); andrewm@5: strncat(totalShortOptions, customShortOptions, 256 - strlen(gDefaultShortOptions) - 1); andrewm@5: andrewm@5: // Copy long options into one array andrewm@5: int n = 0; andrewm@5: while(1) { andrewm@5: if(gDefaultLongOptions[n].name == NULL) andrewm@5: break; andrewm@5: totalLongOptions[n].name = gDefaultLongOptions[n].name; andrewm@5: totalLongOptions[n].has_arg = gDefaultLongOptions[n].has_arg; andrewm@5: totalLongOptions[n].flag = gDefaultLongOptions[n].flag; andrewm@5: totalLongOptions[n].val = gDefaultLongOptions[n].val; andrewm@5: n++; andrewm@5: } andrewm@5: andrewm@5: // Copy custom options into the array, if present andrewm@5: if(customLongOptions == 0) { andrewm@5: // Terminate the array andrewm@5: totalLongOptions[n].name = NULL; andrewm@5: totalLongOptions[n].has_arg = 0; andrewm@5: totalLongOptions[n].flag = NULL; andrewm@5: totalLongOptions[n].val = 0; andrewm@5: } andrewm@5: else { andrewm@5: int customIndex = 0; andrewm@5: while(n < 256) { andrewm@5: if(customLongOptions[customIndex].name == NULL) andrewm@5: break; andrewm@5: totalLongOptions[n].name = customLongOptions[customIndex].name; andrewm@5: totalLongOptions[n].has_arg = customLongOptions[customIndex].has_arg; andrewm@5: totalLongOptions[n].flag = customLongOptions[customIndex].flag; andrewm@5: totalLongOptions[n].val = customLongOptions[customIndex].val; andrewm@5: n++; andrewm@5: customIndex++; andrewm@5: } andrewm@5: andrewm@5: // Terminate the array andrewm@5: totalLongOptions[n].name = NULL; andrewm@5: totalLongOptions[n].has_arg = 0; andrewm@5: totalLongOptions[n].flag = NULL; andrewm@5: totalLongOptions[n].val = 0; andrewm@5: } andrewm@5: } andrewm@5: andrewm@5: while(1) { andrewm@5: if ((c = getopt_long(argc, argv, totalShortOptions, totalLongOptions, NULL)) < 0) andrewm@5: return c; andrewm@5: andrewm@5: switch (c) { andrewm@5: case 'p': andrewm@5: settings->periodSize = atoi(optarg); andrewm@5: if(settings->periodSize < 1) andrewm@5: settings->periodSize = 1; andrewm@5: break; andrewm@5: case 'v': andrewm@5: settings->verbose = 1; andrewm@5: break; andrewm@5: case 'm': andrewm@5: settings->useMatrix = atoi(optarg); andrewm@5: break; andrewm@5: case 'M': andrewm@5: settings->beginMuted = atoi(optarg); andrewm@5: break; andrewm@5: case 'D': andrewm@5: settings->dacLevel = atof(optarg); andrewm@5: break; andrewm@5: case 'A': andrewm@5: settings->adcLevel = atof(optarg); andrewm@5: break; andrewm@5: case 'H': andrewm@5: settings->headphoneLevel = atof(optarg); andrewm@5: break; andrewm@5: case '?': andrewm@5: default: andrewm@5: return c; andrewm@5: } andrewm@5: } andrewm@5: } andrewm@5: andrewm@5: // This function prints standard usage information for default arguments andrewm@5: // Call from within your own usage function andrewm@5: void BeagleRT_usage() andrewm@5: { andrewm@5: std::cerr << " --period [-p] period: Set the hardware period (buffer) size in matrix samples\n"; andrewm@5: std::cerr << " --dac-level [-D] dBs: Set the DAC output level (0dB max; -63.5dB min)\n"; andrewm@5: std::cerr << " --adc-level [-A] dBs: Set the ADC input level (0dB max; -12dB min)\n"; andrewm@5: std::cerr << " --hp-level [-H] dBs: Set the headphone output level (0dB max; -63.5dB min)\n"; andrewm@5: std::cerr << " --mute-speaker [-M] val: Set whether to mute the speaker initially (default: no)\n"; andrewm@5: std::cerr << " --use-matrix [-m] val: Set whether to use ADC/DAC matrix\n"; andrewm@5: std::cerr << " --verbose [-v]: Enable verbose logging information\n"; andrewm@5: }