1*d94ba80eSRichard Cochran /*
2*d94ba80eSRichard Cochran  * PTP 1588 clock support
3*d94ba80eSRichard Cochran  *
4*d94ba80eSRichard Cochran  * Copyright (C) 2010 OMICRON electronics GmbH
5*d94ba80eSRichard Cochran  *
6*d94ba80eSRichard Cochran  *  This program is free software; you can redistribute it and/or modify
7*d94ba80eSRichard Cochran  *  it under the terms of the GNU General Public License as published by
8*d94ba80eSRichard Cochran  *  the Free Software Foundation; either version 2 of the License, or
9*d94ba80eSRichard Cochran  *  (at your option) any later version.
10*d94ba80eSRichard Cochran  *
11*d94ba80eSRichard Cochran  *  This program is distributed in the hope that it will be useful,
12*d94ba80eSRichard Cochran  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
13*d94ba80eSRichard Cochran  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14*d94ba80eSRichard Cochran  *  GNU General Public License for more details.
15*d94ba80eSRichard Cochran  *
16*d94ba80eSRichard Cochran  *  You should have received a copy of the GNU General Public License
17*d94ba80eSRichard Cochran  *  along with this program; if not, write to the Free Software
18*d94ba80eSRichard Cochran  *  Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
19*d94ba80eSRichard Cochran  */
20*d94ba80eSRichard Cochran 
21*d94ba80eSRichard Cochran #ifndef _PTP_CLOCK_KERNEL_H_
22*d94ba80eSRichard Cochran #define _PTP_CLOCK_KERNEL_H_
23*d94ba80eSRichard Cochran 
24*d94ba80eSRichard Cochran #include <linux/ptp_clock.h>
25*d94ba80eSRichard Cochran 
26*d94ba80eSRichard Cochran 
27*d94ba80eSRichard Cochran struct ptp_clock_request {
28*d94ba80eSRichard Cochran 	enum {
29*d94ba80eSRichard Cochran 		PTP_CLK_REQ_EXTTS,
30*d94ba80eSRichard Cochran 		PTP_CLK_REQ_PEROUT,
31*d94ba80eSRichard Cochran 		PTP_CLK_REQ_PPS,
32*d94ba80eSRichard Cochran 	} type;
33*d94ba80eSRichard Cochran 	union {
34*d94ba80eSRichard Cochran 		struct ptp_extts_request extts;
35*d94ba80eSRichard Cochran 		struct ptp_perout_request perout;
36*d94ba80eSRichard Cochran 	};
37*d94ba80eSRichard Cochran };
38*d94ba80eSRichard Cochran 
39*d94ba80eSRichard Cochran /**
40*d94ba80eSRichard Cochran  * struct ptp_clock_info - decribes a PTP hardware clock
41*d94ba80eSRichard Cochran  *
42*d94ba80eSRichard Cochran  * @owner:     The clock driver should set to THIS_MODULE.
43*d94ba80eSRichard Cochran  * @name:      A short name to identify the clock.
44*d94ba80eSRichard Cochran  * @max_adj:   The maximum possible frequency adjustment, in parts per billon.
45*d94ba80eSRichard Cochran  * @n_alarm:   The number of programmable alarms.
46*d94ba80eSRichard Cochran  * @n_ext_ts:  The number of external time stamp channels.
47*d94ba80eSRichard Cochran  * @n_per_out: The number of programmable periodic signals.
48*d94ba80eSRichard Cochran  * @pps:       Indicates whether the clock supports a PPS callback.
49*d94ba80eSRichard Cochran  *
50*d94ba80eSRichard Cochran  * clock operations
51*d94ba80eSRichard Cochran  *
52*d94ba80eSRichard Cochran  * @adjfreq:  Adjusts the frequency of the hardware clock.
53*d94ba80eSRichard Cochran  *            parameter delta: Desired period change in parts per billion.
54*d94ba80eSRichard Cochran  *
55*d94ba80eSRichard Cochran  * @adjtime:  Shifts the time of the hardware clock.
56*d94ba80eSRichard Cochran  *            parameter delta: Desired change in nanoseconds.
57*d94ba80eSRichard Cochran  *
58*d94ba80eSRichard Cochran  * @gettime:  Reads the current time from the hardware clock.
59*d94ba80eSRichard Cochran  *            parameter ts: Holds the result.
60*d94ba80eSRichard Cochran  *
61*d94ba80eSRichard Cochran  * @settime:  Set the current time on the hardware clock.
62*d94ba80eSRichard Cochran  *            parameter ts: Time value to set.
63*d94ba80eSRichard Cochran  *
64*d94ba80eSRichard Cochran  * @enable:   Request driver to enable or disable an ancillary feature.
65*d94ba80eSRichard Cochran  *            parameter request: Desired resource to enable or disable.
66*d94ba80eSRichard Cochran  *            parameter on: Caller passes one to enable or zero to disable.
67*d94ba80eSRichard Cochran  *
68*d94ba80eSRichard Cochran  * Drivers should embed their ptp_clock_info within a private
69*d94ba80eSRichard Cochran  * structure, obtaining a reference to it using container_of().
70*d94ba80eSRichard Cochran  *
71*d94ba80eSRichard Cochran  * The callbacks must all return zero on success, non-zero otherwise.
72*d94ba80eSRichard Cochran  */
73*d94ba80eSRichard Cochran 
74*d94ba80eSRichard Cochran struct ptp_clock_info {
75*d94ba80eSRichard Cochran 	struct module *owner;
76*d94ba80eSRichard Cochran 	char name[16];
77*d94ba80eSRichard Cochran 	s32 max_adj;
78*d94ba80eSRichard Cochran 	int n_alarm;
79*d94ba80eSRichard Cochran 	int n_ext_ts;
80*d94ba80eSRichard Cochran 	int n_per_out;
81*d94ba80eSRichard Cochran 	int pps;
82*d94ba80eSRichard Cochran 	int (*adjfreq)(struct ptp_clock_info *ptp, s32 delta);
83*d94ba80eSRichard Cochran 	int (*adjtime)(struct ptp_clock_info *ptp, s64 delta);
84*d94ba80eSRichard Cochran 	int (*gettime)(struct ptp_clock_info *ptp, struct timespec *ts);
85*d94ba80eSRichard Cochran 	int (*settime)(struct ptp_clock_info *ptp, const struct timespec *ts);
86*d94ba80eSRichard Cochran 	int (*enable)(struct ptp_clock_info *ptp,
87*d94ba80eSRichard Cochran 		      struct ptp_clock_request *request, int on);
88*d94ba80eSRichard Cochran };
89*d94ba80eSRichard Cochran 
90*d94ba80eSRichard Cochran struct ptp_clock;
91*d94ba80eSRichard Cochran 
92*d94ba80eSRichard Cochran /**
93*d94ba80eSRichard Cochran  * ptp_clock_register() - register a PTP hardware clock driver
94*d94ba80eSRichard Cochran  *
95*d94ba80eSRichard Cochran  * @info:  Structure describing the new clock.
96*d94ba80eSRichard Cochran  */
97*d94ba80eSRichard Cochran 
98*d94ba80eSRichard Cochran extern struct ptp_clock *ptp_clock_register(struct ptp_clock_info *info);
99*d94ba80eSRichard Cochran 
100*d94ba80eSRichard Cochran /**
101*d94ba80eSRichard Cochran  * ptp_clock_unregister() - unregister a PTP hardware clock driver
102*d94ba80eSRichard Cochran  *
103*d94ba80eSRichard Cochran  * @ptp:  The clock to remove from service.
104*d94ba80eSRichard Cochran  */
105*d94ba80eSRichard Cochran 
106*d94ba80eSRichard Cochran extern int ptp_clock_unregister(struct ptp_clock *ptp);
107*d94ba80eSRichard Cochran 
108*d94ba80eSRichard Cochran 
109*d94ba80eSRichard Cochran enum ptp_clock_events {
110*d94ba80eSRichard Cochran 	PTP_CLOCK_ALARM,
111*d94ba80eSRichard Cochran 	PTP_CLOCK_EXTTS,
112*d94ba80eSRichard Cochran 	PTP_CLOCK_PPS,
113*d94ba80eSRichard Cochran };
114*d94ba80eSRichard Cochran 
115*d94ba80eSRichard Cochran /**
116*d94ba80eSRichard Cochran  * struct ptp_clock_event - decribes a PTP hardware clock event
117*d94ba80eSRichard Cochran  *
118*d94ba80eSRichard Cochran  * @type:  One of the ptp_clock_events enumeration values.
119*d94ba80eSRichard Cochran  * @index: Identifies the source of the event.
120*d94ba80eSRichard Cochran  * @timestamp: When the event occured.
121*d94ba80eSRichard Cochran  */
122*d94ba80eSRichard Cochran 
123*d94ba80eSRichard Cochran struct ptp_clock_event {
124*d94ba80eSRichard Cochran 	int type;
125*d94ba80eSRichard Cochran 	int index;
126*d94ba80eSRichard Cochran 	u64 timestamp;
127*d94ba80eSRichard Cochran };
128*d94ba80eSRichard Cochran 
129*d94ba80eSRichard Cochran /**
130*d94ba80eSRichard Cochran  * ptp_clock_event() - notify the PTP layer about an event
131*d94ba80eSRichard Cochran  *
132*d94ba80eSRichard Cochran  * @ptp:    The clock obtained from ptp_clock_register().
133*d94ba80eSRichard Cochran  * @event:  Message structure describing the event.
134*d94ba80eSRichard Cochran  */
135*d94ba80eSRichard Cochran 
136*d94ba80eSRichard Cochran extern void ptp_clock_event(struct ptp_clock *ptp,
137*d94ba80eSRichard Cochran 			    struct ptp_clock_event *event);
138*d94ba80eSRichard Cochran 
139*d94ba80eSRichard Cochran #endif
140