gdiplus: Implement GdipSetPathGradientBlend, with tests.
[wine/multimedia.git] / dlls / dmime / performance.c
blob0f76e36dd1fa19c76405fd713c9869d94eeb0176
1 /* IDirectMusicPerformance Implementation
3 * Copyright (C) 2003-2004 Rok Mandeljc
4 * Copyright (C) 2003-2004 Raphael Junqueira
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this program; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
21 #include "dmime_private.h"
23 WINE_DEFAULT_DEBUG_CHANNEL(dmime);
25 typedef struct IDirectMusicPerformance8Impl {
26 IDirectMusicPerformance8 IDirectMusicPerformance8_iface;
27 LONG ref;
28 /* IDirectMusicPerformanceImpl fields */
29 IDirectMusic8 *pDirectMusic;
30 IDirectSound *pDirectSound;
31 IDirectMusicGraph *pToolGraph;
32 DMUS_AUDIOPARAMS pParams;
33 /* global parameters */
34 BOOL fAutoDownload;
35 char cMasterGrooveLevel;
36 float fMasterTempo;
37 long lMasterVolume;
38 /* performance channels */
39 DMUSIC_PRIVATE_PCHANNEL PChannel[32];
40 /* IDirectMusicPerformance8Impl fields */
41 IDirectMusicAudioPath *pDefaultPath;
42 HANDLE hNotification;
43 REFERENCE_TIME rtMinimum;
44 REFERENCE_TIME rtLatencyTime;
45 DWORD dwBumperLength;
46 DWORD dwPrepareTime;
47 /** Message Processing */
48 HANDLE procThread;
49 DWORD procThreadId;
50 REFERENCE_TIME procThreadStartTime;
51 BOOL procThreadTicStarted;
52 CRITICAL_SECTION safe;
53 struct DMUS_PMSGItem *head;
54 struct DMUS_PMSGItem *imm_head;
55 } IDirectMusicPerformance8Impl;
57 typedef struct DMUS_PMSGItem DMUS_PMSGItem;
58 struct DMUS_PMSGItem {
59 DMUS_PMSGItem* next;
60 DMUS_PMSGItem* prev;
62 REFERENCE_TIME rtItemTime;
63 BOOL bInUse;
64 DWORD cb;
65 DMUS_PMSG pMsg;
68 #define DMUS_PMSGToItem(pMSG) ((DMUS_PMSGItem*) (((unsigned char*) pPMSG) - offsetof(DMUS_PMSGItem, pMsg)))
69 #define DMUS_ItemToPMSG(pItem) (&(pItem->pMsg))
70 #define DMUS_ItemRemoveFromQueue(This,pItem) \
72 if (pItem->prev) pItem->prev->next = pItem->next;\
73 if (pItem->next) pItem->next->prev = pItem->prev;\
74 if (This->head == pItem) This->head = pItem->next;\
75 if (This->imm_head == pItem) This->imm_head = pItem->next;\
76 pItem->bInUse = FALSE;\
79 #define PROCESSMSG_START (WM_APP + 0)
80 #define PROCESSMSG_EXIT (WM_APP + 1)
81 #define PROCESSMSG_REMOVE (WM_APP + 2)
82 #define PROCESSMSG_ADD (WM_APP + 4)
85 static DMUS_PMSGItem* ProceedMsg(IDirectMusicPerformance8Impl* This, DMUS_PMSGItem* cur) {
86 if (cur->pMsg.dwType == DMUS_PMSGT_NOTIFICATION) {
87 SetEvent(This->hNotification);
89 DMUS_ItemRemoveFromQueue(This, cur);
90 switch (cur->pMsg.dwType) {
91 case DMUS_PMSGT_WAVE:
92 case DMUS_PMSGT_TEMPO:
93 case DMUS_PMSGT_STOP:
94 default:
95 FIXME("Unhandled PMsg Type: 0x%x\n", cur->pMsg.dwType);
96 break;
98 return cur;
101 static DWORD WINAPI ProcessMsgThread(LPVOID lpParam) {
102 IDirectMusicPerformance8Impl* This = lpParam;
103 DWORD timeOut = INFINITE;
104 MSG msg;
105 HRESULT hr;
106 REFERENCE_TIME rtCurTime;
107 DMUS_PMSGItem* it = NULL;
108 DMUS_PMSGItem* cur = NULL;
109 DMUS_PMSGItem* it_next = NULL;
111 while (TRUE) {
112 DWORD dwDec = This->rtLatencyTime + This->dwBumperLength;
114 if (timeOut > 0) MsgWaitForMultipleObjects(0, NULL, FALSE, timeOut, QS_POSTMESSAGE|QS_SENDMESSAGE|QS_TIMER);
115 timeOut = INFINITE;
117 EnterCriticalSection(&This->safe);
118 hr = IDirectMusicPerformance8_GetTime(&This->IDirectMusicPerformance8_iface, &rtCurTime, NULL);
119 if (FAILED(hr)) {
120 goto outrefresh;
123 for (it = This->imm_head; NULL != it; ) {
124 it_next = it->next;
125 cur = ProceedMsg(This, it);
126 HeapFree(GetProcessHeap(), 0, cur);
127 it = it_next;
130 for (it = This->head; NULL != it && it->rtItemTime < rtCurTime + dwDec; ) {
131 it_next = it->next;
132 cur = ProceedMsg(This, it);
133 HeapFree(GetProcessHeap(), 0, cur);
134 it = it_next;
136 if (NULL != it) {
137 timeOut = ( it->rtItemTime - rtCurTime ) + This->rtLatencyTime;
140 outrefresh:
141 LeaveCriticalSection(&This->safe);
143 while (PeekMessageA(&msg, NULL, 0, 0, PM_REMOVE)) {
144 /** if hwnd we suppose that is a windows event ... */
145 if (NULL != msg.hwnd) {
146 TranslateMessage(&msg);
147 DispatchMessageA(&msg);
148 } else {
149 switch (msg.message) {
150 case WM_QUIT:
151 case PROCESSMSG_EXIT:
152 goto outofthread;
153 case PROCESSMSG_START:
154 break;
155 case PROCESSMSG_ADD:
156 break;
157 case PROCESSMSG_REMOVE:
158 break;
159 default:
160 ERR("Unhandled message %u. Critical Path\n", msg.message);
161 break;
166 /** here we should run a little of current AudioPath */
170 outofthread:
171 TRACE("(%p): Exiting\n", This);
173 return 0;
176 static BOOL PostMessageToProcessMsgThread(IDirectMusicPerformance8Impl* This, UINT iMsg) {
177 if (FALSE == This->procThreadTicStarted && PROCESSMSG_EXIT != iMsg) {
178 BOOL res;
179 This->procThread = CreateThread(NULL, 0, ProcessMsgThread, This, 0, &This->procThreadId);
180 if (NULL == This->procThread) return FALSE;
181 SetThreadPriority(This->procThread, THREAD_PRIORITY_TIME_CRITICAL);
182 This->procThreadTicStarted = TRUE;
183 while(1) {
184 res = PostThreadMessageA(This->procThreadId, iMsg, 0, 0);
185 /* Let the thread creates its message queue (with MsgWaitForMultipleObjects call) by yielding and retrying */
186 if (!res && (GetLastError() == ERROR_INVALID_THREAD_ID))
187 Sleep(0);
188 else
189 break;
191 return res;
193 return PostThreadMessageA(This->procThreadId, iMsg, 0, 0);
196 static inline IDirectMusicPerformance8Impl *impl_from_IDirectMusicPerformance8(IDirectMusicPerformance8 *iface)
198 return CONTAINING_RECORD(iface, IDirectMusicPerformance8Impl, IDirectMusicPerformance8_iface);
201 /* IDirectMusicPerformance8 IUnknown part: */
202 static HRESULT WINAPI IDirectMusicPerformance8Impl_QueryInterface(IDirectMusicPerformance8 *iface,
203 REFIID riid, void **ppv)
205 TRACE("(%p, %s,%p)\n", iface, debugstr_dmguid(riid), ppv);
207 if (IsEqualIID (riid, &IID_IUnknown) ||
208 IsEqualIID (riid, &IID_IDirectMusicPerformance) ||
209 IsEqualIID (riid, &IID_IDirectMusicPerformance2) ||
210 IsEqualIID (riid, &IID_IDirectMusicPerformance8)) {
211 *ppv = iface;
212 IUnknown_AddRef(iface);
213 return S_OK;
216 WARN("(%p, %s,%p): not found\n", iface, debugstr_dmguid(riid), ppv);
217 return E_NOINTERFACE;
220 static ULONG WINAPI IDirectMusicPerformance8Impl_AddRef(IDirectMusicPerformance8 *iface)
222 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
223 ULONG ref = InterlockedIncrement(&This->ref);
225 TRACE("(%p): AddRef from %d\n", This, ref - 1);
227 DMIME_LockModule();
229 return ref;
232 static ULONG WINAPI IDirectMusicPerformance8Impl_Release(IDirectMusicPerformance8 *iface)
234 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
235 ULONG ref = InterlockedDecrement(&This->ref);
237 TRACE("(%p): ReleaseRef to %d\n", This, ref);
239 if (ref == 0) {
240 This->safe.DebugInfo->Spare[0] = 0;
241 DeleteCriticalSection(&This->safe);
242 HeapFree(GetProcessHeap(), 0, This);
245 DMIME_UnlockModule();
247 return ref;
250 /* IDirectMusicPerformanceImpl IDirectMusicPerformance Interface part: */
251 static HRESULT WINAPI IDirectMusicPerformance8Impl_Init(IDirectMusicPerformance8 *iface,
252 IDirectMusic **ppDirectMusic, IDirectSound *pDirectSound, HWND hWnd)
254 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
256 FIXME("(iface = %p, dmusic = %p, dsound = %p, hwnd = %p)\n", This, ppDirectMusic, pDirectSound, hWnd);
257 if (This->pDirectMusic || This->pDirectSound)
258 return DMUS_E_ALREADY_INITED;
260 if (NULL == hWnd) {
261 hWnd = GetForegroundWindow();
264 if (NULL != pDirectSound) {
265 This->pDirectSound = pDirectSound;
266 IDirectSound_AddRef(This->pDirectSound);
267 } else {
268 DirectSoundCreate8(NULL, (LPDIRECTSOUND8*) &This->pDirectSound, NULL);
269 if (!This->pDirectSound) return DSERR_NODRIVER;
271 if (NULL != hWnd) {
272 IDirectSound8_SetCooperativeLevel(This->pDirectSound, hWnd, DSSCL_PRIORITY);
273 } else {
274 /* how to get the ForeGround window handle ? */
275 /*IDirectSound8_SetCooperativeLevel(This->pDirectSound, hWnd, DSSCL_PRIORITY);*/
279 if (NULL != ppDirectMusic && NULL != *ppDirectMusic) {
280 /* app creates it's own dmusic object and gives it to performance */
281 This->pDirectMusic = (IDirectMusic8*) *ppDirectMusic;
282 IDirectMusic8_AddRef(This->pDirectMusic);
283 } else {
284 HRESULT hr;
285 /* App enables the performance to initialize itself and needs a pointer to object */
286 hr = CoCreateInstance(&CLSID_DirectMusic, NULL, CLSCTX_INPROC_SERVER, &IID_IDirectMusic8, (void**)&This->pDirectMusic);
287 if (FAILED(hr))
288 return hr;
289 if (ppDirectMusic) {
290 *ppDirectMusic = (LPDIRECTMUSIC)This->pDirectMusic;
291 IDirectMusic8_AddRef((LPDIRECTMUSIC8)*ppDirectMusic);
295 return S_OK;
298 static HRESULT WINAPI IDirectMusicPerformance8Impl_PlaySegment(IDirectMusicPerformance8 *iface,
299 IDirectMusicSegment *pSegment, DWORD dwFlags, __int64 i64StartTime,
300 IDirectMusicSegmentState **ppSegmentState)
302 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
304 FIXME("(%p, %p, %d, 0x%s, %p): stub\n", This, pSegment, dwFlags,
305 wine_dbgstr_longlong(i64StartTime), ppSegmentState);
306 if (ppSegmentState)
307 return DMUSIC_CreateDirectMusicSegmentStateImpl(&IID_IDirectMusicSegmentState, (LPVOID*)ppSegmentState, NULL);
308 return S_OK;
311 static HRESULT WINAPI IDirectMusicPerformance8Impl_Stop(IDirectMusicPerformance8 *iface,
312 IDirectMusicSegment *pSegment, IDirectMusicSegmentState *pSegmentState, MUSIC_TIME mtTime,
313 DWORD dwFlags)
315 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
317 FIXME("(%p, %p, %p, %d, %d): stub\n", This, pSegment, pSegmentState, mtTime, dwFlags);
318 return S_OK;
321 static HRESULT WINAPI IDirectMusicPerformance8Impl_GetSegmentState(IDirectMusicPerformance8 *iface,
322 IDirectMusicSegmentState **ppSegmentState, MUSIC_TIME mtTime)
324 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
326 FIXME("(%p,%p, %d): stub\n", This, ppSegmentState, mtTime);
327 return S_OK;
330 static HRESULT WINAPI IDirectMusicPerformance8Impl_SetPrepareTime(IDirectMusicPerformance8 *iface,
331 DWORD dwMilliSeconds)
333 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
335 TRACE("(%p, %d)\n", This, dwMilliSeconds);
336 This->dwPrepareTime = dwMilliSeconds;
337 return S_OK;
340 static HRESULT WINAPI IDirectMusicPerformance8Impl_GetPrepareTime(IDirectMusicPerformance8 *iface,
341 DWORD *pdwMilliSeconds)
343 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
345 TRACE("(%p, %p)\n", This, pdwMilliSeconds);
346 if (NULL == pdwMilliSeconds) {
347 return E_POINTER;
349 *pdwMilliSeconds = This->dwPrepareTime;
350 return S_OK;
353 static HRESULT WINAPI IDirectMusicPerformance8Impl_SetBumperLength(IDirectMusicPerformance8 *iface,
354 DWORD dwMilliSeconds)
356 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
358 TRACE("(%p, %d)\n", This, dwMilliSeconds);
359 This->dwBumperLength = dwMilliSeconds;
360 return S_OK;
363 static HRESULT WINAPI IDirectMusicPerformance8Impl_GetBumperLength(IDirectMusicPerformance8 *iface,
364 DWORD *pdwMilliSeconds)
366 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
368 TRACE("(%p, %p)\n", This, pdwMilliSeconds);
369 if (NULL == pdwMilliSeconds) {
370 return E_POINTER;
372 *pdwMilliSeconds = This->dwBumperLength;
373 return S_OK;
376 static HRESULT WINAPI IDirectMusicPerformance8Impl_SendPMsg(IDirectMusicPerformance8 *iface,
377 DMUS_PMSG *pPMSG)
379 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
380 DMUS_PMSGItem* pItem = NULL;
381 DMUS_PMSGItem* it = NULL;
382 DMUS_PMSGItem* prev_it = NULL;
383 DMUS_PMSGItem** queue = NULL;
385 FIXME("(%p, %p): stub\n", This, pPMSG);
387 if (NULL == pPMSG) {
388 return E_POINTER;
390 pItem = DMUS_PMSGToItem(pPMSG);
391 if (NULL == pItem) {
392 return E_POINTER;
394 if (pItem->bInUse) {
395 return DMUS_E_ALREADY_SENT;
398 /* TODO: Valid Flags */
399 /* TODO: DMUS_PMSGF_MUSICTIME */
400 pItem->rtItemTime = pPMSG->rtTime;
402 if (pPMSG->dwFlags & DMUS_PMSGF_TOOL_IMMEDIATE) {
403 queue = &This->imm_head;
404 } else {
405 queue = &This->head;
408 EnterCriticalSection(&This->safe);
409 for (it = *queue; NULL != it && it->rtItemTime < pItem->rtItemTime; it = it->next) {
410 prev_it = it;
412 if (NULL == prev_it) {
413 pItem->prev = NULL;
414 if (NULL != *queue) pItem->next = (*queue)->next;
415 /*assert( NULL == pItem->next->prev );*/
416 if (NULL != pItem->next) pItem->next->prev = pItem;
417 *queue = pItem;
418 } else {
419 pItem->prev = prev_it;
420 pItem->next = prev_it->next;
421 prev_it->next = pItem;
422 if (NULL != pItem->next) pItem->next->prev = pItem;
424 LeaveCriticalSection(&This->safe);
426 /** now in use, prevent from stupid Frees */
427 pItem->bInUse = TRUE;
428 return S_OK;
431 static HRESULT WINAPI IDirectMusicPerformance8Impl_MusicToReferenceTime(IDirectMusicPerformance8 *iface,
432 MUSIC_TIME mtTime, REFERENCE_TIME *prtTime)
434 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
436 FIXME("(%p, %d, %p): stub\n", This, mtTime, prtTime);
437 return S_OK;
440 static HRESULT WINAPI IDirectMusicPerformance8Impl_ReferenceToMusicTime(IDirectMusicPerformance8 *iface,
441 REFERENCE_TIME rtTime, MUSIC_TIME *pmtTime)
443 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
445 FIXME("(%p, 0x%s, %p): stub\n", This, wine_dbgstr_longlong(rtTime), pmtTime);
446 return S_OK;
449 static HRESULT WINAPI IDirectMusicPerformance8Impl_IsPlaying(IDirectMusicPerformance8 *iface,
450 IDirectMusicSegment *pSegment, IDirectMusicSegmentState *pSegState)
452 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
454 FIXME("(%p, %p, %p): stub\n", This, pSegment, pSegState);
455 return S_FALSE;
458 static HRESULT WINAPI IDirectMusicPerformance8Impl_GetTime(IDirectMusicPerformance8 *iface,
459 REFERENCE_TIME *prtNow, MUSIC_TIME *pmtNow)
461 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
462 HRESULT hr = S_OK;
463 REFERENCE_TIME rtCur = 0;
465 /*TRACE("(%p, %p, %p)\n", This, prtNow, pmtNow); */
466 if (This->procThreadTicStarted) {
467 rtCur = ((REFERENCE_TIME) GetTickCount() * 10000) - This->procThreadStartTime;
468 } else {
469 /*return DMUS_E_NO_MASTER_CLOCK;*/
471 if (NULL != prtNow) {
472 *prtNow = rtCur;
474 if (NULL != pmtNow) {
475 hr = IDirectMusicPerformance8_ReferenceToMusicTime(iface, rtCur, pmtNow);
477 return hr;
480 static HRESULT WINAPI IDirectMusicPerformance8Impl_AllocPMsg(IDirectMusicPerformance8 *iface,
481 ULONG cb, DMUS_PMSG **ppPMSG)
483 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
484 DMUS_PMSGItem* pItem = NULL;
486 FIXME("(%p, %d, %p): stub\n", This, cb, ppPMSG);
488 if (sizeof(DMUS_PMSG) > cb) {
489 return E_INVALIDARG;
491 if (NULL == ppPMSG) {
492 return E_POINTER;
494 pItem = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, cb - sizeof(DMUS_PMSG) + sizeof(DMUS_PMSGItem));
495 if (NULL == pItem) {
496 return E_OUTOFMEMORY;
498 pItem->pMsg.dwSize = cb;
499 *ppPMSG = DMUS_ItemToPMSG(pItem);
500 return S_OK;
503 static HRESULT WINAPI IDirectMusicPerformance8Impl_FreePMsg(IDirectMusicPerformance8 *iface,
504 DMUS_PMSG *pPMSG)
506 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
507 DMUS_PMSGItem* pItem = NULL;
509 FIXME("(%p, %p): stub\n", This, pPMSG);
511 if (NULL == pPMSG) {
512 return E_POINTER;
514 pItem = DMUS_PMSGToItem(pPMSG);
515 if (NULL == pItem) {
516 return E_POINTER;
518 if (pItem->bInUse) {
519 /** prevent for freeing PMsg in queue (ie to be processed) */
520 return DMUS_E_CANNOT_FREE;
522 /** now we can remove it safely */
523 EnterCriticalSection(&This->safe);
524 DMUS_ItemRemoveFromQueue( This, pItem );
525 LeaveCriticalSection(&This->safe);
527 /** TODO: see if we should Release the pItem->pMsg->punkUser and others Interfaces */
528 HeapFree(GetProcessHeap(), 0, pItem);
529 return S_OK;
532 static HRESULT WINAPI IDirectMusicPerformance8Impl_GetGraph(IDirectMusicPerformance8 *iface,
533 IDirectMusicGraph **ppGraph)
535 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
537 FIXME("(%p, %p): to check\n", This, ppGraph);
538 if (NULL != This->pToolGraph) {
539 *ppGraph = This->pToolGraph;
540 IDirectMusicGraph_AddRef(*ppGraph);
541 } else {
542 return E_FAIL;
544 return S_OK;
547 static HRESULT WINAPI IDirectMusicPerformance8Impl_SetGraph(IDirectMusicPerformance8 *iface,
548 IDirectMusicGraph *pGraph)
550 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
552 FIXME("(%p, %p): to check\n", This, pGraph);
554 if (NULL != This->pToolGraph) {
555 /* Todo clean buffers and tools before */
556 IDirectMusicGraph_Release(This->pToolGraph);
558 This->pToolGraph = pGraph;
559 if (NULL != This->pToolGraph) {
560 IDirectMusicGraph_AddRef(This->pToolGraph);
562 return S_OK;
565 static HRESULT WINAPI IDirectMusicPerformance8Impl_SetNotificationHandle(IDirectMusicPerformance8 *iface,
566 HANDLE hNotification, REFERENCE_TIME rtMinimum)
568 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
570 FIXME("(%p, %p, 0x%s): stub\n", This, hNotification, wine_dbgstr_longlong(rtMinimum));
571 This->hNotification = hNotification;
572 if (rtMinimum) This->rtMinimum = rtMinimum;
573 return S_OK;
576 static HRESULT WINAPI IDirectMusicPerformance8Impl_GetNotificationPMsg(IDirectMusicPerformance8 *iface,
577 DMUS_NOTIFICATION_PMSG **ppNotificationPMsg)
579 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
581 FIXME("(%p, %p): stub\n", This, ppNotificationPMsg);
582 if (NULL == ppNotificationPMsg) {
583 return E_POINTER;
588 return S_FALSE;
589 /*return S_OK;*/
592 static HRESULT WINAPI IDirectMusicPerformance8Impl_AddNotificationType(IDirectMusicPerformance8 *iface,
593 REFGUID rguidNotificationType)
595 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
597 FIXME("(%p, %s): stub\n", This, debugstr_dmguid(rguidNotificationType));
598 return S_OK;
601 static HRESULT WINAPI IDirectMusicPerformance8Impl_RemoveNotificationType(IDirectMusicPerformance8 *iface,
602 REFGUID rguidNotificationType)
604 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
606 FIXME("(%p, %s): stub\n", This, debugstr_dmguid(rguidNotificationType));
607 return S_OK;
610 static HRESULT WINAPI IDirectMusicPerformance8Impl_AddPort(IDirectMusicPerformance8 *iface,
611 IDirectMusicPort *pPort)
613 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
614 HRESULT hr = E_FAIL;
616 FIXME("(%p, %p): stub\n", This, pPort);
617 if (!This->pDirectMusic || !This->pDirectSound) return DMUS_E_NOT_INIT;
618 if (NULL == pPort) {
619 GUID port_guid;
620 IDirectMusicPort* pDefaultPort = NULL;
621 DMUS_PORTPARAMS params;
622 int i, j;
623 hr = IDirectMusic8_GetDefaultPort(This->pDirectMusic, &port_guid);
624 if (FAILED(hr)) return hr;
625 ZeroMemory(&params, sizeof(params));
626 params.dwSize = sizeof(params);
627 params.dwValidParams = DMUS_PORTPARAMS_CHANNELGROUPS | DMUS_PORTPARAMS_SHARE;
628 params.dwChannelGroups = 1;
629 params.fShare = TRUE;
630 hr = IDirectMusic8_CreatePort(This->pDirectMusic, &port_guid, &params, &pDefaultPort, NULL);
631 if (FAILED(hr)) return hr;
632 hr = IDirectMusicPort_Activate(pDefaultPort, TRUE);
633 if (FAILED(hr)) { IDirectMusicPort_Release(pDefaultPort); return hr; }
634 j = 0;
635 for (i = 0; i < 16; ++i) {
636 if (NULL == This->PChannel[i].port) {
637 This->PChannel[i].port = pPort;
638 This->PChannel[i].group = 0;
639 This->PChannel[i].channel = j; /* FIXME: should this be assigned? */
640 j++;
643 } else {
644 IDirectMusicPort_AddRef(pPort);
647 * We should remember added Ports (for example using a list)
648 * and control if Port is registered for each api who use ports
650 return S_OK;
653 static HRESULT WINAPI IDirectMusicPerformance8Impl_RemovePort(IDirectMusicPerformance8 *iface,
654 IDirectMusicPort *pPort)
656 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
658 FIXME("(%p, %p): stub\n", This, pPort);
659 IDirectMusicPort_Release (pPort);
660 return S_OK;
663 static HRESULT WINAPI IDirectMusicPerformance8Impl_AssignPChannelBlock(IDirectMusicPerformance8 *iface,
664 DWORD dwBlockNum, IDirectMusicPort *pPort, DWORD dwGroup)
666 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
667 int i, j, range /* min value in range */;
669 FIXME("(%p, %d, %p, %d): semi-stub\n", This, dwBlockNum, pPort, dwGroup-1);
670 if (NULL == pPort) return E_POINTER;
672 range = 16 * dwBlockNum;
673 j = 0;
674 for (i = range; i < range+16; i++) {
675 /*TRACE("Setting PChannel[%i] to port %p, group %ld, MIDI port %i\n", i, pPort, dwGroup-1, j); */
676 This->PChannel[i].port = pPort;
677 This->PChannel[i].group = dwGroup - 1; /* first index is always zero */
678 This->PChannel[i].channel = j; /* FIXME: should this be assigned? */
679 j++;
681 /*if (dwGroup > 2) return S_FALSE;*/
683 return S_OK;
686 static HRESULT WINAPI IDirectMusicPerformance8Impl_AssignPChannel(IDirectMusicPerformance8 *iface,
687 DWORD dwPChannel, IDirectMusicPort *pPort, DWORD dwGroup, DWORD dwMChannel)
689 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
691 TRACE("(%p, %d, %p, %d, %d)\n", This, dwPChannel, pPort, dwGroup, dwMChannel);
692 if (NULL == pPort) return E_POINTER;
693 This->PChannel[dwPChannel].port = pPort;
694 This->PChannel[dwPChannel].group = dwGroup;
695 This->PChannel[dwPChannel].channel = dwMChannel;
697 return S_OK;
700 static HRESULT WINAPI IDirectMusicPerformance8Impl_PChannelInfo(IDirectMusicPerformance8 *iface,
701 DWORD dwPChannel, IDirectMusicPort **ppPort, DWORD *pdwGroup, DWORD *pdwMChannel)
703 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
704 DMUS_PORTPARAMS8 dmusportparams;
705 GUID def;
707 FIXME("(%p, %d, %p, %p, %p): stub\n", This, dwPChannel, ppPort, pdwGroup, pdwMChannel);
709 dmusportparams.dwSize = sizeof(DMUS_PORTPARAMS8);
710 dmusportparams.dwValidParams = 0;
711 IDirectMusic8_GetDefaultPort(This->pDirectMusic, &def);
712 IDirectMusic8_CreatePort(This->pDirectMusic, &def, &dmusportparams, ppPort, NULL);
714 return S_OK;
717 static HRESULT WINAPI IDirectMusicPerformance8Impl_DownloadInstrument(IDirectMusicPerformance8 *iface,
718 IDirectMusicInstrument *pInst, DWORD dwPChannel,
719 IDirectMusicDownloadedInstrument **ppDownInst, DMUS_NOTERANGE *pNoteRanges,
720 DWORD dwNumNoteRanges, IDirectMusicPort **ppPort, DWORD *pdwGroup, DWORD *pdwMChannel)
722 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
724 FIXME("(%p, %p, %d, %p, %p, %d, %p, %p, %p): stub\n", This, pInst, dwPChannel, ppDownInst, pNoteRanges, dwNumNoteRanges, ppPort, pdwGroup, pdwMChannel);
725 return S_OK;
728 static HRESULT WINAPI IDirectMusicPerformance8Impl_Invalidate(IDirectMusicPerformance8 *iface,
729 MUSIC_TIME mtTime, DWORD dwFlags)
731 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
733 FIXME("(%p, %d, %d): stub\n", This, mtTime, dwFlags);
734 return S_OK;
737 static HRESULT WINAPI IDirectMusicPerformance8Impl_GetParam(IDirectMusicPerformance8 *iface,
738 REFGUID rguidType, DWORD dwGroupBits, DWORD dwIndex, MUSIC_TIME mtTime,
739 MUSIC_TIME *pmtNext, void *pParam)
741 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
743 FIXME("(%p, %s, %d, %d, %d, %p, %p): stub\n", This, debugstr_dmguid(rguidType), dwGroupBits, dwIndex, mtTime, pmtNext, pParam);
744 return S_OK;
747 static HRESULT WINAPI IDirectMusicPerformance8Impl_SetParam(IDirectMusicPerformance8 *iface,
748 REFGUID rguidType, DWORD dwGroupBits, DWORD dwIndex, MUSIC_TIME mtTime, void *pParam)
750 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
752 FIXME("(%p, %s, %d, %d, %d, %p): stub\n", This, debugstr_dmguid(rguidType), dwGroupBits, dwIndex, mtTime, pParam);
753 return S_OK;
756 static HRESULT WINAPI IDirectMusicPerformance8Impl_GetGlobalParam(IDirectMusicPerformance8 *iface,
757 REFGUID rguidType, void *pParam, DWORD dwSize)
759 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
761 TRACE("(%p, %s, %p, %d): stub\n", This, debugstr_dmguid(rguidType), pParam, dwSize);
763 if (IsEqualGUID (rguidType, &GUID_PerfAutoDownload))
764 memcpy(pParam, &This->fAutoDownload, sizeof(This->fAutoDownload));
765 if (IsEqualGUID (rguidType, &GUID_PerfMasterGrooveLevel))
766 memcpy(pParam, &This->cMasterGrooveLevel, sizeof(This->cMasterGrooveLevel));
767 if (IsEqualGUID (rguidType, &GUID_PerfMasterTempo))
768 memcpy(pParam, &This->fMasterTempo, sizeof(This->fMasterTempo));
769 if (IsEqualGUID (rguidType, &GUID_PerfMasterVolume))
770 memcpy(pParam, &This->lMasterVolume, sizeof(This->lMasterVolume));
772 return S_OK;
775 static HRESULT WINAPI IDirectMusicPerformance8Impl_SetGlobalParam(IDirectMusicPerformance8 *iface,
776 REFGUID rguidType, void *pParam, DWORD dwSize)
778 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
780 TRACE("(%p, %s, %p, %d)\n", This, debugstr_dmguid(rguidType), pParam, dwSize);
782 if (IsEqualGUID (rguidType, &GUID_PerfAutoDownload)) {
783 memcpy(&This->fAutoDownload, pParam, dwSize);
784 TRACE("=> AutoDownload set to %d\n", This->fAutoDownload);
786 if (IsEqualGUID (rguidType, &GUID_PerfMasterGrooveLevel)) {
787 memcpy(&This->cMasterGrooveLevel, pParam, dwSize);
788 TRACE("=> MasterGrooveLevel set to %i\n", This->cMasterGrooveLevel);
790 if (IsEqualGUID (rguidType, &GUID_PerfMasterTempo)) {
791 memcpy(&This->fMasterTempo, pParam, dwSize);
792 TRACE("=> MasterTempo set to %f\n", This->fMasterTempo);
794 if (IsEqualGUID (rguidType, &GUID_PerfMasterVolume)) {
795 memcpy(&This->lMasterVolume, pParam, dwSize);
796 TRACE("=> MasterVolume set to %li\n", This->lMasterVolume);
799 return S_OK;
802 static HRESULT WINAPI IDirectMusicPerformance8Impl_GetLatencyTime(IDirectMusicPerformance8 *iface,
803 REFERENCE_TIME *prtTime)
805 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
807 TRACE("(%p, %p): stub\n", This, prtTime);
808 *prtTime = This->rtLatencyTime;
809 return S_OK;
812 static HRESULT WINAPI IDirectMusicPerformance8Impl_GetQueueTime(IDirectMusicPerformance8 *iface,
813 REFERENCE_TIME *prtTime)
816 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
818 FIXME("(%p, %p): stub\n", This, prtTime);
819 return S_OK;
822 static HRESULT WINAPI IDirectMusicPerformance8Impl_AdjustTime(IDirectMusicPerformance8 *iface,
823 REFERENCE_TIME rtAmount)
825 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
827 FIXME("(%p, 0x%s): stub\n", This, wine_dbgstr_longlong(rtAmount));
828 return S_OK;
831 static HRESULT WINAPI IDirectMusicPerformance8Impl_CloseDown(IDirectMusicPerformance8 *iface)
833 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
835 FIXME("(%p): stub\n", This);
836 if (PostMessageToProcessMsgThread(This, PROCESSMSG_EXIT)) {
837 WaitForSingleObject(This->procThread, INFINITE);
838 This->procThreadTicStarted = FALSE;
839 CloseHandle(This->procThread);
841 if (NULL != This->pDirectSound) {
842 IDirectSound_Release(This->pDirectSound);
843 This->pDirectSound = NULL;
845 if (NULL != This->pDirectMusic) {
846 IDirectMusic8_Release(This->pDirectMusic);
847 This->pDirectMusic = NULL;
849 return S_OK;
852 static HRESULT WINAPI IDirectMusicPerformance8Impl_GetResolvedTime(IDirectMusicPerformance8 *iface,
853 REFERENCE_TIME rtTime, REFERENCE_TIME *prtResolved, DWORD dwTimeResolveFlags)
855 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
857 FIXME("(%p, 0x%s, %p, %d): stub\n", This, wine_dbgstr_longlong(rtTime),
858 prtResolved, dwTimeResolveFlags);
859 return S_OK;
862 static HRESULT WINAPI IDirectMusicPerformance8Impl_MIDIToMusic(IDirectMusicPerformance8 *iface,
863 BYTE bMIDIValue, DMUS_CHORD_KEY *pChord, BYTE bPlayMode, BYTE bChordLevel,
864 WORD *pwMusicValue)
866 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
868 FIXME("(%p, %d, %p, %d, %d, %p): stub\n", This, bMIDIValue, pChord, bPlayMode, bChordLevel, pwMusicValue);
869 return S_OK;
872 static HRESULT WINAPI IDirectMusicPerformance8Impl_MusicToMIDI(IDirectMusicPerformance8 *iface,
873 WORD wMusicValue, DMUS_CHORD_KEY *pChord, BYTE bPlayMode, BYTE bChordLevel,
874 BYTE *pbMIDIValue)
876 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
878 FIXME("(%p, %d, %p, %d, %d, %p): stub\n", This, wMusicValue, pChord, bPlayMode, bChordLevel, pbMIDIValue);
879 return S_OK;
882 static HRESULT WINAPI IDirectMusicPerformance8Impl_TimeToRhythm(IDirectMusicPerformance8 *iface,
883 MUSIC_TIME mtTime, DMUS_TIMESIGNATURE *pTimeSig, WORD *pwMeasure, BYTE *pbBeat,
884 BYTE *pbGrid, short *pnOffset)
886 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
888 FIXME("(%p, %d, %p, %p, %p, %p, %p): stub\n", This, mtTime, pTimeSig, pwMeasure, pbBeat, pbGrid, pnOffset);
889 return S_OK;
892 static HRESULT WINAPI IDirectMusicPerformance8Impl_RhythmToTime(IDirectMusicPerformance8 *iface,
893 WORD wMeasure, BYTE bBeat, BYTE bGrid, short nOffset, DMUS_TIMESIGNATURE *pTimeSig,
894 MUSIC_TIME *pmtTime)
896 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
898 FIXME("(%p, %d, %d, %d, %i, %p, %p): stub\n", This, wMeasure, bBeat, bGrid, nOffset, pTimeSig, pmtTime);
899 return S_OK;
902 /* IDirectMusicPerformance8 Interface part follow: */
903 static HRESULT WINAPI IDirectMusicPerformance8Impl_InitAudio(IDirectMusicPerformance8 *iface,
904 IDirectMusic **ppDirectMusic, IDirectSound **ppDirectSound, HWND hWnd,
905 DWORD dwDefaultPathType, DWORD dwPChannelCount, DWORD dwFlags, DMUS_AUDIOPARAMS* pParams)
907 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
908 IDirectSound* dsound = NULL;
909 HRESULT hr = S_OK;
911 FIXME("(%p, %p, %p, %p, %x, %u, %x, %p): to check\n", This, ppDirectMusic, ppDirectSound, hWnd, dwDefaultPathType, dwPChannelCount, dwFlags, pParams);
913 if (This->pDirectMusic || This->pDirectSound)
914 return DMUS_E_ALREADY_INITED;
916 if (NULL != ppDirectSound && NULL != *ppDirectSound) {
917 dsound = *ppDirectSound;
918 } else {
919 hr = DirectSoundCreate8 (NULL, (LPDIRECTSOUND8*) &dsound, NULL);
920 FIXME("return dsound(%p,%d)\n", dsound, hr);
921 if (FAILED(hr) || !dsound)
922 return DSERR_NODRIVER;
923 if (ppDirectSound)
924 *ppDirectSound = dsound;
927 IDirectMusicPerformance8Impl_Init(iface, ppDirectMusic, dsound, hWnd);
929 /* Init increases the ref count of the dsound object. Decrement it if the app doesn't want a pointer to the object. */
930 if (NULL == ppDirectSound) {
931 IDirectSound_Release(This->pDirectSound);
934 /* as seen in msdn we need params init before audio path creation */
935 if (NULL != pParams) {
936 This->pParams = *pParams;
937 } else {
938 /* TODO, how can i fill the struct as seen on msdn */
939 memset(&This->pParams, 0, sizeof(DMUS_AUDIOPARAMS));
940 This->pParams.dwSize = sizeof(DMUS_AUDIOPARAMS);
941 This->pParams.fInitNow = FALSE;
942 This->pParams.dwValidData = DMUS_AUDIOPARAMS_FEATURES | DMUS_AUDIOPARAMS_VOICES | DMUS_AUDIOPARAMS_SAMPLERATE | DMUS_AUDIOPARAMS_DEFAULTSYNTH;
943 This->pParams.dwVoices = 64;
944 This->pParams.dwSampleRate = (DWORD) 22.050;
945 This->pParams.dwFeatures = dwFlags;
946 This->pParams.clsidDefaultSynth = CLSID_DirectMusicSynthSink;
948 if(dwDefaultPathType != 0)
949 hr = IDirectMusicPerformance8_CreateStandardAudioPath(iface, dwDefaultPathType, dwPChannelCount, FALSE, &This->pDefaultPath);
951 PostMessageToProcessMsgThread(This, PROCESSMSG_START);
953 return hr;
956 static HRESULT WINAPI IDirectMusicPerformance8Impl_PlaySegmentEx(IDirectMusicPerformance8 *iface,
957 IUnknown *pSource, WCHAR *pwzSegmentName, IUnknown *pTransition, DWORD dwFlags,
958 __int64 i64StartTime, IDirectMusicSegmentState **ppSegmentState, IUnknown *pFrom,
959 IUnknown *pAudioPath)
961 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
963 FIXME("(%p, %p, %p, %p, %d, 0x%s, %p, %p, %p): stub\n", This, pSource, pwzSegmentName,
964 pTransition, dwFlags, wine_dbgstr_longlong(i64StartTime), ppSegmentState, pFrom, pAudioPath);
965 if (ppSegmentState)
966 return DMUSIC_CreateDirectMusicSegmentStateImpl(&IID_IDirectMusicSegmentState, (LPVOID*)ppSegmentState, NULL);
967 return S_OK;
970 static HRESULT WINAPI IDirectMusicPerformance8Impl_StopEx(IDirectMusicPerformance8 *iface,
971 IUnknown *pObjectToStop, __int64 i64StopTime, DWORD dwFlags)
973 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
975 FIXME("(%p, %p, 0x%s, %d): stub\n", This, pObjectToStop,
976 wine_dbgstr_longlong(i64StopTime), dwFlags);
977 return S_OK;
980 static HRESULT WINAPI IDirectMusicPerformance8Impl_ClonePMsg(IDirectMusicPerformance8 *iface,
981 DMUS_PMSG *pSourcePMSG, DMUS_PMSG **ppCopyPMSG)
983 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
985 FIXME("(%p, %p, %p): stub\n", This, pSourcePMSG, ppCopyPMSG);
986 return S_OK;
989 static HRESULT WINAPI IDirectMusicPerformance8Impl_CreateAudioPath(IDirectMusicPerformance8 *iface,
990 IUnknown *pSourceConfig, BOOL fActivate, IDirectMusicAudioPath **ppNewPath)
992 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
993 IDirectMusicAudioPathImpl *default_path;
994 IDirectMusicAudioPath *pPath;
996 FIXME("(%p, %p, %d, %p): stub\n", This, pSourceConfig, fActivate, ppNewPath);
998 if (NULL == ppNewPath) {
999 return E_POINTER;
1002 DMUSIC_CreateDirectMusicAudioPathImpl (&IID_IDirectMusicAudioPath, (LPVOID*)&pPath, NULL);
1003 default_path = (IDirectMusicAudioPathImpl*)((char*)(pPath) - offsetof(IDirectMusicAudioPathImpl,AudioPathVtbl));
1004 default_path->pPerf = &This->IDirectMusicPerformance8_iface;
1006 /** TODO */
1008 *ppNewPath = pPath;
1010 return IDirectMusicAudioPath_Activate(*ppNewPath, fActivate);
1013 static HRESULT WINAPI IDirectMusicPerformance8Impl_CreateStandardAudioPath(IDirectMusicPerformance8 *iface,
1014 DWORD dwType, DWORD dwPChannelCount, BOOL fActivate, IDirectMusicAudioPath **ppNewPath)
1016 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
1017 IDirectMusicAudioPathImpl *default_path;
1018 IDirectMusicAudioPath *pPath;
1019 DSBUFFERDESC desc;
1020 WAVEFORMATEX format;
1021 LPDIRECTSOUNDBUFFER buffer;
1022 HRESULT hr = S_OK;
1024 FIXME("(%p)->(%d, %d, %d, %p): semi-stub\n", This, dwType, dwPChannelCount, fActivate, ppNewPath);
1026 if (NULL == ppNewPath) {
1027 return E_POINTER;
1030 DMUSIC_CreateDirectMusicAudioPathImpl (&IID_IDirectMusicAudioPath, (LPVOID*)&pPath, NULL);
1031 default_path = (IDirectMusicAudioPathImpl*)((char*)(pPath) - offsetof(IDirectMusicAudioPathImpl,AudioPathVtbl));
1032 default_path->pPerf = &This->IDirectMusicPerformance8_iface;
1034 /* Secondary buffer description */
1035 memset(&format, 0, sizeof(format));
1036 format.wFormatTag = WAVE_FORMAT_PCM;
1037 format.nChannels = 1;
1038 format.nSamplesPerSec = 44000;
1039 format.nAvgBytesPerSec = 44000*2;
1040 format.nBlockAlign = 2;
1041 format.wBitsPerSample = 16;
1042 format.cbSize = 0;
1044 memset(&desc, 0, sizeof(desc));
1045 desc.dwSize = sizeof(desc);
1046 desc.dwFlags = DSBCAPS_CTRLFX | DSBCAPS_CTRLPAN | DSBCAPS_CTRLVOLUME | DSBCAPS_GLOBALFOCUS;
1047 desc.dwBufferBytes = DSBSIZE_MIN;
1048 desc.dwReserved = 0;
1049 desc.lpwfxFormat = &format;
1050 desc.guid3DAlgorithm = GUID_NULL;
1052 switch(dwType) {
1053 case DMUS_APATH_DYNAMIC_3D:
1054 desc.dwFlags |= DSBCAPS_CTRL3D | DSBCAPS_CTRLFREQUENCY | DSBCAPS_MUTE3DATMAXDISTANCE;
1055 break;
1056 case DMUS_APATH_DYNAMIC_MONO:
1057 desc.dwFlags |= DSBCAPS_CTRLFREQUENCY;
1058 break;
1059 case DMUS_APATH_SHARED_STEREOPLUSREVERB:
1060 /* normally we have to create 2 buffers (one for music other for reverb)
1061 * in this case. See msdn
1063 case DMUS_APATH_DYNAMIC_STEREO:
1064 desc.dwFlags |= DSBCAPS_CTRLFREQUENCY;
1065 format.nChannels = 2;
1066 format.nBlockAlign *= 2;
1067 format.nAvgBytesPerSec *=2;
1068 break;
1069 default:
1070 HeapFree(GetProcessHeap(), 0, default_path);
1071 *ppNewPath = NULL;
1072 return E_INVALIDARG;
1075 /* FIXME: Should we create one secondary buffer for each PChannel? */
1076 hr = IDirectSound8_CreateSoundBuffer ((LPDIRECTSOUND8) This->pDirectSound, &desc, &buffer, NULL);
1077 if (FAILED(hr)) {
1078 HeapFree(GetProcessHeap(), 0, default_path);
1079 *ppNewPath = NULL;
1080 return DSERR_BUFFERLOST;
1082 default_path->pDSBuffer = buffer;
1084 /* Update description for creating primary buffer */
1085 desc.dwFlags |= DSBCAPS_PRIMARYBUFFER;
1086 desc.dwBufferBytes = 0;
1087 desc.lpwfxFormat = NULL;
1089 hr = IDirectSound8_CreateSoundBuffer ((LPDIRECTSOUND8) This->pDirectSound, &desc, &buffer, NULL);
1090 if (FAILED(hr)) {
1091 IDirectSoundBuffer_Release(default_path->pDSBuffer);
1092 HeapFree(GetProcessHeap(), 0, default_path);
1093 *ppNewPath = NULL;
1094 return DSERR_BUFFERLOST;
1096 default_path->pPrimary = buffer;
1098 *ppNewPath = pPath;
1100 TRACE(" returning IDirectMusicAudioPath interface at %p.\n", *ppNewPath);
1102 return IDirectMusicAudioPath_Activate(*ppNewPath, fActivate);
1105 static HRESULT WINAPI IDirectMusicPerformance8Impl_SetDefaultAudioPath(IDirectMusicPerformance8 *iface,
1106 IDirectMusicAudioPath *pAudioPath)
1108 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
1110 FIXME("(%p, %p): semi-stub\n", This, pAudioPath);
1111 if (NULL != This->pDefaultPath) {
1112 IDirectMusicAudioPath_Release(This->pDefaultPath);
1113 ((IDirectMusicAudioPathImpl*) This->pDefaultPath)->pPerf = NULL;
1114 This->pDefaultPath = NULL;
1116 This->pDefaultPath = pAudioPath;
1117 if (NULL != This->pDefaultPath) {
1118 IDirectMusicAudioPath_AddRef(This->pDefaultPath);
1119 ((IDirectMusicAudioPathImpl*)This->pDefaultPath)->pPerf = &This->IDirectMusicPerformance8_iface;
1122 return S_OK;
1125 static HRESULT WINAPI IDirectMusicPerformance8Impl_GetDefaultAudioPath(IDirectMusicPerformance8 *iface,
1126 IDirectMusicAudioPath **ppAudioPath)
1128 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
1130 FIXME("(%p, %p): semi-stub (%p)\n", This, ppAudioPath, This->pDefaultPath);
1132 if (NULL != This->pDefaultPath) {
1133 *ppAudioPath = This->pDefaultPath;
1134 IDirectMusicAudioPath_AddRef(*ppAudioPath);
1135 } else {
1136 *ppAudioPath = NULL;
1138 return S_OK;
1141 static HRESULT WINAPI IDirectMusicPerformance8Impl_GetParamEx(IDirectMusicPerformance8 *iface,
1142 REFGUID rguidType, DWORD dwTrackID, DWORD dwGroupBits, DWORD dwIndex, MUSIC_TIME mtTime,
1143 MUSIC_TIME *pmtNext, void *pParam)
1145 IDirectMusicPerformance8Impl *This = impl_from_IDirectMusicPerformance8(iface);
1147 FIXME("(%p, %s, %d, %d, %d, %d, %p, %p): stub\n", This, debugstr_dmguid(rguidType), dwTrackID, dwGroupBits, dwIndex, mtTime, pmtNext, pParam);
1149 return S_OK;
1152 static const IDirectMusicPerformance8Vtbl DirectMusicPerformance8_Vtbl = {
1153 IDirectMusicPerformance8Impl_QueryInterface,
1154 IDirectMusicPerformance8Impl_AddRef,
1155 IDirectMusicPerformance8Impl_Release,
1156 IDirectMusicPerformance8Impl_Init,
1157 IDirectMusicPerformance8Impl_PlaySegment,
1158 IDirectMusicPerformance8Impl_Stop,
1159 IDirectMusicPerformance8Impl_GetSegmentState,
1160 IDirectMusicPerformance8Impl_SetPrepareTime,
1161 IDirectMusicPerformance8Impl_GetPrepareTime,
1162 IDirectMusicPerformance8Impl_SetBumperLength,
1163 IDirectMusicPerformance8Impl_GetBumperLength,
1164 IDirectMusicPerformance8Impl_SendPMsg,
1165 IDirectMusicPerformance8Impl_MusicToReferenceTime,
1166 IDirectMusicPerformance8Impl_ReferenceToMusicTime,
1167 IDirectMusicPerformance8Impl_IsPlaying,
1168 IDirectMusicPerformance8Impl_GetTime,
1169 IDirectMusicPerformance8Impl_AllocPMsg,
1170 IDirectMusicPerformance8Impl_FreePMsg,
1171 IDirectMusicPerformance8Impl_GetGraph,
1172 IDirectMusicPerformance8Impl_SetGraph,
1173 IDirectMusicPerformance8Impl_SetNotificationHandle,
1174 IDirectMusicPerformance8Impl_GetNotificationPMsg,
1175 IDirectMusicPerformance8Impl_AddNotificationType,
1176 IDirectMusicPerformance8Impl_RemoveNotificationType,
1177 IDirectMusicPerformance8Impl_AddPort,
1178 IDirectMusicPerformance8Impl_RemovePort,
1179 IDirectMusicPerformance8Impl_AssignPChannelBlock,
1180 IDirectMusicPerformance8Impl_AssignPChannel,
1181 IDirectMusicPerformance8Impl_PChannelInfo,
1182 IDirectMusicPerformance8Impl_DownloadInstrument,
1183 IDirectMusicPerformance8Impl_Invalidate,
1184 IDirectMusicPerformance8Impl_GetParam,
1185 IDirectMusicPerformance8Impl_SetParam,
1186 IDirectMusicPerformance8Impl_GetGlobalParam,
1187 IDirectMusicPerformance8Impl_SetGlobalParam,
1188 IDirectMusicPerformance8Impl_GetLatencyTime,
1189 IDirectMusicPerformance8Impl_GetQueueTime,
1190 IDirectMusicPerformance8Impl_AdjustTime,
1191 IDirectMusicPerformance8Impl_CloseDown,
1192 IDirectMusicPerformance8Impl_GetResolvedTime,
1193 IDirectMusicPerformance8Impl_MIDIToMusic,
1194 IDirectMusicPerformance8Impl_MusicToMIDI,
1195 IDirectMusicPerformance8Impl_TimeToRhythm,
1196 IDirectMusicPerformance8Impl_RhythmToTime,
1197 IDirectMusicPerformance8Impl_InitAudio,
1198 IDirectMusicPerformance8Impl_PlaySegmentEx,
1199 IDirectMusicPerformance8Impl_StopEx,
1200 IDirectMusicPerformance8Impl_ClonePMsg,
1201 IDirectMusicPerformance8Impl_CreateAudioPath,
1202 IDirectMusicPerformance8Impl_CreateStandardAudioPath,
1203 IDirectMusicPerformance8Impl_SetDefaultAudioPath,
1204 IDirectMusicPerformance8Impl_GetDefaultAudioPath,
1205 IDirectMusicPerformance8Impl_GetParamEx
1208 /* for ClassFactory */
1209 HRESULT WINAPI DMUSIC_CreateDirectMusicPerformanceImpl (LPCGUID lpcGUID, LPVOID *ppobj, LPUNKNOWN pUnkOuter) {
1210 IDirectMusicPerformance8Impl *obj;
1212 TRACE("(%p,%p,%p)\n", lpcGUID, ppobj, pUnkOuter);
1214 if (pUnkOuter) {
1215 *ppobj = NULL;
1216 return CLASS_E_NOAGGREGATION;
1219 obj = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IDirectMusicPerformance8Impl));
1220 if (NULL == obj) {
1221 *ppobj = NULL;
1222 return E_OUTOFMEMORY;
1224 obj->IDirectMusicPerformance8_iface.lpVtbl = &DirectMusicPerformance8_Vtbl;
1225 obj->ref = 0; /* will be inited by QueryInterface */
1226 obj->pDirectMusic = NULL;
1227 obj->pDirectSound = NULL;
1228 obj->pDefaultPath = NULL;
1229 InitializeCriticalSection(&obj->safe);
1230 obj->safe.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": IDirectMusicPerformance8Impl*->safe");
1232 obj->rtLatencyTime = 100; /* 100ms TO FIX */
1233 obj->dwBumperLength = 50; /* 50ms default */
1234 obj->dwPrepareTime = 1000; /* 1000ms default */
1235 return IDirectMusicPerformance8Impl_QueryInterface(&obj->IDirectMusicPerformance8_iface,
1236 lpcGUID, ppobj);