VxDCall functions do not need to be 'register'.
[wine/multimedia.git] / ole / oleobj.c
blob87519e508cfc5e2d8a0cbdf2f596bc5acc839de1
1 /*
2 * OLE2 COM objects
4 * Copyright 1998 Eric Kohl
5 * Copyright 1999 Francis Beaudet
6 */
9 #include <string.h>
10 #include "winbase.h"
11 #include "winerror.h"
12 #include "oleidl.h"
13 #include "debug.h"
15 DEFAULT_DEBUG_CHANNEL(ole)
18 /**************************************************************************
19 * OleAdviseHolderImpl Implementation
21 typedef struct OleAdviseHolderImpl
23 ICOM_VTABLE(IOleAdviseHolder)* lpvtbl;
25 DWORD ref;
27 DWORD maxSinks;
28 IAdviseSink** arrayOfSinks;
30 } OleAdviseHolderImpl;
32 static LPOLEADVISEHOLDER OleAdviseHolderImpl_Constructor();
33 static void OleAdviseHolderImpl_Destructor(OleAdviseHolderImpl* ptrToDestroy);
34 static HRESULT WINAPI OleAdviseHolderImpl_QueryInterface(LPOLEADVISEHOLDER,REFIID,LPVOID*);
35 static ULONG WINAPI OleAdviseHolderImpl_AddRef(LPOLEADVISEHOLDER);
36 static ULONG WINAPI OleAdviseHolderImpl_Release(LPOLEADVISEHOLDER);
37 static HRESULT WINAPI OleAdviseHolderImpl_Advise(LPOLEADVISEHOLDER, IAdviseSink*, DWORD*);
38 static HRESULT WINAPI OleAdviseHolderImpl_Unadvise (LPOLEADVISEHOLDER, DWORD);
39 static HRESULT WINAPI OleAdviseHolderImpl_EnumAdvise (LPOLEADVISEHOLDER, IEnumSTATDATA **);
40 static HRESULT WINAPI OleAdviseHolderImpl_SendOnRename (LPOLEADVISEHOLDER, IMoniker *);
41 static HRESULT WINAPI OleAdviseHolderImpl_SendOnSave (LPOLEADVISEHOLDER);
42 static HRESULT WINAPI OleAdviseHolderImpl_SendOnClose (LPOLEADVISEHOLDER);
45 /**************************************************************************
46 * OleAdviseHolderImpl_VTable
48 static struct ICOM_VTABLE(IOleAdviseHolder) oahvt =
50 OleAdviseHolderImpl_QueryInterface,
51 OleAdviseHolderImpl_AddRef,
52 OleAdviseHolderImpl_Release,
53 OleAdviseHolderImpl_Advise,
54 OleAdviseHolderImpl_Unadvise,
55 OleAdviseHolderImpl_EnumAdvise,
56 OleAdviseHolderImpl_SendOnRename,
57 OleAdviseHolderImpl_SendOnSave,
58 OleAdviseHolderImpl_SendOnClose
61 /**************************************************************************
62 * OleAdviseHolderImpl_Constructor
65 static LPOLEADVISEHOLDER OleAdviseHolderImpl_Constructor()
67 OleAdviseHolderImpl* lpoah;
68 DWORD index;
70 lpoah= (OleAdviseHolderImpl*)HeapAlloc(GetProcessHeap(),
72 sizeof(OleAdviseHolderImpl));
74 lpoah->lpvtbl = &oahvt;
75 lpoah->ref = 1;
76 lpoah->maxSinks = 10;
77 lpoah->arrayOfSinks = HeapAlloc(GetProcessHeap(),
79 lpoah->maxSinks * sizeof(IAdviseSink*));
81 for (index = 0; index < lpoah->maxSinks; index++)
82 lpoah->arrayOfSinks[index]=0;
84 return (LPOLEADVISEHOLDER)lpoah;
87 /**************************************************************************
88 * OleAdviseHolderImpl_Destructor
90 static void OleAdviseHolderImpl_Destructor(
91 OleAdviseHolderImpl* ptrToDestroy)
93 DWORD index;
95 for (index = 0; index < ptrToDestroy->maxSinks; index++)
97 if (ptrToDestroy->arrayOfSinks[index]!=0)
99 IAdviseSink_Release(ptrToDestroy->arrayOfSinks[index]);
100 ptrToDestroy->arrayOfSinks[index] = NULL;
104 HeapFree(GetProcessHeap(),
106 ptrToDestroy->arrayOfSinks);
109 HeapFree(GetProcessHeap(),
111 ptrToDestroy);
114 /**************************************************************************
115 * OleAdviseHolderImpl_QueryInterface
117 static HRESULT WINAPI OleAdviseHolderImpl_QueryInterface(
118 LPOLEADVISEHOLDER iface,
119 REFIID riid,
120 LPVOID* ppvObj)
122 ICOM_THIS(OleAdviseHolderImpl, iface);
125 * Sanity check
127 if (ppvObj==NULL)
128 return E_POINTER;
130 *ppvObj = NULL;
132 if (IsEqualIID(riid, &IID_IUnknown))
134 /* IUnknown */
135 *ppvObj = This;
137 else if(IsEqualIID(riid, &IID_IOleAdviseHolder))
139 /* IOleAdviseHolder */
140 *ppvObj = (IOleAdviseHolder*) This;
143 if(*ppvObj == NULL)
144 return E_NOINTERFACE;
147 * A successful QI always increments the reference count.
149 IUnknown_AddRef((IUnknown*)*ppvObj);
151 return S_OK;
154 /******************************************************************************
155 * OleAdviseHolderImpl_AddRef
157 static ULONG WINAPI OleAdviseHolderImpl_AddRef(
158 LPOLEADVISEHOLDER iface)
160 ICOM_THIS(OleAdviseHolderImpl, iface);
162 return ++(This->ref);
165 /******************************************************************************
166 * OleAdviseHolderImpl_Release
168 static ULONG WINAPI OleAdviseHolderImpl_Release(
169 LPOLEADVISEHOLDER iface)
171 ICOM_THIS(OleAdviseHolderImpl, iface);
173 This->ref--;
175 if (This->ref == 0)
177 OleAdviseHolderImpl_Destructor(This);
179 return 0;
182 return This->ref;
185 /******************************************************************************
186 * OleAdviseHolderImpl_Advise
188 static HRESULT WINAPI OleAdviseHolderImpl_Advise(
189 LPOLEADVISEHOLDER iface,
190 IAdviseSink* pAdvise,
191 DWORD* pdwConnection)
193 DWORD index;
195 ICOM_THIS(OleAdviseHolderImpl, iface);
197 TRACE(ole, "(%p, %p, %p)\n", This, pAdvise, pdwConnection);
200 * Sanity check
202 if (pdwConnection==NULL)
203 return E_POINTER;
205 *pdwConnection = 0;
208 * Find a free spot in the array.
210 for (index = 0; index < This->maxSinks; index++)
212 if (This->arrayOfSinks[index]==NULL)
213 break;
217 * If the array is full, we need to grow it.
219 if (index == This->maxSinks)
221 DWORD i;
223 This->maxSinks+=10;
225 This->arrayOfSinks = HeapReAlloc(GetProcessHeap(),
227 This->arrayOfSinks,
228 This->maxSinks*sizeof(IAdviseSink*));
230 for (i=index;i < This->maxSinks; i++)
231 This->arrayOfSinks[i]=0;
235 * Store the new sink
237 This->arrayOfSinks[index] = pAdvise;
239 if (This->arrayOfSinks[index]!=NULL)
240 IAdviseSink_AddRef(This->arrayOfSinks[index]);
243 * Return the index as the cookie.
244 * Since 0 is not a valid cookie, we will increment by
245 * 1 the index in the table.
247 *pdwConnection = index+1;
249 return S_OK;
252 /******************************************************************************
253 * OleAdviseHolderImpl_Unadvise
255 static HRESULT WINAPI OleAdviseHolderImpl_Unadvise(
256 LPOLEADVISEHOLDER iface,
257 DWORD dwConnection)
259 ICOM_THIS(OleAdviseHolderImpl, iface);
261 TRACE(ole, "(%p, %lu)\n", This, dwConnection);
264 * So we don't return 0 as a cookie, the index was
265 * incremented by 1 in OleAdviseHolderImpl_Advise
266 * we have to compensate.
268 dwConnection--;
271 * Check for invalid cookies.
273 if ( (dwConnection < 0) ||
274 (dwConnection >= This->maxSinks) )
275 return OLE_E_NOCONNECTION;
277 if (This->arrayOfSinks[dwConnection] == NULL)
278 return OLE_E_NOCONNECTION;
281 * Release the sink and mark the spot in the list as free.
283 IAdviseSink_Release(This->arrayOfSinks[dwConnection]);
284 This->arrayOfSinks[dwConnection] = NULL;
286 return S_OK;
289 /******************************************************************************
290 * OleAdviseHolderImpl_EnumAdvise
292 static HRESULT WINAPI
293 OleAdviseHolderImpl_EnumAdvise (LPOLEADVISEHOLDER iface, IEnumSTATDATA **ppenumAdvise)
295 ICOM_THIS(OleAdviseHolderImpl, iface);
296 FIXME (ole, "(%p)->(%p)\n", This, ppenumAdvise);
298 *ppenumAdvise = NULL;
300 return S_OK;
303 /******************************************************************************
304 * OleAdviseHolderImpl_SendOnRename
306 static HRESULT WINAPI
307 OleAdviseHolderImpl_SendOnRename (LPOLEADVISEHOLDER iface, IMoniker *pmk)
309 ICOM_THIS(OleAdviseHolderImpl, iface);
310 FIXME (ole, "(%p)->(%p)\n", This, pmk);
313 return S_OK;
316 /******************************************************************************
317 * OleAdviseHolderImpl_SendOnSave
319 static HRESULT WINAPI
320 OleAdviseHolderImpl_SendOnSave (LPOLEADVISEHOLDER iface)
322 ICOM_THIS(OleAdviseHolderImpl, iface);
323 FIXME (ole, "(%p)\n", This);
326 return S_OK;
329 /******************************************************************************
330 * OleAdviseHolderImpl_SendOnClose
332 static HRESULT WINAPI
333 OleAdviseHolderImpl_SendOnClose (LPOLEADVISEHOLDER iface)
335 ICOM_THIS(OleAdviseHolderImpl, iface);
336 FIXME (ole, "(%p)\n", This);
339 return S_OK;
342 /**************************************************************************
343 * DataAdviseHolder Implementation
345 typedef struct DataAdviseHolder
347 ICOM_VTABLE(IDataAdviseHolder)* lpvtbl;
349 DWORD ref;
350 } DataAdviseHolder;
352 /**************************************************************************
353 * DataAdviseHolder method prototypes
355 static IDataAdviseHolder* DataAdviseHolder_Constructor();
356 static void DataAdviseHolder_Destructor(DataAdviseHolder* ptrToDestroy);
357 static HRESULT WINAPI DataAdviseHolder_QueryInterface(
358 IDataAdviseHolder* iface,
359 REFIID riid,
360 void** ppvObject);
361 static ULONG WINAPI DataAdviseHolder_AddRef(
362 IDataAdviseHolder* iface);
363 static ULONG WINAPI DataAdviseHolder_Release(
364 IDataAdviseHolder* iface);
365 static HRESULT WINAPI DataAdviseHolder_Advise(
366 IDataAdviseHolder* iface,
367 IDataObject* pDataObject,
368 FORMATETC* pFetc,
369 DWORD advf,
370 IAdviseSink* pAdvise,
371 DWORD* pdwConnection);
372 static HRESULT WINAPI DataAdviseHolder_Unadvise(
373 IDataAdviseHolder* iface,
374 DWORD dwConnection);
375 static HRESULT WINAPI DataAdviseHolder_EnumAdvise(
376 IDataAdviseHolder* iface,
377 IEnumSTATDATA** ppenumAdvise);
378 static HRESULT WINAPI DataAdviseHolder_SendOnDataChange(
379 IDataAdviseHolder* iface,
380 IDataObject* pDataObject,
381 DWORD dwReserved,
382 DWORD advf);
384 /**************************************************************************
385 * DataAdviseHolderImpl_VTable
387 static struct ICOM_VTABLE(IDataAdviseHolder) DataAdviseHolderImpl_VTable =
389 DataAdviseHolder_QueryInterface,
390 DataAdviseHolder_AddRef,
391 DataAdviseHolder_Release,
392 DataAdviseHolder_Advise,
393 DataAdviseHolder_Unadvise,
394 DataAdviseHolder_EnumAdvise,
395 DataAdviseHolder_SendOnDataChange
398 /******************************************************************************
399 * DataAdviseHolder_Constructor
401 static IDataAdviseHolder* DataAdviseHolder_Constructor()
403 DataAdviseHolder* newHolder;
405 newHolder = (DataAdviseHolder*)HeapAlloc(GetProcessHeap(),
407 sizeof(DataAdviseHolder));
409 newHolder->lpvtbl = &DataAdviseHolderImpl_VTable;
410 newHolder->ref = 1;
412 return (IDataAdviseHolder*)newHolder;
415 /******************************************************************************
416 * DataAdviseHolder_Destructor
418 static void DataAdviseHolder_Destructor(
419 DataAdviseHolder* ptrToDestroy)
421 HeapFree(GetProcessHeap(),
423 ptrToDestroy);
426 /************************************************************************
427 * DataAdviseHolder_QueryInterface (IUnknown)
429 * See Windows documentation for more details on IUnknown methods.
431 static HRESULT WINAPI DataAdviseHolder_QueryInterface(
432 IDataAdviseHolder* iface,
433 REFIID riid,
434 void** ppvObject)
436 ICOM_THIS(DataAdviseHolder, iface);
439 * Perform a sanity check on the parameters.
441 if ( (This==0) || (ppvObject==0) )
442 return E_INVALIDARG;
445 * Initialize the return parameter.
447 *ppvObject = 0;
450 * Compare the riid with the interface IDs implemented by this object.
452 if ( (memcmp(&IID_IUnknown, riid, sizeof(IID_IUnknown)) == 0) ||
453 (memcmp(&IID_IDataAdviseHolder, riid, sizeof(IID_IDataAdviseHolder)) == 0) )
455 *ppvObject = iface;
459 * Check that we obtained an interface.
461 if ((*ppvObject)==0)
463 return E_NOINTERFACE;
467 * Query Interface always increases the reference count by one when it is
468 * successful.
470 IUnknown_AddRef((IUnknown*)*ppvObject);
472 return S_OK;;
475 /************************************************************************
476 * DataAdviseHolder_AddRef (IUnknown)
478 * See Windows documentation for more details on IUnknown methods.
480 static ULONG WINAPI DataAdviseHolder_AddRef(
481 IDataAdviseHolder* iface)
483 ICOM_THIS(DataAdviseHolder, iface);
485 This->ref++;
487 return This->ref;
490 /************************************************************************
491 * DataAdviseHolder_Release (IUnknown)
493 * See Windows documentation for more details on IUnknown methods.
495 static ULONG WINAPI DataAdviseHolder_Release(
496 IDataAdviseHolder* iface)
498 ICOM_THIS(DataAdviseHolder, iface);
501 * Decrease the reference count on this object.
503 This->ref--;
506 * If the reference count goes down to 0, perform suicide.
508 if (This->ref==0)
510 DataAdviseHolder_Destructor(This);
512 return 0;
515 return This->ref;
518 static HRESULT WINAPI DataAdviseHolder_Advise(
519 IDataAdviseHolder* iface,
520 IDataObject* pDataObject,
521 FORMATETC* pFetc,
522 DWORD advf,
523 IAdviseSink* pAdvise,
524 DWORD* pdwConnection)
526 FIXME(ole, "(): stub\n");
527 return E_NOTIMPL;
530 static HRESULT WINAPI DataAdviseHolder_Unadvise(
531 IDataAdviseHolder* iface,
532 DWORD dwConnection)
534 FIXME(ole, "(): stub\n");
535 return E_NOTIMPL;
538 static HRESULT WINAPI DataAdviseHolder_EnumAdvise(
539 IDataAdviseHolder* iface,
540 IEnumSTATDATA** ppenumAdvise)
542 FIXME(ole, "(): stub\n");
543 return E_NOTIMPL;
546 static HRESULT WINAPI DataAdviseHolder_SendOnDataChange(
547 IDataAdviseHolder* iface,
548 IDataObject* pDataObject,
549 DWORD dwReserved,
550 DWORD advf)
552 FIXME(ole, "(): stub\n");
553 return E_NOTIMPL;
556 /***********************************************************************
557 * API functions
560 /***********************************************************************
561 * CreateOleAdviseHolder [OLE32.59]
563 HRESULT WINAPI CreateOleAdviseHolder(
564 LPOLEADVISEHOLDER *ppOAHolder)
566 TRACE(ole, "(%p)\n", ppOAHolder);
569 * Sanity check,
571 if (ppOAHolder==NULL)
572 return E_POINTER;
574 *ppOAHolder = OleAdviseHolderImpl_Constructor ();
576 if (*ppOAHolder != NULL)
577 return S_OK;
579 return E_OUTOFMEMORY;
582 /******************************************************************************
583 * CreateDataAdviseHolder [OLE32.53]
585 HRESULT WINAPI CreateDataAdviseHolder(
586 LPDATAADVISEHOLDER* ppDAHolder)
588 TRACE(ole,"(%p)\n", ppDAHolder);
591 * Sanity check,
593 if (ppDAHolder==NULL)
594 return E_POINTER;
596 *ppDAHolder = DataAdviseHolder_Constructor();
598 if (*ppDAHolder != NULL)
599 return S_OK;
601 return E_OUTOFMEMORY;