Linux DRM那些事-平台驱动注册

2020-12-29  本文已影响0人  小田BSP

在RockPI 4A Debian系统的Linux 4.4内核DRM驱动中,有下面一段代码:

static const struct of_device_id rockchip_drm_dt_ids[] = {
    { .compatible = "rockchip,display-subsystem", },
    { /* sentinel */ },
};
MODULE_DEVICE_TABLE(of, rockchip_drm_dt_ids);

static struct platform_driver rockchip_drm_platform_driver = {
    .probe = rockchip_drm_platform_probe,
    .remove = rockchip_drm_platform_remove,
    .shutdown = rockchip_drm_platform_shutdown,
    .driver = {
        .name = "rockchip-drm",
        .of_match_table = rockchip_drm_dt_ids,
        .pm = &rockchip_drm_pm_ops,
    },
};
module_platform_driver(rockchip_drm_platform_driver);

module_platform_driver()在文件include\linux\platform_device.h中定义如下:

#define module_platform_driver(__platform_driver) \
    module_driver(__platform_driver, platform_driver_register, \
            platform_driver_unregister)

#define module_driver(__driver, __register, __unregister, ...) \
static int __init __driver##_init(void) \
{ \
    return __register(&(__driver) , ##__VA_ARGS__); \
} \
module_init(__driver##_init); \
static void __exit __driver##_exit(void) \
{ \
    __unregister(&(__driver) , ##__VA_ARGS__); \
} \
module_exit(__driver##_exit);

宏最终展开如下:

static int __init rockchip_drm_platform_driver_init(void)
{
    return platform_driver_register(&rockchip_drm_platform_driver);
}
module_init(rockchip_drm_platform_driver_init);
static void __exit rockchip_drm_platform_driver_exit(void)
{
    platform_driver_unregister(&rockchip_drm_platform_driver);
}
module_exit(rockchip_drm_platform_driver_exit);

小知识点:##:连接符,连接##前后的两个子串;#:将其后面的宏参数进行字符串化操作。

从上面代码可以看出,宏module_platform_driver()实现功能如下:

1.通过module_init和module_exit实现模块的加载和卸载。

2.通过platform_driver_register()和 platform_driver_unregister()实现平台设备驱动的注册和释放。

有了platform_driver,需要关注下platform_device,DRM模块的platform_devicedts中配置,配置如下:

    display_subsystem: display-subsystem {
        compatible = "rockchip,display-subsystem";
        ports = <&vopl_out>, <&vopb_out>;
        clocks = <&cru PLL_VPLL>, <&cru PLL_CPLL>;
        clock-names = "hdmi-tmds-pll", "default-vop-pll";
        devfreq = <&dmc>;
        status = "disabled";
    };

platform_driverplatform_device之间通过compatible属性进行匹配。若两者相同,即可运行platform_driverprobe()函数。

注:后续会陆续介绍platform总线和dts

注:本文仅在简书、OSCHINA和今日头条发布过,转载请标注原作者和链接。

上一篇下一篇

猜你喜欢

热点阅读