Click here to Skip to main content
15,113,245 members
Articles / Desktop Programming / MFC
Posted 16 Aug 2002


144 bookmarked

Wrapper Library for Windows MIDI API

Rate me:
Please Sign up or sign in to vote.
4.94/5 (67 votes)
28 Jan 2008MIT8 min read
A small library encapsulating the Windows MIDI API
MIDIDeviceDemo Image


Using the Windows low-level MIDI API is not necessarily easy. There are a lot of details to keep track of, and it can get a bit frustrating when you are trying to write a MIDI application. The goal of my small wrapper library is to elevate some of the grunt work that goes into using the Windows low-level MIDI API by encapsulating the functions into C++ classes. With these classes, most of the dirty work is handled behind the scenes. This leaves you free to concentrate on more important matters.

The Library

The MIDI library is encapsulated in the midi namespace. In order to use the library, you must link to the winmm.lib library. There are several parts to the library described in the following header files:

  • midi.h - Useful constants
  • MIDIInDevice.h - Interface for the CMIDIInDevice and CMIDIReceiver classes
  • MIDIOutDevice.h - Interface for the CMIDIOutDevice class
  • MIDIMsg.h - Abstract base class for all MIDI messages
  • ShortMsg.h - Interface for the CShortMsg class
  • LongMsg.h - Interface for the CLongMsg class

Let's look at each part of the library.

midi.h - Useful Constants

The midi.h header file contains many useful constants. For example, the midi::NOTE_OFF constant represents the value for turning off a MIDI note. These constants come in handy when creating and interpreting MIDI messages. You don't have to look up the values in a MIDI reference. Just plug in the constant you need, and you're all set.

MIDIInDevice.h - The CMIDIInDevice and CMIDIReceiver Classes

The CMIDIInDevice class represents MIDI input devices. A MIDI input device can be any hardware device installed on your computer, such as a soundcard, capable of receiving MIDI data. There can be several of these devices present, and the CMIDIInDevice class provides methods for determining the number and characteristics of these devices. The GetNumDevs() function returns the number of MIDI input devices on your system, and each number represents that device's identifier. The GetDevCaps() function returns information about a particular device by storing it in a MIDIINCAPS structure. This is a Windows structure that contains many attributes describing MIDI input devices, such as their names. A good first step before using the CMIDIInDevice class is to determine the number of MIDI input devices on your system and which one you want to use to receive MIDI data.

Before CMIDIInDevice objects can be used, they have to be opened. What do I mean by opened? Think of using the ifstream class from the standard library for reading the contents of a file. Before the file can be read, it has to be opened. The CMIDIInDevice class works in the same way - before it can "read" MIDI data, it has to be opened. You open CMIDIInDevice objects by passing a device identifier to either the constructor or the Open() function. The device identifier is a unique number representing a MIDI input device. You can retrieve this number by using the aforementioned GetNumDevs() and GetDevCaps() functions.

To begin receiving MIDI data, call the StartRecording() function after the MIDI input device has been opened. The device will continue receiving MIDI data until StopRecording() is called to stop the process. Where do the MIDI messages go once they are received? This leads us to the second major component in the input part of the library - the CMIDIReceiver class.

The CMIDIReceiver class is an abstract class representing objects that receive MIDI data from the CMIDIInDevice class. An object derived from the CMIDIReceiver class can register itself with an CMIDIInDevice object. The CMIDIInDevice will then pass on MIDI messages it receives by calling methods in the CMIDIReceiver derived object. The CMIDIReceiver has two methods for receiving short messages, and two methods for receiving system exclusive messages. In both cases, there is one method which is called when things have gone normally and a second method which is called if an error has occurred.

Short messages are represented by the DWORD data type. You don't have to do anything special to receive them. System exclusive messages, on the other hand, are represented by a LPSTR data buffer. In order to receive system exclusive messages, provide the CMIDIInDevice with buffers to store the messages. This is done with the AddSysExBuffer() function. Pass as many buffers as you like during the recording process. When the CMIDIInDevice receives a system exclusive message, it stores it in the buffer and passes it back through the CMIDIReceiver you registered.

Finally, once you are finished using the CMIDIInDevice, close the device simply by calling the Close() function.

To make short messages and system exclusive messages easier to work with, I've added three new classes: CMIDIMsg, CShortMsg, and CLongMsg. The CMIDIMsg class is the abstract base class for all MIDI messages. Both the CShortMsg and CLongMsg classes are derived from it. The CShortMsg class represents MIDI short messages such as note-on and note-off messages. The CLongMsg class represents system exclusive messages. It is essentially a buffer for the bytes that makes up an system exclusive message. It can be used for general purpose system exclusive messages, or you can derive a more specialized class from it.

A code snippet is worth a thousand words, so let's look at an example of using the CMIDIInDevice and CMIDIReceiver classes:

#include <span class="code-string">"StdAfx.h"</span>
#include <span class="code-keyword"><iostream></span>
#include <span class="code-string">"MIDIInDevice.h"</span>
#include <span class="code-string">"ShortMsg.h"</span>

using midi::CMIDIInDevice;
using midi::CMIDIReceiver;

// My class for receiving MIDI messages
class MyReceiver : public CMIDIReceiver
    // Receives short messages
    void ReceiveMsg(DWORD Msg, DWORD TimeStamp);

    // Receives long messages
    void ReceiveMsg(LPSTR Msg, DWORD BytesRecorded, DWORD TimeStamp)

    // Called when an invalid short message is received
    void OnError(DWORD Msg, DWORD TimeStamp)
    { std::cout << "Invalid short message received!\n"; }

    // Called when an invalid long message is received
    void OnError(LPSTR Msg, DWORD BytesRecorded, DWORD TimeStamp) {}

// Function called to receive short messages
void MyReceiver::ReceiveMsg(DWORD Msg, DWORD TimeStamp)
    midi::CShortMsg ShortMsg(Msg, TimeStamp);

    std::cout << "Command: " <<
    std::cout << "\nChannel: " <<
    std::cout << "\nDataByte1: " <<
    std::cout << "\nDataByte2: " <<
    std::cout << "\nTimeStamp: " <<
    std::cout << "\n\n";

int main(void)
        // Make sure there is a MIDI input device present
        if(CMIDIInDevice::GetNumDevs() > 0)
            MyReceiver Receiver;
            CMIDIInDevice InDevice(Receiver);

            // We'll use the first device - we're not picky here

            // Start recording

            // Wait for some MIDI messages

            // Stop recording

            // Close the device
            std::cout << "No MIDI input devices present!\n";
    catch(const std::exception &Err)
        std::cout << Err.what();

    return 0;

Some notes about the code above:

  • All exceptions in the library are derived from std::exception, so catching this exception will catch any exception thrown.
  • Short messages are received as DWORD data types. They must be unpacked before you can do anything interesting with them. You can do this yourself, or you can allow the CShortMsg to take care of this for you. If you decide to pack and unpack short messages, you can use class methods in the CMIDIInDevice class and the CMIDIOutDevice class for unpacking and packing short messages respectively.
  • A time stamp value is associated with each message. The time stamp is measured in milliseconds since the recording process began.
  • The example above doesn't handle system exclusive messages. In order to do that, all you would have to do is pass the input device one or more buffers and implement the methods derived from CMIDIReceiver for receiving system exclusive messages.

MIDIOutDevice.h - The CMIDIOutDevice Class

The CMIDIOutDevice class represents MIDI output devices. Like the CMIDIInDevice class, it has methods for determining the number of devices present and for retrieving information about each device. Using an object of this class is straight forward. Simply open it, create a MIDI message, send the message, and close the device when you are done with it.

Here is another code example:

#include <span class="code-string">"StdAfx.h"</span>
#include <span class="code-keyword"><iostream></span>
#include <span class="code-string">"midi.h"</span>
#include <span class="code-string">"MIDIOutDevice.h"</span>
#include <span class="code-string">"ShortMsg.h"</span>

using midi::CMIDIOutDevice;
using midi::CShortMsg;

// Some useful constants
const unsigned char CHANNEL = 0;
const unsigned char NOTE_ID = 64;
const unsigned char VELOCITY = 127;

int main(void)
        // Make sure there is an output device present
        if(CMIDIOutDevice::GetNumDevs() > 0)
            CMIDIOutDevice OutDevice;

            // Use the first device

            // Create note-on message
            CShortMsg NoteOnMsg(midi::NOTE_ON, CHANNEL, NOTE_ID,
              VELOCITY, 0);

            // Turn note on

            // Wait a bit

            // Create note-off message
            CShortMsg NoteOffMsg(midi::NOTE_OFF, CHANNEL, NOTE_ID, 0, 0);

            // Turn note off

            // Close device
            std::cout << "No MIDI output devices present!\n";
    catch(const std::exception &Err)
        std::cout << Err.what();

    return 0;

Some comments:

  • The code above sends a note, waits for a few seconds, and then turns the note off.
  • To send a system exclusive message, go through the same process as above only prepare your message as a LPSTR buffer or create a CLongMsg object and initialize it with a valid system exclusive message.
  • CMIDIOutDevice doesn't pay any attention to time stamps. In order to send messages at a certain time, you'll have to create a timing scheme of some sort.

The MIDIDeviceDemo Demonstration App

The demo application uses the small MIDI library I've created for input and output. The piano keyboard is interactive. Clicking on a key with the mouse triggers a note. You can also play the piano from the computer keyboard. The keys from 'Z' to '/' correspond to the piano. To change the range of the computer keyboard, press keys 1-5. The '1' key represents the lowest range, and the '5' key represents the highest range.

If a MIDI keyboard is connected to the computer, playing keys on the keyboard will cause the corresponding notes on the keyboard display to change colors. Incoming channel messages are displayed as well. The application has very little practical value, but it does show how the library works.

Hopefully, you will find this small library as useful as I have. I would appreciate any comments or suggestions for future revisions. Take care!


  • 25 Aug 2002 - Updated source code
  • 22 Dec 2002 - Added GetDevID method to both the CMIDIInDevice class and the CMIDIOutDevice class. Added CMIDIMsg, CShortMsg, and CLongMsg classes to the library. Redid demo app from scratch. The keyboard display is derived from my new class CPianoCtrl. All source code is included in the demo project (no need to email any longer for the source!).
  • 9 Sept 2003 - Updated source code and fixed several bugs. The CShortMsg class can now send messages without the status byte. I also changed the way threads are created from using CreateThread to using AfxBeginThread and changed the way thread termination is handled so that it is done more gracefully. Several other changes to the library here and there as well as fixing some bugs in the demo application.
  • 23 Sep 2003 - Updated source code and demo
  • 29 Sep 2003 - Updated source code and demo
  • 25 Jan 2008 - Updated source code and demo
  • 28 Jan 2008 - Bug fix for last update


This article, along with any associated source code and files, is licensed under The MIT License


About the Author

Leslie Sanford
United States United States
Aside from dabbling in BASIC on his old Atari 1040ST years ago, Leslie's programming experience didn't really begin until he discovered the Internet in the late 90s. There he found a treasure trove of information about two of his favorite interests: MIDI and sound synthesis.

After spending a good deal of time calculating formulas he found on the Internet for creating new sounds by hand, he decided that an easier way would be to program the computer to do the work for him. This led him to learn C. He discovered that beyond using programming as a tool for synthesizing sound, he loved programming in and of itself.

Eventually he taught himself C++ and C#, and along the way he immersed himself in the ideas of object oriented programming. Like many of us, he gotten bitten by the design patterns bug and a copy of GOF is never far from his hands.

Now his primary interest is in creating a complete MIDI toolkit using the C# language. He hopes to create something that will become an indispensable tool for those wanting to write MIDI applications for the .NET framework.

Besides programming, his other interests are photography and playing his Les Paul guitars.

Comments and Discussions

QuestionReceiving System Exclusive Pin
Ben Zonneveld2-May-14 10:11
MemberBen Zonneveld2-May-14 10:11 
QuestionMIDI LYRIC Pin
Sudhanta Suryaputra2-Aug-12 5:55
MemberSudhanta Suryaputra2-Aug-12 5:55 
QuestionDemo piano by Leslie Sanford Pin
olaf_2-Jun-12 4:21
Memberolaf_2-Jun-12 4:21 
Questionchannel 10 Pin
mcfenix25-Nov-11 16:53
Membermcfenix25-Nov-11 16:53 
AnswerRe: channel 10 Pin
kabuhilal4-Feb-12 11:22
Memberkabuhilal4-Feb-12 11:22 
QuestionMIDIDevDemo v2 - Unhandled exception [modified] Pin
Yvonne Ieong23-Feb-11 6:43
MemberYvonne Ieong23-Feb-11 6:43 
AnswerRe: MIDIDevDemo v2 - Unhandled exception Pin
Yvonne Ieong23-Feb-11 17:04
MemberYvonne Ieong23-Feb-11 17:04 
GeneralTip: Easy to remove MFC dependency in this library... Pin
Pallium21-Feb-10 8:47
MemberPallium21-Feb-10 8:47 
QuestionThe role of threads Pin
Dukato20-Jan-10 22:48
MemberDukato20-Jan-10 22:48 
GeneralRuntime problem when building with VS2008 Pin
Ben F. Zonneveld15-Jan-10 12:01
MemberBen F. Zonneveld15-Jan-10 12:01 
GeneralRe: Runtime problem when building with VS2008 Pin
Leslie Sanford15-Jan-10 12:31
MemberLeslie Sanford15-Jan-10 12:31 
GeneralRe: Runtime problem when building with VS2008 Pin
Ben F. Zonneveld15-Jan-10 12:45
MemberBen F. Zonneveld15-Jan-10 12:45 
GeneralRe: Runtime problem when building with VS2008 Pin
Leslie Sanford15-Jan-10 14:10
MemberLeslie Sanford15-Jan-10 14:10 
GeneralRelease problem Pin
lipanje6-Oct-09 4:30
Memberlipanje6-Oct-09 4:30 
GeneralRe: Release problem Pin
lipanje8-Oct-09 2:31
Memberlipanje8-Oct-09 2:31 
Questionis there a way to make it work under Windows Mobile? Pin
glook31-Jul-09 4:42
Memberglook31-Jul-09 4:42 
GeneralLINK : fatal error LNK1104: cannot open file 'mfc42d.lib' Pin
gsdagsg15-Jul-09 17:56
Membergsdagsg15-Jul-09 17:56 
GeneralProblem with releasing Pin
lespaulka00117-Nov-08 4:32
Memberlespaulka00117-Nov-08 4:32 
GeneralMFC, Window Form Pin
EllenTCY22-Mar-08 8:01
MemberEllenTCY22-Mar-08 8:01 
QuestionWorking with offsets Pin
Danny Van Hoof9-Feb-08 11:12
MemberDanny Van Hoof9-Feb-08 11:12 
GeneralRe: Working with offsets Pin
Leslie Sanford9-Feb-08 11:16
MemberLeslie Sanford9-Feb-08 11:16 
GeneralCompile Error [modified] Pin
JimD.999926-Jan-08 13:56
MemberJimD.999926-Jan-08 13:56 
GeneralRe: Compile Error Pin
Leslie Sanford26-Jan-08 23:04
MemberLeslie Sanford26-Jan-08 23:04 
GeneralRe: Compile Error Pin
JimD.999927-Jan-08 5:08
MemberJimD.999927-Jan-08 5:08 
GeneralRe: Compile Error Pin
Leslie Sanford27-Jan-08 6:59
MemberLeslie Sanford27-Jan-08 6:59 

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Praise Praise    Rant Rant    Admin Admin   

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads, Ctrl+Shift+Left/Right to switch pages.