1 // SPDX-License-Identifier: GPL-2.0-or-later
5 * Maintained out-of-tree by Huacai Chen <chenhuacai@kernel.org>.
6 * Rewritten for mainline by WANG Xuerui <git@xen0n.name>.
7 * Binbin Zhou <zhoubinbin@loongson.cn>
10 #include <linux/bitfield.h>
11 #include <linux/kernel.h>
12 #include <linux/module.h>
13 #include <linux/platform_device.h>
14 #include <linux/regmap.h>
15 #include <linux/rtc.h>
16 #include <linux/acpi.h>
18 /* Time Of Year(TOY) counters registers */
19 #define TOY_TRIM_REG 0x20 /* Must be initialized to 0 */
20 #define TOY_WRITE0_REG 0x24 /* TOY low 32-bits value (write-only) */
21 #define TOY_WRITE1_REG 0x28 /* TOY high 32-bits value (write-only) */
22 #define TOY_READ0_REG 0x2c /* TOY low 32-bits value (read-only) */
23 #define TOY_READ1_REG 0x30 /* TOY high 32-bits value (read-only) */
24 #define TOY_MATCH0_REG 0x34 /* TOY timing interrupt 0 */
25 #define TOY_MATCH1_REG 0x38 /* TOY timing interrupt 1 */
26 #define TOY_MATCH2_REG 0x3c /* TOY timing interrupt 2 */
28 /* RTC counters registers */
29 #define RTC_CTRL_REG 0x40 /* TOY and RTC control register */
30 #define RTC_TRIM_REG 0x60 /* Must be initialized to 0 */
31 #define RTC_WRITE0_REG 0x64 /* RTC counters value (write-only) */
32 #define RTC_READ0_REG 0x68 /* RTC counters value (read-only) */
33 #define RTC_MATCH0_REG 0x6c /* RTC timing interrupt 0 */
34 #define RTC_MATCH1_REG 0x70 /* RTC timing interrupt 1 */
35 #define RTC_MATCH2_REG 0x74 /* RTC timing interrupt 2 */
37 /* bitmask of TOY_WRITE0_REG */
38 #define TOY_MON GENMASK(31, 26)
39 #define TOY_DAY GENMASK(25, 21)
40 #define TOY_HOUR GENMASK(20, 16)
41 #define TOY_MIN GENMASK(15, 10)
42 #define TOY_SEC GENMASK(9, 4)
43 #define TOY_MSEC GENMASK(3, 0)
45 /* bitmask of TOY_MATCH0/1/2_REG */
46 #define TOY_MATCH_YEAR GENMASK(31, 26)
47 #define TOY_MATCH_MON GENMASK(25, 22)
48 #define TOY_MATCH_DAY GENMASK(21, 17)
49 #define TOY_MATCH_HOUR GENMASK(16, 12)
50 #define TOY_MATCH_MIN GENMASK(11, 6)
51 #define TOY_MATCH_SEC GENMASK(5, 0)
53 /* bitmask of RTC_CTRL_REG */
54 #define RTC_ENABLE BIT(13) /* 1: RTC counters enable */
55 #define TOY_ENABLE BIT(11) /* 1: TOY counters enable */
56 #define OSC_ENABLE BIT(8) /* 1: 32.768k crystal enable */
57 #define TOY_ENABLE_MASK (TOY_ENABLE | OSC_ENABLE)
59 /* PM domain registers */
60 #define PM1_STS_REG 0x0c /* Power management 1 status register */
61 #define RTC_STS BIT(10) /* RTC status */
62 #define PM1_EN_REG 0x10 /* Power management 1 enable register */
63 #define RTC_EN BIT(10) /* RTC event enable */
66 * According to the LS1C manual, RTC_CTRL and alarm-related registers are not defined.
67 * Accessing the relevant registers will cause the system to hang.
69 #define LS1C_RTC_CTRL_WORKAROUND BIT(0)
71 struct loongson_rtc_config
{
72 u32 pm_offset
; /* Offset of PM domain, for RTC alarm wakeup */
73 u32 flags
; /* Workaround bits */
76 struct loongson_rtc_priv
{
77 spinlock_t lock
; /* protects PM registers access */
78 u32 fix_year
; /* RTC alarm year compensation value */
79 struct rtc_device
*rtcdev
;
80 struct regmap
*regmap
;
81 void __iomem
*pm_base
; /* PM domain base, for RTC alarm wakeup */
82 const struct loongson_rtc_config
*config
;
85 static const struct loongson_rtc_config ls1b_rtc_config
= {
90 static const struct loongson_rtc_config ls1c_rtc_config
= {
92 .flags
= LS1C_RTC_CTRL_WORKAROUND
,
95 static const struct loongson_rtc_config generic_rtc_config
= {
100 static const struct loongson_rtc_config ls2k1000_rtc_config
= {
105 static const struct regmap_config loongson_rtc_regmap_config
= {
111 /* RTC alarm irq handler */
112 static irqreturn_t
loongson_rtc_isr(int irq
, void *id
)
114 struct loongson_rtc_priv
*priv
= (struct loongson_rtc_priv
*)id
;
116 rtc_update_irq(priv
->rtcdev
, 1, RTC_AF
| RTC_IRQF
);
120 /* For ACPI fixed event handler */
121 static u32
loongson_rtc_handler(void *id
)
123 struct loongson_rtc_priv
*priv
= (struct loongson_rtc_priv
*)id
;
125 spin_lock(&priv
->lock
);
126 /* Disable RTC alarm wakeup and interrupt */
127 writel(readl(priv
->pm_base
+ PM1_EN_REG
) & ~RTC_EN
,
128 priv
->pm_base
+ PM1_EN_REG
);
130 /* Clear RTC interrupt status */
131 writel(RTC_STS
, priv
->pm_base
+ PM1_STS_REG
);
132 spin_unlock(&priv
->lock
);
135 * The TOY_MATCH0_REG should be cleared 0 here,
136 * otherwise the interrupt cannot be cleared.
138 return regmap_write(priv
->regmap
, TOY_MATCH0_REG
, 0);
141 static int loongson_rtc_set_enabled(struct device
*dev
)
143 struct loongson_rtc_priv
*priv
= dev_get_drvdata(dev
);
145 if (priv
->config
->flags
& LS1C_RTC_CTRL_WORKAROUND
)
148 /* Enable RTC TOY counters and crystal */
149 return regmap_update_bits(priv
->regmap
, RTC_CTRL_REG
, TOY_ENABLE_MASK
,
153 static bool loongson_rtc_get_enabled(struct device
*dev
)
157 struct loongson_rtc_priv
*priv
= dev_get_drvdata(dev
);
159 if (priv
->config
->flags
& LS1C_RTC_CTRL_WORKAROUND
)
162 ret
= regmap_read(priv
->regmap
, RTC_CTRL_REG
, &ctrl_data
);
166 return ctrl_data
& TOY_ENABLE_MASK
;
169 static int loongson_rtc_read_time(struct device
*dev
, struct rtc_time
*tm
)
173 struct loongson_rtc_priv
*priv
= dev_get_drvdata(dev
);
175 if (!loongson_rtc_get_enabled(dev
))
178 ret
= regmap_bulk_read(priv
->regmap
, TOY_READ0_REG
, rtc_data
,
179 ARRAY_SIZE(rtc_data
));
183 tm
->tm_sec
= FIELD_GET(TOY_SEC
, rtc_data
[0]);
184 tm
->tm_min
= FIELD_GET(TOY_MIN
, rtc_data
[0]);
185 tm
->tm_hour
= FIELD_GET(TOY_HOUR
, rtc_data
[0]);
186 tm
->tm_mday
= FIELD_GET(TOY_DAY
, rtc_data
[0]);
187 tm
->tm_mon
= FIELD_GET(TOY_MON
, rtc_data
[0]) - 1;
188 tm
->tm_year
= rtc_data
[1];
190 /* Prepare for RTC alarm year compensation value. */
191 priv
->fix_year
= tm
->tm_year
/ 64 * 64;
195 static int loongson_rtc_set_time(struct device
*dev
, struct rtc_time
*tm
)
199 struct loongson_rtc_priv
*priv
= dev_get_drvdata(dev
);
201 rtc_data
[0] = FIELD_PREP(TOY_SEC
, tm
->tm_sec
)
202 | FIELD_PREP(TOY_MIN
, tm
->tm_min
)
203 | FIELD_PREP(TOY_HOUR
, tm
->tm_hour
)
204 | FIELD_PREP(TOY_DAY
, tm
->tm_mday
)
205 | FIELD_PREP(TOY_MON
, tm
->tm_mon
+ 1);
206 rtc_data
[1] = tm
->tm_year
;
208 ret
= regmap_bulk_write(priv
->regmap
, TOY_WRITE0_REG
, rtc_data
,
209 ARRAY_SIZE(rtc_data
));
213 return loongson_rtc_set_enabled(dev
);
216 static int loongson_rtc_read_alarm(struct device
*dev
, struct rtc_wkalrm
*alrm
)
220 struct loongson_rtc_priv
*priv
= dev_get_drvdata(dev
);
222 ret
= regmap_read(priv
->regmap
, TOY_MATCH0_REG
, &alarm_data
);
226 alrm
->time
.tm_sec
= FIELD_GET(TOY_MATCH_SEC
, alarm_data
);
227 alrm
->time
.tm_min
= FIELD_GET(TOY_MATCH_MIN
, alarm_data
);
228 alrm
->time
.tm_hour
= FIELD_GET(TOY_MATCH_HOUR
, alarm_data
);
229 alrm
->time
.tm_mday
= FIELD_GET(TOY_MATCH_DAY
, alarm_data
);
230 alrm
->time
.tm_mon
= FIELD_GET(TOY_MATCH_MON
, alarm_data
) - 1;
232 * This is a hardware bug: the year field of SYS_TOYMATCH is only 6 bits,
233 * making it impossible to save year values larger than 64.
235 * SYS_TOYMATCH is used to match the alarm time value and determine if
236 * an alarm is triggered, so we must keep the lower 6 bits of the year
237 * value constant during the value conversion.
239 * In summary, we need to manually add 64(or a multiple of 64) to the
240 * year value to avoid the invalid alarm prompt at startup.
242 alrm
->time
.tm_year
= FIELD_GET(TOY_MATCH_YEAR
, alarm_data
) + priv
->fix_year
;
244 alrm
->enabled
= !!(readl(priv
->pm_base
+ PM1_EN_REG
) & RTC_EN
);
248 static int loongson_rtc_alarm_irq_enable(struct device
*dev
, unsigned int enabled
)
251 struct loongson_rtc_priv
*priv
= dev_get_drvdata(dev
);
253 spin_lock(&priv
->lock
);
254 val
= readl(priv
->pm_base
+ PM1_EN_REG
);
255 /* Enable RTC alarm wakeup */
256 writel(enabled
? val
| RTC_EN
: val
& ~RTC_EN
,
257 priv
->pm_base
+ PM1_EN_REG
);
258 spin_unlock(&priv
->lock
);
263 static int loongson_rtc_set_alarm(struct device
*dev
, struct rtc_wkalrm
*alrm
)
267 struct loongson_rtc_priv
*priv
= dev_get_drvdata(dev
);
269 alarm_data
= FIELD_PREP(TOY_MATCH_SEC
, alrm
->time
.tm_sec
)
270 | FIELD_PREP(TOY_MATCH_MIN
, alrm
->time
.tm_min
)
271 | FIELD_PREP(TOY_MATCH_HOUR
, alrm
->time
.tm_hour
)
272 | FIELD_PREP(TOY_MATCH_DAY
, alrm
->time
.tm_mday
)
273 | FIELD_PREP(TOY_MATCH_MON
, alrm
->time
.tm_mon
+ 1)
274 | FIELD_PREP(TOY_MATCH_YEAR
, alrm
->time
.tm_year
- priv
->fix_year
);
276 ret
= regmap_write(priv
->regmap
, TOY_MATCH0_REG
, alarm_data
);
280 return loongson_rtc_alarm_irq_enable(dev
, alrm
->enabled
);
283 static const struct rtc_class_ops loongson_rtc_ops
= {
284 .read_time
= loongson_rtc_read_time
,
285 .set_time
= loongson_rtc_set_time
,
286 .read_alarm
= loongson_rtc_read_alarm
,
287 .set_alarm
= loongson_rtc_set_alarm
,
288 .alarm_irq_enable
= loongson_rtc_alarm_irq_enable
,
291 static int loongson_rtc_probe(struct platform_device
*pdev
)
295 struct loongson_rtc_priv
*priv
;
296 struct device
*dev
= &pdev
->dev
;
298 priv
= devm_kzalloc(dev
, sizeof(*priv
), GFP_KERNEL
);
302 regs
= devm_platform_ioremap_resource(pdev
, 0);
304 return dev_err_probe(dev
, PTR_ERR(regs
),
305 "devm_platform_ioremap_resource failed\n");
307 priv
->regmap
= devm_regmap_init_mmio(dev
, regs
,
308 &loongson_rtc_regmap_config
);
309 if (IS_ERR(priv
->regmap
))
310 return dev_err_probe(dev
, PTR_ERR(priv
->regmap
),
311 "devm_regmap_init_mmio failed\n");
313 priv
->config
= device_get_match_data(dev
);
314 spin_lock_init(&priv
->lock
);
315 platform_set_drvdata(pdev
, priv
);
317 priv
->rtcdev
= devm_rtc_allocate_device(dev
);
318 if (IS_ERR(priv
->rtcdev
))
319 return dev_err_probe(dev
, PTR_ERR(priv
->rtcdev
),
320 "devm_rtc_allocate_device failed\n");
322 /* Get RTC alarm irq */
323 alarm_irq
= platform_get_irq(pdev
, 0);
325 ret
= devm_request_irq(dev
, alarm_irq
, loongson_rtc_isr
,
326 0, "loongson-alarm", priv
);
328 return dev_err_probe(dev
, ret
, "Unable to request irq %d\n",
331 priv
->pm_base
= regs
- priv
->config
->pm_offset
;
332 device_init_wakeup(dev
, 1);
334 if (has_acpi_companion(dev
))
335 acpi_install_fixed_event_handler(ACPI_EVENT_RTC
,
336 loongson_rtc_handler
, priv
);
338 /* Loongson-1C RTC does not support alarm */
339 clear_bit(RTC_FEATURE_ALARM
, priv
->rtcdev
->features
);
342 /* Loongson RTC does not support UIE */
343 clear_bit(RTC_FEATURE_UPDATE_INTERRUPT
, priv
->rtcdev
->features
);
344 priv
->rtcdev
->ops
= &loongson_rtc_ops
;
345 priv
->rtcdev
->range_min
= RTC_TIMESTAMP_BEGIN_2000
;
346 priv
->rtcdev
->range_max
= RTC_TIMESTAMP_END_2099
;
348 return devm_rtc_register_device(priv
->rtcdev
);
351 static void loongson_rtc_remove(struct platform_device
*pdev
)
353 struct device
*dev
= &pdev
->dev
;
354 struct loongson_rtc_priv
*priv
= dev_get_drvdata(dev
);
356 if (!test_bit(RTC_FEATURE_ALARM
, priv
->rtcdev
->features
))
359 if (has_acpi_companion(dev
))
360 acpi_remove_fixed_event_handler(ACPI_EVENT_RTC
,
361 loongson_rtc_handler
);
363 device_init_wakeup(dev
, 0);
364 loongson_rtc_alarm_irq_enable(dev
, 0);
367 static const struct of_device_id loongson_rtc_of_match
[] = {
368 { .compatible
= "loongson,ls1b-rtc", .data
= &ls1b_rtc_config
},
369 { .compatible
= "loongson,ls1c-rtc", .data
= &ls1c_rtc_config
},
370 { .compatible
= "loongson,ls7a-rtc", .data
= &generic_rtc_config
},
371 { .compatible
= "loongson,ls2k1000-rtc", .data
= &ls2k1000_rtc_config
},
374 MODULE_DEVICE_TABLE(of
, loongson_rtc_of_match
);
376 static const struct acpi_device_id loongson_rtc_acpi_match
[] = {
377 { "LOON0001", .driver_data
= (kernel_ulong_t
)&generic_rtc_config
},
380 MODULE_DEVICE_TABLE(acpi
, loongson_rtc_acpi_match
);
382 static struct platform_driver loongson_rtc_driver
= {
383 .probe
= loongson_rtc_probe
,
384 .remove
= loongson_rtc_remove
,
386 .name
= "loongson-rtc",
387 .of_match_table
= loongson_rtc_of_match
,
388 .acpi_match_table
= loongson_rtc_acpi_match
,
391 module_platform_driver(loongson_rtc_driver
);
393 MODULE_DESCRIPTION("Loongson RTC driver");
394 MODULE_AUTHOR("Binbin Zhou <zhoubinbin@loongson.cn>");
395 MODULE_AUTHOR("WANG Xuerui <git@xen0n.name>");
396 MODULE_AUTHOR("Huacai Chen <chenhuacai@kernel.org>");
397 MODULE_LICENSE("GPL");