Mercurial > hg > beaglert
comparison projects/basic_FFT_phase_vocoder/main.cpp @ 250:aec268b5d1b4
Added phase vocoder example (optionally Midi controllable)
author | Giulio Moro <giuliomoro@yahoo.it> |
---|---|
date | Thu, 28 Apr 2016 19:30:12 +0100 |
parents | |
children |
comparison
equal
deleted
inserted
replaced
249:f2d47df23c68 | 250:aec268b5d1b4 |
---|---|
1 /* | |
2 * main.cpp | |
3 * | |
4 * Created on: Oct 24, 2014 | |
5 * Author: parallels | |
6 */ | |
7 | |
8 #include <iostream> | |
9 #include <cstdlib> | |
10 #include <cstdio> | |
11 #include <libgen.h> | |
12 #include <signal.h> | |
13 #include <getopt.h> | |
14 #include <unistd.h> | |
15 #include <sys/time.h> | |
16 #include <sndfile.h> // to load audio files | |
17 #include "SampleData.h" | |
18 #include <BeagleRT.h> | |
19 | |
20 using namespace std; | |
21 | |
22 // Global variables used by getCurrentTime() | |
23 unsigned long long gFirstSeconds, gFirstMicroseconds; | |
24 | |
25 // Load samples from file | |
26 int initFile(string file, SampleData *smp)//float *& smp) | |
27 { | |
28 SNDFILE *sndfile ; | |
29 SF_INFO sfinfo ; | |
30 | |
31 if (!(sndfile = sf_open (file.c_str(), SFM_READ, &sfinfo))) { | |
32 cout << "Couldn't open file " << file << endl; | |
33 return 1; | |
34 } | |
35 | |
36 int numChan = sfinfo.channels; | |
37 if(numChan != 1) | |
38 { | |
39 cout << "Error: " << file << " is not a mono file" << endl; | |
40 return 1; | |
41 } | |
42 | |
43 smp->sampleLen = sfinfo.frames * numChan; | |
44 smp->samples = new float[smp->sampleLen]; | |
45 if(smp == NULL){ | |
46 cout << "Could not allocate buffer" << endl; | |
47 return 1; | |
48 } | |
49 | |
50 int subformat = sfinfo.format & SF_FORMAT_SUBMASK; | |
51 int readcount = sf_read_float(sndfile, smp->samples, smp->sampleLen); | |
52 | |
53 // Pad with zeros in case we couldn't read whole file | |
54 for(int k = readcount; k <smp->sampleLen; k++) | |
55 smp->samples[k] = 0; | |
56 | |
57 if (subformat == SF_FORMAT_FLOAT || subformat == SF_FORMAT_DOUBLE) { | |
58 double scale ; | |
59 int m ; | |
60 | |
61 sf_command (sndfile, SFC_CALC_SIGNAL_MAX, &scale, sizeof (scale)) ; | |
62 if (scale < 1e-10) | |
63 scale = 1.0 ; | |
64 else | |
65 scale = 32700.0 / scale ; | |
66 cout << "File samples scale = " << scale << endl; | |
67 | |
68 for (m = 0; m < smp->sampleLen; m++) | |
69 smp->samples[m] *= scale; | |
70 } | |
71 | |
72 sf_close(sndfile); | |
73 | |
74 return 0; | |
75 } | |
76 | |
77 | |
78 // Handle Ctrl-C by requesting that the audio rendering stop | |
79 void interrupt_handler(int var) | |
80 { | |
81 gShouldStop = true; | |
82 } | |
83 | |
84 // Print usage information | |
85 void usage(const char * processName) | |
86 { | |
87 cerr << "Usage: " << processName << " [options]" << endl; | |
88 | |
89 BeagleRT_usage(); | |
90 | |
91 cerr << " --help [-h]: Print this menu\n"; | |
92 } | |
93 | |
94 /* Function which returns the time since start of the program | |
95 * in (fractional) seconds. | |
96 */ | |
97 double getCurrentTime(void) { | |
98 unsigned long long result; | |
99 struct timeval tv; | |
100 | |
101 gettimeofday(&tv, NULL); | |
102 result = (tv.tv_sec - gFirstSeconds) * 1000000ULL + (tv.tv_usec - gFirstMicroseconds); | |
103 return (double)result / 1000000.0; | |
104 } | |
105 extern SampleData gSampleData; | |
106 int main(int argc, char *argv[]) | |
107 { | |
108 BeagleRTInitSettings settings; // Standard audio settings | |
109 struct timeval tv; | |
110 string fileName; // Name of the sample to load | |
111 | |
112 struct option customOptions[] = | |
113 { | |
114 {"help", 0, NULL, 'h'}, | |
115 {"file", 1, NULL, 'f'}, | |
116 {NULL, 0, NULL, 0} | |
117 }; | |
118 | |
119 gSampleData.samples = 0; | |
120 gSampleData.sampleLen = -1; | |
121 | |
122 // Set default settings | |
123 BeagleRT_defaultSettings(&settings); | |
124 | |
125 settings.periodSize = 32; // Larger period size by default, for testing | |
126 | |
127 // Parse command-line arguments | |
128 while (1) { | |
129 int c; | |
130 if ((c = BeagleRT_getopt_long(argc, argv, "hf:", customOptions, &settings)) < 0) | |
131 break; | |
132 switch (c) { | |
133 case 'h': | |
134 usage(basename(argv[0])); | |
135 exit(0); | |
136 case 'f': | |
137 fileName = string((char *)optarg); | |
138 break; | |
139 case '?': | |
140 default: | |
141 usage(basename(argv[0])); | |
142 exit(1); | |
143 } | |
144 } | |
145 | |
146 if(fileName.empty()){ | |
147 fileName = "sample.wav"; | |
148 } | |
149 | |
150 | |
151 // Load file | |
152 if(initFile(fileName, &gSampleData) != 0) | |
153 { | |
154 cout << "Error: unable to load samples " << endl; | |
155 return -1; | |
156 } | |
157 | |
158 if(settings.verbose) | |
159 cout << "File contains " << gSampleData.sampleLen << " samples" << endl; | |
160 | |
161 | |
162 // Initialise the PRU audio device | |
163 if(BeagleRT_initAudio(&settings, &gSampleData) != 0) { | |
164 cout << "Error: unable to initialise audio" << endl; | |
165 return -1; | |
166 } | |
167 | |
168 // Initialise time | |
169 gettimeofday(&tv, NULL); | |
170 gFirstSeconds = tv.tv_sec; | |
171 gFirstMicroseconds = tv.tv_usec; | |
172 | |
173 // Start the audio device running | |
174 if(BeagleRT_startAudio()) { | |
175 cout << "Error: unable to start real-time audio" << endl; | |
176 return -1; | |
177 } | |
178 | |
179 // Set up interrupt handler to catch Control-C | |
180 signal(SIGINT, interrupt_handler); | |
181 signal(SIGTERM, interrupt_handler); | |
182 | |
183 // Run until told to stop | |
184 while(!gShouldStop) { | |
185 usleep(100000); | |
186 } | |
187 | |
188 // Stop the audio device | |
189 BeagleRT_stopAudio(); | |
190 | |
191 // Clean up any resources allocated for audio | |
192 BeagleRT_cleanupAudio(); | |
193 | |
194 // All done! | |
195 return 0; | |
196 } |