18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later
28c2ecf20Sopenharmony_ci/*
38c2ecf20Sopenharmony_ci * pps-ktimer.c -- kernel timer test client
48c2ecf20Sopenharmony_ci *
58c2ecf20Sopenharmony_ci * Copyright (C) 2005-2006   Rodolfo Giometti <giometti@linux.it>
68c2ecf20Sopenharmony_ci */
78c2ecf20Sopenharmony_ci
88c2ecf20Sopenharmony_ci#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
98c2ecf20Sopenharmony_ci
108c2ecf20Sopenharmony_ci#include <linux/kernel.h>
118c2ecf20Sopenharmony_ci#include <linux/module.h>
128c2ecf20Sopenharmony_ci#include <linux/init.h>
138c2ecf20Sopenharmony_ci#include <linux/time.h>
148c2ecf20Sopenharmony_ci#include <linux/timer.h>
158c2ecf20Sopenharmony_ci#include <linux/pps_kernel.h>
168c2ecf20Sopenharmony_ci
178c2ecf20Sopenharmony_ci/*
188c2ecf20Sopenharmony_ci * Global variables
198c2ecf20Sopenharmony_ci */
208c2ecf20Sopenharmony_ci
218c2ecf20Sopenharmony_cistatic struct pps_device *pps;
228c2ecf20Sopenharmony_cistatic struct timer_list ktimer;
238c2ecf20Sopenharmony_ci
248c2ecf20Sopenharmony_ci/*
258c2ecf20Sopenharmony_ci * The kernel timer
268c2ecf20Sopenharmony_ci */
278c2ecf20Sopenharmony_ci
288c2ecf20Sopenharmony_cistatic void pps_ktimer_event(struct timer_list *unused)
298c2ecf20Sopenharmony_ci{
308c2ecf20Sopenharmony_ci	struct pps_event_time ts;
318c2ecf20Sopenharmony_ci
328c2ecf20Sopenharmony_ci	/* First of all we get the time stamp... */
338c2ecf20Sopenharmony_ci	pps_get_ts(&ts);
348c2ecf20Sopenharmony_ci
358c2ecf20Sopenharmony_ci	pps_event(pps, &ts, PPS_CAPTUREASSERT, NULL);
368c2ecf20Sopenharmony_ci
378c2ecf20Sopenharmony_ci	mod_timer(&ktimer, jiffies + HZ);
388c2ecf20Sopenharmony_ci}
398c2ecf20Sopenharmony_ci
408c2ecf20Sopenharmony_ci/*
418c2ecf20Sopenharmony_ci * The PPS info struct
428c2ecf20Sopenharmony_ci */
438c2ecf20Sopenharmony_ci
448c2ecf20Sopenharmony_cistatic struct pps_source_info pps_ktimer_info = {
458c2ecf20Sopenharmony_ci	.name		= "ktimer",
468c2ecf20Sopenharmony_ci	.path		= "",
478c2ecf20Sopenharmony_ci	.mode		= PPS_CAPTUREASSERT | PPS_OFFSETASSERT |
488c2ecf20Sopenharmony_ci			  PPS_ECHOASSERT |
498c2ecf20Sopenharmony_ci			  PPS_CANWAIT | PPS_TSFMT_TSPEC,
508c2ecf20Sopenharmony_ci	.owner		= THIS_MODULE,
518c2ecf20Sopenharmony_ci};
528c2ecf20Sopenharmony_ci
538c2ecf20Sopenharmony_ci/*
548c2ecf20Sopenharmony_ci * Module staff
558c2ecf20Sopenharmony_ci */
568c2ecf20Sopenharmony_ci
578c2ecf20Sopenharmony_cistatic void __exit pps_ktimer_exit(void)
588c2ecf20Sopenharmony_ci{
598c2ecf20Sopenharmony_ci	dev_info(pps->dev, "ktimer PPS source unregistered\n");
608c2ecf20Sopenharmony_ci
618c2ecf20Sopenharmony_ci	del_timer_sync(&ktimer);
628c2ecf20Sopenharmony_ci	pps_unregister_source(pps);
638c2ecf20Sopenharmony_ci}
648c2ecf20Sopenharmony_ci
658c2ecf20Sopenharmony_cistatic int __init pps_ktimer_init(void)
668c2ecf20Sopenharmony_ci{
678c2ecf20Sopenharmony_ci	pps = pps_register_source(&pps_ktimer_info,
688c2ecf20Sopenharmony_ci				PPS_CAPTUREASSERT | PPS_OFFSETASSERT);
698c2ecf20Sopenharmony_ci	if (IS_ERR(pps)) {
708c2ecf20Sopenharmony_ci		pr_err("cannot register PPS source\n");
718c2ecf20Sopenharmony_ci		return PTR_ERR(pps);
728c2ecf20Sopenharmony_ci	}
738c2ecf20Sopenharmony_ci
748c2ecf20Sopenharmony_ci	timer_setup(&ktimer, pps_ktimer_event, 0);
758c2ecf20Sopenharmony_ci	mod_timer(&ktimer, jiffies + HZ);
768c2ecf20Sopenharmony_ci
778c2ecf20Sopenharmony_ci	dev_info(pps->dev, "ktimer PPS source registered\n");
788c2ecf20Sopenharmony_ci
798c2ecf20Sopenharmony_ci	return 0;
808c2ecf20Sopenharmony_ci}
818c2ecf20Sopenharmony_ci
828c2ecf20Sopenharmony_cimodule_init(pps_ktimer_init);
838c2ecf20Sopenharmony_cimodule_exit(pps_ktimer_exit);
848c2ecf20Sopenharmony_ci
858c2ecf20Sopenharmony_ciMODULE_AUTHOR("Rodolfo Giometti <giometti@linux.it>");
868c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("dummy PPS source by using a kernel timer (just for debug)");
878c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL");
88