/* * Copyright (c) 2019 Winner Microelectronics Co., Ltd. * * SPDX-License-Identifier: Apache-2.0 * * Change Logs: * Date Author Notes * 2019-05-17 tyx 1st version */ #include <rtthread.h> #include <rtdevice.h> #include <stdlib.h> #include <string.h> #include "drv_crypto.h" #include "wm_crypto_hard.h" // #define WM_HWCRYPTO_NOT_LOCK // #define WM_HWCRYPTO_NOT_ALIGN_CHECK extern u32 tls_crypto_crc_update_adv(u32 crc_val, CRYPTO_CRC_TYPE type, u8 mode, const unsigned char *in, u32 len); extern int tls_crypto_des_encrypt_decrypt_adv(unsigned char *key, unsigned char *IV, const unsigned char *in, unsigned char *out, u32 len, CRYPTO_MODE cbc, CRYPTO_WAY dec); extern int tls_crypto_3des_encrypt_decrypt_adv(unsigned char *key, unsigned char *IV, const unsigned char *in, unsigned char *out, u32 len, CRYPTO_MODE cbc, CRYPTO_WAY dec); extern int tls_crypto_aes_encrypt_decrypt_adv(unsigned char *key, unsigned char *IV, const unsigned char *in, unsigned char *out, u32 len, CRYPTO_MODE cbc, CRYPTO_WAY dec); extern int tls_crypto_rc4_adv(unsigned char *key, u32 keylen, const unsigned char *in, unsigned char *out, u32 len); struct wm_hwcrypto_device { struct rt_hwcrypto_device dev; struct rt_mutex mutex; }; struct hash_ctx_des { psDigestContext_t contex; }; static rt_uint32_t _rng_rand(struct hwcrypto_rng *ctx) { rt_uint32_t rand_num; struct wm_hwcrypto_device *_hwcrypto = (struct wm_hwcrypto_device *)ctx->parent.device; #if !defined(WM_HWCRYPTO_NOT_LOCK) rt_mutex_take(&_hwcrypto->mutex, RT_WAITING_FOREVER); #endif tls_crypto_random_init((u32)rt_tick_get(), CRYPTO_RNG_SWITCH_32); tls_crypto_random_bytes((unsigned char *)&rand_num, sizeof(rand_num)); tls_crypto_random_stop(); #if !defined(WM_HWCRYPTO_NOT_LOCK) rt_mutex_release(&_hwcrypto->mutex); #endif return rand_num; } static rt_uint32_t _crc_update(struct hwcrypto_crc *ctx, const rt_uint8_t *in, rt_size_t length) { rt_uint32_t crc_result; CRYPTO_CRC_TYPE type; u8 mode = 0; struct wm_hwcrypto_device *_hwcrypto = (struct wm_hwcrypto_device *)ctx->parent.device; unsigned char align_flag = 0; #if !defined(WM_HWCRYPTO_NOT_ALIGN_CHECK) if (((rt_uint32_t)in % 4) != 0) { void *temp; temp = rt_malloc(length); if (temp) { memcpy(temp, in, length); in = temp; align_flag = 1; } else { return 0; } } #endif switch (ctx->crc_cfg.poly) { case 0x04C11DB7: type = CRYPTO_CRC_TYPE_32; break; case 0x00001021: type = CRYPTO_CRC_TYPE_16_CCITT; break; case 0x00008005: type = CRYPTO_CRC_TYPE_16_MODBUS; break; case 0x00000007: case 0x00000207: type = CRYPTO_CRC_TYPE_8; break; default: return 0; } mode |= ctx->crc_cfg.flags & CRC_FLAG_REFOUT ? OUTPUT_REFLECT : 0; mode |= ctx->crc_cfg.flags & CRC_FLAG_REFIN ? INPUT_REFLECT : 0; #if !defined(WM_HWCRYPTO_NOT_LOCK) rt_mutex_take(&_hwcrypto->mutex, RT_WAITING_FOREVER); #endif crc_result = tls_crypto_crc_update_adv(ctx->crc_cfg.last_val, type, mode, in, length); #if !defined(WM_HWCRYPTO_NOT_LOCK) rt_mutex_release(&_hwcrypto->mutex); #endif #if !defined(WM_HWCRYPTO_NOT_ALIGN_CHECK) if (align_flag) { rt_free((rt_uint8_t *)in); } #endif ctx->crc_cfg.last_val = crc_result; return crc_result ^ 0x00 ^ ctx->crc_cfg.xorout; } static rt_err_t _hash_update(struct hwcrypto_hash *ctx, const rt_uint8_t *in, rt_size_t length) { rt_err_t err = RT_EOK; struct wm_hwcrypto_device *_hwcrypto = (struct wm_hwcrypto_device *)ctx->parent.device; unsigned char align_flag = 0; #if !defined(WM_HWCRYPTO_NOT_ALIGN_CHECK) if (((rt_uint32_t)in % 4) != 0) { void *temp; temp = rt_malloc(length); if (temp) { memcpy(temp, in, length); in = temp; align_flag = 1; } else { return -RT_ENOMEM; } } #endif #if !defined(WM_HWCRYPTO_NOT_LOCK) rt_mutex_take(&_hwcrypto->mutex, RT_WAITING_FOREVER); #endif switch (ctx->parent.type & HWCRYPTO_MAIN_TYPE_MASK) { case HWCRYPTO_TYPE_MD5: tls_crypto_md5_update(&((struct hash_ctx_des *)(ctx->parent.contex))->contex, in, length); break; case HWCRYPTO_TYPE_SHA1: tls_crypto_sha1_update(&((struct hash_ctx_des *)(ctx->parent.contex))->contex, in, length); break; default: err = -RT_ERROR; break; } #if !defined(WM_HWCRYPTO_NOT_LOCK) rt_mutex_release(&_hwcrypto->mutex); #endif #if !defined(WM_HWCRYPTO_NOT_ALIGN_CHECK) if (align_flag) { rt_free((rt_uint8_t *)in); } #endif return err; } static rt_err_t _hash_finish(struct hwcrypto_hash *ctx, rt_uint8_t *out, rt_size_t length) { rt_err_t err = RT_EOK; struct wm_hwcrypto_device *_hwcrypto = (struct wm_hwcrypto_device *)ctx->parent.device; #if !defined(WM_HWCRYPTO_NOT_LOCK) rt_mutex_take(&_hwcrypto->mutex, RT_WAITING_FOREVER); #endif switch (ctx->parent.type & HWCRYPTO_MAIN_TYPE_MASK) { case HWCRYPTO_TYPE_MD5: tls_crypto_md5_final(&((struct hash_ctx_des *)(ctx->parent.contex))->contex, out); break; case HWCRYPTO_TYPE_SHA1: tls_crypto_sha1_final(&((struct hash_ctx_des *)(ctx->parent.contex))->contex, out); break; default: err = -RT_ERROR; break; } #if !defined(WM_HWCRYPTO_NOT_LOCK) rt_mutex_release(&_hwcrypto->mutex); #endif return err; } static rt_err_t _des_crypt(struct hwcrypto_symmetric *symmetric_ctx, struct hwcrypto_symmetric_info *symmetric_info) { CRYPTO_WAY mode; CRYPTO_MODE cbc; struct wm_hwcrypto_device *_hwcrypto = (struct wm_hwcrypto_device *)symmetric_ctx->parent.device; unsigned char *in, *out, align_flag = 0; if ((symmetric_ctx->key_bitlen >> 3) != 8 || (symmetric_info->length % 8) != 0) { return -RT_EINVAL; } in = (unsigned char *)symmetric_info->in; out = (unsigned char *)symmetric_info->out; #if !defined(WM_HWCRYPTO_NOT_ALIGN_CHECK) if (((rt_uint32_t)in % 4) != 0 || ((rt_uint32_t)out % 4) != 0) { in = rt_malloc(symmetric_info->length); if (in) { memcpy(in, symmetric_info->in, symmetric_info->length); out = in; align_flag = 1; } else { return -RT_ENOMEM; } } #endif mode = symmetric_info->mode == HWCRYPTO_MODE_ENCRYPT ? CRYPTO_WAY_ENCRYPT : CRYPTO_WAY_DECRYPT; switch (symmetric_ctx->parent.type & (HWCRYPTO_MAIN_TYPE_MASK | HWCRYPTO_SUB_TYPE_MASK)) { case HWCRYPTO_TYPE_DES_ECB: cbc = CRYPTO_MODE_ECB; break; case HWCRYPTO_TYPE_DES_CBC: cbc = CRYPTO_MODE_CBC; break; default : return -RT_ERROR; } #if !defined(WM_HWCRYPTO_NOT_LOCK) rt_mutex_take(&_hwcrypto->mutex, RT_WAITING_FOREVER); #endif tls_crypto_des_encrypt_decrypt_adv(symmetric_ctx->key, symmetric_ctx->iv, symmetric_info->in, symmetric_info->out, symmetric_info->length, cbc, mode); #if !defined(WM_HWCRYPTO_NOT_LOCK) rt_mutex_release(&_hwcrypto->mutex); #endif #if !defined(WM_HWCRYPTO_NOT_ALIGN_CHECK) if (align_flag) { memcpy(symmetric_info->out, out, symmetric_info->length); rt_free(in); } #endif return RT_EOK; } static rt_err_t _des3_crypt(struct hwcrypto_symmetric *symmetric_ctx, struct hwcrypto_symmetric_info *symmetric_info) { CRYPTO_WAY mode; CRYPTO_MODE cbc; struct wm_hwcrypto_device *_hwcrypto = (struct wm_hwcrypto_device *)symmetric_ctx->parent.device; unsigned char *in, *out, align_flag = 0; if ((symmetric_ctx->key_bitlen >> 3) != 24 || (symmetric_info->length % 8) != 0) { return -RT_EINVAL; } in = (unsigned char *)symmetric_info->in; out = (unsigned char *)symmetric_info->out; #if !defined(WM_HWCRYPTO_NOT_ALIGN_CHECK) if (((rt_uint32_t)in % 4) != 0 || ((rt_uint32_t)out % 4) != 0) { in = rt_malloc(symmetric_info->length); if (in) { memcpy(in, symmetric_info->in, symmetric_info->length); out = in; align_flag = 1; } else { return -RT_ENOMEM; } } #endif mode = symmetric_info->mode == HWCRYPTO_MODE_ENCRYPT ? CRYPTO_WAY_ENCRYPT : CRYPTO_WAY_DECRYPT; switch (symmetric_ctx->parent.type & (HWCRYPTO_MAIN_TYPE_MASK | HWCRYPTO_SUB_TYPE_MASK)) { case HWCRYPTO_TYPE_3DES_ECB: cbc = CRYPTO_MODE_ECB; break; case HWCRYPTO_TYPE_3DES_CBC: cbc = CRYPTO_MODE_CBC; break; default : return -RT_ERROR; } #if !defined(WM_HWCRYPTO_NOT_LOCK) rt_mutex_take(&_hwcrypto->mutex, RT_WAITING_FOREVER); #endif tls_crypto_3des_encrypt_decrypt_adv(symmetric_ctx->key, symmetric_ctx->iv, symmetric_info->in, symmetric_info->out, symmetric_info->length, cbc, mode); #if !defined(WM_HWCRYPTO_NOT_LOCK) rt_mutex_release(&_hwcrypto->mutex); #endif #if !defined(WM_HWCRYPTO_NOT_ALIGN_CHECK) if (align_flag) { memcpy(symmetric_info->out, out, symmetric_info->length); rt_free(in); } #endif return RT_EOK; } static rt_err_t _rc4_crypt(struct hwcrypto_symmetric *symmetric_ctx, struct hwcrypto_symmetric_info *symmetric_info) { struct wm_hwcrypto_device *_hwcrypto = (struct wm_hwcrypto_device *)symmetric_ctx->parent.device; unsigned char *in, *out, align_flag = 0; if ((symmetric_ctx->key_bitlen >> 3) != 16) { return -RT_EINVAL; } in = (unsigned char *)symmetric_info->in; out = (unsigned char *)symmetric_info->out; #if !defined(WM_HWCRYPTO_NOT_ALIGN_CHECK) if (((rt_uint32_t)in % 4) != 0 || ((rt_uint32_t)out % 4) != 0) { in = rt_malloc(symmetric_info->length); if (in) { memcpy(in, symmetric_info->in, symmetric_info->length); out = in; align_flag = 1; } else { return -RT_ENOMEM; } } #endif #if !defined(WM_HWCRYPTO_NOT_LOCK) rt_mutex_take(&_hwcrypto->mutex, RT_WAITING_FOREVER); #endif tls_crypto_rc4_adv(symmetric_ctx->key, symmetric_ctx->key_bitlen >> 3, (unsigned char *)symmetric_info->in, symmetric_info->out, symmetric_info->length); #if !defined(WM_HWCRYPTO_NOT_LOCK) rt_mutex_release(&_hwcrypto->mutex); #endif #if !defined(WM_HWCRYPTO_NOT_ALIGN_CHECK) if (align_flag) { memcpy(symmetric_info->out, out, symmetric_info->length); rt_free(in); } #endif return RT_EOK; } static rt_err_t _aes_crypt(struct hwcrypto_symmetric *symmetric_ctx, struct hwcrypto_symmetric_info *symmetric_info) { CRYPTO_WAY mode; CRYPTO_MODE cbc; struct wm_hwcrypto_device *_hwcrypto = (struct wm_hwcrypto_device *)symmetric_ctx->parent.device; unsigned char *in, *out, align_flag = 0; if ((symmetric_ctx->key_bitlen >> 3) != 16 || (symmetric_info->length % 16) != 0) { return -RT_EINVAL; } in = (unsigned char *)symmetric_info->in; out = (unsigned char *)symmetric_info->out; #if !defined(WM_HWCRYPTO_NOT_ALIGN_CHECK) if (((rt_uint32_t)in % 4) != 0 || ((rt_uint32_t)out % 4) != 0) { in = rt_malloc(symmetric_info->length); if (in) { memcpy(in, symmetric_info->in, symmetric_info->length); out = in; align_flag = 1; } else { return -RT_ENOMEM; } } #endif mode = symmetric_info->mode == HWCRYPTO_MODE_ENCRYPT ? CRYPTO_WAY_ENCRYPT : CRYPTO_WAY_DECRYPT; switch (symmetric_ctx->parent.type & (HWCRYPTO_MAIN_TYPE_MASK | HWCRYPTO_SUB_TYPE_MASK)) { case HWCRYPTO_TYPE_AES_ECB: cbc = CRYPTO_MODE_ECB; break; case HWCRYPTO_TYPE_AES_CBC: cbc = CRYPTO_MODE_CBC; break; case HWCRYPTO_TYPE_AES_CTR: cbc = CRYPTO_MODE_CTR; break; default : return -RT_ERROR; } #if !defined(WM_HWCRYPTO_NOT_LOCK) rt_mutex_take(&_hwcrypto->mutex, RT_WAITING_FOREVER); #endif tls_crypto_aes_encrypt_decrypt_adv(symmetric_ctx->key, symmetric_ctx->iv, in, out, symmetric_info->length, cbc, mode); #if !defined(WM_HWCRYPTO_NOT_LOCK) rt_mutex_release(&_hwcrypto->mutex); #endif #if !defined(WM_HWCRYPTO_NOT_ALIGN_CHECK) if (align_flag) { memcpy(symmetric_info->out, out, symmetric_info->length); rt_free(in); } #endif return RT_EOK; } /**< a = b ^ c (mod d) */ static rt_err_t _bignum_exptmod(struct hwcrypto_bignum *bignum_ctx, struct hw_bignum_mpi *x, const struct hw_bignum_mpi *a, const struct hw_bignum_mpi *b, const struct hw_bignum_mpi *c) { pstm_int pa, pb, pm, pres; u32 * buff_a = NULL; u32 * buff_b = NULL; u32 * buff_m = NULL; int err = -1; void *buff; int buff_len; pstm_init(NULL, &pres); buff_a = tls_mem_alloc(a->total); if(buff_a == NULL) goto out; buff_b = tls_mem_alloc(b->total); if(buff_b == NULL) goto out; buff_m = tls_mem_alloc(c->total); if(buff_m == NULL) goto out; memset(buff_a, 0, a->total); memset(buff_b, 0, b->total); memset(buff_m, 0, c->total); memcpy(buff_a, a->p, a->total); memcpy(buff_b, b->p, b->total); memcpy(buff_m, c->p, c->total); pstm_reverse((unsigned char *)buff_a, a->total); pstm_reverse((unsigned char *)buff_b, b->total); pstm_reverse((unsigned char *)buff_m, c->total); // *((volatile unsigned int *)0x40000710) = *((volatile unsigned int *)0x40000710) | (0x1 << 28); if ((err = pstm_init_for_read_unsigned_bin(NULL, &pa, a->total)) != PS_SUCCESS){ goto out; } if ((err = pstm_read_unsigned_bin(&pa, (unsigned char *)buff_a, a->total)) != PS_SUCCESS) { goto out; } if ((err = pstm_init_for_read_unsigned_bin(NULL, &pb, b->total)) != PS_SUCCESS){ goto out; } if ((err = pstm_read_unsigned_bin(&pb, (unsigned char *)buff_b, b->total)) != PS_SUCCESS) { goto out; } if ((err = pstm_init_for_read_unsigned_bin(NULL, &pm, c->total)) != PS_SUCCESS){ goto out; } if ((err = pstm_read_unsigned_bin(&pm, (unsigned char *)buff_m, c->total)) != PS_SUCCESS) { goto out; } tls_crypto_exptmod(&pa, &pb, &pm, &pres); buff_len = pstm_unsigned_bin_size(&pres); buff = rt_malloc(buff_len); pstm_to_unsigned_bin_nr(NULL, &pres, buff); x->sign = pres.sign; x->p = buff; x->total = buff_len; out: if(buff_a) tls_mem_free(buff_a); if(buff_b) tls_mem_free(buff_b); if(buff_m) tls_mem_free(buff_m); pstm_clear(&pa); pstm_clear(&pb); pstm_clear(&pm); pstm_clear(&pres); if (a->sign < 0) { rt_kprintf("a->sign < 0\n"); } return err == PS_SUCCESS ? RT_EOK : -RT_ERROR; } static const struct hwcrypto_symmetric_ops aes_ops = { .crypt = _aes_crypt, }; static const struct hwcrypto_symmetric_ops rc4_ops = { .crypt = _rc4_crypt, }; static const struct hwcrypto_symmetric_ops des_ops = { .crypt = _des_crypt, }; static const struct hwcrypto_symmetric_ops des3_ops = { .crypt = _des3_crypt, }; static const struct hwcrypto_hash_ops hash_ops = { .update = _hash_update, .finish = _hash_finish, }; static const struct hwcrypto_rng_ops rng_ops = { .update = _rng_rand, }; static const struct hwcrypto_crc_ops crc_ops = { .update = _crc_update, }; static const struct hwcrypto_bignum_ops bignum_ops = { .add = RT_NULL, .sub = RT_NULL, .mul = RT_NULL, .mulmod = RT_NULL, .exptmod = _bignum_exptmod, }; static rt_err_t _crypto_create(struct rt_hwcrypto_ctx *ctx) { rt_err_t res = RT_EOK; void *contex; switch (ctx->type & HWCRYPTO_MAIN_TYPE_MASK) { case HWCRYPTO_TYPE_AES : ctx->contex = RT_NULL; ((struct hwcrypto_symmetric *)ctx)->ops = &aes_ops; break; case HWCRYPTO_TYPE_RC4 : ctx->contex = RT_NULL; ((struct hwcrypto_symmetric *)ctx)->ops = &rc4_ops; break; case HWCRYPTO_TYPE_MD5: case HWCRYPTO_TYPE_SHA1: contex = rt_malloc(sizeof(struct hash_ctx_des)); if (contex == RT_NULL) { return -RT_ENOMEM; } memset(contex, 0, sizeof(struct hash_ctx_des)); ctx->contex = contex; ((struct hwcrypto_hash *)ctx)->ops = &hash_ops; if ((ctx->type & HWCRYPTO_MAIN_TYPE_MASK) == HWCRYPTO_TYPE_MD5) { tls_crypto_md5_init(&((struct hash_ctx_des *)contex)->contex); } else { tls_crypto_sha1_init(&((struct hash_ctx_des *)contex)->contex); } break; case HWCRYPTO_TYPE_DES: ctx->contex = RT_NULL; ((struct hwcrypto_symmetric *)ctx)->ops = &des_ops; break; case HWCRYPTO_TYPE_3DES: ctx->contex = RT_NULL; ((struct hwcrypto_symmetric *)ctx)->ops = &des3_ops; break; case HWCRYPTO_TYPE_RNG: ctx->contex = RT_NULL; ((struct hwcrypto_rng *)ctx)->ops = &rng_ops; ctx->contex = RT_NULL; break; case HWCRYPTO_TYPE_CRC: ctx->contex = RT_NULL; ((struct hwcrypto_crc *)ctx)->ops = &crc_ops; break; case HWCRYPTO_TYPE_BIGNUM: ((struct hwcrypto_bignum *)ctx)->ops = &bignum_ops; break; default: res = -RT_ERROR; break; } return res; } static void _crypto_destroy(struct rt_hwcrypto_ctx *ctx) { rt_free(ctx->contex); } static rt_err_t _crypto_clone(struct rt_hwcrypto_ctx *des, const struct rt_hwcrypto_ctx *src) { rt_err_t res = RT_EOK; switch (src->type & HWCRYPTO_MAIN_TYPE_MASK) { case HWCRYPTO_TYPE_AES: case HWCRYPTO_TYPE_RC4: case HWCRYPTO_TYPE_RNG: case HWCRYPTO_TYPE_CRC: case HWCRYPTO_TYPE_BIGNUM: break; case HWCRYPTO_TYPE_MD5: case HWCRYPTO_TYPE_SHA1: if (des->contex && src->contex) { rt_memcpy(des->contex, src->contex, sizeof(struct hash_ctx_des)); } break; default: res = -RT_ERROR; break; } return res; } static void _crypto_reset(struct rt_hwcrypto_ctx *ctx) { switch (ctx->type & HWCRYPTO_MAIN_TYPE_MASK) { case HWCRYPTO_TYPE_AES: case HWCRYPTO_TYPE_RC4: case HWCRYPTO_TYPE_RNG: case HWCRYPTO_TYPE_CRC: break; case HWCRYPTO_TYPE_MD5: tls_crypto_md5_init(&((struct hash_ctx_des *)(ctx->contex))->contex); break; case HWCRYPTO_TYPE_SHA1: tls_crypto_sha1_init(&((struct hash_ctx_des *)(ctx->contex))->contex); break; default: break; } } static const struct rt_hwcrypto_ops _ops = { .create = _crypto_create, .destroy = _crypto_destroy, .copy = _crypto_clone, .reset = _crypto_reset, }; extern u8 *wpa_supplicant_get_mac(void); int wm_hw_crypto_device_init(void) { static struct wm_hwcrypto_device _crypto_dev; _crypto_dev.dev.ops = &_ops; _crypto_dev.dev.id = 0; rt_memcpy(&_crypto_dev.dev.id, wpa_supplicant_get_mac(), sizeof(_crypto_dev.dev.id) > 6 ? 6 : sizeof(_crypto_dev.dev.id)); _crypto_dev.dev.user_data = &_crypto_dev; if (rt_hwcrypto_register(&_crypto_dev.dev, RT_HWCRYPTO_DEFAULT_NAME) != RT_EOK) { return -1; } rt_mutex_init(&_crypto_dev.mutex, RT_HWCRYPTO_DEFAULT_NAME, RT_IPC_FLAG_PRIO); return 0; } INIT_DEVICE_EXPORT(wm_hw_crypto_device_init);