npcm_rng.c 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Copyright (c) 2022 Nuvoton Technology Corp.
  4. */
  5. #include <common.h>
  6. #include <dm.h>
  7. #include <malloc.h>
  8. #include <rng.h>
  9. #include <uboot_aes.h>
  10. #include <asm/io.h>
  11. #define RNGCS_RNGE BIT(0)
  12. #define RNGCS_DVALID BIT(1)
  13. #define RNGCS_CLKP(range) ((0x0f & (range)) << 2)
  14. #define RNGMODE_M1ROSEL_VAL (0x02) /* Ring Oscillator Select for Method I */
  15. enum {
  16. RNG_CLKP_80_100_MHZ = 0x00, /*default */
  17. RNG_CLKP_60_80_MHZ = 0x01,
  18. RNG_CLKP_50_60_MHZ = 0x02,
  19. RNG_CLKP_40_50_MHZ = 0x03,
  20. RNG_CLKP_30_40_MHZ = 0x04,
  21. RNG_CLKP_25_30_MHZ = 0x05,
  22. RNG_CLKP_20_25_MHZ = 0x06,
  23. RNG_CLKP_5_20_MHZ = 0x07,
  24. RNG_CLKP_2_15_MHZ = 0x08,
  25. RNG_CLKP_9_12_MHZ = 0x09,
  26. RNG_CLKP_7_9_MHZ = 0x0A,
  27. RNG_CLKP_6_7_MHZ = 0x0B,
  28. RNG_CLKP_5_6_MHZ = 0x0C,
  29. RNG_CLKP_4_5_MHZ = 0x0D,
  30. RNG_CLKP_3_4_MHZ = 0x0E,
  31. RNG_NUM_OF_CLKP
  32. };
  33. struct npcm_rng_regs {
  34. unsigned int rngcs;
  35. unsigned int rngd;
  36. unsigned int rngmode;
  37. };
  38. struct npcm_rng_priv {
  39. struct npcm_rng_regs *regs;
  40. };
  41. static struct npcm_rng_priv *rng_priv;
  42. void npcm_rng_init(void)
  43. {
  44. struct npcm_rng_regs *regs = rng_priv->regs;
  45. int init;
  46. /* check if rng enabled */
  47. init = readb(&regs->rngcs);
  48. if ((init & RNGCS_RNGE) == 0) {
  49. /* init rng */
  50. writeb(RNGCS_CLKP(RNG_CLKP_20_25_MHZ) | RNGCS_RNGE, &regs->rngcs);
  51. writeb(RNGMODE_M1ROSEL_VAL, &regs->rngmode);
  52. }
  53. }
  54. void npcm_rng_disable(void)
  55. {
  56. struct npcm_rng_regs *regs = rng_priv->regs;
  57. /* disable rng */
  58. writeb(0, &regs->rngcs);
  59. writeb(0, &regs->rngmode);
  60. }
  61. void srand(unsigned int seed)
  62. {
  63. /* no need to seed for now */
  64. }
  65. int npcm_rng_read(struct udevice *dev, void *data, size_t max)
  66. {
  67. struct npcm_rng_regs *regs = rng_priv->regs;
  68. int i;
  69. int ret_val = 0;
  70. char *buf = data;
  71. npcm_rng_init();
  72. printf("NPCM HW RNG\n");
  73. /* Wait for RNG done (max bytes) */
  74. for (i = 0; i < max; i++) {
  75. /* wait until DVALID is set */
  76. while ((readb(&regs->rngcs) & RNGCS_DVALID) == 0)
  77. ;
  78. buf[i] = ((unsigned int)readb(&regs->rngd) & 0x000000FF);
  79. }
  80. return ret_val;
  81. }
  82. unsigned int rand_r(unsigned int *seedp)
  83. {
  84. struct npcm_rng_regs *regs = rng_priv->regs;
  85. int i;
  86. unsigned int ret_val = 0;
  87. npcm_rng_init();
  88. /* Wait for RNG done (4 bytes) */
  89. for (i = 0; i < 4 ; i++) {
  90. /* wait until DVALID is set */
  91. while ((readb(&regs->rngcs) & RNGCS_DVALID) == 0)
  92. ;
  93. ret_val |= (((unsigned int)readb(&regs->rngd) & 0x000000FF) << (i * 8));
  94. }
  95. return ret_val;
  96. }
  97. unsigned int rand(void)
  98. {
  99. return rand_r(NULL);
  100. }
  101. static int npcm_rng_bind(struct udevice *dev)
  102. {
  103. rng_priv = calloc(1, sizeof(struct npcm_rng_priv));
  104. if (!rng_priv)
  105. return -ENOMEM;
  106. rng_priv->regs = dev_remap_addr_index(dev, 0);
  107. if (!rng_priv->regs) {
  108. printf("Cannot find rng reg address, binding failed\n");
  109. return -EINVAL;
  110. }
  111. printf("RNG: NPCM RNG module bind OK\n");
  112. return 0;
  113. }
  114. static const struct udevice_id npcm_rng_ids[] = {
  115. { .compatible = "nuvoton,npcm845-rng" },
  116. { .compatible = "nuvoton,npcm750-rng" },
  117. { }
  118. };
  119. static const struct dm_rng_ops npcm_rng_ops = {
  120. .read = npcm_rng_read,
  121. };
  122. U_BOOT_DRIVER(npcm_rng) = {
  123. .name = "npcm_rng",
  124. .id = UCLASS_RNG,
  125. .ops = &npcm_rng_ops,
  126. .of_match = npcm_rng_ids,
  127. .priv_auto = sizeof(struct npcm_rng_priv),
  128. .bind = npcm_rng_bind,
  129. };