1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
|
/*
* %CopyrightBegin%
*
* Copyright Ericsson AB 2022. All Rights Reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
* %CopyrightEnd%
*/
/*
* Purpose: CaCert fetcher
*/
#include <erl_nif.h>
#ifdef WINVER /* Windows */
#define WIN32_LEAN_AND_MEAN
#include <windows.h>
#include <Wincrypt.h>
#pragma comment(lib, "crypt32.lib")
#endif
ERL_NIF_TERM ATOM_PKCS_7_ASN_ENCODING;
ERL_NIF_TERM ATOM_X509_ASN_ENCODING;
ERL_NIF_TERM ATOM_UNKNOWN;
ERL_NIF_TERM ATOM_OPEN_ERROR;
static ERL_NIF_TERM os_cacerts(ErlNifEnv* env, int argc, const ERL_NIF_TERM argv[]);
static int load(ErlNifEnv* env, void** priv_data, ERL_NIF_TERM load_info)
{
ATOM_UNKNOWN = enif_make_atom(env,"unknown");
ATOM_OPEN_ERROR = enif_make_atom(env, "internal_error");
ATOM_PKCS_7_ASN_ENCODING = enif_make_atom(env,"pkcs_7_asn_encoding");
ATOM_X509_ASN_ENCODING = enif_make_atom(env,"x509_asn_encoding");
return 0;
}
static int upgrade(ErlNifEnv* env, void** priv_data, void** old_priv_data,
ERL_NIF_TERM load_info)
{
return 0;
}
static void unload(ErlNifEnv* env, void* priv_data)
{
}
static ErlNifFunc nif_funcs[] =
{
{"os_cacerts", 0, os_cacerts, 0},
};
ERL_NIF_INIT(pubkey_os_cacerts, nif_funcs, load, NULL, upgrade, unload)
#ifdef WINVER
ERL_NIF_TERM os_cacerts(ErlNifEnv* env, int argc, const ERL_NIF_TERM argv[])
{
HANDLE hStoreHandle = NULL;
PCCERT_CONTEXT pCertContext = NULL;
ERL_NIF_TERM head, tail, der, enc;
unsigned char * data;
hStoreHandle = CertOpenStore(CERT_STORE_PROV_SYSTEM_W, 0, 0,
// CERT_SYSTEM_STORE_LOCAL_MACHINE,
CERT_SYSTEM_STORE_CURRENT_USER,
L"ROOT");
if (!hStoreHandle) {
return ATOM_OPEN_ERROR;
}
tail = enif_make_list(env, 0);
while((pCertContext = CertEnumCertificatesInStore(hStoreHandle,pCertContext))) {
switch (pCertContext->dwCertEncodingType) {
case X509_ASN_ENCODING:
enc = ATOM_X509_ASN_ENCODING;
break;
case PKCS_7_ASN_ENCODING:
enc = ATOM_PKCS_7_ASN_ENCODING;
break;
default:
enc = ATOM_UNKNOWN;
}
data = enif_make_new_binary(env, pCertContext->cbCertEncoded, &der);
memcpy(data, pCertContext->pbCertEncoded, pCertContext->cbCertEncoded);
head = enif_make_tuple2(env, enc, der);
tail = enif_make_list_cell(env, head, tail);
}
CertCloseStore(hStoreHandle, 0);
return tail;
}
#endif
|