xref: /openbmc/linux/drivers/net/dsa/hirschmann/hellcreek_ptp.c (revision c900529f3d9161bfde5cca0754f83b4d3c3e0220)
1ddd56dfeSKamil Alkhouri // SPDX-License-Identifier: (GPL-2.0 OR MIT)
2ddd56dfeSKamil Alkhouri /*
3ddd56dfeSKamil Alkhouri  * DSA driver for:
4ddd56dfeSKamil Alkhouri  * Hirschmann Hellcreek TSN switch.
5ddd56dfeSKamil Alkhouri  *
6ddd56dfeSKamil Alkhouri  * Copyright (C) 2019,2020 Hochschule Offenburg
7ddd56dfeSKamil Alkhouri  * Copyright (C) 2019,2020 Linutronix GmbH
8ddd56dfeSKamil Alkhouri  * Authors: Kamil Alkhouri <kamil.alkhouri@hs-offenburg.de>
9ddd56dfeSKamil Alkhouri  *	    Kurt Kanzenbach <kurt@linutronix.de>
10ddd56dfeSKamil Alkhouri  */
11ddd56dfeSKamil Alkhouri 
12*f44a9010SRob Herring #include <linux/of.h>
13ddd56dfeSKamil Alkhouri #include <linux/ptp_clock_kernel.h>
14ddd56dfeSKamil Alkhouri #include "hellcreek.h"
15ddd56dfeSKamil Alkhouri #include "hellcreek_ptp.h"
16f0d4ba9eSKamil Alkhouri #include "hellcreek_hwtstamp.h"
17ddd56dfeSKamil Alkhouri 
hellcreek_ptp_read(struct hellcreek * hellcreek,unsigned int offset)18f0d4ba9eSKamil Alkhouri u16 hellcreek_ptp_read(struct hellcreek *hellcreek, unsigned int offset)
19ddd56dfeSKamil Alkhouri {
20ddd56dfeSKamil Alkhouri 	return readw(hellcreek->ptp_base + offset);
21ddd56dfeSKamil Alkhouri }
22ddd56dfeSKamil Alkhouri 
hellcreek_ptp_write(struct hellcreek * hellcreek,u16 data,unsigned int offset)23f0d4ba9eSKamil Alkhouri void hellcreek_ptp_write(struct hellcreek *hellcreek, u16 data,
24ddd56dfeSKamil Alkhouri 			 unsigned int offset)
25ddd56dfeSKamil Alkhouri {
26ddd56dfeSKamil Alkhouri 	writew(data, hellcreek->ptp_base + offset);
27ddd56dfeSKamil Alkhouri }
28ddd56dfeSKamil Alkhouri 
29ddd56dfeSKamil Alkhouri /* Get nanoseconds from PTP clock */
hellcreek_ptp_clock_read(struct hellcreek * hellcreek)30ddd56dfeSKamil Alkhouri static u64 hellcreek_ptp_clock_read(struct hellcreek *hellcreek)
31ddd56dfeSKamil Alkhouri {
32ddd56dfeSKamil Alkhouri 	u16 nsl, nsh;
33ddd56dfeSKamil Alkhouri 
34ddd56dfeSKamil Alkhouri 	/* Take a snapshot */
35ddd56dfeSKamil Alkhouri 	hellcreek_ptp_write(hellcreek, PR_COMMAND_C_SS, PR_COMMAND_C);
36ddd56dfeSKamil Alkhouri 
37ddd56dfeSKamil Alkhouri 	/* The time of the day is saved as 96 bits. However, due to hardware
38ddd56dfeSKamil Alkhouri 	 * limitations the seconds are not or only partly kept in the PTP
39ddd56dfeSKamil Alkhouri 	 * core. Currently only three bits for the seconds are available. That's
40ddd56dfeSKamil Alkhouri 	 * why only the nanoseconds are used and the seconds are tracked in
41ddd56dfeSKamil Alkhouri 	 * software. Anyway due to internal locking all five registers should be
42ddd56dfeSKamil Alkhouri 	 * read.
43ddd56dfeSKamil Alkhouri 	 */
44ddd56dfeSKamil Alkhouri 	nsh = hellcreek_ptp_read(hellcreek, PR_SS_SYNC_DATA_C);
45ddd56dfeSKamil Alkhouri 	nsh = hellcreek_ptp_read(hellcreek, PR_SS_SYNC_DATA_C);
46ddd56dfeSKamil Alkhouri 	nsh = hellcreek_ptp_read(hellcreek, PR_SS_SYNC_DATA_C);
47ddd56dfeSKamil Alkhouri 	nsh = hellcreek_ptp_read(hellcreek, PR_SS_SYNC_DATA_C);
48ddd56dfeSKamil Alkhouri 	nsl = hellcreek_ptp_read(hellcreek, PR_SS_SYNC_DATA_C);
49ddd56dfeSKamil Alkhouri 
50ddd56dfeSKamil Alkhouri 	return (u64)nsl | ((u64)nsh << 16);
51ddd56dfeSKamil Alkhouri }
52ddd56dfeSKamil Alkhouri 
__hellcreek_ptp_gettime(struct hellcreek * hellcreek)53ddd56dfeSKamil Alkhouri static u64 __hellcreek_ptp_gettime(struct hellcreek *hellcreek)
54ddd56dfeSKamil Alkhouri {
55ddd56dfeSKamil Alkhouri 	u64 ns;
56ddd56dfeSKamil Alkhouri 
57ddd56dfeSKamil Alkhouri 	ns = hellcreek_ptp_clock_read(hellcreek);
58ddd56dfeSKamil Alkhouri 	if (ns < hellcreek->last_ts)
59ddd56dfeSKamil Alkhouri 		hellcreek->seconds++;
60ddd56dfeSKamil Alkhouri 	hellcreek->last_ts = ns;
61ddd56dfeSKamil Alkhouri 	ns += hellcreek->seconds * NSEC_PER_SEC;
62ddd56dfeSKamil Alkhouri 
63ddd56dfeSKamil Alkhouri 	return ns;
64ddd56dfeSKamil Alkhouri }
65ddd56dfeSKamil Alkhouri 
66f0d4ba9eSKamil Alkhouri /* Retrieve the seconds parts in nanoseconds for a packet timestamped with @ns.
67f0d4ba9eSKamil Alkhouri  * There has to be a check whether an overflow occurred between the packet
68f0d4ba9eSKamil Alkhouri  * arrival and now. If so use the correct seconds (-1) for calculating the
69f0d4ba9eSKamil Alkhouri  * packet arrival time.
70f0d4ba9eSKamil Alkhouri  */
hellcreek_ptp_gettime_seconds(struct hellcreek * hellcreek,u64 ns)71f0d4ba9eSKamil Alkhouri u64 hellcreek_ptp_gettime_seconds(struct hellcreek *hellcreek, u64 ns)
72f0d4ba9eSKamil Alkhouri {
73f0d4ba9eSKamil Alkhouri 	u64 s;
74f0d4ba9eSKamil Alkhouri 
75f0d4ba9eSKamil Alkhouri 	__hellcreek_ptp_gettime(hellcreek);
76f0d4ba9eSKamil Alkhouri 	if (hellcreek->last_ts > ns)
77f0d4ba9eSKamil Alkhouri 		s = hellcreek->seconds * NSEC_PER_SEC;
78f0d4ba9eSKamil Alkhouri 	else
79f0d4ba9eSKamil Alkhouri 		s = (hellcreek->seconds - 1) * NSEC_PER_SEC;
80f0d4ba9eSKamil Alkhouri 
81f0d4ba9eSKamil Alkhouri 	return s;
82f0d4ba9eSKamil Alkhouri }
83f0d4ba9eSKamil Alkhouri 
hellcreek_ptp_gettime(struct ptp_clock_info * ptp,struct timespec64 * ts)84ddd56dfeSKamil Alkhouri static int hellcreek_ptp_gettime(struct ptp_clock_info *ptp,
85ddd56dfeSKamil Alkhouri 				 struct timespec64 *ts)
86ddd56dfeSKamil Alkhouri {
87ddd56dfeSKamil Alkhouri 	struct hellcreek *hellcreek = ptp_to_hellcreek(ptp);
88ddd56dfeSKamil Alkhouri 	u64 ns;
89ddd56dfeSKamil Alkhouri 
90ddd56dfeSKamil Alkhouri 	mutex_lock(&hellcreek->ptp_lock);
91ddd56dfeSKamil Alkhouri 	ns = __hellcreek_ptp_gettime(hellcreek);
92ddd56dfeSKamil Alkhouri 	mutex_unlock(&hellcreek->ptp_lock);
93ddd56dfeSKamil Alkhouri 
94ddd56dfeSKamil Alkhouri 	*ts = ns_to_timespec64(ns);
95ddd56dfeSKamil Alkhouri 
96ddd56dfeSKamil Alkhouri 	return 0;
97ddd56dfeSKamil Alkhouri }
98ddd56dfeSKamil Alkhouri 
hellcreek_ptp_settime(struct ptp_clock_info * ptp,const struct timespec64 * ts)99ddd56dfeSKamil Alkhouri static int hellcreek_ptp_settime(struct ptp_clock_info *ptp,
100ddd56dfeSKamil Alkhouri 				 const struct timespec64 *ts)
101ddd56dfeSKamil Alkhouri {
102ddd56dfeSKamil Alkhouri 	struct hellcreek *hellcreek = ptp_to_hellcreek(ptp);
103ddd56dfeSKamil Alkhouri 	u16 secl, nsh, nsl;
104ddd56dfeSKamil Alkhouri 
105ddd56dfeSKamil Alkhouri 	secl = ts->tv_sec & 0xffff;
106ddd56dfeSKamil Alkhouri 	nsh  = ((u32)ts->tv_nsec & 0xffff0000) >> 16;
107ddd56dfeSKamil Alkhouri 	nsl  = ts->tv_nsec & 0xffff;
108ddd56dfeSKamil Alkhouri 
109ddd56dfeSKamil Alkhouri 	mutex_lock(&hellcreek->ptp_lock);
110ddd56dfeSKamil Alkhouri 
111ddd56dfeSKamil Alkhouri 	/* Update overflow data structure */
112ddd56dfeSKamil Alkhouri 	hellcreek->seconds = ts->tv_sec;
113ddd56dfeSKamil Alkhouri 	hellcreek->last_ts = ts->tv_nsec;
114ddd56dfeSKamil Alkhouri 
115ddd56dfeSKamil Alkhouri 	/* Set time in clock */
116ddd56dfeSKamil Alkhouri 	hellcreek_ptp_write(hellcreek, 0x00, PR_CLOCK_WRITE_C);
117ddd56dfeSKamil Alkhouri 	hellcreek_ptp_write(hellcreek, 0x00, PR_CLOCK_WRITE_C);
118ddd56dfeSKamil Alkhouri 	hellcreek_ptp_write(hellcreek, secl, PR_CLOCK_WRITE_C);
119ddd56dfeSKamil Alkhouri 	hellcreek_ptp_write(hellcreek, nsh,  PR_CLOCK_WRITE_C);
120ddd56dfeSKamil Alkhouri 	hellcreek_ptp_write(hellcreek, nsl,  PR_CLOCK_WRITE_C);
121ddd56dfeSKamil Alkhouri 
122ddd56dfeSKamil Alkhouri 	mutex_unlock(&hellcreek->ptp_lock);
123ddd56dfeSKamil Alkhouri 
124ddd56dfeSKamil Alkhouri 	return 0;
125ddd56dfeSKamil Alkhouri }
126ddd56dfeSKamil Alkhouri 
hellcreek_ptp_adjfine(struct ptp_clock_info * ptp,long scaled_ppm)127ddd56dfeSKamil Alkhouri static int hellcreek_ptp_adjfine(struct ptp_clock_info *ptp, long scaled_ppm)
128ddd56dfeSKamil Alkhouri {
129ddd56dfeSKamil Alkhouri 	struct hellcreek *hellcreek = ptp_to_hellcreek(ptp);
130ddd56dfeSKamil Alkhouri 	u16 negative = 0, addendh, addendl;
131ddd56dfeSKamil Alkhouri 	u32 addend;
132ddd56dfeSKamil Alkhouri 	u64 adj;
133ddd56dfeSKamil Alkhouri 
134ddd56dfeSKamil Alkhouri 	if (scaled_ppm < 0) {
135ddd56dfeSKamil Alkhouri 		negative = 1;
136ddd56dfeSKamil Alkhouri 		scaled_ppm = -scaled_ppm;
137ddd56dfeSKamil Alkhouri 	}
138ddd56dfeSKamil Alkhouri 
139ddd56dfeSKamil Alkhouri 	/* IP-Core adjusts the nominal frequency by adding or subtracting 1 ns
140ddd56dfeSKamil Alkhouri 	 * from the 8 ns (period of the oscillator) every time the accumulator
141ddd56dfeSKamil Alkhouri 	 * register overflows. The value stored in the addend register is added
142ddd56dfeSKamil Alkhouri 	 * to the accumulator register every 8 ns.
143ddd56dfeSKamil Alkhouri 	 *
144ddd56dfeSKamil Alkhouri 	 * addend value = (2^30 * accumulator_overflow_rate) /
145ddd56dfeSKamil Alkhouri 	 *                oscillator_frequency
146ddd56dfeSKamil Alkhouri 	 * where:
147ddd56dfeSKamil Alkhouri 	 *
148ddd56dfeSKamil Alkhouri 	 * oscillator_frequency = 125 MHz
149ddd56dfeSKamil Alkhouri 	 * accumulator_overflow_rate = 125 MHz * scaled_ppm * 2^-16 * 10^-6 * 8
150ddd56dfeSKamil Alkhouri 	 */
151ddd56dfeSKamil Alkhouri 	adj = scaled_ppm;
152ddd56dfeSKamil Alkhouri 	adj <<= 11;
153ddd56dfeSKamil Alkhouri 	addend = (u32)div_u64(adj, 15625);
154ddd56dfeSKamil Alkhouri 
155ddd56dfeSKamil Alkhouri 	addendh = (addend & 0xffff0000) >> 16;
156ddd56dfeSKamil Alkhouri 	addendl = addend & 0xffff;
157ddd56dfeSKamil Alkhouri 
158ddd56dfeSKamil Alkhouri 	negative = (negative << 15) & 0x8000;
159ddd56dfeSKamil Alkhouri 
160ddd56dfeSKamil Alkhouri 	mutex_lock(&hellcreek->ptp_lock);
161ddd56dfeSKamil Alkhouri 
162ddd56dfeSKamil Alkhouri 	/* Set drift register */
163ddd56dfeSKamil Alkhouri 	hellcreek_ptp_write(hellcreek, negative, PR_CLOCK_DRIFT_C);
164ddd56dfeSKamil Alkhouri 	hellcreek_ptp_write(hellcreek, 0x00, PR_CLOCK_DRIFT_C);
165ddd56dfeSKamil Alkhouri 	hellcreek_ptp_write(hellcreek, 0x00, PR_CLOCK_DRIFT_C);
166ddd56dfeSKamil Alkhouri 	hellcreek_ptp_write(hellcreek, addendh,  PR_CLOCK_DRIFT_C);
167ddd56dfeSKamil Alkhouri 	hellcreek_ptp_write(hellcreek, addendl,  PR_CLOCK_DRIFT_C);
168ddd56dfeSKamil Alkhouri 
169ddd56dfeSKamil Alkhouri 	mutex_unlock(&hellcreek->ptp_lock);
170ddd56dfeSKamil Alkhouri 
171ddd56dfeSKamil Alkhouri 	return 0;
172ddd56dfeSKamil Alkhouri }
173ddd56dfeSKamil Alkhouri 
hellcreek_ptp_adjtime(struct ptp_clock_info * ptp,s64 delta)174ddd56dfeSKamil Alkhouri static int hellcreek_ptp_adjtime(struct ptp_clock_info *ptp, s64 delta)
175ddd56dfeSKamil Alkhouri {
176ddd56dfeSKamil Alkhouri 	struct hellcreek *hellcreek = ptp_to_hellcreek(ptp);
177ddd56dfeSKamil Alkhouri 	u16 negative = 0, counth, countl;
178ddd56dfeSKamil Alkhouri 	u32 count_val;
179ddd56dfeSKamil Alkhouri 
180ddd56dfeSKamil Alkhouri 	/* If the offset is larger than IP-Core slow offset resources. Don't
181ddd56dfeSKamil Alkhouri 	 * consider slow adjustment. Rather, add the offset directly to the
182ddd56dfeSKamil Alkhouri 	 * current time
183ddd56dfeSKamil Alkhouri 	 */
184ddd56dfeSKamil Alkhouri 	if (abs(delta) > MAX_SLOW_OFFSET_ADJ) {
185ddd56dfeSKamil Alkhouri 		struct timespec64 now, then = ns_to_timespec64(delta);
186ddd56dfeSKamil Alkhouri 
187ddd56dfeSKamil Alkhouri 		hellcreek_ptp_gettime(ptp, &now);
188ddd56dfeSKamil Alkhouri 		now = timespec64_add(now, then);
189ddd56dfeSKamil Alkhouri 		hellcreek_ptp_settime(ptp, &now);
190ddd56dfeSKamil Alkhouri 
191ddd56dfeSKamil Alkhouri 		return 0;
192ddd56dfeSKamil Alkhouri 	}
193ddd56dfeSKamil Alkhouri 
194ddd56dfeSKamil Alkhouri 	if (delta < 0) {
195ddd56dfeSKamil Alkhouri 		negative = 1;
196ddd56dfeSKamil Alkhouri 		delta = -delta;
197ddd56dfeSKamil Alkhouri 	}
198ddd56dfeSKamil Alkhouri 
199ddd56dfeSKamil Alkhouri 	/* 'count_val' does not exceed the maximum register size (2^30) */
200ddd56dfeSKamil Alkhouri 	count_val = div_s64(delta, MAX_NS_PER_STEP);
201ddd56dfeSKamil Alkhouri 
202ddd56dfeSKamil Alkhouri 	counth = (count_val & 0xffff0000) >> 16;
203ddd56dfeSKamil Alkhouri 	countl = count_val & 0xffff;
204ddd56dfeSKamil Alkhouri 
205ddd56dfeSKamil Alkhouri 	negative = (negative << 15) & 0x8000;
206ddd56dfeSKamil Alkhouri 
207ddd56dfeSKamil Alkhouri 	mutex_lock(&hellcreek->ptp_lock);
208ddd56dfeSKamil Alkhouri 
209ddd56dfeSKamil Alkhouri 	/* Set offset write register */
210ddd56dfeSKamil Alkhouri 	hellcreek_ptp_write(hellcreek, negative, PR_CLOCK_OFFSET_C);
211ddd56dfeSKamil Alkhouri 	hellcreek_ptp_write(hellcreek, MAX_NS_PER_STEP, PR_CLOCK_OFFSET_C);
212ddd56dfeSKamil Alkhouri 	hellcreek_ptp_write(hellcreek, MIN_CLK_CYCLES_BETWEEN_STEPS,
213ddd56dfeSKamil Alkhouri 			    PR_CLOCK_OFFSET_C);
214ddd56dfeSKamil Alkhouri 	hellcreek_ptp_write(hellcreek, countl,  PR_CLOCK_OFFSET_C);
215ddd56dfeSKamil Alkhouri 	hellcreek_ptp_write(hellcreek, counth,  PR_CLOCK_OFFSET_C);
216ddd56dfeSKamil Alkhouri 
217ddd56dfeSKamil Alkhouri 	mutex_unlock(&hellcreek->ptp_lock);
218ddd56dfeSKamil Alkhouri 
219ddd56dfeSKamil Alkhouri 	return 0;
220ddd56dfeSKamil Alkhouri }
221ddd56dfeSKamil Alkhouri 
hellcreek_ptp_enable(struct ptp_clock_info * ptp,struct ptp_clock_request * rq,int on)222ddd56dfeSKamil Alkhouri static int hellcreek_ptp_enable(struct ptp_clock_info *ptp,
223ddd56dfeSKamil Alkhouri 				struct ptp_clock_request *rq, int on)
224ddd56dfeSKamil Alkhouri {
225ddd56dfeSKamil Alkhouri 	return -EOPNOTSUPP;
226ddd56dfeSKamil Alkhouri }
227ddd56dfeSKamil Alkhouri 
hellcreek_ptp_overflow_check(struct work_struct * work)228ddd56dfeSKamil Alkhouri static void hellcreek_ptp_overflow_check(struct work_struct *work)
229ddd56dfeSKamil Alkhouri {
230ddd56dfeSKamil Alkhouri 	struct delayed_work *dw = to_delayed_work(work);
231ddd56dfeSKamil Alkhouri 	struct hellcreek *hellcreek;
232ddd56dfeSKamil Alkhouri 
233ddd56dfeSKamil Alkhouri 	hellcreek = dw_overflow_to_hellcreek(dw);
234ddd56dfeSKamil Alkhouri 
235ddd56dfeSKamil Alkhouri 	mutex_lock(&hellcreek->ptp_lock);
236ddd56dfeSKamil Alkhouri 	__hellcreek_ptp_gettime(hellcreek);
237ddd56dfeSKamil Alkhouri 	mutex_unlock(&hellcreek->ptp_lock);
238ddd56dfeSKamil Alkhouri 
239ddd56dfeSKamil Alkhouri 	schedule_delayed_work(&hellcreek->overflow_work,
240ddd56dfeSKamil Alkhouri 			      HELLCREEK_OVERFLOW_PERIOD);
241ddd56dfeSKamil Alkhouri }
242ddd56dfeSKamil Alkhouri 
hellcreek_get_brightness(struct hellcreek * hellcreek,int led)2437d9ee2e8SKurt Kanzenbach static enum led_brightness hellcreek_get_brightness(struct hellcreek *hellcreek,
2447d9ee2e8SKurt Kanzenbach 						    int led)
2457d9ee2e8SKurt Kanzenbach {
2467d9ee2e8SKurt Kanzenbach 	return (hellcreek->status_out & led) ? 1 : 0;
2477d9ee2e8SKurt Kanzenbach }
2487d9ee2e8SKurt Kanzenbach 
hellcreek_set_brightness(struct hellcreek * hellcreek,int led,enum led_brightness b)2497d9ee2e8SKurt Kanzenbach static void hellcreek_set_brightness(struct hellcreek *hellcreek, int led,
2507d9ee2e8SKurt Kanzenbach 				     enum led_brightness b)
2517d9ee2e8SKurt Kanzenbach {
2527d9ee2e8SKurt Kanzenbach 	mutex_lock(&hellcreek->ptp_lock);
2537d9ee2e8SKurt Kanzenbach 
2547d9ee2e8SKurt Kanzenbach 	if (b)
2557d9ee2e8SKurt Kanzenbach 		hellcreek->status_out |= led;
2567d9ee2e8SKurt Kanzenbach 	else
2577d9ee2e8SKurt Kanzenbach 		hellcreek->status_out &= ~led;
2587d9ee2e8SKurt Kanzenbach 
2597d9ee2e8SKurt Kanzenbach 	hellcreek_ptp_write(hellcreek, hellcreek->status_out, STATUS_OUT);
2607d9ee2e8SKurt Kanzenbach 
2617d9ee2e8SKurt Kanzenbach 	mutex_unlock(&hellcreek->ptp_lock);
2627d9ee2e8SKurt Kanzenbach }
2637d9ee2e8SKurt Kanzenbach 
hellcreek_led_sync_good_set(struct led_classdev * ldev,enum led_brightness b)2647d9ee2e8SKurt Kanzenbach static void hellcreek_led_sync_good_set(struct led_classdev *ldev,
2657d9ee2e8SKurt Kanzenbach 					enum led_brightness b)
2667d9ee2e8SKurt Kanzenbach {
2677d9ee2e8SKurt Kanzenbach 	struct hellcreek *hellcreek = led_to_hellcreek(ldev, led_sync_good);
2687d9ee2e8SKurt Kanzenbach 
2697d9ee2e8SKurt Kanzenbach 	hellcreek_set_brightness(hellcreek, STATUS_OUT_SYNC_GOOD, b);
2707d9ee2e8SKurt Kanzenbach }
2717d9ee2e8SKurt Kanzenbach 
hellcreek_led_sync_good_get(struct led_classdev * ldev)2727d9ee2e8SKurt Kanzenbach static enum led_brightness hellcreek_led_sync_good_get(struct led_classdev *ldev)
2737d9ee2e8SKurt Kanzenbach {
2747d9ee2e8SKurt Kanzenbach 	struct hellcreek *hellcreek = led_to_hellcreek(ldev, led_sync_good);
2757d9ee2e8SKurt Kanzenbach 
2767d9ee2e8SKurt Kanzenbach 	return hellcreek_get_brightness(hellcreek, STATUS_OUT_SYNC_GOOD);
2777d9ee2e8SKurt Kanzenbach }
2787d9ee2e8SKurt Kanzenbach 
hellcreek_led_is_gm_set(struct led_classdev * ldev,enum led_brightness b)2797d9ee2e8SKurt Kanzenbach static void hellcreek_led_is_gm_set(struct led_classdev *ldev,
2807d9ee2e8SKurt Kanzenbach 				    enum led_brightness b)
2817d9ee2e8SKurt Kanzenbach {
2827d9ee2e8SKurt Kanzenbach 	struct hellcreek *hellcreek = led_to_hellcreek(ldev, led_is_gm);
2837d9ee2e8SKurt Kanzenbach 
2847d9ee2e8SKurt Kanzenbach 	hellcreek_set_brightness(hellcreek, STATUS_OUT_IS_GM, b);
2857d9ee2e8SKurt Kanzenbach }
2867d9ee2e8SKurt Kanzenbach 
hellcreek_led_is_gm_get(struct led_classdev * ldev)2877d9ee2e8SKurt Kanzenbach static enum led_brightness hellcreek_led_is_gm_get(struct led_classdev *ldev)
2887d9ee2e8SKurt Kanzenbach {
2897d9ee2e8SKurt Kanzenbach 	struct hellcreek *hellcreek = led_to_hellcreek(ldev, led_is_gm);
2907d9ee2e8SKurt Kanzenbach 
2917d9ee2e8SKurt Kanzenbach 	return hellcreek_get_brightness(hellcreek, STATUS_OUT_IS_GM);
2927d9ee2e8SKurt Kanzenbach }
2937d9ee2e8SKurt Kanzenbach 
2947d9ee2e8SKurt Kanzenbach /* There two available LEDs internally called sync_good and is_gm. However, the
2957d9ee2e8SKurt Kanzenbach  * user might want to use a different label and specify the default state. Take
2967d9ee2e8SKurt Kanzenbach  * those properties from device tree.
2977d9ee2e8SKurt Kanzenbach  */
hellcreek_led_setup(struct hellcreek * hellcreek)2987d9ee2e8SKurt Kanzenbach static int hellcreek_led_setup(struct hellcreek *hellcreek)
2997d9ee2e8SKurt Kanzenbach {
3007d9ee2e8SKurt Kanzenbach 	struct device_node *leds, *led = NULL;
301d565263bSAndy Shevchenko 	enum led_default_state state;
302d565263bSAndy Shevchenko 	const char *label;
3037d9ee2e8SKurt Kanzenbach 	int ret = -EINVAL;
3047d9ee2e8SKurt Kanzenbach 
30516d584d2SLiang He 	of_node_get(hellcreek->dev->of_node);
3067d9ee2e8SKurt Kanzenbach 	leds = of_find_node_by_name(hellcreek->dev->of_node, "leds");
3077d9ee2e8SKurt Kanzenbach 	if (!leds) {
3087d9ee2e8SKurt Kanzenbach 		dev_err(hellcreek->dev, "No LEDs specified in device tree!\n");
3097d9ee2e8SKurt Kanzenbach 		return ret;
3107d9ee2e8SKurt Kanzenbach 	}
3117d9ee2e8SKurt Kanzenbach 
3127d9ee2e8SKurt Kanzenbach 	hellcreek->status_out = 0;
3137d9ee2e8SKurt Kanzenbach 
3147d9ee2e8SKurt Kanzenbach 	led = of_get_next_available_child(leds, led);
3157d9ee2e8SKurt Kanzenbach 	if (!led) {
3167d9ee2e8SKurt Kanzenbach 		dev_err(hellcreek->dev, "First LED not specified!\n");
3177d9ee2e8SKurt Kanzenbach 		goto out;
3187d9ee2e8SKurt Kanzenbach 	}
3197d9ee2e8SKurt Kanzenbach 
3207d9ee2e8SKurt Kanzenbach 	ret = of_property_read_string(led, "label", &label);
3217d9ee2e8SKurt Kanzenbach 	hellcreek->led_sync_good.name = ret ? "sync_good" : label;
3227d9ee2e8SKurt Kanzenbach 
323d565263bSAndy Shevchenko 	state = led_init_default_state_get(of_fwnode_handle(led));
324d565263bSAndy Shevchenko 	switch (state) {
325d565263bSAndy Shevchenko 	case LEDS_DEFSTATE_ON:
3267d9ee2e8SKurt Kanzenbach 		hellcreek->led_sync_good.brightness = 1;
327d565263bSAndy Shevchenko 		break;
328d565263bSAndy Shevchenko 	case LEDS_DEFSTATE_KEEP:
3297d9ee2e8SKurt Kanzenbach 		hellcreek->led_sync_good.brightness =
330d565263bSAndy Shevchenko 			hellcreek_get_brightness(hellcreek, STATUS_OUT_SYNC_GOOD);
331d565263bSAndy Shevchenko 		break;
332d565263bSAndy Shevchenko 	default:
333d565263bSAndy Shevchenko 		hellcreek->led_sync_good.brightness = 0;
3347d9ee2e8SKurt Kanzenbach 	}
3357d9ee2e8SKurt Kanzenbach 
3367d9ee2e8SKurt Kanzenbach 	hellcreek->led_sync_good.max_brightness = 1;
3377d9ee2e8SKurt Kanzenbach 	hellcreek->led_sync_good.brightness_set = hellcreek_led_sync_good_set;
3387d9ee2e8SKurt Kanzenbach 	hellcreek->led_sync_good.brightness_get = hellcreek_led_sync_good_get;
3397d9ee2e8SKurt Kanzenbach 
3407d9ee2e8SKurt Kanzenbach 	led = of_get_next_available_child(leds, led);
3417d9ee2e8SKurt Kanzenbach 	if (!led) {
3427d9ee2e8SKurt Kanzenbach 		dev_err(hellcreek->dev, "Second LED not specified!\n");
3437d9ee2e8SKurt Kanzenbach 		ret = -EINVAL;
3447d9ee2e8SKurt Kanzenbach 		goto out;
3457d9ee2e8SKurt Kanzenbach 	}
3467d9ee2e8SKurt Kanzenbach 
3477d9ee2e8SKurt Kanzenbach 	ret = of_property_read_string(led, "label", &label);
3487d9ee2e8SKurt Kanzenbach 	hellcreek->led_is_gm.name = ret ? "is_gm" : label;
3497d9ee2e8SKurt Kanzenbach 
350d565263bSAndy Shevchenko 	state = led_init_default_state_get(of_fwnode_handle(led));
351d565263bSAndy Shevchenko 	switch (state) {
352d565263bSAndy Shevchenko 	case LEDS_DEFSTATE_ON:
3537d9ee2e8SKurt Kanzenbach 		hellcreek->led_is_gm.brightness = 1;
354d565263bSAndy Shevchenko 		break;
355d565263bSAndy Shevchenko 	case LEDS_DEFSTATE_KEEP:
3567d9ee2e8SKurt Kanzenbach 		hellcreek->led_is_gm.brightness =
357d565263bSAndy Shevchenko 			hellcreek_get_brightness(hellcreek, STATUS_OUT_IS_GM);
358d565263bSAndy Shevchenko 		break;
359d565263bSAndy Shevchenko 	default:
360d565263bSAndy Shevchenko 		hellcreek->led_is_gm.brightness = 0;
3617d9ee2e8SKurt Kanzenbach 	}
3627d9ee2e8SKurt Kanzenbach 
3637d9ee2e8SKurt Kanzenbach 	hellcreek->led_is_gm.max_brightness = 1;
3647d9ee2e8SKurt Kanzenbach 	hellcreek->led_is_gm.brightness_set = hellcreek_led_is_gm_set;
3657d9ee2e8SKurt Kanzenbach 	hellcreek->led_is_gm.brightness_get = hellcreek_led_is_gm_get;
3667d9ee2e8SKurt Kanzenbach 
3677d9ee2e8SKurt Kanzenbach 	/* Set initial state */
3687d9ee2e8SKurt Kanzenbach 	if (hellcreek->led_sync_good.brightness == 1)
3697d9ee2e8SKurt Kanzenbach 		hellcreek_set_brightness(hellcreek, STATUS_OUT_SYNC_GOOD, 1);
3707d9ee2e8SKurt Kanzenbach 	if (hellcreek->led_is_gm.brightness == 1)
3717d9ee2e8SKurt Kanzenbach 		hellcreek_set_brightness(hellcreek, STATUS_OUT_IS_GM, 1);
3727d9ee2e8SKurt Kanzenbach 
3737d9ee2e8SKurt Kanzenbach 	/* Register both leds */
3747d9ee2e8SKurt Kanzenbach 	led_classdev_register(hellcreek->dev, &hellcreek->led_sync_good);
3757d9ee2e8SKurt Kanzenbach 	led_classdev_register(hellcreek->dev, &hellcreek->led_is_gm);
3767d9ee2e8SKurt Kanzenbach 
3777d9ee2e8SKurt Kanzenbach 	ret = 0;
3787d9ee2e8SKurt Kanzenbach 
3797d9ee2e8SKurt Kanzenbach out:
3807d9ee2e8SKurt Kanzenbach 	of_node_put(leds);
3817d9ee2e8SKurt Kanzenbach 
3827d9ee2e8SKurt Kanzenbach 	return ret;
3837d9ee2e8SKurt Kanzenbach }
3847d9ee2e8SKurt Kanzenbach 
hellcreek_ptp_setup(struct hellcreek * hellcreek)385ddd56dfeSKamil Alkhouri int hellcreek_ptp_setup(struct hellcreek *hellcreek)
386ddd56dfeSKamil Alkhouri {
387ddd56dfeSKamil Alkhouri 	u16 status;
3887d9ee2e8SKurt Kanzenbach 	int ret;
389ddd56dfeSKamil Alkhouri 
390ddd56dfeSKamil Alkhouri 	/* Set up the overflow work */
391ddd56dfeSKamil Alkhouri 	INIT_DELAYED_WORK(&hellcreek->overflow_work,
392ddd56dfeSKamil Alkhouri 			  hellcreek_ptp_overflow_check);
393ddd56dfeSKamil Alkhouri 
394ddd56dfeSKamil Alkhouri 	/* Setup PTP clock */
395ddd56dfeSKamil Alkhouri 	hellcreek->ptp_clock_info.owner = THIS_MODULE;
396ddd56dfeSKamil Alkhouri 	snprintf(hellcreek->ptp_clock_info.name,
397ddd56dfeSKamil Alkhouri 		 sizeof(hellcreek->ptp_clock_info.name),
398ddd56dfeSKamil Alkhouri 		 dev_name(hellcreek->dev));
399ddd56dfeSKamil Alkhouri 
400ddd56dfeSKamil Alkhouri 	/* IP-Core can add up to 0.5 ns per 8 ns cycle, which means
401ddd56dfeSKamil Alkhouri 	 * accumulator_overflow_rate shall not exceed 62.5 MHz (which adjusts
402ddd56dfeSKamil Alkhouri 	 * the nominal frequency by 6.25%)
403ddd56dfeSKamil Alkhouri 	 */
404ddd56dfeSKamil Alkhouri 	hellcreek->ptp_clock_info.max_adj     = 62500000;
405ddd56dfeSKamil Alkhouri 	hellcreek->ptp_clock_info.n_alarm     = 0;
406ddd56dfeSKamil Alkhouri 	hellcreek->ptp_clock_info.n_pins      = 0;
407ddd56dfeSKamil Alkhouri 	hellcreek->ptp_clock_info.n_ext_ts    = 0;
408ddd56dfeSKamil Alkhouri 	hellcreek->ptp_clock_info.n_per_out   = 0;
409ddd56dfeSKamil Alkhouri 	hellcreek->ptp_clock_info.pps	      = 0;
410ddd56dfeSKamil Alkhouri 	hellcreek->ptp_clock_info.adjfine     = hellcreek_ptp_adjfine;
411ddd56dfeSKamil Alkhouri 	hellcreek->ptp_clock_info.adjtime     = hellcreek_ptp_adjtime;
412ddd56dfeSKamil Alkhouri 	hellcreek->ptp_clock_info.gettime64   = hellcreek_ptp_gettime;
413ddd56dfeSKamil Alkhouri 	hellcreek->ptp_clock_info.settime64   = hellcreek_ptp_settime;
414ddd56dfeSKamil Alkhouri 	hellcreek->ptp_clock_info.enable      = hellcreek_ptp_enable;
415f0d4ba9eSKamil Alkhouri 	hellcreek->ptp_clock_info.do_aux_work = hellcreek_hwtstamp_work;
416ddd56dfeSKamil Alkhouri 
417ddd56dfeSKamil Alkhouri 	hellcreek->ptp_clock = ptp_clock_register(&hellcreek->ptp_clock_info,
418ddd56dfeSKamil Alkhouri 						  hellcreek->dev);
419ddd56dfeSKamil Alkhouri 	if (IS_ERR(hellcreek->ptp_clock))
420ddd56dfeSKamil Alkhouri 		return PTR_ERR(hellcreek->ptp_clock);
421ddd56dfeSKamil Alkhouri 
422ddd56dfeSKamil Alkhouri 	/* Enable the offset correction process, if no offset correction is
423ddd56dfeSKamil Alkhouri 	 * already taking place
424ddd56dfeSKamil Alkhouri 	 */
425ddd56dfeSKamil Alkhouri 	status = hellcreek_ptp_read(hellcreek, PR_CLOCK_STATUS_C);
426ddd56dfeSKamil Alkhouri 	if (!(status & PR_CLOCK_STATUS_C_OFS_ACT))
427ddd56dfeSKamil Alkhouri 		hellcreek_ptp_write(hellcreek,
428ddd56dfeSKamil Alkhouri 				    status | PR_CLOCK_STATUS_C_ENA_OFS,
429ddd56dfeSKamil Alkhouri 				    PR_CLOCK_STATUS_C);
430ddd56dfeSKamil Alkhouri 
431ddd56dfeSKamil Alkhouri 	/* Enable the drift correction process */
432ddd56dfeSKamil Alkhouri 	hellcreek_ptp_write(hellcreek, status | PR_CLOCK_STATUS_C_ENA_DRIFT,
433ddd56dfeSKamil Alkhouri 			    PR_CLOCK_STATUS_C);
434ddd56dfeSKamil Alkhouri 
4357d9ee2e8SKurt Kanzenbach 	/* LED setup */
4367d9ee2e8SKurt Kanzenbach 	ret = hellcreek_led_setup(hellcreek);
4377d9ee2e8SKurt Kanzenbach 	if (ret) {
4387d9ee2e8SKurt Kanzenbach 		if (hellcreek->ptp_clock)
4397d9ee2e8SKurt Kanzenbach 			ptp_clock_unregister(hellcreek->ptp_clock);
4407d9ee2e8SKurt Kanzenbach 		return ret;
4417d9ee2e8SKurt Kanzenbach 	}
4427d9ee2e8SKurt Kanzenbach 
443ddd56dfeSKamil Alkhouri 	schedule_delayed_work(&hellcreek->overflow_work,
444ddd56dfeSKamil Alkhouri 			      HELLCREEK_OVERFLOW_PERIOD);
445ddd56dfeSKamil Alkhouri 
446ddd56dfeSKamil Alkhouri 	return 0;
447ddd56dfeSKamil Alkhouri }
448ddd56dfeSKamil Alkhouri 
hellcreek_ptp_free(struct hellcreek * hellcreek)449ddd56dfeSKamil Alkhouri void hellcreek_ptp_free(struct hellcreek *hellcreek)
450ddd56dfeSKamil Alkhouri {
4517d9ee2e8SKurt Kanzenbach 	led_classdev_unregister(&hellcreek->led_is_gm);
4527d9ee2e8SKurt Kanzenbach 	led_classdev_unregister(&hellcreek->led_sync_good);
453ddd56dfeSKamil Alkhouri 	cancel_delayed_work_sync(&hellcreek->overflow_work);
454ddd56dfeSKamil Alkhouri 	if (hellcreek->ptp_clock)
455ddd56dfeSKamil Alkhouri 		ptp_clock_unregister(hellcreek->ptp_clock);
456ddd56dfeSKamil Alkhouri 	hellcreek->ptp_clock = NULL;
457ddd56dfeSKamil Alkhouri }
458