/* * COMPOBJ library * * Copyright 1995 Martin von Loewis * Copyright 1998 Justin Bradford * Copyright 1999 Francis Beaudet * Copyright 1999 Sylvain St-Germain * Copyright 2002 Marcus Meissner * Copyright 2004 Mike Hearn * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include "config.h" #include #include #include #include #include #define COBJMACROS #define NONAMELESSUNION #define NONAMELESSSTRUCT #include "windef.h" #include "winbase.h" #include "winuser.h" #include "objbase.h" #include "ole2.h" #include "ole2ver.h" #include "rpc.h" #include "winerror.h" #include "winreg.h" #include "wownt32.h" #include "wine/unicode.h" #include "objbase.h" #include "ole32_main.h" #include "compobj_private.h" #include "wine/debug.h" WINE_DEFAULT_DEBUG_CHANNEL(ole); typedef LPCSTR LPCOLESTR16; /**************************************************************************** * This section defines variables internal to the COM module. * * TODO: Most of these things will have to be made thread-safe. */ static HRESULT COM_GetRegisteredClassObject(REFCLSID rclsid, DWORD dwClsContext, LPUNKNOWN* ppUnk); static void COM_RevokeAllClasses(void); static void COM_ExternalLockFreeList(void); const CLSID CLSID_StdGlobalInterfaceTable = { 0x00000323, 0, 0, {0xc0, 0, 0, 0, 0, 0, 0, 0x46} }; APARTMENT MTA, *apts; static CRITICAL_SECTION csApartment; static CRITICAL_SECTION_DEBUG critsect_debug = { 0, 0, &csApartment, { &critsect_debug.ProcessLocksList, &critsect_debug.ProcessLocksList }, 0, 0, { 0, (DWORD)(__FILE__ ": csApartment") } }; static CRITICAL_SECTION csApartment = { &critsect_debug, -1, 0, 0, 0, 0 }; /* * This lock count counts the number of times CoInitialize is called. It is * decreased every time CoUninitialize is called. When it hits 0, the COM * libraries are freed */ static LONG s_COMLockCount = 0; /* * This linked list contains the list of registered class objects. These * are mostly used to register the factories for out-of-proc servers of OLE * objects. * * TODO: Make this data structure aware of inter-process communication. This * means that parts of this will be exported to the Wine Server. */ typedef struct tagRegisteredClass { CLSID classIdentifier; LPUNKNOWN classObject; DWORD runContext; DWORD connectFlags; DWORD dwCookie; HANDLE hThread; /* only for localserver */ struct tagRegisteredClass* nextClass; } RegisteredClass; static RegisteredClass* firstRegisteredClass = NULL; static CRITICAL_SECTION csRegisteredClassList; static CRITICAL_SECTION_DEBUG class_cs_debug = { 0, 0, &csRegisteredClassList, { &class_cs_debug.ProcessLocksList, &class_cs_debug.ProcessLocksList }, 0, 0, { 0, (DWORD)(__FILE__ ": csRegisteredClassList") } }; static CRITICAL_SECTION csRegisteredClassList = { &class_cs_debug, -1, 0, 0, 0, 0 }; /***************************************************************************** * This section contains OpenDllList definitions * * The OpenDllList contains only handles of dll loaded by CoGetClassObject or * other functions that do LoadLibrary _without_ giving back a HMODULE. * Without this list these handles would never be freed. * * FIXME: a DLL that says OK when asked for unloading is unloaded in the * next unload-call but not before 600 sec. */ typedef struct tagOpenDll { HINSTANCE hLibrary; struct tagOpenDll *next; } OpenDll; static OpenDll *openDllList = NULL; /* linked list of open dlls */ static CRITICAL_SECTION csOpenDllList; static CRITICAL_SECTION_DEBUG dll_cs_debug = { 0, 0, &csOpenDllList, { &dll_cs_debug.ProcessLocksList, &dll_cs_debug.ProcessLocksList }, 0, 0, { 0, (DWORD)(__FILE__ ": csOpenDllList") } }; static CRITICAL_SECTION csOpenDllList = { &dll_cs_debug, -1, 0, 0, 0, 0 }; static const char aptWinClass[] = "WINE_OLE32_APT_CLASS"; static LRESULT CALLBACK COM_AptWndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam); static void COMPOBJ_DLLList_Add(HANDLE hLibrary); static void COMPOBJ_DllList_FreeUnused(int Timeout); void COMPOBJ_InitProcess( void ) { WNDCLASSA wclass; /* Dispatching to the correct thread in an apartment is done through * window messages rather than RPC transports. When an interface is * marshalled into another apartment in the same process, a window of the * following class is created. The *caller* of CoMarshalInterface (ie the * application) is responsible for pumping the message loop in that thread. * The WM_USER messages which point to the RPCs are then dispatched to * COM_AptWndProc by the user's code from the apartment in which the interface * was unmarshalled. */ memset(&wclass, 0, sizeof(wclass)); wclass.lpfnWndProc = &COM_AptWndProc; wclass.hInstance = OLE32_hInstance; wclass.lpszClassName = aptWinClass; RegisterClassA(&wclass); } void COMPOBJ_UninitProcess( void ) { UnregisterClassA(aptWinClass, OLE32_hInstance); } /****************************************************************************** * Manage apartments. */ /* The multi-threaded apartment (MTA) contains zero or more threads interacting with free threaded (ie thread safe) COM objects. There is only ever one MTA in a process - you can enter it by calling CoInitializeEx(COINIT_MULTITHREADED) */ static void COM_InitMTA(void) { /* OXIDs are object exporter IDs. Each apartment has an OXID, which is unique within a network. That is, two different MTAs on different machines will have different OXIDs. This method of generating an OXID is therefore wrong as it doesn't work across a network, but for local RPC only it's OK. We can distinguish between MTAs and STAs because STAs use the thread ID as well, and no thread can have an ID of zero. The algorithm Microsoft use is currently unknown. */ MTA.oxid = ((OXID)GetCurrentProcessId() << 32); InitializeCriticalSection(&MTA.cs); } static void COM_UninitMTA(void) { DeleteCriticalSection(&MTA.cs); MTA.oxid = 0; } /* creates an apartment structure which stores OLE thread-local * information. Call with COINIT_UNINITIALIZED to create an apartment * that will be initialized with a model later. Note: do not call * with COINIT_UNINITIALIZED if the apartment has already been initialized * with a different COINIT value */ APARTMENT* COM_CreateApartment(DWORD model) { APARTMENT *apt; BOOL create = (NtCurrentTeb()->ReservedForOle == NULL); if (create) { apt = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(APARTMENT)); apt->tid = GetCurrentThreadId(); DuplicateHandle(GetCurrentProcess(), GetCurrentThread(), GetCurrentProcess(), &apt->thread, THREAD_ALL_ACCESS, FALSE, 0); } else apt = NtCurrentTeb()->ReservedForOle; apt->model = model; if (model & COINIT_APARTMENTTHREADED) { /* FIXME: how does windoze create OXIDs? */ apt->oxid = MTA.oxid | GetCurrentThreadId(); apt->win = CreateWindowA(aptWinClass, NULL, 0, 0, 0, 0, 0, 0, 0, OLE32_hInstance, NULL); InitializeCriticalSection(&apt->cs); } else if (!(model & COINIT_UNINITIALIZED)) { apt->parent = &MTA; apt->oxid = MTA.oxid; } EnterCriticalSection(&csApartment); if (create) { if (apts) apts->prev = apt; apt->next = apts; apts = apt; } LeaveCriticalSection(&csApartment); NtCurrentTeb()->ReservedForOle = apt; return apt; } static void COM_DestroyApartment(APARTMENT *apt) { EnterCriticalSection(&csApartment); if (apt->prev) apt->prev->next = apt->next; if (apt->next) apt->next->prev = apt->prev; if (apts == apt) apts = apt->next; apt->prev = NULL; apt->next = NULL; LeaveCriticalSection(&csApartment); if (apt->model & COINIT_APARTMENTTHREADED) { if (apt->win) DestroyWindow(apt->win); DeleteCriticalSection(&apt->cs); } CloseHandle(apt->thread); HeapFree(GetProcessHeap(), 0, apt); } /* The given OXID must be local to this process: you cannot use apartment windows to send RPCs to other processes. This all needs to move to rpcrt4 */ HWND COM_GetApartmentWin(OXID oxid) { APARTMENT *apt; HWND win = 0; EnterCriticalSection(&csApartment); apt = apts; while (apt && apt->oxid != oxid) apt = apt->next; if (apt) win = apt->win; LeaveCriticalSection(&csApartment); return win; } /* Currently inter-thread marshalling is not fully implemented, so this does nothing */ static LRESULT CALLBACK COM_AptWndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam) { return DefWindowProcA(hWnd, msg, wParam, lParam); } /***************************************************************************** * This section contains OpenDllList implemantation */ static void COMPOBJ_DLLList_Add(HANDLE hLibrary) { OpenDll *ptr; OpenDll *tmp; TRACE("\n"); EnterCriticalSection( &csOpenDllList ); if (openDllList == NULL) { /* empty list -- add first node */ openDllList = (OpenDll*)HeapAlloc(GetProcessHeap(),0, sizeof(OpenDll)); openDllList->hLibrary=hLibrary; openDllList->next = NULL; } else { /* search for this dll */ int found = FALSE; for (ptr = openDllList; ptr->next != NULL; ptr=ptr->next) { if (ptr->hLibrary == hLibrary) { found = TRUE; break; } } if (!found) { /* dll not found, add it */ tmp = openDllList; openDllList = (OpenDll*)HeapAlloc(GetProcessHeap(),0, sizeof(OpenDll)); openDllList->hLibrary = hLibrary; openDllList->next = tmp; } } LeaveCriticalSection( &csOpenDllList ); } static void COMPOBJ_DllList_FreeUnused(int Timeout) { OpenDll *curr, *next, *prev = NULL; typedef HRESULT(*DllCanUnloadNowFunc)(void); DllCanUnloadNowFunc DllCanUnloadNow; TRACE("\n"); EnterCriticalSection( &csOpenDllList ); for (curr = openDllList; curr != NULL; ) { DllCanUnloadNow = (DllCanUnloadNowFunc) GetProcAddress(curr->hLibrary, "DllCanUnloadNow"); if ( (DllCanUnloadNow != NULL) && (DllCanUnloadNow() == S_OK) ) { next = curr->next; TRACE("freeing %p\n", curr->hLibrary); FreeLibrary(curr->hLibrary); HeapFree(GetProcessHeap(), 0, curr); if (curr == openDllList) { openDllList = next; } else { prev->next = next; } curr = next; } else { prev = curr; curr = curr->next; } } LeaveCriticalSection( &csOpenDllList ); } /****************************************************************************** * CoBuildVersion [COMPOBJ.1] * CoBuildVersion [OLE32.@] * * RETURNS * Current build version, hiword is majornumber, loword is minornumber */ DWORD WINAPI CoBuildVersion(void) { TRACE("Returning version %d, build %d.\n", rmm, rup); return (rmm<<16)+rup; } /****************************************************************************** * CoInitialize [OLE32.@] * * Initializes the COM libraries by calling CoInitializeEx with * COINIT_APARTMENTTHREADED, ie it enters a STA thread. * * SEE ALSO * CoInitializeEx */ HRESULT WINAPI CoInitialize( LPVOID lpReserved /* [in] pointer to win32 malloc interface (obsolete, should be NULL) */ ) { /* * Just delegate to the newer method. */ return CoInitializeEx(lpReserved, COINIT_APARTMENTTHREADED); } /****************************************************************************** * CoInitializeEx [OLE32.@] * * Initializes the COM libraries. The behavior used to set the win32 * IMalloc used for memory management is obsolete. If * COINIT_APARTMENTTHREADED is specified this thread enters a new STA * (single threaded apartment), otherwise COINIT_MULTITHREADED should * be specified which indicates that the thread will enter the MTA. * * Currently STA threading is only partly implemented. * * RETURNS * S_OK if successful, * S_FALSE if this function was called already. * RPC_E_CHANGED_MODE if a previous call to CoInitializeEx specified another * threading model. * * SEE ALSO * CoUninitialize */ HRESULT WINAPI CoInitializeEx( LPVOID lpReserved, /* [in] pointer to win32 malloc interface (obsolete, should be NULL) */ DWORD dwCoInit /* [in] A value from COINIT specifies the threading model */ ) { HRESULT hr = S_OK; APARTMENT *apt; TRACE("(%p, %x)\n", lpReserved, (int)dwCoInit); if (lpReserved!=NULL) { ERR("(%p, %x) - Bad parameter passed-in %p, must be an old Windows Application\n", lpReserved, (int)dwCoInit, lpReserved); } apt = NtCurrentTeb()->ReservedForOle; if (apt && !(apt->model == COINIT_UNINITIALIZED)) { if (dwCoInit != apt->model) { /* Changing the threading model after it's been set is illegal. If this warning is triggered by Wine code then we are probably using the wrong threading model to implement that API. */ ERR("Attempt to change threading model of this apartment from 0x%lx to 0x%lx\n", apt->model, dwCoInit); return RPC_E_CHANGED_MODE; } hr = S_FALSE; } else hr = S_OK; /* * Check the lock count. If this is the first time going through the initialize * process, we have to initialize the libraries. * * And crank-up that lock count. */ if (InterlockedExchangeAdd(&s_COMLockCount,1)==0) { /* * Initialize the various COM libraries and data structures. */ TRACE("() - Initializing the COM libraries\n"); COM_InitMTA(); RunningObjectTableImpl_Initialize(); } if (!apt || apt->model == COINIT_UNINITIALIZED) apt = COM_CreateApartment(dwCoInit); InterlockedIncrement(&apt->inits); if (hr == S_OK) NtCurrentTeb()->ReservedForOle = apt; return hr; } /* On COM finalization for a STA thread, the message queue is flushed to ensure no pending RPCs are ignored. Non-COM messages are discarded at this point. */ void COM_FlushMessageQueue(void) { MSG message; APARTMENT *apt = NtCurrentTeb()->ReservedForOle; if (!apt || !apt->win) return; TRACE("Flushing STA message queue\n"); while (PeekMessageA(&message, NULL, 0, 0, PM_REMOVE)) { if (message.hwnd != apt->win) continue; TranslateMessage(&message); DispatchMessageA(&message); } } /*********************************************************************** * CoUninitialize [OLE32.@] * * This method will decrement the refcount on the COM libraries, * potentially unloading them. The current thread leaves the apartment * it's currently in. If not in an apartment, the routine does * nothing. * * If COM is to be shut down, any outstanding proxies are * disconnected, all registered class objects are unregistered and the * message queue for the thread is flushed (if native does * this or not is unknown). * * SEE ALSO * CoInitializeEx */ void WINAPI CoUninitialize(void) { LONG lCOMRefCnt; APARTMENT *apt; TRACE("()\n"); apt = NtCurrentTeb()->ReservedForOle; if (!apt) return; if (InterlockedDecrement(&apt->inits)==0) { NtCurrentTeb()->ReservedForOle = NULL; COM_DestroyApartment(apt); apt = NULL; } /* * Decrease the reference count. * If we are back to 0 locks on the COM library, make sure we free * all the associated data structures. */ lCOMRefCnt = InterlockedExchangeAdd(&s_COMLockCount,-1); if (lCOMRefCnt==1) { TRACE("() - Releasing the COM libraries\n"); RunningObjectTableImpl_UnInitialize(); /* disconnect proxies to release the corresponding stubs. * It is confirmed in "Essential COM" in the sub-chapter on * "Lifecycle Management and Marshalling" that the native version also * does some kind of proxy cleanup in this function. * FIXME: does it just disconnect or completely destroy the proxies? * FIXME: should this be in the apartment destructor? */ MARSHAL_Disconnect_Proxies(); /* Release the references to the registered class objects */ COM_RevokeAllClasses(); /* This will free the loaded COM Dlls */ CoFreeAllLibraries(); /* This will free list of external references to COM objects */ COM_ExternalLockFreeList(); /* This ensures we deal with any pending RPCs */ COM_FlushMessageQueue(); COM_UninitMTA(); } else if (lCOMRefCnt<1) { ERR( "CoUninitialize() - not CoInitialized.\n" ); InterlockedExchangeAdd(&s_COMLockCount,1); /* restore the lock count. */ } } /****************************************************************************** * CoDisconnectObject [COMPOBJ.15] * CoDisconnectObject [OLE32.@] * * Disconnects all connections to this object from remote processes. Dispatches * pending RPCs while blocking new RPCs from occurring, and then calls * IMarshal::DisconnectObject on the given object. * * Typically called when the object server is forced to shut down, for instance by * the user. */ HRESULT WINAPI CoDisconnectObject( LPUNKNOWN lpUnk, DWORD reserved ) { FIXME("(%p, %lx): stub - probably harmless\n",lpUnk,reserved); return S_OK; } /****************************************************************************** * CoCreateGuid[OLE32.@] * * Simply forwards to UuidCreate in RPCRT4. * * SEE ALSO * UuidCreate * */ HRESULT WINAPI CoCreateGuid( GUID *pguid /* [out] points to the GUID to initialize */ ) { return UuidCreate(pguid); } /****************************************************************************** * CLSIDFromString [OLE32.@] * IIDFromString [OLE32.@] * * Converts a unique identifier from its string representation into * the GUID struct. * * In Windows, if idstr is not a valid CLSID string then it gets * treated as a ProgID. Wine currently doesn't do this. If idstr is * NULL it's treated as an all-zero GUID. * * RETURNS * S_OK on success * CO_E_CLASSSTRING if idstr is not a valid CLSID */ HRESULT WINAPI __CLSIDFromStringA( LPCSTR idstr, /* [in] string representation of guid */ CLSID *id) /* [out] GUID converted from string */ { const BYTE *s = (const BYTE *) idstr; int i; BYTE table[256]; if (!s) s = "{00000000-0000-0000-0000-000000000000}"; else { /* validate the CLSID string */ if (strlen(s) != 38) return CO_E_CLASSSTRING; if ((s[0]!='{') || (s[9]!='-') || (s[14]!='-') || (s[19]!='-') || (s[24]!='-') || (s[37]!='}')) return CO_E_CLASSSTRING; for (i=1; i<37; i++) { if ((i == 9)||(i == 14)||(i == 19)||(i == 24)) continue; if (!(((s[i] >= '0') && (s[i] <= '9')) || ((s[i] >= 'a') && (s[i] <= 'f')) || ((s[i] >= 'A') && (s[i] <= 'F')))) return CO_E_CLASSSTRING; } } TRACE("%s -> %p\n", s, id); /* quick lookup table */ memset(table, 0, 256); for (i = 0; i < 10; i++) { table['0' + i] = i; } for (i = 0; i < 6; i++) { table['A' + i] = i+10; table['a' + i] = i+10; } /* in form {XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX} */ id->Data1 = (table[s[1]] << 28 | table[s[2]] << 24 | table[s[3]] << 20 | table[s[4]] << 16 | table[s[5]] << 12 | table[s[6]] << 8 | table[s[7]] << 4 | table[s[8]]); id->Data2 = table[s[10]] << 12 | table[s[11]] << 8 | table[s[12]] << 4 | table[s[13]]; id->Data3 = table[s[15]] << 12 | table[s[16]] << 8 | table[s[17]] << 4 | table[s[18]]; /* these are just sequential bytes */ id->Data4[0] = table[s[20]] << 4 | table[s[21]]; id->Data4[1] = table[s[22]] << 4 | table[s[23]]; id->Data4[2] = table[s[25]] << 4 | table[s[26]]; id->Data4[3] = table[s[27]] << 4 | table[s[28]]; id->Data4[4] = table[s[29]] << 4 | table[s[30]]; id->Data4[5] = table[s[31]] << 4 | table[s[32]]; id->Data4[6] = table[s[33]] << 4 | table[s[34]]; id->Data4[7] = table[s[35]] << 4 | table[s[36]]; return S_OK; } /*****************************************************************************/ HRESULT WINAPI CLSIDFromString( LPOLESTR idstr, /* [in] string representation of GUID */ CLSID *id ) /* [out] GUID represented by above string */ { char xid[40]; HRESULT ret; if (!WideCharToMultiByte( CP_ACP, 0, idstr, -1, xid, sizeof(xid), NULL, NULL )) return CO_E_CLASSSTRING; ret = __CLSIDFromStringA(xid,id); if(ret != S_OK) { /* It appears a ProgID is also valid */ ret = CLSIDFromProgID(idstr, id); } return ret; } /* Converts a GUID into the respective string representation. */ HRESULT WINE_StringFromCLSID( const CLSID *id, /* [in] GUID to be converted */ LPSTR idstr /* [out] pointer to buffer to contain converted guid */ ) { static const char *hex = "0123456789ABCDEF"; char *s; int i; if (!id) { ERR("called with id=Null\n"); *idstr = 0x00; return E_FAIL; } sprintf(idstr, "{%08lX-%04X-%04X-%02X%02X-", id->Data1, id->Data2, id->Data3, id->Data4[0], id->Data4[1]); s = &idstr[25]; /* 6 hex bytes */ for (i = 2; i < 8; i++) { *s++ = hex[id->Data4[i]>>4]; *s++ = hex[id->Data4[i] & 0xf]; } *s++ = '}'; *s++ = '\0'; TRACE("%p->%s\n", id, idstr); return S_OK; } /****************************************************************************** * StringFromCLSID [OLE32.@] * StringFromIID [OLE32.@] * * Converts a GUID into the respective string representation. * The target string is allocated using the OLE IMalloc. * * RETURNS * S_OK * E_FAIL */ HRESULT WINAPI StringFromCLSID( REFCLSID id, /* [in] the GUID to be converted */ LPOLESTR *idstr /* [out] a pointer to a to-be-allocated pointer pointing to the resulting string */ ) { char buf[80]; HRESULT ret; LPMALLOC mllc; if ((ret = CoGetMalloc(0,&mllc))) return ret; ret=WINE_StringFromCLSID(id,buf); if (!ret) { DWORD len = MultiByteToWideChar( CP_ACP, 0, buf, -1, NULL, 0 ); *idstr = IMalloc_Alloc( mllc, len * sizeof(WCHAR) ); MultiByteToWideChar( CP_ACP, 0, buf, -1, *idstr, len ); } return ret; } /****************************************************************************** * StringFromGUID2 [COMPOBJ.76] * StringFromGUID2 [OLE32.@] * * Modified version of StringFromCLSID that allows you to specify max * buffer size. * * RETURNS * The length of the resulting string, 0 if there was any problem. */ INT WINAPI StringFromGUID2( REFGUID id, /* [in] GUID to convert to string */ LPOLESTR str, /* [out] Unicode buffer to hold result */ INT cmax) { char xguid[80]; if (WINE_StringFromCLSID(id,xguid)) return 0; return MultiByteToWideChar( CP_ACP, 0, xguid, -1, str, cmax ); } /****************************************************************************** * ProgIDFromCLSID [OLE32.@] * * Converts a class id into the respective Program ID. (By using a * registry lookup) * * RETURNS * S_OK * E_OUTOFMEMORY * REGDB_E_CLASSNOTREG if the given clsid has no associated ProgID */ HRESULT WINAPI ProgIDFromCLSID( REFCLSID clsid, /* [in] class id as found in registry */ LPOLESTR *lplpszProgID/* [out] associated Prog ID */ ) { char strCLSID[50], *buf, *buf2; DWORD buf2len; HKEY xhkey; LPMALLOC mllc; HRESULT ret = S_OK; WINE_StringFromCLSID(clsid, strCLSID); buf = HeapAlloc(GetProcessHeap(), 0, strlen(strCLSID)+14); sprintf(buf,"CLSID\\%s\\ProgID", strCLSID); if (RegOpenKeyA(HKEY_CLASSES_ROOT, buf, &xhkey)) ret = REGDB_E_CLASSNOTREG; HeapFree(GetProcessHeap(), 0, buf); if (ret == S_OK) { buf2 = HeapAlloc(GetProcessHeap(), 0, 255); buf2len = 255; if (RegQueryValueA(xhkey, NULL, buf2, &buf2len)) ret = REGDB_E_CLASSNOTREG; if (ret == S_OK) { if (CoGetMalloc(0,&mllc)) ret = E_OUTOFMEMORY; else { DWORD len = MultiByteToWideChar( CP_ACP, 0, buf2, -1, NULL, 0 ); *lplpszProgID = IMalloc_Alloc(mllc, len * sizeof(WCHAR) ); MultiByteToWideChar( CP_ACP, 0, buf2, -1, *lplpszProgID, len ); } } HeapFree(GetProcessHeap(), 0, buf2); } RegCloseKey(xhkey); return ret; } HRESULT WINAPI CLSIDFromProgID16( LPCOLESTR16 progid, /* [in] program id as found in registry */ LPCLSID riid /* [out] associated CLSID */ ) { char *buf,buf2[80]; DWORD buf2len; HRESULT err; HKEY xhkey; buf = HeapAlloc(GetProcessHeap(),0,strlen(progid)+8); sprintf(buf,"%s\\CLSID",progid); if ((err=RegOpenKeyA(HKEY_CLASSES_ROOT,buf,&xhkey))) { HeapFree(GetProcessHeap(),0,buf); return CO_E_CLASSSTRING; } HeapFree(GetProcessHeap(),0,buf); buf2len = sizeof(buf2); if ((err=RegQueryValueA(xhkey,NULL,buf2,&buf2len))) { RegCloseKey(xhkey); return CO_E_CLASSSTRING; } RegCloseKey(xhkey); return __CLSIDFromStringA(buf2,riid); } /****************************************************************************** * CLSIDFromProgID [COMPOBJ.61] * * Converts a program id into the respective GUID. (By using a * registry lookup) * * RETURNS * S_OK * CO_E_CLASSSTRING if the given ProgID cannot be found */ HRESULT WINAPI CLSIDFromProgID( LPCOLESTR progid, /* [in] Unicode program id as found in registry */ LPCLSID riid ) /* [out] associated CLSID */ { static const WCHAR clsidW[] = { '\\','C','L','S','I','D',0 }; char buf2[80]; DWORD buf2len = sizeof(buf2); HKEY xhkey; WCHAR *buf = HeapAlloc( GetProcessHeap(),0,(strlenW(progid)+8) * sizeof(WCHAR) ); strcpyW( buf, progid ); strcatW( buf, clsidW ); if (RegOpenKeyW(HKEY_CLASSES_ROOT,buf,&xhkey)) { HeapFree(GetProcessHeap(),0,buf); return CO_E_CLASSSTRING; } HeapFree(GetProcessHeap(),0,buf); if (RegQueryValueA(xhkey,NULL,buf2,&buf2len)) { RegCloseKey(xhkey); return CO_E_CLASSSTRING; } RegCloseKey(xhkey); return __CLSIDFromStringA(buf2,riid); } /***************************************************************************** * CoGetPSClsid [OLE32.@] * * This function returns the CLSID of the proxy/stub factory that * implements IPSFactoryBuffer for the specified interface. * * The standard marshaller activates the object with the CLSID * returned and uses the CreateProxy and CreateStub methods on its * IPSFactoryBuffer interface to construct the proxies and stubs for a * given object. * * CoGetPSClsid determines this CLSID by searching the * HKEY_CLASSES_ROOT\Interface\{string form of riid}\ProxyStubClsid32 * in the registry and any interface id registered by * CoRegisterPSClsid within the current process. * * FIXME: We only search the registry, not ids registered with * CoRegisterPSClsid. * * RETURNS * S_OK * E_OUTOFMEMORY * E_INVALIDARG if no PSFactoryBuffer is associated with the IID, or it could not be parsed */ HRESULT WINAPI CoGetPSClsid( REFIID riid, /* [in] Interface whose proxy/stub CLSID is to be returned */ CLSID *pclsid ) /* [out] Where to store returned proxy/stub CLSID */ { char *buf, buf2[40]; DWORD buf2len; HKEY xhkey; TRACE("() riid=%s, pclsid=%p\n", debugstr_guid(riid), pclsid); /* Get the input iid as a string */ WINE_StringFromCLSID(riid, buf2); /* Allocate memory for the registry key we will construct. (length of iid string plus constant length of static text */ buf = HeapAlloc(GetProcessHeap(), 0, strlen(buf2)+27); if (buf == NULL) { return (E_OUTOFMEMORY); } /* Construct the registry key we want */ sprintf(buf,"Interface\\%s\\ProxyStubClsid32", buf2); /* Open the key.. */ if (RegOpenKeyA(HKEY_CLASSES_ROOT, buf, &xhkey)) { WARN("No PSFactoryBuffer object is registered for this IID\n"); HeapFree(GetProcessHeap(),0,buf); return (E_INVALIDARG); } HeapFree(GetProcessHeap(),0,buf); /* ... Once we have the key, query the registry to get the value of CLSID as a string, and convert it into a proper CLSID structure to be passed back to the app */ buf2len = sizeof(buf2); if ( (RegQueryValueA(xhkey,NULL,buf2,&buf2len)) ) { RegCloseKey(xhkey); return E_INVALIDARG; } RegCloseKey(xhkey); /* We have the CLSid we want back from the registry as a string, so lets convert it into a CLSID structure */ if ( (__CLSIDFromStringA(buf2,pclsid)) != NOERROR) { return E_INVALIDARG; } TRACE ("() Returning CLSID=%s\n", debugstr_guid(pclsid)); return (S_OK); } /*********************************************************************** * WriteClassStm (OLE32.@) * * This function write a CLSID on stream */ HRESULT WINAPI WriteClassStm(IStream *pStm,REFCLSID rclsid) { TRACE("(%p,%p)\n",pStm,rclsid); if (rclsid==NULL) return E_INVALIDARG; return IStream_Write(pStm,rclsid,sizeof(CLSID),NULL); } /*********************************************************************** * ReadClassStm (OLE32.@) * * This function read a CLSID from a stream */ HRESULT WINAPI ReadClassStm(IStream *pStm,CLSID *pclsid) { ULONG nbByte; HRESULT res; TRACE("(%p,%p)\n",pStm,pclsid); if (pclsid==NULL) return E_INVALIDARG; res = IStream_Read(pStm,(void*)pclsid,sizeof(CLSID),&nbByte); if (FAILED(res)) return res; if (nbByte != sizeof(CLSID)) return S_FALSE; else return S_OK; } /*** * COM_GetRegisteredClassObject * * This internal method is used to scan the registered class list to * find a class object. * * Params: * rclsid Class ID of the class to find. * dwClsContext Class context to match. * ppv [out] returns a pointer to the class object. Complying * to normal COM usage, this method will increase the * reference count on this object. */ static HRESULT COM_GetRegisteredClassObject( REFCLSID rclsid, DWORD dwClsContext, LPUNKNOWN* ppUnk) { HRESULT hr = S_FALSE; RegisteredClass* curClass; EnterCriticalSection( &csRegisteredClassList ); /* * Sanity check */ assert(ppUnk!=0); /* * Iterate through the whole list and try to match the class ID. */ curClass = firstRegisteredClass; while (curClass != 0) { /* * Check if we have a match on the class ID. */ if (IsEqualGUID(&(curClass->classIdentifier), rclsid)) { /* * Since we don't do out-of process or DCOM just right away, let's ignore the * class context. */ /* * We have a match, return the pointer to the class object. */ *ppUnk = curClass->classObject; IUnknown_AddRef(curClass->classObject); hr = S_OK; goto end; } /* * Step to the next class in the list. */ curClass = curClass->nextClass; } end: LeaveCriticalSection( &csRegisteredClassList ); /* * If we get to here, we haven't found our class. */ return hr; } static DWORD WINAPI _LocalServerThread(LPVOID param) { HANDLE hPipe; char pipefn[200]; RegisteredClass *newClass = (RegisteredClass*)param; HRESULT hres; IStream *pStm; STATSTG ststg; unsigned char *buffer; int buflen; IClassFactory *classfac; LARGE_INTEGER seekto; ULARGE_INTEGER newpos; ULONG res; TRACE("Starting classfactory server thread for %s.\n",debugstr_guid(&newClass->classIdentifier)); strcpy(pipefn,PIPEPREF); WINE_StringFromCLSID(&newClass->classIdentifier,pipefn+strlen(PIPEPREF)); hPipe = CreateNamedPipeA( pipefn, PIPE_ACCESS_DUPLEX, PIPE_TYPE_BYTE|PIPE_WAIT, PIPE_UNLIMITED_INSTANCES, 4096, 4096, NMPWAIT_USE_DEFAULT_WAIT, NULL ); if (hPipe == INVALID_HANDLE_VALUE) { FIXME("pipe creation failed for %s, le is %lx\n",pipefn,GetLastError()); return 1; } while (1) { if (!ConnectNamedPipe(hPipe,NULL)) { ERR("Failure during ConnectNamedPipe %lx, ABORT!\n",GetLastError()); break; } TRACE("marshalling IClassFactory to client\n"); hres = IUnknown_QueryInterface(newClass->classObject,&IID_IClassFactory,(LPVOID*)&classfac); if (hres) return hres; hres = CreateStreamOnHGlobal(0,TRUE,&pStm); if (hres) { FIXME("Failed to create stream on hglobal.\n"); return hres; } hres = CoMarshalInterface(pStm,&IID_IClassFactory,(LPVOID)classfac,0,NULL,0); if (hres) { FIXME("CoMarshalInterface failed, %lx!\n",hres); return hres; } IUnknown_Release(classfac); /* is this right? */ hres = IStream_Stat(pStm,&ststg,0); if (hres) return hres; buflen = ststg.cbSize.u.LowPart; buffer = HeapAlloc(GetProcessHeap(),0,buflen); seekto.u.LowPart = 0; seekto.u.HighPart = 0; hres = IStream_Seek(pStm,seekto,SEEK_SET,&newpos); if (hres) { FIXME("IStream_Seek failed, %lx\n",hres); return hres; } hres = IStream_Read(pStm,buffer,buflen,&res); if (hres) { FIXME("Stream Read failed, %lx\n",hres); return hres; } IStream_Release(pStm); WriteFile(hPipe,buffer,buflen,&res,NULL); FlushFileBuffers(hPipe); DisconnectNamedPipe(hPipe); TRACE("done marshalling IClassFactory\n"); } CloseHandle(hPipe); return 0; } /****************************************************************************** * CoRegisterClassObject [OLE32.@] * * This method will register the class object for a given class * ID. Servers housed in EXE files use this method instead of * exporting DllGetClassObject to allow other code to connect to their * objects. * * When a class object (an object which implements IClassFactory) is * registered in this way, a new thread is started which listens for * connections on a named pipe specific to the registered CLSID. When * something else connects to it, it writes out the marshalled * IClassFactory interface to the pipe. The code on the other end uses * this buffer to unmarshal the class factory, and can then call * methods on it. * * In Windows, such objects are registered with the RPC endpoint * mapper, not with a unique named pipe. * * MSDN claims that multiple interface registrations are legal, but we * can't do that with our current implementation. * * RETURNS * S_OK on success, * E_INVALIDARG if lpdwRegister or pUnk are NULL, * CO_E_OBJISREG if the object is already registered. We should not return this. * * SEE ALSO * CoRevokeClassObject, CoGetClassObject */ HRESULT WINAPI CoRegisterClassObject( REFCLSID rclsid, /* [in] CLSID of the object to register */ LPUNKNOWN pUnk, /* [in] IUnknown of the object */ DWORD dwClsContext, /* [in] CLSCTX flags indicating the context in which to run the executable */ DWORD flags, /* [in] REGCLS flags indicating how connections are made */ LPDWORD lpdwRegister) /* [out] A unique cookie that can be passed to CoRevokeClassObject */ { RegisteredClass* newClass; LPUNKNOWN foundObject; HRESULT hr; TRACE("(%s,%p,0x%08lx,0x%08lx,%p)\n", debugstr_guid(rclsid),pUnk,dwClsContext,flags,lpdwRegister); if ( (lpdwRegister==0) || (pUnk==0) ) return E_INVALIDARG; *lpdwRegister = 0; /* * First, check if the class is already registered. * If it is, this should cause an error. */ hr = COM_GetRegisteredClassObject(rclsid, dwClsContext, &foundObject); if (hr == S_OK) { IUnknown_Release(foundObject); return CO_E_OBJISREG; } newClass = HeapAlloc(GetProcessHeap(), 0, sizeof(RegisteredClass)); if ( newClass == NULL ) return E_OUTOFMEMORY; EnterCriticalSection( &csRegisteredClassList ); newClass->classIdentifier = *rclsid; newClass->runContext = dwClsContext; newClass->connectFlags = flags; /* * Use the address of the chain node as the cookie since we are sure it's * unique. */ newClass->dwCookie = (DWORD)newClass; newClass->nextClass = firstRegisteredClass; /* * Since we're making a copy of the object pointer, we have to increase its * reference count. */ newClass->classObject = pUnk; IUnknown_AddRef(newClass->classObject); firstRegisteredClass = newClass; LeaveCriticalSection( &csRegisteredClassList ); *lpdwRegister = newClass->dwCookie; if (dwClsContext & CLSCTX_LOCAL_SERVER) { DWORD tid; start_apartment_listener_thread(); newClass->hThread = CreateThread(NULL,0,_LocalServerThread,newClass,0,&tid); } return S_OK; } /*********************************************************************** * CoRevokeClassObject [OLE32.@] * * This method will remove a class object from the class registry * * See the Windows documentation for more details. */ HRESULT WINAPI CoRevokeClassObject( DWORD dwRegister) { HRESULT hr = E_INVALIDARG; RegisteredClass** prevClassLink; RegisteredClass* curClass; TRACE("(%08lx)\n",dwRegister); EnterCriticalSection( &csRegisteredClassList ); /* * Iterate through the whole list and try to match the cookie. */ curClass = firstRegisteredClass; prevClassLink = &firstRegisteredClass; while (curClass != 0) { /* * Check if we have a match on the cookie. */ if (curClass->dwCookie == dwRegister) { /* * Remove the class from the chain. */ *prevClassLink = curClass->nextClass; /* * Release the reference to the class object. */ IUnknown_Release(curClass->classObject); /* * Free the memory used by the chain node. */ HeapFree(GetProcessHeap(), 0, curClass); hr = S_OK; goto end; } /* * Step to the next class in the list. */ prevClassLink = &(curClass->nextClass); curClass = curClass->nextClass; } end: LeaveCriticalSection( &csRegisteredClassList ); /* * If we get to here, we haven't found our class. */ return hr; } /*********************************************************************** * compobj_RegReadPath [internal] * * Reads a registry value and expands it when necessary */ HRESULT compobj_RegReadPath(char * keyname, char * valuename, char * dst, DWORD dstlen) { HRESULT hres; HKEY key; DWORD keytype; char src[MAX_PATH]; DWORD dwLength = dstlen; if((hres = RegOpenKeyExA(HKEY_CLASSES_ROOT, keyname, 0, KEY_READ, &key)) == ERROR_SUCCESS) { if( (hres = RegQueryValueExA(key, NULL, NULL, &keytype, (LPBYTE)src, &dwLength)) == ERROR_SUCCESS ) { if (keytype == REG_EXPAND_SZ) { if (dstlen <= ExpandEnvironmentStringsA(src, dst, dstlen)) hres = ERROR_MORE_DATA; } else { lstrcpynA(dst, src, dstlen); } } RegCloseKey (key); } return hres; } /*********************************************************************** * CoGetClassObject [COMPOBJ.7] * CoGetClassObject [OLE32.@] * * FIXME. If request allows of several options and there is a failure * with one (other than not being registered) do we try the * others or return failure? (E.g. inprocess is registered but * the DLL is not found but the server version works) */ HRESULT WINAPI CoGetClassObject( REFCLSID rclsid, DWORD dwClsContext, COSERVERINFO *pServerInfo, REFIID iid, LPVOID *ppv ) { LPUNKNOWN regClassObject; HRESULT hres = E_UNEXPECTED; char xclsid[80]; HINSTANCE hLibrary; typedef HRESULT (CALLBACK *DllGetClassObjectFunc)(REFCLSID clsid, REFIID iid, LPVOID *ppv); DllGetClassObjectFunc DllGetClassObject; WINE_StringFromCLSID((LPCLSID)rclsid,xclsid); TRACE("\n\tCLSID:\t%s,\n\tIID:\t%s\n", debugstr_guid(rclsid), debugstr_guid(iid)); if (pServerInfo) { FIXME("\tpServerInfo: name=%s\n",debugstr_w(pServerInfo->pwszName)); FIXME("\t\tpAuthInfo=%p\n",pServerInfo->pAuthInfo); } /* * First, try and see if we can't match the class ID with one of the * registered classes. */ if (S_OK == COM_GetRegisteredClassObject(rclsid, dwClsContext, ®ClassObject)) { /* * Get the required interface from the retrieved pointer. */ hres = IUnknown_QueryInterface(regClassObject, iid, ppv); /* * Since QI got another reference on the pointer, we want to release the * one we already have. If QI was unsuccessful, this will release the object. This * is good since we are not returning it in the "out" parameter. */ IUnknown_Release(regClassObject); return hres; } /* first try: in-process */ if ((CLSCTX_INPROC_SERVER | CLSCTX_INPROC_HANDLER) & dwClsContext) { char keyname[MAX_PATH]; char dllpath[MAX_PATH+1]; sprintf(keyname,"CLSID\\%s\\InprocServer32",xclsid); if ( compobj_RegReadPath(keyname, NULL, dllpath, sizeof(dllpath)) != ERROR_SUCCESS) { /* failure: CLSID is not found in registry */ WARN("class %s not registered inproc\n", xclsid); hres = REGDB_E_CLASSNOTREG; } else { if ((hLibrary = LoadLibraryExA(dllpath, 0, LOAD_WITH_ALTERED_SEARCH_PATH)) == 0) { /* failure: DLL could not be loaded */ ERR("couldn't load InprocServer32 dll %s\n", dllpath); hres = E_ACCESSDENIED; /* FIXME: or should this be CO_E_DLLNOTFOUND? */ } else if (!(DllGetClassObject = (DllGetClassObjectFunc)GetProcAddress(hLibrary, "DllGetClassObject"))) { /* failure: the dll did not export DllGetClassObject */ ERR("couldn't find function DllGetClassObject in %s\n", dllpath); FreeLibrary( hLibrary ); hres = CO_E_DLLNOTFOUND; } else { /* OK: get the ClassObject */ COMPOBJ_DLLList_Add( hLibrary ); return DllGetClassObject(rclsid, iid, ppv); } } } /* Next try out of process */ if (CLSCTX_LOCAL_SERVER & dwClsContext) { return create_marshalled_proxy(rclsid,iid,ppv); } /* Finally try remote: this requires networked DCOM (a lot of work) */ if (CLSCTX_REMOTE_SERVER & dwClsContext) { FIXME ("CLSCTX_REMOTE_SERVER not supported\n"); hres = E_NOINTERFACE; } return hres; } /*********************************************************************** * CoResumeClassObjects (OLE32.@) * * Resumes classobjects registered with REGCLS suspended */ HRESULT WINAPI CoResumeClassObjects(void) { FIXME("stub\n"); return S_OK; } /*********************************************************************** * GetClassFile (OLE32.@) * * This function supplies the CLSID associated with the given filename. */ HRESULT WINAPI GetClassFile(LPCOLESTR filePathName,CLSID *pclsid) { IStorage *pstg=0; HRESULT res; int nbElm, length, i; LONG sizeProgId; LPOLESTR *pathDec=0,absFile=0,progId=0; LPWSTR extension; static const WCHAR bkslashW[] = {'\\',0}; static const WCHAR dotW[] = {'.',0}; TRACE("%s, %p\n", debugstr_w(filePathName), pclsid); /* if the file contain a storage object the return the CLSID written by IStorage_SetClass method*/ if((StgIsStorageFile(filePathName))==S_OK){ res=StgOpenStorage(filePathName,NULL,STGM_READ | STGM_SHARE_DENY_WRITE,NULL,0,&pstg); if (SUCCEEDED(res)) res=ReadClassStg(pstg,pclsid); IStorage_Release(pstg); return res; } /* if the file is not a storage object then attemps to match various bits in the file against a pattern in the registry. this case is not frequently used ! so I present only the psodocode for this case for(i=0;i= 0) && *(extension = &absFile[i]) != '.'; i--) /* nothing */; if (!extension || !lstrcmpW(extension, dotW)) return MK_E_INVALIDEXTENSION; res=RegQueryValueW(HKEY_CLASSES_ROOT, extension, NULL, &sizeProgId); /* get the progId associated to the extension */ progId = CoTaskMemAlloc(sizeProgId); res = RegQueryValueW(HKEY_CLASSES_ROOT, extension, progId, &sizeProgId); if (res==ERROR_SUCCESS) /* return the clsid associated to the progId */ res= CLSIDFromProgID(progId,pclsid); for(i=0; pathDec[i]!=NULL;i++) CoTaskMemFree(pathDec[i]); CoTaskMemFree(pathDec); CoTaskMemFree(progId); if (res==ERROR_SUCCESS) return res; return MK_E_INVALIDEXTENSION; } /*********************************************************************** * CoCreateInstance [COMPOBJ.13] * CoCreateInstance [OLE32.@] */ HRESULT WINAPI CoCreateInstance( REFCLSID rclsid, LPUNKNOWN pUnkOuter, DWORD dwClsContext, REFIID iid, LPVOID *ppv) { HRESULT hres; LPCLASSFACTORY lpclf = 0; if (!COM_CurrentApt()) return CO_E_NOTINITIALIZED; /* * Sanity check */ if (ppv==0) return E_POINTER; /* * Initialize the "out" parameter */ *ppv = 0; /* * The Standard Global Interface Table (GIT) object is a process-wide singleton. * Rather than create a class factory, we can just check for it here */ if (IsEqualIID(rclsid, &CLSID_StdGlobalInterfaceTable)) { if (StdGlobalInterfaceTableInstance == NULL) StdGlobalInterfaceTableInstance = StdGlobalInterfaceTable_Construct(); hres = IGlobalInterfaceTable_QueryInterface( (IGlobalInterfaceTable*) StdGlobalInterfaceTableInstance, iid, ppv); if (hres) return hres; TRACE("Retrieved GIT (%p)\n", *ppv); return S_OK; } /* * Get a class factory to construct the object we want. */ hres = CoGetClassObject(rclsid, dwClsContext, NULL, &IID_IClassFactory, (LPVOID)&lpclf); if (FAILED(hres)) { FIXME("no classfactory created for CLSID %s, hres is 0x%08lx\n", debugstr_guid(rclsid),hres); return hres; } /* * Create the object and don't forget to release the factory */ hres = IClassFactory_CreateInstance(lpclf, pUnkOuter, iid, ppv); IClassFactory_Release(lpclf); if(FAILED(hres)) FIXME("no instance created for interface %s of class %s, hres is 0x%08lx\n", debugstr_guid(iid), debugstr_guid(rclsid),hres); return hres; } /*********************************************************************** * CoCreateInstanceEx [OLE32.@] */ HRESULT WINAPI CoCreateInstanceEx( REFCLSID rclsid, LPUNKNOWN pUnkOuter, DWORD dwClsContext, COSERVERINFO* pServerInfo, ULONG cmq, MULTI_QI* pResults) { IUnknown* pUnk = NULL; HRESULT hr; ULONG index; ULONG successCount = 0; /* * Sanity check */ if ( (cmq==0) || (pResults==NULL)) return E_INVALIDARG; if (pServerInfo!=NULL) FIXME("() non-NULL pServerInfo not supported!\n"); /* * Initialize all the "out" parameters. */ for (index = 0; index < cmq; index++) { pResults[index].pItf = NULL; pResults[index].hr = E_NOINTERFACE; } /* * Get the object and get its IUnknown pointer. */ hr = CoCreateInstance(rclsid, pUnkOuter, dwClsContext, &IID_IUnknown, (VOID**)&pUnk); if (hr) return hr; /* * Then, query for all the interfaces requested. */ for (index = 0; index < cmq; index++) { pResults[index].hr = IUnknown_QueryInterface(pUnk, pResults[index].pIID, (VOID**)&(pResults[index].pItf)); if (pResults[index].hr == S_OK) successCount++; } /* * Release our temporary unknown pointer. */ IUnknown_Release(pUnk); if (successCount == 0) return E_NOINTERFACE; if (successCount!=cmq) return CO_S_NOTALLINTERFACES; return S_OK; } /*********************************************************************** * CoLoadLibrary (OLE32.@) */ HINSTANCE WINAPI CoLoadLibrary(LPOLESTR lpszLibName, BOOL bAutoFree) { TRACE("(%s, %d)\n", debugstr_w(lpszLibName), bAutoFree); return LoadLibraryExW(lpszLibName, 0, LOAD_WITH_ALTERED_SEARCH_PATH); } /*********************************************************************** * CoFreeLibrary [OLE32.@] * * NOTES: don't believe the documentation */ void WINAPI CoFreeLibrary(HINSTANCE hLibrary) { FreeLibrary(hLibrary); } /*********************************************************************** * CoFreeAllLibraries [OLE32.@] * * NOTES: don't believe the documentation */ void WINAPI CoFreeAllLibraries(void) { /* NOP */ } /*********************************************************************** * CoFreeUnusedLibraries [COMPOBJ.17] * CoFreeUnusedLibraries [OLE32.@] * * FIXME: Calls to CoFreeUnusedLibraries from any thread always route * through the main apartment's thread to call DllCanUnloadNow */ void WINAPI CoFreeUnusedLibraries(void) { COMPOBJ_DllList_FreeUnused(0); } /*********************************************************************** * CoFileTimeNow [COMPOBJ.82] * CoFileTimeNow [OLE32.@] * * RETURNS * the current system time in lpFileTime */ HRESULT WINAPI CoFileTimeNow( FILETIME *lpFileTime ) /* [out] the current time */ { GetSystemTimeAsFileTime( lpFileTime ); return S_OK; } /*********************************************************************** * CoLoadLibrary (OLE32.@) */ static void COM_RevokeAllClasses() { EnterCriticalSection( &csRegisteredClassList ); while (firstRegisteredClass!=0) { CoRevokeClassObject(firstRegisteredClass->dwCookie); } LeaveCriticalSection( &csRegisteredClassList ); } /**************************************************************************** * COM External Lock methods implementation * * This api provides a linked list to managed external references to * COM objects. * * The public interface consists of three calls: * COM_ExternalLockAddRef * COM_ExternalLockRelease * COM_ExternalLockFreeList */ #define EL_END_OF_LIST 0 #define EL_NOT_FOUND 0 /* * Declaration of the static structure that manage the * external lock to COM objects. */ typedef struct COM_ExternalLock COM_ExternalLock; typedef struct COM_ExternalLockList COM_ExternalLockList; struct COM_ExternalLock { IUnknown *pUnk; /* IUnknown referenced */ ULONG uRefCount; /* external lock counter to IUnknown object*/ COM_ExternalLock *next; /* Pointer to next element in list */ }; struct COM_ExternalLockList { COM_ExternalLock *head; /* head of list */ }; /* * Declaration and initialization of the static structure that manages * the external lock to COM objects. */ static COM_ExternalLockList elList = { EL_END_OF_LIST }; /* * Private methods used to managed the linked list */ static COM_ExternalLock* COM_ExternalLockLocate( COM_ExternalLock *element, IUnknown *pUnk); /**************************************************************************** * Internal - Insert a new IUnknown* to the linked list */ static BOOL COM_ExternalLockInsert( IUnknown *pUnk) { COM_ExternalLock *newLock = NULL; COM_ExternalLock *previousHead = NULL; /* * Allocate space for the new storage object */ newLock = HeapAlloc(GetProcessHeap(), 0, sizeof(COM_ExternalLock)); if (newLock!=NULL) { if ( elList.head == EL_END_OF_LIST ) { elList.head = newLock; /* The list is empty */ } else { /* insert does it at the head */ previousHead = elList.head; elList.head = newLock; } /* Set new list item data member */ newLock->pUnk = pUnk; newLock->uRefCount = 1; newLock->next = previousHead; return TRUE; } return FALSE; } /**************************************************************************** * Internal - Method that removes an item from the linked list. */ static void COM_ExternalLockDelete( COM_ExternalLock *itemList) { COM_ExternalLock *current = elList.head; if ( current == itemList ) { /* this section handles the deletion of the first node */ elList.head = itemList->next; HeapFree( GetProcessHeap(), 0, itemList); } else { do { if ( current->next == itemList ){ /* We found the item to free */ current->next = itemList->next; /* readjust the list pointers */ HeapFree( GetProcessHeap(), 0, itemList); break; } /* Skip to the next item */ current = current->next; } while ( current != EL_END_OF_LIST ); } } /**************************************************************************** * Internal - Recursivity agent for IUnknownExternalLockList_Find * * NOTES: how long can the list be ?? (recursive!!!) */ static COM_ExternalLock* COM_ExternalLockLocate( COM_ExternalLock *element, IUnknown *pUnk) { if ( element == EL_END_OF_LIST ) return EL_NOT_FOUND; else if ( element->pUnk == pUnk ) /* We found it */ return element; else /* Not the right guy, keep on looking */ return COM_ExternalLockLocate( element->next, pUnk); } /**************************************************************************** * Public - Method that increments the count for a IUnknown* in the linked * list. The item is inserted if not already in the list. */ static void COM_ExternalLockAddRef(IUnknown *pUnk) { COM_ExternalLock *externalLock = COM_ExternalLockLocate(elList.head, pUnk); /* * Add an external lock to the object. If it was already externally * locked, just increase the reference count. If it was not. * add the item to the list. */ if ( externalLock == EL_NOT_FOUND ) COM_ExternalLockInsert(pUnk); else externalLock->uRefCount++; /* * Add an internal lock to the object */ IUnknown_AddRef(pUnk); } /**************************************************************************** * Public - Method that decrements the count for a IUnknown* in the linked * list. The item is removed from the list if its count end up at zero or if * bRelAll is TRUE. */ static void COM_ExternalLockRelease( IUnknown *pUnk, BOOL bRelAll) { COM_ExternalLock *externalLock = COM_ExternalLockLocate(elList.head, pUnk); if ( externalLock != EL_NOT_FOUND ) { do { externalLock->uRefCount--; /* release external locks */ IUnknown_Release(pUnk); /* release local locks as well */ if ( bRelAll == FALSE ) break; /* perform single release */ } while ( externalLock->uRefCount > 0 ); if ( externalLock->uRefCount == 0 ) /* get rid of the list entry */ COM_ExternalLockDelete(externalLock); } } /**************************************************************************** * Public - Method that frees the content of the list. */ static void COM_ExternalLockFreeList() { COM_ExternalLock *head; head = elList.head; /* grab it by the head */ while ( head != EL_END_OF_LIST ) { COM_ExternalLockDelete(head); /* get rid of the head stuff */ head = elList.head; /* get the new head... */ } } /**************************************************************************** * Public - Method that dump the content of the list. */ void COM_ExternalLockDump() { COM_ExternalLock *current = elList.head; DPRINTF("\nExternal lock list contains:\n"); while ( current != EL_END_OF_LIST ) { DPRINTF( "\t%p with %lu references count.\n", current->pUnk, current->uRefCount); /* Skip to the next item */ current = current->next; } } /****************************************************************************** * CoLockObjectExternal [OLE32.@] */ HRESULT WINAPI CoLockObjectExternal( LPUNKNOWN pUnk, /* [in] object to be locked */ BOOL fLock, /* [in] do lock */ BOOL fLastUnlockReleases) /* [in] unlock all */ { if (fLock) { /* * Increment the external lock coutner, COM_ExternalLockAddRef also * increment the object's internal lock counter. */ COM_ExternalLockAddRef( pUnk); } else { /* * Decrement the external lock coutner, COM_ExternalLockRelease also * decrement the object's internal lock counter. */ COM_ExternalLockRelease( pUnk, fLastUnlockReleases); } return S_OK; } /*********************************************************************** * CoInitializeWOW (OLE32.@) */ HRESULT WINAPI CoInitializeWOW(DWORD x,DWORD y) { FIXME("(0x%08lx,0x%08lx),stub!\n",x,y); return 0; } /*********************************************************************** * CoGetState [OLE32.@] * * NOTES: might be incomplete */ HRESULT WINAPI CoGetState(IUnknown ** ppv) { APARTMENT * apt = COM_CurrentInfo(); FIXME("\n"); if(apt && apt->state) { IUnknown_AddRef(apt->state); *ppv = apt->state; FIXME("-- %p\n", *ppv); return S_OK; } *ppv = NULL; return E_FAIL; } /*********************************************************************** * CoSetState [OLE32.@] * */ HRESULT WINAPI CoSetState(IUnknown * pv) { APARTMENT * apt = COM_CurrentInfo(); if (!apt) apt = COM_CreateApartment(COINIT_UNINITIALIZED); FIXME("(%p),stub!\n", pv); if (pv) { IUnknown_AddRef(pv); } if (apt->state) { TRACE("-- release %p now\n", apt->state); IUnknown_Release(apt->state); } apt->state = pv; return S_OK; } /****************************************************************************** * OleGetAutoConvert [OLE32.@] */ HRESULT WINAPI OleGetAutoConvert(REFCLSID clsidOld, LPCLSID pClsidNew) { HKEY hkey = 0; char buf[200]; WCHAR wbuf[200]; DWORD len; HRESULT res = S_OK; sprintf(buf,"CLSID\\");WINE_StringFromCLSID(clsidOld,&buf[6]); if (RegOpenKeyA(HKEY_CLASSES_ROOT,buf,&hkey)) { res = REGDB_E_CLASSNOTREG; goto done; } len = 200; /* we can just query for the default value of AutoConvertTo key like that, without opening the AutoConvertTo key and querying for NULL (default) */ if (RegQueryValueA(hkey,"AutoConvertTo",buf,&len)) { res = REGDB_E_KEYMISSING; goto done; } MultiByteToWideChar( CP_ACP, 0, buf, -1, wbuf, sizeof(wbuf)/sizeof(WCHAR) ); CLSIDFromString(wbuf,pClsidNew); done: if (hkey) RegCloseKey(hkey); return res; } /****************************************************************************** * OleSetAutoConvert [OLE32.@] */ HRESULT WINAPI OleSetAutoConvert(REFCLSID clsidOld, REFCLSID clsidNew) { HKEY hkey = 0; char buf[200], szClsidNew[200]; HRESULT res = S_OK; TRACE("(%s,%s)\n", debugstr_guid(clsidOld), debugstr_guid(clsidNew)); sprintf(buf,"CLSID\\");WINE_StringFromCLSID(clsidOld,&buf[6]); WINE_StringFromCLSID(clsidNew, szClsidNew); if (RegOpenKeyA(HKEY_CLASSES_ROOT,buf,&hkey)) { res = REGDB_E_CLASSNOTREG; goto done; } if (RegSetValueA(hkey, "AutoConvertTo", REG_SZ, szClsidNew, strlen(szClsidNew)+1)) { res = REGDB_E_WRITEREGDB; goto done; } done: if (hkey) RegCloseKey(hkey); return res; } /****************************************************************************** * OleDoAutoConvert [OLE32.@] */ HRESULT WINAPI OleDoAutoConvert(IStorage *pStg, LPCLSID pClsidNew) { FIXME("(%p,%p) : stub\n",pStg,pClsidNew); return E_NOTIMPL; } /****************************************************************************** * CoTreatAsClass [OLE32.@] * * Sets TreatAs value of a class */ HRESULT WINAPI CoTreatAsClass(REFCLSID clsidOld, REFCLSID clsidNew) { HKEY hkey = 0; char buf[47]; char szClsidNew[39]; HRESULT res = S_OK; char auto_treat_as[39]; LONG auto_treat_as_size = sizeof(auto_treat_as); CLSID id; sprintf(buf,"CLSID\\");WINE_StringFromCLSID(clsidOld,&buf[6]); WINE_StringFromCLSID(clsidNew, szClsidNew); if (RegOpenKeyA(HKEY_CLASSES_ROOT,buf,&hkey)) { res = REGDB_E_CLASSNOTREG; goto done; } if (!memcmp( clsidOld, clsidNew, sizeof(*clsidOld) )) { if (!RegQueryValueA(hkey, "AutoTreatAs", auto_treat_as, &auto_treat_as_size) && !__CLSIDFromStringA(auto_treat_as, &id)) { if (RegSetValueA(hkey, "TreatAs", REG_SZ, auto_treat_as, strlen(auto_treat_as)+1)) { res = REGDB_E_WRITEREGDB; goto done; } } else { RegDeleteKeyA(hkey, "TreatAs"); goto done; } } else if (RegSetValueA(hkey, "TreatAs", REG_SZ, szClsidNew, strlen(szClsidNew)+1)) { res = REGDB_E_WRITEREGDB; goto done; } done: if (hkey) RegCloseKey(hkey); return res; } /****************************************************************************** * CoGetTreatAsClass [OLE32.@] * * Reads the TreatAs value from a class. */ HRESULT WINAPI CoGetTreatAsClass(REFCLSID clsidOld, LPCLSID clsidNew) { HKEY hkey = 0; char buf[200], szClsidNew[200]; HRESULT res = S_OK; LONG len = sizeof(szClsidNew); FIXME("(%s,%p)\n", debugstr_guid(clsidOld), clsidNew); sprintf(buf,"CLSID\\");WINE_StringFromCLSID(clsidOld,&buf[6]); memcpy(clsidNew,clsidOld,sizeof(CLSID)); /* copy over old value */ if (RegOpenKeyA(HKEY_CLASSES_ROOT,buf,&hkey)) { res = REGDB_E_CLASSNOTREG; goto done; } if (RegQueryValueA(hkey, "TreatAs", szClsidNew, &len)) { res = S_FALSE; goto done; } res = __CLSIDFromStringA(szClsidNew,clsidNew); if (FAILED(res)) FIXME("Failed CLSIDFromStringA(%s), hres %lx?\n",szClsidNew,res); done: if (hkey) RegCloseKey(hkey); return res; } /*********************************************************************** * IsEqualGUID [OLE32.@] * * Compares two Unique Identifiers. * * RETURNS * TRUE if equal */ #undef IsEqualGUID BOOL WINAPI IsEqualGUID( REFGUID rguid1, /* [in] unique id 1 */ REFGUID rguid2 /* [in] unique id 2 */ ) { return !memcmp(rguid1,rguid2,sizeof(GUID)); } /*********************************************************************** * CoInitializeSecurity [OLE32.@] */ HRESULT WINAPI CoInitializeSecurity(PSECURITY_DESCRIPTOR pSecDesc, LONG cAuthSvc, SOLE_AUTHENTICATION_SERVICE* asAuthSvc, void* pReserved1, DWORD dwAuthnLevel, DWORD dwImpLevel, void* pReserved2, DWORD dwCapabilities, void* pReserved3) { FIXME("(%p,%ld,%p,%p,%ld,%ld,%p,%ld,%p) - stub!\n", pSecDesc, cAuthSvc, asAuthSvc, pReserved1, dwAuthnLevel, dwImpLevel, pReserved2, dwCapabilities, pReserved3); return S_OK; } /*********************************************************************** * CoSuspendClassObjects [OLE32.@] */ HRESULT WINAPI CoSuspendClassObjects(void) { FIXME("\n"); return S_OK; } /*********************************************************************** * CoAddRefServerProcess [OLE32.@] */ ULONG WINAPI CoAddRefServerProcess(void) { FIXME("\n"); return 2; } /*********************************************************************** * CoReleaseServerProcess [OLE32.@] */ ULONG WINAPI CoReleaseServerProcess(void) { FIXME("\n"); return 1; }