|  | /* SPDX-License-Identifier: GPL-2.0+ */ | 
|  | /* | 
|  | * PTP hardware clock driver for the IDT ClockMatrix(TM) family of timing and | 
|  | * synchronization devices. | 
|  | * | 
|  | * Copyright (C) 2019 Integrated Device Technology, Inc., a Renesas Company. | 
|  | */ | 
|  | #ifndef PTP_IDTCLOCKMATRIX_H | 
|  | #define PTP_IDTCLOCKMATRIX_H | 
|  |  | 
|  | #include <linux/ktime.h> | 
|  | #include <linux/mfd/idt8a340_reg.h> | 
|  | #include <linux/ptp_clock.h> | 
|  | #include <linux/regmap.h> | 
|  |  | 
|  | #define FW_FILENAME	"idtcm.bin" | 
|  | #define MAX_TOD		(4) | 
|  | #define MAX_PLL		(8) | 
|  | #define MAX_REF_CLK	(16) | 
|  |  | 
|  | #define MAX_ABS_WRITE_PHASE_NANOSECONDS (107374182L) | 
|  |  | 
|  | #define TOD_MASK_ADDR		(0xFFA5) | 
|  | #define DEFAULT_TOD_MASK	(0x04) | 
|  |  | 
|  | #define SET_U16_LSB(orig, val8) (orig = (0xff00 & (orig)) | (val8)) | 
|  | #define SET_U16_MSB(orig, val8) (orig = (0x00ff & (orig)) | (val8 << 8)) | 
|  |  | 
|  | #define TOD0_PTP_PLL_ADDR		(0xFFA8) | 
|  | #define TOD1_PTP_PLL_ADDR		(0xFFA9) | 
|  | #define TOD2_PTP_PLL_ADDR		(0xFFAA) | 
|  | #define TOD3_PTP_PLL_ADDR		(0xFFAB) | 
|  |  | 
|  | #define TOD0_OUT_ALIGN_MASK_ADDR	(0xFFB0) | 
|  | #define TOD1_OUT_ALIGN_MASK_ADDR	(0xFFB2) | 
|  | #define TOD2_OUT_ALIGN_MASK_ADDR	(0xFFB4) | 
|  | #define TOD3_OUT_ALIGN_MASK_ADDR	(0xFFB6) | 
|  |  | 
|  | #define DEFAULT_OUTPUT_MASK_PLL0	(0x003) | 
|  | #define DEFAULT_OUTPUT_MASK_PLL1	(0x00c) | 
|  | #define DEFAULT_OUTPUT_MASK_PLL2	(0x030) | 
|  | #define DEFAULT_OUTPUT_MASK_PLL3	(0x0c0) | 
|  |  | 
|  | #define DEFAULT_TOD0_PTP_PLL		(0) | 
|  | #define DEFAULT_TOD1_PTP_PLL		(1) | 
|  | #define DEFAULT_TOD2_PTP_PLL		(2) | 
|  | #define DEFAULT_TOD3_PTP_PLL		(3) | 
|  |  | 
|  | #define PHASE_PULL_IN_THRESHOLD_NS_DEPRECATED	(150000) | 
|  | #define PHASE_PULL_IN_THRESHOLD_NS		(15000) | 
|  | #define TOD_WRITE_OVERHEAD_COUNT_MAX		(2) | 
|  | #define TOD_BYTE_COUNT				(11) | 
|  |  | 
|  | #define LOCK_TIMEOUT_MS			(2000) | 
|  | #define LOCK_POLL_INTERVAL_MS		(10) | 
|  |  | 
|  | #define IDTCM_MAX_WRITE_COUNT		(512) | 
|  |  | 
|  | #define PHASE_PULL_IN_MAX_PPB		(144000) | 
|  | #define PHASE_PULL_IN_MIN_THRESHOLD_NS	(2) | 
|  |  | 
|  | /* | 
|  | * Return register address based on passed in firmware version | 
|  | */ | 
|  | #define IDTCM_FW_REG(FW, VER, REG)	(((FW) < (VER)) ? (REG) : (REG##_##VER)) | 
|  | enum fw_version { | 
|  | V_DEFAULT = 0, | 
|  | V487 = 1, | 
|  | V520 = 2, | 
|  | }; | 
|  |  | 
|  | /* PTP PLL Mode */ | 
|  | enum ptp_pll_mode { | 
|  | PTP_PLL_MODE_MIN = 0, | 
|  | PTP_PLL_MODE_WRITE_FREQUENCY = PTP_PLL_MODE_MIN, | 
|  | PTP_PLL_MODE_WRITE_PHASE, | 
|  | PTP_PLL_MODE_UNSUPPORTED, | 
|  | PTP_PLL_MODE_MAX = PTP_PLL_MODE_UNSUPPORTED, | 
|  | }; | 
|  |  | 
|  | struct idtcm; | 
|  |  | 
|  | struct idtcm_channel { | 
|  | struct ptp_clock_info	caps; | 
|  | struct ptp_clock	*ptp_clock; | 
|  | struct idtcm		*idtcm; | 
|  | u16			dpll_phase; | 
|  | u16			dpll_freq; | 
|  | u16			dpll_n; | 
|  | u16			dpll_ctrl_n; | 
|  | u16			dpll_phase_pull_in; | 
|  | u16			tod_read_primary; | 
|  | u16			tod_read_secondary; | 
|  | u16			tod_write; | 
|  | u16			tod_n; | 
|  | u16			hw_dpll_n; | 
|  | u8			sync_src; | 
|  | enum ptp_pll_mode	mode; | 
|  | int			(*configure_write_frequency)(struct idtcm_channel *channel); | 
|  | int			(*configure_write_phase)(struct idtcm_channel *channel); | 
|  | int			(*do_phase_pull_in)(struct idtcm_channel *channel, | 
|  | s32 offset_ns, u32 max_ffo_ppb); | 
|  | s32			current_freq_scaled_ppm; | 
|  | bool			phase_pull_in; | 
|  | u32			dco_delay; | 
|  | /* last input trigger for extts */ | 
|  | u8			refn; | 
|  | u8			pll; | 
|  | u8			tod; | 
|  | u16			output_mask; | 
|  | }; | 
|  |  | 
|  | struct idtcm { | 
|  | struct idtcm_channel	channel[MAX_TOD]; | 
|  | struct device		*dev; | 
|  | u8			tod_mask; | 
|  | char			version[16]; | 
|  | enum fw_version		fw_ver; | 
|  | /* Polls for external time stamps */ | 
|  | u8			extts_mask; | 
|  | bool			extts_single_shot; | 
|  | struct delayed_work	extts_work; | 
|  | /* Remember the ptp channel to report extts */ | 
|  | struct idtcm_channel	*event_channel[MAX_TOD]; | 
|  | /* Mutex to protect operations from being interrupted */ | 
|  | struct mutex		*lock; | 
|  | struct device		*mfd; | 
|  | struct regmap		*regmap; | 
|  | /* Overhead calculation for adjtime */ | 
|  | u8			calculate_overhead_flag; | 
|  | s64			tod_write_overhead_ns; | 
|  | ktime_t			start_time; | 
|  | }; | 
|  |  | 
|  | struct idtcm_fwrc { | 
|  | u8 hiaddr; | 
|  | u8 loaddr; | 
|  | u8 value; | 
|  | u8 reserved; | 
|  | } __packed; | 
|  |  | 
|  | #endif /* PTP_IDTCLOCKMATRIX_H */ |