2012-11-15 04:01:05 +00:00
# ifndef cmProc4_h
# define cmProc4_h
# ifdef __cplusplus
extern " C " {
# endif
typedef struct
{
unsigned smpIdx ; // time tag sample index for val
cmMidiByte_t val ; //
2012-12-13 16:46:48 +00:00
bool validFl ; //
2012-11-19 06:41:27 +00:00
} cmScFolBufEle_t ;
2012-11-15 04:01:05 +00:00
typedef struct
{
2012-12-13 16:46:48 +00:00
unsigned pitch ;
unsigned scEvtIdx ;
} cmScFolEvt_t ;
typedef struct
{
unsigned evtCnt ; //
cmScFolEvt_t * evtV ; // pitchV[pitchCnt]
unsigned scIdx ; // index of the score loc (into cmScoreEvt[]) at this location
int barNumb ; // bar number of this location
2012-11-15 04:01:05 +00:00
} cmScFolLoc_t ;
2012-11-19 06:41:27 +00:00
typedef struct
{
cmObj obj ;
2012-12-04 01:18:50 +00:00
cmReal_t srate ; //
cmScH_t scH ; // score handle
unsigned bufN ; // event buffer count
cmScFolBufEle_t * bufV ; // event buffer bufV[bufN] - bufV[bufN-1] newest event, bufV[boi] oldest event
int locN ; // count of score locations
cmScFolLoc_t * loc ; // score loc[locN]
unsigned sbi ; // oldest score window index
unsigned sei ; // newest score window index
unsigned msln ; // minimum score look ahead count
unsigned mswn ; // maximum score window length
unsigned forwardCnt ; // count of score loc's to look ahead for a match to the current pitch when the optimal edit-dist alignment does not produce a match for the current pitch
unsigned maxDist ; // max. dist allowed to still consider matching
unsigned minVel ; // notes < minVel are ignored
bool printFl ; // true if pitch tracker reporting should be included
bool noBackFl ; // prevent the tracker from going backwards in time
2012-11-19 06:41:27 +00:00
unsigned * edWndMtx ;
2012-12-04 01:18:50 +00:00
unsigned missCnt ; // current consecutive unmatched notes
unsigned matchCnt ; // current consecutive matched notes
unsigned eventIdx ; // events since reset
unsigned skipCnt ; // notes skipped due to velocity
unsigned ret_idx ; // last tracked location
2012-11-19 06:41:27 +00:00
2012-11-15 04:01:05 +00:00
} cmScFol ;
2012-12-04 01:18:50 +00:00
cmScFol * cmScFolAlloc ( cmCtx * ctx , cmScFol * p , cmReal_t srate , cmScH_t scH , unsigned bufN , unsigned minWndLookAhead , unsigned maxWndCnt , unsigned minVel ) ;
2012-11-15 04:01:05 +00:00
cmRC_t cmScFolFree ( cmScFol * * pp ) ;
2012-12-04 01:18:50 +00:00
cmRC_t cmScFolInit ( cmScFol * p , cmReal_t srate , cmScH_t scH , unsigned bufN , unsigned minWndLookAhead , unsigned maxWndCnt , unsigned minVel ) ;
2012-11-15 04:01:05 +00:00
cmRC_t cmScFolFinal ( cmScFol * p ) ;
2012-12-04 01:18:50 +00:00
// Jump to a score location and reset the internal state of the follower.
2012-11-15 04:01:05 +00:00
cmRC_t cmScFolReset ( cmScFol * p , unsigned scoreIndex ) ;
2012-12-04 01:18:50 +00:00
// Give the follower a MIDI performance event. Only MIDI note-on events are acted upon;
// all others are ignored.
2012-11-15 04:01:05 +00:00
unsigned cmScFolExec ( cmScFol * p , unsigned smpIdx , unsigned status , cmMidiByte_t d0 , cmMidiByte_t d1 ) ;
2012-12-13 16:46:48 +00:00
//=======================================================================================================================
typedef struct
{
unsigned pitch ;
unsigned scEvtIdx ;
bool matchFl ;
} cmScTrkEvt_t ;
typedef struct
{
unsigned evtCnt ; //
cmScTrkEvt_t * evtV ; // evtV[evtCnt]
unsigned scIdx ; // index of the score event (into cmScoreEvt[]) at this location
int barNumb ; // bar number of this location
} cmScTrkLoc_t ;
typedef struct
{
cmObj obj ;
cmScFol * sfp ;
double srate ;
cmScH_t scH ;
unsigned locN ;
cmScTrkLoc_t * loc ;
unsigned minVel ;
unsigned maxWndCnt ;
unsigned minWndLookAhead ;
bool printFl ;
int curLocIdx ;
unsigned evtIndex ;
} cmScTrk ;
cmScTrk * cmScTrkAlloc ( cmCtx * ctx , cmScTrk * p , cmReal_t srate , cmScH_t scH , unsigned bufN , unsigned minWndLookAhead , unsigned maxWndCnt , unsigned minVel ) ;
cmRC_t cmScTrkFree ( cmScTrk * * pp ) ;
cmRC_t cmScTrkInit ( cmScTrk * p , cmReal_t srate , cmScH_t scH , unsigned bufN , unsigned minWndLookAhead , unsigned maxWndCnt , unsigned minVel ) ;
cmRC_t cmScTrkFinal ( cmScTrk * p ) ;
// Jump to a score location and reset the internal state of the follower.
cmRC_t cmScTrkReset ( cmScTrk * p , unsigned scoreIndex ) ;
// Give the follower a MIDI performance event. Only MIDI note-on events are acted upon;
// all others are ignored.
unsigned cmScTrkExec ( cmScTrk * p , unsigned smpIdx , unsigned status , cmMidiByte_t d0 , cmMidiByte_t d1 ) ;
2013-01-14 00:47:50 +00:00
//=======================================================================================================================
//
// Simplified string alignment function based on Levenshtein edit distance.
//
enum { kEdMinIdx , kEdSubIdx , kEdDelIdx , kEdInsIdx , kEdCnt } ;
typedef struct
{
unsigned v [ kEdCnt ] ;
bool matchFl ;
bool transFl ;
} ed_val ;
typedef struct ed_path_str
{
2013-01-30 19:56:56 +00:00
unsigned code ;
unsigned ri ;
unsigned ci ;
bool matchFl ;
bool transFl ;
2013-01-14 00:47:50 +00:00
struct ed_path_str * next ;
} ed_path ;
/*
Backtracking :
m [ rn , cn ] is organized to indicate the mutation operations
on s0 [ 0 : rn - 1 ] or s1 [ 0 : cn - 1 ] during backtracking .
Backtracking begins at cell m [ rn - 1 , cn - 1 ] and proceeds
up and left toward m [ 0 , 0 ] . The action to perform during
backtracking is determined by examinging which values
int m [ ] . v [ 1 : 3 ] match m [ ] . v [ 0 ] .
Match Next Cell
Index Operation Location
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
1 Substitute char s0 [ ri - 1 ] move diagonally ; up - left
2 Delete char s0 [ ri - 1 ] move up .
3 Delete char s1 [ ci - 1 ] move left .
( same as inserting blank
into after s [ ri - 1 ]
Note that more than one value in m [ ] . v [ 1 : 3 ] may match
m [ ] . v [ 0 ] . In this case the candidate solution branches
at this point in the candidate selection processes .
*/
typedef struct
{
const char * s0 ; // forms rows of m[] - mutate to match s1 - rn=strlen(s0)
const char * s1 ; // forms columns of m[] - target string - cn=strlen(s1)
unsigned rn ; // length of s0 + 1
unsigned cn ; // length of s1 + 1
ed_val * m ; // m[rn,cn]
unsigned pn ; // rn+cn
ed_path * p_mem ; // pmem[ 2*pn ];
ed_path * p_avl ; // available path record linked list
ed_path * p_cur ; // current path linked list
ed_path * p_opt ; // p_opt[pn] current best alignment
double s_opt ; // score of the current best alignment
} ed_r ;
// print the DP matrix ed_r.m[rn,cn].
void ed_print_mtx ( ed_r * r ) ;
// Initialize ed_r.
void ed_init ( ed_r * r , const char * s0 , const char * s1 ) ;
// Fill in the DP matrix.
void ed_calc_mtx ( ed_r * r ) ;
// Traverse the possible alignments in the DP matrix and determine the optimal alignment.
void ed_align ( ed_r * r ) ;
// Print the optimal alignment p_opt[]
void ed_print_opt ( ed_r * r ) ;
// Free resource allocated by ed_init().
void ed_free ( ed_r * r ) ;
// Main test function.
void ed_main ( ) ;
2013-01-16 01:38:21 +00:00
2013-01-14 00:47:50 +00:00
//=======================================================================================================================
2013-01-16 01:38:21 +00:00
enum
{
2013-01-18 17:28:33 +00:00
kSmMinIdx , //
2013-01-16 01:38:21 +00:00
kSmSubIdx , // 'substitute' - may or may not match
kSmDelIdx , // 'delete' - delete a MIDI note
kSmInsIdx , // 'insert' - insert a space in the score
kSmCnt
} ;
enum
{
2013-01-16 23:12:54 +00:00
kSmMatchFl = 0x01 ,
kSmTransFl = 0x02 ,
kSmTruePosFl = 0x04 ,
kSmFalsePosFl = 0x08 ,
kSmBarFl = 0x10 ,
kSmNoteFl = 0x20
2013-01-16 01:38:21 +00:00
} ;
// Dynamic Programming (DP) matrix element
typedef struct
{
2013-01-18 17:28:33 +00:00
unsigned v [ kSmCnt ] ; // cost for each operation
unsigned flags ; // cmSmMatchFl | cmSmTransFl
2013-01-30 19:56:56 +00:00
unsigned scEvtIdx ;
2013-01-16 01:38:21 +00:00
} cmScMatchVal_t ;
// List record used to track a path through the DP matrix p->m[,]
typedef struct cmScMatchPath_str
{
2013-01-30 19:56:56 +00:00
unsigned code ; // kSmXXXIdx
unsigned ri ; // matrix row index
unsigned ci ; // matrix col index
unsigned flags ; // cmSmMatchFl | cmSmTransFl
unsigned locIdx ; // p->loc index or cmInvalidIdx
unsigned scEvtIdx ; // scScore event index
struct cmScMatchPath_str * next ; //
2013-01-16 01:38:21 +00:00
} cmScMatchPath_t ;
typedef struct cmScMatchEvt_str
{
2013-01-18 17:28:33 +00:00
unsigned pitch ; //
unsigned scEvtIdx ; // scScore event index
2013-01-16 01:38:21 +00:00
} cmScMatchEvt_t ;
// Score location record.
typedef struct
{
unsigned evtCnt ; //
cmScMatchEvt_t * evtV ; // evtV[evtCnt]
unsigned scLocIdx ; // scH score location index
int barNumb ; // bar number of this location
} cmScMatchLoc_t ;
2013-01-30 19:56:56 +00:00
typedef struct
{
unsigned mni ; // unique identifier for this MIDI note - used to recognize when the cmScMatcher backtracks.
unsigned smpIdx ; // time stamp of this event
unsigned pitch ; // MIDI note pitch
unsigned vel ; // " " velocity
unsigned locIdx ; // location assoc'd with this MIDI evt (cmInvalidIdx if not a matching or non-matching 'substitute')
unsigned scEvtIdx ; // cmScore event index assoc'd with this event
} cmScMatchMidi_t ;
2013-01-16 01:38:21 +00:00
typedef struct
{
2013-01-18 17:28:33 +00:00
cmObj obj ; //
cmScH_t scH ; // cmScore handle
2013-01-16 01:38:21 +00:00
unsigned locN ; //
cmScMatchLoc_t * loc ; // loc[locN]
2013-01-30 19:56:56 +00:00
unsigned mrn ; // max m[] row count (midi)
unsigned rn ; // cur m[] row count
unsigned mcn ; // max m[] column count (score)
unsigned cn ; // cur m[] column count
2013-01-16 01:38:21 +00:00
unsigned mmn ; // max length of midiBuf[] (mrn-1)
unsigned msn ; // max length of score window (mcn-1)
cmScMatchVal_t * m ; // m[mrn,mcn] DP matrix
unsigned pn ; // mrn+mcn
2013-01-30 19:56:56 +00:00
cmScMatchPath_t * p_mem ; // pmem[ 2*pn ] - path memory
2013-01-16 01:38:21 +00:00
cmScMatchPath_t * p_avl ; // available path record linked list
cmScMatchPath_t * p_cur ; // current path linked list
2013-01-30 19:56:56 +00:00
cmScMatchPath_t * p_opt ; // p_opt[pn] - current best alignment as a linked list
double opt_cost ; // last p_opt cost set by cmScMatchExec()
2013-01-16 01:38:21 +00:00
} cmScMatch ;
2013-01-30 19:56:56 +00:00
/*
1 ) This matcher cannot handle multiple instances of the same pitch occuring
at the same ' location ' .
2 ) Because each note of a chord is spread out over multiple locations , and
there is no way to indicate that a note in the chord is already ' in - use ' .
If a MIDI note which is part of the chord is repeated , in error , it will
apear to be correct ( a positive match will be assigned to
the second ( and possible successive notes ) ) .
*/
2013-01-16 01:38:21 +00:00
cmScMatch * cmScMatchAlloc ( cmCtx * c , cmScMatch * p , cmScH_t scH , unsigned maxScWndN , unsigned maxMidiWndN ) ;
cmRC_t cmScMatchFree ( cmScMatch * * pp ) ;
cmRC_t cmScMatchInit ( cmScMatch * p , cmScH_t scH , unsigned maxScWndN , unsigned maxMidiWndN ) ;
cmRC_t cmScMatchFinal ( cmScMatch * p ) ;
2013-01-30 19:56:56 +00:00
// Locate the position in p->loc[locIdx:locIdx+locN-1] which bests
// matches midiV[midiN].
// The result of this function is to update p_opt[]
// The optimal path p_opt[] will only be updated if the edit_cost associated 'midiV[midiN]'.
// with the best match is less than 'min_cost'.
// Set 'min_cost' to DBL_MAX to force p_opt[] to be updated.
// Returns cmEofRC if locIdx + locN > p->locN - note that this is not
// necessarily an error.
cmRC_t cmScMatchExec ( cmScMatch * p , unsigned locIdx , unsigned locN , const cmScMatchMidi_t * midiV , unsigned midiN , double min_cost ) ;
2013-01-16 01:38:21 +00:00
//=======================================================================================================================
typedef struct
{
unsigned locIdx ;
2013-01-30 19:56:56 +00:00
unsigned scEvtIdx ;
2013-01-16 01:38:21 +00:00
unsigned mni ;
2013-01-18 17:28:33 +00:00
unsigned smpIdx ;
2013-01-16 01:38:21 +00:00
unsigned pitch ;
unsigned vel ;
2013-01-16 23:12:54 +00:00
unsigned flags ;
2013-01-16 01:38:21 +00:00
} cmScMatcherResult_t ;
2013-01-18 17:28:33 +00:00
struct cmScMatcher_str ;
typedef void ( * cmScMatcherCb_t ) ( struct cmScMatcher_str * p , void * arg , cmScMatcherResult_t * rp ) ;
typedef struct cmScMatcher_str
2013-01-16 01:38:21 +00:00
{
2013-01-18 17:28:33 +00:00
cmObj obj ;
cmScMatcherCb_t cbFunc ;
void * cbArg ;
cmScMatch * mp ;
unsigned mn ;
2013-01-30 19:56:56 +00:00
cmScMatchMidi_t * midiBuf ; // midiBuf[mn]
2013-01-16 01:38:21 +00:00
2013-01-31 01:33:44 +00:00
cmScMatcherResult_t * res ; // res[rn]
unsigned rn ; // length of res[]
unsigned ri ; // next avail res[] recd.
2013-01-16 01:38:21 +00:00
2013-01-18 17:28:33 +00:00
double s_opt ; //
2013-03-02 01:40:04 +00:00
unsigned missCnt ; // current count of consecutive trailing non-matches
2013-01-31 01:33:44 +00:00
unsigned ili ; // index into loc[] to start scan following reset
unsigned eli ; // index into loc[] of the last positive match.
2013-03-02 01:40:04 +00:00
unsigned mni ; // current count of MIDI events since the last call to cmScMatcherReset()
unsigned mbi ; // index of oldest MIDI event in midiBuf[]; stays at 0 when the buffer is full.
2013-01-18 17:28:33 +00:00
unsigned begSyncLocIdx ; // start of score window, in mp->loc[], of best match in previous scan
2013-03-02 01:40:04 +00:00
unsigned initHopCnt ; // max window hops during the initial (when the MIDI buffer fills for first time) sync scan
2013-01-18 17:28:33 +00:00
unsigned stepCnt ; // count of forward/backward score loc's to examine for a match during cmScMatcherStep().
unsigned maxMissCnt ; // max. number of consecutive non-matches during step prior to executing a scan.
2013-03-02 01:40:04 +00:00
unsigned scanCnt ; // current count of times a resync-scan was executed during cmScMatcherStep()
2013-01-31 01:33:44 +00:00
bool printFl ;
2013-01-16 01:38:21 +00:00
} cmScMatcher ;
2013-01-18 17:28:33 +00:00
2013-01-31 01:33:44 +00:00
cmScMatcher * cmScMatcherAlloc ( cmCtx * c , cmScMatcher * p , double srate , cmScH_t scH , unsigned scWndN , unsigned midiWndN , cmScMatcherCb_t cbFunc , void * cbArg ) ;
2013-01-16 01:38:21 +00:00
cmRC_t cmScMatcherFree ( cmScMatcher * * pp ) ;
2013-01-31 01:33:44 +00:00
cmRC_t cmScMatcherInit ( cmScMatcher * p , double srate , cmScH_t scH , unsigned scWndN , unsigned midiWndN , cmScMatcherCb_t cbFunc , void * cbArg ) ;
2013-01-16 01:38:21 +00:00
cmRC_t cmScMatcherFinal ( cmScMatcher * p ) ;
2013-01-31 01:33:44 +00:00
// 'scLocIdx' is a score index as used by cmScoreLoc(scH) not into p->mp->loc[].
cmRC_t cmScMatcherReset ( cmScMatcher * p , unsigned scLocIdx ) ;
2013-03-02 01:40:04 +00:00
// Slide a score window hopCnt times, beginning at 'bli' (an
2013-01-31 01:33:44 +00:00
// index int p->mp->loc[]) looking for the best match to p->midiBuf[].
// The score window contain scWndN (p->mp->mcn-1) score locations.
2013-01-16 01:38:21 +00:00
// Returns the index into p->mp->loc[] of the start of the best
// match score window. The score associated
// with this match is stored in s_opt.
2013-03-02 01:40:04 +00:00
unsigned cmScMatcherScan ( cmScMatcher * p , unsigned bli , unsigned hopCnt ) ;
2013-01-16 01:38:21 +00:00
2013-01-31 01:33:44 +00:00
// Step forward/back by p->stepCnt from p->eli.
// p->eli must therefore be valid prior to calling this function.
2013-01-16 01:38:21 +00:00
// If more than p->maxMissCnt consecutive MIDI events are
// missed then automatically run cmScAlignScan().
// Return cmEofRC if the end of the score is encountered.
// Return cmSubSysFailRC if an internal scan resync. failed.
cmRC_t cmScMatcherStep ( cmScMatcher * p ) ;
2013-01-31 01:33:44 +00:00
// This function calls cmScMatcherScan() and cmScMatcherStep() internally.
// If 'status' is not kNonMidiMdId then the function returns without changing the
// state of the object.
// If the MIDI note passed by the call results in a successful match then
// p->eli will be updated to the location in p->mp->loc[] of the latest
// match, the MIDI note in p->midiBuf[] associated with this match
2013-03-02 01:40:04 +00:00
// will be assigned a valid locIdx and scLocIdx values, and *scLocIdxPtr
2013-01-31 01:33:44 +00:00
// will be set with the matched scLocIdx of the match.
// If this call does not result in a successful match *scLocIdxPtr is set
// to cmInvalidIdx.
// Return:
// cmOkRC - Continue processing MIDI events.
// cmEofRC - The end of the score was encountered.
// cmInvalidArgRC - scan failed or the object was in an invalid state to attempt a match.
// cmSubSysFailRC - a scan resync failed in cmScMatcherStep().
cmRC_t cmScMatcherExec ( cmScMatcher * p , unsigned smpIdx , unsigned status , cmMidiByte_t d0 , cmMidiByte_t d1 , unsigned * scLocIdxPtr ) ;
2013-01-16 01:38:21 +00:00
2013-01-14 00:47:50 +00:00
2013-01-14 23:15:53 +00:00
//=======================================================================================================================
typedef struct
{
2013-01-30 19:56:56 +00:00
cmScoreSet_t * sp ; // ptr to this set in the score
2013-01-18 17:28:33 +00:00
unsigned bsei ; // begin score event index
unsigned esei ; // end score event index
2013-01-30 19:56:56 +00:00
unsigned bsli ; // beg score loc index
2013-01-18 17:28:33 +00:00
unsigned esli ; // end score loc index
2013-01-30 19:56:56 +00:00
unsigned bli ; //
2013-01-18 17:28:33 +00:00
unsigned eli ; //
2013-01-31 01:33:44 +00:00
double value ; // DBL_MAX if the value has not yet been set
2013-03-02 01:40:04 +00:00
double tempo ; //
double match_cost ; // cost of the match to the performance divided by sp->eleCnt
2013-01-31 01:33:44 +00:00
2013-01-14 23:15:53 +00:00
} cmScMeasSet_t ;
typedef struct
{
cmObj obj ;
2013-01-31 01:33:44 +00:00
double srate ; //
2013-01-30 19:56:56 +00:00
cmScMatch * mp ; //
2013-01-31 01:33:44 +00:00
unsigned mii ; // next avail recd in midiBuf[]
2013-01-18 17:28:33 +00:00
unsigned mn ; // length of of midiBuf[]
2013-01-30 19:56:56 +00:00
cmScMatchMidi_t * midiBuf ; // midiBuf[mn]
2013-01-18 17:28:33 +00:00
unsigned sn ; // length of set[]
cmScMeasSet_t * set ; // set[sn]
2013-01-30 19:56:56 +00:00
unsigned dn ; // length of dynRef[]
unsigned * dynRef ; // dynRef[dn]
2013-01-31 01:33:44 +00:00
unsigned nsi ; // next set index
unsigned nsli ; // next score location index
unsigned vsi ; // set[vsi:nsi-1] indicates sets with new values following a call to cmScMeasExec()
2013-02-09 07:48:54 +00:00
unsigned vsli ; // vsli:nsli-1 indicates cmScore loc's to check for section triggers following a call to cmScMeasExec()
2013-01-14 23:15:53 +00:00
} cmScMeas ;
2013-01-31 01:33:44 +00:00
//
2013-01-30 19:56:56 +00:00
// Notes:
2013-01-31 01:33:44 +00:00
//
2013-01-30 19:56:56 +00:00
// 1) midiBuf[] stores all MIDI notes for the duration of the performance
// it is initialized to 2*score_event_count.
2013-01-31 01:33:44 +00:00
//
2013-01-30 19:56:56 +00:00
// 2) dynRef][ is the gives the MIDI velocity range for each dynamics
// category: pppp-fff
//
2013-02-09 07:48:54 +00:00
// 3) See a cmDspKr.c _cmScFolMatcherCb() for an example of how
// cmScMeas.vsi and cmScMeas.vsli are used to act on the results of
// a call to cmMeasExec().
2013-01-30 19:56:56 +00:00
cmScMeas * cmScMeasAlloc ( cmCtx * c , cmScMeas * p , cmScH_t scH , double srate , const unsigned * dynRefArray , unsigned dynRefCnt ) ;
2013-01-14 23:15:53 +00:00
cmRC_t cmScMeasFree ( cmScMeas * * pp ) ;
2013-01-30 19:56:56 +00:00
cmRC_t cmScMeasInit ( cmScMeas * p , cmScH_t scH , double srate , const unsigned * dynRefArray , unsigned dynRefCnt ) ;
2013-01-14 23:15:53 +00:00
cmRC_t cmScMeasFinal ( cmScMeas * p ) ;
2013-01-30 19:56:56 +00:00
2013-01-31 01:33:44 +00:00
// Empty MIDI buffer and set the next set nsi and nsli to zero.
cmRC_t cmScMeasReset ( cmScMeas * p ) ;
2013-01-30 19:56:56 +00:00
// This function is called for each input MIDI note which is assigned a
// score location by cmScMatcher.
// 'mni' is the MIDI event index which uniquely identifies this MIDI event.
// 'locIdx' is the location index into cmScMatcher.mp->loc[] associated with
// this event.
cmRC_t cmScMeasExec ( cmScMeas * p , unsigned mni , unsigned locIdx , unsigned scEvtIdx , unsigned flags , unsigned smpIdx , unsigned pitch , unsigned vel ) ;
2013-01-18 17:28:33 +00:00
2013-01-14 23:15:53 +00:00
2013-01-18 17:28:33 +00:00
//=======================================================================================================================
unsigned cmScAlignScanToTimeLineEvent ( cmScMatcher * p , cmTlH_t tlH , cmTlObj_t * top , unsigned endSmpIdx ) ;
// Given a score, a time-line, and a marker on the time line scan the
// entire score looking for the best match between the first 'midiN'
// notes in each marker region and the score.
void cmScAlignScanMarkers ( cmRpt_t * rpt , cmTlH_t tlH , cmScH_t s cH ) ;
2013-01-14 23:15:53 +00:00
2013-02-09 07:48:54 +00:00
//=======================================================================================================================
enum
{
kInvalidModTId ,
kSetModTId , // set variable to parray[0] at scLocIdx
kLineModTId , // linear ramp variable to parray[0] over parray[1] seconds
kSetLineModTId , // set variable to parray[0] and ramp to parray[1] over parray[2] seconds
} ;
enum
{
2013-03-02 01:40:04 +00:00
kActiveModFl = 0x01 , // this variable is on the 'active' list
kCalcModFl = 0x02 // when this variable is used as a parameter it's value must be calculated rather than used directly.
2013-02-09 07:48:54 +00:00
} ;
struct cmScModEntry_str ;
2013-03-02 01:40:04 +00:00
typedef enum
{
kInvalidModPId ,
kLiteralModPId , // this is a literal value
kSymbolModPId //
} cmScModPId_t ;
typedef struct cmScModParam_str
{
cmScModPId_t pid ; // parameter type: literal or symbol
unsigned symId ; // symbol of external and internal variables
double val ; // value of literals
} cmScModParam_t ;
2013-02-09 07:48:54 +00:00
typedef struct cmScModVar_str
{
unsigned flags ; // see kXXXModFl flags above.
unsigned varSymId ; // variable name
2013-03-02 01:40:04 +00:00
unsigned outVarId ; // output var id
double value ; // current value of this variable
2013-02-09 07:48:54 +00:00
double v0 ; // reserved internal variable
2013-03-02 01:40:04 +00:00
unsigned phase ; // cycle phase since activation
double min ;
double max ;
double rate ; // output rate in milliseconds
2013-02-09 07:48:54 +00:00
struct cmScModEntry_str * entry ; // last entry assoc'd with this value
struct cmScModVar_str * vlink ; // p->vlist link
struct cmScModVar_str * alink ; // p->alist link
} cmScModVar_t ;
2013-03-02 01:40:04 +00:00
// Each entry gives a time tagged location and some parameters
// for an algorthm which is used to set/modulate a value.
2013-02-09 07:48:54 +00:00
typedef struct cmScModEntry_str
{
2013-03-02 01:40:04 +00:00
unsigned scLocIdx ; // entry start time
unsigned typeId ; // variable type
cmScModParam_t beg ; // parameter values
cmScModParam_t end ; //
cmScModParam_t dur ; //
cmScModParam_t min ; // min value for this variable
cmScModParam_t max ; // max value for this variable
cmScModParam_t rate ; // update rate in milliseconds (DBL_MAX to disable)
cmScModVar_t * varPtr ; // target variable
2013-02-09 07:48:54 +00:00
} cmScModEntry_t ;
typedef void ( * cmScModCb_t ) ( void * cbArg , unsigned varSymId , double value ) ;
typedef struct
{
cmObj obj ;
2013-03-02 01:40:04 +00:00
cmChar_t * fn ; // modulator score file
2013-02-09 07:48:54 +00:00
unsigned modSymId ; // modulator name
2013-03-02 01:40:04 +00:00
cmSymTblH_t stH ; // symbol table used by this modulator
2013-02-09 07:48:54 +00:00
cmScModCb_t cbFunc ; // active value callback function
void * cbArg ; // first arg to cbFunc()
unsigned samplesPerCycle ; // interval in samples between calls to cmScModulatorExec()
double srate ; // system sample rate
cmScModEntry_t * earray ; // earray[en] - entry array sorted on ascending cmScModEntry_t.scLocIdx
unsigned en ; // count
cmScModVar_t * vlist ; // variable list
cmScModVar_t * alist ; // active variable list
2013-03-02 01:40:04 +00:00
cmScModVar_t * elist ; // last element on the active list
2013-02-09 07:48:54 +00:00
unsigned nei ; // next entry index
2013-03-02 01:40:04 +00:00
unsigned outVarCnt ; // count of unique vars that are targets of entry recds
2013-02-09 07:48:54 +00:00
} cmScModulator ;
cmScModulator * cmScModulatorAlloc ( cmCtx * c , cmScModulator * p , cmCtx_t * ctx , cmSymTblH_t stH , double srate , unsigned samplesPerCycle , const cmChar_t * fn , const cmChar_t * modLabel , cmScModCb_t cbFunc , void * cbArg ) ;
cmRC_t cmScModulatorFree ( cmScModulator * * pp ) ;
cmRC_t cmScModulatorInit ( cmScModulator * p , cmCtx_t * ctx , cmSymTblH_t stH , double srate , unsigned samplesPerCycle , const cmChar_t * fn , const cmChar_t * modLabel , cmScModCb_t cbFunc , void * cbArg ) ;
cmRC_t cmScModulatorFinal ( cmScModulator * p ) ;
// Return count of variables.
2013-03-02 01:40:04 +00:00
unsigned cmScModulatorOutVarCount ( cmScModulator * p ) ;
2013-02-09 07:48:54 +00:00
// Return a pointer to the variable at vlist[idx].
2013-03-02 01:40:04 +00:00
cmScModVar_t * cmScModulatorOutVar ( cmScModulator * p , unsigned idx ) ;
cmRC_t cmScModulatorSetValue ( cmScModulator * p , unsigned varSymId , double value , double min , double max ) ;
2013-02-09 07:48:54 +00:00
2013-03-02 01:40:04 +00:00
cmRC_t cmScModulatorReset ( cmScModulator * p , cmCtx_t * ctx , unsigned scLocIdx ) ;
2013-02-09 07:48:54 +00:00
cmRC_t cmScModulatorExec ( cmScModulator * p , unsigned scLocIdx ) ;
2013-03-02 01:40:04 +00:00
cmRC_t cmScModulatorDump ( cmScModulator * p ) ;
2013-02-09 07:48:54 +00:00
2012-11-15 04:01:05 +00:00
# ifdef __cplusplus
}
# endif
# endif