/****************************************************************************** Copyright (c) 1985-1998 Microsoft Corporation Title: foramts.c - Multimedia Systems Media Control Interface Contains specific mci time format conversion functions Version: 1.00 Date: 7-MAR-1991 Author: Greg Simons ------------------------------------------------------------------------------ Change log: DATE REV DESCRIPTION ----------- ----- ----------------------------------------------------------- 7-MAR-1991 GregSi Original *****************************************************************************/ #define UNICODE //MMSYSTEM #define MMNOSOUND - Sound support #define MMNOWAVE - Waveform support #define MMNOAUX - Auxiliary output support #define MMNOJOY - Joystick support //MMDDK #define NOWAVEDEV - Waveform support #define NOAUXDEV - Auxiliary output support #define NOJOYDEV - Joystick support #include #include #include #include "mmsys.h" #include "list.h" #include "mciseq.h" /**************************** PRIVATE PROTOTYPES *************************/ PRIVATE DWORD NEAR PASCAL FPSDisplay(DWORD dwDisplayType); PRIVATE DWORD NEAR PASCAL FPSFile(int wFileDiv); PRIVATE DWORD NEAR PASCAL HMSFToFrames(DWORD dwCurrent, DWORD dwFormat); PRIVATE DWORD NEAR PASCAL HMSFToMS(DWORD dwHmsf, DWORD dwFormat); PRIVATE DWORD NEAR PASCAL FramesToHMSF(DWORD dwFrames, DWORD dwFormat); /**************************** PRIVATE FUNCTIONS *************************/ PRIVATE DWORD NEAR PASCAL FPSDisplay(DWORD dwDisplayType) // return the frames per second for smpte display types // (utility function for format conversion routines) { switch (dwDisplayType) { case MCI_FORMAT_SMPTE_24: return 24; case MCI_FORMAT_SMPTE_25: return 25; case MCI_FORMAT_SMPTE_30DROP: case MCI_FORMAT_SMPTE_30: return 30; #ifdef DEBUG default: return 0; #endif } //switch return 0; } PRIVATE DWORD NEAR PASCAL FPSFile(int wFileDiv) { // returns frames per second based file division type switch (wFileDiv) { case SEQ_DIV_SMPTE_24: return 24; case SEQ_DIV_SMPTE_25: return 25; case SEQ_DIV_SMPTE_30: case SEQ_DIV_SMPTE_30DROP: return 30; #ifdef DEBUG default: return 0; #endif } //switch return 0; } PRIVATE DWORD NEAR PASCAL HMSFToFrames(DWORD dwCurrent, DWORD dwFormat) /* convert from hmsf (colonized) format to raw frames */ { DWORD dwReturn; HMSF hmsf = * ((HMSF FAR *) &dwCurrent); // cast dwCurrent to hmsf int fps = (int)FPSDisplay(dwFormat); dwReturn = ((DWORD)hmsf.hours * 60 * 60 * fps) + ((DWORD)hmsf.minutes * 60 * fps) + ((DWORD)hmsf.seconds * fps) + hmsf.frames; return dwReturn; } PRIVATE DWORD NEAR PASCAL HMSFToMS(DWORD dwHmsf, DWORD dwFormat) // convert hmsf (colonized) format to milliseconds { DWORD dwReturn; DWORD dwFrames = HMSFToFrames(dwHmsf, dwFormat); int fps = (int)FPSDisplay(dwFormat); dwReturn = ((dwFrames * 1000) + (fps/2)) / fps; // (fps/2) for rounding return dwReturn; } PRIVATE DWORD NEAR PASCAL FramesToHMSF(DWORD dwFrames, DWORD dwFormat) // convert from frames to hmsf (colonized) format { HMSF hmsf; int fps = (int)FPSDisplay(dwFormat); hmsf.hours = (BYTE)(dwFrames / ((DWORD) 60 * 60 * fps)); hmsf.minutes = (BYTE)((dwFrames % ((DWORD) 60 * 60 * fps)) / (60 * fps)); hmsf.seconds = (BYTE)((dwFrames % ((DWORD) 60 * fps)) / fps); hmsf.frames = (BYTE)((dwFrames % fps)); return * ((DWORD FAR *) &hmsf); } /**************************** PUBLIC FUNCTIONS *************************/ PUBLIC BOOL NEAR PASCAL ColonizeOutput(pSeqStreamType pStream) // tells whether the user display type is such that the output should // displayed colonoized (i.e. "hh:mm:ss:ff") { if ((pStream->userDisplayType == MCI_FORMAT_SMPTE_24) || (pStream->userDisplayType == MCI_FORMAT_SMPTE_25) || (pStream->userDisplayType == MCI_FORMAT_SMPTE_30DROP) || (pStream->userDisplayType == MCI_FORMAT_SMPTE_30)) // smpte times are the only colonized formats return TRUE; else return FALSE; } PUBLIC BOOL NEAR PASCAL FormatsEqual(pSeqStreamType pStream) // tells whether the display format is compatible with the file format // (i.e. will conversion have to be done in interaction with the user) { BOOL bReturn; // Essentially, ppqn file type only compatible with song pointer display; // SMPTE compatible with anything but ppqn if (pStream->fileDivType == SEQ_DIV_PPQN) { if (pStream->userDisplayType == MCI_SEQ_FORMAT_SONGPTR) bReturn = TRUE; else bReturn = FALSE; } else { if ((pStream->userDisplayType == MCI_FORMAT_SMPTE_24) || (pStream->userDisplayType == MCI_FORMAT_SMPTE_25) || (pStream->userDisplayType == MCI_FORMAT_SMPTE_30DROP) || (pStream->userDisplayType == MCI_FORMAT_SMPTE_30)) bReturn = TRUE; else bReturn = FALSE; } return bReturn; } PUBLIC DWORD NEAR PASCAL CnvtTimeToSeq(pSeqStreamType pStream, DWORD dwCurrent, MIDISEQINFO FAR * pSeqInfo) /* The sequencer understands two time units: Ticks for ppqn files, or frames for smpte files. The user data is presented either as song pointer, milliseconds, or H(our)M(inute)S(econd)F(rame). This routine converts FROM user time TO sequencer time. */ { DWORD dwMs; // milliseconds DWORD fps; // frames per second; DWORD dwReturn; DWORD dwTicks; if (FormatsEqual(pStream)) { if ((pStream->userDisplayType == MCI_FORMAT_SMPTE_24) || (pStream->userDisplayType == MCI_FORMAT_SMPTE_25) || (pStream->userDisplayType == MCI_FORMAT_SMPTE_30) || (pStream->userDisplayType == MCI_FORMAT_SMPTE_30DROP)) // both file and display are smpte dwReturn = pSeqInfo->wResolution * HMSFToFrames(dwCurrent, pStream->userDisplayType); else //both file and display are ppqn (convert song pointer to ppqn) dwReturn = (pSeqInfo->wResolution * dwCurrent) / 4; } else if (pStream->fileDivType == SEQ_DIV_PPQN) //ppqn file, display format !ppqn { if (pStream->userDisplayType != MCI_FORMAT_MILLISECONDS) // must be smpte -- convert hmsf to milliseconds dwMs = HMSFToMS(dwCurrent, pStream->userDisplayType); else // must be milliseconds already dwMs = dwCurrent; // now that we have milliseconds, we must ask the sequencer to // convert them to ppqn (using it's internal tempo map) midiSeqMessage((HMIDISEQ) pStream->hSeq, SEQ_MSTOTICKS, dwMs, (DWORD_PTR)(LPSTR) &dwTicks); // passed back in dwTicks dwReturn = dwTicks; } else // smpte file, display format != smpte { // NB: don't worry about SONGPTR -- it's illegal here // also, don't worry about HMSF -- it's equal // The only possible case is ms -> ticks fps = FPSFile(pStream->fileDivType); dwReturn = ((dwCurrent * fps * pSeqInfo->wResolution) + 500) / 1000; // add 500 to round } return dwReturn; } PUBLIC DWORD NEAR PASCAL CnvtTimeFromSeq(pSeqStreamType pStream, DWORD dwTicks, MIDISEQINFO FAR * pSeqInfo) // This routine converts FROM sequencer time TO user time. { DWORD dwMs; // milliseconds DWORD fps; // frames per second; DWORD dwReturn; DWORD dwFrames; DWORD dwNativeUnits; if (pSeqInfo->wDivType == SEQ_DIV_PPQN) dwNativeUnits = (dwTicks * 4) / pSeqInfo->wResolution; else dwNativeUnits = dwTicks / pSeqInfo->wResolution; if (FormatsEqual(pStream)) { if ((pStream->userDisplayType == MCI_FORMAT_SMPTE_24) || (pStream->userDisplayType == MCI_FORMAT_SMPTE_25) || (pStream->userDisplayType == MCI_FORMAT_SMPTE_30) || (pStream->userDisplayType == MCI_FORMAT_SMPTE_30DROP)) dwReturn = FramesToHMSF(dwNativeUnits, pStream->userDisplayType); else dwReturn = dwNativeUnits; // no conversion needed } else if (pStream->fileDivType == SEQ_DIV_PPQN) { // convert song ptr to ms midiSeqMessage((HMIDISEQ) pStream->hSeq, SEQ_TICKSTOMS, dwTicks, (DWORD_PTR)(LPSTR) &dwMs); // passed back in dwSongPtr if (pStream->userDisplayType == MCI_FORMAT_MILLISECONDS) dwReturn = dwMs; else // convert from ms to frames, and then frames to hmsf format { fps = FPSDisplay(pStream->userDisplayType); dwFrames = (dwMs * fps) / 1000; dwReturn = FramesToHMSF(dwFrames, pStream->userDisplayType); } } else // smpte file { // NB: don't worry about SONGPTR -- it's illegal here // also, don't worry about HMSF -- it's "equal" // The only possible case is frames->ms // set stream display type default based on file div type fps = FPSFile(pStream->fileDivType); dwReturn = ((dwTicks * 1000) + (fps/2)) / (fps * pSeqInfo->wResolution); // add (fps/2) to round // BTW: it would take > 39 hours to overflow @ 30fps (23:59:59:29 is max) } return dwReturn; } PUBLIC BOOL NEAR PASCAL RangeCheck(pSeqStreamType pStream, DWORD dwValue) /* range checks raw, unconverted data. checks if data is negative, or past end of file length. also checks if smpte hours, minutes, seconds and frames are all valid. Returns TRUE if legal, otherwise FALSE */ { int fps; HMSF hmsf; DWORD dwLength; MIDISEQINFO seqInfo; // Get length, and convert it to display format midiSeqMessage((HMIDISEQ) pStream->hSeq, SEQ_GETINFO, (DWORD_PTR) (LPMIDISEQINFO) &seqInfo, 0L); dwLength = CnvtTimeFromSeq(pStream, seqInfo.dwLength, &seqInfo); switch (pStream->userDisplayType) // check length based on user format { case MCI_FORMAT_SMPTE_24: case MCI_FORMAT_SMPTE_25: case MCI_FORMAT_SMPTE_30: case MCI_FORMAT_SMPTE_30DROP: hmsf = * ((HMSF FAR *) &dwValue); fps = (int)FPSDisplay(pStream->userDisplayType); // get frames per second // check for format errors if (((int)hmsf.frames >= fps) || (hmsf.seconds >= 60) || (hmsf.minutes >= 60) || (hmsf.hours > 24)) return FALSE; // don't check for negative values, since using unsigned bytes // (2's comp. negs would get caught above anyway) // check for length error if (HMSFToMS(* ((DWORD FAR *) &dwValue), pStream->userDisplayType) > HMSFToMS(dwLength, pStream->userDisplayType)) return FALSE; break; case MCI_SEQ_FORMAT_SONGPTR: case MCI_FORMAT_MILLISECONDS: if (dwValue > dwLength) return FALSE; // past end } return TRUE; }