mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2024-12-29 04:50:03 +01:00
343 lines
7.5 KiB
C
343 lines
7.5 KiB
C
/*
|
|
* Copyright (C) 2014 Hamburg University of Applied Sciences
|
|
*
|
|
* 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 tests
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Test application for nrf24l01p lowlevel functions
|
|
*
|
|
* @author Peter Kietzmann <peter.kietzmann@haw-hamburg.de>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#ifndef SPI_PORT
|
|
#error "SPI_PORT not defined"
|
|
#endif
|
|
#ifndef CE_PIN
|
|
#error "CE_PIN not defined"
|
|
#endif
|
|
#ifndef CS_PIN
|
|
#error "CS_PIN not defined"
|
|
#endif
|
|
#ifndef IRQ_PIN
|
|
#error "IRQ_PIN not defined"
|
|
#endif
|
|
|
|
|
|
#include <stdio.h>
|
|
#include <stdint.h>
|
|
#include <inttypes.h>
|
|
#include <board.h>
|
|
#include <time.h>
|
|
|
|
#include "nrf24l01p.h"
|
|
#include "nrf24l01p_settings.h"
|
|
#include "periph/spi.h"
|
|
#include "periph/gpio.h"
|
|
#include "vtimer.h"
|
|
#include "hwtimer.h"
|
|
#include "shell.h"
|
|
#include "shell_commands.h"
|
|
#include "thread.h"
|
|
#include "msg.h"
|
|
|
|
#define TEST_RX_MSG 1
|
|
|
|
#define SHELL_BUFFER_SIZE 128
|
|
|
|
static int shell_read(void);
|
|
static void shell_write(int);
|
|
static int cmd_send(int argc, char **argv);
|
|
static int cmd_print_regs(int argc, char **argv);
|
|
static int cmd_its(int argc, char **argv);
|
|
|
|
|
|
void printbin(unsigned byte);
|
|
void print_register(char reg, int num_bytes);
|
|
|
|
|
|
static nrf24l01p_t nrf24l01p_0;
|
|
|
|
/**
|
|
* define some additional shell commands
|
|
*/
|
|
static const shell_command_t shell_commands[] = {
|
|
{ "prgs", "print registers", cmd_print_regs },
|
|
{ "it", "init transceiver", cmd_its },
|
|
{ "send", "send 32 bytes data", cmd_send },
|
|
{ NULL, NULL, NULL }
|
|
};
|
|
|
|
void prtbin(unsigned byte)
|
|
{
|
|
for (char i = 0; i < 8; i++) {
|
|
printf("%u", (byte >> (7 - i)) & 0x0001);
|
|
}
|
|
|
|
puts("\n");
|
|
}
|
|
|
|
/**
|
|
* @print register
|
|
*/
|
|
void print_register(char reg, int num_bytes)
|
|
{
|
|
|
|
vtimer_init();
|
|
|
|
char buf_return[num_bytes];
|
|
int ret;
|
|
|
|
|
|
gpio_clear(CS_PIN);
|
|
vtimer_usleep(1);
|
|
ret = spi_transfer_regs(SPI_PORT, (CMD_R_REGISTER | (REGISTER_MASK & reg)), 0, buf_return, num_bytes);
|
|
gpio_set(CS_PIN);
|
|
|
|
if (ret < 0) {
|
|
printf("Error in read access\n");
|
|
}
|
|
else {
|
|
if (num_bytes < 2) {
|
|
printf("0x%x returned: ", reg);
|
|
|
|
for (int i = 0; i < num_bytes; i++) {
|
|
prtbin(buf_return[i]);
|
|
}
|
|
}
|
|
else {
|
|
printf("0x%x returned: ", reg);
|
|
|
|
for (int i = 0; i < num_bytes; i++) {
|
|
printf("%x ", buf_return[i]);
|
|
}
|
|
|
|
printf("\n\n");
|
|
}
|
|
}
|
|
}
|
|
|
|
char rx_handler_stack[THREAD_STACKSIZE_MAIN];
|
|
|
|
/* RX handler that waits for a message from the ISR */
|
|
void *nrf24l01p_rx_handler(void *arg)
|
|
{
|
|
msg_t msg_q[1];
|
|
msg_init_queue(msg_q, 1);
|
|
unsigned int pid = thread_getpid();
|
|
char rx_buf[NRF24L01P_MAX_DATA_LENGTH];
|
|
|
|
puts("Registering nrf24l01p_rx_handler thread...");
|
|
nrf24l01p_register(&nrf24l01p_0, &pid);
|
|
|
|
msg_t m;
|
|
|
|
while (msg_receive(&m)) {
|
|
printf("nrf24l01p_rx_handler got a message: ");
|
|
|
|
switch (m.type) {
|
|
case RCV_PKT_NRF24L01P:
|
|
puts("Received packet.");
|
|
|
|
/* CE low */
|
|
nrf24l01p_stop((nrf24l01p_t *)m.content.ptr);
|
|
|
|
/* read payload */
|
|
nrf24l01p_read_payload((nrf24l01p_t *)m.content.ptr, rx_buf, NRF24L01P_MAX_DATA_LENGTH);
|
|
|
|
/* flush rx fifo */
|
|
nrf24l01p_flush_rx_fifo((nrf24l01p_t *)m.content.ptr);
|
|
|
|
/* CE high */
|
|
nrf24l01p_start((nrf24l01p_t *)m.content.ptr);
|
|
|
|
/* print rx buffer */
|
|
for (int i = 0; i < NRF24L01P_MAX_DATA_LENGTH; i++) {
|
|
printf("%i ", rx_buf[i]);
|
|
}
|
|
|
|
puts("");
|
|
|
|
break;
|
|
|
|
default:
|
|
puts("stray message.");
|
|
break;
|
|
}
|
|
}
|
|
|
|
puts("nrf24l01p_rx_handler: this should not have happened!");
|
|
|
|
return NULL;
|
|
}
|
|
|
|
/**
|
|
* @init transceiver
|
|
*/
|
|
int cmd_its(int argc, char **argv)
|
|
{
|
|
(void) argc;
|
|
(void) argv;
|
|
|
|
puts("Init Transceiver\n");
|
|
|
|
/* initialize transceiver device */
|
|
if (nrf24l01p_init(&nrf24l01p_0, SPI_PORT, CE_PIN, CS_PIN, IRQ_PIN) < 0) {
|
|
puts("Error in nrf24l01p_init");
|
|
return;
|
|
}
|
|
|
|
/* create thread that gets msg when data arrives */
|
|
if (thread_create(
|
|
rx_handler_stack, sizeof(rx_handler_stack), THREAD_PRIORITY_MAIN - 1, 0,
|
|
nrf24l01p_rx_handler, 0, "nrf24l01p_rx_handler") < 0) {
|
|
puts("Error in thread_create");
|
|
return;
|
|
}
|
|
|
|
/* setup device as receiver */
|
|
if (nrf24l01p_set_rxmode(&nrf24l01p_0) < 0) {
|
|
puts("Error in nrf24l01p_set_rxmode");
|
|
return 1;
|
|
}
|
|
|
|
return cmd_print_regs(0, 0);
|
|
}
|
|
|
|
/**
|
|
* @set TX mode
|
|
*/
|
|
int cmd_send(int argc, char **argv)
|
|
{
|
|
(void) argc;
|
|
(void) argv;
|
|
|
|
puts("Send");
|
|
|
|
int status = 0;
|
|
char tx_buf[NRF24L01P_MAX_DATA_LENGTH];
|
|
|
|
/* fill TX buffer with numbers 32..1 */
|
|
for (int i = 0; i < sizeof(tx_buf); i++) {
|
|
tx_buf[i] = NRF24L01P_MAX_DATA_LENGTH - i;
|
|
}
|
|
/* power on the device */
|
|
if (nrf24l01p_on(&nrf24l01p_0) < 0) {
|
|
puts("Error in nrf24l01p_on");
|
|
return;
|
|
}
|
|
/* setup device as transmitter */
|
|
if (nrf24l01p_set_txmode(&nrf24l01p_0) < 0) {
|
|
puts("Error in nrf24l01p_set_txmode");
|
|
return;
|
|
}
|
|
/* load data to transmit into device */
|
|
if (nrf24l01p_preload(&nrf24l01p_0, tx_buf, NRF24L01P_MAX_DATA_LENGTH) < 0) {
|
|
puts("Error in nrf24l01p_preload");
|
|
return;
|
|
}
|
|
/* trigger transmitting */
|
|
nrf24l01p_transmit(&nrf24l01p_0);
|
|
/* wait while data is pysically transmitted */
|
|
hwtimer_wait(DELAY_DATA_ON_AIR);
|
|
/* get status of the transceiver */
|
|
status = nrf24l01p_get_status(&nrf24l01p_0);
|
|
if (status < 0) {
|
|
puts("Error in nrf24l01p_get_status");
|
|
}
|
|
if (status & TX_DS) {
|
|
puts("Sent Packet");
|
|
}
|
|
/* setup device as receiver */
|
|
if (nrf24l01p_set_rxmode(&nrf24l01p_0) < 0) {
|
|
puts("Error in nrf24l01p_set_rxmode");
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* @print registers
|
|
*/
|
|
int cmd_print_regs(int argc, char **argv)
|
|
{
|
|
(void) argc;
|
|
(void) argv;
|
|
|
|
printf("################## Print Registers ###################\n");
|
|
|
|
|
|
puts("REG_CONFIG: ");
|
|
print_register(REG_CONFIG, 1);
|
|
|
|
puts("REG_EN_AA: ");
|
|
print_register(REG_EN_AA, 1);
|
|
|
|
puts("REG_EN_RXADDR: ");
|
|
print_register(REG_EN_RXADDR, 1);
|
|
|
|
puts("REG_SETUP_AW: ");
|
|
print_register(REG_SETUP_AW, 1);
|
|
|
|
puts("REG_SETUP_RETR: ");
|
|
print_register(REG_SETUP_RETR, 1);
|
|
|
|
puts("REG_RF_CH: ");
|
|
print_register(REG_RF_CH, 1);
|
|
|
|
puts("REG_RF_SETUP: ");
|
|
print_register(REG_RF_SETUP, 1);
|
|
|
|
puts("REG_STATUS: ");
|
|
print_register(REG_STATUS, 1);
|
|
|
|
puts("REG_OBSERVE_TX: ");
|
|
print_register(REG_OBSERVE_TX, 1);
|
|
|
|
puts("REG_RPD: ");
|
|
print_register(REG_RPD, 1);
|
|
|
|
puts("REG_RX_ADDR_P0: ");
|
|
print_register(REG_RX_ADDR_P0, INITIAL_ADDRESS_WIDTH);
|
|
|
|
puts("REG_TX_ADDR: ");
|
|
print_register(REG_TX_ADDR, INITIAL_ADDRESS_WIDTH);
|
|
|
|
puts("REG_RX_PW_P0: ");
|
|
print_register(REG_RX_PW_P0, 1);
|
|
|
|
puts("REG_FIFO_STATUS: ");
|
|
print_register(REG_FIFO_STATUS, 1);
|
|
|
|
puts("REG_DYNPD: ");
|
|
print_register(REG_DYNPD, 1);
|
|
|
|
puts("REG_FEATURE: ");
|
|
print_register(REG_FEATURE, 1);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int main(void)
|
|
{
|
|
shell_t shell;
|
|
|
|
puts("Welcome to RIOT!");
|
|
|
|
puts("Initializing shell...");
|
|
shell_init(&shell, shell_commands, SHELL_BUFFER_SIZE, getchar, putchar);
|
|
|
|
puts("Starting shell...");
|
|
shell_run(&shell);
|
|
return 0;
|
|
}
|