Skip to content

Commit da95b34

Browse files
graebmprateek-y
andauthored
Support PKCS#11 for mutual TLS on Unix platforms (#451)
Merging the `pkcs11` feature branch to `main`. Previous Pull Requests can be found here: - #410 - Add PKCS#11 headers - #408 - Public API first pass - #412 - Implement library load/unload - #413 - Get tests running in CI - #425 - Find private key / begin s2n integration - #428 - Finish integration with s2n - #430 - Add tests. Each test now sets up its own tokendir. - #431 - Misc fixes - #432 - Add TLS test - #434 - Handle connection failure during PKCS#11 operations - #439 - Support multiple digest types for RSA - #440 - Misc fixes - #445 - Each CKR_ return value has its own AWS error-code - #443 - Add license for PKCS#11 headers - #442 - Behavior enum controls how C_Initialize() and C_Finalize() are called. - #450 - Misc fixes Co-authored-by: Prateek Yadav <[email protected]>
1 parent bb63077 commit da95b34

25 files changed

+7290
-119
lines changed

.builder/actions/pkcs11_test_setup.py

+106
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,106 @@
1+
"""
2+
Prepare for PKCS#11 tests by configuring SoftHSM2, if it is installed.
3+
"""
4+
5+
import Builder
6+
7+
import os
8+
import re
9+
10+
11+
class Pkcs11TestSetup(Builder.Action):
12+
"""
13+
Set up this machine for running the PKCS#11 tests.
14+
If SoftHSM2 cannot be installed, the tests are skipped.
15+
16+
This action should be run in the 'pre_build_steps' or 'build_steps' stage.
17+
"""
18+
19+
def run(self, env):
20+
self.env = env
21+
22+
# total hack: don't run PKCS#11 tests when building all C libs with -DBUILD_SHARED_LIBS=ON.
23+
# here's what happens: libsofthsm2.so loads the system libcrypto.so and
24+
# s2n loads the aws-lc's libcrypto.so and really strange things start happening.
25+
# this wouldn't happen in the real world, just in our tests, so just bail out
26+
if any('BUILD_SHARED_LIBS=ON' in arg for arg in env.args.args):
27+
print("WARNING: PKCS#11 tests disabled when BUILD_SHARED_LIBS=ON due to weird libcrypto.so behavior")
28+
return
29+
30+
# try to install softhsm
31+
try:
32+
softhsm_install_acion = Builder.InstallPackages(['softhsm'])
33+
softhsm_install_acion.run(env)
34+
except:
35+
print("WARNING: softhsm could not be installed. PKCS#11 tests are disabled")
36+
return
37+
38+
softhsm_lib = self._find_softhsm_lib()
39+
if softhsm_lib is None:
40+
print("WARNING: libsofthsm2.so not found. PKCS#11 tests are disabled")
41+
return
42+
43+
# set cmake flag so PKCS#11 tests are enabled
44+
env.project.config['cmake_args'].append('-DENABLE_PKCS11_TESTS=ON')
45+
46+
# put SoftHSM config file and token directory under the build dir.
47+
softhsm2_dir = os.path.join(env.build_dir, 'softhsm2')
48+
conf_path = os.path.join(softhsm2_dir, 'softhsm2.conf')
49+
token_dir = os.path.join(softhsm2_dir, 'tokens')
50+
env.shell.mkdir(token_dir)
51+
self._setenv('SOFTHSM2_CONF', conf_path)
52+
with open(conf_path, 'w') as conf_file:
53+
conf_file.write(f"directories.tokendir = {token_dir}\n")
54+
55+
# print SoftHSM version
56+
self._exec_softhsm2_util('--version')
57+
58+
# sanity check SoftHSM is working
59+
self._exec_softhsm2_util('--show-slots')
60+
61+
# set env vars for tests
62+
self._setenv('TEST_PKCS11_LIB', softhsm_lib)
63+
self._setenv('TEST_PKCS11_TOKEN_DIR', token_dir)
64+
65+
def _find_softhsm_lib(self):
66+
"""Return path to SoftHSM2 shared lib, or None if not found"""
67+
68+
# note: not using `ldconfig --print-cache` to find it because
69+
# some installers put it in weird places where ldconfig doesn't look
70+
# (like in a subfolder under lib/)
71+
72+
for lib_dir in ['lib64', 'lib']: # search lib64 before lib
73+
for base_dir in ['/usr/local', '/usr', '/',]:
74+
search_dir = os.path.join(base_dir, lib_dir)
75+
for root, dirs, files in os.walk(search_dir):
76+
for file_name in files:
77+
if 'libsofthsm2.so' in file_name:
78+
return os.path.join(root, file_name)
79+
return None
80+
81+
82+
def _exec_softhsm2_util(self, *args, **kwargs):
83+
if not 'check' in kwargs:
84+
kwargs['check'] = True
85+
86+
result = self.env.shell.exec('softhsm2-util', *args, **kwargs)
87+
88+
# older versions of softhsm2-util (2.1.0 is a known offender)
89+
# return error code 0 and print the help if invalid args are passed.
90+
# This should be an error.
91+
#
92+
# invalid args can happen because newer versions of softhsm2-util
93+
# support more args than older versions, so what works on your
94+
# machine might not work on some ancient docker image.
95+
if 'Usage: softhsm2-util' in result.output:
96+
raise Exception('softhsm2-util failed')
97+
98+
return result
99+
100+
def _setenv(self, var, value):
101+
"""
102+
Set environment variable now,
103+
and ensure the environment variable is set again when tests run
104+
"""
105+
self.env.shell.setenv(var, value)
106+
self.env.project.config['test_env'][var] = value

.github/workflows/ci.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ on:
77
- '!main'
88

99
env:
10-
BUILDER_VERSION: v0.8.27
10+
BUILDER_VERSION: v0.8.31
1111
BUILDER_SOURCE: releases
1212
BUILDER_HOST: https://d19elf31gohf1l.cloudfront.net
1313
PACKAGE_NAME: aws-c-io

CMakeLists.txt

+9-5
Original file line numberDiff line numberDiff line change
@@ -147,17 +147,21 @@ elseif (CMAKE_SYSTEM_NAME STREQUAL "FreeBSD" OR CMAKE_SYSTEM_NAME STREQUAL "NetB
147147

148148
endif()
149149

150-
if (NOT BYO_CRYPTO AND USE_S2N)
150+
if (BYO_CRYPTO)
151+
set(USE_S2N OFF)
152+
153+
if (APPLE OR WIN32)
154+
message(FATAL_ERROR "BYO_CRYPTO is only for use with unix systems. It cannot be used on your current platform target")
155+
endif()
156+
endif()
157+
158+
if (USE_S2N)
151159
file(GLOB AWS_IO_TLS_SRC
152160
"source/s2n/*.c"
153161
)
154162
aws_use_package(s2n)
155163
endif()
156164

157-
if (BYO_CRYPTO AND (APPLE OR WIN32))
158-
message(FATAL_ERROR "BYO_CRYPTO is only for use with unix systems. It cannot be used on your current platform target")
159-
endif()
160-
161165
file(GLOB IO_HEADERS
162166
${AWS_IO_HEADERS}
163167
${AWS_IO_OS_HEADERS}

PKCS11.md

+55
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,55 @@
1+
# PKCS#11 tests
2+
3+
To run the PKCS#11 tests, configure cmake with: `-DENABLE_PKCS11_TESTS=ON`
4+
5+
and set the following environment variables:
6+
7+
```
8+
TEST_PKCS11_LIB = <path-to-shared-lib>
9+
TEST_PKCS11_TOKEN_DIR = <path-to-softhsm-token-dir>
10+
```
11+
TEST_PKCS11_LIB is used by the tests to peform pkcs11 operations.
12+
13+
TEST_PKCS11_TOKEN_DIR is used by the tests to clear the softhsm tokens before a test begins. This is achieved by cleaning the token directory <b>NOTE: Any tokens created outside the tests will be cleaned up along with all the objects/keys on it as part of the tests.</b>
14+
15+
16+
## The suggested way to set up your machine
17+
1) Install [SoftHSM2](https://www.opendnssec.org/softhsm/) via brew / apt / apt-get / yum:
18+
```
19+
> apt install softhsm
20+
```
21+
22+
Check that it's working:
23+
```
24+
> softhsm2-util --show-slots
25+
```
26+
27+
If this spits out an error message, create a config file:
28+
* Default location: `~/.config/softhsm2/softhsm2.conf`
29+
* This file must specify token dir, default value is:
30+
```
31+
directories.tokendir = /usr/local/var/lib/softhsm/tokens/
32+
```
33+
34+
2) Set env vars like so:
35+
```
36+
TEST_PKCS11_LIB = <path to libsofthsm2.so>
37+
TEST_PKCS11_TOKEN_DIR = /usr/local/var/lib/softhsm/tokens/
38+
```
39+
40+
41+
3) [Example to import your keys, Not used by tests] Create token and private key
42+
43+
You can use any values for the labels, pin, key, cert, CA etc.
44+
Here are copy-paste friendly commands for using files available in this repo.
45+
```
46+
> softhsm2-util --init-token --free --label my-test-token --pin 0000 --so-pin 0000
47+
```
48+
49+
Note which slot the token ended up in
50+
51+
```
52+
> softhsm2-util --import tests/resources/unittests.p8 --slot <slot-with-token> --label my-test-key --id BEEFCAFE --pin 0000
53+
```
54+
<b>WARN: All tokens created outside the tests would be cleaned up as part of the tests, Use a separate token directory for running the tests if you would like to keep your tokens intact.</b>
55+

THIRD-PARTY-LICENSES.txt

+31
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
** PKCS#11 Headers; version 2.40 -- http://docs.oasis-open.org/pkcs11/pkcs11-base/v2.40/errata01/os/include/pkcs11-v2.40/
2+
3+
Copyright (c) OASIS Open 2016. All Rights Reserved.
4+
5+
All capitalized terms in the following text have the meanings assigned to them
6+
in the OASIS Intellectual Property Rights Policy (the "OASIS IPR Policy"). The
7+
full Policy may be found at the OASIS website:
8+
[http://www.oasis-open.org/policies-guidelines/ipr]
9+
10+
This document and translations of it may be copied and furnished to others, and
11+
derivative works that comment on or otherwise explain it or assist in its
12+
implementation may be prepared, copied, published, and distributed, in whole or
13+
in part, without restriction of any kind, provided that the above copyright
14+
notice and this section are included on all such copies and derivative works.
15+
However, this document itself may not be modified in any way, including by
16+
removing the copyright notice or references to OASIS, except as needed for the
17+
purpose of developing any document or deliverable produced by an OASIS
18+
Technical Committee (in which case the rules applicable to copyrights, as set
19+
forth in the OASIS IPR Policy, must be followed) or as required to translate it
20+
into languages other than English.
21+
22+
The limited permissions granted above are perpetual and will not be revoked by
23+
OASIS or its successors or assigns.
24+
25+
This document and the information contained herein is provided on an "AS IS"
26+
basis and OASIS DISCLAIMS ALL WARRANTIES, EXPRESS OR IMPLIED, INCLUDING BUT NOT
27+
LIMITED TO ANY WARRANTY THAT THE USE OF THE INFORMATION HEREIN WILL NOT
28+
INFRINGE ANY OWNERSHIP RIGHTS OR ANY IMPLIED WARRANTIES OF MERCHANTABILITY OR
29+
FITNESS FOR A PARTICULAR PURPOSE. OASIS AND ITS MEMBERS WILL NOT BE LIABLE FOR
30+
ANY DIRECT, INDIRECT, SPECIAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF ANY USE
31+
OF THIS DOCUMENT OR ANY PART THEREOF.

builder.json

+8-2
Original file line numberDiff line numberDiff line change
@@ -12,8 +12,14 @@
1212
{ "name": "aws-c-mqtt" },
1313
{ "name": "aws-c-http" }
1414
],
15+
"targets": {
16+
"linux": {
17+
"_comment": "set up SoftHSM2 for PKCS#11 tests (see: ./builder/actions/pkcs11_test_setup.py)",
18+
"+pre_build_steps": ["pkcs11-test-setup"]
19+
}
20+
},
1521
"build_env": {
16-
"LSAN_OPTIONS": "suppressions={source_dir}/tests/resources/suppressions.txt:allow_addr2line=1",
17-
"CMAKE_BUILD_PARALLEL_LEVEL": "2"
22+
"LSAN_OPTIONS": "suppressions={source_dir}/tests/resources/suppressions-lsan.txt:allow_addr2line=1",
23+
"ASAN_OPTIONS": "suppressions={source_dir}/tests/resources/suppressions-asan.txt"
1824
}
1925
}

include/aws/io/io.h

+103
Original file line numberDiff line numberDiff line change
@@ -134,6 +134,109 @@ enum aws_io_errors {
134134
AWS_IO_TLS_ALERT_NOT_GRACEFUL,
135135
AWS_IO_MAX_RETRIES_EXCEEDED,
136136
AWS_IO_RETRY_PERMISSION_DENIED,
137+
AWS_IO_TLS_DIGEST_ALGORITHM_UNSUPPORTED,
138+
AWS_IO_TLS_SIGNATURE_ALGORITHM_UNSUPPORTED,
139+
140+
AWS_ERROR_PKCS11_VERSION_UNSUPPORTED,
141+
AWS_ERROR_PKCS11_TOKEN_NOT_FOUND,
142+
AWS_ERROR_PKCS11_KEY_NOT_FOUND,
143+
AWS_ERROR_PKCS11_KEY_TYPE_UNSUPPORTED,
144+
AWS_ERROR_PKCS11_UNKNOWN_CRYPTOKI_RETURN_VALUE,
145+
146+
/* PKCS#11 "CKR_" (Cryptoki Return Value) as AWS error-codes */
147+
AWS_ERROR_PKCS11_CKR_CANCEL,
148+
AWS_ERROR_PKCS11_CKR_HOST_MEMORY,
149+
AWS_ERROR_PKCS11_CKR_SLOT_ID_INVALID,
150+
AWS_ERROR_PKCS11_CKR_GENERAL_ERROR,
151+
AWS_ERROR_PKCS11_CKR_FUNCTION_FAILED,
152+
AWS_ERROR_PKCS11_CKR_ARGUMENTS_BAD,
153+
AWS_ERROR_PKCS11_CKR_NO_EVENT,
154+
AWS_ERROR_PKCS11_CKR_NEED_TO_CREATE_THREADS,
155+
AWS_ERROR_PKCS11_CKR_CANT_LOCK,
156+
AWS_ERROR_PKCS11_CKR_ATTRIBUTE_READ_ONLY,
157+
AWS_ERROR_PKCS11_CKR_ATTRIBUTE_SENSITIVE,
158+
AWS_ERROR_PKCS11_CKR_ATTRIBUTE_TYPE_INVALID,
159+
AWS_ERROR_PKCS11_CKR_ATTRIBUTE_VALUE_INVALID,
160+
AWS_ERROR_PKCS11_CKR_ACTION_PROHIBITED,
161+
AWS_ERROR_PKCS11_CKR_DATA_INVALID,
162+
AWS_ERROR_PKCS11_CKR_DATA_LEN_RANGE,
163+
AWS_ERROR_PKCS11_CKR_DEVICE_ERROR,
164+
AWS_ERROR_PKCS11_CKR_DEVICE_MEMORY,
165+
AWS_ERROR_PKCS11_CKR_DEVICE_REMOVED,
166+
AWS_ERROR_PKCS11_CKR_ENCRYPTED_DATA_INVALID,
167+
AWS_ERROR_PKCS11_CKR_ENCRYPTED_DATA_LEN_RANGE,
168+
AWS_ERROR_PKCS11_CKR_FUNCTION_CANCELED,
169+
AWS_ERROR_PKCS11_CKR_FUNCTION_NOT_PARALLEL,
170+
AWS_ERROR_PKCS11_CKR_FUNCTION_NOT_SUPPORTED,
171+
AWS_ERROR_PKCS11_CKR_KEY_HANDLE_INVALID,
172+
AWS_ERROR_PKCS11_CKR_KEY_SIZE_RANGE,
173+
AWS_ERROR_PKCS11_CKR_KEY_TYPE_INCONSISTENT,
174+
AWS_ERROR_PKCS11_CKR_KEY_NOT_NEEDED,
175+
AWS_ERROR_PKCS11_CKR_KEY_CHANGED,
176+
AWS_ERROR_PKCS11_CKR_KEY_NEEDED,
177+
AWS_ERROR_PKCS11_CKR_KEY_INDIGESTIBLE,
178+
AWS_ERROR_PKCS11_CKR_KEY_FUNCTION_NOT_PERMITTED,
179+
AWS_ERROR_PKCS11_CKR_KEY_NOT_WRAPPABLE,
180+
AWS_ERROR_PKCS11_CKR_KEY_UNEXTRACTABLE,
181+
AWS_ERROR_PKCS11_CKR_MECHANISM_INVALID,
182+
AWS_ERROR_PKCS11_CKR_MECHANISM_PARAM_INVALID,
183+
AWS_ERROR_PKCS11_CKR_OBJECT_HANDLE_INVALID,
184+
AWS_ERROR_PKCS11_CKR_OPERATION_ACTIVE,
185+
AWS_ERROR_PKCS11_CKR_OPERATION_NOT_INITIALIZED,
186+
AWS_ERROR_PKCS11_CKR_PIN_INCORRECT,
187+
AWS_ERROR_PKCS11_CKR_PIN_INVALID,
188+
AWS_ERROR_PKCS11_CKR_PIN_LEN_RANGE,
189+
AWS_ERROR_PKCS11_CKR_PIN_EXPIRED,
190+
AWS_ERROR_PKCS11_CKR_PIN_LOCKED,
191+
AWS_ERROR_PKCS11_CKR_SESSION_CLOSED,
192+
AWS_ERROR_PKCS11_CKR_SESSION_COUNT,
193+
AWS_ERROR_PKCS11_CKR_SESSION_HANDLE_INVALID,
194+
AWS_ERROR_PKCS11_CKR_SESSION_PARALLEL_NOT_SUPPORTED,
195+
AWS_ERROR_PKCS11_CKR_SESSION_READ_ONLY,
196+
AWS_ERROR_PKCS11_CKR_SESSION_EXISTS,
197+
AWS_ERROR_PKCS11_CKR_SESSION_READ_ONLY_EXISTS,
198+
AWS_ERROR_PKCS11_CKR_SESSION_READ_WRITE_SO_EXISTS,
199+
AWS_ERROR_PKCS11_CKR_SIGNATURE_INVALID,
200+
AWS_ERROR_PKCS11_CKR_SIGNATURE_LEN_RANGE,
201+
AWS_ERROR_PKCS11_CKR_TEMPLATE_INCOMPLETE,
202+
AWS_ERROR_PKCS11_CKR_TEMPLATE_INCONSISTENT,
203+
AWS_ERROR_PKCS11_CKR_TOKEN_NOT_PRESENT,
204+
AWS_ERROR_PKCS11_CKR_TOKEN_NOT_RECOGNIZED,
205+
AWS_ERROR_PKCS11_CKR_TOKEN_WRITE_PROTECTED,
206+
AWS_ERROR_PKCS11_CKR_UNWRAPPING_KEY_HANDLE_INVALID,
207+
AWS_ERROR_PKCS11_CKR_UNWRAPPING_KEY_SIZE_RANGE,
208+
AWS_ERROR_PKCS11_CKR_UNWRAPPING_KEY_TYPE_INCONSISTENT,
209+
AWS_ERROR_PKCS11_CKR_USER_ALREADY_LOGGED_IN,
210+
AWS_ERROR_PKCS11_CKR_USER_NOT_LOGGED_IN,
211+
AWS_ERROR_PKCS11_CKR_USER_PIN_NOT_INITIALIZED,
212+
AWS_ERROR_PKCS11_CKR_USER_TYPE_INVALID,
213+
AWS_ERROR_PKCS11_CKR_USER_ANOTHER_ALREADY_LOGGED_IN,
214+
AWS_ERROR_PKCS11_CKR_USER_TOO_MANY_TYPES,
215+
AWS_ERROR_PKCS11_CKR_WRAPPED_KEY_INVALID,
216+
AWS_ERROR_PKCS11_CKR_WRAPPED_KEY_LEN_RANGE,
217+
AWS_ERROR_PKCS11_CKR_WRAPPING_KEY_HANDLE_INVALID,
218+
AWS_ERROR_PKCS11_CKR_WRAPPING_KEY_SIZE_RANGE,
219+
AWS_ERROR_PKCS11_CKR_WRAPPING_KEY_TYPE_INCONSISTENT,
220+
AWS_ERROR_PKCS11_CKR_RANDOM_SEED_NOT_SUPPORTED,
221+
AWS_ERROR_PKCS11_CKR_RANDOM_NO_RNG,
222+
AWS_ERROR_PKCS11_CKR_DOMAIN_PARAMS_INVALID,
223+
AWS_ERROR_PKCS11_CKR_CURVE_NOT_SUPPORTED,
224+
AWS_ERROR_PKCS11_CKR_BUFFER_TOO_SMALL,
225+
AWS_ERROR_PKCS11_CKR_SAVED_STATE_INVALID,
226+
AWS_ERROR_PKCS11_CKR_INFORMATION_SENSITIVE,
227+
AWS_ERROR_PKCS11_CKR_STATE_UNSAVEABLE,
228+
AWS_ERROR_PKCS11_CKR_CRYPTOKI_NOT_INITIALIZED,
229+
AWS_ERROR_PKCS11_CKR_CRYPTOKI_ALREADY_INITIALIZED,
230+
AWS_ERROR_PKCS11_CKR_MUTEX_BAD,
231+
AWS_ERROR_PKCS11_CKR_MUTEX_NOT_LOCKED,
232+
AWS_ERROR_PKCS11_CKR_NEW_PIN_MODE,
233+
AWS_ERROR_PKCS11_CKR_NEXT_OTP,
234+
AWS_ERROR_PKCS11_CKR_EXCEEDED_MAX_ITERATIONS,
235+
AWS_ERROR_PKCS11_CKR_FIPS_SELF_TEST_FAILED,
236+
AWS_ERROR_PKCS11_CKR_LIBRARY_LOAD_FAILED,
237+
AWS_ERROR_PKCS11_CKR_PIN_TOO_WEAK,
238+
AWS_ERROR_PKCS11_CKR_PUBLIC_KEY_INVALID,
239+
AWS_ERROR_PKCS11_CKR_FUNCTION_REJECTED,
137240

138241
AWS_IO_ERROR_END_RANGE = AWS_ERROR_ENUM_END_RANGE(AWS_C_IO_PACKAGE_ID),
139242
AWS_IO_INVALID_FILE_HANDLE = AWS_ERROR_INVALID_FILE_HANDLE,

include/aws/io/logging.h

+1
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,7 @@ enum aws_io_log_subject {
2929
AWS_LS_IO_SHARED_LIBRARY,
3030
AWS_LS_IO_EXPONENTIAL_BACKOFF_RETRY_STRATEGY,
3131
AWS_LS_IO_STANDARD_RETRY_STRATEGY,
32+
AWS_LS_IO_PKCS11,
3233
AWS_IO_LS_LAST = AWS_LOG_SUBJECT_END_RANGE(AWS_C_IO_PACKAGE_ID)
3334
};
3435

0 commit comments

Comments
 (0)