xref: /openbmc/qemu/hw/rtc/ls7a_rtc.c (revision 9ea0f206b7e292c46d0767c3ef1a5ac1d0c6fecb)
1c117f68aSXiaojuan Yang /* SPDX-License-Identifier: GPL-2.0-or-later */
2c117f68aSXiaojuan Yang /*
3*9ea0f206SPhilippe Mathieu-Daudé  * LoongArch LS7A Real Time Clock emulation
4c117f68aSXiaojuan Yang  *
5c117f68aSXiaojuan Yang  * Copyright (C) 2021 Loongson Technology Corporation Limited
6c117f68aSXiaojuan Yang  */
7c117f68aSXiaojuan Yang 
8c117f68aSXiaojuan Yang #include "qemu/osdep.h"
9c117f68aSXiaojuan Yang #include "hw/sysbus.h"
10c117f68aSXiaojuan Yang #include "hw/irq.h"
118b4d80bbSPhilippe Mathieu-Daudé #include "hw/register.h"
12c117f68aSXiaojuan Yang #include "qemu/timer.h"
13c117f68aSXiaojuan Yang #include "sysemu/sysemu.h"
14c117f68aSXiaojuan Yang #include "qemu/cutils.h"
15c117f68aSXiaojuan Yang #include "qemu/log.h"
16c117f68aSXiaojuan Yang #include "migration/vmstate.h"
17c117f68aSXiaojuan Yang #include "hw/misc/unimp.h"
18c117f68aSXiaojuan Yang #include "sysemu/rtc.h"
19c117f68aSXiaojuan Yang #include "hw/registerfields.h"
20c117f68aSXiaojuan Yang 
21c117f68aSXiaojuan Yang #define SYS_TOYTRIM        0x20
22c117f68aSXiaojuan Yang #define SYS_TOYWRITE0      0x24
23c117f68aSXiaojuan Yang #define SYS_TOYWRITE1      0x28
24c117f68aSXiaojuan Yang #define SYS_TOYREAD0       0x2C
25c117f68aSXiaojuan Yang #define SYS_TOYREAD1       0x30
26c117f68aSXiaojuan Yang #define SYS_TOYMATCH0      0x34
27c117f68aSXiaojuan Yang #define SYS_TOYMATCH1      0x38
28c117f68aSXiaojuan Yang #define SYS_TOYMATCH2      0x3C
29c117f68aSXiaojuan Yang #define SYS_RTCCTRL        0x40
30c117f68aSXiaojuan Yang #define SYS_RTCTRIM        0x60
31c117f68aSXiaojuan Yang #define SYS_RTCWRTIE0      0x64
32c117f68aSXiaojuan Yang #define SYS_RTCREAD0       0x68
33c117f68aSXiaojuan Yang #define SYS_RTCMATCH0      0x6C
34c117f68aSXiaojuan Yang #define SYS_RTCMATCH1      0x70
35c117f68aSXiaojuan Yang #define SYS_RTCMATCH2      0x74
36c117f68aSXiaojuan Yang 
37c117f68aSXiaojuan Yang #define LS7A_RTC_FREQ     32768
38c117f68aSXiaojuan Yang #define TIMER_NUMS        3
39c117f68aSXiaojuan Yang /*
40c117f68aSXiaojuan Yang  * Shift bits and filed mask
41c117f68aSXiaojuan Yang  */
42c117f68aSXiaojuan Yang 
43c117f68aSXiaojuan Yang FIELD(TOY, MON, 26, 6)
44c117f68aSXiaojuan Yang FIELD(TOY, DAY, 21, 5)
45c117f68aSXiaojuan Yang FIELD(TOY, HOUR, 16, 5)
46c117f68aSXiaojuan Yang FIELD(TOY, MIN, 10, 6)
47c117f68aSXiaojuan Yang FIELD(TOY, SEC, 4, 6)
48c117f68aSXiaojuan Yang FIELD(TOY, MSEC, 0, 4)
49c117f68aSXiaojuan Yang 
50c117f68aSXiaojuan Yang FIELD(TOY_MATCH, YEAR, 26, 6)
51c117f68aSXiaojuan Yang FIELD(TOY_MATCH, MON, 22, 4)
52c117f68aSXiaojuan Yang FIELD(TOY_MATCH, DAY, 17, 5)
53c117f68aSXiaojuan Yang FIELD(TOY_MATCH, HOUR, 12, 5)
54c117f68aSXiaojuan Yang FIELD(TOY_MATCH, MIN, 6, 6)
55c117f68aSXiaojuan Yang FIELD(TOY_MATCH, SEC, 0, 6)
56c117f68aSXiaojuan Yang 
57c117f68aSXiaojuan Yang FIELD(RTC_CTRL, RTCEN, 13, 1)
58c117f68aSXiaojuan Yang FIELD(RTC_CTRL, TOYEN, 11, 1)
59c117f68aSXiaojuan Yang FIELD(RTC_CTRL, EO, 8, 1)
60c117f68aSXiaojuan Yang 
61c117f68aSXiaojuan Yang #define TYPE_LS7A_RTC "ls7a_rtc"
62c117f68aSXiaojuan Yang OBJECT_DECLARE_SIMPLE_TYPE(LS7ARtcState, LS7A_RTC)
63c117f68aSXiaojuan Yang 
64c117f68aSXiaojuan Yang struct LS7ARtcState {
65c117f68aSXiaojuan Yang     SysBusDevice parent_obj;
66c117f68aSXiaojuan Yang 
67c117f68aSXiaojuan Yang     MemoryRegion iomem;
68c117f68aSXiaojuan Yang     /*
69c117f68aSXiaojuan Yang      * Needed to preserve the tick_count across migration, even if the
70c117f68aSXiaojuan Yang      * absolute value of the rtc_clock is different on the source and
71c117f68aSXiaojuan Yang      * destination.
72c117f68aSXiaojuan Yang      */
73c117f68aSXiaojuan Yang     int64_t offset_toy;
74c117f68aSXiaojuan Yang     int64_t offset_rtc;
75c117f68aSXiaojuan Yang     int64_t data;
76c117f68aSXiaojuan Yang     int tidx;
77c117f68aSXiaojuan Yang     uint32_t toymatch[3];
78c117f68aSXiaojuan Yang     uint32_t toytrim;
79c117f68aSXiaojuan Yang     uint32_t cntrctl;
80c117f68aSXiaojuan Yang     uint32_t rtctrim;
81c117f68aSXiaojuan Yang     uint32_t rtccount;
82c117f68aSXiaojuan Yang     uint32_t rtcmatch[3];
83c117f68aSXiaojuan Yang     QEMUTimer *toy_timer[TIMER_NUMS];
84c117f68aSXiaojuan Yang     QEMUTimer *rtc_timer[TIMER_NUMS];
85c117f68aSXiaojuan Yang     qemu_irq irq;
86c117f68aSXiaojuan Yang };
87c117f68aSXiaojuan Yang 
88c117f68aSXiaojuan Yang /* switch nanoseconds time to rtc ticks */
89c1ca312aSRichard Henderson static uint64_t ls7a_rtc_ticks(void)
90c117f68aSXiaojuan Yang {
91c117f68aSXiaojuan Yang     return qemu_clock_get_ns(rtc_clock) * LS7A_RTC_FREQ / NANOSECONDS_PER_SECOND;
92c117f68aSXiaojuan Yang }
93c117f68aSXiaojuan Yang 
94c117f68aSXiaojuan Yang /* switch rtc ticks to nanoseconds */
95c1ca312aSRichard Henderson static uint64_t ticks_to_ns(uint64_t ticks)
96c117f68aSXiaojuan Yang {
97c117f68aSXiaojuan Yang     return ticks * NANOSECONDS_PER_SECOND / LS7A_RTC_FREQ;
98c117f68aSXiaojuan Yang }
99c117f68aSXiaojuan Yang 
100c1ca312aSRichard Henderson static bool toy_enabled(LS7ARtcState *s)
101c117f68aSXiaojuan Yang {
102c117f68aSXiaojuan Yang     return FIELD_EX32(s->cntrctl, RTC_CTRL, TOYEN) &&
103c117f68aSXiaojuan Yang            FIELD_EX32(s->cntrctl, RTC_CTRL, EO);
104c117f68aSXiaojuan Yang }
105c117f68aSXiaojuan Yang 
106c1ca312aSRichard Henderson static bool rtc_enabled(LS7ARtcState *s)
107c117f68aSXiaojuan Yang {
108c117f68aSXiaojuan Yang     return FIELD_EX32(s->cntrctl, RTC_CTRL, RTCEN) &&
109c117f68aSXiaojuan Yang            FIELD_EX32(s->cntrctl, RTC_CTRL, EO);
110c117f68aSXiaojuan Yang }
111c117f68aSXiaojuan Yang 
112c117f68aSXiaojuan Yang /* parse struct tm to toy value */
113c1ca312aSRichard Henderson static uint64_t toy_time_to_val_mon(const struct tm *tm)
114c117f68aSXiaojuan Yang {
115c117f68aSXiaojuan Yang     uint64_t val = 0;
116c117f68aSXiaojuan Yang 
117582788c3SXiaojuan Yang     val = FIELD_DP32(val, TOY, MON, tm->tm_mon + 1);
118582788c3SXiaojuan Yang     val = FIELD_DP32(val, TOY, DAY, tm->tm_mday);
119582788c3SXiaojuan Yang     val = FIELD_DP32(val, TOY, HOUR, tm->tm_hour);
120582788c3SXiaojuan Yang     val = FIELD_DP32(val, TOY, MIN, tm->tm_min);
121582788c3SXiaojuan Yang     val = FIELD_DP32(val, TOY, SEC, tm->tm_sec);
122c117f68aSXiaojuan Yang     return val;
123c117f68aSXiaojuan Yang }
124c117f68aSXiaojuan Yang 
125c1ca312aSRichard Henderson static void toymatch_val_to_time(LS7ARtcState *s, uint64_t val, struct tm *tm)
126c117f68aSXiaojuan Yang {
1274f2c6587SXiaojuan Yang     qemu_get_timedate(tm, s->offset_toy);
128c117f68aSXiaojuan Yang     tm->tm_sec = FIELD_EX32(val, TOY_MATCH, SEC);
129c117f68aSXiaojuan Yang     tm->tm_min = FIELD_EX32(val, TOY_MATCH, MIN);
130c117f68aSXiaojuan Yang     tm->tm_hour = FIELD_EX32(val, TOY_MATCH, HOUR);
131c117f68aSXiaojuan Yang     tm->tm_mday = FIELD_EX32(val, TOY_MATCH, DAY);
132c117f68aSXiaojuan Yang     tm->tm_mon = FIELD_EX32(val, TOY_MATCH, MON) - 1;
133c117f68aSXiaojuan Yang     tm->tm_year += (FIELD_EX32(val, TOY_MATCH, YEAR) - (tm->tm_year & 0x3f));
134c117f68aSXiaojuan Yang }
135c117f68aSXiaojuan Yang 
1364f2c6587SXiaojuan Yang static void toymatch_write(LS7ARtcState *s, uint64_t val, int num)
137c117f68aSXiaojuan Yang {
138c117f68aSXiaojuan Yang     int64_t now, expire_time;
1394f2c6587SXiaojuan Yang     struct tm tm = {};
140c117f68aSXiaojuan Yang 
141c117f68aSXiaojuan Yang     /* it do not support write when toy disabled */
142c117f68aSXiaojuan Yang     if (toy_enabled(s)) {
143c117f68aSXiaojuan Yang         s->toymatch[num] = val;
14459e52dcfSXiaojuan Yang         /* calculate expire time */
145c117f68aSXiaojuan Yang         now = qemu_clock_get_ms(rtc_clock);
1464f2c6587SXiaojuan Yang         toymatch_val_to_time(s, val, &tm);
1474f2c6587SXiaojuan Yang         expire_time = now + (qemu_timedate_diff(&tm) - s->offset_toy) * 1000;
148c117f68aSXiaojuan Yang         timer_mod(s->toy_timer[num], expire_time);
149c117f68aSXiaojuan Yang     }
150c117f68aSXiaojuan Yang }
151c117f68aSXiaojuan Yang 
152c117f68aSXiaojuan Yang static void rtcmatch_write(LS7ARtcState *s, uint64_t val, int num)
153c117f68aSXiaojuan Yang {
154c117f68aSXiaojuan Yang     uint64_t expire_ns;
155c117f68aSXiaojuan Yang 
156c117f68aSXiaojuan Yang     /* it do not support write when toy disabled */
157c117f68aSXiaojuan Yang     if (rtc_enabled(s)) {
158c117f68aSXiaojuan Yang         s->rtcmatch[num] = val;
15959e52dcfSXiaojuan Yang         /* calculate expire time */
160c117f68aSXiaojuan Yang         expire_ns = ticks_to_ns(val) - ticks_to_ns(s->offset_rtc);
161c117f68aSXiaojuan Yang         timer_mod_ns(s->rtc_timer[num], expire_ns);
162c117f68aSXiaojuan Yang     }
163c117f68aSXiaojuan Yang }
164c117f68aSXiaojuan Yang 
165c117f68aSXiaojuan Yang static void ls7a_toy_stop(LS7ARtcState *s)
166c117f68aSXiaojuan Yang {
167c117f68aSXiaojuan Yang     int i;
168c117f68aSXiaojuan Yang 
16959e52dcfSXiaojuan Yang     /* delete timers, and when re-enabled, recalculate expire time */
170c117f68aSXiaojuan Yang     for (i = 0; i < TIMER_NUMS; i++) {
171c117f68aSXiaojuan Yang         timer_del(s->toy_timer[i]);
172c117f68aSXiaojuan Yang     }
173c117f68aSXiaojuan Yang }
174c117f68aSXiaojuan Yang 
175c117f68aSXiaojuan Yang static void ls7a_rtc_stop(LS7ARtcState *s)
176c117f68aSXiaojuan Yang {
177c117f68aSXiaojuan Yang     int i;
178c117f68aSXiaojuan Yang 
17959e52dcfSXiaojuan Yang     /* delete timers, and when re-enabled, recalculate expire time */
180c117f68aSXiaojuan Yang     for (i = 0; i < TIMER_NUMS; i++) {
181c117f68aSXiaojuan Yang         timer_del(s->rtc_timer[i]);
182c117f68aSXiaojuan Yang     }
183c117f68aSXiaojuan Yang }
184c117f68aSXiaojuan Yang 
185c117f68aSXiaojuan Yang static void ls7a_toy_start(LS7ARtcState *s)
186c117f68aSXiaojuan Yang {
187c117f68aSXiaojuan Yang     int i;
188c117f68aSXiaojuan Yang     uint64_t expire_time, now;
1894f2c6587SXiaojuan Yang     struct tm tm = {};
190c117f68aSXiaojuan Yang 
191c117f68aSXiaojuan Yang     now = qemu_clock_get_ms(rtc_clock);
192c117f68aSXiaojuan Yang 
19359e52dcfSXiaojuan Yang     /* recalculate expire time and enable timer */
194c117f68aSXiaojuan Yang     for (i = 0; i < TIMER_NUMS; i++) {
1954f2c6587SXiaojuan Yang         toymatch_val_to_time(s, s->toymatch[i], &tm);
196c117f68aSXiaojuan Yang         expire_time = now + (qemu_timedate_diff(&tm) - s->offset_toy) * 1000;
197c117f68aSXiaojuan Yang         timer_mod(s->toy_timer[i], expire_time);
198c117f68aSXiaojuan Yang     }
199c117f68aSXiaojuan Yang }
200c117f68aSXiaojuan Yang 
201c117f68aSXiaojuan Yang static void ls7a_rtc_start(LS7ARtcState *s)
202c117f68aSXiaojuan Yang {
203c117f68aSXiaojuan Yang     int i;
2046935f132SXiaojuan Yang     uint64_t expire_time;
205c117f68aSXiaojuan Yang 
20659e52dcfSXiaojuan Yang     /* recalculate expire time and enable timer */
207c117f68aSXiaojuan Yang     for (i = 0; i < TIMER_NUMS; i++) {
208c117f68aSXiaojuan Yang         expire_time = ticks_to_ns(s->rtcmatch[i]) - ticks_to_ns(s->offset_rtc);
209c117f68aSXiaojuan Yang         timer_mod_ns(s->rtc_timer[i], expire_time);
210c117f68aSXiaojuan Yang     }
211c117f68aSXiaojuan Yang }
212c117f68aSXiaojuan Yang 
213c117f68aSXiaojuan Yang static uint64_t ls7a_rtc_read(void *opaque, hwaddr addr, unsigned size)
214c117f68aSXiaojuan Yang {
215c117f68aSXiaojuan Yang     LS7ARtcState *s = LS7A_RTC(opaque);
216c117f68aSXiaojuan Yang     struct tm tm;
217c117f68aSXiaojuan Yang     int val = 0;
218c117f68aSXiaojuan Yang 
219c117f68aSXiaojuan Yang     switch (addr) {
220c117f68aSXiaojuan Yang     case SYS_TOYREAD0:
221c117f68aSXiaojuan Yang         if (toy_enabled(s)) {
222c117f68aSXiaojuan Yang             qemu_get_timedate(&tm, s->offset_toy);
223582788c3SXiaojuan Yang             val = toy_time_to_val_mon(&tm);
224c117f68aSXiaojuan Yang         } else {
2256935f132SXiaojuan Yang             /* return 0 when toy disabled */
2266935f132SXiaojuan Yang             val = 0;
227c117f68aSXiaojuan Yang         }
228c117f68aSXiaojuan Yang         break;
229c117f68aSXiaojuan Yang     case SYS_TOYREAD1:
230c117f68aSXiaojuan Yang         if (toy_enabled(s)) {
231c117f68aSXiaojuan Yang             qemu_get_timedate(&tm, s->offset_toy);
232c117f68aSXiaojuan Yang             val = tm.tm_year;
233c117f68aSXiaojuan Yang         } else {
2346935f132SXiaojuan Yang             /* return 0 when toy disabled */
2356935f132SXiaojuan Yang             val = 0;
236c117f68aSXiaojuan Yang         }
237c117f68aSXiaojuan Yang         break;
238c117f68aSXiaojuan Yang     case SYS_TOYMATCH0:
239c117f68aSXiaojuan Yang         val = s->toymatch[0];
240c117f68aSXiaojuan Yang         break;
241c117f68aSXiaojuan Yang     case SYS_TOYMATCH1:
242c117f68aSXiaojuan Yang         val = s->toymatch[1];
243c117f68aSXiaojuan Yang         break;
244c117f68aSXiaojuan Yang     case SYS_TOYMATCH2:
245c117f68aSXiaojuan Yang         val = s->toymatch[2];
246c117f68aSXiaojuan Yang         break;
247c117f68aSXiaojuan Yang     case SYS_RTCCTRL:
248c117f68aSXiaojuan Yang         val = s->cntrctl;
249c117f68aSXiaojuan Yang         break;
250c117f68aSXiaojuan Yang     case SYS_RTCREAD0:
251c117f68aSXiaojuan Yang         if (rtc_enabled(s)) {
252c117f68aSXiaojuan Yang             val = ls7a_rtc_ticks() + s->offset_rtc;
253c117f68aSXiaojuan Yang         } else {
2546935f132SXiaojuan Yang             /* return 0 when rtc disabled */
2556935f132SXiaojuan Yang             val = 0;
256c117f68aSXiaojuan Yang         }
257c117f68aSXiaojuan Yang         break;
258c117f68aSXiaojuan Yang     case SYS_RTCMATCH0:
259c117f68aSXiaojuan Yang         val = s->rtcmatch[0];
260c117f68aSXiaojuan Yang         break;
261c117f68aSXiaojuan Yang     case SYS_RTCMATCH1:
262c117f68aSXiaojuan Yang         val = s->rtcmatch[1];
263c117f68aSXiaojuan Yang         break;
264c117f68aSXiaojuan Yang     case SYS_RTCMATCH2:
265c117f68aSXiaojuan Yang         val = s->rtcmatch[2];
266c117f68aSXiaojuan Yang         break;
267c117f68aSXiaojuan Yang     default:
268c117f68aSXiaojuan Yang         val = 0;
269c117f68aSXiaojuan Yang         break;
270c117f68aSXiaojuan Yang     }
271c117f68aSXiaojuan Yang     return val;
272c117f68aSXiaojuan Yang }
273c117f68aSXiaojuan Yang 
274c117f68aSXiaojuan Yang static void ls7a_rtc_write(void *opaque, hwaddr addr,
275c117f68aSXiaojuan Yang                            uint64_t val, unsigned size)
276c117f68aSXiaojuan Yang {
277c117f68aSXiaojuan Yang     int old_toyen, old_rtcen, new_toyen, new_rtcen;
278c117f68aSXiaojuan Yang     LS7ARtcState *s = LS7A_RTC(opaque);
279c117f68aSXiaojuan Yang     struct tm tm;
280c117f68aSXiaojuan Yang 
281c117f68aSXiaojuan Yang     switch (addr) {
282c117f68aSXiaojuan Yang     case SYS_TOYWRITE0:
283c117f68aSXiaojuan Yang         /* it do not support write when toy disabled */
284c117f68aSXiaojuan Yang         if (toy_enabled(s)) {
285c117f68aSXiaojuan Yang             qemu_get_timedate(&tm, s->offset_toy);
286c117f68aSXiaojuan Yang             tm.tm_sec = FIELD_EX32(val, TOY, SEC);
287c117f68aSXiaojuan Yang             tm.tm_min = FIELD_EX32(val, TOY, MIN);
288c117f68aSXiaojuan Yang             tm.tm_hour = FIELD_EX32(val, TOY, HOUR);
289c117f68aSXiaojuan Yang             tm.tm_mday = FIELD_EX32(val, TOY, DAY);
290c117f68aSXiaojuan Yang             tm.tm_mon = FIELD_EX32(val, TOY, MON) - 1;
291c117f68aSXiaojuan Yang             s->offset_toy = qemu_timedate_diff(&tm);
292c117f68aSXiaojuan Yang         }
293c117f68aSXiaojuan Yang     break;
294c117f68aSXiaojuan Yang     case SYS_TOYWRITE1:
295c117f68aSXiaojuan Yang         if (toy_enabled(s)) {
296c117f68aSXiaojuan Yang             qemu_get_timedate(&tm, s->offset_toy);
297c117f68aSXiaojuan Yang             tm.tm_year = val;
298c117f68aSXiaojuan Yang             s->offset_toy = qemu_timedate_diff(&tm);
299c117f68aSXiaojuan Yang         }
300c117f68aSXiaojuan Yang         break;
301c117f68aSXiaojuan Yang     case SYS_TOYMATCH0:
3024f2c6587SXiaojuan Yang         toymatch_write(s, val, 0);
303c117f68aSXiaojuan Yang         break;
304c117f68aSXiaojuan Yang     case SYS_TOYMATCH1:
3054f2c6587SXiaojuan Yang         toymatch_write(s, val, 1);
306c117f68aSXiaojuan Yang         break;
307c117f68aSXiaojuan Yang     case SYS_TOYMATCH2:
3084f2c6587SXiaojuan Yang         toymatch_write(s, val, 2);
309c117f68aSXiaojuan Yang         break;
310c117f68aSXiaojuan Yang     case SYS_RTCCTRL:
311c117f68aSXiaojuan Yang         /* get old ctrl */
312c117f68aSXiaojuan Yang         old_toyen = toy_enabled(s);
313c117f68aSXiaojuan Yang         old_rtcen = rtc_enabled(s);
314c117f68aSXiaojuan Yang 
315c117f68aSXiaojuan Yang         s->cntrctl = val;
316c117f68aSXiaojuan Yang         /* get new ctrl */
317c117f68aSXiaojuan Yang         new_toyen = toy_enabled(s);
318c117f68aSXiaojuan Yang         new_rtcen = rtc_enabled(s);
319c117f68aSXiaojuan Yang 
320c117f68aSXiaojuan Yang         /*
321c117f68aSXiaojuan Yang          * we do not consider if EO changed, as it always set at most time.
322c117f68aSXiaojuan Yang          * toy or rtc enabled should start timer. otherwise, stop timer
323c117f68aSXiaojuan Yang          */
324c117f68aSXiaojuan Yang         if (old_toyen != new_toyen) {
325c117f68aSXiaojuan Yang             if (new_toyen) {
326c117f68aSXiaojuan Yang                 ls7a_toy_start(s);
327c117f68aSXiaojuan Yang             } else {
328c117f68aSXiaojuan Yang                 ls7a_toy_stop(s);
329c117f68aSXiaojuan Yang             }
330c117f68aSXiaojuan Yang         }
331c117f68aSXiaojuan Yang         if (old_rtcen != new_rtcen) {
332c117f68aSXiaojuan Yang             if (new_rtcen) {
333c117f68aSXiaojuan Yang                 ls7a_rtc_start(s);
334c117f68aSXiaojuan Yang             } else {
335c117f68aSXiaojuan Yang                 ls7a_rtc_stop(s);
336c117f68aSXiaojuan Yang             }
337c117f68aSXiaojuan Yang         }
338c117f68aSXiaojuan Yang         break;
339c117f68aSXiaojuan Yang     case SYS_RTCWRTIE0:
340c117f68aSXiaojuan Yang         if (rtc_enabled(s)) {
341c117f68aSXiaojuan Yang             s->offset_rtc = val - ls7a_rtc_ticks();
342c117f68aSXiaojuan Yang         }
343c117f68aSXiaojuan Yang         break;
344c117f68aSXiaojuan Yang     case SYS_RTCMATCH0:
345c117f68aSXiaojuan Yang         rtcmatch_write(s, val, 0);
346c117f68aSXiaojuan Yang         break;
347c117f68aSXiaojuan Yang     case SYS_RTCMATCH1:
348c117f68aSXiaojuan Yang         rtcmatch_write(s, val, 1);
349c117f68aSXiaojuan Yang         break;
350c117f68aSXiaojuan Yang     case SYS_RTCMATCH2:
351c117f68aSXiaojuan Yang         rtcmatch_write(s, val, 2);
352c117f68aSXiaojuan Yang         break;
353c117f68aSXiaojuan Yang     default:
354c117f68aSXiaojuan Yang         break;
355c117f68aSXiaojuan Yang     }
356c117f68aSXiaojuan Yang }
357c117f68aSXiaojuan Yang 
358c117f68aSXiaojuan Yang static const MemoryRegionOps ls7a_rtc_ops = {
359c117f68aSXiaojuan Yang     .read = ls7a_rtc_read,
360c117f68aSXiaojuan Yang     .write = ls7a_rtc_write,
361c117f68aSXiaojuan Yang     .endianness = DEVICE_LITTLE_ENDIAN,
362c117f68aSXiaojuan Yang     .valid = {
363c117f68aSXiaojuan Yang         .min_access_size = 4,
364c117f68aSXiaojuan Yang         .max_access_size = 4,
365c117f68aSXiaojuan Yang     },
366c117f68aSXiaojuan Yang };
367c117f68aSXiaojuan Yang 
368c117f68aSXiaojuan Yang static void toy_timer_cb(void *opaque)
369c117f68aSXiaojuan Yang {
370c117f68aSXiaojuan Yang     LS7ARtcState *s = opaque;
371c117f68aSXiaojuan Yang 
372c117f68aSXiaojuan Yang     if (toy_enabled(s)) {
373df11f3eaSXiaojuan Yang         qemu_irq_raise(s->irq);
374c117f68aSXiaojuan Yang     }
375c117f68aSXiaojuan Yang }
376c117f68aSXiaojuan Yang 
377c117f68aSXiaojuan Yang static void rtc_timer_cb(void *opaque)
378c117f68aSXiaojuan Yang {
379c117f68aSXiaojuan Yang     LS7ARtcState *s = opaque;
380c117f68aSXiaojuan Yang 
381c117f68aSXiaojuan Yang     if (rtc_enabled(s)) {
382df11f3eaSXiaojuan Yang         qemu_irq_raise(s->irq);
383c117f68aSXiaojuan Yang     }
384c117f68aSXiaojuan Yang }
385c117f68aSXiaojuan Yang 
386c117f68aSXiaojuan Yang static void ls7a_rtc_realize(DeviceState *dev, Error **errp)
387c117f68aSXiaojuan Yang {
388c117f68aSXiaojuan Yang     int i;
389c117f68aSXiaojuan Yang     SysBusDevice *sbd = SYS_BUS_DEVICE(dev);
390c117f68aSXiaojuan Yang     LS7ARtcState *d = LS7A_RTC(sbd);
391c117f68aSXiaojuan Yang     memory_region_init_io(&d->iomem, NULL, &ls7a_rtc_ops,
392c117f68aSXiaojuan Yang                          (void *)d, "ls7a_rtc", 0x100);
393c117f68aSXiaojuan Yang 
394c117f68aSXiaojuan Yang     sysbus_init_irq(sbd, &d->irq);
395c117f68aSXiaojuan Yang 
396c117f68aSXiaojuan Yang     sysbus_init_mmio(sbd, &d->iomem);
397c117f68aSXiaojuan Yang     for (i = 0; i < TIMER_NUMS; i++) {
398c117f68aSXiaojuan Yang         d->toymatch[i] = 0;
399c117f68aSXiaojuan Yang         d->rtcmatch[i] = 0;
400c117f68aSXiaojuan Yang         d->toy_timer[i] = timer_new_ms(rtc_clock, toy_timer_cb, d);
401c117f68aSXiaojuan Yang         d->rtc_timer[i] = timer_new_ms(rtc_clock, rtc_timer_cb, d);
402c117f68aSXiaojuan Yang     }
403c117f68aSXiaojuan Yang     d->offset_toy = 0;
404c117f68aSXiaojuan Yang     d->offset_rtc = 0;
405c117f68aSXiaojuan Yang 
406c117f68aSXiaojuan Yang }
407c117f68aSXiaojuan Yang 
408e5c0367eSXiaojuan Yang /* delete timer and clear reg when reset */
409e5c0367eSXiaojuan Yang static void ls7a_rtc_reset(DeviceState *dev)
410e5c0367eSXiaojuan Yang {
411e5c0367eSXiaojuan Yang     int i;
412e5c0367eSXiaojuan Yang     SysBusDevice *sbd = SYS_BUS_DEVICE(dev);
413e5c0367eSXiaojuan Yang     LS7ARtcState *d = LS7A_RTC(sbd);
414e5c0367eSXiaojuan Yang     for (i = 0; i < TIMER_NUMS; i++) {
415e5c0367eSXiaojuan Yang         if (toy_enabled(d)) {
416e5c0367eSXiaojuan Yang             timer_del(d->toy_timer[i]);
417e5c0367eSXiaojuan Yang         }
418e5c0367eSXiaojuan Yang         if (rtc_enabled(d)) {
419e5c0367eSXiaojuan Yang             timer_del(d->rtc_timer[i]);
420e5c0367eSXiaojuan Yang         }
421e5c0367eSXiaojuan Yang         d->toymatch[i] = 0;
422e5c0367eSXiaojuan Yang         d->rtcmatch[i] = 0;
423e5c0367eSXiaojuan Yang     }
424e5c0367eSXiaojuan Yang     d->cntrctl = 0;
425e5c0367eSXiaojuan Yang }
426e5c0367eSXiaojuan Yang 
427c117f68aSXiaojuan Yang static int ls7a_rtc_pre_save(void *opaque)
428c117f68aSXiaojuan Yang {
429c117f68aSXiaojuan Yang     LS7ARtcState *s = LS7A_RTC(opaque);
430c117f68aSXiaojuan Yang 
431c117f68aSXiaojuan Yang     ls7a_toy_stop(s);
432c117f68aSXiaojuan Yang     ls7a_rtc_stop(s);
433c117f68aSXiaojuan Yang 
434c117f68aSXiaojuan Yang     return 0;
435c117f68aSXiaojuan Yang }
436c117f68aSXiaojuan Yang 
437c117f68aSXiaojuan Yang static int ls7a_rtc_post_load(void *opaque, int version_id)
438c117f68aSXiaojuan Yang {
439c117f68aSXiaojuan Yang     LS7ARtcState *s = LS7A_RTC(opaque);
440c117f68aSXiaojuan Yang     if (toy_enabled(s)) {
441c117f68aSXiaojuan Yang         ls7a_toy_start(s);
442c117f68aSXiaojuan Yang     }
443c117f68aSXiaojuan Yang 
444c117f68aSXiaojuan Yang     if (rtc_enabled(s)) {
445c117f68aSXiaojuan Yang         ls7a_rtc_start(s);
446c117f68aSXiaojuan Yang     }
447c117f68aSXiaojuan Yang 
448c117f68aSXiaojuan Yang     return 0;
449c117f68aSXiaojuan Yang }
450c117f68aSXiaojuan Yang 
451c117f68aSXiaojuan Yang static const VMStateDescription vmstate_ls7a_rtc = {
452c117f68aSXiaojuan Yang     .name = "ls7a_rtc",
453c117f68aSXiaojuan Yang     .version_id = 1,
454c117f68aSXiaojuan Yang     .minimum_version_id = 1,
455c117f68aSXiaojuan Yang     .pre_save = ls7a_rtc_pre_save,
456c117f68aSXiaojuan Yang     .post_load = ls7a_rtc_post_load,
457a80cc662SRichard Henderson     .fields = (const VMStateField[]) {
458c117f68aSXiaojuan Yang         VMSTATE_INT64(offset_toy, LS7ARtcState),
459c117f68aSXiaojuan Yang         VMSTATE_INT64(offset_rtc, LS7ARtcState),
460c117f68aSXiaojuan Yang         VMSTATE_UINT32_ARRAY(toymatch, LS7ARtcState, TIMER_NUMS),
461c117f68aSXiaojuan Yang         VMSTATE_UINT32_ARRAY(rtcmatch, LS7ARtcState, TIMER_NUMS),
462c117f68aSXiaojuan Yang         VMSTATE_UINT32(cntrctl, LS7ARtcState),
463c117f68aSXiaojuan Yang         VMSTATE_END_OF_LIST()
464c117f68aSXiaojuan Yang     }
465c117f68aSXiaojuan Yang };
466c117f68aSXiaojuan Yang 
467c117f68aSXiaojuan Yang static void ls7a_rtc_class_init(ObjectClass *klass, void *data)
468c117f68aSXiaojuan Yang {
469c117f68aSXiaojuan Yang     DeviceClass *dc = DEVICE_CLASS(klass);
470c117f68aSXiaojuan Yang     dc->vmsd = &vmstate_ls7a_rtc;
471c117f68aSXiaojuan Yang     dc->realize = ls7a_rtc_realize;
472e5c0367eSXiaojuan Yang     dc->reset = ls7a_rtc_reset;
473c117f68aSXiaojuan Yang     dc->desc = "ls7a rtc";
474c117f68aSXiaojuan Yang }
475c117f68aSXiaojuan Yang 
476c117f68aSXiaojuan Yang static const TypeInfo ls7a_rtc_info = {
477c117f68aSXiaojuan Yang     .name          = TYPE_LS7A_RTC,
478c117f68aSXiaojuan Yang     .parent        = TYPE_SYS_BUS_DEVICE,
479c117f68aSXiaojuan Yang     .instance_size = sizeof(LS7ARtcState),
480c117f68aSXiaojuan Yang     .class_init    = ls7a_rtc_class_init,
481c117f68aSXiaojuan Yang };
482c117f68aSXiaojuan Yang 
483c117f68aSXiaojuan Yang static void ls7a_rtc_register_types(void)
484c117f68aSXiaojuan Yang {
485c117f68aSXiaojuan Yang     type_register_static(&ls7a_rtc_info);
486c117f68aSXiaojuan Yang }
487c117f68aSXiaojuan Yang 
488c117f68aSXiaojuan Yang type_init(ls7a_rtc_register_types)
489