小知识:详解linux pwm驱动编写

pwm方波可以用来控制很多的设备,比如它可以被用来控制电机。简单来说,就是单位时间内的方波越多,那么电机的转速就会越快;反之就越慢。通过这个特性,soc就可以轻松地利用pwm对外设进行自动控制。所以,今天的主题就是pwm驱动。

1、驱动目录

?
1
drivers/pwm

2、查看对应目录下的Kconfig

?
1
2
3
4
5
6
7
8
config PWM_SAMSUNG
tristate “Samsung PWM support”
depends on PLAT_SAMSUNG || ARCH_EXYNOS
help
Generic PWM framework driver for Samsung.
To compile this driver as a module, choose M here: the module
will be called pwm-samsung.

3、确认PWM_SAMSUNG只依赖于自己之外,继续看Makefile

?
1
2
3
obj-$(CONFIG_PWM) += core.o
obj-$(CONFIG_PWM_SAMSUNG) += pwm-samsung.o

4、根据Makefile查阅pwm-samsung.c文件,结构比较清楚

?
1
2
3
4
5
6
7
8
9
10
static struct platform_driver pwm_samsung_driver = {
.driver = {
.name = “samsung-pwm”,
.pm = &pwm_samsung_pm_ops,
.of_match_table = of_match_ptr(samsung_pwm_matches),
},
.probe = pwm_samsung_probe,
.remove = pwm_samsung_remove,
};
module_platform_driver(pwm_samsung_driver);

5、soc设备大多数是platform设备,继续寻找probe函数中的有用信息

?
1
2
3
4
5
6
ret = pwmchip_add(&chip->chip);
if (ret < 0) {
dev_err(dev, “failed to register PWM chip\n”);
clk_disable_unprepare(chip->base_clk);
return ret;
}

6、找到注册函数后,接续看看函数接口点在什么地方

?
1
2
3
4
5
6
7
8
9
static const struct pwm_ops pwm_samsung_ops = {
.request = pwm_samsung_request,
.free = pwm_samsung_free,
.enable = pwm_samsung_enable,
.disable = pwm_samsung_disable,
.config = pwm_samsung_config,
.set_polarity = pwm_samsung_set_polarity,
.owner = THIS_MODULE,
};

7、pwm设备没有中断函数,一般是立马生效,除此之外,代码中还是有设备树的内容,可以看看

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
static const struct of_device_id samsung_pwm_matches[] = {
{ .compatible = “samsung,s3c2410-pwm”, .data = &s3c24xx_variant },
{ .compatible = “samsung,s3c6400-pwm”, .data = &s3c64xx_variant },
{ .compatible = “samsung,s5p6440-pwm”, .data = &s5p64x0_variant },
{ .compatible = “samsung,s5pc100-pwm”, .data = &s5pc100_variant },
{ .compatible = “samsung,exynos4210-pwm”, .data = &s5p64x0_variant },
{},
};
MODULE_DEVICE_TABLE(of, samsung_pwm_matches);
static int pwm_samsung_parse_dt(struct samsung_pwm_chip *chip)
{
struct device_node *np = chip->chip.dev->of_node;
const struct of_device_id *match;
struct property *prop;
const __be32 *cur;
u32 val;
match = of_match_node(samsung_pwm_matches, np);
if (!match)
return -ENODEV;
memcpy(&chip->variant, match->data, sizeof(chip->variant));
of_property_for_each_u32(np, “samsung,pwm-outputs”, prop, cur, val) {
if (val >= SAMSUNG_PWM_NUM) {
dev_err(chip->chip.dev,
“%s: invalid channel index in samsung,pwm-outputs property\n”,
__func__);
continue;
}
chip->variant.output_mask |= BIT(val);
}
return 0;
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。

原文链接:https://blog.csdn.net/feixiaoxing/article/details/79889240

声明: 猿站网有关资源均来自网络搜集与网友提供,任何涉及商业盈利目的的均不得使用,否则产生的一切后果将由您自己承担! 本平台资源仅供个人学习交流、测试使用 所有内容请在下载后24小时内删除,制止非法恶意传播,不对任何下载或转载者造成的危害负任何法律责任!也请大家支持、购置正版! 。本站一律禁止以任何方式发布或转载任何违法的相关信息访客发现请向站长举报,会员发帖仅代表会员个人观点,并不代表本站赞同其观点和对其真实性负责。本网站的资源部分来源于网络,如有侵权烦请发送邮件至:2697268773@qq.com进行处理。
建站知识

小知识:linux 触摸屏驱动编写

2023-3-26 5:39:44

建站知识

小知识:Linux常用命令mkdir详解

2023-3-26 5:56:16

0 条回复 A文章作者 M管理员
    暂无讨论,说说你的看法吧
个人中心
购物车
优惠劵
今日签到
有新私信 私信列表
搜索