comparison data/midi/rtmidi/RtMidi.h @ 1527:710e6250a401 zoom

Merge from default branch
author Chris Cannam
date Mon, 17 Sep 2018 13:51:14 +0100
parents 08bbbd2023c4
children
comparison
equal deleted inserted replaced
1324:d4a28d1479a8 1527:710e6250a401
6 MIDI input/output subclasses RtMidiIn and RtMidiOut. 6 MIDI input/output subclasses RtMidiIn and RtMidiOut.
7 7
8 RtMidi WWW site: http://music.mcgill.ca/~gary/rtmidi/ 8 RtMidi WWW site: http://music.mcgill.ca/~gary/rtmidi/
9 9
10 RtMidi: realtime MIDI i/o C++ classes 10 RtMidi: realtime MIDI i/o C++ classes
11 Copyright (c) 2003-2009 Gary P. Scavone 11 Copyright (c) 2003-2016 Gary P. Scavone
12 12
13 Permission is hereby granted, free of charge, to any person 13 Permission is hereby granted, free of charge, to any person
14 obtaining a copy of this software and associated documentation files 14 obtaining a copy of this software and associated documentation files
15 (the "Software"), to deal in the Software without restriction, 15 (the "Software"), to deal in the Software without restriction,
16 including without limitation the rights to use, copy, modify, merge, 16 including without limitation the rights to use, copy, modify, merge,
20 20
21 The above copyright notice and this permission notice shall be 21 The above copyright notice and this permission notice shall be
22 included in all copies or substantial portions of the Software. 22 included in all copies or substantial portions of the Software.
23 23
24 Any person wishing to distribute modifications to the Software is 24 Any person wishing to distribute modifications to the Software is
25 requested to send the modifications to the original developer so that 25 asked to send the modifications to the original developer so that
26 they can be incorporated into the canonical version. 26 they can be incorporated into the canonical version. This is,
27 however, not a binding provision of this license.
27 28
28 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 29 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
29 EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 30 EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
30 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. 31 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
31 IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR 32 IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR
33 CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION 34 CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
34 WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 35 WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
35 */ 36 */
36 /**********************************************************************/ 37 /**********************************************************************/
37 38
38 // RtMidi: Version 1.0.8 39 /*!
40 \file RtMidi.h
41 */
39 42
40 #ifndef RTMIDI_H 43 #ifndef RTMIDI_H
41 #define RTMIDI_H 44 #define RTMIDI_H
42 45
43 #include "RtError.h" 46 #define RTMIDI_VERSION "2.1.1"
47
48 #include <exception>
49 #include <iostream>
44 #include <string> 50 #include <string>
51 #include <vector>
52
53 /************************************************************************/
54 /*! \class RtMidiError
55 \brief Exception handling class for RtMidi.
56
57 The RtMidiError class is quite simple but it does allow errors to be
58 "caught" by RtMidiError::Type. See the RtMidi documentation to know
59 which methods can throw an RtMidiError.
60 */
61 /************************************************************************/
62
63 class RtMidiError : public std::exception
64 {
65 public:
66 //! Defined RtMidiError types.
67 enum Type {
68 WARNING, /*!< A non-critical error. */
69 DEBUG_WARNING, /*!< A non-critical error which might be useful for debugging. */
70 UNSPECIFIED, /*!< The default, unspecified error type. */
71 NO_DEVICES_FOUND, /*!< No devices found on system. */
72 INVALID_DEVICE, /*!< An invalid device ID was specified. */
73 MEMORY_ERROR, /*!< An error occured during memory allocation. */
74 INVALID_PARAMETER, /*!< An invalid parameter was specified to a function. */
75 INVALID_USE, /*!< The function was called incorrectly. */
76 DRIVER_ERROR, /*!< A system driver error occured. */
77 SYSTEM_ERROR, /*!< A system error occured. */
78 THREAD_ERROR /*!< A thread error occured. */
79 };
80
81 //! The constructor.
82 RtMidiError( const std::string& message, Type type = RtMidiError::UNSPECIFIED ) throw() : message_(message), type_(type) {}
83
84 //! The destructor.
85 virtual ~RtMidiError( void ) throw() {}
86
87 //! Prints thrown error message to stderr.
88 virtual void printMessage( void ) const throw() { std::cerr << '\n' << message_ << "\n\n"; }
89
90 //! Returns the thrown error message type.
91 virtual const Type& getType(void) const throw() { return type_; }
92
93 //! Returns the thrown error message string.
94 virtual const std::string& getMessage(void) const throw() { return message_; }
95
96 //! Returns the thrown error message as a c-style string.
97 virtual const char* what( void ) const throw() { return message_.c_str(); }
98
99 protected:
100 std::string message_;
101 Type type_;
102 };
103
104 //! RtMidi error callback function prototype.
105 /*!
106 \param type Type of error.
107 \param errorText Error description.
108
109 Note that class behaviour is undefined after a critical error (not
110 a warning) is reported.
111 */
112 typedef void (*RtMidiErrorCallback)( RtMidiError::Type type, const std::string &errorText, void *userData );
113
114 class MidiApi;
45 115
46 class RtMidi 116 class RtMidi
47 { 117 {
48 public: 118 public:
119
120 //! MIDI API specifier arguments.
121 enum Api {
122 UNSPECIFIED, /*!< Search for a working compiled API. */
123 MACOSX_CORE, /*!< Macintosh OS-X Core Midi API. */
124 LINUX_ALSA, /*!< The Advanced Linux Sound Architecture API. */
125 UNIX_JACK, /*!< The JACK Low-Latency MIDI Server API. */
126 WINDOWS_MM, /*!< The Microsoft Multimedia MIDI API. */
127 RTMIDI_DUMMY /*!< A compilable but non-functional API. */
128 };
129
130 //! A static function to determine the current RtMidi version.
131 static std::string getVersion( void ) throw();
132
133 //! A static function to determine the available compiled MIDI APIs.
134 /*!
135 The values returned in the std::vector can be compared against
136 the enumerated list values. Note that there can be more than one
137 API compiled for certain operating systems.
138 */
139 static void getCompiledApi( std::vector<RtMidi::Api> &apis ) throw();
49 140
50 //! Pure virtual openPort() function. 141 //! Pure virtual openPort() function.
51 virtual void openPort( unsigned int portNumber = 0, const std::string portName = std::string( "RtMidi" ) ) = 0; 142 virtual void openPort( unsigned int portNumber = 0, const std::string portName = std::string( "RtMidi" ) ) = 0;
52 143
53 //! Pure virtual openVirtualPort() function. 144 //! Pure virtual openVirtualPort() function.
60 virtual std::string getPortName( unsigned int portNumber = 0 ) = 0; 151 virtual std::string getPortName( unsigned int portNumber = 0 ) = 0;
61 152
62 //! Pure virtual closePort() function. 153 //! Pure virtual closePort() function.
63 virtual void closePort( void ) = 0; 154 virtual void closePort( void ) = 0;
64 155
156 //! Returns true if a port is open and false if not.
157 virtual bool isPortOpen( void ) const = 0;
158
159 //! Set an error callback function to be invoked when an error has occured.
160 /*!
161 The callback function will be called whenever an error has occured. It is best
162 to set the error callback function before opening a port.
163 */
164 virtual void setErrorCallback( RtMidiErrorCallback errorCallback = NULL, void *userData = 0 ) = 0;
165
65 protected: 166 protected:
66 167
67 RtMidi(); 168 RtMidi();
68 virtual ~RtMidi() {}; 169 virtual ~RtMidi();
69 170
70 // A basic error reporting function for internal use in the RtMidi 171 MidiApi *rtapi_;
71 // subclasses. The behavior of this function can be modified to
72 // suit specific needs.
73 void error( RtError::Type type );
74
75 void *apiData_;
76 bool connected_;
77 std::string errorString_;
78 }; 172 };
79 173
80 /**********************************************************************/ 174 /**********************************************************************/
81 /*! \class RtMidiIn 175 /*! \class RtMidiIn
82 \brief A realtime MIDI input class. 176 \brief A realtime MIDI input class.
85 realtime MIDI input. It allows access to a single MIDI input 179 realtime MIDI input. It allows access to a single MIDI input
86 port. Incoming MIDI messages are either saved to a queue for 180 port. Incoming MIDI messages are either saved to a queue for
87 retrieval using the getMessage() function or immediately passed to 181 retrieval using the getMessage() function or immediately passed to
88 a user-specified callback function. Create multiple instances of 182 a user-specified callback function. Create multiple instances of
89 this class to connect to more than one MIDI device at the same 183 this class to connect to more than one MIDI device at the same
90 time. With the OS-X and Linux ALSA MIDI APIs, it is also possible 184 time. With the OS-X, Linux ALSA, and JACK MIDI APIs, it is also
91 to open a virtual input port to which other MIDI software clients 185 possible to open a virtual input port to which other MIDI software
92 can connect. 186 clients can connect.
93 187
94 by Gary P. Scavone, 2003-2008. 188 by Gary P. Scavone, 2003-2014.
95 */ 189 */
96 /**********************************************************************/ 190 /**********************************************************************/
97 191
98 #include <vector> 192 // **************************************************************** //
99 #include <queue> 193 //
194 // RtMidiIn and RtMidiOut class declarations.
195 //
196 // RtMidiIn / RtMidiOut are "controllers" used to select an available
197 // MIDI input or output interface. They present common APIs for the
198 // user to call but all functionality is implemented by the classes
199 // MidiInApi, MidiOutApi and their subclasses. RtMidiIn and RtMidiOut
200 // each create an instance of a MidiInApi or MidiOutApi subclass based
201 // on the user's API choice. If no choice is made, they attempt to
202 // make a "logical" API selection.
203 //
204 // **************************************************************** //
100 205
101 class RtMidiIn : public RtMidi 206 class RtMidiIn : public RtMidi
102 { 207 {
103 public: 208 public:
104 209
105 //! User callback function type definition. 210 //! User callback function type definition.
106 typedef void (*RtMidiCallback)( double timeStamp, std::vector<unsigned char> *message, void *userData); 211 typedef void (*RtMidiCallback)( double timeStamp, std::vector<unsigned char> *message, void *userData);
107 212
213 //! Default constructor that allows an optional api, client name and queue size.
214 /*!
215 An exception will be thrown if a MIDI system initialization
216 error occurs. The queue size defines the maximum number of
217 messages that can be held in the MIDI queue (when not using a
218 callback function). If the queue size limit is reached,
219 incoming messages will be ignored.
220
221 If no API argument is specified and multiple API support has been
222 compiled, the default order of use is ALSA, JACK (Linux) and CORE,
223 JACK (OS-X).
224
225 \param api An optional API id can be specified.
226 \param clientName An optional client name can be specified. This
227 will be used to group the ports that are created
228 by the application.
229 \param queueSizeLimit An optional size of the MIDI input queue can be specified.
230 */
231 RtMidiIn( RtMidi::Api api=UNSPECIFIED,
232 const std::string clientName = std::string( "RtMidi Input Client"),
233 unsigned int queueSizeLimit = 100 );
234
235 //! If a MIDI connection is still open, it will be closed by the destructor.
236 ~RtMidiIn ( void ) throw();
237
238 //! Returns the MIDI API specifier for the current instance of RtMidiIn.
239 RtMidi::Api getCurrentApi( void ) throw();
240
241 //! Open a MIDI input connection given by enumeration number.
242 /*!
243 \param portNumber An optional port number greater than 0 can be specified.
244 Otherwise, the default or first port found is opened.
245 \param portName An optional name for the application port that is used to connect to portId can be specified.
246 */
247 void openPort( unsigned int portNumber = 0, const std::string portName = std::string( "RtMidi Input" ) );
248
249 //! Create a virtual input port, with optional name, to allow software connections (OS X, JACK and ALSA only).
250 /*!
251 This function creates a virtual MIDI input port to which other
252 software applications can connect. This type of functionality
253 is currently only supported by the Macintosh OS-X, any JACK,
254 and Linux ALSA APIs (the function returns an error for the other APIs).
255
256 \param portName An optional name for the application port that is
257 used to connect to portId can be specified.
258 */
259 void openVirtualPort( const std::string portName = std::string( "RtMidi Input" ) );
260
261 //! Set a callback function to be invoked for incoming MIDI messages.
262 /*!
263 The callback function will be called whenever an incoming MIDI
264 message is received. While not absolutely necessary, it is best
265 to set the callback function before opening a MIDI port to avoid
266 leaving some messages in the queue.
267
268 \param callback A callback function must be given.
269 \param userData Optionally, a pointer to additional data can be
270 passed to the callback function whenever it is called.
271 */
272 void setCallback( RtMidiCallback callback, void *userData = 0 );
273
274 //! Cancel use of the current callback function (if one exists).
275 /*!
276 Subsequent incoming MIDI messages will be written to the queue
277 and can be retrieved with the \e getMessage function.
278 */
279 void cancelCallback();
280
281 //! Close an open MIDI connection (if one exists).
282 void closePort( void );
283
284 //! Returns true if a port is open and false if not.
285 virtual bool isPortOpen() const;
286
287 //! Return the number of available MIDI input ports.
288 /*!
289 \return This function returns the number of MIDI ports of the selected API.
290 */
291 unsigned int getPortCount();
292
293 //! Return a string identifier for the specified MIDI input port number.
294 /*!
295 \return The name of the port with the given Id is returned.
296 \retval An empty string is returned if an invalid port specifier is provided.
297 */
298 std::string getPortName( unsigned int portNumber = 0 );
299
300 //! Specify whether certain MIDI message types should be queued or ignored during input.
301 /*!
302 By default, MIDI timing and active sensing messages are ignored
303 during message input because of their relative high data rates.
304 MIDI sysex messages are ignored by default as well. Variable
305 values of "true" imply that the respective message type will be
306 ignored.
307 */
308 void ignoreTypes( bool midiSysex = true, bool midiTime = true, bool midiSense = true );
309
310 //! Fill the user-provided vector with the data bytes for the next available MIDI message in the input queue and return the event delta-time in seconds.
311 /*!
312 This function returns immediately whether a new message is
313 available or not. A valid message is indicated by a non-zero
314 vector size. An exception is thrown if an error occurs during
315 message retrieval or an input connection was not previously
316 established.
317 */
318 double getMessage( std::vector<unsigned char> *message );
319
320 //! Set an error callback function to be invoked when an error has occured.
321 /*!
322 The callback function will be called whenever an error has occured. It is best
323 to set the error callback function before opening a port.
324 */
325 virtual void setErrorCallback( RtMidiErrorCallback errorCallback = NULL, void *userData = 0 );
326
327 protected:
328 void openMidiApi( RtMidi::Api api, const std::string clientName, unsigned int queueSizeLimit );
329
330 };
331
332 /**********************************************************************/
333 /*! \class RtMidiOut
334 \brief A realtime MIDI output class.
335
336 This class provides a common, platform-independent API for MIDI
337 output. It allows one to probe available MIDI output ports, to
338 connect to one such port, and to send MIDI bytes immediately over
339 the connection. Create multiple instances of this class to
340 connect to more than one MIDI device at the same time. With the
341 OS-X, Linux ALSA and JACK MIDI APIs, it is also possible to open a
342 virtual port to which other MIDI software clients can connect.
343
344 by Gary P. Scavone, 2003-2014.
345 */
346 /**********************************************************************/
347
348 class RtMidiOut : public RtMidi
349 {
350 public:
351
108 //! Default constructor that allows an optional client name. 352 //! Default constructor that allows an optional client name.
109 /*! 353 /*!
110 An exception will be thrown if a MIDI system initialization error occurs. 354 An exception will be thrown if a MIDI system initialization error occurs.
111 */ 355
112 RtMidiIn( const std::string clientName = std::string( "RtMidi Input Client") ); 356 If no API argument is specified and multiple API support has been
113 357 compiled, the default order of use is ALSA, JACK (Linux) and CORE,
114 //! If a MIDI connection is still open, it will be closed by the destructor. 358 JACK (OS-X).
115 ~RtMidiIn(); 359 */
116 360 RtMidiOut( RtMidi::Api api=UNSPECIFIED,
117 //! Open a MIDI input connection. 361 const std::string clientName = std::string( "RtMidi Output Client") );
362
363 //! The destructor closes any open MIDI connections.
364 ~RtMidiOut( void ) throw();
365
366 //! Returns the MIDI API specifier for the current instance of RtMidiOut.
367 RtMidi::Api getCurrentApi( void ) throw();
368
369 //! Open a MIDI output connection.
118 /*! 370 /*!
119 An optional port number greater than 0 can be specified. 371 An optional port number greater than 0 can be specified.
120 Otherwise, the default or first port found is opened. 372 Otherwise, the default or first port found is opened. An
121 */ 373 exception is thrown if an error occurs while attempting to make
122 void openPort( unsigned int portNumber = 0, const std::string Portname = std::string( "RtMidi Input" ) ); 374 the port connection.
123 375 */
124 //! Create a virtual input port, with optional name, to allow software connections (OS X and ALSA only). 376 void openPort( unsigned int portNumber = 0, const std::string portName = std::string( "RtMidi Output" ) );
125 /*! 377
126 This function creates a virtual MIDI input port to which other 378 //! Close an open MIDI connection (if one exists).
379 void closePort( void );
380
381 //! Returns true if a port is open and false if not.
382 virtual bool isPortOpen() const;
383
384 //! Create a virtual output port, with optional name, to allow software connections (OS X, JACK and ALSA only).
385 /*!
386 This function creates a virtual MIDI output port to which other
127 software applications can connect. This type of functionality 387 software applications can connect. This type of functionality
128 is currently only supported by the Macintosh OS-X and Linux ALSA 388 is currently only supported by the Macintosh OS-X, Linux ALSA
129 APIs (the function does nothing for the other APIs). 389 and JACK APIs (the function does nothing with the other APIs).
130 */ 390 An exception is thrown if an error occurs while attempting to
131 void openVirtualPort( const std::string portName = std::string( "RtMidi Input" ) ); 391 create the virtual port.
132 392 */
133 //! Set a callback function to be invoked for incoming MIDI messages. 393 void openVirtualPort( const std::string portName = std::string( "RtMidi Output" ) );
134 /*! 394
135 The callback function will be called whenever an incoming MIDI 395 //! Return the number of available MIDI output ports.
136 message is received. While not absolutely necessary, it is best 396 unsigned int getPortCount( void );
137 to set the callback function before opening a MIDI port to avoid 397
138 leaving some messages in the queue. 398 //! Return a string identifier for the specified MIDI port type and number.
139 */ 399 /*!
140 void setCallback( RtMidiCallback callback, void *userData = 0 ); 400 An empty string is returned if an invalid port specifier is provided.
141
142 //! Cancel use of the current callback function (if one exists).
143 /*!
144 Subsequent incoming MIDI messages will be written to the queue
145 and can be retrieved with the \e getMessage function.
146 */
147 void cancelCallback();
148
149 //! Close an open MIDI connection (if one exists).
150 void closePort( void );
151
152 //! Return the number of available MIDI input ports.
153 unsigned int getPortCount();
154
155 //! Return a string identifier for the specified MIDI input port number.
156 /*!
157 An exception is thrown if an invalid port specifier is provided.
158 */ 401 */
159 std::string getPortName( unsigned int portNumber = 0 ); 402 std::string getPortName( unsigned int portNumber = 0 );
160 403
161 //! Set the maximum number of MIDI messages to be saved in the queue. 404 //! Immediately send a single message out an open MIDI output port.
162 /*! 405 /*!
163 If the queue size limit is reached, incoming messages will be 406 An exception is thrown if an error occurs during output or an
164 ignored. The default limit is 1024. 407 output connection was not previously established.
165 */ 408 */
166 void setQueueSizeLimit( unsigned int queueSize ); 409 void sendMessage( std::vector<unsigned char> *message );
167 410
168 //! Specify whether certain MIDI message types should be queued or ignored during input. 411 //! Set an error callback function to be invoked when an error has occured.
169 /*! 412 /*!
170 By default, MIDI timing and active sensing messages are ignored 413 The callback function will be called whenever an error has occured. It is best
171 during message input because of their relative high data rates. 414 to set the error callback function before opening a port.
172 MIDI sysex messages are ignored by default as well. Variable 415 */
173 values of "true" imply that the respective message type will be 416 virtual void setErrorCallback( RtMidiErrorCallback errorCallback = NULL, void *userData = 0 );
174 ignored. 417
175 */ 418 protected:
176 void ignoreTypes( bool midiSysex = true, bool midiTime = true, bool midiSense = true ); 419 void openMidiApi( RtMidi::Api api, const std::string clientName );
177 420 };
178 //! Fill the user-provided vector with the data bytes for the next available MIDI message in the input queue and return the event delta-time in seconds. 421
179 /*! 422
180 This function returns immediately whether a new message is 423 // **************************************************************** //
181 available or not. A valid message is indicated by a non-zero 424 //
182 vector size. An exception is thrown if an error occurs during 425 // MidiInApi / MidiOutApi class declarations.
183 message retrieval or an input connection was not previously 426 //
184 established. 427 // Subclasses of MidiInApi and MidiOutApi contain all API- and
185 */ 428 // OS-specific code necessary to fully implement the RtMidi API.
429 //
430 // Note that MidiInApi and MidiOutApi are abstract base classes and
431 // cannot be explicitly instantiated. RtMidiIn and RtMidiOut will
432 // create instances of a MidiInApi or MidiOutApi subclass.
433 //
434 // **************************************************************** //
435
436 class MidiApi
437 {
438 public:
439
440 MidiApi();
441 virtual ~MidiApi();
442 virtual RtMidi::Api getCurrentApi( void ) = 0;
443 virtual void openPort( unsigned int portNumber, const std::string portName ) = 0;
444 virtual void openVirtualPort( const std::string portName ) = 0;
445 virtual void closePort( void ) = 0;
446
447 virtual unsigned int getPortCount( void ) = 0;
448 virtual std::string getPortName( unsigned int portNumber ) = 0;
449
450 inline bool isPortOpen() const { return connected_; }
451 void setErrorCallback( RtMidiErrorCallback errorCallback, void *userData );
452
453 //! A basic error reporting function for RtMidi classes.
454 void error( RtMidiError::Type type, std::string errorString );
455
456 protected:
457 virtual void initialize( const std::string& clientName ) = 0;
458
459 void *apiData_;
460 bool connected_;
461 std::string errorString_;
462 RtMidiErrorCallback errorCallback_;
463 bool firstErrorOccurred_;
464 void *errorCallbackUserData_;
465 };
466
467 class MidiInApi : public MidiApi
468 {
469 public:
470
471 MidiInApi( unsigned int queueSizeLimit );
472 virtual ~MidiInApi( void );
473 void setCallback( RtMidiIn::RtMidiCallback callback, void *userData );
474 void cancelCallback( void );
475 virtual void ignoreTypes( bool midiSysex, bool midiTime, bool midiSense );
186 double getMessage( std::vector<unsigned char> *message ); 476 double getMessage( std::vector<unsigned char> *message );
187 477
188 // A MIDI structure used internally by the class to store incoming 478 // A MIDI structure used internally by the class to store incoming
189 // messages. Each message represents one and only one MIDI message. 479 // messages. Each message represents one and only one MIDI message.
190 struct MidiMessage { 480 struct MidiMessage {
191 std::vector<unsigned char> bytes; 481 std::vector<unsigned char> bytes;
192 double timeStamp; 482 double timeStamp;
193 483
194 // Default constructor. 484 // Default constructor.
195 MidiMessage() 485 MidiMessage()
196 :bytes(3), timeStamp(0.0) {} 486 :bytes(0), timeStamp(0.0) {}
487 };
488
489 struct MidiQueue {
490 unsigned int front;
491 unsigned int back;
492 unsigned int size;
493 unsigned int ringSize;
494 MidiMessage *ring;
495
496 // Default constructor.
497 MidiQueue()
498 :front(0), back(0), size(0), ringSize(0) {}
197 }; 499 };
198 500
199 // The RtMidiInData structure is used to pass private class data to 501 // The RtMidiInData structure is used to pass private class data to
200 // the MIDI input handling function or thread. 502 // the MIDI input handling function or thread.
201 struct RtMidiInData { 503 struct RtMidiInData {
202 std::queue<MidiMessage> queue; 504 MidiQueue queue;
203 MidiMessage message; 505 MidiMessage message;
204 unsigned int queueLimit;
205 unsigned char ignoreFlags; 506 unsigned char ignoreFlags;
206 bool doInput; 507 bool doInput;
207 bool firstMessage; 508 bool firstMessage;
208 void *apiData; 509 void *apiData;
209 bool usingCallback; 510 bool usingCallback;
210 void *userCallback; 511 RtMidiIn::RtMidiCallback userCallback;
211 void *userData; 512 void *userData;
212 bool continueSysex; 513 bool continueSysex;
213 514
214 // Default constructor. 515 // Default constructor.
215 RtMidiInData() 516 RtMidiInData()
216 : queueLimit(1024), ignoreFlags(7), doInput(false), firstMessage(true), 517 : ignoreFlags(7), doInput(false), firstMessage(true),
217 apiData(0), usingCallback(false), userCallback(0), userData(0), 518 apiData(0), usingCallback(false), userCallback(0), userData(0),
218 continueSysex(false) {} 519 continueSysex(false) {}
219 }; 520 };
220 521
221 private: 522 protected:
222 523 RtMidiInData inputData_;
524 };
525
526 class MidiOutApi : public MidiApi
527 {
528 public:
529
530 MidiOutApi( void );
531 virtual ~MidiOutApi( void );
532 virtual void sendMessage( std::vector<unsigned char> *message ) = 0;
533 };
534
535 // **************************************************************** //
536 //
537 // Inline RtMidiIn and RtMidiOut definitions.
538 //
539 // **************************************************************** //
540
541 inline RtMidi::Api RtMidiIn :: getCurrentApi( void ) throw() { return rtapi_->getCurrentApi(); }
542 inline void RtMidiIn :: openPort( unsigned int portNumber, const std::string portName ) { rtapi_->openPort( portNumber, portName ); }
543 inline void RtMidiIn :: openVirtualPort( const std::string portName ) { rtapi_->openVirtualPort( portName ); }
544 inline void RtMidiIn :: closePort( void ) { rtapi_->closePort(); }
545 inline bool RtMidiIn :: isPortOpen() const { return rtapi_->isPortOpen(); }
546 inline void RtMidiIn :: setCallback( RtMidiCallback callback, void *userData ) { ((MidiInApi *)rtapi_)->setCallback( callback, userData ); }
547 inline void RtMidiIn :: cancelCallback( void ) { ((MidiInApi *)rtapi_)->cancelCallback(); }
548 inline unsigned int RtMidiIn :: getPortCount( void ) { return rtapi_->getPortCount(); }
549 inline std::string RtMidiIn :: getPortName( unsigned int portNumber ) { return rtapi_->getPortName( portNumber ); }
550 inline void RtMidiIn :: ignoreTypes( bool midiSysex, bool midiTime, bool midiSense ) { ((MidiInApi *)rtapi_)->ignoreTypes( midiSysex, midiTime, midiSense ); }
551 inline double RtMidiIn :: getMessage( std::vector<unsigned char> *message ) { return ((MidiInApi *)rtapi_)->getMessage( message ); }
552 inline void RtMidiIn :: setErrorCallback( RtMidiErrorCallback errorCallback, void *userData ) { rtapi_->setErrorCallback(errorCallback, userData); }
553
554 inline RtMidi::Api RtMidiOut :: getCurrentApi( void ) throw() { return rtapi_->getCurrentApi(); }
555 inline void RtMidiOut :: openPort( unsigned int portNumber, const std::string portName ) { rtapi_->openPort( portNumber, portName ); }
556 inline void RtMidiOut :: openVirtualPort( const std::string portName ) { rtapi_->openVirtualPort( portName ); }
557 inline void RtMidiOut :: closePort( void ) { rtapi_->closePort(); }
558 inline bool RtMidiOut :: isPortOpen() const { return rtapi_->isPortOpen(); }
559 inline unsigned int RtMidiOut :: getPortCount( void ) { return rtapi_->getPortCount(); }
560 inline std::string RtMidiOut :: getPortName( unsigned int portNumber ) { return rtapi_->getPortName( portNumber ); }
561 inline void RtMidiOut :: sendMessage( std::vector<unsigned char> *message ) { ((MidiOutApi *)rtapi_)->sendMessage( message ); }
562 inline void RtMidiOut :: setErrorCallback( RtMidiErrorCallback errorCallback, void *userData ) { rtapi_->setErrorCallback(errorCallback, userData); }
563
564 // **************************************************************** //
565 //
566 // MidiInApi and MidiOutApi subclass prototypes.
567 //
568 // **************************************************************** //
569
570 #if !defined(__LINUX_ALSA__) && !defined(__UNIX_JACK__) && !defined(__MACOSX_CORE__) && !defined(__WINDOWS_MM__)
571 #define __RTMIDI_DUMMY__
572 #endif
573
574 #if defined(__MACOSX_CORE__)
575
576 class MidiInCore: public MidiInApi
577 {
578 public:
579 MidiInCore( const std::string clientName, unsigned int queueSizeLimit );
580 ~MidiInCore( void );
581 RtMidi::Api getCurrentApi( void ) { return RtMidi::MACOSX_CORE; };
582 void openPort( unsigned int portNumber, const std::string portName );
583 void openVirtualPort( const std::string portName );
584 void closePort( void );
585 unsigned int getPortCount( void );
586 std::string getPortName( unsigned int portNumber );
587
588 protected:
223 void initialize( const std::string& clientName ); 589 void initialize( const std::string& clientName );
224 RtMidiInData inputData_; 590 };
225 591
226 }; 592 class MidiOutCore: public MidiOutApi
227 593 {
228 /**********************************************************************/ 594 public:
229 /*! \class RtMidiOut 595 MidiOutCore( const std::string clientName );
230 \brief A realtime MIDI output class. 596 ~MidiOutCore( void );
231 597 RtMidi::Api getCurrentApi( void ) { return RtMidi::MACOSX_CORE; };
232 This class provides a common, platform-independent API for MIDI 598 void openPort( unsigned int portNumber, const std::string portName );
233 output. It allows one to probe available MIDI output ports, to 599 void openVirtualPort( const std::string portName );
234 connect to one such port, and to send MIDI bytes immediately over 600 void closePort( void );
235 the connection. Create multiple instances of this class to 601 unsigned int getPortCount( void );
236 connect to more than one MIDI device at the same time. 602 std::string getPortName( unsigned int portNumber );
237
238 by Gary P. Scavone, 2003-2008.
239 */
240 /**********************************************************************/
241
242 class RtMidiOut : public RtMidi
243 {
244 public:
245
246 //! Default constructor that allows an optional client name.
247 /*!
248 An exception will be thrown if a MIDI system initialization error occurs.
249 */
250 RtMidiOut( const std::string clientName = std::string( "RtMidi Output Client" ) );
251
252 //! The destructor closes any open MIDI connections.
253 ~RtMidiOut();
254
255 //! Open a MIDI output connection.
256 /*!
257 An optional port number greater than 0 can be specified.
258 Otherwise, the default or first port found is opened. An
259 exception is thrown if an error occurs while attempting to make
260 the port connection.
261 */
262 void openPort( unsigned int portNumber = 0, const std::string portName = std::string( "RtMidi Output" ) );
263
264 //! Close an open MIDI connection (if one exists).
265 void closePort();
266
267 //! Create a virtual output port, with optional name, to allow software connections (OS X and ALSA only).
268 /*!
269 This function creates a virtual MIDI output port to which other
270 software applications can connect. This type of functionality
271 is currently only supported by the Macintosh OS-X and Linux ALSA
272 APIs (the function does nothing with the other APIs). An
273 exception is thrown if an error occurs while attempting to create
274 the virtual port.
275 */
276 void openVirtualPort( const std::string portName = std::string( "RtMidi Output" ) );
277
278 //! Return the number of available MIDI output ports.
279 unsigned int getPortCount();
280
281 //! Return a string identifier for the specified MIDI port type and number.
282 /*!
283 An exception is thrown if an invalid port specifier is provided.
284 */
285 std::string getPortName( unsigned int portNumber = 0 );
286
287 //! Immediately send a single message out an open MIDI output port.
288 /*!
289 An exception is thrown if an error occurs during output or an
290 output connection was not previously established.
291 */
292 void sendMessage( std::vector<unsigned char> *message ); 603 void sendMessage( std::vector<unsigned char> *message );
293 604
294 private: 605 protected:
295
296 void initialize( const std::string& clientName ); 606 void initialize( const std::string& clientName );
297 }; 607 };
298 608
299 #endif 609 #endif
610
611 #if defined(__UNIX_JACK__)
612
613 class MidiInJack: public MidiInApi
614 {
615 public:
616 MidiInJack( const std::string clientName, unsigned int queueSizeLimit );
617 ~MidiInJack( void );
618 RtMidi::Api getCurrentApi( void ) { return RtMidi::UNIX_JACK; };
619 void openPort( unsigned int portNumber, const std::string portName );
620 void openVirtualPort( const std::string portName );
621 void closePort( void );
622 unsigned int getPortCount( void );
623 std::string getPortName( unsigned int portNumber );
624
625 protected:
626 std::string clientName;
627
628 void connect( void );
629 void initialize( const std::string& clientName );
630 };
631
632 class MidiOutJack: public MidiOutApi
633 {
634 public:
635 MidiOutJack( const std::string clientName );
636 ~MidiOutJack( void );
637 RtMidi::Api getCurrentApi( void ) { return RtMidi::UNIX_JACK; };
638 void openPort( unsigned int portNumber, const std::string portName );
639 void openVirtualPort( const std::string portName );
640 void closePort( void );
641 unsigned int getPortCount( void );
642 std::string getPortName( unsigned int portNumber );
643 void sendMessage( std::vector<unsigned char> *message );
644
645 protected:
646 std::string clientName;
647
648 void connect( void );
649 void initialize( const std::string& clientName );
650 };
651
652 #endif
653
654 #if defined(__LINUX_ALSA__)
655
656 class MidiInAlsa: public MidiInApi
657 {
658 public:
659 MidiInAlsa( const std::string clientName, unsigned int queueSizeLimit );
660 ~MidiInAlsa( void );
661 RtMidi::Api getCurrentApi( void ) { return RtMidi::LINUX_ALSA; };
662 void openPort( unsigned int portNumber, const std::string portName );
663 void openVirtualPort( const std::string portName );
664 void closePort( void );
665 unsigned int getPortCount( void );
666 std::string getPortName( unsigned int portNumber );
667
668 protected:
669 void initialize( const std::string& clientName );
670 };
671
672 class MidiOutAlsa: public MidiOutApi
673 {
674 public:
675 MidiOutAlsa( const std::string clientName );
676 ~MidiOutAlsa( void );
677 RtMidi::Api getCurrentApi( void ) { return RtMidi::LINUX_ALSA; };
678 void openPort( unsigned int portNumber, const std::string portName );
679 void openVirtualPort( const std::string portName );
680 void closePort( void );
681 unsigned int getPortCount( void );
682 std::string getPortName( unsigned int portNumber );
683 void sendMessage( std::vector<unsigned char> *message );
684
685 protected:
686 void initialize( const std::string& clientName );
687 };
688
689 #endif
690
691 #if defined(__WINDOWS_MM__)
692
693 class MidiInWinMM: public MidiInApi
694 {
695 public:
696 MidiInWinMM( const std::string clientName, unsigned int queueSizeLimit );
697 ~MidiInWinMM( void );
698 RtMidi::Api getCurrentApi( void ) { return RtMidi::WINDOWS_MM; };
699 void openPort( unsigned int portNumber, const std::string portName );
700 void openVirtualPort( const std::string portName );
701 void closePort( void );
702 unsigned int getPortCount( void );
703 std::string getPortName( unsigned int portNumber );
704
705 protected:
706 void initialize( const std::string& clientName );
707 };
708
709 class MidiOutWinMM: public MidiOutApi
710 {
711 public:
712 MidiOutWinMM( const std::string clientName );
713 ~MidiOutWinMM( void );
714 RtMidi::Api getCurrentApi( void ) { return RtMidi::WINDOWS_MM; };
715 void openPort( unsigned int portNumber, const std::string portName );
716 void openVirtualPort( const std::string portName );
717 void closePort( void );
718 unsigned int getPortCount( void );
719 std::string getPortName( unsigned int portNumber );
720 void sendMessage( std::vector<unsigned char> *message );
721
722 protected:
723 void initialize( const std::string& clientName );
724 };
725
726 #endif
727
728 #if defined(__RTMIDI_DUMMY__)
729
730 class MidiInDummy: public MidiInApi
731 {
732 public:
733 MidiInDummy( const std::string /*clientName*/, unsigned int queueSizeLimit ) : MidiInApi( queueSizeLimit ) { errorString_ = "MidiInDummy: This class provides no functionality."; error( RtMidiError::WARNING, errorString_ ); }
734 RtMidi::Api getCurrentApi( void ) { return RtMidi::RTMIDI_DUMMY; }
735 void openPort( unsigned int /*portNumber*/, const std::string /*portName*/ ) {}
736 void openVirtualPort( const std::string /*portName*/ ) {}
737 void closePort( void ) {}
738 unsigned int getPortCount( void ) { return 0; }
739 std::string getPortName( unsigned int /*portNumber*/ ) { return ""; }
740
741 protected:
742 void initialize( const std::string& /*clientName*/ ) {}
743 };
744
745 class MidiOutDummy: public MidiOutApi
746 {
747 public:
748 MidiOutDummy( const std::string /*clientName*/ ) { errorString_ = "MidiOutDummy: This class provides no functionality."; error( RtMidiError::WARNING, errorString_ ); }
749 RtMidi::Api getCurrentApi( void ) { return RtMidi::RTMIDI_DUMMY; }
750 void openPort( unsigned int /*portNumber*/, const std::string /*portName*/ ) {}
751 void openVirtualPort( const std::string /*portName*/ ) {}
752 void closePort( void ) {}
753 unsigned int getPortCount( void ) { return 0; }
754 std::string getPortName( unsigned int /*portNumber*/ ) { return ""; }
755 void sendMessage( std::vector<unsigned char> * /*message*/ ) {}
756
757 protected:
758 void initialize( const std::string& /*clientName*/ ) {}
759 };
760
761 #endif
762
763 #endif