/* * Copyright 2004-2007 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 */ #include <stdarg.h> #include <assert.h> #include "windef.h" #include "winbase.h" #include "wincrypt.h" #include "wine/debug.h" #include "crypt32_private.h" WINE_DEFAULT_DEBUG_CHANNEL(crypt); typedef struct _WINE_PROVIDERSTORE { WINECRYPT_CERTSTORE hdr; DWORD dwStoreProvFlags; WINECRYPT_CERTSTORE *memStore; HCERTSTOREPROV hStoreProv; PFN_CERT_STORE_PROV_CLOSE provCloseStore; PFN_CERT_STORE_PROV_WRITE_CERT provWriteCert; PFN_CERT_STORE_PROV_DELETE_CERT provDeleteCert; PFN_CERT_STORE_PROV_WRITE_CRL provWriteCrl; PFN_CERT_STORE_PROV_DELETE_CRL provDeleteCrl; PFN_CERT_STORE_PROV_WRITE_CTL provWriteCtl; PFN_CERT_STORE_PROV_DELETE_CTL provDeleteCtl; PFN_CERT_STORE_PROV_CONTROL provControl; } WINE_PROVIDERSTORE; static void ProvStore_addref(WINECRYPT_CERTSTORE *store) { LONG ref = InterlockedIncrement(&store->ref); TRACE("ref = %d\n", ref); } static DWORD ProvStore_release(WINECRYPT_CERTSTORE *cert_store, DWORD flags) { WINE_PROVIDERSTORE *store = (WINE_PROVIDERSTORE*)cert_store; LONG ref; if(flags) FIXME("Unimplemented flags %x\n", flags); ref = InterlockedDecrement(&store->hdr.ref); TRACE("(%p) ref=%d\n", store, ref); if(ref) return ERROR_SUCCESS; if (store->provCloseStore) store->provCloseStore(store->hStoreProv, flags); if (!(store->dwStoreProvFlags & CERT_STORE_PROV_EXTERNAL_FLAG)) store->memStore->vtbl->release(store->memStore, flags); CRYPT_FreeStore(&store->hdr); return ERROR_SUCCESS; } static void ProvStore_releaseContext(WINECRYPT_CERTSTORE *store, context_t *context) { /* As long as we don't have contexts properly stored (and hack around hCertStore in add* and enum* functions), this function should never be called. */ assert(0); } static BOOL ProvStore_addCert(WINECRYPT_CERTSTORE *store, context_t *cert, context_t *toReplace, context_t **ppStoreContext, BOOL use_link) { WINE_PROVIDERSTORE *ps = (WINE_PROVIDERSTORE*)store; BOOL ret; TRACE("(%p, %p, %p, %p)\n", store, cert, toReplace, ppStoreContext); if (toReplace) ret = ps->memStore->vtbl->certs.addContext(ps->memStore, cert, toReplace, ppStoreContext, TRUE); else { ret = TRUE; if (ps->provWriteCert) ret = ps->provWriteCert(ps->hStoreProv, context_ptr(cert), CERT_STORE_PROV_WRITE_ADD_FLAG); if (ret) ret = ps->memStore->vtbl->certs.addContext(ps->memStore, cert, NULL, ppStoreContext, TRUE); } /* dirty trick: replace the returned context's hCertStore with * store. */ if (ret && ppStoreContext) (*(cert_t**)ppStoreContext)->ctx.hCertStore = store; return ret; } static context_t *ProvStore_enumCert(WINECRYPT_CERTSTORE *store, context_t *prev) { WINE_PROVIDERSTORE *ps = (WINE_PROVIDERSTORE*)store; cert_t *ret; ret = (cert_t*)ps->memStore->vtbl->certs.enumContext(ps->memStore, prev); if (!ret) return NULL; /* same dirty trick: replace the returned context's hCertStore with * store. */ ret->ctx.hCertStore = store; return &ret->base; } static BOOL ProvStore_deleteCert(WINECRYPT_CERTSTORE *store, context_t *context) { WINE_PROVIDERSTORE *ps = (WINE_PROVIDERSTORE*)store; BOOL ret = TRUE; TRACE("(%p, %p)\n", store, context); if (ps->provDeleteCert) ret = ps->provDeleteCert(ps->hStoreProv, context_ptr(context), 0); if (ret) ret = ps->memStore->vtbl->certs.delete(ps->memStore, context); return ret; } static BOOL ProvStore_addCRL(WINECRYPT_CERTSTORE *store, context_t *crl, context_t *toReplace, context_t **ppStoreContext, BOOL use_link) { WINE_PROVIDERSTORE *ps = (WINE_PROVIDERSTORE*)store; BOOL ret; TRACE("(%p, %p, %p, %p)\n", store, crl, toReplace, ppStoreContext); if (toReplace) ret = ps->memStore->vtbl->crls.addContext(ps->memStore, crl, toReplace, ppStoreContext, TRUE); else { if (ps->hdr.dwOpenFlags & CERT_STORE_READONLY_FLAG) { SetLastError(ERROR_ACCESS_DENIED); ret = FALSE; } else { ret = TRUE; if (ps->provWriteCrl) ret = ps->provWriteCrl(ps->hStoreProv, context_ptr(crl), CERT_STORE_PROV_WRITE_ADD_FLAG); if (ret) ret = ps->memStore->vtbl->crls.addContext(ps->memStore, crl, NULL, ppStoreContext, TRUE); } } /* dirty trick: replace the returned context's hCertStore with * store. */ if (ret && ppStoreContext) (*(crl_t**)ppStoreContext)->ctx.hCertStore = store; return ret; } static context_t *ProvStore_enumCRL(WINECRYPT_CERTSTORE *store, context_t *prev) { WINE_PROVIDERSTORE *ps = (WINE_PROVIDERSTORE*)store; crl_t *ret; ret = (crl_t*)ps->memStore->vtbl->crls.enumContext(ps->memStore, prev); if (!ret) return NULL; /* same dirty trick: replace the returned context's hCertStore with * store. */ ret->ctx.hCertStore = store; return &ret->base; } static BOOL ProvStore_deleteCRL(WINECRYPT_CERTSTORE *store, context_t *crl) { WINE_PROVIDERSTORE *ps = (WINE_PROVIDERSTORE*)store; BOOL ret = TRUE; TRACE("(%p, %p)\n", store, crl); if (ps->provDeleteCrl) ret = ps->provDeleteCrl(ps->hStoreProv, context_ptr(crl), 0); if (ret) ret = ps->memStore->vtbl->crls.delete(ps->memStore, crl); return ret; } static BOOL ProvStore_addCTL(WINECRYPT_CERTSTORE *store, context_t *ctl, context_t *toReplace, context_t **ppStoreContext, BOOL use_link) { WINE_PROVIDERSTORE *ps = (WINE_PROVIDERSTORE*)store; BOOL ret; TRACE("(%p, %p, %p, %p)\n", store, ctl, toReplace, ppStoreContext); if (toReplace) ret = ps->memStore->vtbl->ctls.addContext(ps->memStore, ctl, toReplace, ppStoreContext, TRUE); else { if (ps->hdr.dwOpenFlags & CERT_STORE_READONLY_FLAG) { SetLastError(ERROR_ACCESS_DENIED); ret = FALSE; } else { ret = TRUE; if (ps->provWriteCtl) ret = ps->provWriteCtl(ps->hStoreProv, context_ptr(ctl), CERT_STORE_PROV_WRITE_ADD_FLAG); if (ret) ret = ps->memStore->vtbl->ctls.addContext(ps->memStore, ctl, NULL, ppStoreContext, TRUE); } } /* dirty trick: replace the returned context's hCertStore with * store. */ if (ret && ppStoreContext) (*(ctl_t**)ppStoreContext)->ctx.hCertStore = store; return ret; } static context_t *ProvStore_enumCTL(WINECRYPT_CERTSTORE *store, context_t *prev) { WINE_PROVIDERSTORE *ps = (WINE_PROVIDERSTORE*)store; ctl_t *ret; ret = (ctl_t*)ps->memStore->vtbl->ctls.enumContext(ps->memStore, prev); if (!ret) return NULL; /* same dirty trick: replace the returned context's hCertStore with * store. */ ret->ctx.hCertStore = store; return &ret->base; } static BOOL ProvStore_deleteCTL(WINECRYPT_CERTSTORE *store, context_t *ctl) { WINE_PROVIDERSTORE *ps = (WINE_PROVIDERSTORE*)store; BOOL ret = TRUE; TRACE("(%p, %p)\n", store, ctl); if (ps->provDeleteCtl) ret = ps->provDeleteCtl(ps->hStoreProv, context_ptr(ctl), 0); if (ret) ret = ps->memStore->vtbl->ctls.delete(ps->memStore, ctl); return ret; } static BOOL ProvStore_control(WINECRYPT_CERTSTORE *cert_store, DWORD dwFlags, DWORD dwCtrlType, void const *pvCtrlPara) { WINE_PROVIDERSTORE *store = (WINE_PROVIDERSTORE*)cert_store; BOOL ret = TRUE; TRACE("(%p, %08x, %d, %p)\n", store, dwFlags, dwCtrlType, pvCtrlPara); if (store->provControl) ret = store->provControl(store->hStoreProv, dwFlags, dwCtrlType, pvCtrlPara); return ret; } static const store_vtbl_t ProvStoreVtbl = { ProvStore_addref, ProvStore_release, ProvStore_releaseContext, ProvStore_control, { ProvStore_addCert, ProvStore_enumCert, ProvStore_deleteCert }, { ProvStore_addCRL, ProvStore_enumCRL, ProvStore_deleteCRL }, { ProvStore_addCTL, ProvStore_enumCTL, ProvStore_deleteCTL } }; WINECRYPT_CERTSTORE *CRYPT_ProvCreateStore(DWORD dwFlags, WINECRYPT_CERTSTORE *memStore, const CERT_STORE_PROV_INFO *pProvInfo) { WINE_PROVIDERSTORE *ret = CryptMemAlloc(sizeof(WINE_PROVIDERSTORE)); if (ret) { CRYPT_InitStore(&ret->hdr, dwFlags, StoreTypeProvider, &ProvStoreVtbl); ret->dwStoreProvFlags = pProvInfo->dwStoreProvFlags; if (ret->dwStoreProvFlags & CERT_STORE_PROV_EXTERNAL_FLAG) { CertCloseStore(memStore, 0); ret->memStore = NULL; } else ret->memStore = memStore; ret->hStoreProv = pProvInfo->hStoreProv; if (pProvInfo->cStoreProvFunc > CERT_STORE_PROV_CLOSE_FUNC) ret->provCloseStore = pProvInfo->rgpvStoreProvFunc[CERT_STORE_PROV_CLOSE_FUNC]; else ret->provCloseStore = NULL; if (pProvInfo->cStoreProvFunc > CERT_STORE_PROV_WRITE_CERT_FUNC) ret->provWriteCert = pProvInfo->rgpvStoreProvFunc[ CERT_STORE_PROV_WRITE_CERT_FUNC]; else ret->provWriteCert = NULL; if (pProvInfo->cStoreProvFunc > CERT_STORE_PROV_DELETE_CERT_FUNC) ret->provDeleteCert = pProvInfo->rgpvStoreProvFunc[ CERT_STORE_PROV_DELETE_CERT_FUNC]; else ret->provDeleteCert = NULL; if (pProvInfo->cStoreProvFunc > CERT_STORE_PROV_WRITE_CRL_FUNC) ret->provWriteCrl = pProvInfo->rgpvStoreProvFunc[ CERT_STORE_PROV_WRITE_CRL_FUNC]; else ret->provWriteCrl = NULL; if (pProvInfo->cStoreProvFunc > CERT_STORE_PROV_DELETE_CRL_FUNC) ret->provDeleteCrl = pProvInfo->rgpvStoreProvFunc[ CERT_STORE_PROV_DELETE_CRL_FUNC]; else ret->provDeleteCrl = NULL; if (pProvInfo->cStoreProvFunc > CERT_STORE_PROV_WRITE_CTL_FUNC) ret->provWriteCtl = pProvInfo->rgpvStoreProvFunc[ CERT_STORE_PROV_WRITE_CTL_FUNC]; else ret->provWriteCtl = NULL; if (pProvInfo->cStoreProvFunc > CERT_STORE_PROV_DELETE_CTL_FUNC) ret->provDeleteCtl = pProvInfo->rgpvStoreProvFunc[ CERT_STORE_PROV_DELETE_CTL_FUNC]; else ret->provDeleteCtl = NULL; if (pProvInfo->cStoreProvFunc > CERT_STORE_PROV_CONTROL_FUNC) ret->provControl = pProvInfo->rgpvStoreProvFunc[ CERT_STORE_PROV_CONTROL_FUNC]; else ret->provControl = NULL; } return (WINECRYPT_CERTSTORE*)ret; } WINECRYPT_CERTSTORE *CRYPT_ProvOpenStore(LPCSTR lpszStoreProvider, DWORD dwEncodingType, HCRYPTPROV hCryptProv, DWORD dwFlags, const void *pvPara) { static HCRYPTOIDFUNCSET set = NULL; PFN_CERT_DLL_OPEN_STORE_PROV_FUNC provOpenFunc; HCRYPTOIDFUNCADDR hFunc; WINECRYPT_CERTSTORE *ret = NULL; if (!set) set = CryptInitOIDFunctionSet(CRYPT_OID_OPEN_STORE_PROV_FUNC, 0); CryptGetOIDFunctionAddress(set, dwEncodingType, lpszStoreProvider, 0, (void **)&provOpenFunc, &hFunc); if (provOpenFunc) { CERT_STORE_PROV_INFO provInfo = { 0 }; provInfo.cbSize = sizeof(provInfo); if (dwFlags & CERT_STORE_DELETE_FLAG) provOpenFunc(lpszStoreProvider, dwEncodingType, hCryptProv, dwFlags, pvPara, NULL, &provInfo); else { HCERTSTORE memStore; memStore = CertOpenStore(CERT_STORE_PROV_MEMORY, 0, 0, CERT_STORE_CREATE_NEW_FLAG, NULL); if (memStore) { if (provOpenFunc(lpszStoreProvider, dwEncodingType, hCryptProv, dwFlags, pvPara, memStore, &provInfo)) ret = CRYPT_ProvCreateStore(dwFlags, memStore, &provInfo); else CertCloseStore(memStore, 0); } } CryptFreeOIDFunctionAddress(hFunc, 0); } else SetLastError(ERROR_FILE_NOT_FOUND); return ret; }