-
Notifications
You must be signed in to change notification settings - Fork 47
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Register an encoder for writing the PrivateKeyInfo as PEM. The encoder writes the pcsk11-uri to the PEM file. The file has the following structure: SEQUENCE { ASN1_VISIBLESTRING desc; // set to "PKCS#11 Provider URI v1.0" ASN1_UTF8STRING uri; // pkcs11-uri as produced by p11prov_key_to_uri() } The feature has to be enabled explicitly by setting pkcs11-module-encode-provider-uri-to-pem. Signed-off-by: Florian Wernli <[email protected]>
- Loading branch information
Showing
17 changed files
with
785 additions
and
8 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -30,6 +30,7 @@ jobs: | |
thkdf | ||
toaepsha2 | ||
top_state | ||
tpem_encoder | ||
tpubkey | ||
trand | ||
trsapss | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,230 @@ | ||
/* Copyright (C) 2023 Simo Sorce <[email protected]> | ||
SPDX-License-Identifier: Apache-2.0 | ||
*/ | ||
|
||
#include "provider.h" | ||
#include "decoder.h" | ||
#include "store.h" | ||
#include "util.h" | ||
#include "pk11_uri.h" | ||
#include <openssl/asn1t.h> | ||
#include <openssl/bio.h> | ||
|
||
typedef struct p11prov_decoder_ctx { | ||
P11PROV_CTX *provctx; | ||
} P11PROV_DECODER_CTX; | ||
|
||
static void *p11prov_decoder_newctx(void *provctx) | ||
{ | ||
P11PROV_DECODER_CTX *dctx; | ||
dctx = OPENSSL_zalloc(sizeof(P11PROV_DECODER_CTX)); | ||
if (!dctx) { | ||
return NULL; | ||
} | ||
|
||
dctx->provctx = provctx; | ||
return dctx; | ||
} | ||
|
||
static void p11prov_decoder_freectx(void *ctx) | ||
{ | ||
OPENSSL_clear_free(ctx, sizeof(P11PROV_DECODER_CTX)); | ||
} | ||
|
||
static int obj_desc_verify(P11PROV_PK11_URI *obj) | ||
{ | ||
const char *desc = NULL; | ||
int desc_len; | ||
desc = (const char *)ASN1_STRING_get0_data(obj->desc); | ||
desc_len = ASN1_STRING_length(obj->desc); | ||
if (!desc || desc_len <= 0) { | ||
P11PROV_debug("Failed to get description"); | ||
return RET_OSSL_ERR; | ||
} | ||
|
||
if (desc_len != (sizeof(P11PROV_DESCS_URI_FILE) - 1) | ||
|| 0 != strncmp(desc, P11PROV_DESCS_URI_FILE, desc_len)) { | ||
P11PROV_debug("Description string does not match"); | ||
return RET_OSSL_ERR; | ||
} | ||
return RET_OSSL_OK; | ||
} | ||
|
||
static char *obj_uri_get1(P11PROV_PK11_URI *obj) | ||
{ | ||
const unsigned char *uri = ASN1_STRING_get0_data(obj->uri); | ||
int uri_len = ASN1_STRING_length(obj->uri); | ||
if (!uri || uri_len <= 0) { | ||
P11PROV_debug("Failed to get URI"); | ||
return NULL; | ||
} | ||
return p11prov_alloc_sprintf(uri_len, "%*s", uri_len, uri); | ||
} | ||
|
||
struct desired_data_type_cbdata { | ||
const char *desired_data_type; | ||
OSSL_CALLBACK *cb; | ||
void *cbarg; | ||
}; | ||
|
||
static int filter_for_desired_data_type(const OSSL_PARAM params[], void *arg) | ||
{ | ||
struct desired_data_type_cbdata *cbdata = arg; | ||
const OSSL_PARAM *p = | ||
OSSL_PARAM_locate_const(params, OSSL_OBJECT_PARAM_DATA_TYPE); | ||
const char *data_type = NULL; | ||
|
||
if (p && OSSL_PARAM_get_utf8_string_ptr(p, &data_type) | ||
&& 0 == strcmp(cbdata->desired_data_type, data_type)) { | ||
return cbdata->cb(params, cbdata->cbarg); | ||
} | ||
|
||
return RET_OSSL_CARRY_ON_DECODING; | ||
} | ||
|
||
static int load_obj(const P11PROV_DECODER_CTX *ctx, const unsigned char *der, | ||
long der_len, struct desired_data_type_cbdata *cbdata, | ||
OSSL_PASSPHRASE_CALLBACK *pw_cb, void *pw_cbarg) | ||
{ | ||
P11PROV_PK11_URI *obj = NULL; | ||
char *uri = NULL; | ||
|
||
obj = d2i_P11PROV_PK11_URI(NULL, &der, der_len); | ||
if (!obj) { | ||
P11PROV_debug("P11 KEY DECODER d2i_P11PROV_PK11_URI failed"); | ||
goto done; | ||
} | ||
|
||
if (!obj_desc_verify(obj)) { | ||
goto done; | ||
} | ||
|
||
uri = obj_uri_get1(obj); | ||
if (!uri) { | ||
goto done; | ||
} | ||
|
||
p11prov_store_direct_fetch(ctx->provctx, uri, filter_for_desired_data_type, | ||
cbdata, pw_cb, pw_cbarg); | ||
done: | ||
OPENSSL_free(uri); | ||
P11PROV_PK11_URI_free(obj); | ||
return RET_OSSL_CARRY_ON_DECODING; | ||
} | ||
|
||
static int p11prov_der_decoder_p11prov_obj_decode( | ||
const char *desired_data_type, void *inctx, OSSL_CORE_BIO *cin, | ||
int selection, OSSL_CALLBACK *object_cb, void *object_cbarg, | ||
OSSL_PASSPHRASE_CALLBACK *pw_cb, void *pw_cbarg) | ||
{ | ||
const P11PROV_DECODER_CTX *ctx = inctx; | ||
BIO *bin; | ||
unsigned char *der = NULL; | ||
long der_len; | ||
int ret = RET_OSSL_CARRY_ON_DECODING; | ||
|
||
bin = BIO_new_from_core_bio(p11prov_ctx_get_libctx(ctx->provctx), cin); | ||
if (!bin) { | ||
P11PROV_debug("P11 DECODER BIO_new_from_core_bio failed"); | ||
goto done; | ||
} | ||
|
||
der_len = BIO_get_mem_data(bin, &der); | ||
if (der_len <= 0) { | ||
P11PROV_debug("P11 DECODER BIO_get_mem_data failed"); | ||
goto done; | ||
} | ||
|
||
struct desired_data_type_cbdata cbdata = { | ||
.desired_data_type = desired_data_type, | ||
.cb = object_cb, | ||
.cbarg = object_cbarg, | ||
}; | ||
|
||
ret = load_obj(ctx, der, der_len, &cbdata, pw_cb, pw_cbarg); | ||
|
||
done: | ||
BIO_free(bin); | ||
P11PROV_debug("der decoder (carry on:%d)", ret); | ||
return ret; | ||
} | ||
|
||
static int p11prov_der_decoder_p11prov_rsa_decode( | ||
void *inctx, OSSL_CORE_BIO *cin, int selection, OSSL_CALLBACK *object_cb, | ||
void *object_cbarg, OSSL_PASSPHRASE_CALLBACK *pw_cb, void *pw_cbarg) | ||
{ | ||
return p11prov_der_decoder_p11prov_obj_decode( | ||
P11PROV_NAME_RSA, inctx, cin, selection, object_cb, object_cbarg, pw_cb, | ||
pw_cbarg); | ||
} | ||
|
||
const OSSL_DISPATCH p11prov_der_decoder_p11prov_rsa_functions[] = { | ||
DISPATCH_BASE_DECODER_ELEM(NEWCTX, newctx), | ||
DISPATCH_BASE_DECODER_ELEM(FREECTX, freectx), | ||
DISPATCH_DECODER_ELEM(DECODE, der, p11prov, rsa, decode), | ||
{ 0, NULL } | ||
}; | ||
|
||
static int p11prov_der_decoder_p11prov_ec_decode( | ||
void *inctx, OSSL_CORE_BIO *cin, int selection, OSSL_CALLBACK *object_cb, | ||
void *object_cbarg, OSSL_PASSPHRASE_CALLBACK *pw_cb, void *pw_cbarg) | ||
{ | ||
return p11prov_der_decoder_p11prov_obj_decode( | ||
P11PROV_NAME_EC, inctx, cin, selection, object_cb, object_cbarg, pw_cb, | ||
pw_cbarg); | ||
} | ||
|
||
const OSSL_DISPATCH p11prov_der_decoder_p11prov_ec_functions[] = { | ||
DISPATCH_BASE_DECODER_ELEM(NEWCTX, newctx), | ||
DISPATCH_BASE_DECODER_ELEM(FREECTX, freectx), | ||
DISPATCH_DECODER_ELEM(DECODE, der, p11prov, ec, decode), | ||
{ 0, NULL } | ||
}; | ||
|
||
static int p11prov_pem_decoder_p11prov_der_decode( | ||
void *inctx, OSSL_CORE_BIO *cin, int selection, OSSL_CALLBACK *object_cb, | ||
void *object_cbarg, OSSL_PASSPHRASE_CALLBACK *pw_cb, void *pw_cbarg) | ||
{ | ||
|
||
BIO *bin; | ||
char *pem_label; | ||
char *pem_header; | ||
unsigned char *der_data; | ||
long der_len; | ||
OSSL_PARAM params[3]; | ||
int ret = RET_OSSL_CARRY_ON_DECODING; | ||
P11PROV_DECODER_CTX *ctx = inctx; | ||
|
||
bin = BIO_new_from_core_bio(p11prov_ctx_get_libctx(ctx->provctx), cin); | ||
if (!bin) { | ||
P11PROV_debug("BIO_new_from_core_bio failed"); | ||
return RET_OSSL_CARRY_ON_DECODING; | ||
} | ||
|
||
P11PROV_debug("PEM_read_pio (fpos:%u)", BIO_tell(bin)); | ||
|
||
if (PEM_read_bio(bin, &pem_label, &pem_header, &der_data, &der_len) > 0 | ||
&& strcmp(pem_label, P11PROV_PEM_LABEL) == 0) { | ||
params[0] = OSSL_PARAM_construct_octet_string(OSSL_OBJECT_PARAM_DATA, | ||
der_data, der_len); | ||
params[1] = OSSL_PARAM_construct_utf8_string( | ||
OSSL_OBJECT_PARAM_DATA_STRUCTURE, (char *)P11PROV_DER_STRUCTURE, 0); | ||
params[2] = OSSL_PARAM_construct_end(); | ||
ret = object_cb(params, object_cbarg); | ||
} | ||
|
||
OPENSSL_free(pem_label); | ||
OPENSSL_free(pem_header); | ||
OPENSSL_free(der_data); | ||
BIO_free(bin); | ||
|
||
P11PROV_debug("pem decoder (carry on:%d)", ret); | ||
return ret; | ||
} | ||
|
||
const OSSL_DISPATCH p11prov_pem_decoder_p11prov_der_functions[] = { | ||
DISPATCH_BASE_DECODER_ELEM(NEWCTX, newctx), | ||
DISPATCH_BASE_DECODER_ELEM(FREECTX, freectx), | ||
DISPATCH_DECODER_ELEM(DECODE, pem, p11prov, der, decode), | ||
{ 0, NULL } | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,39 @@ | ||
/* Copyright (C) 2022 Simo Sorce <[email protected]> | ||
SPDX-License-Identifier: Apache-2.0 */ | ||
|
||
#ifndef _DECODER_H | ||
#define _DECODER_H | ||
|
||
#include <openssl/core.h> | ||
|
||
#define RET_OSSL_CARRY_ON_DECODING 1 | ||
#define RET_OSSL_STOP_DECODING 0 | ||
|
||
/* DECODERs */ | ||
#define DISPATCH_TEXT_DECODER_FN(type, name) \ | ||
static OSSL_FUNC_DECODER_##name##_fn p11prov_##type##_DECODER_##name##_text | ||
#define DISPATCH_TEXT_DECODER_ELEM(NAME, type, name) \ | ||
{ \ | ||
OSSL_FUNC_DECODER_##NAME, \ | ||
(void (*)(void))p11prov_##type##_DECODER_##name \ | ||
} | ||
#define DISPATCH_BASE_DECODER_FN(name) \ | ||
DECL_DISPATCH_FUNC(DECODER, p11prov_decoder_, name) | ||
#define DISPATCH_BASE_DECODER_ELEM(NAME, name) \ | ||
{ \ | ||
OSSL_FUNC_DECODER_##NAME, (void (*)(void))p11prov_decoder_##name \ | ||
} | ||
#define DISPATCH_DECODER_FN(type, structure, format, name) \ | ||
DECL_DISPATCH_FUNC(DECODER, \ | ||
p11prov_##type##_decoder_##structure##_##format, name) | ||
#define DISPATCH_DECODER_ELEM(NAME, type, structure, format, name) \ | ||
{ \ | ||
OSSL_FUNC_DECODER_##NAME, \ | ||
(void (*)( \ | ||
void))p11prov_##type##_decoder_##structure##_##format##_##name \ | ||
} | ||
extern const OSSL_DISPATCH p11prov_der_decoder_p11prov_rsa_functions[]; | ||
extern const OSSL_DISPATCH p11prov_der_decoder_p11prov_ec_functions[]; | ||
extern const OSSL_DISPATCH p11prov_pem_decoder_p11prov_der_functions[]; | ||
|
||
#endif /* _DECODER_H */ |
Oops, something went wrong.