1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
RIOT/boards/iot-lab_M3/drivers/at86rf231_driver.c

172 lines
3.5 KiB
C
Raw Normal View History

2014-07-03 10:36:46 +02:00
/*
* Copyright (C) 2014 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 board_iot-lab_M3
* @{
*
* @file at86rf231_driver.c
* @brief Board specific implementations for the at86rf231 radio driver
*
* @author Thomas Eichinger <thomas.eichinger@fu-berlin.de>
*
* @}
*/
2014-06-11 14:59:24 +02:00
#include <stdio.h>
#include "cpu.h"
#include "sched.h"
#include "vtimer.h"
2014-07-03 10:36:46 +02:00
#include "arch/thread_arch.h"
2014-06-11 14:59:24 +02:00
#include "periph/gpio.h"
2014-08-19 15:21:51 +02:00
#include "periph/spi.h"
2014-06-11 14:59:24 +02:00
#include "periph_conf.h"
#include "at86rf231.h"
#include "at86rf231_spi.h"
/*
SPI1
SCLK : PA5
MISO : PA6
MOSI : PA7
CS : PA4
GPIO
IRQ0 : PC4 : Frame buff empty indicator
DIG2 : ? : RX Frame Time stamping XXX : NOT USED
Reset : PC1 : active low, enable chip
SLEEP : PA2 : control sleep, tx & rx state
*/
2014-07-24 20:08:58 +02:00
uint8_t at86rf231_get_status(void)
2014-06-11 14:59:24 +02:00
{
2014-07-24 20:08:58 +02:00
return at86rf231_reg_read(AT86RF231_REG__TRX_STATUS)
& AT86RF231_TRX_STATUS_MASK__TRX_STATUS;
2014-06-11 14:59:24 +02:00
}
2014-07-24 20:08:58 +02:00
void at86rf231_spi_select(void)
2014-06-11 14:59:24 +02:00
{
gpio_clear(SPI_0_CS_GPIO);
2014-06-11 14:59:24 +02:00
}
2014-07-24 20:08:58 +02:00
void at86rf231_spi_unselect(void)
2014-06-11 14:59:24 +02:00
{
gpio_set(SPI_0_CS_GPIO);
2014-06-11 14:59:24 +02:00
}
2014-07-24 20:08:58 +02:00
void at86rf231_slp_set(void)
2014-06-11 14:59:24 +02:00
{
gpio_set(SPI_0_SLEEP_GPIO);
2014-06-11 14:59:24 +02:00
}
2014-07-24 20:08:58 +02:00
void at86rf231_slp_clear(void)
2014-06-11 14:59:24 +02:00
{
gpio_clear(SPI_0_SLEEP_GPIO);
2014-06-11 14:59:24 +02:00
}
2014-07-24 20:08:58 +02:00
void at86rf231_rst_set(void)
2014-06-11 14:59:24 +02:00
{
gpio_clear(SPI_0_RESET_GPIO);
2014-06-11 14:59:24 +02:00
}
2014-07-24 20:08:58 +02:00
void at86rf231_rst_clear(void)
{
gpio_set(SPI_0_RESET_GPIO);
2014-07-24 20:08:58 +02:00
}
2014-06-11 14:59:24 +02:00
2014-07-24 20:08:58 +02:00
void at86rf231_enable_interrupts(void)
2014-06-11 14:59:24 +02:00
{
2014-07-03 10:36:46 +02:00
gpio_irq_enable(SPI_0_IRQ0_GPIO);
2014-06-11 14:59:24 +02:00
}
2014-07-03 10:36:46 +02:00
2014-07-24 20:08:58 +02:00
void at86rf231_disable_interrupts(void)
2014-06-11 14:59:24 +02:00
{
2014-07-03 10:36:46 +02:00
gpio_irq_disable(SPI_0_IRQ0_GPIO);
2014-06-11 14:59:24 +02:00
}
void at86rf231_gpio_spi_interrupts_init(void)
{
2014-07-03 10:36:46 +02:00
/* set up GPIO pins */
/* SCLK and MOSI*/
GPIOA->CRL &= ~(0xf << (5 * 4));
GPIOA->CRL |= (0xb << (5 * 4));
GPIOA->CRL &= ~(0xf << (7 * 4));
GPIOA->CRL |= (0xb << (7 * 4));
/* MISO */
gpio_init_in(SPI_0_MISO_GPIO, GPIO_NOPULL);
2014-06-11 14:59:24 +02:00
2014-07-03 10:36:46 +02:00
/* SPI init */
2014-08-19 15:21:51 +02:00
spi_init_master(SPI_0, SPI_CONF_FIRST_RISING, SPI_SPEED_5MHZ);
2014-06-11 14:59:24 +02:00
2014-07-03 10:36:46 +02:00
spi_poweron(SPI_0);
2014-06-11 14:59:24 +02:00
2014-07-03 10:36:46 +02:00
/* IRQ0 */
gpio_init_in(SPI_0_IRQ0_GPIO, GPIO_NOPULL);
2014-07-24 20:08:58 +02:00
gpio_init_int(SPI_0_IRQ0_GPIO, GPIO_NOPULL, GPIO_RISING, (gpio_cb_t)at86rf231_rx_irq, NULL);
2014-06-11 14:59:24 +02:00
/* Connect EXTI4 Line to PC4 pin */
2014-07-24 20:08:58 +02:00
at86rf231_enable_interrupts();
2014-06-11 14:59:24 +02:00
2014-07-03 10:36:46 +02:00
/* CS */
gpio_init_out(SPI_0_CS_GPIO, GPIO_NOPULL);
/* SLEEP */
gpio_init_out(SPI_0_SLEEP_GPIO, GPIO_NOPULL);
2014-06-11 14:59:24 +02:00
/* RESET */
2014-07-03 10:36:46 +02:00
gpio_init_out(SPI_0_RESET_GPIO, GPIO_NOPULL);
2014-06-11 14:59:24 +02:00
}
void at86rf231_reset(void)
{
/* force reset */
2014-07-24 20:08:58 +02:00
at86rf231_rst_set();
2014-06-11 14:59:24 +02:00
2014-07-24 20:08:58 +02:00
/* put pins to default values */
at86rf231_spi_unselect();
at86rf231_slp_clear();
2014-06-11 14:59:24 +02:00
2014-07-24 20:08:58 +02:00
/* additional waiting to comply to min rst pulse width */
uint8_t delay = 50;
while (delay--){}
2014-06-11 14:59:24 +02:00
2014-07-24 20:08:58 +02:00
at86rf231_rst_clear();
2014-06-11 14:59:24 +02:00
/* Send a FORCE TRX OFF command */
at86rf231_reg_write(AT86RF231_REG__TRX_STATE, AT86RF231_TRX_STATE__FORCE_TRX_OFF);
/* busy wait for TRX_OFF state */
uint8_t status;
2014-07-24 20:08:58 +02:00
uint8_t max_wait = 100;
2014-06-11 14:59:24 +02:00
do {
status = at86rf231_get_status();
if (!--max_wait) {
2014-07-03 10:36:46 +02:00
printf("at86rf231 : ERROR : could not enter TRX_OFF mode\n");
2014-06-11 14:59:24 +02:00
break;
}
2014-07-03 10:36:46 +02:00
} while ((status & AT86RF231_TRX_STATUS_MASK__TRX_STATUS)
!= AT86RF231_TRX_STATUS__TRX_OFF);
2014-06-11 14:59:24 +02:00
}
2014-07-24 20:08:58 +02:00
uint8_t at86rf231_spi_transfer_byte(uint8_t byte)
2014-06-11 14:59:24 +02:00
{
2014-07-24 20:08:58 +02:00
char ret;
spi_transfer_byte(SPI_0, byte, &ret);
return ret;
2014-06-11 14:59:24 +02:00
}
2014-07-03 10:36:46 +02:00
2014-07-24 20:08:58 +02:00
void at86rf231_spi_transfer(const uint8_t *data_out, uint8_t *data_in, uint16_t length)
2014-06-11 14:59:24 +02:00
{
2014-07-24 20:08:58 +02:00
spi_transfer_bytes(SPI_0, (char*)data_out, (char*)data_in, length);
2014-06-11 14:59:24 +02:00
}