rtc-ssd202d.c 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Real time clocks driver for MStar/SigmaStar SSD202D SoCs.
  4. *
  5. * (C) 2021 Daniel Palmer
  6. * (C) 2023 Romain Perier
  7. */
  8. #include <linux/clk.h>
  9. #include <linux/delay.h>
  10. #include <linux/module.h>
  11. #include <linux/mod_devicetable.h>
  12. #include <linux/platform_device.h>
  13. #include <linux/rtc.h>
  14. #include <linux/regmap.h>
  15. #include <linux/pm.h>
  16. #define REG_CTRL 0x0
  17. #define REG_CTRL1 0x4
  18. #define REG_ISO_CTRL 0xc
  19. #define REG_WRDATA_L 0x10
  20. #define REG_WRDATA_H 0x14
  21. #define REG_ISOACK 0x20
  22. #define REG_RDDATA_L 0x24
  23. #define REG_RDDATA_H 0x28
  24. #define REG_RDCNT_L 0x30
  25. #define REG_RDCNT_H 0x34
  26. #define REG_CNT_TRIG 0x38
  27. #define REG_PWRCTRL 0x3c
  28. #define REG_RTC_TEST 0x54
  29. #define CNT_RD_TRIG_BIT BIT(0)
  30. #define CNT_RD_BIT BIT(0)
  31. #define BASE_WR_BIT BIT(1)
  32. #define BASE_RD_BIT BIT(2)
  33. #define CNT_RST_BIT BIT(3)
  34. #define ISO_CTRL_ACK_MASK BIT(3)
  35. #define ISO_CTRL_ACK_SHIFT 3
  36. #define SW0_WR_BIT BIT(5)
  37. #define SW1_WR_BIT BIT(6)
  38. #define SW0_RD_BIT BIT(7)
  39. #define SW1_RD_BIT BIT(8)
  40. #define ISO_CTRL_MASK GENMASK(2, 0)
  41. struct ssd202d_rtc {
  42. struct rtc_device *rtc_dev;
  43. void __iomem *base;
  44. };
  45. static u8 read_iso_en(void __iomem *base)
  46. {
  47. return readb(base + REG_RTC_TEST) & 0x1;
  48. }
  49. static u8 read_iso_ctrl_ack(void __iomem *base)
  50. {
  51. return (readb(base + REG_ISOACK) & ISO_CTRL_ACK_MASK) >> ISO_CTRL_ACK_SHIFT;
  52. }
  53. static int ssd202d_rtc_isoctrl(struct ssd202d_rtc *priv)
  54. {
  55. static const unsigned int sequence[] = { 0x0, 0x1, 0x3, 0x7, 0x5, 0x1, 0x0 };
  56. unsigned int val;
  57. struct device *dev = &priv->rtc_dev->dev;
  58. int i, ret;
  59. /*
  60. * This gates iso_en by writing a special sequence of bytes to iso_ctrl
  61. * and ensuring that it has been correctly applied by reading iso_ctrl_ack
  62. */
  63. for (i = 0; i < ARRAY_SIZE(sequence); i++) {
  64. writeb(sequence[i] & ISO_CTRL_MASK, priv->base + REG_ISO_CTRL);
  65. ret = read_poll_timeout(read_iso_ctrl_ack, val, val == (i % 2), 100,
  66. 20 * 100, true, priv->base);
  67. if (ret) {
  68. dev_dbg(dev, "Timeout waiting for ack byte %i (%x) of sequence\n", i,
  69. sequence[i]);
  70. return ret;
  71. }
  72. }
  73. /*
  74. * At this point iso_en should be raised for 1ms
  75. */
  76. ret = read_poll_timeout(read_iso_en, val, val, 100, 22 * 100, true, priv->base);
  77. if (ret)
  78. dev_dbg(dev, "Timeout waiting for iso_en\n");
  79. mdelay(2);
  80. return 0;
  81. }
  82. static void ssd202d_rtc_read_reg(struct ssd202d_rtc *priv, unsigned int reg,
  83. unsigned int field, unsigned int *base)
  84. {
  85. unsigned int l, h;
  86. u16 val;
  87. /* Ask for the content of an RTC value into RDDATA by gating iso_en,
  88. * then iso_en is gated and the content of RDDATA can be read
  89. */
  90. val = readw(priv->base + reg);
  91. writew(val | field, priv->base + reg);
  92. ssd202d_rtc_isoctrl(priv);
  93. writew(val & ~field, priv->base + reg);
  94. l = readw(priv->base + REG_RDDATA_L);
  95. h = readw(priv->base + REG_RDDATA_H);
  96. *base = (h << 16) | l;
  97. }
  98. static void ssd202d_rtc_write_reg(struct ssd202d_rtc *priv, unsigned int reg,
  99. unsigned int field, u32 base)
  100. {
  101. u16 val;
  102. /* Set the content of an RTC value from WRDATA by gating iso_en */
  103. val = readw(priv->base + reg);
  104. writew(val | field, priv->base + reg);
  105. writew(base, priv->base + REG_WRDATA_L);
  106. writew(base >> 16, priv->base + REG_WRDATA_H);
  107. ssd202d_rtc_isoctrl(priv);
  108. writew(val & ~field, priv->base + reg);
  109. }
  110. static int ssd202d_rtc_read_counter(struct ssd202d_rtc *priv, unsigned int *counter)
  111. {
  112. unsigned int l, h;
  113. u16 val;
  114. val = readw(priv->base + REG_CTRL1);
  115. writew(val | CNT_RD_BIT, priv->base + REG_CTRL1);
  116. ssd202d_rtc_isoctrl(priv);
  117. writew(val & ~CNT_RD_BIT, priv->base + REG_CTRL1);
  118. val = readw(priv->base + REG_CTRL1);
  119. writew(val | CNT_RD_TRIG_BIT, priv->base + REG_CNT_TRIG);
  120. writew(val & ~CNT_RD_TRIG_BIT, priv->base + REG_CNT_TRIG);
  121. l = readw(priv->base + REG_RDCNT_L);
  122. h = readw(priv->base + REG_RDCNT_H);
  123. *counter = (h << 16) | l;
  124. return 0;
  125. }
  126. static int ssd202d_rtc_read_time(struct device *dev, struct rtc_time *tm)
  127. {
  128. struct ssd202d_rtc *priv = dev_get_drvdata(dev);
  129. unsigned int sw0, base, counter;
  130. u32 seconds;
  131. int ret;
  132. /* Check that RTC is enabled by SW */
  133. ssd202d_rtc_read_reg(priv, REG_CTRL, SW0_RD_BIT, &sw0);
  134. if (sw0 != 1)
  135. return -EINVAL;
  136. /* Get RTC base value from RDDATA */
  137. ssd202d_rtc_read_reg(priv, REG_CTRL, BASE_RD_BIT, &base);
  138. /* Get RTC counter value from RDDATA */
  139. ret = ssd202d_rtc_read_counter(priv, &counter);
  140. if (ret)
  141. return ret;
  142. seconds = base + counter;
  143. rtc_time64_to_tm(seconds, tm);
  144. return 0;
  145. }
  146. static int ssd202d_rtc_reset_counter(struct ssd202d_rtc *priv)
  147. {
  148. u16 val;
  149. val = readw(priv->base + REG_CTRL);
  150. writew(val | CNT_RST_BIT, priv->base + REG_CTRL);
  151. ssd202d_rtc_isoctrl(priv);
  152. writew(val & ~CNT_RST_BIT, priv->base + REG_CTRL);
  153. ssd202d_rtc_isoctrl(priv);
  154. return 0;
  155. }
  156. static int ssd202d_rtc_set_time(struct device *dev, struct rtc_time *tm)
  157. {
  158. struct ssd202d_rtc *priv = dev_get_drvdata(dev);
  159. unsigned long seconds = rtc_tm_to_time64(tm);
  160. ssd202d_rtc_write_reg(priv, REG_CTRL, BASE_WR_BIT, seconds);
  161. ssd202d_rtc_reset_counter(priv);
  162. ssd202d_rtc_write_reg(priv, REG_CTRL, SW0_WR_BIT, 1);
  163. return 0;
  164. }
  165. static const struct rtc_class_ops ssd202d_rtc_ops = {
  166. .read_time = ssd202d_rtc_read_time,
  167. .set_time = ssd202d_rtc_set_time,
  168. };
  169. static int ssd202d_rtc_probe(struct platform_device *pdev)
  170. {
  171. struct device *dev = &pdev->dev;
  172. struct ssd202d_rtc *priv;
  173. priv = devm_kzalloc(&pdev->dev, sizeof(struct ssd202d_rtc), GFP_KERNEL);
  174. if (!priv)
  175. return -ENOMEM;
  176. priv->base = devm_platform_ioremap_resource(pdev, 0);
  177. if (IS_ERR(priv->base))
  178. return PTR_ERR(priv->base);
  179. priv->rtc_dev = devm_rtc_allocate_device(dev);
  180. if (IS_ERR(priv->rtc_dev))
  181. return PTR_ERR(priv->rtc_dev);
  182. priv->rtc_dev->ops = &ssd202d_rtc_ops;
  183. priv->rtc_dev->range_max = U32_MAX;
  184. platform_set_drvdata(pdev, priv);
  185. return devm_rtc_register_device(priv->rtc_dev);
  186. }
  187. static const struct of_device_id ssd202d_rtc_of_match_table[] = {
  188. { .compatible = "mstar,ssd202d-rtc" },
  189. { }
  190. };
  191. MODULE_DEVICE_TABLE(of, ssd202d_rtc_of_match_table);
  192. static struct platform_driver ssd202d_rtc_driver = {
  193. .probe = ssd202d_rtc_probe,
  194. .driver = {
  195. .name = "ssd202d-rtc",
  196. .of_match_table = ssd202d_rtc_of_match_table,
  197. },
  198. };
  199. module_platform_driver(ssd202d_rtc_driver);
  200. MODULE_AUTHOR("Daniel Palmer <daniel@thingy.jp>");
  201. MODULE_AUTHOR("Romain Perier <romain.perier@gmail.com>");
  202. MODULE_DESCRIPTION("MStar SSD202D RTC Driver");
  203. MODULE_LICENSE("GPL");