373 lines
		
	
	
		
			9.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			373 lines
		
	
	
		
			9.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
#include <string.h>
 | 
						||
#include <stdlib.h>
 | 
						||
#include "rtmidi_c.h"
 | 
						||
#include "RtMidi.h"
 | 
						||
 | 
						||
/* Compile-time assertions that will break if the enums are changed in
 | 
						||
 * the future without synchronizing them properly.  If you get (g++)
 | 
						||
 * "error: ‘StaticAssert<b>::StaticAssert() [with bool b = false]’ is
 | 
						||
 * private within this context", it means enums are not aligned. */
 | 
						||
template<bool b> class StaticAssert { private: StaticAssert() {} };
 | 
						||
template<> class StaticAssert<true>{ public: StaticAssert() {} };
 | 
						||
#define ENUM_EQUAL(x,y) StaticAssert<(int)x==(int)y>()
 | 
						||
class StaticAssertions { StaticAssertions() {
 | 
						||
    ENUM_EQUAL( RTMIDI_API_UNSPECIFIED,     RtMidi::UNSPECIFIED );
 | 
						||
    ENUM_EQUAL( RTMIDI_API_MACOSX_CORE,     RtMidi::MACOSX_CORE );
 | 
						||
    ENUM_EQUAL( RTMIDI_API_LINUX_ALSA,      RtMidi::LINUX_ALSA );
 | 
						||
    ENUM_EQUAL( RTMIDI_API_UNIX_JACK,       RtMidi::UNIX_JACK );
 | 
						||
    ENUM_EQUAL( RTMIDI_API_WINDOWS_MM,      RtMidi::WINDOWS_MM );
 | 
						||
    ENUM_EQUAL( RTMIDI_API_RTMIDI_DUMMY,    RtMidi::RTMIDI_DUMMY );
 | 
						||
 | 
						||
    ENUM_EQUAL( RTMIDI_ERROR_WARNING,            RtMidiError::WARNING );
 | 
						||
    ENUM_EQUAL( RTMIDI_ERROR_DEBUG_WARNING,      RtMidiError::DEBUG_WARNING );
 | 
						||
    ENUM_EQUAL( RTMIDI_ERROR_UNSPECIFIED,        RtMidiError::UNSPECIFIED );
 | 
						||
    ENUM_EQUAL( RTMIDI_ERROR_NO_DEVICES_FOUND,   RtMidiError::NO_DEVICES_FOUND );
 | 
						||
    ENUM_EQUAL( RTMIDI_ERROR_INVALID_DEVICE,     RtMidiError::INVALID_DEVICE );
 | 
						||
    ENUM_EQUAL( RTMIDI_ERROR_MEMORY_ERROR,       RtMidiError::MEMORY_ERROR );
 | 
						||
    ENUM_EQUAL( RTMIDI_ERROR_INVALID_PARAMETER,  RtMidiError::INVALID_PARAMETER );
 | 
						||
    ENUM_EQUAL( RTMIDI_ERROR_INVALID_USE,        RtMidiError::INVALID_USE );
 | 
						||
    ENUM_EQUAL( RTMIDI_ERROR_DRIVER_ERROR,       RtMidiError::DRIVER_ERROR );
 | 
						||
    ENUM_EQUAL( RTMIDI_ERROR_SYSTEM_ERROR,       RtMidiError::SYSTEM_ERROR );
 | 
						||
    ENUM_EQUAL( RTMIDI_ERROR_THREAD_ERROR,       RtMidiError::THREAD_ERROR );
 | 
						||
}};
 | 
						||
 | 
						||
class CallbackProxyUserData
 | 
						||
{
 | 
						||
  public:
 | 
						||
  CallbackProxyUserData (RtMidiCCallback cCallback, void *userData)
 | 
						||
    : c_callback (cCallback), user_data (userData)
 | 
						||
  {
 | 
						||
  }
 | 
						||
  RtMidiCCallback c_callback;
 | 
						||
  void *user_data;
 | 
						||
};
 | 
						||
 | 
						||
extern "C" const enum RtMidiApi rtmidi_compiled_apis[]; // casting from RtMidi::Api[]
 | 
						||
extern "C" const unsigned int rtmidi_num_compiled_apis;
 | 
						||
 | 
						||
/* RtMidi API */
 | 
						||
int rtmidi_get_compiled_api (enum RtMidiApi *apis, unsigned int apis_size)
 | 
						||
{
 | 
						||
    unsigned num = rtmidi_num_compiled_apis;
 | 
						||
    if (apis) {
 | 
						||
        num = (num < apis_size) ? num : apis_size;
 | 
						||
        memcpy(apis, rtmidi_compiled_apis, num * sizeof(enum RtMidiApi));
 | 
						||
    }
 | 
						||
    return (int)num;
 | 
						||
}
 | 
						||
 | 
						||
extern "C" const char* rtmidi_api_names[][2];
 | 
						||
const char *rtmidi_api_name(enum RtMidiApi api) {
 | 
						||
    if (api < 0 || api >= RTMIDI_API_NUM)
 | 
						||
        return NULL;
 | 
						||
    return rtmidi_api_names[api][0];
 | 
						||
}
 | 
						||
 | 
						||
const char *rtmidi_api_display_name(enum RtMidiApi api)
 | 
						||
{
 | 
						||
    if (api < 0 || api >= RTMIDI_API_NUM)
 | 
						||
        return "Unknown";
 | 
						||
    return rtmidi_api_names[api][1];
 | 
						||
}
 | 
						||
 | 
						||
enum RtMidiApi rtmidi_compiled_api_by_name(const char *name) {
 | 
						||
    RtMidi::Api api = RtMidi::UNSPECIFIED;
 | 
						||
    if (name) {
 | 
						||
        api = RtMidi::getCompiledApiByName(name);
 | 
						||
    }
 | 
						||
    return (enum RtMidiApi)api;
 | 
						||
}
 | 
						||
 | 
						||
void rtmidi_error (MidiApi *api, enum RtMidiErrorType type, const char* errorString)
 | 
						||
{
 | 
						||
  std::string msg = errorString;
 | 
						||
  api->error ((RtMidiError::Type) type, msg);
 | 
						||
}
 | 
						||
 | 
						||
void rtmidi_open_port (RtMidiPtr device, unsigned int portNumber, const char *portName)
 | 
						||
{
 | 
						||
    std::string name = portName;
 | 
						||
    try {
 | 
						||
        ((RtMidi*) device->ptr)->openPort (portNumber, name);
 | 
						||
 | 
						||
    } catch (const RtMidiError & err) {
 | 
						||
        device->ok  = false;
 | 
						||
        device->msg = err.what ();
 | 
						||
    }
 | 
						||
}
 | 
						||
 | 
						||
void rtmidi_open_virtual_port (RtMidiPtr device, const char *portName)
 | 
						||
{
 | 
						||
    std::string name = portName;
 | 
						||
    try {
 | 
						||
        ((RtMidi*) device->ptr)->openVirtualPort (name);
 | 
						||
 | 
						||
    } catch (const RtMidiError & err) {
 | 
						||
        device->ok  = false;
 | 
						||
        device->msg = err.what ();
 | 
						||
    }
 | 
						||
 | 
						||
}
 | 
						||
 | 
						||
void rtmidi_close_port (RtMidiPtr device)
 | 
						||
{
 | 
						||
    try {
 | 
						||
        ((RtMidi*) device->ptr)->closePort ();
 | 
						||
 | 
						||
    } catch (const RtMidiError & err) {
 | 
						||
        device->ok  = false;
 | 
						||
        device->msg = err.what ();
 | 
						||
    }
 | 
						||
}
 | 
						||
 | 
						||
unsigned int rtmidi_get_port_count (RtMidiPtr device)
 | 
						||
{
 | 
						||
    try {
 | 
						||
        return ((RtMidi*) device->ptr)->getPortCount ();
 | 
						||
 | 
						||
    } catch (const RtMidiError & err) {
 | 
						||
        device->ok  = false;
 | 
						||
        device->msg = err.what ();
 | 
						||
        return -1;
 | 
						||
    }
 | 
						||
}
 | 
						||
 | 
						||
int rtmidi_get_port_name (RtMidiPtr device, unsigned int portNumber, char * bufOut, int * bufLen)
 | 
						||
{
 | 
						||
    if (bufOut == nullptr && bufLen == nullptr) {
 | 
						||
        return -1;
 | 
						||
    }
 | 
						||
 | 
						||
    std::string name;
 | 
						||
    try {
 | 
						||
        name = ((RtMidi*) device->ptr)->getPortName (portNumber);
 | 
						||
    } catch (const RtMidiError & err) {
 | 
						||
        device->ok  = false;
 | 
						||
        device->msg = err.what ();
 | 
						||
        return -1;
 | 
						||
    }
 | 
						||
 | 
						||
    if (bufOut == nullptr) {
 | 
						||
        *bufLen = static_cast<int>(name.size()) + 1;
 | 
						||
        return 0;
 | 
						||
    }
 | 
						||
 | 
						||
    return snprintf(bufOut, static_cast<size_t>(*bufLen), "%s", name.c_str());
 | 
						||
}
 | 
						||
 | 
						||
/* RtMidiIn API */
 | 
						||
RtMidiInPtr rtmidi_in_create_default ()
 | 
						||
{
 | 
						||
    RtMidiWrapper* wrp = new RtMidiWrapper;
 | 
						||
 | 
						||
    try {
 | 
						||
        RtMidiIn* rIn = new RtMidiIn ();
 | 
						||
 | 
						||
        wrp->ptr = (void*) rIn;
 | 
						||
        wrp->data = 0;
 | 
						||
        wrp->ok  = true;
 | 
						||
        wrp->msg = "";
 | 
						||
 | 
						||
    } catch (const RtMidiError & err) {
 | 
						||
        wrp->ptr = 0;
 | 
						||
        wrp->data = 0;
 | 
						||
        wrp->ok  = false;
 | 
						||
        wrp->msg = err.what ();
 | 
						||
    }
 | 
						||
 | 
						||
    return wrp;
 | 
						||
}
 | 
						||
 | 
						||
RtMidiInPtr rtmidi_in_create (enum RtMidiApi api, const char *clientName, unsigned int queueSizeLimit)
 | 
						||
{
 | 
						||
    std::string name = clientName;
 | 
						||
    RtMidiWrapper* wrp = new RtMidiWrapper;
 | 
						||
 | 
						||
    try {
 | 
						||
        RtMidiIn* rIn = new RtMidiIn ((RtMidi::Api) api, name, queueSizeLimit);
 | 
						||
 | 
						||
        wrp->ptr = (void*) rIn;
 | 
						||
        wrp->data = 0;
 | 
						||
        wrp->ok  = true;
 | 
						||
        wrp->msg = "";
 | 
						||
 | 
						||
    } catch (const RtMidiError & err) {
 | 
						||
        wrp->ptr = 0;
 | 
						||
        wrp->data = 0;
 | 
						||
        wrp->ok  = false;
 | 
						||
        wrp->msg = err.what ();
 | 
						||
    }
 | 
						||
 | 
						||
    return wrp;
 | 
						||
}
 | 
						||
 | 
						||
void rtmidi_in_free (RtMidiInPtr device)
 | 
						||
{
 | 
						||
    if (device->data)
 | 
						||
      delete (CallbackProxyUserData*) device->data;
 | 
						||
    delete (RtMidiIn*) device->ptr;
 | 
						||
    delete device;
 | 
						||
}
 | 
						||
 | 
						||
enum RtMidiApi rtmidi_in_get_current_api (RtMidiPtr device)
 | 
						||
{
 | 
						||
    try {
 | 
						||
        return (RtMidiApi) ((RtMidiIn*) device->ptr)->getCurrentApi ();
 | 
						||
 | 
						||
    } catch (const RtMidiError & err) {
 | 
						||
        device->ok  = false;
 | 
						||
        device->msg = err.what ();
 | 
						||
 | 
						||
        return RTMIDI_API_UNSPECIFIED;
 | 
						||
    }
 | 
						||
}
 | 
						||
 | 
						||
static
 | 
						||
void callback_proxy (double timeStamp, std::vector<unsigned char> *message, void *userData)
 | 
						||
{
 | 
						||
  CallbackProxyUserData* data = reinterpret_cast<CallbackProxyUserData*> (userData);
 | 
						||
  data->c_callback (timeStamp, message->data (), message->size (), data->user_data);
 | 
						||
}
 | 
						||
 | 
						||
void rtmidi_in_set_callback (RtMidiInPtr device, RtMidiCCallback callback, void *userData)
 | 
						||
{
 | 
						||
    device->data = (void*) new CallbackProxyUserData (callback, userData);
 | 
						||
    try {
 | 
						||
        ((RtMidiIn*) device->ptr)->setCallback (callback_proxy, device->data);
 | 
						||
    } catch (const RtMidiError & err) {
 | 
						||
        device->ok  = false;
 | 
						||
        device->msg = err.what ();
 | 
						||
        delete (CallbackProxyUserData*) device->data;
 | 
						||
        device->data = 0;
 | 
						||
    }
 | 
						||
}
 | 
						||
 | 
						||
void rtmidi_in_cancel_callback (RtMidiInPtr device)
 | 
						||
{
 | 
						||
    try {
 | 
						||
        ((RtMidiIn*) device->ptr)->cancelCallback ();
 | 
						||
        delete (CallbackProxyUserData*) device->data;
 | 
						||
        device->data = 0;
 | 
						||
    } catch (const RtMidiError & err) {
 | 
						||
        device->ok  = false;
 | 
						||
        device->msg = err.what ();
 | 
						||
    }
 | 
						||
}
 | 
						||
 | 
						||
void rtmidi_in_ignore_types (RtMidiInPtr device, bool midiSysex, bool midiTime, bool midiSense)
 | 
						||
{
 | 
						||
  ((RtMidiIn*) device->ptr)->ignoreTypes (midiSysex, midiTime, midiSense);
 | 
						||
}
 | 
						||
 | 
						||
double rtmidi_in_get_message (RtMidiInPtr device,
 | 
						||
                              unsigned char *message,
 | 
						||
                              size_t *size)
 | 
						||
{
 | 
						||
    try {
 | 
						||
        // FIXME: use allocator to achieve efficient buffering
 | 
						||
        std::vector<unsigned char> v;
 | 
						||
        double ret = ((RtMidiIn*) device->ptr)->getMessage (&v);
 | 
						||
 | 
						||
        if (v.size () > 0 && v.size() <= *size) {
 | 
						||
            memcpy (message, v.data (), (int) v.size ());
 | 
						||
        }
 | 
						||
 | 
						||
        *size = v.size();
 | 
						||
        return ret;
 | 
						||
    }
 | 
						||
    catch (const RtMidiError & err) {
 | 
						||
        device->ok  = false;
 | 
						||
        device->msg = err.what ();
 | 
						||
        return -1;
 | 
						||
    }
 | 
						||
    catch (...) {
 | 
						||
        device->ok  = false;
 | 
						||
        device->msg = "Unknown error";
 | 
						||
        return -1;
 | 
						||
    }
 | 
						||
}
 | 
						||
 | 
						||
/* RtMidiOut API */
 | 
						||
RtMidiOutPtr rtmidi_out_create_default ()
 | 
						||
{
 | 
						||
    RtMidiWrapper* wrp = new RtMidiWrapper;
 | 
						||
 | 
						||
    try {
 | 
						||
        RtMidiOut* rOut = new RtMidiOut ();
 | 
						||
 | 
						||
        wrp->ptr = (void*) rOut;
 | 
						||
        wrp->data = 0;
 | 
						||
        wrp->ok  = true;
 | 
						||
        wrp->msg = "";
 | 
						||
 | 
						||
    } catch (const RtMidiError & err) {
 | 
						||
        wrp->ptr = 0;
 | 
						||
        wrp->data = 0;
 | 
						||
        wrp->ok  = false;
 | 
						||
        wrp->msg = err.what ();
 | 
						||
    }
 | 
						||
 | 
						||
    return wrp;
 | 
						||
}
 | 
						||
 | 
						||
RtMidiOutPtr rtmidi_out_create (enum RtMidiApi api, const char *clientName)
 | 
						||
{
 | 
						||
    RtMidiWrapper* wrp = new RtMidiWrapper;
 | 
						||
    std::string name = clientName;
 | 
						||
 | 
						||
    try {
 | 
						||
        RtMidiOut* rOut = new RtMidiOut ((RtMidi::Api) api, name);
 | 
						||
 | 
						||
        wrp->ptr = (void*) rOut;
 | 
						||
        wrp->data = 0;
 | 
						||
        wrp->ok  = true;
 | 
						||
        wrp->msg = "";
 | 
						||
 | 
						||
    } catch (const RtMidiError & err) {
 | 
						||
        wrp->ptr = 0;
 | 
						||
        wrp->data = 0;
 | 
						||
        wrp->ok  = false;
 | 
						||
        wrp->msg = err.what ();
 | 
						||
    }
 | 
						||
 | 
						||
 | 
						||
    return wrp;
 | 
						||
}
 | 
						||
 | 
						||
void rtmidi_out_free (RtMidiOutPtr device)
 | 
						||
{
 | 
						||
    delete (RtMidiOut*) device->ptr;
 | 
						||
    delete device;
 | 
						||
}
 | 
						||
 | 
						||
enum RtMidiApi rtmidi_out_get_current_api (RtMidiPtr device)
 | 
						||
{
 | 
						||
    try {
 | 
						||
        return (RtMidiApi) ((RtMidiOut*) device->ptr)->getCurrentApi ();
 | 
						||
 | 
						||
    } catch (const RtMidiError & err) {
 | 
						||
        device->ok  = false;
 | 
						||
        device->msg = err.what ();
 | 
						||
 | 
						||
        return RTMIDI_API_UNSPECIFIED;
 | 
						||
    }
 | 
						||
}
 | 
						||
 | 
						||
int rtmidi_out_send_message (RtMidiOutPtr device, const unsigned char *message, int length)
 | 
						||
{
 | 
						||
    try {
 | 
						||
        ((RtMidiOut*) device->ptr)->sendMessage (message, length);
 | 
						||
        return 0;
 | 
						||
    }
 | 
						||
    catch (const RtMidiError & err) {
 | 
						||
        device->ok  = false;
 | 
						||
        device->msg = err.what ();
 | 
						||
        return -1;
 | 
						||
    }
 | 
						||
    catch (...) {
 | 
						||
        device->ok  = false;
 | 
						||
        device->msg = "Unknown error";
 | 
						||
        return -1;
 | 
						||
    }
 | 
						||
}
 |