Changeset 524 for OCO/branches/DAZ/drv16/midistrm.cpp
- Timestamp:
- Jul 22, 2010, 8:20:48 PM (15 years ago)
- File:
-
- 1 edited
Legend:
- Unmodified
- Added
- Removed
-
TabularUnified OCO/branches/DAZ/drv16/midistrm.cpp ¶
r468 r524 3 3 /* SCCSID = %W% %E% */ 4 4 /**************************************************************************** 5 * 6 * Copyright (c) IBM Corporation 1994 - 1997. 7 * 8 * The following IBM OS/2 source code is provided to you solely for the 5 * * 6 * Copyright (c) IBM Corporation 1994 - 1997. * 7 * * 8 * The following IBM OS/2 source code is provided to you solely for the * 9 9 * the purpose of assisting you in your development of OS/2 device drivers. * 10 * You may use this code in accordance with the IBM License Agreement 11 * provided in the IBM Device Driver Source Kit for OS/2. 12 * 10 * You may use this code in accordance with the IBM License Agreement * 11 * provided in the IBM Device Driver Source Kit for OS/2. * 12 * * 13 13 ****************************************************************************/ 14 14 /**@internal %W% 15 15 * @notes 16 * 17 * 16 * MIDISTREAM class implementation. The Midi Stream class is derived 17 * from the Stream class. 18 18 * @version %I% 19 19 * @context Unless otherwise noted, all interfaces are Ring-0, 16-bit, 20 * 20 * kernel stack. 21 21 * @history 22 22 * 23 23 */ 24 24 #define INCL_NOPMAPI 25 #define INCL_DOSERRORS 25 #define INCL_DOSERRORS // for ERROR_INVALID_FUNCTION 26 26 #include <os2.h> 27 27 #include <os2me.h> 28 #include <audio.h> 28 #include <audio.h> // for #define MIDI 29 29 30 30 #include <include.h> … … 39 39 USHORT MIDISTREAM::_usBitNumber[ NUM_MidiChannels ] = 40 40 { 0x0001, 0x0002, 0x0004, 0x0008, 41 42 43 41 0x0010, 0x0020, 0x0040, 0x0080, 42 0x0100, 0x0200, 0x0400, 0x0800, 43 0x1000, 0x2000, 0x4000, 0x8000 }; 44 44 45 45 46 46 /**@internal CalcDelay 47 * @param 48 * @return 47 * @param None 48 * @return None 49 49 * @notes 50 * 51 *---------------------------------------------------------- == 52 * 53 * 54 * 55 * 56 * 57 * 50 * 600,000,000 microseconds/10 minutes 51 *---------------------------------------------------------- == X microseconds/clock 52 * usCPQNnum 53 * (ulTempo beats/10 min) * ( 24 * ------------- clocks/beat ) 54 * usCPQNden 55 * 56 * 57 * 25,000,000 * usCPQNden 58 58 *== -------------------------- 59 * 59 * ulTempo * usCPQNnum 60 60 * 61 61 * where 62 62 * usCPQNden = ((usCPQN & 0x3F) + 1) * 3 63 * usCPQNnum = 1 64 * 65 * 63 * usCPQNnum = 1 if bit 6 of usCPQN is set 64 * 65 * or 66 66 * 67 67 * usCPQNden = 1 68 * usCPQNnum = usCPQN + 1 68 * usCPQNnum = usCPQN + 1 if bit 6 is not set 69 69 */ 70 70 void MIDISTREAM::CalcDelay(void) … … 72 72 ULONG ul; 73 73 74 if (usCPQN & 0x40) { 75 76 77 74 if (usCPQN & 0x40) { // bit 6 is set if it's a denominator 75 ul = 25000000 * ((usCPQN & 0x3F) + 1); 76 ulPerClock = ul / ulTempo; 77 ulPerClock *= 3; 78 78 } else { 79 80 79 ul = ulTempo * (usCPQN+1); 80 ulPerClock = 25000000 / ul; 81 81 } 82 82 } … … 84 84 85 85 /**@external MIDISTRM::Process 86 * 87 * 86 * Consume MIDI bytes from the MMPM/2 stream buffers and send 87 * them off to the MIDI parser to be interpreted. 88 88 * @param void 89 89 * @return void 90 90 * @notes Runs at Task time on a global context hook; does not run 91 * on an interrupt level.Interacts with the Timer object defined92 * 93 * 91 * on an interrupt level. Interacts with the Timer object defined 92 * for this stream to obtain current time and to request next Stream 93 * time to be scheduled. 94 94 */ 95 95 void MIDISTREAM::Process( void ) 96 96 { 97 ULONG ulNewTime; 98 ULONG ulElapsedTime; 97 ULONG ulNewTime; // Time, in mSec, on entry. 98 ULONG ulElapsedTime; // Elapsed time, last tick to this one. 99 99 100 100 // Update time variables. 101 101 ulNewTime = ((MIDIAUDIO*) pahw)->getTimer()->ulGetTime(); 102 102 if ( ulNewTime > _ulLastProcess ) 103 103 ulElapsedTime = ulNewTime - _ulLastProcess; 104 104 else 105 105 ulElapsedTime = 0; 106 106 107 107 _ulLastProcess = ulNewTime; 108 108 ulCurrentTime = ulNewTime; 109 109 110 if (qhInProcess.IsElements() == 0) 111 112 113 if (ulStreamState == STREAM_PAUSED) 114 110 if (qhInProcess.IsElements() == 0) // no buffers to process? 111 return; 112 113 if (ulStreamState == STREAM_PAUSED) // is the stream paused? 114 return; 115 115 116 116 ProcessEvents(); … … 123 123 124 124 while (lWait <= 0 && qhInProcess.IsElements()) { 125 126 127 128 129 130 131 132 125 parse(*(pbuff + (pstreambuff->ulBuffpos)++)); 126 127 if (pstreambuff->ulBuffpos >= buffsz) { 128 qhDone.PushOnTail(qhInProcess.PopHead()); 129 pstreambuff = (PSTREAMBUFFER) qhInProcess.Head(); 130 pbuff = pstreambuff->pBuffptr; 131 buffsz = pstreambuff->ulBuffsz; 132 } 133 133 } 134 134 while (qhDone.IsElements()) 135 135 ReturnBuffer(); 136 136 137 137 // Determine next time to run. If we submit a time that has already … … 163 163 } 164 164 /**@internal MIDISTREAM::_allNotesOff 165 * 165 * Shut off all notes that are currently playing. 166 166 * @param None. 167 167 * @return void 168 168 * @notes This function walks the _notesOn array and shuts off any note 169 * 169 * that is flagged as being actively played. 170 170 */ 171 171 void MIDISTREAM::_allNotesOff( void ) 172 172 { 173 173 for ( USHORT noteNum=0; noteNum < NUM_MidiNotes; ++noteNum) 174 175 176 177 178 179 pahw->noteOff(StreamId,(BYTE)mchan, (BYTE)noteNum, 0 );180 181 174 if (_notesOn[noteNum]) 175 // This note number is playing on one or more channels. 176 // Shut the note off on all channels on which it is playing. 177 for ( USHORT mchan=0; mchan < NUM_MidiChannels; ++mchan) 178 if (_notesOn[noteNum] & _usBitNumber[mchan]) { 179 pahw->noteOff(StreamId, (BYTE)mchan, (BYTE)noteNum, 0 ); 180 _notesOn[noteNum] &= ~(_usBitNumber[mchan]); 181 } 182 182 } 183 183 //****************************************************************************** … … 185 185 ULONG MIDISTREAM::StartStream(void) 186 186 { 187 //dprintf(("MIDISTREAM::StartStream %lx %lx", (void far *)this, StreamId));188 189 state = S_Init;// Reset parser state.190 message.clear();// Clear current message.191 192 193 194 195 196 197 198 199 200 201 187 //dprintf(("MIDISTREAM::StartStream %lx %lx", (void far *)this, StreamId)); 188 189 state = S_Init; // Reset parser state. 190 message.clear(); // Clear current message. 191 lWait = 0; //SvL, reset this too 192 193 pahw->getTimer()->vSetTime(ulCurrentTime); 194 if (!pahw->Start(StreamId)) { 195 DebugInt3(); 196 return ERROR_START_STREAM; 197 } 198 199 ulStreamState = STREAM_STREAMING; 200 pahw->SetVolume(StreamId, getMixerStreamId(), volume); 201 return NO_ERROR; 202 202 } 203 203 //****************************************************************************** … … 205 205 ULONG MIDISTREAM::StopStream(PCONTROL_PARM pControl) 206 206 { 207 208 //dprintf(("MIDISTREAM::StopStream %lx (already stopped)", StreamId));209 210 211 212 213 //dprintf(("MIDISTREAM::StopStream %lx %lx", (void far *)this, StreamId));214 215 216 217 218 219 207 if(ulStreamState == STREAM_STOPPED) { 208 //dprintf(("MIDISTREAM::StopStream %lx (already stopped)", StreamId)); 209 pControl->ulTime = GetCurrentTime(); 210 return NO_ERROR; 211 } 212 213 //dprintf(("MIDISTREAM::StopStream %lx %lx", (void far *)this, StreamId)); 214 ulStreamState = STREAM_STOPPED; 215 pahw->Stop(StreamId); 216 _allNotesOff(); 217 ReturnBuffers(); 218 pControl->ulTime = GetCurrentTime(); 219 return NO_ERROR; 220 220 } 221 221 //****************************************************************************** … … 223 223 ULONG MIDISTREAM::PauseStream(PCONTROL_PARM pControl) 224 224 { 225 //dprintf(("MIDISTREAM::PauseStream %lx %lx", (void far *)this, StreamId));226 if (ulStreamState == STREAM_PAUSED) {// is the stream paused?227 228 229 230 231 232 233 234 235 225 //dprintf(("MIDISTREAM::PauseStream %lx %lx", (void far *)this, StreamId)); 226 if (ulStreamState == STREAM_PAUSED) { // is the stream paused? 227 DebugInt3(); 228 return ERROR_INVALID_SEQUENCE; 229 } 230 231 pahw->Stop(StreamId); 232 _allNotesOff(); 233 pControl->ulTime = GetCurrentTime(); 234 ulStreamState = STREAM_PAUSED; 235 return NO_ERROR; 236 236 } 237 237 //****************************************************************************** … … 239 239 ULONG MIDISTREAM::ResumeStream(void) 240 240 { 241 //dprintf(("MIDISTREAM::ResumeStream %lx %lx", (void far *)this, StreamId));242 if (ulStreamState != STREAM_PAUSED) {// is the stream paused?243 244 245 246 247 state = S_Init;// Reset parser state.248 message.clear();// Clear current message.249 250 251 252 253 241 //dprintf(("MIDISTREAM::ResumeStream %lx %lx", (void far *)this, StreamId)); 242 if (ulStreamState != STREAM_PAUSED) { // is the stream paused? 243 DebugInt3(); 244 return ERROR_INVALID_SEQUENCE; 245 } 246 247 state = S_Init; // Reset parser state. 248 message.clear(); // Clear current message. 249 pahw->getTimer()->vSetTime(ulCurrentTime); 250 pahw->Start(StreamId); 251 pahw->SetVolume(StreamId, getMixerStreamId(), volume); 252 ulStreamState = STREAM_STREAMING; 253 return NO_ERROR; 254 254 } 255 255 //****************************************************************************** … … 257 257 BOOL MIDISTREAM::SetProperty(int type, ULONG value, ULONG reserved) 258 258 { 259 260 261 volume = value; 262 263 264 265 266 267 268 269 270 259 switch(type) { 260 case PROPERTY_VOLUME: 261 volume = value; 262 if(ulStreamState == STREAM_STREAMING) { 263 pahw->SetVolume(StreamId, getMixerStreamId(), volume); 264 } 265 break; 266 267 default: 268 return STREAM::SetProperty(type, value, reserved); 269 } 270 return TRUE; 271 271 } 272 272 //****************************************************************************** … … 275 275 STREAM(streamtype, filesysnum, mixerStreamId) 276 276 { 277 //dprintf(("MIDISTREAM::ctor"));278 279 280 281 282 283 // 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 277 //dprintf(("MIDISTREAM::ctor")); 278 279 // get the pointer to the hardware object 280 pahw = (MIDIAUDIO *)GetHardwareDevice(streamtype); 281 if(pahw) { 282 if(pahw->Open(0, streamtype, filesysnum, &StreamId) == FALSE) { 283 // dprintf(("MIDISTREAM ctor: open failed!!")); 284 DebugInt3(); 285 } 286 } 287 else DebugInt3(); 288 289 // Initialize tempo & scheduling information. 290 ulTempo = 1200; 291 usCPQN = 0; 292 CalcDelay(); 293 lWait = 0; 294 _ulLastProcess = 0; 295 296 // Reset the parser. 297 state = S_Init; 298 message.clear(); 299 300 // Reset our tracking of which notes are currently on. 301 for (int i=0; i<NUM_MidiNotes; ++i) 302 _notesOn[ i ] = 0; 303 303 } 304 304 //****************************************************************************** … … 306 306 MIDISTREAM::~MIDISTREAM() 307 307 { 308 309 310 311 } 312 //****************************************************************************** 313 //****************************************************************************** 314 308 if(StreamId && pahw) { 309 pahw->Close(StreamId); 310 } 311 } 312 //****************************************************************************** 313 //****************************************************************************** 314
Note:
See TracChangeset
for help on using the changeset viewer.