1*758e9fbaSOystein Eftevaag /* SPDX-License-Identifier: BSD-2-Clause */
2*758e9fbaSOystein Eftevaag /***********************************************************************
3*758e9fbaSOystein Eftevaag * Copyright (c) 2017-2018, Intel Corporation
4*758e9fbaSOystein Eftevaag *
5*758e9fbaSOystein Eftevaag * All rights reserved.
6*758e9fbaSOystein Eftevaag ***********************************************************************/
7*758e9fbaSOystein Eftevaag #ifdef HAVE_CONFIG_H
8*758e9fbaSOystein Eftevaag #include <config.h>
9*758e9fbaSOystein Eftevaag #endif
10*758e9fbaSOystein Eftevaag
11*758e9fbaSOystein Eftevaag #include <stdlib.h>
12*758e9fbaSOystein Eftevaag #include <stdio.h>
13*758e9fbaSOystein Eftevaag #include <string.h>
14*758e9fbaSOystein Eftevaag #include <setjmp.h>
15*758e9fbaSOystein Eftevaag #include <cmocka.h>
16*758e9fbaSOystein Eftevaag #include "tss2_mu.h"
17*758e9fbaSOystein Eftevaag #include "util/tss2_endian.h"
18*758e9fbaSOystein Eftevaag
19*758e9fbaSOystein Eftevaag /*
20*758e9fbaSOystein Eftevaag * Success case
21*758e9fbaSOystein Eftevaag */
22*758e9fbaSOystein Eftevaag static void
tpm2b_marshal_success(void ** state)23*758e9fbaSOystein Eftevaag tpm2b_marshal_success(void **state) {
24*758e9fbaSOystein Eftevaag TPM2B_DIGEST dgst = {4, {0}};
25*758e9fbaSOystein Eftevaag TPM2B_ECC_POINT point = {0};
26*758e9fbaSOystein Eftevaag uint8_t buffer[sizeof(dgst) + sizeof(point)] = {0};
27*758e9fbaSOystein Eftevaag size_t buffer_size = sizeof(buffer);
28*758e9fbaSOystein Eftevaag uint16_t *size_ptr = (uint16_t *) buffer;
29*758e9fbaSOystein Eftevaag uint32_t *ptr = (uint32_t *) (buffer + 2);
30*758e9fbaSOystein Eftevaag uint32_t value = 0xdeadbeef;
31*758e9fbaSOystein Eftevaag uint64_t value2 = 0xdeadbeefdeadbeefULL;
32*758e9fbaSOystein Eftevaag uint64_t *ptr2;
33*758e9fbaSOystein Eftevaag TSS2_RC rc;
34*758e9fbaSOystein Eftevaag
35*758e9fbaSOystein Eftevaag memcpy(dgst.buffer, &value, sizeof(value));
36*758e9fbaSOystein Eftevaag memcpy(point.point.x.buffer, &value, sizeof(value));
37*758e9fbaSOystein Eftevaag point.point.x.size = sizeof(value);
38*758e9fbaSOystein Eftevaag memcpy(point.point.y.buffer, &value2, sizeof(value2));
39*758e9fbaSOystein Eftevaag point.point.y.size = sizeof(value2);
40*758e9fbaSOystein Eftevaag
41*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_DIGEST_Marshal(&dgst, buffer, buffer_size, NULL);
42*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_RC_SUCCESS);
43*758e9fbaSOystein Eftevaag assert_int_equal (*size_ptr, HOST_TO_BE_16(4));
44*758e9fbaSOystein Eftevaag assert_int_equal (*ptr, value);
45*758e9fbaSOystein Eftevaag
46*758e9fbaSOystein Eftevaag size_ptr = (uint16_t *) buffer;
47*758e9fbaSOystein Eftevaag ptr = (uint32_t *) (buffer + 4);
48*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_ECC_POINT_Marshal(&point, buffer, buffer_size, NULL);
49*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_RC_SUCCESS);
50*758e9fbaSOystein Eftevaag /*
51*758e9fbaSOystein Eftevaag * size_ptr points to the size of the whole TPMS_ECC_POINT:
52*758e9fbaSOystein Eftevaag * sizeof(unit16) + sizeof(value) + sizeof(unit16) + sizeof(value2)
53*758e9fbaSOystein Eftevaag */
54*758e9fbaSOystein Eftevaag assert_int_equal (*size_ptr, HOST_TO_BE_16(2 + 4 + 2 + 8));
55*758e9fbaSOystein Eftevaag /* check point.x: */
56*758e9fbaSOystein Eftevaag assert_int_equal (*(size_ptr + 1), HOST_TO_BE_16(4));
57*758e9fbaSOystein Eftevaag assert_int_equal (*ptr, value);
58*758e9fbaSOystein Eftevaag size_ptr = (uint16_t *) (buffer + 2 + 2 + 4);
59*758e9fbaSOystein Eftevaag ptr2 = (uint64_t *) (buffer + 2 + 2 + 2 + 4);
60*758e9fbaSOystein Eftevaag /* check point.y: */
61*758e9fbaSOystein Eftevaag assert_int_equal (*size_ptr, HOST_TO_BE_16(8));
62*758e9fbaSOystein Eftevaag assert_int_equal (*ptr2, value2);
63*758e9fbaSOystein Eftevaag }
64*758e9fbaSOystein Eftevaag
65*758e9fbaSOystein Eftevaag /*
66*758e9fbaSOystein Eftevaag * Success case with a valid offset
67*758e9fbaSOystein Eftevaag */
68*758e9fbaSOystein Eftevaag static void
tpm2b_marshal_success_offset(void ** state)69*758e9fbaSOystein Eftevaag tpm2b_marshal_success_offset(void **state) {
70*758e9fbaSOystein Eftevaag TPM2B_DIGEST dgst = {4, {0}};
71*758e9fbaSOystein Eftevaag TPM2B_ECC_POINT point = {0};
72*758e9fbaSOystein Eftevaag size_t offset = 10;
73*758e9fbaSOystein Eftevaag uint8_t buffer[sizeof(dgst) + sizeof(point) + 10] = {0};
74*758e9fbaSOystein Eftevaag size_t buffer_size = sizeof(buffer);
75*758e9fbaSOystein Eftevaag uint16_t *size_ptr = (uint16_t *) (buffer + 10);
76*758e9fbaSOystein Eftevaag uint32_t *ptr = (uint32_t *) (buffer + 2 + 10);
77*758e9fbaSOystein Eftevaag uint32_t value = 0xdeadbeef;
78*758e9fbaSOystein Eftevaag uint64_t value2 = 0xdeadbeefdeadbeefULL;
79*758e9fbaSOystein Eftevaag uint64_t *ptr2;
80*758e9fbaSOystein Eftevaag TSS2_RC rc;
81*758e9fbaSOystein Eftevaag
82*758e9fbaSOystein Eftevaag memcpy(dgst.buffer, &value, sizeof(value));
83*758e9fbaSOystein Eftevaag memcpy(point.point.x.buffer, &value, sizeof(value));
84*758e9fbaSOystein Eftevaag point.point.x.size = sizeof(value);
85*758e9fbaSOystein Eftevaag memcpy(point.point.y.buffer, &value2, sizeof(value2));
86*758e9fbaSOystein Eftevaag point.point.y.size = sizeof(value2);
87*758e9fbaSOystein Eftevaag
88*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_DIGEST_Marshal(&dgst, buffer, buffer_size, &offset);
89*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_RC_SUCCESS);
90*758e9fbaSOystein Eftevaag assert_int_equal (*size_ptr, HOST_TO_BE_16(4));
91*758e9fbaSOystein Eftevaag assert_int_equal (*ptr, value);
92*758e9fbaSOystein Eftevaag /* check the offset */
93*758e9fbaSOystein Eftevaag assert_int_equal (offset, 10 + 2 + 4);
94*758e9fbaSOystein Eftevaag
95*758e9fbaSOystein Eftevaag size_ptr = (uint16_t *) (buffer + offset);
96*758e9fbaSOystein Eftevaag ptr = (uint32_t *) (buffer + offset + 4);
97*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_ECC_POINT_Marshal(&point, buffer, buffer_size, &offset);
98*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_RC_SUCCESS);
99*758e9fbaSOystein Eftevaag /*
100*758e9fbaSOystein Eftevaag * size_ptr points to the size of the whole TPMS_ECC_POINT:
101*758e9fbaSOystein Eftevaag * sizeof(unit16) + sizeof(value) + sizeof(unit16) + sizeof(value2)
102*758e9fbaSOystein Eftevaag */
103*758e9fbaSOystein Eftevaag assert_int_equal (*size_ptr, HOST_TO_BE_16(2 + 4 + 2 + 8));
104*758e9fbaSOystein Eftevaag /* check point.x: */
105*758e9fbaSOystein Eftevaag assert_int_equal (*(size_ptr + 1), HOST_TO_BE_16(4));
106*758e9fbaSOystein Eftevaag assert_int_equal (*ptr, value);
107*758e9fbaSOystein Eftevaag size_ptr = (uint16_t *) (buffer + 10 + 2 + 4 + 2 + 2 + 4);
108*758e9fbaSOystein Eftevaag ptr2 = (uint64_t *) (buffer + 10 + 2 + 4 + 2 + 2 + 4 + 2);
109*758e9fbaSOystein Eftevaag
110*758e9fbaSOystein Eftevaag /* check point.y: */
111*758e9fbaSOystein Eftevaag assert_int_equal (*size_ptr, HOST_TO_BE_16(8));
112*758e9fbaSOystein Eftevaag assert_int_equal (*ptr2, value2);
113*758e9fbaSOystein Eftevaag /* check the offset */
114*758e9fbaSOystein Eftevaag assert_int_equal (offset, 10 + 2 + 4 + 2 + 2 + 4 + 2 + 8);
115*758e9fbaSOystein Eftevaag }
116*758e9fbaSOystein Eftevaag
117*758e9fbaSOystein Eftevaag /*
118*758e9fbaSOystein Eftevaag * Success case with a null buffer
119*758e9fbaSOystein Eftevaag */
120*758e9fbaSOystein Eftevaag static void
tpm2b_marshal_buffer_null_with_offset(void ** state)121*758e9fbaSOystein Eftevaag tpm2b_marshal_buffer_null_with_offset(void **state)
122*758e9fbaSOystein Eftevaag {
123*758e9fbaSOystein Eftevaag TPM2B_DIGEST dgst = {4, {0}};
124*758e9fbaSOystein Eftevaag TPM2B_ECC_POINT point = {0};
125*758e9fbaSOystein Eftevaag size_t offset = 10;
126*758e9fbaSOystein Eftevaag size_t buffer_size = sizeof(dgst) + sizeof(point) + 10;
127*758e9fbaSOystein Eftevaag uint32_t value = 0xdeadbeef;
128*758e9fbaSOystein Eftevaag uint64_t value2 = 0xdeadbeefdeadbeefULL;
129*758e9fbaSOystein Eftevaag
130*758e9fbaSOystein Eftevaag TSS2_RC rc;
131*758e9fbaSOystein Eftevaag
132*758e9fbaSOystein Eftevaag memcpy(dgst.buffer, &value, sizeof(value));
133*758e9fbaSOystein Eftevaag memcpy(point.point.x.buffer, &value, sizeof(value));
134*758e9fbaSOystein Eftevaag point.point.x.size = sizeof(value);
135*758e9fbaSOystein Eftevaag memcpy(point.point.y.buffer, &value2, sizeof(value2));
136*758e9fbaSOystein Eftevaag point.point.y.size = sizeof(value2);
137*758e9fbaSOystein Eftevaag
138*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_DIGEST_Marshal(&dgst, NULL, buffer_size, &offset);
139*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_RC_SUCCESS);
140*758e9fbaSOystein Eftevaag assert_int_equal (offset, 10 + 2 + 4);
141*758e9fbaSOystein Eftevaag
142*758e9fbaSOystein Eftevaag offset = 10;
143*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_ECC_POINT_Marshal(&point, NULL, buffer_size, &offset);
144*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_RC_SUCCESS);
145*758e9fbaSOystein Eftevaag assert_int_equal (offset, 10 + 2 + 2 + sizeof(value) + 2 + sizeof(value2));
146*758e9fbaSOystein Eftevaag offset = 0;
147*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_DIGEST_Marshal(&dgst, NULL, buffer_size, &offset);
148*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_RC_SUCCESS);
149*758e9fbaSOystein Eftevaag /*
150*758e9fbaSOystein Eftevaag * TSS MU spec states:
151*758e9fbaSOystein Eftevaag * If the 'buffer' parameter is NULL the implementation shall not write
152*758e9fbaSOystein Eftevaag * any marshaled data but the 'offset' parameter shall be updated as
153*758e9fbaSOystein Eftevaag * though it had.
154*758e9fbaSOystein Eftevaag * The offset of call with NULL and not NULL buffer will be compared.
155*758e9fbaSOystein Eftevaag */
156*758e9fbaSOystein Eftevaag uint8_t buffer[offset];
157*758e9fbaSOystein Eftevaag
158*758e9fbaSOystein Eftevaag size_t offset1 = 0;
159*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_DIGEST_Marshal(&dgst, NULL, buffer_size, &offset1);
160*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_RC_SUCCESS);
161*758e9fbaSOystein Eftevaag
162*758e9fbaSOystein Eftevaag size_t offset2 = 0;
163*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_DIGEST_Marshal(&dgst, buffer, buffer_size, &offset2);
164*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_RC_SUCCESS);
165*758e9fbaSOystein Eftevaag assert_int_equal(offset1, offset2);
166*758e9fbaSOystein Eftevaag
167*758e9fbaSOystein Eftevaag }
168*758e9fbaSOystein Eftevaag
169*758e9fbaSOystein Eftevaag /*
170*758e9fbaSOystein Eftevaag * Invalid case with a null buffer and a null offset
171*758e9fbaSOystein Eftevaag */
172*758e9fbaSOystein Eftevaag static void
tpm2b_marshal_buffer_null_offset_null(void ** state)173*758e9fbaSOystein Eftevaag tpm2b_marshal_buffer_null_offset_null(void **state)
174*758e9fbaSOystein Eftevaag {
175*758e9fbaSOystein Eftevaag TPM2B_DIGEST dgst = {4, {0}};
176*758e9fbaSOystein Eftevaag TPM2B_ECC_POINT point = {0};
177*758e9fbaSOystein Eftevaag size_t buffer_size = 1024;
178*758e9fbaSOystein Eftevaag TSS2_RC rc;
179*758e9fbaSOystein Eftevaag
180*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_DIGEST_Marshal(&dgst, NULL, buffer_size, NULL);
181*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_MU_RC_BAD_REFERENCE);
182*758e9fbaSOystein Eftevaag
183*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_ECC_POINT_Marshal(&point, NULL, buffer_size, NULL);
184*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_MU_RC_BAD_REFERENCE);
185*758e9fbaSOystein Eftevaag }
186*758e9fbaSOystein Eftevaag
187*758e9fbaSOystein Eftevaag /*
188*758e9fbaSOystein Eftevaag * Invalid case with not big enough buffer
189*758e9fbaSOystein Eftevaag */
190*758e9fbaSOystein Eftevaag static void
tpm2b_marshal_buffer_size_lt_data_nad_lt_offset(void ** state)191*758e9fbaSOystein Eftevaag tpm2b_marshal_buffer_size_lt_data_nad_lt_offset(void **state) {
192*758e9fbaSOystein Eftevaag TPM2B_DIGEST dgst = {4, {0}};
193*758e9fbaSOystein Eftevaag TPM2B_ECC_POINT point = {0};
194*758e9fbaSOystein Eftevaag size_t offset = 10;
195*758e9fbaSOystein Eftevaag uint8_t buffer[sizeof(dgst) + sizeof(point)] = {0};
196*758e9fbaSOystein Eftevaag size_t buffer_size = sizeof(buffer);
197*758e9fbaSOystein Eftevaag uint32_t value = 0xdeadbeef;
198*758e9fbaSOystein Eftevaag uint64_t value2 = 0xdeadbeefdeadbeefULL;
199*758e9fbaSOystein Eftevaag TSS2_RC rc;
200*758e9fbaSOystein Eftevaag
201*758e9fbaSOystein Eftevaag memcpy(dgst.buffer, &value, sizeof(value));
202*758e9fbaSOystein Eftevaag memcpy(point.point.x.buffer, &value, sizeof(value));
203*758e9fbaSOystein Eftevaag point.point.x.size = sizeof(value);
204*758e9fbaSOystein Eftevaag memcpy(point.point.y.buffer, &value2, sizeof(value2));
205*758e9fbaSOystein Eftevaag point.point.y.size = sizeof(value2);
206*758e9fbaSOystein Eftevaag
207*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_DIGEST_Marshal(&dgst, buffer, buffer_size, &offset);
208*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_RC_SUCCESS);
209*758e9fbaSOystein Eftevaag
210*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_ECC_POINT_Marshal(&point, buffer, buffer_size, &offset);
211*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_RC_SUCCESS);
212*758e9fbaSOystein Eftevaag }
213*758e9fbaSOystein Eftevaag
214*758e9fbaSOystein Eftevaag /*
215*758e9fbaSOystein Eftevaag * Unmarshal success case
216*758e9fbaSOystein Eftevaag */
217*758e9fbaSOystein Eftevaag static void
tpm2b_unmarshal_success(void ** state)218*758e9fbaSOystein Eftevaag tpm2b_unmarshal_success(void **state)
219*758e9fbaSOystein Eftevaag {
220*758e9fbaSOystein Eftevaag TPM2B_DIGEST dgst = {0};
221*758e9fbaSOystein Eftevaag TPM2B_ECC_POINT point = {0};
222*758e9fbaSOystein Eftevaag size_t offset = 0;
223*758e9fbaSOystein Eftevaag uint8_t buffer[] = { 0x00, 0x04, 0xef, 0xbe, 0xad, 0xde, /* digest of 4 bytes */
224*758e9fbaSOystein Eftevaag 0x00, 0x0c, /* size of TPM2B_ECC_POINT */
225*758e9fbaSOystein Eftevaag 0x00, 0x04, 0xef, 0xbe, 0xad, 0xde, /* ECC_POINT.x - 4 bytes */
226*758e9fbaSOystein Eftevaag 0x00, 0x04, 0x44, 0x33, 0x22, 0x11 }; /* ECC_POINT.y - 4 bytes */
227*758e9fbaSOystein Eftevaag
228*758e9fbaSOystein Eftevaag size_t buffer_size = sizeof(buffer);
229*758e9fbaSOystein Eftevaag uint32_t value = 0xdeadbeef;
230*758e9fbaSOystein Eftevaag uint32_t value2 = 0x11223344;
231*758e9fbaSOystein Eftevaag uint32_t val;
232*758e9fbaSOystein Eftevaag TSS2_RC rc;
233*758e9fbaSOystein Eftevaag
234*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_DIGEST_Unmarshal(buffer, buffer_size, &offset, &dgst);
235*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_RC_SUCCESS);
236*758e9fbaSOystein Eftevaag assert_int_equal (dgst.size, 4);
237*758e9fbaSOystein Eftevaag memcpy(&val, dgst.buffer, sizeof(val));
238*758e9fbaSOystein Eftevaag assert_int_equal (le32toh(val), value);
239*758e9fbaSOystein Eftevaag assert_int_equal (offset, 6);
240*758e9fbaSOystein Eftevaag
241*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_ECC_POINT_Unmarshal(buffer, buffer_size, &offset, &point);
242*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_RC_SUCCESS);
243*758e9fbaSOystein Eftevaag assert_int_equal (point.point.x.size, 4);
244*758e9fbaSOystein Eftevaag memcpy(&val, point.point.x.buffer, sizeof(val));
245*758e9fbaSOystein Eftevaag assert_int_equal (le32toh(val), value);
246*758e9fbaSOystein Eftevaag assert_int_equal (point.point.y.size, 4);
247*758e9fbaSOystein Eftevaag memcpy(&val, point.point.y.buffer, sizeof(val));
248*758e9fbaSOystein Eftevaag assert_int_equal (le32toh(val), value2);
249*758e9fbaSOystein Eftevaag assert_int_equal (offset, 20);
250*758e9fbaSOystein Eftevaag }
251*758e9fbaSOystein Eftevaag
252*758e9fbaSOystein Eftevaag /*
253*758e9fbaSOystein Eftevaag * Unmarshal success case with offset
254*758e9fbaSOystein Eftevaag */
255*758e9fbaSOystein Eftevaag static void
tpm2b_unmarshal_success_offset(void ** state)256*758e9fbaSOystein Eftevaag tpm2b_unmarshal_success_offset(void **state)
257*758e9fbaSOystein Eftevaag {
258*758e9fbaSOystein Eftevaag TPM2B_DIGEST dgst = {0};
259*758e9fbaSOystein Eftevaag TPM2B_ECC_POINT point = {0};
260*758e9fbaSOystein Eftevaag size_t offset = 6;
261*758e9fbaSOystein Eftevaag uint8_t buffer[] = { 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, /* random 6 bytes offset */
262*758e9fbaSOystein Eftevaag 0x00, 0x04, 0xef, 0xbe, 0xad, 0xde, /* digest of 4 bytes */
263*758e9fbaSOystein Eftevaag 0x00, 0x10, /* size of TPM2B_ECC_POINT - 16 bytes */
264*758e9fbaSOystein Eftevaag 0x00, 0x08, 0xef, 0xbe, 0xad, 0xde, 0xef, 0xbe, 0xad, 0xde, /* ECC_POINT.x - 8 bytes */
265*758e9fbaSOystein Eftevaag 0x00, 0x04, 0x44, 0x33, 0x22, 0x11 }; /* ECC_POINT.y - 4 bytes */
266*758e9fbaSOystein Eftevaag
267*758e9fbaSOystein Eftevaag size_t buffer_size = sizeof(buffer);
268*758e9fbaSOystein Eftevaag uint32_t value = 0xdeadbeef;
269*758e9fbaSOystein Eftevaag uint64_t value2 = 0xdeadbeefdeadbeefULL;
270*758e9fbaSOystein Eftevaag uint32_t value3 = 0x11223344;
271*758e9fbaSOystein Eftevaag uint32_t val;
272*758e9fbaSOystein Eftevaag uint64_t val2;
273*758e9fbaSOystein Eftevaag TSS2_RC rc;
274*758e9fbaSOystein Eftevaag
275*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_DIGEST_Unmarshal(buffer, buffer_size, &offset, &dgst);
276*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_RC_SUCCESS);
277*758e9fbaSOystein Eftevaag assert_int_equal (dgst.size, 4);
278*758e9fbaSOystein Eftevaag memcpy(&val, dgst.buffer, sizeof(val));
279*758e9fbaSOystein Eftevaag assert_int_equal (le32toh(val), value);
280*758e9fbaSOystein Eftevaag assert_int_equal (offset, 12);
281*758e9fbaSOystein Eftevaag
282*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_ECC_POINT_Unmarshal(buffer, buffer_size, &offset, &point);
283*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_RC_SUCCESS);
284*758e9fbaSOystein Eftevaag assert_int_equal (point.point.x.size, 8);
285*758e9fbaSOystein Eftevaag memcpy(&val2, point.point.x.buffer, sizeof(val2));
286*758e9fbaSOystein Eftevaag assert_int_equal (le64toh(val2), value2);
287*758e9fbaSOystein Eftevaag assert_int_equal (point.point.y.size, 4);
288*758e9fbaSOystein Eftevaag memcpy(&val, point.point.y.buffer, sizeof(val));
289*758e9fbaSOystein Eftevaag assert_int_equal (le32toh(val), value3);
290*758e9fbaSOystein Eftevaag assert_int_equal (offset, 30);
291*758e9fbaSOystein Eftevaag }
292*758e9fbaSOystein Eftevaag
293*758e9fbaSOystein Eftevaag /*
294*758e9fbaSOystein Eftevaag * Test case ensures a NULL buffer parameter produces a BAD_REFERENCE RC.
295*758e9fbaSOystein Eftevaag */
296*758e9fbaSOystein Eftevaag void
tpm2b_unmarshal_buffer_null(void ** state)297*758e9fbaSOystein Eftevaag tpm2b_unmarshal_buffer_null (void **state)
298*758e9fbaSOystein Eftevaag {
299*758e9fbaSOystein Eftevaag TPM2B_DIGEST dgst = {0};
300*758e9fbaSOystein Eftevaag TPM2B_ECC_POINT point = {0};
301*758e9fbaSOystein Eftevaag TSS2_RC rc;
302*758e9fbaSOystein Eftevaag size_t offset = 0;
303*758e9fbaSOystein Eftevaag
304*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_DIGEST_Unmarshal (NULL, 1, NULL, NULL);
305*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_MU_RC_BAD_REFERENCE);
306*758e9fbaSOystein Eftevaag
307*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_ECC_POINT_Unmarshal (NULL, 1, NULL, NULL);
308*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_MU_RC_BAD_REFERENCE);
309*758e9fbaSOystein Eftevaag
310*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_DIGEST_Unmarshal (NULL, 1, &offset, NULL);
311*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_MU_RC_BAD_REFERENCE);
312*758e9fbaSOystein Eftevaag
313*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_ECC_POINT_Unmarshal (NULL, 1, &offset, NULL);
314*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_MU_RC_BAD_REFERENCE);
315*758e9fbaSOystein Eftevaag
316*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_DIGEST_Unmarshal (NULL, 1, NULL, &dgst);
317*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_MU_RC_BAD_REFERENCE);
318*758e9fbaSOystein Eftevaag
319*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_ECC_POINT_Unmarshal (NULL, 1, NULL, &point);
320*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_MU_RC_BAD_REFERENCE);
321*758e9fbaSOystein Eftevaag }
322*758e9fbaSOystein Eftevaag /*
323*758e9fbaSOystein Eftevaag * Test case ensures a NULL dest and offset parameters produce an
324*758e9fbaSOystein Eftevaag * INSUFFICIENT_BUFFER RC.
325*758e9fbaSOystein Eftevaag */
326*758e9fbaSOystein Eftevaag void
tpm2b_unmarshal_dest_null(void ** state)327*758e9fbaSOystein Eftevaag tpm2b_unmarshal_dest_null (void **state)
328*758e9fbaSOystein Eftevaag {
329*758e9fbaSOystein Eftevaag uint8_t buffer [1];
330*758e9fbaSOystein Eftevaag TSS2_RC rc;
331*758e9fbaSOystein Eftevaag
332*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_DIGEST_Unmarshal (buffer, sizeof (buffer), NULL, NULL);
333*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_MU_RC_BAD_REFERENCE);
334*758e9fbaSOystein Eftevaag
335*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_ECC_POINT_Unmarshal (buffer, 1, NULL, NULL);
336*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_MU_RC_BAD_REFERENCE);
337*758e9fbaSOystein Eftevaag }
338*758e9fbaSOystein Eftevaag
339*758e9fbaSOystein Eftevaag /*
340*758e9fbaSOystein Eftevaag * Test case ensures the offset is updated when dest is NULL
341*758e9fbaSOystein Eftevaag * and offset is valid
342*758e9fbaSOystein Eftevaag */
343*758e9fbaSOystein Eftevaag void
tpm2b_unmarshal_dest_null_offset_valid(void ** state)344*758e9fbaSOystein Eftevaag tpm2b_unmarshal_dest_null_offset_valid (void **state)
345*758e9fbaSOystein Eftevaag {
346*758e9fbaSOystein Eftevaag size_t offset = 0;
347*758e9fbaSOystein Eftevaag uint8_t buffer[] = { 0x00, 0x04, 0xef, 0xbe, 0xad, 0xde, /* digest of 4 bytes */
348*758e9fbaSOystein Eftevaag 0x00, 0x10, /* size of TPM2B_ECC_POINT - 16 bytes */
349*758e9fbaSOystein Eftevaag 0x00, 0x08, 0xef, 0xbe, 0xad, 0xde, 0xef, 0xbe, 0xad, 0xde, /* ECC_POINT.x - 8 bytes */
350*758e9fbaSOystein Eftevaag 0x00, 0x04, 0x44, 0x33, 0x22, 0x11 }; /* ECC_POINT.y - 4 bytes */
351*758e9fbaSOystein Eftevaag TSS2_RC rc;
352*758e9fbaSOystein Eftevaag
353*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_DIGEST_Unmarshal (buffer, sizeof (buffer), &offset, NULL);
354*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_RC_SUCCESS);
355*758e9fbaSOystein Eftevaag assert_int_equal (offset, 6);
356*758e9fbaSOystein Eftevaag
357*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_ECC_POINT_Unmarshal (buffer, sizeof (buffer), &offset, NULL);
358*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_RC_SUCCESS);
359*758e9fbaSOystein Eftevaag assert_int_equal (offset, 24);
360*758e9fbaSOystein Eftevaag }
361*758e9fbaSOystein Eftevaag
362*758e9fbaSOystein Eftevaag /*
363*758e9fbaSOystein Eftevaag * Invalid case with not big enough buffer
364*758e9fbaSOystein Eftevaag */
365*758e9fbaSOystein Eftevaag static void
tpm2b_unmarshal_buffer_size_lt_data_nad_lt_offset(void ** state)366*758e9fbaSOystein Eftevaag tpm2b_unmarshal_buffer_size_lt_data_nad_lt_offset(void **state)
367*758e9fbaSOystein Eftevaag {
368*758e9fbaSOystein Eftevaag TPM2B_DIGEST dgst = {4, {0x00, 0x01, 0x02, 0x03}};
369*758e9fbaSOystein Eftevaag TPM2B_ECC_POINT point = {0};
370*758e9fbaSOystein Eftevaag uint8_t buffer[sizeof(dgst) + sizeof(point)] = { 0 };
371*758e9fbaSOystein Eftevaag size_t offset = sizeof(dgst) - 5;
372*758e9fbaSOystein Eftevaag TSS2_RC rc;
373*758e9fbaSOystein Eftevaag
374*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_DIGEST_Unmarshal (buffer, 6, &offset, &dgst);
375*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_MU_RC_INSUFFICIENT_BUFFER);
376*758e9fbaSOystein Eftevaag assert_int_equal (offset, sizeof(dgst) - 5);
377*758e9fbaSOystein Eftevaag
378*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_ECC_POINT_Unmarshal (buffer, 1, &offset, &point);
379*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_MU_RC_INSUFFICIENT_BUFFER);
380*758e9fbaSOystein Eftevaag assert_int_equal (offset, sizeof(dgst) - 5);
381*758e9fbaSOystein Eftevaag }
382*758e9fbaSOystein Eftevaag
383*758e9fbaSOystein Eftevaag /*
384*758e9fbaSOystein Eftevaag * Success case
385*758e9fbaSOystein Eftevaag */
386*758e9fbaSOystein Eftevaag static void
tpm2b_public_rsa_marshal_success(void ** state)387*758e9fbaSOystein Eftevaag tpm2b_public_rsa_marshal_success(void **state) {
388*758e9fbaSOystein Eftevaag TPM2B_PUBLIC pub2b = {0};
389*758e9fbaSOystein Eftevaag TPMT_PUBLIC *pub = &pub2b.publicArea;
390*758e9fbaSOystein Eftevaag uint8_t buffer[sizeof(pub2b)] = { 0 };
391*758e9fbaSOystein Eftevaag size_t buffer_size = sizeof(buffer);
392*758e9fbaSOystein Eftevaag TPM2B_PUBLIC *ptr1;
393*758e9fbaSOystein Eftevaag TSS2_RC rc;
394*758e9fbaSOystein Eftevaag
395*758e9fbaSOystein Eftevaag pub->type = TPM2_ALG_RSA;
396*758e9fbaSOystein Eftevaag pub->parameters.rsaDetail.symmetric.algorithm = TPM2_ALG_AES;
397*758e9fbaSOystein Eftevaag pub->parameters.rsaDetail.symmetric.keyBits.aes = 128;
398*758e9fbaSOystein Eftevaag pub->parameters.rsaDetail.symmetric.mode.aes = TPM2_ALG_CBC;
399*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_PUBLIC_Marshal(&pub2b, buffer, buffer_size, NULL);
400*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_RC_SUCCESS);
401*758e9fbaSOystein Eftevaag ptr1 = (TPM2B_PUBLIC *)buffer;
402*758e9fbaSOystein Eftevaag assert_int_equal (ptr1->size, HOST_TO_BE_16(0x1a));
403*758e9fbaSOystein Eftevaag }
404*758e9fbaSOystein Eftevaag
405*758e9fbaSOystein Eftevaag /*
406*758e9fbaSOystein Eftevaag * Success case
407*758e9fbaSOystein Eftevaag */
408*758e9fbaSOystein Eftevaag static void
tpm2b_public_rsa_unique_size_marshal_success(void ** state)409*758e9fbaSOystein Eftevaag tpm2b_public_rsa_unique_size_marshal_success(void **state) {
410*758e9fbaSOystein Eftevaag TPM2B_PUBLIC pub2b = {0};
411*758e9fbaSOystein Eftevaag TPMT_PUBLIC *pub = &pub2b.publicArea;
412*758e9fbaSOystein Eftevaag uint8_t buffer[sizeof(pub2b)] = { 0 };
413*758e9fbaSOystein Eftevaag size_t buffer_size = sizeof(buffer);
414*758e9fbaSOystein Eftevaag TPM2B_PUBLIC *ptr1;
415*758e9fbaSOystein Eftevaag TSS2_RC rc;
416*758e9fbaSOystein Eftevaag
417*758e9fbaSOystein Eftevaag pub->type = TPM2_ALG_RSA;
418*758e9fbaSOystein Eftevaag pub->parameters.rsaDetail.symmetric.algorithm = TPM2_ALG_AES;
419*758e9fbaSOystein Eftevaag pub->parameters.rsaDetail.symmetric.keyBits.aes = 128;
420*758e9fbaSOystein Eftevaag pub->parameters.rsaDetail.symmetric.mode.aes = TPM2_ALG_CBC;
421*758e9fbaSOystein Eftevaag pub->unique.rsa.size = 0x100;
422*758e9fbaSOystein Eftevaag rc = Tss2_MU_TPM2B_PUBLIC_Marshal(&pub2b, buffer, buffer_size, NULL);
423*758e9fbaSOystein Eftevaag assert_int_equal (rc, TSS2_RC_SUCCESS);
424*758e9fbaSOystein Eftevaag ptr1 = (TPM2B_PUBLIC *)buffer;
425*758e9fbaSOystein Eftevaag assert_int_equal (ptr1->size, HOST_TO_BE_16(0x11a));
426*758e9fbaSOystein Eftevaag }
427*758e9fbaSOystein Eftevaag
main(void)428*758e9fbaSOystein Eftevaag int main(void) {
429*758e9fbaSOystein Eftevaag const struct CMUnitTest tests[] = {
430*758e9fbaSOystein Eftevaag cmocka_unit_test(tpm2b_marshal_success),
431*758e9fbaSOystein Eftevaag cmocka_unit_test(tpm2b_marshal_success_offset),
432*758e9fbaSOystein Eftevaag cmocka_unit_test(tpm2b_marshal_buffer_null_with_offset),
433*758e9fbaSOystein Eftevaag cmocka_unit_test(tpm2b_marshal_buffer_null_offset_null),
434*758e9fbaSOystein Eftevaag cmocka_unit_test(tpm2b_marshal_buffer_size_lt_data_nad_lt_offset),
435*758e9fbaSOystein Eftevaag cmocka_unit_test(tpm2b_unmarshal_success),
436*758e9fbaSOystein Eftevaag cmocka_unit_test(tpm2b_unmarshal_success_offset),
437*758e9fbaSOystein Eftevaag cmocka_unit_test(tpm2b_unmarshal_buffer_null),
438*758e9fbaSOystein Eftevaag cmocka_unit_test(tpm2b_unmarshal_dest_null),
439*758e9fbaSOystein Eftevaag cmocka_unit_test(tpm2b_unmarshal_dest_null_offset_valid),
440*758e9fbaSOystein Eftevaag cmocka_unit_test(tpm2b_unmarshal_buffer_size_lt_data_nad_lt_offset),
441*758e9fbaSOystein Eftevaag cmocka_unit_test(tpm2b_public_rsa_marshal_success),
442*758e9fbaSOystein Eftevaag cmocka_unit_test(tpm2b_public_rsa_unique_size_marshal_success),
443*758e9fbaSOystein Eftevaag };
444*758e9fbaSOystein Eftevaag return cmocka_run_group_tests(tests, NULL, NULL);
445*758e9fbaSOystein Eftevaag }
446