|
Packit |
aea12f |
/*
|
|
Packit |
aea12f |
* Copyright (C) 2015 Nikos Mavrogiannopoulos
|
|
Packit |
aea12f |
*
|
|
Packit |
aea12f |
* Author: Nikos Mavrogiannopoulos
|
|
Packit |
aea12f |
*
|
|
Packit |
aea12f |
* This file is part of GnuTLS.
|
|
Packit |
aea12f |
*
|
|
Packit |
aea12f |
* GnuTLS is free software; you can redistribute it and/or modify it
|
|
Packit |
aea12f |
* under the terms of the GNU General Public License as published by
|
|
Packit |
aea12f |
* the Free Software Foundation; either version 3 of the License, or
|
|
Packit |
aea12f |
* (at your option) any later version.
|
|
Packit |
aea12f |
*
|
|
Packit |
aea12f |
* GnuTLS is distributed in the hope that it will be useful, but
|
|
Packit |
aea12f |
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
Packit |
aea12f |
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Packit |
aea12f |
* General Public License for more details.
|
|
Packit |
aea12f |
*
|
|
Packit |
aea12f |
* You should have received a copy of the GNU General Public License
|
|
Packit |
aea12f |
* along with GnuTLS; if not, write to the Free Software Foundation,
|
|
Packit |
aea12f |
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
Packit |
aea12f |
*/
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
/* Parts copied from GnuTLS example programs. */
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
#ifdef HAVE_CONFIG_H
|
|
Packit |
aea12f |
#include <config.h>
|
|
Packit |
aea12f |
#endif
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
#include <windows.h>
|
|
Packit |
aea12f |
#include <wincrypt.h>
|
|
Packit |
aea12f |
#include <winbase.h>
|
|
Packit |
aea12f |
#include <ncrypt.h>
|
|
Packit |
aea12f |
#include <string.h>
|
|
Packit |
aea12f |
#include <gnutls/gnutls.h>
|
|
Packit |
aea12f |
#include <gnutls/x509.h>
|
|
Packit |
aea12f |
#include <gnutls/abstract.h>
|
|
Packit |
aea12f |
#include <gnutls/system-keys.h>
|
|
Packit |
aea12f |
#include <stdio.h>
|
|
Packit |
aea12f |
#include <assert.h>
|
|
Packit |
aea12f |
#include "../cert-common.h"
|
|
Packit |
aea12f |
#include "ncrypt-int.h"
|
|
Packit |
aea12f |
#include <utils.h>
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
/* This is a dummy replacement of ncrypt. It will pretend to open a specified
|
|
Packit |
aea12f |
* key by using a hardcoded one, and perform operations using that key.
|
|
Packit |
aea12f |
*/
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
#define debug_func() fprintf(stderr, "%s: %d\n", __func__, __LINE__);
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
__declspec(dllexport)
|
|
Packit |
aea12f |
SECURITY_STATUS WINAPI NCryptDeleteKey(NCRYPT_KEY_HANDLE hKey,DWORD dwFlags)
|
|
Packit |
aea12f |
{
|
|
Packit |
aea12f |
debug_func();
|
|
Packit |
aea12f |
return 0;
|
|
Packit |
aea12f |
}
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
__declspec(dllexport)
|
|
Packit |
aea12f |
SECURITY_STATUS WINAPI NCryptOpenStorageProvider(
|
|
Packit |
aea12f |
NCRYPT_PROV_HANDLE *phProvider, LPCWSTR pszProviderName,
|
|
Packit |
aea12f |
DWORD dwFlags)
|
|
Packit |
aea12f |
{
|
|
Packit |
aea12f |
debug_func();
|
|
Packit |
aea12f |
*phProvider = 0;
|
|
Packit |
aea12f |
return 0x0000ffff;
|
|
Packit |
aea12f |
}
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
__declspec(dllexport)
|
|
Packit |
aea12f |
SECURITY_STATUS WINAPI NCryptOpenKey(
|
|
Packit |
aea12f |
NCRYPT_PROV_HANDLE hProvider, NCRYPT_KEY_HANDLE *phKey,
|
|
Packit |
aea12f |
LPCWSTR pszKeyName, DWORD dwLegacyKeySpec,
|
|
Packit |
aea12f |
DWORD dwFlags)
|
|
Packit |
aea12f |
{
|
|
Packit |
aea12f |
gnutls_privkey_t p;
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
debug_func();
|
|
Packit |
aea12f |
assert_int_nequal(gnutls_privkey_init(&p), 0);
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
assert_int_nequal(gnutls_privkey_import_x509_raw(p, &key_dat, GNUTLS_X509_FMT_PEM, NULL, 0), 0);
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
*phKey = (NCRYPT_KEY_HANDLE)p;
|
|
Packit |
aea12f |
return 1;
|
|
Packit |
aea12f |
}
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
__declspec(dllexport)
|
|
Packit |
aea12f |
SECURITY_STATUS WINAPI NCryptGetProperty(
|
|
Packit |
aea12f |
NCRYPT_HANDLE hObject, LPCWSTR pszProperty,
|
|
Packit |
aea12f |
PBYTE pbOutput, DWORD cbOutput,
|
|
Packit |
aea12f |
DWORD *pcbResult, DWORD dwFlags)
|
|
Packit |
aea12f |
{
|
|
Packit |
aea12f |
debug_func();
|
|
Packit |
aea12f |
//assert_int_nequal(pszProperty, NCRYPT_ALGORITHM_PROPERTY);
|
|
Packit |
aea12f |
assert(pbOutput!=NULL);
|
|
Packit |
aea12f |
memcpy(pbOutput, BCRYPT_RSA_ALGORITHM, sizeof(BCRYPT_RSA_ALGORITHM));
|
|
Packit |
aea12f |
return 1;
|
|
Packit |
aea12f |
}
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
__declspec(dllexport)
|
|
Packit |
aea12f |
SECURITY_STATUS WINAPI NCryptFreeObject(
|
|
Packit |
aea12f |
NCRYPT_HANDLE hObject)
|
|
Packit |
aea12f |
{
|
|
Packit |
aea12f |
debug_func();
|
|
Packit |
aea12f |
if (hObject != 0)
|
|
Packit |
aea12f |
gnutls_privkey_deinit((gnutls_privkey_t)hObject);
|
|
Packit |
aea12f |
return 1;
|
|
Packit |
aea12f |
}
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
__declspec(dllexport)
|
|
Packit |
aea12f |
SECURITY_STATUS WINAPI NCryptDecrypt(
|
|
Packit |
aea12f |
NCRYPT_KEY_HANDLE hKey, PBYTE pbInput,
|
|
Packit |
aea12f |
DWORD cbInput, VOID *pPaddingInfo,
|
|
Packit |
aea12f |
PBYTE pbOutput, DWORD cbOutput,
|
|
Packit |
aea12f |
DWORD *pcbResult, DWORD dwFlags)
|
|
Packit |
aea12f |
{
|
|
Packit |
aea12f |
gnutls_datum_t c, p;
|
|
Packit |
aea12f |
assert_int_nequal(dwFlags, NCRYPT_PAD_PKCS1_FLAG);
|
|
Packit |
aea12f |
c.data = pbInput;
|
|
Packit |
aea12f |
c.size = cbInput;
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
debug_func();
|
|
Packit |
aea12f |
if (pbOutput == NULL || cbOutput == 0) {
|
|
Packit |
aea12f |
*pcbResult = 256;
|
|
Packit |
aea12f |
return 1;
|
|
Packit |
aea12f |
}
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
assert_int_nequal(gnutls_privkey_decrypt_data((gnutls_privkey_t)hKey, 0,
|
|
Packit |
aea12f |
&c, &p), 0);
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
*pcbResult = p.size;
|
|
Packit |
aea12f |
memcpy(pbOutput, p.data, p.size);
|
|
Packit |
aea12f |
gnutls_free(p.data);
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
return 1;
|
|
Packit |
aea12f |
}
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
static int StrCmpW(const WCHAR *str1, const WCHAR *str2 )
|
|
Packit |
aea12f |
{
|
|
Packit |
aea12f |
while (*str1 && (*str1 == *str2)) { str1++; str2++; }
|
|
Packit |
aea12f |
return *str1 - *str2;
|
|
Packit |
aea12f |
}
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
__declspec(dllexport)
|
|
Packit |
aea12f |
SECURITY_STATUS WINAPI NCryptSignHash(
|
|
Packit |
aea12f |
NCRYPT_KEY_HANDLE hKey, VOID* pPaddingInfo,
|
|
Packit |
aea12f |
PBYTE pbHashValue, DWORD cbHashValue,
|
|
Packit |
aea12f |
PBYTE pbSignature, DWORD cbSignature,
|
|
Packit |
aea12f |
DWORD* pcbResult, DWORD dwFlags)
|
|
Packit |
aea12f |
{
|
|
Packit |
aea12f |
BCRYPT_PKCS1_PADDING_INFO *info;
|
|
Packit |
aea12f |
int hash = 0;
|
|
Packit |
aea12f |
gnutls_privkey_t p = (gnutls_privkey_t)hKey;
|
|
Packit |
aea12f |
gnutls_datum_t v = {pbHashValue, cbHashValue}, s;
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
debug_func();
|
|
Packit |
aea12f |
info = pPaddingInfo;
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
if (info != NULL) {
|
|
Packit |
aea12f |
if (info->pszAlgId && StrCmpW(info->pszAlgId, NCRYPT_SHA1_ALGORITHM) == 0)
|
|
Packit |
aea12f |
hash = GNUTLS_DIG_SHA1;
|
|
Packit |
aea12f |
else if (info->pszAlgId && StrCmpW(info->pszAlgId, NCRYPT_SHA256_ALGORITHM) == 0)
|
|
Packit |
aea12f |
hash = GNUTLS_DIG_SHA256;
|
|
Packit |
aea12f |
else if (info->pszAlgId != NULL) {
|
|
Packit |
aea12f |
assert(0);
|
|
Packit |
aea12f |
}
|
|
Packit |
aea12f |
}
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
if (pbSignature == NULL || cbSignature == 0) {
|
|
Packit |
aea12f |
*pcbResult = 256;
|
|
Packit |
aea12f |
return 1;
|
|
Packit |
aea12f |
}
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
assert(p!=NULL);
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
if (info == NULL || info->pszAlgId == NULL) {
|
|
Packit |
aea12f |
assert_int_nequal(gnutls_privkey_sign_hash(p, 0, GNUTLS_PRIVKEY_SIGN_FLAG_TLS1_RSA, &v, &s), 0);
|
|
Packit |
aea12f |
} else if (info != NULL) {
|
|
Packit |
aea12f |
assert_int_nequal(gnutls_privkey_sign_hash(p, hash, 0, &v, &s), 0);
|
|
Packit |
aea12f |
}
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
*pcbResult = s.size;
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
if (pbSignature) {
|
|
Packit |
aea12f |
assert(cbSignature >= s.size);
|
|
Packit |
aea12f |
memcpy(pbSignature, s.data, s.size);
|
|
Packit |
aea12f |
}
|
|
Packit |
aea12f |
gnutls_free(s.data);
|
|
Packit |
aea12f |
|
|
Packit |
aea12f |
return 1;
|
|
Packit |
aea12f |
}
|
|
Packit |
aea12f |
|