summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorJohn Glover <j@johnglover.net>2012-10-24 17:31:53 +0200
committerJohn Glover <j@johnglover.net>2012-10-24 17:31:53 +0200
commit1023eb542bb7b43fb49fdfbff59c88e75b81415f (patch)
treedc59bc3f664e9154a72fb2565c9978ccaee2e2ac /src
parent03d7fafff27767d0d5707cb17166963d652cb953 (diff)
downloadsimpl-1023eb542bb7b43fb49fdfbff59c88e75b81415f.tar.gz
simpl-1023eb542bb7b43fb49fdfbff59c88e75b81415f.tar.bz2
simpl-1023eb542bb7b43fb49fdfbff59c88e75b81415f.zip
[mq] Tidy up C++ MQ implementation.
Diffstat (limited to 'src')
-rw-r--r--src/mq/mq.cpp434
-rw-r--r--src/mq/mq.h50
2 files changed, 222 insertions, 262 deletions
diff --git a/src/mq/mq.cpp b/src/mq/mq.cpp
index 26b2674..8e2166d 100644
--- a/src/mq/mq.cpp
+++ b/src/mq/mq.cpp
@@ -1,59 +1,80 @@
#include "mq.h"
-/* -------------------------------------------------------------------------
- * Initialisation and destruction
- */
-
-int init_mq(MQParameters* params)
-{
- /* allocate memory for FFT */
- params->fft_in = (sample*) fftw_malloc(sizeof(sample) * params->frame_size);
- params->fft_out = (fftw_complex*) fftw_malloc(sizeof(fftw_complex) * params->num_bins);
- params->fft_plan = fftw_plan_dft_r2c_1d(params->frame_size, params->fft_in,
+using namespace simpl;
+
+// ----------------------------------------------------------------------------
+// Windowing
+
+void hamming_window(int window_size, sample* window) {
+ sample sum = 0;
+
+ for(int i = 0; i < window_size; i++) {
+ window[i] = 0.54 - (0.46 * cos(2.0 * M_PI * i / (window_size - 1)));
+ sum += window[i];
+ }
+
+ for(int i = 0; i < window_size; i++) {
+ window[i] /= sum;
+ }
+}
+
+// ----------------------------------------------------------------------------
+// Initialisation and destruction
+
+int simpl::init_mq(MQParameters* params) {
+ // allocate memory for window
+ params->window = (sample*) malloc(sizeof(sample) * params->frame_size);
+ int i;
+ for(i = 0; i < params->frame_size; i++) {
+ params->window[i] = 1.0;
+ }
+ hamming_window(params->frame_size, params->window);
+
+ // allocate memory for FFT
+ params->fft_in = (sample*) fftw_malloc(sizeof(sample) *
+ params->frame_size);
+ params->fft_out = (fftw_complex*) fftw_malloc(sizeof(fftw_complex) *
+ params->num_bins);
+ params->fft_plan = fftw_plan_dft_r2c_1d(params->frame_size, params->fft_in,
params->fft_out, FFTW_ESTIMATE);
- /* set other variables to defaults */
+ // set other variables to defaults
reset_mq(params);
return 0;
}
-void reset_mq(MQParameters* params)
-{
+void simpl::reset_mq(MQParameters* params) {
params->prev_peaks = NULL;
}
-int destroy_mq(MQParameters* params)
-{
- if(params)
- {
+int simpl::destroy_mq(MQParameters* params) {
+ if(params) {
+ if(params->window) free(params->window);
if(params->fft_in) free(params->fft_in);
if(params->fft_out) free(params->fft_out);
fftw_destroy_plan(params->fft_plan);
+
+ params->window = NULL;
+ params->fft_in = NULL;
+ params->fft_out = NULL;
}
return 0;
}
-/* -------------------------------------------------------------------------
- * Peak Detection
- */
-
-/* Add new_peak to the doubly linked list of peaks, keeping peaks sorted
- * with the largest amplitude peaks at the start of the list
- */
-void add_peak(Peak* new_peak, PeakList* peak_list)
-{
- do
- {
- if(peak_list->peak)
- {
- if(peak_list->peak->amplitude > new_peak->amplitude)
- {
- if(peak_list->next)
- {
+// ----------------------------------------------------------------------------
+// Peak Detection
+
+// Add new_peak to the doubly linked list of peaks, keeping peaks sorted
+// with the largest amplitude peaks at the start of the list
+void add_peak(MQPeak* new_peak, MQPeakList* peak_list) {
+ do {
+ if(peak_list->peak) {
+ if(peak_list->peak->amplitude > new_peak->amplitude) {
+ if(peak_list->next) {
peak_list = peak_list->next;
}
- else
- {
- PeakList* new_node = (PeakList*)malloc(sizeof(PeakList));
+ else {
+ MQPeakList* new_node =
+ (MQPeakList*)malloc(sizeof(MQPeakList));
new_node->peak = new_peak;
new_node->prev = peak_list;
new_node->next = NULL;
@@ -61,9 +82,8 @@ void add_peak(Peak* new_peak, PeakList* peak_list)
return;
}
}
- else
- {
- PeakList* new_node = (PeakList*)malloc(sizeof(PeakList));
+ else {
+ MQPeakList* new_node = (MQPeakList*)malloc(sizeof(MQPeakList));
new_node->peak = peak_list->peak;
new_node->prev = peak_list;
new_node->next = peak_list->next;
@@ -72,9 +92,8 @@ void add_peak(Peak* new_peak, PeakList* peak_list)
return;
}
}
- else
- {
- /* should only happen for the first peak */
+ else {
+ // should only happen for the first peak
peak_list->peak = new_peak;
return;
}
@@ -82,63 +101,59 @@ void add_peak(Peak* new_peak, PeakList* peak_list)
while(1);
}
-/* delete the given PeakList */
-void delete_peak_list(PeakList* peak_list)
-{
- /* destroy list of peaks */
- while(peak_list && peak_list->next)
- {
- if(peak_list->peak)
- {
+// delete the given PeakList
+void simpl::delete_peak_list(MQPeakList* peak_list) {
+ // destroy list of peaks
+ while(peak_list && peak_list->next) {
+ if(peak_list->peak) {
free(peak_list->peak);
}
- PeakList* temp = peak_list->next;
+ MQPeakList* temp = peak_list->next;
free(peak_list);
peak_list = temp;
}
free(peak_list);
}
-sample get_magnitude(sample x, sample y)
-{
+sample get_magnitude(sample x, sample y) {
return sqrt((x*x) + (y*y));
}
-sample get_phase(sample x, sample y)
-{
+sample get_phase(sample x, sample y) {
return atan2(y, x);
}
-PeakList* find_peaks(int signal_size, sample* signal, MQParameters* params)
-{
+MQPeakList* simpl::mq_find_peaks(int signal_size, sample* signal,
+ MQParameters* params) {
int i;
int num_peaks = 0;
sample prev_amp, current_amp, next_amp;
- PeakList* peak_list = (PeakList*)malloc(sizeof(PeakList));
+ MQPeakList* peak_list = (MQPeakList*)malloc(sizeof(MQPeakList));
peak_list->next = NULL;
peak_list->prev = NULL;
peak_list->peak = NULL;
- /* take fft of the signal */
+ // take fft of the signal
memcpy(params->fft_in, signal, sizeof(sample)*params->frame_size);
- hann_window(params->frame_size, params->fft_in);
+ for(i = 0; i < params->frame_size; i++) {
+ params->fft_in[i] *= params->window[i];
+ }
fftw_execute(params->fft_plan);
- /* get initial magnitudes */
- prev_amp = get_magnitude(params->fft_out[0][0], params->fft_out[0][1]);
- current_amp = get_magnitude(params->fft_out[1][0], params->fft_out[1][1]);
+ // get initial magnitudes
+ prev_amp = get_magnitude(params->fft_out[0][0], params->fft_out[0][1]);
+ current_amp = get_magnitude(params->fft_out[1][0], params->fft_out[1][1]);
- /* find all peaks in the amplitude spectrum */
- for(i = 1; i < params->num_bins-1; i++)
- {
- next_amp = get_magnitude(params->fft_out[i+1][0], params->fft_out[i+1][1]);
+ // find all peaks in the amplitude spectrum
+ for(i = 1; i < params->num_bins - 1; i++) {
+ next_amp = get_magnitude(params->fft_out[i+1][0],
+ params->fft_out[i+1][1]);
- if((current_amp > prev_amp) &&
+ if((current_amp > prev_amp) &&
(current_amp > next_amp) &&
- (current_amp > params->peak_threshold))
- {
- /* create a new Peak */
- Peak* p = (Peak*)malloc(sizeof(Peak));
+ (current_amp > params->peak_threshold)) {
+ // create a new MQPeak
+ MQPeak* p = (MQPeak*)malloc(sizeof(MQPeak));
p->amplitude = current_amp;
p->frequency = i * params->fundamental;
p->phase = get_phase(params->fft_out[i][0], params->fft_out[i][1]);
@@ -146,7 +161,7 @@ PeakList* find_peaks(int signal_size, sample* signal, MQParameters* params)
p->next = NULL;
p->prev = NULL;
- /* add it to the appropriate position in the list of Peaks */
+ // add it to the appropriate position in the list of Peaks
add_peak(p, peak_list);
num_peaks++;
}
@@ -154,13 +169,10 @@ PeakList* find_peaks(int signal_size, sample* signal, MQParameters* params)
current_amp = next_amp;
}
- /* limit peaks to a maximum of max_peaks
- */
- if(num_peaks > params->max_peaks)
- {
- PeakList* current = peak_list;
- for(i = 0; i < params->max_peaks-1; i++)
- {
+ // limit peaks to a maximum of max_peaks
+ if(num_peaks > params->max_peaks) {
+ MQPeakList* current = peak_list;
+ for(i = 0; i < params->max_peaks-1; i++) {
current = current->next;
}
@@ -169,46 +181,36 @@ PeakList* find_peaks(int signal_size, sample* signal, MQParameters* params)
num_peaks = params->max_peaks;
}
- return sort_peaks_by_frequency(peak_list, num_peaks);
+ return simpl::mq_sort_peaks_by_frequency(peak_list, num_peaks);
}
-/* -------------------------------------------------------------------------
- * Sorting
- */
-
-PeakList* merge(PeakList* list1, PeakList* list2)
-{
- PeakList* merged_head = NULL;
- PeakList* merged_tail;
-
- while(list1 || list2)
- {
- if(list1 && list2)
- {
- if(list1->peak->frequency <= list2->peak->frequency)
- {
- if(!merged_head)
- {
+// ----------------------------------------------------------------------------
+// Sorting
+
+MQPeakList* merge(MQPeakList* list1, MQPeakList* list2) {
+ MQPeakList* merged_head = NULL;
+ MQPeakList* merged_tail;
+
+ while(list1 || list2) {
+ if(list1 && list2) {
+ if(list1->peak->frequency <= list2->peak->frequency) {
+ if(!merged_head) {
merged_head = list1;
merged_tail = merged_head;
}
- else
- {
+ else {
merged_tail->next = list1;
merged_tail = merged_tail->next;
}
list1 = list1->next;
merged_tail->next = NULL;
}
- else
- {
- if(!merged_head)
- {
+ else {
+ if(!merged_head) {
merged_head = list2;
merged_tail = merged_head;
}
- else
- {
+ else {
merged_tail->next = list2;
merged_tail = merged_tail->next;
}
@@ -216,30 +218,24 @@ PeakList* merge(PeakList* list1, PeakList* list2)
merged_tail->next = NULL;
}
}
- else if(list1)
- {
- if(!merged_head)
- {
+ else if(list1) {
+ if(!merged_head) {
merged_head = list1;
merged_tail = merged_head;
}
- else
- {
+ else {
merged_tail->next = list1;
merged_tail = merged_tail->next;
}
list1 = list1->next;
merged_tail->next = NULL;
}
- else if(list2)
- {
- if(!merged_head)
- {
+ else if(list2) {
+ if(!merged_head) {
merged_head = list2;
merged_tail = merged_head;
}
- else
- {
+ else {
merged_tail->next = list2;
merged_tail = merged_tail->next;
}
@@ -251,36 +247,30 @@ PeakList* merge(PeakList* list1, PeakList* list2)
return merged_head;
}
-PeakList* merge_sort(PeakList* peak_list, int num_peaks)
-{
- if(num_peaks <= 1)
- {
+MQPeakList* merge_sort(MQPeakList* peak_list, int num_peaks) {
+ if(num_peaks <= 1) {
return peak_list;
}
- PeakList* left;
- PeakList* right;
- PeakList* current = peak_list;
+ MQPeakList* left;
+ MQPeakList* right;
+ MQPeakList* current = peak_list;
int n = 0;
- /* find the index of the middle peak. If we have an odd number,
- * give the extra peak to the left */
+ // find the index of the middle peak. If we have an odd number,
+ // give the extra peak to the left
int middle;
- if(num_peaks % 2 == 0)
- {
+ if(num_peaks % 2 == 0) {
middle = num_peaks/2;
}
- else
- {
+ else {
middle = (num_peaks/2) + 1;
}
- /* split the peak list into left and right at the middle value */
+ // split the peak list into left and right at the middle value
left = peak_list;
- while(current)
- {
- if(n == middle-1)
- {
+ while(current) {
+ if(n == middle-1) {
right = current->next;
current->next = NULL;
break;
@@ -290,56 +280,44 @@ PeakList* merge_sort(PeakList* peak_list, int num_peaks)
current = current->next;
}
- /* recursively sort and merge */
+ // recursively sort and merge
left = merge_sort(left, middle);
right = merge_sort(right, num_peaks-middle);
return merge(left, right);
}
-/*
- * Sort peak_list into a list order from smaller to largest frequency.
- */
-PeakList* sort_peaks_by_frequency(PeakList* peak_list, int num_peaks)
-{
- if(!peak_list || num_peaks == 0)
- {
+// Sort peak_list into a list order from smaller to largest frequency.
+MQPeakList* simpl::mq_sort_peaks_by_frequency(MQPeakList* peak_list,
+ int num_peaks) {
+ if(!peak_list || num_peaks == 0) {
return NULL;
}
- else
- {
+ else {
return merge_sort(peak_list, num_peaks);
}
}
-/* -------------------------------------------------------------------------
- * Partial Tracking
- */
-
-/*
- * Find a candidate match for peak in frame if one exists. This is the closest
- * (in frequency) match that is within the matching interval.
- */
-Peak* find_closest_match(Peak* p, PeakList* peak_list, MQParameters* params, int backwards)
-{
- PeakList* current = peak_list;
- Peak* match = NULL;
- sample best_distance = 44100.0; /* TODO: set this to params->sampling_rate */
+// ----------------------------------------------------------------------------
+// Partial Tracking
+
+// Find a candidate match for peak in frame if one exists. This is the closest
+// (in frequency) match that is within the matching interval.
+MQPeak* find_closest_match(MQPeak* p, MQPeakList* peak_list,
+ MQParameters* params, int backwards) {
+ MQPeakList* current = peak_list;
+ MQPeak* match = NULL;
+ sample best_distance = 44100.0;
sample distance;
- while(current && current->peak)
- {
- if(backwards)
- {
- if(current->peak->prev)
- {
+ while(current && current->peak) {
+ if(backwards) {
+ if(current->peak->prev) {
current = current->next;
continue;
}
}
- else
- {
- if(current->peak->next)
- {
+ else {
+ if(current->peak->next) {
current = current->next;
continue;
}
@@ -347,8 +325,7 @@ Peak* find_closest_match(Peak* p, PeakList* peak_list, MQParameters* params, int
distance = fabs(current->peak->frequency - p->frequency);
if((distance < params->matching_interval) &&
- (distance < best_distance))
- {
+ (distance < best_distance)) {
best_distance = distance;
match = current->peak;
}
@@ -358,25 +335,23 @@ Peak* find_closest_match(Peak* p, PeakList* peak_list, MQParameters* params, int
return match;
}
-/*
- * Returns the closest unmatched peak in frame with a frequency less than p.frequency.
- */
-Peak* free_peak_below(Peak* p, PeakList* peak_list)
-{
- PeakList* current = peak_list;
- Peak* free_peak = NULL;
- sample closest_frequency = 44100; /* TODO: set this to params->sampling_rate */
-
- while(current && current->peak)
- {
- if(current->peak != p)
- {
- /* if current peak is unmatched, and it is closer to p than the last
- * unmatched peak that we saw, save it */
- if(!current->peak->prev && (current->peak->frequency < p->frequency) &&
- (fabs(current->peak->frequency - p->frequency) < closest_frequency))
- {
- closest_frequency = fabs(current->peak->frequency - p->frequency);
+// Returns the closest unmatched peak in frame with a frequency less
+// than p.frequency.
+MQPeak* free_peak_below(MQPeak* p, MQPeakList* peak_list) {
+ MQPeakList* current = peak_list;
+ MQPeak* free_peak = NULL;
+ sample closest_frequency = 44100;
+
+ while(current && current->peak) {
+ if(current->peak != p) {
+ // if current peak is unmatched, and it is closer to p than the
+ // last unmatched peak that we saw, save it
+ if(!current->peak->prev &&
+ (current->peak->frequency < p->frequency) &&
+ (fabs(current->peak->frequency - p->frequency)
+ < closest_frequency)) {
+ closest_frequency = fabs(current->peak->frequency -
+ p->frequency);
free_peak = current->peak;
}
}
@@ -385,44 +360,39 @@ Peak* free_peak_below(Peak* p, PeakList* peak_list)
return free_peak;
}
-/*
- * A simplified version of MQ Partial Tracking.
- */
-PeakList* track_peaks(PeakList* peak_list, MQParameters* params)
-{
- PeakList* current = peak_list;
-
- /* MQ algorithm needs 2 frames of data, so return if this is the first frame */
- if(params->prev_peaks)
- {
- /* find all matches for previous peaks in the current frame */
+
+// MQ Partial Tracking
+MQPeakList* simpl::mq_track_peaks(MQPeakList* peak_list,
+ MQParameters* params) {
+ MQPeakList* current = peak_list;
+
+ // MQ algorithm needs 2 frames of data, so return if this is the
+ // first frame
+ if(params->prev_peaks) {
+ // find all matches for previous peaks in the current frame
current = params->prev_peaks;
- while(current && current->peak)
- {
- Peak* match = find_closest_match(current->peak, peak_list, params, 1);
- /*printf("peak: %f\n", current->peak->frequency);*/
- if(match)
- {
- Peak* closest_to_cand = find_closest_match(match, params->prev_peaks, params, 0);
- if(closest_to_cand != current->peak)
- {
- /*printf("%f closer to %f\n", match->frequency, closest_to_cand->frequency);*/
- /* see if the closest peak with lower frequency to the candidate is within
- * the matching interval
- */
- Peak* lower = free_peak_below(match, peak_list);
- if(lower)
- {
- if(fabs(lower->frequency - current->peak->frequency) < params->matching_interval)
- {
+ while(current && current->peak) {
+ MQPeak* match = find_closest_match(
+ current->peak, peak_list, params, 1
+ );
+ if(match) {
+ MQPeak* closest_to_cand = find_closest_match(
+ match, params->prev_peaks, params, 0
+ );
+ if(closest_to_cand != current->peak) {
+ // see if the closest peak with lower frequency to the
+ // candidate is within the matching interval
+ MQPeak* lower = free_peak_below(match, peak_list);
+ if(lower) {
+ if(fabs(lower->frequency - current->peak->frequency)
+ < params->matching_interval) {
lower->prev = current->peak;
current->peak->next = lower;
}
}
}
- /* if closest_peak == peak, it is a definitive match */
- else
- {
+ // if closest_peak == peak, it is a definitive match
+ else {
match->prev = current->peak;
current->peak->next = match;
}
@@ -434,21 +404,3 @@ PeakList* track_peaks(PeakList* peak_list, MQParameters* params)
params->prev_peaks = peak_list;
return peak_list;
}
-
-/* -------------------------------------------------------------------------
- * Testing
- */
-
-/*
- * Prints the frequency values of all peaks in p.
- */
-void print_list(PeakList* p)
-{
- int i = 0;
- while(p)
- {
- printf("%d: freq=%f\n", i, p->peak->frequency);
- p = p->next;
- i++;
- }
-}
diff --git a/src/mq/mq.h b/src/mq/mq.h
index 611203d..d2f24cc 100644
--- a/src/mq/mq.h
+++ b/src/mq/mq.h
@@ -1,50 +1,58 @@
-#ifndef _MQ_H
-#define _MQ_H
+#ifndef _SIMPL_MQ_H
+#define _SIMPL_MQ_H
+
+#include <cstdlib>
#include <fftw3.h>
#include <math.h>
+#include <string.h>
+
+namespace simpl
+{
+
typedef double sample;
-typedef struct Peak
-{
+typedef struct MQPeak {
float amplitude;
float frequency;
float phase;
int bin;
- struct Peak* next;
- struct Peak* prev;
-} Peak;
+ struct MQPeak* next;
+ struct MQPeak* prev;
+} MQPeak;
-typedef struct PeakList
-{
- struct PeakList* next;
- struct PeakList* prev;
- struct Peak* peak;
-} PeakList;
+typedef struct MQPeakList {
+ struct MQPeakList* next;
+ struct MQPeakList* prev;
+ struct MQPeak* peak;
+} MQPeakList;
-typedef struct MQParameters
-{
+typedef struct MQParameters {
int frame_size;
int max_peaks;
int num_bins;
sample peak_threshold;
sample fundamental;
sample matching_interval;
+ sample* window;
sample* fft_in;
fftw_complex* fft_out;
fftw_plan fft_plan;
- PeakList* prev_peaks;
+ MQPeakList* prev_peaks;
} MQParameters;
int init_mq(MQParameters* params);
void reset_mq(MQParameters* params);
int destroy_mq(MQParameters* params);
-void delete_peak_list(PeakList* peak_list);
+void delete_peak_list(MQPeakList* peak_list);
+
+MQPeakList* mq_sort_peaks_by_frequency(MQPeakList* peak_list, int num_peaks);
+MQPeakList* mq_find_peaks(int signal_size, sample* signal,
+ MQParameters* params);
+MQPeakList* mq_track_peaks(MQPeakList* peak_list, MQParameters* params);
+
+} // end of namespace simpl
-PeakList* sort_peaks_by_frequency(PeakList* peak_list, int num_peaks);
-PeakList* find_peaks(int signal_size, sample* signal, MQParameters* params);
-PeakList* track_peaks(PeakList* peak_list, MQParameters* params);
-
#endif