1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/drivers/ata8520e/ata8520e.c

508 lines
14 KiB
C

/*
* Copyright (C) 2017 Inria
*
* 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_ata8520e
* @{
*
* @file
* @brief Device driver for Microchip ATA8520E transceiver
*
* @author Alexandre Abadie <alexandre.abadie@inria.fr>
*
* @}
*/
#include <stdio.h>
#include <stdlib.h>
#include <inttypes.h>
#include <string.h>
#include "xtimer.h"
#include "fmt.h"
#include "periph/spi.h"
#include "periph/gpio.h"
#include "board.h"
#include "ata8520e_internals.h"
#include "ata8520e.h"
#define ENABLE_DEBUG (0)
/* Warning: to correctly display the debug messages in callbacks,
add CFLAGS+=-DTHREAD_STACKSIZE_IDLE=THREAD_STACKSIZE_DEFAULT to the build
command.
*/
#include "debug.h"
#define SPIDEV (dev->params.spi)
#define CSPIN (dev->params.cs_pin)
#define INTPIN (dev->params.int_pin)
#define RESETPIN (dev->params.reset_pin)
#define POWERPIN (dev->params.power_pin)
#define SPI_CS_DELAY (US_PER_MS)
#define DELAY_10_MS (10 * US_PER_MS) /* 10 ms */
#define TX_TIMEOUT (8U) /* 8 s */
#define TX_RX_TIMEOUT (50U) /* 50 s */
static void _print_atmel_status(uint8_t status)
{
DEBUG("[ata8520e] Atmel status: %d\n", status);
if (status & ATA8520E_ATMEL_PA_MASK) {
DEBUG("[ata8520e] Atmel: PA ON\n");
}
else {
DEBUG("[ata8520e] Atmel: PA OFF\n");
}
if ((status >> 1) & ATA8520E_ATMEL_SYSTEM_READY_MASK) {
DEBUG("[ata8520e] Atmel: System ready to operate\n");
return;
}
if ((status >> 1) & ATA8520E_ATMEL_FRAME_SENT_MASK) {
DEBUG("[ata8520e] Atmel: Frame sent\n");
return;
}
switch ((status >> 1) & 0x0F) {
case ATA8520E_ATMEL_OK:
DEBUG("[ata8520e] Atmel: System is ready\n");
break;
case ATA8520E_ATMEL_COMMAND_ERROR:
DEBUG("[ata8520e] Atmel: Command error / not supported\n");
break;
case ATA8520E_ATMEL_GENERIC_ERROR:
DEBUG("[ata8520e] Atmel: Generic error\n");
break;
case ATA8520E_ATMEL_FREQUENCY_ERROR:
DEBUG("[ata8520e] Atmel: Frequency error\n");
break;
case ATA8520E_ATMEL_USAGE_ERROR:
DEBUG("[ata8520e] Atmel: Usage error\n");
break;
case ATA8520E_ATMEL_OPENING_ERROR:
DEBUG("[ata8520e] Atmel: Opening error\n");
break;
case ATA8520E_ATMEL_CLOSING_ERROR:
DEBUG("[ata8520e] Atmel: Closing error\n");
break;
case ATA8520E_ATMEL_SEND_ERROR:
DEBUG("[ata8520e] Atmel: Send error\n");
break;
default:
DEBUG("[ata8520e] Atmel: Unknown status code '%02X'\n",
(status >> 1) & 0x0F);
break;
}
}
static void _print_sigfox_status(uint8_t status)
{
DEBUG("[ata8520e] Sigfox status: %d\n", status);
switch (status) {
case ATA8520E_SIGFOX_NO_ERROR:
DEBUG("[ata8520e] Sigfox: OK\n");
break;
case ATA8520E_SIGFOX_TX_LEN_TOO_LONG:
DEBUG("[ata8520e] Sigfox: TX data length exceeds 12 bytes\n");
break;
case ATA8520E_SIGFOX_RX_TIMEOUT:
DEBUG("[ata8520e] Sigfox: Timeout for downlink message\n");
break;
case ATA8520E_SIGFOX_RX_BIT_TIMEOUT:
DEBUG("[ata8520e] Sigfox: Timeout for bit downlink\n");
break;
case ATA8520E_SIGFOX2_INIT_ERROR:
DEBUG("[ata8520e] Sigfox2: Initialization error\n");
break;
case ATA8520E_SIGFOX2_TX_ERROR:
DEBUG("[ata8520e] Sigfox2: Error during send\n");
break;
case ATA8520E_SIGFOX2_RF_ERROR:
DEBUG("[ata8520e] Sigfox2: Error in RF frequency\n");
break;
case ATA8520E_SIGFOX2_DF_WAIT_ERROR:
DEBUG("[ata8520e] Sigfox2: Error during wait for data frame\n");
break;
default:
DEBUG("[ata8520e] Sigfox: Internal error '%02X'\n", status);
break;
}
}
static void _irq_handler(void *arg)
{
(void) arg;
ata8520e_t * dev = (ata8520e_t *)arg;
DEBUG("[ata8520e] Event received !\n");
dev->event_received = 1;
switch (dev->internal_state) {
case ATA8520E_STATE_IDLE:
DEBUG("[ata8520e] IDLE state\n");
break;
case ATA8520E_STATE_INIT:
DEBUG("[ata8520e] INIT state\n");
break;
case ATA8520E_STATE_TX:
DEBUG("[ata8520e] TX state\n");
break;
case ATA8520E_STATE_RX:
DEBUG("[ata8520e] RX state\n");
break;
}
}
static void _getbus(const ata8520e_t *dev)
{
if (spi_acquire(SPIDEV, CSPIN, SPI_MODE_0, dev->params.spi_clk) < 0) {
DEBUG("[ata8520e] ERROR: Cannot acquire SPI bus!\n");
}
}
static void _spi_transfer_byte(const ata8520e_t *dev, bool cont, uint8_t out)
{
/* Manually triggering CS because of a required delay, see datasheet,
section 2.1.1, page 10 */
gpio_clear((gpio_t)CSPIN);
xtimer_usleep(SPI_CS_DELAY);
spi_transfer_byte(SPIDEV, SPI_CS_UNDEF, cont, out);
xtimer_usleep(SPI_CS_DELAY);
gpio_set((gpio_t)CSPIN);
}
static void _spi_transfer_bytes(const ata8520e_t *dev, bool cont,
const void *out, void *in, size_t len)
{
/* Manually triggering CS because of a required delay, see datasheet,
section 2.1.1, page 10 */
gpio_clear((gpio_t)CSPIN);
xtimer_usleep(SPI_CS_DELAY);
spi_transfer_bytes(SPIDEV, SPI_CS_UNDEF, cont, out, in, len);
xtimer_usleep(SPI_CS_DELAY);
gpio_set((gpio_t)CSPIN);
}
static void _send_command(const ata8520e_t *dev, uint8_t command)
{
_getbus(dev);
_spi_transfer_byte(dev, false, command);
spi_release(SPIDEV);
}
static void _status(const ata8520e_t *dev)
{
/* clear the event line and check the device status,
see datasheet, section 2.1.2.10, page 12 */
_getbus(dev);
_spi_transfer_byte(dev, true, ATA8520E_GET_STATUS);
_spi_transfer_byte(dev, true, 0);
_spi_transfer_byte(dev, true, 0); /* SSM unused */
uint8_t atmel = spi_transfer_byte(SPIDEV, CSPIN, true, 0);
uint8_t sigfox = spi_transfer_byte(SPIDEV, CSPIN, true, 0);
uint8_t sigfox2 = spi_transfer_byte(SPIDEV, CSPIN, false, 0);
spi_release(SPIDEV);
if (ENABLE_DEBUG) {
_print_atmel_status(atmel);
_print_sigfox_status(sigfox);
_print_sigfox_status(sigfox2);
}
}
static void _reset(const ata8520e_t *dev)
{
gpio_set(RESETPIN);
xtimer_usleep(DELAY_10_MS);
gpio_clear(RESETPIN);
xtimer_usleep(DELAY_10_MS);
gpio_set(RESETPIN);
}
static void _poweron(const ata8520e_t *dev)
{
/* power up procedure, see datasheet, section 2.2.1, page 24 */
gpio_set(POWERPIN);
_reset(dev);
}
static void _poweroff(const ata8520e_t *dev)
{
/* power down procedure, see datasheet, section 2.2.2, page 24 */
_status(dev);
gpio_clear(POWERPIN);
_send_command(dev, ATA8520E_OFF_MODE);
}
int ata8520e_init(ata8520e_t *dev, const ata8520e_params_t *params)
{
/* write config params to device descriptor */
dev->params = *params;
/* Initialize pins*/
if (gpio_init_int(INTPIN, GPIO_IN_PD,
GPIO_FALLING, _irq_handler, dev) < 0 ) {
DEBUG("[ata8520e] ERROR: Interrupt pin not initialized\n");
return -ATA8520E_ERR_GPIO_INT;
}
if (gpio_init(POWERPIN, GPIO_OUT) < 0) {
DEBUG("[ata8520e] ERROR: Power pin not initialized\n");
return -ATA8520E_ERR_GPIO_POWER;
}
if (gpio_init(RESETPIN, GPIO_OUT) < 0) {
DEBUG("[ata8520e] ERROR: Reset pin not initialized\n");
return -ATA8520E_ERR_GPIO_RESET;
}
dev->internal_state = ATA8520E_STATE_INIT;
_poweron(dev);
/* Initialize SPI bus*/
if (spi_init_cs(dev->params.spi, CSPIN) < 0) {
DEBUG("[ata8520e] ERROR: SPI not initialized\n");
return -ATA8520E_ERR_SPI;
}
xtimer_usleep(100 * US_PER_MS); /* 100 ms */
if (ENABLE_DEBUG) {
char sigfox_id[SIGFOX_ID_LENGTH + 1];
ata8520e_read_id(dev, sigfox_id);
char sigfox_pac[SIGFOX_PAC_LENGTH + 1];
ata8520e_read_pac(dev, sigfox_pac);
uint8_t atmel_version[2];
ata8520e_read_atmel_version(dev, atmel_version);
char sigfox_version[11];
ata8520e_read_sigfox_version(dev, sigfox_version);
DEBUG("[ata8520e] Atmel version: %d.%d\n",
atmel_version[0], atmel_version[1]);
DEBUG("[ata8520e] Sigfox version: %s\n", sigfox_version);
DEBUG("[ata8520e] Sigfox ID: %s\n", sigfox_id);
DEBUG("[ata8520e] Sigfox PAC: %s\n", sigfox_pac);
}
/* clear event line */
_status(dev);
dev->internal_state = ATA8520E_STATE_IDLE;
_poweroff(dev);
return ATA8520E_OK;
}
void ata8520e_system_reset(const ata8520e_t *dev)
{
_send_command(dev, ATA8520E_SYSTEM_RESET);
}
void ata8520e_read_atmel_version(const ata8520e_t *dev, uint8_t *version)
{
_poweron(dev);
_getbus(dev);
_spi_transfer_byte(dev, true, ATA8520E_ATMEL_VERSION);
_spi_transfer_byte(dev, true, 0);
_spi_transfer_bytes(dev, false, 0, version, 2);
spi_release(SPIDEV);
_poweroff(dev);
}
void ata8520e_read_sigfox_version(const ata8520e_t *dev, char *version)
{
_poweron(dev);
_getbus(dev);
_spi_transfer_byte(dev, true, ATA8520E_SIGFOX_VERSION);
_spi_transfer_byte(dev, true, 0);
_spi_transfer_bytes(dev, false, 0, version, 11);
spi_release(SPIDEV);
_poweroff(dev);
}
void ata8520e_read_pac(const ata8520e_t *dev, char *pac)
{
_poweron(dev);
uint8_t pac_bytes[SIGFOX_PAC_LENGTH];
_getbus(dev);
_spi_transfer_byte(dev, true, ATA8520E_GET_PAC);
_spi_transfer_byte(dev, true, 0);
_spi_transfer_bytes(dev, false, NULL, pac_bytes, SIGFOX_PAC_LENGTH / 2);
spi_release(SPIDEV);
fmt_bytes_hex(pac, pac_bytes, SIGFOX_PAC_LENGTH >> 1);
pac[SIGFOX_PAC_LENGTH] = '\0';
_poweroff(dev);
}
void ata8520e_read_id(const ata8520e_t *dev, char *id)
{
_poweron(dev);
uint8_t id_bytes[SIGFOX_ID_LENGTH];
_getbus(dev);
_spi_transfer_byte(dev, true, ATA8520E_GET_ID);
_spi_transfer_byte(dev, true, 0);
_spi_transfer_bytes(dev, false, NULL, id_bytes, SIGFOX_ID_LENGTH);
spi_release(SPIDEV);
/* create id string (4 hexadecimal values in reversed order) */
fmt_bytes_hex_reverse(id, id_bytes, SIGFOX_ID_LENGTH >> 1);
id[SIGFOX_ID_LENGTH] = '\0';
_poweroff(dev);
}
static void isr_event_timeout(void *arg)
{
ata8520e_t *dev = (ata8520e_t *)arg;
mutex_unlock(&(dev->event_lock));
}
static bool _wait_event(ata8520e_t *dev, uint8_t timeout)
{
dev->event_received = 0;
xtimer_ticks64_t start_time = xtimer_now64();
xtimer_t event_timer;
event_timer.callback = isr_event_timeout;
event_timer.arg = dev;
xtimer_set(&event_timer, (uint32_t)timeout * US_PER_SEC);
/* waiting for the event */
while ((!dev->event_received) &&
xtimer_less(xtimer_diff32_64(xtimer_now64(), start_time),
xtimer_ticks_from_usec(timeout * US_PER_SEC))) {
mutex_lock(&(dev->event_lock));
}
xtimer_remove(&event_timer);
if (dev->event_received == 0) {
DEBUG("[ata8520e] event timeout\n");
return true;
}
return false;
}
static void _prepare_send_frame(ata8520e_t *dev, uint8_t *msg, uint8_t msg_len)
{
_poweron(dev);
xtimer_usleep(5 * US_PER_MS);
_status(dev);
/* Verify message length */
if (msg_len > SIGFOX_MAX_TX_LENGTH) {
DEBUG("[ata8520e] Message exceeds the maximum %d characters length "
"allowed. It will be truncated.\n", SIGFOX_MAX_TX_LENGTH);
msg_len = SIGFOX_MAX_TX_LENGTH;
}
dev->internal_state = ATA8520E_STATE_TX;
/* Write message in TX buffer */
DEBUG("[ata8520e] Writing send frame to RX buffer\n");
_getbus(dev);
_spi_transfer_byte(dev, true, ATA8520E_WRITE_TX_BUFFER);
_spi_transfer_byte(dev, true, msg_len);
_spi_transfer_bytes(dev, false, msg, NULL, msg_len);
spi_release(SPIDEV);
}
static int _wait_send(ata8520e_t *dev, uint8_t timeout)
{
DEBUG("[ata8520e] waiting for TX to complete\n");
/* Wait 8s maximum for the message to be sent */
int ret = ATA8520E_OK;
if (_wait_event(dev, timeout)) {
DEBUG("[ata8520e] failed to send message\n");
ret = -ATA8520E_ERR_EVENT_TIMEOUT;
}
else {
DEBUG("[ata8520e] message sent\n");
}
/* Clear event line */
_status(dev);
/* back to idle state */
dev->internal_state = ATA8520E_STATE_IDLE;
return ret;
}
int ata8520e_send_frame(ata8520e_t *dev, uint8_t *msg, uint8_t msg_len)
{
DEBUG("[ata8520e] Sending frame '%s', length: %d\n", (char*)msg, msg_len);
_prepare_send_frame(dev, msg, msg_len);
/* Trigger TX */
_send_command(dev, ATA8520E_SEND_FRAME);
int ret = _wait_send(dev, TX_TIMEOUT);
/* switch off transceiver before returning */
_poweroff(dev);
return ret;
}
int ata8520e_send_receive_frame(ata8520e_t *dev, uint8_t *msg, uint8_t msg_len,
uint8_t *rx_payload)
{
DEBUG("[ata8520e] Sending frame '%s', length: %d\n", (char*)msg, msg_len);
_prepare_send_frame(dev, msg, msg_len);
/* Trigger TX */
_send_command(dev, ATA8520E_SEND_RECEIVE_FRAME);
int ret = _wait_send(dev, TX_RX_TIMEOUT);
if (ret == -ATA8520E_ERR_EVENT_TIMEOUT) {
return ret;
}
/* Read RX message */
dev->internal_state = ATA8520E_STATE_RX;
DEBUG("[ata8520e] Reading RX buffer\n");
_getbus(dev);
_spi_transfer_byte(dev, true, ATA8520E_READ_RX_BUFFER);
_spi_transfer_byte(dev, true, 0);
_spi_transfer_bytes(dev, false, NULL, rx_payload, SIGFOX_RX_LENGTH);
spi_release(SPIDEV);
/* back to idle state */
dev->internal_state = ATA8520E_STATE_IDLE;
/* switch off transceiver before returning */
_poweroff(dev);
return ret;
}
int ata8520e_send_bit(ata8520e_t *dev, bool bit)
{
DEBUG("[ata8520e] Sending bit '%d'\n", bit);
_poweron(dev);
xtimer_usleep(5 * US_PER_MS);
_status(dev);
dev->internal_state = ATA8520E_STATE_TX;
/* Sends the bit */
_getbus(dev);
_spi_transfer_byte(dev, true, ATA8520E_SEND_BIT);
if (bit) {
_spi_transfer_byte(dev, false, 0x01);
}
else {
_spi_transfer_byte(dev, false, 0x00);
}
spi_release(SPIDEV);
int ret = _wait_send(dev, TX_TIMEOUT);
/* switch off transceiver before returning */
_poweroff(dev);
return ret;
}