点击↑深色口袋物联,选择关注公众号,获取更多内容,不迷路

最近帮同事 Review 一款LVDS转eDp Bridge 芯片的驱动代码,发现一个很有意思的现象。同事为了“省事”,把网上能找到的结构体定义都塞进去了:
struct my_bridge {
struct drm_bridge base;
struct drm_connector connector; // 这个是为了抄 HDMI 驱动加的
struct drm_bridge *panel_bridge; // 这个是为了抄 Panel 驱动加的
struct drm_bridge *next_bridge; // 这个是为了抄其他 Bridge 驱动加的
// ...
};结果编译是过了,但跑起来一团糟:probe 时不知道该初始化谁,attach 时逻辑混乱,甚至出现了两个 Connector 互相打架的情况。
这其实是很多驱动开发者,尤其是刚接触 DRM 子系统,容易犯错的地方:看着内核代码里的结构体,定义的五花八门,不知道自己驱动,到底该定义哪些字段。
今天,我们就以 DRM Bridge 驱动中的“私有结构体封装”为主题,把这些看似复杂的组合,像剥洋葱一样,层层剥开,让你彻底看懂其中的设计门道。
问题背景:
我们在开发一个显示板卡驱动,硬件链路是:SoC (LVDS) -> 转换芯片 -> eDp 屏幕。
直觉写法:
同事觉得,转换芯片既然要输出图像,那肯定需要 connector;后面接着屏幕,那肯定需要 panel_bridge。于是他定义了包含所有字段的结构体,试图把所有角色都演了。
现象:
系统启动后,modetest 命令看到了两个 Connector,一个 connected,一个 disconnected。应用程序打开设备时,经常选错 Connector,导致黑屏。
核心问题:
DRM 的显示链路是有严格“角色分工”的。你的私有结构体,必须精准地反映你在链路中的位置和职责。“越俎代庖”,不仅浪费内存,更会破坏drm子系统对链路的拓扑管理。
要搞懂结构体怎么定义,先得搞懂 DRM 的链路拓扑。每一个字段,都代表了一个“坑位”。
常见的错误理解:
“结构体就是存数据的,多定义点字段备用总没错。”
原理剖析:
struct drm_bridge:这是你的本体。无论你是啥芯片,只要是个 Bridge,这个字段必须有,且通常放在结构体首部(方便 container_of 寻址)。struct drm_connector:这是终点站的标志。只有当你的输出接口是“可插拔”的(如 HDMI 母座、DP 接口),或者你是链路的终点时,才需要封装它。如果你只是个中间的协议转换器,不需要它。struct drm_bridge *next_bridge:这是指向下级的指针。当你不是终点,后面还连着其他 Bridge 时,需要保存这个指针,以便把数据流“接力”传下去。struct drm_bridge *panel_bridge:这是一个特殊的下级。当你后面挂的是一块固定的屏幕,内核为了统一管理,把屏幕封装成了一个 Bridge。这个字段通常指向这个“屏幕 Bridge”。对比图示:直觉 vs 正确
【直觉写法:大杂烩】
+-------------------------------------------------------+
| struct my_bridge { |
| struct drm_bridge base; |
| struct drm_connector connector; // 多余! |
| struct drm_bridge *next; // 冗余! |
| ... |
| } |
+-------------------------------------------------------+
问题:角色不清,逻辑冲突,注册了不该注册的资源。
【正确做法:按需定岗】
场景 A (中间节点):
SoC -> [Bridge A] -> [Bridge B] -> ...
结构体:bridge + *next_bridge
场景 B (终点接口):
SoC -> [Bridge] -> HDMI 母座
结构体:bridge + connector
场景 C (驱动面板):
SoC -> [Bridge] -> 固定屏幕
结构体:bridge + *panel_bridge
场景 D (复杂):
SoC -> [Bridge] -> HDMI 母座 (且内部还有逻辑子链路)
结构体:bridge + connector + *next_bridge理解了角色,我们就可以总结出私有结构体的“四大封装定式”。
关键设计要点(铁律):
struct drm_bridge 永远内嵌,这是入场券。struct drm_connector。*next_bridge 或 *panel_bridge 来保存引用。next_bridge 和 panel_bridge 通常是指针,因为它们指向内核其他地方分配的对象;而 connector 和 bridge 是你自己的对象,所以要内嵌,是变量。内核工作流程:
[Probe 阶段]
|
v
分析设备树 -> 判断硬件拓扑
|
+---> 后面是 HDMI 接口? -> 封装 connector
|
+---> 后面是另一个 Bridge? -> 定义 *next_bridge
|
+---> 后面是固定屏幕? -> 定义 *panel_bridge
|
v
[Attach 阶段]
|
v
如果定义了 connector -> drm_connector_init() (创建终点)
如果定义了 *next -> drm_bridge_attach() (连接下级)
如果定义了 *panel -> devm_drm_panel_bridge_add() (包装屏幕)下面我们通过代码实例,展示这几种封装形式的具体写法。
drm_bridge)场景:最简单的协议转换器,或者不需要管理下级设备的驱动(极少见,通常至少有个指针)。
/*
* 场景:一个极其简单的 Level Shifter(电平转换器)
* 特点:透明传输,无需关心后面是谁
*/
struct simple_shifter {
struct drm_bridge bridge; // 本体
struct gpio_desc *enable_gpio;
};
static const struct drm_bridge_funcs shifter_funcs = {
.enable = shifter_enable,
// ...
};drm_bridge + next_bridge)场景:显示链路中间的一环,比如 RGB 转 LVDS 芯片,后面可能还挂着其他 Bridge。
/*
* 场景:RGB to LVDS Converter
* 特点:需要把数据传给下一个 Bridge
*/
struct lvds_converter {
struct drm_bridge bridge; // 内嵌内核对象
struct drm_bridge *next_bridge; // 指向链路中的下一个 bridge
struct i2c_client *client;
};
// 辅助宏
#define to_lvds_converter(b) container_of(b, struct lvds_converter, bridge)
static int lvds_attach(struct drm_bridge *bridge,
enum drm_bridge_attach_flags flags)
{
struct lvds_converter *lvds = to_lvds_converter(bridge);
// 核心:调用内核函数挂载下一级,返回下一级 bridge 的指针
// 注意:这里传入 flags,如果下级需要创建 connector,由下级决定
lvds->next_bridge = drm_bridge_attach(bridge->encoder,
lvds->next_bridge,
bridge, flags);
if (IS_ERR(lvds->next_bridge))
return PTR_ERR(lvds->next_bridge);
return 0;
}为什么这样设计?
保存 next_bridge 指针是为了在 enable、mode_set 等回调中,通过 drm_bridge_chain_xxx 系列函数手动通知下级设备,或者仅仅是确认挂载成功。
drm_bridge + drm_connector)场景:HDMI 发送器,输出到母座,需要处理热插拔(HPD)和 EDID。
/*
* 场景:HDMI Transmitter
* 特点:链路终点,需要创建 Connector
*/
struct hdmi_transmitter {
struct drm_bridge bridge;
struct drm_connector connector; // 自己管理 Connector
struct i2c_client *client;
};
#define to_hdmi_transmitter(b) container_of(b, struct hdmi_transmitter, bridge)
static int hdmi_attach(struct drm_bridge *bridge,
enum drm_bridge_attach_flags flags)
{
struct hdmi_transmitter *hdmi = to_hdmi_transmitter(bridge);
int ret;
// 如果内核没有禁止创建 Connector (通常 HDMI 驱动必须创建)
if (!(flags & DRM_BRIDGE_ATTACH_NO_CONNECTOR)) {
// 1. 初始化 Connector
ret = drm_connector_init(bridge->dev, &hdmi->connector,
&hdmi_connector_funcs,
DRM_MODE_CONNECTOR_HDMIA);
if (ret)
return ret;
// 2. 将 Connector 挂在 Encoder 上,并关联当前的 Bridge
drm_connector_attach_encoder(&hdmi->connector, bridge->encoder);
}
return 0;
}
// Connector 的操作函数,用于 HPD 和 EDID
static int hdmi_get_modes(struct drm_connector *connector) {
// 读取 EDID ...
return 0;
}为什么这样设计?
HDMI 接口对应物理上的“接口”而非“屏”。用户空间需要通过 Connector 来获知“屏幕插上了吗?”、“支持什么分辨率?”。所以驱动必须封装 Connector。
drm_bridge + panel_bridge)场景:Bridge 后面直接挂了一块固定屏幕(Panel)。这是目前最主流的做法。
/*
* 场景:DPI to MIPI Bridge + 固定屏
* 特点:利用 panel_bridge 将 Panel 纳入 Bridge 链路
*/
struct dpi_to_mipi {
struct drm_bridge bridge;
struct drm_bridge *panel_bridge; // 用于包装 Panel
struct mipi_dsi_host *host;
};
static int dpi_to_mipi_probe(struct i2c_client *client)
{
struct dpi_to_mipi *ctx;
struct drm_panel *panel;
ctx = devm_kzalloc(...);
// 1. 获取设备树中引用的 Panel 节点
ret = drm_of_find_panel_or_bridge(client->dev.of_node, 1, 0, &panel, NULL);
if (ret)
return ret;
// 2. 关键:将 Panel 包装成一个 Bridge
// 这个 panel_bridge 内部实现了 Panel 的时序控制
ctx->panel_bridge = devm_drm_panel_bridge_add(&client->dev, panel);
if (IS_ERR(ctx->panel_bridge))
return PTR_ERR(ctx->panel_bridge);
// 注册自己
return devm_drm_bridge_add(&client->dev, &ctx->bridge);
}
static int dpi_to_mipi_attach(struct drm_bridge *bridge,
enum drm_bridge_attach_flags flags)
{
struct dpi_to_mipi *ctx = to_ctx(bridge);
// 3. 直接挂载 panel_bridge
// 因为 panel_bridge 是内核标准的 bridge,这里逻辑就统一了
return drm_bridge_attach(bridge->encoder, ctx->panel_bridge, bridge, flags);
}为什么这样设计?
这是目前 Linux DRM 最推荐的“最佳实践”。把 Panel 包装成 Bridge 后,整个显示链路变成了清一色的 Bridge 链表。驱动不需要关心 Panel 的上下电细节,只需要把数据流传给 panel_bridge 即可。
drm_bridge + drm_connector + next_bridge)场景:相对复杂的 Bridge 芯片。比如一个 HDMI 转 LVDS 芯片,它一方面需要一个 Connector 来接收 HDMI 输入(Input,这种叫 mux),或者它作为输出端创建 Connector,同时内部还有逻辑子链路需要管理。
更常见的例子:它是输出端(需要 Connector),同时它内部又集成了一个驱动逻辑,需要挂接下一级 Bridge。
/*
* 场景:智能 HDMI 转换器
* 特点:自己创建 Connector,同时后面还挂着下一级 Bridge
*/
struct smart_bridge {
struct drm_bridge bridge;
struct drm_connector connector; // 自己的 Connector
struct drm_bridge *next_bridge; // 下级 Bridge
};
static int smart_attach(struct drm_bridge *bridge,
enum drm_bridge_attach_flags flags)
{
struct smart_bridge *s = ...;
// 1. 创建 Connector (自己的职责)
if (!(flags & DRM_BRIDGE_ATTACH_NO_CONNECTOR)) {
drm_connector_init(..., &s->connector, ...);
}
// 2. 挂接下级 (转发职责)
// 注意:下级通常不需要创建 Connector,因为我们已经有了
s->next_bridge = drm_bridge_attach(bridge->encoder, s->next_bridge,
bridge, DRM_BRIDGE_ATTACH_NO_CONNECTOR);
return PTR_ERR_OR_ZERO(s->next_bridge);
}在实际开发中,结构体定义只是第一步,怎么用才是关键。
1. DRM_BRIDGE_ATTACH_NO_CONNECTOR 标志位陷阱
这是最容易踩的坑。
flags。如果上层传了这个标志,说明上层(如 DRM Core 或 Writeback Connector)想自己管理 Connector,你就别创建了。drm_bridge_attach 挂接 Panel Bridge 时,通常要手动加上这个标志,防止 Panel Bridge 内部再冒出一个 Connector 来。2. 生命周期管理
bridge 和 connector 是内嵌的,随私有结构体一起分配(devm_kzalloc),一起释放。next_bridge 和 panel_bridge 是借来的指针,指向内核其他地方的对象,不要尝试释放它们。3. 调试技巧
在 attach 函数里打印一下关键信息,能帮你理清拓扑:
DRM_DEV_INFO(dev, "Attach: self=%p, next=%p, flags=0x%x\n",
&ctx->bridge, ctx->next_bridge, flags);最后,我们用一张图把“结构体定义决策”串起来:
[开始编写 Bridge 驱动]
|
v
[硬件拓扑分析]
|
+---> 我是链路终点吗?(输出接口是 HDMI/DP 母座)
| |
| +--YES--> 结构体内嵌: drm_connector
| |
| +--NO---> 结构体不包含 connector
|
+---> 我后面有下级设备吗?
|
+--YES--> 下级是 Bridge? -> 定义 *next_bridge
| 下级是 Panel? -> 定义 *panel_bridge (并通过 API 获取)
|
+--NO---> 只需要 drm_bridge
|
v
[编写 Probe]
分配结构体 -> 初始化 bridge.funcs -> 处理 panel 指针
|
v
[编写 Attach]
根据 flags 决定是否 init connector
调用 drm_bridge_attach 连接下级Linux DRM Bridge 的私有结构体封装,本质上就是**“链路拓扑的 C 语言映射”**。
next_bridge 是为了把控制权传递下去,定义 connector 是为了终止链路并向上暴露接口。所以,drm驱动的私有结构体定义,是根据硬件拓扑图来得,也可以说根据设备树定义来的,多定义过少定义,都会影响整个drm链路。如果没有定义好,也许跑Qt应用程序可以,但Weston可能会奔溃!
如果对次感兴趣,关注我,分享更多调试过程!