/* * Copyright (C) 2014 Loci Controls Inc. * 2016 Freie Universität Berlin * * 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. */ /** * @ingroup cpu_cc2538 * @{ * * @file * @brief HWRNG peripheral driver interface implementation * * @author Ian Martin * @author Hauke Petersen * * @} */ #include "cpu.h" #include "periph/hwrng.h" void hwrng_init(void) { uint16_t seed = 0; int i; /* Make sure the RNG is on */ SOC_ADC->cc2538_adc_adccon1.ADCCON1bits.RCTRL = 0; /* Enable clock for the RF Core */ SYS_CTRL_RCGCRFC = 1; /* Wait for the clock ungating to take effect */ while (SYS_CTRL_RCGCRFC != 1); /* Infinite RX - FRMCTRL0[3:2] = 10. This will mess with radio operation */ RFCORE_XREG_FRMCTRL0 = 0x00000008; /* Turn RF on */ RFCORE_SFR_RFST = ISRXON; /* * Wait until "the chip has been in RX long enough for the transients to * have died out. A convenient way to do this is to wait for the RSSI-valid * signal to go high." */ while (!RFCORE->XREG_RSSISTATbits.RSSI_VALID); /* * Form the seed by concatenating bits from IF_ADC in the RF receive path. * Keep sampling until we have read at least 16 bits AND the seed is valid * * Invalid seeds are 0 and 0x8003 and should not be used. */ for (i = 0; (i < 8) || (seed == 0) || (seed == 0x8003); i++) { seed <<= 2; seed ^= RFCORE->XREG_RFRND; } /* Seed the high byte first: */ SOC_ADC_RNDL = (seed >> 8) & 0xff; SOC_ADC_RNDL = seed & 0xff; /* Turn RF off: */ RFCORE_SFR_RFST = ISRFOFF; } void hwrng_read(void *buf, unsigned int num) { unsigned count; uint8_t *b = (uint8_t *)buf; for (count = 0; count < num; ) { /* Clock the RNG LSFR once: */ SOC_ADC->cc2538_adc_adccon1.ADCCON1bits.RCTRL = 1; /* Read up to 2 bytes of hwrng data: */ b[count++] = SOC_ADC_RNDL; if (count >= num) break; b[count++] = SOC_ADC_RNDH; } }