1*cfb92d14SAndroid Build Coastguard Worker /*
2*cfb92d14SAndroid Build Coastguard Worker * Copyright (c) 2024, The OpenThread Authors.
3*cfb92d14SAndroid Build Coastguard Worker * All rights reserved.
4*cfb92d14SAndroid Build Coastguard Worker *
5*cfb92d14SAndroid Build Coastguard Worker * Redistribution and use in source and binary forms, with or without
6*cfb92d14SAndroid Build Coastguard Worker * modification, are permitted provided that the following conditions are met:
7*cfb92d14SAndroid Build Coastguard Worker * 1. Redistributions of source code must retain the above copyright
8*cfb92d14SAndroid Build Coastguard Worker * notice, this list of conditions and the following disclaimer.
9*cfb92d14SAndroid Build Coastguard Worker * 2. Redistributions in binary form must reproduce the above copyright
10*cfb92d14SAndroid Build Coastguard Worker * notice, this list of conditions and the following disclaimer in the
11*cfb92d14SAndroid Build Coastguard Worker * documentation and/or other materials provided with the distribution.
12*cfb92d14SAndroid Build Coastguard Worker * 3. Neither the name of the copyright holder nor the
13*cfb92d14SAndroid Build Coastguard Worker * names of its contributors may be used to endorse or promote products
14*cfb92d14SAndroid Build Coastguard Worker * derived from this software without specific prior written permission.
15*cfb92d14SAndroid Build Coastguard Worker *
16*cfb92d14SAndroid Build Coastguard Worker * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
17*cfb92d14SAndroid Build Coastguard Worker * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18*cfb92d14SAndroid Build Coastguard Worker * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19*cfb92d14SAndroid Build Coastguard Worker * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
20*cfb92d14SAndroid Build Coastguard Worker * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
21*cfb92d14SAndroid Build Coastguard Worker * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
22*cfb92d14SAndroid Build Coastguard Worker * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
23*cfb92d14SAndroid Build Coastguard Worker * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
24*cfb92d14SAndroid Build Coastguard Worker * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
25*cfb92d14SAndroid Build Coastguard Worker * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
26*cfb92d14SAndroid Build Coastguard Worker * POSSIBILITY OF SUCH DAMAGE.
27*cfb92d14SAndroid Build Coastguard Worker */
28*cfb92d14SAndroid Build Coastguard Worker
29*cfb92d14SAndroid Build Coastguard Worker #include "openthread-core-config.h"
30*cfb92d14SAndroid Build Coastguard Worker
31*cfb92d14SAndroid Build Coastguard Worker #include "test_platform.h"
32*cfb92d14SAndroid Build Coastguard Worker #include "test_util.h"
33*cfb92d14SAndroid Build Coastguard Worker
34*cfb92d14SAndroid Build Coastguard Worker #if OPENTHREAD_CONFIG_BLE_TCAT_ENABLE
35*cfb92d14SAndroid Build Coastguard Worker
36*cfb92d14SAndroid Build Coastguard Worker #include <openthread/ble_secure.h>
37*cfb92d14SAndroid Build Coastguard Worker
38*cfb92d14SAndroid Build Coastguard Worker #define OT_TCAT_X509_CERT \
39*cfb92d14SAndroid Build Coastguard Worker "-----BEGIN CERTIFICATE-----\n" \
40*cfb92d14SAndroid Build Coastguard Worker "MIIB6TCCAZCgAwIBAgICNekwCgYIKoZIzj0EAwIwcTEmMCQGA1UEAwwdVGhyZWFk\n" \
41*cfb92d14SAndroid Build Coastguard Worker "IENlcnRpZmljYXRpb24gRGV2aWNlQ0ExGTAXBgNVBAoMEFRocmVhZCBHcm91cCBJ\n" \
42*cfb92d14SAndroid Build Coastguard Worker "bmMxEjAQBgNVBAcMCVNhbiBSYW1vbjELMAkGA1UECAwCQ0ExCzAJBgNVBAYTAlVT\n" \
43*cfb92d14SAndroid Build Coastguard Worker "MCAXDTI0MDUwNzA5Mzk0NVoYDzI5OTkxMjMxMDkzOTQ1WjA8MSEwHwYDVQQDDBhU\n" \
44*cfb92d14SAndroid Build Coastguard Worker "Q0FUIEV4YW1wbGUgRGV2aWNlQ2VydDExFzAVBgNVBAUTDjQ3MjMtOTgzMy0wMDAx\n" \
45*cfb92d14SAndroid Build Coastguard Worker "MFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcDQgAE11h/4vKZXVXv+1GDZo066spItloT\n" \
46*cfb92d14SAndroid Build Coastguard Worker "dpCi0bux0jvpQSHLdQBIc+40zVCxMDRUvbX//vJKGsSJKOVUlCojQ2wIdqNLMEkw\n" \
47*cfb92d14SAndroid Build Coastguard Worker "HwYDVR0jBBgwFoAUX6sbKWiIodS0MaiGYefnZlnt+BkwEAYJKwYBBAGC3yoCBAMC\n" \
48*cfb92d14SAndroid Build Coastguard Worker "AQUwFAYJKwYBBAGC3yoDBAcEBSABAQEBMAoGCCqGSM49BAMCA0cAMEQCIHWu+Rd1\n" \
49*cfb92d14SAndroid Build Coastguard Worker "VRlzrD8KbuyJcJFTXh2sQ9UIrFIA7+4e/GVcAiAVBdGqTxbt3TGkBBllpafAUB2/\n" \
50*cfb92d14SAndroid Build Coastguard Worker "s0GJj7E33oblqy5eHQ==\n" \
51*cfb92d14SAndroid Build Coastguard Worker "-----END CERTIFICATE-----\n"
52*cfb92d14SAndroid Build Coastguard Worker
53*cfb92d14SAndroid Build Coastguard Worker #define OT_TCAT_PRIV_KEY \
54*cfb92d14SAndroid Build Coastguard Worker "-----BEGIN EC PRIVATE KEY-----\n" \
55*cfb92d14SAndroid Build Coastguard Worker "MHcCAQEEIIqKM1QTlNaquV74W6Viz/ggXoLqlPOP6LagSyaFO3oUoAoGCCqGSM49\n" \
56*cfb92d14SAndroid Build Coastguard Worker "AwEHoUQDQgAE11h/4vKZXVXv+1GDZo066spItloTdpCi0bux0jvpQSHLdQBIc+40\n" \
57*cfb92d14SAndroid Build Coastguard Worker "zVCxMDRUvbX//vJKGsSJKOVUlCojQ2wIdg==\n" \
58*cfb92d14SAndroid Build Coastguard Worker "-----END EC PRIVATE KEY-----\n"
59*cfb92d14SAndroid Build Coastguard Worker
60*cfb92d14SAndroid Build Coastguard Worker #define OT_TCAT_TRUSTED_ROOT_CERTIFICATE \
61*cfb92d14SAndroid Build Coastguard Worker "-----BEGIN CERTIFICATE-----\n" \
62*cfb92d14SAndroid Build Coastguard Worker "MIICOzCCAeGgAwIBAgIJAKOc2hehOGoBMAoGCCqGSM49BAMCMHExJjAkBgNVBAMM\n" \
63*cfb92d14SAndroid Build Coastguard Worker "HVRocmVhZCBDZXJ0aWZpY2F0aW9uIERldmljZUNBMRkwFwYDVQQKDBBUaHJlYWQg\n" \
64*cfb92d14SAndroid Build Coastguard Worker "R3JvdXAgSW5jMRIwEAYDVQQHDAlTYW4gUmFtb24xCzAJBgNVBAgMAkNBMQswCQYD\n" \
65*cfb92d14SAndroid Build Coastguard Worker "VQQGEwJVUzAeFw0yNDA1MDMyMDAyMThaFw00NDA0MjgyMDAyMThaMHExJjAkBgNV\n" \
66*cfb92d14SAndroid Build Coastguard Worker "BAMMHVRocmVhZCBDZXJ0aWZpY2F0aW9uIERldmljZUNBMRkwFwYDVQQKDBBUaHJl\n" \
67*cfb92d14SAndroid Build Coastguard Worker "YWQgR3JvdXAgSW5jMRIwEAYDVQQHDAlTYW4gUmFtb24xCzAJBgNVBAgMAkNBMQsw\n" \
68*cfb92d14SAndroid Build Coastguard Worker "CQYDVQQGEwJVUzBZMBMGByqGSM49AgEGCCqGSM49AwEHA0IABGy850VBIPTkN3oL\n" \
69*cfb92d14SAndroid Build Coastguard Worker "x++zIUsZk2k26w4fuieFz9oNvjdb5W14+Yf3mvGWsl4NHyLxqhmamVAR4h7zWRlZ\n" \
70*cfb92d14SAndroid Build Coastguard Worker "0XyMVpKjYjBgMB4GA1UdEQQXMBWBE3RvbUB0aHJlYWRncm91cC5vcmcwDgYDVR0P\n" \
71*cfb92d14SAndroid Build Coastguard Worker "AQH/BAQDAgGGMA8GA1UdEwEB/wQFMAMBAf8wHQYDVR0OBBYEFF+rGyloiKHUtDGo\n" \
72*cfb92d14SAndroid Build Coastguard Worker "hmHn52ZZ7fgZMAoGCCqGSM49BAMCA0gAMEUCIQCTq1qjPZs9fAJB6ppTXs588Pnu\n" \
73*cfb92d14SAndroid Build Coastguard Worker "eVFOwC8bd//D99KiHAIgU84kwFHIyDvFqu6y+u1hFqBGsiuTmKwZ2PHhVe/xK1k=\n" \
74*cfb92d14SAndroid Build Coastguard Worker "-----END CERTIFICATE-----\n"
75*cfb92d14SAndroid Build Coastguard Worker
76*cfb92d14SAndroid Build Coastguard Worker namespace ot {
77*cfb92d14SAndroid Build Coastguard Worker
78*cfb92d14SAndroid Build Coastguard Worker class TestBleSecure
79*cfb92d14SAndroid Build Coastguard Worker {
80*cfb92d14SAndroid Build Coastguard Worker public:
TestBleSecure(void)81*cfb92d14SAndroid Build Coastguard Worker TestBleSecure(void)
82*cfb92d14SAndroid Build Coastguard Worker : mIsConnected(false)
83*cfb92d14SAndroid Build Coastguard Worker , mIsBleConnectionOpen(false)
84*cfb92d14SAndroid Build Coastguard Worker {
85*cfb92d14SAndroid Build Coastguard Worker }
86*cfb92d14SAndroid Build Coastguard Worker
HandleBleSecureConnect(bool aConnected,bool aBleConnectionOpen)87*cfb92d14SAndroid Build Coastguard Worker void HandleBleSecureConnect(bool aConnected, bool aBleConnectionOpen)
88*cfb92d14SAndroid Build Coastguard Worker {
89*cfb92d14SAndroid Build Coastguard Worker mIsConnected = aConnected;
90*cfb92d14SAndroid Build Coastguard Worker mIsBleConnectionOpen = aBleConnectionOpen;
91*cfb92d14SAndroid Build Coastguard Worker }
92*cfb92d14SAndroid Build Coastguard Worker
IsConnected(void) const93*cfb92d14SAndroid Build Coastguard Worker bool IsConnected(void) const { return mIsConnected; }
IsBleConnectionOpen(void) const94*cfb92d14SAndroid Build Coastguard Worker bool IsBleConnectionOpen(void) const { return mIsBleConnectionOpen; }
95*cfb92d14SAndroid Build Coastguard Worker
96*cfb92d14SAndroid Build Coastguard Worker private:
97*cfb92d14SAndroid Build Coastguard Worker bool mIsConnected;
98*cfb92d14SAndroid Build Coastguard Worker bool mIsBleConnectionOpen;
99*cfb92d14SAndroid Build Coastguard Worker };
100*cfb92d14SAndroid Build Coastguard Worker
HandleBleSecureConnect(otInstance * aInstance,bool aConnected,bool aBleConnectionOpen,void * aContext)101*cfb92d14SAndroid Build Coastguard Worker static void HandleBleSecureConnect(otInstance *aInstance, bool aConnected, bool aBleConnectionOpen, void *aContext)
102*cfb92d14SAndroid Build Coastguard Worker {
103*cfb92d14SAndroid Build Coastguard Worker OT_UNUSED_VARIABLE(aInstance);
104*cfb92d14SAndroid Build Coastguard Worker
105*cfb92d14SAndroid Build Coastguard Worker static_cast<TestBleSecure *>(aContext)->HandleBleSecureConnect(aConnected, aBleConnectionOpen);
106*cfb92d14SAndroid Build Coastguard Worker }
107*cfb92d14SAndroid Build Coastguard Worker
TestTcat(void)108*cfb92d14SAndroid Build Coastguard Worker void TestTcat(void)
109*cfb92d14SAndroid Build Coastguard Worker {
110*cfb92d14SAndroid Build Coastguard Worker const char kPskdVendor[] = "J01NM3";
111*cfb92d14SAndroid Build Coastguard Worker const char kUrl[] = "dummy_url";
112*cfb92d14SAndroid Build Coastguard Worker constexpr uint16_t kConnectionId = 0;
113*cfb92d14SAndroid Build Coastguard Worker const int kCertificateThreadVersion = 2;
114*cfb92d14SAndroid Build Coastguard Worker const int kCertificateAuthorizationField = 3;
115*cfb92d14SAndroid Build Coastguard Worker const uint8_t expectedTcatAuthField[5] = {0x20, 0x01, 0x01, 0x01, 0x01};
116*cfb92d14SAndroid Build Coastguard Worker uint8_t attributeBuffer[8];
117*cfb92d14SAndroid Build Coastguard Worker size_t attributeLen;
118*cfb92d14SAndroid Build Coastguard Worker
119*cfb92d14SAndroid Build Coastguard Worker TestBleSecure ble;
120*cfb92d14SAndroid Build Coastguard Worker Instance *instance = testInitInstance();
121*cfb92d14SAndroid Build Coastguard Worker
122*cfb92d14SAndroid Build Coastguard Worker otTcatVendorInfo vendorInfo = {.mProvisioningUrl = kUrl, .mPskdString = kPskdVendor};
123*cfb92d14SAndroid Build Coastguard Worker
124*cfb92d14SAndroid Build Coastguard Worker otBleSecureSetCertificate(instance, reinterpret_cast<const uint8_t *>(OT_TCAT_X509_CERT), sizeof(OT_TCAT_X509_CERT),
125*cfb92d14SAndroid Build Coastguard Worker reinterpret_cast<const uint8_t *>(OT_TCAT_PRIV_KEY), sizeof(OT_TCAT_PRIV_KEY));
126*cfb92d14SAndroid Build Coastguard Worker
127*cfb92d14SAndroid Build Coastguard Worker otBleSecureSetCaCertificateChain(instance, reinterpret_cast<const uint8_t *>(OT_TCAT_TRUSTED_ROOT_CERTIFICATE),
128*cfb92d14SAndroid Build Coastguard Worker sizeof(OT_TCAT_TRUSTED_ROOT_CERTIFICATE));
129*cfb92d14SAndroid Build Coastguard Worker
130*cfb92d14SAndroid Build Coastguard Worker otBleSecureSetSslAuthMode(instance, true);
131*cfb92d14SAndroid Build Coastguard Worker
132*cfb92d14SAndroid Build Coastguard Worker // Validate BLE secure and Tcat start APIs
133*cfb92d14SAndroid Build Coastguard Worker SuccessOrQuit(otBleSecureSetTcatVendorInfo(instance, &vendorInfo));
134*cfb92d14SAndroid Build Coastguard Worker VerifyOrQuit(otBleSecureTcatStart(instance, nullptr) == kErrorInvalidState);
135*cfb92d14SAndroid Build Coastguard Worker SuccessOrQuit(otBleSecureStart(instance, HandleBleSecureConnect, nullptr, true, &ble));
136*cfb92d14SAndroid Build Coastguard Worker VerifyOrQuit(otBleSecureStart(instance, HandleBleSecureConnect, nullptr, true, nullptr) == kErrorAlready);
137*cfb92d14SAndroid Build Coastguard Worker SuccessOrQuit(otBleSecureTcatStart(instance, nullptr));
138*cfb92d14SAndroid Build Coastguard Worker
139*cfb92d14SAndroid Build Coastguard Worker // Validate connection callbacks when platform informs that peer has connected/disconnected
140*cfb92d14SAndroid Build Coastguard Worker VerifyOrQuit(!otBleSecureIsConnected(instance));
141*cfb92d14SAndroid Build Coastguard Worker otPlatBleGapOnConnected(instance, kConnectionId);
142*cfb92d14SAndroid Build Coastguard Worker VerifyOrQuit(!ble.IsConnected() && ble.IsBleConnectionOpen());
143*cfb92d14SAndroid Build Coastguard Worker otPlatBleGapOnDisconnected(instance, kConnectionId);
144*cfb92d14SAndroid Build Coastguard Worker VerifyOrQuit(!ble.IsConnected() && !ble.IsBleConnectionOpen());
145*cfb92d14SAndroid Build Coastguard Worker
146*cfb92d14SAndroid Build Coastguard Worker // Verify that Thread-attribute parsing isn't available yet when not connected as client or server.
147*cfb92d14SAndroid Build Coastguard Worker attributeLen = sizeof(attributeBuffer);
148*cfb92d14SAndroid Build Coastguard Worker VerifyOrQuit(otBleSecureGetThreadAttributeFromPeerCertificate(instance, kCertificateAuthorizationField,
149*cfb92d14SAndroid Build Coastguard Worker &attributeBuffer[0],
150*cfb92d14SAndroid Build Coastguard Worker &attributeLen) == kErrorInvalidState);
151*cfb92d14SAndroid Build Coastguard Worker attributeLen = sizeof(attributeBuffer);
152*cfb92d14SAndroid Build Coastguard Worker VerifyOrQuit(otBleSecureGetThreadAttributeFromOwnCertificate(
153*cfb92d14SAndroid Build Coastguard Worker instance, kCertificateThreadVersion, &attributeBuffer[0], &attributeLen) == kErrorInvalidState);
154*cfb92d14SAndroid Build Coastguard Worker
155*cfb92d14SAndroid Build Coastguard Worker // Validate connection callbacks when calling `otBleSecureDisconnect()`
156*cfb92d14SAndroid Build Coastguard Worker otPlatBleGapOnConnected(instance, kConnectionId);
157*cfb92d14SAndroid Build Coastguard Worker VerifyOrQuit(!ble.IsConnected() && ble.IsBleConnectionOpen());
158*cfb92d14SAndroid Build Coastguard Worker otBleSecureDisconnect(instance);
159*cfb92d14SAndroid Build Coastguard Worker VerifyOrQuit(!ble.IsConnected() && !ble.IsBleConnectionOpen());
160*cfb92d14SAndroid Build Coastguard Worker
161*cfb92d14SAndroid Build Coastguard Worker // Validate TLS connection can be started only when peer is connected
162*cfb92d14SAndroid Build Coastguard Worker otPlatBleGapOnConnected(instance, kConnectionId);
163*cfb92d14SAndroid Build Coastguard Worker SuccessOrQuit(otBleSecureConnect(instance));
164*cfb92d14SAndroid Build Coastguard Worker VerifyOrQuit(otBleSecureIsConnectionActive(instance));
165*cfb92d14SAndroid Build Coastguard Worker
166*cfb92d14SAndroid Build Coastguard Worker // Once in TLS client connecting state, the below cert eval functions are available.
167*cfb92d14SAndroid Build Coastguard Worker // Test that the Thread-specific attributes can be decoded properly.
168*cfb92d14SAndroid Build Coastguard Worker attributeLen = 1;
169*cfb92d14SAndroid Build Coastguard Worker SuccessOrQuit(otBleSecureGetThreadAttributeFromOwnCertificate(instance, kCertificateThreadVersion,
170*cfb92d14SAndroid Build Coastguard Worker &attributeBuffer[0], &attributeLen));
171*cfb92d14SAndroid Build Coastguard Worker VerifyOrQuit(attributeLen == 1 && attributeBuffer[0] >= kThreadVersion1p4);
172*cfb92d14SAndroid Build Coastguard Worker
173*cfb92d14SAndroid Build Coastguard Worker static_assert(5 == sizeof(expectedTcatAuthField), "expectedTcatAuthField size incorrect for test");
174*cfb92d14SAndroid Build Coastguard Worker attributeLen = 5;
175*cfb92d14SAndroid Build Coastguard Worker SuccessOrQuit(otBleSecureGetThreadAttributeFromOwnCertificate(instance, kCertificateAuthorizationField,
176*cfb92d14SAndroid Build Coastguard Worker &attributeBuffer[0], &attributeLen));
177*cfb92d14SAndroid Build Coastguard Worker VerifyOrQuit(attributeLen == 5 && memcmp(&expectedTcatAuthField, &attributeBuffer, attributeLen) == 0);
178*cfb92d14SAndroid Build Coastguard Worker
179*cfb92d14SAndroid Build Coastguard Worker // Validate TLS connection can be started only when peer is connected
180*cfb92d14SAndroid Build Coastguard Worker otBleSecureDisconnect(instance);
181*cfb92d14SAndroid Build Coastguard Worker VerifyOrQuit(otBleSecureConnect(instance) == kErrorInvalidState);
182*cfb92d14SAndroid Build Coastguard Worker
183*cfb92d14SAndroid Build Coastguard Worker // Validate Tcat state changes after stopping BLE secure
184*cfb92d14SAndroid Build Coastguard Worker VerifyOrQuit(otBleSecureIsTcatEnabled(instance));
185*cfb92d14SAndroid Build Coastguard Worker otBleSecureStop(instance);
186*cfb92d14SAndroid Build Coastguard Worker VerifyOrQuit(!otBleSecureIsTcatEnabled(instance));
187*cfb92d14SAndroid Build Coastguard Worker
188*cfb92d14SAndroid Build Coastguard Worker testFreeInstance(instance);
189*cfb92d14SAndroid Build Coastguard Worker }
190*cfb92d14SAndroid Build Coastguard Worker
191*cfb92d14SAndroid Build Coastguard Worker } // namespace ot
192*cfb92d14SAndroid Build Coastguard Worker
193*cfb92d14SAndroid Build Coastguard Worker #endif // OPENTHREAD_CONFIG_BLE_TCAT_ENABLE
194*cfb92d14SAndroid Build Coastguard Worker
main(void)195*cfb92d14SAndroid Build Coastguard Worker int main(void)
196*cfb92d14SAndroid Build Coastguard Worker {
197*cfb92d14SAndroid Build Coastguard Worker #if OPENTHREAD_CONFIG_BLE_TCAT_ENABLE
198*cfb92d14SAndroid Build Coastguard Worker ot::TestTcat();
199*cfb92d14SAndroid Build Coastguard Worker printf("All tests passed\n");
200*cfb92d14SAndroid Build Coastguard Worker #else
201*cfb92d14SAndroid Build Coastguard Worker printf("Tcat is not enabled\n");
202*cfb92d14SAndroid Build Coastguard Worker return -1;
203*cfb92d14SAndroid Build Coastguard Worker #endif
204*cfb92d14SAndroid Build Coastguard Worker return 0;
205*cfb92d14SAndroid Build Coastguard Worker }
206