pps-ldisc.c 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141
  1. // SPDX-License-Identifier: GPL-2.0-or-later
  2. /*
  3. * pps-ldisc.c -- PPS line discipline
  4. *
  5. * Copyright (C) 2008 Rodolfo Giometti <[email protected]>
  6. */
  7. #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
  8. #include <linux/module.h>
  9. #include <linux/serial_core.h>
  10. #include <linux/tty.h>
  11. #include <linux/pps_kernel.h>
  12. #include <linux/bug.h>
  13. static void pps_tty_dcd_change(struct tty_struct *tty, unsigned int status)
  14. {
  15. struct pps_device *pps;
  16. struct pps_event_time ts;
  17. pps_get_ts(&ts);
  18. pps = pps_lookup_dev(tty);
  19. /*
  20. * This should never fail, but the ldisc locking is very
  21. * convoluted, so don't crash just in case.
  22. */
  23. if (WARN_ON_ONCE(pps == NULL))
  24. return;
  25. /* Now do the PPS event report */
  26. pps_event(pps, &ts, status ? PPS_CAPTUREASSERT :
  27. PPS_CAPTURECLEAR, NULL);
  28. dev_dbg(pps->dev, "PPS %s at %lu\n",
  29. status ? "assert" : "clear", jiffies);
  30. }
  31. static int (*alias_n_tty_open)(struct tty_struct *tty);
  32. static int pps_tty_open(struct tty_struct *tty)
  33. {
  34. struct pps_source_info info;
  35. struct tty_driver *drv = tty->driver;
  36. int index = tty->index + drv->name_base;
  37. struct pps_device *pps;
  38. int ret;
  39. info.owner = THIS_MODULE;
  40. info.dev = NULL;
  41. snprintf(info.name, PPS_MAX_NAME_LEN, "%s%d", drv->driver_name, index);
  42. snprintf(info.path, PPS_MAX_NAME_LEN, "/dev/%s%d", drv->name, index);
  43. info.mode = PPS_CAPTUREBOTH | \
  44. PPS_OFFSETASSERT | PPS_OFFSETCLEAR | \
  45. PPS_CANWAIT | PPS_TSFMT_TSPEC;
  46. pps = pps_register_source(&info, PPS_CAPTUREBOTH | \
  47. PPS_OFFSETASSERT | PPS_OFFSETCLEAR);
  48. if (IS_ERR(pps)) {
  49. pr_err("cannot register PPS source \"%s\"\n", info.path);
  50. return PTR_ERR(pps);
  51. }
  52. pps->lookup_cookie = tty;
  53. /* Now open the base class N_TTY ldisc */
  54. ret = alias_n_tty_open(tty);
  55. if (ret < 0) {
  56. pr_err("cannot open tty ldisc \"%s\"\n", info.path);
  57. goto err_unregister;
  58. }
  59. dev_info(pps->dev, "source \"%s\" added\n", info.path);
  60. return 0;
  61. err_unregister:
  62. pps_unregister_source(pps);
  63. return ret;
  64. }
  65. static void (*alias_n_tty_close)(struct tty_struct *tty);
  66. static void pps_tty_close(struct tty_struct *tty)
  67. {
  68. struct pps_device *pps = pps_lookup_dev(tty);
  69. alias_n_tty_close(tty);
  70. if (WARN_ON(!pps))
  71. return;
  72. dev_info(pps->dev, "removed\n");
  73. pps_unregister_source(pps);
  74. }
  75. static struct tty_ldisc_ops pps_ldisc_ops;
  76. /*
  77. * Module stuff
  78. */
  79. static int __init pps_tty_init(void)
  80. {
  81. int err;
  82. /* Inherit the N_TTY's ops */
  83. n_tty_inherit_ops(&pps_ldisc_ops);
  84. /* Save N_TTY's open()/close() methods */
  85. alias_n_tty_open = pps_ldisc_ops.open;
  86. alias_n_tty_close = pps_ldisc_ops.close;
  87. /* Init PPS_TTY data */
  88. pps_ldisc_ops.owner = THIS_MODULE;
  89. pps_ldisc_ops.num = N_PPS;
  90. pps_ldisc_ops.name = "pps_tty";
  91. pps_ldisc_ops.dcd_change = pps_tty_dcd_change;
  92. pps_ldisc_ops.open = pps_tty_open;
  93. pps_ldisc_ops.close = pps_tty_close;
  94. err = tty_register_ldisc(&pps_ldisc_ops);
  95. if (err)
  96. pr_err("can't register PPS line discipline\n");
  97. else
  98. pr_info("PPS line discipline registered\n");
  99. return err;
  100. }
  101. static void __exit pps_tty_cleanup(void)
  102. {
  103. tty_unregister_ldisc(&pps_ldisc_ops);
  104. }
  105. module_init(pps_tty_init);
  106. module_exit(pps_tty_cleanup);
  107. MODULE_ALIAS_LDISC(N_PPS);
  108. MODULE_AUTHOR("Rodolfo Giometti <[email protected]>");
  109. MODULE_DESCRIPTION("PPS TTY device driver");
  110. MODULE_LICENSE("GPL");