1*49cdfc7eSAndroid Build Coastguard Worker // SPDX-License-Identifier: GPL-2.0-or-later
2*49cdfc7eSAndroid Build Coastguard Worker /*
3*49cdfc7eSAndroid Build Coastguard Worker * Copyright 2019 Google LLC
4*49cdfc7eSAndroid Build Coastguard Worker * Copyright (c) Linux Test Project, 2019-2021
5*49cdfc7eSAndroid Build Coastguard Worker */
6*49cdfc7eSAndroid Build Coastguard Worker
7*49cdfc7eSAndroid Build Coastguard Worker #include <errno.h>
8*49cdfc7eSAndroid Build Coastguard Worker #include <stdlib.h>
9*49cdfc7eSAndroid Build Coastguard Worker
10*49cdfc7eSAndroid Build Coastguard Worker #define TST_NO_DEFAULT_MAIN
11*49cdfc7eSAndroid Build Coastguard Worker #include "tst_test.h"
12*49cdfc7eSAndroid Build Coastguard Worker #include "tst_af_alg.h"
13*49cdfc7eSAndroid Build Coastguard Worker #include "lapi/socket.h"
14*49cdfc7eSAndroid Build Coastguard Worker
tst_alg_create(void)15*49cdfc7eSAndroid Build Coastguard Worker int tst_alg_create(void)
16*49cdfc7eSAndroid Build Coastguard Worker {
17*49cdfc7eSAndroid Build Coastguard Worker const long ret = socket(AF_ALG, SOCK_SEQPACKET, 0);
18*49cdfc7eSAndroid Build Coastguard Worker
19*49cdfc7eSAndroid Build Coastguard Worker if (ret >= 0)
20*49cdfc7eSAndroid Build Coastguard Worker return ret;
21*49cdfc7eSAndroid Build Coastguard Worker if (errno == EAFNOSUPPORT)
22*49cdfc7eSAndroid Build Coastguard Worker tst_brk(TCONF, "kernel doesn't support AF_ALG");
23*49cdfc7eSAndroid Build Coastguard Worker tst_brk(TBROK | TERRNO, "unexpected error creating AF_ALG socket");
24*49cdfc7eSAndroid Build Coastguard Worker return -1;
25*49cdfc7eSAndroid Build Coastguard Worker }
26*49cdfc7eSAndroid Build Coastguard Worker
tst_alg_bind_addr(int algfd,const struct sockaddr_alg * addr)27*49cdfc7eSAndroid Build Coastguard Worker void tst_alg_bind_addr(int algfd, const struct sockaddr_alg *addr)
28*49cdfc7eSAndroid Build Coastguard Worker {
29*49cdfc7eSAndroid Build Coastguard Worker const long ret = bind(algfd, (const struct sockaddr *)addr,
30*49cdfc7eSAndroid Build Coastguard Worker sizeof(*addr));
31*49cdfc7eSAndroid Build Coastguard Worker
32*49cdfc7eSAndroid Build Coastguard Worker if (ret == 0)
33*49cdfc7eSAndroid Build Coastguard Worker return;
34*49cdfc7eSAndroid Build Coastguard Worker
35*49cdfc7eSAndroid Build Coastguard Worker if (errno == ELIBBAD && tst_fips_enabled()) {
36*49cdfc7eSAndroid Build Coastguard Worker tst_brk(TCONF,
37*49cdfc7eSAndroid Build Coastguard Worker "FIPS enabled => %s algorithm '%s' disabled",
38*49cdfc7eSAndroid Build Coastguard Worker addr->salg_type, addr->salg_name);
39*49cdfc7eSAndroid Build Coastguard Worker }
40*49cdfc7eSAndroid Build Coastguard Worker
41*49cdfc7eSAndroid Build Coastguard Worker if (errno == ENOENT) {
42*49cdfc7eSAndroid Build Coastguard Worker tst_brk(TCONF, "kernel doesn't support %s algorithm '%s'",
43*49cdfc7eSAndroid Build Coastguard Worker addr->salg_type, addr->salg_name);
44*49cdfc7eSAndroid Build Coastguard Worker }
45*49cdfc7eSAndroid Build Coastguard Worker
46*49cdfc7eSAndroid Build Coastguard Worker tst_brk(TBROK | TERRNO,
47*49cdfc7eSAndroid Build Coastguard Worker "unexpected error binding AF_ALG socket to %s algorithm '%s'",
48*49cdfc7eSAndroid Build Coastguard Worker addr->salg_type, addr->salg_name);
49*49cdfc7eSAndroid Build Coastguard Worker }
50*49cdfc7eSAndroid Build Coastguard Worker
init_sockaddr_alg(struct sockaddr_alg * addr,const char * algtype,const char * algname)51*49cdfc7eSAndroid Build Coastguard Worker static void init_sockaddr_alg(struct sockaddr_alg *addr,
52*49cdfc7eSAndroid Build Coastguard Worker const char *algtype, const char *algname)
53*49cdfc7eSAndroid Build Coastguard Worker {
54*49cdfc7eSAndroid Build Coastguard Worker memset(addr, 0, sizeof(*addr));
55*49cdfc7eSAndroid Build Coastguard Worker
56*49cdfc7eSAndroid Build Coastguard Worker addr->salg_family = AF_ALG;
57*49cdfc7eSAndroid Build Coastguard Worker
58*49cdfc7eSAndroid Build Coastguard Worker strncpy((char *)addr->salg_type, algtype, sizeof(addr->salg_type));
59*49cdfc7eSAndroid Build Coastguard Worker if (addr->salg_type[sizeof(addr->salg_type) - 1] != '\0')
60*49cdfc7eSAndroid Build Coastguard Worker tst_brk(TBROK, "algorithm type too long: '%s'", algtype);
61*49cdfc7eSAndroid Build Coastguard Worker
62*49cdfc7eSAndroid Build Coastguard Worker strncpy((char *)addr->salg_name, algname, sizeof(addr->salg_name));
63*49cdfc7eSAndroid Build Coastguard Worker if (addr->salg_name[sizeof(addr->salg_name) - 1] != '\0')
64*49cdfc7eSAndroid Build Coastguard Worker tst_brk(TBROK, "algorithm name too long: '%s'", algname);
65*49cdfc7eSAndroid Build Coastguard Worker }
66*49cdfc7eSAndroid Build Coastguard Worker
tst_alg_bind(int algfd,const char * algtype,const char * algname)67*49cdfc7eSAndroid Build Coastguard Worker void tst_alg_bind(int algfd, const char *algtype, const char *algname)
68*49cdfc7eSAndroid Build Coastguard Worker {
69*49cdfc7eSAndroid Build Coastguard Worker struct sockaddr_alg addr;
70*49cdfc7eSAndroid Build Coastguard Worker
71*49cdfc7eSAndroid Build Coastguard Worker init_sockaddr_alg(&addr, algtype, algname);
72*49cdfc7eSAndroid Build Coastguard Worker
73*49cdfc7eSAndroid Build Coastguard Worker tst_alg_bind_addr(algfd, &addr);
74*49cdfc7eSAndroid Build Coastguard Worker }
75*49cdfc7eSAndroid Build Coastguard Worker
tst_try_alg(const char * algtype,const char * algname)76*49cdfc7eSAndroid Build Coastguard Worker int tst_try_alg(const char *algtype, const char *algname)
77*49cdfc7eSAndroid Build Coastguard Worker {
78*49cdfc7eSAndroid Build Coastguard Worker long ret;
79*49cdfc7eSAndroid Build Coastguard Worker int retval = 0;
80*49cdfc7eSAndroid Build Coastguard Worker int algfd;
81*49cdfc7eSAndroid Build Coastguard Worker struct sockaddr_alg addr;
82*49cdfc7eSAndroid Build Coastguard Worker
83*49cdfc7eSAndroid Build Coastguard Worker algfd = tst_alg_create();
84*49cdfc7eSAndroid Build Coastguard Worker
85*49cdfc7eSAndroid Build Coastguard Worker init_sockaddr_alg(&addr, algtype, algname);
86*49cdfc7eSAndroid Build Coastguard Worker
87*49cdfc7eSAndroid Build Coastguard Worker ret = bind(algfd, (const struct sockaddr *)&addr, sizeof(addr));
88*49cdfc7eSAndroid Build Coastguard Worker
89*49cdfc7eSAndroid Build Coastguard Worker if (ret != 0)
90*49cdfc7eSAndroid Build Coastguard Worker retval = errno;
91*49cdfc7eSAndroid Build Coastguard Worker
92*49cdfc7eSAndroid Build Coastguard Worker close(algfd);
93*49cdfc7eSAndroid Build Coastguard Worker return retval;
94*49cdfc7eSAndroid Build Coastguard Worker }
95*49cdfc7eSAndroid Build Coastguard Worker
tst_have_alg(const char * algtype,const char * algname)96*49cdfc7eSAndroid Build Coastguard Worker bool tst_have_alg(const char *algtype, const char *algname)
97*49cdfc7eSAndroid Build Coastguard Worker {
98*49cdfc7eSAndroid Build Coastguard Worker int ret;
99*49cdfc7eSAndroid Build Coastguard Worker
100*49cdfc7eSAndroid Build Coastguard Worker ret = tst_try_alg(algtype, algname);
101*49cdfc7eSAndroid Build Coastguard Worker
102*49cdfc7eSAndroid Build Coastguard Worker switch (ret) {
103*49cdfc7eSAndroid Build Coastguard Worker case 0:
104*49cdfc7eSAndroid Build Coastguard Worker return true;
105*49cdfc7eSAndroid Build Coastguard Worker case ENOENT:
106*49cdfc7eSAndroid Build Coastguard Worker tst_res(TCONF, "kernel doesn't have %s algorithm '%s'",
107*49cdfc7eSAndroid Build Coastguard Worker algtype, algname);
108*49cdfc7eSAndroid Build Coastguard Worker return false;
109*49cdfc7eSAndroid Build Coastguard Worker case ELIBBAD:
110*49cdfc7eSAndroid Build Coastguard Worker if (tst_fips_enabled()) {
111*49cdfc7eSAndroid Build Coastguard Worker tst_res(TCONF,
112*49cdfc7eSAndroid Build Coastguard Worker "FIPS enabled => %s algorithm '%s' disabled",
113*49cdfc7eSAndroid Build Coastguard Worker algtype, algname);
114*49cdfc7eSAndroid Build Coastguard Worker return false;
115*49cdfc7eSAndroid Build Coastguard Worker }
116*49cdfc7eSAndroid Build Coastguard Worker /* fallthrough */
117*49cdfc7eSAndroid Build Coastguard Worker default:
118*49cdfc7eSAndroid Build Coastguard Worker errno = ret;
119*49cdfc7eSAndroid Build Coastguard Worker tst_brk(TBROK | TERRNO,
120*49cdfc7eSAndroid Build Coastguard Worker "unexpected error binding AF_ALG socket to %s algorithm '%s'",
121*49cdfc7eSAndroid Build Coastguard Worker algtype, algname);
122*49cdfc7eSAndroid Build Coastguard Worker return false;
123*49cdfc7eSAndroid Build Coastguard Worker }
124*49cdfc7eSAndroid Build Coastguard Worker }
125*49cdfc7eSAndroid Build Coastguard Worker
tst_require_alg(const char * algtype,const char * algname)126*49cdfc7eSAndroid Build Coastguard Worker void tst_require_alg(const char *algtype, const char *algname)
127*49cdfc7eSAndroid Build Coastguard Worker {
128*49cdfc7eSAndroid Build Coastguard Worker int algfd = tst_alg_create();
129*49cdfc7eSAndroid Build Coastguard Worker
130*49cdfc7eSAndroid Build Coastguard Worker tst_alg_bind(algfd, algtype, algname);
131*49cdfc7eSAndroid Build Coastguard Worker
132*49cdfc7eSAndroid Build Coastguard Worker close(algfd);
133*49cdfc7eSAndroid Build Coastguard Worker }
134*49cdfc7eSAndroid Build Coastguard Worker
tst_alg_setkey(int algfd,const uint8_t * key,unsigned int keylen)135*49cdfc7eSAndroid Build Coastguard Worker void tst_alg_setkey(int algfd, const uint8_t *key, unsigned int keylen)
136*49cdfc7eSAndroid Build Coastguard Worker {
137*49cdfc7eSAndroid Build Coastguard Worker long ret;
138*49cdfc7eSAndroid Build Coastguard Worker uint8_t *keybuf = NULL;
139*49cdfc7eSAndroid Build Coastguard Worker unsigned int i;
140*49cdfc7eSAndroid Build Coastguard Worker
141*49cdfc7eSAndroid Build Coastguard Worker if (key == NULL) {
142*49cdfc7eSAndroid Build Coastguard Worker /* generate a random key */
143*49cdfc7eSAndroid Build Coastguard Worker keybuf = SAFE_MALLOC(keylen);
144*49cdfc7eSAndroid Build Coastguard Worker for (i = 0; i < keylen; i++)
145*49cdfc7eSAndroid Build Coastguard Worker keybuf[i] = rand();
146*49cdfc7eSAndroid Build Coastguard Worker key = keybuf;
147*49cdfc7eSAndroid Build Coastguard Worker }
148*49cdfc7eSAndroid Build Coastguard Worker ret = setsockopt(algfd, SOL_ALG, ALG_SET_KEY, key, keylen);
149*49cdfc7eSAndroid Build Coastguard Worker if (ret != 0) {
150*49cdfc7eSAndroid Build Coastguard Worker tst_brk(TBROK | TERRNO,
151*49cdfc7eSAndroid Build Coastguard Worker "unexpected error setting key (len=%u)", keylen);
152*49cdfc7eSAndroid Build Coastguard Worker }
153*49cdfc7eSAndroid Build Coastguard Worker free(keybuf);
154*49cdfc7eSAndroid Build Coastguard Worker }
155*49cdfc7eSAndroid Build Coastguard Worker
tst_alg_accept(int algfd)156*49cdfc7eSAndroid Build Coastguard Worker int tst_alg_accept(int algfd)
157*49cdfc7eSAndroid Build Coastguard Worker {
158*49cdfc7eSAndroid Build Coastguard Worker const long ret = accept(algfd, NULL, NULL);
159*49cdfc7eSAndroid Build Coastguard Worker
160*49cdfc7eSAndroid Build Coastguard Worker if (ret < 0) {
161*49cdfc7eSAndroid Build Coastguard Worker tst_brk(TBROK | TERRNO,
162*49cdfc7eSAndroid Build Coastguard Worker "unexpected error accept()ing AF_ALG request socket");
163*49cdfc7eSAndroid Build Coastguard Worker }
164*49cdfc7eSAndroid Build Coastguard Worker return ret;
165*49cdfc7eSAndroid Build Coastguard Worker }
166*49cdfc7eSAndroid Build Coastguard Worker
tst_alg_setup(const char * algtype,const char * algname,const uint8_t * key,unsigned int keylen)167*49cdfc7eSAndroid Build Coastguard Worker int tst_alg_setup(const char *algtype, const char *algname,
168*49cdfc7eSAndroid Build Coastguard Worker const uint8_t *key, unsigned int keylen)
169*49cdfc7eSAndroid Build Coastguard Worker {
170*49cdfc7eSAndroid Build Coastguard Worker int algfd = tst_alg_create();
171*49cdfc7eSAndroid Build Coastguard Worker
172*49cdfc7eSAndroid Build Coastguard Worker tst_alg_bind(algfd, algtype, algname);
173*49cdfc7eSAndroid Build Coastguard Worker
174*49cdfc7eSAndroid Build Coastguard Worker if (keylen != 0)
175*49cdfc7eSAndroid Build Coastguard Worker tst_alg_setkey(algfd, key, keylen);
176*49cdfc7eSAndroid Build Coastguard Worker
177*49cdfc7eSAndroid Build Coastguard Worker return algfd;
178*49cdfc7eSAndroid Build Coastguard Worker }
179*49cdfc7eSAndroid Build Coastguard Worker
tst_alg_setup_reqfd(const char * algtype,const char * algname,const uint8_t * key,unsigned int keylen)180*49cdfc7eSAndroid Build Coastguard Worker int tst_alg_setup_reqfd(const char *algtype, const char *algname,
181*49cdfc7eSAndroid Build Coastguard Worker const uint8_t *key, unsigned int keylen)
182*49cdfc7eSAndroid Build Coastguard Worker {
183*49cdfc7eSAndroid Build Coastguard Worker int algfd = tst_alg_setup(algtype, algname, key, keylen);
184*49cdfc7eSAndroid Build Coastguard Worker int reqfd = tst_alg_accept(algfd);
185*49cdfc7eSAndroid Build Coastguard Worker
186*49cdfc7eSAndroid Build Coastguard Worker close(algfd);
187*49cdfc7eSAndroid Build Coastguard Worker return reqfd;
188*49cdfc7eSAndroid Build Coastguard Worker }
189*49cdfc7eSAndroid Build Coastguard Worker
tst_alg_sendmsg(int reqfd,const void * data,size_t datalen,const struct tst_alg_sendmsg_params * params)190*49cdfc7eSAndroid Build Coastguard Worker void tst_alg_sendmsg(int reqfd, const void *data, size_t datalen,
191*49cdfc7eSAndroid Build Coastguard Worker const struct tst_alg_sendmsg_params *params)
192*49cdfc7eSAndroid Build Coastguard Worker {
193*49cdfc7eSAndroid Build Coastguard Worker struct iovec iov = {
194*49cdfc7eSAndroid Build Coastguard Worker .iov_base = (void *)data,
195*49cdfc7eSAndroid Build Coastguard Worker .iov_len = datalen,
196*49cdfc7eSAndroid Build Coastguard Worker };
197*49cdfc7eSAndroid Build Coastguard Worker struct msghdr msg = {
198*49cdfc7eSAndroid Build Coastguard Worker .msg_iov = &iov,
199*49cdfc7eSAndroid Build Coastguard Worker .msg_iovlen = 1,
200*49cdfc7eSAndroid Build Coastguard Worker .msg_flags = params->msg_flags,
201*49cdfc7eSAndroid Build Coastguard Worker };
202*49cdfc7eSAndroid Build Coastguard Worker size_t controllen;
203*49cdfc7eSAndroid Build Coastguard Worker uint8_t *control;
204*49cdfc7eSAndroid Build Coastguard Worker struct cmsghdr *cmsg;
205*49cdfc7eSAndroid Build Coastguard Worker struct af_alg_iv *alg_iv;
206*49cdfc7eSAndroid Build Coastguard Worker
207*49cdfc7eSAndroid Build Coastguard Worker if (params->encrypt && params->decrypt)
208*49cdfc7eSAndroid Build Coastguard Worker tst_brk(TBROK, "Both encrypt and decrypt are specified");
209*49cdfc7eSAndroid Build Coastguard Worker
210*49cdfc7eSAndroid Build Coastguard Worker controllen = 0;
211*49cdfc7eSAndroid Build Coastguard Worker if (params->encrypt || params->decrypt)
212*49cdfc7eSAndroid Build Coastguard Worker controllen += CMSG_SPACE(sizeof(uint32_t));
213*49cdfc7eSAndroid Build Coastguard Worker if (params->ivlen)
214*49cdfc7eSAndroid Build Coastguard Worker controllen += CMSG_SPACE(sizeof(struct af_alg_iv) +
215*49cdfc7eSAndroid Build Coastguard Worker params->ivlen);
216*49cdfc7eSAndroid Build Coastguard Worker if (params->assoclen)
217*49cdfc7eSAndroid Build Coastguard Worker controllen += CMSG_SPACE(sizeof(uint32_t));
218*49cdfc7eSAndroid Build Coastguard Worker
219*49cdfc7eSAndroid Build Coastguard Worker control = SAFE_MALLOC(controllen);
220*49cdfc7eSAndroid Build Coastguard Worker memset(control, 0, controllen);
221*49cdfc7eSAndroid Build Coastguard Worker msg.msg_control = control;
222*49cdfc7eSAndroid Build Coastguard Worker msg.msg_controllen = controllen;
223*49cdfc7eSAndroid Build Coastguard Worker cmsg = CMSG_FIRSTHDR(&msg);
224*49cdfc7eSAndroid Build Coastguard Worker
225*49cdfc7eSAndroid Build Coastguard Worker if (params->encrypt || params->decrypt) {
226*49cdfc7eSAndroid Build Coastguard Worker cmsg->cmsg_level = SOL_ALG;
227*49cdfc7eSAndroid Build Coastguard Worker cmsg->cmsg_type = ALG_SET_OP;
228*49cdfc7eSAndroid Build Coastguard Worker cmsg->cmsg_len = CMSG_LEN(sizeof(uint32_t));
229*49cdfc7eSAndroid Build Coastguard Worker *(uint32_t *)CMSG_DATA(cmsg) =
230*49cdfc7eSAndroid Build Coastguard Worker params->encrypt ? ALG_OP_ENCRYPT : ALG_OP_DECRYPT;
231*49cdfc7eSAndroid Build Coastguard Worker cmsg = CMSG_NXTHDR(&msg, cmsg);
232*49cdfc7eSAndroid Build Coastguard Worker }
233*49cdfc7eSAndroid Build Coastguard Worker if (params->ivlen) {
234*49cdfc7eSAndroid Build Coastguard Worker cmsg->cmsg_level = SOL_ALG;
235*49cdfc7eSAndroid Build Coastguard Worker cmsg->cmsg_type = ALG_SET_IV;
236*49cdfc7eSAndroid Build Coastguard Worker cmsg->cmsg_len = CMSG_LEN(sizeof(struct af_alg_iv) +
237*49cdfc7eSAndroid Build Coastguard Worker params->ivlen);
238*49cdfc7eSAndroid Build Coastguard Worker alg_iv = (struct af_alg_iv *)CMSG_DATA(cmsg);
239*49cdfc7eSAndroid Build Coastguard Worker alg_iv->ivlen = params->ivlen;
240*49cdfc7eSAndroid Build Coastguard Worker memcpy(alg_iv->iv, params->iv, params->ivlen);
241*49cdfc7eSAndroid Build Coastguard Worker cmsg = CMSG_NXTHDR(&msg, cmsg);
242*49cdfc7eSAndroid Build Coastguard Worker }
243*49cdfc7eSAndroid Build Coastguard Worker if (params->assoclen) {
244*49cdfc7eSAndroid Build Coastguard Worker cmsg->cmsg_level = SOL_ALG;
245*49cdfc7eSAndroid Build Coastguard Worker cmsg->cmsg_type = ALG_SET_AEAD_ASSOCLEN;
246*49cdfc7eSAndroid Build Coastguard Worker cmsg->cmsg_len = CMSG_LEN(sizeof(uint32_t));
247*49cdfc7eSAndroid Build Coastguard Worker *(uint32_t *)CMSG_DATA(cmsg) = params->assoclen;
248*49cdfc7eSAndroid Build Coastguard Worker cmsg = CMSG_NXTHDR(&msg, cmsg);
249*49cdfc7eSAndroid Build Coastguard Worker }
250*49cdfc7eSAndroid Build Coastguard Worker
251*49cdfc7eSAndroid Build Coastguard Worker SAFE_SENDMSG(datalen, reqfd, &msg, 0);
252*49cdfc7eSAndroid Build Coastguard Worker }
253