push 5b1efc32b5a8acb1d5b5e60584746392dd0c436e
[wine/hacks.git] / dlls / oleaut32 / dispatch.c
blobed3fad47019d8c22e7b238e4460803eb510518be
1 /**
2 * Dispatch API functions
4 * Copyright 2000 Francois Jacques, Macadamian Technologies Inc.
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.
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.
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
22 #include "config.h"
24 #include <stdlib.h>
25 #include <string.h>
26 #include <stdarg.h>
27 #include <stdio.h>
28 #include <ctype.h>
30 #define COBJMACROS
32 #include "windef.h"
33 #include "winbase.h"
34 #include "objbase.h"
35 #include "oleauto.h"
36 #include "winerror.h"
38 #include "wine/debug.h"
40 WINE_DEFAULT_DEBUG_CHANNEL(ole);
42 static IDispatch * StdDispatch_Construct(IUnknown * punkOuter, void * pvThis, ITypeInfo * pTypeInfo);
44 /******************************************************************************
45 * DispInvoke (OLEAUT32.30)
47 * Call an object method using the information from its type library.
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().
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().
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 */
77 TRACE("\n");
79 return ITypeInfo_Invoke(ptinfo, _this, dispidMember, wFlags,
80 pparams, pvarResult, pexcepinfo, puArgErr);
83 /******************************************************************************
84 * DispGetIDsOfNames (OLEAUT32.29)
86 * Convert a set of parameter names to DISPIDs for DispInvoke().
88 * RETURNS
89 * Success: S_OK.
90 * Failure: An HRESULT error code.
92 * NOTES
93 * This call defers to ITypeInfo_GetIDsOfNames(). The ITypeInfo interface passed
94 * as ptinfo contains the information to map names to DISPIDs.
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 */
102 return ITypeInfo_GetIDsOfNames(ptinfo, rgszNames, cNames, rgdispid);
105 /******************************************************************************
106 * DispGetParam (OLEAUT32.28)
108 * Retrieve a parameter from a DISPPARAMS structure and coerce it to the
109 * specified variant type.
111 * NOTES
112 * Coercion is done using system (0) locale.
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.
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 */
127 /* position is counted backwards */
128 UINT pos;
129 HRESULT hr;
131 TRACE("position=%d, cArgs=%d, cNamedArgs=%d\n",
132 position, pdispparams->cArgs, pdispparams->cNamedArgs);
134 if (position < pdispparams->cArgs)
136 /* positional arg? */
137 pos = pdispparams->cArgs - position - 1;
139 else
141 /* FIXME: is this how to handle named args? */
142 for (pos=0; pos<pdispparams->cNamedArgs; pos++)
143 if (pdispparams->rgdispidNamedArgs[pos] == position) break;
145 if (pos==pdispparams->cNamedArgs)
146 return DISP_E_PARAMNOTFOUND;
149 if (pdispparams->cArgs > 0 && !pdispparams->rgvarg)
151 hr = E_INVALIDARG;
152 goto done;
155 if (!pvarResult)
157 hr = E_INVALIDARG;
158 goto done;
161 hr = VariantChangeType(pvarResult,
162 &pdispparams->rgvarg[pos],
163 0, vtTarg);
165 done:
166 if (FAILED(hr))
167 *puArgErr = pos;
169 return hr;
172 /******************************************************************************
173 * CreateStdDispatch [OLEAUT32.32]
175 * Create and return a standard IDispatch object.
177 * RETURNS
178 * Success: S_OK. ppunkStdDisp contains the new object.
179 * Failure: An HRESULT error code.
181 * NOTES
182 * Outer unknown appears to be completely ignored.
184 HRESULT WINAPI CreateStdDispatch(
185 IUnknown* punkOuter,
186 void* pvThis,
187 ITypeInfo* ptinfo,
188 IUnknown** ppunkStdDisp)
190 TRACE("(%p, %p, %p, %p)\n", punkOuter, pvThis, ptinfo, ppunkStdDisp);
192 *ppunkStdDisp = (LPUNKNOWN)StdDispatch_Construct(punkOuter, pvThis, ptinfo);
193 if (!*ppunkStdDisp)
194 return E_OUTOFMEMORY;
195 return S_OK;
199 /******************************************************************************
200 * IDispatch {OLEAUT32}
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).
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.
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().
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.
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.
230 * METHODS
233 typedef struct
235 const IDispatchVtbl *lpVtbl;
236 void * pvThis;
237 ITypeInfo * pTypeInfo;
238 LONG ref;
239 } StdDispatch;
241 /******************************************************************************
242 * IDispatch_QueryInterface {OLEAUT32}
244 * See IUnknown_QueryInterface.
246 static HRESULT WINAPI StdDispatch_QueryInterface(
247 LPDISPATCH iface,
248 REFIID riid,
249 void** ppvObject)
251 StdDispatch *This = (StdDispatch *)iface;
252 TRACE("(%p)->(%s, %p)\n", iface, debugstr_guid(riid), ppvObject);
254 if (IsEqualIID(riid, &IID_IDispatch) ||
255 IsEqualIID(riid, &IID_IUnknown))
257 *ppvObject = This;
258 IUnknown_AddRef((LPUNKNOWN)*ppvObject);
259 return S_OK;
261 return E_NOINTERFACE;
264 /******************************************************************************
265 * IDispatch_AddRef {OLEAUT32}
267 * See IUnknown_AddRef.
269 static ULONG WINAPI StdDispatch_AddRef(LPDISPATCH iface)
271 StdDispatch *This = (StdDispatch *)iface;
272 ULONG refCount = InterlockedIncrement(&This->ref);
274 TRACE("(%p)->(ref before=%u)\n",This, refCount - 1);
276 return refCount;
279 /******************************************************************************
280 * IDispatch_Release {OLEAUT32}
282 * See IUnknown_Release.
284 static ULONG WINAPI StdDispatch_Release(LPDISPATCH iface)
286 StdDispatch *This = (StdDispatch *)iface;
287 ULONG refCount = InterlockedDecrement(&This->ref);
289 TRACE("(%p)->(ref before=%u)\n", This, refCount + 1);
291 if (!refCount)
293 ITypeInfo_Release(This->pTypeInfo);
294 CoTaskMemFree(This);
297 return refCount;
300 /******************************************************************************
301 * IDispatch_GetTypeInfoCount {OLEAUT32}
303 * Get the count of type information in an IDispatch interface.
305 * PARAMS
306 * iface [I] IDispatch interface
307 * pctinfo [O] Destination for the count
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.
314 * NOTES
315 * See IDispatch() and IDispatch_GetTypeInfo().
317 static HRESULT WINAPI StdDispatch_GetTypeInfoCount(LPDISPATCH iface, UINT * pctinfo)
319 StdDispatch *This = (StdDispatch *)iface;
320 TRACE("(%p)\n", pctinfo);
322 *pctinfo = This->pTypeInfo ? 1 : 0;
323 return S_OK;
326 /******************************************************************************
327 * IDispatch_GetTypeInfo {OLEAUT32}
329 * Get type information from an IDispatch interface.
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
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.
341 * NOTES
342 * See IDispatch.
344 static HRESULT WINAPI StdDispatch_GetTypeInfo(LPDISPATCH iface, UINT iTInfo, LCID lcid, ITypeInfo** ppTInfo)
346 StdDispatch *This = (StdDispatch *)iface;
347 TRACE("(%d, %x, %p)\n", iTInfo, lcid, ppTInfo);
349 *ppTInfo = NULL;
350 if (iTInfo != 0)
351 return DISP_E_BADINDEX;
353 if (This->pTypeInfo)
355 *ppTInfo = This->pTypeInfo;
356 ITypeInfo_AddRef(*ppTInfo);
358 return S_OK;
361 /******************************************************************************
362 * IDispatch_GetIDsOfNames {OLEAUT32}
364 * Convert a methods name and an optional set of parameter names into DISPIDs
365 * for passing to IDispatch_Invoke().
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.
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.
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.
387 static HRESULT WINAPI StdDispatch_GetIDsOfNames(LPDISPATCH iface, REFIID riid, LPOLESTR * rgszNames, UINT cNames, LCID lcid, DISPID * rgDispId)
389 StdDispatch *This = (StdDispatch *)iface;
390 TRACE("(%s, %p, %d, 0x%x, %p)\n", debugstr_guid(riid), rgszNames, cNames, lcid, rgDispId);
392 if (!IsEqualGUID(riid, &IID_NULL))
394 FIXME(" expected riid == IID_NULL\n");
395 return E_INVALIDARG;
397 return DispGetIDsOfNames(This->pTypeInfo, rgszNames, cNames, rgDispId);
400 /******************************************************************************
401 * IDispatch_Invoke {OLEAUT32}
403 * Call an object method.
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
416 * RETURNS
417 * Success: S_OK.
418 * Failure: See DispInvoke() for failure cases.
420 * NOTES
421 * See DispInvoke() and IDispatch().
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)
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);
430 if (!IsEqualGUID(riid, &IID_NULL))
432 FIXME(" expected riid == IID_NULL\n");
433 return E_INVALIDARG;
435 return DispInvoke(This->pvThis, This->pTypeInfo, dispIdMember, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
438 static const IDispatchVtbl StdDispatch_VTable =
440 StdDispatch_QueryInterface,
441 StdDispatch_AddRef,
442 StdDispatch_Release,
443 StdDispatch_GetTypeInfoCount,
444 StdDispatch_GetTypeInfo,
445 StdDispatch_GetIDsOfNames,
446 StdDispatch_Invoke
449 static IDispatch * StdDispatch_Construct(
450 IUnknown * punkOuter,
451 void * pvThis,
452 ITypeInfo * pTypeInfo)
454 StdDispatch * pStdDispatch;
456 pStdDispatch = CoTaskMemAlloc(sizeof(StdDispatch));
457 if (!pStdDispatch)
458 return (IDispatch *)pStdDispatch;
460 pStdDispatch->lpVtbl = &StdDispatch_VTable;
461 pStdDispatch->pvThis = pvThis;
462 pStdDispatch->pTypeInfo = pTypeInfo;
463 pStdDispatch->ref = 1;
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);
469 return (IDispatch *)pStdDispatch;