1. 程式人生 > >GPIO編程2:使用GPIO監聽中斷完整程序

GPIO編程2:使用GPIO監聽中斷完整程序

fcntl reading case port io編程 con 編程 \n roman

一個完整的使用GPIO捕捉中斷的程序:

#include <stdlib.h> 
#include <stdio.h>  
#include <string.h>
#include <unistd.h>
#include <fcntl.h>
#include <poll.h>

#define SYSFS_GPIO_EXPORT              "/sys/class/gpio/export"  
#define SYSFS_GPIO_UNEXPORT            "/sys/class/gpio/unexport"

int
gpio_export(int pin); int gpio_unexport(int pin); int gpio_direction(int pin, int dir); int gpio_write(int pin, int value); int gpio_read(int pin); int gpio_export(int pin) { char buffer[64]; int len; int fd; fd = open(SYSFS_GPIO_EXPORT, O_WRONLY);
if (fd < 0) { printf("Failed to open export for writing!\n"); return(-1); } len = snprintf(buffer, sizeof(buffer), "%d", pin); if (write(fd, buffer, len) < 0) { printf("Failed to export gpio!\n"); } close(fd);
return 0; } int gpio_unexport(int pin) { char buffer[64]; int len; int fd; fd = open(SYSFS_GPIO_UNEXPORT, O_WRONLY); if (fd < 0) { printf("Failed to open unexport for writing!\n"); return -1; } len = snprintf(buffer, sizeof(buffer), "%d", pin); if (write(fd, buffer, len) < 0) { printf("Failed to unexport gpio!\n"); return -1; } close(fd); return 0; } //dir: 0-->IN, 1-->OUT int gpio_direction(int pin, int dir) { static const char dir_str[] = "in\0out"; char path[64]; int fd; snprintf(path, sizeof(path), "/sys/class/gpio/gpio%d/direction\n", pin); fd = open(path, O_WRONLY); if (fd < 0) { printf("Failed to open gpio direction for writing!\n"); return -1; } if (write(fd, &dir_str[dir == 0 ? 0 : 3], dir == 0 ? 2 : 3) < 0) { printf("Failed to set direction!\n"); return -1; } close(fd); return 0; } //value: 0-->LOW, 1-->HIGH int gpio_write(int pin, int value) { static const char values_str[] = "01"; char path[64]; int fd; snprintf(path, sizeof(path), "/sys/class/gpio/gpio%d/value", pin); fd = open(path, O_WRONLY); if (fd < 0) { printf("Failed to open gpio value for writing!\n"); return -1; } if (write(fd, &values_str[value == 0 ? 0 : 1], 1) < 0) { printf("Failed to write value!\n"); return -1; } close(fd); return 0; } int gpio_read(int pin) { char path[64]; char value_str[3]; int fd; snprintf(path, sizeof(path), "/sys/class/gpio/gpio%d/value", pin); fd = open(path, O_RDONLY); if (fd < 0) { printf("Failed to open gpio value for reading!\n"); return -1; } if (read(fd, value_str, 3) < 0) { printf("Failed to read value!\n"); return -1; } close(fd); return (atoi(value_str)); } /* none表示引腳為輸入,不是中斷引腳 rising表示引腳為中斷輸入,上升沿觸發 falling表示引腳為中斷輸入,下降沿觸發 both表示引腳為中斷輸入,邊沿觸發 */ // 0-->none, 1-->rising, 2-->falling, 3-->both int gpio_edge(int pin, int edge) { const char dir_str[] = "none\0rising\0falling\0both"; char ptr; char path[64]; int fd; switch(edge) { case 0: ptr = 0; break; case 1: ptr = 5; break; case 2: ptr = 12; break; case 3: ptr = 20; break; default: ptr = 0; break; } snprintf(path, sizeof(path), "/sys/class/gpio/gpio%d/edge", pin); fd = open(path, O_WRONLY); if (fd < 0) { printf("Failed to open gpio edge for writing!\n"); return -1; } if (write(fd, &dir_str[ptr], strlen(&dir_str[ptr])) < 0) { printf("Failed to set edge!\n"); return -1; } close(fd); return 0; } int main() { int PIN = 481; int gpio_fd, ret; struct pollfd fds[1]; char buff[10]; unsigned char cnt = 0; //按鍵引腳初始化 gpio_export(PIN); gpio_direction(PIN, 0); gpio_edge(PIN,2); gpio_fd = open("/sys/class/gpio/gpio481/value",O_RDONLY); if(gpio_fd < 0) { printf("Failed to open value!\n"); return -1; } fds[0].fd = gpio_fd; fds[0].events = POLLPRI; ret = read(gpio_fd,buff,10); if( ret == -1 ) printf("read\n"); while(1) { ret = poll(fds,1,0); if(ret == -1 ) printf("poll fail\n"); if(fds[0].revents & POLLPRI) { printf("%d\n",gpio_read(PIN)); } } return 0; }

GPIO編程2:使用GPIO監聽中斷完整程序