一点英文资料<br><br>To use the MIDI Stream API, you need to first call midiStreamOpen() once to open some MIDI stream device for output. (This is similiar to the Low level MIDI API's midiOutOpen). You pass the Device ID of that desired stream device. Then, you can subsequently call other Stream functions, such as midiStreamOut() to queue MIDI messages for playback upon that device, and midiStreamRestart() to start the actual playback. Windows and that device will take care of timing out each event and outputting its MIDI bytes. <br>After you're done outputting to a stream device (and have no further use for it), you must close that device.<br><br>Think of a stream device like a file. You open it, you write to it (ie, queue MIDI events to it which Windows "plays"), and then you close it.<br><br><br>--------------------------------------------------------------------------------<br><br>Opening the default Stream for playback<br>This is extremely similiar to the Low level MIDI API's midiOutOpen() (except that a pointer to the device ID is passed rather than the device ID directly passed, and also, there is one extra arg that is currently set to 1). To open the default device, use a Device ID of 0 as so: <br>unsigned long result, deviceID;<br>HMIDISTRM outHandle;<br><br>/* Open default MIDI Out stream device */<br>deviceID = 0;<br>result = midiStreamOpen(&outHandle, &deviceID, 1, 0, 0, CALLBACK_NULL);<br>if (result)<br>{<br> printf("There was an error opening the default MIDI stream device!/r/n");<br>}<br><br>The most flexible way to choose a MIDI Stream device for input or output<br>Windows Stream Manager is capable of streaming MIDI data to any one of the MIDI Output Devices on a system. So, in order to get a list of MIDI Out Devices that you can open with midiStreamOpen(), you simply do things exactly as you would with the low level's midiOutOpen(). You query all of the MIDI Output devices in the system using midiOutGetDevCaps(), and remember that the first device always has an ID of 0, and subsequent devices have ascending ID numbers. <br>Individual Windows 95 drivers for any installed cards may be specially written to directly support the Stream API. When you query a particular MIDI Output device (using midiOutGetDevCaps), you can check the dwSupport field of the MIDIOUTCAPS structure. If the driver directly supports the Stream API, the MIDICAPS_STREAM bit of dwSupport will be set. (Download my ListMidiDevs C example to show how to query the MIDI Output devices for such support). Such a driver may offer special features such as allowing the Stream Manager to sync playback to incoming SMPTE or MIDI Time Code. Those features would be determined by what code has been added to the driver, as well as perhaps hardware support by the MIDI card itself.<br><br>One caveat is that the MIDI Mapper cannot be opened as the output for a MIDI stream. The Windows MIDI Stream Manager supports only one MIDI output at a time (unless the card's driver directly supports the Stream API and adds some sort of routing feature based upon the MIDIEVENT's dwStreamID field. Currently, the Stream Manager doesn't support this routing).<br><br><br>--------------------------------------------------------------------------------<br><br>General overview of stream playback<br>The process of passing the MIDI events that you want Windows to time out and output is similiar to passing a buffer of data to midiOutLongMsg(). In fact, you use a MIDIHDR structure, as well as calls to midiOutPrepareHeader() and midiOutUnprepareHeader(). But, you also use one or more MIDIEVENT structures, and the actual time out and output of the MIDI messages doesn't begin until you call midiStreamRestart(). (ie, Until you actually call midiStreamRestart, Windows simply queues the MIDI messages in the order you pass them). <br>Here's how you pass a MIDI message to be played back:<br><br>Place one MIDI message into a structure called a MIDIEVENT. <br><br>Place a pointer to this MIDIEVENT into the lpData field of a MIDIHDR structure. Also set the MIDIHDR's dwBufferLength and dwBytesRecorded fields to the size of this MIDIEVENT structure. Set the dwFlags field to 0.<br><br><br>Pass the MIDIHDR to midiOutPrepareHeader().<br><br><br>Pass the MIDIHDR to midiStreamOut().<br><br>The MIDI message is now queued for playback. If you haven't called midiStreamRestart() yet, then Windows does not yet start timing out and outputting this event. So, you can queue several MIDI messages prior to the start of playback by repeating the above steps, using a separate MIDIEVENT and MIDIHDR for each message.<br><br>It's also possible to queue several MIDI messages using one MIDIHDR and one call to midiOutPrepareHeader() and midiStreamOut(). You do this by using an array of MIDIEVENT structures (ie, one for each of your MIDI messages). Place each MIDI message into one of the MIDIEVENT structures in that array. Then place a pointer to the entire array in the MIDIHDR's lpData field. Of course, the MIDIHDR's dwBufferLength and dwBytesRecorded fields are set to the size of the entire array of MIDIEVENT structures. It is much more efficient and memory-conserving to combine all MIDI messages that occur upon the same musical beat into one array of MIDIEVENTS, and cue them with one MIDIHDR.<br><br>Note that after Windows finishes playing all of the events queued with a particular MIDIHDR, it sets the MHDR_DONE bit of the dwFlags field. This may be very useful later when you're dealing with processing MIDIHDRs as a result of being notified by Windows that a MIDIHDR's events have finished playing. Also note that a MIDIHDR has a dwUser field that you can use for your own purposes.<br><br>Now it's certainly feasible to queue up all of the MIDI messages of a sequence prior to calling midiStreamRestart() by simply putting them into one gigantic array of MIDIEVENT structures, and passing the whole array in one call to midiStreamOut(). You can download my Simple Stream C example which illustrates this approach. Included are the Project Workspace files for Visual C++ 4.0, but since it is a console app, any Windows C compiler should be able to compile it. My code takes this approach in order to present the simplest possible example of using the Stream API to play a musical sequence, and therefore give you an easy introduction to how the Stream API works.<br><br>But that approach is sort of like using the High level MIDI API. After all, by passing all of the data all at once, you then lose control over too much of the playback process. The whole point of the Stream API is to feed Windows a few events at a time so that the playback doesn't get too far ahead that you lose the sense of real-time control over individual events. If you want to be doing real-time mixing of several "tracks" of MIDI messages into a single stream, giving the user the option to mute one of the tracks at any time for example, then obviously you don't want to queue MIDI messages too far in advance.<br><br>But, if you want a smooth playback, you obviously have to queue those events before they need to be played. The best approach to take is to use a sort of double-buffering scheme. In other words, you'll always have one "block" of MIDI messages queued while the current block of MIDI messages is playing. When the current block is finished playing (and the queued block starts playing), then you'll queue another block.<br><br>You'll select a brief "time window", for example 1 quarter note. (ie, Assume our "musical beat" is a quarter note). Then, before playback is started with midiStreamRestart(), you'll place all of the MIDI messages whose timing falls within the first quarter note (ie, the first beat) into an array of MIDIEVENTS (and a MIDIHDR) and queue it with midiStreamOut(). You'll also place all of the MIDI messages whose timing falls within the second quarter note (ie, second beat) into another array of MIDIEVENTS (and another MIDIHDR) and queue it with midiStreamOut(). Then, you'll call midiStreamRestart() to start playback. As soon as the last MIDI message in the first array of MIDIEVENTS is finished playing, you'll place all of the MIDI messages whose timing falls within the third quarter note into that same array of MIDIEVENTS and queue it with midiStreamOut(). Of course, while you're doing this, the second queued array has been playing. After that array's last MIDI message is played, you'll use that array to queue another "block" of MIDI messages whose timing falls within the fourth quarter note. Etc. In this way, you always have data queued for continuous, smooth playback, but the playback is only 1 musical beat ahead of any real-time action the user instructs you to perform upon the data. So, for example, if he tells you to mute one of the tracks, maybe it won't effectively work out that way for one musical beat (some events on that track may already be queued for the next beat), but that's plenty close enough.<br><br>So what if there happens to be no MIDI events that fall within the next musical beat? What do you queue? As you'll see in the next section, you can queue a single NOP event, timed to delay for an entire beat.<br><br><br>--------------------------------------------------------------------------------<br><br>The MIDIEVENT structure<br>Now let's take a closer look at the MIDIEVENT structure to see how various MIDI messages are stored in it. The Stream API documents it as so: <br>typedef struct { <br> DWORD dwDeltaTime;<br> DWORD dwStreamID;<br> DWORD dwEvent;<br> DWORD dwParms[];<br>} MIDIEVENT;<br><br>The dwDeltaTime field is just like the timing in MIDI File Format (except that it's an unsigned long rather than a variable length quantity). It represents the amount of time to delay before outputting this MIDI message. You can specify the time in terms of PPQN clocks, or a SMPTE time. (When using the former, you can send "Tempo Events" to the stream device to change the tempo, or call a Stream function that allows you to change the tempo on the fly. When using the latter, Windows can sync the MIDI playback to streaming video or other SMPTE cues).<br><br>Currently, the dwStreamID field isn't used and should be set to 0. (My own tests show that whatever value you stuff into this field is ignored, but unless you're using a driver that directly supports the MIDI Stream API and the driver uses this field, it is safest to set it to 0 in case some future version of Windows utilizes this field).<br><br>Although declared as an unsigned long, the 4 bytes of the dwEvent field are actually individual pieces of information. The highest byte contains some flag bits and some bits that form an "event type" value. I'll refer to this as the Event Type byte.<br><br>If this MIDIEVENT contains a normal MIDI Voice message such as a Note-On, Program Change, Aftertouch, or any of the other MIDI messages that are 3 or less bytes, then the Event Type byte is set to the value MEVT_SHORTMSG.<br><br>In this case, the remaining 3 bytes of this field are the MIDI Status byte, the first MIDI data byte (if any), and the second MIDI data byte (if any). Note that this is packed up exactly the way that a MIDI message is passed to midiOutShortMsg(). Do not use running status. The stream device will implement running status when it outputs the MIDI messages.<br><br>Here then is how you would initialize a MIDIEVENT with a Note-On MIDI message for middle C on channel 1 (with velocity of 0x40) and a delta time of 0 (gets output as soon as Windows finishes playing the previously queued event):<br><br>MIDIEVENT mevt;<br><br>mevt.dwDeltaTime = 0;<br>mevt.dwStreamID = 0;<br>mevt.dwEvent = ((unsigned long)MEVT_SHORTMSG<<24) | 0x00403C90;<br><br>Note that since MEVT_SHORTMSG is really 0x00, we can simply the above by actually removing the setting of the Event Type byte (since its already 0 in our packed MIDI message): <br>mevt.dwEvent = 0x00403C90;<br><br>What about that dwParms field? Well, if you look closely at the declaration, it isn't really there. There's no size for this dwParms array. Say what?!? That's right. There is no dwParms field on the above MIDIEVENT. The structure really has only 3 fields. It should have been declared as something like this (which I'll call MY_MIDI_EVT):<br><br>typedef struct { <br> DWORD dwDeltaTime;<br> DWORD dwStreamID;<br> DWORD dwEvent;<br>} MY_MIDI_EVT;<br><br>So what was that dwParms field doing there? Well, here's where it gets tricky. There are other types of events that you can pass. For example, consider a System Exclusive event. It can have many more than 3 bytes. Where do you put all of those bytes? Well, now you use that extra dwParms array appended to the end. Its size is set to however many bytes you need to pass. In essense, you're appending the data to the end of the MIDIEVENT structure. That's right. The size of the MIDIEVENT structure you pass will vary depending upon the type of event it contains.<br><br>Let's take an example where we want to pass a System Exclusive event that consists of 7 bytes. Well, the first thing that we need to do is redeclare the MIDIEVENT structure, thereby creating a new structure. I'll call this MY_SYSEX_EVT.<br><br>typedef struct { <br> DWORD dwDeltaTime;<br> DWORD dwStreamID;<br> DWORD dwEvent;<br> unsigned char dwParams[8];<br>} MY_SYSEX_EVT;<br><br>See? I declared an array large enough to hold my extra bytes. Why did I use a size of 8 rather than 7? Well, the Stream API requires that the size of all MIDIEVENT structures passed to it be aligned on a doubleword boundary. So, if I'm going to stuff this MIDIEVENT into a buffer containing another MIDIEVENT that comes after it, I want that second MIDIEVENT to be properly aligned. In essense, I've added a pad byte above to make the structure's size a multiple of 4, and thereby ensure that subsequent MIDIEVENT structures in the same array are properly aligned. <br>OK, now let's initialize it. For such a MIDIEVENT, its Event Type byte must be the value MEVT_LONGMSG. Furthermore, the remaining 3 bytes of the dwEvent field must be a 24-bit count of the number of bytes in our System Exclusive message (ie, 7 in this case).<br><br>MY_SYSEX_EVT mevt;<br>unsigned char sysEx[] = {0xF0, 0x7F, 0x7F, 0x01, 0x02, 0xF7};<br><br>mevt.dwDeltaTime = 100; /* Just for the hell of it, delay 100 clocks before sending it */<br>mevt.dwStreamID = 0;<br>mevt.dwEvent = ((unsigned long)MEVT_LONGMSG<<24) | sizeof(sysEx);<br>memcpy(&mevt.dwParams[0], &sysEx[0], sizeof(sysEx));<br><br>What other Event Types are there (besides the "short" type for MIDI messages of 3 bytes or less, and the "long" type for System Exclusive)? The most useful is a Tempo event. For such a MIDIEVENT, its Event Type byte must be the value MEVT_TEMPO. Furthermore, the remaining 3 bytes of the dwEvent field must be a Tempo value as a 24-bit value. It is expressed in microseconds per quarter note, just like in the MIDI File Format's MetaTempo event. <br>Another Event Type is a comment. Like a System Exclusive MIDIEVENT, the characters that form the comment are appended to the end of the MIDIEVENT structure itself. Its Event Type byte must be the value MEVT_COMMENT. Furthermore, the remaining 3 bytes of the dwEvent field must be a 24-bit count of the number of chars in the comment. Again, pad out the structure's size to a multiple of 4 if you're going to put it in an array with other MIDIEVENTs after it. Windows ignores comment events, so you can use them for your own purposes.<br><br>Another Event Type is a NOP (no operation). It's a "short" type of event, like Tempo and regular MIDI Voice messages, so no extra fields are appended to the MIDIEVENT structure. Its Event Type byte must be the value MEVT_NOP. Furthermore, the remaining 3 bytes of the dwEvent field can be used for any purpose you wish. Windows ignores NOP events, so you can use them for your own purposes.<br><br>The final Event Type is a version event. It has a MIDISTRMBUFFVER structure appended to the end of the MIDIEVENT structure. This extra structure just contains version information about the Stream. Its Event Type byte must be the value MEVT_VERSION. Furthermore, the remaining 3 bytes of the dwEvent field must be a 24-bit count of the size of the MIDISTRMBUFFVER structure.<br><br>OK, I'm sure that there is one nagging question in your mind. If MIDIEVENT structures can be variable size, depending upon the Event Type, then how do you declare a static array of them? Well, you can't really (unless you happen to stick to only using the "short" types). What you'll need to do is just copy them into one large char buffer, using a pointer to that buffer, and do some creative casting. For example, here I copy two MIDIEVENTs into one buffer:<br><br>MY_SYSEX_EVT * xevt;<br>MY_MIDI_EVT * mevt;<br>unsigned char sysEx[] = {0xF0, 0x7F, 0x7F, 0x01, 0x02, 0xF7};<br>unsigned char buffer[20];<br><br>/* Format for a Note-On */<br>mevt = (MY_MIDI_EVT *)&buffer[0];<br>mevt->dwDeltaTime = 0;<br>mevt->dwStreamID = 0;<br>mevt->dwEvent = 0x00403C90;<br>mevt++;<br><br>/* Format for a System Exclusive */<br>xevt = (MY_SYSEX_EVT *)mevt;<br>xevt->dwDeltaTime = 100;<br>xevt->dwStreamID = 0;<br>xevt->dwEvent = ((unsigned long)MEVT_LONGMSG<<24) | sizeof(sysEx);<br>memcpy(&xevt->dwParams[0], &sysEx[0], sizeof(sysEx));<br><br>Of course, I should make sure that buffer is aligned upon a doubleword boundary (which you can do with your compiler's alignment directive). <br>But another approach to take is to simply declare your array to be an array of unsigned longs. After all, since all MIDIEVENT structures need to be padded out to a multiple of 4 bytes, then the net result is that each MIDIEVENT structure consists 3 or more unsigned longs. So, here's the above array initialized this way:<br><br>unsigned long buffer[] = { 0, 0, 0x00403C90, /* The Note-On */<br> 0, 0, ((unsigned long)MEVT_LONGMSG<<24) | 7, 0x017F7FF0, 0x00F70201}; /* The SysEx */<br><br>Note the order of the packed SysEx bytes. Each unsigned long contains the next 4 bytes, and remember that we're using little endian order on each unsigned long. <br><br><br>--------------------------------------------------------------------------------<br><br>Starting playback<br>You start playback by calling midiStreamRestart() as so: <br>unsigned long err;<br><br>err = midiStreamRestart(outHandle);<br>if (err)<br>{<br> printf("An error starting playback!/n");<br>}<br><br>Note that when playback begins, the stream device's current time is set to 0 (if the playback hasn't been paused). <br>Stop the current playback<br>You can stop a playback in progress by calling midiStreamStop() as so: <br>unsigned long err;<br><br>err = midiStreamStop(outHandle);<br>if (err)<br>{<br> printf("An error stopping playback!/n");<br>}<br><br>This flushes all of the queued MIDIEVENTs, and you can subsequently midiOutUnprepareHeader() the MIDIHDRs. (The MHDR_DONE bit is set in the dwFlags field of all queued MIDIHDR structures). <br>It also turns off any notes that are still turned on. (By contrast, midiOutReset() turns off all notes regardless, and is more of a "panic" button type of response to turn off any "stuck notes").<br><br>Calling this function when the output is already stopped has no effect, and doesn't return an error.<br><br>WARNING! WARNING! WARNING! The Windows Stream Manager appears to be severely broken. Calling midiStreamStop() (or midiStreamPause) does indeed stop playback, but it appears to also close the stream handle that you pass to it. The result is that a subsequent call to another function using that handle results in the Stream Manager returning an error that the handle is no longer valid. The only options you have are:<br><br>Open a stream device immediately before you intend to queue and play MIDI events. Play those events. Then close the stream device. <br><br>Once you call midiStreamRestart(), never call midiStreamStop() or midiStreamPause(). Simply maintain your own clock variable and "play flag" variable that your callback can test to see whether it should continue queueing MIDI events for playback and increment its clock, or just do nothing. In other words, you're going to start up the stream once and leave it constantly running in the background, waiting for more MIDIEVENTs to be queued.<br><br>Pause/Resume the current playback<br>You can pause a playback in progress by calling midiStreamPause() as so: <br>unsigned long err;<br><br>err = midiStreamPause(outHandle);<br>if (err)<br>{<br> printf("An error pausing playback!/n");<br>}<br><br>If you wish to subsequently resume playback from the point at which it was paused, then call midiStreamRestart(). The stream device's time is not reset to 0, and playback resumes. If instead, you wish to stop the device and flush the queued MIDIEVENTs (perhaps in order to reset its time to 0 and start playback from the beginning of a sequence), then instead call midiStreamStop(). <br>Calling this function when the output is already paused has no effect, and doesn't return an error.<br><br><br>--------------------------------------------------------------------------------<br><br>Setting/Querying the Tempo and Timebase<br>Before playing queued MIDIEVENTs, you'll want to set the Timebase for the stream device. This is equivalent to the MIDI File Format's Division. It tells the stream device how to scale the dwDeltaTime field of each MIDIEVENT. (ie, Consider it a SMPTE time in 30 fps, or a time-stamp at 96 PPQN, or a time-stamp at 120 PPQN, etc). You use midiStreamProperty() with the MIDIPROP_SET and MIDIPROP_TIMEDIV flags as the third arg. The second arg is a pointer to a MIDIPROPTIMEDIV structure whose dwTimeDiv field is initialized to your desired Timebase. For PPQN Timebases, just specify the PPQN value, for example 96 PPQN. For SMPTE, the low two bytes are as per the MIDI File Format's Division field when SMPTE is specified. <br>Here I set the Timebase to 96 PPQN (which is the default if you don't specify a Timebase):<br><br>unsigned long err;<br>MIDIPROPTIMEDIV prop;<br><br>prop.cbStruct = sizeof(MIDIPROPTIMEDIV);<br>prop.dwTimeDiv = 96;<br>err = midiStreamProperty(outHandle, (LPBYTE)&prop, MIDIPROP_SET|MIDIPROP_TIMEDIV);<br>if (err)<br>{<br> printf("An error setting the timebase!/n");<br>}<br><br>Besides putting Tempo events in the stream in order to set tempo at any given time, you can also call midiStreamProperty() with the MIDIPROP_SET and MIDIPROP_TEMPO flags as the third arg to set tempo. The second arg is a pointer to a MIDIPROPTEMPO structure whose dwTempo field is initialized to your desired Tempo. Note that this is irrelevant when using a SMPTE Timebase. <br>Here I set the Tempo to 120 BPM (which is the default if you don't specify a Tempo):<br><br>unsigned long err;<br>MIDIPROPTEMPO prop;<br><br>prop.cbStruct = sizeof(MIDIPROPTEMPO);<br>prop.dwTempo = 0x0007A120;<br>err = midiStreamProperty(outHandle, (LPBYTE)&prop, MIDIPROP_SET|MIDIPROP_TEMPO);<br>if (err)<br>{<br> printf("An error setting the tempo!/n");<br>}<br><br>Of course, you can query a stream device's current timebase or tempo by using the MIDIPROP_GET flag instead of MIDIPROP_SET. Windows fills in the MIDIPROPTIMEDIV or MIDIPROPTEMPO you pass. <br>Here I query the current Tempo:<br><br>unsigned long err;<br>MIDIPROPTEMPO prop;<br><br>prop.cbStruct = sizeof(MIDIPROPTEMPO);<br>err = midiStreamProperty(outHandle, (LPBYTE)&prop, MIDIPROP_GET|MIDIPROP_TEMPO);<br>if (err)<br>{<br> printf("An error requesting the tempo!/n");<br>}<br>else<br>{<br> printf("Tempo = %u/n", prop.dwTempo);<br>}<br><br><br>--------------------------------------------------------------------------------<br><br>Notification during playback<br>I previously talked about how you want to queue up another array of MIDIEVENTs as soon as one array finishes playing. So how do you receive such notification from Windows? When you call midiStreamOpen(), you can tell it how you want Windows to notify you. In our above example, we specified CALLBACK_NULL (ie, we didn't want Windows to notify us). But there are other choices as follows: <br>CALLBACK_EVENT -- You allocate some Event with CreateEvent(), and Windows uses this to signal your app. (ie, Your app can wait on that Event signal, for example with WaitForSingleObject). You pass the handle of the Event as the 4th arg to midiStreamOpen(). <br><br>CALLBACK_THREAD -- Windows causes some suspended thread within your app to run. (ie, Your app's thread can suspend itself via SuspendThread). You pass the Thread ID of the desired thread to be run as the 4th arg to midiStreamOpen().<br><br><br>CALLBACK_WINDOW -- Windows sends a message to some open window in your app. The parameters for the message will contain additional information about what caused Windows to send that message. You pass the desired window's handle as the 4th arg to midiStreamOpen().<br><br><br>CALLBACK_FUNCTION -- Windows directly calls some function in your app. It passes args that contain additional information about what caused Windows to call your function. You pass a pointer to the desired function as the 4th arg to midiStreamOpen(). The 5th arg to midiStreamOpen() can be anything you desire, and this will be passed to your callback function each time that Windows calls your callback.<br><br>The latter two methods allow you to better determine what exactly caused Windows to notify you, because they supply additional information to you.<br><br>So when does Windows notify you? Here are the times when Windows notifies you:<br><br>When you open a Stream device via midiStreamOpen(). <br><br>When you close a Stream Device via midiStreamClose().<br><br><br>When midiStreamOut() encounters an event that has its MEVT_F_CALLBACK flag set.<br><br><br>When midiStreamOut() finishes playing a MIDIHDR's block of data.<br><br>Here's an example of setting the MEVT_F_CALLBACK flag of an event.<br><br>xevt->dwEvent = ((unsigned long)MEVT_LONGMSG<<24) | sizeof(sysEx) | MEVT_F_CALLBACK;<br><br>Now, after Windows plays that above event, it notifies you. For example, if you've chosen CALLBACK_FUNCTION method, Windows calls your callback function.<br><br>Windows also will notify you when the last event in a MIDIHDR's block of events is played. This is the point at which you will queue the next array of MIDIEVENTs using that same array (and its associated MIDIHDR).<br><br>In fact, you could strategically place NOP events with their dwDeltaTime set so that they just happen to fall upon each musical downbeat, and set the MEVT_F_CALLBACK flag of each of those event's dwEvent fields. Then, everytime that your callback is called, you can update a graphical display counting off the beat. (My Stream Callback C example shows that using CALLBACK_FUNCTION method). <br><br><br>--------------------------------------------------------------------------------<br><br>Here's an example of using the CALLBACK_EVENT method to play an array of MIDIEVENTs. This is a complete example that shows you all the bare minimum details you need to know to play a stream of MIDIEVENTs.<br><br>/* The array of MIDIEVENTs to be output. We only have 2 */<br>unsigned long myNotes[] = {0, 0, 0x007F3C90, /* A note-on */<br>192, 0, 0x00003C90}; /* A note-off. It's the last event in the array */<br><br>HANDLE event;<br>HMIDISTRM outHandle;<br>MIDIHDR midiHdr;<br>MIDIPROPTIMEDIV prop;<br>unsigned long err;<br><br>/* Allocate an Event signal */<br>if ((event = CreateEvent(0, FALSE, FALSE, 0)))<br>{<br> /* Open default MIDI Out stream device. Tell it to notify via CALLBACK_EVENT and use my created Event */<br> err = 0;<br> if (!(err = midiStreamOpen(&outHandle, &err, 1, (DWORD)event, 0, CALLBACK_EVENT)))<br> {<br> /* Windows signals me once when the driver is opened. Clear that now */<br> ResetEvent(event);<br><br> /* Set the timebase. Here I use 96 PPQN */<br> prop.cbStruct = sizeof(MIDIPROPTIMEDIV);<br> prop.dwTimeDiv = 96;<br> midiStreamProperty(outHandle, (LPBYTE)&prop, MIDIPROP_SET|MIDIPROP_TIMEDIV);<br><br> /* If you wanted something other than 120 BPM, here you should also set the tempo */<br><br> /* Store pointer to our stream (ie, array) of messages in MIDIHDR */<br> midiHdr.lpData = (LPBYTE)&myNotes[0];<br><br> /* Store its size in the MIDIHDR */<br> midiHdr.dwBufferLength = midiHdr.dwBytesRecorded = sizeof(myNotes);<br><br> /* Flags must be set to 0 */<br> midiHdr.dwFlags = 0;<br><br> /* Prepare the buffer and MIDIHDR */<br> err = midiOutPrepareHeader(outHandle, &midiHdr, sizeof(MIDIHDR));<br> if (!err)<br> {<br> /* Queue the Stream of messages. Output doesn't actually start<br> until we later call midiStreamRestart().<br> */<br> err = midiStreamOut(outHandle, &midiHdr, sizeof(MIDIHDR));<br> if (!err)<br> {<br> /* Start outputting the Stream of messages. This will return immediately<br> as the stream device will time out and output the messages on its own in<br> the background.<br> */<br> err = midiStreamRestart(outHandle);<br> if (!err)<br><br> /* Wait for playback to stop. Windows signals me using that Event I created */<br> WaitForSingleObject(event, INFINITE);<br> }<br><br> /* Unprepare the buffer and MIDIHDR */<br> midiOutUnprepareHeader(outHandle, &midiHdr, sizeof(MIDIHDR));<br> }<br> <br> /* Close the MIDI Stream */<br> midiStreamClose(outHandle);<br> }<br><br> /* Free the Event */<br> CloseHandle(event);<br>}<br><br><br>--------------------------------------------------------------------------------<br><br>If using the CALLBACK_FUNCTION method, then you need to write a function that has the following declaration (although you can name the function anything you like):<br><br>void CALLBACK midiCallback(HMIDIOUT handle, UINT uMsg, DWORD dwInstance, DWORD dwParam1, DWORD dwParam2);<br><br>As mentioned, you pass a pointer to this function as the 4th arg to midiStreamOpen(). The 5th arg to midiStreamOpen() can be anything you desire, and this will be passed to your callback function each time that Windows calls your callback. Windows calls your function whenever 1 of 4 possible things happen: <br>When you open a Stream device via midiStreamOpen(). In this case, the uMsg arg to your callback will be MOM_OPEN. The handle arg will be the same as what is returned to midiStreamOpen(). The dwInstance arg is whatever I passed to midiStreamOpen() as its dwInstance arg. <br><br>When you close a Stream Device via midiStreamClose().In this case, the uMsg arg to your callback will be MOM_CLOSE. The handle arg will be the same as what was passed to midiStreamClose(). The dwInstance arg is the 5th arg you passed to midiStreamOpen() when you initially opened this handle.<br><br><br>When midiStreamOut() encounters an event that has its MEVT_F_CALLBACK flag set. In this case, the uMsg arg to your callback will be MOM_POSITIONCB. The handle arg will be the same as what is passed to midiStreamOut(). The dwInstance arg is the 5th arg you passed to midiStreamOpen() when you initially opened this handle. The dwParam1 arg points to the MIDIHDR that you passed to midiStreamOut(). The dwOffset field of the MIDIHDR will indicate the byte offset into the buffer for the event that caused this callback to be called. See my note below.<br><br><br>When midiStreamOut() finishes playing a MIDIHDR's block of data. In this case, the uMsg arg to my callback will be MOM_DONE. The handle arg will be the same as what is passed to midiStreamOut(). The dwInstance arg is the 5th arg you passed to midiStreamOpen() when you initially opened this handle. The dwParam1 arg points to the MIDIHDR that you passed to midiStreamOut().<br><br>NOTE: The dwParam2 arg is not used. This is reserved for future use.<br><br>Here's an example of using the CALLBACK_FUNCTION method to play an array of MIDIEVENTs. This is a complete example that shows you all the bare minimum details you need to know to play a stream of MIDIEVENTs.<br><br>/* The array of MIDIEVENTs to be output. We only have 2 */<br>unsigned long myNotes[] = {0, 0, 0x007F3C90, /* A note-on */<br>192, 0, 0x00003C90}; /* A note-off. It's the last event in the array */<br><br>HANDLE Event;<br><br>void CALLBACK midiCallback(HMIDIOUT handle, UINT uMsg, DWORD dwInstance, DWORD dwParam1, DWORD dwParam2)<br>{<br> LPMIDIHDR lpMIDIHeader;<br> MIDIEVENT * lpMIDIEvent;<br><br> /* Determine why Windows called me */<br> switch (uMsg)<br> {<br> /* Got some event with its MEVT_F_CALLBACK flag set */<br> case MOM_POSITIONCB:<br><br> /* Assign address of MIDIHDR to a LPMIDIHDR variable. Makes it easier to access the<br> field that contains the pointer to our block of MIDI events */<br> lpMIDIHeader = (LPMIDIHDR)dwParam1;<br><br> /* Get address of the MIDI event that caused this call */<br> lpMIDIEvent = (MIDIEVENT *)&(lpMIDIHeader->lpData[lpMIDIHeader->dwOffset]);<br><br> /* Normally, if you had several different types of events with the<br> MEVT_F_CALLBACK flag set, you'd likely now do a switch on the highest<br> byte of the dwEvent field, assuming that you need to do different<br> things for different types of events.<br> */<br><br> break;<br><br> /* The last event in the MIDIHDR has played */<br> case MOM_DONE:<br><br> /* Wake up main() */<br> SetEvent(Event);<br><br> break;<br><br><br> /* Process these messages if you desire */<br> case MOM_OPEN:<br> case MOM_CLOSE:<br><br> break;<br> }<br>}<br><br>int main(int argc, char **argv)<br>{<br> HMIDISTRM outHandle;<br> MIDIHDR midiHdr;<br> MIDIPROPTIMEDIV prop;<br> unsigned long err;<br><br> /* Allocate an Event signal */<br> if ((event = CreateEvent(0, FALSE, FALSE, 0)))<br> {<br> /* Open default MIDI Out stream device. Tell it to notify via CALLBACK_EVENT and use my created Event */<br> err = 0;<br> if (!(err = midiStreamOpen(&outHandle, &err, 1, (DWORD)midiCallback, 0, CALLBACK_FUNCTION)))<br> {<br> /* Set the timebase. Here I use 96 PPQN */<br> prop.cbStruct = sizeof(MIDIPROPTIMEDIV);<br> prop.dwTimeDiv = 96;<br> midiStreamProperty(outHandle, (LPBYTE)&prop, MIDIPROP_SET|MIDIPROP_TIMEDIV);<br><br> /* If you wanted something other than 120 BPM, here you should also set the tempo */<br><br> /* Store pointer to our stream (ie, array) of messages in MIDIHDR */<br> midiHdr.lpData = (LPBYTE)&myNotes[0];<br><br> /* Store its size in the MIDIHDR */<br> midiHdr.dwBufferLength = midiHdr.dwBytesRecorded = sizeof(myNotes);<br><br> /* Flags must be set to 0 */<br> midiHdr.dwFlags = 0;<br><br> /* Prepare the buffer and MIDIHDR */<br> err = midiOutPrepareHeader(outHandle, &midiHdr, sizeof(MIDIHDR));<br> if (!err)<br> {<br> /* Queue the Stream of messages. Output doesn't actually start<br> until we later call midiStreamRestart().<br> */<br> err = midiStreamOut(outHandle, &midiHdr, sizeof(MIDIHDR));<br> if (!err)<br> {<br> /* Start outputting the Stream of messages. This will return immediately<br> as the stream device will time out and output the messages on its own in<br> the background.<br> */<br> err = midiStreamRestart(outHandle);<br> if (!err)<br><br> /* Wait for playback to stop. Windows calls my callback, which will set this signal */<br> WaitForSingleObject(event, INFINITE);<br> }<br><br> /* Unprepare the buffer and MIDIHDR */<br> midiOutUnprepareHeader(outHandle, &midiHdr, sizeof(MIDIHDR));<br> }<br> <br> /* Close the MIDI Stream */<br> midiStreamClose(outHandle);<br> }<br><br> /* Free the Event */<br> CloseHandle(event);<br> }<br><br> return(0);<br>}<br><br>NOTE: If you happen to have two or more events that occur upon the same beat (or close enough such that Windows doesn't even have time to call your callback once before both events time out), and both their MEVT_F_CALLBACK flags are set, then Windows only calls your callback once for all of those events. The dwOffset of the MIDIHDR would reference the last such event. Therefore, don't bother setting the MEVT_F_CALLBACK flag of more than one event that occurs upon a given time. <br>Furthermore, while Windows calls your callback, it continues timing out the next event in the background. If it happens that the next event happens to have its MEVT_F_CALLBACK flag also set, and it times out while your callback is still processing that previous MEVT_F_CALLBACK event, then Windows will call your callback again. In other words, if you're not sure that you've placed enough time inbetween each of your MEVT_F_CALLBACK flagged events in order to do what you need to get done in your callback, then you had better make sure your callback handles reentrancy properly. (ie, Don't use global variables whose value changes, or use some sort of mechanism to arbitrate access to those variables such as a Mutex). For practical purposes, you should avoid setting the MEVT_F_CALLBACK flag of an event whose time makes it occur less than 25 milliseconds after the previous event with its MEVT_F_CALLBACK flag set. Microsoft recommends the following approach to handling a situation where MEVT_F_CALLBACK events are timing out quicker than your callback can finish its work: "The callback should use the MIDIHDR's dwOffset to decide what to do. If it is expecting this to reference a particular location (ie, event) but it instead references a location somewhere further along in the array of MIDIEVENTs, it means that your callback is not keeping up or that the MEVT_F_CALLBACK flagged events are scheduled too closely together for the capabilities of the computer. The best thing to do in this case is to just throw the callback away because there will be another one in the queue that will be delivered just as soon as processing of the current callback is completed."<br><br><br>--------------------------------------------------------------------------------<br><br>Querying the current playback time<br>You can retrieve the current playback time by calling midiStreamPosition(). It fills in an MMTIME structure. <br>Before calling midiStreamPosition(), you set the wType field of the MMTIME structure to indicate the time format you desire returned. After calling midiStreamPosition(), you should check the wType field. Some stream devices may not support returning certain time formats, for example, a SMPTE format. In that case, midiStreamPosition() will set the wType field to the closest supported format, and return that time.<br><br>The allowable time formats (ie, for the wType field) are:<br><br>TIME_BYTES Current byte offset from beginning of the playback.<br>TIME_MIDI MIDI time (ie, MIDI Song Position Pointer).<br>TIME_MS Time in milliseconds.<br>TIME_SAMPLES Number of waveform-audio samples (for syncing to WAVE playback).<br>TIME_SMPTE SMPTE time.<br>TIME_TICKS PPQN clocks.<br><br>Here I query the current Time. (Note that the Time is set to 0 when playback starts, except for a paused playback that is resumed). <br>unsigned long err;<br>MMTIME time;<br><br>/* Request millisecond time returned */<br>time.wType = TIME_MS;<br>err = midiStreamPosition(outHandle, &time, sizeof(MMTIME));<br>if (err)<br>{<br> printf("An error requesting the time!/n");<br>}<br>else switch (time.wType)<br>{<br> case TIME_BYTES:<br> printf("%u bytes have been played so far./n", time.cb);<br> break;<br><br> case TIME_MIDI:<br> printf("Midi Song Position Pointer = %u/n", time.midi.songptrpos);<br> break;<br><br> case TIME_MS:<br> printf("Millisecond Time = %u/n", time.ms);<br> break;<br><br> case TIME_SAMPLES:<br> printf("%u digital audio samples have been played so far./n", time.sample);<br> break;<br><br> case TIME_SMPTE:<br> printf("SMPTE Time (%u fps) = %u:%u:%u:%u/n", time.smpte.fps, time.smpte.hour, time.smpte.min, time.smpte.sec, time.smpte.frame);<br> break;<br><br> case TIME_TICKS:<br> printf("PPQN Clocks = %u/n", time.ticks);<br>}<br><br>Outputting MIDI data immediately while a Stream is playing<br>Note that you can call midiOutShortMsg() or midiOutLongMsg() while the Stream (ie, queued MIDI events) is playing. In such a case, the MIDI data you send via these two functions will be output as soon as possible. (But don't utilize running status with midiOutShortMsg. Always specify the full MIDI message with Status). <br><br><br>--------------------------------------------------------------------------------<br><br>Stream recording<br>Currently, the Stream API does not support recording. If you need to do MIDI recording, you'll have to use the Low level MIDI API and a MultiMedia Timer instead of the Stream API.