1/*-
2 * SPDX-License-Identifier: BSD-2-Clause
3 *
4 * Copyright (c) 2021, 2022 Soren Schmidt <sos@deepcore.dk>
5 *
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions
8 * are met:
9 * 1. Redistributions of source code must retain the above copyright
10 *    notice, this list of conditions and the following disclaimer.
11 * 2. Redistributions in binary form must reproduce the above copyright
12 *    notice, this list of conditions and the following disclaimer in the
13 *    documentation and/or other materials provided with the distribution.
14 *
15 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
16 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
17 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
18 * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
19 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
20 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
21 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
22 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
23 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
24 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
25 * SUCH DAMAGE.
26 *
27 */
28
29#include <sys/param.h>
30#include <sys/bus.h>
31#include <sys/kernel.h>
32#include <sys/module.h>
33#include <sys/mutex.h>
34#include <sys/rman.h>
35#include <machine/bus.h>
36
37#include <dev/ofw/openfirm.h>
38#include <dev/ofw/ofw_bus.h>
39#include <dev/ofw/ofw_bus_subr.h>
40
41#include <dev/fdt/simple_mfd.h>
42
43#include <dev/clk/clk.h>
44#include <dev/hwreset/hwreset.h>
45#include <dev/regulator/regulator.h>
46#include <dev/syscon/syscon.h>
47#include <dev/phy/phy.h>
48
49#include <contrib/device-tree/include/dt-bindings/phy/phy.h>
50
51#include "syscon_if.h"
52#include "phydev_if.h"
53#include "phynode_if.h"
54
55
56static struct ofw_compat_data compat_data[] = {
57	{"rockchip,rk3568-naneng-combphy",	1},
58	{NULL, 0}
59};
60
61struct rk3568_combphy_softc {
62	device_t	dev;
63	phandle_t	node;
64	struct resource	*mem;
65	struct phynode	*phynode;
66	struct syscon	*pipe_grf;
67	struct syscon	*pipe_phy_grf;
68	clk_t		ref_clk;
69	clk_t		apb_clk;
70	clk_t		pipe_clk;
71	hwreset_t	phy_reset;
72	int		mode;
73};
74
75#define	PHYREG6				0x14
76#define	 PHYREG6_PLL_DIV_MASK			0xc0
77#define	 PHYREG6_PLL_DIV_2			(1 << 6)
78#define	PHYREG7				0x18
79#define	 PHYREG7_TX_RTERM_50OHM			(8 << 4)
80#define	 PHYREG7_RX_RTERM_44OHM			(15 << 0)
81#define	PHYREG8				0x1c
82#define	 PHYREG8_SSC_EN			0x10
83#define	PHYREG11			0x28
84#define	 PHYREG11_SU_TRIM_0_7			0xf0
85#define	PHYREG12			0x2c
86#define	 PHYREG12_PLL_LPF_ADJ_VALUE		4
87#define	PHYREG15			0x38
88#define	 PHYREG15_CTLE_EN			0x01
89#define	 PHYREG15_SSC_CNT_MASK			0xc0
90#define	 PHYREG15_SSC_CNT_VALUE			(1 << 6)
91#define	PHYREG16			0x3c
92#define	 PHYREG16_SSC_CNT_VALUE			0x5f
93#define	PHYREG18			0x44
94#define	 PHYREG18_PLL_LOOP			0x32
95#define	PHYREG32			0x7c
96#define	 PHYREG32_SSC_MASK			0xf0
97#define	 PHYREG32_SSC_UPWARD			(0 << 4)
98#define	 PHYREG32_SSC_DOWNWARD			(1 << 4)
99#define	 PHYREG32_SSC_OFFSET_500PPM		(1 << 6)
100#define	PHYREG33			0x80
101#define	 PHYREG33_PLL_KVCO_MASK			0x1c
102#define	 PHYREG33_PLL_KVCO_VALUE		(2 << 2)
103
104#define	PIPE_MASK_ALL			(0xffff << 16)
105#define	PIPE_PHY_GRF_PIPE_CON0		0x00
106#define	 PIPE_DATABUSWIDTH_MASK			0x3
107#define	 PIPE_DATABUSWIDTH_32BIT		0
108#define	 PIPE_DATABUSWIDTH_16BIT		1
109#define	 PIPE_PHYMODE_MASK			(3 << 2)
110#define	 PIPE_PHYMODE_PCIE			(0 << 2)
111#define	 PIPE_PHYMODE_USB3			(1 << 2)
112#define	 PIPE_PHYMODE_SATA			(2 << 2)
113#define	 PIPE_RATE_MASK				(3 << 4)
114#define	 PIPE_RATE_PCIE_2_5GBPS			(0 << 4)
115#define	 PIPE_RATE_PCIE_5GBPS			(1 << 4)
116#define	 PIPE_RATE_USB3_5GBPS			(0 << 4)
117#define	 PIPE_RATE_SATA_1GBPS5			(0 << 4)
118#define	 PIPE_RATE_SATA_3GBPS			(1 << 4)
119#define	 PIPE_RATE_SATA_6GBPS			(2 << 4)
120#define	 PIPE_MAC_PCLKREQ_N			(1 << 8)
121#define	 PIPE_L1SUB_ENTREQ			(1 << 9)
122#define	 PIPE_RXTERM				(1 << 12)
123#define	PIPE_PHY_GRF_PIPE_CON1		0x04
124#define	 PHY_CLK_SEL_MASK			(3 << 13)
125#define	 PHY_CLK_SEL_24M			(0 << 13)
126#define	 PHY_CLK_SEL_25M			(1 << 13)
127#define	 PHY_CLK_SEL_100M			(2 << 13)
128#define	PIPE_PHY_GRF_PIPE_CON2		0x08
129#define	 SEL_PIPE_TXCOMPLIANCE_I		(1 << 15)
130#define	 SEL_PIPE_TXELECIDLE			(1 << 12)
131#define	 SEL_PIPE_RXTERM			(1 << 8)
132#define	 SEL_PIPE_BYPASS_CODEC			(1 << 7)
133#define	 SEL_PIPE_PIPE_EBUF			(1 << 6)
134#define	 SEL_PIPE_PIPE_PHYMODE			(1 << 1)
135#define	 SEL_PIPE_DATABUSWIDTH			(1 << 0)
136#define	PIPE_PHY_GRF_PIPE_CON3		0x0c
137#define	 PIPE_SEL_MASK				(3 << 13)
138#define	 PIPE_SEL_PCIE				(0 << 13)
139#define	 PIPE_SEL_USB3				(1 << 13)
140#define	 PIPE_SEL_SATA				(2 << 13)
141#define	 PIPE_CLK_REF_SRC_I_MASK		(3 << 8)
142#define	 PIPE_CLK_REF_SRC_I_PLL_CKREF_INNER	(2 << 8)
143#define	 PIPE_RXELECIDLE			(1 << 10)
144#define	 PIPE_FROM_PCIE_IO			(1 << 11)
145
146#define	PIPE_GRF_PIPE_CON0		0x00
147#define	 SATA2_PHY_SPDMODE_1GBPS5		(0 << 12)
148#define	 SATA2_PHY_SPDMODE_3GBPS		(1 << 12)
149#define	 SATA2_PHY_SPDMODE_6GBPS		(2 << 12)
150#define	 SATA1_PHY_SPDMODE_1GBPS5		(0 << 8)
151#define	 SATA1_PHY_SPDMODE_3GBPS		(1 << 8)
152#define	 SATA1_PHY_SPDMODE_6GBPS		(2 << 8)
153#define	 SATA0_PHY_SPDMODE_1GBPS5		(0 << 4)
154#define	 SATA0_PHY_SPDMODE_3GBPS		(1 << 4)
155#define	 SATA0_PHY_SPDMODE_6GBPS		(2 << 4)
156
157#define	PIPE_GRF_SATA_CON0		0x10
158#define	PIPE_GRF_SATA_CON1		0x14
159#define	PIPE_GRF_SATA_CON2		0x18
160#define	PIPE_GRF_XPCS_CON0		0x40
161
162
163/* PHY class and methods */
164static int
165rk3568_combphy_enable(struct phynode *phynode, bool enable)
166{
167	device_t dev = phynode_get_device(phynode);
168	struct rk3568_combphy_softc *sc = device_get_softc(dev);
169	uint64_t rate;
170
171	if (enable == false)
172		return (0);
173
174	switch (sc->mode) {
175	case PHY_TYPE_SATA:
176		device_printf(dev, "configuring for SATA");
177
178		/* tx_rterm 50 ohm & rx_rterm 44 ohm */
179		bus_write_4(sc->mem, PHYREG7,
180		    PHYREG7_TX_RTERM_50OHM | PHYREG7_RX_RTERM_44OHM);
181
182		/* Adaptive CTLE */
183		bus_write_4(sc->mem, PHYREG15,
184		    bus_read_4(sc->mem, PHYREG15) | PHYREG15_CTLE_EN);
185
186		/* config grf_pipe for PCIe */
187		SYSCON_WRITE_4(sc->pipe_phy_grf, PIPE_PHY_GRF_PIPE_CON3,
188		    PIPE_MASK_ALL | PIPE_SEL_SATA | PIPE_RXELECIDLE | 0x7);
189
190		SYSCON_WRITE_4(sc->pipe_phy_grf, PIPE_PHY_GRF_PIPE_CON2,
191		    PIPE_MASK_ALL | SEL_PIPE_TXCOMPLIANCE_I |
192		    SEL_PIPE_DATABUSWIDTH | 0xc3);
193
194		SYSCON_WRITE_4(sc->pipe_phy_grf, PIPE_PHY_GRF_PIPE_CON0,
195		    PIPE_MASK_ALL | PIPE_RXTERM | PIPE_DATABUSWIDTH_16BIT |
196		    PIPE_RATE_SATA_3GBPS | PIPE_PHYMODE_SATA);
197
198		SYSCON_WRITE_4(sc->pipe_grf, PIPE_GRF_PIPE_CON0,
199		    PIPE_MASK_ALL | SATA0_PHY_SPDMODE_6GBPS |
200		    SATA1_PHY_SPDMODE_6GBPS | SATA2_PHY_SPDMODE_6GBPS);
201		break;
202
203	case PHY_TYPE_PCIE:
204		device_printf(dev, "configuring for PCIe");
205
206		/* Set SSC downward spread spectrum */
207		bus_write_4(sc->mem, PHYREG32,
208		    (bus_read_4(sc->mem, PHYREG32) & PHYREG32_SSC_MASK) |
209		    PHYREG32_SSC_DOWNWARD);
210
211		/* config grf_pipe for PCIe */
212		SYSCON_WRITE_4(sc->pipe_phy_grf, PIPE_PHY_GRF_PIPE_CON3,
213		    PIPE_MASK_ALL | PIPE_SEL_PCIE |
214		    PIPE_CLK_REF_SRC_I_PLL_CKREF_INNER);
215		SYSCON_WRITE_4(sc->pipe_phy_grf, PIPE_PHY_GRF_PIPE_CON2,
216		    PIPE_MASK_ALL | SEL_PIPE_RXTERM | SEL_PIPE_DATABUSWIDTH);
217		SYSCON_WRITE_4(sc->pipe_phy_grf, PIPE_PHY_GRF_PIPE_CON0,
218		    PIPE_MASK_ALL | PIPE_RXTERM | PIPE_DATABUSWIDTH_32BIT |
219		    PIPE_RATE_PCIE_2_5GBPS | PIPE_PHYMODE_PCIE);
220		break;
221
222	case PHY_TYPE_USB3:
223		device_printf(dev, "configuring for USB3");
224
225		/* Set SSC downward spread spectrum */
226		bus_write_4(sc->mem, PHYREG32,
227		    (bus_read_4(sc->mem, PHYREG32) & PHYREG32_SSC_MASK) |
228		    PHYREG32_SSC_DOWNWARD);
229
230		/* Adaptive CTLE */
231		bus_write_4(sc->mem, PHYREG15,
232		    bus_read_4(sc->mem, PHYREG15) | PHYREG15_CTLE_EN);
233
234		/* Set PLL KVCO fine tuning signals */
235		bus_write_4(sc->mem, PHYREG33,
236		    (bus_read_4(sc->mem, PHYREG33) & PHYREG33_PLL_KVCO_MASK) |
237		    PHYREG33_PLL_KVCO_VALUE);
238
239		/* Enable controlling random jitter. */
240		bus_write_4(sc->mem, PHYREG12, PHYREG12_PLL_LPF_ADJ_VALUE);
241
242		/* Set PLL input clock divider 1/2 */
243		bus_write_4(sc->mem, PHYREG6,
244		    (bus_read_4(sc->mem, PHYREG6) & PHYREG6_PLL_DIV_MASK) |
245		    PHYREG6_PLL_DIV_2);
246
247		/* Set PLL loop divider */
248		bus_write_4(sc->mem, PHYREG18, PHYREG18_PLL_LOOP);
249
250		/* Set PLL LPF R1 to su_trim[0:7] */
251		bus_write_4(sc->mem, PHYREG11, PHYREG11_SU_TRIM_0_7);
252
253		/* config grf_pipe for USB3 */
254		SYSCON_WRITE_4(sc->pipe_phy_grf, PIPE_PHY_GRF_PIPE_CON3,
255		    PIPE_MASK_ALL | PIPE_SEL_USB3);
256		SYSCON_WRITE_4(sc->pipe_phy_grf, PIPE_PHY_GRF_PIPE_CON2,
257		    PIPE_MASK_ALL);
258		SYSCON_WRITE_4(sc->pipe_phy_grf, PIPE_PHY_GRF_PIPE_CON0,
259		    PIPE_MASK_ALL | PIPE_DATABUSWIDTH_16BIT |
260		    PIPE_PHYMODE_USB3 | PIPE_RATE_USB3_5GBPS);
261		break;
262
263	default:
264		printf("Unsupported mode=%d\n", sc->mode);
265		return (-1);
266	}
267
268	clk_get_freq(sc->ref_clk, &rate);
269	printf(" ref_clk=%lu\n", rate);
270
271	switch (rate) {
272	case 24000000:
273		SYSCON_WRITE_4(sc->pipe_phy_grf, PIPE_PHY_GRF_PIPE_CON1,
274		    (PHY_CLK_SEL_MASK << 16) | PHY_CLK_SEL_24M);
275
276		if (sc->mode == PHY_TYPE_USB3 || sc->mode == PHY_TYPE_SATA) {
277			/* Adaptive CTLE */
278			bus_write_4(sc->mem, PHYREG15,
279			    (bus_read_4(sc->mem, PHYREG15) &
280			    PHYREG15_SSC_CNT_MASK) | PHYREG15_SSC_CNT_VALUE);
281
282			/* SSC control period */
283			bus_write_4(sc->mem, PHYREG16, PHYREG16_SSC_CNT_VALUE);
284		}
285		break;
286
287	case 25000000:
288		SYSCON_WRITE_4(sc->pipe_phy_grf, PIPE_PHY_GRF_PIPE_CON1,
289		    (PHY_CLK_SEL_MASK << 16) | PHY_CLK_SEL_25M);
290		break;
291
292	case 100000000:
293		SYSCON_WRITE_4(sc->pipe_phy_grf, PIPE_PHY_GRF_PIPE_CON1,
294		    (PHY_CLK_SEL_MASK << 16) | PHY_CLK_SEL_100M);
295
296		if (sc->mode == PHY_TYPE_PCIE) {
297			/* Set PLL KVCO fine tuning signals */
298			bus_write_4(sc->mem, PHYREG33,
299			    (bus_read_4(sc->mem, PHYREG33) &
300			    PHYREG33_PLL_KVCO_MASK) | PHYREG33_PLL_KVCO_VALUE);
301
302			/* Enable controlling random jitter. */
303			bus_write_4(sc->mem, PHYREG12,
304			    PHYREG12_PLL_LPF_ADJ_VALUE);
305
306			/* Set PLL input clock divider 1/2 */
307			bus_write_4(sc->mem, PHYREG6,
308			    (bus_read_4(sc->mem, PHYREG6) &
309			    PHYREG6_PLL_DIV_MASK) | PHYREG6_PLL_DIV_2);
310
311			/* Set PLL loop divider */
312			bus_write_4(sc->mem, PHYREG18, PHYREG18_PLL_LOOP);
313
314			/* Set PLL LPF R1 to su_trim[0:7] */
315			bus_write_4(sc->mem, PHYREG11, PHYREG11_SU_TRIM_0_7);
316		}
317		if (sc->mode == PHY_TYPE_SATA) {
318			/* Set SSC downward spread spectrum */
319			bus_write_4(sc->mem, PHYREG32,
320			    (bus_read_4(sc->mem, PHYREG32) & ~0x000000f0) |
321			    PHYREG32_SSC_DOWNWARD | PHYREG32_SSC_OFFSET_500PPM);
322		}
323		break;
324
325	default:
326		device_printf(dev, "unknown ref rate=%lu\n", rate);
327		break;
328	}
329
330	if (OF_hasprop(sc->node, "rockchip,ext-refclk")) {
331		device_printf(dev, "UNSUPPORTED rockchip,ext-refclk\n");
332	}
333	if (OF_hasprop(sc->node, "rockchip,enable-ssc")) {
334		device_printf(dev, "setting rockchip,enable-ssc\n");
335		bus_write_4(sc->mem, PHYREG8,
336		    bus_read_4(sc->mem, PHYREG8) | PHYREG8_SSC_EN);
337	}
338
339	if (hwreset_deassert(sc->phy_reset))
340		device_printf(dev, "phy_reset failed to clear\n");
341
342	return (0);
343}
344
345static phynode_method_t rk3568_combphy_phynode_methods[] = {
346	PHYNODEMETHOD(phynode_enable,	rk3568_combphy_enable),
347
348	PHYNODEMETHOD_END
349};
350DEFINE_CLASS_1(rk3568_combphy_phynode, rk3568_combphy_phynode_class,
351    rk3568_combphy_phynode_methods, 0, phynode_class);
352
353
354/* Device class and methods */
355static int
356rk3568_combphy_probe(device_t dev)
357{
358
359	if (!ofw_bus_status_okay(dev))
360		return (ENXIO);
361	if (ofw_bus_search_compatible(dev, compat_data)->ocd_data == 0)
362		return (ENXIO);
363	device_set_desc(dev, "RockChip combo PHY");
364	return (BUS_PROBE_DEFAULT);
365}
366
367static int
368rk3568_combphy_attach(device_t dev)
369{
370	struct rk3568_combphy_softc *sc = device_get_softc(dev);
371	struct phynode_init_def phy_init;
372	struct phynode *phynode;
373	int rid = 0;
374
375	sc->dev = dev;
376	sc->node = ofw_bus_get_node(dev);
377
378	/* Get memory resource */
379	if (!(sc->mem = bus_alloc_resource_any(dev, SYS_RES_MEMORY,
380	    &rid, RF_ACTIVE))) {
381		device_printf(dev, "Cannot allocate memory resources\n");
382		return (ENXIO);
383	}
384
385	/* Get syncons handles */
386	if (OF_hasprop(sc->node, "rockchip,pipe-grf") &&
387	    syscon_get_by_ofw_property(dev, sc->node, "rockchip,pipe-grf",
388	    &sc->pipe_grf))
389		return (ENXIO);
390	if (OF_hasprop(sc->node, "rockchip,pipe-phy-grf") &&
391	    syscon_get_by_ofw_property(dev, sc->node, "rockchip,pipe-phy-grf",
392	    &sc->pipe_phy_grf))
393		return (ENXIO);
394
395	/* Get & enable clocks */
396	if (clk_get_by_ofw_name(dev, 0, "ref", &sc->ref_clk)) {
397		device_printf(dev, "getting ref failed\n");
398		return (ENXIO);
399	}
400	if (clk_enable(sc->ref_clk))
401		device_printf(dev, "enable ref failed\n");
402	if (clk_get_by_ofw_name(dev, 0, "apb", &sc->apb_clk)) {
403		device_printf(dev, "getting apb failed\n");
404		return (ENXIO);
405	}
406	if (clk_enable(sc->apb_clk))
407		device_printf(dev, "enable apb failed\n");
408	if (clk_get_by_ofw_name(dev, 0, "pipe", &sc->pipe_clk)) {
409		device_printf(dev, "getting pipe failed\n");
410		return (ENXIO);
411	}
412	if (clk_enable(sc->pipe_clk))
413		device_printf(dev, "enable pipe failed\n");
414
415	/* get & assert reset */
416	if (hwreset_get_by_ofw_idx(dev, sc->node, 0, &sc->phy_reset)) {
417		device_printf(dev, "Cannot get reset\n");
418		return (ENXIO);
419	}
420	hwreset_assert(sc->phy_reset);
421
422	bzero(&phy_init, sizeof(phy_init));
423	phy_init.id = 0;
424	phy_init.ofw_node = sc->node;
425	if (!(phynode = phynode_create(dev, &rk3568_combphy_phynode_class,
426	    &phy_init))) {
427		device_printf(dev, "failed to create combphy PHY\n");
428		return (ENXIO);
429	}
430	if (!phynode_register(phynode)) {
431		device_printf(dev, "failed to register combphy PHY\n");
432		return (ENXIO);
433	}
434	sc->phynode = phynode;
435	sc->mode = 0;
436
437	return (0);
438}
439
440static int
441rk3568_combphy_map(device_t dev, phandle_t xref, int ncells, pcell_t *cells,
442    intptr_t *id)
443{
444	struct rk3568_combphy_softc *sc = device_get_softc(dev);
445
446	if (phydev_default_ofw_map(dev, xref, ncells, cells, id))
447		return (ERANGE);
448
449	/* Store the phy mode that is handed to us in id */
450	sc->mode = *id;
451
452	/* Set our id to 0 so the std phy_get_*() works as usual */
453	*id = 0;
454
455	return (0);
456}
457
458static device_method_t rk3568_combphy_methods[] = {
459	DEVMETHOD(device_probe,		rk3568_combphy_probe),
460	DEVMETHOD(device_attach,	rk3568_combphy_attach),
461	DEVMETHOD(phydev_map,		rk3568_combphy_map),
462
463	DEVMETHOD_END
464};
465
466DEFINE_CLASS_1(rk3568_combphy, rk3568_combphy_driver, rk3568_combphy_methods,
467    sizeof(struct simple_mfd_softc), simple_mfd_driver);
468EARLY_DRIVER_MODULE(rk3568_combphy, simplebus, rk3568_combphy_driver,
469    0, 0, BUS_PASS_RESOURCE + BUS_PASS_ORDER_LATE);
470