xref: /reactos/dll/win32/oleaut32/dispatch.c (revision c2d0d784)
1 /**
2  * Dispatch API functions
3  *
4  * Copyright 2000  Francois Jacques, Macadamian Technologies Inc.
5  *
6  * This library 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.
10  *
11  * This library 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.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with this library; if not, write to the Free Software
18  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19  *
20  */
21 
22 #include "config.h"
23 
24 #include <stdlib.h>
25 #include <string.h>
26 #include <stdarg.h>
27 #include <stdio.h>
28 #include <ctype.h>
29 
30 #define COBJMACROS
31 
32 #include "windef.h"
33 #include "winbase.h"
34 #include "objbase.h"
35 #include "oleauto.h"
36 #include "winerror.h"
37 
38 #include "wine/debug.h"
39 
40 WINE_DEFAULT_DEBUG_CHANNEL(ole);
41 
42 static IDispatch * StdDispatch_Construct(IUnknown * punkOuter, void * pvThis, ITypeInfo * pTypeInfo);
43 
44 /******************************************************************************
45  *		DispInvoke (OLEAUT32.30)
46  *
47  * Call an object method using the information from its type library.
48  *
49  * RETURNS
50  *  Success: S_OK.
51  *  Failure: Returns DISP_E_EXCEPTION and updates pexcepinfo if an exception occurs.
52  *           DISP_E_BADPARAMCOUNT if the number of parameters is incorrect.
53  *           DISP_E_MEMBERNOTFOUND if the method does not exist.
54  *           puArgErr is updated if a parameter error (see notes) occurs.
55  *           Otherwise, returns the result of calling ITypeInfo_Invoke().
56  *
57  * NOTES
58  *  Parameter errors include the following:
59  *| DISP_E_BADVARTYPE
60  *| E_INVALIDARG            An argument was invalid
61  *| DISP_E_TYPEMISMATCH,
62  *| DISP_E_OVERFLOW         An argument was valid but could not be coerced
63  *| DISP_E_PARAMNOTOPTIONAL A non optional parameter was not passed
64  *| DISP_E_PARAMNOTFOUND    A parameter was passed that was not expected by the method
65  *  This call defers to ITypeInfo_Invoke().
66  */
67 HRESULT WINAPI DispInvoke(
68 	VOID       *_this,        /* [in] Object to call method on */
69 	ITypeInfo  *ptinfo,       /* [in] Object type info */
70 	DISPID      dispidMember, /* [in] DISPID of the member (e.g. from GetIDsOfNames()) */
71 	USHORT      wFlags,       /* [in] Kind of method call (DISPATCH_ flags from "oaidl.h") */
72 	DISPPARAMS *pparams,      /* [in] Array of method arguments */
73 	VARIANT    *pvarResult,   /* [out] Destination for the result of the call */
74 	EXCEPINFO  *pexcepinfo,   /* [out] Destination for exception information */
75 	UINT       *puArgErr)     /* [out] Destination for bad argument */
76 {
77     TRACE("\n");
78 
79     return ITypeInfo_Invoke(ptinfo, _this, dispidMember, wFlags,
80                             pparams, pvarResult, pexcepinfo, puArgErr);
81 }
82 
83 /******************************************************************************
84  *		DispGetIDsOfNames (OLEAUT32.29)
85  *
86  * Convert a set of parameter names to DISPIDs for DispInvoke().
87  *
88  * RETURNS
89  *  Success: S_OK.
90  *  Failure: An HRESULT error code.
91  *
92  * NOTES
93  *  This call defers to ITypeInfo_GetIDsOfNames(). The ITypeInfo interface passed
94  *  as ptinfo contains the information to map names to DISPIDs.
95  */
96 HRESULT WINAPI DispGetIDsOfNames(
97 	ITypeInfo  *ptinfo,    /* [in] Object's type info */
98 	OLECHAR   **rgszNames, /* [in] Array of names to get DISPIDs for */
99 	UINT        cNames,    /* [in] Number of names in rgszNames */
100 	DISPID     *rgdispid)  /* [out] Destination for converted DISPIDs */
101 {
102     return ITypeInfo_GetIDsOfNames(ptinfo, rgszNames, cNames, rgdispid);
103 }
104 
105 /******************************************************************************
106  *		DispGetParam (OLEAUT32.28)
107  *
108  * Retrieve a parameter from a DISPPARAMS structure and coerce it to the
109  * specified variant type.
110  *
111  * NOTES
112  *  Coercion is done using system (0) locale.
113  *
114  * RETURNS
115  *  Success: S_OK.
116  *  Failure: DISP_E_PARAMNOTFOUND, if position is invalid. or
117  *           DISP_E_TYPEMISMATCH, if the coercion failed. puArgErr is
118  *           set to the index of the argument in pdispparams.
119  */
120 HRESULT WINAPI DispGetParam(
121 	DISPPARAMS *pdispparams, /* [in] Parameter list */
122 	UINT        position,    /* [in] Position of parameter to coerce in pdispparams */
123 	VARTYPE     vtTarg,      /* [in] Type of value to coerce to */
124 	VARIANT    *pvarResult,  /* [out] Destination for resulting variant */
125 	UINT       *puArgErr)    /* [out] Destination for error code */
126 {
127     /* position is counted backwards */
128     UINT pos;
129     HRESULT hr;
130 
131     TRACE("position=%d, cArgs=%d, cNamedArgs=%d\n",
132           position, pdispparams->cArgs, pdispparams->cNamedArgs);
133 
134     if (position < pdispparams->cArgs)
135     {
136       /* positional arg? */
137       pos = pdispparams->cArgs - position - 1;
138     }
139     else
140     {
141       /* FIXME: is this how to handle named args? */
142       for (pos=0; pos<pdispparams->cNamedArgs; pos++)
143         if (pdispparams->rgdispidNamedArgs[pos] == position) break;
144 
145       if (pos==pdispparams->cNamedArgs)
146         return DISP_E_PARAMNOTFOUND;
147     }
148 
149     if (pdispparams->cArgs > 0 && !pdispparams->rgvarg)
150     {
151         hr = E_INVALIDARG;
152         goto done;
153     }
154 
155     if (!pvarResult)
156     {
157         hr = E_INVALIDARG;
158         goto done;
159     }
160 
161     hr = VariantChangeType(pvarResult,
162                            &pdispparams->rgvarg[pos],
163                            0, vtTarg);
164 
165 done:
166     if (FAILED(hr))
167         *puArgErr = pos;
168 
169     return hr;
170 }
171 
172 /******************************************************************************
173  * CreateStdDispatch [OLEAUT32.32]
174  *
175  * Create and return a standard IDispatch object.
176  *
177  * RETURNS
178  *  Success: S_OK. ppunkStdDisp contains the new object.
179  *  Failure: An HRESULT error code.
180  *
181  * NOTES
182  *  Outer unknown appears to be completely ignored.
183  */
184 HRESULT WINAPI CreateStdDispatch(
185         IUnknown* punkOuter,
186         void* pvThis,
187 	ITypeInfo* ptinfo,
188 	IUnknown** ppunkStdDisp)
189 {
190     TRACE("(%p, %p, %p, %p)\n", punkOuter, pvThis, ptinfo, ppunkStdDisp);
191 
192     *ppunkStdDisp = (LPUNKNOWN)StdDispatch_Construct(punkOuter, pvThis, ptinfo);
193     if (!*ppunkStdDisp)
194         return E_OUTOFMEMORY;
195     return S_OK;
196 }
197 
198 
199 /******************************************************************************
200  * IDispatch {OLEAUT32}
201  *
202  * NOTES
203  *  The IDispatch interface provides a single interface to dispatch method calls,
204  *  regardless of whether the object to be called is in or out of process,
205  *  local or remote (e.g. being called over a network). This interface is late-bound
206  *  (linked at run-time), as opposed to early-bound (linked at compile time).
207  *
208  *  The interface is used by objects that wish to called by scripting
209  *  languages such as VBA, in order to minimise the amount of COM and C/C++
210  *  knowledge required, or by objects that wish to live out of process from code
211  *  that will call their methods.
212  *
213  *  Method, property and parameter names can be localised. The details required to
214  *  map names to methods and parameters are collected in a type library, usually
215  *  output by an IDL compiler using the objects IDL description. This information is
216  *  accessible programmatically through the ITypeLib interface (for a type library),
217  *  and the ITypeInfo interface (for an object within the type library). Type information
218  *  can also be created at run-time using CreateDispTypeInfo().
219  *
220  * WRAPPERS
221  *  Instead of using IDispatch directly, there are several wrapper functions available
222  *  to simplify the process of calling an objects methods through IDispatch.
223  *
224  *  A standard implementation of an IDispatch object is created by calling
225  *  CreateStdDispatch(). Numeric Id values for the parameters and methods (DISPIDs)
226  *  of an object of interest are retrieved by calling DispGetIDsOfNames(). DispGetParam()
227  *  retrieves information about a particular parameter. Finally the DispInvoke()
228  *  function is responsible for actually calling methods on an object.
229  *
230  * METHODS
231  */
232 
233 typedef struct
234 {
235     const IDispatchVtbl *lpVtbl;
236     void * pvThis;
237     ITypeInfo * pTypeInfo;
238     LONG ref;
239 } StdDispatch;
240 
241 /******************************************************************************
242  * IDispatch_QueryInterface {OLEAUT32}
243  *
244  * See IUnknown_QueryInterface.
245  */
246 static HRESULT WINAPI StdDispatch_QueryInterface(
247   LPDISPATCH iface,
248   REFIID riid,
249   void** ppvObject)
250 {
251     StdDispatch *This = (StdDispatch *)iface;
252     TRACE("(%p)->(%s, %p)\n", iface, debugstr_guid(riid), ppvObject);
253 
254     if (IsEqualIID(riid, &IID_IDispatch) ||
255         IsEqualIID(riid, &IID_IUnknown))
256     {
257         *ppvObject = This;
258         IUnknown_AddRef((LPUNKNOWN)*ppvObject);
259         return S_OK;
260     }
261     return E_NOINTERFACE;
262 }
263 
264 /******************************************************************************
265  * IDispatch_AddRef {OLEAUT32}
266  *
267  * See IUnknown_AddRef.
268  */
269 static ULONG WINAPI StdDispatch_AddRef(LPDISPATCH iface)
270 {
271     StdDispatch *This = (StdDispatch *)iface;
272     ULONG refCount = InterlockedIncrement(&This->ref);
273 
274     TRACE("(%p)->(ref before=%u)\n",This, refCount - 1);
275 
276     return refCount;
277 }
278 
279 /******************************************************************************
280  * IDispatch_Release {OLEAUT32}
281  *
282  * See IUnknown_Release.
283  */
284 static ULONG WINAPI StdDispatch_Release(LPDISPATCH iface)
285 {
286     StdDispatch *This = (StdDispatch *)iface;
287     ULONG refCount = InterlockedDecrement(&This->ref);
288 
289     TRACE("(%p)->(ref before=%u)\n", This, refCount + 1);
290 
291     if (!refCount)
292     {
293         ITypeInfo_Release(This->pTypeInfo);
294         CoTaskMemFree(This);
295     }
296 
297     return refCount;
298 }
299 
300 /******************************************************************************
301  * IDispatch_GetTypeInfoCount {OLEAUT32}
302  *
303  * Get the count of type information in an IDispatch interface.
304  *
305  * PARAMS
306  *  iface   [I] IDispatch interface
307  *  pctinfo [O] Destination for the count
308  *
309  * RETURNS
310  *  Success: S_OK. pctinfo is updated with the count. This is always 1 if
311  *           the object provides type information, and 0 if it does not.
312  *  Failure: E_NOTIMPL. The object does not provide type information.
313  *
314  * NOTES
315  *  See IDispatch() and IDispatch_GetTypeInfo().
316  */
317 static HRESULT WINAPI StdDispatch_GetTypeInfoCount(LPDISPATCH iface, UINT * pctinfo)
318 {
319     StdDispatch *This = (StdDispatch *)iface;
320     TRACE("(%p)\n", pctinfo);
321 
322     *pctinfo = This->pTypeInfo ? 1 : 0;
323     return S_OK;
324 }
325 
326 /******************************************************************************
327  * IDispatch_GetTypeInfo {OLEAUT32}
328  *
329  * Get type information from an IDispatch interface.
330  *
331  * PARAMS
332  *  iface   [I] IDispatch interface
333  *  iTInfo  [I] Index of type information.
334  *  lcid    [I] Locale of the type information to get
335  *  ppTInfo [O] Destination for the ITypeInfo object
336  *
337  * RETURNS
338  *  Success: S_OK. ppTInfo is updated with the objects type information
339  *  Failure: DISP_E_BADINDEX, if iTInfo is any value other than 0.
340  *
341  * NOTES
342  *  See IDispatch.
343  */
344 static HRESULT WINAPI StdDispatch_GetTypeInfo(LPDISPATCH iface, UINT iTInfo, LCID lcid, ITypeInfo** ppTInfo)
345 {
346     StdDispatch *This = (StdDispatch *)iface;
347     TRACE("(%d, %x, %p)\n", iTInfo, lcid, ppTInfo);
348 
349     *ppTInfo = NULL;
350     if (iTInfo != 0)
351         return DISP_E_BADINDEX;
352 
353     if (This->pTypeInfo)
354     {
355       *ppTInfo = This->pTypeInfo;
356       ITypeInfo_AddRef(*ppTInfo);
357     }
358     return S_OK;
359 }
360 
361 /******************************************************************************
362  * IDispatch_GetIDsOfNames {OLEAUT32}
363  *
364  * Convert a methods name and an optional set of parameter names into DISPIDs
365  * for passing to IDispatch_Invoke().
366  *
367  * PARAMS
368  *  iface     [I] IDispatch interface
369  *  riid      [I] Reserved, set to IID_NULL
370  *  rgszNames [I] Name to convert
371  *  cNames    [I] Number of names in rgszNames
372  *  lcid      [I] Locale of the type information to convert from
373  *  rgDispId  [O] Destination for converted DISPIDs.
374  *
375  * RETURNS
376  *  Success: S_OK.
377  *  Failure: DISP_E_UNKNOWNNAME, if any of the names is invalid.
378  *           DISP_E_UNKNOWNLCID if lcid is invalid.
379  *           Otherwise, an HRESULT error code.
380  *
381  * NOTES
382  *  This call defers to ITypeInfo_GetIDsOfNames(), using the ITypeInfo object
383  *  contained within the IDispatch object.
384  *  The first member of the names list must be a method name. The names following
385  *  the method name are the parameters for that method.
386  */
387 static HRESULT WINAPI StdDispatch_GetIDsOfNames(LPDISPATCH iface, REFIID riid, LPOLESTR * rgszNames, UINT cNames, LCID lcid, DISPID * rgDispId)
388 {
389     StdDispatch *This = (StdDispatch *)iface;
390     TRACE("(%s, %p, %d, 0x%x, %p)\n", debugstr_guid(riid), rgszNames, cNames, lcid, rgDispId);
391 
392     if (!IsEqualGUID(riid, &IID_NULL))
393     {
394         FIXME(" expected riid == IID_NULL\n");
395         return E_INVALIDARG;
396     }
397     return DispGetIDsOfNames(This->pTypeInfo, rgszNames, cNames, rgDispId);
398 }
399 
400 /******************************************************************************
401  * IDispatch_Invoke {OLEAUT32}
402  *
403  * Call an object method.
404  *
405  * PARAMS
406  *  iface        [I] IDispatch interface
407  *  dispIdMember [I] DISPID of the method (from GetIDsOfNames())
408  *  riid         [I] Reserved, set to IID_NULL
409  *  lcid         [I] Locale of the type information to convert parameters with
410  *  wFlags,      [I] Kind of method call (DISPATCH_ flags from "oaidl.h")
411  *  pDispParams  [I] Array of method arguments
412  *  pVarResult   [O] Destination for the result of the call
413  *  pExcepInfo   [O] Destination for exception information
414  *  puArgErr     [O] Destination for bad argument
415  *
416  * RETURNS
417  *  Success: S_OK.
418  *  Failure: See DispInvoke() for failure cases.
419  *
420  * NOTES
421  *  See DispInvoke() and IDispatch().
422  */
423 static HRESULT WINAPI StdDispatch_Invoke(LPDISPATCH iface, DISPID dispIdMember, REFIID riid, LCID lcid,
424                                          WORD wFlags, DISPPARAMS * pDispParams, VARIANT * pVarResult,
425                                          EXCEPINFO * pExcepInfo, UINT * puArgErr)
426 {
427     StdDispatch *This = (StdDispatch *)iface;
428     TRACE("(%d, %s, 0x%x, 0x%x, %p, %p, %p, %p)\n", dispIdMember, debugstr_guid(riid), lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
429 
430     if (!IsEqualGUID(riid, &IID_NULL))
431     {
432         FIXME(" expected riid == IID_NULL\n");
433         return E_INVALIDARG;
434     }
435     return DispInvoke(This->pvThis, This->pTypeInfo, dispIdMember, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
436 }
437 
438 static const IDispatchVtbl StdDispatch_VTable =
439 {
440   StdDispatch_QueryInterface,
441   StdDispatch_AddRef,
442   StdDispatch_Release,
443   StdDispatch_GetTypeInfoCount,
444   StdDispatch_GetTypeInfo,
445   StdDispatch_GetIDsOfNames,
446   StdDispatch_Invoke
447 };
448 
449 static IDispatch * StdDispatch_Construct(
450   IUnknown * punkOuter,
451   void * pvThis,
452   ITypeInfo * pTypeInfo)
453 {
454     StdDispatch * pStdDispatch;
455 
456     pStdDispatch = CoTaskMemAlloc(sizeof(StdDispatch));
457     if (!pStdDispatch)
458         return (IDispatch *)pStdDispatch;
459 
460     pStdDispatch->lpVtbl = &StdDispatch_VTable;
461     pStdDispatch->pvThis = pvThis;
462     pStdDispatch->pTypeInfo = pTypeInfo;
463     pStdDispatch->ref = 1;
464 
465     /* we keep a reference to the type info so prevent it from
466      * being destroyed until we are done with it */
467     ITypeInfo_AddRef(pTypeInfo);
468 
469     return (IDispatch *)pStdDispatch;
470 }
471