coreboot
coreboot is an Open Source project aimed at replacing the proprietary BIOS found in most computers.
qmpv3_usb_phy.c
Go to the documentation of this file.
1 /* SPDX-License-Identifier: GPL-2.0-only */
2 
3 #include <timer.h>
4 #include <soc/usb/qmp_usb_phy.h>
5 #include <soc/addressmap.h>
6 
7 /* Only for QMP V3 PHY - QSERDES COM registers */
63 };
64 check_member(usb3_phy_qserdes_com_reg_layout, com_ssc_en_center, 0x010);
69 check_member(usb3_phy_qserdes_com_reg_layout, com_bias_en_clkbuflr_en, 0x034);
71 check_member(usb3_phy_qserdes_com_reg_layout, com_cp_ctrl_mode0, 0x060);
72 check_member(usb3_phy_qserdes_com_reg_layout, com_sysclk_en_sel, 0x080);
73 check_member(usb3_phy_qserdes_com_reg_layout, com_resetsm_ctrl2, 0x08c);
74 check_member(usb3_phy_qserdes_com_reg_layout, com_dec_start_mode0, 0x0b0);
75 check_member(usb3_phy_qserdes_com_reg_layout, com_div_frac_start1_mode0, 0x0b8);
76 check_member(usb3_phy_qserdes_com_reg_layout, com_integloop_gain0_mode0, 0x0d8);
79 check_member(usb3_phy_qserdes_com_reg_layout, com_coreclk_div_mode0, 0x148);
81 check_member(usb3_phy_qserdes_com_reg_layout, com_svs_mode_clk_sel, 0x164);
82 
83 /* Only for QMP V3 PHY - TX registers */
94 };
95 check_member(usb3_phy_qserdes_tx_reg_layout, tx_res_code_lane_offset_tx, 0x044);
96 check_member(usb3_phy_qserdes_tx_reg_layout, tx_res_code_lane_offset_rx, 0x048);
97 check_member(usb3_phy_qserdes_tx_reg_layout, tx_highz_drvr_en, 0x060);
99 check_member(usb3_phy_qserdes_tx_reg_layout, tx_rcv_detect_lvl_2, 0x0a4);
100 
101 /* Only for QMP V3 PHY - RX registers */
127 };
129 check_member(usb3_phy_qserdes_rx_reg_layout, rx_ucdr_so_gain_half, 0x00c);
130 check_member(usb3_phy_qserdes_rx_reg_layout, rx_ucdr_fastlock_fo_gain, 0x030);
131 check_member(usb3_phy_qserdes_rx_reg_layout, rx_ucdr_so_saturtn_and_en, 0x034);
132 check_member(usb3_phy_qserdes_rx_reg_layout, rx_ucdr_pi_cntrls, 0x044);
134 check_member(usb3_phy_qserdes_rx_reg_layout, rx_rx_equ_adap_ctrl2, 0x0d4);
135 check_member(usb3_phy_qserdes_rx_reg_layout, rx_rx_equ_adap_ctrl3, 0x0d8);
136 check_member(usb3_phy_qserdes_rx_reg_layout, rx_rx_equ_adap_ctrl4, 0x0dc);
137 check_member(usb3_phy_qserdes_rx_reg_layout, rx_rx_eq_offset_adap_ctrl1, 0x0f8);
138 check_member(usb3_phy_qserdes_rx_reg_layout, rx_rx_offset_adap_ctrl2, 0x0fc);
139 check_member(usb3_phy_qserdes_rx_reg_layout, rx_sigdet_enables, 0x100);
141 check_member(usb3_phy_qserdes_rx_reg_layout, rx_sigdet_deglitch_ctrl, 0x10c);
144 
145 /* Only for QMP V3 PHY - PCS registers */
202 };
203 check_member(usb3_phy_pcs_reg_layout, pcs_sw_reset, 0x000);
204 check_member(usb3_phy_pcs_reg_layout, pcs_txmgn_v0, 0x00c);
205 check_member(usb3_phy_pcs_reg_layout, pcs_txmgn_v1, 0x010);
206 check_member(usb3_phy_pcs_reg_layout, pcs_txmgn_v2, 0x014);
207 check_member(usb3_phy_pcs_reg_layout, pcs_txmgn_v3, 0x018);
208 check_member(usb3_phy_pcs_reg_layout, pcs_txmgn_v4, 0x01c);
209 check_member(usb3_phy_pcs_reg_layout, pcs_txmgn_ls, 0x020);
210 check_member(usb3_phy_pcs_reg_layout, pcs_txdeemph_m6db_v0, 0x024);
211 check_member(usb3_phy_pcs_reg_layout, pcs_txdeemph_m3p5db_v0, 0x028);
212 check_member(usb3_phy_pcs_reg_layout, pcs_txdeemph_m6db_v1, 0x02c);
213 check_member(usb3_phy_pcs_reg_layout, pcs_txdeemph_m3p5db_v1, 0x030);
214 check_member(usb3_phy_pcs_reg_layout, pcs_txdeemph_m6db_v2, 0x034);
215 check_member(usb3_phy_pcs_reg_layout, pcs_txdeemph_m3p5db_v2, 0x038);
216 check_member(usb3_phy_pcs_reg_layout, pcs_txdeemph_m6db_v3, 0x03c);
217 check_member(usb3_phy_pcs_reg_layout, pcs_txdeemph_m3p5db_v3, 0x040);
218 check_member(usb3_phy_pcs_reg_layout, pcs_txdeemph_m6db_v4, 0x044);
219 check_member(usb3_phy_pcs_reg_layout, pcs_txdeemph_m3p5db_v4, 0x048);
220 check_member(usb3_phy_pcs_reg_layout, pcs_txdeemph_m6db_ls, 0x04c);
221 check_member(usb3_phy_pcs_reg_layout, pcs_txdeemph_m3p5db_ls, 0x050);
222 check_member(usb3_phy_pcs_reg_layout, pcs_rate_slew_cntrl, 0x05c);
223 check_member(usb3_phy_pcs_reg_layout, pcs_power_state_config2, 0x064);
224 check_member(usb3_phy_pcs_reg_layout, pcs_rcvr_dtct_dly_p1u2_l, 0x070);
225 check_member(usb3_phy_pcs_reg_layout, pcs_rcvr_dtct_dly_p1u2_h, 0x074);
226 check_member(usb3_phy_pcs_reg_layout, pcs_rcvr_dtct_dly_u3_l, 0x078);
227 check_member(usb3_phy_pcs_reg_layout, pcs_rcvr_dtct_dly_u3_h, 0x07c);
228 check_member(usb3_phy_pcs_reg_layout, pcs_lock_detect_config1, 0x080);
229 check_member(usb3_phy_pcs_reg_layout, pcs_lock_detect_config2, 0x084);
230 check_member(usb3_phy_pcs_reg_layout, pcs_lock_detect_config3, 0x088);
231 check_member(usb3_phy_pcs_reg_layout, pcs_pwrup_reset_dly_time_auxclk, 0x0a0);
232 check_member(usb3_phy_pcs_reg_layout, pcs_rxeqtraining_wait_time, 0x0b8);
233 check_member(usb3_phy_pcs_reg_layout, pcs_fll_cnt_val_h_tol, 0x0d0);
234 check_member(usb3_phy_pcs_reg_layout, pcs_autonomous_mode_ctrl, 0x0d8);
235 check_member(usb3_phy_pcs_reg_layout, pcs_ready_status, 0x174);
236 check_member(usb3_phy_pcs_reg_layout, pcs_refgen_req_config2, 0x210);
237 
245  (void *)QMP_PHY_PCS_REG_BASE;
246 
247 
248 static const struct qmp_phy_init_tbl qmp_v3_usb3_serdes_tbl[] = {
285 };
286 
287 static const struct qmp_phy_init_tbl qmp_v3_usb3_tx_tbl[] = {
293 };
294 
295 static const struct qmp_phy_init_tbl qmp_v3_usb3_rx_tbl[] = {
309 };
310 
311 static const struct qmp_phy_init_tbl qmp_v3_usb3_pcs_tbl[] = {
312  /* FLL settings */
313  {&pcs_reg_layout->pcs_fll_ctrl2, 0x83},
317  {&pcs_reg_layout->pcs_fll_ctrl1, 0x02},
318 
319  /* Lock Det settings */
324 
326  {&pcs_reg_layout->pcs_txmgn_v0, 0x9f},
327  {&pcs_reg_layout->pcs_txmgn_v1, 0x9f},
328  {&pcs_reg_layout->pcs_txmgn_v2, 0xb7},
329  {&pcs_reg_layout->pcs_txmgn_v3, 0x4e},
330  {&pcs_reg_layout->pcs_txmgn_v4, 0x65},
331  {&pcs_reg_layout->pcs_txmgn_ls, 0x6b},
354 };
355 
358  .serdes_tbl_num = ARRAY_SIZE(qmp_v3_usb3_serdes_tbl),
359  .tx_tbl = qmp_v3_usb3_tx_tbl,
360  .tx_tbl_num = ARRAY_SIZE(qmp_v3_usb3_tx_tbl),
361  .rx_tbl = qmp_v3_usb3_rx_tbl,
362  .rx_tbl_num = ARRAY_SIZE(qmp_v3_usb3_rx_tbl),
363  .pcs_tbl = qmp_v3_usb3_pcs_tbl,
364  .pcs_tbl_num = ARRAY_SIZE(qmp_v3_usb3_pcs_tbl),
365  .qmp_pcs_reg = (void *)QMP_PHY_PCS_REG_BASE,
366 };
367 
368 static void qcom_qmp_phy_configure(const struct qmp_phy_init_tbl tbl[],
369  int num)
370 {
371  int i;
372  const struct qmp_phy_init_tbl *t = tbl;
373 
374  if (!t)
375  return;
376 
377  for (i = 0; i < num; i++, t++)
378  write32(t->address, t->val);
379 }
380 
381 void ss_qmp_phy_init(void)
382 {
383  struct ss_usb_phy_reg *ss_phy_reg;
384 
385  ss_phy_reg = &qmp_v3_usb_phy;
386  /* power up USB3 PHY */
387  write32(&ss_phy_reg->qmp_pcs_reg->pcs_power_down_control, 0x01);
388 
389  /* Serdes configuration */
391  ss_phy_reg->serdes_tbl_num);
392  /* Tx, Rx, and PCS configurations */
393  qcom_qmp_phy_configure(ss_phy_reg->tx_tbl, ss_phy_reg->tx_tbl_num);
394  qcom_qmp_phy_configure(ss_phy_reg->rx_tbl, ss_phy_reg->rx_tbl_num);
395  qcom_qmp_phy_configure(ss_phy_reg->pcs_tbl, ss_phy_reg->pcs_tbl_num);
396 
397  /* perform software reset of PCS/Serdes */
398  write32(&ss_phy_reg->qmp_pcs_reg->pcs_sw_reset, 0x00);
399  /* start PCS/Serdes to operation mode */
400  write32(&ss_phy_reg->qmp_pcs_reg->pcs_start_control, 0x03);
401 
402  /*
403  * Wait for PHY initialization to be done
404  * PCS_STATUS: wait for 1ms for PHY STATUS;
405  * SW can continuously check for PHYSTATUS = 1.b0.
406  */
407  long lock_us = wait_us(10000,
408  !(read32(&ss_phy_reg->qmp_pcs_reg->pcs_ready_status) &
410  if (!lock_us)
411  printk(BIOS_ERR, "QMP PHY PLL LOCK fails:\n");
412  else
413  printk(BIOS_DEBUG, "QMP PHY initialized and locked in %ldus\n",
414  lock_us);
415 }
static void write32(void *addr, uint32_t val)
Definition: mmio.h:40
static uint32_t read32(const void *addr)
Definition: mmio.h:22
#define ARRAY_SIZE(a)
Definition: helpers.h:12
#define printk(level,...)
Definition: stdlib.h:16
#define wait_us(timeout_us, condition)
Definition: timer.h:198
#define BIOS_DEBUG
BIOS_DEBUG - Verbose output.
Definition: loglevel.h:128
#define BIOS_ERR
BIOS_ERR - System in incomplete state.
Definition: loglevel.h:72
#define USB3_PCS_PHYSTATUS
Definition: qmp_usb_phy.h:7
static struct usb3_phy_qserdes_com_reg_layout *const qserdes_com_reg_layout
static void qcom_qmp_phy_configure(const struct qmp_phy_init_tbl tbl[], int num)
void ss_qmp_phy_init(void)
struct ss_usb_phy_reg qmp_v3_usb_phy
static const struct qmp_phy_init_tbl qmp_v3_usb3_rx_tbl[]
static const struct qmp_phy_init_tbl qmp_v3_usb3_serdes_tbl[]
static const struct qmp_phy_init_tbl qmp_v3_usb3_tx_tbl[]
check_member(usb3_phy_qserdes_com_reg_layout, com_ssc_en_center, 0x010)
static const struct qmp_phy_init_tbl qmp_v3_usb3_pcs_tbl[]
static struct usb3_phy_pcs_reg_layout *const pcs_reg_layout
static struct usb3_phy_qserdes_tx_reg_layout *const qserdes_tx_reg_layout
static struct usb3_phy_qserdes_rx_reg_layout *const qserdes_rx_reg_layout
#define QMP_PHY_QSERDES_RX_REG_BASE
Definition: addressmap.h:45
#define QMP_PHY_PCS_REG_BASE
Definition: addressmap.h:46
#define QMP_PHY_QSERDES_COM_REG_BASE
Definition: addressmap.h:43
#define QMP_PHY_QSERDES_TX_REG_BASE
Definition: addressmap.h:44
uint32_t u32
Definition: stdint.h:51
uint8_t u8
Definition: stdint.h:45
struct usb3_phy_pcs_reg_layout * qmp_pcs_reg
Definition: qmp_usb_phy.h:28
const struct qmp_phy_init_tbl * rx_tbl
Definition: qmp_usb_phy.h:22
const struct qmp_phy_init_tbl * serdes_tbl
Definition: qmp_usb_phy.h:16
const struct qmp_phy_init_tbl * tx_tbl
Definition: qmp_usb_phy.h:19
const struct qmp_phy_init_tbl * pcs_tbl
Definition: qmp_usb_phy.h:25