Blame view

RIOT/cpu/mips_pic32_common/periph/hwrng.c 1.7 KB
a752c7ab   elopes   add first test an...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
  /*
   * Copyright(C) 2017 Francois Berder <fberder@outlook.fr>
   *
   * This file is subject to the terms and conditions of the GNU Lesser
   * General Public License v2.1. See the file LICENSE in the top level
   * directory for more details.
   *
   */
  
  #include <string.h>
  #include "board.h"
  #include "periph/hwrng.h"
  
  #ifdef _RNG
  
  static void wait_plen_cycles(void)
  {
      unsigned int i;
      for (i = 0; i < (RNGCON & _RNGCON_PLEN_MASK); ++i)
          __asm volatile ("nop");
  }
  
  void hwrng_init(void)
  {
      RNGCON = _RNGCON_TRNGEN_MASK;
  
      /*
      * Wait to have at least 64 bits before setting the 64-bit seed
      * of the pseudo random generator.
      */
      while (RNGCNT < 64) {}
  
      /* Load seed from the TRNG */
      RNGCON |= _RNGCON_LOAD_MASK;
      while (RNGCON & _RNGCON_LOAD_MASK) {}
  
      RNGCON &= ~_RNGCON_TRNGEN_MASK;
  
      RNGPOLY1 = 0x00C00003;
      RNGPOLY2 = 0x00000000;
  
      RNGCON |= 42;   /* Set PLEN to 42 */
      RNGCON |= _RNGCON_CONT_MASK;
  }
  
  void hwrng_read(void *buf, unsigned int num)
  {
      unsigned int i = 0;
      uint8_t *buffer = (uint8_t *)buf;
  
      RNGCON |= _RNGCON_PRNGEN_MASK;
  
      for (i = 0; i < (num >> 3); ++i) {
          uint32_t rng1, rng2;
  
          wait_plen_cycles();
          rng1 = RNGNUMGEN1;
          rng2 = RNGNUMGEN2;
          memcpy(buffer, &rng1, sizeof(rng1));
          memcpy(buffer + 4, &rng2, sizeof(rng2));
          buffer += 8;
      }
  
      num &= 0x7;
      if (num) {
          uint32_t rng1, n = num & 0x3;
  
          wait_plen_cycles();
          rng1 = RNGNUMGEN1;
          memcpy(buffer, &rng1, n);
          num -= n;
          buffer += n;
  
          if (num) {
              uint32_t rng2 = RNGNUMGEN2;
              memcpy(buffer, &rng2, num);
          }
      }
  
      RNGCON &= ~_RNGCON_PRNGEN_MASK;
  }
  
  #endif /* _RNG */