static void test_installOIDFunctionAddress(void) { BOOL ret; CRYPT_OID_FUNC_ENTRY entry = { CRYPT_DEFAULT_OID, funky }; HCRYPTOIDFUNCSET set; /* This crashes ret = CryptInstallOIDFunctionAddress(NULL, 0, NULL, 0, NULL, 0); */ /* Installing zero functions should work */ SetLastError(0xdeadbeef); ret = CryptInstallOIDFunctionAddress(NULL, 0, "CryptDllEncodeObject", 0, NULL, 0); ok(ret && GetLastError() == 0xdeadbeef, "Expected success, got %08x\n", GetLastError()); /* The function name doesn't much matter */ SetLastError(0xdeadbeef); ret = CryptInstallOIDFunctionAddress(NULL, 0, "OhSoFunky", 0, NULL, 0); ok(ret && GetLastError() == 0xdeadbeef, "Expected success, got %08x\n", GetLastError()); SetLastError(0xdeadbeef); entry.pszOID = X509_CERT; ret = CryptInstallOIDFunctionAddress(NULL, 0, "OhSoFunky", 1, &entry, 0); ok(ret && GetLastError() == 0xdeadbeef, "Expected success, got %08x\n", GetLastError()); set = CryptInitOIDFunctionSet("OhSoFunky", 0); ok(set != 0, "CryptInitOIDFunctionSet failed: %08x\n", GetLastError()); if (set) { funcY funcAddr = NULL; HCRYPTOIDFUNCADDR hFuncAddr = NULL; /* This crashes ret = CryptGetOIDFunctionAddress(set, X509_ASN_ENCODING, 0, 0, NULL, NULL); */ ret = CryptGetOIDFunctionAddress(set, X509_ASN_ENCODING, 0, 0, (void **)&funcAddr, &hFuncAddr); ok(!ret && (GetLastError() == ERROR_FILE_NOT_FOUND || GetLastError() == E_INVALIDARG /* some Win98 */), "Expected ERROR_FILE_NOT_FOUND or E_INVALIDARG, got %d\n", GetLastError()); ret = CryptGetOIDFunctionAddress(set, X509_ASN_ENCODING, X509_CERT, 0, (void **)&funcAddr, &hFuncAddr); ok(!ret && GetLastError() == ERROR_FILE_NOT_FOUND, "Expected ERROR_FILE_NOT_FOUND, got %d\n", GetLastError()); ret = CryptGetOIDFunctionAddress(set, 0, X509_CERT, 0, (void **)&funcAddr, &hFuncAddr); ok(ret, "CryptGetOIDFunctionAddress failed: %d\n", GetLastError()); if (funcAddr) { int y = funcAddr(0xabadc0da); ok(y == 0xabadc0da, "Unexpected return (%d) from function\n", y); CryptFreeOIDFunctionAddress(hFuncAddr, 0); } } }
static void test_getDefaultOIDFunctionAddress(void) { BOOL ret; HCRYPTOIDFUNCSET set; void *funcAddr; HCRYPTOIDFUNCADDR hFuncAddr; /* Crash ret = CryptGetDefaultOIDFunctionAddress(0, 0, NULL, 0, NULL, NULL); ret = CryptGetDefaultOIDFunctionAddress(0, 0, NULL, 0, &funcAddr, NULL); ret = CryptGetDefaultOIDFunctionAddress(0, 0, NULL, 0, NULL, &hFuncAddr); ret = CryptGetDefaultOIDFunctionAddress(0, 0, NULL, 0, &funcAddr, &hFuncAddr); */ set = CryptInitOIDFunctionSet("CertDllOpenStoreProv", 0); ok(set != 0, "CryptInitOIDFunctionSet failed: %d\n", GetLastError()); /* This crashes if hFuncAddr is not 0 to begin with */ hFuncAddr = 0; ret = CryptGetDefaultOIDFunctionAddress(set, 0, NULL, 0, &funcAddr, &hFuncAddr); ok(!ret && GetLastError() == ERROR_FILE_NOT_FOUND, "Expected ERROR_FILE_NOT_FOUND, got %d\n", GetLastError()); /* This fails with the normal encoding too, so built-in functions aren't * returned. */ ret = CryptGetDefaultOIDFunctionAddress(set, X509_ASN_ENCODING, NULL, 0, &funcAddr, &hFuncAddr); ok(!ret && GetLastError() == ERROR_FILE_NOT_FOUND, "Expected ERROR_FILE_NOT_FOUND, got %d\n", GetLastError()); /* Even with a registered dll, this fails (since the dll doesn't exist) */ SetLastError(0xdeadbeef); ret = CryptRegisterDefaultOIDFunction(0, "CertDllOpenStoreProv", 0, bogusDll); if (!ret && GetLastError() == ERROR_ACCESS_DENIED) skip("Need admin rights\n"); else ok(ret, "CryptRegisterDefaultOIDFunction failed: %08x\n", GetLastError()); ret = CryptGetDefaultOIDFunctionAddress(set, 0, NULL, 0, &funcAddr, &hFuncAddr); ok(!ret && GetLastError() == ERROR_FILE_NOT_FOUND, "Expected ERROR_FILE_NOT_FOUND, got %d\n", GetLastError()); CryptUnregisterDefaultOIDFunction(0, "CertDllOpenStoreProv", bogusDll); }
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; }
static void test_oidFunctionSet(void) { HCRYPTOIDFUNCSET set1, set2; BOOL ret; LPWSTR buf = NULL; DWORD size; /* This crashes set = CryptInitOIDFunctionSet(NULL, 0); */ /* The name doesn't mean much */ set1 = CryptInitOIDFunctionSet("funky", 0); ok(set1 != 0, "CryptInitOIDFunctionSet failed: %08x\n", GetLastError()); if (set1) { /* These crash ret = CryptGetDefaultOIDDllList(NULL, 0, NULL, NULL); ret = CryptGetDefaultOIDDllList(NULL, 0, NULL, &size); */ size = 0; ret = CryptGetDefaultOIDDllList(set1, 0, NULL, &size); ok(ret, "CryptGetDefaultOIDDllList failed: %08x\n", GetLastError()); if (ret) { buf = HeapAlloc(GetProcessHeap(), 0, size * sizeof(WCHAR)); if (buf) { ret = CryptGetDefaultOIDDllList(set1, 0, buf, &size); ok(ret, "CryptGetDefaultOIDDllList failed: %08x\n", GetLastError()); ok(!*buf, "Expected empty DLL list\n"); HeapFree(GetProcessHeap(), 0, buf); } } } /* MSDN says flags must be 0, but it's not checked */ set1 = CryptInitOIDFunctionSet("", 1); ok(set1 != 0, "CryptInitOIDFunctionSet failed: %08x\n", GetLastError()); set2 = CryptInitOIDFunctionSet("", 0); ok(set2 != 0, "CryptInitOIDFunctionSet failed: %08x\n", GetLastError()); /* There isn't a free function, so there must be only one set per name to * limit leaks. (I guess the sets are freed when crypt32 is unloaded.) */ ok(set1 == set2, "Expected identical sets\n"); if (set1) { /* The empty name function set used here seems to correspond to * DEFAULT. */ } /* There's no installed function for a built-in encoding. */ set1 = CryptInitOIDFunctionSet("CryptDllEncodeObject", 0); ok(set1 != 0, "CryptInitOIDFunctionSet failed: %08x\n", GetLastError()); if (set1) { void *funcAddr; HCRYPTOIDFUNCADDR hFuncAddr; ret = CryptGetOIDFunctionAddress(set1, X509_ASN_ENCODING, X509_CERT, 0, &funcAddr, &hFuncAddr); ok((!ret && GetLastError() == ERROR_FILE_NOT_FOUND) || broken(ret) /* some Win98 */, "Expected ERROR_FILE_NOT_FOUND, got %08x\n", GetLastError()); } }