annotate projects/filter_IIR/main.cpp @ 60:ad6b2767beaf newapi

Command-line argument support for BeagleRT program in scripts, plus a simple script for halting the BBB
author andrewm
date Wed, 15 Jul 2015 18:53:10 +0100
parents 3c3a1357657d
children
rev   line source
victor@3 1 /*
victor@3 2 * main.cpp
victor@3 3 *
victor@3 4 * Created on: Oct 24, 2014
victor@3 5 * Author: Andrew McPherson and Victor Zappi
victor@3 6 */
victor@3 7
victor@3 8 #include <iostream>
victor@3 9 #include <cstdlib>
victor@3 10 #include <libgen.h>
victor@3 11 #include <signal.h>
victor@3 12 #include <string>
andrewm@5 13 #include <getopt.h>
victor@3 14 #include <sndfile.h> // to load audio files
andrewm@56 15
andrewm@56 16 #include <BeagleRT.h>
victor@3 17 #include "SampleData.h"
victor@3 18
victor@3 19 using namespace std;
victor@3 20
victor@3 21 float gCutFreq = 100;
victor@3 22
victor@3 23 // Load samples from file
victor@3 24 int initFile(string file, SampleData *smp)//float *& smp)
victor@3 25 {
victor@3 26 SNDFILE *sndfile ;
victor@3 27 SF_INFO sfinfo ;
victor@3 28
victor@3 29 if (!(sndfile = sf_open (file.c_str(), SFM_READ, &sfinfo))) {
victor@3 30 cout << "Couldn't open file " << file << endl;
victor@3 31 return 1;
victor@3 32 }
victor@3 33
victor@3 34 int numChan = sfinfo.channels;
victor@3 35 if(numChan != 1)
victor@3 36 {
victor@3 37 cout << "Error: " << file << " is not a mono file" << endl;
victor@3 38 return 1;
victor@3 39 }
victor@3 40
victor@3 41 smp->sampleLen = sfinfo.frames * numChan;
victor@3 42 smp->samples = new float[smp->sampleLen];
victor@3 43 if(smp == NULL){
victor@3 44 cout << "Could not allocate buffer" << endl;
victor@3 45 return 1;
victor@3 46 }
victor@3 47
victor@3 48 int subformat = sfinfo.format & SF_FORMAT_SUBMASK;
victor@3 49 int readcount = sf_read_float(sndfile, smp->samples, smp->sampleLen);
victor@3 50
victor@3 51 // Pad with zeros in case we couldn't read whole file
victor@3 52 for(int k = readcount; k <smp->sampleLen; k++)
victor@3 53 smp->samples[k] = 0;
victor@3 54
victor@3 55 if (subformat == SF_FORMAT_FLOAT || subformat == SF_FORMAT_DOUBLE) {
victor@3 56 double scale ;
victor@3 57 int m ;
victor@3 58
victor@3 59 sf_command (sndfile, SFC_CALC_SIGNAL_MAX, &scale, sizeof (scale)) ;
victor@3 60 if (scale < 1e-10)
victor@3 61 scale = 1.0 ;
victor@3 62 else
victor@3 63 scale = 32700.0 / scale ;
victor@3 64 cout << "File samples scale = " << scale << endl;
victor@3 65
victor@3 66 for (m = 0; m < smp->sampleLen; m++)
victor@3 67 smp->samples[m] *= scale;
victor@3 68 }
victor@3 69
victor@3 70 sf_close(sndfile);
victor@3 71
victor@3 72 return 0;
victor@3 73 }
victor@3 74
victor@3 75 // Handle Ctrl-C by requesting that the audio rendering stop
victor@3 76 void interrupt_handler(int var)
victor@3 77 {
victor@3 78 //rt_task_delete ((RT_TASK *) &gTriggerSamplesTask);
victor@3 79 gShouldStop = true;
victor@3 80 }
victor@3 81
victor@3 82 // Print usage information
victor@3 83 void usage(const char * processName)
victor@3 84 {
andrewm@5 85 cerr << "Usage: " << processName << " [options]" << endl;
andrewm@5 86
andrewm@5 87 BeagleRT_usage();
andrewm@5 88
andrewm@5 89 cerr << " --file [-f] filename: Name of the file to load (default is \"longsample.wav\")\n";
andrewm@5 90 cerr << " --cutfreq [-c] freq: Set the cut off frequency of the filter in Hz\n";
andrewm@5 91 cerr << " --help [-h]: Print this menu\n";
victor@3 92 }
victor@3 93
victor@3 94 int main(int argc, char *argv[])
victor@3 95 {
andrewm@52 96 BeagleRTInitSettings settings; // Standard audio settings
victor@3 97 string fileName; // Name of the sample to load
victor@3 98
victor@3 99 SampleData sampleData; // User define structure to pass data retrieved from file to render function
victor@3 100 sampleData.samples = 0;
victor@3 101 sampleData.sampleLen = -1;
victor@3 102
andrewm@5 103
andrewm@5 104 struct option customOptions[] =
andrewm@5 105 {
andrewm@5 106 {"help", 0, NULL, 'h'},
andrewm@5 107 {"cutfreq", 1, NULL, 'c'},
andrewm@5 108 {"file", 1, NULL, 'f'},
andrewm@5 109 {NULL, 0, NULL, 0}
andrewm@5 110 };
andrewm@5 111
andrewm@5 112 // Set default settings
andrewm@5 113 BeagleRT_defaultSettings(&settings);
andrewm@5 114
victor@3 115 // Parse command-line arguments
victor@3 116 while (1) {
victor@3 117 int c;
andrewm@5 118 if ((c = BeagleRT_getopt_long(argc, argv, "hf:c:", customOptions, &settings)) < 0)
victor@3 119 break;
victor@3 120 switch (c) {
victor@3 121 case 'h':
victor@3 122 usage(basename(argv[0]));
victor@3 123 exit(0);
victor@3 124 case 'f':
victor@3 125 fileName = string((char *)optarg);
victor@3 126 break;
victor@3 127 case 'c':
victor@3 128 gCutFreq = atof(optarg);
victor@3 129 break;
victor@3 130 case '?':
victor@3 131 default:
victor@3 132 usage(basename(argv[0]));
victor@3 133 exit(1);
victor@3 134 }
victor@3 135 }
victor@3 136
victor@3 137 if(fileName.empty()){
victor@9 138 fileName = "filter/longsample.wav";
victor@3 139 }
victor@3 140
andrewm@5 141 if(settings.verbose) {
victor@3 142 cout << "Loading file " << fileName << endl;
victor@3 143 }
victor@3 144
victor@3 145 // Load file
victor@3 146 if(initFile(fileName, &sampleData) != 0)
victor@3 147 {
victor@3 148 cout << "Error: unable to load samples " << endl;
victor@3 149 return -1;
victor@3 150 }
victor@3 151
andrewm@5 152 if(settings.verbose)
victor@3 153 cout << "File contains " << sampleData.sampleLen << " samples" << endl;
victor@3 154
andrewm@5 155
victor@3 156 // Initialise the PRU audio device
andrewm@5 157 if(BeagleRT_initAudio(&settings, &sampleData) != 0) {
victor@3 158 cout << "Error: unable to initialise audio" << endl;
victor@3 159 return -1;
victor@3 160 }
victor@3 161
victor@3 162 // Start the audio device running
andrewm@5 163 if(BeagleRT_startAudio()) {
victor@3 164 cout << "Error: unable to start real-time audio" << endl;
victor@3 165 return -1;
victor@3 166 }
victor@3 167
andrewm@15 168 // Set up interrupt handler to catch Control-C and SIGTERM
victor@3 169 signal(SIGINT, interrupt_handler);
andrewm@15 170 signal(SIGTERM, interrupt_handler);
victor@3 171
victor@3 172 // Run until told to stop
victor@3 173 while(!gShouldStop) {
victor@3 174 usleep(100000);
victor@3 175 }
victor@3 176
victor@3 177 // Stop the audio device
andrewm@5 178 BeagleRT_stopAudio();
victor@3 179
victor@3 180 // Clean up any resources allocated for audio
andrewm@5 181 BeagleRT_cleanupAudio();
victor@3 182
victor@3 183 // All done!
victor@3 184 return 0;
victor@3 185 }
andrewm@5 186