| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014 |
- // 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);
|