1*7bb1a6cfStb /* $OpenBSD: asn1evp.c,v 1.5 2022/09/05 21:06:31 tb Exp $ */
2bda5caccSjsing /*
3bda5caccSjsing * Copyright (c) 2017 Joel Sing <jsing@openbsd.org>
4bda5caccSjsing *
5bda5caccSjsing * Permission to use, copy, modify, and distribute this software for any
6bda5caccSjsing * purpose with or without fee is hereby granted, provided that the above
7bda5caccSjsing * copyright notice and this permission notice appear in all copies.
8bda5caccSjsing *
9bda5caccSjsing * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
10bda5caccSjsing * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
11bda5caccSjsing * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
12bda5caccSjsing * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
13bda5caccSjsing * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
14bda5caccSjsing * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
15bda5caccSjsing * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
16bda5caccSjsing */
17bda5caccSjsing
18bda5caccSjsing #include <limits.h>
19bda5caccSjsing #include <stdio.h>
20bda5caccSjsing #include <string.h>
21bda5caccSjsing
22bda5caccSjsing #include <openssl/asn1.h>
23bda5caccSjsing
244254f679Sjsing #define TEST_NUM 0x7fffffffL
254254f679Sjsing
26bda5caccSjsing unsigned char asn1_atios[] = {
274254f679Sjsing 0x30, 0x10, 0x02, 0x04, 0x7f, 0xff, 0xff, 0xff,
284254f679Sjsing 0x04, 0x08, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05,
294254f679Sjsing 0x06, 0x07,
30bda5caccSjsing };
31bda5caccSjsing
32bda5caccSjsing unsigned char test_octetstring[] = {
33bda5caccSjsing 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07,
34bda5caccSjsing };
35bda5caccSjsing
36bda5caccSjsing static void
hexdump(const unsigned char * buf,size_t len)37bda5caccSjsing hexdump(const unsigned char *buf, size_t len)
38bda5caccSjsing {
39bda5caccSjsing size_t i;
40bda5caccSjsing
41bda5caccSjsing for (i = 1; i <= len; i++)
42bda5caccSjsing fprintf(stderr, " 0x%02hhx,%s", buf[i - 1], i % 8 ? "" : "\n");
43bda5caccSjsing
44bda5caccSjsing fprintf(stderr, "\n");
45bda5caccSjsing }
46bda5caccSjsing
47bda5caccSjsing static int
compare_data(const char * label,const unsigned char * d1,size_t d1_len,const unsigned char * d2,size_t d2_len)48bda5caccSjsing compare_data(const char *label, const unsigned char *d1, size_t d1_len,
49bda5caccSjsing const unsigned char *d2, size_t d2_len)
50bda5caccSjsing {
51bda5caccSjsing if (d1_len != d2_len) {
52bda5caccSjsing fprintf(stderr, "FAIL: got %s with length %zu, want %zu\n",
53bda5caccSjsing label, d1_len, d2_len);
54bda5caccSjsing return -1;
55bda5caccSjsing }
56bda5caccSjsing if (memcmp(d1, d2, d1_len) != 0) {
57bda5caccSjsing fprintf(stderr, "FAIL: %s differs\n", label);
58bda5caccSjsing fprintf(stderr, "got:\n");
59bda5caccSjsing hexdump(d1, d1_len);
60bda5caccSjsing fprintf(stderr, "want:\n");
61bda5caccSjsing hexdump(d2, d2_len);
62bda5caccSjsing return -1;
63bda5caccSjsing }
64bda5caccSjsing return 0;
65bda5caccSjsing }
66bda5caccSjsing
67bda5caccSjsing int
main(int argc,char ** argv)68bda5caccSjsing main(int argc, char **argv)
69bda5caccSjsing {
70bda5caccSjsing unsigned char data[16];
714254f679Sjsing long num = TEST_NUM;
72c1d56b54Sjsing ASN1_TYPE *at = NULL;
73bda5caccSjsing int failed = 1;
74bda5caccSjsing int len;
75bda5caccSjsing
76c1d56b54Sjsing if ((at = ASN1_TYPE_new()) == NULL) {
77c1d56b54Sjsing fprintf(stderr, "FAIL: ASN1_TYPE_new returned NULL\n");
78c1d56b54Sjsing goto done;
79c1d56b54Sjsing }
80bda5caccSjsing
81c1d56b54Sjsing if (!ASN1_TYPE_set_int_octetstring(at, num, test_octetstring,
82bda5caccSjsing sizeof(test_octetstring))) {
83bda5caccSjsing fprintf(stderr, "FAIL: ASN1_TYPE_set_int_octetstring failed\n");
84bda5caccSjsing goto done;
85bda5caccSjsing }
86c1d56b54Sjsing if (at->type != V_ASN1_SEQUENCE) {
87*7bb1a6cfStb fprintf(stderr, "FAIL: not a V_ASN1_SEQUENCE (%d != %d)\n",
88c1d56b54Sjsing at->type, V_ASN1_SEQUENCE);
89bda5caccSjsing goto done;
90bda5caccSjsing }
91c1d56b54Sjsing if (at->value.sequence->type != V_ASN1_OCTET_STRING) {
92*7bb1a6cfStb fprintf(stderr, "FAIL: not a V_ASN1_OCTET_STRING (%d != %d)\n",
93c1d56b54Sjsing at->type, V_ASN1_OCTET_STRING);
94bda5caccSjsing goto done;
95bda5caccSjsing }
96c1d56b54Sjsing if (compare_data("sequence", at->value.sequence->data,
97c1d56b54Sjsing at->value.sequence->length, asn1_atios, sizeof(asn1_atios)) == -1)
98bda5caccSjsing goto done;
99bda5caccSjsing
100bda5caccSjsing memset(&data, 0, sizeof(data));
101bda5caccSjsing num = 0;
102bda5caccSjsing
103c1d56b54Sjsing if ((len = ASN1_TYPE_get_int_octetstring(at, &num, data,
104bda5caccSjsing sizeof(data))) < 0) {
105bda5caccSjsing fprintf(stderr, "FAIL: ASN1_TYPE_get_int_octetstring failed\n");
106bda5caccSjsing goto done;
107bda5caccSjsing }
1084254f679Sjsing if (num != TEST_NUM) {
109*7bb1a6cfStb fprintf(stderr, "FAIL: got num %ld, want %ld\n", num, TEST_NUM);
110bda5caccSjsing goto done;
111bda5caccSjsing }
112bda5caccSjsing if (compare_data("octet string", data, len,
113bda5caccSjsing test_octetstring, sizeof(test_octetstring)) == -1)
114bda5caccSjsing goto done;
115bda5caccSjsing if (data[len] != 0) {
116bda5caccSjsing fprintf(stderr, "FAIL: octet string overflowed buffer\n");
117bda5caccSjsing goto done;
118bda5caccSjsing }
119bda5caccSjsing
120bda5caccSjsing memset(&data, 0, sizeof(data));
121bda5caccSjsing num = 0;
122bda5caccSjsing
123bda5caccSjsing /* With a limit buffer, the output should be truncated... */
124c1d56b54Sjsing if ((len = ASN1_TYPE_get_int_octetstring(at, &num, data, 4)) < 0) {
125bda5caccSjsing fprintf(stderr, "FAIL: ASN1_TYPE_get_int_octetstring failed\n");
126bda5caccSjsing goto done;
127bda5caccSjsing }
1284254f679Sjsing if (num != TEST_NUM) {
129*7bb1a6cfStb fprintf(stderr, "FAIL: got num %ld, want %ld\n", num, TEST_NUM);
130bda5caccSjsing goto done;
131bda5caccSjsing }
132bda5caccSjsing if (len != sizeof(test_octetstring)) {
133*7bb1a6cfStb fprintf(stderr, "FAIL: got length mismatch (%d != %zu)\n",
134bda5caccSjsing len, sizeof(test_octetstring));
135bda5caccSjsing goto done;
136bda5caccSjsing }
137bda5caccSjsing if (compare_data("octet string", data, 4, test_octetstring, 4) == -1)
138bda5caccSjsing goto done;
139bda5caccSjsing if (data[4] != 0) {
140bda5caccSjsing fprintf(stderr, "FAIL: octet string overflowed buffer\n");
141bda5caccSjsing goto done;
142bda5caccSjsing }
143bda5caccSjsing
144bda5caccSjsing failed = 0;
145bda5caccSjsing
146bda5caccSjsing done:
147c1d56b54Sjsing ASN1_TYPE_free(at);
148c1d56b54Sjsing
149bda5caccSjsing return failed;
150bda5caccSjsing }
151