home *** CD-ROM | disk | FTP | other *** search
/ Multimedia Classic / MultimediaClassic.mdf / app_midi / drum / mididrum.asc < prev    next >
Encoding:
Text File  |  1993-01-19  |  54.4 KB  |  1,073 lines

  1.  
  2.  
  3.  *****  Computer Select, October 1992 : Doc #24357  *****
  4.  
  5. Journal:   PC Magazine  May 12 1992 v11 n9 p394(3)
  6.            * Full Text COPYRIGHT Ziff-Davis Publishing Co. 1992.
  7. -----------------------------------------------------------------------------
  8. Title:     Creating a computer drum machine, part 1. (Environments)(column)
  9.            (Tutorial)
  10. Author:    Petzold, Charles.
  11. AttFile:    Program:  DRUM1.EXE  Self extracting archive.
  12.             Table:  L09ENT1.WKS  Drum's source files.
  13.  
  14. Abstract:  A guide to creating a Musical Instrument Digital Interface
  15.            (MIDI)-based drum machine is presented.  Drum machines are a
  16.            special type of sequencer that is somewhat simpler than a complete
  17.            musical sequencer because it replicates percussion and does not
  18.            need to define musical pitch.  The two parameters of a drum
  19.            machine are percussion instrument and time.  Microsoft's
  20.            Multimedia Windows supports several MIDI channels, one of which is
  21.            usually reserved as a non-melodic drum channel.  The programmer
  22.            should send a percussion channel the Program Change message for
  23.            instrument voice 0; there are 47 different percussion sounds
  24.            available as defined by the MIDI Manufacturers Association.  A
  25.            sample program is discussed.
  26. -----------------------------------------------------------------------------
  27. Descriptors..
  28. Topic:     MIDI
  29.            Tutorial
  30.            Musical Instruments
  31.            Programming Instruction.
  32. Feature:   illustration
  33.            table.
  34.  
  35. Record#:   12 162 077.
  36. -----------------------------------------------------------------------------
  37. Full Text:
  38.  
  39. by Charles Petzold
  40.  
  41. A strong, regular drumbeat is the crucial rhythmic foundation of much of
  42. American popular music.  Under the influence of African-American culture,
  43. jazz, rhythm and blues, and rock 'n' roll have all been built around the
  44. driving pulse of percussion.
  45.  
  46. Drummers are often highly regarded for their rhythmic accuracy and precision,
  47. and it is not uncomplimentary to say that a drummer "plays like a machine."
  48. Although electronic drum machines can't capture the improvised variations
  49. that make good drumming so enjoyable, a drum machine can still be a
  50. reasonable compromise.  Standalone electronic drum machines have been around
  51. for years, and now even inexpensive small synthesizers often include several
  52. built-in percussion patterns, letting you switch from tango to rock to bossa
  53. nova riffs with the push of a button.
  54.  
  55. While you may associate these incessant, repetitive rhythms with tacky lounge
  56. singers or the worst of house music, real drum machines can be much more
  57. versatile.  They let you customize your rhythms with a variety of percussion
  58. instruments, and can be very useful for experimentation when composing or
  59. arranging music, even if you switch to real drummers for performances or
  60. recording sessions.  As an accompaniment to a solo guitar, piano, saxophone,
  61. or whatever, drum machines can be a lot of fun.
  62.  
  63. DRUM MACHINES AND SEQUENCING
  64.  
  65. A drum machine is a sequencer, but a special case.  Back in the old days (two
  66. decades ago), sequencers were stand- alone devices that could store a
  67. sequence of musical notes and play them back, very often repeated in a cyclic
  68. pattern.  Several forward-looking rock bands used sequencers during that
  69. period, including Pink Floyd, The Who, and Tangerine Dream.  During the mid
  70. and late 1970s, sequencers became quite common in disco and some varieties of
  71. New Wave rock.
  72.  
  73. Standalone sequencers are still used in performance settings, although for
  74. studio or home recording they've largely been replaced with computers.
  75. Because of the increased storage capacity of a computer, sequencing software
  76. can handle entire multi-instrument compositions rather than just basic
  77. repeated patterns of notes.
  78.  
  79. Drum machines are somewhat simpler than full-fledged sequencers, though.
  80. Compositions played by sequencing software involve (at the very least) three
  81. parameters: pitch, time, and the instrument voice.  Because each of the
  82. various percussion instruments played by drum machines has a single pitch (or
  83. no definable musical pitch at all), drum machines can be limited to two
  84. parameters: the percussion instrument and time.
  85.  
  86. In this column and the next two, I'll describe a computer drum machine
  87. program called DRUM that I wrote for Windows with Multimedia Extensions.
  88. This program lets you construct a sequence of up to 32 notes using 47
  89. different percussion sounds.  The program plays the sequence repetitively at
  90. a selectable tempo and volume.  For accurate timing, DRUM requires a dynamic
  91. link library called DRUMDLL, which I'll describe in the next column.  The
  92. final column will discuss a second module used in the program that has code
  93. to store percussion patterns on-disk.
  94.  
  95. THE MIDI CHANNELS
  96.  
  97. As I've discussed in previous columns, a multimedia PC requires an internal
  98. synthesizer based on the industry-standard Musical Instrument Digital
  99. Interface (MIDI).  A multimedia PC also has at least one MIDI Out port for
  100. connecting an external MIDI synthesizer, and a MIDI In port for connecting an
  101. external MIDI controller (such as a keyboard).
  102.  
  103. Multimedia Windows includes support for MIDI through function calls beginning
  104. with the prefixes midiOut and midiIn, and through the Media Control Interface
  105. (MCI).  As we've seen, the primary purpose of the midiOut functions is to
  106. send 2-, 3-, and 4-byte MIDI messages to a synthesizer to play musical notes.
  107. Most MIDI messages include a 4-bit code that identifies a MIDI channel.
  108. Generally, a Program Change message is sent to a synthesizer to select an
  109. instrument voice for a particular channel.  Then, Note On and Note Off
  110. messages for that channel play particular pitches using that instrument
  111. voice.  These messages include key numbers to identify the note, where key
  112. number 60 is middle C. The use of 16 channels allows (in theory) 16 different
  113. instrument sounds to play simultaneously.
  114.  
  115. One channel, however, is usually reserved as a nonmelodic percussion channel.
  116. In this case, Note On messages do not trigger different pitches for a
  117. particular instrument voice, but instead trigger different percussion sounds
  118. based on the key numbers.
  119.  
  120. GENERAL MIDI MODE
  121.  
  122. The MIDI Manufacturers Association has recently defined a standard for
  123. synthesizers to facilitate the interchange of computer-based MIDI sequences.
  124. This standard is called General MIDI Mode.  In particular, General MIDI Mode
  125. defines a collection of 128 instrument voices selectable with Program Change
  126. messages.  It also assigns channels 0 through 8 and 10 through 15 as the
  127. melodic channels for which these instrument voices may be chosen.  (The term
  128. melodic implies that different key numbers play different pitches.) Channel 9
  129. is the percussion channel, for which 47 different percussion sounds are
  130. defined.
  131.  
  132. When using the percussion channel, you should send it a Program Change
  133. message for instrument voice 0.  You can then play the 47 different
  134. percussion sounds by sending Note On and Note Off messages using key numbers
  135. ranging from 35 (an acoustic bass drum) through 81 (a triangle).
  136.  
  137. Multimedia Windows generally adheres to the General MIDI Mode specification,
  138. with one significant departure: Because many inexpensive MIDI synthesizers
  139. cannot match the level of polyphony (the number of simultaneous sounds)
  140. implied by General MIDI Mode, Multimedia Windows classifies synthesizers as
  141. either base-level or extended.
  142.  
  143. An extended synthesizer (capable of playing nine melodic instruments with
  144. 16-note polyphony and percussion with 16-note polyphony) uses channels 0
  145. through 8 for the melodic voices and channel 9 for percussion.  This is the
  146. same as General MIDI Mode, but without the top six channels.  A base-level
  147. synthesizer (capable of playing three melodic instruments with 6-note
  148. polyphony and percussion with 3-note polyphony) uses channels 12 through 14
  149. for the melodic voices and channel 15 for percussion.
  150.  
  151. If you have a synthesizer that uses a different channel for percussion, or if
  152. it has a percussion channel in which the key numbers trigger percussion
  153. sounds other than the ones defined by Multimedia Windows, that's a job for
  154. the MIDI Mapper.  This MIDI output device uses mapping tables (which can be
  155. created in the MIDI Mapper utility invoked from the Control Panel) to alter
  156. MIDI messages on their way to a synthesizer.
  157.  
  158. USING THE DRUM PROGRAM
  159.  
  160. We are now ready to start up the program.  When you first run DRUM, it looks
  161. like Figure 1.  The 47 different percussion instruments are listed by name in
  162. two columns on the left half of the window.  (Figuring out how to fit these
  163. names on the screen was one of the hardest jobs in designing this program!)
  164. The grid on the right is a two-dimensional array of percussion sound versus
  165. time.  Each instrument is associated with a row in the grid.  The 32 columns
  166. are 32 beats.  If you think of these 32 beats as occurring within a measure
  167. of 4/4 time (four quarter notes per measure), then each beat corresponds to a
  168. 32nd note.
  169.  
  170. When you select Running from the Sequence menu, the program will attempt to
  171. open the MIDI Mapper device.  If it's unsuccessful, you'll get a message box.
  172. Otherwise, you'll see a little "bouncing ball" skip across the bottom of the
  173. grid as each beat is played.
  174.  
  175. You can click with the mouse anywhere within the grid to play a percussion
  176. sound during that beat.  Use the left mouse button for a base-level
  177. synthesizer; the square will turn light gray.  Use the right mouse button for
  178. an extended synthesizer; the square will turn dark gray.  If you click with
  179. both mouse buttons (either together or independently), the square will turn
  180. black and the percussion sound will be heard on both a base-level and an
  181. extended synthesizer.  Clicking again with either or both buttons will turn
  182. off the sound for that beat.
  183.  
  184. Across the top of the grid is a dot every four beats.  Those dots simply make
  185. it easier to pinpoint your button clicks without too much counting.  At the
  186. upper-right-hand corner of the grid is a colon and bar (:|) that resembles a
  187. repeat sign used in traditional music notation and indicates the length of
  188. the sequence.  You can click with the mouse anywhere above the grid to put
  189. the repeat sign someplace else.  The sequence plays up to but not including
  190. the beat beneath the repeat sign.  If you want to create a waltz rhythm, for
  191. example, you should set the repeat mark for some multiple of three beats.
  192.  
  193. The horizontal scroll bar controls the velocity byte in the MIDI Note On
  194. messages.  This generally affects the volume of the sounds, although it may
  195. also affect timbre in some synthesizers.  The program initially sets the
  196. velocity scroll bar thumb in the center position.
  197.  
  198. The vertical scroll bar controls the tempo.  This is a logarithmic scale
  199. ranging from 1 second per beat when the thumb is at the bottom to 10
  200. milliseconds per beat at the top.  The program initially sets the tempo at
  201. 100 milliseconds (one-tenth of a second) per beat, with the scroll bar thumb
  202. in the center.
  203.  
  204. The File menu allows you to save and retrieve files with the extension .DRM,
  205. a format I invented.  These files are fairly small (512 bytes) and use the
  206. RIFF file format, which is recommended for all new Multimedia Windows data
  207. files.  The .DRM files include the data for both base-level and extended
  208. synthesizers, the number of beats for the sequence, the velocity, and the
  209. tempo.
  210.  
  211. The About option from the Help menu displays a dialog box that contains a
  212. very brief summary of the use of the mouse on the two scroll bars and their
  213. functions.
  214.  
  215. Finally, the Stopped option from the Sequence menu stops the music and closes
  216. the MIDI Mapper device after finishing the sequence being executed.
  217.  
  218. DEVICE-INDEPENDENT .DRM FILES
  219.  
  220. Let's say you and some of your friends all have my DRUM program and use it to
  221. create new and fascinating percussion sequences, which you pass around to
  222. each other in .DRM files.  Or perhaps you're interested in creating a
  223. collection of .DRM files yourself and uploading them to PC MagNet.  If you
  224. do, each .DRM file should include sequences for both base-level and extended
  225. synthesizers.  Otherwise, some people who try to play the file won't hear
  226. anything!
  227.  
  228. The sequence for base-level synthesizers should have 3 or fewer percussion
  229. sounds per beat; those for extended synthesizers can include up to 16
  230. percussion sounds per beat.  The result will look something like the DRUM
  231. screen display shown in Figure 2.
  232.  
  233. Though it's not readily discernible in the screenshot, some of the grid
  234. squares are black (meaning that the percussion sound is played on both
  235. base-level and extended synthesizers), and others are dark gray (meaning that
  236. the percussion sound is played only on extended synthesizers).  At no time
  237. are more than three percussion sounds played on the base-level synthesizer.
  238.  
  239. There's nothing wrong with trying to play more than three simultaneous
  240. percussion sounds on a base-level synthesizer, of course, but there's no
  241. guarantee that you'll hear them all.  Some synthesizers will give highest
  242. priority to the notes that come first and discard the rest when the limit has
  243. been reached, while others give highest priority to notes that come later and
  244. turn off the earlier notes.
  245.  
  246. It's probably easiest to make the sequence for a base-level synthesizer a
  247. subset of the extended-synthesizer sequence.  But there are alternatives.
  248. You may discover a more satisfactory solution in completely rethinking an
  249. arrangement for a base-level synthesizer.
  250.  
  251. The preparation of MIDI sequences for both base-level and extended
  252. syn-thesizers is very important in creating device-independent MIDI data for
  253. Multimedia Windows.  And keep in mind that you have the freedom to make the
  254. base-level sequence much different from the extended-synthesizer sequence to
  255. account for the lower number of simultaneous voices.  It is for such
  256. flexibility that the designers of Multimedia Windows made the base-level
  257. synthesizer channels separate from and not a subset of the
  258. extended-synthesizer channels.
  259.  
  260. THE DRUM SOURCE CODE
  261.  
  262. You can download all source code for DRUM, including the DRUM.EXE executable
  263. and two dynamic link libraries from PC MagNet.  Figure 3 lists all the files
  264. involved, with brief descriptions.
  265.  
  266. In the next column, I'll show code for the DRUMDLL dynamic link library.
  267. This DLL performs all the midiOut function calls.  To achieve the precision
  268. required of a program such as this (which can play sequences as quickly as 10
  269. ms.  per beat), DRUMDLL makes use of the Multimedia Windows high-resolution
  270. timer, using functions beginning with the prefix time.  Without these
  271. functions, DRUM would not be able to play faster than 55 ms.  per beat, and
  272. would be dependent on the mushy, imprecise nature of the normal Windows
  273. timer.
  274.  
  275. Meanwhile, have fun with the program.  Class resumes in two weeks.
  276.  
  277.  
  278.  *****  Computer Select, October 1992 : Doc #22006  *****
  279.  
  280. Journal:   PC Magazine  May 26 1992 v11 n10 p367(5)
  281.            * Full Text COPYRIGHT Ziff-Davis Publishing Co. 1992.
  282. -----------------------------------------------------------------------------
  283. Title:     Creating a computer drum machine, part 2. (DRUM utility software)
  284.            (Tutorial)
  285. Author:    Petzold, Charles.
  286. AttFile:    Program:  DRUMDLL.BCP  Make file for Borland C++ 3.0.
  287.             Program:  DRUMDLL.C  C source code.
  288.             Program:  DRUMDLL.DEF  Definition file.
  289.             Program:  DRUMDLL.H  Header file.
  290.             Program:  DRUMDLL.MSC  Make file for Microsoft C 7.0.
  291.  
  292. Abstract:  The timer provided in Microsoft Corp's Windows graphical user
  293.            interface (GUI) called SetTimer is easy to use and suffices for
  294.            clock programs and simple animation, but it is a faulty tool when
  295.            implemented for time-critical applications.  It is based on the
  296.            microcomputer's system clock and can only display elapsed time in
  297.            multiples of 55m seconds.  Music applications are typical cases
  298.            that require a better clock than the time provided by Windows.  A
  299.            utility called DRUM is a drum machine program for the Windows
  300.            environment written with Multimedia Extensions.  It utilizes the
  301.            percussion channel on Musical Instrument Digital Interface
  302.            (MIDI)-compliant musical instruments.
  303. -----------------------------------------------------------------------------
  304. Descriptors..
  305. Topic:     Clocks
  306.            Tutorial
  307.            Shareware
  308.            Programming
  309.            Music
  310.            Software Packages.
  311. Feature:   illustration
  312.            program.
  313.  
  314. Record#:   12 220 583.
  315. -----------------------------------------------------------------------------
  316. Full Text:
  317.  
  318. The standard Windows timer is certainly easy to use.  Just call SetTimer,
  319. specifying an elapsed time in milliseconds, and you'll receive WM_TIMER
  320. messages periodically.  It's a fine facility for clock programs and even for
  321. simple animation.
  322.  
  323. For time-critical applications, though, the Windows timer is a disaster.  In
  324. the first place, the timer is based on the PC system clock, which means that
  325. you can't receive WM_TIMER messages more frequently than every 55
  326. milliseconds or with an elapsed time that is not a multiple of 55
  327. milliseconds.  Second, the timer messages are not asynchronous.  They come
  328. through the program's message queue, and processing may be delayed if another
  329. program is at work.  Third, because a message queue can hold only one
  330. WM_TIMER message at any time, you're not even guaranteed that you'll get all
  331. the timer messages you expect.
  332.  
  333. Playing music is one such time-critical application for which the Windows
  334. timer is simply inadequate.  I tried to get away with it in the BACHTOCC
  335. program discussed in the April 14, 1992, column, but it was obvious that
  336. using the Windows timer was not a good solution, and I admitted that at the
  337. time.  Now it's time to do the job right.
  338.  
  339. In the previous issue I began discussing a drum machine program written for
  340. Microsoft Windows with Multimedia Extensions.  DRUM uses the percussion
  341. channel on an electronic music synthesizer conforming to the Musical
  342. Instrument Digital Interface (MIDI) specification.  The program will also run
  343. under Windows 3.1 (which includes Multimedia Extensions) if you have a MIDI
  344. board and a driver for it.
  345.  
  346. As shown in Figure 1, DRUM lets you interactively create a percussion pattern
  347. by clicking on a grid that represents 47 percussion sounds and 32 beats.  You
  348. can also adjust the velocity (which generally corresponds to the volume), the
  349. tempo (as fast as 10 milliseconds per beat), and the number of beats in the
  350. sequence.  In this column I'll discuss a dynamic link library called DRUMDLL
  351. that DRUM uses to play the MIDI sequence.
  352.  
  353. THE MULTIMEDIA TIME FUNCTIONS
  354.  
  355. To provide the accuracy we need to play MIDI sequences on the PC, Multimedia
  356. Windows includes a high-resolution timer that is implemented using seven
  357. functions beginning with the prefix time.  One of these functions is
  358. superfluous; DRUMDLL demonstrates the use of the other six.
  359.  
  360. The timer functions work with a callback function located in your code,
  361. specifically in a dynamic link library that your program uses.  This callback
  362. function is called by the timer device driver according to a timer delay
  363. value specified by the program.
  364.  
  365. When dealing with the multimedia timer, you specify two different times, both
  366. in milliseconds.  The first is the delay time, and the second is called the
  367. resolution.  You can think of the resolution as a tolerable error.  If you
  368. specify a delay of 100 milliseconds with a resolution of 10 milliseconds, the
  369. actual timer delay may range anywhere from 90 to 110 milliseconds.
  370.  
  371. There is sometimes a terminology problem when talking about resolution.  The
  372. higher the resolution, the lower the time value.  For example, a
  373. 5-millisecond resolution is higher than a 10-ms.  resolution.  To avoid this
  374. problem, I'll use the words higher and lower in conjuction with the term
  375. resolution value to indicate a numerical relationship.  For example, a 5-ms.
  376. resolution value is lower than a 10-ms.  resolution value.
  377.  
  378. Before you use the timer, you should obtain the timer device capabilities:
  379.  
  380. timeGetDevCaps (&timecaps, wSize) ;
  381.  
  382. The first parameter is a pointer to a structure of type TIMECAPS, and the
  383. second parameter is the size of this structure.  The TIMECAPS structure has
  384. only two fields, wPeriodMin and wPeriodMax.  These are the minimum and
  385. maximum resolution values supported by the timer device driv-er.  If you look
  386. at these values after calling timeGetDevCaps, you'll find that wPeriodMin is
  387. 1 and wPeriodMax is 65535, so this function may not seem crucial.  It's a
  388. good idea, however, to get these resolution values anyway and use them in the
  389. other timer function calls.
  390.  
  391. The next step is to call
  392.  
  393. timeBeginPeriod (wResolution) ;
  394.  
  395. to indicate the lowest timer resolution value that your program requires,
  396. within the range given in the TIMECAPS structure.  This call lets the timer
  397. device driver best provide for multiple programs that may be using the timer.
  398. Every call to timeBeginPeriod must be paired with a later call to
  399. timeEndPeriod (which we'll get to shortly).
  400.  
  401. Now you're ready to set a timer event:
  402.  
  403. wTimerID = timeSetEvent (wDelay,
  404.  
  405. wResolution, CallBackFunc,
  406.  
  407. dwData, wFlag) ;
  408.  
  409. The wTimerID return from the call will be 0 if an error occurs.  Following
  410. this call, the function CallBackFunc will be called from Windows in wDelay
  411. milliseconds with an allowable error specified by wResolution.  The
  412. wResolution value must be greater than or equal to the resolution value
  413. passed to timeBeginPeriod.  The dwData parameter is program- defined data,
  414. which is later passed to CallBackFunc.  The last parameter can be either
  415. TIME_ONESHOT, to get a single call to CallBackFunc in wDelay milliseconds, or
  416. TIME_PERIODIC, to get calls to CallBackFunc every wDelay milliseconds.  I'll
  417. discuss CallBackFunc shortly.
  418.  
  419. To stop a one-shot timer event before CallBackFunc is called, or to halt
  420. periodic timer events, call
  421.  
  422. timeKillEvent (wTimerID) ;
  423.  
  424. You don't need to kill a one-shot timer event after CallBackFunc is called.
  425.  
  426. When you're finished using the timer in your program, call
  427.  
  428. timeEndPeriod (wResolution) ;
  429.  
  430. with the same param-eter passed to time-BeginPeriod.
  431.  
  432. Try not to be greedy when specifying resolution values in these timer
  433. function calls.  Selecting a 1-ms.  resolution requires that the timer device
  434. driver process timer hardware interrupts every millisecond.  This can cause a
  435. serious drain on the rest of the system.
  436.  
  437. Two other functions begin with the prefix time.  The function
  438.  
  439. dwSysTime = timeGetTime () ;
  440.  
  441. returns the system time in milliseconds since Windows first started up.  The
  442. function
  443.  
  444. timeGetSystemTime (&mmtime, wSize);
  445.  
  446. requires a pointer to an MMTIME structure as the first parameter and the size
  447. of this structure as the second.  Although MMTIME can be used in other
  448. circumstances to get the system time in formats other than milliseconds, in
  449. this case it always returns the time in milliseconds.  So timeGetSystemTime
  450. is superfluous.
  451.  
  452. THE TIMER CALLBACK FUNCTION
  453.  
  454. As I mentioned, the multimedia timer works by calling a callback function
  455. located in a dynamic link library that your program uses.  The address of the
  456. callback function is passed as a parameter to time-SetEvent.  The timer
  457. device driver calls the callback function during hardware timer interrupts.
  458. This has several important implications: nThe callback function must be
  459. located in a dynamic link library, and it must be exported.  (The easiest way
  460. to export the callback function is to use the _export keyword when defining
  461. the function.) Un-like the case with other callback functions used in
  462. Windows, you don't need to call MakeProcInstance for the callback.  nAs
  463. stated in the documentation, the code segment in which the callback function
  464. resides must be fixed in memory, and any data that it uses must be in a fixed
  465. data segment.  Segments are flagged as fixed or movable in the dynamic link
  466. library's module definition file.  The seg-ments must be defined as fixed so
  467. they are not swapped out to disk and require reloading during the interrupt.
  468. nThe callback function is limited in the Windows function calls it can make.
  469. The callback function can only call Post- Message, four timer functions
  470. (timeSet-Event, timeKillEvent, timeGetTime, and the superfluous
  471. timeGetSystemTime), two MIDI output functions (midiOutShort-Msg and
  472. midiOutLongMsg), and the debugging function OutputDebugStr.  nAnd remember,
  473. don't even think about making any DOS function calls from the callback
  474. function.
  475.  
  476. As you can see, the multimedia timer is designed specifically for playing
  477. MIDI sequences and is limited in its usefulness beyond that.  You can use
  478. PostMessage for informing a window procedure of timer events, and the window
  479. procedure can do whatever it likes with the information, but it won't be
  480. responding with the accuracy of the timer callback itself.
  481.  
  482. The callback function has five param-eters, but only two of them are used:
  483. the timer ID number returned from time-SetEvent and the dwData value origi-
  484. nally passed as a parameter to time-SetEvent.
  485.  
  486. THE DRUMDLL SOURCE CODE
  487.  
  488. The source code for the DRUMDLL dynamic link library is shown in Figures 2
  489. through 6.  All source code for DRUM and the DRUMDLL dynamic link library can
  490. be downloaded from PC MagNet.  If you want to recreate DRUMDLL.DLL using the
  491. Microsoft C 7.0 compiler, run
  492.  
  493. NMAKE DRUMDLL.MSC
  494.  
  495. With the Borland C++ 3.0 compiler, use
  496.  
  497. MAKE -f DRUMDLL.BCP
  498.  
  499. If you're familiar with the compiler flags for these two products, you'll
  500. notice that DRUMDLL.C is compiled for C++ mode.  Although I'm not using any
  501. C++- specific code in DRUMDLL.C, adding such code is easier if you don't have
  502. to make further changes to account for the better type checking in C++.
  503.  
  504. Currently, it appears as if the retail version of Microsoft C 7.0 will
  505. include much of what is now included in the Windows 3.0 Software Development
  506. Kit (SDK), but beta versions have not included the multimedia MMSYSTEM.H
  507. header file or MMSYSTEM.LIB import libraries.  These files, however, are
  508. being included in the beta version of the Windows 3.1 SDK, indicating that
  509. the separate Multimedia Development Kit (MDK) is being phased out.  It's not
  510. clear now how all this will shake out; at any rate, an upgrade to Borland C++
  511. 3.0 is expected to account for the enhancements to Windows 3.1, including
  512. multimedia.
  513.  
  514. DRUMDLL.C contains four exported functions.  Three of them--DrumSet- Params,
  515. DrumBeginSequence, and Drum-EndSequence--are defined in DRUM-DLL.H and are
  516. called from the DRUM program.  The other is the timer callback function,
  517. DrumTimerFunc.
  518.  
  519. The three functions called from the DRUM program are defined in DRUMDLL.H
  520. within a construction that looks like this:
  521.  
  522. extern "C"
  523.  
  524. {
  525.  
  526. ....
  527.  
  528. }
  529.  
  530. This is to prevent C++ from "name-mangling" these functions.  C++ usually
  531. alters function names to include codes that indicate the parameter data types
  532. and return value, allowing for C++ function overloading and intermodule type
  533. checking during linking.  Because different compilers perform name mangling
  534. in different ways, you'll probably want to avoid it for functions exported
  535. from dyna-mic link libraries for use by Windows programs.  Otherwise you
  536. won't be able to use a dynamic link library from a program compiled with a
  537. compiler different from that used for the DLL.  In DRUM-DLL.C, I've also used
  538. the extern "C" statement for the MMSYSTEM.H header file, because the version
  539. I have is not C++-ready.
  540.  
  541. DRUMDLL also has a LibMain function.  Such a function usually unlocks the
  542. DLL's data segment, but here it needn't do anything because the data segment
  543. is fixed.  The only other function is MidiOutMessage, which simplifies the
  544. construction of MIDI messages in preparation for calling the Multimedia
  545. Windows function midiOutShortMsg.
  546.  
  547. I usually don't like to use global variables very much, but this DLL is made
  548. much simpler with them.  Global variables in a DLL are shared among all
  549. pro-grams that use the DLL, so only one program (or one instance of a
  550. program) can use DRUMDLL at any time.  The DRUM program does not allow
  551. running a second instance, and I've used the -p flag when running the
  552. resource compiler in the DRUMDLL Make files so that other programs can't use
  553. DRUMDLL while DRUM is running.
  554.  
  555. THE WORKINGS OF DRUMDLL
  556.  
  557. The DRUM program calls the DrumSet-Params function in DRUMDLL at various
  558. times: when DRUM's window is cre-ated, when the user clicks on the grid or
  559. manipulates the scroll bars, when the pro-gram loads a .DRM file from disk,
  560. or when the grid is cleared.  The single parameter to DrumSetParams is a
  561. pointer to a structure of type DRUM, defined in DRUMDLL.H.  This structure
  562. stores the beat time in milliseconds, the velocity, the number of beats in
  563. the sequence, and two sets of 32-bit integers (47 integers per set) for
  564. storing the grid settings for the base-level and extended synthesizers.  Each
  565. bit in these 32-bit integers corresponds to a beat of the sequence.
  566.  
  567. The DRUM program maintains a structure of type DRUM in static memory and
  568. passes a pointer to it when calling DrumSetParams.  DrumSetParams simply
  569. copies the contents of the structure to static memory in the DLL.  You might
  570. think that it would only be necessary for DRUMDLL to save a pointer to the
  571. DRUM structure located in DRUM, but this would violate one of the rules for
  572. using a timer callback function: Any data the timer callback function
  573. accesses must be in a fixed data segment.
  574.  
  575. To start the sequence, DRUM calls the DrumBeginSequence function in DRUMDLL.
  576. The only parameter is a window handle.  This is used for notification.
  577.  
  578. DrumBeginSequence opens the MIDI Mapper output device and, if successful,
  579. sends Program Change messages to select instrument voice 0 for MIDI channels
  580. 9 and 15.  As you may recall from the last issue, channel 9 is the percussion
  581. channel for extended synthesizers, and channel 15 is the percussion channel
  582. for base-level synthesizers.  To use the percussion channel, you select voice
  583. 0.  Different key numbers in the Note On messages then select different
  584. percussion sounds.
  585.  
  586. DrumBeginSequence continues by calling timeGetDevCaps and then
  587. time-BeginPeriod.  The desired timer resolution defined in the TIMER_RES
  588. constant is 5 ms., but I've defined a macro called minmax to calculate a
  589. resolution within the limits returned from timeGetDevCaps.
  590.  
  591. The next call is timeSetEvent, which specifies the beat time, the calculated
  592. resolution, the callback function Drum-TimerFunc, and the constant
  593. TIME_ONE-SHOT.  DRUMDLL uses a one-shot timer rather than a periodic timer so
  594. that the tempo can be dynamically changed while a sequence is running.  After
  595. the timeSet-Event call, the timer device driver will call DrumTimerFunc when
  596. the delay time has elapsed.
  597.  
  598. PLAYING THE SEQUENCE
  599.  
  600. The DrumTimerFunc callback is the func-tion where most of the heavy action
  601. takes place.  The variable iIndex stores the cur-rent beat in the sequence.
  602. The callback begins by sending MIDI Note Off messages for the drum sounds
  603. currently playing.  (iIndex has an initial value of -1 to prevent this when
  604. the sequence first be-gins.) Note Off messages are sent to chan-nel 9 for
  605. extended synthesizers and channel 15 for base-level synthesizers.
  606.  
  607. Next, iIndex is incremented, and its value is delivered to the window
  608. procedure in DRUM with a user-defined message called WM_USER_NOTIFY.  The
  609. wParam message parameter is set to iIndex so that WndProc in DRUM.C can move
  610. the "bouncing ball" at the bottom of the grid.  The lParam parameter is set
  611. to the current system time obtained from time-GetTime.  WndProc doesn't do
  612. anything with the system time, but I used it for debugging purposes and left
  613. it in so I could claim that DRUMDLL illustrates all six essential multimedia
  614. timer functions!
  615.  
  616. DrumTimerFunc finishes by sending Note On messages to the synthesizer for
  617. channels 9 and 15, saving the grid values so the sounds can be turned off the
  618. next time through, and then setting a new one-shot timer event by calling
  619. timeSetEvent.
  620.  
  621. To stop the sequence, DRUM calls DrumEndSequence with a single parameter that
  622. can be set to either TRUE or FALSE.  If it is TRUE, then DrumEnd-Sequence
  623. ends the sequence right away by killing any pending timer event, calling
  624. timeEndPeriod, sending "all notes off" messages to the two MIDI channels, and
  625. closing the MIDI output port.
  626.  
  627. DRUM calls DrumEndSequence with a TRUE parameter when the user decides to
  628. terminate the program.  If the user selects Stop from the Sequence menu in
  629. DRUM, however, the program instead calls Drum-EndSequence with a FALSE
  630. parameter.  This allows the sequence to complete the current cycle before
  631. ending.  DrumEndSequence responds to this call by setting the bEndSequence
  632. global variable to NULL.  If bEndSequence is TRUE and the beat index has been
  633. set to 0, DrumTimerFunc posts a user-defined message called WM_USER_FINISHED
  634. to WndProc.  WndProc must respond to this message by calling DrumEndSequence
  635. with a TRUE parameter to terminate the use of the timer and the MIDI port
  636. properly.
  637.  
  638. In an earlier version of the program, DrumTimerFunc itself called
  639. DrumEndSequence with a TRUE parameter to shut down.  It seemed to work fine,
  640. but I realized that DrumEndSequence was calling functions not listed as those
  641. allowed within timer callbacks.
  642.  
  643. The lesson: Be careful when writing a timer callback function.  Because
  644. you're coding in C, it's easy to forget that you're writing code that has to
  645. be executed during a hardware timer interrupt.  Keep the list of allowable
  646. functions close at hand.  It could well be that other functions can safely be
  647. called from a timer callback, but do you really want to take a chance that
  648. the next version of Windows will still work that way? Most C runtime
  649. functions should be okay, but don't try any file I/O during the timer
  650. callback.
  651.  
  652. If you need to perform a task based on the multimedia timer that you can't do
  653. without calling some other Windows function, use PostMessage to inform a
  654. window procedure of the timer event and let the window procedure do it.
  655. PostMessage is fine because all it does is place a message in a message
  656. queue.  The window procedure retrieves the message later--not during the
  657. timer interrupt.
  658.  
  659. Keep in mind also that the hardware timer could interrupt code in your
  660. program or other code in the DLL.  This might be a problem if you're altering
  661. data that the callback function accesses.  I decided this would not be a
  662. problem in DRUMDLL, but if you have critical sections of your code that
  663. should not be interrupted by a timer interrupt, surround the code with calls
  664. to the _disable and _enable functions provided in both the Microsoft C and
  665. Borland C++ runtime libraries.
  666.  
  667. FINISHING UP
  668.  
  669. We're not finished with DRUM just yet.  The DRUM program can also save and
  670. retrieve files containing the information stored in the DRUM structure.
  671. These files are in the Resource Interchange File Format (RIFF) recommended
  672. for multimedia file types.  You can read and write RIFF files using standard
  673. file I/O functions, of course, but an easier approach is provided by
  674. functions beginning with the prefix mmio (multimedia input/output).  In the
  675. next issue, I'll discuss these functions and the new "common dialog box"
  676. library included in Windows 3.1.
  677.  
  678.  
  679.  *****  Computer Select, October 1992 : Doc #14307  *****
  680.  
  681. Journal:   PC Magazine  June 30 1992 v11 n12 p367(6)
  682.            * Full Text COPYRIGHT Ziff-Davis Publishing Co. 1992.
  683. -----------------------------------------------------------------------------
  684. Title:     Creating a computer drum machine, part 3. (Environment)(column)
  685.            (Tutorial)
  686. Author:    Petzold, Charles.
  687. AttFile:    Program:  DRUMFILE.C  File I/O Routines for DRUM.
  688.  
  689. Abstract:  A guide to programming Microsoft Windows to create electronic
  690.            'drums' in multimedia applications is presented.  The multimedia
  691.            I/O functions provided with Microsoft's Windows Multimedia
  692.            Extensions are specifically tailored for reading and writing the
  693.            RIFF file format.  A RIFF file includes nested chunks of
  694.            information, each of which has a type, size and data.  Basic
  695.            techniques for using 'mmio' functions are discussed.  The mmioOpen
  696.            function returns a handle to a file and can include parameters for
  697.            data structures.  It works with the mmioCreateChunk function to
  698.            write information to a RIFF file.  Reading RIFF files involves the
  699.            mmioRead and mmioDescend functions.
  700. -----------------------------------------------------------------------------
  701. Descriptors..
  702. Product:   Microsoft Windows (Graphical user interface) (Programming).
  703. Topic:     Music
  704.            Multimedia technology
  705.            Tutorial
  706.            Programming Instruction
  707.            Graphical user interface.
  708. Feature:   illustration
  709.            program.
  710.  
  711. Record#:   12 216 586.
  712. -----------------------------------------------------------------------------
  713. Full Text:
  714.  
  715. File I/O under Windows has always been a little peculiar.  Until Windows 3.0,
  716. the only documented file I/O function in Windows was OpenFile, which returned
  717. a DOS file handle.  This function had the advantage of being able to create a
  718. fully qualified path and filename based on the current directory, but it was
  719. often clumsy to use.
  720.  
  721. Some programmers used the DOS file handle obtained from OpenFile with the
  722. low-level C file I/O functions (read, write, close, and so forth).  I took
  723. this approach in one of the programs in my book Programming Windows
  724. (Microsoft Press, 1990).  The program worked fine with Microsoft C 6.0, but
  725. it had problems when compiled under Borland C++ 2.0.  Under Borland's
  726. compiler, file handles associated with the low-level C file I/O functions
  727. were not the same as the DOS file handles.
  728.  
  729. Windows programmers often need to use global memory blocks with files.  The
  730. easiest way to do this is to pass far pointers to file I/O functions.  But
  731. when you write small- or medium-model programs (which is highly recommended
  732. for Windows programming), all the standard C runtime functions accept near
  733. pointers.
  734.  
  735. Windows 3.0 finally documented some file I/O functions ( _lcreate, _lopen,
  736. _lwrite, _lread, _llseek, and _lclose) that had actu-ally been part of
  737. Windows since Version 1.0.  These functions accept far pointers and hook
  738. almost directly into the equivalent DOS file I/O function calls.  They
  739. probably have come to be the most commonly used file I/O functions in Windows
  740. application programs.  It does seem strange, though, to be making raw DOS
  741. function calls in the middle of a sophisticated graphical Windows program.
  742.  
  743. PROBLEMS AND SOLUTIONS
  744.  
  745. Beyond the actual mechanics of file I/O, Windows posed another problem for
  746. programmers who wanted to add file support to their applications.  Windows
  747. programs should display a File Open dialog box for the user to select a file,
  748. but such dialog boxes proved rather complex to write.  Many programmers
  749. wished for a standard File Open dialog box to be built into Windows itself.
  750.  
  751. With Windows 3.1, this finally happened.  This version includes a
  752. common-dialog-box library that contains dialog-box templates and procedures
  753. for opening files, saving files, printing, search-and-replace operations,
  754. choosing colors, and choosing fonts.  In most cases, all you have to do is
  755. set up a structure's fields, pass the structure to a function, and get back
  756. your results when the user closes the dialog box.
  757.  
  758. These dialog boxes are implemented in a dynamic link library called COM-
  759. MDLG.DLL.  COMMDLG.H is the header file defining the functions and
  760. structures, and COMMDLG.LIB is the import library for linking the program to
  761. use the DLL.  Microsoft allows COMMDLG.DLL to be shipped with applications
  762. that use the common-dialog-box library but continue to run under Windows 3.0.
  763. (For more information on the Windows common-dialog library, see the May 26,
  764. 1992, Power Programming column.)
  765.  
  766. Moreover, because Windows 3.1 includes Multimedia Extensions, it makes
  767. available 16 functions, beginning with the prefix mmio (multimedia
  768. input/output), that are specifically designed for working with RIFF (Resource
  769. Interchange File Format) files.  As I discussed in the previous column, RIFF
  770. files are used to store multimedia data.
  771.  
  772. FINISHING UP DRUM
  773.  
  774. With the common-dialog-box library and the mmio functions, adding file I/O to
  775. DRUM requires little more than the DRUMFILE.C module shown in Figure 1.  DRUM
  776. maintains two filenames: The szTitleName variable stores the filename and
  777. extension only.  This is used for displaying the filename in the title bar of
  778. the program's window and in message boxes.  The szFileName variable stores
  779. the fully qualified path and filename and is used for reading from and
  780. writing to the file.
  781.  
  782. GetOpenFileName and GetSaveFileName are the two functions in the
  783. common-dialog-box library that display the dialog boxes.  Both of these
  784. functions ac-cept structures of type OPENFILENAME.  Although the structure is
  785. large, for typical use most of its fields can be set to 0.  Basically, as
  786. shown in the functions Drum-FileOpenDlg and DrumFileSaveDlg in DRUMFILE.C,
  787. you need pointers to buffers to receive the fully qualified filename, the
  788. title name, and the size of these buffers, as well as a filter string
  789. containing the file types for the dialog.
  790.  
  791. MMIO BASICS
  792.  
  793. In comparison with other sets of file I/O functions, the mmio functions are
  794. very versatile.  If you like, you can use the mmioOpen, mmioRead, mmioWrite,
  795. mmioSeek, and mmioClose functions for general-purpose file I/O.  One
  796. advantage of mmio over both
  797.  
  798. Windows and C file   I/O is that the mmioRead and mmioWrite
  799.  
  800. functions accept huge pointers.  You can read or write a global memory block
  801. greater than 64K in one function call.  The mmio functions also let you
  802. directly manipulate the file I/O buffer, supply your own I/O procedures, and
  803. create memory files.
  804.  
  805. Here I'll restrict use of the mmio func-tions to those few required for
  806. writing and reading RIFF files.  As I described last time, RIFF files are
  807. composed of nested chunks of information, each of which has a chunk type (a
  808. four-character name), a chunk size (32 bits long), and chunk data.  I also
  809. presented the RDUMP program, which displays the layout of a RIFF file.
  810. Figure 2 shows RDUMP output for one of DRUM's data files.
  811.  
  812. The DRUM data file uses a form type of DRUM.  As recommended in the Microsoft
  813. Windows Multimedia Programmer's Reference, I registered this form type with
  814. the Product Marketing area of the Multimedia Systems Group at Microsoft.
  815.  
  816. Although you can create RIFF files using normal file I/O functions,
  817. calculating and storing the chunk sizes can be a nuisance.  That's one of the
  818. things the mmio functions handle very well.  When reading a RIFF file, the
  819. mmio functions also hide the process of skipping over unnecessary chunks.
  820.  
  821. When you open a file using the mmio functions, you get a file handle of type
  822. HMMIO.  Don't attempt to use this handle with DOS file I/O function calls or
  823. C file functions.
  824.  
  825. The four-character chunk names, form types, and list types are treated as
  826. 32-bit integers in the mmio functions.  These have a data type of FOURCC,
  827. which stands for four-character code.
  828.  
  829. Two methods are provided for obtaining a 32-bit integer from characters or
  830. strings.  One approach is to use the following macro:
  831.  
  832. fcc = mmioFOURCC (ch1, ch2, ch3, ch4) ;
  833.  
  834. The four parameters are the four characters individually; for example:
  835.  
  836. fcc = mmioFOURCC ('W','A','V', 'E') ;
  837.  
  838. You can also use a function that accepts an entire string:
  839.  
  840. fcc = mmioStringToFOURCC (szFCC, wFlag) ;
  841.  
  842. In this case:
  843.  
  844. fcc = mmioStringToFOURCC ("WAVE",  0) ;
  845.  
  846. The wFlag parameter can be MMIO_TOUPPER to make the characters of the string
  847. uppercase before conversion.
  848.  
  849. Some of the mmio functions require a pointer to a structure of type MMCKINFO
  850. (multimedia chunk information).  This structure is shown in Figure 3.  In the
  851. examples below, I'll use the variable name mmckinfo as a structure of type
  852. MMCKINFO.
  853.  
  854. CREATING A RIFF FILE
  855.  
  856. To open a file using the mmio functions, the first step is to call
  857.  
  858. hmmio = mmioOpen (szFilename, &mmioinfo, dwFlags) ;
  859.  
  860. The function returns a handle to the file.  The second parameter can be a
  861. pointer to an MMIOINFO structure (not the MMCKINFO structure discussed
  862. above), but in many cases you can set that parameter to NULL.  The dwFlags
  863. parameter specifies whether you want to create the file or open the file for
  864. reading or writing; it also lets you indicate sharing modes and other
  865. information.
  866.  
  867. The following function creates a chunk in the file:
  868.  
  869. wError = mmioCreateChunk (hmmio, &mmckinfo, wFlags) ;
  870.  
  871. There are two ways to use this function:
  872.  
  873. If you are creating a chunk that is not a RIFF chunk or a LIST chunk, prepare
  874. for the call by doing the following:
  875.  
  876. *    Set the ckid field of the MMCKINFO structure to the
  877.  
  878. four-character code of the chunk type.
  879.  
  880. *    Optionally set the cksize field to the chunk size of the data
  881.  
  882. in the chunk.
  883.  
  884. *    Set the wFlags parameter of mmio-CreateChunk to 0.
  885.  
  886. In this case, mmioCreateChunk writes 8 bytes to the file--the chunk type
  887. (from the ckid field) and the chunk size (from the cksize field).  The file
  888. pointer is left positioned at the end of the chunk size.
  889.  
  890. If you are creating a RIFF or a LIST chunk, then prepare as follows:
  891.  
  892. *    Set the fccType field of the MMCKINFO structure to the
  893.  
  894. four-character code of the form type (for a RIFF chunk) or the list type (for
  895. a LIST chunk).
  896.  
  897. *    Optionally set the cksize field to the chunk size of the data
  898.  
  899. in the chunk.
  900.  
  901. *    Set the wFlags parameter of the function to MMIO_CREATERIFF or
  902.  
  903. MMIO_CREATELIST to create a RIFF chunk or a LIST chunk.
  904.  
  905. In this case, mmioCreateChunk writes 12 bytes to the file--the chunk-type
  906. ("RIFF" or "LIST"), chunk size (from the cksize field), and form type (from
  907. the fccType field).  The file pointer is left positioned at the end of the
  908. form type.
  909.  
  910. In either case, you don't have to set the cksize field of the structure.  The
  911. file's chunk-size field can be filled in at a later time by calling
  912. mmioAscend (as we'll see).  On return from the mmioCreate-Chunk function, the
  913. file pointer and the dwDataOffset field of the structure will be set to the
  914. beginning of the chunk-data area relative to the beginning of the file.
  915.  
  916. Next write the chunk data using
  917.  
  918. lWritten = mmioWrite (hmmio, pData, lSize) ;
  919.  
  920. The pointer can be to a memory block greater than 64K.  The size of the data
  921. is given in the last parameter, and the function returns the number of bytes
  922. written.  This may be less than the number of bytes requested if you run out
  923. of disk space.
  924.  
  925. After writing the chunk data, you call
  926.  
  927. wError = mmioAscend (hmmio, &mmckinfo, 0) ;
  928.  
  929. The mmioAscend function has two distinct purposes, one when reading a RIFF
  930. file and another when writing to one.  When use in conjunction with
  931. mmioCreateChunk and mmioWrite while writing to a file, mmioAscend fills in
  932. the chunk size for the chunk originally created by mmio-CreateChunk.  The
  933. MMCKINFO structure passed to mmioAscend must be the same MMCKINFO structure
  934. passed earlier to mmioCreateChunk to create the chunk.  The mmioAscend
  935. function works by subtracting the dwDataOffset field of the structure from
  936. the current file pointer (which will now be at the end of the chunk data) and
  937. storing that value before the data.  The mmioAscend function also takes care
  938. of data padding if the chunk data is not a multiple of 2 bytes in length.
  939.  
  940. RIFF files are composed of nested levels of chunks.  For mmioAscend to work
  941. correctly, you must maintain multiple MMCKINFO structures, each of which is
  942. associated with a level in the file.  The DRUM data files have three levels.
  943. Hence, in the DrumFileWrite function in DRUMFILE.C, I've defined an array of
  944. three MMCKINFO structures, which can be referenced as mmckinfo[0],
  945. mmckinfo[1], and mmckinfo[2].
  946.  
  947. Mmckinfo[0] is used in the first mmio-CreateChunk call to create a RIFF chunk
  948. type with a DRUM form type.  This is followed by a second mmioCreateChunk
  949. call using mmckinfo[1] to create a LIST chunk type with an INFO list type.
  950.  
  951. A third mmioCreateChunk call using mmckinfo[2] creates a chunk type of ISFT,
  952. identifying the software that created the data file.  Following the mmioWrite
  953. call to write the string szSoftware, a call to mmioAscent using mmckinfo[2]
  954. fills in the chunk size field for this chunk.  This is the first completed
  955. chunk.
  956.  
  957. The next chunk is also within the LIST chunk.  The program proceeds with
  958. another mmioCreateChunk call to create an ISCD (creation data) chunk, using
  959. mmckinfo[2].  After the mmioWrite call to write the chunk data, a call to
  960. mmioAscend using mmckinfo[2] fills in the chunk size.
  961.  
  962. That's the end of this chunk, and it is also the end of the LIST chunk.  To
  963. fill in the chunk-size field of the LIST chunk, mmioAscend is called again,
  964. this time using mmckinfo[1], which was originally used to create the LIST
  965. chunk.
  966.  
  967. To create the "fmt " and "data" chunks, mmioCreateChunk uses mmckinfo[1]; the
  968. mmioWrite calls are followed by mmioAscend, also using mmckinfo[1].  Now all
  969. the chunk sizes have been filled in except for the RIFF chunk itself.  That
  970. requires one more call to mmioAscend using mmckinfo[0].  There's only one
  971. call left, and that's to mmioClose.
  972.  
  973. It probably seems as if an mmioAscend call changes the current file pointer,
  974. and it certainly may in order to fill in the chunk size, but by the time the
  975. function returns, the file pointer is restored to its position at the end of
  976. the chunk data (or perhaps incremented by 1 byte for data padding).  All
  977. writing to the file is sequential from beginning to end.
  978.  
  979. After a successful mmioOpen call, nothing can really go wrong except running
  980. out of disk space.  I use the variable wError to accumulate error codes from
  981. the mmioCreateChunk, mmioWrite, mmioAscend, and mmioClose call, each of which
  982. could fail if insufficient disk space is available.  If that happens, the
  983. file is deleted using mmioOpen with the MMIO_DELETE constant, and an error
  984. message is returned to the caller.
  985.  
  986. READING A RIFF FILE
  987.  
  988. Reading a RIFF file is similar to creating one, except that mmioRead is
  989. called instead of mmioWrite, and mmioDescend is called rather than
  990. mmioCreateChunk.  To descend into a chunk means to locate a chunk and put the
  991. file pointer after the chunk size (or after the form type or list type for a
  992. RIFF or LIST chunk type).  To ascend from a chunk means to move the file
  993. pointer to the end of the chunk data.  Neither the mmioDescend nor mmioAscend
  994. functions move the file pointer to an earlier position in the file.
  995.  
  996. The documentation for mmioDescend is confusing.  Let's use a cookbook
  997. approach to examine the function: You want a separate MMCKINFO structure for
  998. each level in the file.  I'll refer to these structures as mmckinfo[0],
  999. mmckinfo[1], and mmckinfo[2].  The first thing to do after opening a file is
  1000. to verify that it is in the RIFF format and, optionally, that the form type
  1001. is the one you want.  If you're only interested in a specific form type, set
  1002. the fccType field of the MMCKINFO structure to that form type and call
  1003.  
  1004. wError = mmioDescend (hmmio, &mmckinfo[0], NULL, MMIO_FINDRIFF) ;
  1005.  
  1006. Watch out: The documentation indicates that the ckid field should be set to
  1007. the form type.  But the mmioDescend function called with the MMIO_FINDRIFF
  1008. parameter does not even check that field.  If the fccType field is not set to
  1009. a form type, mmioDescend will return successfully for a RIFF file with any
  1010. form type.  Then you will have to check the fccType field explicitly.
  1011.  
  1012. After this function call, the ckid field will contain "RIFF", the cksize
  1013. field will have the chunk size, the fccType will be the form type you
  1014. specified (or the form type of the file), and dwDataOffset will indicate the
  1015. position in the file just after the form type (that is, 12 bytes).
  1016.  
  1017. Now you're ready to search for chunk types within the RIFF chunk.  If the
  1018. chunk types that interest you occur in a fixed order (as in a waveform audio
  1019. file or a DRUM file, where the "fmt " chunk must precede the "data" chunk),
  1020. set the ckid field of the mmckinfo[1] structure to the first chunk type you
  1021. need (for example, the four-character code for "fmt")  and call
  1022.  
  1023. wError = mmioDescend (hmmio, &mmckinfo[1], &mmckinfo[0], MMIO_FINDCHUNK) ;
  1024.  
  1025. The third parameter indicates a "parent" chunk as defined by the MMCKINFO
  1026. structure.  Notice that the last parameter is MMIO_FINDCHUNK.  After this
  1027. function call, the fields of the mmckinfo[1] structure are set as follows:
  1028. ckid is the chunk type you specified, cksize is the size of the chunk,
  1029. fccType is set to 0, and dwOffset indicates the offset from the beginning of
  1030. the file to the beginning of the chunk data.
  1031.  
  1032. Now you can read the chunk using mmioRead:
  1033.  
  1034. lRead = mmioRead (hmmio, pData, lSize) ;
  1035.  
  1036. After reading the chunk data, call
  1037.  
  1038. mmioAscend (hmmio, mmckinfo[1], 0);
  1039.  
  1040. This moves the file pointer to the end of the chunk data, accounting for
  1041. possible padding.  You can now search for the next chunk you need using
  1042. mmioDescend, read it with mmioRead, and move to the end of the chunk data
  1043. with mmioAscend.
  1044.  
  1045. If the ordering of the chunk types is not defined, set the ckid field of the
  1046. mmckinfo[1] structure to 0 before calling the mmioDescend function with the
  1047. MMIO_FINDCHUNK parameter.  The function will find the next subchunk, and you
  1048. can then check the ckid field of the structure to determine the chunk type.
  1049. If you're not interested in the chunk, call mmioAscend and mmioDescend again.
  1050. MmioDescend will return an error code if no more subchunks are present.
  1051.  
  1052. If the chunk type is "LIST", mmio- Descend will correctly increment the file
  1053. pointer to follow the form type.  You can check the fccType field of the
  1054. mmckinfo[1] structure for the list type.  If you're interested in this list
  1055. type, you can descend into the subchunks by calling the mmioDescend function
  1056. with mmckinfo[3].  (The chunks probably won't be in a specific order, so you
  1057. will set the ckid field to 0L.) Otherwise, just call mmioAscend with
  1058. mmckinfo[2] to skip the LIST chunk.
  1059.  
  1060. You can also search for a specific LIST type by calling mmioDescend with the
  1061. MMIO_FINDLIST parameter.  If you are interested in a particular list type,
  1062. set the fccType field of the MMCKINFO structure to that list type.
  1063.  
  1064. DRUM is interested only in the "fmt data" chunks of the RIFF file it uses for
  1065. storing data.  The code in DRUMFILE.C follows the cookbook approach, and most
  1066. of the work involves verifying that the file follows the correct format.
  1067.  
  1068. COMING UP: MIDI Input
  1069.  
  1070. If you attach a MIDI keyboard (or other MIDI controller) to the MIDI In port
  1071. of a MIDI board, you can also use Multimedia Windows to read MIDI messages.
  1072. In the next column, we'll begin an exploration of MIDI input.
  1073.