2 * Copyright (c) 2018 Cisco and/or its affiliates.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at:
7 * http://www.apache.org/licenses/LICENSE-2.0
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
17 #include <vlib/vlib.h>
18 #include <vnet/crypto/crypto.h>
20 vnet_crypto_main_t crypto_main;
22 static_always_inline u32
23 vnet_crypto_process_ops_call_handler (vlib_main_t * vm,
24 vnet_crypto_main_t * cm,
25 vnet_crypto_op_id_t opt,
26 vnet_crypto_op_t * ops[], u32 n_ops)
31 if (cm->ops_handlers[opt] == 0)
35 ops[0]->status = VNET_CRYPTO_OP_STATUS_FAIL_NO_HANDLER;
41 return (cm->ops_handlers[opt]) (vm, ops, n_ops);
46 vnet_crypto_process_ops (vlib_main_t * vm, vnet_crypto_op_t ops[], u32 n_ops)
48 vnet_crypto_main_t *cm = &crypto_main;
49 const int op_q_size = VLIB_FRAME_SIZE;
50 vnet_crypto_op_t *op_queue[op_q_size];
51 vnet_crypto_op_id_t opt, current_op_type = ~0;
57 for (i = 0; i < n_ops; i++)
61 if (current_op_type != opt || n_op_queue >= op_q_size)
63 rv += vnet_crypto_process_ops_call_handler (vm, cm, current_op_type,
64 op_queue, n_op_queue);
66 current_op_type = opt;
69 op_queue[n_op_queue++] = &ops[i];
72 rv += vnet_crypto_process_ops_call_handler (vm, cm, current_op_type,
73 op_queue, n_op_queue);
78 vnet_crypto_register_engine (vlib_main_t * vm, char *name, int prio,
81 vnet_crypto_main_t *cm = &crypto_main;
82 vnet_crypto_engine_t *p;
84 vec_add2 (cm->engines, p, 1);
89 hash_set_mem (cm->engine_index_by_name, p->name, p - cm->engines);
91 return p - cm->engines;
95 vnet_crypto_set_handler (char *alg_name, char *engine)
98 vnet_crypto_main_t *cm = &crypto_main;
99 vnet_crypto_alg_data_t *ad;
100 vnet_crypto_engine_t *ce;
103 p = hash_get_mem (cm->alg_index_by_name, alg_name);
107 ad = vec_elt_at_index (cm->algs, p[0]);
109 p = hash_get_mem (cm->engine_index_by_name, engine);
113 ce = vec_elt_at_index (cm->engines, p[0]);
115 for (i = 0; i < VNET_CRYPTO_OP_N_TYPES; i++)
117 vnet_crypto_op_data_t *od;
118 vnet_crypto_op_id_t id = ad->op_by_type[i];
121 od = vec_elt_at_index (cm->opt_data, id);
122 if (ce->ops_handlers[id])
124 od->active_engine_index = p[0];
125 cm->ops_handlers[id] = ce->ops_handlers[id];
133 vnet_crypto_register_ops_handler (vlib_main_t * vm, u32 engine_index,
134 vnet_crypto_op_id_t opt,
135 vnet_crypto_ops_handler_t * fn)
137 vnet_crypto_main_t *cm = &crypto_main;
138 vnet_crypto_engine_t *ae, *e = vec_elt_at_index (cm->engines, engine_index);
139 vnet_crypto_op_data_t *otd = cm->opt_data + opt;
140 vec_validate_aligned (cm->ops_handlers, VNET_CRYPTO_N_OP_IDS - 1,
141 CLIB_CACHE_LINE_BYTES);
142 e->ops_handlers[opt] = fn;
144 if (otd->active_engine_index == ~0)
146 otd->active_engine_index = engine_index;
147 cm->ops_handlers[opt] = fn;
150 ae = vec_elt_at_index (cm->engines, otd->active_engine_index);
151 if (ae->priority < e->priority)
153 otd->active_engine_index = engine_index;
154 cm->ops_handlers[opt] = fn;
161 vnet_crypto_register_key_handler (vlib_main_t * vm, u32 engine_index,
162 vnet_crypto_key_handler_t * key_handler)
164 vnet_crypto_main_t *cm = &crypto_main;
165 vnet_crypto_engine_t *e = vec_elt_at_index (cm->engines, engine_index);
166 e->key_op_handler = key_handler;
171 vnet_crypto_key_add (vlib_main_t * vm, vnet_crypto_alg_t alg, u8 * data,
175 vnet_crypto_main_t *cm = &crypto_main;
176 vnet_crypto_engine_t *engine;
177 vnet_crypto_key_t *key;
178 pool_get_zero (cm->keys, key);
179 index = key - cm->keys;
181 vec_validate_aligned (key->data, length - 1, CLIB_CACHE_LINE_BYTES);
182 clib_memcpy (key->data, data, length);
185 vec_foreach (engine, cm->engines)
186 if (engine->key_op_handler)
187 engine->key_op_handler (vm, VNET_CRYPTO_KEY_OP_ADD, index);
193 vnet_crypto_key_del (vlib_main_t * vm, vnet_crypto_key_index_t index)
195 vnet_crypto_main_t *cm = &crypto_main;
196 vnet_crypto_engine_t *engine;
197 vnet_crypto_key_t *key = pool_elt_at_index (cm->keys, index);
200 vec_foreach (engine, cm->engines)
201 if (engine->key_op_handler)
202 engine->key_op_handler (vm, VNET_CRYPTO_KEY_OP_DEL, index);
205 clib_memset (key->data, 0, vec_len (key->data));
206 vec_free (key->data);
207 pool_put (cm->keys, key);
211 vnet_crypto_key_modify (vlib_main_t * vm, vnet_crypto_key_index_t index,
212 vnet_crypto_alg_t alg, u8 * data, u16 length)
214 vnet_crypto_main_t *cm = &crypto_main;
215 vnet_crypto_engine_t *engine;
216 vnet_crypto_key_t *key = pool_elt_at_index (cm->keys, index);
218 if (vec_len (key->data))
219 clib_memset (key->data, 0, vec_len (key->data));
220 vec_free (key->data);
221 vec_validate_aligned (key->data, length - 1, CLIB_CACHE_LINE_BYTES);
222 clib_memcpy (key->data, data, length);
226 vec_foreach (engine, cm->engines)
227 if (engine->key_op_handler)
228 engine->key_op_handler (vm, VNET_CRYPTO_KEY_OP_MODIFY, index);
233 vnet_crypto_init_cipher_data (vnet_crypto_alg_t alg, vnet_crypto_op_id_t eid,
234 vnet_crypto_op_id_t did, char *name, u8 is_aead)
236 vnet_crypto_op_type_t eopt, dopt;
237 vnet_crypto_main_t *cm = &crypto_main;
238 cm->algs[alg].name = name;
239 cm->opt_data[eid].alg = cm->opt_data[did].alg = alg;
240 cm->opt_data[eid].active_engine_index = ~0;
241 cm->opt_data[did].active_engine_index = ~0;
244 eopt = VNET_CRYPTO_OP_TYPE_AEAD_ENCRYPT;
245 dopt = VNET_CRYPTO_OP_TYPE_AEAD_DECRYPT;
249 eopt = VNET_CRYPTO_OP_TYPE_ENCRYPT;
250 dopt = VNET_CRYPTO_OP_TYPE_DECRYPT;
252 cm->opt_data[eid].type = eopt;
253 cm->opt_data[did].type = dopt;
254 cm->algs[alg].op_by_type[eopt] = eid;
255 cm->algs[alg].op_by_type[dopt] = did;
256 hash_set_mem (cm->alg_index_by_name, name, alg);
260 vnet_crypto_init_hmac_data (vnet_crypto_alg_t alg,
261 vnet_crypto_op_id_t id, char *name)
263 vnet_crypto_main_t *cm = &crypto_main;
264 cm->algs[alg].name = name;
265 cm->algs[alg].op_by_type[VNET_CRYPTO_OP_TYPE_HMAC] = id;
266 cm->opt_data[id].alg = alg;
267 cm->opt_data[id].active_engine_index = ~0;
268 cm->opt_data[id].type = VNET_CRYPTO_OP_TYPE_HMAC;
269 hash_set_mem (cm->alg_index_by_name, name, alg);
273 vnet_crypto_init (vlib_main_t * vm)
275 vnet_crypto_main_t *cm = &crypto_main;
276 vlib_thread_main_t *tm = vlib_get_thread_main ();
277 cm->engine_index_by_name = hash_create_string ( /* size */ 0,
279 cm->alg_index_by_name = hash_create_string (0, sizeof (uword));
280 vec_validate_aligned (cm->threads, tm->n_vlib_mains, CLIB_CACHE_LINE_BYTES);
281 vec_validate (cm->algs, VNET_CRYPTO_N_ALGS);
283 vnet_crypto_init_cipher_data (VNET_CRYPTO_ALG_##n, \
284 VNET_CRYPTO_OP_##n##_ENC, \
285 VNET_CRYPTO_OP_##n##_DEC, s, 0);
286 foreach_crypto_cipher_alg;
289 vnet_crypto_init_cipher_data (VNET_CRYPTO_ALG_##n, \
290 VNET_CRYPTO_OP_##n##_ENC, \
291 VNET_CRYPTO_OP_##n##_DEC, s, 1);
292 foreach_crypto_aead_alg;
295 vnet_crypto_init_hmac_data (VNET_CRYPTO_ALG_HMAC_##n, \
296 VNET_CRYPTO_OP_##n##_HMAC, "hmac-" s);
297 foreach_crypto_hmac_alg;
302 VLIB_INIT_FUNCTION (vnet_crypto_init);
305 * fd.io coding-style-patch-verification: ON
308 * eval: (c-set-style "gnu")