mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
344 lines
9.8 KiB
C
344 lines
9.8 KiB
C
/*
|
|
* Copyright (C) 2013 Alaeddine Weslati <alaeddine.weslati@inria.fr>
|
|
* Copyright (C) 2015 Freie Universität Berlin
|
|
* 2017 HAW Hamburg
|
|
*
|
|
* 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 drivers_at86rf2xx
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Internal interfaces for AT86RF2xx drivers
|
|
*
|
|
* @author Alaeddine Weslati <alaeddine.weslati@inria.fr>
|
|
* @author Thomas Eichinger <thomas.eichinger@fu-berlin.de>
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
* @author Sebastian Meiling <s@mlng.net>
|
|
*/
|
|
|
|
#ifndef AT86RF2XX_INTERNAL_H
|
|
#define AT86RF2XX_INTERNAL_H
|
|
|
|
#include <stdint.h>
|
|
|
|
#include "at86rf2xx.h"
|
|
|
|
#if AT86RF2XX_IS_PERIPH
|
|
#include <string.h>
|
|
#include "irq.h"
|
|
#include "time_units.h"
|
|
#endif
|
|
#include "at86rf2xx_registers.h"
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
/**
|
|
* @brief Max. allowed transmit power for the transceiver
|
|
*/
|
|
#ifdef MODULE_AT86RF212B
|
|
#define AT86RF2XX_TXPOWER_MAX (36)
|
|
#elif MODULE_AT86RF233
|
|
#define AT86RF2XX_TXPOWER_MAX (21)
|
|
#else
|
|
#define AT86RF2XX_TXPOWER_MAX (20)
|
|
#endif
|
|
|
|
/**
|
|
* @brief Transmit power offset
|
|
*/
|
|
#ifdef MODULE_AT86RF212B
|
|
#define AT86RF2XX_TXPOWER_OFF (25)
|
|
#else
|
|
#define AT86RF2XX_TXPOWER_OFF (17)
|
|
#endif
|
|
|
|
/**
|
|
* @brief Transition time from SLEEP to TRX_OFF in us, refer figure 7-4, p.42.
|
|
* For different environments refer figure 13-13, p.201
|
|
*/
|
|
#define AT86RF2XX_WAKEUP_DELAY (306U)
|
|
|
|
/**
|
|
* @brief Minimum reset pulse width, refer p.190. We use 62us so
|
|
* that it is at least one tick on platforms with coarse ztimers
|
|
*/
|
|
#define AT86RF2XX_RESET_PULSE_WIDTH (62U)
|
|
|
|
/**
|
|
* @brief The typical transition time to TRX_OFF after reset pulse is 26 us,
|
|
* refer to figure 7-8, p. 44. We use 62 us so that it is at least one
|
|
* tick on platforms that use a 16384 Hz oscillator or have slow start
|
|
* up times due to parasitic capacitance on the oscillator
|
|
*/
|
|
#define AT86RF2XX_RESET_DELAY (62U)
|
|
|
|
/**
|
|
* @brief Read from a register at address `addr` from device `dev`.
|
|
*
|
|
* @param[in] dev device to read from
|
|
* @param[in] addr address of the register to read
|
|
*
|
|
* @return the value of the specified register
|
|
*/
|
|
#if AT86RF2XX_IS_PERIPH
|
|
static inline uint8_t at86rf2xx_reg_read(const at86rf2xx_t *dev, volatile uint8_t *addr) {
|
|
(void) dev;
|
|
return *addr;
|
|
}
|
|
#else
|
|
uint8_t at86rf2xx_reg_read(const at86rf2xx_t *dev, uint8_t addr);
|
|
#endif
|
|
|
|
/**
|
|
* @brief Write to a register at address `addr` from device `dev`.
|
|
*
|
|
* @param[in] dev device to write to
|
|
* @param[in] addr address of the register to write
|
|
* @param[in] value value to write to the given register
|
|
*/
|
|
#if AT86RF2XX_IS_PERIPH
|
|
static inline void at86rf2xx_reg_write(const at86rf2xx_t *dev, volatile uint8_t *addr,
|
|
const uint8_t value) {
|
|
(void) dev;
|
|
*addr = value;
|
|
}
|
|
#else
|
|
void at86rf2xx_reg_write(const at86rf2xx_t *dev, uint8_t addr, uint8_t value);
|
|
#endif
|
|
|
|
/**
|
|
* @brief Read a chunk of data from the SRAM of the given device
|
|
*
|
|
* @param[in] dev device to read from
|
|
* @param[in] offset starting address to read from [valid 0x00-0x7f]
|
|
* @param[out] data buffer to read data into
|
|
* @param[in] len number of bytes to read from SRAM
|
|
*/
|
|
#if AT86RF2XX_IS_PERIPH
|
|
static inline void at86rf2xx_sram_read(const at86rf2xx_t *dev, uint8_t offset,
|
|
uint8_t *data, size_t len) {
|
|
(void)dev;
|
|
memcpy(data, (void*)(AT86RF2XX_REG__TRXFBST + offset), len);
|
|
}
|
|
#else
|
|
void at86rf2xx_sram_read(const at86rf2xx_t *dev, uint8_t offset,
|
|
uint8_t *data, size_t len);
|
|
#endif
|
|
/**
|
|
* @brief Write a chunk of data into the SRAM of the given device
|
|
*
|
|
* @param[in] dev device to write to
|
|
* @param[in] offset address in the SRAM to write to [valid 0x00-0x7f]
|
|
* @param[in] data data to copy into SRAM
|
|
* @param[in] len number of bytes to write to SRAM
|
|
*/
|
|
#if AT86RF2XX_IS_PERIPH
|
|
static inline void at86rf2xx_sram_write(const at86rf2xx_t *dev, uint8_t offset,
|
|
const uint8_t *data, size_t len) {
|
|
(void)dev;
|
|
memcpy((void*)(AT86RF2XX_REG__TRXFBST + offset), data, len);
|
|
}
|
|
#else
|
|
void at86rf2xx_sram_write(const at86rf2xx_t *dev, uint8_t offset,
|
|
const uint8_t *data, size_t len);
|
|
#endif
|
|
/**
|
|
* @brief Start a read transaction internal frame buffer of the given device
|
|
*
|
|
* Reading the frame buffer returns some extra bytes that are not accessible
|
|
* through reading the RAM directly. This locks the used SPI.
|
|
*
|
|
* @param[in] dev device to start read
|
|
*/
|
|
#if AT86RF2XX_IS_PERIPH
|
|
static inline void at86rf2xx_fb_start(const at86rf2xx_t *dev) {
|
|
(void) dev;
|
|
}
|
|
#else
|
|
void at86rf2xx_fb_start(const at86rf2xx_t *dev);
|
|
#endif
|
|
/**
|
|
* @brief Read the internal frame buffer of the given device
|
|
*
|
|
* Each read advances the position in the buffer by @p len.
|
|
*
|
|
* @param[in] dev device to read from
|
|
* @param[out] data buffer to copy the data to
|
|
* @param[in] len number of bytes to read from the frame buffer
|
|
*/
|
|
#if AT86RF2XX_IS_PERIPH
|
|
static inline void at86rf2xx_fb_read(const at86rf2xx_t *dev, uint8_t *data, size_t len) {
|
|
(void)dev;
|
|
memcpy(data, (void*)AT86RF2XX_REG__TRXFBST, len);
|
|
}
|
|
#else
|
|
void at86rf2xx_fb_read(const at86rf2xx_t *dev, uint8_t *data, size_t len);
|
|
#endif
|
|
/**
|
|
* @brief Stop a read transaction internal frame buffer of the given device
|
|
*
|
|
* Release the SPI device and unlock frame buffer protection.
|
|
*
|
|
* @param[in] dev device to stop read
|
|
*/
|
|
#if AT86RF2XX_IS_PERIPH
|
|
static inline void at86rf2xx_fb_stop(const at86rf2xx_t *dev) {
|
|
(void) dev;
|
|
}
|
|
#else
|
|
void at86rf2xx_fb_stop(const at86rf2xx_t *dev);
|
|
#endif
|
|
/**
|
|
* @brief Convenience function for reading the status of the given device
|
|
*
|
|
* @param[in] dev device to read the status from
|
|
*
|
|
* @return internal status of the given device
|
|
*/
|
|
uint8_t at86rf2xx_get_status(const at86rf2xx_t *dev);
|
|
|
|
/**
|
|
* @brief Make sure that device is not sleeping
|
|
*
|
|
* @param[in,out] dev device to eventually wake up
|
|
*/
|
|
void at86rf2xx_assert_awake(at86rf2xx_t *dev);
|
|
|
|
/**
|
|
* @brief Trigger a hardware reset
|
|
*
|
|
* @param[in,out] dev device to reset
|
|
*/
|
|
void at86rf2xx_hardware_reset(at86rf2xx_t *dev);
|
|
|
|
/**
|
|
* @brief Set PHY parameters based on channel and page number
|
|
*
|
|
* @param[in,out] dev device to configure
|
|
* @param[in] chan channel number to be set
|
|
* @param[in] page channel page
|
|
* @param[in] txpower TX power in dBm
|
|
*/
|
|
void at86rf2xx_configure_phy(at86rf2xx_t *dev, uint8_t chan, uint8_t page, int16_t txpower);
|
|
|
|
#if AT86RF2XX_RANDOM_NUMBER_GENERATOR || defined(DOXYGEN)
|
|
/**
|
|
* @brief Read random data from the RNG
|
|
*
|
|
* @note According to the data sheet this function only works properly in
|
|
* Basic Operation Mode. However, sporadic testing has shown that even
|
|
* in Extended Operation Mode this returns random enough data to be
|
|
* used as a seed for @ref sys_random if no cryptographically secure
|
|
* randomness is required.
|
|
* Any further use-case needs to be evaluated, especially if
|
|
* crypto-relevant randomness is required.
|
|
*
|
|
* @param[in] dev device to configure
|
|
* @param[out] data buffer to copy the random data to
|
|
* @param[in] len number of random bytes to store in data
|
|
*/
|
|
void at86rf2xx_get_random(at86rf2xx_t *dev, uint8_t *data, size_t len);
|
|
#endif
|
|
|
|
/**
|
|
* @brief Initialize AT86RF2XX SPI communication
|
|
*
|
|
* @param[in,out] dev device to initialize
|
|
* @param[in] irq_handler IRQ handler
|
|
*/
|
|
void at86rf2xx_spi_init(at86rf2xx_t *dev, void (*irq_handler)(void *arg));
|
|
|
|
/**
|
|
* @brief Get the PSDU length of the received frame.
|
|
*
|
|
* @param[in] dev pointer to the device descriptor
|
|
*
|
|
* @return the PSDU length
|
|
*/
|
|
static inline uint8_t at86rf2xx_get_rx_len(at86rf2xx_t *dev)
|
|
{
|
|
(void) dev;
|
|
#if AT86RF2XX_IS_PERIPH
|
|
return TST_RX_LENGTH;
|
|
#else
|
|
uint8_t phr;
|
|
at86rf2xx_fb_read(dev, &phr, 1);
|
|
return phr;
|
|
#endif
|
|
}
|
|
|
|
/**
|
|
* @brief Get the IRQ flags.
|
|
*
|
|
* This function clears the IRQ flags
|
|
* @param[in,out] dev pointer to the device descriptor
|
|
*
|
|
* @return IRQ flags
|
|
*/
|
|
static inline uint8_t at86rf2xx_get_irq_flags(at86rf2xx_t *dev)
|
|
{
|
|
(void) dev;
|
|
#if AT86RF2XX_IS_PERIPH
|
|
uint8_t irq_mask = dev->irq_status;
|
|
dev->irq_status = 0;
|
|
return irq_mask;
|
|
#else
|
|
return at86rf2xx_reg_read(dev, AT86RF2XX_REG__IRQ_STATUS);
|
|
#endif
|
|
|
|
}
|
|
|
|
#if AT86RF2XX_IS_PERIPH
|
|
/*
|
|
* Read a 32 bit register as described in section 10.3 of the datasheet: A read
|
|
* of the least significant byte causes the current value to be atomically
|
|
* captured in a temporary 32 bit registers. The remaining reads will access this
|
|
* register instead. Only a single 32 bit temporary register is used to provide
|
|
* means to atomically access them. Thus, interrupts must be disabled during the
|
|
* read sequence in order to prevent other threads (or ISRs) from updating the
|
|
* temporary 32 bit register before the reading sequence has completed.
|
|
*/
|
|
static inline uint32_t reg32_read(volatile uint8_t *reg_ll)
|
|
{
|
|
le_uint32_t reg;
|
|
unsigned state = irq_disable();
|
|
reg.u8[0] = reg_ll[0];
|
|
reg.u8[1] = reg_ll[1];
|
|
reg.u8[2] = reg_ll[2];
|
|
reg.u8[3] = reg_ll[3];
|
|
irq_restore(state);
|
|
return reg.u32;
|
|
}
|
|
|
|
/**
|
|
* @brief Get the timestamp of the packet in symbol counter ticks
|
|
*
|
|
* @param[in] dev pointer to the device descriptor
|
|
*
|
|
* @return the symbol counter value
|
|
*/
|
|
static inline uint32_t at86rf2xx_get_sc(const at86rf2xx_t *dev)
|
|
{
|
|
(void) dev;
|
|
return reg32_read(&SCCNTLL);
|
|
}
|
|
|
|
/* Symbol counter frequency is 62500Hz. One symbol counter tick is 16us = 16000 ns*/
|
|
#define SC_TO_NS (16000LU)
|
|
|
|
#endif
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#endif /* AT86RF2XX_INTERNAL_H */
|
|
/** @} */
|