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-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;
|
|
|
|
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
|
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-03-20 14:27:46 +00:00
|
|
|
if((rc = ele->getv_opt( "vel_table", velTable,
|
|
|
|
"pedal", pedalRecd)) != kOkRC )
|
|
|
|
{
|
|
|
|
rc = cwLogError(kSyntaxErrorRC,"MIDI record play device optional argument parsing failed.");
|
|
|
|
goto errLabel;
|
|
|
|
}
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
// 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;
|
|
|
|
bool supress_fl = is_note_on_fl && after_stop_time_fl;
|
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 ];
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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;
|
|
|
|
else
|
|
|
|
cwLogError(kInvalidIdRC,"Unexpected pedal down velocity (%i) during pedal velocity mapping.",d1);
|
2022-03-20 14:27:46 +00:00
|
|
|
}
|
|
|
|
}
|
2022-05-14 14:22:29 +00:00
|
|
|
|
|
|
|
if( !supress_fl )
|
|
|
|
io::midiDeviceSend( p->ioH, p->midiDevA[i].midiOutDevIdx, p->midiDevA[i].midiOutPortIdx, status + ch, d0, out_d1 );
|
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;
|
|
|
|
bool print_fl = false;
|
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 )
|
|
|
|
{
|
|
|
|
for(unsigned i=0; i<p->msgArrayInIdx; ++i)
|
|
|
|
{
|
|
|
|
am_midi_msg_t* mm = p->msgArray + i;
|
|
|
|
_print_midi_msg(mm);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
rc_t _stop( midi_record_play_t* p )
|
|
|
|
{
|
|
|
|
rc_t rc = kOkRC;
|
|
|
|
|
|
|
|
p->startedFl = false;
|
|
|
|
|
|
|
|
time::spec_t t1;
|
|
|
|
time::get(t1);
|
|
|
|
|
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-03-20 14:27:46 +00:00
|
|
|
p->msgArray[i].microsec = time::elapsedMicros(p->iMsgArray[0].timestamp,p->iMsgArray[i].timestamp);
|
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 )
|
|
|
|
{
|
|
|
|
}
|
|
|
|
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-03-20 14:27:46 +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;
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
}
|