xref: /reactos/dll/directx/wine/dmusic/dmusic_main.c (revision 8a978a17)
1 /* DirectMusic Main
2  *
3  * Copyright (C) 2003-2004 Rok Mandeljc
4  *
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.
9  *
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.
14  *
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
18  */
19 
20 
21 #include <stdio.h>
22 #include <stdarg.h>
23 
24 #define COBJMACROS
25 
26 #include "windef.h"
27 #include "winbase.h"
28 #include "winnt.h"
29 #include "wingdi.h"
30 #include "winuser.h"
31 #include "winreg.h"
32 #include "objbase.h"
33 #include "rpcproxy.h"
34 #include "initguid.h"
35 #include "dmusici.h"
36 
37 #include "dmusic_private.h"
38 
39 WINE_DEFAULT_DEBUG_CHANNEL(dmusic);
40 
41 static HINSTANCE instance;
42 LONG DMUSIC_refCount = 0;
43 
44 typedef struct {
45         IClassFactory IClassFactory_iface;
46         HRESULT (WINAPI *fnCreateInstance)(REFIID riid, void **ppv, IUnknown *pUnkOuter);
47 } IClassFactoryImpl;
48 
49 /******************************************************************
50  *      IClassFactory implementation
51  */
52 static inline IClassFactoryImpl *impl_from_IClassFactory(IClassFactory *iface)
53 {
54         return CONTAINING_RECORD(iface, IClassFactoryImpl, IClassFactory_iface);
55 }
56 
57 static HRESULT WINAPI ClassFactory_QueryInterface(IClassFactory *iface, REFIID riid, void **ppv)
58 {
59         if (ppv == NULL)
60                 return E_POINTER;
61 
62         if (IsEqualGUID(&IID_IUnknown, riid))
63                 TRACE("(%p)->(IID_IUnknown %p)\n", iface, ppv);
64         else if (IsEqualGUID(&IID_IClassFactory, riid))
65                 TRACE("(%p)->(IID_IClassFactory %p)\n", iface, ppv);
66         else {
67                 FIXME("(%p)->(%s %p)\n", iface, debugstr_guid(riid), ppv);
68                 *ppv = NULL;
69                 return E_NOINTERFACE;
70         }
71 
72         *ppv = iface;
73         IUnknown_AddRef((IUnknown*)*ppv);
74         return S_OK;
75 }
76 
77 static ULONG WINAPI ClassFactory_AddRef(IClassFactory *iface)
78 {
79         DMUSIC_LockModule();
80 
81         return 2; /* non-heap based object */
82 }
83 
84 static ULONG WINAPI ClassFactory_Release(IClassFactory *iface)
85 {
86         DMUSIC_UnlockModule();
87 
88         return 1; /* non-heap based object */
89 }
90 
91 static HRESULT WINAPI ClassFactory_CreateInstance(IClassFactory *iface, IUnknown *pUnkOuter,
92         REFIID riid, void **ppv)
93 {
94         IClassFactoryImpl *This = impl_from_IClassFactory(iface);
95 
96         TRACE ("(%p, %s, %p)\n", pUnkOuter, debugstr_dmguid(riid), ppv);
97 
98         return This->fnCreateInstance(riid, ppv, pUnkOuter);
99 }
100 
101 static HRESULT WINAPI ClassFactory_LockServer(IClassFactory *iface, BOOL dolock)
102 {
103         TRACE("(%d)\n", dolock);
104 
105         if (dolock)
106                 DMUSIC_LockModule();
107         else
108                 DMUSIC_UnlockModule();
109 
110         return S_OK;
111 }
112 
113 static const IClassFactoryVtbl classfactory_vtbl = {
114         ClassFactory_QueryInterface,
115         ClassFactory_AddRef,
116         ClassFactory_Release,
117         ClassFactory_CreateInstance,
118         ClassFactory_LockServer
119 };
120 
121 static IClassFactoryImpl DirectMusic_CF = {{&classfactory_vtbl}, DMUSIC_CreateDirectMusicImpl};
122 static IClassFactoryImpl Collection_CF = {{&classfactory_vtbl},
123                                           DMUSIC_CreateDirectMusicCollectionImpl};
124 
125 /******************************************************************
126  *		DllMain
127  *
128  *
129  */
130 BOOL WINAPI DllMain(HINSTANCE hinstDLL, DWORD fdwReason, LPVOID lpvReserved) {
131 	if (fdwReason == DLL_PROCESS_ATTACH) {
132             instance = hinstDLL;
133             DisableThreadLibraryCalls(hinstDLL);
134 	}
135 
136 	return TRUE;
137 }
138 
139 
140 /******************************************************************
141  *		DllCanUnloadNow (DMUSIC.@)
142  *
143  *
144  */
145 HRESULT WINAPI DllCanUnloadNow(void)
146 {
147 	return DMUSIC_refCount != 0 ? S_FALSE : S_OK;
148 }
149 
150 
151 /******************************************************************
152  *		DllGetClassObject (DMUSIC.@)
153  *
154  *
155  */
156 HRESULT WINAPI DllGetClassObject(REFCLSID rclsid, REFIID riid, LPVOID *ppv)
157 {
158 	TRACE("(%s, %s, %p)\n", debugstr_dmguid(rclsid), debugstr_dmguid(riid), ppv);
159 	if (IsEqualCLSID (rclsid, &CLSID_DirectMusic) && IsEqualIID (riid, &IID_IClassFactory)) {
160 		*ppv = &DirectMusic_CF;
161 		IClassFactory_AddRef((IClassFactory*)*ppv);
162 		return S_OK;
163 	} else if (IsEqualCLSID (rclsid, &CLSID_DirectMusicCollection) && IsEqualIID (riid, &IID_IClassFactory)) {
164 		*ppv = &Collection_CF;
165 		IClassFactory_AddRef((IClassFactory*)*ppv);
166 		return S_OK;
167 	}
168 
169     WARN("(%s, %s, %p): no interface found.\n", debugstr_dmguid(rclsid), debugstr_dmguid(riid), ppv);
170     return CLASS_E_CLASSNOTAVAILABLE;
171 }
172 
173 /***********************************************************************
174  *		DllRegisterServer (DMUSIC.@)
175  */
176 HRESULT WINAPI DllRegisterServer(void)
177 {
178     return __wine_register_resources( instance );
179 }
180 
181 /***********************************************************************
182  *		DllUnregisterServer (DMUSIC.@)
183  */
184 HRESULT WINAPI DllUnregisterServer(void)
185 {
186     return __wine_unregister_resources( instance );
187 }
188 
189 /******************************************************************
190  *		Helper functions
191  *
192  *
193  */
194 /* dwPatch from MIDILOCALE */
195 DWORD MIDILOCALE2Patch (const MIDILOCALE *pLocale) {
196 	DWORD dwPatch = 0;
197 	if (!pLocale) return 0;
198 	dwPatch |= (pLocale->ulBank & F_INSTRUMENT_DRUMS); /* set drum bit */
199 	dwPatch |= ((pLocale->ulBank & 0x00007F7F) << 8); /* set MIDI bank location */
200 	dwPatch |= (pLocale->ulInstrument & 0x0000007F); /* set PC value */
201 	return dwPatch;
202 }
203 
204 /* MIDILOCALE from dwPatch */
205 void Patch2MIDILOCALE (DWORD dwPatch, LPMIDILOCALE pLocale) {
206 	memset (pLocale, 0, sizeof(MIDILOCALE));
207 
208 	pLocale->ulInstrument = (dwPatch & 0x7F); /* get PC value */
209 	pLocale->ulBank = ((dwPatch & 0x007F7F00) >> 8); /* get MIDI bank location */
210 	pLocale->ulBank |= (dwPatch & F_INSTRUMENT_DRUMS); /* get drum bit */
211 }
212 
213 /* check whether the given DWORD is even (return 0) or odd (return 1) */
214 int even_or_odd (DWORD number) {
215 	return (number & 0x1); /* basically, check if bit 0 is set ;) */
216 }
217 
218 /* FOURCC to string conversion for debug messages */
219 const char *debugstr_fourcc (DWORD fourcc) {
220     if (!fourcc) return "'null'";
221     return wine_dbg_sprintf ("\'%c%c%c%c\'",
222 		(char)(fourcc), (char)(fourcc >> 8),
223         (char)(fourcc >> 16), (char)(fourcc >> 24));
224 }
225 
226 /* DMUS_VERSION struct to string conversion for debug messages */
227 static const char *debugstr_dmversion(const DMUS_VERSION *version)
228 {
229     if (!version)
230         return "'null'";
231     return wine_dbg_sprintf("'%hu,%hu,%hu,%hu'",
232             HIWORD(version->dwVersionMS), LOWORD(version->dwVersionMS),
233             HIWORD(version->dwVersionLS), LOWORD(version->dwVersionLS));
234 }
235 
236 /* returns name of given GUID */
237 const char *debugstr_dmguid (const GUID *id) {
238 	static const guid_info guids[] = {
239 		/* CLSIDs */
240 		GE(CLSID_AudioVBScript),
241 		GE(CLSID_DirectMusic),
242 		GE(CLSID_DirectMusicAudioPathConfig),
243 		GE(CLSID_DirectMusicAuditionTrack),
244 		GE(CLSID_DirectMusicBand),
245 		GE(CLSID_DirectMusicBandTrack),
246 		GE(CLSID_DirectMusicChordMapTrack),
247 		GE(CLSID_DirectMusicChordMap),
248 		GE(CLSID_DirectMusicChordTrack),
249 		GE(CLSID_DirectMusicCollection),
250 		GE(CLSID_DirectMusicCommandTrack),
251 		GE(CLSID_DirectMusicComposer),
252 		GE(CLSID_DirectMusicContainer),
253 		GE(CLSID_DirectMusicGraph),
254 		GE(CLSID_DirectMusicLoader),
255 		GE(CLSID_DirectMusicLyricsTrack),
256 		GE(CLSID_DirectMusicMarkerTrack),
257 		GE(CLSID_DirectMusicMelodyFormulationTrack),
258 		GE(CLSID_DirectMusicMotifTrack),
259 		GE(CLSID_DirectMusicMuteTrack),
260 		GE(CLSID_DirectMusicParamControlTrack),
261 		GE(CLSID_DirectMusicPatternTrack),
262 		GE(CLSID_DirectMusicPerformance),
263 		GE(CLSID_DirectMusicScript),
264 		GE(CLSID_DirectMusicScriptAutoImpSegment),
265 		GE(CLSID_DirectMusicScriptAutoImpPerformance),
266 		GE(CLSID_DirectMusicScriptAutoImpSegmentState),
267 		GE(CLSID_DirectMusicScriptAutoImpAudioPathConfig),
268 		GE(CLSID_DirectMusicScriptAutoImpAudioPath),
269 		GE(CLSID_DirectMusicScriptAutoImpSong),
270 		GE(CLSID_DirectMusicScriptSourceCodeLoader),
271 		GE(CLSID_DirectMusicScriptTrack),
272 		GE(CLSID_DirectMusicSection),
273 		GE(CLSID_DirectMusicSegment),
274 		GE(CLSID_DirectMusicSegmentState),
275 		GE(CLSID_DirectMusicSegmentTriggerTrack),
276 		GE(CLSID_DirectMusicSegTriggerTrack),
277 		GE(CLSID_DirectMusicSeqTrack),
278 		GE(CLSID_DirectMusicSignPostTrack),
279 		GE(CLSID_DirectMusicSong),
280 		GE(CLSID_DirectMusicStyle),
281 		GE(CLSID_DirectMusicStyleTrack),
282 		GE(CLSID_DirectMusicSynth),
283 		GE(CLSID_DirectMusicSynthSink),
284 		GE(CLSID_DirectMusicSysExTrack),
285 		GE(CLSID_DirectMusicTemplate),
286 		GE(CLSID_DirectMusicTempoTrack),
287 		GE(CLSID_DirectMusicTimeSigTrack),
288 		GE(CLSID_DirectMusicWaveTrack),
289 		GE(CLSID_DirectSoundWave),
290 		/* IIDs */
291 		GE(IID_IDirectMusic),
292 		GE(IID_IDirectMusic2),
293 		GE(IID_IDirectMusic8),
294 		GE(IID_IDirectMusicAudioPath),
295 		GE(IID_IDirectMusicBand),
296 		GE(IID_IDirectMusicBuffer),
297 		GE(IID_IDirectMusicChordMap),
298 		GE(IID_IDirectMusicCollection),
299 		GE(IID_IDirectMusicComposer),
300 		GE(IID_IDirectMusicContainer),
301 		GE(IID_IDirectMusicDownload),
302 		GE(IID_IDirectMusicDownloadedInstrument),
303 		GE(IID_IDirectMusicGetLoader),
304 		GE(IID_IDirectMusicGraph),
305 		GE(IID_IDirectMusicInstrument),
306 		GE(IID_IDirectMusicLoader),
307 		GE(IID_IDirectMusicLoader8),
308 		GE(IID_IDirectMusicObject),
309 		GE(IID_IDirectMusicPatternTrack),
310 		GE(IID_IDirectMusicPerformance),
311 		GE(IID_IDirectMusicPerformance2),
312 		GE(IID_IDirectMusicPerformance8),
313 		GE(IID_IDirectMusicPort),
314 		GE(IID_IDirectMusicPortDownload),
315 		GE(IID_IDirectMusicScript),
316 		GE(IID_IDirectMusicSegment),
317 		GE(IID_IDirectMusicSegment2),
318 		GE(IID_IDirectMusicSegment8),
319 		GE(IID_IDirectMusicSegmentState),
320 		GE(IID_IDirectMusicSegmentState8),
321 		GE(IID_IDirectMusicStyle),
322 		GE(IID_IDirectMusicStyle8),
323 		GE(IID_IDirectMusicSynth),
324 		GE(IID_IDirectMusicSynth8),
325 		GE(IID_IDirectMusicSynthSink),
326 		GE(IID_IDirectMusicThru),
327 		GE(IID_IDirectMusicTool),
328 		GE(IID_IDirectMusicTool8),
329 		GE(IID_IDirectMusicTrack),
330 		GE(IID_IDirectMusicTrack8),
331 		GE(IID_IUnknown),
332 		GE(IID_IPersistStream),
333 		GE(IID_IStream),
334 		GE(IID_IClassFactory),
335 		/* GUIDs */
336 		GE(GUID_DirectMusicAllTypes),
337 		GE(GUID_NOTIFICATION_CHORD),
338 		GE(GUID_NOTIFICATION_COMMAND),
339 		GE(GUID_NOTIFICATION_MEASUREANDBEAT),
340 		GE(GUID_NOTIFICATION_PERFORMANCE),
341 		GE(GUID_NOTIFICATION_RECOMPOSE),
342 		GE(GUID_NOTIFICATION_SEGMENT),
343 		GE(GUID_BandParam),
344 		GE(GUID_ChordParam),
345 		GE(GUID_CommandParam),
346 		GE(GUID_CommandParam2),
347 		GE(GUID_CommandParamNext),
348 		GE(GUID_IDirectMusicBand),
349 		GE(GUID_IDirectMusicChordMap),
350 		GE(GUID_IDirectMusicStyle),
351 		GE(GUID_MuteParam),
352 		GE(GUID_Play_Marker),
353 		GE(GUID_RhythmParam),
354 		GE(GUID_TempoParam),
355 		GE(GUID_TimeSignature),
356 		GE(GUID_Valid_Start_Time),
357 		GE(GUID_Clear_All_Bands),
358 		GE(GUID_ConnectToDLSCollection),
359 		GE(GUID_Disable_Auto_Download),
360 		GE(GUID_DisableTempo),
361 		GE(GUID_DisableTimeSig),
362 		GE(GUID_Download),
363 		GE(GUID_DownloadToAudioPath),
364 		GE(GUID_Enable_Auto_Download),
365 		GE(GUID_EnableTempo),
366 		GE(GUID_EnableTimeSig),
367 		GE(GUID_IgnoreBankSelectForGM),
368 		GE(GUID_SeedVariations),
369 		GE(GUID_StandardMIDIFile),
370 		GE(GUID_Unload),
371 		GE(GUID_UnloadFromAudioPath),
372 		GE(GUID_Variations),
373 		GE(GUID_PerfMasterTempo),
374 		GE(GUID_PerfMasterVolume),
375 		GE(GUID_PerfMasterGrooveLevel),
376 		GE(GUID_PerfAutoDownload),
377 		GE(GUID_DefaultGMCollection),
378 		GE(GUID_Synth_Default),
379 		GE(GUID_Buffer_Reverb),
380 		GE(GUID_Buffer_EnvReverb),
381 		GE(GUID_Buffer_Stereo),
382 		GE(GUID_Buffer_3D_Dry),
383 		GE(GUID_Buffer_Mono),
384 		GE(GUID_DMUS_PROP_GM_Hardware),
385 		GE(GUID_DMUS_PROP_GS_Capable),
386 		GE(GUID_DMUS_PROP_GS_Hardware),
387 		GE(GUID_DMUS_PROP_DLS1),
388 		GE(GUID_DMUS_PROP_DLS2),
389 		GE(GUID_DMUS_PROP_Effects),
390 		GE(GUID_DMUS_PROP_INSTRUMENT2),
391 		GE(GUID_DMUS_PROP_LegacyCaps),
392 		GE(GUID_DMUS_PROP_MemorySize),
393 		GE(GUID_DMUS_PROP_SampleMemorySize),
394 		GE(GUID_DMUS_PROP_SamplePlaybackRate),
395 		GE(GUID_DMUS_PROP_SetSynthSink),
396 		GE(GUID_DMUS_PROP_SinkUsesDSound),
397 		GE(GUID_DMUS_PROP_SynthSink_DSOUND),
398 		GE(GUID_DMUS_PROP_SynthSink_WAVE),
399 		GE(GUID_DMUS_PROP_Volume),
400 		GE(GUID_DMUS_PROP_WavesReverb),
401 		GE(GUID_DMUS_PROP_WriteLatency),
402 		GE(GUID_DMUS_PROP_WritePeriod),
403 		GE(GUID_DMUS_PROP_XG_Capable),
404 		GE(GUID_DMUS_PROP_XG_Hardware)
405 	};
406 
407 	unsigned int i;
408 
409         if (!id) return "(null)";
410 
411 	for (i = 0; i < ARRAY_SIZE(guids); i++) {
412 		if (IsEqualGUID(id, guids[i].guid))
413 			return guids[i].name;
414 	}
415 	/* if we didn't find it, act like standard debugstr_guid */
416 	return debugstr_guid(id);
417 }
418 
419 /* generic flag-dumping function */
420 static const char* debugstr_flags (DWORD flags, const flag_info* names, size_t num_names){
421 	char buffer[128] = "", *ptr = &buffer[0];
422 	unsigned int i;
423 	int size = sizeof(buffer);
424 
425 	for (i=0; i < num_names; i++)
426 	{
427 		if ((flags & names[i].val) ||	/* standard flag*/
428 			((!flags) && (!names[i].val))) { /* zero value only */
429 				int cnt = snprintf(ptr, size, "%s ", names[i].name);
430 				if (cnt < 0 || cnt >= size) break;
431 				size -= cnt;
432 				ptr += cnt;
433 		}
434 	}
435 
436 	return wine_dbg_sprintf("%s", buffer);
437 }
438 
439 /* dump DMUS_OBJ flags */
440 static const char *debugstr_DMUS_OBJ_FLAGS (DWORD flagmask) {
441     static const flag_info flags[] = {
442 	    FE(DMUS_OBJ_OBJECT),
443 	    FE(DMUS_OBJ_CLASS),
444 	    FE(DMUS_OBJ_NAME),
445 	    FE(DMUS_OBJ_CATEGORY),
446 	    FE(DMUS_OBJ_FILENAME),
447 	    FE(DMUS_OBJ_FULLPATH),
448 	    FE(DMUS_OBJ_URL),
449 	    FE(DMUS_OBJ_VERSION),
450 	    FE(DMUS_OBJ_DATE),
451 	    FE(DMUS_OBJ_LOADED),
452 	    FE(DMUS_OBJ_MEMORY),
453 	    FE(DMUS_OBJ_STREAM)
454 	};
455     return debugstr_flags(flagmask, flags, ARRAY_SIZE(flags));
456 }
457 
458 /* Dump whole DMUS_OBJECTDESC struct */
459 void dump_DMUS_OBJECTDESC(LPDMUS_OBJECTDESC desc)
460 {
461     TRACE("DMUS_OBJECTDESC (%p):\n", desc);
462     TRACE(" - dwSize = %d\n", desc->dwSize);
463     TRACE(" - dwValidData = %s\n", debugstr_DMUS_OBJ_FLAGS (desc->dwValidData));
464     if (desc->dwValidData & DMUS_OBJ_CLASS)    TRACE(" - guidClass = %s\n", debugstr_dmguid(&desc->guidClass));
465     if (desc->dwValidData & DMUS_OBJ_OBJECT)   TRACE(" - guidObject = %s\n", debugstr_guid(&desc->guidObject));
466     if (desc->dwValidData & DMUS_OBJ_DATE)     TRACE(" - ftDate = FIXME\n");
467     if (desc->dwValidData & DMUS_OBJ_VERSION)  TRACE(" - vVersion = %s\n", debugstr_dmversion(&desc->vVersion));
468     if (desc->dwValidData & DMUS_OBJ_NAME)     TRACE(" - wszName = %s\n", debugstr_w(desc->wszName));
469     if (desc->dwValidData & DMUS_OBJ_CATEGORY) TRACE(" - wszCategory = %s\n", debugstr_w(desc->wszCategory));
470     if (desc->dwValidData & DMUS_OBJ_FILENAME) TRACE(" - wszFileName = %s\n", debugstr_w(desc->wszFileName));
471     if (desc->dwValidData & DMUS_OBJ_MEMORY)   TRACE(" - llMemLength = 0x%s\n  - pbMemData = %p\n",
472                                                      wine_dbgstr_longlong(desc->llMemLength), desc->pbMemData);
473     if (desc->dwValidData & DMUS_OBJ_STREAM)   TRACE(" - pStream = %p\n", desc->pStream);
474 }
475 
476 /* Dump DMUS_PORTPARAMS flags */
477 static const char* debugstr_DMUS_PORTPARAMS_FLAGS(DWORD flagmask)
478 {
479     static const flag_info flags[] = {
480         FE(DMUS_PORTPARAMS_VOICES),
481         FE(DMUS_PORTPARAMS_CHANNELGROUPS),
482         FE(DMUS_PORTPARAMS_AUDIOCHANNELS),
483         FE(DMUS_PORTPARAMS_SAMPLERATE),
484         FE(DMUS_PORTPARAMS_EFFECTS),
485         FE(DMUS_PORTPARAMS_SHARE)
486     };
487     return debugstr_flags(flagmask, flags, ARRAY_SIZE(flags));
488 }
489 
490 /* Dump whole DMUS_PORTPARAMS struct */
491 void dump_DMUS_PORTPARAMS(LPDMUS_PORTPARAMS params)
492 {
493     TRACE("DMUS_PORTPARAMS (%p):\n", params);
494     TRACE(" - dwSize = %d\n", params->dwSize);
495     TRACE(" - dwValidParams = %s\n", debugstr_DMUS_PORTPARAMS_FLAGS(params->dwValidParams));
496     if (params->dwValidParams & DMUS_PORTPARAMS_VOICES)        TRACE(" - dwVoices = %u\n", params->dwVoices);
497     if (params->dwValidParams & DMUS_PORTPARAMS_CHANNELGROUPS) TRACE(" - dwChannelGroup = %u\n", params->dwChannelGroups);
498     if (params->dwValidParams & DMUS_PORTPARAMS_AUDIOCHANNELS) TRACE(" - dwAudioChannels = %u\n", params->dwAudioChannels);
499     if (params->dwValidParams & DMUS_PORTPARAMS_SAMPLERATE)    TRACE(" - dwSampleRate = %u\n", params->dwSampleRate);
500     if (params->dwValidParams & DMUS_PORTPARAMS_EFFECTS)       TRACE(" - dwEffectFlags = %x\n", params->dwEffectFlags);
501     if (params->dwValidParams & DMUS_PORTPARAMS_SHARE)         TRACE(" - fShare = %u\n", params->fShare);
502 }
503