mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 11:52:44 +01:00
512 lines
12 KiB
C
512 lines
12 KiB
C
/*
|
|
* 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 tests
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Test application for the low-level I2C peripheral driver
|
|
*
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
* @author Kevin Weiss <kevin.weiss@haw-hamburg.de>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <errno.h>
|
|
|
|
#include "periph_conf.h"
|
|
#include "periph/gpio.h"
|
|
#include "periph/i2c.h"
|
|
#include "shell.h"
|
|
|
|
#include <xtimer.h>
|
|
|
|
#ifndef I2C_ACK
|
|
#define I2C_ACK (0)
|
|
#endif
|
|
|
|
#define INVALID_ARGS puts("Error: Invalid number of arguments");
|
|
|
|
#define BUFSIZE (128U)
|
|
|
|
#define CONVERT_ERROR (-32768)
|
|
|
|
#define ARG_ERROR (-1)
|
|
|
|
/* i2c_buf is global to reduce stack memory consumption */
|
|
static uint8_t i2c_buf[BUFSIZE];
|
|
|
|
static inline void _print_i2c_read(i2c_t dev, uint16_t *reg, uint8_t *buf,
|
|
int len)
|
|
{
|
|
printf("Success: i2c_%i read %i byte(s) ", dev, len);
|
|
if (reg != NULL) {
|
|
printf("from reg 0x%02x ", *reg);
|
|
}
|
|
printf(": [");
|
|
for (int i = 0; i < len; i++) {
|
|
if (i != 0) {
|
|
printf(", ");
|
|
}
|
|
printf("0x%02x", buf[i]);
|
|
}
|
|
printf("]\n");
|
|
}
|
|
|
|
static inline int _get_num(const char *str)
|
|
{
|
|
errno = 0;
|
|
char *temp;
|
|
long val = strtol(str, &temp, 0);
|
|
|
|
if (temp == str || *temp != '\0' ||
|
|
((val == LONG_MIN || val == LONG_MAX) && errno == ERANGE)) {
|
|
val = CONVERT_ERROR;
|
|
}
|
|
return (int)val;
|
|
}
|
|
|
|
static int _check_param(int argc, char **argv, int c_min, int c_max, char *use)
|
|
{
|
|
int dev;
|
|
|
|
if (argc - 1 < c_min || argc - 1 > c_max) {
|
|
printf("Usage: %s %s\n", argv[0], use);
|
|
INVALID_ARGS;
|
|
return ARG_ERROR;
|
|
}
|
|
|
|
dev = _get_num(argv[1]);
|
|
if (dev < 0 || dev >= (int)I2C_NUMOF) {
|
|
printf("Error: No device, only %d supported\n", (int)I2C_NUMOF);
|
|
return ARG_ERROR;
|
|
}
|
|
return dev;
|
|
}
|
|
|
|
static int _print_i2c_error(int res)
|
|
{
|
|
if (res == -EOPNOTSUPP) {
|
|
printf("Error: EOPNOTSUPP [%d]\n", -res);
|
|
return 1;
|
|
}
|
|
else if (res == -EINVAL) {
|
|
printf("Error: EINVAL [%d]\n", -res);
|
|
return 1;
|
|
}
|
|
else if (res == -EAGAIN) {
|
|
printf("Error: EAGAIN [%d]\n", -res);
|
|
return 1;
|
|
}
|
|
else if (res == -ENXIO) {
|
|
printf("Error: ENXIO [%d]\n", -res);
|
|
return 1;
|
|
}
|
|
else if (res == -EIO) {
|
|
printf("Error: EIO [%d]\n", -res);
|
|
return 1;
|
|
}
|
|
else if (res == -ETIMEDOUT) {
|
|
printf("Error: ETIMEDOUT [%d]\n", -res);
|
|
return 1;
|
|
}
|
|
else if (res == I2C_ACK) {
|
|
printf("Success: I2C_ACK [%d]\n", res);
|
|
return 0;
|
|
}
|
|
printf("Error: Unknown error [%d]\n", res);
|
|
return 1;
|
|
}
|
|
|
|
int cmd_i2c_acquire(int argc, char **argv)
|
|
{
|
|
int dev;
|
|
|
|
dev = _check_param(argc, argv, 1, 1, "DEV");
|
|
if (dev == ARG_ERROR) {
|
|
return 1;
|
|
}
|
|
|
|
printf("Command: i2c_acquire(%i)\n", dev);
|
|
i2c_acquire(dev);
|
|
|
|
printf("Success: i2c_%i acquired\n", dev);
|
|
return 0;
|
|
}
|
|
|
|
int cmd_i2c_release(int argc, char **argv)
|
|
{
|
|
int dev;
|
|
|
|
dev = _check_param(argc, argv, 1, 1, "DEV");
|
|
if (dev == ARG_ERROR) {
|
|
return 1;
|
|
}
|
|
|
|
printf("Command: i2c_release(%i)\n", dev);
|
|
i2c_release(dev);
|
|
|
|
printf("Success: i2c_%i released\n", dev);
|
|
return 0;
|
|
}
|
|
|
|
#ifdef MODULE_PERIPH_I2C_RECONFIGURE
|
|
int cmd_i2c_gpio(int argc, char **argv)
|
|
{
|
|
int dev;
|
|
|
|
dev = _check_param(argc, argv, 1, 1, "DEV");
|
|
if (dev == ARG_ERROR) {
|
|
return 1;
|
|
}
|
|
|
|
gpio_t sda_pin = i2c_pin_sda(dev);
|
|
gpio_t scl_pin = i2c_pin_scl(dev);
|
|
|
|
printf("Command: i2c_deinit_pins(%i)\n", dev);
|
|
i2c_deinit_pins(dev);
|
|
|
|
gpio_init(sda_pin, GPIO_OUT);
|
|
gpio_init(scl_pin, GPIO_OUT);
|
|
|
|
xtimer_sleep(1);
|
|
|
|
printf("Command: gpio_set()\n");
|
|
gpio_set(sda_pin);
|
|
gpio_set(scl_pin);
|
|
|
|
xtimer_sleep(1);
|
|
|
|
printf("Command: gpio_clear()\n");
|
|
gpio_clear(sda_pin);
|
|
gpio_clear(scl_pin);
|
|
|
|
xtimer_sleep(1);
|
|
|
|
printf("Command: i2c_init_pins(%i)\n", dev);
|
|
i2c_init_pins(dev);
|
|
|
|
printf("Success: i2c_%i re-init\n", dev);
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
int cmd_i2c_read_reg(int argc, char **argv)
|
|
{
|
|
int res;
|
|
uint16_t addr;
|
|
uint16_t reg;
|
|
uint8_t flags = 0;
|
|
uint8_t data;
|
|
int dev;
|
|
|
|
dev = _check_param(argc, argv, 4, 4, "DEV ADDR REG FLAG");
|
|
if (dev == ARG_ERROR) {
|
|
return 1;
|
|
}
|
|
|
|
addr = _get_num(argv[2]);
|
|
reg = _get_num(argv[3]);
|
|
flags = _get_num(argv[4]);
|
|
|
|
printf("Command: i2c_read_reg(%i, 0x%02x, 0x%02x, 0x%02x)\n", dev, addr,
|
|
reg, flags);
|
|
res = i2c_read_reg(dev, addr, reg, &data, flags);
|
|
|
|
if (res == I2C_ACK) {
|
|
_print_i2c_read(dev, ®, &data, 1);
|
|
return 0;
|
|
}
|
|
return _print_i2c_error(res);
|
|
}
|
|
|
|
int cmd_i2c_read_regs(int argc, char **argv)
|
|
{
|
|
int res;
|
|
uint16_t addr;
|
|
uint16_t reg;
|
|
uint8_t flags = 0;
|
|
int len;
|
|
int dev;
|
|
|
|
dev = _check_param(argc, argv, 5, 5, "DEV ADDR REG LEN FLAG");
|
|
if (dev == ARG_ERROR) {
|
|
return 1;
|
|
}
|
|
|
|
addr = _get_num(argv[2]);
|
|
reg = _get_num(argv[3]);
|
|
len = _get_num(argv[4]);
|
|
flags = _get_num(argv[5]);
|
|
|
|
if (len < 1 || len > (int)BUFSIZE) {
|
|
puts("Error: invalid LENGTH parameter given");
|
|
return 1;
|
|
}
|
|
else {
|
|
printf("Command: i2c_read_regs(%i, 0x%02x, 0x%02x, %i, 0x%02x)\n", dev,
|
|
addr, reg, len, flags);
|
|
res = i2c_read_regs(dev, addr, reg, i2c_buf, len, flags);
|
|
}
|
|
|
|
if (res == I2C_ACK) {
|
|
_print_i2c_read(dev, ®, i2c_buf, len);
|
|
return 0;
|
|
}
|
|
return _print_i2c_error(res);
|
|
}
|
|
|
|
int cmd_i2c_read_byte(int argc, char **argv)
|
|
{
|
|
int res;
|
|
uint16_t addr;
|
|
uint8_t flags = 0;
|
|
uint8_t data;
|
|
int dev;
|
|
|
|
dev = _check_param(argc, argv, 3, 3, "DEV ADDR FLAG");
|
|
if (dev == ARG_ERROR) {
|
|
return 1;
|
|
}
|
|
|
|
addr = _get_num(argv[2]);
|
|
flags = _get_num(argv[3]);
|
|
|
|
printf("Command: i2c_read_byte(%i, 0x%02x, 0x%02x)\n", dev, addr, flags);
|
|
res = i2c_read_byte(dev, addr, &data, flags);
|
|
|
|
if (res == I2C_ACK) {
|
|
_print_i2c_read(dev, NULL, &data, 1);
|
|
return 0;
|
|
}
|
|
return _print_i2c_error(res);
|
|
}
|
|
|
|
int cmd_i2c_read_bytes(int argc, char **argv)
|
|
{
|
|
int res;
|
|
uint16_t addr;
|
|
uint8_t flags = 0;
|
|
int len;
|
|
int dev;
|
|
|
|
dev = _check_param(argc, argv, 4, 4, "DEV ADDR LENGTH FLAG");
|
|
if (dev == ARG_ERROR) {
|
|
return 1;
|
|
}
|
|
|
|
addr = _get_num(argv[2]);
|
|
len = _get_num(argv[3]);
|
|
flags = _get_num(argv[4]);
|
|
|
|
if (len < 1 || len > (int)BUFSIZE) {
|
|
puts("Error: invalid LENGTH parameter given");
|
|
return 1;
|
|
}
|
|
else {
|
|
printf("Command: i2c_read_bytes(%i, 0x%02x, %i, 0x%02x)\n", dev,
|
|
addr, len, flags);
|
|
res = i2c_read_bytes(dev, addr, i2c_buf, len, flags);
|
|
}
|
|
|
|
if (res == I2C_ACK) {
|
|
_print_i2c_read(dev, NULL, i2c_buf, len);
|
|
return 0;
|
|
}
|
|
return _print_i2c_error(res);
|
|
}
|
|
|
|
int cmd_i2c_write_byte(int argc, char **argv)
|
|
{
|
|
int res;
|
|
uint16_t addr;
|
|
uint8_t flags = 0;
|
|
uint8_t data;
|
|
int dev;
|
|
|
|
dev = _check_param(argc, argv, 4, 4, "DEV ADDR BYTE FLAG");
|
|
if (dev == ARG_ERROR) {
|
|
return 1;
|
|
}
|
|
|
|
addr = _get_num(argv[2]);
|
|
data = _get_num(argv[3]);
|
|
flags = _get_num(argv[4]);
|
|
|
|
printf("Command: i2c_write_byte(%i, 0x%02x, 0x%02x, [0x%02x", dev, addr,
|
|
flags, data);
|
|
puts("])");
|
|
res = i2c_write_byte(dev, addr, data, flags);
|
|
|
|
if (res == I2C_ACK) {
|
|
printf("Success: i2c_%i wrote 1 byte to the bus\n", dev);
|
|
return 0;
|
|
}
|
|
return _print_i2c_error(res);
|
|
}
|
|
|
|
int cmd_i2c_write_bytes(int argc, char **argv)
|
|
{
|
|
int res;
|
|
uint16_t addr;
|
|
uint8_t flags = 0;
|
|
int len = argc - 4;
|
|
int dev;
|
|
|
|
dev = _check_param(argc, argv, 4, 3 + BUFSIZE,
|
|
"DEV ADDR FLAG BYTE0 [BYTE1 [BYTE_n [...]]]");
|
|
if (dev == ARG_ERROR) {
|
|
return 1;
|
|
}
|
|
|
|
addr = _get_num(argv[2]);
|
|
flags = _get_num(argv[3]);
|
|
for (int i = 0; i < len; i++) {
|
|
i2c_buf[i] = _get_num(argv[i + 4]);
|
|
}
|
|
|
|
printf("Command: i2c_write_bytes(%i, 0x%02x, 0x%02x, [", dev, addr,
|
|
flags);
|
|
for (int i = 0; i < len; i++) {
|
|
if (i != 0) {
|
|
printf(", ");
|
|
}
|
|
printf("0x%02x", i2c_buf[i]);
|
|
}
|
|
puts("])");
|
|
res = i2c_write_bytes(dev, addr, i2c_buf, len, flags);
|
|
|
|
if (res == I2C_ACK) {
|
|
printf("Success: i2c_%i wrote %i bytes\n", dev, len);
|
|
return 0;
|
|
}
|
|
return _print_i2c_error(res);
|
|
}
|
|
|
|
int cmd_i2c_write_reg(int argc, char **argv)
|
|
{
|
|
int res;
|
|
uint16_t addr;
|
|
uint16_t reg;
|
|
uint8_t flags = 0;
|
|
uint8_t data;
|
|
int dev;
|
|
|
|
dev = _check_param(argc, argv, 5, 5, "DEV ADDR REG BYTE FLAG");
|
|
if (dev == ARG_ERROR) {
|
|
return 1;
|
|
}
|
|
|
|
addr = _get_num(argv[2]);
|
|
reg = _get_num(argv[3]);
|
|
data = _get_num(argv[4]);
|
|
flags = _get_num(argv[5]);
|
|
|
|
printf("Command: i2c_write_reg(%i, 0x%02x, 0x%02x, 0x%02x, [0x%02x", dev,
|
|
addr, reg, flags, data);
|
|
puts("])");
|
|
res = i2c_write_reg(dev, addr, reg, data, flags);
|
|
|
|
if (res == I2C_ACK) {
|
|
printf("Success: i2c_%i wrote 1 byte\n", dev);
|
|
return 0;
|
|
}
|
|
return _print_i2c_error(res);
|
|
}
|
|
|
|
int cmd_i2c_write_regs(int argc, char **argv)
|
|
{
|
|
int res;
|
|
uint16_t addr;
|
|
uint16_t reg;
|
|
uint8_t flags = 0;
|
|
int len = argc - 5;
|
|
int dev;
|
|
|
|
dev = _check_param(argc, argv, 5, 4 + BUFSIZE,
|
|
"DEV ADDR REG FLAG BYTE0 [BYTE1 ...]");
|
|
if (dev == ARG_ERROR) {
|
|
return 1;
|
|
}
|
|
|
|
addr = _get_num(argv[2]);
|
|
reg = _get_num(argv[3]);
|
|
flags = _get_num(argv[4]);
|
|
for (int i = 0; i < len; i++) {
|
|
i2c_buf[i] = _get_num(argv[i + 5]);
|
|
}
|
|
|
|
printf("Command: i2c_write_regs(%i, 0x%02x, 0x%02x, 0x%02x, [", dev,
|
|
addr, reg, flags);
|
|
for (int i = 0; i < len; i++) {
|
|
if (i != 0) {
|
|
printf(", ");
|
|
}
|
|
printf("0x%02x", i2c_buf[i]);
|
|
}
|
|
puts("])");
|
|
res = i2c_write_regs(dev, addr, reg, i2c_buf, len, flags);
|
|
|
|
if (res == I2C_ACK) {
|
|
printf("Success: i2c_%i wrote %i bytes to reg 0x%02x\n",
|
|
dev, len, reg);
|
|
return 0;
|
|
}
|
|
return _print_i2c_error(res);
|
|
}
|
|
|
|
int cmd_i2c_get_devs(int argc, char **argv)
|
|
{
|
|
(void)argv;
|
|
(void)argc;
|
|
printf("Command: return I2C_NUMOF\n");
|
|
printf("Success: Amount of i2c devices: [%d]\n", I2C_NUMOF);
|
|
return 0;
|
|
}
|
|
|
|
int cmd_i2c_get_id(int argc, char **argv)
|
|
{
|
|
(void)argv;
|
|
(void)argc;
|
|
puts("Success: [periph_i2c]");
|
|
return 0;
|
|
}
|
|
|
|
static const shell_command_t shell_commands[] = {
|
|
{ "i2c_acquire", "Get access to the I2C bus", cmd_i2c_acquire },
|
|
{ "i2c_release", "Release to the I2C bus", cmd_i2c_release },
|
|
#ifdef MODULE_PERIPH_I2C_RECONFIGURE
|
|
{ "i2c_gpio", "Re-configures I2C pins to GPIO mode and back.", cmd_i2c_gpio },
|
|
#endif
|
|
{ "i2c_read_reg", "Read byte from register", cmd_i2c_read_reg },
|
|
{ "i2c_read_regs", "Read bytes from registers", cmd_i2c_read_regs },
|
|
{ "i2c_read_byte", "Read byte from the I2C device", cmd_i2c_read_byte },
|
|
{ "i2c_read_bytes", "Read bytes from the I2C device", cmd_i2c_read_bytes },
|
|
{ "i2c_write_byte", "Write byte to the I2C device", cmd_i2c_write_byte },
|
|
{ "i2c_write_bytes", "Write bytes to the I2C device", cmd_i2c_write_bytes },
|
|
{ "i2c_write_reg", "Write byte to register", cmd_i2c_write_reg },
|
|
{ "i2c_write_regs", "Write bytes to registers", cmd_i2c_write_regs },
|
|
{ "i2c_get_devs", "Gets amount of supported i2c devices", cmd_i2c_get_devs },
|
|
{ "i2c_get_id", "Get the id of the fw", cmd_i2c_get_id },
|
|
{ NULL, NULL, NULL }
|
|
};
|
|
|
|
int main(void)
|
|
{
|
|
puts("Start: Test for the low-level I2C driver");
|
|
|
|
char line_buf[SHELL_DEFAULT_BUFSIZE];
|
|
shell_run(shell_commands, line_buf, SHELL_DEFAULT_BUFSIZE);
|
|
|
|
return 0;
|
|
}
|