diff --git a/.github/workflows/build-and-test.yml b/.github/workflows/build-and-test.yml index 396e2352..22965368 100644 --- a/.github/workflows/build-and-test.yml +++ b/.github/workflows/build-and-test.yml @@ -33,6 +33,11 @@ jobs: - name: Build and test run: cd test && make clean && make WOLFSSL_DIR=../wolfssl run + # Build and test ASAN build, with wolfCrypt tests enabled. FORCE SUCCESS + # since wolfCrypt tests aren't all passing yet - this is just a status indicator + - name: Build and test ASAN TESTWOLFCRYPT (wolfCrypt tests OK to fail) + run: cd test && make clean && make ASAN=1 TESTWOLFCRYPT=1 WOLFSSL_DIR=../wolfssl run || true + # Build and test debug build with ASAN and NOCRYPTO - name: Build and test ASAN DEBUG NOCRYPTO run: cd test && make clean && make DEBUG=1 ASAN=1 NOCRYPTO=1 WOLFSSL_DIR=../wolfssl run diff --git a/test/Makefile b/test/Makefile index faa64a40..100fb17c 100644 --- a/test/Makefile +++ b/test/Makefile @@ -13,7 +13,7 @@ BIN = wh_test USER_SETTINGS_DIR ?= ./ INC = -I$(WOLFHSM_DIR) \ -I$(USER_SETTINGS_DIR) \ - -I$(WOLFSSL_DIR) \ + -I$(WOLFSSL_DIR) # Library configuration defines to use additional files. DEF = -DWOLFSSL_USER_SETTINGS -DWOLFHSM_CFG @@ -22,6 +22,10 @@ DEF = -DWOLFSSL_USER_SETTINGS -DWOLFHSM_CFG #DEF += -DWOLFHSM_CFG_TEST_VERBOSE DEF += -DWOLFHSM_CFG_TEST_POSIX +ifeq ($(TESTWOLFCRYPT),1) +DEF += -DWOLFHSM_CFG_TEST_WOLFCRYPTTEST +endif + # Architecture ARCHFLAGS ?= @@ -82,8 +86,14 @@ SRC_C += \ $(WOLFSSL_DIR)/wolfcrypt/src/sha256.c \ $(WOLFSSL_DIR)/wolfcrypt/src/aes.c \ $(WOLFSSL_DIR)/wolfcrypt/src/ecc.c \ - $(WOLFSSL_DIR)/wolfcrypt/src/cmac.c \ + $(WOLFSSL_DIR)/wolfcrypt/src/cmac.c +endif +ifeq ($(TESTWOLFCRYPT),1) +# wolfCrypt test source files +SRC_C += $(WOLFSSL_DIR)/wolfcrypt/test/test.c +SRC_C += $(wildcard $(WOLFSSL_DIR)/wolfcrypt/src/*.c) +CFLAGS += -DWC_USE_DEVID=0x5748534D endif # wolfHSM source files @@ -105,14 +115,14 @@ SRC_C += \ $(WOLFHSM_DIR)/src/wh_message_customcb.c \ $(WOLFHSM_DIR)/src/wh_message_nvm.c \ $(WOLFHSM_DIR)/src/wh_transport_mem.c \ - $(WOLFHSM_DIR)/src/wh_flash_ramsim.c \ + $(WOLFHSM_DIR)/src/wh_flash_ramsim.c ifeq ($(SHE),1) SRC_C += \ $(WOLFHSM_DIR)/src/wh_client_she.c \ $(WOLFHSM_DIR)/src/wh_server_she.c \ $(WOLFHSM_DIR)/src/wh_she_common.c \ - $(WOLFHSM_DIR)/src/wh_she_crypto.c \ + $(WOLFHSM_DIR)/src/wh_she_crypto.c endif @@ -123,7 +133,7 @@ SRC_C += \ $(WOLFHSM_DIR)/src/wh_flash_ramsim.c \ $(WOLFHSM_DIR)/src/wh_transport_mem.c \ $(WOLFHSM_DIR)/port/posix/posix_flash_file.c \ - $(WOLFHSM_DIR)/port/posix/posix_transport_tcp.c \ + $(WOLFHSM_DIR)/port/posix/posix_transport_tcp.c # APP SRC_C += \ @@ -134,6 +144,10 @@ SRC_C += \ ./src/wh_test_nvm_flash.c \ ./src/wh_test_clientserver.c \ ./src/wh_test_flash_ramsim.c \ + +ifeq ($(TESTWOLFCRYPT),1) +SRC_C += ./src/wh_test_wolfcrypt_test.c +endif FILENAMES_C = $(notdir $(SRC_C)) #FILENAMES_C := $(filter-out evp.c, $(FILENAMES_C)) diff --git a/test/wh_test.c b/test/wh_test.c index 2eff0018..3eae52ab 100644 --- a/test/wh_test.c +++ b/test/wh_test.c @@ -34,6 +34,10 @@ #include "wh_test_she.h" #include "wh_test_clientserver.h" +#if defined(WOLFHSM_CFG_TEST_WOLFCRYPTTEST) +#include "wh_test_wolfcrypt_test.h" +#endif + /* Default test args */ @@ -56,6 +60,11 @@ int whTest_Unit(void) #if defined(WOLFHSM_CFG_SHE_EXTENSION) WH_TEST_ASSERT(0 == whTest_She()); #endif /* WOLFHSM_SHE_EXTENTION */ + +#if defined(WOLFHSM_CFG_TEST_WOLFCRYPTTEST) + WH_TEST_ASSERT(0 == whTest_WolfCryptTest()); +#endif + #endif /* !WOLFHSM_CFG_NO_CRYPTO */ return 0; diff --git a/test/wh_test_wolfcrypt_test.c b/test/wh_test_wolfcrypt_test.c new file mode 100644 index 00000000..3ecfb3db --- /dev/null +++ b/test/wh_test_wolfcrypt_test.c @@ -0,0 +1,246 @@ +/* + * Copyright (C) 2024 wolfSSL Inc. + * + * This file is part of wolfHSM. + * + * wolfHSM is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 3 of the License, or + * (at your option) any later version. + * + * wolfHSM is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with wolfHSM. If not, see . + */ +/* + * test/wh_test_wolfcrypt_test.c + * + */ + +#include +#include /* For printf */ +#include /* For memset, memcpy */ + +#ifndef WOLFHSM_CFG_NO_CRYPTO + +#include "wolfssl/wolfcrypt/settings.h" +#include "wolfssl/wolfcrypt/types.h" +#include "wolfcrypt/test/test.h" + +#include "wolfhsm/wh_error.h" +#include "wolfhsm/wh_nvm.h" +#include "wolfhsm/wh_nvm_flash.h" +#include "wolfhsm/wh_flash_ramsim.h" +#include "wolfhsm/wh_comm.h" +#include "wolfhsm/wh_message.h" +#include "wolfhsm/wh_server.h" +#include "wolfhsm/wh_client.h" +#include "wolfhsm/wh_transport_mem.h" + +#include "wh_test_common.h" + +#if defined(WOLFHSM_CFG_TEST_POSIX) +#include /* For pthread_create/cancel/join/_t */ +#include "port/posix/posix_transport_tcp.h" +#include "port/posix/posix_flash_file.h" +#endif + +#if defined(WOLFHSM_CFG_TEST_POSIX) +#include /* For sleep */ +#include /* For pthread_create/cancel/join/_t */ +#include "port/posix/posix_transport_tcp.h" +#include "port/posix/posix_flash_file.h" +#endif + + +#define BUFFER_SIZE 4096 +#define FLASH_RAM_SIZE (1024 * 1024) /* 1MB */ + +int whTest_WolfCryptTestCfg(whClientConfig* config) +{ + whClientContext client[1] = {0}; + + if (config == NULL) { + return WH_ERROR_BADARGS; + } + + WH_TEST_RETURN_ON_FAIL(wh_Client_Init(client, config)); + WH_TEST_RETURN_ON_FAIL(wh_Client_CommInit(client, NULL, NULL)); + + /* assumes wolfCrypt has been initialized before this function */ + WH_TEST_RETURN_ON_FAIL(wolfcrypt_test(NULL)); + + return WH_ERROR_OK; +} + +static int whTest_ServerCfgLoop(whServerConfig* serverCfg) +{ + whServerContext server[1] = {0}; + whCommConnected am_connected = WH_COMM_CONNECTED; + int ret = 0; + + WH_TEST_RETURN_ON_FAIL(wh_Server_Init(server, serverCfg)); + WH_TEST_RETURN_ON_FAIL(wh_Server_SetConnected(server, am_connected)); + + while (am_connected == WH_COMM_CONNECTED) { + ret = wh_Server_HandleRequestMessage(server); + if ((ret != WH_ERROR_NOTREADY) && (ret != WH_ERROR_OK)) { + WH_ERROR_PRINT( + "[server] Failed to wh_Server_HandleRequestMessage ret=%d\n", + ret); + break; + } + wh_Server_GetConnected(server, &am_connected); + } + + if ((ret == 0) || (ret == WH_ERROR_NOTREADY)) { + ret = 0; + WH_TEST_RETURN_ON_FAIL(wh_Server_Cleanup(server)); + } + else { + ret = wh_Server_Cleanup(server); + } + + return ret; +} + + +#if defined(WOLFHSM_CFG_TEST_POSIX) +static void* _whClientTask(void* cf) +{ + WH_TEST_ASSERT(0 == whTest_WolfCryptTestCfg(cf)); + return NULL; +} + +static void* _whServerTask(void* cf) +{ + WH_TEST_ASSERT(0 == whTest_ServerCfgLoop(cf)); + return NULL; +} + + +static void _whClientServerThreadTest(whClientConfig* c_conf, + whServerConfig* s_conf) +{ + pthread_t cthread = {0}; + pthread_t sthread = {0}; + + void* retval; + int rc = 0; + + rc = pthread_create(&sthread, NULL, _whServerTask, s_conf); + if (rc == 0) { + rc = pthread_create(&cthread, NULL, _whClientTask, c_conf); + if (rc == 0) { + /* All good. Block on joining */ + pthread_join(cthread, &retval); + pthread_cancel(sthread); + } + else { + /* Cancel the server thread */ + pthread_cancel(sthread); + } + } +} + +static int wh_ClientServer_MemThreadTest(void) +{ + uint8_t req[BUFFER_SIZE] = {0}; + uint8_t resp[BUFFER_SIZE] = {0}; + + whTransportMemConfig tmcf[1] = {{ + .req = (whTransportMemCsr*)req, + .req_size = sizeof(req), + .resp = (whTransportMemCsr*)resp, + .resp_size = sizeof(resp), + }}; + /* Client configuration/contexts */ + whTransportClientCb tccb[1] = {WH_TRANSPORT_MEM_CLIENT_CB}; + whTransportMemClientContext tmcc[1] = {0}; + whCommClientConfig cc_conf[1] = {{ + .transport_cb = tccb, + .transport_context = (void*)tmcc, + .transport_config = (void*)tmcf, + .client_id = 123, + }}; + whClientConfig c_conf[1] = {{ + .comm = cc_conf, + }}; + /* Server configuration/contexts */ + whTransportServerCb tscb[1] = {WH_TRANSPORT_MEM_SERVER_CB}; + whTransportMemServerContext tmsc[1] = {0}; + whCommServerConfig cs_conf[1] = {{ + .transport_cb = tscb, + .transport_context = (void*)tmsc, + .transport_config = (void*)tmcf, + .server_id = 124, + }}; + + /* RamSim Flash state and configuration */ + whFlashRamsimCtx fc[1] = {0}; + whFlashRamsimCfg fc_conf[1] = {{ + .size = FLASH_RAM_SIZE, + .sectorSize = FLASH_RAM_SIZE / 2, + .pageSize = 8, + .erasedByte = (uint8_t)0, + }}; + const whFlashCb fcb[1] = {WH_FLASH_RAMSIM_CB}; + + /* NVM Flash Configuration using RamSim HAL Flash */ + whNvmFlashConfig nf_conf[1] = {{ + .cb = fcb, + .context = fc, + .config = fc_conf, + }}; + whNvmFlashContext nfc[1] = {0}; + whNvmCb nfcb[1] = {WH_NVM_FLASH_CB}; + + whNvmConfig n_conf[1] = {{ + .cb = nfcb, + .context = nfc, + .config = nf_conf, + }}; + whNvmContext nvm[1] = {{0}}; + + /* Crypto context */ + whServerCryptoContext crypto[1] = {{ + .devId = INVALID_DEVID, + }}; + + whServerConfig s_conf[1] = {{ + .comm_config = cs_conf, + .nvm = nvm, + .crypto = crypto, + }}; + + WH_TEST_RETURN_ON_FAIL(wh_Nvm_Init(nvm, n_conf)); + + WH_TEST_RETURN_ON_FAIL(wolfCrypt_Init()); + WH_TEST_RETURN_ON_FAIL(wc_InitRng_ex(crypto->rng, NULL, crypto->devId)); + + _whClientServerThreadTest(c_conf, s_conf); + + wh_Nvm_Cleanup(nvm); + + wc_FreeRng(crypto->rng); + wolfCrypt_Cleanup(); + + return WH_ERROR_OK; +} +#endif /* WOLFHSM_CFG_TEST_POSIX */ + + +int whTest_WolfCryptTest(void) +{ +#if defined(WOLFHSM_CFG_TEST_POSIX) + printf("Testing wolfCrypt tests: (pthread) mem...\n"); + WH_TEST_RETURN_ON_FAIL(wh_ClientServer_MemThreadTest()); +#endif + return 0; +} + +#endif /* !WOLFHSM_CFG_NO_CRYPTO */ diff --git a/test/wh_test_wolfcrypt_test.h b/test/wh_test_wolfcrypt_test.h new file mode 100644 index 00000000..df5b7d10 --- /dev/null +++ b/test/wh_test_wolfcrypt_test.h @@ -0,0 +1,36 @@ +/* + * Copyright (C) 2024 wolfSSL Inc. + * + * This file is part of wolfHSM. + * + * wolfHSM is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 3 of the License, or + * (at your option) any later version. + * + * wolfHSM is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with wolfHSM. If not, see . + */ +/* + * test/wh_test_wolfcrypt_test.h + * + */ + +#ifndef TEST_WH_TEST_WOLFCRYPT_TEST_H_ +#define TEST_WH_TEST_WOLFCRYPT_TEST_H_ + +/* + * Runs wolfCrypt test suite on top of wolfHSM using a memory transport backend. + * and optionally the POSIX TCP backent if WOLFHSM_CFG_TEST_POSIX is defined + * + * Returns 0 on success and a non-zero error code on failure + */ +int whTest_WolfCryptTestCfg(whClientConfig* config); +int whTest_WolfCryptTest(void); + +#endif /* TEST_WH_TEST_WOLFCRYPT_TEST_H_ */