标准linux4.4--驱动开发(一)ADC驱动编写
RK3308-驱动开发(一)ADC驱动编写简介DTS配置在驱动文件中关联DTS的配置驱动说明FAQs简介Firefly-RK3308开发板上的 AD 接口有两种TS-ADC(Temperature Sensor):支持两通道,时钟频率必须低于800KHZSAR-ADC(Successive Approximation Register):支持六通道单端10位的SAR-ADC,时钟频率必须小于13M
标准linux4.4--驱动开发(一)ADC驱动编写
简介
Firefly-RK3308开发板上的 AD 接口有两种
TS-ADC(Temperature Sensor):支持两通道,时钟频率必须低于800KHZ
SAR-ADC(Successive Approximation Register):支持六通道单端10位的SAR-ADC,时钟频率必须小于13MHZ。(常用)
内核采用工业 I/O 子系统来控制 ADC,该子系统主要为 AD 转换或者 DA 转换的传感器设计。 下面以 SAR-ADC 为例子,介绍 ADC 的基本配置方法。
DTS配置
SAR-ADC 的 DTS 节点在 kernel/arch/arm64/boot/dts/rockchip/rk3308.dtsi 文件中定义,如下所示:
saradc: saradc@ff1e0000 {
compatible = "rockchip,rk3308-saradc", "rockchip,rk3399-saradc";
reg = <0x0 0xff1e0000 0x0 0x100>;
interrupts = <GIC_SPI 37 IRQ_TYPE_LEVEL_HIGH>;
#io-channel-cells = <1>;
clocks = <&cru SCLK_SARADC>, <&cru PCLK_SARADC>;
clock-names = "saradc", "apb_pclk";
resets = <&cru SRST_SARADC_P>;
reset-names = "saradc-apb";
status = "disabled";
};
想要使用CPU上的ADC资源,必须要添加DTS配置及设备树,所以用户首先需在 DTS 文件中添加 ADC 的资源描述:
这里我举2个例子
(1)adc模拟的按键驱动(RK适配的)
(2)萤火虫firefly适配的adc使用demo
接下来看下他们分别的设备树如何描述
(1)adc模拟的按键(简单逻辑根据adc检测的电压的不同,转换成数字信号范围去检测按了哪个按键)
adc-keys {
compatible = "adc-keys";
io-channels = <&saradc 1>;//这里申请的是 SARADC 通道1。
io-channel-names = "buttons";
poll-interval = <100>;
keyup-threshold-microvolt = <1800000>;
esc-key {
linux,code = <KEY_MICMUTE>;
label = "micmute";
press-threshold-microvolt = <1130000>;
};
home-key {
linux,code = <KEY_MODE>;
label = "mode";
press-threshold-microvolt = <901000>;
};
menu-key {
linux,code = <KEY_PLAY>;
label = "play";
press-threshold-microvolt = <624000>;
};
vol-down-key {
linux,code = <KEY_VOLUMEDOWN>;
label = "volume down";
press-threshold-microvolt = <300000>;
};
vol-up-key {
linux,code = <KEY_VOLUMEUP>;
label = "volume up";
press-threshold-microvolt = <18000>;
};
};
(2)adc-demo
adc_demo: adc_demo{
status = "disabled";
compatible = "firefly,rk3399-adc";
io-channels = <&saradc 3>;//这里申请的是 SARADC 通道3。
};
在驱动文件中关联DTS的配置
用户驱动可参考 RK adc Key驱动 :这是一个侦测 模拟出的5个按键状态的驱动。首先在驱动文件中定义 of_device_id 结构体数组:
static const struct of_device_id adc_keys_of_match[] = {
{ .compatible = "adc-keys", },
{ }
};
然后将该结构体数组填充到要使用 ADC 的 platform_driver 中:
static struct platform_driver __refdata adc_keys_driver = {
.driver = {
.name = "adc_keys",
.of_match_table = of_match_ptr(adc_keys_of_match),
},
.probe = adc_keys_probe,
};
module_platform_driver(adc_keys_driver);
接着在 probe函数 中对 DTS 所添加的资源进行解析:
简单逻辑是:获取道设备树描述的节点,遍历节点(有多少个节点都会读取到),在获取节点中相应的属性,分别保存起来。
例句相关部分驱动:(具体可以下载看)
ADC相关函数:
st = devm_kzalloc(dev, sizeof(*st), GFP_KERNEL);//申请内核空间内核
if (!st)
return -ENOMEM;
st->channel = devm_iio_channel_get(dev, "buttons");//获取设备IIO通道 与str匹配
if (IS_ERR(st->channel))
return PTR_ERR(st->channel);
if (!st->channel->indio_dev)
return -ENXIO;
error = iio_get_channel_type(st->channel, &type);//获取io通道类型
if (error < 0)
return error;
if (type != IIO_VOLTAGE) {
dev_err(dev, "Incompatible channel type %d\n", type);
return -EINVAL;
}
读取设备树中的属性
device_for_each_child_node(dev, child) {//循环去读子节点按键
if (fwnode_property_read_u32(child, "press-threshold-microvolt",//读取dtsi中子节点中press-threshold-microvolt属性
&map[i].voltage)) {
dev_err(dev, "Key with invalid or missing voltage\n");
fwnode_handle_put(child);
return -EINVAL;
}
map[i].voltage /= 1000;
if (fwnode_property_read_u32(child, "linux,code",//读取dtsi中子节点中linux,code属性
&map[i].keycode)) {
dev_err(dev, "Key with invalid or missing linux,code\n");
fwnode_handle_put(child);
return -EINVAL;
}
i++;
}
驱动说明
st->channel = devm_iio_channel_get(dev, "buttons");//获取设备IIO通道 与str匹配
iio_get_channel_type(st->channel, &type);//获取io通道类型
iio_read_channel_processed(st->channel, &value);//读ADC通道的值 存到value中
ADC驱动用到的函数比较简单。
FAQs
为何按上面的步骤申请 SARADC,会出现申请报错的情况?
驱动需要获取ADC通道来使用时,需要对驱动的加载时间进行控制,必须要在saradc初始化之后。saradc是使用module_platform_driver()进行平台设备驱动注册,最终调用的是module_init()。所以用户的驱动加载函数只需使用比module_init()优先级低的,例如:late_initcall(),就能保证驱动的加载的时间比saradc初始化时间晚,可避免出错。
更多推荐
所有评论(0)