drv_trng.c 1.9 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889
  1. /**************************************************************************//**
  2. *
  3. * @copyright (C) 2020 Nuvoton Technology Corp. All rights reserved.
  4. *
  5. * SPDX-License-Identifier: Apache-2.0
  6. *
  7. * Change Logs:
  8. * Date Author Notes
  9. * 2020-3-4 CHChen First version
  10. *
  11. ******************************************************************************/
  12. #include <rtconfig.h>
  13. #if (defined(BSP_USING_TRNG) && defined(RT_HWCRYPTO_USING_RNG))
  14. #include <rtdevice.h>
  15. #include "NuMicro.h"
  16. #include <stdlib.h>
  17. #define NU_CRYPTO_TRNG_NAME "nu_TRNG"
  18. /* Private variables ------------------------------------------------------------*/
  19. static struct rt_mutex s_TRNG_mutex;
  20. static int s_i32TRNGEnable = 0;
  21. static rt_uint32_t nu_trng_run(void)
  22. {
  23. uint32_t u32RNGValue;
  24. rt_err_t result;
  25. result = rt_mutex_take(&s_TRNG_mutex, RT_WAITING_FOREVER);
  26. RT_ASSERT(result == RT_EOK);
  27. TRNG_Open();
  28. if (TRNG_GenWord(&u32RNGValue) < 0)
  29. {
  30. //Failed, use software rand
  31. u32RNGValue = rand();
  32. }
  33. result = rt_mutex_release(&s_TRNG_mutex);
  34. RT_ASSERT(result == RT_EOK);
  35. return u32RNGValue;
  36. }
  37. rt_err_t nu_trng_init(void)
  38. {
  39. rt_err_t result;
  40. result = rt_mutex_init(&s_TRNG_mutex, NU_CRYPTO_TRNG_NAME, RT_IPC_FLAG_FIFO);
  41. RT_ASSERT(result == RT_EOK);
  42. if ((SYS->CSERVER & SYS_CSERVER_VERSION_Msk) == 0x0)
  43. {
  44. rt_kprintf("This chip does not support TRNG!\n");
  45. return -RT_ERROR;
  46. }
  47. s_i32TRNGEnable = 1;
  48. SYS_ResetModule(TRNG_RST);
  49. return RT_EOK;
  50. }
  51. void nu_trng_open(void)
  52. {
  53. #if defined(NU_PRNG_USE_SEED)
  54. srand(NU_PRNG_SEED_VALUE);
  55. #else
  56. srand(rt_tick_get());
  57. #endif
  58. }
  59. rt_uint32_t nu_trng_rand(struct hwcrypto_rng *ctx)
  60. {
  61. if (!s_i32TRNGEnable)
  62. {
  63. uint32_t u32RNGValue;
  64. //use software rand
  65. u32RNGValue = rand();
  66. return u32RNGValue;
  67. }
  68. return nu_trng_run();
  69. }
  70. #endif //#if (defined(BSP_USING_TRNG) && defined(RT_HWCRYPTO_USING_RNG))