ptp: ptp_clockmatrix: Add wait_for_sys_apll_dpll_lock.
authorVincent Cheng <vincent.cheng.xh@renesas.com>
Wed, 17 Feb 2021 05:42:12 +0000 (00:42 -0500)
committerDavid S. Miller <davem@davemloft.net>
Wed, 17 Feb 2021 21:49:25 +0000 (13:49 -0800)
Part of the device initialization aligns the rising edge of the output
clock to the internal 1 PPS clock. If the system APLL and DPLL is not
locked, then the alignment will fail and there will be a fixed offset
between the internal 1 PPS clock and the output clock.

After loading the device firmware, poll the system APLL and DPLL for
locked state prior to initialization, timing out after 2 seconds.

Signed-off-by: Vincent Cheng <vincent.cheng.xh@renesas.com>
Acked-by: Richard Cochran <richardcochran@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
drivers/ptp/idt8a340_reg.h
drivers/ptp/ptp_clockmatrix.c
drivers/ptp/ptp_clockmatrix.h

index a664dfe5fd2f8ead2725f42f3e65529cf040cb3b..ac524cf0f31fbf710594c9929c958dc90e3a4112 100644 (file)
 #define OTP_SCSR_CONFIG_SELECT            0x0022
 
 #define STATUS                            0xc03c
+#define DPLL_SYS_STATUS                   0x0020
+#define DPLL_SYS_APLL_STATUS              0x0021
 #define USER_GPIO0_TO_7_STATUS            0x008a
 #define USER_GPIO8_TO_15_STATUS           0x008b
 
 /* Bit definitions for the DPLL_CTRL_COMBO_MASTER_CFG register */
 #define COMBO_MASTER_HOLD                 BIT(0)
 
+/* Bit definitions for DPLL_SYS_STATUS register */
+#define DPLL_SYS_STATE_MASK               (0xf)
+
+/* Bit definitions for SYS_APLL_STATUS register */
+#define SYS_APLL_LOSS_LOCK_LIVE_MASK       BIT(0)
+#define SYS_APLL_LOSS_LOCK_LIVE_LOCKED     0
+#define SYS_APLL_LOSS_LOCK_LIVE_UNLOCKED   1
+
 #endif
index 051511f5e1f28e32e093baf0002db3fe1bc68497..9bfd32baad92c01ab345d5064f191b5a9bad9ad5 100644 (file)
@@ -335,6 +335,67 @@ static int wait_for_boot_status_ready(struct idtcm *idtcm)
        return -EBUSY;
 }
 
+static int read_sys_apll_status(struct idtcm *idtcm, u8 *status)
+{
+       return idtcm_read(idtcm, STATUS, DPLL_SYS_APLL_STATUS, status,
+                         sizeof(u8));
+}
+
+static int read_sys_dpll_status(struct idtcm *idtcm, u8 *status)
+{
+       return idtcm_read(idtcm, STATUS, DPLL_SYS_STATUS, status, sizeof(u8));
+}
+
+static int wait_for_sys_apll_dpll_lock(struct idtcm *idtcm)
+{
+       unsigned long timeout = jiffies + msecs_to_jiffies(LOCK_TIMEOUT_MS);
+       u8 apll = 0;
+       u8 dpll = 0;
+       int err;
+
+       do {
+               err = read_sys_apll_status(idtcm, &apll);
+               if (err)
+                       return err;
+
+               err = read_sys_dpll_status(idtcm, &dpll);
+               if (err)
+                       return err;
+
+               apll &= SYS_APLL_LOSS_LOCK_LIVE_MASK;
+               dpll &= DPLL_SYS_STATE_MASK;
+
+               if (apll == SYS_APLL_LOSS_LOCK_LIVE_LOCKED &&
+                   dpll == DPLL_STATE_LOCKED) {
+                       return 0;
+               } else if (dpll == DPLL_STATE_FREERUN ||
+                          dpll == DPLL_STATE_HOLDOVER ||
+                          dpll == DPLL_STATE_OPEN_LOOP) {
+                       dev_warn(&idtcm->client->dev,
+                               "No wait state: DPLL_SYS_STATE %d", dpll);
+                       return -EPERM;
+               }
+
+               msleep(LOCK_POLL_INTERVAL_MS);
+       } while (time_is_after_jiffies(timeout));
+
+       dev_warn(&idtcm->client->dev,
+                "%d ms lock timeout: SYS APLL Loss Lock %d  SYS DPLL state %d",
+                LOCK_TIMEOUT_MS, apll, dpll);
+
+       return -ETIME;
+}
+
+static void wait_for_chip_ready(struct idtcm *idtcm)
+{
+       if (wait_for_boot_status_ready(idtcm))
+               dev_warn(&idtcm->client->dev, "BOOT_STATUS != 0xA0");
+
+       if (wait_for_sys_apll_dpll_lock(idtcm))
+               dev_warn(&idtcm->client->dev,
+                        "Continuing while SYS APLL/DPLL is not locked");
+}
+
 static int _idtcm_gettime(struct idtcm_channel *channel,
                          struct timespec64 *ts)
 {
@@ -2235,8 +2296,7 @@ static int idtcm_probe(struct i2c_client *client,
                dev_warn(&idtcm->client->dev,
                         "loading firmware failed with %d\n", err);
 
-       if (wait_for_boot_status_ready(idtcm))
-               dev_warn(&idtcm->client->dev, "BOOT_STATUS != 0xA0\n");
+       wait_for_chip_ready(idtcm);
 
        if (idtcm->tod_mask) {
                for (i = 0; i < MAX_TOD; i++) {
index 645de2c66b64aa0b4f2ee90e2ec5b89267927787..023323680db7f6a20c400ad2cd42e989e720c230 100644 (file)
@@ -51,6 +51,9 @@
 #define TOD_WRITE_OVERHEAD_COUNT_MAX           (2)
 #define TOD_BYTE_COUNT                         (11)
 
+#define LOCK_TIMEOUT_MS                        (2000)
+#define LOCK_POLL_INTERVAL_MS          (10)
+
 #define PEROUT_ENABLE_OUTPUT_MASK      (0xdeadbeef)
 
 #define IDTCM_MAX_WRITE_COUNT          (512)
@@ -105,6 +108,18 @@ enum scsr_tod_write_type_sel {
        SCSR_TOD_WR_TYPE_SEL_MAX = SCSR_TOD_WR_TYPE_SEL_DELTA_MINUS,
 };
 
+/* Values STATUS.DPLL_SYS_STATUS.DPLL_SYS_STATE */
+enum dpll_state {
+       DPLL_STATE_MIN = 0,
+       DPLL_STATE_FREERUN = DPLL_STATE_MIN,
+       DPLL_STATE_LOCKACQ = 1,
+       DPLL_STATE_LOCKREC = 2,
+       DPLL_STATE_LOCKED = 3,
+       DPLL_STATE_HOLDOVER = 4,
+       DPLL_STATE_OPEN_LOOP = 5,
+       DPLL_STATE_MAX = DPLL_STATE_OPEN_LOOP,
+};
+
 struct idtcm;
 
 struct idtcm_channel {