2021-05-10 12:37:29 +00:00
# include "cwCommon.h"
# include "cwLog.h"
# include "cwCommonImpl.h"
# include "cwMem.h"
# include "cwObject.h"
# include "cwFileSys.h"
# include "cwFile.h"
# include "cwTime.h"
# include "cwMidiDecls.h"
# include "cwMidi.h"
2021-08-15 20:02:45 +00:00
# include "cwMidiFile.h"
2021-05-10 12:37:29 +00:00
# include "cwUiDecls.h"
# include "cwIo.h"
# include "cwIoMidiRecordPlay.h"
2021-12-30 02:37:07 +00:00
# define TIMER_LABEL "midi_record_play_timer"
2021-05-10 12:37:29 +00:00
namespace cw
{
namespace midi_record_play
{
typedef struct am_midi_msg_str
{
unsigned devIdx ;
unsigned portIdx ;
2021-11-01 14:13:48 +00:00
unsigned microsec ;
unsigned id ;
2021-05-10 12:37:29 +00:00
time : : spec_t timestamp ;
2022-05-06 20:03:43 +00:00
unsigned loc ;
2021-05-10 12:37:29 +00:00
uint8_t ch ;
uint8_t status ;
uint8_t d0 ;
uint8_t d1 ;
} am_midi_msg_t ;
2022-01-22 14:42:21 +00:00
typedef struct midi_device_str
{
char * midiOutDevLabel ;
char * midiOutPortLabel ;
unsigned midiOutDevIdx ;
unsigned midiOutPortIdx ;
bool enableFl ;
2022-03-20 14:27:46 +00:00
unsigned velTableN ;
uint8_t * velTableArray ;
bool pedalMapEnableFl ;
2022-05-06 20:03:43 +00:00
2022-03-20 14:27:46 +00:00
unsigned pedalDownVelId ;
unsigned pedalDownVel ;
2022-05-06 20:03:43 +00:00
unsigned pedalDownHalfVelId ;
2022-03-20 14:27:46 +00:00
unsigned pedalDownHalfVel ;
2022-05-06 20:03:43 +00:00
unsigned pedalUpHalfVelId ;
unsigned pedalUpHalfVel ;
2022-10-15 13:26:35 +00:00
unsigned velHistogram [ midi : : kMidiVelCnt ] ;
bool force_damper_down_fl ;
unsigned force_damper_down_threshold ;
unsigned force_damper_down_velocity ;
2022-11-11 19:04:44 +00:00
bool damper_dead_band_enable_fl ;
unsigned damper_dead_band_min_value ;
unsigned damper_dead_band_max_value ;
2022-05-06 20:03:43 +00:00
2022-01-22 14:42:21 +00:00
} midi_device_t ;
2022-03-20 14:27:46 +00:00
enum
{
kHalfPedalDone ,
kWaitForBegin ,
kWaitForNoteOn ,
kWaitForNoteOff ,
kWaitForPedalUp ,
kWaitForPedalDown ,
} ;
2021-05-10 12:37:29 +00:00
typedef struct midi_record_play_str
{
io : : handle_t ioH ;
2022-01-22 14:42:21 +00:00
am_midi_msg_t * msgArray ; // msgArray[ msgArrayN ]
unsigned msgArrayN ; // Count of messages allocated in msgArray.
2022-03-20 14:27:46 +00:00
unsigned msgArrayInIdx ; // Next available space for loaded MIDI messages (also the current count of msgs in msgArray[])
2022-01-22 14:42:21 +00:00
unsigned msgArrayOutIdx ; // Next message to transmit in msgArray[]
unsigned midi_timer_period_micro_sec ; // Timer period in microseconds
2022-05-14 14:22:29 +00:00
unsigned all_off_delay_ms ; // Wait this long before turning all notes off after the last note-on has played
2021-05-10 12:37:29 +00:00
2022-03-20 14:27:46 +00:00
am_midi_msg_t * iMsgArray ; // msgArray[ msgArrayN ]
unsigned iMsgArrayN ; // Count of messages allocated in msgArray.
unsigned iMsgArrayInIdx ; // Next available space for incoming MIDI messages (also the current count of msgs in msgArray[])
2022-01-22 14:42:21 +00:00
midi_device_t * midiDevA ;
unsigned midiDevN ;
2021-05-10 12:37:29 +00:00
bool startedFl ;
bool recordFl ;
2022-11-11 19:04:44 +00:00
bool muteFl ;
2021-05-10 12:37:29 +00:00
bool thruFl ;
2022-03-20 14:27:46 +00:00
bool logInFl ; // log incoming message when not in 'record' mode.
bool logOutFl ; // log outgoing messages
2022-10-15 13:26:35 +00:00
bool velHistogramEnableFl ;
2021-05-10 12:37:29 +00:00
2022-03-20 14:27:46 +00:00
bool halfPedalFl ;
unsigned halfPedalState ;
unsigned halfPedalNextUs ;
unsigned halfPedalNoteDelayUs ;
unsigned halfPedalNoteDurUs ;
unsigned halfPedalUpDelayUs ;
unsigned halfPedalDownDelayUs ;
uint8_t halfPedalMidiPitch ;
uint8_t halfPedalMidiNoteVel ;
uint8_t halfPedalMidiPedalVel ;
2021-05-10 12:37:29 +00:00
time : : spec_t play_time ;
2021-08-15 20:02:45 +00:00
time : : spec_t start_time ;
2021-11-14 16:52:24 +00:00
time : : spec_t end_play_event_timestamp ;
2022-05-14 14:22:29 +00:00
time : : spec_t all_off_timestamp ;
2022-03-20 14:27:46 +00:00
time : : spec_t store_time ;
2021-08-15 20:02:45 +00:00
2021-11-01 14:13:48 +00:00
event_callback_t cb ;
void * cb_arg ;
2021-05-10 12:37:29 +00:00
} midi_record_play_t ;
2022-03-20 14:27:46 +00:00
2021-05-10 12:37:29 +00:00
enum
{
kMidiRecordPlayTimerId
} ;
midi_record_play_t * _handleToPtr ( handle_t h )
{ return handleToPtr < handle_t , midi_record_play_t > ( h ) ; }
rc_t _destroy ( midi_record_play_t * p )
{
rc_t rc = kOkRC ;
2021-12-30 02:37:07 +00:00
unsigned timerIdx ;
if ( ( timerIdx = io : : timerLabelToIndex ( p - > ioH , TIMER_LABEL ) ) ! = kInvalidIdx )
io : : timerDestroy ( p - > ioH , timerIdx ) ;
2022-01-22 14:42:21 +00:00
for ( unsigned i = 0 ; i < p - > midiDevN ; + + i )
{
mem : : release ( p - > midiDevA [ i ] . midiOutDevLabel ) ;
mem : : release ( p - > midiDevA [ i ] . midiOutPortLabel ) ;
2022-03-20 14:27:46 +00:00
mem : : release ( p - > midiDevA [ i ] . velTableArray ) ;
2022-01-22 14:42:21 +00:00
}
mem : : release ( p - > midiDevA ) ;
2021-05-10 12:37:29 +00:00
mem : : release ( p - > msgArray ) ;
2022-03-20 14:27:46 +00:00
mem : : release ( p - > iMsgArray ) ;
2021-05-10 12:37:29 +00:00
mem : : release ( p ) ;
return rc ;
}
rc_t _parseCfg ( midi_record_play_t * p , const object_t & cfg )
{
rc_t rc = kOkRC ;
2022-01-22 14:42:21 +00:00
const object_t * midiDevL = nullptr ;
2021-05-10 12:37:29 +00:00
if ( ( rc = cfg . getv (
" max_midi_msg_count " , p - > msgArrayN ,
" midi_timer_period_micro_sec " , p - > midi_timer_period_micro_sec ,
2022-05-14 14:22:29 +00:00
" all_off_delay_ms " , p - > all_off_delay_ms ,
2022-03-20 14:27:46 +00:00
" midi_device_list " , midiDevL ,
" log_in_flag " , p - > logInFl ,
" log_out_flag " , p - > logOutFl ,
" half_pedal_flag " , p - > halfPedalFl ) ) ! = kOkRC )
2021-05-10 12:37:29 +00:00
{
rc = cwLogError ( kSyntaxErrorRC , " MIDI record play configuration parse failed. " ) ;
goto errLabel ;
}
2022-03-20 14:27:46 +00:00
p - > iMsgArrayN = p - > msgArrayN ;
2022-01-22 14:42:21 +00:00
if ( midiDevL - > child_count ( ) > 0 )
{
p - > midiDevN = midiDevL - > child_count ( ) ;
p - > midiDevA = mem : : allocZ < midi_device_t > ( p - > midiDevN ) ;
printf ( " Midi record play devices:%i \n " , p - > midiDevN ) ;
for ( unsigned i = 0 ; i < p - > midiDevN ; + + i )
{
const object_t * ele = midiDevL - > child_ele ( i ) ;
const char * midiOutDevLabel = nullptr ;
const char * midiOutPortLabel = nullptr ;
2022-03-20 14:27:46 +00:00
const object_t * velTable = nullptr ;
const object_t * pedalRecd = nullptr ;
2022-01-22 14:42:21 +00:00
bool enableFl = false ;
if ( ( rc = ele - > getv ( " midi_out_device " , midiOutDevLabel ,
" midi_out_port " , midiOutPortLabel ,
" enableFl " , enableFl ) ) ! = kOkRC )
{
rc = cwLogError ( kSyntaxErrorRC , " MIDI record play device list configuration parse failed. " ) ;
goto errLabel ;
}
2022-03-20 14:27:46 +00:00
2022-01-22 14:42:21 +00:00
2022-11-11 19:04:44 +00:00
if ( ( rc = ele - > getv_opt (
" vel_table " , velTable ,
" pedal " , pedalRecd ,
" force_damper_down_fl " , p - > midiDevA [ i ] . force_damper_down_fl ,
" force_damper_down_threshold " , p - > midiDevA [ i ] . force_damper_down_threshold ,
" force_damper_down_velocity " , p - > midiDevA [ i ] . force_damper_down_velocity ,
" damper_dead_band_enable_fl " , p - > midiDevA [ i ] . damper_dead_band_enable_fl ,
" damper_dead_band_min_value " , p - > midiDevA [ i ] . damper_dead_band_min_value ,
" damper_dead_band_max_value " , p - > midiDevA [ i ] . damper_dead_band_max_value ) ) ! = kOkRC )
2022-03-20 14:27:46 +00:00
{
rc = cwLogError ( kSyntaxErrorRC , " MIDI record play device optional argument parsing failed. " ) ;
goto errLabel ;
}
2022-10-15 13:26:35 +00:00
2022-11-11 19:04:44 +00:00
cwLogInfo ( " Force Pedal: enabled:%i thresh:%i veloc:%i dead band: enable:%i min:%i max:%i " ,
p - > midiDevA [ i ] . force_damper_down_fl ,
p - > midiDevA [ i ] . force_damper_down_threshold ,
p - > midiDevA [ i ] . force_damper_down_velocity ,
p - > midiDevA [ i ] . damper_dead_band_enable_fl ,
p - > midiDevA [ i ] . damper_dead_band_min_value ,
p - > midiDevA [ i ] . damper_dead_band_max_value ) ;
2022-03-20 14:27:46 +00:00
2022-01-22 14:42:21 +00:00
p - > midiDevA [ i ] . midiOutDevLabel = mem : : duplStr ( midiOutDevLabel ) ;
p - > midiDevA [ i ] . midiOutPortLabel = mem : : duplStr ( midiOutPortLabel ) ;
p - > midiDevA [ i ] . enableFl = enableFl ;
2022-03-20 14:27:46 +00:00
if ( velTable ! = nullptr )
{
p - > midiDevA [ i ] . velTableN = velTable - > child_count ( ) ;
p - > midiDevA [ i ] . velTableArray = mem : : allocZ < uint8_t > ( p - > midiDevA [ i ] . velTableN ) ;
for ( unsigned j = 0 ; j < p - > midiDevA [ i ] . velTableN ; + + j )
{
if ( ( rc = velTable - > child_ele ( j ) - > value ( p - > midiDevA [ i ] . velTableArray [ j ] ) ) ! = kOkRC )
{
rc = cwLogError ( kSyntaxErrorRC , " An error occured while parsing the velocity table for MIDI device:'%s' port:'%s'. " , midiOutDevLabel , midiOutPortLabel ) ;
goto errLabel ;
}
}
}
if ( pedalRecd ! = nullptr )
{
if ( ( rc = pedalRecd - > getv ( " down_id " , p - > midiDevA [ i ] . pedalDownVelId ,
" down_vel " , p - > midiDevA [ i ] . pedalDownVel ,
2022-05-06 20:03:43 +00:00
" half_down_id " , p - > midiDevA [ i ] . pedalDownHalfVelId ,
" half_down_vel " , p - > midiDevA [ i ] . pedalDownHalfVel ,
" half_up_id " , p - > midiDevA [ i ] . pedalUpHalfVelId ,
" half_up_vel " , p - > midiDevA [ i ] . pedalUpHalfVel
) ) ! = kOkRC )
2022-03-20 14:27:46 +00:00
{
rc = cwLogError ( kSyntaxErrorRC , " An error occured while parsing the pedal record for MIDI device:'%s' port:'%s'. " , midiOutDevLabel , midiOutPortLabel ) ;
goto errLabel ;
}
else
{
p - > midiDevA [ i ] . pedalMapEnableFl = true ;
}
}
2022-01-22 14:42:21 +00:00
}
}
2021-05-10 12:37:29 +00:00
// allocate the MIDI msg buffer
p - > msgArray = mem : : allocZ < am_midi_msg_t > ( p - > msgArrayN ) ;
2022-03-20 14:27:46 +00:00
p - > iMsgArray = mem : : allocZ < am_midi_msg_t > ( p - > iMsgArrayN ) ;
2021-05-10 12:37:29 +00:00
errLabel :
return rc ;
}
2021-11-14 16:52:24 +00:00
rc_t _stop ( midi_record_play_t * p ) ;
2022-03-20 14:27:46 +00:00
const am_midi_msg_t * _midi_store ( midi_record_play_t * p , unsigned devIdx , unsigned portIdx , const time : : spec_t & ts , uint8_t ch , uint8_t status , uint8_t d0 , uint8_t d1 )
{
am_midi_msg_t * am = nullptr ;
2022-10-15 19:07:06 +00:00
//if( !midi::isPedal(status,d0) )
// printf("MIDI store: %i : ch:%i st:%i d0:%i d1:%i\n",p->iMsgArrayInIdx,ch,status,d0,d1);
2022-03-20 14:27:46 +00:00
// verify that space exists in the record buffer
if ( p - > iMsgArrayInIdx < p - > iMsgArrayN )
{
// MAKE THIS ATOMIC
unsigned id = p - > iMsgArrayInIdx ;
+ + p - > iMsgArrayInIdx ;
am = p - > iMsgArray + id ;
am - > id = id ;
am - > devIdx = devIdx ;
am - > portIdx = portIdx ;
am - > timestamp = ts ;
am - > ch = ch ;
am - > status = status ;
am - > d0 = d0 ;
am - > d1 = d1 ;
}
return am ;
}
2021-11-01 14:13:48 +00:00
2022-05-06 20:03:43 +00:00
rc_t _event_callback ( midi_record_play_t * p , unsigned id , const time : : spec_t timestamp , unsigned loc , uint8_t ch , uint8_t status , uint8_t d0 , uint8_t d1 , bool log_fl = true )
2021-11-01 14:13:48 +00:00
{
2021-11-14 16:52:24 +00:00
rc_t rc = kOkRC ;
2022-05-14 14:22:29 +00:00
// if we have arrived at the stop time
bool after_stop_time_fl = ! time : : isZero ( p - > end_play_event_timestamp ) & & time : : isGT ( timestamp , p - > end_play_event_timestamp ) ;
bool after_all_off_fl = after_stop_time_fl & & time : : isGT ( timestamp , p - > all_off_timestamp ) ;
bool is_note_on_fl = status = = midi : : kNoteOnMdId and d1 ! = 0 ;
2022-10-15 13:26:35 +00:00
bool is_damper_fl = status = = midi : : kCtlMdId and d0 = = midi : : kSustainCtlMdId ;
2022-11-11 19:04:44 +00:00
bool supress_fl = ( is_note_on_fl & & after_stop_time_fl ) | | p - > muteFl ;
bool is_pedal_fl = midi : : isPedal ( status , d0 ) ;
2021-11-14 16:52:24 +00:00
2022-05-14 14:22:29 +00:00
if ( after_all_off_fl )
2021-11-14 16:52:24 +00:00
{
rc = _stop ( p ) ;
}
2022-05-14 14:22:29 +00:00
else
2021-11-14 16:52:24 +00:00
{
2022-03-20 14:27:46 +00:00
if ( p - > halfPedalFl )
{
if ( status = = midi : : kCtlMdId & & d0 = = midi : : kSustainCtlMdId & & d1 ! = 0 )
d1 = p - > halfPedalMidiPedalVel ;
}
2022-05-14 14:22:29 +00:00
// for each midi device
2022-01-22 14:42:21 +00:00
for ( unsigned i = 0 ; i < p - > midiDevN ; + + i )
if ( p - > midiDevA [ i ] . enableFl )
2022-03-20 14:27:46 +00:00
{
uint8_t out_d1 = d1 ;
2022-05-14 14:22:29 +00:00
2022-03-20 14:27:46 +00:00
if ( ! p - > halfPedalFl )
{
// map the note on velocity
2022-05-14 14:22:29 +00:00
if ( is_note_on_fl and p - > midiDevA [ i ] . velTableArray ! = nullptr )
2022-03-20 14:27:46 +00:00
{
if ( d1 > = p - > midiDevA [ i ] . velTableN )
cwLogError ( kInvalidIdRC , " A MIDI note-on velocity (%i) outside the velocity table range was encountered. " , d1 ) ;
else
out_d1 = p - > midiDevA [ i ] . velTableArray [ d1 ] ;
}
2022-11-11 19:04:44 +00:00
// store the note-on velocity histogram data
2022-10-15 13:26:35 +00:00
if ( p - > velHistogramEnableFl & & is_note_on_fl & & out_d1 < midi : : kMidiVelCnt )
p - > midiDevA [ i ] . velHistogram [ out_d1 ] + = 1 ;
2022-11-11 19:04:44 +00:00
// if the damper pedal velocity is in the dead band then don't send it
if ( p - > midiDevA [ i ] . damper_dead_band_enable_fl & & is_pedal_fl & & p - > midiDevA [ i ] . damper_dead_band_min_value < = out_d1 & & out_d1 < = p - > midiDevA [ i ] . damper_dead_band_max_value )
out_d1 = 0 ;
// if the damper pedal velocity is over the 'forcing' threshold then force the damper down
2022-10-15 13:26:35 +00:00
if ( p - > midiDevA [ i ] . force_damper_down_fl & & is_damper_fl & & out_d1 > p - > midiDevA [ i ] . force_damper_down_threshold )
out_d1 = p - > midiDevA [ i ] . force_damper_down_velocity ;
2022-11-11 19:04:44 +00:00
2022-03-20 14:27:46 +00:00
// map the pedal down velocity
if ( status = = midi : : kCtlMdId & & d0 = = midi : : kSustainCtlMdId & & p - > midiDevA [ i ] . pedalMapEnableFl )
{
2022-05-06 20:03:43 +00:00
if ( d1 = = 0 )
out_d1 = 0 ;
2022-03-20 14:27:46 +00:00
else
2022-05-06 20:03:43 +00:00
if ( d1 = = p - > midiDevA [ i ] . pedalDownVelId )
out_d1 = p - > midiDevA [ i ] . pedalDownVel ;
2022-03-20 14:27:46 +00:00
else
2022-05-06 20:03:43 +00:00
if ( d1 = = p - > midiDevA [ i ] . pedalDownHalfVelId )
out_d1 = p - > midiDevA [ i ] . pedalDownHalfVel ;
2022-10-01 22:51:30 +00:00
else
{
2022-10-15 13:26:35 +00:00
cwLogError ( kInvalidIdRC , " Unexpected pedal down velocity (%i) during pedal velocity mapping. Remove the 'pedal' stanza from the MIDI device cfg to prevent pedal mapping. " , d1 ) ;
2022-10-01 22:51:30 +00:00
}
2022-03-20 14:27:46 +00:00
}
}
2022-05-14 14:22:29 +00:00
2022-11-11 19:04:44 +00:00
if ( ! supress_fl )
{
2022-05-14 14:22:29 +00:00
io : : midiDeviceSend ( p - > ioH , p - > midiDevA [ i ] . midiOutDevIdx , p - > midiDevA [ i ] . midiOutPortIdx , status + ch , d0 , out_d1 ) ;
2022-11-11 19:04:44 +00:00
}
2022-03-20 14:27:46 +00:00
}
2021-11-01 14:13:48 +00:00
2022-05-14 14:22:29 +00:00
if ( ! after_stop_time_fl and p - > cb )
2022-05-21 13:26:23 +00:00
p - > cb ( p - > cb_arg , kMidiEventActionId , id , timestamp , loc , ch , status , d0 , d1 ) ;
2022-03-20 14:27:46 +00:00
if ( log_fl & & p - > logOutFl )
{
// Note: The device of outgoing messages is set to p->midiDevN + 1 to distinguish it from
// incoming messages.
_midi_store ( p , p - > midiDevN , 0 , timestamp , ch , status , d0 , d1 ) ;
}
2021-11-14 16:52:24 +00:00
}
return rc ;
2021-11-01 14:13:48 +00:00
}
2022-03-20 14:27:46 +00:00
rc_t _transmit_msg ( midi_record_play_t * p , const am_midi_msg_t * am , bool log_fl = true )
{
2022-05-06 20:03:43 +00:00
return _event_callback ( p , am - > id , am - > timestamp , am - > loc , am - > ch , am - > status , am - > d0 , am - > d1 , log_fl ) ;
2022-03-20 14:27:46 +00:00
}
rc_t _transmit_note ( midi_record_play_t * p , unsigned ch , unsigned pitch , unsigned vel , unsigned microsecs )
2021-11-01 14:13:48 +00:00
{
2022-03-20 14:27:46 +00:00
time : : spec_t ts = { 0 } ;
time : : microsecondsToSpec ( ts , microsecs ) ;
2022-05-06 20:03:43 +00:00
return _event_callback ( p , kInvalidId , ts , kInvalidId , ch , midi : : kNoteOnMdId , pitch , vel ) ;
2021-11-01 14:13:48 +00:00
}
2022-03-20 14:27:46 +00:00
rc_t _transmit_ctl ( midi_record_play_t * p , unsigned ch , unsigned ctlId , unsigned ctlVal , unsigned microsecs )
2021-11-01 14:13:48 +00:00
{
time : : spec_t ts = { 0 } ;
2022-03-20 14:27:46 +00:00
time : : microsecondsToSpec ( ts , microsecs ) ;
2022-05-06 20:03:43 +00:00
return _event_callback ( p , kInvalidId , ts , kInvalidId , ch , midi : : kCtlMdId , ctlId , ctlVal ) ;
2021-11-01 14:13:48 +00:00
}
2022-03-20 14:27:46 +00:00
rc_t _transmit_pedal ( midi_record_play_t * p , unsigned ch , unsigned pedalCtlId , bool pedalDownFl , unsigned microsecs )
2021-11-01 14:13:48 +00:00
{
2022-03-20 14:27:46 +00:00
return _transmit_ctl ( p , ch , pedalCtlId , pedalDownFl ? 127 : 0 , microsecs ) ;
2021-11-01 14:13:48 +00:00
}
2022-03-20 14:27:46 +00:00
void _half_pedal_update ( midi_record_play_t * p , unsigned cur_time_us )
{
if ( cur_time_us > = p - > halfPedalNextUs )
{
unsigned midi_ch = 0 ;
switch ( p - > halfPedalState )
{
case kWaitForBegin :
printf ( " down: %i %i \n " , cur_time_us / 1000 , p - > halfPedalMidiPedalVel ) ;
_transmit_ctl ( p , midi_ch , midi : : kSustainCtlMdId , p - > halfPedalMidiPedalVel , cur_time_us ) ;
p - > halfPedalState = kWaitForNoteOn ;
p - > halfPedalNextUs + = p - > halfPedalNoteDelayUs ;
break ;
case kWaitForNoteOn :
printf ( " note: %i \n " , cur_time_us / 1000 ) ;
_transmit_note ( p , midi_ch , p - > halfPedalMidiPitch , p - > halfPedalMidiNoteVel , cur_time_us ) ;
p - > halfPedalNextUs + = p - > halfPedalNoteDurUs ;
p - > halfPedalState = kWaitForNoteOff ;
break ;
case kWaitForNoteOff :
printf ( " off: %i \n " , cur_time_us / 1000 ) ;
_transmit_note ( p , midi_ch , p - > halfPedalMidiPitch , 0 , cur_time_us ) ;
p - > halfPedalNextUs + = p - > halfPedalUpDelayUs ;
p - > halfPedalState = kWaitForPedalUp ;
break ;
case kWaitForPedalUp :
printf ( " up: %i \n " , cur_time_us / 1000 ) ;
_transmit_ctl ( p , midi_ch , midi : : kSustainCtlMdId , 0 , cur_time_us ) ;
p - > halfPedalNextUs + = p - > halfPedalDownDelayUs ;
p - > halfPedalState = kWaitForPedalDown ;
break ;
case kWaitForPedalDown :
//printf("down: %i\n",cur_time_us/1000);
//_transmit_ctl( p, midi_ch, midi::kSustainCtlMdId, p->halfPedalMidiPedalVel, cur_time_us);
//_stop(p);
p - > halfPedalState = kHalfPedalDone ;
break ;
case kHalfPedalDone :
break ;
}
}
}
2021-11-01 14:13:48 +00:00
2022-03-20 14:27:46 +00:00
// Set the next location to store an incoming MIDI message
2021-05-10 12:37:29 +00:00
void _set_midi_msg_next_index ( midi_record_play_t * p , unsigned next_idx )
{
2022-03-20 14:27:46 +00:00
p - > iMsgArrayInIdx = next_idx ;
2021-05-10 12:37:29 +00:00
}
2022-03-20 14:27:46 +00:00
// Set the next index of the next MIDI message to transmit
2021-05-10 12:37:29 +00:00
void _set_midi_msg_next_play_index ( midi_record_play_t * p , unsigned next_idx )
{
p - > msgArrayOutIdx = next_idx ;
}
2021-08-15 20:02:45 +00:00
2022-09-10 12:54:17 +00:00
cw : : rc_t _am_file_read_version_0 ( const char * fn , file : : handle_t fH , am_midi_msg_t * amMsgArray , unsigned msgN )
{
// version 0 record type
typedef struct msg_str
{
unsigned dev_idx ;
unsigned port_idx ;
time : : spec_t timestamp ;
uint8_t ch ;
uint8_t st ;
uint8_t d0 ;
uint8_t d1 ;
unsigned microsecs ;
} zero_msg_t ;
cw : : rc_t rc = kOkRC ;
zero_msg_t * zMsgArray = mem : : allocZ < zero_msg_t > ( msgN ) ;
unsigned fileByteN = msgN * sizeof ( zero_msg_t ) ;
if ( ( rc = file : : read ( fH , zMsgArray , fileByteN ) ) ! = kOkRC )
{
rc = cwLogError ( kReadFailRC , " Data read failed on Audio-MIDI file: '%s'. " , fn ) ;
goto errLabel ;
}
for ( unsigned i = 0 ; i < msgN ; + + i )
{
am_midi_msg_t * am = amMsgArray + i ;
zero_msg_t * zm = zMsgArray + i ;
am - > devIdx = zm - > dev_idx ;
am - > portIdx = zm - > port_idx ;
am - > microsec = zm - > microsecs ;
am - > id = i ;
am - > timestamp = zm - > timestamp ;
am - > loc = i ;
am - > ch = zm - > ch ;
am - > status = zm - > st ;
am - > d0 = zm - > d0 ;
am - > d1 = zm - > d1 ;
}
errLabel :
mem : : release ( zMsgArray ) ;
return rc ;
}
2021-08-15 20:02:45 +00:00
// Read the am_midi_msg_t records from a file written by _midi_write()
// If msgArrayCntRef==0 and msgArrayRef==NULL then an array will be allocated and it is up
// to the caller to release it, otherwise the msgArrayCntRef should be set to the count
// of available records in msgArrayRef[]. Note the if there are more records in the file
// than there are record in msgArrayRef[] then a warning will be issued and only
// msgArrayCntRef records will be returned.
cw : : rc_t _am_file_read ( const char * fn , unsigned & msgArrayCntRef , am_midi_msg_t * & msgArrayRef )
{
2022-09-10 12:54:17 +00:00
rc_t rc = kOkRC ;
unsigned recordN = 0 ; // count of records in the ifle
unsigned fileByteN = 0 ; // count of bytes in the file
int version = 0 ; // version id (always a negative number)
bool alloc_fl = false ;
2022-10-15 19:07:06 +00:00
bool print_fl = true ;
2021-08-15 20:02:45 +00:00
file : : handle_t fH ;
2022-09-10 12:54:17 +00:00
2021-08-15 20:02:45 +00:00
if ( ( rc = file : : open ( fH , fn , file : : kReadFl ) ) ! = kOkRC )
{
rc = cwLogError ( kOpenFailRC , " Unable to locate the AM file: '%s'. " , fn ) ;
goto errLabel ;
}
2022-09-10 12:54:17 +00:00
// read the first word - which is either a version id or the count of records in the file
// if this is a version 0 file
if ( ( rc = file : : read ( fH , version ) ) ! = kOkRC )
2021-08-15 20:02:45 +00:00
{
2022-09-10 12:54:17 +00:00
rc = cwLogError ( kReadFailRC , " Version read failed on Audio-MIDI file: '%s'. " , fn ) ;
2021-08-15 20:02:45 +00:00
goto errLabel ;
}
2022-09-10 12:54:17 +00:00
// if the version is greater than 0 then this is a version 0 file (and 'version' holds the record count.
if ( version > 0 )
{
recordN = ( unsigned ) version ;
version = 0 ;
}
else // otherwise the second word in the file holds the size of the file
{
if ( ( rc = file : : read ( fH , recordN ) ) ! = kOkRC )
{
rc = cwLogError ( kReadFailRC , " Header read failed on Audio-MIDI file: '%s'. " , fn ) ;
goto errLabel ;
}
}
// if the output msg array was not allocated - then allocate it here
2021-08-15 20:02:45 +00:00
if ( msgArrayCntRef = = 0 | | msgArrayRef = = nullptr )
{
2022-09-10 12:54:17 +00:00
alloc_fl = true ;
msgArrayRef = mem : : allocZ < am_midi_msg_t > ( recordN ) ;
2021-08-15 20:02:45 +00:00
}
2022-09-10 12:54:17 +00:00
else // if the msg array was allocated but is too small - then decrease the count of records to be read from the file
2021-08-15 20:02:45 +00:00
{
2022-09-10 12:54:17 +00:00
if ( recordN > msgArrayCntRef )
2021-08-15 20:02:45 +00:00
{
2022-09-10 12:54:17 +00:00
cwLogWarning ( " The count of message in Audio-MIDI file '%s' reduced from %i to %i. " , fn , recordN , msgArrayCntRef ) ;
recordN = msgArrayCntRef ;
2021-08-15 20:02:45 +00:00
}
}
2022-09-10 12:54:17 +00:00
if ( version = = 0 )
2021-08-15 20:02:45 +00:00
{
2022-09-10 12:54:17 +00:00
// read the version 0 file into a temporary buffer then translate to am_midi_msg_t records
if ( ( rc = _am_file_read_version_0 ( fn , fH , msgArrayRef , recordN ) ) ! = kOkRC )
goto errLabel ;
}
else
{
if ( ( rc = file : : read ( fH , msgArrayRef , fileByteN ) ) ! = kOkRC )
{
rc = cwLogError ( kReadFailRC , " Data read failed on Audio-MIDI file: '%s'. " , fn ) ;
goto errLabel ;
}
2021-08-15 20:02:45 +00:00
}
2022-09-10 12:54:17 +00:00
if ( print_fl )
{
for ( unsigned i = 0 ; i < recordN ; + + i )
{
am_midi_msg_t * m = msgArrayRef ;
double dt = time : : elapsedSecs ( m [ 0 ] . timestamp , m [ i ] . timestamp ) ;
printf ( " %4i %4i : %6.2f %2x %2x %2x %2x : %6.2f \n " , m [ i ] . devIdx , m [ i ] . portIdx , dt , m [ i ] . ch , m [ i ] . status , m [ i ] . d0 , m [ i ] . d1 , m [ i ] . microsec / ( 1000.0 * 1000.0 ) ) ;
}
}
msgArrayCntRef = recordN ;
2021-08-15 20:02:45 +00:00
errLabel :
2022-09-10 12:54:17 +00:00
if ( rc ! = kOkRC and alloc_fl )
{
mem : : release ( msgArrayRef ) ;
msgArrayRef = nullptr ;
msgArrayCntRef = 0 ;
}
2021-08-15 20:02:45 +00:00
return rc ;
}
2022-03-20 14:27:46 +00:00
// Fill the play buffer from a previously store AM file.
2021-05-10 12:37:29 +00:00
rc_t _midi_read ( midi_record_play_t * p , const char * fn )
{
rc_t rc = kOkRC ;
unsigned n = 0 ;
2022-09-10 12:54:17 +00:00
int version ;
2021-05-10 12:37:29 +00:00
file : : handle_t fH ;
2022-09-10 12:54:17 +00:00
2021-05-10 12:37:29 +00:00
if ( ( rc = file : : open ( fH , fn , file : : kReadFl ) ) ! = kOkRC )
{
rc = cwLogError ( kOpenFailRC , " Unable to locate the file: '%s'. " , fn ) ;
goto errLabel ;
}
2022-09-10 12:54:17 +00:00
if ( ( rc = file : : read ( fH , version ) ) ! = kOkRC )
{
rc = cwLogError ( kReadFailRC , " Version read failed on Audio-MIDI file: '%s'. " , fn ) ;
goto errLabel ;
}
2021-05-10 12:37:29 +00:00
if ( ( rc = file : : read ( fH , n ) ) ! = kOkRC )
{
rc = cwLogError ( kReadFailRC , " Header read failed on Audio-MIDI file: '%s'. " , fn ) ;
goto errLabel ;
}
if ( n > p - > msgArrayN )
{
cwLogWarning ( " The count of message in Audio-MIDI file '%s' reduced from %i to %i. " , fn , n , p - > msgArrayN ) ;
n = p - > msgArrayN ;
}
if ( ( rc = file : : read ( fH , p - > msgArray , n * sizeof ( am_midi_msg_t ) ) ) ! = kOkRC )
{
rc = cwLogError ( kReadFailRC , " Data read failed on Audio-MIDI file: '%s'. " , fn ) ;
goto errLabel ;
}
2022-03-20 14:27:46 +00:00
p - > msgArrayInIdx = n ;
2021-05-10 12:37:29 +00:00
cwLogInfo ( " Read %i from '%s'. " , n , fn ) ;
errLabel :
2022-03-20 14:27:46 +00:00
file : : close ( fH ) ;
2021-05-10 12:37:29 +00:00
return rc ;
}
2022-03-20 14:27:46 +00:00
// Write the record buffer to an AM file
2021-05-10 12:37:29 +00:00
rc_t _midi_write ( midi_record_play_t * p , const char * fn )
{
rc_t rc = kOkRC ;
file : : handle_t fH ;
2022-09-10 12:54:17 +00:00
// NOTE: version must be a small negative number to differentiate from file version that
// whose first word is the count of records in the file, rather than the version number
int version = - 1 ;
2022-03-20 14:27:46 +00:00
if ( p - > iMsgArrayInIdx = = 0 )
2021-05-10 12:37:29 +00:00
{
cwLogWarning ( " Nothing to write. " ) ;
return rc ;
}
// open the file
if ( ( rc = file : : open ( fH , fn , file : : kWriteFl ) ) ! = kOkRC )
{
rc = cwLogError ( kOpenFailRC , " Unable to create the file '%s'. " , cwStringNullGuard ( fn ) ) ;
goto errLabel ;
}
2022-09-10 12:54:17 +00:00
// write the file version
if ( ( rc = write ( fH , version ) ) ! = kOkRC )
{
rc = cwLogError ( kWriteFailRC , " Version write to '%s' failed. " , cwStringNullGuard ( fn ) ) ;
goto errLabel ;
}
2021-05-10 12:37:29 +00:00
// write the file header
2022-03-20 14:27:46 +00:00
if ( ( rc = write ( fH , p - > iMsgArrayInIdx ) ) ! = kOkRC )
2021-05-10 12:37:29 +00:00
{
rc = cwLogError ( kWriteFailRC , " Header write to '%s' failed. " , cwStringNullGuard ( fn ) ) ;
goto errLabel ;
}
// write the file data
2022-03-20 14:27:46 +00:00
if ( ( rc = write ( fH , p - > iMsgArray , sizeof ( am_midi_msg_t ) * p - > iMsgArrayInIdx ) ) ! = kOkRC )
2021-05-10 12:37:29 +00:00
{
rc = cwLogError ( kWriteFailRC , " Data write to '%s' failed. " , cwStringNullGuard ( fn ) ) ;
goto errLabel ;
}
2022-03-20 14:27:46 +00:00
errLabel :
2021-05-10 12:37:29 +00:00
file : : close ( fH ) ;
2022-03-20 14:27:46 +00:00
cwLogInfo ( " Saved %i events to '%s'. " , p - > iMsgArrayInIdx , fn ) ;
return rc ;
}
rc_t _write_csv ( midi_record_play_t * p , const char * fn )
{
rc_t rc = kOkRC ;
file : : handle_t fH ;
if ( p - > iMsgArrayInIdx = = 0 )
{
cwLogWarning ( " Nothing to write. " ) ;
return rc ;
}
// open the file
if ( ( rc = file : : open ( fH , fn , file : : kWriteFl ) ) ! = kOkRC )
{
rc = cwLogError ( kOpenFailRC , " Unable to create the file '%s'. " , cwStringNullGuard ( fn ) ) ;
goto errLabel ;
}
file : : printf ( fH , " dev,port,microsec,id,sec,ch,status,d0,d1 \n " ) ;
for ( unsigned i = 0 ; i < p - > iMsgArrayInIdx ; + + i )
{
const am_midi_msg_t * m = p - > iMsgArray + i ;
double secs = time : : elapsedSecs ( p - > iMsgArray [ 0 ] . timestamp , p - > iMsgArray [ i ] . timestamp ) ;
2021-05-10 12:37:29 +00:00
2022-03-20 14:27:46 +00:00
char sciPitch [ midi : : kMidiSciPitchCharCnt + 1 ] ;
if ( m - > status = = midi : : kNoteOnMdId )
midi : : midiToSciPitch ( m - > d0 , sciPitch , midi : : kMidiSciPitchCharCnt ) ;
else
strcpy ( sciPitch , " " ) ;
if ( ( rc = file : : printf ( fH , " %3i,%3i,%8i,%3i,%8.4f,%2i,0x%2x,%5s,%3i,%3i \n " ,
m - > devIdx , m - > portIdx , m - > microsec , m - > id , secs ,
m - > ch , m - > status , sciPitch , m - > d0 , m - > d1 ) ) ! = kOkRC )
{
rc = cwLogError ( rc , " Write failed on line:%i " , i + 1 ) ;
goto errLabel ;
}
}
2021-05-10 12:37:29 +00:00
errLabel :
2022-03-20 14:27:46 +00:00
file : : close ( fH ) ;
cwLogInfo ( " Saved %i events to '%s'. " , p - > iMsgArrayInIdx , fn ) ;
2021-05-10 12:37:29 +00:00
return rc ;
2022-03-20 14:27:46 +00:00
2021-05-10 12:37:29 +00:00
}
2021-08-15 20:02:45 +00:00
rc_t _midi_file_write ( const char * fn , const am_midi_msg_t * msgArray , unsigned msgArrayCnt )
{
rc_t rc = kOkRC ;
const unsigned midiFileTrackCnt = 1 ;
const unsigned midiFileTicksPerQN = 192 ;
const unsigned midiFileTempoBpm = 120 ;
const unsigned midiFileTrkIdx = 0 ;
file : : handle_t fH ;
midi : : file : : handle_t mfH ;
time : : spec_t t0 ;
if ( msgArrayCnt = = 0 )
{
cwLogWarning ( " Nothing to write. " ) ;
return rc ;
}
if ( ( rc = midi : : file : : create ( mfH , midiFileTrackCnt , midiFileTicksPerQN ) ) ! = kOkRC )
{
rc = cwLogError ( rc , " MIDI file create failed. File:'%s' " , cwStringNullGuard ( fn ) ) ;
goto errLabel ;
}
if ( ( rc = midi : : file : : insertTrackTempoMsg ( mfH , midiFileTrkIdx , 0 , midiFileTempoBpm ) ) ! = kOkRC )
{
rc = cwLogError ( rc , " MIDI file tempo message insert failed. File:'%s' " , cwStringNullGuard ( fn ) ) ;
goto errLabel ;
}
t0 = msgArray [ 0 ] . timestamp ;
for ( unsigned i = 0 ; i < msgArrayCnt ; + + i )
{
double secs = time : : elapsedMicros ( t0 , msgArray [ i ] . timestamp ) / 1000000.0 ;
unsigned atick = secs * midiFileTicksPerQN * midiFileTempoBpm / 60.0 ;
if ( ( rc = insertTrackChMsg ( mfH , midiFileTrkIdx , atick , msgArray [ i ] . ch + msgArray [ i ] . status , msgArray [ i ] . d0 , msgArray [ i ] . d1 ) ) ! = kOkRC )
{
rc = cwLogError ( rc , " MIDI file message insert failed. File: '%s'. " , cwStringNullGuard ( fn ) ) ;
goto errLabel ;
}
}
if ( ( rc = midi : : file : : write ( mfH , fn ) ) ! = kOkRC )
{
rc = cwLogError ( rc , " MIDI file write failed on '%s'. " , cwStringNullGuard ( fn ) ) ;
goto errLabel ;
}
if ( ( rc = midi : : file : : close ( mfH ) ) ! = kOkRC )
{
rc = cwLogError ( rc , " MIDI file close failed on '%s'. " , cwStringNullGuard ( fn ) ) ;
goto errLabel ;
}
errLabel :
return rc ;
}
2021-05-10 12:37:29 +00:00
void _print_midi_msg ( const am_midi_msg_t * mm )
{
printf ( " %i %i : %10i : %2i 0x%02x 0x%02x 0x%02x \n " , mm - > devIdx , mm - > portIdx , mm - > microsec , mm - > ch , mm - > status , mm - > d0 , mm - > d1 ) ;
}
void _report_midi ( midi_record_play_t * p )
{
2022-10-15 19:07:06 +00:00
printf ( " omsg cnt:%i \n " , p - > msgArrayInIdx ) ;
2021-05-10 12:37:29 +00:00
for ( unsigned i = 0 ; i < p - > msgArrayInIdx ; + + i )
{
am_midi_msg_t * mm = p - > msgArray + i ;
_print_midi_msg ( mm ) ;
}
2022-10-15 19:07:06 +00:00
printf ( " imsg cnt:%i \n " , p - > iMsgArrayInIdx ) ;
for ( unsigned i = 0 ; i < p - > iMsgArrayInIdx ; + + i )
{
am_midi_msg_t * mm = p - > iMsgArray + i ;
_print_midi_msg ( mm ) ;
}
2021-05-10 12:37:29 +00:00
}
2022-10-15 13:26:35 +00:00
rc_t _write_vel_histogram ( midi_record_play_t * p )
{
const char * fname = " /home/kevin/temp/vel_histogram.txt " ;
rc_t rc = kOkRC ;
if ( ! p - > velHistogramEnableFl )
return rc ;
file : : handle_t h ;
if ( ( rc = file : : open ( h , fname , file : : kWriteFl ) ) ! = kOkRC )
return rc ;
for ( unsigned i = 0 ; i < p - > midiDevN ; + + i )
if ( p - > midiDevA [ i ] . enableFl )
{
for ( unsigned j = 0 ; j < midi : : kMidiVelCnt ; + + j )
if ( ( rc = file : : printf ( h , " %i, " , p - > midiDevA [ i ] . velHistogram [ j ] ) ) ! = kOkRC )
{
rc = cwLogError ( rc , " Histogram output file (%s) write failed. " , fname ) ;
goto errLabel ;
}
file : : printf ( h , " \n " ) ;
}
errLabel :
file : : close ( h ) ;
return rc ;
}
2022-10-15 19:07:06 +00:00
// Fill the play buffer (msgArray) from the record buffer (iMsgArray)
void _iMsgArray_to_msgArray ( midi_record_play_t * p )
{
if ( p - > msgArrayN < p - > iMsgArrayN )
{
mem : : resize ( p - > msgArray , p - > iMsgArrayN , mem : : kZeroAllFl ) ;
p - > msgArrayN = p - > iMsgArrayN ;
}
p - > msgArrayOutIdx = 0 ;
p - > msgArrayInIdx = p - > iMsgArrayInIdx ;
memcpy ( p - > msgArray , p - > iMsgArray , p - > iMsgArrayInIdx * sizeof ( am_midi_msg_t ) ) ;
}
2021-05-10 12:37:29 +00:00
rc_t _stop ( midi_record_play_t * p )
{
rc_t rc = kOkRC ;
p - > startedFl = false ;
time : : spec_t t1 ;
time : : get ( t1 ) ;
2022-10-15 13:26:35 +00:00
_write_vel_histogram ( p ) ;
2022-03-20 14:27:46 +00:00
// if we were recording
2021-05-10 12:37:29 +00:00
if ( p - > recordFl )
{
2022-03-20 14:27:46 +00:00
// set the 'microsec' value for each MIDI msg as an offset from the first message[]
for ( unsigned i = 0 ; i < p - > iMsgArrayInIdx ; + + i )
2021-05-10 12:37:29 +00:00
{
2022-10-15 19:07:06 +00:00
p - > iMsgArray [ i ] . microsec = time : : elapsedMicros ( p - > iMsgArray [ 0 ] . timestamp , p - > iMsgArray [ i ] . timestamp ) ;
2021-05-10 12:37:29 +00:00
}
2022-10-15 19:07:06 +00:00
// copy the recorded messages from the input buffer to the output buffer
_iMsgArray_to_msgArray ( p ) ;
2021-05-10 12:37:29 +00:00
cwLogInfo ( " MIDI messages recorded: %i " , p - > msgArrayInIdx ) ;
}
else
{
io : : timerStop ( p - > ioH , io : : timerIdToIndex ( p - > ioH , kMidiRecordPlayTimerId ) ) ;
2022-03-20 14:27:46 +00:00
// TODO:
// BUG BUG BUG: should work for all channels
2021-05-10 12:37:29 +00:00
// all notes off
2022-03-20 14:27:46 +00:00
_transmit_ctl ( p , 0 , 121 , 0 , 0 ) ; // reset all controllers
_transmit_ctl ( p , 0 , 123 , 0 , 0 ) ; // all notes off
_transmit_ctl ( p , 0 , 0 , 0 , 0 ) ; // switch to bank 0
2021-11-01 14:13:48 +00:00
2021-05-10 12:37:29 +00:00
}
2022-05-21 13:26:23 +00:00
if ( p - > cb ! = nullptr )
p - > cb ( p - > cb_arg , kPlayerStoppedActionId , kInvalidId , t1 , kInvalidId , 0 , 0 , 0 , 0 ) ;
2021-05-10 12:37:29 +00:00
return rc ;
}
2022-03-20 14:27:46 +00:00
2021-11-01 14:13:48 +00:00
rc_t _midi_receive ( midi_record_play_t * p , const io : : midi_msg_t & m )
2021-05-10 12:37:29 +00:00
{
rc_t rc = kOkRC ;
const midi : : packet_t * pkt = m . pkt ;
// for each midi msg
for ( unsigned j = 0 ; j < pkt - > msgCnt ; + + j )
{
// if this is a sys-ex msg
if ( pkt - > msgArray = = NULL )
{
2022-10-15 19:07:06 +00:00
cwLogError ( kNotImplementedRC , " Sys-ex recording not implemented. " ) ;
2021-05-10 12:37:29 +00:00
}
else // this is a triple
{
2021-11-01 14:13:48 +00:00
//if( !midi::isPedal(pkt->msgArray[j].status,pkt->msgArray[j].d0) )
2022-10-15 19:07:06 +00:00
// printf("IN: 0x%x 0x%x 0x%x\n", pkt->msgArray[j].status, pkt->msgArray[j].d0, pkt->msgArray[j].d1 );
2021-11-01 14:13:48 +00:00
2022-03-20 14:27:46 +00:00
if ( ( p - > recordFl | | p - > logInFl ) & & p - > startedFl )
2021-05-10 12:37:29 +00:00
{
// verify that space exists in the record buffer
2022-03-20 14:27:46 +00:00
if ( p - > iMsgArrayInIdx > = p - > iMsgArrayN )
2021-05-10 12:37:29 +00:00
{
_stop ( p ) ;
2022-03-20 14:27:46 +00:00
rc = cwLogError ( kBufTooSmallRC , " MIDI message record buffer is full. % messages. " , p - > iMsgArrayN ) ;
2021-05-10 12:37:29 +00:00
goto errLabel ;
}
else
{
// copy the msg into the record buffer
midi : : msg_t * mm = pkt - > msgArray + j ;
if ( midi : : isChStatus ( mm - > status ) )
{
2022-03-20 14:27:46 +00:00
const am_midi_msg_t * am = _midi_store ( p , pkt - > devIdx , pkt - > portIdx , mm - > timeStamp , mm - > status & 0x0f , mm - > status & 0xf0 , mm - > d0 , mm - > d1 ) ;
2021-11-01 14:13:48 +00:00
2022-03-20 14:27:46 +00:00
if ( p - > thruFl & & am ! = nullptr )
_transmit_msg ( p , am , false ) ;
2021-05-10 12:37:29 +00:00
}
}
}
}
}
errLabel :
return rc ;
}
2022-03-20 14:27:46 +00:00
2021-05-10 12:37:29 +00:00
rc_t _timer_callback ( midi_record_play_t * p , io : : timer_msg_t & m )
{
rc_t rc = kOkRC ;
// if the MIDI player is started and in 'play' mode and msg remain to be played
if ( p - > startedFl & & ( p - > recordFl = = false ) & & ( p - > msgArrayOutIdx < p - > msgArrayInIdx ) )
{
time : : spec_t t ;
time : : get ( t ) ;
unsigned cur_time_us = time : : elapsedMicros ( p - > play_time , t ) ;
2022-03-20 14:27:46 +00:00
if ( p - > halfPedalFl )
_half_pedal_update ( p , cur_time_us ) ;
else
while ( p - > msgArray [ p - > msgArrayOutIdx ] . microsec < = cur_time_us )
{
2021-05-10 12:37:29 +00:00
2022-03-20 14:27:46 +00:00
am_midi_msg_t * mm = p - > msgArray + p - > msgArrayOutIdx ;
2021-05-10 12:37:29 +00:00
2022-03-20 14:27:46 +00:00
//_print_midi_msg(mm);
2021-05-10 12:37:29 +00:00
2022-03-20 14:27:46 +00:00
_transmit_msg ( p , mm ) ;
_set_midi_msg_next_play_index ( p , p - > msgArrayOutIdx + 1 ) ;
2021-11-01 14:13:48 +00:00
2022-03-20 14:27:46 +00:00
// if all MIDI messages have been played
if ( p - > msgArrayOutIdx > = p - > msgArrayInIdx )
2021-08-15 20:02:45 +00:00
{
2022-03-20 14:27:46 +00:00
_stop ( p ) ;
break ;
2021-08-15 20:02:45 +00:00
}
}
2021-05-10 12:37:29 +00:00
}
return rc ;
}
}
}
2021-11-01 14:13:48 +00:00
cw : : rc_t cw : : midi_record_play : : create ( handle_t & hRef , io : : handle_t ioH , const object_t & cfg , event_callback_t cb , void * cb_arg )
2021-05-10 12:37:29 +00:00
{
midi_record_play_t * p = nullptr ;
rc_t rc ;
if ( ( rc = destroy ( hRef ) ) ! = kOkRC )
return rc ;
p = mem : : allocZ < midi_record_play_t > ( ) ;
if ( ( rc = _parseCfg ( p , cfg ) ) ! = kOkRC )
goto errLabel ;
p - > ioH = ioH ;
2021-11-01 14:13:48 +00:00
p - > cb = cb ;
p - > cb_arg = cb_arg ;
2022-03-20 14:27:46 +00:00
p - > halfPedalState = kHalfPedalDone ;
p - > halfPedalNextUs = 0 ;
p - > halfPedalNoteDelayUs = 100 * 1000 ;
p - > halfPedalNoteDurUs = 1000 * 1000 ;
p - > halfPedalUpDelayUs = 1000 * 1000 ;
p - > halfPedalDownDelayUs = 1000 * 1000 ;
p - > halfPedalMidiPitch = 64 ;
p - > halfPedalMidiNoteVel = 64 ;
p - > halfPedalMidiPedalVel = 127 ;
2022-01-22 14:42:21 +00:00
2022-05-14 16:37:11 +00:00
for ( unsigned i = 0 ; i < p - > midiDevN ; + + i )
2021-05-10 12:37:29 +00:00
{
2022-01-22 14:42:21 +00:00
midi_device_t * dev = p - > midiDevA + i ;
2022-05-14 16:37:11 +00:00
if ( ! p - > midiDevA [ i ] . enableFl )
continue ;
2022-01-22 14:42:21 +00:00
if ( ( dev - > midiOutDevIdx = io : : midiDeviceIndex ( p - > ioH , dev - > midiOutDevLabel ) ) = = kInvalidIdx )
{
rc = cwLogError ( kInvalidArgRC , " The MIDI output device: '%s' was not found. " , cwStringNullGuard ( dev - > midiOutDevLabel ) ) ;
goto errLabel ;
}
if ( ( dev - > midiOutPortIdx = io : : midiDevicePortIndex ( p - > ioH , dev - > midiOutDevIdx , false , dev - > midiOutPortLabel ) ) = = kInvalidIdx )
{
rc = cwLogError ( kInvalidArgRC , " The MIDI output port: '%s' was not found. " , cwStringNullGuard ( dev - > midiOutPortLabel ) ) ;
goto errLabel ;
}
printf ( " %s %s : %i %i \n " , dev - > midiOutDevLabel , dev - > midiOutPortLabel , dev - > midiOutDevIdx , dev - > midiOutPortIdx ) ;
2021-05-10 12:37:29 +00:00
}
// create the MIDI playback timer
2021-12-30 02:37:07 +00:00
if ( ( rc = timerCreate ( p - > ioH , TIMER_LABEL , kMidiRecordPlayTimerId , p - > midi_timer_period_micro_sec ) ) ! = kOkRC )
2021-05-10 12:37:29 +00:00
{
cwLogError ( rc , " Audio-MIDI timer create failed. " ) ;
goto errLabel ;
}
errLabel :
if ( rc = = kOkRC )
hRef . set ( p ) ;
else
_destroy ( p ) ;
return rc ;
}
cw : : rc_t cw : : midi_record_play : : destroy ( handle_t & hRef )
{
rc_t rc = kOkRC ;
if ( ! hRef . isValid ( ) )
return kOkRC ;
midi_record_play_t * p = _handleToPtr ( hRef ) ;
if ( ( rc = _destroy ( p ) ) ! = kOkRC )
return rc ;
hRef . clear ( ) ;
return rc ;
}
2021-11-14 16:52:24 +00:00
cw : : rc_t cw : : midi_record_play : : start ( handle_t h , bool rewindFl , const time : : spec_t * end_play_event_timestamp )
2021-05-10 12:37:29 +00:00
{
midi_record_play_t * p = _handleToPtr ( h ) ;
p - > startedFl = true ;
2022-10-15 13:26:35 +00:00
if ( p - > velHistogramEnableFl )
for ( unsigned i = 0 ; i < p - > midiDevN ; + + i )
if ( p - > midiDevA [ i ] . enableFl )
memset ( p - > midiDevA [ i ] . velHistogram , 0 , sizeof ( p - > midiDevA [ i ] . velHistogram ) ) ;
2021-11-14 16:52:24 +00:00
// set the end play time
if ( end_play_event_timestamp = = nullptr or time : : isZero ( * end_play_event_timestamp ) )
time : : setZero ( p - > end_play_event_timestamp ) ;
else
2022-05-14 14:22:29 +00:00
{
2021-11-14 16:52:24 +00:00
p - > end_play_event_timestamp = * end_play_event_timestamp ;
2022-05-14 14:22:29 +00:00
p - > all_off_timestamp = * end_play_event_timestamp ;
time : : advanceMs ( p - > all_off_timestamp , p - > all_off_delay_ms ) ;
}
2021-11-14 16:52:24 +00:00
2021-05-10 12:37:29 +00:00
time : : get ( p - > start_time ) ;
2022-03-20 14:27:46 +00:00
if ( p - > recordFl | | p - > logInFl or p - > logOutFl )
2021-05-10 12:37:29 +00:00
{
_set_midi_msg_next_index ( p , 0 ) ;
}
2022-03-20 14:27:46 +00:00
if ( ! p - > recordFl )
2021-05-10 12:37:29 +00:00
{
time : : get ( p - > play_time ) ;
2021-11-01 14:13:48 +00:00
if ( rewindFl )
_set_midi_msg_next_play_index ( p , 0 ) ;
else
{
// Set the begin play time back by the time offset of the current output event.
// This will cause that event to be played back immediately.
time : : subtractMicros ( p - > play_time , p - > msgArray [ p - > msgArrayOutIdx ] . microsec ) ;
}
2022-03-20 14:27:46 +00:00
if ( p - > halfPedalFl )
{
p - > halfPedalNextUs = 0 ;
p - > halfPedalState = kWaitForBegin ;
}
2021-11-01 14:13:48 +00:00
io : : timerStart ( p - > ioH , io : : timerIdToIndex ( p - > ioH , kMidiRecordPlayTimerId ) ) ;
2021-05-10 12:37:29 +00:00
}
return kOkRC ;
}
cw : : rc_t cw : : midi_record_play : : stop ( handle_t h )
{
midi_record_play_t * p = _handleToPtr ( h ) ;
return _stop ( p ) ;
}
bool cw : : midi_record_play : : is_started ( handle_t h )
{
midi_record_play_t * p = _handleToPtr ( h ) ;
return p - > startedFl ;
}
cw : : rc_t cw : : midi_record_play : : clear ( handle_t h )
{
rc_t rc = kOkRC ;
midi_record_play_t * p = _handleToPtr ( h ) ;
_set_midi_msg_next_index ( p , 0 ) ;
return rc ;
}
cw : : rc_t cw : : midi_record_play : : set_record_state ( handle_t h , bool record_fl )
{
rc_t rc = kOkRC ;
midi_record_play_t * p = _handleToPtr ( h ) ;
p - > recordFl = record_fl ;
return rc ;
}
bool cw : : midi_record_play : : record_state ( handle_t h )
{
midi_record_play_t * p = _handleToPtr ( h ) ;
return p - > recordFl ;
}
2022-11-11 19:04:44 +00:00
cw : : rc_t cw : : midi_record_play : : set_mute_state ( handle_t h , bool mute_fl )
{
rc_t rc = kOkRC ;
midi_record_play_t * p = _handleToPtr ( h ) ;
p - > muteFl = mute_fl ;
return rc ;
}
bool cw : : midi_record_play : : mute_state ( handle_t h )
{
midi_record_play_t * p = _handleToPtr ( h ) ;
return p - > muteFl ;
}
2021-05-10 12:37:29 +00:00
cw : : rc_t cw : : midi_record_play : : set_thru_state ( handle_t h , bool thru_fl )
{
rc_t rc = kOkRC ;
midi_record_play_t * p = _handleToPtr ( h ) ;
p - > thruFl = thru_fl ;
return rc ;
}
bool cw : : midi_record_play : : thru_state ( handle_t h )
{
midi_record_play_t * p = _handleToPtr ( h ) ;
return p - > thruFl ;
}
cw : : rc_t cw : : midi_record_play : : save ( handle_t h , const char * fn )
{
midi_record_play_t * p = _handleToPtr ( h ) ;
return _midi_write ( p , fn ) ;
}
2022-03-20 14:27:46 +00:00
cw : : rc_t cw : : midi_record_play : : save_csv ( handle_t h , const char * fn )
{
midi_record_play_t * p = _handleToPtr ( h ) ;
return _write_csv ( p , fn ) ;
}
2021-05-10 12:37:29 +00:00
cw : : rc_t cw : : midi_record_play : : open ( handle_t h , const char * fn )
{
midi_record_play_t * p = _handleToPtr ( h ) ;
return _midi_read ( p , fn ) ;
}
2021-11-01 14:13:48 +00:00
cw : : rc_t cw : : midi_record_play : : load ( handle_t h , const midi_msg_t * msg , unsigned msg_count )
{
rc_t rc = kOkRC ;
midi_record_play_t * p = _handleToPtr ( h ) ;
if ( msg_count > p - > msgArrayN )
{
mem : : release ( p - > msgArray ) ;
p - > msgArray = mem : : allocZ < am_midi_msg_t > ( msg_count ) ;
p - > msgArrayN = msg_count ;
}
for ( unsigned i = 0 ; i < msg_count ; + + i )
{
p - > msgArray [ i ] . id = msg [ i ] . id ;
p - > msgArray [ i ] . timestamp = msg [ i ] . timestamp ;
2022-05-06 20:03:43 +00:00
p - > msgArray [ i ] . loc = msg [ i ] . loc ;
2021-11-01 14:13:48 +00:00
p - > msgArray [ i ] . ch = msg [ i ] . ch ;
p - > msgArray [ i ] . status = msg [ i ] . status ;
p - > msgArray [ i ] . d0 = msg [ i ] . d0 ;
p - > msgArray [ i ] . d1 = msg [ i ] . d1 ;
2022-01-22 14:42:21 +00:00
p - > msgArray [ i ] . devIdx = kInvalidIdx ;
p - > msgArray [ i ] . portIdx = kInvalidIdx ;
2021-11-01 14:13:48 +00:00
p - > msgArray [ i ] . microsec = time : : elapsedMicros ( p - > msgArray [ 0 ] . timestamp , p - > msgArray [ i ] . timestamp ) ;
2022-03-20 14:27:46 +00:00
2021-11-01 14:13:48 +00:00
}
p - > msgArrayInIdx = msg_count ;
p - > msgArrayOutIdx = 0 ;
return rc ;
}
cw : : rc_t cw : : midi_record_play : : seek ( handle_t h , time : : spec_t seek_timestamp )
{
rc_t rc = kOkRC ;
bool damp_down_fl = false ; // TODO: track pedals on all channels
bool sost_down_fl = false ;
bool soft_down_fl = false ;
midi_record_play_t * p = _handleToPtr ( h ) ;
for ( unsigned i = 0 ; i < p - > msgArrayInIdx ; + + i )
{
am_midi_msg_t * mm = p - > msgArray + i ;
if ( time : : isLTE ( seek_timestamp , mm - > timestamp ) )
{
p - > msgArrayOutIdx = i ;
2022-03-20 14:27:46 +00:00
_transmit_pedal ( p , mm - > ch , midi : : kSustainCtlMdId , damp_down_fl , 0 ) ;
_transmit_pedal ( p , mm - > ch , midi : : kSostenutoCtlMdId , sost_down_fl , 0 ) ;
_transmit_pedal ( p , mm - > ch , midi : : kSoftPedalCtlMdId , soft_down_fl , 0 ) ;
2022-06-12 16:15:08 +00:00
//cwLogInfo("damper: %s.", damp_down_fl ? "down" : "up");
2021-11-01 14:13:48 +00:00
break ;
}
if ( mm - > status = = midi : : kCtlMdId )
{
switch ( mm - > d0 )
{
case midi : : kSustainCtlMdId :
damp_down_fl = mm - > d1 > 64 ;
break ;
case midi : : kSostenutoCtlMdId :
sost_down_fl = mm - > d1 > 64 ;
break ;
case midi : : kSoftPedalCtlMdId :
soft_down_fl = mm - > d1 > 64 ;
break ;
default :
break ;
}
}
}
return rc ;
}
2021-05-10 12:37:29 +00:00
unsigned cw : : midi_record_play : : event_count ( handle_t h )
{
midi_record_play_t * p = _handleToPtr ( h ) ;
return p - > msgArrayInIdx ;
}
unsigned cw : : midi_record_play : : event_index ( handle_t h )
{
midi_record_play_t * p = _handleToPtr ( h ) ;
2022-03-20 14:27:46 +00:00
return p - > recordFl ? p - > iMsgArrayInIdx : p - > msgArrayOutIdx ;
2021-05-10 12:37:29 +00:00
}
2022-05-06 20:03:43 +00:00
unsigned cw : : midi_record_play : : event_loc ( handle_t h )
{
midi_record_play_t * p = _handleToPtr ( h ) ;
if ( ! p - > recordFl & & 0 < = p - > msgArrayOutIdx & & p - > msgArrayOutIdx < p - > msgArrayN )
return p - > msgArray [ p - > msgArrayOutIdx ] . loc ;
return kInvalidId ;
}
2021-05-10 12:37:29 +00:00
cw : : rc_t cw : : midi_record_play : : exec ( handle_t h , const io : : msg_t & m )
{
rc_t rc = kOkRC ;
midi_record_play_t * p = _handleToPtr ( h ) ;
switch ( m . tid )
{
case io : : kTimerTId :
if ( m . u . timer ! = nullptr )
rc = _timer_callback ( p , * m . u . timer ) ;
break ;
case io : : kMidiTId :
if ( m . u . midi ! = nullptr )
2021-11-01 14:13:48 +00:00
_midi_receive ( p , * m . u . midi ) ;
2021-05-10 12:37:29 +00:00
break ;
default :
rc = kOkRC ;
}
return rc ;
}
2022-01-22 14:42:21 +00:00
unsigned cw : : midi_record_play : : device_count ( handle_t h )
{
midi_record_play_t * p = _handleToPtr ( h ) ;
return p - > midiDevN ;
}
bool cw : : midi_record_play : : is_device_enabled ( handle_t h , unsigned devIdx )
{
midi_record_play_t * p = _handleToPtr ( h ) ;
bool fl = false ;
if ( devIdx > = p - > midiDevN )
cwLogError ( kInvalidArgRC , " The MIDI record-play device index '%i' is invalid. " , devIdx ) ;
else
fl = p - > midiDevA [ devIdx ] . enableFl ;
return fl ;
}
2021-08-15 20:02:45 +00:00
2022-01-22 14:42:21 +00:00
void cw : : midi_record_play : : enable_device ( handle_t h , unsigned devIdx , bool enableFl )
{
midi_record_play_t * p = _handleToPtr ( h ) ;
if ( devIdx > = p - > midiDevN )
cwLogError ( kInvalidArgRC , " The MIDI record-play device index '%i' is invalid. " , devIdx ) ;
else
{
p - > midiDevA [ devIdx ] . enableFl = enableFl ;
printf ( " Enable: %i = %i \n " , devIdx , enableFl ) ;
}
}
2021-08-15 20:02:45 +00:00
2022-03-20 14:27:46 +00:00
void cw : : midi_record_play : : half_pedal_params ( handle_t h , unsigned noteDelayMs , unsigned pitch , unsigned vel , unsigned pedal_vel , unsigned noteDurMs , unsigned downDelayMs )
{
midi_record_play_t * p = _handleToPtr ( h ) ;
p - > halfPedalNoteDelayUs = noteDelayMs * 1000 ;
p - > halfPedalNoteDurUs = noteDurMs * 1000 ;
p - > halfPedalDownDelayUs = downDelayMs * 1000 ;
p - > halfPedalMidiPitch = pitch ;
p - > halfPedalMidiNoteVel = vel ;
p - > halfPedalMidiPedalVel = pedal_vel ;
}
2021-08-15 20:02:45 +00:00
cw : : rc_t cw : : midi_record_play : : am_to_midi_file ( const char * am_filename , const char * midi_filename )
{
rc_t rc = kOkRC ;
unsigned msgArrayCnt = 0 ;
am_midi_msg_t * msgArray = nullptr ;
2022-10-01 23:00:33 +00:00
if ( ! filesys : : isFile ( am_filename ) )
{
cwLogError ( kOpenFailRC , " The AM file '%s' does not exist. " , am_filename ) ;
goto errLabel ;
}
2021-08-15 20:02:45 +00:00
if ( ( rc = _am_file_read ( am_filename , msgArrayCnt , msgArray ) ) ! = kOkRC )
{
rc = cwLogError ( rc , " Unable to read AM file '%s'. " , cwStringNullGuard ( am_filename ) ) ;
goto errLabel ;
}
if ( ( rc = _midi_file_write ( midi_filename , msgArray , msgArrayCnt ) ) ! = kOkRC )
{
rc = cwLogError ( rc , " Unable to write AM file '%s' to '%s'. " , cwStringNullGuard ( am_filename ) , cwStringNullGuard ( midi_filename ) ) ;
goto errLabel ;
}
errLabel :
mem : : release ( msgArray ) ;
return rc ;
}
cw : : rc_t cw : : midi_record_play : : am_to_midi_dir ( const char * inDir )
{
rc_t rc = kOkRC ;
filesys : : dirEntry_t * dirEntryArray = nullptr ;
unsigned dirEntryCnt = 0 ;
2022-09-10 12:54:17 +00:00
if ( ( dirEntryArray = filesys : : dirEntries ( inDir , filesys : : kDirFsFl | filesys : : kFullPathFsFl , & dirEntryCnt ) ) = = nullptr )
2021-08-15 20:02:45 +00:00
goto errLabel ;
2022-09-10 12:54:17 +00:00
for ( unsigned i = 0 ; i < dirEntryCnt and rc = = kOkRC ; + + i )
2021-08-15 20:02:45 +00:00
{
2022-09-10 12:54:17 +00:00
char * am_fn = filesys : : makeFn ( dirEntryArray [ i ] . name , " midi " , " am " , NULL ) ;
char * midi_fn = filesys : : makeFn ( dirEntryArray [ i ] . name , " midi " , " mid " , NULL ) ;
cwLogInfo ( " 0x%x AM:%s MIDI:%s " , dirEntryArray [ i ] . flags , dirEntryArray [ i ] . name , midi_fn ) ;
rc = am_to_midi_file ( am_fn , midi_fn ) ;
mem : : release ( am_fn ) ;
mem : : release ( midi_fn ) ;
2021-08-15 20:02:45 +00:00
}
errLabel :
mem : : release ( dirEntryArray ) ;
return rc ;
}
cw : : rc_t cw : : midi_record_play : : am_to_midi_file ( const object_t * cfg )
{
rc_t rc = kOkRC ;
const char * inDir = nullptr ;
//
if ( cfg = = nullptr )
{
rc = cwLogError ( kInvalidArgRC , " AM to MIDI file: No input directory. " ) ;
goto errLabel ;
}
//
if ( ( rc = cfg - > getv ( " inDir " , inDir ) ) ! = kOkRC )
{
rc = cwLogError ( rc , " AM to MIDI file: Unable to parse input arg's. " ) ;
goto errLabel ;
}
//
if ( ( rc = am_to_midi_dir ( inDir ) ) ! = kOkRC )
{
rc = cwLogError ( rc , " AM to MIDI file conversion on directory:'%s' failed. " , cwStringNullGuard ( inDir ) ) ;
goto errLabel ;
}
errLabel :
return rc ;
}
2022-10-15 19:07:06 +00:00
void cw : : midi_record_play : : report ( handle_t h )
{
midi_record_play_t * p = _handleToPtr ( h ) ;
_report_midi ( p ) ;
}