gpio-event-mon.c 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192
  1. /*
  2. * gpio-event-mon - monitor GPIO line events from userspace
  3. *
  4. * Copyright (C) 2016 Linus Walleij
  5. *
  6. * This program is free software; you can redistribute it and/or modify it
  7. * under the terms of the GNU General Public License version 2 as published by
  8. * the Free Software Foundation.
  9. *
  10. * Usage:
  11. * gpio-event-mon -n <device-name> -o <offset>
  12. */
  13. #include <unistd.h>
  14. #include <stdlib.h>
  15. #include <stdbool.h>
  16. #include <stdio.h>
  17. #include <dirent.h>
  18. #include <errno.h>
  19. #include <string.h>
  20. #include <poll.h>
  21. #include <fcntl.h>
  22. #include <getopt.h>
  23. #include <inttypes.h>
  24. #include <sys/ioctl.h>
  25. #include <linux/gpio.h>
  26. int monitor_device(const char *device_name,
  27. unsigned int line,
  28. u_int32_t handleflags,
  29. u_int32_t eventflags,
  30. unsigned int loops)
  31. {
  32. struct gpioevent_request req;
  33. struct gpiohandle_data data;
  34. char *chrdev_name;
  35. int fd;
  36. int ret;
  37. int i = 0;
  38. ret = asprintf(&chrdev_name, "/dev/%s", device_name);
  39. if (ret < 0)
  40. return -ENOMEM;
  41. fd = open(chrdev_name, 0);
  42. if (fd == -1) {
  43. ret = -errno;
  44. fprintf(stderr, "Failed to open %s\n", chrdev_name);
  45. goto exit_close_error;
  46. }
  47. req.lineoffset = line;
  48. req.handleflags = handleflags;
  49. req.eventflags = eventflags;
  50. strcpy(req.consumer_label, "gpio-event-mon");
  51. ret = ioctl(fd, GPIO_GET_LINEEVENT_IOCTL, &req);
  52. if (ret == -1) {
  53. ret = -errno;
  54. fprintf(stderr, "Failed to issue GET EVENT "
  55. "IOCTL (%d)\n",
  56. ret);
  57. goto exit_close_error;
  58. }
  59. /* Read initial states */
  60. ret = ioctl(req.fd, GPIOHANDLE_GET_LINE_VALUES_IOCTL, &data);
  61. if (ret == -1) {
  62. ret = -errno;
  63. fprintf(stderr, "Failed to issue GPIOHANDLE GET LINE "
  64. "VALUES IOCTL (%d)\n",
  65. ret);
  66. goto exit_close_error;
  67. }
  68. fprintf(stdout, "Monitoring line %d on %s\n", line, device_name);
  69. fprintf(stdout, "Initial line value: %d\n", data.values[0]);
  70. while (1) {
  71. struct gpioevent_data event;
  72. ret = read(req.fd, &event, sizeof(event));
  73. if (ret == -1) {
  74. if (errno == -EAGAIN) {
  75. fprintf(stderr, "nothing available\n");
  76. continue;
  77. } else {
  78. ret = -errno;
  79. fprintf(stderr, "Failed to read event (%d)\n",
  80. ret);
  81. break;
  82. }
  83. }
  84. if (ret != sizeof(event)) {
  85. fprintf(stderr, "Reading event failed\n");
  86. ret = -EIO;
  87. break;
  88. }
  89. fprintf(stdout, "GPIO EVENT %" PRIu64 ": ", event.timestamp);
  90. switch (event.id) {
  91. case GPIOEVENT_EVENT_RISING_EDGE:
  92. fprintf(stdout, "rising edge");
  93. break;
  94. case GPIOEVENT_EVENT_FALLING_EDGE:
  95. fprintf(stdout, "falling edge");
  96. break;
  97. default:
  98. fprintf(stdout, "unknown event");
  99. }
  100. fprintf(stdout, "\n");
  101. i++;
  102. if (i == loops)
  103. break;
  104. }
  105. exit_close_error:
  106. if (close(fd) == -1)
  107. perror("Failed to close GPIO character device file");
  108. free(chrdev_name);
  109. return ret;
  110. }
  111. void print_usage(void)
  112. {
  113. fprintf(stderr, "Usage: gpio-event-mon [options]...\n"
  114. "Listen to events on GPIO lines, 0->1 1->0\n"
  115. " -n <name> Listen on GPIOs on a named device (must be stated)\n"
  116. " -o <n> Offset to monitor\n"
  117. " -d Set line as open drain\n"
  118. " -s Set line as open source\n"
  119. " -r Listen for rising edges\n"
  120. " -f Listen for falling edges\n"
  121. " [-c <n>] Do <n> loops (optional, infinite loop if not stated)\n"
  122. " -? This helptext\n"
  123. "\n"
  124. "Example:\n"
  125. "gpio-event-mon -n gpiochip0 -o 4 -r -f\n"
  126. );
  127. }
  128. int main(int argc, char **argv)
  129. {
  130. const char *device_name = NULL;
  131. unsigned int line = -1;
  132. unsigned int loops = 0;
  133. u_int32_t handleflags = GPIOHANDLE_REQUEST_INPUT;
  134. u_int32_t eventflags = 0;
  135. int c;
  136. while ((c = getopt(argc, argv, "c:n:o:dsrf?")) != -1) {
  137. switch (c) {
  138. case 'c':
  139. loops = strtoul(optarg, NULL, 10);
  140. break;
  141. case 'n':
  142. device_name = optarg;
  143. break;
  144. case 'o':
  145. line = strtoul(optarg, NULL, 10);
  146. break;
  147. case 'd':
  148. handleflags |= GPIOHANDLE_REQUEST_OPEN_DRAIN;
  149. break;
  150. case 's':
  151. handleflags |= GPIOHANDLE_REQUEST_OPEN_SOURCE;
  152. break;
  153. case 'r':
  154. eventflags |= GPIOEVENT_REQUEST_RISING_EDGE;
  155. break;
  156. case 'f':
  157. eventflags |= GPIOEVENT_REQUEST_FALLING_EDGE;
  158. break;
  159. case '?':
  160. print_usage();
  161. return -1;
  162. }
  163. }
  164. if (!device_name || line == -1) {
  165. print_usage();
  166. return -1;
  167. }
  168. if (!eventflags) {
  169. printf("No flags specified, listening on both rising and "
  170. "falling edges\n");
  171. eventflags = GPIOEVENT_REQUEST_BOTH_EDGES;
  172. }
  173. return monitor_device(device_name, line, handleflags,
  174. eventflags, loops);
  175. }