winecoreaudio.drv: Add support for waveOutBreakLoop/WODM_BREAKLOOP.
[wine.git] / dlls / dswave / dswave.c
blob6e0f9851589eecbc8b58d0216cc469384a669957
1 /* IDirectMusicWave Implementation
3 * Copyright (C) 2003-2004 Rok Mandeljc
5 * This program is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2.1 of the License, or (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this program; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
20 #include "dswave_private.h"
22 WINE_DEFAULT_DEBUG_CHANNEL(dswave);
23 WINE_DECLARE_DEBUG_CHANNEL(dmfile);
25 /* an interface that is, according to my tests, obtained by loader after loading object; is it acting
26 as some sort of bridge between object and loader? */
27 static const GUID IID_IDirectMusicWavePRIVATE = {0x69e934e4,0x97f1,0x4f1d,{0x88,0xe8,0xf2,0xac,0x88,0x67,0x13,0x27}};
29 static ULONG WINAPI IDirectMusicWaveImpl_IUnknown_AddRef (LPUNKNOWN iface);
30 static ULONG WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_AddRef (LPDIRECTMUSICSEGMENT8 iface);
31 static ULONG WINAPI IDirectMusicWaveImpl_IDirectMusicObject_AddRef (LPDIRECTMUSICOBJECT iface);
32 static ULONG WINAPI IDirectMusicWaveImpl_IPersistStream_AddRef (LPPERSISTSTREAM iface);
34 /*****************************************************************************
35 * IDirectMusicWaveImpl implementation
37 /* IDirectMusicWaveImpl IUnknown part: */
38 static HRESULT WINAPI IDirectMusicWaveImpl_IUnknown_QueryInterface (LPUNKNOWN iface, REFIID riid, LPVOID *ppobj) {
39 ICOM_THIS_MULTI(IDirectMusicWaveImpl, UnknownVtbl, iface);
41 TRACE("(%p, %s, %p)\n", This, debugstr_dmguid(riid), ppobj);
43 if (IsEqualIID (riid, &IID_IUnknown)) {
44 *ppobj = &This->UnknownVtbl;
45 IDirectMusicWaveImpl_IUnknown_AddRef ((LPUNKNOWN)&This->UnknownVtbl);
46 return S_OK;
47 } else if (IsEqualIID (riid, &IID_IDirectMusicSegment)
48 || IsEqualIID (riid, &IID_IDirectMusicSegment2)
49 || IsEqualIID (riid, &IID_IDirectMusicSegment8)) {
50 *ppobj = &This->SegmentVtbl;
51 IDirectMusicWaveImpl_IDirectMusicSegment8_AddRef ((LPDIRECTMUSICSEGMENT8)&This->SegmentVtbl);
52 return S_OK;
53 } else if (IsEqualIID (riid, &IID_IDirectMusicObject)) {
54 *ppobj = &This->ObjectVtbl;
55 IDirectMusicWaveImpl_IDirectMusicObject_AddRef ((LPDIRECTMUSICOBJECT)&This->ObjectVtbl);
56 return S_OK;
57 } else if (IsEqualIID (riid, &IID_IPersistStream)) {
58 *ppobj = &This->PersistStreamVtbl;
59 IDirectMusicWaveImpl_IPersistStream_AddRef ((LPPERSISTSTREAM)&This->PersistStreamVtbl);
60 return S_OK;
61 } else if (IsEqualIID (riid, &IID_IDirectMusicWavePRIVATE)) {
62 WARN(": requested private interface, expect crash\n");
63 return E_NOINTERFACE;
66 WARN("(%p, %s, %p): not found\n", This, debugstr_dmguid(riid), ppobj);
67 return E_NOINTERFACE;
70 static ULONG WINAPI IDirectMusicWaveImpl_IUnknown_AddRef (LPUNKNOWN iface) {
71 ICOM_THIS_MULTI(IDirectMusicWaveImpl, UnknownVtbl, iface);
72 ULONG refCount = InterlockedIncrement(&This->ref);
74 TRACE("(%p)->(ref before=%u)\n", This, refCount - 1);
76 DSWAVE_LockModule();
78 return refCount;
81 static ULONG WINAPI IDirectMusicWaveImpl_IUnknown_Release (LPUNKNOWN iface) {
82 ICOM_THIS_MULTI(IDirectMusicWaveImpl, UnknownVtbl, iface);
83 ULONG refCount = InterlockedDecrement(&This->ref);
85 TRACE("(%p)->(ref before=%u)\n", This, refCount + 1);
87 if (!refCount) {
88 HeapFree(GetProcessHeap(), 0, This);
91 DSWAVE_UnlockModule();
93 return refCount;
96 static const IUnknownVtbl DirectMusicWave_Unknown_Vtbl = {
97 IDirectMusicWaveImpl_IUnknown_QueryInterface,
98 IDirectMusicWaveImpl_IUnknown_AddRef,
99 IDirectMusicWaveImpl_IUnknown_Release
102 /* IDirectMusicSegment8Impl IDirectMusicSegment part: */
103 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_QueryInterface (LPDIRECTMUSICSEGMENT8 iface, REFIID riid, LPVOID *ppobj) {
104 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
105 return IDirectMusicWaveImpl_IUnknown_QueryInterface ((LPUNKNOWN)&This->UnknownVtbl, riid, ppobj);
108 static ULONG WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_AddRef (LPDIRECTMUSICSEGMENT8 iface) {
109 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
110 return IDirectMusicWaveImpl_IUnknown_AddRef ((LPUNKNOWN)&This->UnknownVtbl);
113 static ULONG WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_Release (LPDIRECTMUSICSEGMENT8 iface) {
114 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
115 return IDirectMusicWaveImpl_IUnknown_Release ((LPUNKNOWN)&This->UnknownVtbl);
118 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_GetLength (LPDIRECTMUSICSEGMENT8 iface, MUSIC_TIME* pmtLength) {
119 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
120 FIXME("(%p, %p): stub\n", This, pmtLength);
121 return S_OK;
124 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_SetLength (LPDIRECTMUSICSEGMENT8 iface, MUSIC_TIME mtLength) {
125 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
126 FIXME("(%p, %d): stub\n", This, mtLength);
127 return S_OK;
130 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_GetRepeats (LPDIRECTMUSICSEGMENT8 iface, DWORD* pdwRepeats) {
131 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
132 FIXME("(%p, %p): stub\n", This, pdwRepeats);
133 return S_OK;
136 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_SetRepeats (LPDIRECTMUSICSEGMENT8 iface, DWORD dwRepeats) {
137 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
138 FIXME("(%p, %d): stub\n", This, dwRepeats);
139 return S_OK;
142 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_GetDefaultResolution (LPDIRECTMUSICSEGMENT8 iface, DWORD* pdwResolution) {
143 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
144 FIXME("(%p, %p): stub\n", This, pdwResolution);
145 return S_OK;
148 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_SetDefaultResolution (LPDIRECTMUSICSEGMENT8 iface, DWORD dwResolution) {
149 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
150 FIXME("(%p, %d): stub\n", This, dwResolution);
151 return S_OK;
154 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_GetTrack (LPDIRECTMUSICSEGMENT8 iface, REFGUID rguidType, DWORD dwGroupBits, DWORD dwIndex, IDirectMusicTrack** ppTrack) {
155 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
156 FIXME("(%p, %s, %d, %d, %p): stub\n", This, debugstr_dmguid(rguidType), dwGroupBits, dwIndex, ppTrack);
157 return S_OK;
160 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_GetTrackGroup (LPDIRECTMUSICSEGMENT8 iface, IDirectMusicTrack* pTrack, DWORD* pdwGroupBits) {
161 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
162 FIXME("(%p, %p, %p): stub\n", This, pTrack, pdwGroupBits);
163 return S_OK;
166 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_InsertTrack (LPDIRECTMUSICSEGMENT8 iface, IDirectMusicTrack* pTrack, DWORD dwGroupBits) {
167 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
168 FIXME("(%p, %p, %d): stub\n", This, pTrack, dwGroupBits);
169 return S_OK;
172 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_RemoveTrack (LPDIRECTMUSICSEGMENT8 iface, IDirectMusicTrack* pTrack) {
173 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
174 FIXME("(%p, %p): stub\n", This, pTrack);
175 return S_OK;
178 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_InitPlay (LPDIRECTMUSICSEGMENT8 iface, IDirectMusicSegmentState** ppSegState, IDirectMusicPerformance* pPerformance, DWORD dwFlags) {
179 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
180 FIXME("(%p, %p, %p, %d): stub\n", This, ppSegState, pPerformance, dwFlags);
181 return S_OK;
184 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_GetGraph (LPDIRECTMUSICSEGMENT8 iface, IDirectMusicGraph** ppGraph) {
185 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
186 FIXME("(%p, %p): stub\n", This, ppGraph);
187 return S_OK;
190 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_SetGraph (LPDIRECTMUSICSEGMENT8 iface, IDirectMusicGraph* pGraph) {
191 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
192 FIXME("(%p, %p): stub\n", This, pGraph);
193 return S_OK;
196 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_AddNotificationType (LPDIRECTMUSICSEGMENT8 iface, REFGUID rguidNotificationType) {
197 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
198 FIXME("(%p, %s): stub\n", This, debugstr_dmguid(rguidNotificationType));
199 return S_OK;
202 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_RemoveNotificationType (LPDIRECTMUSICSEGMENT8 iface, REFGUID rguidNotificationType) {
203 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
204 FIXME("(%p, %s): stub\n", This, debugstr_dmguid(rguidNotificationType));
205 return S_OK;
208 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_GetParam (LPDIRECTMUSICSEGMENT8 iface, REFGUID rguidType, DWORD dwGroupBits, DWORD dwIndex, MUSIC_TIME mtTime, MUSIC_TIME* pmtNext, void* pParam) {
209 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
210 FIXME("(%p, %s, %d, %d, %d, %p, %p): stub\n", This, debugstr_dmguid(rguidType), dwGroupBits, dwIndex, mtTime, pmtNext, pParam);
211 return S_OK;
214 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_SetParam (LPDIRECTMUSICSEGMENT8 iface, REFGUID rguidType, DWORD dwGroupBits, DWORD dwIndex, MUSIC_TIME mtTime, void* pParam) {
215 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
216 FIXME("(%p, %s, %d, %d, %d, %p): stub\n", This, debugstr_dmguid(rguidType), dwGroupBits, dwIndex, mtTime, pParam);
217 return S_OK;
220 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_Clone (LPDIRECTMUSICSEGMENT8 iface, MUSIC_TIME mtStart, MUSIC_TIME mtEnd, IDirectMusicSegment** ppSegment) {
221 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
222 FIXME("(%p, %d, %d, %p): stub\n", This, mtStart, mtEnd, ppSegment);
223 return S_OK;
226 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_SetStartPoint (LPDIRECTMUSICSEGMENT8 iface, MUSIC_TIME mtStart) {
227 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
228 FIXME("(%p, %d): stub\n", This, mtStart);
229 return S_OK;
232 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_GetStartPoint (LPDIRECTMUSICSEGMENT8 iface, MUSIC_TIME* pmtStart) {
233 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
234 FIXME("(%p, %p): stub\n", This, pmtStart);
235 return S_OK;
238 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_SetLoopPoints (LPDIRECTMUSICSEGMENT8 iface, MUSIC_TIME mtStart, MUSIC_TIME mtEnd) {
239 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
240 FIXME("(%p, %d, %d): stub\n", This, mtStart, mtEnd);
241 return S_OK;
244 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_GetLoopPoints (LPDIRECTMUSICSEGMENT8 iface, MUSIC_TIME* pmtStart, MUSIC_TIME* pmtEnd) {
245 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
246 FIXME("(%p, %p, %p): stub\n", This, pmtStart, pmtEnd);
247 return S_OK;
250 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_SetPChannelsUsed (LPDIRECTMUSICSEGMENT8 iface, DWORD dwNumPChannels, DWORD* paPChannels) {
251 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
252 FIXME("(%p, %d, %p): stub\n", This, dwNumPChannels, paPChannels);
253 return S_OK;
256 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_SetTrackConfig (LPDIRECTMUSICSEGMENT8 iface, REFGUID rguidTrackClassID, DWORD dwGroupBits, DWORD dwIndex, DWORD dwFlagsOn, DWORD dwFlagsOff) {
257 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
258 FIXME("(%p, %s, %d, %d, %d, %d): stub\n", This, debugstr_dmguid(rguidTrackClassID), dwGroupBits, dwIndex, dwFlagsOn, dwFlagsOff);
259 return S_OK;
262 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_GetAudioPathConfig (LPDIRECTMUSICSEGMENT8 iface, IUnknown** ppAudioPathConfig){
263 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
264 FIXME("(%p, %p): stub\n", This, ppAudioPathConfig);
265 return S_OK;
268 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_Compose (LPDIRECTMUSICSEGMENT8 iface, MUSIC_TIME mtTime, IDirectMusicSegment* pFromSegment, IDirectMusicSegment* pToSegment, IDirectMusicSegment** ppComposedSegment) {
269 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
270 FIXME("(%p, %d, %p, %p, %p): stub\n", This, mtTime, pFromSegment, pToSegment, ppComposedSegment);
271 return S_OK;
274 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_Download (LPDIRECTMUSICSEGMENT8 iface, IUnknown *pAudioPath) {
275 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
276 FIXME("(%p, %p): stub\n", This, pAudioPath);
277 return S_OK;
280 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicSegment8_Unload (LPDIRECTMUSICSEGMENT8 iface, IUnknown *pAudioPath) {
281 ICOM_THIS_MULTI(IDirectMusicWaveImpl, SegmentVtbl, iface);
282 FIXME("(%p, %p): stub\n", This, pAudioPath);
283 return S_OK;
286 static const IDirectMusicSegment8Vtbl DirectMusicSegment8_Segment_Vtbl = {
287 IDirectMusicWaveImpl_IDirectMusicSegment8_QueryInterface,
288 IDirectMusicWaveImpl_IDirectMusicSegment8_AddRef,
289 IDirectMusicWaveImpl_IDirectMusicSegment8_Release,
290 IDirectMusicWaveImpl_IDirectMusicSegment8_GetLength,
291 IDirectMusicWaveImpl_IDirectMusicSegment8_SetLength,
292 IDirectMusicWaveImpl_IDirectMusicSegment8_GetRepeats,
293 IDirectMusicWaveImpl_IDirectMusicSegment8_SetRepeats,
294 IDirectMusicWaveImpl_IDirectMusicSegment8_GetDefaultResolution,
295 IDirectMusicWaveImpl_IDirectMusicSegment8_SetDefaultResolution,
296 IDirectMusicWaveImpl_IDirectMusicSegment8_GetTrack,
297 IDirectMusicWaveImpl_IDirectMusicSegment8_GetTrackGroup,
298 IDirectMusicWaveImpl_IDirectMusicSegment8_InsertTrack,
299 IDirectMusicWaveImpl_IDirectMusicSegment8_RemoveTrack,
300 IDirectMusicWaveImpl_IDirectMusicSegment8_InitPlay,
301 IDirectMusicWaveImpl_IDirectMusicSegment8_GetGraph,
302 IDirectMusicWaveImpl_IDirectMusicSegment8_SetGraph,
303 IDirectMusicWaveImpl_IDirectMusicSegment8_AddNotificationType,
304 IDirectMusicWaveImpl_IDirectMusicSegment8_RemoveNotificationType,
305 IDirectMusicWaveImpl_IDirectMusicSegment8_GetParam,
306 IDirectMusicWaveImpl_IDirectMusicSegment8_SetParam,
307 IDirectMusicWaveImpl_IDirectMusicSegment8_Clone,
308 IDirectMusicWaveImpl_IDirectMusicSegment8_SetStartPoint,
309 IDirectMusicWaveImpl_IDirectMusicSegment8_GetStartPoint,
310 IDirectMusicWaveImpl_IDirectMusicSegment8_SetLoopPoints,
311 IDirectMusicWaveImpl_IDirectMusicSegment8_GetLoopPoints,
312 IDirectMusicWaveImpl_IDirectMusicSegment8_SetPChannelsUsed,
313 IDirectMusicWaveImpl_IDirectMusicSegment8_SetTrackConfig,
314 IDirectMusicWaveImpl_IDirectMusicSegment8_GetAudioPathConfig,
315 IDirectMusicWaveImpl_IDirectMusicSegment8_Compose,
316 IDirectMusicWaveImpl_IDirectMusicSegment8_Download,
317 IDirectMusicWaveImpl_IDirectMusicSegment8_Unload
320 /* IDirectMusicWaveImpl IDirectMusicObject part: */
321 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicObject_QueryInterface (LPDIRECTMUSICOBJECT iface, REFIID riid, LPVOID *ppobj) {
322 ICOM_THIS_MULTI(IDirectMusicWaveImpl, ObjectVtbl, iface);
323 return IDirectMusicWaveImpl_IUnknown_QueryInterface ((LPUNKNOWN)&This->UnknownVtbl, riid, ppobj);
326 static ULONG WINAPI IDirectMusicWaveImpl_IDirectMusicObject_AddRef (LPDIRECTMUSICOBJECT iface) {
327 ICOM_THIS_MULTI(IDirectMusicWaveImpl, ObjectVtbl, iface);
328 return IDirectMusicWaveImpl_IUnknown_AddRef ((LPUNKNOWN)&This->UnknownVtbl);
331 static ULONG WINAPI IDirectMusicWaveImpl_IDirectMusicObject_Release (LPDIRECTMUSICOBJECT iface) {
332 ICOM_THIS_MULTI(IDirectMusicWaveImpl, ObjectVtbl, iface);
333 return IDirectMusicWaveImpl_IUnknown_Release ((LPUNKNOWN)&This->UnknownVtbl);
336 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicObject_GetDescriptor (LPDIRECTMUSICOBJECT iface, LPDMUS_OBJECTDESC pDesc) {
337 ICOM_THIS_MULTI(IDirectMusicWaveImpl, ObjectVtbl, iface);
338 TRACE("(%p, %p)\n", This, pDesc);
339 /* I think we shouldn't return pointer here since then values can be changed; it'd be a mess */
340 memcpy (pDesc, This->pDesc, This->pDesc->dwSize);
341 return S_OK;
344 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicObject_SetDescriptor (LPDIRECTMUSICOBJECT iface, LPDMUS_OBJECTDESC pDesc) {
345 ICOM_THIS_MULTI(IDirectMusicWaveImpl, ObjectVtbl, iface);
346 TRACE("(%p, %p, %s): setting descriptor:\n", This, pDesc, debugstr_DMUS_OBJECTDESC(pDesc));
348 /* According to MSDN, we should copy only given values, not whole struct */
349 if (pDesc->dwValidData & DMUS_OBJ_OBJECT)
350 This->pDesc->guidObject = pDesc->guidObject;
351 if (pDesc->dwValidData & DMUS_OBJ_CLASS)
352 This->pDesc->guidClass = pDesc->guidClass;
353 if (pDesc->dwValidData & DMUS_OBJ_NAME)
354 lstrcpynW (This->pDesc->wszName, pDesc->wszName, DMUS_MAX_NAME);
355 if (pDesc->dwValidData & DMUS_OBJ_CATEGORY)
356 lstrcpynW (This->pDesc->wszCategory, pDesc->wszCategory, DMUS_MAX_CATEGORY);
357 if (pDesc->dwValidData & DMUS_OBJ_FILENAME)
358 lstrcpynW (This->pDesc->wszFileName, pDesc->wszFileName, DMUS_MAX_FILENAME);
359 if (pDesc->dwValidData & DMUS_OBJ_VERSION)
360 This->pDesc->vVersion = pDesc->vVersion;
361 if (pDesc->dwValidData & DMUS_OBJ_DATE)
362 This->pDesc->ftDate = pDesc->ftDate;
363 if (pDesc->dwValidData & DMUS_OBJ_MEMORY) {
364 memcpy (&This->pDesc->llMemLength, &pDesc->llMemLength, sizeof (pDesc->llMemLength));
365 memcpy (This->pDesc->pbMemData, pDesc->pbMemData, sizeof (pDesc->pbMemData));
367 if (pDesc->dwValidData & DMUS_OBJ_STREAM) {
368 /* according to MSDN, we copy the stream */
369 IStream_Clone (pDesc->pStream, &This->pDesc->pStream);
372 /* add new flags */
373 This->pDesc->dwValidData |= pDesc->dwValidData;
375 return S_OK;
378 static HRESULT WINAPI IDirectMusicWaveImpl_IDirectMusicObject_ParseDescriptor (LPDIRECTMUSICOBJECT iface, LPSTREAM pStream, LPDMUS_OBJECTDESC pDesc) {
379 DMUS_PRIVATE_CHUNK Chunk;
380 DWORD StreamSize, StreamCount, ListSize[1], ListCount[1];
381 LARGE_INTEGER liMove; /* used when skipping chunks */
383 TRACE("(%p, %p)\n", pStream, pDesc);
385 /* FIXME: should this be determined from stream? */
386 pDesc->dwValidData |= DMUS_OBJ_CLASS;
387 pDesc->guidClass = CLSID_DirectMusicSegment;
389 IStream_Read (pStream, &Chunk, sizeof(FOURCC)+sizeof(DWORD), NULL);
390 TRACE_(dmfile)(": %s chunk (size = 0x%04x)", debugstr_fourcc (Chunk.fccID), Chunk.dwSize);
391 switch (Chunk.fccID) {
392 case FOURCC_RIFF: {
393 IStream_Read (pStream, &Chunk.fccID, sizeof(FOURCC), NULL);
394 TRACE_(dmfile)(": RIFF chunk of type %s", debugstr_fourcc(Chunk.fccID));
395 StreamSize = Chunk.dwSize - sizeof(FOURCC);
396 StreamCount = 0;
397 if (Chunk.fccID == mmioFOURCC('W','A','V','E')) {
398 TRACE_(dmfile)(": wave form\n");
399 do {
400 IStream_Read (pStream, &Chunk, sizeof(FOURCC)+sizeof(DWORD), NULL);
401 StreamCount += sizeof(FOURCC) + sizeof(DWORD) + Chunk.dwSize;
402 TRACE_(dmfile)(": %s chunk (size = 0x%04x)", debugstr_fourcc (Chunk.fccID), Chunk.dwSize);
403 switch (Chunk.fccID) {
404 case DMUS_FOURCC_GUID_CHUNK: {
405 TRACE_(dmfile)(": GUID chunk\n");
406 pDesc->dwValidData |= DMUS_OBJ_OBJECT;
407 IStream_Read (pStream, &pDesc->guidObject, Chunk.dwSize, NULL);
408 break;
410 case DMUS_FOURCC_VERSION_CHUNK: {
411 TRACE_(dmfile)(": version chunk\n");
412 pDesc->dwValidData |= DMUS_OBJ_VERSION;
413 IStream_Read (pStream, &pDesc->vVersion, Chunk.dwSize, NULL);
414 break;
416 case DMUS_FOURCC_CATEGORY_CHUNK: {
417 TRACE_(dmfile)(": category chunk\n");
418 pDesc->dwValidData |= DMUS_OBJ_CATEGORY;
419 IStream_Read (pStream, pDesc->wszCategory, Chunk.dwSize, NULL);
420 break;
422 case FOURCC_LIST: {
423 IStream_Read (pStream, &Chunk.fccID, sizeof(FOURCC), NULL);
424 TRACE_(dmfile)(": LIST chunk of type %s", debugstr_fourcc(Chunk.fccID));
425 ListSize[0] = Chunk.dwSize - sizeof(FOURCC);
426 ListCount[0] = 0;
427 switch (Chunk.fccID) {
428 /* evil M$ UNFO list, which can (!?) contain INFO elements */
429 case DMUS_FOURCC_UNFO_LIST: {
430 TRACE_(dmfile)(": UNFO list\n");
431 do {
432 IStream_Read (pStream, &Chunk, sizeof(FOURCC)+sizeof(DWORD), NULL);
433 ListCount[0] += sizeof(FOURCC) + sizeof(DWORD) + Chunk.dwSize;
434 TRACE_(dmfile)(": %s chunk (size = 0x%04x)", debugstr_fourcc (Chunk.fccID), Chunk.dwSize);
435 switch (Chunk.fccID) {
436 /* don't ask me why, but M$ puts INFO elements in UNFO list sometimes
437 (though strings seem to be valid unicode) */
438 case mmioFOURCC('I','N','A','M'):
439 case DMUS_FOURCC_UNAM_CHUNK: {
440 TRACE_(dmfile)(": name chunk\n");
441 pDesc->dwValidData |= DMUS_OBJ_NAME;
442 IStream_Read (pStream, pDesc->wszName, Chunk.dwSize, NULL);
443 break;
445 case mmioFOURCC('I','A','R','T'):
446 case DMUS_FOURCC_UART_CHUNK: {
447 TRACE_(dmfile)(": artist chunk (ignored)\n");
448 liMove.QuadPart = Chunk.dwSize;
449 IStream_Seek (pStream, liMove, STREAM_SEEK_CUR, NULL);
450 break;
452 case mmioFOURCC('I','C','O','P'):
453 case DMUS_FOURCC_UCOP_CHUNK: {
454 TRACE_(dmfile)(": copyright chunk (ignored)\n");
455 liMove.QuadPart = Chunk.dwSize;
456 IStream_Seek (pStream, liMove, STREAM_SEEK_CUR, NULL);
457 break;
459 case mmioFOURCC('I','S','B','J'):
460 case DMUS_FOURCC_USBJ_CHUNK: {
461 TRACE_(dmfile)(": subject chunk (ignored)\n");
462 liMove.QuadPart = Chunk.dwSize;
463 IStream_Seek (pStream, liMove, STREAM_SEEK_CUR, NULL);
464 break;
466 case mmioFOURCC('I','C','M','T'):
467 case DMUS_FOURCC_UCMT_CHUNK: {
468 TRACE_(dmfile)(": comment chunk (ignored)\n");
469 liMove.QuadPart = Chunk.dwSize;
470 IStream_Seek (pStream, liMove, STREAM_SEEK_CUR, NULL);
471 break;
473 default: {
474 TRACE_(dmfile)(": unknown chunk (irrevelant & skipping)\n");
475 liMove.QuadPart = Chunk.dwSize;
476 IStream_Seek (pStream, liMove, STREAM_SEEK_CUR, NULL);
477 break;
480 TRACE_(dmfile)(": ListCount[0] = %d < ListSize[0] = %d\n", ListCount[0], ListSize[0]);
481 } while (ListCount[0] < ListSize[0]);
482 break;
484 default: {
485 TRACE_(dmfile)(": unknown (skipping)\n");
486 liMove.QuadPart = Chunk.dwSize - sizeof(FOURCC);
487 IStream_Seek (pStream, liMove, STREAM_SEEK_CUR, NULL);
488 break;
491 break;
493 default: {
494 TRACE_(dmfile)(": unknown chunk (irrevelant & skipping)\n");
495 liMove.QuadPart = Chunk.dwSize;
496 IStream_Seek (pStream, liMove, STREAM_SEEK_CUR, NULL);
497 break;
500 TRACE_(dmfile)(": StreamCount[0] = %d < StreamSize[0] = %d\n", StreamCount, StreamSize);
501 } while (StreamCount < StreamSize);
502 } else {
503 TRACE_(dmfile)(": unexpected chunk; loading failed)\n");
504 liMove.QuadPart = StreamSize;
505 IStream_Seek (pStream, liMove, STREAM_SEEK_CUR, NULL); /* skip the rest of the chunk */
506 return E_FAIL;
509 TRACE_(dmfile)(": reading finished\n");
510 break;
512 default: {
513 TRACE_(dmfile)(": unexpected chunk; loading failed)\n");
514 liMove.QuadPart = Chunk.dwSize;
515 IStream_Seek (pStream, liMove, STREAM_SEEK_CUR, NULL); /* skip the rest of the chunk */
516 return DMUS_E_INVALIDFILE;
520 TRACE(": returning descriptor: %s\n", debugstr_DMUS_OBJECTDESC (pDesc));
522 return S_OK;
525 static const IDirectMusicObjectVtbl DirectMusicWave_Object_Vtbl = {
526 IDirectMusicWaveImpl_IDirectMusicObject_QueryInterface,
527 IDirectMusicWaveImpl_IDirectMusicObject_AddRef,
528 IDirectMusicWaveImpl_IDirectMusicObject_Release,
529 IDirectMusicWaveImpl_IDirectMusicObject_GetDescriptor,
530 IDirectMusicWaveImpl_IDirectMusicObject_SetDescriptor,
531 IDirectMusicWaveImpl_IDirectMusicObject_ParseDescriptor
534 /* IDirectMusicWaveImpl IPersistStream part: */
535 static HRESULT WINAPI IDirectMusicWaveImpl_IPersistStream_QueryInterface (LPPERSISTSTREAM iface, REFIID riid, LPVOID *ppobj) {
536 ICOM_THIS_MULTI(IDirectMusicWaveImpl, PersistStreamVtbl, iface);
537 return IDirectMusicWaveImpl_IUnknown_QueryInterface ((LPUNKNOWN)&This->UnknownVtbl, riid, ppobj);
540 static ULONG WINAPI IDirectMusicWaveImpl_IPersistStream_AddRef (LPPERSISTSTREAM iface) {
541 ICOM_THIS_MULTI(IDirectMusicWaveImpl, PersistStreamVtbl, iface);
542 return IDirectMusicWaveImpl_IUnknown_AddRef ((LPUNKNOWN)&This->UnknownVtbl);
545 static ULONG WINAPI IDirectMusicWaveImpl_IPersistStream_Release (LPPERSISTSTREAM iface) {
546 ICOM_THIS_MULTI(IDirectMusicWaveImpl, PersistStreamVtbl, iface);
547 return IDirectMusicWaveImpl_IUnknown_Release ((LPUNKNOWN)&This->UnknownVtbl);
550 static HRESULT WINAPI IDirectMusicWaveImpl_IPersistStream_GetClassID (LPPERSISTSTREAM iface, CLSID* pClassID) {
551 return E_NOTIMPL;
554 static HRESULT WINAPI IDirectMusicWaveImpl_IPersistStream_IsDirty (LPPERSISTSTREAM iface) {
555 return E_NOTIMPL;
558 static HRESULT WINAPI IDirectMusicWaveImpl_IPersistStream_Load (LPPERSISTSTREAM iface, IStream* pStm) {
559 ICOM_THIS_MULTI(IDirectMusicWaveImpl, PersistStreamVtbl, iface);
561 DMUS_PRIVATE_CHUNK Chunk;
562 DWORD StreamSize, StreamCount, ListSize[1], ListCount[1];
563 LARGE_INTEGER liMove; /* used when skipping chunks */
565 FIXME("(%p, %p): loading not implemented yet (only descriptor is loaded)\n", This, pStm);
567 /* FIXME: should this be determined from stream? */
568 This->pDesc->dwValidData |= DMUS_OBJ_CLASS;
569 This->pDesc->guidClass = CLSID_DirectMusicSegment;
571 IStream_Read (pStm, &Chunk, sizeof(FOURCC)+sizeof(DWORD), NULL);
572 TRACE_(dmfile)(": %s chunk (size = 0x%04x)", debugstr_fourcc (Chunk.fccID), Chunk.dwSize);
573 switch (Chunk.fccID) {
574 case FOURCC_RIFF: {
575 IStream_Read (pStm, &Chunk.fccID, sizeof(FOURCC), NULL);
576 TRACE_(dmfile)(": RIFF chunk of type %s", debugstr_fourcc(Chunk.fccID));
577 StreamSize = Chunk.dwSize - sizeof(FOURCC);
578 StreamCount = 0;
579 if (Chunk.fccID == mmioFOURCC('W','A','V','E')) {
580 TRACE_(dmfile)(": wave form\n");
581 do {
582 IStream_Read (pStm, &Chunk, sizeof(FOURCC)+sizeof(DWORD), NULL);
583 StreamCount += sizeof(FOURCC) + sizeof(DWORD) + Chunk.dwSize;
584 TRACE_(dmfile)(": %s chunk (size = 0x%04x)", debugstr_fourcc (Chunk.fccID), Chunk.dwSize);
585 switch (Chunk.fccID) {
586 case DMUS_FOURCC_GUID_CHUNK: {
587 TRACE_(dmfile)(": GUID chunk\n");
588 This->pDesc->dwValidData |= DMUS_OBJ_OBJECT;
589 IStream_Read (pStm, &This->pDesc->guidObject, Chunk.dwSize, NULL);
590 break;
592 case DMUS_FOURCC_VERSION_CHUNK: {
593 TRACE_(dmfile)(": version chunk\n");
594 This->pDesc->dwValidData |= DMUS_OBJ_VERSION;
595 IStream_Read (pStm, &This->pDesc->vVersion, Chunk.dwSize, NULL);
596 break;
598 case DMUS_FOURCC_CATEGORY_CHUNK: {
599 TRACE_(dmfile)(": category chunk\n");
600 This->pDesc->dwValidData |= DMUS_OBJ_CATEGORY;
601 IStream_Read (pStm, This->pDesc->wszCategory, Chunk.dwSize, NULL);
602 break;
604 case FOURCC_LIST: {
605 IStream_Read (pStm, &Chunk.fccID, sizeof(FOURCC), NULL);
606 TRACE_(dmfile)(": LIST chunk of type %s", debugstr_fourcc(Chunk.fccID));
607 ListSize[0] = Chunk.dwSize - sizeof(FOURCC);
608 ListCount[0] = 0;
609 switch (Chunk.fccID) {
610 /* evil M$ UNFO list, which can (!?) contain INFO elements */
611 case DMUS_FOURCC_UNFO_LIST: {
612 TRACE_(dmfile)(": UNFO list\n");
613 do {
614 IStream_Read (pStm, &Chunk, sizeof(FOURCC)+sizeof(DWORD), NULL);
615 ListCount[0] += sizeof(FOURCC) + sizeof(DWORD) + Chunk.dwSize;
616 TRACE_(dmfile)(": %s chunk (size = 0x%04x)", debugstr_fourcc (Chunk.fccID), Chunk.dwSize);
617 switch (Chunk.fccID) {
618 /* don't ask me why, but M$ puts INFO elements in UNFO list sometimes
619 (though strings seem to be valid unicode) */
620 case mmioFOURCC('I','N','A','M'):
621 case DMUS_FOURCC_UNAM_CHUNK: {
622 TRACE_(dmfile)(": name chunk\n");
623 This->pDesc->dwValidData |= DMUS_OBJ_NAME;
624 IStream_Read (pStm, This->pDesc->wszName, Chunk.dwSize, NULL);
625 break;
627 case mmioFOURCC('I','A','R','T'):
628 case DMUS_FOURCC_UART_CHUNK: {
629 TRACE_(dmfile)(": artist chunk (ignored)\n");
630 liMove.QuadPart = Chunk.dwSize;
631 IStream_Seek (pStm, liMove, STREAM_SEEK_CUR, NULL);
632 break;
634 case mmioFOURCC('I','C','O','P'):
635 case DMUS_FOURCC_UCOP_CHUNK: {
636 TRACE_(dmfile)(": copyright chunk (ignored)\n");
637 liMove.QuadPart = Chunk.dwSize;
638 IStream_Seek (pStm, liMove, STREAM_SEEK_CUR, NULL);
639 break;
641 case mmioFOURCC('I','S','B','J'):
642 case DMUS_FOURCC_USBJ_CHUNK: {
643 TRACE_(dmfile)(": subject chunk (ignored)\n");
644 liMove.QuadPart = Chunk.dwSize;
645 IStream_Seek (pStm, liMove, STREAM_SEEK_CUR, NULL);
646 break;
648 case mmioFOURCC('I','C','M','T'):
649 case DMUS_FOURCC_UCMT_CHUNK: {
650 TRACE_(dmfile)(": comment chunk (ignored)\n");
651 liMove.QuadPart = Chunk.dwSize;
652 IStream_Seek (pStm, liMove, STREAM_SEEK_CUR, NULL);
653 break;
655 default: {
656 TRACE_(dmfile)(": unknown chunk (irrevelant & skipping)\n");
657 liMove.QuadPart = Chunk.dwSize;
658 IStream_Seek (pStm, liMove, STREAM_SEEK_CUR, NULL);
659 break;
662 TRACE_(dmfile)(": ListCount[0] = %d < ListSize[0] = %d\n", ListCount[0], ListSize[0]);
663 } while (ListCount[0] < ListSize[0]);
664 break;
666 default: {
667 TRACE_(dmfile)(": unknown (skipping)\n");
668 liMove.QuadPart = Chunk.dwSize - sizeof(FOURCC);
669 IStream_Seek (pStm, liMove, STREAM_SEEK_CUR, NULL);
670 break;
673 break;
675 default: {
676 TRACE_(dmfile)(": unknown chunk (irrevelant & skipping)\n");
677 liMove.QuadPart = Chunk.dwSize;
678 IStream_Seek (pStm, liMove, STREAM_SEEK_CUR, NULL);
679 break;
682 TRACE_(dmfile)(": StreamCount[0] = %d < StreamSize[0] = %d\n", StreamCount, StreamSize);
683 } while (StreamCount < StreamSize);
684 } else {
685 TRACE_(dmfile)(": unexpected chunk; loading failed)\n");
686 liMove.QuadPart = StreamSize;
687 IStream_Seek (pStm, liMove, STREAM_SEEK_CUR, NULL); /* skip the rest of the chunk */
688 return E_FAIL;
691 TRACE_(dmfile)(": reading finished\n");
692 break;
694 default: {
695 TRACE_(dmfile)(": unexpected chunk; loading failed)\n");
696 liMove.QuadPart = Chunk.dwSize;
697 IStream_Seek (pStm, liMove, STREAM_SEEK_CUR, NULL); /* skip the rest of the chunk */
698 return DMUS_E_INVALIDFILE;
702 return S_OK;
705 static HRESULT WINAPI IDirectMusicWaveImpl_IPersistStream_Save (LPPERSISTSTREAM iface, IStream* pStm, BOOL fClearDirty) {
706 return E_NOTIMPL;
709 static HRESULT WINAPI IDirectMusicWaveImpl_IPersistStream_GetSizeMax (LPPERSISTSTREAM iface, ULARGE_INTEGER* pcbSize) {
710 return E_NOTIMPL;
713 static const IPersistStreamVtbl DirectMusicWave_PersistStream_Vtbl = {
714 IDirectMusicWaveImpl_IPersistStream_QueryInterface,
715 IDirectMusicWaveImpl_IPersistStream_AddRef,
716 IDirectMusicWaveImpl_IPersistStream_Release,
717 IDirectMusicWaveImpl_IPersistStream_GetClassID,
718 IDirectMusicWaveImpl_IPersistStream_IsDirty,
719 IDirectMusicWaveImpl_IPersistStream_Load,
720 IDirectMusicWaveImpl_IPersistStream_Save,
721 IDirectMusicWaveImpl_IPersistStream_GetSizeMax
725 /* for ClassFactory */
726 HRESULT WINAPI DMUSIC_CreateDirectMusicWaveImpl (LPCGUID lpcGUID, LPVOID* ppobj, LPUNKNOWN pUnkOuter) {
727 IDirectMusicWaveImpl* obj;
729 obj = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IDirectMusicWaveImpl));
730 if (NULL == obj) {
731 *ppobj = NULL;
732 return E_OUTOFMEMORY;
734 obj->UnknownVtbl = &DirectMusicWave_Unknown_Vtbl;
735 obj->SegmentVtbl = &DirectMusicSegment8_Segment_Vtbl;
736 obj->ObjectVtbl = &DirectMusicWave_Object_Vtbl;
737 obj->PersistStreamVtbl = &DirectMusicWave_PersistStream_Vtbl;
738 obj->pDesc = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(DMUS_OBJECTDESC));
739 DM_STRUCT_INIT(obj->pDesc);
740 obj->pDesc->dwValidData |= DMUS_OBJ_CLASS;
741 obj->pDesc->guidClass = CLSID_DirectMusicSegment; /* shown by tests */
742 obj->ref = 0; /* will be inited by QueryInterface */
744 return IDirectMusicWaveImpl_IUnknown_QueryInterface ((LPUNKNOWN)&obj->UnknownVtbl, lpcGUID, ppobj);