1091876ccSKishon Vijay Abraham I // SPDX-License-Identifier: GPL-2.0
27adaf921SRandy Dunlap /*
3091876ccSKishon Vijay Abraham I * Wrapper driver for SERDES used in J721E
4091876ccSKishon Vijay Abraham I *
5091876ccSKishon Vijay Abraham I * Copyright (C) 2019 Texas Instruments Incorporated - http://www.ti.com/
6091876ccSKishon Vijay Abraham I * Author: Kishon Vijay Abraham I <kishon@ti.com>
7091876ccSKishon Vijay Abraham I */
8091876ccSKishon Vijay Abraham I
9091876ccSKishon Vijay Abraham I #include <dt-bindings/phy/phy.h>
10040cbe76SKishon Vijay Abraham I #include <dt-bindings/phy/phy-ti.h>
1176b45387SShixin Liu #include <linux/slab.h>
12091876ccSKishon Vijay Abraham I #include <linux/clk.h>
13091876ccSKishon Vijay Abraham I #include <linux/clk-provider.h>
14c9f9eba0SRoger Quadros #include <linux/gpio.h>
15c9f9eba0SRoger Quadros #include <linux/gpio/consumer.h>
16091876ccSKishon Vijay Abraham I #include <linux/io.h>
17091876ccSKishon Vijay Abraham I #include <linux/module.h>
18edd473d4SRoger Quadros #include <linux/mfd/syscon.h>
19091876ccSKishon Vijay Abraham I #include <linux/mux/consumer.h>
20091876ccSKishon Vijay Abraham I #include <linux/of_address.h>
21091876ccSKishon Vijay Abraham I #include <linux/of_platform.h>
22091876ccSKishon Vijay Abraham I #include <linux/platform_device.h>
23091876ccSKishon Vijay Abraham I #include <linux/pm_runtime.h>
24091876ccSKishon Vijay Abraham I #include <linux/regmap.h>
25091876ccSKishon Vijay Abraham I #include <linux/reset-controller.h>
26091876ccSKishon Vijay Abraham I
2786d11e22SRoger Quadros #define REF_CLK_19_2MHZ 19200000
2886d11e22SRoger Quadros #define REF_CLK_25MHZ 25000000
2986d11e22SRoger Quadros #define REF_CLK_100MHZ 100000000
3086d11e22SRoger Quadros #define REF_CLK_156_25MHZ 156250000
3186d11e22SRoger Quadros
32edd473d4SRoger Quadros /* SCM offsets */
33edd473d4SRoger Quadros #define SERDES_SUP_CTRL 0x4400
34edd473d4SRoger Quadros
35edd473d4SRoger Quadros /* SERDES offsets */
36091876ccSKishon Vijay Abraham I #define WIZ_SERDES_CTRL 0x404
37091876ccSKishon Vijay Abraham I #define WIZ_SERDES_TOP_CTRL 0x408
38091876ccSKishon Vijay Abraham I #define WIZ_SERDES_RST 0x40c
39c9f9eba0SRoger Quadros #define WIZ_SERDES_TYPEC 0x410
40091876ccSKishon Vijay Abraham I #define WIZ_LANECTL(n) (0x480 + (0x40 * (n)))
4148820990SKishon Vijay Abraham I #define WIZ_LANEDIV(n) (0x484 + (0x40 * (n)))
42091876ccSKishon Vijay Abraham I
43040cbe76SKishon Vijay Abraham I #define WIZ_MAX_INPUT_CLOCKS 4
44040cbe76SKishon Vijay Abraham I /* To include mux clocks, divider clocks and gate clocks */
45040cbe76SKishon Vijay Abraham I #define WIZ_MAX_OUTPUT_CLOCKS 32
46040cbe76SKishon Vijay Abraham I
47091876ccSKishon Vijay Abraham I #define WIZ_MAX_LANES 4
48091876ccSKishon Vijay Abraham I #define WIZ_MUX_NUM_CLOCKS 3
49091876ccSKishon Vijay Abraham I #define WIZ_DIV_NUM_CLOCKS_16G 2
50091876ccSKishon Vijay Abraham I #define WIZ_DIV_NUM_CLOCKS_10G 1
51091876ccSKishon Vijay Abraham I
52c9f9eba0SRoger Quadros #define WIZ_SERDES_TYPEC_LN10_SWAP BIT(30)
53c9f9eba0SRoger Quadros
54091876ccSKishon Vijay Abraham I enum wiz_lane_standard_mode {
55091876ccSKishon Vijay Abraham I LANE_MODE_GEN1,
56091876ccSKishon Vijay Abraham I LANE_MODE_GEN2,
57091876ccSKishon Vijay Abraham I LANE_MODE_GEN3,
58091876ccSKishon Vijay Abraham I LANE_MODE_GEN4,
59091876ccSKishon Vijay Abraham I };
60091876ccSKishon Vijay Abraham I
61494de1ddSSinthu Raja /*
62494de1ddSSinthu Raja * List of master lanes used for lane swapping
63494de1ddSSinthu Raja */
64494de1ddSSinthu Raja enum wiz_typec_master_lane {
65494de1ddSSinthu Raja LANE0 = 0,
66494de1ddSSinthu Raja LANE2 = 2,
67494de1ddSSinthu Raja };
68494de1ddSSinthu Raja
69091876ccSKishon Vijay Abraham I enum wiz_refclk_mux_sel {
70091876ccSKishon Vijay Abraham I PLL0_REFCLK,
71091876ccSKishon Vijay Abraham I PLL1_REFCLK,
72091876ccSKishon Vijay Abraham I REFCLK_DIG,
73091876ccSKishon Vijay Abraham I };
74091876ccSKishon Vijay Abraham I
75091876ccSKishon Vijay Abraham I enum wiz_refclk_div_sel {
76091876ccSKishon Vijay Abraham I CMN_REFCLK_DIG_DIV,
77091876ccSKishon Vijay Abraham I CMN_REFCLK1_DIG_DIV,
78091876ccSKishon Vijay Abraham I };
79091876ccSKishon Vijay Abraham I
80040cbe76SKishon Vijay Abraham I enum wiz_clock_input {
81040cbe76SKishon Vijay Abraham I WIZ_CORE_REFCLK,
82040cbe76SKishon Vijay Abraham I WIZ_EXT_REFCLK,
83040cbe76SKishon Vijay Abraham I WIZ_CORE_REFCLK1,
84040cbe76SKishon Vijay Abraham I WIZ_EXT_REFCLK1,
85040cbe76SKishon Vijay Abraham I };
86040cbe76SKishon Vijay Abraham I
87091876ccSKishon Vijay Abraham I static const struct reg_field por_en = REG_FIELD(WIZ_SERDES_CTRL, 31, 31);
88091876ccSKishon Vijay Abraham I static const struct reg_field phy_reset_n = REG_FIELD(WIZ_SERDES_RST, 31, 31);
899e405f87SKishon Vijay Abraham I static const struct reg_field phy_en_refclk = REG_FIELD(WIZ_SERDES_RST, 30, 30);
90091876ccSKishon Vijay Abraham I static const struct reg_field pll1_refclk_mux_sel =
91091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_SERDES_RST, 29, 29);
92e27ecef8SMatt Ranostay static const struct reg_field pll1_refclk_mux_sel_2 =
93e27ecef8SMatt Ranostay REG_FIELD(WIZ_SERDES_RST, 22, 23);
94091876ccSKishon Vijay Abraham I static const struct reg_field pll0_refclk_mux_sel =
95091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_SERDES_RST, 28, 28);
96e27ecef8SMatt Ranostay static const struct reg_field pll0_refclk_mux_sel_2 =
97e27ecef8SMatt Ranostay REG_FIELD(WIZ_SERDES_RST, 28, 29);
98091876ccSKishon Vijay Abraham I static const struct reg_field refclk_dig_sel_16g =
99091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_SERDES_RST, 24, 25);
100091876ccSKishon Vijay Abraham I static const struct reg_field refclk_dig_sel_10g =
101091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_SERDES_RST, 24, 24);
102091876ccSKishon Vijay Abraham I static const struct reg_field pma_cmn_refclk_int_mode =
103091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_SERDES_TOP_CTRL, 28, 29);
104e27ecef8SMatt Ranostay static const struct reg_field pma_cmn_refclk1_int_mode =
105e27ecef8SMatt Ranostay REG_FIELD(WIZ_SERDES_TOP_CTRL, 20, 21);
106091876ccSKishon Vijay Abraham I static const struct reg_field pma_cmn_refclk_mode =
107091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_SERDES_TOP_CTRL, 30, 31);
108091876ccSKishon Vijay Abraham I static const struct reg_field pma_cmn_refclk_dig_div =
109091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_SERDES_TOP_CTRL, 26, 27);
110091876ccSKishon Vijay Abraham I static const struct reg_field pma_cmn_refclk1_dig_div =
111091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_SERDES_TOP_CTRL, 24, 25);
112edd473d4SRoger Quadros
113edd473d4SRoger Quadros static const struct reg_field sup_pll0_refclk_mux_sel =
114edd473d4SRoger Quadros REG_FIELD(SERDES_SUP_CTRL, 0, 1);
115edd473d4SRoger Quadros static const struct reg_field sup_pll1_refclk_mux_sel =
116edd473d4SRoger Quadros REG_FIELD(SERDES_SUP_CTRL, 2, 3);
117edd473d4SRoger Quadros static const struct reg_field sup_pma_cmn_refclk1_int_mode =
118edd473d4SRoger Quadros REG_FIELD(SERDES_SUP_CTRL, 4, 5);
119edd473d4SRoger Quadros static const struct reg_field sup_refclk_dig_sel_10g =
120edd473d4SRoger Quadros REG_FIELD(SERDES_SUP_CTRL, 6, 7);
121edd473d4SRoger Quadros static const struct reg_field sup_legacy_clk_override =
122edd473d4SRoger Quadros REG_FIELD(SERDES_SUP_CTRL, 8, 8);
123edd473d4SRoger Quadros
124040cbe76SKishon Vijay Abraham I static const char * const output_clk_names[] = {
125040cbe76SKishon Vijay Abraham I [TI_WIZ_PLL0_REFCLK] = "pll0-refclk",
126040cbe76SKishon Vijay Abraham I [TI_WIZ_PLL1_REFCLK] = "pll1-refclk",
127040cbe76SKishon Vijay Abraham I [TI_WIZ_REFCLK_DIG] = "refclk-dig",
1289e405f87SKishon Vijay Abraham I [TI_WIZ_PHY_EN_REFCLK] = "phy-en-refclk",
129040cbe76SKishon Vijay Abraham I };
130091876ccSKishon Vijay Abraham I
131091876ccSKishon Vijay Abraham I static const struct reg_field p_enable[WIZ_MAX_LANES] = {
132091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_LANECTL(0), 30, 31),
133091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_LANECTL(1), 30, 31),
134091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_LANECTL(2), 30, 31),
135091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_LANECTL(3), 30, 31),
136091876ccSKishon Vijay Abraham I };
137091876ccSKishon Vijay Abraham I
1387ae14cf5SJyri Sarha enum p_enable { P_ENABLE = 2, P_ENABLE_FORCE = 1, P_ENABLE_DISABLE = 0 };
1397ae14cf5SJyri Sarha
140091876ccSKishon Vijay Abraham I static const struct reg_field p_align[WIZ_MAX_LANES] = {
141091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_LANECTL(0), 29, 29),
142091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_LANECTL(1), 29, 29),
143091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_LANECTL(2), 29, 29),
144091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_LANECTL(3), 29, 29),
145091876ccSKishon Vijay Abraham I };
146091876ccSKishon Vijay Abraham I
147091876ccSKishon Vijay Abraham I static const struct reg_field p_raw_auto_start[WIZ_MAX_LANES] = {
148091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_LANECTL(0), 28, 28),
149091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_LANECTL(1), 28, 28),
150091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_LANECTL(2), 28, 28),
151091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_LANECTL(3), 28, 28),
152091876ccSKishon Vijay Abraham I };
153091876ccSKishon Vijay Abraham I
154091876ccSKishon Vijay Abraham I static const struct reg_field p_standard_mode[WIZ_MAX_LANES] = {
155091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_LANECTL(0), 24, 25),
156091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_LANECTL(1), 24, 25),
157091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_LANECTL(2), 24, 25),
158091876ccSKishon Vijay Abraham I REG_FIELD(WIZ_LANECTL(3), 24, 25),
159091876ccSKishon Vijay Abraham I };
160091876ccSKishon Vijay Abraham I
1616ecac2f8SKishon Vijay Abraham I static const struct reg_field p0_fullrt_div[WIZ_MAX_LANES] = {
1626ecac2f8SKishon Vijay Abraham I REG_FIELD(WIZ_LANECTL(0), 22, 23),
1636ecac2f8SKishon Vijay Abraham I REG_FIELD(WIZ_LANECTL(1), 22, 23),
1646ecac2f8SKishon Vijay Abraham I REG_FIELD(WIZ_LANECTL(2), 22, 23),
1656ecac2f8SKishon Vijay Abraham I REG_FIELD(WIZ_LANECTL(3), 22, 23),
1666ecac2f8SKishon Vijay Abraham I };
1676ecac2f8SKishon Vijay Abraham I
168b64a85fbSTanmay Patil static const struct reg_field p0_mac_src_sel[WIZ_MAX_LANES] = {
169b64a85fbSTanmay Patil REG_FIELD(WIZ_LANECTL(0), 20, 21),
170b64a85fbSTanmay Patil REG_FIELD(WIZ_LANECTL(1), 20, 21),
171b64a85fbSTanmay Patil REG_FIELD(WIZ_LANECTL(2), 20, 21),
172b64a85fbSTanmay Patil REG_FIELD(WIZ_LANECTL(3), 20, 21),
173b64a85fbSTanmay Patil };
174b64a85fbSTanmay Patil
175b64a85fbSTanmay Patil static const struct reg_field p0_rxfclk_sel[WIZ_MAX_LANES] = {
176b64a85fbSTanmay Patil REG_FIELD(WIZ_LANECTL(0), 6, 7),
177b64a85fbSTanmay Patil REG_FIELD(WIZ_LANECTL(1), 6, 7),
178b64a85fbSTanmay Patil REG_FIELD(WIZ_LANECTL(2), 6, 7),
179b64a85fbSTanmay Patil REG_FIELD(WIZ_LANECTL(3), 6, 7),
180b64a85fbSTanmay Patil };
181b64a85fbSTanmay Patil
182b64a85fbSTanmay Patil static const struct reg_field p0_refclk_sel[WIZ_MAX_LANES] = {
183b64a85fbSTanmay Patil REG_FIELD(WIZ_LANECTL(0), 18, 19),
184b64a85fbSTanmay Patil REG_FIELD(WIZ_LANECTL(1), 18, 19),
185b64a85fbSTanmay Patil REG_FIELD(WIZ_LANECTL(2), 18, 19),
186b64a85fbSTanmay Patil REG_FIELD(WIZ_LANECTL(3), 18, 19),
187b64a85fbSTanmay Patil };
18848820990SKishon Vijay Abraham I static const struct reg_field p_mac_div_sel0[WIZ_MAX_LANES] = {
18948820990SKishon Vijay Abraham I REG_FIELD(WIZ_LANEDIV(0), 16, 22),
19048820990SKishon Vijay Abraham I REG_FIELD(WIZ_LANEDIV(1), 16, 22),
19148820990SKishon Vijay Abraham I REG_FIELD(WIZ_LANEDIV(2), 16, 22),
19248820990SKishon Vijay Abraham I REG_FIELD(WIZ_LANEDIV(3), 16, 22),
19348820990SKishon Vijay Abraham I };
19448820990SKishon Vijay Abraham I
19548820990SKishon Vijay Abraham I static const struct reg_field p_mac_div_sel1[WIZ_MAX_LANES] = {
19648820990SKishon Vijay Abraham I REG_FIELD(WIZ_LANEDIV(0), 0, 8),
19748820990SKishon Vijay Abraham I REG_FIELD(WIZ_LANEDIV(1), 0, 8),
19848820990SKishon Vijay Abraham I REG_FIELD(WIZ_LANEDIV(2), 0, 8),
19948820990SKishon Vijay Abraham I REG_FIELD(WIZ_LANEDIV(3), 0, 8),
20048820990SKishon Vijay Abraham I };
20148820990SKishon Vijay Abraham I
202c9f9eba0SRoger Quadros static const struct reg_field typec_ln10_swap =
203c9f9eba0SRoger Quadros REG_FIELD(WIZ_SERDES_TYPEC, 30, 30);
204c9f9eba0SRoger Quadros
205494de1ddSSinthu Raja static const struct reg_field typec_ln23_swap =
206494de1ddSSinthu Raja REG_FIELD(WIZ_SERDES_TYPEC, 31, 31);
207494de1ddSSinthu Raja
208091876ccSKishon Vijay Abraham I struct wiz_clk_mux {
209091876ccSKishon Vijay Abraham I struct clk_hw hw;
210091876ccSKishon Vijay Abraham I struct regmap_field *field;
211549cb1aeSKishon Vijay Abraham I const u32 *table;
212091876ccSKishon Vijay Abraham I struct clk_init_data clk_data;
213091876ccSKishon Vijay Abraham I };
214091876ccSKishon Vijay Abraham I
215091876ccSKishon Vijay Abraham I #define to_wiz_clk_mux(_hw) container_of(_hw, struct wiz_clk_mux, hw)
216091876ccSKishon Vijay Abraham I
217091876ccSKishon Vijay Abraham I struct wiz_clk_divider {
218091876ccSKishon Vijay Abraham I struct clk_hw hw;
219091876ccSKishon Vijay Abraham I struct regmap_field *field;
2205a72122dSRikard Falkeborn const struct clk_div_table *table;
221091876ccSKishon Vijay Abraham I struct clk_init_data clk_data;
222091876ccSKishon Vijay Abraham I };
223091876ccSKishon Vijay Abraham I
224091876ccSKishon Vijay Abraham I #define to_wiz_clk_div(_hw) container_of(_hw, struct wiz_clk_divider, hw)
225091876ccSKishon Vijay Abraham I
226091876ccSKishon Vijay Abraham I struct wiz_clk_mux_sel {
227040cbe76SKishon Vijay Abraham I u32 table[WIZ_MAX_INPUT_CLOCKS];
228091876ccSKishon Vijay Abraham I const char *node_name;
229040cbe76SKishon Vijay Abraham I u32 num_parents;
230040cbe76SKishon Vijay Abraham I u32 parents[WIZ_MAX_INPUT_CLOCKS];
231091876ccSKishon Vijay Abraham I };
232091876ccSKishon Vijay Abraham I
233091876ccSKishon Vijay Abraham I struct wiz_clk_div_sel {
2345a72122dSRikard Falkeborn const struct clk_div_table *table;
235091876ccSKishon Vijay Abraham I const char *node_name;
236091876ccSKishon Vijay Abraham I };
237091876ccSKishon Vijay Abraham I
2389e405f87SKishon Vijay Abraham I struct wiz_phy_en_refclk {
2399e405f87SKishon Vijay Abraham I struct clk_hw hw;
2409e405f87SKishon Vijay Abraham I struct regmap_field *phy_en_refclk;
2419e405f87SKishon Vijay Abraham I struct clk_init_data clk_data;
2429e405f87SKishon Vijay Abraham I };
2439e405f87SKishon Vijay Abraham I
2449e405f87SKishon Vijay Abraham I #define to_wiz_phy_en_refclk(_hw) container_of(_hw, struct wiz_phy_en_refclk, hw)
2459e405f87SKishon Vijay Abraham I
246549cb1aeSKishon Vijay Abraham I static const struct wiz_clk_mux_sel clk_mux_sel_16g[] = {
247091876ccSKishon Vijay Abraham I {
248091876ccSKishon Vijay Abraham I /*
249091876ccSKishon Vijay Abraham I * Mux value to be configured for each of the input clocks
250091876ccSKishon Vijay Abraham I * in the order populated in device tree
251091876ccSKishon Vijay Abraham I */
252091876ccSKishon Vijay Abraham I .table = { 1, 0 },
253091876ccSKishon Vijay Abraham I .node_name = "pll0-refclk",
254091876ccSKishon Vijay Abraham I },
255091876ccSKishon Vijay Abraham I {
256091876ccSKishon Vijay Abraham I .table = { 1, 0 },
257091876ccSKishon Vijay Abraham I .node_name = "pll1-refclk",
258091876ccSKishon Vijay Abraham I },
259091876ccSKishon Vijay Abraham I {
260091876ccSKishon Vijay Abraham I .table = { 1, 3, 0, 2 },
261091876ccSKishon Vijay Abraham I .node_name = "refclk-dig",
262091876ccSKishon Vijay Abraham I },
263091876ccSKishon Vijay Abraham I };
264091876ccSKishon Vijay Abraham I
265549cb1aeSKishon Vijay Abraham I static const struct wiz_clk_mux_sel clk_mux_sel_10g[] = {
266091876ccSKishon Vijay Abraham I {
267091876ccSKishon Vijay Abraham I /*
268091876ccSKishon Vijay Abraham I * Mux value to be configured for each of the input clocks
269091876ccSKishon Vijay Abraham I * in the order populated in device tree
270091876ccSKishon Vijay Abraham I */
271040cbe76SKishon Vijay Abraham I .num_parents = 2,
272040cbe76SKishon Vijay Abraham I .parents = { WIZ_CORE_REFCLK, WIZ_EXT_REFCLK },
273091876ccSKishon Vijay Abraham I .table = { 1, 0 },
274091876ccSKishon Vijay Abraham I .node_name = "pll0-refclk",
275091876ccSKishon Vijay Abraham I },
276091876ccSKishon Vijay Abraham I {
277040cbe76SKishon Vijay Abraham I .num_parents = 2,
278040cbe76SKishon Vijay Abraham I .parents = { WIZ_CORE_REFCLK, WIZ_EXT_REFCLK },
279091876ccSKishon Vijay Abraham I .table = { 1, 0 },
280091876ccSKishon Vijay Abraham I .node_name = "pll1-refclk",
281091876ccSKishon Vijay Abraham I },
282091876ccSKishon Vijay Abraham I {
283040cbe76SKishon Vijay Abraham I .num_parents = 2,
284040cbe76SKishon Vijay Abraham I .parents = { WIZ_CORE_REFCLK, WIZ_EXT_REFCLK },
285091876ccSKishon Vijay Abraham I .table = { 1, 0 },
286091876ccSKishon Vijay Abraham I .node_name = "refclk-dig",
287091876ccSKishon Vijay Abraham I },
288091876ccSKishon Vijay Abraham I };
289091876ccSKishon Vijay Abraham I
290edd473d4SRoger Quadros static const struct wiz_clk_mux_sel clk_mux_sel_10g_2_refclk[] = {
291edd473d4SRoger Quadros {
292edd473d4SRoger Quadros .num_parents = 3,
293edd473d4SRoger Quadros .parents = { WIZ_CORE_REFCLK, WIZ_CORE_REFCLK1, WIZ_EXT_REFCLK },
294edd473d4SRoger Quadros .table = { 2, 3, 0 },
295edd473d4SRoger Quadros .node_name = "pll0-refclk",
296edd473d4SRoger Quadros },
297edd473d4SRoger Quadros {
298edd473d4SRoger Quadros .num_parents = 3,
299edd473d4SRoger Quadros .parents = { WIZ_CORE_REFCLK, WIZ_CORE_REFCLK1, WIZ_EXT_REFCLK },
300edd473d4SRoger Quadros .table = { 2, 3, 0 },
301edd473d4SRoger Quadros .node_name = "pll1-refclk",
302edd473d4SRoger Quadros },
303edd473d4SRoger Quadros {
304edd473d4SRoger Quadros .num_parents = 3,
305edd473d4SRoger Quadros .parents = { WIZ_CORE_REFCLK, WIZ_CORE_REFCLK1, WIZ_EXT_REFCLK },
306edd473d4SRoger Quadros .table = { 2, 3, 0 },
307edd473d4SRoger Quadros .node_name = "refclk-dig",
308edd473d4SRoger Quadros },
309edd473d4SRoger Quadros };
310edd473d4SRoger Quadros
3115a72122dSRikard Falkeborn static const struct clk_div_table clk_div_table[] = {
312091876ccSKishon Vijay Abraham I { .val = 0, .div = 1, },
313091876ccSKishon Vijay Abraham I { .val = 1, .div = 2, },
314091876ccSKishon Vijay Abraham I { .val = 2, .div = 4, },
315091876ccSKishon Vijay Abraham I { .val = 3, .div = 8, },
3166d1e6bcbSKishon Vijay Abraham I { /* sentinel */ },
317091876ccSKishon Vijay Abraham I };
318091876ccSKishon Vijay Abraham I
319549cb1aeSKishon Vijay Abraham I static const struct wiz_clk_div_sel clk_div_sel[] = {
320091876ccSKishon Vijay Abraham I {
321091876ccSKishon Vijay Abraham I .table = clk_div_table,
322091876ccSKishon Vijay Abraham I .node_name = "cmn-refclk-dig-div",
323091876ccSKishon Vijay Abraham I },
324091876ccSKishon Vijay Abraham I {
325091876ccSKishon Vijay Abraham I .table = clk_div_table,
326091876ccSKishon Vijay Abraham I .node_name = "cmn-refclk1-dig-div",
327091876ccSKishon Vijay Abraham I },
328091876ccSKishon Vijay Abraham I };
329091876ccSKishon Vijay Abraham I
330091876ccSKishon Vijay Abraham I enum wiz_type {
331091876ccSKishon Vijay Abraham I J721E_WIZ_16G,
332edd473d4SRoger Quadros J721E_WIZ_10G, /* Also for J7200 SR1.0 */
3336ecac2f8SKishon Vijay Abraham I AM64_WIZ_10G,
334edd473d4SRoger Quadros J7200_WIZ_10G, /* J7200 SR2.0 */
335e27ecef8SMatt Ranostay J784S4_WIZ_10G,
336bea3ce75SMatt Ranostay J721S2_WIZ_10G,
337091876ccSKishon Vijay Abraham I };
338091876ccSKishon Vijay Abraham I
3394daa43e9SRoger Quadros struct wiz_data {
3404daa43e9SRoger Quadros enum wiz_type type;
341edd473d4SRoger Quadros const struct reg_field *pll0_refclk_mux_sel;
342edd473d4SRoger Quadros const struct reg_field *pll1_refclk_mux_sel;
3434daa43e9SRoger Quadros const struct reg_field *refclk_dig_sel;
3444daa43e9SRoger Quadros const struct reg_field *pma_cmn_refclk1_dig_div;
345edd473d4SRoger Quadros const struct reg_field *pma_cmn_refclk1_int_mode;
3464daa43e9SRoger Quadros const struct wiz_clk_mux_sel *clk_mux_sel;
3474daa43e9SRoger Quadros unsigned int clk_div_sel_num;
3484daa43e9SRoger Quadros };
3494daa43e9SRoger Quadros
350c9f9eba0SRoger Quadros #define WIZ_TYPEC_DIR_DEBOUNCE_MIN 100 /* ms */
351c9f9eba0SRoger Quadros #define WIZ_TYPEC_DIR_DEBOUNCE_MAX 1000
352c9f9eba0SRoger Quadros
353091876ccSKishon Vijay Abraham I struct wiz {
354091876ccSKishon Vijay Abraham I struct regmap *regmap;
355edd473d4SRoger Quadros struct regmap *scm_regmap;
356091876ccSKishon Vijay Abraham I enum wiz_type type;
357549cb1aeSKishon Vijay Abraham I const struct wiz_clk_mux_sel *clk_mux_sel;
358549cb1aeSKishon Vijay Abraham I const struct wiz_clk_div_sel *clk_div_sel;
359091876ccSKishon Vijay Abraham I unsigned int clk_div_sel_num;
360091876ccSKishon Vijay Abraham I struct regmap_field *por_en;
361091876ccSKishon Vijay Abraham I struct regmap_field *phy_reset_n;
3629e405f87SKishon Vijay Abraham I struct regmap_field *phy_en_refclk;
363091876ccSKishon Vijay Abraham I struct regmap_field *p_enable[WIZ_MAX_LANES];
364091876ccSKishon Vijay Abraham I struct regmap_field *p_align[WIZ_MAX_LANES];
365091876ccSKishon Vijay Abraham I struct regmap_field *p_raw_auto_start[WIZ_MAX_LANES];
366091876ccSKishon Vijay Abraham I struct regmap_field *p_standard_mode[WIZ_MAX_LANES];
36748820990SKishon Vijay Abraham I struct regmap_field *p_mac_div_sel0[WIZ_MAX_LANES];
36848820990SKishon Vijay Abraham I struct regmap_field *p_mac_div_sel1[WIZ_MAX_LANES];
3696ecac2f8SKishon Vijay Abraham I struct regmap_field *p0_fullrt_div[WIZ_MAX_LANES];
370b64a85fbSTanmay Patil struct regmap_field *p0_mac_src_sel[WIZ_MAX_LANES];
371b64a85fbSTanmay Patil struct regmap_field *p0_rxfclk_sel[WIZ_MAX_LANES];
372b64a85fbSTanmay Patil struct regmap_field *p0_refclk_sel[WIZ_MAX_LANES];
373091876ccSKishon Vijay Abraham I struct regmap_field *pma_cmn_refclk_int_mode;
374edd473d4SRoger Quadros struct regmap_field *pma_cmn_refclk1_int_mode;
375091876ccSKishon Vijay Abraham I struct regmap_field *pma_cmn_refclk_mode;
376091876ccSKishon Vijay Abraham I struct regmap_field *pma_cmn_refclk_dig_div;
377091876ccSKishon Vijay Abraham I struct regmap_field *pma_cmn_refclk1_dig_div;
378549cb1aeSKishon Vijay Abraham I struct regmap_field *mux_sel_field[WIZ_MUX_NUM_CLOCKS];
379549cb1aeSKishon Vijay Abraham I struct regmap_field *div_sel_field[WIZ_DIV_NUM_CLOCKS_16G];
380c9f9eba0SRoger Quadros struct regmap_field *typec_ln10_swap;
381494de1ddSSinthu Raja struct regmap_field *typec_ln23_swap;
382edd473d4SRoger Quadros struct regmap_field *sup_legacy_clk_override;
383091876ccSKishon Vijay Abraham I
384091876ccSKishon Vijay Abraham I struct device *dev;
385091876ccSKishon Vijay Abraham I u32 num_lanes;
386091876ccSKishon Vijay Abraham I struct platform_device *serdes_pdev;
387091876ccSKishon Vijay Abraham I struct reset_controller_dev wiz_phy_reset_dev;
388c9f9eba0SRoger Quadros struct gpio_desc *gpio_typec_dir;
389c9f9eba0SRoger Quadros int typec_dir_delay;
3907ae14cf5SJyri Sarha u32 lane_phy_type[WIZ_MAX_LANES];
3919a8a54b9SSinthu Raja u32 master_lane_num[WIZ_MAX_LANES];
392040cbe76SKishon Vijay Abraham I struct clk *input_clks[WIZ_MAX_INPUT_CLOCKS];
393040cbe76SKishon Vijay Abraham I struct clk *output_clks[WIZ_MAX_OUTPUT_CLOCKS];
394040cbe76SKishon Vijay Abraham I struct clk_onecell_data clk_data;
3954daa43e9SRoger Quadros const struct wiz_data *data;
396091876ccSKishon Vijay Abraham I };
397091876ccSKishon Vijay Abraham I
wiz_reset(struct wiz * wiz)398091876ccSKishon Vijay Abraham I static int wiz_reset(struct wiz *wiz)
399091876ccSKishon Vijay Abraham I {
400091876ccSKishon Vijay Abraham I int ret;
401091876ccSKishon Vijay Abraham I
402091876ccSKishon Vijay Abraham I ret = regmap_field_write(wiz->por_en, 0x1);
403091876ccSKishon Vijay Abraham I if (ret)
404091876ccSKishon Vijay Abraham I return ret;
405091876ccSKishon Vijay Abraham I
406091876ccSKishon Vijay Abraham I mdelay(1);
407091876ccSKishon Vijay Abraham I
408091876ccSKishon Vijay Abraham I ret = regmap_field_write(wiz->por_en, 0x0);
409091876ccSKishon Vijay Abraham I if (ret)
410091876ccSKishon Vijay Abraham I return ret;
411091876ccSKishon Vijay Abraham I
412091876ccSKishon Vijay Abraham I return 0;
413091876ccSKishon Vijay Abraham I }
414091876ccSKishon Vijay Abraham I
wiz_p_mac_div_sel(struct wiz * wiz)41548820990SKishon Vijay Abraham I static int wiz_p_mac_div_sel(struct wiz *wiz)
41648820990SKishon Vijay Abraham I {
41748820990SKishon Vijay Abraham I u32 num_lanes = wiz->num_lanes;
41848820990SKishon Vijay Abraham I int ret;
41948820990SKishon Vijay Abraham I int i;
42048820990SKishon Vijay Abraham I
42148820990SKishon Vijay Abraham I for (i = 0; i < num_lanes; i++) {
42227ecdd07SSiddharth Vadapalli if (wiz->lane_phy_type[i] == PHY_TYPE_SGMII ||
423b64a85fbSTanmay Patil wiz->lane_phy_type[i] == PHY_TYPE_QSGMII ||
424b64a85fbSTanmay Patil wiz->lane_phy_type[i] == PHY_TYPE_USXGMII) {
42548820990SKishon Vijay Abraham I ret = regmap_field_write(wiz->p_mac_div_sel0[i], 1);
42648820990SKishon Vijay Abraham I if (ret)
42748820990SKishon Vijay Abraham I return ret;
42848820990SKishon Vijay Abraham I
42948820990SKishon Vijay Abraham I ret = regmap_field_write(wiz->p_mac_div_sel1[i], 2);
43048820990SKishon Vijay Abraham I if (ret)
43148820990SKishon Vijay Abraham I return ret;
43248820990SKishon Vijay Abraham I }
43348820990SKishon Vijay Abraham I }
43448820990SKishon Vijay Abraham I
43548820990SKishon Vijay Abraham I return 0;
43648820990SKishon Vijay Abraham I }
43748820990SKishon Vijay Abraham I
wiz_mode_select(struct wiz * wiz)438091876ccSKishon Vijay Abraham I static int wiz_mode_select(struct wiz *wiz)
439091876ccSKishon Vijay Abraham I {
440091876ccSKishon Vijay Abraham I u32 num_lanes = wiz->num_lanes;
4417ae14cf5SJyri Sarha enum wiz_lane_standard_mode mode;
442091876ccSKishon Vijay Abraham I int ret;
443091876ccSKishon Vijay Abraham I int i;
444091876ccSKishon Vijay Abraham I
445091876ccSKishon Vijay Abraham I for (i = 0; i < num_lanes; i++) {
44657c0e136SSiddharth Vadapalli if (wiz->lane_phy_type[i] == PHY_TYPE_DP) {
4477ae14cf5SJyri Sarha mode = LANE_MODE_GEN1;
44857c0e136SSiddharth Vadapalli } else if (wiz->lane_phy_type[i] == PHY_TYPE_QSGMII) {
44948820990SKishon Vijay Abraham I mode = LANE_MODE_GEN2;
45057c0e136SSiddharth Vadapalli } else if (wiz->lane_phy_type[i] == PHY_TYPE_USXGMII) {
451b64a85fbSTanmay Patil ret = regmap_field_write(wiz->p0_mac_src_sel[i], 0x3);
452b64a85fbSTanmay Patil ret = regmap_field_write(wiz->p0_rxfclk_sel[i], 0x3);
453b64a85fbSTanmay Patil ret = regmap_field_write(wiz->p0_refclk_sel[i], 0x3);
454b64a85fbSTanmay Patil mode = LANE_MODE_GEN1;
45557c0e136SSiddharth Vadapalli } else {
45657c0e136SSiddharth Vadapalli continue;
457b64a85fbSTanmay Patil }
458b64a85fbSTanmay Patil
4597ae14cf5SJyri Sarha ret = regmap_field_write(wiz->p_standard_mode[i], mode);
460091876ccSKishon Vijay Abraham I if (ret)
461091876ccSKishon Vijay Abraham I return ret;
462091876ccSKishon Vijay Abraham I }
463091876ccSKishon Vijay Abraham I
464091876ccSKishon Vijay Abraham I return 0;
465091876ccSKishon Vijay Abraham I }
466091876ccSKishon Vijay Abraham I
wiz_init_raw_interface(struct wiz * wiz,bool enable)467091876ccSKishon Vijay Abraham I static int wiz_init_raw_interface(struct wiz *wiz, bool enable)
468091876ccSKishon Vijay Abraham I {
469091876ccSKishon Vijay Abraham I u32 num_lanes = wiz->num_lanes;
470091876ccSKishon Vijay Abraham I int i;
471091876ccSKishon Vijay Abraham I int ret;
472091876ccSKishon Vijay Abraham I
473091876ccSKishon Vijay Abraham I for (i = 0; i < num_lanes; i++) {
474091876ccSKishon Vijay Abraham I ret = regmap_field_write(wiz->p_align[i], enable);
475091876ccSKishon Vijay Abraham I if (ret)
476091876ccSKishon Vijay Abraham I return ret;
477091876ccSKishon Vijay Abraham I
478091876ccSKishon Vijay Abraham I ret = regmap_field_write(wiz->p_raw_auto_start[i], enable);
479091876ccSKishon Vijay Abraham I if (ret)
480091876ccSKishon Vijay Abraham I return ret;
481091876ccSKishon Vijay Abraham I }
482091876ccSKishon Vijay Abraham I
483091876ccSKishon Vijay Abraham I return 0;
484091876ccSKishon Vijay Abraham I }
485091876ccSKishon Vijay Abraham I
wiz_init(struct wiz * wiz)486091876ccSKishon Vijay Abraham I static int wiz_init(struct wiz *wiz)
487091876ccSKishon Vijay Abraham I {
488091876ccSKishon Vijay Abraham I struct device *dev = wiz->dev;
489091876ccSKishon Vijay Abraham I int ret;
490091876ccSKishon Vijay Abraham I
491091876ccSKishon Vijay Abraham I ret = wiz_reset(wiz);
492091876ccSKishon Vijay Abraham I if (ret) {
493091876ccSKishon Vijay Abraham I dev_err(dev, "WIZ reset failed\n");
494091876ccSKishon Vijay Abraham I return ret;
495091876ccSKishon Vijay Abraham I }
496091876ccSKishon Vijay Abraham I
497091876ccSKishon Vijay Abraham I ret = wiz_mode_select(wiz);
498091876ccSKishon Vijay Abraham I if (ret) {
499091876ccSKishon Vijay Abraham I dev_err(dev, "WIZ mode select failed\n");
500091876ccSKishon Vijay Abraham I return ret;
501091876ccSKishon Vijay Abraham I }
502091876ccSKishon Vijay Abraham I
50348820990SKishon Vijay Abraham I ret = wiz_p_mac_div_sel(wiz);
50448820990SKishon Vijay Abraham I if (ret) {
50548820990SKishon Vijay Abraham I dev_err(dev, "Configuring P0 MAC DIV SEL failed\n");
50648820990SKishon Vijay Abraham I return ret;
50748820990SKishon Vijay Abraham I }
50848820990SKishon Vijay Abraham I
509091876ccSKishon Vijay Abraham I ret = wiz_init_raw_interface(wiz, true);
510091876ccSKishon Vijay Abraham I if (ret) {
511091876ccSKishon Vijay Abraham I dev_err(dev, "WIZ interface initialization failed\n");
512091876ccSKishon Vijay Abraham I return ret;
513091876ccSKishon Vijay Abraham I }
514091876ccSKishon Vijay Abraham I
515091876ccSKishon Vijay Abraham I return 0;
516091876ccSKishon Vijay Abraham I }
517091876ccSKishon Vijay Abraham I
wiz_regfield_init(struct wiz * wiz)518091876ccSKishon Vijay Abraham I static int wiz_regfield_init(struct wiz *wiz)
519091876ccSKishon Vijay Abraham I {
520091876ccSKishon Vijay Abraham I struct regmap *regmap = wiz->regmap;
521edd473d4SRoger Quadros struct regmap *scm_regmap = wiz->regmap; /* updated later to scm_regmap if applicable */
522091876ccSKishon Vijay Abraham I int num_lanes = wiz->num_lanes;
523091876ccSKishon Vijay Abraham I struct device *dev = wiz->dev;
5244daa43e9SRoger Quadros const struct wiz_data *data = wiz->data;
525091876ccSKishon Vijay Abraham I int i;
526091876ccSKishon Vijay Abraham I
527091876ccSKishon Vijay Abraham I wiz->por_en = devm_regmap_field_alloc(dev, regmap, por_en);
528091876ccSKishon Vijay Abraham I if (IS_ERR(wiz->por_en)) {
529091876ccSKishon Vijay Abraham I dev_err(dev, "POR_EN reg field init failed\n");
530091876ccSKishon Vijay Abraham I return PTR_ERR(wiz->por_en);
531091876ccSKishon Vijay Abraham I }
532091876ccSKishon Vijay Abraham I
533091876ccSKishon Vijay Abraham I wiz->phy_reset_n = devm_regmap_field_alloc(dev, regmap,
534091876ccSKishon Vijay Abraham I phy_reset_n);
535091876ccSKishon Vijay Abraham I if (IS_ERR(wiz->phy_reset_n)) {
536091876ccSKishon Vijay Abraham I dev_err(dev, "PHY_RESET_N reg field init failed\n");
537091876ccSKishon Vijay Abraham I return PTR_ERR(wiz->phy_reset_n);
538091876ccSKishon Vijay Abraham I }
539091876ccSKishon Vijay Abraham I
540091876ccSKishon Vijay Abraham I wiz->pma_cmn_refclk_int_mode =
541091876ccSKishon Vijay Abraham I devm_regmap_field_alloc(dev, regmap, pma_cmn_refclk_int_mode);
542091876ccSKishon Vijay Abraham I if (IS_ERR(wiz->pma_cmn_refclk_int_mode)) {
543091876ccSKishon Vijay Abraham I dev_err(dev, "PMA_CMN_REFCLK_INT_MODE reg field init failed\n");
544091876ccSKishon Vijay Abraham I return PTR_ERR(wiz->pma_cmn_refclk_int_mode);
545091876ccSKishon Vijay Abraham I }
546091876ccSKishon Vijay Abraham I
547091876ccSKishon Vijay Abraham I wiz->pma_cmn_refclk_mode =
548091876ccSKishon Vijay Abraham I devm_regmap_field_alloc(dev, regmap, pma_cmn_refclk_mode);
549091876ccSKishon Vijay Abraham I if (IS_ERR(wiz->pma_cmn_refclk_mode)) {
550091876ccSKishon Vijay Abraham I dev_err(dev, "PMA_CMN_REFCLK_MODE reg field init failed\n");
551091876ccSKishon Vijay Abraham I return PTR_ERR(wiz->pma_cmn_refclk_mode);
552091876ccSKishon Vijay Abraham I }
553091876ccSKishon Vijay Abraham I
554549cb1aeSKishon Vijay Abraham I wiz->div_sel_field[CMN_REFCLK_DIG_DIV] =
555549cb1aeSKishon Vijay Abraham I devm_regmap_field_alloc(dev, regmap, pma_cmn_refclk_dig_div);
556549cb1aeSKishon Vijay Abraham I if (IS_ERR(wiz->div_sel_field[CMN_REFCLK_DIG_DIV])) {
557091876ccSKishon Vijay Abraham I dev_err(dev, "PMA_CMN_REFCLK_DIG_DIV reg field init failed\n");
558549cb1aeSKishon Vijay Abraham I return PTR_ERR(wiz->div_sel_field[CMN_REFCLK_DIG_DIV]);
559091876ccSKishon Vijay Abraham I }
560091876ccSKishon Vijay Abraham I
5614daa43e9SRoger Quadros if (data->pma_cmn_refclk1_dig_div) {
562549cb1aeSKishon Vijay Abraham I wiz->div_sel_field[CMN_REFCLK1_DIG_DIV] =
563091876ccSKishon Vijay Abraham I devm_regmap_field_alloc(dev, regmap,
5644daa43e9SRoger Quadros *data->pma_cmn_refclk1_dig_div);
565549cb1aeSKishon Vijay Abraham I if (IS_ERR(wiz->div_sel_field[CMN_REFCLK1_DIG_DIV])) {
566091876ccSKishon Vijay Abraham I dev_err(dev, "PMA_CMN_REFCLK1_DIG_DIV reg field init failed\n");
567549cb1aeSKishon Vijay Abraham I return PTR_ERR(wiz->div_sel_field[CMN_REFCLK1_DIG_DIV]);
568091876ccSKishon Vijay Abraham I }
569091876ccSKishon Vijay Abraham I }
570091876ccSKishon Vijay Abraham I
571edd473d4SRoger Quadros if (wiz->scm_regmap) {
572edd473d4SRoger Quadros scm_regmap = wiz->scm_regmap;
573edd473d4SRoger Quadros wiz->sup_legacy_clk_override =
574edd473d4SRoger Quadros devm_regmap_field_alloc(dev, scm_regmap, sup_legacy_clk_override);
575edd473d4SRoger Quadros if (IS_ERR(wiz->sup_legacy_clk_override)) {
576edd473d4SRoger Quadros dev_err(dev, "SUP_LEGACY_CLK_OVERRIDE reg field init failed\n");
577edd473d4SRoger Quadros return PTR_ERR(wiz->sup_legacy_clk_override);
578edd473d4SRoger Quadros }
579edd473d4SRoger Quadros }
580edd473d4SRoger Quadros
581549cb1aeSKishon Vijay Abraham I wiz->mux_sel_field[PLL0_REFCLK] =
582edd473d4SRoger Quadros devm_regmap_field_alloc(dev, scm_regmap, *data->pll0_refclk_mux_sel);
583549cb1aeSKishon Vijay Abraham I if (IS_ERR(wiz->mux_sel_field[PLL0_REFCLK])) {
584091876ccSKishon Vijay Abraham I dev_err(dev, "PLL0_REFCLK_SEL reg field init failed\n");
585549cb1aeSKishon Vijay Abraham I return PTR_ERR(wiz->mux_sel_field[PLL0_REFCLK]);
586091876ccSKishon Vijay Abraham I }
587091876ccSKishon Vijay Abraham I
588549cb1aeSKishon Vijay Abraham I wiz->mux_sel_field[PLL1_REFCLK] =
589edd473d4SRoger Quadros devm_regmap_field_alloc(dev, scm_regmap, *data->pll1_refclk_mux_sel);
590549cb1aeSKishon Vijay Abraham I if (IS_ERR(wiz->mux_sel_field[PLL1_REFCLK])) {
591091876ccSKishon Vijay Abraham I dev_err(dev, "PLL1_REFCLK_SEL reg field init failed\n");
592549cb1aeSKishon Vijay Abraham I return PTR_ERR(wiz->mux_sel_field[PLL1_REFCLK]);
593091876ccSKishon Vijay Abraham I }
594091876ccSKishon Vijay Abraham I
595edd473d4SRoger Quadros wiz->mux_sel_field[REFCLK_DIG] = devm_regmap_field_alloc(dev, scm_regmap,
5964daa43e9SRoger Quadros *data->refclk_dig_sel);
597549cb1aeSKishon Vijay Abraham I if (IS_ERR(wiz->mux_sel_field[REFCLK_DIG])) {
598091876ccSKishon Vijay Abraham I dev_err(dev, "REFCLK_DIG_SEL reg field init failed\n");
599549cb1aeSKishon Vijay Abraham I return PTR_ERR(wiz->mux_sel_field[REFCLK_DIG]);
600091876ccSKishon Vijay Abraham I }
601091876ccSKishon Vijay Abraham I
602edd473d4SRoger Quadros if (data->pma_cmn_refclk1_int_mode) {
603edd473d4SRoger Quadros wiz->pma_cmn_refclk1_int_mode =
604edd473d4SRoger Quadros devm_regmap_field_alloc(dev, scm_regmap, *data->pma_cmn_refclk1_int_mode);
605edd473d4SRoger Quadros if (IS_ERR(wiz->pma_cmn_refclk1_int_mode)) {
606edd473d4SRoger Quadros dev_err(dev, "PMA_CMN_REFCLK1_INT_MODE reg field init failed\n");
607edd473d4SRoger Quadros return PTR_ERR(wiz->pma_cmn_refclk1_int_mode);
608edd473d4SRoger Quadros }
609edd473d4SRoger Quadros }
610edd473d4SRoger Quadros
611091876ccSKishon Vijay Abraham I for (i = 0; i < num_lanes; i++) {
612091876ccSKishon Vijay Abraham I wiz->p_enable[i] = devm_regmap_field_alloc(dev, regmap,
613091876ccSKishon Vijay Abraham I p_enable[i]);
614091876ccSKishon Vijay Abraham I if (IS_ERR(wiz->p_enable[i])) {
615091876ccSKishon Vijay Abraham I dev_err(dev, "P%d_ENABLE reg field init failed\n", i);
616091876ccSKishon Vijay Abraham I return PTR_ERR(wiz->p_enable[i]);
617091876ccSKishon Vijay Abraham I }
618091876ccSKishon Vijay Abraham I
619091876ccSKishon Vijay Abraham I wiz->p_align[i] = devm_regmap_field_alloc(dev, regmap,
620091876ccSKishon Vijay Abraham I p_align[i]);
621091876ccSKishon Vijay Abraham I if (IS_ERR(wiz->p_align[i])) {
622091876ccSKishon Vijay Abraham I dev_err(dev, "P%d_ALIGN reg field init failed\n", i);
623091876ccSKishon Vijay Abraham I return PTR_ERR(wiz->p_align[i]);
624091876ccSKishon Vijay Abraham I }
625091876ccSKishon Vijay Abraham I
626091876ccSKishon Vijay Abraham I wiz->p_raw_auto_start[i] =
627091876ccSKishon Vijay Abraham I devm_regmap_field_alloc(dev, regmap, p_raw_auto_start[i]);
628091876ccSKishon Vijay Abraham I if (IS_ERR(wiz->p_raw_auto_start[i])) {
629091876ccSKishon Vijay Abraham I dev_err(dev, "P%d_RAW_AUTO_START reg field init fail\n",
630091876ccSKishon Vijay Abraham I i);
631091876ccSKishon Vijay Abraham I return PTR_ERR(wiz->p_raw_auto_start[i]);
632091876ccSKishon Vijay Abraham I }
633091876ccSKishon Vijay Abraham I
634091876ccSKishon Vijay Abraham I wiz->p_standard_mode[i] =
635091876ccSKishon Vijay Abraham I devm_regmap_field_alloc(dev, regmap, p_standard_mode[i]);
636091876ccSKishon Vijay Abraham I if (IS_ERR(wiz->p_standard_mode[i])) {
637091876ccSKishon Vijay Abraham I dev_err(dev, "P%d_STANDARD_MODE reg field init fail\n",
638091876ccSKishon Vijay Abraham I i);
639091876ccSKishon Vijay Abraham I return PTR_ERR(wiz->p_standard_mode[i]);
640091876ccSKishon Vijay Abraham I }
6416ecac2f8SKishon Vijay Abraham I
6426ecac2f8SKishon Vijay Abraham I wiz->p0_fullrt_div[i] = devm_regmap_field_alloc(dev, regmap, p0_fullrt_div[i]);
6436ecac2f8SKishon Vijay Abraham I if (IS_ERR(wiz->p0_fullrt_div[i])) {
6446ecac2f8SKishon Vijay Abraham I dev_err(dev, "P%d_FULLRT_DIV reg field init failed\n", i);
6456ecac2f8SKishon Vijay Abraham I return PTR_ERR(wiz->p0_fullrt_div[i]);
6466ecac2f8SKishon Vijay Abraham I }
64748820990SKishon Vijay Abraham I
648b64a85fbSTanmay Patil wiz->p0_mac_src_sel[i] = devm_regmap_field_alloc(dev, regmap, p0_mac_src_sel[i]);
649b64a85fbSTanmay Patil if (IS_ERR(wiz->p0_mac_src_sel[i])) {
650b64a85fbSTanmay Patil dev_err(dev, "P%d_MAC_SRC_SEL reg field init failed\n", i);
651b64a85fbSTanmay Patil return PTR_ERR(wiz->p0_mac_src_sel[i]);
652b64a85fbSTanmay Patil }
653b64a85fbSTanmay Patil
654b64a85fbSTanmay Patil wiz->p0_rxfclk_sel[i] = devm_regmap_field_alloc(dev, regmap, p0_rxfclk_sel[i]);
655b64a85fbSTanmay Patil if (IS_ERR(wiz->p0_rxfclk_sel[i])) {
656b64a85fbSTanmay Patil dev_err(dev, "P%d_RXFCLK_SEL reg field init failed\n", i);
657b64a85fbSTanmay Patil return PTR_ERR(wiz->p0_rxfclk_sel[i]);
658b64a85fbSTanmay Patil }
659b64a85fbSTanmay Patil
660b64a85fbSTanmay Patil wiz->p0_refclk_sel[i] = devm_regmap_field_alloc(dev, regmap, p0_refclk_sel[i]);
661b64a85fbSTanmay Patil if (IS_ERR(wiz->p0_refclk_sel[i])) {
662b64a85fbSTanmay Patil dev_err(dev, "P%d_REFCLK_SEL reg field init failed\n", i);
663b64a85fbSTanmay Patil return PTR_ERR(wiz->p0_refclk_sel[i]);
664b64a85fbSTanmay Patil }
665b64a85fbSTanmay Patil
66648820990SKishon Vijay Abraham I wiz->p_mac_div_sel0[i] =
66748820990SKishon Vijay Abraham I devm_regmap_field_alloc(dev, regmap, p_mac_div_sel0[i]);
66848820990SKishon Vijay Abraham I if (IS_ERR(wiz->p_mac_div_sel0[i])) {
66948820990SKishon Vijay Abraham I dev_err(dev, "P%d_MAC_DIV_SEL0 reg field init fail\n",
67048820990SKishon Vijay Abraham I i);
67148820990SKishon Vijay Abraham I return PTR_ERR(wiz->p_mac_div_sel0[i]);
67248820990SKishon Vijay Abraham I }
67348820990SKishon Vijay Abraham I
67448820990SKishon Vijay Abraham I wiz->p_mac_div_sel1[i] =
67548820990SKishon Vijay Abraham I devm_regmap_field_alloc(dev, regmap, p_mac_div_sel1[i]);
67648820990SKishon Vijay Abraham I if (IS_ERR(wiz->p_mac_div_sel1[i])) {
67748820990SKishon Vijay Abraham I dev_err(dev, "P%d_MAC_DIV_SEL1 reg field init fail\n",
67848820990SKishon Vijay Abraham I i);
67948820990SKishon Vijay Abraham I return PTR_ERR(wiz->p_mac_div_sel1[i]);
68048820990SKishon Vijay Abraham I }
681091876ccSKishon Vijay Abraham I }
682091876ccSKishon Vijay Abraham I
683c9f9eba0SRoger Quadros wiz->typec_ln10_swap = devm_regmap_field_alloc(dev, regmap,
684c9f9eba0SRoger Quadros typec_ln10_swap);
685c9f9eba0SRoger Quadros if (IS_ERR(wiz->typec_ln10_swap)) {
686c9f9eba0SRoger Quadros dev_err(dev, "LN10_SWAP reg field init failed\n");
687c9f9eba0SRoger Quadros return PTR_ERR(wiz->typec_ln10_swap);
688c9f9eba0SRoger Quadros }
689c9f9eba0SRoger Quadros
690494de1ddSSinthu Raja wiz->typec_ln23_swap = devm_regmap_field_alloc(dev, regmap,
691494de1ddSSinthu Raja typec_ln23_swap);
692494de1ddSSinthu Raja if (IS_ERR(wiz->typec_ln23_swap)) {
693494de1ddSSinthu Raja dev_err(dev, "LN23_SWAP reg field init failed\n");
694494de1ddSSinthu Raja return PTR_ERR(wiz->typec_ln23_swap);
695494de1ddSSinthu Raja }
696494de1ddSSinthu Raja
6979e405f87SKishon Vijay Abraham I wiz->phy_en_refclk = devm_regmap_field_alloc(dev, regmap, phy_en_refclk);
6989e405f87SKishon Vijay Abraham I if (IS_ERR(wiz->phy_en_refclk)) {
6999e405f87SKishon Vijay Abraham I dev_err(dev, "PHY_EN_REFCLK reg field init failed\n");
7009e405f87SKishon Vijay Abraham I return PTR_ERR(wiz->phy_en_refclk);
7019e405f87SKishon Vijay Abraham I }
7029e405f87SKishon Vijay Abraham I
7039e405f87SKishon Vijay Abraham I return 0;
7049e405f87SKishon Vijay Abraham I }
7059e405f87SKishon Vijay Abraham I
wiz_phy_en_refclk_enable(struct clk_hw * hw)7069e405f87SKishon Vijay Abraham I static int wiz_phy_en_refclk_enable(struct clk_hw *hw)
7079e405f87SKishon Vijay Abraham I {
7089e405f87SKishon Vijay Abraham I struct wiz_phy_en_refclk *wiz_phy_en_refclk = to_wiz_phy_en_refclk(hw);
7099e405f87SKishon Vijay Abraham I struct regmap_field *phy_en_refclk = wiz_phy_en_refclk->phy_en_refclk;
7109e405f87SKishon Vijay Abraham I
7119e405f87SKishon Vijay Abraham I regmap_field_write(phy_en_refclk, 1);
7129e405f87SKishon Vijay Abraham I
7139e405f87SKishon Vijay Abraham I return 0;
7149e405f87SKishon Vijay Abraham I }
7159e405f87SKishon Vijay Abraham I
wiz_phy_en_refclk_disable(struct clk_hw * hw)7169e405f87SKishon Vijay Abraham I static void wiz_phy_en_refclk_disable(struct clk_hw *hw)
7179e405f87SKishon Vijay Abraham I {
7189e405f87SKishon Vijay Abraham I struct wiz_phy_en_refclk *wiz_phy_en_refclk = to_wiz_phy_en_refclk(hw);
7199e405f87SKishon Vijay Abraham I struct regmap_field *phy_en_refclk = wiz_phy_en_refclk->phy_en_refclk;
7209e405f87SKishon Vijay Abraham I
7219e405f87SKishon Vijay Abraham I regmap_field_write(phy_en_refclk, 0);
7229e405f87SKishon Vijay Abraham I }
7239e405f87SKishon Vijay Abraham I
wiz_phy_en_refclk_is_enabled(struct clk_hw * hw)7249e405f87SKishon Vijay Abraham I static int wiz_phy_en_refclk_is_enabled(struct clk_hw *hw)
7259e405f87SKishon Vijay Abraham I {
7269e405f87SKishon Vijay Abraham I struct wiz_phy_en_refclk *wiz_phy_en_refclk = to_wiz_phy_en_refclk(hw);
7279e405f87SKishon Vijay Abraham I struct regmap_field *phy_en_refclk = wiz_phy_en_refclk->phy_en_refclk;
7289e405f87SKishon Vijay Abraham I int val;
7299e405f87SKishon Vijay Abraham I
7309e405f87SKishon Vijay Abraham I regmap_field_read(phy_en_refclk, &val);
7319e405f87SKishon Vijay Abraham I
7329e405f87SKishon Vijay Abraham I return !!val;
7339e405f87SKishon Vijay Abraham I }
7349e405f87SKishon Vijay Abraham I
7359e405f87SKishon Vijay Abraham I static const struct clk_ops wiz_phy_en_refclk_ops = {
7369e405f87SKishon Vijay Abraham I .enable = wiz_phy_en_refclk_enable,
7379e405f87SKishon Vijay Abraham I .disable = wiz_phy_en_refclk_disable,
7389e405f87SKishon Vijay Abraham I .is_enabled = wiz_phy_en_refclk_is_enabled,
7399e405f87SKishon Vijay Abraham I };
7409e405f87SKishon Vijay Abraham I
wiz_phy_en_refclk_register(struct wiz * wiz)7419e405f87SKishon Vijay Abraham I static int wiz_phy_en_refclk_register(struct wiz *wiz)
7429e405f87SKishon Vijay Abraham I {
7439e405f87SKishon Vijay Abraham I struct wiz_phy_en_refclk *wiz_phy_en_refclk;
7449e405f87SKishon Vijay Abraham I struct device *dev = wiz->dev;
7459e405f87SKishon Vijay Abraham I struct clk_init_data *init;
7469e405f87SKishon Vijay Abraham I struct clk *clk;
7471fbef61bSMatt Ranostay char *clk_name;
7481fbef61bSMatt Ranostay unsigned int sz;
7499e405f87SKishon Vijay Abraham I
7509e405f87SKishon Vijay Abraham I wiz_phy_en_refclk = devm_kzalloc(dev, sizeof(*wiz_phy_en_refclk), GFP_KERNEL);
7519e405f87SKishon Vijay Abraham I if (!wiz_phy_en_refclk)
7529e405f87SKishon Vijay Abraham I return -ENOMEM;
7539e405f87SKishon Vijay Abraham I
7549e405f87SKishon Vijay Abraham I init = &wiz_phy_en_refclk->clk_data;
7559e405f87SKishon Vijay Abraham I
7569e405f87SKishon Vijay Abraham I init->ops = &wiz_phy_en_refclk_ops;
7579e405f87SKishon Vijay Abraham I init->flags = 0;
7581fbef61bSMatt Ranostay
7591fbef61bSMatt Ranostay sz = strlen(dev_name(dev)) + strlen(output_clk_names[TI_WIZ_PHY_EN_REFCLK]) + 2;
7601fbef61bSMatt Ranostay
7611fbef61bSMatt Ranostay clk_name = kzalloc(sz, GFP_KERNEL);
7621fbef61bSMatt Ranostay if (!clk_name)
7631fbef61bSMatt Ranostay return -ENOMEM;
7641fbef61bSMatt Ranostay
7651fbef61bSMatt Ranostay snprintf(clk_name, sz, "%s_%s", dev_name(dev), output_clk_names[TI_WIZ_PHY_EN_REFCLK]);
7661fbef61bSMatt Ranostay init->name = clk_name;
7679e405f87SKishon Vijay Abraham I
7689e405f87SKishon Vijay Abraham I wiz_phy_en_refclk->phy_en_refclk = wiz->phy_en_refclk;
7699e405f87SKishon Vijay Abraham I wiz_phy_en_refclk->hw.init = init;
7709e405f87SKishon Vijay Abraham I
7719e405f87SKishon Vijay Abraham I clk = devm_clk_register(dev, &wiz_phy_en_refclk->hw);
7721fbef61bSMatt Ranostay
7731fbef61bSMatt Ranostay kfree(clk_name);
7741fbef61bSMatt Ranostay
7759e405f87SKishon Vijay Abraham I if (IS_ERR(clk))
7769e405f87SKishon Vijay Abraham I return PTR_ERR(clk);
7779e405f87SKishon Vijay Abraham I
7789e405f87SKishon Vijay Abraham I wiz->output_clks[TI_WIZ_PHY_EN_REFCLK] = clk;
7799e405f87SKishon Vijay Abraham I
780091876ccSKishon Vijay Abraham I return 0;
781091876ccSKishon Vijay Abraham I }
782091876ccSKishon Vijay Abraham I
wiz_clk_mux_get_parent(struct clk_hw * hw)783091876ccSKishon Vijay Abraham I static u8 wiz_clk_mux_get_parent(struct clk_hw *hw)
784091876ccSKishon Vijay Abraham I {
785091876ccSKishon Vijay Abraham I struct wiz_clk_mux *mux = to_wiz_clk_mux(hw);
786091876ccSKishon Vijay Abraham I struct regmap_field *field = mux->field;
787091876ccSKishon Vijay Abraham I unsigned int val;
788091876ccSKishon Vijay Abraham I
789091876ccSKishon Vijay Abraham I regmap_field_read(field, &val);
790549cb1aeSKishon Vijay Abraham I return clk_mux_val_to_index(hw, (u32 *)mux->table, 0, val);
791091876ccSKishon Vijay Abraham I }
792091876ccSKishon Vijay Abraham I
wiz_clk_mux_set_parent(struct clk_hw * hw,u8 index)793091876ccSKishon Vijay Abraham I static int wiz_clk_mux_set_parent(struct clk_hw *hw, u8 index)
794091876ccSKishon Vijay Abraham I {
795091876ccSKishon Vijay Abraham I struct wiz_clk_mux *mux = to_wiz_clk_mux(hw);
796091876ccSKishon Vijay Abraham I struct regmap_field *field = mux->field;
797091876ccSKishon Vijay Abraham I int val;
798091876ccSKishon Vijay Abraham I
799091876ccSKishon Vijay Abraham I val = mux->table[index];
800091876ccSKishon Vijay Abraham I return regmap_field_write(field, val);
801091876ccSKishon Vijay Abraham I }
802091876ccSKishon Vijay Abraham I
803091876ccSKishon Vijay Abraham I static const struct clk_ops wiz_clk_mux_ops = {
804091876ccSKishon Vijay Abraham I .determine_rate = __clk_mux_determine_rate,
805091876ccSKishon Vijay Abraham I .set_parent = wiz_clk_mux_set_parent,
806091876ccSKishon Vijay Abraham I .get_parent = wiz_clk_mux_get_parent,
807091876ccSKishon Vijay Abraham I };
808040cbe76SKishon Vijay Abraham I
wiz_mux_clk_register(struct wiz * wiz,struct regmap_field * field,const struct wiz_clk_mux_sel * mux_sel,int clk_index)809040cbe76SKishon Vijay Abraham I static int wiz_mux_clk_register(struct wiz *wiz, struct regmap_field *field,
810040cbe76SKishon Vijay Abraham I const struct wiz_clk_mux_sel *mux_sel, int clk_index)
811040cbe76SKishon Vijay Abraham I {
812040cbe76SKishon Vijay Abraham I struct device *dev = wiz->dev;
813040cbe76SKishon Vijay Abraham I struct clk_init_data *init;
814040cbe76SKishon Vijay Abraham I const char **parent_names;
815040cbe76SKishon Vijay Abraham I unsigned int num_parents;
816040cbe76SKishon Vijay Abraham I struct wiz_clk_mux *mux;
817040cbe76SKishon Vijay Abraham I char clk_name[100];
818040cbe76SKishon Vijay Abraham I struct clk *clk;
819040cbe76SKishon Vijay Abraham I int ret = 0, i;
820040cbe76SKishon Vijay Abraham I
821040cbe76SKishon Vijay Abraham I mux = devm_kzalloc(dev, sizeof(*mux), GFP_KERNEL);
822040cbe76SKishon Vijay Abraham I if (!mux)
823040cbe76SKishon Vijay Abraham I return -ENOMEM;
824040cbe76SKishon Vijay Abraham I
825040cbe76SKishon Vijay Abraham I num_parents = mux_sel->num_parents;
826040cbe76SKishon Vijay Abraham I
827040cbe76SKishon Vijay Abraham I parent_names = kzalloc((sizeof(char *) * num_parents), GFP_KERNEL);
828040cbe76SKishon Vijay Abraham I if (!parent_names)
829040cbe76SKishon Vijay Abraham I return -ENOMEM;
830040cbe76SKishon Vijay Abraham I
831040cbe76SKishon Vijay Abraham I for (i = 0; i < num_parents; i++) {
832040cbe76SKishon Vijay Abraham I clk = wiz->input_clks[mux_sel->parents[i]];
833040cbe76SKishon Vijay Abraham I if (IS_ERR_OR_NULL(clk)) {
834040cbe76SKishon Vijay Abraham I dev_err(dev, "Failed to get parent clk for %s\n",
835040cbe76SKishon Vijay Abraham I output_clk_names[clk_index]);
836040cbe76SKishon Vijay Abraham I ret = -EINVAL;
837040cbe76SKishon Vijay Abraham I goto err;
838040cbe76SKishon Vijay Abraham I }
839040cbe76SKishon Vijay Abraham I parent_names[i] = __clk_get_name(clk);
840040cbe76SKishon Vijay Abraham I }
841040cbe76SKishon Vijay Abraham I
842040cbe76SKishon Vijay Abraham I snprintf(clk_name, sizeof(clk_name), "%s_%s", dev_name(dev), output_clk_names[clk_index]);
843040cbe76SKishon Vijay Abraham I
844040cbe76SKishon Vijay Abraham I init = &mux->clk_data;
845040cbe76SKishon Vijay Abraham I
846040cbe76SKishon Vijay Abraham I init->ops = &wiz_clk_mux_ops;
847040cbe76SKishon Vijay Abraham I init->flags = CLK_SET_RATE_NO_REPARENT;
848040cbe76SKishon Vijay Abraham I init->parent_names = parent_names;
849040cbe76SKishon Vijay Abraham I init->num_parents = num_parents;
850040cbe76SKishon Vijay Abraham I init->name = clk_name;
851040cbe76SKishon Vijay Abraham I
852040cbe76SKishon Vijay Abraham I mux->field = field;
853040cbe76SKishon Vijay Abraham I mux->table = mux_sel->table;
854040cbe76SKishon Vijay Abraham I mux->hw.init = init;
855040cbe76SKishon Vijay Abraham I
856040cbe76SKishon Vijay Abraham I clk = devm_clk_register(dev, &mux->hw);
857040cbe76SKishon Vijay Abraham I if (IS_ERR(clk)) {
858040cbe76SKishon Vijay Abraham I ret = PTR_ERR(clk);
859040cbe76SKishon Vijay Abraham I goto err;
860040cbe76SKishon Vijay Abraham I }
861040cbe76SKishon Vijay Abraham I
862040cbe76SKishon Vijay Abraham I wiz->output_clks[clk_index] = clk;
863040cbe76SKishon Vijay Abraham I
864040cbe76SKishon Vijay Abraham I err:
865040cbe76SKishon Vijay Abraham I kfree(parent_names);
866040cbe76SKishon Vijay Abraham I
867040cbe76SKishon Vijay Abraham I return ret;
868040cbe76SKishon Vijay Abraham I }
869040cbe76SKishon Vijay Abraham I
wiz_mux_of_clk_register(struct wiz * wiz,struct device_node * node,struct regmap_field * field,const u32 * table)870549cb1aeSKishon Vijay Abraham I static int wiz_mux_of_clk_register(struct wiz *wiz, struct device_node *node,
871091876ccSKishon Vijay Abraham I struct regmap_field *field, const u32 *table)
872091876ccSKishon Vijay Abraham I {
873091876ccSKishon Vijay Abraham I struct device *dev = wiz->dev;
874091876ccSKishon Vijay Abraham I struct clk_init_data *init;
875091876ccSKishon Vijay Abraham I const char **parent_names;
876091876ccSKishon Vijay Abraham I unsigned int num_parents;
877091876ccSKishon Vijay Abraham I struct wiz_clk_mux *mux;
878091876ccSKishon Vijay Abraham I char clk_name[100];
879091876ccSKishon Vijay Abraham I struct clk *clk;
880091876ccSKishon Vijay Abraham I int ret;
881091876ccSKishon Vijay Abraham I
882091876ccSKishon Vijay Abraham I mux = devm_kzalloc(dev, sizeof(*mux), GFP_KERNEL);
883091876ccSKishon Vijay Abraham I if (!mux)
884091876ccSKishon Vijay Abraham I return -ENOMEM;
885091876ccSKishon Vijay Abraham I
886091876ccSKishon Vijay Abraham I num_parents = of_clk_get_parent_count(node);
887091876ccSKishon Vijay Abraham I if (num_parents < 2) {
888091876ccSKishon Vijay Abraham I dev_err(dev, "SERDES clock must have parents\n");
889091876ccSKishon Vijay Abraham I return -EINVAL;
890091876ccSKishon Vijay Abraham I }
891091876ccSKishon Vijay Abraham I
892091876ccSKishon Vijay Abraham I parent_names = devm_kzalloc(dev, (sizeof(char *) * num_parents),
893091876ccSKishon Vijay Abraham I GFP_KERNEL);
894091876ccSKishon Vijay Abraham I if (!parent_names)
895091876ccSKishon Vijay Abraham I return -ENOMEM;
896091876ccSKishon Vijay Abraham I
897091876ccSKishon Vijay Abraham I of_clk_parent_fill(node, parent_names, num_parents);
898091876ccSKishon Vijay Abraham I
899091876ccSKishon Vijay Abraham I snprintf(clk_name, sizeof(clk_name), "%s_%s", dev_name(dev),
900091876ccSKishon Vijay Abraham I node->name);
901091876ccSKishon Vijay Abraham I
902091876ccSKishon Vijay Abraham I init = &mux->clk_data;
903091876ccSKishon Vijay Abraham I
904091876ccSKishon Vijay Abraham I init->ops = &wiz_clk_mux_ops;
905091876ccSKishon Vijay Abraham I init->flags = CLK_SET_RATE_NO_REPARENT;
906091876ccSKishon Vijay Abraham I init->parent_names = parent_names;
907091876ccSKishon Vijay Abraham I init->num_parents = num_parents;
908091876ccSKishon Vijay Abraham I init->name = clk_name;
909091876ccSKishon Vijay Abraham I
910091876ccSKishon Vijay Abraham I mux->field = field;
911091876ccSKishon Vijay Abraham I mux->table = table;
912091876ccSKishon Vijay Abraham I mux->hw.init = init;
913091876ccSKishon Vijay Abraham I
914091876ccSKishon Vijay Abraham I clk = devm_clk_register(dev, &mux->hw);
915091876ccSKishon Vijay Abraham I if (IS_ERR(clk))
916091876ccSKishon Vijay Abraham I return PTR_ERR(clk);
917091876ccSKishon Vijay Abraham I
918091876ccSKishon Vijay Abraham I ret = of_clk_add_provider(node, of_clk_src_simple_get, clk);
919091876ccSKishon Vijay Abraham I if (ret)
920091876ccSKishon Vijay Abraham I dev_err(dev, "Failed to add clock provider: %s\n", clk_name);
921091876ccSKishon Vijay Abraham I
922091876ccSKishon Vijay Abraham I return ret;
923091876ccSKishon Vijay Abraham I }
924091876ccSKishon Vijay Abraham I
wiz_clk_div_recalc_rate(struct clk_hw * hw,unsigned long parent_rate)925091876ccSKishon Vijay Abraham I static unsigned long wiz_clk_div_recalc_rate(struct clk_hw *hw,
926091876ccSKishon Vijay Abraham I unsigned long parent_rate)
927091876ccSKishon Vijay Abraham I {
928091876ccSKishon Vijay Abraham I struct wiz_clk_divider *div = to_wiz_clk_div(hw);
929091876ccSKishon Vijay Abraham I struct regmap_field *field = div->field;
930091876ccSKishon Vijay Abraham I int val;
931091876ccSKishon Vijay Abraham I
932091876ccSKishon Vijay Abraham I regmap_field_read(field, &val);
933091876ccSKishon Vijay Abraham I
934091876ccSKishon Vijay Abraham I return divider_recalc_rate(hw, parent_rate, val, div->table, 0x0, 2);
935091876ccSKishon Vijay Abraham I }
936091876ccSKishon Vijay Abraham I
wiz_clk_div_round_rate(struct clk_hw * hw,unsigned long rate,unsigned long * prate)937091876ccSKishon Vijay Abraham I static long wiz_clk_div_round_rate(struct clk_hw *hw, unsigned long rate,
938091876ccSKishon Vijay Abraham I unsigned long *prate)
939091876ccSKishon Vijay Abraham I {
940091876ccSKishon Vijay Abraham I struct wiz_clk_divider *div = to_wiz_clk_div(hw);
941091876ccSKishon Vijay Abraham I
942091876ccSKishon Vijay Abraham I return divider_round_rate(hw, rate, prate, div->table, 2, 0x0);
943091876ccSKishon Vijay Abraham I }
944091876ccSKishon Vijay Abraham I
wiz_clk_div_set_rate(struct clk_hw * hw,unsigned long rate,unsigned long parent_rate)945091876ccSKishon Vijay Abraham I static int wiz_clk_div_set_rate(struct clk_hw *hw, unsigned long rate,
946091876ccSKishon Vijay Abraham I unsigned long parent_rate)
947091876ccSKishon Vijay Abraham I {
948091876ccSKishon Vijay Abraham I struct wiz_clk_divider *div = to_wiz_clk_div(hw);
949091876ccSKishon Vijay Abraham I struct regmap_field *field = div->field;
950091876ccSKishon Vijay Abraham I int val;
951091876ccSKishon Vijay Abraham I
952091876ccSKishon Vijay Abraham I val = divider_get_val(rate, parent_rate, div->table, 2, 0x0);
953091876ccSKishon Vijay Abraham I if (val < 0)
954091876ccSKishon Vijay Abraham I return val;
955091876ccSKishon Vijay Abraham I
956091876ccSKishon Vijay Abraham I return regmap_field_write(field, val);
957091876ccSKishon Vijay Abraham I }
958091876ccSKishon Vijay Abraham I
959091876ccSKishon Vijay Abraham I static const struct clk_ops wiz_clk_div_ops = {
960091876ccSKishon Vijay Abraham I .recalc_rate = wiz_clk_div_recalc_rate,
961091876ccSKishon Vijay Abraham I .round_rate = wiz_clk_div_round_rate,
962091876ccSKishon Vijay Abraham I .set_rate = wiz_clk_div_set_rate,
963091876ccSKishon Vijay Abraham I };
964091876ccSKishon Vijay Abraham I
wiz_div_clk_register(struct wiz * wiz,struct device_node * node,struct regmap_field * field,const struct clk_div_table * table)965091876ccSKishon Vijay Abraham I static int wiz_div_clk_register(struct wiz *wiz, struct device_node *node,
9665a72122dSRikard Falkeborn struct regmap_field *field,
967091876ccSKishon Vijay Abraham I const struct clk_div_table *table)
968091876ccSKishon Vijay Abraham I {
969091876ccSKishon Vijay Abraham I struct device *dev = wiz->dev;
970091876ccSKishon Vijay Abraham I struct wiz_clk_divider *div;
971091876ccSKishon Vijay Abraham I struct clk_init_data *init;
972091876ccSKishon Vijay Abraham I const char **parent_names;
973091876ccSKishon Vijay Abraham I char clk_name[100];
974091876ccSKishon Vijay Abraham I struct clk *clk;
975091876ccSKishon Vijay Abraham I int ret;
976091876ccSKishon Vijay Abraham I
977091876ccSKishon Vijay Abraham I div = devm_kzalloc(dev, sizeof(*div), GFP_KERNEL);
978091876ccSKishon Vijay Abraham I if (!div)
979091876ccSKishon Vijay Abraham I return -ENOMEM;
980091876ccSKishon Vijay Abraham I
981091876ccSKishon Vijay Abraham I snprintf(clk_name, sizeof(clk_name), "%s_%s", dev_name(dev),
982091876ccSKishon Vijay Abraham I node->name);
983091876ccSKishon Vijay Abraham I
984091876ccSKishon Vijay Abraham I parent_names = devm_kzalloc(dev, sizeof(char *), GFP_KERNEL);
985091876ccSKishon Vijay Abraham I if (!parent_names)
986091876ccSKishon Vijay Abraham I return -ENOMEM;
987091876ccSKishon Vijay Abraham I
988091876ccSKishon Vijay Abraham I of_clk_parent_fill(node, parent_names, 1);
989091876ccSKishon Vijay Abraham I
990091876ccSKishon Vijay Abraham I init = &div->clk_data;
991091876ccSKishon Vijay Abraham I
992091876ccSKishon Vijay Abraham I init->ops = &wiz_clk_div_ops;
993091876ccSKishon Vijay Abraham I init->flags = 0;
994091876ccSKishon Vijay Abraham I init->parent_names = parent_names;
995091876ccSKishon Vijay Abraham I init->num_parents = 1;
996091876ccSKishon Vijay Abraham I init->name = clk_name;
997091876ccSKishon Vijay Abraham I
998091876ccSKishon Vijay Abraham I div->field = field;
999091876ccSKishon Vijay Abraham I div->table = table;
1000091876ccSKishon Vijay Abraham I div->hw.init = init;
1001091876ccSKishon Vijay Abraham I
1002091876ccSKishon Vijay Abraham I clk = devm_clk_register(dev, &div->hw);
1003091876ccSKishon Vijay Abraham I if (IS_ERR(clk))
1004091876ccSKishon Vijay Abraham I return PTR_ERR(clk);
1005091876ccSKishon Vijay Abraham I
1006091876ccSKishon Vijay Abraham I ret = of_clk_add_provider(node, of_clk_src_simple_get, clk);
1007091876ccSKishon Vijay Abraham I if (ret)
1008091876ccSKishon Vijay Abraham I dev_err(dev, "Failed to add clock provider: %s\n", clk_name);
1009091876ccSKishon Vijay Abraham I
1010091876ccSKishon Vijay Abraham I return ret;
1011091876ccSKishon Vijay Abraham I }
1012091876ccSKishon Vijay Abraham I
wiz_clock_cleanup(struct wiz * wiz,struct device_node * node)1013091876ccSKishon Vijay Abraham I static void wiz_clock_cleanup(struct wiz *wiz, struct device_node *node)
1014549cb1aeSKishon Vijay Abraham I {
1015040cbe76SKishon Vijay Abraham I const struct wiz_clk_mux_sel *clk_mux_sel = wiz->clk_mux_sel;
1016091876ccSKishon Vijay Abraham I struct device *dev = wiz->dev;
1017091876ccSKishon Vijay Abraham I struct device_node *clk_node;
1018091876ccSKishon Vijay Abraham I int i;
1019edd473d4SRoger Quadros
1020edd473d4SRoger Quadros switch (wiz->type) {
1021edd473d4SRoger Quadros case AM64_WIZ_10G:
1022e27ecef8SMatt Ranostay case J7200_WIZ_10G:
1023bea3ce75SMatt Ranostay case J784S4_WIZ_10G:
1024040cbe76SKishon Vijay Abraham I case J721S2_WIZ_10G:
1025040cbe76SKishon Vijay Abraham I of_clk_del_provider(dev->of_node);
1026edd473d4SRoger Quadros return;
1027edd473d4SRoger Quadros default:
1028040cbe76SKishon Vijay Abraham I break;
1029040cbe76SKishon Vijay Abraham I }
1030091876ccSKishon Vijay Abraham I
1031091876ccSKishon Vijay Abraham I for (i = 0; i < WIZ_MUX_NUM_CLOCKS; i++) {
1032091876ccSKishon Vijay Abraham I clk_node = of_get_child_by_name(node, clk_mux_sel[i].node_name);
1033091876ccSKishon Vijay Abraham I of_clk_del_provider(clk_node);
1034091876ccSKishon Vijay Abraham I of_node_put(clk_node);
10357e52a39fSKishon Vijay Abraham I }
10367e52a39fSKishon Vijay Abraham I
10377e52a39fSKishon Vijay Abraham I for (i = 0; i < wiz->clk_div_sel_num; i++) {
10387e52a39fSKishon Vijay Abraham I clk_node = of_get_child_by_name(node, clk_div_sel[i].node_name);
10397e52a39fSKishon Vijay Abraham I of_clk_del_provider(clk_node);
10407e52a39fSKishon Vijay Abraham I of_node_put(clk_node);
10419e405f87SKishon Vijay Abraham I }
10429e405f87SKishon Vijay Abraham I
1043091876ccSKishon Vijay Abraham I of_clk_del_provider(wiz->dev->of_node);
1044091876ccSKishon Vijay Abraham I }
1045040cbe76SKishon Vijay Abraham I
wiz_clock_register(struct wiz * wiz)1046040cbe76SKishon Vijay Abraham I static int wiz_clock_register(struct wiz *wiz)
1047040cbe76SKishon Vijay Abraham I {
1048040cbe76SKishon Vijay Abraham I const struct wiz_clk_mux_sel *clk_mux_sel = wiz->clk_mux_sel;
1049040cbe76SKishon Vijay Abraham I struct device *dev = wiz->dev;
1050040cbe76SKishon Vijay Abraham I struct device_node *node = dev->of_node;
1051040cbe76SKishon Vijay Abraham I int clk_index;
1052040cbe76SKishon Vijay Abraham I int ret;
1053040cbe76SKishon Vijay Abraham I int i;
1054040cbe76SKishon Vijay Abraham I
1055040cbe76SKishon Vijay Abraham I clk_index = TI_WIZ_PLL0_REFCLK;
1056040cbe76SKishon Vijay Abraham I for (i = 0; i < WIZ_MUX_NUM_CLOCKS; i++, clk_index++) {
1057040cbe76SKishon Vijay Abraham I ret = wiz_mux_clk_register(wiz, wiz->mux_sel_field[i], &clk_mux_sel[i], clk_index);
1058040cbe76SKishon Vijay Abraham I if (ret) {
1059040cbe76SKishon Vijay Abraham I dev_err(dev, "Failed to register clk: %s\n", output_clk_names[clk_index]);
1060040cbe76SKishon Vijay Abraham I return ret;
1061040cbe76SKishon Vijay Abraham I }
1062040cbe76SKishon Vijay Abraham I }
10639e405f87SKishon Vijay Abraham I
10649e405f87SKishon Vijay Abraham I ret = wiz_phy_en_refclk_register(wiz);
10659e405f87SKishon Vijay Abraham I if (ret) {
10669e405f87SKishon Vijay Abraham I dev_err(dev, "Failed to add phy-en-refclk\n");
10679e405f87SKishon Vijay Abraham I return ret;
10689e405f87SKishon Vijay Abraham I }
1069040cbe76SKishon Vijay Abraham I
1070040cbe76SKishon Vijay Abraham I wiz->clk_data.clks = wiz->output_clks;
1071040cbe76SKishon Vijay Abraham I wiz->clk_data.clk_num = WIZ_MAX_OUTPUT_CLOCKS;
1072040cbe76SKishon Vijay Abraham I ret = of_clk_add_provider(node, of_clk_src_onecell_get, &wiz->clk_data);
1073040cbe76SKishon Vijay Abraham I if (ret)
1074040cbe76SKishon Vijay Abraham I dev_err(dev, "Failed to add clock provider: %s\n", node->name);
1075040cbe76SKishon Vijay Abraham I
1076040cbe76SKishon Vijay Abraham I return ret;
1077040cbe76SKishon Vijay Abraham I }
1078091876ccSKishon Vijay Abraham I
wiz_clock_init(struct wiz * wiz,struct device_node * node)1079091876ccSKishon Vijay Abraham I static int wiz_clock_init(struct wiz *wiz, struct device_node *node)
1080549cb1aeSKishon Vijay Abraham I {
1081091876ccSKishon Vijay Abraham I const struct wiz_clk_mux_sel *clk_mux_sel = wiz->clk_mux_sel;
1082091876ccSKishon Vijay Abraham I struct device *dev = wiz->dev;
1083091876ccSKishon Vijay Abraham I struct device_node *clk_node;
1084091876ccSKishon Vijay Abraham I const char *node_name;
1085091876ccSKishon Vijay Abraham I unsigned long rate;
1086091876ccSKishon Vijay Abraham I struct clk *clk;
1087091876ccSKishon Vijay Abraham I int ret;
1088091876ccSKishon Vijay Abraham I int i;
1089091876ccSKishon Vijay Abraham I
1090091876ccSKishon Vijay Abraham I clk = devm_clk_get(dev, "core_ref_clk");
1091091876ccSKishon Vijay Abraham I if (IS_ERR(clk)) {
1092091876ccSKishon Vijay Abraham I dev_err(dev, "core_ref_clk clock not found\n");
1093091876ccSKishon Vijay Abraham I ret = PTR_ERR(clk);
1094091876ccSKishon Vijay Abraham I return ret;
1095040cbe76SKishon Vijay Abraham I }
1096091876ccSKishon Vijay Abraham I wiz->input_clks[WIZ_CORE_REFCLK] = clk;
1097091876ccSKishon Vijay Abraham I
1098091876ccSKishon Vijay Abraham I rate = clk_get_rate(clk);
1099091876ccSKishon Vijay Abraham I if (rate >= 100000000)
1100091876ccSKishon Vijay Abraham I regmap_field_write(wiz->pma_cmn_refclk_int_mode, 0x1);
1101091876ccSKishon Vijay Abraham I else
1102091876ccSKishon Vijay Abraham I regmap_field_write(wiz->pma_cmn_refclk_int_mode, 0x3);
110386d11e22SRoger Quadros
110486d11e22SRoger Quadros switch (wiz->type) {
110586d11e22SRoger Quadros case AM64_WIZ_10G:
110686d11e22SRoger Quadros case J7200_WIZ_10G:
110786d11e22SRoger Quadros switch (rate) {
110886d11e22SRoger Quadros case REF_CLK_100MHZ:
110986d11e22SRoger Quadros regmap_field_write(wiz->div_sel_field[CMN_REFCLK_DIG_DIV], 0x2);
111086d11e22SRoger Quadros break;
111186d11e22SRoger Quadros case REF_CLK_156_25MHZ:
111286d11e22SRoger Quadros regmap_field_write(wiz->div_sel_field[CMN_REFCLK_DIG_DIV], 0x3);
111386d11e22SRoger Quadros break;
111486d11e22SRoger Quadros default:
111586d11e22SRoger Quadros regmap_field_write(wiz->div_sel_field[CMN_REFCLK_DIG_DIV], 0);
111686d11e22SRoger Quadros break;
111786d11e22SRoger Quadros }
111886d11e22SRoger Quadros break;
111986d11e22SRoger Quadros default:
112086d11e22SRoger Quadros break;
112186d11e22SRoger Quadros }
1122edd473d4SRoger Quadros
1123edd473d4SRoger Quadros if (wiz->data->pma_cmn_refclk1_int_mode) {
1124edd473d4SRoger Quadros clk = devm_clk_get(dev, "core_ref1_clk");
1125edd473d4SRoger Quadros if (IS_ERR(clk)) {
1126edd473d4SRoger Quadros dev_err(dev, "core_ref1_clk clock not found\n");
1127edd473d4SRoger Quadros ret = PTR_ERR(clk);
1128edd473d4SRoger Quadros return ret;
1129edd473d4SRoger Quadros }
1130edd473d4SRoger Quadros wiz->input_clks[WIZ_CORE_REFCLK1] = clk;
1131edd473d4SRoger Quadros
1132edd473d4SRoger Quadros rate = clk_get_rate(clk);
1133edd473d4SRoger Quadros if (rate >= 100000000)
1134edd473d4SRoger Quadros regmap_field_write(wiz->pma_cmn_refclk1_int_mode, 0x1);
1135edd473d4SRoger Quadros else
1136edd473d4SRoger Quadros regmap_field_write(wiz->pma_cmn_refclk1_int_mode, 0x3);
1137edd473d4SRoger Quadros }
1138091876ccSKishon Vijay Abraham I
1139091876ccSKishon Vijay Abraham I clk = devm_clk_get(dev, "ext_ref_clk");
1140091876ccSKishon Vijay Abraham I if (IS_ERR(clk)) {
1141091876ccSKishon Vijay Abraham I dev_err(dev, "ext_ref_clk clock not found\n");
1142091876ccSKishon Vijay Abraham I ret = PTR_ERR(clk);
1143091876ccSKishon Vijay Abraham I return ret;
1144040cbe76SKishon Vijay Abraham I }
1145091876ccSKishon Vijay Abraham I wiz->input_clks[WIZ_EXT_REFCLK] = clk;
1146091876ccSKishon Vijay Abraham I
1147091876ccSKishon Vijay Abraham I rate = clk_get_rate(clk);
1148091876ccSKishon Vijay Abraham I if (rate >= 100000000)
1149091876ccSKishon Vijay Abraham I regmap_field_write(wiz->pma_cmn_refclk_mode, 0x0);
1150091876ccSKishon Vijay Abraham I else
1151091876ccSKishon Vijay Abraham I regmap_field_write(wiz->pma_cmn_refclk_mode, 0x2);
1152edd473d4SRoger Quadros
1153edd473d4SRoger Quadros switch (wiz->type) {
1154edd473d4SRoger Quadros case AM64_WIZ_10G:
1155e27ecef8SMatt Ranostay case J7200_WIZ_10G:
1156bea3ce75SMatt Ranostay case J784S4_WIZ_10G:
1157040cbe76SKishon Vijay Abraham I case J721S2_WIZ_10G:
1158040cbe76SKishon Vijay Abraham I ret = wiz_clock_register(wiz);
1159040cbe76SKishon Vijay Abraham I if (ret)
1160040cbe76SKishon Vijay Abraham I dev_err(dev, "Failed to register wiz clocks\n");
1161edd473d4SRoger Quadros return ret;
1162edd473d4SRoger Quadros default:
1163040cbe76SKishon Vijay Abraham I break;
1164040cbe76SKishon Vijay Abraham I }
1165091876ccSKishon Vijay Abraham I
1166091876ccSKishon Vijay Abraham I for (i = 0; i < WIZ_MUX_NUM_CLOCKS; i++) {
1167091876ccSKishon Vijay Abraham I node_name = clk_mux_sel[i].node_name;
1168091876ccSKishon Vijay Abraham I clk_node = of_get_child_by_name(node, node_name);
1169091876ccSKishon Vijay Abraham I if (!clk_node) {
1170091876ccSKishon Vijay Abraham I dev_err(dev, "Unable to get %s node\n", node_name);
1171091876ccSKishon Vijay Abraham I ret = -EINVAL;
1172091876ccSKishon Vijay Abraham I goto err;
1173091876ccSKishon Vijay Abraham I }
1174040cbe76SKishon Vijay Abraham I
1175091876ccSKishon Vijay Abraham I ret = wiz_mux_of_clk_register(wiz, clk_node, wiz->mux_sel_field[i],
1176091876ccSKishon Vijay Abraham I clk_mux_sel[i].table);
1177091876ccSKishon Vijay Abraham I if (ret) {
1178091876ccSKishon Vijay Abraham I dev_err(dev, "Failed to register %s clock\n",
1179091876ccSKishon Vijay Abraham I node_name);
1180091876ccSKishon Vijay Abraham I of_node_put(clk_node);
1181091876ccSKishon Vijay Abraham I goto err;
1182091876ccSKishon Vijay Abraham I }
1183091876ccSKishon Vijay Abraham I
1184091876ccSKishon Vijay Abraham I of_node_put(clk_node);
1185091876ccSKishon Vijay Abraham I }
1186091876ccSKishon Vijay Abraham I
1187091876ccSKishon Vijay Abraham I for (i = 0; i < wiz->clk_div_sel_num; i++) {
1188091876ccSKishon Vijay Abraham I node_name = clk_div_sel[i].node_name;
1189091876ccSKishon Vijay Abraham I clk_node = of_get_child_by_name(node, node_name);
1190091876ccSKishon Vijay Abraham I if (!clk_node) {
1191091876ccSKishon Vijay Abraham I dev_err(dev, "Unable to get %s node\n", node_name);
1192091876ccSKishon Vijay Abraham I ret = -EINVAL;
1193091876ccSKishon Vijay Abraham I goto err;
1194091876ccSKishon Vijay Abraham I }
1195549cb1aeSKishon Vijay Abraham I
1196091876ccSKishon Vijay Abraham I ret = wiz_div_clk_register(wiz, clk_node, wiz->div_sel_field[i],
1197091876ccSKishon Vijay Abraham I clk_div_sel[i].table);
1198091876ccSKishon Vijay Abraham I if (ret) {
1199091876ccSKishon Vijay Abraham I dev_err(dev, "Failed to register %s clock\n",
1200091876ccSKishon Vijay Abraham I node_name);
1201091876ccSKishon Vijay Abraham I of_node_put(clk_node);
1202091876ccSKishon Vijay Abraham I goto err;
1203091876ccSKishon Vijay Abraham I }
1204091876ccSKishon Vijay Abraham I
1205091876ccSKishon Vijay Abraham I of_node_put(clk_node);
1206091876ccSKishon Vijay Abraham I }
1207091876ccSKishon Vijay Abraham I
1208091876ccSKishon Vijay Abraham I return 0;
1209091876ccSKishon Vijay Abraham I err:
1210091876ccSKishon Vijay Abraham I wiz_clock_cleanup(wiz, node);
1211091876ccSKishon Vijay Abraham I
1212091876ccSKishon Vijay Abraham I return ret;
1213091876ccSKishon Vijay Abraham I }
1214091876ccSKishon Vijay Abraham I
wiz_phy_reset_assert(struct reset_controller_dev * rcdev,unsigned long id)1215091876ccSKishon Vijay Abraham I static int wiz_phy_reset_assert(struct reset_controller_dev *rcdev,
1216091876ccSKishon Vijay Abraham I unsigned long id)
1217091876ccSKishon Vijay Abraham I {
1218091876ccSKishon Vijay Abraham I struct device *dev = rcdev->dev;
1219091876ccSKishon Vijay Abraham I struct wiz *wiz = dev_get_drvdata(dev);
1220091876ccSKishon Vijay Abraham I int ret = 0;
1221091876ccSKishon Vijay Abraham I
1222091876ccSKishon Vijay Abraham I if (id == 0) {
1223091876ccSKishon Vijay Abraham I ret = regmap_field_write(wiz->phy_reset_n, false);
1224091876ccSKishon Vijay Abraham I return ret;
1225091876ccSKishon Vijay Abraham I }
12267ae14cf5SJyri Sarha
1227091876ccSKishon Vijay Abraham I ret = regmap_field_write(wiz->p_enable[id - 1], P_ENABLE_DISABLE);
1228091876ccSKishon Vijay Abraham I return ret;
1229091876ccSKishon Vijay Abraham I }
12306ecac2f8SKishon Vijay Abraham I
wiz_phy_fullrt_div(struct wiz * wiz,int lane)12316ecac2f8SKishon Vijay Abraham I static int wiz_phy_fullrt_div(struct wiz *wiz, int lane)
123227ecdd07SSiddharth Vadapalli {
123327ecdd07SSiddharth Vadapalli switch (wiz->type) {
12346ecac2f8SKishon Vijay Abraham I case AM64_WIZ_10G:
12356ecac2f8SKishon Vijay Abraham I if (wiz->lane_phy_type[lane] == PHY_TYPE_PCIE)
123627ecdd07SSiddharth Vadapalli return regmap_field_write(wiz->p0_fullrt_div[lane], 0x1);
1237*ec318c51SSiddharth Vadapalli break;
1238*ec318c51SSiddharth Vadapalli
123927ecdd07SSiddharth Vadapalli case J721E_WIZ_16G:
1240edd473d4SRoger Quadros case J721E_WIZ_10G:
1241bea3ce75SMatt Ranostay case J7200_WIZ_10G:
124227ecdd07SSiddharth Vadapalli case J721S2_WIZ_10G:
124327ecdd07SSiddharth Vadapalli if (wiz->lane_phy_type[lane] == PHY_TYPE_SGMII)
124427ecdd07SSiddharth Vadapalli return regmap_field_write(wiz->p0_fullrt_div[lane], 0x2);
124527ecdd07SSiddharth Vadapalli break;
124627ecdd07SSiddharth Vadapalli default:
124727ecdd07SSiddharth Vadapalli return 0;
12486ecac2f8SKishon Vijay Abraham I }
12496ecac2f8SKishon Vijay Abraham I return 0;
12506ecac2f8SKishon Vijay Abraham I }
1251091876ccSKishon Vijay Abraham I
wiz_phy_reset_deassert(struct reset_controller_dev * rcdev,unsigned long id)1252091876ccSKishon Vijay Abraham I static int wiz_phy_reset_deassert(struct reset_controller_dev *rcdev,
1253091876ccSKishon Vijay Abraham I unsigned long id)
1254091876ccSKishon Vijay Abraham I {
1255091876ccSKishon Vijay Abraham I struct device *dev = rcdev->dev;
1256091876ccSKishon Vijay Abraham I struct wiz *wiz = dev_get_drvdata(dev);
1257091876ccSKishon Vijay Abraham I int ret;
12589a8a54b9SSinthu Raja
1259c9f9eba0SRoger Quadros if (id == 0) {
12609a8a54b9SSinthu Raja /* if typec-dir gpio was specified, set LN10 SWAP bit based on that */
1261c9f9eba0SRoger Quadros if (wiz->gpio_typec_dir) {
1262c9f9eba0SRoger Quadros if (wiz->typec_dir_delay)
1263c9f9eba0SRoger Quadros msleep_interruptible(wiz->typec_dir_delay);
1264c9f9eba0SRoger Quadros
1265c9f9eba0SRoger Quadros if (gpiod_get_value_cansleep(wiz->gpio_typec_dir))
1266c9f9eba0SRoger Quadros regmap_field_write(wiz->typec_ln10_swap, 1);
1267c9f9eba0SRoger Quadros else
12689a8a54b9SSinthu Raja regmap_field_write(wiz->typec_ln10_swap, 0);
1269494de1ddSSinthu Raja } else {
1270494de1ddSSinthu Raja /* if no typec-dir gpio is specified and PHY type is USB3
1271494de1ddSSinthu Raja * with master lane number is '0' or '2', then set LN10 or
12729a8a54b9SSinthu Raja * LN23 SWAP bit to '1' respectively.
12739a8a54b9SSinthu Raja */
12749a8a54b9SSinthu Raja u32 num_lanes = wiz->num_lanes;
12759a8a54b9SSinthu Raja int i;
12769a8a54b9SSinthu Raja
1277494de1ddSSinthu Raja for (i = 0; i < num_lanes; i++) {
1278494de1ddSSinthu Raja if (wiz->lane_phy_type[i] == PHY_TYPE_USB3) {
1279494de1ddSSinthu Raja switch (wiz->master_lane_num[i]) {
12809a8a54b9SSinthu Raja case LANE0:
1281494de1ddSSinthu Raja regmap_field_write(wiz->typec_ln10_swap, 1);
1282494de1ddSSinthu Raja break;
1283494de1ddSSinthu Raja case LANE2:
1284494de1ddSSinthu Raja regmap_field_write(wiz->typec_ln23_swap, 1);
1285494de1ddSSinthu Raja break;
1286494de1ddSSinthu Raja default:
1287494de1ddSSinthu Raja break;
1288494de1ddSSinthu Raja }
12899a8a54b9SSinthu Raja }
12909a8a54b9SSinthu Raja }
1291c9f9eba0SRoger Quadros }
1292c9f9eba0SRoger Quadros }
1293091876ccSKishon Vijay Abraham I
1294091876ccSKishon Vijay Abraham I if (id == 0) {
1295091876ccSKishon Vijay Abraham I ret = regmap_field_write(wiz->phy_reset_n, true);
1296091876ccSKishon Vijay Abraham I return ret;
1297091876ccSKishon Vijay Abraham I }
12986ecac2f8SKishon Vijay Abraham I
12996ecac2f8SKishon Vijay Abraham I ret = wiz_phy_fullrt_div(wiz, id - 1);
13006ecac2f8SKishon Vijay Abraham I if (ret)
13016ecac2f8SKishon Vijay Abraham I return ret;
13027ae14cf5SJyri Sarha
13037ae14cf5SJyri Sarha if (wiz->lane_phy_type[id - 1] == PHY_TYPE_DP)
13047ae14cf5SJyri Sarha ret = regmap_field_write(wiz->p_enable[id - 1], P_ENABLE);
13057ae14cf5SJyri Sarha else
13067ae14cf5SJyri Sarha ret = regmap_field_write(wiz->p_enable[id - 1], P_ENABLE_FORCE);
1307091876ccSKishon Vijay Abraham I
1308091876ccSKishon Vijay Abraham I return ret;
1309091876ccSKishon Vijay Abraham I }
1310091876ccSKishon Vijay Abraham I
1311091876ccSKishon Vijay Abraham I static const struct reset_control_ops wiz_phy_reset_ops = {
1312091876ccSKishon Vijay Abraham I .assert = wiz_phy_reset_assert,
1313091876ccSKishon Vijay Abraham I .deassert = wiz_phy_reset_deassert,
1314091876ccSKishon Vijay Abraham I };
13155a72122dSRikard Falkeborn
1316091876ccSKishon Vijay Abraham I static const struct regmap_config wiz_regmap_config = {
1317091876ccSKishon Vijay Abraham I .reg_bits = 32,
1318091876ccSKishon Vijay Abraham I .val_bits = 32,
1319091876ccSKishon Vijay Abraham I .reg_stride = 4,
1320091876ccSKishon Vijay Abraham I .fast_io = true,
1321091876ccSKishon Vijay Abraham I };
13224daa43e9SRoger Quadros
13234daa43e9SRoger Quadros static struct wiz_data j721e_16g_data = {
1324edd473d4SRoger Quadros .type = J721E_WIZ_16G,
1325edd473d4SRoger Quadros .pll0_refclk_mux_sel = &pll0_refclk_mux_sel,
13264daa43e9SRoger Quadros .pll1_refclk_mux_sel = &pll1_refclk_mux_sel,
13274daa43e9SRoger Quadros .refclk_dig_sel = &refclk_dig_sel_16g,
13284daa43e9SRoger Quadros .pma_cmn_refclk1_dig_div = &pma_cmn_refclk1_dig_div,
13294daa43e9SRoger Quadros .clk_mux_sel = clk_mux_sel_16g,
13304daa43e9SRoger Quadros .clk_div_sel_num = WIZ_DIV_NUM_CLOCKS_16G,
13314daa43e9SRoger Quadros };
13324daa43e9SRoger Quadros
13334daa43e9SRoger Quadros static struct wiz_data j721e_10g_data = {
1334edd473d4SRoger Quadros .type = J721E_WIZ_10G,
1335edd473d4SRoger Quadros .pll0_refclk_mux_sel = &pll0_refclk_mux_sel,
13364daa43e9SRoger Quadros .pll1_refclk_mux_sel = &pll1_refclk_mux_sel,
13374daa43e9SRoger Quadros .refclk_dig_sel = &refclk_dig_sel_10g,
13384daa43e9SRoger Quadros .clk_mux_sel = clk_mux_sel_10g,
13394daa43e9SRoger Quadros .clk_div_sel_num = WIZ_DIV_NUM_CLOCKS_10G,
13404daa43e9SRoger Quadros };
13414daa43e9SRoger Quadros
13424daa43e9SRoger Quadros static struct wiz_data am64_10g_data = {
1343edd473d4SRoger Quadros .type = AM64_WIZ_10G,
1344edd473d4SRoger Quadros .pll0_refclk_mux_sel = &pll0_refclk_mux_sel,
13454daa43e9SRoger Quadros .pll1_refclk_mux_sel = &pll1_refclk_mux_sel,
13464daa43e9SRoger Quadros .refclk_dig_sel = &refclk_dig_sel_10g,
13474daa43e9SRoger Quadros .clk_mux_sel = clk_mux_sel_10g,
13484daa43e9SRoger Quadros .clk_div_sel_num = WIZ_DIV_NUM_CLOCKS_10G,
13494daa43e9SRoger Quadros };
1350edd473d4SRoger Quadros
1351edd473d4SRoger Quadros static struct wiz_data j7200_pg2_10g_data = {
1352edd473d4SRoger Quadros .type = J7200_WIZ_10G,
1353edd473d4SRoger Quadros .pll0_refclk_mux_sel = &sup_pll0_refclk_mux_sel,
1354edd473d4SRoger Quadros .pll1_refclk_mux_sel = &sup_pll1_refclk_mux_sel,
1355edd473d4SRoger Quadros .refclk_dig_sel = &sup_refclk_dig_sel_10g,
1356edd473d4SRoger Quadros .pma_cmn_refclk1_int_mode = &sup_pma_cmn_refclk1_int_mode,
1357edd473d4SRoger Quadros .clk_mux_sel = clk_mux_sel_10g_2_refclk,
1358edd473d4SRoger Quadros .clk_div_sel_num = WIZ_DIV_NUM_CLOCKS_10G,
1359edd473d4SRoger Quadros };
1360e27ecef8SMatt Ranostay
1361e27ecef8SMatt Ranostay static struct wiz_data j784s4_10g_data = {
1362e27ecef8SMatt Ranostay .type = J784S4_WIZ_10G,
1363e27ecef8SMatt Ranostay .pll0_refclk_mux_sel = &pll0_refclk_mux_sel_2,
1364e27ecef8SMatt Ranostay .pll1_refclk_mux_sel = &pll1_refclk_mux_sel_2,
1365e27ecef8SMatt Ranostay .refclk_dig_sel = &refclk_dig_sel_16g,
1366e27ecef8SMatt Ranostay .pma_cmn_refclk1_int_mode = &pma_cmn_refclk1_int_mode,
1367e27ecef8SMatt Ranostay .clk_mux_sel = clk_mux_sel_10g_2_refclk,
1368e27ecef8SMatt Ranostay .clk_div_sel_num = WIZ_DIV_NUM_CLOCKS_10G,
1369e27ecef8SMatt Ranostay };
1370bea3ce75SMatt Ranostay
1371bea3ce75SMatt Ranostay static struct wiz_data j721s2_10g_data = {
1372bea3ce75SMatt Ranostay .type = J721S2_WIZ_10G,
1373bea3ce75SMatt Ranostay .pll0_refclk_mux_sel = &pll0_refclk_mux_sel,
1374bea3ce75SMatt Ranostay .pll1_refclk_mux_sel = &pll1_refclk_mux_sel,
1375bea3ce75SMatt Ranostay .refclk_dig_sel = &refclk_dig_sel_10g,
1376bea3ce75SMatt Ranostay .clk_mux_sel = clk_mux_sel_10g,
1377bea3ce75SMatt Ranostay .clk_div_sel_num = WIZ_DIV_NUM_CLOCKS_10G,
1378bea3ce75SMatt Ranostay };
1379091876ccSKishon Vijay Abraham I
1380091876ccSKishon Vijay Abraham I static const struct of_device_id wiz_id_table[] = {
13814daa43e9SRoger Quadros {
1382091876ccSKishon Vijay Abraham I .compatible = "ti,j721e-wiz-16g", .data = &j721e_16g_data,
1383091876ccSKishon Vijay Abraham I },
13844daa43e9SRoger Quadros {
1385091876ccSKishon Vijay Abraham I .compatible = "ti,j721e-wiz-10g", .data = &j721e_10g_data,
13866ecac2f8SKishon Vijay Abraham I },
13874daa43e9SRoger Quadros {
13886ecac2f8SKishon Vijay Abraham I .compatible = "ti,am64-wiz-10g", .data = &am64_10g_data,
1389edd473d4SRoger Quadros },
1390edd473d4SRoger Quadros {
1391edd473d4SRoger Quadros .compatible = "ti,j7200-wiz-10g", .data = &j7200_pg2_10g_data,
1392e27ecef8SMatt Ranostay },
1393e27ecef8SMatt Ranostay {
1394e27ecef8SMatt Ranostay .compatible = "ti,j784s4-wiz-10g", .data = &j784s4_10g_data,
1395bea3ce75SMatt Ranostay },
1396bea3ce75SMatt Ranostay {
1397bea3ce75SMatt Ranostay .compatible = "ti,j721s2-wiz-10g", .data = &j721s2_10g_data,
1398091876ccSKishon Vijay Abraham I },
1399091876ccSKishon Vijay Abraham I {}
1400091876ccSKishon Vijay Abraham I };
1401091876ccSKishon Vijay Abraham I MODULE_DEVICE_TABLE(of, wiz_id_table);
14027ae14cf5SJyri Sarha
wiz_get_lane_phy_types(struct device * dev,struct wiz * wiz)14037ae14cf5SJyri Sarha static int wiz_get_lane_phy_types(struct device *dev, struct wiz *wiz)
14047ae14cf5SJyri Sarha {
14057ae14cf5SJyri Sarha struct device_node *serdes, *subnode;
14067ae14cf5SJyri Sarha
14077ae14cf5SJyri Sarha serdes = of_get_child_by_name(dev->of_node, "serdes");
14087ae14cf5SJyri Sarha if (!serdes) {
14097ae14cf5SJyri Sarha dev_err(dev, "%s: Getting \"serdes\"-node failed\n", __func__);
14107ae14cf5SJyri Sarha return -EINVAL;
14117ae14cf5SJyri Sarha }
14127ae14cf5SJyri Sarha
14137ae14cf5SJyri Sarha for_each_child_of_node(serdes, subnode) {
14147ae14cf5SJyri Sarha u32 reg, num_lanes = 1, phy_type = PHY_NONE;
14157ae14cf5SJyri Sarha int ret, i;
14163fb95456SKishon Vijay Abraham I
14173fb95456SKishon Vijay Abraham I if (!(of_node_name_eq(subnode, "phy") ||
14183fb95456SKishon Vijay Abraham I of_node_name_eq(subnode, "link")))
14193fb95456SKishon Vijay Abraham I continue;
14207ae14cf5SJyri Sarha
14217ae14cf5SJyri Sarha ret = of_property_read_u32(subnode, "reg", ®);
1422f26fde36SYang Li if (ret) {
14237ae14cf5SJyri Sarha of_node_put(subnode);
14247ae14cf5SJyri Sarha dev_err(dev,
14257ae14cf5SJyri Sarha "%s: Reading \"reg\" from \"%s\" failed: %d\n",
14267ae14cf5SJyri Sarha __func__, subnode->name, ret);
14277ae14cf5SJyri Sarha return ret;
14287ae14cf5SJyri Sarha }
14297ae14cf5SJyri Sarha of_property_read_u32(subnode, "cdns,num-lanes", &num_lanes);
14307ae14cf5SJyri Sarha of_property_read_u32(subnode, "cdns,phy-type", &phy_type);
14317ae14cf5SJyri Sarha
14327ae14cf5SJyri Sarha dev_dbg(dev, "%s: Lanes %u-%u have phy-type %u\n", __func__,
14337ae14cf5SJyri Sarha reg, reg + num_lanes - 1, phy_type);
14349a8a54b9SSinthu Raja
14359a8a54b9SSinthu Raja for (i = reg; i < reg + num_lanes; i++) {
14367ae14cf5SJyri Sarha wiz->master_lane_num[i] = reg;
14377ae14cf5SJyri Sarha wiz->lane_phy_type[i] = phy_type;
14389a8a54b9SSinthu Raja }
14397ae14cf5SJyri Sarha }
14407ae14cf5SJyri Sarha
14417ae14cf5SJyri Sarha return 0;
14427ae14cf5SJyri Sarha }
1443091876ccSKishon Vijay Abraham I
wiz_probe(struct platform_device * pdev)1444091876ccSKishon Vijay Abraham I static int wiz_probe(struct platform_device *pdev)
1445091876ccSKishon Vijay Abraham I {
1446091876ccSKishon Vijay Abraham I struct reset_controller_dev *phy_reset_dev;
1447091876ccSKishon Vijay Abraham I struct device *dev = &pdev->dev;
1448091876ccSKishon Vijay Abraham I struct device_node *node = dev->of_node;
1449725c7b8dSFaiz Abbas struct platform_device *serdes_pdev;
1450091876ccSKishon Vijay Abraham I bool already_configured = false;
1451091876ccSKishon Vijay Abraham I struct device_node *child_node;
1452091876ccSKishon Vijay Abraham I struct regmap *regmap;
1453091876ccSKishon Vijay Abraham I struct resource res;
1454091876ccSKishon Vijay Abraham I void __iomem *base;
1455725c7b8dSFaiz Abbas struct wiz *wiz;
1456091876ccSKishon Vijay Abraham I int ret, val, i;
14574daa43e9SRoger Quadros u32 num_lanes;
1458091876ccSKishon Vijay Abraham I const struct wiz_data *data;
1459091876ccSKishon Vijay Abraham I
1460091876ccSKishon Vijay Abraham I wiz = devm_kzalloc(dev, sizeof(*wiz), GFP_KERNEL);
1461091876ccSKishon Vijay Abraham I if (!wiz)
1462091876ccSKishon Vijay Abraham I return -ENOMEM;
14634daa43e9SRoger Quadros
14644daa43e9SRoger Quadros data = of_device_get_match_data(dev);
14654daa43e9SRoger Quadros if (!data) {
14664daa43e9SRoger Quadros dev_err(dev, "NULL device data\n");
14674daa43e9SRoger Quadros return -EINVAL;
14684daa43e9SRoger Quadros }
14694daa43e9SRoger Quadros
14704daa43e9SRoger Quadros wiz->data = data;
1471091876ccSKishon Vijay Abraham I wiz->type = data->type;
1472091876ccSKishon Vijay Abraham I
1473091876ccSKishon Vijay Abraham I child_node = of_get_child_by_name(node, "serdes");
1474091876ccSKishon Vijay Abraham I if (!child_node) {
1475091876ccSKishon Vijay Abraham I dev_err(dev, "Failed to get SERDES child DT node\n");
1476091876ccSKishon Vijay Abraham I return -ENODEV;
1477091876ccSKishon Vijay Abraham I }
1478091876ccSKishon Vijay Abraham I
1479091876ccSKishon Vijay Abraham I ret = of_address_to_resource(child_node, 0, &res);
1480091876ccSKishon Vijay Abraham I if (ret) {
1481091876ccSKishon Vijay Abraham I dev_err(dev, "Failed to get memory resource\n");
1482091876ccSKishon Vijay Abraham I goto err_addr_to_resource;
1483091876ccSKishon Vijay Abraham I }
1484091876ccSKishon Vijay Abraham I
1485e2ae8bcaSWei Yongjun base = devm_ioremap(dev, res.start, resource_size(&res));
1486e2ae8bcaSWei Yongjun if (!base) {
1487091876ccSKishon Vijay Abraham I ret = -ENOMEM;
1488e2ae8bcaSWei Yongjun goto err_addr_to_resource;
1489091876ccSKishon Vijay Abraham I }
1490091876ccSKishon Vijay Abraham I
1491091876ccSKishon Vijay Abraham I regmap = devm_regmap_init_mmio(dev, base, &wiz_regmap_config);
1492091876ccSKishon Vijay Abraham I if (IS_ERR(regmap)) {
1493091876ccSKishon Vijay Abraham I dev_err(dev, "Failed to initialize regmap\n");
1494091876ccSKishon Vijay Abraham I ret = PTR_ERR(regmap);
1495091876ccSKishon Vijay Abraham I goto err_addr_to_resource;
1496091876ccSKishon Vijay Abraham I }
1497edd473d4SRoger Quadros
1498edd473d4SRoger Quadros wiz->scm_regmap = syscon_regmap_lookup_by_phandle(node, "ti,scm");
1499edd473d4SRoger Quadros if (IS_ERR(wiz->scm_regmap)) {
1500edd473d4SRoger Quadros if (wiz->type == J7200_WIZ_10G) {
1501a270cc13SDan Carpenter dev_err(dev, "Couldn't get ti,scm regmap\n");
1502a270cc13SDan Carpenter ret = -ENODEV;
1503edd473d4SRoger Quadros goto err_addr_to_resource;
1504edd473d4SRoger Quadros }
1505edd473d4SRoger Quadros
1506edd473d4SRoger Quadros wiz->scm_regmap = NULL;
1507edd473d4SRoger Quadros }
1508091876ccSKishon Vijay Abraham I
1509091876ccSKishon Vijay Abraham I ret = of_property_read_u32(node, "num-lanes", &num_lanes);
1510091876ccSKishon Vijay Abraham I if (ret) {
1511091876ccSKishon Vijay Abraham I dev_err(dev, "Failed to read num-lanes property\n");
1512091876ccSKishon Vijay Abraham I goto err_addr_to_resource;
1513091876ccSKishon Vijay Abraham I }
1514091876ccSKishon Vijay Abraham I
1515091876ccSKishon Vijay Abraham I if (num_lanes > WIZ_MAX_LANES) {
1516e2ae8bcaSWei Yongjun dev_err(dev, "Cannot support %d lanes\n", num_lanes);
1517091876ccSKishon Vijay Abraham I ret = -ENODEV;
1518091876ccSKishon Vijay Abraham I goto err_addr_to_resource;
1519091876ccSKishon Vijay Abraham I }
1520c9f9eba0SRoger Quadros
1521c9f9eba0SRoger Quadros wiz->gpio_typec_dir = devm_gpiod_get_optional(dev, "typec-dir",
1522c9f9eba0SRoger Quadros GPIOD_IN);
1523c9f9eba0SRoger Quadros if (IS_ERR(wiz->gpio_typec_dir)) {
1524c9f9eba0SRoger Quadros ret = PTR_ERR(wiz->gpio_typec_dir);
1525c9f9eba0SRoger Quadros if (ret != -EPROBE_DEFER)
1526c9f9eba0SRoger Quadros dev_err(dev, "Failed to request typec-dir gpio: %d\n",
1527c9f9eba0SRoger Quadros ret);
1528c9f9eba0SRoger Quadros goto err_addr_to_resource;
1529c9f9eba0SRoger Quadros }
1530c9f9eba0SRoger Quadros
1531c9f9eba0SRoger Quadros if (wiz->gpio_typec_dir) {
1532c9f9eba0SRoger Quadros ret = of_property_read_u32(node, "typec-dir-debounce-ms",
1533c9f9eba0SRoger Quadros &wiz->typec_dir_delay);
1534c9f9eba0SRoger Quadros if (ret && ret != -EINVAL) {
1535c9f9eba0SRoger Quadros dev_err(dev, "Invalid typec-dir-debounce property\n");
1536c9f9eba0SRoger Quadros goto err_addr_to_resource;
1537c9f9eba0SRoger Quadros }
1538c9f9eba0SRoger Quadros
1539c9f9eba0SRoger Quadros /* use min. debounce from Type-C spec if not provided in DT */
1540c9f9eba0SRoger Quadros if (ret == -EINVAL)
1541c9f9eba0SRoger Quadros wiz->typec_dir_delay = WIZ_TYPEC_DIR_DEBOUNCE_MIN;
1542c9f9eba0SRoger Quadros
1543c9f9eba0SRoger Quadros if (wiz->typec_dir_delay < WIZ_TYPEC_DIR_DEBOUNCE_MIN ||
1544b8203ec7SYang Li wiz->typec_dir_delay > WIZ_TYPEC_DIR_DEBOUNCE_MAX) {
1545c9f9eba0SRoger Quadros ret = -EINVAL;
1546c9f9eba0SRoger Quadros dev_err(dev, "Invalid typec-dir-debounce property\n");
1547c9f9eba0SRoger Quadros goto err_addr_to_resource;
1548c9f9eba0SRoger Quadros }
1549c9f9eba0SRoger Quadros }
15507ae14cf5SJyri Sarha
15517ae14cf5SJyri Sarha ret = wiz_get_lane_phy_types(dev, wiz);
1552a270cc13SDan Carpenter if (ret)
15537ae14cf5SJyri Sarha goto err_addr_to_resource;
1554091876ccSKishon Vijay Abraham I
1555091876ccSKishon Vijay Abraham I wiz->dev = dev;
1556091876ccSKishon Vijay Abraham I wiz->regmap = regmap;
15574daa43e9SRoger Quadros wiz->num_lanes = num_lanes;
1558091876ccSKishon Vijay Abraham I wiz->clk_mux_sel = data->clk_mux_sel;
15594daa43e9SRoger Quadros wiz->clk_div_sel = clk_div_sel;
1560091876ccSKishon Vijay Abraham I wiz->clk_div_sel_num = data->clk_div_sel_num;
1561091876ccSKishon Vijay Abraham I
1562091876ccSKishon Vijay Abraham I platform_set_drvdata(pdev, wiz);
1563091876ccSKishon Vijay Abraham I
1564091876ccSKishon Vijay Abraham I ret = wiz_regfield_init(wiz);
1565091876ccSKishon Vijay Abraham I if (ret) {
1566091876ccSKishon Vijay Abraham I dev_err(dev, "Failed to initialize regfields\n");
1567091876ccSKishon Vijay Abraham I goto err_addr_to_resource;
1568091876ccSKishon Vijay Abraham I }
1569edd473d4SRoger Quadros
1570edd473d4SRoger Quadros /* Enable supplemental Control override if available */
1571edd473d4SRoger Quadros if (wiz->scm_regmap)
1572edd473d4SRoger Quadros regmap_field_write(wiz->sup_legacy_clk_override, 1);
1573091876ccSKishon Vijay Abraham I
1574091876ccSKishon Vijay Abraham I phy_reset_dev = &wiz->wiz_phy_reset_dev;
1575091876ccSKishon Vijay Abraham I phy_reset_dev->dev = dev;
1576091876ccSKishon Vijay Abraham I phy_reset_dev->ops = &wiz_phy_reset_ops,
1577091876ccSKishon Vijay Abraham I phy_reset_dev->owner = THIS_MODULE,
1578091876ccSKishon Vijay Abraham I phy_reset_dev->of_node = node;
1579091876ccSKishon Vijay Abraham I /* Reset for each of the lane and one for the entire SERDES */
1580091876ccSKishon Vijay Abraham I phy_reset_dev->nr_resets = num_lanes + 1;
1581091876ccSKishon Vijay Abraham I
1582091876ccSKishon Vijay Abraham I ret = devm_reset_controller_register(dev, phy_reset_dev);
1583091876ccSKishon Vijay Abraham I if (ret < 0) {
1584091876ccSKishon Vijay Abraham I dev_warn(dev, "Failed to register reset controller\n");
1585091876ccSKishon Vijay Abraham I goto err_addr_to_resource;
1586091876ccSKishon Vijay Abraham I }
1587091876ccSKishon Vijay Abraham I
1588091876ccSKishon Vijay Abraham I pm_runtime_enable(dev);
1589091876ccSKishon Vijay Abraham I ret = pm_runtime_get_sync(dev);
1590091876ccSKishon Vijay Abraham I if (ret < 0) {
1591091876ccSKishon Vijay Abraham I dev_err(dev, "pm_runtime_get_sync failed\n");
1592091876ccSKishon Vijay Abraham I goto err_get_sync;
1593091876ccSKishon Vijay Abraham I }
1594091876ccSKishon Vijay Abraham I
1595091876ccSKishon Vijay Abraham I ret = wiz_clock_init(wiz, node);
1596091876ccSKishon Vijay Abraham I if (ret < 0) {
1597091876ccSKishon Vijay Abraham I dev_warn(dev, "Failed to initialize clocks\n");
1598091876ccSKishon Vijay Abraham I goto err_get_sync;
1599091876ccSKishon Vijay Abraham I }
1600725c7b8dSFaiz Abbas
1601725c7b8dSFaiz Abbas for (i = 0; i < wiz->num_lanes; i++) {
1602725c7b8dSFaiz Abbas regmap_field_read(wiz->p_enable[i], &val);
1603725c7b8dSFaiz Abbas if (val & (P_ENABLE | P_ENABLE_FORCE)) {
1604725c7b8dSFaiz Abbas already_configured = true;
1605725c7b8dSFaiz Abbas break;
1606725c7b8dSFaiz Abbas }
1607725c7b8dSFaiz Abbas }
1608725c7b8dSFaiz Abbas
1609091876ccSKishon Vijay Abraham I if (!already_configured) {
1610091876ccSKishon Vijay Abraham I ret = wiz_init(wiz);
1611091876ccSKishon Vijay Abraham I if (ret) {
1612091876ccSKishon Vijay Abraham I dev_err(dev, "WIZ initialization failed\n");
1613091876ccSKishon Vijay Abraham I goto err_wiz_init;
1614725c7b8dSFaiz Abbas }
1615091876ccSKishon Vijay Abraham I }
1616f7eb147dSKishon Vijay Abraham I
1617f7eb147dSKishon Vijay Abraham I serdes_pdev = of_platform_device_create(child_node, NULL, dev);
1618f7eb147dSKishon Vijay Abraham I if (!serdes_pdev) {
1619f7eb147dSKishon Vijay Abraham I dev_WARN(dev, "Unable to create SERDES platform device\n");
1620f7eb147dSKishon Vijay Abraham I ret = -ENOMEM;
1621f7eb147dSKishon Vijay Abraham I goto err_wiz_init;
1622f7eb147dSKishon Vijay Abraham I }
1623f7eb147dSKishon Vijay Abraham I wiz->serdes_pdev = serdes_pdev;
1624091876ccSKishon Vijay Abraham I
1625091876ccSKishon Vijay Abraham I of_node_put(child_node);
1626091876ccSKishon Vijay Abraham I return 0;
1627091876ccSKishon Vijay Abraham I
1628091876ccSKishon Vijay Abraham I err_wiz_init:
1629091876ccSKishon Vijay Abraham I wiz_clock_cleanup(wiz, node);
1630091876ccSKishon Vijay Abraham I
1631091876ccSKishon Vijay Abraham I err_get_sync:
1632091876ccSKishon Vijay Abraham I pm_runtime_put(dev);
1633091876ccSKishon Vijay Abraham I pm_runtime_disable(dev);
1634091876ccSKishon Vijay Abraham I
1635091876ccSKishon Vijay Abraham I err_addr_to_resource:
1636091876ccSKishon Vijay Abraham I of_node_put(child_node);
1637091876ccSKishon Vijay Abraham I
1638091876ccSKishon Vijay Abraham I return ret;
1639091876ccSKishon Vijay Abraham I }
1640db850ea9SUwe Kleine-König
wiz_remove(struct platform_device * pdev)1641091876ccSKishon Vijay Abraham I static void wiz_remove(struct platform_device *pdev)
1642091876ccSKishon Vijay Abraham I {
1643091876ccSKishon Vijay Abraham I struct device *dev = &pdev->dev;
1644091876ccSKishon Vijay Abraham I struct device_node *node = dev->of_node;
1645091876ccSKishon Vijay Abraham I struct platform_device *serdes_pdev;
1646091876ccSKishon Vijay Abraham I struct wiz *wiz;
1647091876ccSKishon Vijay Abraham I
1648091876ccSKishon Vijay Abraham I wiz = dev_get_drvdata(dev);
1649091876ccSKishon Vijay Abraham I serdes_pdev = wiz->serdes_pdev;
1650091876ccSKishon Vijay Abraham I
1651091876ccSKishon Vijay Abraham I of_platform_device_destroy(&serdes_pdev->dev, NULL);
1652091876ccSKishon Vijay Abraham I wiz_clock_cleanup(wiz, node);
1653091876ccSKishon Vijay Abraham I pm_runtime_put(dev);
1654091876ccSKishon Vijay Abraham I pm_runtime_disable(dev);
1655091876ccSKishon Vijay Abraham I }
1656091876ccSKishon Vijay Abraham I
1657091876ccSKishon Vijay Abraham I static struct platform_driver wiz_driver = {
1658db850ea9SUwe Kleine-König .probe = wiz_probe,
1659091876ccSKishon Vijay Abraham I .remove_new = wiz_remove,
1660091876ccSKishon Vijay Abraham I .driver = {
1661091876ccSKishon Vijay Abraham I .name = "wiz",
1662091876ccSKishon Vijay Abraham I .of_match_table = wiz_id_table,
1663091876ccSKishon Vijay Abraham I },
1664091876ccSKishon Vijay Abraham I };
1665091876ccSKishon Vijay Abraham I module_platform_driver(wiz_driver);
1666091876ccSKishon Vijay Abraham I
1667091876ccSKishon Vijay Abraham I MODULE_AUTHOR("Texas Instruments Inc.");
1668091876ccSKishon Vijay Abraham I MODULE_DESCRIPTION("TI J721E WIZ driver");
1669 MODULE_LICENSE("GPL v2");
1670