Linux系统PWM应用编程

本章我们将学习如何对开发板上的PWM 设备进行应用编程。

应用层如何操控PWM

与LED 设备一样,PWM 同样也是通过sysfs 方式进行操控,进入到/sys/class/pwm 目录下,如下所示:
在这里插入图片描述
这里列举出了8 个以pwmchipX(X 表示数字0~7)命名的文件夹,这八个文件夹其实就对应了I.MX6U的8 个PWM 控制器,I.MX6U 总共有8 个PWM 控制器,大家可以通过查询I.MX6U 参考手册得知。

我们随便以其中一个为例,进入到pwmchip0 目录下:
在这里插入图片描述
在这个目录下我们重点关注的是export、npwm 以及unexport 这三个属性文件,下面一一进行介绍:
⚫ npwm:这是一个只读属性,读取该文件可以得知该PWM 控制器下共有几路PWM 输出,如下所示:
在这里插入图片描述
I.MX6U 每个PWM 控制器只有1 路PWM 输出,所以总共有8 路PWM,分别对应I.MX6U 的
PWM1~PWM8 这8 路输出(pwmchip0 对应PWM1,pwmchip1 对应PWM2,以此类推,开发板出厂系统中,PWM1 已经被用作LCD 背光控制了,应用层不能直接对它进行控制了;而其它PWM 均不能使用,原因在于I/O 资源不够,为了满足板子上其它外设对I/O 引脚的需求,取舍情况下只能如此!)。
⚫ export:与GPIO 控制一样,在使用PWM 之前,也需要将其导出,通过export 属性进行导出,以下所示:

echo 0 > export

在这里插入图片描述
0 表示一个编号,注意,每个PWM 控制器(pwmchipX)下,使用export 属性文件导出PWM 时,编号都是从0 开始;因为I.MX6U 每个控制器都只有一路PWM,所以都只能使用编号0,如下所示:

echo 0 > /sys/class/pwm/pwmchip0/export #导出PWM1
echo 0 > /sys/class/pwm/pwmchip1/export #导出PWM2
echo 0 > /sys/class/pwm/pwmchip2/export #导出PWM3
echo 0 > /sys/class/pwm/pwmchip3/export #导出PWM4
echo 0 > /sys/class/pwm/pwmchip4/export #导出PWM5
echo 0 > /sys/class/pwm/pwmchip5/export #导出PWM6

导出成功后会在pwmchipX(X 表示数字0~7)目录下生成一个名为pwm0 的目录,如图24.1.4 所示,稍后介绍。
⚫ unexport:将导出的PWM 删除。当使用完PWM 之后,我们需要将导出的PWM 删除,譬如:
echo 0 > unexport
写入到unexport 文件中的编号与写入到export 文件中的编号是相对应的;需要注意的是,export 文件和unexport 文件都是只写的、没有读权限。
如何控制PWM
通过export 导出之后,便会生成pwm0 这个目录,我们进入到该目录下看看:

在这里插入图片描述

该目录下也有一些属性文件,我们重点关注duty_cycle、enable、period 以及polarity 这四个属性文件,接下来一一进行介绍。
⚫ enable:可读可写,写入"0"表示禁止PWM;写入"1"表示使能PWM。读取该文件获取PWM 当前是禁止还是使能状态。

echo 0 > enable #禁止PWM 输出
echo 1 > enable #使能PWM 输出

通常配置好PWM 之后,再使能PWM。
⚫ polarity:用于设置极性,可读可写,可写入的值如下:
“normal”:普通;
“inversed”:反转;

echo normal > polarity #默认极性
echo inversed > polarity #极性反转

很多SoC 的PWM 外设其硬件上并不支持极性配置,所以对应的驱动程序中并未实现这个接口,应用层自然也就无法通过polarity 属性文件对PWM 极性进行配置,ALPHA/Mini I.MX6U 开发板出厂系统便是如此!
⚫ period:用于配置PWM 周期,可读可写;写入一个字符串数字值,以ns(纳秒)为单位,譬如配置PWM 周期为10us(微秒):

echo 10000 > period #PWM 周期设置为10us(10 * 1000ns)

⚫ duty_cycle:用于配置PWM 的占空比,可读可写;写入一个字符串数字值,同样也是以ns 为单位,譬如:

echo 5000 > duty_cycle #PWM 占空比设置为5us

编写应用程序

通过上面的介绍,我们已经知道在应用层如何去使用PWM 外设了,本小节我们来编写一个简单的测试代码,来控制开发板上的PWM 外设,示例代码如下所示:
本例程源码对应的路径为:开发板光盘->11、Linux C 应用编程例程源码->24_pwm->pwm.c。

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

static char pwm_path[100];

static int pwm_config(const char *attr, const char *val)
{
    char file_path[100];
    int len;
    int fd;

    sprintf(file_path, "%s/%s", pwm_path, attr);
    if (0 > (fd = open(file_path, O_WRONLY))) {
        perror("open error");
        return fd;
    }

    len = strlen(val);
    if (len != write(fd, val, len)) {
        perror("write error");
        close(fd);
        return -1;
    }

    close(fd);  //关闭文件
    return 0;
}

int main(int argc, char *argv[])
{
    /* 校验传参 */
    if (4 != argc) {
        fprintf(stderr, "usage: %s <id> <period> <duty>\n",
                argv[0]);
        exit(-1);
    }

    /* 打印配置信息 */
    printf("PWM config: id<%s>, period<%s>, duty<%s>\n",
            argv[1], argv[2],
            argv[3]);

    /* 导出pwm */
    sprintf(pwm_path, "/sys/class/pwm/pwmchip%s/pwm0", argv[1]);

    if (access(pwm_path, F_OK)) {//如果pwm0目录不存在, 则导出

        char temp[100];
        int fd;

        sprintf(temp, "/sys/class/pwm/pwmchip%s/export", argv[1]);
        if (0 > (fd = open(temp, O_WRONLY))) {
            perror("open error");
            exit(-1);
        }

        if (1 != write(fd, "0", 1)) {//导出pwm
            perror("write error");
            close(fd);
            exit(-1);
        }

        close(fd);  //关闭文件
    }

    /* 配置PWM周期 */
    if (pwm_config("period", argv[2]))
        exit(-1);

    /* 配置占空比 */
    if (pwm_config("duty_cycle", argv[3]))
        exit(-1);

    /* 使能pwm */
    pwm_config("enable", "1");

    /* 退出程序 */
    exit(0);
}

main()函数中,首先对传参进行校验,执行该应用程序的时候需要用户传入3 个参数,分别是编号(0、
1、2、3 等,分别表示I.MX6U 的PWM1、PWM2、PWM3…)、周期(以ns 为单位)、PWM 占空比(以
ns 为单位)。譬如:

./testApp 0 500000 250000

接下来需要导出pwm,首先使用access()函数判断pwm0 目录是否存在,如果存在表示pwm 已经导出,如果不存在,则表示未导出,那么就需要通过export 文件将其导出。
导出成功之后,接着配置PWM 周期、占空比,最后使能PWM。
编译示例代码:
在这里插入图片描述

在开发板上测试

将上小节编译得到的可执行文件拷贝到开发板Linux 系统/home/root 目录下,如下所示:

在这里插入图片描述

前面提到了,开发板出厂系统没法使用PWM,如果大家想要测试PWM,可以对出厂系统的内核源码进行配置、需修改设备树,禁用LCD 和backlight 背光设备(status 属性设置为disabled 即可),修改完之后重新编译设备树,用编译得到的设备树镜像文件(dtb 文件)替换掉开发板启动文件中的dtb 文件。也可以参考《I.MX6U 嵌入式Linux 驱动开发指南》第七十三章内容,自行配置PWM。
这里笔者告诉大家一个简单地方法,不用重新编译设备树文件,直接把禁用LCD 和backlight 背光设备,将PWM1 腾出来给我们测试使用,直接操作呢?
首先我们需要重启开发板,进入到u-boot 命令行模式下,如下:
在这里插入图片描述
我们要做什么呢?其实就是去修改内核设备树文件,将LCD 和backlight 设备的status 属性修改为“disabled”,禁用这两个设备;怎么修改呢?u-boot 中提供了查看、修改设备树的命令,u-boot 启动时,会将内核设备树(dtb)拷贝到内存中,当拷贝到内存中之后呢,我们就可以去查看或修改设备树了,这里笔者直接把需要执行的命令贴出来,如下所示:
SD/eMMC 启动方式:

setenv disable_lcd 'fdt addr ${fdt_addr}; fdt set /backlight status disable; fdt set /soc/aips-bus@02100000/lcdif@021c8000 status disable'
下一条命令
setenv mmcboot 'echo Booting from mmc ...; run mmcargs; if test ${boot_fdt} = yes || test ${boot_fdt} = try; then if run loadfdt; then run disable_lcd; bootz ${loadaddr} - ${fdt_addr}; else if test ${boot_fdt} = try; then bootz; else echo WARN: Cannot load the DT; fi; fi; else bootz; fi;'

NAND 启动方式:

setenv disable_lcd 'fdt addr ${fdt_addr}; fdt set /backlight status disable; fdt set /soc/aips-bus@02100000/lcdif@021c8000 status disable'
下一条命令
setenv bootcmd 'nand read ${loadaddr} 0x620000 0x800000;nand read ${fdt_addr} ${fdt_offset} 0x20000; run disable_lcd; bootz ${loadaddr} - ${fdt_addr}'

拷贝时注意格式的问题,分为SD/eMMC 启动方式和NAND 启动方式;笔者测试用的开发板是eMMC方式启动的,在u-boot 命令行模式下执行如下命令:
在这里插入图片描述
执行完两条命令后,接着执行boot 命令启动开发板:

boot

在这里插入图片描述
Tips:这种修改方式只对本次启动生效,因为我们修改的是内存中的那份设备树文件,下一次重启开发板时将又恢复到未修改前的状态,请悉知!
系统启动之后,PWM1 就已经腾出来给我们测试使用了,此时LCD 被禁用了!
执行上小节编译得到的可执行文件:
在这里插入图片描述
本实验测试的是PWM1,开发板出厂系统已经将PWM1 输出绑定到了GPIO1_IO08 引脚(也就是LCD
背光引脚),该引脚已经通过开发板上的扩展口引出,如下所示:
在这里插入图片描述
Mini 开发板可以通过背面丝印标注的名称或原理图进行确认。
接下来使用示波器来检测GPIO1_IO08 引脚输出的PWM 波形,如下所示:
在这里插入图片描述
此时GPIO1_IO08 引脚输出了PWM 波形,其周期为500us(也就是500000ns),对应的频率为2KHz,占空比为50%,与我们配置的情况是一样的。

  • 5
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 5
    评论
PWM-GPIO驱动程序是Linux内核中的一个驱动模块,用于控制嵌入式系统中的GPIO引脚产生PWM信号。该驱动程序允许开发人员通过编程的方式来控制GPIO引脚的电平变化,从而产生不同占空比的PWM信号。 在Linux内核中,PWM-GPIO驱动程序通过向用户空间提供了相应的接口来实现PWM信号的控制。开发人员可以通过打开相应的设备节点,并使用相应的系统调用函数来设置PWM的频率、占空比等参数,从而实现对GPIO引脚的PWM信号的控制。 驱动程序的核心部分是一个PWM系统,它与GPIO子系统紧密集成。PWM系统负责管理PWM信号的生成和控制,而GPIO子系统负责管理GPIO引脚的配置和操作。PWM-GPIO驱动程序在这两个子系统之间起着桥梁的作用。 PWM-GPIO驱动程序的实现方式与硬件平台相关,每个平台可能有不同的具体实现。在驱动程序的初始化过程中,必须先配置GPIO引脚的功能为PWM模式,并将相应的寄存器映射到内核中,以便能够通过对寄存器的操作来控制GPIO引脚。驱动程序还需要初始化PWM系统,为每个GPIO引脚分配相应的PWM通道,并根据需求设置PWM的频率、占空比等参数。 通过PWM-GPIO驱动程序,开发人员可以方便地利用Linux内核的功能来实现对嵌入式系统中GPIO引脚产生PWM信号的控制。这为开发PWM驱动、控制舵机、LED等应用提供了便捷的方式。
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

行稳方能走远

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值