xref: /linux/Documentation/translations/zh_CN/driver-api/phy/phy.rst (revision c2a96b7f187fb6a455836d4a6e113947ff11de97)
1.. SPDX-License-Identifier: GPL-2.0
2.. include:: ../../disclaimer-zh_CN.rst
3
4:Original: Documentation/driver-api/phy/phy.rst
5
6:翻译:
7
8 司延腾 Yanteng Si <siyanteng@loongson.cn>
9
10=========
11PHY子系统
12=========
13
14:作者: Kishon Vijay Abraham I <kishon@ti.com>
15
16本文档解释了 PHY 的通用框架和提供的API,以及使用方法。
17
18简介
19====
20
21*PHY* 是物理层的缩写,它被用来把设备连接到一个物理媒介,例如USB控制器
22有一个提供序列化、反序列化、编码、解码和负责获取所需的数据传输速率的 PHY。
23注意,有些USB控制器内嵌了 PHY 的功能,其它的则使用了一个外置的PHY,此外
24使用 PHY 的设备还有无线网、以太网、SATA等(控制器)。
25
26创建这个框架的目的是将遍布 Linux 内核的 PHY 驱动程序融入到 drivers/phy27以增加代码的可复用性,进而提高代码的可维护性。
28
29该框架仅适用于使用外部 PHY(PHY 功能未嵌入控制器内)的设备。
30
31注册/注销PHY provider
32=====================
33
34PHY provider是指实现一个或多个 PHY 实例的实体。对于 PHY provider 仅
35实现单个 PHY 实例的简单情况,框架在 of_phy_simple_xlate 中提供其自己
36的 of_xlate 实现。如果 PHY provider 实现多个实例,则应提供其自己的
37of_xlate 实现。of_xlate 仅用于 dt 启动情况。
38
39::
40
41	#define of_phy_provider_register(dev, xlate)    \
42		__of_phy_provider_register((dev), NULL, THIS_MODULE, (xlate))
43
44	#define devm_of_phy_provider_register(dev, xlate)       \
45		__devm_of_phy_provider_register((dev), NULL, THIS_MODULE,
46						(xlate))
47
48of_phy_provider_register 和 devm_of_phy_provider_register 宏
49可用于注册 phy_provider,它以 device 和 of_xlate 作为参数。对于 dt
50启动情况,所有 PHY provider 都应使用上述两个宏之一来注册 PHY provider。
51
52与 PHY provider 关联的设备树节点通常包含一组子节点,每个子节点代表一个
53PHY。某些绑定可能会为了上下文和可扩展性将子节点嵌套在特别的层级中,在这种
54情况下,可以使用低级别的 of_phy_provider_register_full() 和
55devm_of_phy_provider_register_full() 宏来覆盖包含子节点的节点。
56
57::
58
59	#define of_phy_provider_register_full(dev, children, xlate) \
60		__of_phy_provider_register(dev, children, THIS_MODULE, xlate)
61
62	#define devm_of_phy_provider_register_full(dev, children, xlate) \
63		__devm_of_phy_provider_register_full(dev, children,
64						     THIS_MODULE, xlate)
65
66	void devm_of_phy_provider_unregister(struct device *dev,
67		struct phy_provider *phy_provider);
68	void of_phy_provider_unregister(struct phy_provider *phy_provider);
69
70devm_of_phy_provider_unregister 和 of_phy_provider_unregister
71可以被用来注销PHY.
72
73创建PHY
74=======
75
76PHY 驱动程序应创建 PHY,以便其他外围(芯片)控制器能够使用它。PHY 框架
77提供了 2 个 API 来创建 PHY。
78
79::
80
81	struct phy *phy_create(struct device *dev, struct device_node *node,
82			       const struct phy_ops *ops);
83	struct phy *devm_phy_create(struct device *dev,
84				    struct device_node *node,
85				    const struct phy_ops *ops);
86
87PHY 驱动程序可以使用上述两个 API 之一,通过传递设备指针和 phy_ops
88来创建 PHY。
89
90phy_ops 是一组用于执行 PHY 操作(例如 init、exit、power_on 和
91power_off)的函数指针。
92
93在 phy_ops 中,PHY provider驱动程序在创建 PHY 后使用 phy_set_drvdata()
94设置私有数据,使用 phy_get_drvdata() 获取私有数据。
95
96获取对 PHY 的引用
97=================
98
99控制器必须先获取对 PHY 的引用,然后才能使用 PHY。此框架提供以下 API
100来获取对 PHY 的引用。
101
102::
103
104	struct phy *phy_get(struct device *dev, const char *string);
105	struct phy *devm_phy_get(struct device *dev, const char *string);
106	struct phy *devm_phy_optional_get(struct device *dev,
107					  const char *string);
108	struct phy *devm_of_phy_get(struct device *dev, struct device_node *np,
109				    const char *con_id);
110	struct phy *devm_of_phy_optional_get(struct device *dev,
111					     struct device_node *np,
112					     const char *con_id);
113	struct phy *devm_of_phy_get_by_index(struct device *dev,
114					     struct device_node *np,
115					     int index);
116
117phy_get、devm_phy_get 和 devm_phy_optional_get 可用于在 dt
118启动的情况下获取 PHY,字符串参数应包含 dt 数据中给出的 phy 名称,在
119非 dt 启动的情况下,它应包含 PHY 的标签。两个 devm_phy_get 在成功
120获取 PHY 后使用 devres 将设备与 PHY 关联。在驱动程序分离时,将在
121devres 数据上调用 release 函数并释放 devres 数据。当 phy 是可选
122的时,应使用 _optional_get 变体。这些函数永远不会返回 -ENODEV,而
123是在找不到 phy 时返回 NULL。一些通用驱动程序(例如 ehci)可能使用
124多个 phy。在这种情况下,devm_of_phy_get 或 devm_of_phy_get_by_index
125用于根据名称或索引获取 phy 引用。
126
127需要注意的是,NULL 是有效的 phy 引用。NULL phy 上的所有 phy 使用
128者调用都将成为 NOP。也就是说释放调用,当应用于 NULL phy 时,release
129调用、phy_init()/phy_exit() 调用、phy_power_on()/phy_power_off()
130调用都是 NOP。NULL phy 在处理可选的 phy 设备中很有用。
131
132API的调用顺序
133=============
134
135通常,调用顺序应该是::
136
137    [devm_][of_]phy_get()
138    phy_init()
139    phy_power_on()
140    [phy_set_mode[_ext]()]
141    ...
142    phy_power_off()
143    phy_exit()
144    [[of_]phy_put()]
145
146一些PHY驱动可能没有实现 :c:func:`phy_init` 或 :c:func:`phy_power_on`,
147但是控制器应该总是调用这些函数以兼容其它PHY,有些PHY可能要求
148:c:func:`phy_set_mode <phy_set_mode_ext>` 而其他 PHY 可能使用
149默认模式(通常通过设备树或其他固件配置)。出于兼容性考虑,如果您知道将
150使用哪种模式,则应始终调用此函数。通常,应在 :c:func:`phy_power_on`
151之后调用此函数,尽管某些 PHY 驱动程序可能随时允许调用它。
152
153释放对 PHY 的引用
154=================
155
156当控制器不再需要 PHY 时,它必须使用上一节中提到的 API 释放对已获得
157的 PHY 的引用。PHY 框架提供了 2 个 API 来释放对 PHY 的引用。
158
159::
160
161	void phy_put(struct phy *phy);
162	void devm_phy_put(struct device *dev, struct phy *phy);
163
164这两个 API 都用于释放对 PHY 的引用,并且 devm_phy_put 会销毁与此
165PHY 关联的设备资源。
166
167销毁 PHY
168========
169
170当创建 PHY 的驱动程序被卸载时,它应该使用以下 2 个 API 之一销毁其创
171建的 PHY::
172
173	void phy_destroy(struct phy *phy);
174	void devm_phy_destroy(struct device *dev, struct phy *phy);
175
176这两个 API 都会销毁 PHY,并且 devm_phy_destroy 会销毁与此 PHY 关
177联的 devres。
178
179PM Runtime
180==========
181
182这个子系统启用了pm runtime。 所以,在创建PHY 时,将调用此子系统创建的
183phy 设备的 pm_runtime_enable 函数,在销毁 PHY 时,将调用
184pm_runtime_disable。请注意,此子系统创建的 phy 设备将是调用 phy_create
185(PHY provider 设备)的设备的子设备。
186
187因此,由于父子关系,此子系统创建的 phy_device 的 pm_runtime_get_sync
188调用 PHY provider 设备的 pm_runtime_get_sync。还应注意,
189phy_power_on 和 phy_power_off 分别执行 phy_pm_runtime_get_sync 和
190phy_pm_runtime_put。有导出的 API,如 phy_pm_runtime_get、
191phy_pm_runtime_get_sync、phy_pm_runtime_put、phy_pm_runtime_put_sync、
192phy_pm_runtime_allow 和 phy_pm_runtime_forbid,用于执行 PM 操作。
193
194PHY映射
195=======
196
197为了在没有 DeviceTree 帮助的情况下获取对 PHY 的引用,框架提供了可与
198clkdev 进行比较的查找,允许将 clk 结构体绑定到设备。当 struct phy 的
199句柄已存在时,可以在运行时进行查找。
200
201该框架提供以下 API 用于注册和注销查找::
202
203	int phy_create_lookup(struct phy *phy, const char *con_id,
204			      const char *dev_id);
205	void phy_remove_lookup(struct phy *phy, const char *con_id,
206			       const char *dev_id);
207
208DeviceTree绑定
209==============
210
211PHY dt 绑定的文档可以在以下位置找到 @
212Documentation/devicetree/bindings/phy/phy-bindings.txt
213