xref: /openbmc/linux/drivers/phy/ti/phy-j721e-wiz.c (revision 1ac731c529cd4d6adbce134754b51ff7d822b145)
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", &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