xref: /f-stack/freebsd/sys/timepps.h (revision 22ce4aff)
1a9643ea8Slogwang /*-
2*22ce4affSfengbojiang  * SPDX-License-Identifier: Beerware
3*22ce4affSfengbojiang  *
4a9643ea8Slogwang  * ----------------------------------------------------------------------------
5a9643ea8Slogwang  * "THE BEER-WARE LICENSE" (Revision 42):
6a9643ea8Slogwang  * <[email protected]> wrote this file.  As long as you retain this notice you
7a9643ea8Slogwang  * can do whatever you want with this stuff. If we meet some day, and you think
8a9643ea8Slogwang  * this stuff is worth it, you can buy me a beer in return.   Poul-Henning Kamp
9a9643ea8Slogwang  * ----------------------------------------------------------------------------
10a9643ea8Slogwang  *
11a9643ea8Slogwang  * Copyright (c) 2011 The FreeBSD Foundation
12a9643ea8Slogwang  * All rights reserved.
13a9643ea8Slogwang  *
14a9643ea8Slogwang  * Portions of this software were developed by Julien Ridoux at the University
15a9643ea8Slogwang  * of Melbourne under sponsorship from the FreeBSD Foundation.
16a9643ea8Slogwang  *
17a9643ea8Slogwang  * $FreeBSD$
18a9643ea8Slogwang  *
19a9643ea8Slogwang  * The is a FreeBSD version of the RFC 2783 API for Pulse Per Second
20a9643ea8Slogwang  * timing interfaces.
21a9643ea8Slogwang  */
22a9643ea8Slogwang 
23a9643ea8Slogwang #ifndef _SYS_TIMEPPS_H_
24a9643ea8Slogwang #define _SYS_TIMEPPS_H_
25a9643ea8Slogwang 
26a9643ea8Slogwang #include <sys/_ffcounter.h>
27a9643ea8Slogwang #include <sys/ioccom.h>
28a9643ea8Slogwang #include <sys/time.h>
29a9643ea8Slogwang 
30a9643ea8Slogwang #define PPS_API_VERS_1	1
31a9643ea8Slogwang 
32a9643ea8Slogwang typedef int pps_handle_t;
33a9643ea8Slogwang 
34a9643ea8Slogwang typedef unsigned pps_seq_t;
35a9643ea8Slogwang 
36a9643ea8Slogwang typedef struct ntp_fp {
37a9643ea8Slogwang 	unsigned int	integral;
38a9643ea8Slogwang 	unsigned int	fractional;
39a9643ea8Slogwang } ntp_fp_t;
40a9643ea8Slogwang 
41a9643ea8Slogwang typedef union pps_timeu {
42a9643ea8Slogwang 	struct timespec	tspec;
43a9643ea8Slogwang 	ntp_fp_t	ntpfp;
44a9643ea8Slogwang 	unsigned long	longpad[3];
45a9643ea8Slogwang } pps_timeu_t;
46a9643ea8Slogwang 
47a9643ea8Slogwang typedef struct {
48a9643ea8Slogwang 	pps_seq_t	assert_sequence;	/* assert event seq # */
49a9643ea8Slogwang 	pps_seq_t	clear_sequence;		/* clear event seq # */
50a9643ea8Slogwang 	pps_timeu_t	assert_tu;
51a9643ea8Slogwang 	pps_timeu_t	clear_tu;
52a9643ea8Slogwang 	int		current_mode;		/* current mode bits */
53a9643ea8Slogwang } pps_info_t;
54a9643ea8Slogwang 
55a9643ea8Slogwang typedef struct {
56a9643ea8Slogwang 	pps_seq_t	assert_sequence;	/* assert event seq # */
57a9643ea8Slogwang 	pps_seq_t	clear_sequence;		/* clear event seq # */
58a9643ea8Slogwang 	pps_timeu_t	assert_tu;
59a9643ea8Slogwang 	pps_timeu_t	clear_tu;
60a9643ea8Slogwang 	ffcounter	assert_ffcount;		/* ffcounter on assert event */
61a9643ea8Slogwang 	ffcounter	clear_ffcount;		/* ffcounter on clear event */
62a9643ea8Slogwang 	int		current_mode;		/* current mode bits */
63a9643ea8Slogwang } pps_info_ffc_t;
64a9643ea8Slogwang 
65a9643ea8Slogwang #define assert_timestamp        assert_tu.tspec
66a9643ea8Slogwang #define clear_timestamp         clear_tu.tspec
67a9643ea8Slogwang 
68a9643ea8Slogwang #define assert_timestamp_ntpfp  assert_tu.ntpfp
69a9643ea8Slogwang #define clear_timestamp_ntpfp   clear_tu.ntpfp
70a9643ea8Slogwang 
71a9643ea8Slogwang typedef struct {
72a9643ea8Slogwang 	int api_version;			/* API version # */
73a9643ea8Slogwang 	int mode;				/* mode bits */
74a9643ea8Slogwang 	pps_timeu_t assert_off_tu;
75a9643ea8Slogwang 	pps_timeu_t clear_off_tu;
76a9643ea8Slogwang } pps_params_t;
77a9643ea8Slogwang 
78a9643ea8Slogwang #define assert_offset   assert_off_tu.tspec
79a9643ea8Slogwang #define clear_offset    clear_off_tu.tspec
80a9643ea8Slogwang 
81a9643ea8Slogwang #define assert_offset_ntpfp     assert_off_tu.ntpfp
82a9643ea8Slogwang #define clear_offset_ntpfp      clear_off_tu.ntpfp
83a9643ea8Slogwang 
84a9643ea8Slogwang #define PPS_CAPTUREASSERT	0x01
85a9643ea8Slogwang #define PPS_CAPTURECLEAR	0x02
86a9643ea8Slogwang #define PPS_CAPTUREBOTH		0x03
87a9643ea8Slogwang 
88a9643ea8Slogwang #define PPS_OFFSETASSERT	0x10
89a9643ea8Slogwang #define PPS_OFFSETCLEAR		0x20
90a9643ea8Slogwang 
91a9643ea8Slogwang #define PPS_ECHOASSERT		0x40
92a9643ea8Slogwang #define PPS_ECHOCLEAR		0x80
93a9643ea8Slogwang 
94a9643ea8Slogwang #define PPS_CANWAIT		0x100
95a9643ea8Slogwang #define PPS_CANPOLL		0x200
96a9643ea8Slogwang 
97a9643ea8Slogwang #define PPS_TSFMT_TSPEC		0x1000
98a9643ea8Slogwang #define PPS_TSFMT_NTPFP		0x2000
99a9643ea8Slogwang 
100a9643ea8Slogwang #define	PPS_TSCLK_FBCK		0x10000
101a9643ea8Slogwang #define	PPS_TSCLK_FFWD		0x20000
102a9643ea8Slogwang #define	PPS_TSCLK_MASK		0x30000
103a9643ea8Slogwang 
104a9643ea8Slogwang #define PPS_KC_HARDPPS		0
105a9643ea8Slogwang #define PPS_KC_HARDPPS_PLL	1
106a9643ea8Slogwang #define PPS_KC_HARDPPS_FLL	2
107a9643ea8Slogwang 
108a9643ea8Slogwang struct pps_fetch_args {
109a9643ea8Slogwang 	int tsformat;
110a9643ea8Slogwang 	pps_info_t	pps_info_buf;
111a9643ea8Slogwang 	struct timespec	timeout;
112a9643ea8Slogwang };
113a9643ea8Slogwang 
114a9643ea8Slogwang struct pps_fetch_ffc_args {
115a9643ea8Slogwang 	int		tsformat;
116a9643ea8Slogwang 	pps_info_ffc_t	pps_info_buf_ffc;
117a9643ea8Slogwang 	struct timespec	timeout;
118a9643ea8Slogwang };
119a9643ea8Slogwang 
120a9643ea8Slogwang struct pps_kcbind_args {
121a9643ea8Slogwang 	int kernel_consumer;
122a9643ea8Slogwang 	int edge;
123a9643ea8Slogwang 	int tsformat;
124a9643ea8Slogwang };
125a9643ea8Slogwang 
126a9643ea8Slogwang #define PPS_IOC_CREATE		_IO('1', 1)
127a9643ea8Slogwang #define PPS_IOC_DESTROY		_IO('1', 2)
128a9643ea8Slogwang #define PPS_IOC_SETPARAMS	_IOW('1', 3, pps_params_t)
129a9643ea8Slogwang #define PPS_IOC_GETPARAMS	_IOR('1', 4, pps_params_t)
130a9643ea8Slogwang #define PPS_IOC_GETCAP		_IOR('1', 5, int)
131a9643ea8Slogwang #define PPS_IOC_FETCH		_IOWR('1', 6, struct pps_fetch_args)
132a9643ea8Slogwang #define PPS_IOC_KCBIND		_IOW('1', 7, struct pps_kcbind_args)
133a9643ea8Slogwang #define	PPS_IOC_FETCH_FFCOUNTER	_IOWR('1', 8, struct pps_fetch_ffc_args)
134a9643ea8Slogwang 
135a9643ea8Slogwang #ifdef _KERNEL
136a9643ea8Slogwang 
137a9643ea8Slogwang struct mtx;
138a9643ea8Slogwang 
139a9643ea8Slogwang #define	KCMODE_EDGEMASK		0x03
140a9643ea8Slogwang #define	KCMODE_ABIFLAG		0x80000000 /* Internal use: abi-aware driver. */
141a9643ea8Slogwang 
142a9643ea8Slogwang #define	PPS_ABI_VERSION		1
143a9643ea8Slogwang 
144a9643ea8Slogwang #define	PPSFLAG_MTX_SPIN	0x01	/* Driver mtx is MTX_SPIN type. */
145a9643ea8Slogwang 
146a9643ea8Slogwang struct pps_state {
147a9643ea8Slogwang 	/* Capture information. */
148a9643ea8Slogwang 	struct timehands *capth;
149a9643ea8Slogwang 	struct fftimehands *capffth;
150a9643ea8Slogwang 	unsigned	capgen;
151a9643ea8Slogwang 	unsigned	capcount;
152a9643ea8Slogwang 
153a9643ea8Slogwang 	/* State information. */
154a9643ea8Slogwang 	pps_params_t	ppsparam;
155a9643ea8Slogwang 	pps_info_t	ppsinfo;
156a9643ea8Slogwang 	pps_info_ffc_t	ppsinfo_ffc;
157a9643ea8Slogwang 	int		kcmode;
158a9643ea8Slogwang 	int		ppscap;
159a9643ea8Slogwang 	struct timecounter *ppstc;
160a9643ea8Slogwang 	unsigned	ppscount[3];
161a9643ea8Slogwang 	/*
162a9643ea8Slogwang 	 * The following fields are valid if the driver calls pps_init_abi().
163a9643ea8Slogwang 	 */
164a9643ea8Slogwang 	uint16_t	driver_abi;	/* Driver sets before pps_init_abi(). */
165a9643ea8Slogwang 	uint16_t	kernel_abi;	/* Kernel sets during pps_init_abi(). */
166a9643ea8Slogwang 	struct mtx	*driver_mtx;	/* Optional, valid if non-NULL. */
167a9643ea8Slogwang 	uint32_t	flags;
168a9643ea8Slogwang };
169a9643ea8Slogwang 
170a9643ea8Slogwang void pps_capture(struct pps_state *pps);
171a9643ea8Slogwang void pps_event(struct pps_state *pps, int event);
172a9643ea8Slogwang void pps_init(struct pps_state *pps);
173a9643ea8Slogwang void pps_init_abi(struct pps_state *pps);
174a9643ea8Slogwang int pps_ioctl(unsigned long cmd, caddr_t data, struct pps_state *pps);
175a9643ea8Slogwang void hardpps(struct timespec *tsp, long nsec);
176a9643ea8Slogwang 
177a9643ea8Slogwang #else /* !_KERNEL */
178a9643ea8Slogwang 
179a9643ea8Slogwang static __inline int
time_pps_create(int filedes,pps_handle_t * handle)180a9643ea8Slogwang time_pps_create(int filedes, pps_handle_t *handle)
181a9643ea8Slogwang {
182a9643ea8Slogwang 	int error;
183a9643ea8Slogwang 
184a9643ea8Slogwang 	*handle = -1;
185a9643ea8Slogwang 	error = ioctl(filedes, PPS_IOC_CREATE, 0);
186a9643ea8Slogwang 	if (error < 0)
187a9643ea8Slogwang 		return (-1);
188a9643ea8Slogwang 	*handle = filedes;
189a9643ea8Slogwang 	return (0);
190a9643ea8Slogwang }
191a9643ea8Slogwang 
192a9643ea8Slogwang static __inline int
time_pps_destroy(pps_handle_t handle)193a9643ea8Slogwang time_pps_destroy(pps_handle_t handle)
194a9643ea8Slogwang {
195a9643ea8Slogwang 	return (ioctl(handle, PPS_IOC_DESTROY, 0));
196a9643ea8Slogwang }
197a9643ea8Slogwang 
198a9643ea8Slogwang static __inline int
time_pps_setparams(pps_handle_t handle,const pps_params_t * ppsparams)199a9643ea8Slogwang time_pps_setparams(pps_handle_t handle, const pps_params_t *ppsparams)
200a9643ea8Slogwang {
201a9643ea8Slogwang 	return (ioctl(handle, PPS_IOC_SETPARAMS, ppsparams));
202a9643ea8Slogwang }
203a9643ea8Slogwang 
204a9643ea8Slogwang static __inline int
time_pps_getparams(pps_handle_t handle,pps_params_t * ppsparams)205a9643ea8Slogwang time_pps_getparams(pps_handle_t handle, pps_params_t *ppsparams)
206a9643ea8Slogwang {
207a9643ea8Slogwang 	return (ioctl(handle, PPS_IOC_GETPARAMS, ppsparams));
208a9643ea8Slogwang }
209a9643ea8Slogwang 
210a9643ea8Slogwang static __inline int
time_pps_getcap(pps_handle_t handle,int * mode)211a9643ea8Slogwang time_pps_getcap(pps_handle_t handle, int *mode)
212a9643ea8Slogwang {
213a9643ea8Slogwang 	return (ioctl(handle, PPS_IOC_GETCAP, mode));
214a9643ea8Slogwang }
215a9643ea8Slogwang 
216a9643ea8Slogwang static __inline int
time_pps_fetch(pps_handle_t handle,const int tsformat,pps_info_t * ppsinfobuf,const struct timespec * timeout)217a9643ea8Slogwang time_pps_fetch(pps_handle_t handle, const int tsformat,
218a9643ea8Slogwang 	pps_info_t *ppsinfobuf, const struct timespec *timeout)
219a9643ea8Slogwang {
220a9643ea8Slogwang 	int error;
221a9643ea8Slogwang 	struct pps_fetch_args arg;
222a9643ea8Slogwang 
223a9643ea8Slogwang 	arg.tsformat = tsformat;
224a9643ea8Slogwang 	if (timeout == NULL) {
225a9643ea8Slogwang 		arg.timeout.tv_sec = -1;
226a9643ea8Slogwang 		arg.timeout.tv_nsec = -1;
227a9643ea8Slogwang 	} else
228a9643ea8Slogwang 		arg.timeout = *timeout;
229a9643ea8Slogwang 	error = ioctl(handle, PPS_IOC_FETCH, &arg);
230a9643ea8Slogwang 	*ppsinfobuf = arg.pps_info_buf;
231a9643ea8Slogwang 	return (error);
232a9643ea8Slogwang }
233a9643ea8Slogwang 
234a9643ea8Slogwang static __inline int
time_pps_fetch_ffc(pps_handle_t handle,const int tsformat,pps_info_ffc_t * ppsinfobuf,const struct timespec * timeout)235a9643ea8Slogwang time_pps_fetch_ffc(pps_handle_t handle, const int tsformat,
236a9643ea8Slogwang 	pps_info_ffc_t *ppsinfobuf, const struct timespec *timeout)
237a9643ea8Slogwang {
238a9643ea8Slogwang 	struct pps_fetch_ffc_args arg;
239a9643ea8Slogwang 	int error;
240a9643ea8Slogwang 
241a9643ea8Slogwang 	arg.tsformat = tsformat;
242a9643ea8Slogwang 	if (timeout == NULL) {
243a9643ea8Slogwang 		arg.timeout.tv_sec = -1;
244a9643ea8Slogwang 		arg.timeout.tv_nsec = -1;
245a9643ea8Slogwang 	} else {
246a9643ea8Slogwang 		arg.timeout = *timeout;
247a9643ea8Slogwang 	}
248a9643ea8Slogwang 	error = ioctl(handle, PPS_IOC_FETCH_FFCOUNTER, &arg);
249a9643ea8Slogwang 	*ppsinfobuf = arg.pps_info_buf_ffc;
250a9643ea8Slogwang 	return (error);
251a9643ea8Slogwang }
252a9643ea8Slogwang 
253a9643ea8Slogwang static __inline int
time_pps_kcbind(pps_handle_t handle,const int kernel_consumer,const int edge,const int tsformat)254a9643ea8Slogwang time_pps_kcbind(pps_handle_t handle, const int kernel_consumer,
255a9643ea8Slogwang 	const int edge, const int tsformat)
256a9643ea8Slogwang {
257a9643ea8Slogwang 	struct pps_kcbind_args arg;
258a9643ea8Slogwang 
259a9643ea8Slogwang 	arg.kernel_consumer = kernel_consumer;
260a9643ea8Slogwang 	arg.edge = edge;
261a9643ea8Slogwang 	arg.tsformat = tsformat;
262a9643ea8Slogwang 	return (ioctl(handle, PPS_IOC_KCBIND, &arg));
263a9643ea8Slogwang }
264a9643ea8Slogwang 
265a9643ea8Slogwang #endif /* KERNEL */
266a9643ea8Slogwang 
267a9643ea8Slogwang #endif /* !_SYS_TIMEPPS_H_ */
268