Commit 381012f7 authored by Akihiro Sagawa's avatar Akihiro Sagawa Committed by Alexandre Julliard

winmm: Preserve fractions in microseconds to compensate rounding errors.

Otherwise, a song is played faster than expected in some cases. For instance, we assume that 40 pan control change (CC#10) messages are expressed in 12.5ms each, * Prior art omits fractions, i.e. 12ms + 12ms + ... = 480ms. * Proposal technique saves fractions, i.e. 12ms + 13ms + ... = 500ms. Ideal duration is 500ms. So, the proposed method is better than prior one. Signed-off-by: 's avatarAkihiro Sagawa <sagawa.aki@gmail.com> Signed-off-by: 's avatarAndrew Eikum <aeikum@codeweavers.com> Signed-off-by: 's avatarAlexandre Julliard <julliard@winehq.org>
parent 2cb32626
...@@ -921,6 +921,7 @@ typedef struct WINE_MIDIStream { ...@@ -921,6 +921,7 @@ typedef struct WINE_MIDIStream {
DWORD dwElapsedMS; DWORD dwElapsedMS;
WORD wFlags; WORD wFlags;
WORD status; WORD status;
WORD remainder;
HANDLE hEvent; HANDLE hEvent;
LPMIDIHDR lpMidiHdr; LPMIDIHDR lpMidiHdr;
} WINE_MIDIStream; } WINE_MIDIStream;
...@@ -965,13 +966,13 @@ static DWORD MMSYSTEM_MidiStream_Convert(WINE_MIDIStream* lpMidiStrm, DWORD puls ...@@ -965,13 +966,13 @@ static DWORD MMSYSTEM_MidiStream_Convert(WINE_MIDIStream* lpMidiStrm, DWORD puls
} else if (lpMidiStrm->dwTimeDiv > 0x8000) { /* SMPTE, unchecked FIXME? */ } else if (lpMidiStrm->dwTimeDiv > 0x8000) { /* SMPTE, unchecked FIXME? */
int nf = 256 - HIBYTE(lpMidiStrm->dwTimeDiv); /* number of frames */ int nf = 256 - HIBYTE(lpMidiStrm->dwTimeDiv); /* number of frames */
int nsf = LOBYTE(lpMidiStrm->dwTimeDiv); /* number of sub-frames */ int nsf = LOBYTE(lpMidiStrm->dwTimeDiv); /* number of sub-frames */
ret = (pulse * 1000) / (nf * nsf); ret = (pulse * 1000000) / (nf * nsf);
} else { } else {
ret = (DWORD)((double)pulse * ((double)lpMidiStrm->dwTempo / 1000) / ret = (DWORD)((double)pulse * (double)lpMidiStrm->dwTempo /
(double)lpMidiStrm->dwTimeDiv); (double)lpMidiStrm->dwTimeDiv);
} }
return ret; return ret; /* in microseconds */
} }
static DWORD midistream_get_playing_position(WINE_MIDIStream* lpMidiStrm) static DWORD midistream_get_playing_position(WINE_MIDIStream* lpMidiStrm)
...@@ -1007,6 +1008,8 @@ static BOOL MMSYSTEM_MidiStream_MessageHandler(WINE_MIDIStream* lpMidiStrm, LPWI ...@@ -1007,6 +1008,8 @@ static BOOL MMSYSTEM_MidiStream_MessageHandler(WINE_MIDIStream* lpMidiStrm, LPWI
lpMidiStrm->status = MSM_STATUS_STOPPED; lpMidiStrm->status = MSM_STATUS_STOPPED;
lpMidiStrm->dwPulses = 0; lpMidiStrm->dwPulses = 0;
lpMidiStrm->dwElapsedMS = 0; lpMidiStrm->dwElapsedMS = 0;
lpMidiStrm->dwPositionMS = 0;
lpMidiStrm->remainder = 0;
LeaveCriticalSection(&lpMidiStrm->lock); LeaveCriticalSection(&lpMidiStrm->lock);
/* this is not quite what MS doc says... */ /* this is not quite what MS doc says... */
midiOutReset(lpMidiStrm->hDevice); midiOutReset(lpMidiStrm->hDevice);
...@@ -1178,8 +1181,11 @@ start_header: ...@@ -1178,8 +1181,11 @@ start_header:
/* do we have to wait ? */ /* do we have to wait ? */
if (me->dwDeltaTime) { if (me->dwDeltaTime) {
DWORD delta;
EnterCriticalSection(&lpMidiStrm->lock); EnterCriticalSection(&lpMidiStrm->lock);
lpMidiStrm->dwPositionMS += MMSYSTEM_MidiStream_Convert(lpMidiStrm, me->dwDeltaTime); delta = lpMidiStrm->remainder + MMSYSTEM_MidiStream_Convert(lpMidiStrm, me->dwDeltaTime);
lpMidiStrm->dwPositionMS += delta / 1000;
lpMidiStrm->remainder = delta % 1000;
lpMidiStrm->dwPulses += me->dwDeltaTime; lpMidiStrm->dwPulses += me->dwDeltaTime;
LeaveCriticalSection(&lpMidiStrm->lock); LeaveCriticalSection(&lpMidiStrm->lock);
...@@ -1331,6 +1337,7 @@ MMRESULT WINAPI midiStreamOpen(HMIDISTRM* lphMidiStrm, LPUINT lpuDeviceID, ...@@ -1331,6 +1337,7 @@ MMRESULT WINAPI midiStreamOpen(HMIDISTRM* lphMidiStrm, LPUINT lpuDeviceID,
lpMidiStrm->dwTempo = 500000; /* micro seconds per quarter note, i.e. 120 BPM */ lpMidiStrm->dwTempo = 500000; /* micro seconds per quarter note, i.e. 120 BPM */
lpMidiStrm->dwTimeDiv = 24; /* ticks per quarter note */ lpMidiStrm->dwTimeDiv = 24; /* ticks per quarter note */
lpMidiStrm->dwPositionMS = 0; lpMidiStrm->dwPositionMS = 0;
lpMidiStrm->remainder = 0;
lpMidiStrm->status = MSM_STATUS_PAUSED; lpMidiStrm->status = MSM_STATUS_PAUSED;
lpMidiStrm->dwElapsedMS = 0; lpMidiStrm->dwElapsedMS = 0;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment