153 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			153 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|   | //*****************************************//
 | ||
|  | //  sysextest.cpp
 | ||
|  | //  by Gary Scavone, 2003-2005.
 | ||
|  | //
 | ||
|  | //  Simple program to test MIDI sysex sending and receiving.
 | ||
|  | //
 | ||
|  | //*****************************************//
 | ||
|  | 
 | ||
|  | #include <iostream>
 | ||
|  | #include <cstdlib>
 | ||
|  | #include <typeinfo>
 | ||
|  | #include "RtMidi.h"
 | ||
|  | 
 | ||
|  | void usage( void ) { | ||
|  |   std::cout << "\nuseage: sysextest N\n"; | ||
|  |   std::cout << "    where N = length of sysex data to send / receive.\n\n"; | ||
|  |   exit( 0 ); | ||
|  | } | ||
|  | 
 | ||
|  | // Platform-dependent sleep routines.
 | ||
|  | #if defined(WIN32)
 | ||
|  |   #include <windows.h>
 | ||
|  |   #define SLEEP( milliseconds ) Sleep( (DWORD) milliseconds ) 
 | ||
|  | #else // Unix variants
 | ||
|  |   #include <unistd.h>
 | ||
|  |   #define SLEEP( milliseconds ) usleep( (unsigned long) (milliseconds * 1000.0) )
 | ||
|  | #endif
 | ||
|  | 
 | ||
|  | // This function should be embedded in a try/catch block in case of
 | ||
|  | // an exception.  It offers the user a choice of MIDI ports to open.
 | ||
|  | // It returns false if there are no ports available.
 | ||
|  | bool chooseMidiPort( RtMidi *rtmidi ); | ||
|  | 
 | ||
|  | void mycallback( double deltatime, std::vector< unsigned char > *message, void * /*userData*/ ) | ||
|  | { | ||
|  |   unsigned int nBytes = message->size(); | ||
|  |   for ( unsigned int i=0; i<nBytes; i++ ) | ||
|  |     std::cout << "Byte " << i << " = " << (int)message->at(i) << ", "; | ||
|  |   if ( nBytes > 0 ) | ||
|  |     std::cout << "# of bytes = " << nBytes << ", stamp = " << deltatime << std::endl; | ||
|  | } | ||
|  | 
 | ||
|  | int main( int argc, char *argv[] ) | ||
|  | { | ||
|  |   RtMidiOut *midiout = 0; | ||
|  |   RtMidiIn *midiin = 0; | ||
|  |   std::vector<unsigned char> message; | ||
|  |   unsigned int i, nBytes; | ||
|  | 
 | ||
|  |   // Minimal command-line check.
 | ||
|  |   if ( argc != 2 ) usage(); | ||
|  |   nBytes = (unsigned int) atoi( argv[1] ); | ||
|  | 
 | ||
|  |   // RtMidiOut and RtMidiIn constructors
 | ||
|  |   try { | ||
|  |     midiout = new RtMidiOut(); | ||
|  |     midiin = new RtMidiIn(); | ||
|  |   } | ||
|  |   catch ( RtMidiError &error ) { | ||
|  |     error.printMessage(); | ||
|  |     goto cleanup; | ||
|  |   } | ||
|  | 
 | ||
|  |   // Don't ignore sysex, timing, or active sensing messages.
 | ||
|  |   midiin->ignoreTypes( false, true, true ); | ||
|  | 
 | ||
|  |   try { | ||
|  |     if ( chooseMidiPort( midiin ) == false ) goto cleanup; | ||
|  |     if ( chooseMidiPort( midiout ) == false ) goto cleanup; | ||
|  | 
 | ||
|  |     midiin->setCallback( &mycallback ); | ||
|  | 
 | ||
|  |     message.push_back( 0xF6 ); | ||
|  |     midiout->sendMessage( &message ); | ||
|  |     SLEEP( 500 ); // pause a little
 | ||
|  | 
 | ||
|  |     // Create a long sysex message of numbered bytes and send it out ... twice.
 | ||
|  |     for ( int n=0; n<2; n++ ) { | ||
|  |       message.clear(); | ||
|  |       message.push_back( 240 ); | ||
|  |       for ( i=0; i<nBytes; i++ ) | ||
|  |         message.push_back( i % 128 ); | ||
|  |       message.push_back( 247 ); | ||
|  |       midiout->sendMessage( &message ); | ||
|  | 
 | ||
|  |       SLEEP( 500 ); // pause a little
 | ||
|  |     } | ||
|  |   } | ||
|  |   catch ( RtMidiError &error ) { | ||
|  |     error.printMessage(); | ||
|  |     goto cleanup; | ||
|  |   } | ||
|  | 
 | ||
|  |   // Clean up
 | ||
|  |  cleanup: | ||
|  |   delete midiout; | ||
|  |   delete midiin; | ||
|  | 
 | ||
|  |   return 0; | ||
|  | } | ||
|  | 
 | ||
|  | bool chooseMidiPort( RtMidi *rtmidi ) | ||
|  | { | ||
|  |   bool isInput = false; | ||
|  |   if ( typeid( *rtmidi ) == typeid( RtMidiIn ) ) | ||
|  |     isInput = true; | ||
|  | 
 | ||
|  |   if ( isInput ) | ||
|  |     std::cout << "\nWould you like to open a virtual input port? [y/N] "; | ||
|  |   else | ||
|  |     std::cout << "\nWould you like to open a virtual output port? [y/N] "; | ||
|  | 
 | ||
|  |   std::string keyHit; | ||
|  |   std::getline( std::cin, keyHit ); | ||
|  |   if ( keyHit == "y" ) { | ||
|  |     rtmidi->openVirtualPort(); | ||
|  |     return true; | ||
|  |   } | ||
|  | 
 | ||
|  |   std::string portName; | ||
|  |   unsigned int i = 0, nPorts = rtmidi->getPortCount(); | ||
|  |   if ( nPorts == 0 ) { | ||
|  |     if ( isInput ) | ||
|  |       std::cout << "No input ports available!" << std::endl; | ||
|  |     else | ||
|  |       std::cout << "No output ports available!" << std::endl; | ||
|  |     return false; | ||
|  |   } | ||
|  | 
 | ||
|  |   if ( nPorts == 1 ) { | ||
|  |     std::cout << "\nOpening " << rtmidi->getPortName() << std::endl; | ||
|  |   } | ||
|  |   else { | ||
|  |     for ( i=0; i<nPorts; i++ ) { | ||
|  |       portName = rtmidi->getPortName(i); | ||
|  |       if ( isInput ) | ||
|  |         std::cout << "  Input port #" << i << ": " << portName << '\n'; | ||
|  |       else | ||
|  |         std::cout << "  Output port #" << i << ": " << portName << '\n'; | ||
|  |     } | ||
|  | 
 | ||
|  |     do { | ||
|  |       std::cout << "\nChoose a port number: "; | ||
|  |       std::cin >> i; | ||
|  |     } while ( i >= nPorts ); | ||
|  |   } | ||
|  | 
 | ||
|  |   std::cout << std::endl; | ||
|  |   rtmidi->openPort( i ); | ||
|  | 
 | ||
|  |   return true; | ||
|  | } |