22 #include <openssl/objects.h>
23 #include <openssl/x509v3.h>
24 #include <openssl/pem.h>
25 #include <openssl/err.h>
27 #include <freerdp/config.h>
29 #include <winpr/crt.h>
30 #include <winpr/string.h>
31 #include <winpr/assert.h>
33 #include <freerdp/log.h>
35 #include "x509_utils.h"
37 #define TAG FREERDP_TAG("crypto")
39 BYTE* x509_utils_get_hash(
const X509* xcert,
const char* hash,
size_t* length)
41 UINT32 fp_len = EVP_MAX_MD_SIZE;
43 const EVP_MD* md = EVP_get_digestbyname(hash);
46 WLog_ERR(TAG,
"System does not support %s hash!", hash);
49 if (!xcert || !length)
51 WLog_ERR(TAG,
"Invalid arugments: xcert=%p, length=%p", xcert, length);
55 fp = calloc(fp_len + 1,
sizeof(BYTE));
58 WLog_ERR(TAG,
"could not allocate %" PRIuz
" bytes", fp_len);
62 if (X509_digest(xcert, md, fp, &fp_len) != 1)
65 WLog_ERR(TAG,
"certificate does not have a %s hash!", hash);
73 static char* crypto_print_name(
const X509_NAME* name)
76 BIO* outBIO = BIO_new(BIO_s_mem());
78 if (X509_NAME_print_ex(outBIO, name, 0, XN_FLAG_ONELINE) > 0)
80 UINT64 size = BIO_number_written(outBIO);
83 buffer = calloc(1, (
size_t)size + 1);
89 const int rc = BIO_read(outBIO, buffer, (
int)size);
103 char* x509_utils_get_subject(
const X509* xcert)
105 char* subject = NULL;
108 WLog_ERR(TAG,
"Invalid certificate %p", xcert);
111 subject = crypto_print_name(X509_get_subject_name(xcert));
113 WLog_WARN(TAG,
"certificate does not have a subject!");
119 static const char* general_name_type_labels[] = {
"OTHERNAME",
"EMAIL ",
"DNS ",
120 "X400 ",
"DIRNAME ",
"EDIPARTY ",
121 "URI ",
"IPADD ",
"RID " };
123 static const char* general_name_type_label(
int general_name_type)
125 if ((0 <= general_name_type) &&
126 ((
size_t)general_name_type < ARRAYSIZE(general_name_type_labels)))
128 return general_name_type_labels[general_name_type];
132 static char buffer[80] = { 0 };
133 (void)snprintf(buffer,
sizeof(buffer),
"Unknown general name type (%d)", general_name_type);
183 typedef int (*general_name_mapper_pr)(GENERAL_NAME* name,
void* data,
int index,
int count);
185 static void map_subject_alt_name(
const X509* x509,
int general_name_type,
186 general_name_mapper_pr mapper,
void* data)
189 STACK_OF(GENERAL_NAME)* gens = NULL;
190 gens = X509_get_ext_d2i(x509, NID_subject_alt_name, NULL, NULL);
197 num = sk_GENERAL_NAME_num(gens);
199 for (
int i = 0; (i < num); i++)
201 GENERAL_NAME* name = sk_GENERAL_NAME_value(gens, i);
205 if ((general_name_type == GEN_ALL) || (general_name_type == name->type))
207 if (!mapper(name, data, i, num))
215 sk_GENERAL_NAME_pop_free(gens, GENERAL_NAME_free);
237 typedef struct string_list
245 static void string_list_initialize(string_list* list)
250 list->maximum = INT_MAX;
253 static void string_list_allocate(string_list* list,
int allocate_count)
255 if (!list->strings && list->allocated == 0)
257 list->strings = calloc((
size_t)allocate_count,
sizeof(
char*));
258 list->allocated = list->strings ? allocate_count : -1;
263 static void string_list_free(string_list* list)
271 static int extract_string(GENERAL_NAME* name,
void* data,
int index,
int count)
273 string_list* list = data;
274 unsigned char* cstring = 0;
275 ASN1_STRING* str = NULL;
280 str = name->d.uniformResourceIdentifier;
284 str = name->d.dNSName;
288 str = name->d.rfc822Name;
295 if ((ASN1_STRING_to_UTF8(&cstring, str)) < 0)
297 WLog_ERR(TAG,
"ASN1_STRING_to_UTF8() failed for %s: %s",
298 general_name_type_label(name->type), ERR_error_string(ERR_get_error(), NULL));
302 string_list_allocate(list, count);
304 if (list->allocated <= 0)
306 OPENSSL_free(cstring);
310 list->strings[list->count] = (
char*)cstring;
313 if (list->count >= list->maximum)
339 typedef struct object_list
341 ASN1_OBJECT* type_id;
348 static void object_list_initialize(object_list* list)
354 list->maximum = INT_MAX;
357 static void object_list_allocate(object_list* list,
int allocate_count)
359 if (!list->strings && list->allocated == 0)
361 list->strings = calloc(allocate_count,
sizeof(list->strings[0]));
362 list->allocated = list->strings ? allocate_count : -1;
367 static char* object_string(ASN1_TYPE*
object)
370 unsigned char* utf8String = NULL;
373 length = ASN1_STRING_to_UTF8(&utf8String, object->value.asn1_string);
380 result = _strdup((
char*)utf8String);
381 OPENSSL_free(utf8String);
385 static void object_list_free(object_list* list)
390 static int extract_othername_object_as_string(GENERAL_NAME* name,
void* data,
int index,
int count)
392 object_list* list = data;
394 if (name->type != GEN_OTHERNAME)
399 if (0 != OBJ_cmp(name->d.otherName->type_id, list->type_id))
404 object_list_allocate(list, count);
406 if (list->allocated <= 0)
411 list->strings[list->count] = object_string(name->d.otherName->value);
413 if (list->strings[list->count])
418 if (list->count >= list->maximum)
426 char* x509_utils_get_email(
const X509* x509)
430 string_list_initialize(&list);
432 map_subject_alt_name(x509, GEN_EMAIL, extract_string, &list);
436 string_list_free(&list);
440 result = _strdup(list.strings[0]);
441 OPENSSL_free(list.strings[0]);
442 string_list_free(&list);
446 char* x509_utils_get_upn(
const X509* x509)
450 object_list_initialize(&list);
451 list.type_id = OBJ_nid2obj(NID_ms_upn);
453 map_subject_alt_name(x509, GEN_OTHERNAME, extract_othername_object_as_string, &list);
457 object_list_free(&list);
461 result = list.strings[0];
462 object_list_free(&list);
466 char* x509_utils_get_date(
const X509* x509, BOOL startDate)
470 const ASN1_TIME* date = startDate ? X509_get0_notBefore(x509) : X509_get0_notAfter(x509);
474 BIO* bmem = BIO_new(BIO_s_mem());
479 if (ASN1_TIME_print(bmem, date))
481 BUF_MEM* bptr = NULL;
483 BIO_get_mem_ptr(bmem, &bptr);
484 str = strndup(bptr->data, bptr->length);
493 void x509_utils_dns_names_free(
size_t count,
size_t* lengths,
char** dns_names)
499 for (
size_t i = 0; i < count; i++)
503 OPENSSL_free(dns_names[i]);
511 char** x509_utils_get_dns_names(
const X509* x509,
size_t* count,
size_t** lengths)
515 string_list_initialize(&list);
516 map_subject_alt_name(x509, GEN_DNS, extract_string, &list);
517 (*count) = list.count;
521 string_list_free(&list);
527 result = calloc(list.count,
sizeof(*result));
528 (*lengths) = calloc(list.count,
sizeof(**lengths));
530 if (!result || !(*lengths))
532 string_list_free(&list);
540 for (
int i = 0; i < list.count; i++)
542 result[i] = list.strings[i];
543 (*lengths)[i] = strlen(result[i]);
546 string_list_free(&list);
550 char* x509_utils_get_issuer(
const X509* xcert)
555 WLog_ERR(TAG,
"Invalid certificate %p", xcert);
558 issuer = crypto_print_name(X509_get_issuer_name(xcert));
560 WLog_WARN(TAG,
"certificate does not have an issuer!");
564 BOOL x509_utils_check_eku(
const X509* xcert,
int nid)
567 STACK_OF(ASN1_OBJECT)* oid_stack = NULL;
568 ASN1_OBJECT* oid = NULL;
573 oid = OBJ_nid2obj(nid);
577 oid_stack = X509_get_ext_d2i(xcert, NID_ext_key_usage, NULL, NULL);
581 if (sk_ASN1_OBJECT_find(oid_stack, oid) >= 0)
584 sk_ASN1_OBJECT_pop_free(oid_stack, ASN1_OBJECT_free);
588 void x509_utils_print_info(
const X509* xcert)
592 char* subject = NULL;
593 subject = x509_utils_get_subject(xcert);
594 issuer = x509_utils_get_issuer(xcert);
595 fp = (
char*)x509_utils_get_hash(xcert,
"sha256", NULL);
599 WLog_ERR(TAG,
"error computing fingerprint");
600 goto out_free_issuer;
603 WLog_INFO(TAG,
"Certificate details:");
604 WLog_INFO(TAG,
"\tSubject: %s", subject);
605 WLog_INFO(TAG,
"\tIssuer: %s", issuer);
606 WLog_INFO(TAG,
"\tThumbprint: %s", fp);
608 "The above X.509 certificate could not be verified, possibly because you do not have "
609 "the CA certificate in your certificate store, or the certificate has expired. "
610 "Please look at the OpenSSL documentation on how to add a private CA to the store.");
617 X509* x509_utils_from_pem(
const char* data,
size_t len, BOOL fromFile)
622 bio = BIO_new_file(data,
"rb");
628 bio = BIO_new_mem_buf(data, (
int)len);
633 WLog_ERR(TAG,
"BIO_new failed for certificate");
637 x509 = PEM_read_bio_X509(bio, NULL, NULL, 0);
640 WLog_ERR(TAG,
"PEM_read_bio_X509 returned NULL [input length %" PRIuz
"]", len);
645 static WINPR_MD_TYPE hash_nid_to_winpr(
int hash_nid)
656 return WINPR_MD_SHA1;
658 return WINPR_MD_SHA224;
660 return WINPR_MD_SHA256;
662 return WINPR_MD_SHA384;
664 return WINPR_MD_SHA512;
666 return WINPR_MD_RIPEMD160;
667 #if (OPENSSL_VERSION_NUMBER >= 0x1010101fL) && !defined(LIBRESSL_VERSION_NUMBER)
669 return WINPR_MD_SHA3_224;
671 return WINPR_MD_SHA3_256;
673 return WINPR_MD_SHA3_384;
675 return WINPR_MD_SHA3_512;
677 return WINPR_MD_SHAKE128;
679 return WINPR_MD_SHAKE256;
683 return WINPR_MD_NONE;
687 static WINPR_MD_TYPE get_rsa_pss_digest(
const X509_ALGOR* alg)
689 WINPR_MD_TYPE ret = WINPR_MD_NONE;
690 WINPR_MD_TYPE message_digest = WINPR_MD_NONE;
691 WINPR_MD_TYPE mgf1_digest = WINPR_MD_NONE;
693 const void* param_value = NULL;
694 const ASN1_STRING* sequence = NULL;
695 const unsigned char* inp = NULL;
696 RSA_PSS_PARAMS* params = NULL;
697 X509_ALGOR* mgf1_digest_alg = NULL;
701 X509_ALGOR_get0(NULL, ¶m_type, ¶m_value, alg);
706 if (param_type != V_ASN1_SEQUENCE)
708 sequence = param_value;
711 inp = ASN1_STRING_get0_data(sequence);
712 params = d2i_RSA_PSS_PARAMS(NULL, &inp, ASN1_STRING_length(sequence));
719 message_digest = WINPR_MD_SHA1;
720 if (params->hashAlgorithm != NULL)
722 const ASN1_OBJECT* obj = NULL;
723 X509_ALGOR_get0(&obj, NULL, NULL, params->hashAlgorithm);
724 message_digest = hash_nid_to_winpr(OBJ_obj2nid(obj));
725 if (message_digest == WINPR_MD_NONE)
729 mgf1_digest = WINPR_MD_SHA1;
730 if (params->maskGenAlgorithm != NULL)
732 const ASN1_OBJECT* obj = NULL;
733 int mgf_param_type = 0;
734 const void* mgf_param_value = NULL;
735 const ASN1_STRING* mgf_param_sequence = NULL;
737 X509_ALGOR_get0(&obj, &mgf_param_type, &mgf_param_value, params->maskGenAlgorithm);
738 if (OBJ_obj2nid(obj) != NID_mgf1)
742 if (mgf_param_type != V_ASN1_SEQUENCE)
744 mgf_param_sequence = mgf_param_value;
745 inp = ASN1_STRING_get0_data(mgf_param_sequence);
746 mgf1_digest_alg = d2i_X509_ALGOR(NULL, &inp, ASN1_STRING_length(mgf_param_sequence));
747 if (mgf1_digest_alg == NULL)
751 X509_ALGOR_get0(&obj, NULL, NULL, mgf1_digest_alg);
752 mgf1_digest = hash_nid_to_winpr(OBJ_obj2nid(obj));
753 if (mgf1_digest == WINPR_MD_NONE)
760 if (message_digest != mgf1_digest)
762 ret = message_digest;
765 RSA_PSS_PARAMS_free(params);
766 X509_ALGOR_free(mgf1_digest_alg);
770 WINPR_MD_TYPE x509_utils_get_signature_alg(
const X509* xcert)
774 const int nid = X509_get_signature_nid(xcert);
776 if (nid == NID_rsassaPss)
778 const X509_ALGOR* alg = NULL;
779 X509_get0_signature(NULL, &alg, xcert);
780 return get_rsa_pss_digest(alg);
784 if (OBJ_find_sigid_algs(nid, &hash_nid, NULL) != 1)
785 return WINPR_MD_NONE;
787 return hash_nid_to_winpr(hash_nid);
790 char* x509_utils_get_common_name(
const X509* xcert,
size_t* plength)
792 X509_NAME* subject_name = X509_get_subject_name(xcert);
793 if (subject_name == NULL)
796 const int index = X509_NAME_get_index_by_NID(subject_name, NID_commonName, -1);
800 const X509_NAME_ENTRY* entry = X509_NAME_get_entry(subject_name, index);
804 const ASN1_STRING* entry_data = X509_NAME_ENTRY_get_data(entry);
805 if (entry_data == NULL)
808 BYTE* common_name_raw = NULL;
809 const int length = ASN1_STRING_to_UTF8(&common_name_raw, entry_data);
814 *plength = (size_t)length;
816 char* common_name = _strdup((
char*)common_name_raw);
817 OPENSSL_free(common_name_raw);
821 static int verify_cb(
int ok, X509_STORE_CTX* csc)
826 int err = X509_STORE_CTX_get_error(csc);
827 int derr = X509_STORE_CTX_get_error_depth(csc);
828 X509* where = X509_STORE_CTX_get_current_cert(csc);
829 const char* what = X509_verify_cert_error_string(err);
830 char* name = x509_utils_get_subject(where);
832 WLog_WARN(TAG,
"Certificate verification failure '%s (%d)' at stack position %d", what, err,
834 WLog_WARN(TAG,
"%s", name);
841 BOOL x509_utils_verify(X509* xcert, STACK_OF(X509) * chain,
const char* certificate_store_path)
843 const int purposes[3] = { X509_PURPOSE_SSL_SERVER, X509_PURPOSE_SSL_CLIENT, X509_PURPOSE_ANY };
844 X509_STORE_CTX* csc = NULL;
846 X509_LOOKUP* lookup = NULL;
851 X509_STORE* cert_ctx = X509_STORE_new();
853 if (cert_ctx == NULL)
856 #if OPENSSL_VERSION_NUMBER < 0x10100000L || defined(LIBRESSL_VERSION_NUMBER)
857 OpenSSL_add_all_algorithms();
859 OPENSSL_init_crypto(OPENSSL_INIT_ADD_ALL_CIPHERS | OPENSSL_INIT_ADD_ALL_DIGESTS |
860 OPENSSL_INIT_LOAD_CONFIG,
864 if (X509_STORE_set_default_paths(cert_ctx) != 1)
867 lookup = X509_STORE_add_lookup(cert_ctx, X509_LOOKUP_hash_dir());
872 X509_LOOKUP_add_dir(lookup, NULL, X509_FILETYPE_DEFAULT);
874 if (certificate_store_path != NULL)
876 X509_LOOKUP_add_dir(lookup, certificate_store_path, X509_FILETYPE_PEM);
879 X509_STORE_set_flags(cert_ctx, 0);
881 for (
size_t i = 0; i < ARRAYSIZE(purposes); i++)
885 int purpose = purposes[i];
886 csc = X509_STORE_CTX_new();
890 if (!X509_STORE_CTX_init(csc, cert_ctx, xcert, chain))
893 X509_STORE_CTX_set_purpose(csc, purpose);
894 X509_STORE_CTX_set_verify_cb(csc, verify_cb);
896 rc = X509_verify_cert(csc);
897 err = X509_STORE_CTX_get_error(csc);
899 X509_STORE_CTX_free(csc);
905 else if (err != X509_V_ERR_INVALID_PURPOSE)
909 X509_STORE_free(cert_ctx);