2016-08-01 21:53:20 +02:00
|
|
|
/****************************************************************************
|
|
|
|
* examples/gpio/gpio_main.c
|
|
|
|
*
|
|
|
|
* Copyright (C) 2016 Gregory Nutt. All rights reserved.
|
|
|
|
* Author: Gregory Nutt <gnutt@nuttx.org>
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
*
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in
|
|
|
|
* the documentation and/or other materials provided with the
|
|
|
|
* distribution.
|
|
|
|
* 3. Neither the name NuttX nor the names of its contributors may be
|
|
|
|
* used to endorse or promote products derived from this software
|
|
|
|
* without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
|
|
|
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
|
|
|
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
|
|
|
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
|
|
|
|
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
|
|
|
|
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
|
|
|
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Included Files
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include <nuttx/config.h>
|
|
|
|
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <signal.h>
|
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
#include <nuttx/ioexpander/gpio.h>
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static void show_usage(FAR const char *progname)
|
|
|
|
{
|
2020-07-09 12:10:49 +02:00
|
|
|
fprintf(stderr, "USAGE: %s [-w <signo>] [-o <value>] <driver-path>\n",
|
|
|
|
progname);
|
2016-08-01 23:27:48 +02:00
|
|
|
fprintf(stderr, " %s -h\n", progname);
|
2016-08-01 21:53:20 +02:00
|
|
|
fprintf(stderr, "Where:\n");
|
2020-07-09 12:10:49 +02:00
|
|
|
fprintf(stderr, "\t<driver-path>: The full path to the GPIO pin "
|
|
|
|
"driver.\n");
|
|
|
|
fprintf(stderr,
|
|
|
|
"\t-w <signo>: Wait for an signal if this is an interrupt pin.\n");
|
|
|
|
fprintf(stderr,
|
|
|
|
"\t-o <value>: Write this value (0 or 1) if this is an output "
|
|
|
|
"pin.\n");
|
2016-08-01 21:53:20 +02:00
|
|
|
fprintf(stderr, "\t-h: Print this usage information and exit.\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Public Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* gpio_main
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
int main(int argc, FAR char *argv[])
|
|
|
|
{
|
|
|
|
FAR char *devpath = NULL;
|
|
|
|
enum gpio_pintype_e pintype;
|
|
|
|
bool havesigno = false;
|
|
|
|
bool invalue;
|
|
|
|
bool outvalue = false;
|
|
|
|
bool haveout = false;
|
|
|
|
int signo = 0;
|
|
|
|
int ndx;
|
|
|
|
int ret;
|
|
|
|
int fd;
|
|
|
|
|
|
|
|
/* Parse command line */
|
|
|
|
|
|
|
|
if (argc < 2)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "ERROR: Missing required arguments\n");
|
|
|
|
show_usage(argv[0]);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
ndx = 1;
|
|
|
|
if (strcmp(argv[ndx], "-h") == 0)
|
|
|
|
{
|
|
|
|
show_usage(argv[0]);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strcmp(argv[ndx], "-w") == 0)
|
|
|
|
{
|
|
|
|
havesigno = true;
|
|
|
|
|
|
|
|
if (++ndx >= argc)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "ERROR: Missing argument to -o\n");
|
|
|
|
show_usage(argv[0]);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
signo = atoi(argv[ndx]);
|
|
|
|
|
|
|
|
if (++ndx >= argc)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "ERROR: Missing required <driver-path>\n");
|
|
|
|
show_usage(argv[0]);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-01 23:27:48 +02:00
|
|
|
if (ndx < argc && strcmp(argv[ndx], "-o") == 0)
|
2016-08-01 21:53:20 +02:00
|
|
|
{
|
|
|
|
if (++ndx >= argc)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "ERROR: Missing argument to -o\n");
|
|
|
|
show_usage(argv[0]);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
2016-08-01 23:27:48 +02:00
|
|
|
if (strcmp(argv[ndx], "0") == 0)
|
2016-08-01 21:53:20 +02:00
|
|
|
{
|
|
|
|
outvalue = false;
|
|
|
|
haveout = true;
|
|
|
|
}
|
2016-08-01 23:27:48 +02:00
|
|
|
else if (strcmp(argv[ndx], "1") == 0)
|
2016-08-01 21:53:20 +02:00
|
|
|
{
|
|
|
|
outvalue = true;
|
|
|
|
haveout = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf(stderr, "ERROR: Invalid argument to -o\n");
|
|
|
|
show_usage(argv[0]);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (++ndx >= argc)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "ERROR: Missing required <driver-path>\n");
|
|
|
|
show_usage(argv[0]);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
devpath = argv[ndx];
|
|
|
|
printf("Driver: %s\n", devpath);
|
|
|
|
|
|
|
|
/* Open the pin driver */
|
|
|
|
|
|
|
|
fd = open(devpath, O_RDWR);
|
|
|
|
if (fd < 0)
|
|
|
|
{
|
|
|
|
int errcode = errno;
|
|
|
|
fprintf(stderr, "ERROR: Failed to open %s: %d\n", devpath, errcode);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get the pin type */
|
|
|
|
|
|
|
|
ret = ioctl(fd, GPIOC_PINTYPE, (unsigned long)((uintptr_t)&pintype));
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
int errcode = errno;
|
2020-07-09 12:10:49 +02:00
|
|
|
fprintf(stderr, "ERROR: Failed to read pintype from %s: %d\n",
|
|
|
|
devpath, errcode);
|
2016-08-01 21:53:20 +02:00
|
|
|
close(fd);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Read the pin value */
|
|
|
|
|
|
|
|
ret = ioctl(fd, GPIOC_READ, (unsigned long)((uintptr_t)&invalue));
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
int errcode = errno;
|
2020-07-09 12:10:49 +02:00
|
|
|
fprintf(stderr, "ERROR: Failed to read value from %s: %d\n",
|
|
|
|
devpath, errcode);
|
2016-08-01 21:53:20 +02:00
|
|
|
close(fd);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Perform the test based on the pintype and on command line options */
|
|
|
|
|
|
|
|
switch (pintype)
|
|
|
|
{
|
|
|
|
case GPIO_INPUT_PIN:
|
|
|
|
{
|
2020-07-09 12:10:49 +02:00
|
|
|
printf(" Input pin: Value=%u\n",
|
|
|
|
(unsigned int)invalue);
|
2016-08-01 21:53:20 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2020-07-09 11:50:14 +02:00
|
|
|
case GPIO_INPUT_PIN_PULLUP:
|
|
|
|
{
|
2020-07-09 12:10:49 +02:00
|
|
|
printf(" Input pin (pull-up): Value=%u\n",
|
|
|
|
(unsigned int)invalue);
|
2020-07-09 11:50:14 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GPIO_INPUT_PIN_PULLDOWN:
|
|
|
|
{
|
2020-07-09 12:10:49 +02:00
|
|
|
printf(" Input pin (pull-down): Value=%u\n",
|
|
|
|
(unsigned int)invalue);
|
2020-07-09 11:50:14 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2016-08-01 21:53:20 +02:00
|
|
|
case GPIO_OUTPUT_PIN:
|
2020-07-09 11:50:14 +02:00
|
|
|
case GPIO_OUTPUT_PIN_OPENDRAIN:
|
2016-08-01 21:53:20 +02:00
|
|
|
{
|
|
|
|
printf(" Output pin: Value=%u\n", (unsigned int)invalue);
|
|
|
|
|
|
|
|
if (haveout)
|
|
|
|
{
|
|
|
|
printf(" Writing: Value=%u\n", (unsigned int)outvalue);
|
|
|
|
|
|
|
|
/* Write the pin value */
|
|
|
|
|
|
|
|
ret = ioctl(fd, GPIOC_WRITE, (unsigned long)outvalue);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
int errcode = errno;
|
2020-07-09 12:10:49 +02:00
|
|
|
fprintf(stderr,
|
|
|
|
"ERROR: Failed to write value %u from %s: %d\n",
|
2016-08-01 21:53:20 +02:00
|
|
|
devpath, (unsigned int)outvalue, errcode);
|
|
|
|
close(fd);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Re-read the pin value */
|
|
|
|
|
2020-07-09 12:10:49 +02:00
|
|
|
ret = ioctl(fd, GPIOC_READ,
|
|
|
|
(unsigned long)((uintptr_t)&invalue));
|
2016-08-01 21:53:20 +02:00
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
int errcode = errno;
|
2020-07-09 12:10:49 +02:00
|
|
|
fprintf(stderr,
|
|
|
|
"ERROR: Failed to re-read value from %s: %d\n",
|
2016-08-01 21:53:20 +02:00
|
|
|
devpath, errcode);
|
|
|
|
close(fd);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf(" Verify: Value=%u\n", (unsigned int)invalue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GPIO_INTERRUPT_PIN:
|
|
|
|
{
|
|
|
|
printf(" Interrupt pin: Value=%u\n", invalue);
|
|
|
|
|
|
|
|
if (havesigno)
|
|
|
|
{
|
2018-11-08 14:50:42 +01:00
|
|
|
struct sigevent notify;
|
2016-08-01 21:53:20 +02:00
|
|
|
struct timespec ts;
|
|
|
|
sigset_t set;
|
|
|
|
|
2018-11-08 14:50:42 +01:00
|
|
|
notify.sigev_notify = SIGEV_SIGNAL;
|
|
|
|
notify.sigev_signo = signo;
|
|
|
|
|
2016-08-01 21:53:20 +02:00
|
|
|
/* Set up to receive signal */
|
|
|
|
|
2018-11-08 14:50:42 +01:00
|
|
|
ret = ioctl(fd, GPIOC_REGISTER, (unsigned long)¬ify);
|
2016-08-01 21:53:20 +02:00
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
int errcode = errno;
|
2018-11-08 14:50:42 +01:00
|
|
|
|
2020-07-09 12:10:49 +02:00
|
|
|
fprintf(stderr,
|
|
|
|
"ERROR: Failed to setup for signal from %s: %d\n",
|
2016-08-01 21:53:20 +02:00
|
|
|
devpath, errcode);
|
2018-11-08 14:50:42 +01:00
|
|
|
|
2016-08-01 21:53:20 +02:00
|
|
|
close(fd);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Wait up to 5 seconds for the signal */
|
|
|
|
|
2020-01-02 13:09:50 +01:00
|
|
|
sigemptyset(&set);
|
|
|
|
sigaddset(&set, signo);
|
2016-08-01 21:53:20 +02:00
|
|
|
|
|
|
|
ts.tv_sec = 5;
|
|
|
|
ts.tv_nsec = 0;
|
|
|
|
|
2017-03-02 05:40:43 +01:00
|
|
|
ret = sigtimedwait(&set, NULL, &ts);
|
2020-01-02 13:09:50 +01:00
|
|
|
ioctl(fd, GPIOC_UNREGISTER, 0);
|
2016-08-01 21:53:20 +02:00
|
|
|
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
int errcode = errno;
|
|
|
|
if (errcode == EAGAIN)
|
|
|
|
{
|
|
|
|
printf(" [Five second timeout with no signal]\n");
|
|
|
|
close(fd);
|
|
|
|
return EXIT_SUCCESS;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-07-09 12:10:49 +02:00
|
|
|
fprintf(stderr, "ERROR: Failed to wait signal %d "
|
|
|
|
"from %s: %d\n", signo, devpath, errcode);
|
2016-08-01 21:53:20 +02:00
|
|
|
close(fd);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Re-read the pin value */
|
|
|
|
|
2020-07-09 12:10:49 +02:00
|
|
|
ret = ioctl(fd, GPIOC_READ,
|
|
|
|
(unsigned long)((uintptr_t)&invalue));
|
2016-08-01 21:53:20 +02:00
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
int errcode = errno;
|
2020-07-09 12:10:49 +02:00
|
|
|
fprintf(stderr,
|
|
|
|
"ERROR: Failed to re-read value from %s: %d\n",
|
2016-08-01 21:53:20 +02:00
|
|
|
devpath, errcode);
|
|
|
|
close(fd);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf(" Verify: Value=%u\n", (unsigned int)invalue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
fprintf(stderr, "ERROR: Unrecognized pintype: %d\n", (int)pintype);
|
|
|
|
close(fd);
|
2018-08-13 15:47:26 +02:00
|
|
|
return EXIT_FAILURE;
|
2016-08-01 21:53:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
close(fd);
|
|
|
|
return EXIT_SUCCESS;
|
|
|
|
}
|