| /* |
| * Copyright 2002 Mike McCormack for CodeWeavers |
| * Copyright 2004-2006 Juan Lang |
| * |
| * 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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA |
| * |
| * FIXME: |
| * - The concept of physical stores and locations isn't implemented. (This |
| * doesn't mean registry stores et al aren't implemented. See the PSDK for |
| * registering and enumerating physical stores and locations.) |
| * - Many flags, options and whatnot are unimplemented. |
| */ |
| |
| #include "config.h" |
| #include "wine/port.h" |
| |
| #include <assert.h> |
| #include <stdarg.h> |
| #include "windef.h" |
| #include "winbase.h" |
| #include "winnls.h" |
| #include "winreg.h" |
| #include "winuser.h" |
| #include "wincrypt.h" |
| #include "wine/debug.h" |
| #include "wine/exception.h" |
| #include "crypt32_private.h" |
| |
| WINE_DEFAULT_DEBUG_CHANNEL(crypt); |
| |
| static const WINE_CONTEXT_INTERFACE gCertInterface = { |
| (CreateContextFunc)CertCreateCertificateContext, |
| (AddContextToStoreFunc)CertAddCertificateContextToStore, |
| (AddEncodedContextToStoreFunc)CertAddEncodedCertificateToStore, |
| (EnumContextsInStoreFunc)CertEnumCertificatesInStore, |
| (EnumPropertiesFunc)CertEnumCertificateContextProperties, |
| (GetContextPropertyFunc)CertGetCertificateContextProperty, |
| (SetContextPropertyFunc)CertSetCertificateContextProperty, |
| (SerializeElementFunc)CertSerializeCertificateStoreElement, |
| (DeleteContextFunc)CertDeleteCertificateFromStore, |
| }; |
| const WINE_CONTEXT_INTERFACE *pCertInterface = &gCertInterface; |
| |
| static const WINE_CONTEXT_INTERFACE gCRLInterface = { |
| (CreateContextFunc)CertCreateCRLContext, |
| (AddContextToStoreFunc)CertAddCRLContextToStore, |
| (AddEncodedContextToStoreFunc)CertAddEncodedCRLToStore, |
| (EnumContextsInStoreFunc)CertEnumCRLsInStore, |
| (EnumPropertiesFunc)CertEnumCRLContextProperties, |
| (GetContextPropertyFunc)CertGetCRLContextProperty, |
| (SetContextPropertyFunc)CertSetCRLContextProperty, |
| (SerializeElementFunc)CertSerializeCRLStoreElement, |
| (DeleteContextFunc)CertDeleteCRLFromStore, |
| }; |
| const WINE_CONTEXT_INTERFACE *pCRLInterface = &gCRLInterface; |
| |
| static const WINE_CONTEXT_INTERFACE gCTLInterface = { |
| (CreateContextFunc)CertCreateCTLContext, |
| (AddContextToStoreFunc)CertAddCTLContextToStore, |
| (AddEncodedContextToStoreFunc)CertAddEncodedCTLToStore, |
| (EnumContextsInStoreFunc)CertEnumCTLsInStore, |
| (EnumPropertiesFunc)CertEnumCTLContextProperties, |
| (GetContextPropertyFunc)CertGetCTLContextProperty, |
| (SetContextPropertyFunc)CertSetCTLContextProperty, |
| (SerializeElementFunc)CertSerializeCTLStoreElement, |
| (DeleteContextFunc)CertDeleteCTLFromStore, |
| }; |
| const WINE_CONTEXT_INTERFACE *pCTLInterface = &gCTLInterface; |
| |
| typedef struct _WINE_MEMSTORE |
| { |
| WINECRYPT_CERTSTORE hdr; |
| CRITICAL_SECTION cs; |
| struct list certs; |
| struct list crls; |
| struct list ctls; |
| } WINE_MEMSTORE; |
| |
| void CRYPT_InitStore(WINECRYPT_CERTSTORE *store, DWORD dwFlags, CertStoreType type, const store_vtbl_t *vtbl) |
| { |
| store->ref = 1; |
| store->dwMagic = WINE_CRYPTCERTSTORE_MAGIC; |
| store->type = type; |
| store->dwOpenFlags = dwFlags; |
| store->vtbl = vtbl; |
| store->properties = NULL; |
| } |
| |
| void CRYPT_FreeStore(WINECRYPT_CERTSTORE *store) |
| { |
| if (store->properties) |
| ContextPropertyList_Free(store->properties); |
| store->dwMagic = 0; |
| CryptMemFree(store); |
| } |
| |
| BOOL WINAPI I_CertUpdateStore(HCERTSTORE store1, HCERTSTORE store2, DWORD unk0, |
| DWORD unk1) |
| { |
| static BOOL warned = FALSE; |
| const WINE_CONTEXT_INTERFACE * const interfaces[] = { pCertInterface, |
| pCRLInterface, pCTLInterface }; |
| DWORD i; |
| |
| TRACE("(%p, %p, %08x, %08x)\n", store1, store2, unk0, unk1); |
| if (!warned) |
| { |
| FIXME("semi-stub\n"); |
| warned = TRUE; |
| } |
| |
| /* Poor-man's resync: empty first store, then add everything from second |
| * store to it. |
| */ |
| for (i = 0; i < sizeof(interfaces) / sizeof(interfaces[0]); i++) |
| { |
| const void *context; |
| |
| do { |
| context = interfaces[i]->enumContextsInStore(store1, NULL); |
| if (context) |
| interfaces[i]->deleteFromStore(context); |
| } while (context); |
| do { |
| context = interfaces[i]->enumContextsInStore(store2, context); |
| if (context) |
| interfaces[i]->addContextToStore(store1, context, |
| CERT_STORE_ADD_ALWAYS, NULL); |
| } while (context); |
| } |
| return TRUE; |
| } |
| |
| static BOOL MemStore_addContext(WINE_MEMSTORE *store, struct list *list, context_t *orig_context, |
| context_t *existing, context_t **ret_context, BOOL use_link) |
| { |
| context_t *context; |
| |
| context = orig_context->vtbl->clone(orig_context, &store->hdr, use_link); |
| if (!context) |
| return FALSE; |
| |
| TRACE("adding %p\n", context); |
| EnterCriticalSection(&store->cs); |
| if (existing) { |
| context->u.entry.prev = existing->u.entry.prev; |
| context->u.entry.next = existing->u.entry.next; |
| context->u.entry.prev->next = &context->u.entry; |
| context->u.entry.next->prev = &context->u.entry; |
| list_init(&existing->u.entry); |
| if(!existing->ref) |
| Context_Release(existing); |
| }else { |
| list_add_head(list, &context->u.entry); |
| } |
| LeaveCriticalSection(&store->cs); |
| |
| if(ret_context) |
| *ret_context = context; |
| else |
| Context_Release(context); |
| return TRUE; |
| } |
| |
| static context_t *MemStore_enumContext(WINE_MEMSTORE *store, struct list *list, context_t *prev) |
| { |
| struct list *next; |
| context_t *ret; |
| |
| EnterCriticalSection(&store->cs); |
| if (prev) { |
| next = list_next(list, &prev->u.entry); |
| Context_Release(prev); |
| }else { |
| next = list_next(list, list); |
| } |
| LeaveCriticalSection(&store->cs); |
| |
| if (!next) { |
| SetLastError(CRYPT_E_NOT_FOUND); |
| return NULL; |
| } |
| |
| ret = LIST_ENTRY(next, context_t, u.entry); |
| Context_AddRef(ret); |
| return ret; |
| } |
| |
| static BOOL MemStore_deleteContext(WINE_MEMSTORE *store, context_t *context) |
| { |
| BOOL in_list = FALSE; |
| |
| EnterCriticalSection(&store->cs); |
| if (!list_empty(&context->u.entry)) { |
| list_remove(&context->u.entry); |
| list_init(&context->u.entry); |
| in_list = TRUE; |
| } |
| LeaveCriticalSection(&store->cs); |
| |
| if(in_list && !context->ref) |
| Context_Free(context); |
| return TRUE; |
| } |
| |
| static void free_contexts(struct list *list) |
| { |
| context_t *context, *next; |
| |
| LIST_FOR_EACH_ENTRY_SAFE(context, next, list, context_t, u.entry) |
| { |
| TRACE("freeing %p\n", context); |
| list_remove(&context->u.entry); |
| Context_Free(context); |
| } |
| } |
| |
| static void MemStore_releaseContext(WINECRYPT_CERTSTORE *store, context_t *context) |
| { |
| /* Free the context only if it's not in a list. Otherwise it may be reused later. */ |
| if(list_empty(&context->u.entry)) |
| Context_Free(context); |
| } |
| |
| static BOOL MemStore_addCert(WINECRYPT_CERTSTORE *store, context_t *cert, |
| context_t *toReplace, context_t **ppStoreContext, BOOL use_link) |
| { |
| WINE_MEMSTORE *ms = (WINE_MEMSTORE *)store; |
| |
| TRACE("(%p, %p, %p, %p)\n", store, cert, toReplace, ppStoreContext); |
| return MemStore_addContext(ms, &ms->certs, cert, toReplace, ppStoreContext, use_link); |
| } |
| |
| static context_t *MemStore_enumCert(WINECRYPT_CERTSTORE *store, context_t *prev) |
| { |
| WINE_MEMSTORE *ms = (WINE_MEMSTORE *)store; |
| |
| TRACE("(%p, %p)\n", store, prev); |
| |
| return MemStore_enumContext(ms, &ms->certs, prev); |
| } |
| |
| static BOOL MemStore_deleteCert(WINECRYPT_CERTSTORE *store, context_t *context) |
| { |
| WINE_MEMSTORE *ms = (WINE_MEMSTORE *)store; |
| |
| TRACE("(%p, %p)\n", store, context); |
| |
| return MemStore_deleteContext(ms, context); |
| } |
| |
| static BOOL MemStore_addCRL(WINECRYPT_CERTSTORE *store, context_t *crl, |
| context_t *toReplace, context_t **ppStoreContext, BOOL use_link) |
| { |
| WINE_MEMSTORE *ms = (WINE_MEMSTORE *)store; |
| |
| TRACE("(%p, %p, %p, %p)\n", store, crl, toReplace, ppStoreContext); |
| |
| return MemStore_addContext(ms, &ms->crls, crl, toReplace, ppStoreContext, use_link); |
| } |
| |
| static context_t *MemStore_enumCRL(WINECRYPT_CERTSTORE *store, context_t *prev) |
| { |
| WINE_MEMSTORE *ms = (WINE_MEMSTORE *)store; |
| |
| TRACE("(%p, %p)\n", store, prev); |
| |
| return MemStore_enumContext(ms, &ms->crls, prev); |
| } |
| |
| static BOOL MemStore_deleteCRL(WINECRYPT_CERTSTORE *store, context_t *context) |
| { |
| WINE_MEMSTORE *ms = (WINE_MEMSTORE *)store; |
| |
| TRACE("(%p, %p)\n", store, context); |
| |
| return MemStore_deleteContext(ms, context); |
| } |
| |
| static BOOL MemStore_addCTL(WINECRYPT_CERTSTORE *store, context_t *ctl, |
| context_t *toReplace, context_t **ppStoreContext, BOOL use_link) |
| { |
| WINE_MEMSTORE *ms = (WINE_MEMSTORE *)store; |
| |
| TRACE("(%p, %p, %p, %p)\n", store, ctl, toReplace, ppStoreContext); |
| |
| return MemStore_addContext(ms, &ms->ctls, ctl, toReplace, ppStoreContext, use_link); |
| } |
| |
| static context_t *MemStore_enumCTL(WINECRYPT_CERTSTORE *store, context_t *prev) |
| { |
| WINE_MEMSTORE *ms = (WINE_MEMSTORE *)store; |
| |
| TRACE("(%p, %p)\n", store, prev); |
| |
| return MemStore_enumContext(ms, &ms->ctls, prev); |
| } |
| |
| static BOOL MemStore_deleteCTL(WINECRYPT_CERTSTORE *store, context_t *context) |
| { |
| WINE_MEMSTORE *ms = (WINE_MEMSTORE *)store; |
| |
| TRACE("(%p, %p)\n", store, context); |
| |
| return MemStore_deleteContext(ms, context); |
| } |
| |
| static void MemStore_addref(WINECRYPT_CERTSTORE *store) |
| { |
| LONG ref = InterlockedIncrement(&store->ref); |
| TRACE("ref = %d\n", ref); |
| } |
| |
| static DWORD MemStore_release(WINECRYPT_CERTSTORE *cert_store, DWORD flags) |
| { |
| WINE_MEMSTORE *store = (WINE_MEMSTORE*)cert_store; |
| LONG ref; |
| |
| if(flags & ~CERT_CLOSE_STORE_CHECK_FLAG) |
| FIXME("Unimplemented flags %x\n", flags); |
| |
| ref = InterlockedDecrement(&store->hdr.ref); |
| TRACE("(%p) ref=%d\n", store, ref); |
| if(ref) |
| return (flags & CERT_CLOSE_STORE_CHECK_FLAG) ? CRYPT_E_PENDING_CLOSE : ERROR_SUCCESS; |
| |
| free_contexts(&store->certs); |
| free_contexts(&store->crls); |
| free_contexts(&store->ctls); |
| store->cs.DebugInfo->Spare[0] = 0; |
| DeleteCriticalSection(&store->cs); |
| CRYPT_FreeStore(&store->hdr); |
| return ERROR_SUCCESS; |
| } |
| |
| static BOOL MemStore_control(WINECRYPT_CERTSTORE *store, DWORD dwFlags, |
| DWORD dwCtrlType, void const *pvCtrlPara) |
| { |
| SetLastError(ERROR_CALL_NOT_IMPLEMENTED); |
| return FALSE; |
| } |
| |
| static const store_vtbl_t MemStoreVtbl = { |
| MemStore_addref, |
| MemStore_release, |
| MemStore_releaseContext, |
| MemStore_control, |
| { |
| MemStore_addCert, |
| MemStore_enumCert, |
| MemStore_deleteCert |
| }, { |
| MemStore_addCRL, |
| MemStore_enumCRL, |
| MemStore_deleteCRL |
| }, { |
| MemStore_addCTL, |
| MemStore_enumCTL, |
| MemStore_deleteCTL |
| } |
| }; |
| |
| static WINECRYPT_CERTSTORE *CRYPT_MemOpenStore(HCRYPTPROV hCryptProv, |
| DWORD dwFlags, const void *pvPara) |
| { |
| WINE_MEMSTORE *store; |
| |
| TRACE("(%ld, %08x, %p)\n", hCryptProv, dwFlags, pvPara); |
| |
| if (dwFlags & CERT_STORE_DELETE_FLAG) |
| { |
| SetLastError(ERROR_CALL_NOT_IMPLEMENTED); |
| store = NULL; |
| } |
| else |
| { |
| store = CryptMemAlloc(sizeof(WINE_MEMSTORE)); |
| if (store) |
| { |
| memset(store, 0, sizeof(WINE_MEMSTORE)); |
| CRYPT_InitStore(&store->hdr, dwFlags, StoreTypeMem, &MemStoreVtbl); |
| InitializeCriticalSection(&store->cs); |
| store->cs.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": ContextList.cs"); |
| list_init(&store->certs); |
| list_init(&store->crls); |
| list_init(&store->ctls); |
| /* Mem store doesn't need crypto provider, so close it */ |
| if (hCryptProv && !(dwFlags & CERT_STORE_NO_CRYPT_RELEASE_FLAG)) |
| CryptReleaseContext(hCryptProv, 0); |
| } |
| } |
| return (WINECRYPT_CERTSTORE*)store; |
| } |
| |
| static const WCHAR rootW[] = { 'R','o','o','t',0 }; |
| |
| static WINECRYPT_CERTSTORE *CRYPT_SysRegOpenStoreW(HCRYPTPROV hCryptProv, |
| DWORD dwFlags, const void *pvPara) |
| { |
| static const WCHAR fmt[] = { '%','s','\\','%','s',0 }; |
| LPCWSTR storeName = pvPara; |
| LPWSTR storePath; |
| WINECRYPT_CERTSTORE *store = NULL; |
| HKEY root; |
| LPCWSTR base; |
| |
| TRACE("(%ld, %08x, %s)\n", hCryptProv, dwFlags, |
| debugstr_w(pvPara)); |
| |
| if (!pvPara) |
| { |
| SetLastError(E_INVALIDARG); |
| return NULL; |
| } |
| /* FIXME: In Windows, the root store (even the current user location) is |
| * protected: adding to it or removing from it present a user interface, |
| * and the keys are owned by the system process, not the current user. |
| * Wine's registry doesn't implement access controls, so a similar |
| * mechanism isn't possible yet. |
| */ |
| if ((dwFlags & CERT_SYSTEM_STORE_LOCATION_MASK) == |
| CERT_SYSTEM_STORE_LOCAL_MACHINE && !lstrcmpiW(storeName, rootW)) |
| return CRYPT_RootOpenStore(hCryptProv, dwFlags); |
| |
| switch (dwFlags & CERT_SYSTEM_STORE_LOCATION_MASK) |
| { |
| case CERT_SYSTEM_STORE_LOCAL_MACHINE: |
| root = HKEY_LOCAL_MACHINE; |
| base = CERT_LOCAL_MACHINE_SYSTEM_STORE_REGPATH; |
| break; |
| case CERT_SYSTEM_STORE_CURRENT_USER: |
| root = HKEY_CURRENT_USER; |
| base = CERT_LOCAL_MACHINE_SYSTEM_STORE_REGPATH; |
| break; |
| case CERT_SYSTEM_STORE_CURRENT_SERVICE: |
| /* hklm\Software\Microsoft\Cryptography\Services\servicename\ |
| * SystemCertificates |
| */ |
| FIXME("CERT_SYSTEM_STORE_CURRENT_SERVICE, %s: stub\n", |
| debugstr_w(storeName)); |
| return NULL; |
| case CERT_SYSTEM_STORE_SERVICES: |
| /* hklm\Software\Microsoft\Cryptography\Services\servicename\ |
| * SystemCertificates |
| */ |
| FIXME("CERT_SYSTEM_STORE_SERVICES, %s: stub\n", |
| debugstr_w(storeName)); |
| return NULL; |
| case CERT_SYSTEM_STORE_USERS: |
| /* hku\user sid\Software\Microsoft\SystemCertificates */ |
| FIXME("CERT_SYSTEM_STORE_USERS, %s: stub\n", |
| debugstr_w(storeName)); |
| return NULL; |
| case CERT_SYSTEM_STORE_CURRENT_USER_GROUP_POLICY: |
| root = HKEY_CURRENT_USER; |
| base = CERT_GROUP_POLICY_SYSTEM_STORE_REGPATH; |
| break; |
| case CERT_SYSTEM_STORE_LOCAL_MACHINE_GROUP_POLICY: |
| root = HKEY_LOCAL_MACHINE; |
| base = CERT_GROUP_POLICY_SYSTEM_STORE_REGPATH; |
| break; |
| case CERT_SYSTEM_STORE_LOCAL_MACHINE_ENTERPRISE: |
| /* hklm\Software\Microsoft\EnterpriseCertificates */ |
| FIXME("CERT_SYSTEM_STORE_LOCAL_MACHINE_ENTERPRISE, %s: stub\n", |
| debugstr_w(storeName)); |
| return NULL; |
| default: |
| SetLastError(E_INVALIDARG); |
| return NULL; |
| } |
| |
| storePath = CryptMemAlloc((lstrlenW(base) + lstrlenW(storeName) + 2) * |
| sizeof(WCHAR)); |
| if (storePath) |
| { |
| LONG rc; |
| HKEY key; |
| REGSAM sam = dwFlags & CERT_STORE_READONLY_FLAG ? KEY_READ : |
| KEY_ALL_ACCESS; |
| |
| wsprintfW(storePath, fmt, base, storeName); |
| if (dwFlags & CERT_STORE_OPEN_EXISTING_FLAG) |
| rc = RegOpenKeyExW(root, storePath, 0, sam, &key); |
| else |
| { |
| DWORD disp; |
| |
| rc = RegCreateKeyExW(root, storePath, 0, NULL, 0, sam, NULL, |
| &key, &disp); |
| if (!rc && dwFlags & CERT_STORE_CREATE_NEW_FLAG && |
| disp == REG_OPENED_EXISTING_KEY) |
| { |
| RegCloseKey(key); |
| rc = ERROR_FILE_EXISTS; |
| } |
| } |
| if (!rc) |
| { |
| store = CRYPT_RegOpenStore(hCryptProv, dwFlags, key); |
| RegCloseKey(key); |
| } |
| else |
| SetLastError(rc); |
| CryptMemFree(storePath); |
| } |
| return store; |
| } |
| |
| static WINECRYPT_CERTSTORE *CRYPT_SysRegOpenStoreA(HCRYPTPROV hCryptProv, |
| DWORD dwFlags, const void *pvPara) |
| { |
| int len; |
| WINECRYPT_CERTSTORE *ret = NULL; |
| |
| TRACE("(%ld, %08x, %s)\n", hCryptProv, dwFlags, |
| debugstr_a(pvPara)); |
| |
| if (!pvPara) |
| { |
| SetLastError(ERROR_FILE_NOT_FOUND); |
| return NULL; |
| } |
| len = MultiByteToWideChar(CP_ACP, 0, pvPara, -1, NULL, 0); |
| if (len) |
| { |
| LPWSTR storeName = CryptMemAlloc(len * sizeof(WCHAR)); |
| |
| if (storeName) |
| { |
| MultiByteToWideChar(CP_ACP, 0, pvPara, -1, storeName, len); |
| ret = CRYPT_SysRegOpenStoreW(hCryptProv, dwFlags, storeName); |
| CryptMemFree(storeName); |
| } |
| } |
| return ret; |
| } |
| |
| static WINECRYPT_CERTSTORE *CRYPT_SysOpenStoreW(HCRYPTPROV hCryptProv, |
| DWORD dwFlags, const void *pvPara) |
| { |
| HCERTSTORE store = 0; |
| BOOL ret; |
| |
| TRACE("(%ld, %08x, %s)\n", hCryptProv, dwFlags, |
| debugstr_w(pvPara)); |
| |
| if (!pvPara) |
| { |
| SetLastError(ERROR_FILE_NOT_FOUND); |
| return NULL; |
| } |
| /* This returns a different error than system registry stores if the |
| * location is invalid. |
| */ |
| switch (dwFlags & CERT_SYSTEM_STORE_LOCATION_MASK) |
| { |
| case CERT_SYSTEM_STORE_LOCAL_MACHINE: |
| case CERT_SYSTEM_STORE_CURRENT_USER: |
| case CERT_SYSTEM_STORE_CURRENT_SERVICE: |
| case CERT_SYSTEM_STORE_SERVICES: |
| case CERT_SYSTEM_STORE_USERS: |
| case CERT_SYSTEM_STORE_CURRENT_USER_GROUP_POLICY: |
| case CERT_SYSTEM_STORE_LOCAL_MACHINE_GROUP_POLICY: |
| case CERT_SYSTEM_STORE_LOCAL_MACHINE_ENTERPRISE: |
| ret = TRUE; |
| break; |
| default: |
| SetLastError(ERROR_FILE_NOT_FOUND); |
| ret = FALSE; |
| } |
| if (ret) |
| { |
| HCERTSTORE regStore = CertOpenStore(CERT_STORE_PROV_SYSTEM_REGISTRY_W, |
| 0, 0, dwFlags, pvPara); |
| |
| if (regStore) |
| { |
| store = CertOpenStore(CERT_STORE_PROV_COLLECTION, 0, 0, |
| CERT_STORE_CREATE_NEW_FLAG, NULL); |
| CertAddStoreToCollection(store, regStore, |
| dwFlags & CERT_STORE_READONLY_FLAG ? 0 : |
| CERT_PHYSICAL_STORE_ADD_ENABLE_FLAG, 0); |
| CertCloseStore(regStore, 0); |
| /* CERT_SYSTEM_STORE_CURRENT_USER returns both the HKCU and HKLM |
| * stores. |
| */ |
| if ((dwFlags & CERT_SYSTEM_STORE_LOCATION_MASK) == |
| CERT_SYSTEM_STORE_CURRENT_USER) |
| { |
| dwFlags &= ~CERT_SYSTEM_STORE_CURRENT_USER; |
| dwFlags |= CERT_SYSTEM_STORE_LOCAL_MACHINE; |
| regStore = CertOpenStore(CERT_STORE_PROV_SYSTEM_REGISTRY_W, 0, |
| 0, dwFlags, pvPara); |
| if (regStore) |
| { |
| CertAddStoreToCollection(store, regStore, |
| dwFlags & CERT_STORE_READONLY_FLAG ? 0 : |
| CERT_PHYSICAL_STORE_ADD_ENABLE_FLAG, 0); |
| CertCloseStore(regStore, 0); |
| } |
| } |
| /* System store doesn't need crypto provider, so close it */ |
| if (hCryptProv && !(dwFlags & CERT_STORE_NO_CRYPT_RELEASE_FLAG)) |
| CryptReleaseContext(hCryptProv, 0); |
| } |
| } |
| return store; |
| } |
| |
| static WINECRYPT_CERTSTORE *CRYPT_SysOpenStoreA(HCRYPTPROV hCryptProv, |
| DWORD dwFlags, const void *pvPara) |
| { |
| int len; |
| WINECRYPT_CERTSTORE *ret = NULL; |
| |
| TRACE("(%ld, %08x, %s)\n", hCryptProv, dwFlags, |
| debugstr_a(pvPara)); |
| |
| if (!pvPara) |
| { |
| SetLastError(ERROR_FILE_NOT_FOUND); |
| return NULL; |
| } |
| len = MultiByteToWideChar(CP_ACP, 0, pvPara, -1, NULL, 0); |
| if (len) |
| { |
| LPWSTR storeName = CryptMemAlloc(len * sizeof(WCHAR)); |
| |
| if (storeName) |
| { |
| MultiByteToWideChar(CP_ACP, 0, pvPara, -1, storeName, len); |
| ret = CRYPT_SysOpenStoreW(hCryptProv, dwFlags, storeName); |
| CryptMemFree(storeName); |
| } |
| } |
| return ret; |
| } |
| |
| static void WINAPI CRYPT_MsgCloseStore(HCERTSTORE hCertStore, DWORD dwFlags) |
| { |
| HCRYPTMSG msg = hCertStore; |
| |
| TRACE("(%p, %08x)\n", msg, dwFlags); |
| CryptMsgClose(msg); |
| } |
| |
| static void *msgProvFuncs[] = { |
| CRYPT_MsgCloseStore, |
| }; |
| |
| static WINECRYPT_CERTSTORE *CRYPT_MsgOpenStore(HCRYPTPROV hCryptProv, |
| DWORD dwFlags, const void *pvPara) |
| { |
| WINECRYPT_CERTSTORE *store = NULL; |
| HCRYPTMSG msg = (HCRYPTMSG)pvPara; |
| WINECRYPT_CERTSTORE *memStore; |
| |
| TRACE("(%ld, %08x, %p)\n", hCryptProv, dwFlags, pvPara); |
| |
| memStore = CertOpenStore(CERT_STORE_PROV_MEMORY, 0, 0, |
| CERT_STORE_CREATE_NEW_FLAG, NULL); |
| if (memStore) |
| { |
| BOOL ret; |
| DWORD size, count, i; |
| |
| size = sizeof(count); |
| ret = CryptMsgGetParam(msg, CMSG_CERT_COUNT_PARAM, 0, &count, &size); |
| for (i = 0; ret && i < count; i++) |
| { |
| size = 0; |
| ret = CryptMsgGetParam(msg, CMSG_CERT_PARAM, i, NULL, &size); |
| if (ret) |
| { |
| LPBYTE buf = CryptMemAlloc(size); |
| |
| if (buf) |
| { |
| ret = CryptMsgGetParam(msg, CMSG_CERT_PARAM, i, buf, &size); |
| if (ret) |
| ret = CertAddEncodedCertificateToStore(memStore, |
| X509_ASN_ENCODING, buf, size, CERT_STORE_ADD_ALWAYS, |
| NULL); |
| CryptMemFree(buf); |
| } |
| } |
| } |
| size = sizeof(count); |
| ret = CryptMsgGetParam(msg, CMSG_CRL_COUNT_PARAM, 0, &count, &size); |
| for (i = 0; ret && i < count; i++) |
| { |
| size = 0; |
| ret = CryptMsgGetParam(msg, CMSG_CRL_PARAM, i, NULL, &size); |
| if (ret) |
| { |
| LPBYTE buf = CryptMemAlloc(size); |
| |
| if (buf) |
| { |
| ret = CryptMsgGetParam(msg, CMSG_CRL_PARAM, i, buf, &size); |
| if (ret) |
| ret = CertAddEncodedCRLToStore(memStore, |
| X509_ASN_ENCODING, buf, size, CERT_STORE_ADD_ALWAYS, |
| NULL); |
| CryptMemFree(buf); |
| } |
| } |
| } |
| if (ret) |
| { |
| CERT_STORE_PROV_INFO provInfo = { 0 }; |
| |
| provInfo.cbSize = sizeof(provInfo); |
| provInfo.cStoreProvFunc = sizeof(msgProvFuncs) / |
| sizeof(msgProvFuncs[0]); |
| provInfo.rgpvStoreProvFunc = msgProvFuncs; |
| provInfo.hStoreProv = CryptMsgDuplicate(msg); |
| store = CRYPT_ProvCreateStore(dwFlags, memStore, &provInfo); |
| /* Msg store doesn't need crypto provider, so close it */ |
| if (hCryptProv && !(dwFlags & CERT_STORE_NO_CRYPT_RELEASE_FLAG)) |
| CryptReleaseContext(hCryptProv, 0); |
| } |
| else |
| CertCloseStore(memStore, 0); |
| } |
| TRACE("returning %p\n", store); |
| return store; |
| } |
| |
| static WINECRYPT_CERTSTORE *CRYPT_PKCSOpenStore(HCRYPTPROV hCryptProv, |
| DWORD dwFlags, const void *pvPara) |
| { |
| HCRYPTMSG msg; |
| WINECRYPT_CERTSTORE *store = NULL; |
| const CRYPT_DATA_BLOB *data = pvPara; |
| BOOL ret; |
| DWORD msgOpenFlags = dwFlags & CERT_STORE_NO_CRYPT_RELEASE_FLAG ? 0 : |
| CMSG_CRYPT_RELEASE_CONTEXT_FLAG; |
| |
| TRACE("(%ld, %08x, %p)\n", hCryptProv, dwFlags, pvPara); |
| |
| msg = CryptMsgOpenToDecode(PKCS_7_ASN_ENCODING, msgOpenFlags, CMSG_SIGNED, |
| hCryptProv, NULL, NULL); |
| ret = CryptMsgUpdate(msg, data->pbData, data->cbData, TRUE); |
| if (!ret) |
| { |
| CryptMsgClose(msg); |
| msg = CryptMsgOpenToDecode(PKCS_7_ASN_ENCODING, msgOpenFlags, 0, |
| hCryptProv, NULL, NULL); |
| ret = CryptMsgUpdate(msg, data->pbData, data->cbData, TRUE); |
| if (ret) |
| { |
| DWORD type, size = sizeof(type); |
| |
| /* Only signed messages are allowed, check type */ |
| ret = CryptMsgGetParam(msg, CMSG_TYPE_PARAM, 0, &type, &size); |
| if (ret && type != CMSG_SIGNED) |
| { |
| SetLastError(CRYPT_E_INVALID_MSG_TYPE); |
| ret = FALSE; |
| } |
| } |
| } |
| if (ret) |
| store = CRYPT_MsgOpenStore(0, dwFlags, msg); |
| CryptMsgClose(msg); |
| TRACE("returning %p\n", store); |
| return store; |
| } |
| |
| static WINECRYPT_CERTSTORE *CRYPT_SerializedOpenStore(HCRYPTPROV hCryptProv, |
| DWORD dwFlags, const void *pvPara) |
| { |
| HCERTSTORE store; |
| const CRYPT_DATA_BLOB *data = pvPara; |
| |
| TRACE("(%ld, %08x, %p)\n", hCryptProv, dwFlags, pvPara); |
| |
| if (dwFlags & CERT_STORE_DELETE_FLAG) |
| { |
| SetLastError(ERROR_CALL_NOT_IMPLEMENTED); |
| return NULL; |
| } |
| |
| store = CertOpenStore(CERT_STORE_PROV_MEMORY, 0, 0, |
| CERT_STORE_CREATE_NEW_FLAG, NULL); |
| if (store) |
| { |
| if (!CRYPT_ReadSerializedStoreFromBlob(data, store)) |
| { |
| CertCloseStore(store, 0); |
| store = NULL; |
| } |
| } |
| TRACE("returning %p\n", store); |
| return (WINECRYPT_CERTSTORE*)store; |
| } |
| |
| static WINECRYPT_CERTSTORE *CRYPT_PhysOpenStoreW(HCRYPTPROV hCryptProv, |
| DWORD dwFlags, const void *pvPara) |
| { |
| if (dwFlags & CERT_SYSTEM_STORE_RELOCATE_FLAG) |
| FIXME("(%ld, %08x, %p): stub\n", hCryptProv, dwFlags, pvPara); |
| else |
| FIXME("(%ld, %08x, %s): stub\n", hCryptProv, dwFlags, |
| debugstr_w(pvPara)); |
| return NULL; |
| } |
| |
| HCERTSTORE WINAPI CertOpenStore(LPCSTR lpszStoreProvider, |
| DWORD dwMsgAndCertEncodingType, HCRYPTPROV_LEGACY hCryptProv, DWORD dwFlags, |
| const void* pvPara) |
| { |
| WINECRYPT_CERTSTORE *hcs; |
| StoreOpenFunc openFunc = NULL; |
| |
| TRACE("(%s, %08x, %08lx, %08x, %p)\n", debugstr_a(lpszStoreProvider), |
| dwMsgAndCertEncodingType, hCryptProv, dwFlags, pvPara); |
| |
| if (IS_INTOID(lpszStoreProvider)) |
| { |
| switch (LOWORD(lpszStoreProvider)) |
| { |
| case LOWORD(CERT_STORE_PROV_MSG): |
| openFunc = CRYPT_MsgOpenStore; |
| break; |
| case LOWORD(CERT_STORE_PROV_MEMORY): |
| openFunc = CRYPT_MemOpenStore; |
| break; |
| case LOWORD(CERT_STORE_PROV_FILE): |
| openFunc = CRYPT_FileOpenStore; |
| break; |
| case LOWORD(CERT_STORE_PROV_PKCS7): |
| openFunc = CRYPT_PKCSOpenStore; |
| break; |
| case LOWORD(CERT_STORE_PROV_SERIALIZED): |
| openFunc = CRYPT_SerializedOpenStore; |
| break; |
| case LOWORD(CERT_STORE_PROV_REG): |
| openFunc = CRYPT_RegOpenStore; |
| break; |
| case LOWORD(CERT_STORE_PROV_FILENAME_A): |
| openFunc = CRYPT_FileNameOpenStoreA; |
| break; |
| case LOWORD(CERT_STORE_PROV_FILENAME_W): |
| openFunc = CRYPT_FileNameOpenStoreW; |
| break; |
| case LOWORD(CERT_STORE_PROV_COLLECTION): |
| openFunc = CRYPT_CollectionOpenStore; |
| break; |
| case LOWORD(CERT_STORE_PROV_SYSTEM_A): |
| openFunc = CRYPT_SysOpenStoreA; |
| break; |
| case LOWORD(CERT_STORE_PROV_SYSTEM_W): |
| openFunc = CRYPT_SysOpenStoreW; |
| break; |
| case LOWORD(CERT_STORE_PROV_SYSTEM_REGISTRY_A): |
| openFunc = CRYPT_SysRegOpenStoreA; |
| break; |
| case LOWORD(CERT_STORE_PROV_SYSTEM_REGISTRY_W): |
| openFunc = CRYPT_SysRegOpenStoreW; |
| break; |
| case LOWORD(CERT_STORE_PROV_PHYSICAL_W): |
| openFunc = CRYPT_PhysOpenStoreW; |
| break; |
| default: |
| if (LOWORD(lpszStoreProvider)) |
| FIXME("unimplemented type %d\n", LOWORD(lpszStoreProvider)); |
| } |
| } |
| else if (!strcasecmp(lpszStoreProvider, sz_CERT_STORE_PROV_MEMORY)) |
| openFunc = CRYPT_MemOpenStore; |
| else if (!strcasecmp(lpszStoreProvider, sz_CERT_STORE_PROV_FILENAME_W)) |
| openFunc = CRYPT_FileOpenStore; |
| else if (!strcasecmp(lpszStoreProvider, sz_CERT_STORE_PROV_SYSTEM)) |
| openFunc = CRYPT_SysOpenStoreW; |
| else if (!strcasecmp(lpszStoreProvider, sz_CERT_STORE_PROV_PKCS7)) |
| openFunc = CRYPT_PKCSOpenStore; |
| else if (!strcasecmp(lpszStoreProvider, sz_CERT_STORE_PROV_SERIALIZED)) |
| openFunc = CRYPT_SerializedOpenStore; |
| else if (!strcasecmp(lpszStoreProvider, sz_CERT_STORE_PROV_COLLECTION)) |
| openFunc = CRYPT_CollectionOpenStore; |
| else if (!strcasecmp(lpszStoreProvider, sz_CERT_STORE_PROV_SYSTEM_REGISTRY)) |
| openFunc = CRYPT_SysRegOpenStoreW; |
| else |
| { |
| FIXME("unimplemented type %s\n", lpszStoreProvider); |
| openFunc = NULL; |
| } |
| |
| if (!openFunc) |
| hcs = CRYPT_ProvOpenStore(lpszStoreProvider, dwMsgAndCertEncodingType, |
| hCryptProv, dwFlags, pvPara); |
| else |
| hcs = openFunc(hCryptProv, dwFlags, pvPara); |
| return hcs; |
| } |
| |
| HCERTSTORE WINAPI CertOpenSystemStoreA(HCRYPTPROV_LEGACY hProv, |
| LPCSTR szSubSystemProtocol) |
| { |
| if (!szSubSystemProtocol) |
| { |
| SetLastError(E_INVALIDARG); |
| return 0; |
| } |
| return CertOpenStore(CERT_STORE_PROV_SYSTEM_A, 0, hProv, |
| CERT_SYSTEM_STORE_CURRENT_USER, szSubSystemProtocol); |
| } |
| |
| HCERTSTORE WINAPI CertOpenSystemStoreW(HCRYPTPROV_LEGACY hProv, |
| LPCWSTR szSubSystemProtocol) |
| { |
| if (!szSubSystemProtocol) |
| { |
| SetLastError(E_INVALIDARG); |
| return 0; |
| } |
| return CertOpenStore(CERT_STORE_PROV_SYSTEM_W, 0, hProv, |
| CERT_SYSTEM_STORE_CURRENT_USER, szSubSystemProtocol); |
| } |
| |
| PCCERT_CONTEXT WINAPI CertEnumCertificatesInStore(HCERTSTORE hCertStore, PCCERT_CONTEXT pPrev) |
| { |
| cert_t *prev = pPrev ? cert_from_ptr(pPrev) : NULL, *ret; |
| WINECRYPT_CERTSTORE *hcs = hCertStore; |
| |
| TRACE("(%p, %p)\n", hCertStore, pPrev); |
| if (!hCertStore) |
| ret = NULL; |
| else if (hcs->dwMagic != WINE_CRYPTCERTSTORE_MAGIC) |
| ret = NULL; |
| else |
| ret = (cert_t*)hcs->vtbl->certs.enumContext(hcs, prev ? &prev->base : NULL); |
| return ret ? &ret->ctx : NULL; |
| } |
| |
| BOOL WINAPI CertDeleteCertificateFromStore(PCCERT_CONTEXT pCertContext) |
| { |
| WINECRYPT_CERTSTORE *hcs; |
| |
| TRACE("(%p)\n", pCertContext); |
| |
| if (!pCertContext) |
| return TRUE; |
| |
| hcs = pCertContext->hCertStore; |
| |
| if (hcs->dwMagic != WINE_CRYPTCERTSTORE_MAGIC) |
| return FALSE; |
| |
| return hcs->vtbl->certs.delete(hcs, &cert_from_ptr(pCertContext)->base); |
| } |
| |
| BOOL WINAPI CertAddCRLContextToStore(HCERTSTORE hCertStore, |
| PCCRL_CONTEXT pCrlContext, DWORD dwAddDisposition, |
| PCCRL_CONTEXT* ppStoreContext) |
| { |
| WINECRYPT_CERTSTORE *store = hCertStore; |
| BOOL ret = TRUE; |
| PCCRL_CONTEXT toAdd = NULL, existing = NULL; |
| |
| TRACE("(%p, %p, %08x, %p)\n", hCertStore, pCrlContext, |
| dwAddDisposition, ppStoreContext); |
| |
| /* Weird case to pass a test */ |
| if (dwAddDisposition == 0) |
| { |
| SetLastError(STATUS_ACCESS_VIOLATION); |
| return FALSE; |
| } |
| if (dwAddDisposition != CERT_STORE_ADD_ALWAYS) |
| { |
| existing = CertFindCRLInStore(hCertStore, 0, 0, CRL_FIND_EXISTING, |
| pCrlContext, NULL); |
| } |
| |
| switch (dwAddDisposition) |
| { |
| case CERT_STORE_ADD_ALWAYS: |
| toAdd = CertDuplicateCRLContext(pCrlContext); |
| break; |
| case CERT_STORE_ADD_NEW: |
| if (existing) |
| { |
| TRACE("found matching CRL, not adding\n"); |
| SetLastError(CRYPT_E_EXISTS); |
| ret = FALSE; |
| } |
| else |
| toAdd = CertDuplicateCRLContext(pCrlContext); |
| break; |
| case CERT_STORE_ADD_NEWER: |
| if (existing) |
| { |
| LONG newer = CompareFileTime(&existing->pCrlInfo->ThisUpdate, |
| &pCrlContext->pCrlInfo->ThisUpdate); |
| |
| if (newer < 0) |
| toAdd = CertDuplicateCRLContext(pCrlContext); |
| else |
| { |
| TRACE("existing CRL is newer, not adding\n"); |
| SetLastError(CRYPT_E_EXISTS); |
| ret = FALSE; |
| } |
| } |
| else |
| toAdd = CertDuplicateCRLContext(pCrlContext); |
| break; |
| case CERT_STORE_ADD_NEWER_INHERIT_PROPERTIES: |
| if (existing) |
| { |
| LONG newer = CompareFileTime(&existing->pCrlInfo->ThisUpdate, |
| &pCrlContext->pCrlInfo->ThisUpdate); |
| |
| if (newer < 0) |
| { |
| toAdd = CertDuplicateCRLContext(pCrlContext); |
| Context_CopyProperties(toAdd, existing); |
| } |
| else |
| { |
| TRACE("existing CRL is newer, not adding\n"); |
| SetLastError(CRYPT_E_EXISTS); |
| ret = FALSE; |
| } |
| } |
| else |
| toAdd = CertDuplicateCRLContext(pCrlContext); |
| break; |
| case CERT_STORE_ADD_REPLACE_EXISTING: |
| toAdd = CertDuplicateCRLContext(pCrlContext); |
| break; |
| case CERT_STORE_ADD_REPLACE_EXISTING_INHERIT_PROPERTIES: |
| toAdd = CertDuplicateCRLContext(pCrlContext); |
| if (existing) |
| Context_CopyProperties(toAdd, existing); |
| break; |
| case CERT_STORE_ADD_USE_EXISTING: |
| if (existing) |
| { |
| Context_CopyProperties(existing, pCrlContext); |
| if (ppStoreContext) |
| *ppStoreContext = CertDuplicateCRLContext(existing); |
| } |
| else |
| toAdd = CertDuplicateCRLContext(pCrlContext); |
| break; |
| default: |
| FIXME("Unimplemented add disposition %d\n", dwAddDisposition); |
| ret = FALSE; |
| } |
| |
| if (toAdd) |
| { |
| if (store) { |
| context_t *ret_context; |
| ret = store->vtbl->crls.addContext(store, context_from_ptr(toAdd), |
| existing ? context_from_ptr(existing) : NULL, ppStoreContext ? &ret_context : NULL, FALSE); |
| if (ret && ppStoreContext) |
| *ppStoreContext = context_ptr(ret_context); |
| }else if (ppStoreContext) { |
| *ppStoreContext = CertDuplicateCRLContext(toAdd); |
| } |
| CertFreeCRLContext(toAdd); |
| } |
| if (existing) |
| CertFreeCRLContext(existing); |
| |
| TRACE("returning %d\n", ret); |
| return ret; |
| } |
| |
| BOOL WINAPI CertDeleteCRLFromStore(PCCRL_CONTEXT pCrlContext) |
| { |
| WINECRYPT_CERTSTORE *hcs; |
| BOOL ret; |
| |
| TRACE("(%p)\n", pCrlContext); |
| |
| if (!pCrlContext) |
| return TRUE; |
| |
| hcs = pCrlContext->hCertStore; |
| |
| if (hcs->dwMagic != WINE_CRYPTCERTSTORE_MAGIC) |
| return FALSE; |
| |
| ret = hcs->vtbl->crls.delete(hcs, &crl_from_ptr(pCrlContext)->base); |
| if (ret) |
| ret = CertFreeCRLContext(pCrlContext); |
| return ret; |
| } |
| |
| PCCRL_CONTEXT WINAPI CertEnumCRLsInStore(HCERTSTORE hCertStore, PCCRL_CONTEXT pPrev) |
| { |
| crl_t *ret, *prev = pPrev ? crl_from_ptr(pPrev) : NULL; |
| WINECRYPT_CERTSTORE *hcs = hCertStore; |
| |
| TRACE("(%p, %p)\n", hCertStore, pPrev); |
| if (!hCertStore) |
| ret = NULL; |
| else if (hcs->dwMagic != WINE_CRYPTCERTSTORE_MAGIC) |
| ret = NULL; |
| else |
| ret = (crl_t*)hcs->vtbl->crls.enumContext(hcs, prev ? &prev->base : NULL); |
| return ret ? &ret->ctx : NULL; |
| } |
| |
| HCERTSTORE WINAPI CertDuplicateStore(HCERTSTORE hCertStore) |
| { |
| WINECRYPT_CERTSTORE *hcs = hCertStore; |
| |
| TRACE("(%p)\n", hCertStore); |
| |
| if (hcs && hcs->dwMagic == WINE_CRYPTCERTSTORE_MAGIC) |
| hcs->vtbl->addref(hcs); |
| return hCertStore; |
| } |
| |
| BOOL WINAPI CertCloseStore(HCERTSTORE hCertStore, DWORD dwFlags) |
| { |
| WINECRYPT_CERTSTORE *hcs = hCertStore; |
| DWORD res; |
| |
| TRACE("(%p, %08x)\n", hCertStore, dwFlags); |
| |
| if( ! hCertStore ) |
| return TRUE; |
| |
| if ( hcs->dwMagic != WINE_CRYPTCERTSTORE_MAGIC ) |
| return FALSE; |
| |
| res = hcs->vtbl->release(hcs, dwFlags); |
| if (res != ERROR_SUCCESS) { |
| SetLastError(res); |
| return FALSE; |
| } |
| |
| return TRUE; |
| } |
| |
| BOOL WINAPI CertControlStore(HCERTSTORE hCertStore, DWORD dwFlags, |
| DWORD dwCtrlType, void const *pvCtrlPara) |
| { |
| WINECRYPT_CERTSTORE *hcs = hCertStore; |
| BOOL ret; |
| |
| TRACE("(%p, %08x, %d, %p)\n", hCertStore, dwFlags, dwCtrlType, |
| pvCtrlPara); |
| |
| if (!hcs) |
| ret = FALSE; |
| else if (hcs->dwMagic != WINE_CRYPTCERTSTORE_MAGIC) |
| ret = FALSE; |
| else |
| { |
| if (hcs->vtbl->control) |
| ret = hcs->vtbl->control(hcs, dwFlags, dwCtrlType, pvCtrlPara); |
| else |
| ret = TRUE; |
| } |
| return ret; |
| } |
| |
| BOOL WINAPI CertGetStoreProperty(HCERTSTORE hCertStore, DWORD dwPropId, |
| void *pvData, DWORD *pcbData) |
| { |
| WINECRYPT_CERTSTORE *store = hCertStore; |
| BOOL ret = FALSE; |
| |
| TRACE("(%p, %d, %p, %p)\n", hCertStore, dwPropId, pvData, pcbData); |
| |
| switch (dwPropId) |
| { |
| case CERT_ACCESS_STATE_PROP_ID: |
| if (!pvData) |
| { |
| *pcbData = sizeof(DWORD); |
| ret = TRUE; |
| } |
| else if (*pcbData < sizeof(DWORD)) |
| { |
| SetLastError(ERROR_MORE_DATA); |
| *pcbData = sizeof(DWORD); |
| } |
| else |
| { |
| DWORD state = 0; |
| |
| if (store->type != StoreTypeMem && |
| !(store->dwOpenFlags & CERT_STORE_READONLY_FLAG)) |
| state |= CERT_ACCESS_STATE_WRITE_PERSIST_FLAG; |
| *(DWORD *)pvData = state; |
| ret = TRUE; |
| } |
| break; |
| default: |
| if (store->properties) |
| { |
| CRYPT_DATA_BLOB blob; |
| |
| ret = ContextPropertyList_FindProperty(store->properties, dwPropId, |
| &blob); |
| if (ret) |
| { |
| if (!pvData) |
| *pcbData = blob.cbData; |
| else if (*pcbData < blob.cbData) |
| { |
| SetLastError(ERROR_MORE_DATA); |
| *pcbData = blob.cbData; |
| ret = FALSE; |
| } |
| else |
| { |
| memcpy(pvData, blob.pbData, blob.cbData); |
| *pcbData = blob.cbData; |
| } |
| } |
| else |
| SetLastError(CRYPT_E_NOT_FOUND); |
| } |
| else |
| SetLastError(CRYPT_E_NOT_FOUND); |
| } |
| return ret; |
| } |
| |
| BOOL WINAPI CertSetStoreProperty(HCERTSTORE hCertStore, DWORD dwPropId, |
| DWORD dwFlags, const void *pvData) |
| { |
| WINECRYPT_CERTSTORE *store = hCertStore; |
| BOOL ret = FALSE; |
| |
| TRACE("(%p, %d, %08x, %p)\n", hCertStore, dwPropId, dwFlags, pvData); |
| |
| if (!store->properties) |
| store->properties = ContextPropertyList_Create(); |
| switch (dwPropId) |
| { |
| case CERT_ACCESS_STATE_PROP_ID: |
| SetLastError(E_INVALIDARG); |
| break; |
| default: |
| if (pvData) |
| { |
| const CRYPT_DATA_BLOB *blob = pvData; |
| |
| ret = ContextPropertyList_SetProperty(store->properties, dwPropId, |
| blob->pbData, blob->cbData); |
| } |
| else |
| { |
| ContextPropertyList_RemoveProperty(store->properties, dwPropId); |
| ret = TRUE; |
| } |
| } |
| return ret; |
| } |
| |
| static LONG CRYPT_OpenParentStore(DWORD dwFlags, |
| void *pvSystemStoreLocationPara, HKEY *key) |
| { |
| HKEY root; |
| LPCWSTR base; |
| |
| TRACE("(%08x, %p)\n", dwFlags, pvSystemStoreLocationPara); |
| |
| switch (dwFlags & CERT_SYSTEM_STORE_LOCATION_MASK) |
| { |
| case CERT_SYSTEM_STORE_LOCAL_MACHINE: |
| root = HKEY_LOCAL_MACHINE; |
| base = CERT_LOCAL_MACHINE_SYSTEM_STORE_REGPATH; |
| break; |
| case CERT_SYSTEM_STORE_CURRENT_USER: |
| root = HKEY_CURRENT_USER; |
| base = CERT_LOCAL_MACHINE_SYSTEM_STORE_REGPATH; |
| break; |
| case CERT_SYSTEM_STORE_CURRENT_SERVICE: |
| /* hklm\Software\Microsoft\Cryptography\Services\servicename\ |
| * SystemCertificates |
| */ |
| FIXME("CERT_SYSTEM_STORE_CURRENT_SERVICE\n"); |
| return ERROR_FILE_NOT_FOUND; |
| case CERT_SYSTEM_STORE_SERVICES: |
| /* hklm\Software\Microsoft\Cryptography\Services\servicename\ |
| * SystemCertificates |
| */ |
| FIXME("CERT_SYSTEM_STORE_SERVICES\n"); |
| return ERROR_FILE_NOT_FOUND; |
| case CERT_SYSTEM_STORE_USERS: |
| /* hku\user sid\Software\Microsoft\SystemCertificates */ |
| FIXME("CERT_SYSTEM_STORE_USERS\n"); |
| return ERROR_FILE_NOT_FOUND; |
| case CERT_SYSTEM_STORE_CURRENT_USER_GROUP_POLICY: |
| root = HKEY_CURRENT_USER; |
| base = CERT_GROUP_POLICY_SYSTEM_STORE_REGPATH; |
| break; |
| case CERT_SYSTEM_STORE_LOCAL_MACHINE_GROUP_POLICY: |
| root = HKEY_LOCAL_MACHINE; |
| base = CERT_GROUP_POLICY_SYSTEM_STORE_REGPATH; |
| break; |
| case CERT_SYSTEM_STORE_LOCAL_MACHINE_ENTERPRISE: |
| /* hklm\Software\Microsoft\EnterpriseCertificates */ |
| FIXME("CERT_SYSTEM_STORE_LOCAL_MACHINE_ENTERPRISE\n"); |
| return ERROR_FILE_NOT_FOUND; |
| default: |
| return ERROR_FILE_NOT_FOUND; |
| } |
| |
| return RegOpenKeyExW(root, base, 0, KEY_READ, key); |
| } |
| |
| BOOL WINAPI CertEnumSystemStore(DWORD dwFlags, void *pvSystemStoreLocationPara, |
| void *pvArg, PFN_CERT_ENUM_SYSTEM_STORE pfnEnum) |
| { |
| BOOL ret = FALSE; |
| LONG rc; |
| HKEY key; |
| CERT_SYSTEM_STORE_INFO info = { sizeof(info) }; |
| |
| TRACE("(%08x, %p, %p, %p)\n", dwFlags, pvSystemStoreLocationPara, pvArg, |
| pfnEnum); |
| |
| rc = CRYPT_OpenParentStore(dwFlags, pvArg, &key); |
| if (!rc) |
| { |
| DWORD index = 0; |
| |
| ret = TRUE; |
| do { |
| WCHAR name[MAX_PATH]; |
| DWORD size = sizeof(name) / sizeof(name[0]); |
| |
| rc = RegEnumKeyExW(key, index++, name, &size, NULL, NULL, NULL, |
| NULL); |
| if (!rc) |
| ret = pfnEnum(name, dwFlags, &info, NULL, pvArg); |
| } while (ret && !rc); |
| if (ret && rc != ERROR_NO_MORE_ITEMS) |
| SetLastError(rc); |
| } |
| else |
| SetLastError(rc); |
| /* Include root store for the local machine location (it isn't in the |
| * registry) |
| */ |
| if (ret && (dwFlags & CERT_SYSTEM_STORE_LOCATION_MASK) == |
| CERT_SYSTEM_STORE_LOCAL_MACHINE) |
| ret = pfnEnum(rootW, dwFlags, &info, NULL, pvArg); |
| return ret; |
| } |
| |
| BOOL WINAPI CertEnumPhysicalStore(const void *pvSystemStore, DWORD dwFlags, |
| void *pvArg, PFN_CERT_ENUM_PHYSICAL_STORE pfnEnum) |
| { |
| if (dwFlags & CERT_SYSTEM_STORE_RELOCATE_FLAG) |
| FIXME("(%p, %08x, %p, %p): stub\n", pvSystemStore, dwFlags, pvArg, |
| pfnEnum); |
| else |
| FIXME("(%s, %08x, %p, %p): stub\n", debugstr_w(pvSystemStore), |
| dwFlags, pvArg, |
| pfnEnum); |
| return FALSE; |
| } |
| |
| BOOL WINAPI CertRegisterPhysicalStore(const void *pvSystemStore, DWORD dwFlags, |
| LPCWSTR pwszStoreName, PCERT_PHYSICAL_STORE_INFO pStoreInfo, void *pvReserved) |
| { |
| if (dwFlags & CERT_SYSTEM_STORE_RELOCATE_FLAG) |
| FIXME("(%p, %08x, %s, %p, %p): stub\n", pvSystemStore, dwFlags, |
| debugstr_w(pwszStoreName), pStoreInfo, pvReserved); |
| else |
| FIXME("(%s, %08x, %s, %p, %p): stub\n", debugstr_w(pvSystemStore), |
| dwFlags, debugstr_w(pwszStoreName), pStoreInfo, pvReserved); |
| return FALSE; |
| } |
| |
| static void EmptyStore_addref(WINECRYPT_CERTSTORE *store) |
| { |
| TRACE("(%p)\n", store); |
| } |
| |
| static DWORD EmptyStore_release(WINECRYPT_CERTSTORE *store, DWORD flags) |
| { |
| TRACE("(%p)\n", store); |
| return E_UNEXPECTED; |
| } |
| |
| static void EmptyStore_releaseContext(WINECRYPT_CERTSTORE *store, context_t *context) |
| { |
| Context_Free(context); |
| } |
| |
| static BOOL EmptyStore_add(WINECRYPT_CERTSTORE *store, context_t *context, |
| context_t *replace, context_t **ret_context, BOOL use_link) |
| { |
| TRACE("(%p, %p, %p, %p)\n", store, context, replace, ret_context); |
| |
| /* FIXME: We should clone the context */ |
| if(ret_context) { |
| Context_AddRef(context); |
| *ret_context = context; |
| } |
| |
| return TRUE; |
| } |
| |
| static context_t *EmptyStore_enum(WINECRYPT_CERTSTORE *store, context_t *prev) |
| { |
| TRACE("(%p, %p)\n", store, prev); |
| |
| SetLastError(CRYPT_E_NOT_FOUND); |
| return NULL; |
| } |
| |
| static BOOL EmptyStore_delete(WINECRYPT_CERTSTORE *store, context_t *context) |
| { |
| return TRUE; |
| } |
| |
| static BOOL EmptyStore_control(WINECRYPT_CERTSTORE *store, DWORD flags, DWORD ctrl_type, void const *ctrl_para) |
| { |
| TRACE("()\n"); |
| |
| SetLastError(ERROR_CALL_NOT_IMPLEMENTED); |
| return FALSE; |
| } |
| |
| static const store_vtbl_t EmptyStoreVtbl = { |
| EmptyStore_addref, |
| EmptyStore_release, |
| EmptyStore_releaseContext, |
| EmptyStore_control, |
| { |
| EmptyStore_add, |
| EmptyStore_enum, |
| EmptyStore_delete |
| }, { |
| EmptyStore_add, |
| EmptyStore_enum, |
| EmptyStore_delete |
| }, { |
| EmptyStore_add, |
| EmptyStore_enum, |
| EmptyStore_delete |
| } |
| }; |
| |
| WINECRYPT_CERTSTORE empty_store; |
| |
| void init_empty_store(void) |
| { |
| CRYPT_InitStore(&empty_store, CERT_STORE_READONLY_FLAG, StoreTypeEmpty, &EmptyStoreVtbl); |
| } |