1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
RIOT/tests/drivers/pcf857x/main.c

435 lines
13 KiB
C

/*
* Copyright (C) 2018 Gunar Schorcht
*
* 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
* @brief Test application for Texas Instruments PCF857X I2C I/O expanders
* @author Gunar Schorcht <gunar@schorcht.net>
* @file
*
* ## Overview
*
* This test application demonstrates the usage of the PCF857X driver interface
* and can be used to test each PCF857X expander I/O pin with shell commands.
*
* The application bases on the test application for GPIO peripheral drivers
* which is under following copyright:
*
* Copyright (C) 2014,2017 Freie Universität Berlin
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
*
* ## Compilation
*
* To use the test application, compile it with one or more of the pseudomodules
* `pcf8574`, `pcf8574a` or `pcf8575` to enable the driver for your
* expander modules. Please check the default configuration parameters in
* `$(RIOTBASE)/drivers/pcf857x/include/pcf857x_params.h` and modify them
* if necessary. Alternatively, a modified version of this file could be
* placed in the directory of this test application to override it.
* ```
* USEMODULE=pcf8575 make -C tests/drivers/pcf857x BOARD=...
* ```
* @note When no pseudomodule is given, `pcf8575` is used by default.
*
* To use external interrupts with the expander I/O pins, the PCF857X
* low-active open-drain interrupt signal has to be enabled. Add module
* `pcf857x_irq` for this purpose and define the MCU interrupt pin by
* parameter `PCF857X_PARAM_INT_PIN`, e.g.
* ```
* CFLAGS="-DPCF857X_PARAM_INT_PIN=\(GPIO_PIN\(0,6\)\)" \
* USEMODULE="pcf8575 pcf857x_irq" make -C tests/drivers/pcf857x BOARD=...
* ```
* @note Since interrupts are handled in the context of a separate event thread,
* enabling interrupts requires more RAM.
*
* ## Usage
*
* The test allows to use commands as known from GPIO test application for
* PCF857X expanders:
* ```
* > help
* Command Description
* ---------------------------------------
* init_out init as output (push-pull mode)
* init_in init as input w/o pull resistor
* init_in_pu init as input with pull-up
* init_od init as output (open-drain without pull resistor)
* init_od_pu init as output (open-drain with pull-up)
* init_int init as external INT w/o pull resistor
* enable_int enable or disable gpio interrupt
* read read pin status
* set set pin to HIGH
* clear set pin to LOW
* toggle toggle pin
* bench run a set of predefined benchmarks
* ```
* The number of the first PCF857X expander port used by the test application
* is defined by the macro `PCF857X_PORT_0`, which is 16 by default. This value
* can be overridden during compilation, e.g.:
* ```
* CFLAGS="-DPCF857X_PORT_0=8" \
* USEMODULE=pcf8575 make -C tests/drivers/pcf857x BOARD=...
* ```
* Using the port number defined by `PCF857X_PORT_0` and the following port
* numbers, you can apply the command to the PCF857X expander ports. For
* example, the following command initializes I/O pin 7 of the first PCF857X
* expander:
* ```
* init_out 16 7
* ```
* Commands with port numbers less than `PCF857X_PORT_0` refer to GPIO
* peripheral ports. Thus, both the I/O pins of the PCF857X expanders as well
* as the GPIO peripheral pins of the MCU can be addressed by all commands.
*/
#include <stdio.h>
#include <stdlib.h>
#include "pcf857x.h"
#include "pcf857x_params.h"
#include "irq.h"
#include "shell.h"
#include "benchmark.h"
#define BENCH_RUNS_DEFAULT (100UL * 100)
/* Number of configured PCF857X I/O expander devices */
#define PCF857X_NUM ARRAY_SIZE(pcf857x_params)
/* Port number of the first PCF857X I/O expander device */
#ifndef PCF857X_PORT_0
#define PCF857X_PORT_0 (16)
#endif
/* PCF857X devices allocation */
pcf857x_t pcf857x_dev[PCF857X_NUM];
#ifdef MODULE_PCF857X_IRQ
static void cb(void *arg)
{
printf("INT: external interrupt from pin %i\n", (int)arg);
}
#endif
static int init_pin(int argc, char **argv, gpio_mode_t mode)
{
if (argc < 3) {
printf("usage: %s <port> <pin>\n", argv[0]);
return 1;
}
int po = atoi(argv[1]);
int pi = atoi(argv[2]);
if (po < PCF857X_PORT_0) {
gpio_init(GPIO_PIN(po, pi), mode);
}
else if (pcf857x_gpio_init(&pcf857x_dev[po - PCF857X_PORT_0],
pi, mode) < 0) {
printf("error: init PCF857X pin (dev %i, pin %02i) failed\n", po, pi);
return 1;
}
return 0;
}
static int init_out(int argc, char **argv)
{
return init_pin(argc, argv, GPIO_OUT);
}
static int init_in(int argc, char **argv)
{
return init_pin(argc, argv, GPIO_IN);
}
static int init_in_pu(int argc, char **argv)
{
return init_pin(argc, argv, GPIO_IN_PU);
}
static int init_od(int argc, char **argv)
{
return init_pin(argc, argv, GPIO_OD);
}
static int init_od_pu(int argc, char **argv)
{
return init_pin(argc, argv, GPIO_OD_PU);
}
#ifdef MODULE_PCF857X_IRQ
static int init_int(int argc, char **argv)
{
gpio_mode_t mode = GPIO_IN;
gpio_flank_t flank;
int fl;
if (argc < 4) {
printf("usage: %s <port> <pin> <flank>\n", argv[0]);
puts("\tflank:\n"
"\t0: falling\n"
"\t1: rising\n"
"\t2: both\n");
return 1;
}
int po = atoi(argv[1]);
int pi = atoi(argv[2]);
fl = atoi(argv[3]);
switch (fl) {
case 0:
flank = GPIO_FALLING;
break;
case 1:
flank = GPIO_RISING;
break;
case 2:
flank = GPIO_BOTH;
break;
default:
puts("error: invalid value for active flank");
return 1;
}
if (po < PCF857X_PORT_0) {
gpio_init_int(GPIO_PIN(po, pi), mode, flank, cb, (void *)pi);
}
else if (pcf857x_gpio_init_int(&pcf857x_dev[po - PCF857X_PORT_0], pi,
mode, flank, cb, (void *)pi) < 0) {
printf("error: init_int PCF857X pin (dev %i, pin %02i) failed\n",
po, pi);
return 1;
}
return 0;
}
static int enable_int(int argc, char **argv)
{
int status;
if (argc < 4) {
printf("usage: %s <port> <pin> <status>\n", argv[0]);
puts("\tstatus:\n"
"\t0: disable\n"
"\t1: enable\n");
return 1;
}
int po = atoi(argv[1]);
int pi = atoi(argv[2]);
status = atoi(argv[3]);
switch (status) {
case 0:
puts("disabling GPIO interrupt");
if (po < PCF857X_PORT_0) {
gpio_irq_disable(GPIO_PIN(po, pi));
}
else {
pcf857x_gpio_irq_disable(&pcf857x_dev[po - PCF857X_PORT_0], pi);
}
break;
case 1:
puts("enabling GPIO interrupt");
if (po < PCF857X_PORT_0) {
gpio_irq_enable(GPIO_PIN(po, pi));
}
else {
pcf857x_gpio_irq_enable(&pcf857x_dev[po - PCF857X_PORT_0], pi);
}
break;
default:
puts("error: invalid status");
return 1;
}
return 0;
}
#endif /* MODULE_PCF857X_IRQ */
static int read_pin(int argc, char **argv)
{
if (argc < 3) {
printf("usage: %s <port> <pin>\n", argv[0]);
return 1;
}
int po = atoi(argv[1]);
int pi = atoi(argv[2]);
if (po < PCF857X_PORT_0) {
if (gpio_read(GPIO_PIN(po, pi))) {
printf("GPIO pin (port %i, pin %02i) is HIGH\n", po, pi);
}
else {
printf("GPIO pin (port %i, pin %02i) is LOW\n", po, pi);
} }
else {
if (pcf857x_gpio_read(&pcf857x_dev[po - PCF857X_PORT_0], pi)) {
printf("PCF857X pin (dev %i, pin %02i) is HIGH\n", po, pi);
}
else {
printf("PCF857X pin (dev %i, pin %02i) is LOW\n", po, pi);
}
}
return 0;
}
static int set_pin(int argc, char **argv)
{
if (argc < 3) {
printf("usage: %s <port> <pin>\n", argv[0]);
return 1;
}
int po = atoi(argv[1]);
int pi = atoi(argv[2]);
if (po < PCF857X_PORT_0) {
gpio_set(GPIO_PIN(po, pi));
}
else {
pcf857x_gpio_set(&pcf857x_dev[po - PCF857X_PORT_0], pi);
}
return 0;
}
static int clear_pin(int argc, char **argv)
{
if (argc < 3) {
printf("usage: %s <port> <pin>\n", argv[0]);
return 1;
}
int po = atoi(argv[1]);
int pi = atoi(argv[2]);
if (po < PCF857X_PORT_0) {
gpio_clear(GPIO_PIN(po, pi));
}
else {
pcf857x_gpio_clear(&pcf857x_dev[po - PCF857X_PORT_0], pi);
}
return 0;
}
static int toggle_pin(int argc, char **argv)
{
if (argc < 3) {
printf("usage: %s <port> <pin>\n", argv[0]);
return 1;
}
int po = atoi(argv[1]);
int pi = atoi(argv[2]);
if (po < PCF857X_PORT_0) {
gpio_toggle(GPIO_PIN(po, pi));
}
else {
pcf857x_gpio_toggle(&pcf857x_dev[po - PCF857X_PORT_0], pi);
}
return 0;
}
static int bench(int argc, char **argv)
{
if (argc < 3) {
printf("usage: %s <port> <pin> [# of runs]\n", argv[0]);
return 1;
}
int po = atoi(argv[1]);
int pi = atoi(argv[2]);
unsigned long runs = BENCH_RUNS_DEFAULT;
if (argc > 3) {
runs = (unsigned long)atol(argv[3]);
}
puts("\nGPIO driver run-time performance benchmark\n");
if (po < PCF857X_PORT_0) {
BENCHMARK_FUNC("nop loop", runs, __asm__ volatile("nop"));
gpio_init(GPIO_PIN(po, pi), GPIO_OUT);
BENCHMARK_FUNC("gpio_set", runs, gpio_set(GPIO_PIN(po, pi)));
BENCHMARK_FUNC("gpio_clear", runs, gpio_clear(GPIO_PIN(po, pi)));
BENCHMARK_FUNC("gpio_toggle", runs, gpio_toggle(GPIO_PIN(po, pi)));
gpio_init(GPIO_PIN(po, pi), GPIO_IN);
BENCHMARK_FUNC("gpio_read", runs, (void)gpio_read(GPIO_PIN(po, pi)));
gpio_init(GPIO_PIN(po, pi), GPIO_OUT);
BENCHMARK_FUNC("gpio_write", runs, gpio_write(GPIO_PIN(po, pi), 1));
}
else {
pcf857x_t* dev = &pcf857x_dev[po - PCF857X_PORT_0];
BENCHMARK_FUNC("nop loop", runs, __asm__ volatile("nop"));
pcf857x_gpio_init(dev, pi, GPIO_OUT);
BENCHMARK_FUNC("gpio_set", runs, pcf857x_gpio_set(dev, pi));
BENCHMARK_FUNC("gpio_clear", runs, pcf857x_gpio_clear(dev, pi));
BENCHMARK_FUNC("gpio_toggle", runs, pcf857x_gpio_toggle(dev, pi));
pcf857x_gpio_init(dev, pi, GPIO_IN);
BENCHMARK_FUNC("gpio_read", runs, (void)pcf857x_gpio_read(dev, pi));
pcf857x_gpio_init(dev, pi, GPIO_OUT);
BENCHMARK_FUNC("gpio_write", runs, pcf857x_gpio_write(dev, pi, 1));
puts("\n --- DONE ---");
}
return 0;
}
static const shell_command_t shell_commands[] = {
{ "init_out", "init as output (push-pull mode)", init_out },
{ "init_in", "init as input w/o pull resistor", init_in },
{ "init_in_pu", "init as input with pull-up", init_in_pu },
{ "init_od", "init as output (open-drain without pull resistor)", init_od },
{ "init_od_pu", "init as output (open-drain with pull-up)", init_od_pu },
#ifdef MODULE_PCF857X_IRQ
{ "init_int", "init as external INT w/o pull resistor", init_int },
{ "enable_int", "enable or disable gpio interrupt", enable_int },
#endif
{ "read", "read pin status", read_pin },
{ "set", "set pin to HIGH", set_pin },
{ "clear", "set pin to LOW", clear_pin },
{ "toggle", "toggle pin", toggle_pin },
{ "bench", "run a set of predefined benchmarks", bench },
{ NULL, NULL, NULL }
};
int main(void)
{
puts("PCF857X I/O expander GPIO peripheral driver test\n");
puts("Initializing PCF857X");
/* initialize configured PCF857X devices */
for (unsigned i = 0; i < PCF857X_NUM; i++) {
if (pcf857x_init(&pcf857x_dev[i], &pcf857x_params[i]) != PCF857X_OK) {
puts("[Failed]");
return 1;
}
}
puts("[OK]\n");
printf("In this test, pins are specified by integer port and pin numbers.\n"
"PCF8574 has 8 I/O pins labeled P00...P07.\n"
"PCF8575 has 16 I/O pins labeled P00...P07 and P10...P17\n"
"Use port %d and pin 0...15 in all commands to access them.\n\n"
"NOTE: make sure the values you use exist! The\n"
" behavior for not existing ports/pins is not defined!\n",
PCF857X_PORT_0);
/* start the shell */
char line_buf[SHELL_DEFAULT_BUFSIZE];
shell_run(shell_commands, line_buf, SHELL_DEFAULT_BUFSIZE);
return 0;
}