| // SPDX-License-Identifier: GPL-2.0+ |
| /* |
| * PTP hardware clock driver for the FemtoClock3 family of timing and |
| * synchronization devices. |
| * |
| * Copyright (C) 2023 Integrated Device Technology, Inc., a Renesas Company. |
| */ |
| #include <linux/firmware.h> |
| #include <linux/platform_device.h> |
| #include <linux/module.h> |
| #include <linux/ptp_clock_kernel.h> |
| #include <linux/delay.h> |
| #include <linux/jiffies.h> |
| #include <linux/kernel.h> |
| #include <linux/timekeeping.h> |
| #include <linux/string.h> |
| #include <linux/of.h> |
| #include <linux/bitfield.h> |
| #include <linux/mfd/rsmu.h> |
| #include <linux/mfd/idtRC38xxx_reg.h> |
| #include <linux/unaligned.h> |
| |
| #include "ptp_private.h" |
| #include "ptp_fc3.h" |
| |
| MODULE_DESCRIPTION("Driver for IDT FemtoClock3(TM) family"); |
| MODULE_AUTHOR("IDT support-1588 <IDT-support-1588@lm.renesas.com>"); |
| MODULE_VERSION("1.0"); |
| MODULE_LICENSE("GPL"); |
| |
| /* |
| * The name of the firmware file to be loaded |
| * over-rides any automatic selection |
| */ |
| static char *firmware; |
| module_param(firmware, charp, 0); |
| |
| static s64 ns2counters(struct idtfc3 *idtfc3, s64 nsec, u32 *sub_ns) |
| { |
| s64 sync; |
| s32 rem; |
| |
| if (likely(nsec >= 0)) { |
| sync = div_u64_rem(nsec, idtfc3->ns_per_sync, &rem); |
| *sub_ns = rem; |
| } else { |
| sync = -div_u64_rem(-nsec - 1, idtfc3->ns_per_sync, &rem) - 1; |
| *sub_ns = idtfc3->ns_per_sync - rem - 1; |
| } |
| |
| return sync * idtfc3->ns_per_sync; |
| } |
| |
| static s64 tdc_meas2offset(struct idtfc3 *idtfc3, u64 meas_read) |
| { |
| s64 coarse, fine; |
| |
| fine = sign_extend64(FIELD_GET(FINE_MEAS_MASK, meas_read), 12); |
| coarse = sign_extend64(FIELD_GET(COARSE_MEAS_MASK, meas_read), (39 - 13)); |
| |
| fine = div64_s64(fine * NSEC_PER_SEC, idtfc3->tdc_apll_freq * 62LL); |
| coarse = div64_s64(coarse * NSEC_PER_SEC, idtfc3->time_ref_freq); |
| |
| return coarse + fine; |
| } |
| |
| static s64 tdc_offset2phase(struct idtfc3 *idtfc3, s64 offset_ns) |
| { |
| if (offset_ns > idtfc3->ns_per_sync / 2) |
| offset_ns -= idtfc3->ns_per_sync; |
| |
| return offset_ns * idtfc3->tdc_offset_sign; |
| } |
| |
| static int idtfc3_set_lpf_mode(struct idtfc3 *idtfc3, u8 mode) |
| { |
| int err; |
| |
| if (mode >= LPF_INVALID) |
| return -EINVAL; |
| |
| if (idtfc3->lpf_mode == mode) |
| return 0; |
| |
| err = regmap_bulk_write(idtfc3->regmap, LPF_MODE_CNFG, &mode, sizeof(mode)); |
| if (err) |
| return err; |
| |
| idtfc3->lpf_mode = mode; |
| |
| return 0; |
| } |
| |
| static int idtfc3_enable_lpf(struct idtfc3 *idtfc3, bool enable) |
| { |
| u8 val; |
| int err; |
| |
| err = regmap_bulk_read(idtfc3->regmap, LPF_CTRL, &val, sizeof(val)); |
| if (err) |
| return err; |
| |
| if (enable == true) |
| val |= LPF_EN; |
| else |
| val &= ~LPF_EN; |
| |
| return regmap_bulk_write(idtfc3->regmap, LPF_CTRL, &val, sizeof(val)); |
| } |
| |
| static int idtfc3_get_time_ref_freq(struct idtfc3 *idtfc3) |
| { |
| int err; |
| u8 buf[4]; |
| u8 time_ref_div; |
| u8 time_clk_div; |
| |
| err = regmap_bulk_read(idtfc3->regmap, TIME_CLOCK_MEAS_DIV_CNFG, buf, sizeof(buf)); |
| if (err) |
| return err; |
| time_ref_div = FIELD_GET(TIME_REF_DIV_MASK, get_unaligned_le32(buf)) + 1; |
| |
| err = regmap_bulk_read(idtfc3->regmap, TIME_CLOCK_COUNT, buf, 1); |
| if (err) |
| return err; |
| time_clk_div = (buf[0] & TIME_CLOCK_COUNT_MASK) + 1; |
| idtfc3->time_ref_freq = idtfc3->hw_param.time_clk_freq * |
| time_clk_div / time_ref_div; |
| |
| return 0; |
| } |
| |
| static int idtfc3_get_tdc_offset_sign(struct idtfc3 *idtfc3) |
| { |
| int err; |
| u8 buf[4]; |
| u32 val; |
| u8 sig1, sig2; |
| |
| err = regmap_bulk_read(idtfc3->regmap, TIME_CLOCK_TDC_FANOUT_CNFG, buf, sizeof(buf)); |
| if (err) |
| return err; |
| |
| val = get_unaligned_le32(buf); |
| if ((val & TIME_SYNC_TO_TDC_EN) != TIME_SYNC_TO_TDC_EN) { |
| dev_err(idtfc3->dev, "TIME_SYNC_TO_TDC_EN is off !!!"); |
| return -EINVAL; |
| } |
| |
| sig1 = FIELD_GET(SIG1_MUX_SEL_MASK, val); |
| sig2 = FIELD_GET(SIG2_MUX_SEL_MASK, val); |
| |
| if ((sig1 == sig2) || ((sig1 != TIME_SYNC) && (sig2 != TIME_SYNC))) { |
| dev_err(idtfc3->dev, "Invalid tdc_mux_sel sig1=%d sig2=%d", sig1, sig2); |
| return -EINVAL; |
| } else if (sig1 == TIME_SYNC) { |
| idtfc3->tdc_offset_sign = 1; |
| } else if (sig2 == TIME_SYNC) { |
| idtfc3->tdc_offset_sign = -1; |
| } |
| |
| return 0; |
| } |
| |
| static int idtfc3_lpf_bw(struct idtfc3 *idtfc3, u8 shift, u8 mult) |
| { |
| u8 val = FIELD_PREP(LPF_BW_SHIFT, shift) | FIELD_PREP(LPF_BW_MULT, mult); |
| |
| return regmap_bulk_write(idtfc3->regmap, LPF_BW_CNFG, &val, sizeof(val)); |
| } |
| |
| static int idtfc3_enable_tdc(struct idtfc3 *idtfc3, bool enable, u8 meas_mode) |
| { |
| int err; |
| u8 val = 0; |
| |
| /* Disable TDC first */ |
| err = regmap_bulk_write(idtfc3->regmap, TIME_CLOCK_MEAS_CTRL, &val, sizeof(val)); |
| if (err) |
| return err; |
| |
| if (enable == false) |
| return idtfc3_lpf_bw(idtfc3, LPF_BW_SHIFT_DEFAULT, LPF_BW_MULT_DEFAULT); |
| |
| if (meas_mode >= MEAS_MODE_INVALID) |
| return -EINVAL; |
| |
| /* Change TDC meas mode */ |
| err = regmap_bulk_write(idtfc3->regmap, TIME_CLOCK_MEAS_CNFG, |
| &meas_mode, sizeof(meas_mode)); |
| if (err) |
| return err; |
| |
| /* Enable TDC */ |
| val = TDC_MEAS_EN; |
| if (meas_mode == CONTINUOUS) |
| val |= TDC_MEAS_START; |
| err = regmap_bulk_write(idtfc3->regmap, TIME_CLOCK_MEAS_CTRL, &val, sizeof(val)); |
| if (err) |
| return err; |
| |
| return idtfc3_lpf_bw(idtfc3, LPF_BW_SHIFT_1PPS, LPF_BW_MULT_DEFAULT); |
| } |
| |
| static bool get_tdc_meas(struct idtfc3 *idtfc3, s64 *offset_ns) |
| { |
| bool valid = false; |
| u8 buf[9]; |
| u8 val; |
| int err; |
| |
| while (true) { |
| err = regmap_bulk_read(idtfc3->regmap, TDC_FIFO_STS, |
| &val, sizeof(val)); |
| if (err) |
| return false; |
| |
| if (val & FIFO_EMPTY) |
| break; |
| |
| err = regmap_bulk_read(idtfc3->regmap, TDC_FIFO_READ_REQ, |
| &buf, sizeof(buf)); |
| if (err) |
| return false; |
| |
| valid = true; |
| } |
| |
| if (valid) |
| *offset_ns = tdc_meas2offset(idtfc3, get_unaligned_le64(&buf[1])); |
| |
| return valid; |
| } |
| |
| static int check_tdc_fifo_overrun(struct idtfc3 *idtfc3) |
| { |
| u8 val; |
| int err; |
| |
| /* Check if FIFO is overrun */ |
| err = regmap_bulk_read(idtfc3->regmap, TDC_FIFO_STS, &val, sizeof(val)); |
| if (err) |
| return err; |
| |
| if (!(val & FIFO_FULL)) |
| return 0; |
| |
| dev_warn(idtfc3->dev, "TDC FIFO overrun !!!"); |
| |
| err = idtfc3_enable_tdc(idtfc3, true, CONTINUOUS); |
| if (err) |
| return err; |
| |
| return 0; |
| } |
| |
| static int get_tdc_meas_continuous(struct idtfc3 *idtfc3) |
| { |
| int err; |
| s64 offset_ns; |
| struct ptp_clock_event event; |
| |
| err = check_tdc_fifo_overrun(idtfc3); |
| if (err) |
| return err; |
| |
| if (get_tdc_meas(idtfc3, &offset_ns) && offset_ns >= 0) { |
| event.index = 0; |
| event.offset = tdc_offset2phase(idtfc3, offset_ns); |
| event.type = PTP_CLOCK_EXTOFF; |
| ptp_clock_event(idtfc3->ptp_clock, &event); |
| } |
| |
| return 0; |
| } |
| |
| static int idtfc3_read_subcounter(struct idtfc3 *idtfc3) |
| { |
| u8 buf[5] = {0}; |
| int err; |
| |
| err = regmap_bulk_read(idtfc3->regmap, TOD_COUNTER_READ_REQ, |
| &buf, sizeof(buf)); |
| if (err) |
| return err; |
| |
| /* sync_counter_value is [31:82] and sub_sync_counter_value is [0:30] */ |
| return get_unaligned_le32(&buf[1]) & SUB_SYNC_COUNTER_MASK; |
| } |
| |
| static int idtfc3_tod_update_is_done(struct idtfc3 *idtfc3) |
| { |
| int err; |
| u8 req; |
| |
| err = read_poll_timeout_atomic(regmap_bulk_read, err, !req, USEC_PER_MSEC, |
| idtfc3->tc_write_timeout, true, idtfc3->regmap, |
| TOD_SYNC_LOAD_REQ_CTRL, &req, 1); |
| if (err) |
| dev_err(idtfc3->dev, "TOD counter write timeout !!!"); |
| |
| return err; |
| } |
| |
| static int idtfc3_write_subcounter(struct idtfc3 *idtfc3, u32 counter) |
| { |
| u8 buf[18] = {0}; |
| int err; |
| |
| /* sync_counter_value is [31:82] and sub_sync_counter_value is [0:30] */ |
| put_unaligned_le32(counter & SUB_SYNC_COUNTER_MASK, &buf[0]); |
| |
| buf[16] = SUB_SYNC_LOAD_ENABLE | SYNC_LOAD_ENABLE; |
| buf[17] = SYNC_LOAD_REQ; |
| |
| err = regmap_bulk_write(idtfc3->regmap, TOD_SYNC_LOAD_VAL_CTRL, |
| &buf, sizeof(buf)); |
| if (err) |
| return err; |
| |
| return idtfc3_tod_update_is_done(idtfc3); |
| } |
| |
| static int idtfc3_timecounter_update(struct idtfc3 *idtfc3, u32 counter, s64 ns) |
| { |
| int err; |
| |
| err = idtfc3_write_subcounter(idtfc3, counter); |
| if (err) |
| return err; |
| |
| /* Update time counter */ |
| idtfc3->ns = ns; |
| idtfc3->last_counter = counter; |
| |
| return 0; |
| } |
| |
| static int idtfc3_timecounter_read(struct idtfc3 *idtfc3) |
| { |
| int now, delta; |
| |
| now = idtfc3_read_subcounter(idtfc3); |
| if (now < 0) |
| return now; |
| |
| /* calculate the delta since the last idtfc3_timecounter_read(): */ |
| if (now >= idtfc3->last_counter) |
| delta = now - idtfc3->last_counter; |
| else |
| delta = idtfc3->sub_sync_count - idtfc3->last_counter + now; |
| |
| /* Update time counter */ |
| idtfc3->ns += delta * idtfc3->ns_per_counter; |
| idtfc3->last_counter = now; |
| |
| return 0; |
| } |
| |
| static int _idtfc3_gettime(struct idtfc3 *idtfc3, struct timespec64 *ts) |
| { |
| int err; |
| |
| err = idtfc3_timecounter_read(idtfc3); |
| if (err) |
| return err; |
| |
| *ts = ns_to_timespec64(idtfc3->ns); |
| |
| return 0; |
| } |
| |
| static int idtfc3_gettime(struct ptp_clock_info *ptp, struct timespec64 *ts) |
| { |
| struct idtfc3 *idtfc3 = container_of(ptp, struct idtfc3, caps); |
| int err; |
| |
| mutex_lock(idtfc3->lock); |
| err = _idtfc3_gettime(idtfc3, ts); |
| mutex_unlock(idtfc3->lock); |
| |
| return err; |
| } |
| |
| static int _idtfc3_settime(struct idtfc3 *idtfc3, const struct timespec64 *ts) |
| { |
| s64 offset_ns, now_ns; |
| u32 counter, sub_ns; |
| int now; |
| |
| if (timespec64_valid(ts) == false) { |
| dev_err(idtfc3->dev, "%s: invalid timespec", __func__); |
| return -EINVAL; |
| } |
| |
| now = idtfc3_read_subcounter(idtfc3); |
| if (now < 0) |
| return now; |
| |
| offset_ns = (idtfc3->sub_sync_count - now) * idtfc3->ns_per_counter; |
| now_ns = timespec64_to_ns(ts); |
| (void)ns2counters(idtfc3, offset_ns + now_ns, &sub_ns); |
| |
| counter = sub_ns / idtfc3->ns_per_counter; |
| return idtfc3_timecounter_update(idtfc3, counter, now_ns); |
| } |
| |
| static int idtfc3_settime(struct ptp_clock_info *ptp, const struct timespec64 *ts) |
| { |
| struct idtfc3 *idtfc3 = container_of(ptp, struct idtfc3, caps); |
| int err; |
| |
| mutex_lock(idtfc3->lock); |
| err = _idtfc3_settime(idtfc3, ts); |
| mutex_unlock(idtfc3->lock); |
| |
| return err; |
| } |
| |
| static int _idtfc3_adjtime(struct idtfc3 *idtfc3, s64 delta) |
| { |
| /* |
| * The TOD counter can be synchronously loaded with any value, |
| * to be loaded on the next Time Sync pulse |
| */ |
| s64 sync_ns; |
| u32 sub_ns; |
| u32 counter; |
| |
| if (idtfc3->ns + delta < 0) { |
| dev_err(idtfc3->dev, "%lld ns adj is too large", delta); |
| return -EINVAL; |
| } |
| |
| sync_ns = ns2counters(idtfc3, delta + idtfc3->ns_per_sync, &sub_ns); |
| |
| counter = sub_ns / idtfc3->ns_per_counter; |
| return idtfc3_timecounter_update(idtfc3, counter, idtfc3->ns + sync_ns + |
| counter * idtfc3->ns_per_counter); |
| } |
| |
| static int idtfc3_adjtime(struct ptp_clock_info *ptp, s64 delta) |
| { |
| struct idtfc3 *idtfc3 = container_of(ptp, struct idtfc3, caps); |
| int err; |
| |
| mutex_lock(idtfc3->lock); |
| err = _idtfc3_adjtime(idtfc3, delta); |
| mutex_unlock(idtfc3->lock); |
| |
| return err; |
| } |
| |
| static int _idtfc3_adjphase(struct idtfc3 *idtfc3, s32 delta) |
| { |
| u8 buf[8] = {0}; |
| int err; |
| s64 pcw; |
| |
| err = idtfc3_set_lpf_mode(idtfc3, LPF_WP); |
| if (err) |
| return err; |
| |
| /* |
| * Phase Control Word unit is: 10^9 / (TDC_APLL_FREQ * 124) |
| * |
| * delta * TDC_APLL_FREQ * 124 |
| * PCW = --------------------------- |
| * 10^9 |
| * |
| */ |
| pcw = div_s64((s64)delta * idtfc3->tdc_apll_freq * 124, NSEC_PER_SEC); |
| |
| put_unaligned_le64(pcw, buf); |
| |
| return regmap_bulk_write(idtfc3->regmap, LPF_WR_PHASE_CTRL, buf, sizeof(buf)); |
| } |
| |
| static int idtfc3_adjphase(struct ptp_clock_info *ptp, s32 delta) |
| { |
| struct idtfc3 *idtfc3 = container_of(ptp, struct idtfc3, caps); |
| int err; |
| |
| mutex_lock(idtfc3->lock); |
| err = _idtfc3_adjphase(idtfc3, delta); |
| mutex_unlock(idtfc3->lock); |
| |
| return err; |
| } |
| |
| static int _idtfc3_adjfine(struct idtfc3 *idtfc3, long scaled_ppm) |
| { |
| u8 buf[8] = {0}; |
| int err; |
| s64 fcw; |
| |
| err = idtfc3_set_lpf_mode(idtfc3, LPF_WF); |
| if (err) |
| return err; |
| |
| /* |
| * Frequency Control Word unit is: 2^-44 * 10^6 ppm |
| * |
| * adjfreq: |
| * ppb * 2^44 |
| * FCW = ---------- |
| * 10^9 |
| * |
| * adjfine: |
| * ppm_16 * 2^28 |
| * FCW = ------------- |
| * 10^6 |
| */ |
| fcw = scaled_ppm * BIT(28); |
| fcw = div_s64(fcw, 1000000); |
| |
| put_unaligned_le64(fcw, buf); |
| |
| return regmap_bulk_write(idtfc3->regmap, LPF_WR_FREQ_CTRL, buf, sizeof(buf)); |
| } |
| |
| static int idtfc3_adjfine(struct ptp_clock_info *ptp, long scaled_ppm) |
| { |
| struct idtfc3 *idtfc3 = container_of(ptp, struct idtfc3, caps); |
| int err; |
| |
| mutex_lock(idtfc3->lock); |
| err = _idtfc3_adjfine(idtfc3, scaled_ppm); |
| mutex_unlock(idtfc3->lock); |
| |
| return err; |
| } |
| |
| static int idtfc3_enable(struct ptp_clock_info *ptp, |
| struct ptp_clock_request *rq, int on) |
| { |
| struct idtfc3 *idtfc3 = container_of(ptp, struct idtfc3, caps); |
| int err = -EOPNOTSUPP; |
| |
| mutex_lock(idtfc3->lock); |
| switch (rq->type) { |
| case PTP_CLK_REQ_PEROUT: |
| if (!on) |
| err = 0; |
| /* Only accept a 1-PPS aligned to the second. */ |
| else if (rq->perout.start.nsec || rq->perout.period.sec != 1 || |
| rq->perout.period.nsec) |
| err = -ERANGE; |
| else |
| err = 0; |
| break; |
| case PTP_CLK_REQ_EXTTS: |
| if (on) { |
| /* Only accept requests for external phase offset */ |
| if ((rq->extts.flags & PTP_EXT_OFFSET) != (PTP_EXT_OFFSET)) |
| err = -EOPNOTSUPP; |
| else |
| err = idtfc3_enable_tdc(idtfc3, true, CONTINUOUS); |
| } else { |
| err = idtfc3_enable_tdc(idtfc3, false, MEAS_MODE_INVALID); |
| } |
| break; |
| default: |
| break; |
| } |
| mutex_unlock(idtfc3->lock); |
| |
| if (err) |
| dev_err(idtfc3->dev, "Failed in %s with err %d!", __func__, err); |
| |
| return err; |
| } |
| |
| static long idtfc3_aux_work(struct ptp_clock_info *ptp) |
| { |
| struct idtfc3 *idtfc3 = container_of(ptp, struct idtfc3, caps); |
| static int tdc_get; |
| |
| mutex_lock(idtfc3->lock); |
| tdc_get %= TDC_GET_PERIOD; |
| if ((tdc_get == 0) || (tdc_get == TDC_GET_PERIOD / 2)) |
| idtfc3_timecounter_read(idtfc3); |
| get_tdc_meas_continuous(idtfc3); |
| tdc_get++; |
| mutex_unlock(idtfc3->lock); |
| |
| return idtfc3->tc_update_period; |
| } |
| |
| static const struct ptp_clock_info idtfc3_caps = { |
| .owner = THIS_MODULE, |
| .max_adj = MAX_FFO_PPB, |
| .n_per_out = 1, |
| .n_ext_ts = 1, |
| .adjphase = &idtfc3_adjphase, |
| .adjfine = &idtfc3_adjfine, |
| .adjtime = &idtfc3_adjtime, |
| .gettime64 = &idtfc3_gettime, |
| .settime64 = &idtfc3_settime, |
| .enable = &idtfc3_enable, |
| .do_aux_work = &idtfc3_aux_work, |
| }; |
| |
| static int idtfc3_hw_calibrate(struct idtfc3 *idtfc3) |
| { |
| int err = 0; |
| u8 val; |
| |
| mdelay(10); |
| /* |
| * Toggle TDC_DAC_RECAL_REQ: |
| * (1) set tdc_en to 1 |
| * (2) set tdc_dac_recal_req to 0 |
| * (3) set tdc_dac_recal_req to 1 |
| */ |
| val = TDC_EN; |
| err = regmap_bulk_write(idtfc3->regmap, TDC_CTRL, |
| &val, sizeof(val)); |
| if (err) |
| return err; |
| val = TDC_EN | TDC_DAC_RECAL_REQ; |
| err = regmap_bulk_write(idtfc3->regmap, TDC_CTRL, |
| &val, sizeof(val)); |
| if (err) |
| return err; |
| mdelay(10); |
| |
| /* |
| * Toggle APLL_REINIT: |
| * (1) set apll_reinit to 0 |
| * (2) set apll_reinit to 1 |
| */ |
| val = 0; |
| err = regmap_bulk_write(idtfc3->regmap, SOFT_RESET_CTRL, |
| &val, sizeof(val)); |
| if (err) |
| return err; |
| val = APLL_REINIT; |
| err = regmap_bulk_write(idtfc3->regmap, SOFT_RESET_CTRL, |
| &val, sizeof(val)); |
| if (err) |
| return err; |
| mdelay(10); |
| |
| return err; |
| } |
| |
| static int idtfc3_init_timecounter(struct idtfc3 *idtfc3) |
| { |
| int err; |
| u32 period_ms; |
| |
| period_ms = idtfc3->sub_sync_count * MSEC_PER_SEC / |
| idtfc3->hw_param.time_clk_freq; |
| |
| idtfc3->tc_update_period = msecs_to_jiffies(period_ms / TDC_GET_PERIOD); |
| idtfc3->tc_write_timeout = period_ms * USEC_PER_MSEC; |
| |
| err = idtfc3_timecounter_update(idtfc3, 0, 0); |
| if (err) |
| return err; |
| |
| err = idtfc3_timecounter_read(idtfc3); |
| if (err) |
| return err; |
| |
| ptp_schedule_worker(idtfc3->ptp_clock, idtfc3->tc_update_period); |
| |
| return 0; |
| } |
| |
| static int idtfc3_get_tdc_apll_freq(struct idtfc3 *idtfc3) |
| { |
| int err; |
| u8 tdc_fb_div_int; |
| u8 tdc_ref_div; |
| struct idtfc3_hw_param *param = &idtfc3->hw_param; |
| |
| err = regmap_bulk_read(idtfc3->regmap, TDC_REF_DIV_CNFG, |
| &tdc_ref_div, sizeof(tdc_ref_div)); |
| if (err) |
| return err; |
| |
| err = regmap_bulk_read(idtfc3->regmap, TDC_FB_DIV_INT_CNFG, |
| &tdc_fb_div_int, sizeof(tdc_fb_div_int)); |
| if (err) |
| return err; |
| |
| tdc_fb_div_int &= TDC_FB_DIV_INT_MASK; |
| tdc_ref_div &= TDC_REF_DIV_CONFIG_MASK; |
| |
| idtfc3->tdc_apll_freq = div_u64(param->xtal_freq * (u64)tdc_fb_div_int, |
| 1 << tdc_ref_div); |
| |
| return 0; |
| } |
| |
| static int idtfc3_get_fod(struct idtfc3 *idtfc3) |
| { |
| int err; |
| u8 fod; |
| |
| err = regmap_bulk_read(idtfc3->regmap, TIME_CLOCK_SRC, &fod, sizeof(fod)); |
| if (err) |
| return err; |
| |
| switch (fod) { |
| case 0: |
| idtfc3->fod_n = FOD_0; |
| break; |
| case 1: |
| idtfc3->fod_n = FOD_1; |
| break; |
| case 2: |
| idtfc3->fod_n = FOD_2; |
| break; |
| default: |
| return -EINVAL; |
| } |
| |
| return 0; |
| } |
| |
| static int idtfc3_get_sync_count(struct idtfc3 *idtfc3) |
| { |
| int err; |
| u8 buf[4]; |
| |
| err = regmap_bulk_read(idtfc3->regmap, SUB_SYNC_GEN_CNFG, buf, sizeof(buf)); |
| if (err) |
| return err; |
| |
| idtfc3->sub_sync_count = (get_unaligned_le32(buf) & SUB_SYNC_COUNTER_MASK) + 1; |
| idtfc3->ns_per_counter = NSEC_PER_SEC / idtfc3->hw_param.time_clk_freq; |
| idtfc3->ns_per_sync = idtfc3->sub_sync_count * idtfc3->ns_per_counter; |
| |
| return 0; |
| } |
| |
| static int idtfc3_setup_hw_param(struct idtfc3 *idtfc3) |
| { |
| int err; |
| |
| err = idtfc3_get_fod(idtfc3); |
| if (err) |
| return err; |
| |
| err = idtfc3_get_sync_count(idtfc3); |
| if (err) |
| return err; |
| |
| err = idtfc3_get_time_ref_freq(idtfc3); |
| if (err) |
| return err; |
| |
| return idtfc3_get_tdc_apll_freq(idtfc3); |
| } |
| |
| static int idtfc3_configure_hw(struct idtfc3 *idtfc3) |
| { |
| int err = 0; |
| |
| err = idtfc3_hw_calibrate(idtfc3); |
| if (err) |
| return err; |
| |
| err = idtfc3_enable_lpf(idtfc3, true); |
| if (err) |
| return err; |
| |
| err = idtfc3_enable_tdc(idtfc3, false, MEAS_MODE_INVALID); |
| if (err) |
| return err; |
| |
| err = idtfc3_get_tdc_offset_sign(idtfc3); |
| if (err) |
| return err; |
| |
| return idtfc3_setup_hw_param(idtfc3); |
| } |
| |
| static int idtfc3_set_overhead(struct idtfc3 *idtfc3) |
| { |
| s64 current_ns = 0; |
| s64 lowest_ns = 0; |
| int err; |
| u8 i; |
| ktime_t start; |
| ktime_t stop; |
| ktime_t diff; |
| |
| char buf[18] = {0}; |
| |
| for (i = 0; i < 5; i++) { |
| start = ktime_get_raw(); |
| |
| err = regmap_bulk_write(idtfc3->regmap, TOD_SYNC_LOAD_VAL_CTRL, |
| &buf, sizeof(buf)); |
| if (err) |
| return err; |
| |
| stop = ktime_get_raw(); |
| |
| diff = ktime_sub(stop, start); |
| |
| current_ns = ktime_to_ns(diff); |
| |
| if (i == 0) { |
| lowest_ns = current_ns; |
| } else { |
| if (current_ns < lowest_ns) |
| lowest_ns = current_ns; |
| } |
| } |
| |
| idtfc3->tod_write_overhead = lowest_ns; |
| |
| return err; |
| } |
| |
| static int idtfc3_enable_ptp(struct idtfc3 *idtfc3) |
| { |
| int err; |
| |
| idtfc3->caps = idtfc3_caps; |
| snprintf(idtfc3->caps.name, sizeof(idtfc3->caps.name), "IDT FC3W"); |
| idtfc3->ptp_clock = ptp_clock_register(&idtfc3->caps, NULL); |
| |
| if (IS_ERR(idtfc3->ptp_clock)) { |
| err = PTR_ERR(idtfc3->ptp_clock); |
| idtfc3->ptp_clock = NULL; |
| return err; |
| } |
| |
| err = idtfc3_set_overhead(idtfc3); |
| if (err) |
| return err; |
| |
| err = idtfc3_init_timecounter(idtfc3); |
| if (err) |
| return err; |
| |
| dev_info(idtfc3->dev, "TIME_SYNC_CHANNEL registered as ptp%d", |
| idtfc3->ptp_clock->index); |
| |
| return 0; |
| } |
| |
| static int idtfc3_load_firmware(struct idtfc3 *idtfc3) |
| { |
| char fname[128] = FW_FILENAME; |
| const struct firmware *fw; |
| struct idtfc3_fwrc *rec; |
| u16 addr; |
| u8 val; |
| int err; |
| s32 len; |
| |
| idtfc3_default_hw_param(&idtfc3->hw_param); |
| |
| if (firmware) /* module parameter */ |
| snprintf(fname, sizeof(fname), "%s", firmware); |
| |
| dev_info(idtfc3->dev, "requesting firmware '%s'\n", fname); |
| |
| err = request_firmware(&fw, fname, idtfc3->dev); |
| |
| if (err) { |
| dev_err(idtfc3->dev, |
| "requesting firmware failed with err %d!\n", err); |
| return err; |
| } |
| |
| dev_dbg(idtfc3->dev, "firmware size %zu bytes\n", fw->size); |
| |
| rec = (struct idtfc3_fwrc *)fw->data; |
| |
| for (len = fw->size; len > 0; len -= sizeof(*rec)) { |
| if (rec->reserved) { |
| dev_err(idtfc3->dev, |
| "bad firmware, reserved field non-zero\n"); |
| err = -EINVAL; |
| } else { |
| val = rec->value; |
| addr = rec->hiaddr << 8 | rec->loaddr; |
| |
| rec++; |
| |
| err = idtfc3_set_hw_param(&idtfc3->hw_param, addr, val); |
| } |
| |
| if (err != -EINVAL) { |
| err = 0; |
| |
| /* Max register */ |
| if (addr >= 0xE88) |
| continue; |
| |
| err = regmap_bulk_write(idtfc3->regmap, addr, |
| &val, sizeof(val)); |
| } |
| |
| if (err) |
| goto out; |
| } |
| |
| err = idtfc3_configure_hw(idtfc3); |
| out: |
| release_firmware(fw); |
| return err; |
| } |
| |
| static int idtfc3_read_device_id(struct idtfc3 *idtfc3, u16 *device_id) |
| { |
| int err; |
| u8 buf[2] = {0}; |
| |
| err = regmap_bulk_read(idtfc3->regmap, DEVICE_ID, |
| &buf, sizeof(buf)); |
| if (err) { |
| dev_err(idtfc3->dev, "%s failed with %d", __func__, err); |
| return err; |
| } |
| |
| *device_id = get_unaligned_le16(buf); |
| |
| return 0; |
| } |
| |
| static int idtfc3_check_device_compatibility(struct idtfc3 *idtfc3) |
| { |
| int err; |
| u16 device_id; |
| |
| err = idtfc3_read_device_id(idtfc3, &device_id); |
| if (err) |
| return err; |
| |
| if ((device_id & DEVICE_ID_MASK) == 0) { |
| dev_err(idtfc3->dev, "invalid device"); |
| return -EINVAL; |
| } |
| |
| return 0; |
| } |
| |
| static int idtfc3_probe(struct platform_device *pdev) |
| { |
| struct rsmu_ddata *ddata = dev_get_drvdata(pdev->dev.parent); |
| struct idtfc3 *idtfc3; |
| int err; |
| |
| idtfc3 = devm_kzalloc(&pdev->dev, sizeof(struct idtfc3), GFP_KERNEL); |
| |
| if (!idtfc3) |
| return -ENOMEM; |
| |
| idtfc3->dev = &pdev->dev; |
| idtfc3->mfd = pdev->dev.parent; |
| idtfc3->lock = &ddata->lock; |
| idtfc3->regmap = ddata->regmap; |
| |
| mutex_lock(idtfc3->lock); |
| |
| err = idtfc3_check_device_compatibility(idtfc3); |
| if (err) { |
| mutex_unlock(idtfc3->lock); |
| return err; |
| } |
| |
| err = idtfc3_load_firmware(idtfc3); |
| if (err) { |
| if (err == -ENOENT) { |
| mutex_unlock(idtfc3->lock); |
| return -EPROBE_DEFER; |
| } |
| dev_warn(idtfc3->dev, "loading firmware failed with %d", err); |
| } |
| |
| err = idtfc3_enable_ptp(idtfc3); |
| if (err) { |
| dev_err(idtfc3->dev, "idtfc3_enable_ptp failed with %d", err); |
| mutex_unlock(idtfc3->lock); |
| return err; |
| } |
| |
| mutex_unlock(idtfc3->lock); |
| |
| if (err) { |
| ptp_clock_unregister(idtfc3->ptp_clock); |
| return err; |
| } |
| |
| platform_set_drvdata(pdev, idtfc3); |
| |
| return 0; |
| } |
| |
| static void idtfc3_remove(struct platform_device *pdev) |
| { |
| struct idtfc3 *idtfc3 = platform_get_drvdata(pdev); |
| |
| ptp_clock_unregister(idtfc3->ptp_clock); |
| } |
| |
| static struct platform_driver idtfc3_driver = { |
| .driver = { |
| .name = "rc38xxx-phc", |
| }, |
| .probe = idtfc3_probe, |
| .remove_new = idtfc3_remove, |
| }; |
| |
| module_platform_driver(idtfc3_driver); |