The Battle for Wesnoth  1.19.22+dev
tls_root_store.cpp
Go to the documentation of this file.
1 /*
2  Copyright (C) 2021 - 2025
3  Part of the Battle for Wesnoth Project https://www.wesnoth.org/
4 
5  This program is free software; you can redistribute it and/or modify
6  it under the terms of the GNU General Public License as published by
7  the Free Software Foundation; either version 2 of the License, or
8  (at your option) any later version.
9  This program is distributed in the hope that it will be useful,
10  but WITHOUT ANY WARRANTY.
11 
12  See the COPYING file for more details.
13 */
14 
15 #include "tls_root_store.hpp"
16 
17 #include "log.hpp"
18 
19 #ifdef _WIN32
20 #include <wincrypt.h>
21 #elif defined(__APPLE__)
22 #include <TargetConditionals.h>
23 #include <Security/Security.h>
24 #elif defined(__ANDROID__)
25 #include "filesystem.hpp"
26 #endif
27 
28 static lg::log_domain log_network("network");
29 #define DBG_NW LOG_STREAM(debug, log_network)
30 #define LOG_NW LOG_STREAM(info, log_network)
31 #define WRN_NW LOG_STREAM(warn, log_network)
32 #define ERR_NW LOG_STREAM(err, log_network)
33 
34 namespace network_asio
35 {
36 
37 void load_tls_root_certs(boost::asio::ssl::context &ctx)
38 {
39 #ifdef _WIN32
40  HCERTSTORE hStore = CertOpenSystemStore(0, TEXT("ROOT"));
41  assert(hStore != NULL);
42 
43  X509_STORE *store = X509_STORE_new();
44  PCCERT_CONTEXT pContext = NULL;
45  while ((pContext = CertEnumCertificatesInStore(hStore, pContext)) != NULL) {
46  X509 *x509 = d2i_X509(NULL,
47  const_cast<const unsigned char**>(&pContext->pbCertEncoded),
48  pContext->cbCertEncoded);
49  if(x509 != NULL) {
50  X509_STORE_add_cert(store, x509);
51  X509_free(x509);
52  }
53  }
54 
55  CertFreeCertificateContext(pContext);
56  CertCloseStore(hStore, 0);
57 
58  SSL_CTX_set_cert_store(ctx.native_handle(), store);
59 #elif defined(__APPLE__) && !TARGET_OS_IPHONE
60  X509_STORE *store = X509_STORE_new();
61  CFArrayRef certs = NULL;
62  // copy all system certs
63  OSStatus os_status = SecTrustCopyAnchorCertificates(&certs);
64 
65  // check for any problems copying the certs
66  if(os_status != 0) {
67  ERR_NW << "Error enumerating certificates.";
68 
69  if (certs != NULL) {
70  CFRelease(certs);
71  }
72  return;
73  }
74 
75  for(CFIndex i = 0; i < CFArrayGetCount(certs); i++) {
76  SecCertificateRef cert = (SecCertificateRef)CFArrayGetValueAtIndex(certs, i);
77 
78  // convert the cert to DER format
79  CFDataRef der_cert = SecCertificateCopyData(cert);
80  if(!der_cert) {
81  ERR_NW << "Error getting a DER representation of a certificate.";
82  continue;
83  }
84 
85  // decode each cert to an openssl X509 object
86  const uint8_t* der_cert_ptr = CFDataGetBytePtr(der_cert);
87  X509* x509_cert = d2i_X509(NULL, &der_cert_ptr, CFDataGetLength(der_cert));
88  if(!x509_cert) {
89  ERR_NW << "Error deciding the X509 certificate.";
90  CFRelease(der_cert);
91  continue;
92  }
93 
94  // Add the X509 openssl object to the verification store
95  if(X509_STORE_add_cert(store, x509_cert) != 1) {
96  CFRelease(der_cert);
97  X509_free(x509_cert);
98  ERR_NW << "Error adding the X509 certificate to the store.";
99  continue;
100  }
101  }
102 
103  CFRelease(certs);
104  SSL_CTX_set_cert_store(ctx.native_handle(), store);
105 #elif defined(__APPLE__) && TARGET_OS_IPHONE
106  ctx.set_default_verify_paths();
107 #elif defined(__ANDROID__)
108  ctx.load_verify_file(game_config::path + "/certificates/cacert.pem");
109 #else
110  ctx.set_default_verify_paths();
111 #endif
112 }
113 
114 }
Declarations for File-IO.
std::size_t i
Definition: function.cpp:1031
Standard logging facilities (interface).
std::string path
Definition: filesystem.cpp:106
void load_tls_root_certs(boost::asio::ssl::context &ctx)
#define ERR_NW
static lg::log_domain log_network("network")