shithub: aubio

Download patch

ref: 2ba3440468478fdaaf5b4fbc11e7103d54f7500e
parent: f162cf9b6087f77398c97b7b16b3079460f06a14
author: Paul Brossier <piem@piem.org>
date: Thu Oct 8 16:24:43 EDT 2009

src/pitch/pitchyin.{c,h}: add proper aubio_pitchyin_t object, clean and update prototypes, make multichannel

--- a/src/pitch/pitchyin.c
+++ b/src/pitch/pitchyin.c
@@ -30,6 +30,45 @@
 #include "mathutils.h"
 #include "pitch/pitchyin.h"
 
+struct _aubio_pitchyin_t {
+  fvec_t * yin;
+  smpl_t tol;
+};
+
+/** compute difference function
+  
+  \param input input signal 
+  \param yinbuf output buffer to store difference function (half shorter than input)
+
+*/
+void aubio_pitchyin_diff(fvec_t * input, fvec_t * yinbuf);
+
+/** in place computation of the YIN cumulative normalised function 
+  
+  \param yinbuf input signal (a square difference function), also used to store function 
+
+*/
+void aubio_pitchyin_getcum(fvec_t * yinbuf);
+
+/** detect pitch in a YIN function
+  
+  \param yinbuf input buffer as computed by aubio_pitchyin_getcum
+
+*/
+uint_t aubio_pitchyin_getpitch(fvec_t *yinbuf);
+
+aubio_pitchyin_t * new_aubio_pitchyin (uint_t bufsize) {
+  aubio_pitchyin_t * o = AUBIO_NEW(aubio_pitchyin_t);
+  o->yin = new_fvec (bufsize/2, 1);
+  o->tol = 0.15;
+  return o;
+}
+
+void del_aubio_pitchyin (aubio_pitchyin_t *o) {
+  del_fvec(o->yin);
+  AUBIO_FREE(o);
+}
+
 /* outputs the difference function */
 void aubio_pitchyin_diff(fvec_t * input, fvec_t * yin){
   uint_t c,j,tau;
@@ -88,28 +127,43 @@
 
 
 /* all the above in one */
-smpl_t aubio_pitchyin_getpitchfast(fvec_t * input, fvec_t * yin, smpl_t tol){
-  uint_t c=0,j,tau = 0;
+void aubio_pitchyin_do(aubio_pitchyin_t *o, fvec_t * input, fvec_t * out){
+  smpl_t tol = o->tol;
+  fvec_t * yin = o->yin;
+  uint_t c , j,tau = 0;
   sint_t period;
   smpl_t tmp = 0., tmp2 = 0.;
-  yin->data[c][0] = 1.;
-  for (tau=1;tau<yin->length;tau++)
-  {
-    yin->data[c][tau] = 0.;
-    for (j=0;j<yin->length;j++)
+  for (c = 0; c < input->channels; c++) {
+    yin->data[c][0] = 1.;
+    for (tau=1;tau<yin->length;tau++)
     {
-      tmp = input->data[c][j] - input->data[c][j+tau];
-      yin->data[c][tau] += SQR(tmp);
+      yin->data[c][tau] = 0.;
+      for (j=0;j<yin->length;j++)
+      {
+        tmp = input->data[c][j] - input->data[c][j+tau];
+        yin->data[c][tau] += SQR(tmp);
+      }
+      tmp2 += yin->data[c][tau];
+      yin->data[c][tau] *= tau/tmp2;
+      period = tau-3;
+      if(tau > 4 && (yin->data[c][period] < tol) && 
+          (yin->data[c][period] < yin->data[c][period+1])) {
+        out->data[c][0] = fvec_quadint(yin,period,1);
+        goto beach;
+      }
     }
-    tmp2 += yin->data[c][tau];
-    yin->data[c][tau] *= tau/tmp2;
-    period = tau-3;
-    if(tau > 4 && (yin->data[c][period] < tol) && 
-        (yin->data[c][period] < yin->data[c][period+1])) {
-      return fvec_quadint(yin,period,1);
-    }
+    out->data[c][0] = fvec_quadint(yin,fvec_min_elem(yin),1);
+beach:
+    continue;
   }
-  return fvec_quadint(yin,fvec_min_elem(yin),1);
   //return 0;
 }
 
+uint_t aubio_pitchyin_set_tolerance (aubio_pitchyin_t *o, smpl_t tol) {
+  o->tol = tol;
+  return 0;
+}
+
+smpl_t aubio_pitchyin_get_tolerance (aubio_pitchyin_t *o) {
+  return o->tol;
+}
--- a/src/pitch/pitchyin.h
+++ b/src/pitch/pitchyin.h
@@ -37,36 +37,48 @@
 extern "C" {
 #endif
 
-/** compute difference function
-  
-  \param input input signal 
-  \param yinbuf output buffer to store difference function (half shorter than input)
+/** pitch detection object */
+typedef struct _aubio_pitchyin_t aubio_pitchyin_t;
 
+/** creation of the pitch detection object
+ 
+  \param bufsize size of the input buffer to analyse 
+ 
 */
-void aubio_pitchyin_diff(fvec_t * input, fvec_t * yinbuf);
+aubio_pitchyin_t * new_aubio_pitchyin (uint_t bufsize);
 
-/** in place computation of the YIN cumulative normalised function 
-  
-  \param yinbuf input signal (a square difference function), also used to store function 
+/** deletion of the pitch detection object
+ 
+  \param p pitch detection object as returned by new_aubio_pitchyin()
+ 
+*/
+void del_aubio_pitchyin (aubio_pitchyin_t * o);
 
+/** execute pitch detection on an input buffer 
+ 
+  \param p pitch detection object as returned by new_aubio_pitchyin()
+  \param input input signal window (length as specified at creation time) 
+  \param tol tolerance parameter for minima selection [default 0.85] 
+ 
 */
-void aubio_pitchyin_getcum(fvec_t * yinbuf);
+void aubio_pitchyin_do (aubio_pitchyin_t * o, fvec_t *in, fvec_t *out);
 
-/** detect pitch in a YIN function
+
+/** set tolerance parameter for YIN algorithm 
   
-  \param yinbuf input buffer as computed by aubio_pitchyin_getcum
+  \param o YIN pitch detection object 
+  \param tol tolerance parameter for minima selection [default 0.15]
 
 */
-uint_t aubio_pitchyin_getpitch(fvec_t *yinbuf);
+uint_t aubio_pitchyin_set_tolerance (aubio_pitchyin_t *o, smpl_t tol);
 
-/** fast implementation of the YIN algorithm 
+/** get tolerance parameter for YIN algorithm 
   
-  \param input input signal 
-  \param yinbuf input buffer used to compute the YIN function
-  \param tol tolerance parameter for minima selection [default 0.15]
+  \param o YIN pitch detection object 
+  \return tolerance parameter for minima selection [default 0.15]
 
 */
-smpl_t aubio_pitchyin_getpitchfast(fvec_t * input, fvec_t *yinbuf, smpl_t tol);
+smpl_t aubio_pitchyin_get_tolerance (aubio_pitchyin_t *o);
 
 #ifdef __cplusplus
 }
--- a/tests/src/test-pitchyin.c
+++ b/tests/src/test-pitchyin.c
@@ -6,13 +6,11 @@
         uint_t channels   = 1;                          /* number of channel */
         fvec_t * in       = new_fvec (win_s, channels); /* input buffer */
         fvec_t * out      = new_fvec (win_s/2, channels); /* input buffer */
+        aubio_pitchyin_t *p = new_aubio_pitchyin (win_s);
         uint_t i = 0;
 
         while (i < 10) {
-          aubio_pitchyin_diff   (in,out);
-          aubio_pitchyin_getcum (out);
-          aubio_pitchyin_getpitch (out);
-          aubio_pitchyin_getpitchfast (in,out,0.2);
+          aubio_pitchyin_do (p, in,out);
           i++;
         };