新唐NUC980使用记录:在用户应用中使用GPIO

Posted Naisu Xu

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了新唐NUC980使用记录:在用户应用中使用GPIO相关的知识,希望对你有一定的参考价值。

文章目录

目的

GPIO是最基础的外设,使用频率也非常高,这篇文章将简单体验在NUC980 Liunx用户应用中使用GPIO功能。

这篇文章中内容均在下面的开发板上进行测试:
《新唐NUC980使用记录:自制开发板(基于NUC980DK61YC)》

开发板中提供了两组共四个直连到GPIO口上的轻触按钮和发光二极管,可以方便地进行GPIO功能测试:

这篇文章主要是在下面文章基础上进行的:
《新唐NUC980使用记录:访问以太网(LAN8720A) & 启用SSH》

基础说明

默认情况下NUC980官方的内核是配置启用了sysfs文件系统GPIO支持的:

所以可以通过读写 /sys/class/gpio/ 目录下指定GPIO口编号的文件来操作GPIO口。GPIO口编号换算如下:
PB13 = 32 x 1(PA) + 13 = 45
PF10 = 32 x 5(PA/PB/PC/PD/PE) + 10 = 170
PE10 = 32 x 4(PA/PB/PC/PD) + 10 = 138
PE12 = 32 x 4(PA/PB/PC/PD) + 12 = 140

需要注意的是默认情况下PE10和PE12是被设置成USB相关功能的,需要修改内核进行调整:

在终端中操作

基于上面内容我们就可以直接在终端中操作GPIO口了:

# 导出以使用GPIO45
echo 45 > /sys/class/gpio/export
# 导出后将在 /sys/class/gpio/ 目录下出现 gpio45 目录,读写其中的文件即可操作该GPIO口

# 将GPIO45设置为输出模式
echo out > /sys/class/gpio/gpio45/direction
# 将GPIO45设置为输出高电平
echo 1 > /sys/class/gpio/gpio45/value
# 将GPIO45设置为输出低电平
echo 0 > /sys/class/gpio/gpio45/value

# ====================

# 导出以使用GPIO140
echo 140 > /sys/class/gpio/export
# 将GPIO140设置为输入模式
echo in > /sys/class/gpio/gpio140/direction
# 打印GPIO140端口电平
cat /sys/class/gpio/gpio140/value

# ====================

# 取消使用GPIO45
echo 45 > /sys/class/gpio/unexport
# 取消使用GPIO45
echo 140 > /sys/class/gpio/unexport


上面演示中按钮按下和松开时可以读取到不同的电平值。

使用程序操作

除了在终端中使用,也可以通过程序进行操作:

cd ~/nuc980-sdk/
mkdir -p apps/gpio
cd apps/gpio/
gedit main.c

在 main.c 文件中写入下面代码:

#include <stdlib.h>
#include <unistd.h>

int main(void)

    int i;

    system("echo 45 > /sys/class/gpio/export");
    system("echo out > /sys/class/gpio/gpio45/direction");

    for (i = 0; i < 8; i++)
    
        system("echo 0 > /sys/class/gpio/gpio45/value");
        sleep(1);
        system("echo 1 > /sys/class/gpio/gpio45/value");
        sleep(1);
    

    system("echo 45 > /sys/class/gpio/unexport");

    return 0;

编译生成程序并拷贝到开发板中:

export PATH=$PATH:/home/nx/nuc980-sdk/buildroot-2022.02.3/output/host/bin
arm-linux-gcc main.c
# 开发板启用了SSH的话可以使用SCP命令将程序通过网络拷贝到开发板中
scp a.out root@192.168.31.142:/root/

在开发板上运行程序:

/root/a.out

上面代码也可以使用下面这种传统的文件操作方式:

#include <string.h>
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>

#define OPEN_FILE(fd, file, flag) \\
    (fd) = open((file), (flag)); \\
    if ((fd) < 0)  printf("open %s failed!\\n", (file)); return -1; 

#define WRITE_FILE(fd, str) \\
    if(write((fd), (str), strlen(str)) != strlen(str)) \\
     printf("write %s failed!\\n", (str)); return -1; 

int main(void)

    int i, fd;

    if(access("/sys/class/gpio/gpio45/", F_OK))
    
        OPEN_FILE(fd, "/sys/class/gpio/export", O_WRONLY);
        WRITE_FILE(fd, "45");
        close(fd);
    

    OPEN_FILE(fd, "/sys/class/gpio/gpio45/direction", O_WRONLY);
    WRITE_FILE(fd, "out");
    close(fd);

	OPEN_FILE(fd, "/sys/class/gpio/gpio45/value", O_RDWR);
    for (i = 0; i < 8; i++)
    
        lseek(fd, 0, SEEK_SET); // 移动文件操作指针到文件开头
        WRITE_FILE(fd, "0");
        sleep(1);
        lseek(fd, 0, SEEK_SET); // 移动文件操作指针到文件开头
        WRITE_FILE(fd, "1");
        sleep(1);
    
    close(fd);

    OPEN_FILE(fd, "/sys/class/gpio/unexport", O_WRONLY);
    WRITE_FILE(fd, "45");
    close(fd);

    return 0;

上面的C语言程序都是输出控制LED的,也可以用程序来读取按键输入电平:

#include <string.h>
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>

#define OPEN_FILE(fd, file, flag) \\
    (fd) = open((file), (flag)); \\
    if ((fd) < 0)  printf("open %s failed!\\n", (file)); return -1; 

#define WRITE_FILE(fd, str) \\
    if(write((fd), (str), strlen(str)) != strlen(str)) \\
     printf("write %s failed!\\n", (str)); return -1; 

#define READ_FILE(fd, bufptr, size) \\
    if (read((fd), bufptr, size) != size) \\
     printf("read failed!\\n"); return -1; 
    
int main(void)

    int i, fd;
    char value[2] = 0;

    if(access("/sys/class/gpio/gpio140/", F_OK))
    
        OPEN_FILE(fd, "/sys/class/gpio/export", O_WRONLY);
        WRITE_FILE(fd, "140");
        close(fd);
    

    OPEN_FILE(fd, "/sys/class/gpio/gpio140/direction", O_WRONLY);
    WRITE_FILE(fd, "in");
    close(fd);

    OPEN_FILE(fd, "/sys/class/gpio/gpio140/value", O_RDWR);
    for (i = 0; i < 8; i++)
    
        lseek(fd, 0, SEEK_SET); // 移动文件操作指针到文件开头
        READ_FILE(fd, value, 1);
        printf("value is %s\\n", value);
        sleep(2);
    
    close(fd);

    OPEN_FILE(fd, "/sys/class/gpio/unexport", O_WRONLY);
    WRITE_FILE(fd, "140");
    close(fd);

    return 0;


在程序运行过程中按下按钮可以看到输出的值改变。

上面对于输入值获取操作属于轮询方式,也可以使用中断方式来获取(开发板上两个按键的引脚都是有外部中断功能的):

#include <string.h>
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <poll.h> 

#define OPEN_FILE(fd, file, flag) \\
    (fd) = open((file), (flag)); \\
    if ((fd) < 0)  printf("open %s failed!\\n", (file)); return -1; 

#define WRITE_FILE(fd, str) \\
    if(write((fd), (str), strlen(str)) != strlen(str)) \\
     printf("write %s failed!\\n", (str)); return -1; 

#define READ_FILE(fd, bufptr, size) \\
    if (read((fd), bufptr, size) != size) \\
     printf("read failed!\\n"); return -1; 
    
int main(void)

    int i, fd, ret;
    char value[2] = 0;
    struct pollfd fds[1];
    nfds_t nfds = 1;
    

    if(access("/sys/class/gpio/gpio140/", F_OK))
    
        OPEN_FILE(fd, "/sys/class/gpio/export", O_WRONLY);
        WRITE_FILE(fd, "140");
        close(fd);
    

    OPEN_FILE(fd, "/sys/class/gpio/gpio140/direction", O_WRONLY);
    WRITE_FILE(fd, "in");
    close(fd);

    OPEN_FILE(fd, "/sys/class/gpio/gpio140/edge", O_WRONLY); // edge文件用于设置外部中断触发方式
    // none 无; rising 上升沿触发; falling 下降沿触发(实际测试有点问题); both 双边触发
    WRITE_FILE(fd, "rising"); // 使能下降沿中断
    close(fd);

    OPEN_FILE(fd, "/sys/class/gpio/gpio140/value", O_RDWR | O_NONBLOCK);
    READ_FILE(fd, value, 1); // 先读取一次,以免触发第一次不期望的中断
    fds[0].fd = fd;
    fds[0].events = POLLPRI;
    for (i = 0; i < 8; i++)
    
        ret = poll(fds, nfds, 5000);
        // ret = poll(fds, nfds, -1); // timeout=-1 无限等待
        if (ret > 0)
        
            // 这里返回的fds[0].revents其实是 POLLERR | POLLPRI
            if (fds[0].revents & POLLPRI)
            
                lseek(fd, 0, SEEK_SET); // 移动文件操作指针到文件开头
                READ_FILE(fd, value, 1);
                printf("value is %s\\n", value);
            
        
        else if (ret == 0)
        
            printf("timeout!\\n");
        
        else
        
            printf("poll error!\\n");
        
    
    close(fd);

    OPEN_FILE(fd, "/sys/class/gpio/unexport", O_WRONLY);
    WRITE_FILE(fd, "140");
    close(fd);

    return 0;

总结

GPIO是最基础的外设,使用频率非常高,同时在用户应用中使用GPIO也是比较简单的,基础的使用参考上面这些内容就差不多了。

以上是关于新唐NUC980使用记录:在用户应用中使用GPIO的主要内容,如果未能解决你的问题,请参考以下文章

新唐NUC980使用记录:开发环境准备与编译配置基础说明

新唐NUC980使用记录:自制开发板(基于NUC980DK61YC)

新唐NUC980使用记录:访问以太网(LAN8720A) & 启用SSH

新唐NUC980使用记录:向内核添加USB无线网卡驱动(基于RTL8188EUS)

新唐NUC980使用记录:使用wpa_supplicant访问无线网络

新唐NUC980使用记录:基础说明与资料索引