/* * 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 * @author Kevin Weiss * * @} */ #include #include #include #include "periph_conf.h" #include "periph/gpio.h" #include "periph/i2c.h" #include "shell.h" #include #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 consumtion */ 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 res; int dev; dev = _check_param(argc, argv, 1, 1, "DEV"); if (dev == ARG_ERROR) { return 1; } printf("Command: i2c_acquire(%i)\n", dev); res = i2c_acquire(dev); if (res == I2C_ACK) { printf("Success: i2c_%i acquired\n", dev); return 0; } return _print_i2c_error(res); } 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; }