Statistics
| Branch: | Revision:

root / hw / mc146818rtc.c @ a1bc20df

History | View | Annotate | Download (27.6 kB)

1 80cabfad bellard
/*
2 80cabfad bellard
 * QEMU MC146818 RTC emulation
3 5fafdf24 ths
 *
4 80cabfad bellard
 * Copyright (c) 2003-2004 Fabrice Bellard
5 5fafdf24 ths
 *
6 80cabfad bellard
 * Permission is hereby granted, free of charge, to any person obtaining a copy
7 80cabfad bellard
 * of this software and associated documentation files (the "Software"), to deal
8 80cabfad bellard
 * in the Software without restriction, including without limitation the rights
9 80cabfad bellard
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10 80cabfad bellard
 * copies of the Software, and to permit persons to whom the Software is
11 80cabfad bellard
 * furnished to do so, subject to the following conditions:
12 80cabfad bellard
 *
13 80cabfad bellard
 * The above copyright notice and this permission notice shall be included in
14 80cabfad bellard
 * all copies or substantial portions of the Software.
15 80cabfad bellard
 *
16 80cabfad bellard
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 80cabfad bellard
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 80cabfad bellard
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19 80cabfad bellard
 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 80cabfad bellard
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21 80cabfad bellard
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
22 80cabfad bellard
 * THE SOFTWARE.
23 80cabfad bellard
 */
24 87ecb68b pbrook
#include "hw.h"
25 87ecb68b pbrook
#include "qemu-timer.h"
26 87ecb68b pbrook
#include "sysemu.h"
27 1d914fa0 Isaku Yamahata
#include "mc146818rtc.h"
28 80cabfad bellard
29 d362e757 Jan Kiszka
#ifdef TARGET_I386
30 d362e757 Jan Kiszka
#include "apic.h"
31 d362e757 Jan Kiszka
#endif
32 d362e757 Jan Kiszka
33 80cabfad bellard
//#define DEBUG_CMOS
34 aa6f63ff Blue Swirl
//#define DEBUG_COALESCED
35 80cabfad bellard
36 ec51e364 Isaku Yamahata
#ifdef DEBUG_CMOS
37 ec51e364 Isaku Yamahata
# define CMOS_DPRINTF(format, ...)      printf(format, ## __VA_ARGS__)
38 ec51e364 Isaku Yamahata
#else
39 ec51e364 Isaku Yamahata
# define CMOS_DPRINTF(format, ...)      do { } while (0)
40 ec51e364 Isaku Yamahata
#endif
41 ec51e364 Isaku Yamahata
42 aa6f63ff Blue Swirl
#ifdef DEBUG_COALESCED
43 aa6f63ff Blue Swirl
# define DPRINTF_C(format, ...)      printf(format, ## __VA_ARGS__)
44 aa6f63ff Blue Swirl
#else
45 aa6f63ff Blue Swirl
# define DPRINTF_C(format, ...)      do { } while (0)
46 aa6f63ff Blue Swirl
#endif
47 aa6f63ff Blue Swirl
48 56038ef6 Yang Zhang
#define NSEC_PER_SEC    1000000000LL
49 00cf5774 Paolo Bonzini
#define SEC_PER_MIN     60
50 00cf5774 Paolo Bonzini
#define MIN_PER_HOUR    60
51 00cf5774 Paolo Bonzini
#define SEC_PER_HOUR    3600
52 00cf5774 Paolo Bonzini
#define HOUR_PER_DAY    24
53 00cf5774 Paolo Bonzini
#define SEC_PER_DAY     86400
54 56038ef6 Yang Zhang
55 dd17765b Gleb Natapov
#define RTC_REINJECT_ON_ACK_COUNT 20
56 e46deaba Paolo Bonzini
#define RTC_CLOCK_RATE            32768
57 56038ef6 Yang Zhang
#define UIP_HOLD_LENGTH           (8 * NSEC_PER_SEC / 32768)
58 ba32edab Gleb Natapov
59 1d914fa0 Isaku Yamahata
typedef struct RTCState {
60 32e0c826 Gerd Hoffmann
    ISADevice dev;
61 b2c5009b Richard Henderson
    MemoryRegion io;
62 dff38e7b bellard
    uint8_t cmos_data[128];
63 dff38e7b bellard
    uint8_t cmos_index;
64 32e0c826 Gerd Hoffmann
    int32_t base_year;
65 56038ef6 Yang Zhang
    uint64_t base_rtc;
66 56038ef6 Yang Zhang
    uint64_t last_update;
67 56038ef6 Yang Zhang
    int64_t offset;
68 d537cf6c pbrook
    qemu_irq irq;
69 100d9891 aurel32
    qemu_irq sqw_irq;
70 18c6e2ff ths
    int it_shift;
71 dff38e7b bellard
    /* periodic timer */
72 dff38e7b bellard
    QEMUTimer *periodic_timer;
73 dff38e7b bellard
    int64_t next_periodic_time;
74 56038ef6 Yang Zhang
    /* update-ended timer */
75 56038ef6 Yang Zhang
    QEMUTimer *update_timer;
76 00cf5774 Paolo Bonzini
    uint64_t next_alarm_time;
77 ba32edab Gleb Natapov
    uint16_t irq_reinject_on_ack_count;
78 73822ec8 aliguori
    uint32_t irq_coalesced;
79 73822ec8 aliguori
    uint32_t period;
80 93b66569 aliguori
    QEMUTimer *coalesced_timer;
81 17604dac Jan Kiszka
    Notifier clock_reset_notifier;
82 433acf0d Jan Kiszka
    LostTickPolicy lost_tick_policy;
83 da98c8eb Gerd Hoffmann
    Notifier suspend_notifier;
84 1d914fa0 Isaku Yamahata
} RTCState;
85 dff38e7b bellard
86 dff38e7b bellard
static void rtc_set_time(RTCState *s);
87 56038ef6 Yang Zhang
static void rtc_update_time(RTCState *s);
88 e2826cf4 Paolo Bonzini
static void rtc_set_cmos(RTCState *s, const struct tm *tm);
89 56038ef6 Yang Zhang
static inline int rtc_from_bcd(RTCState *s, int a);
90 00cf5774 Paolo Bonzini
static uint64_t get_next_alarm(RTCState *s);
91 56038ef6 Yang Zhang
92 41a9b8b2 Yang Zhang
static inline bool rtc_running(RTCState *s)
93 41a9b8b2 Yang Zhang
{
94 41a9b8b2 Yang Zhang
    return (!(s->cmos_data[RTC_REG_B] & REG_B_SET) &&
95 41a9b8b2 Yang Zhang
            (s->cmos_data[RTC_REG_A] & 0x70) <= 0x20);
96 41a9b8b2 Yang Zhang
}
97 41a9b8b2 Yang Zhang
98 56038ef6 Yang Zhang
static uint64_t get_guest_rtc_ns(RTCState *s)
99 56038ef6 Yang Zhang
{
100 56038ef6 Yang Zhang
    uint64_t guest_rtc;
101 56038ef6 Yang Zhang
    uint64_t guest_clock = qemu_get_clock_ns(rtc_clock);
102 56038ef6 Yang Zhang
103 56038ef6 Yang Zhang
    guest_rtc = s->base_rtc * NSEC_PER_SEC
104 56038ef6 Yang Zhang
                 + guest_clock - s->last_update + s->offset;
105 56038ef6 Yang Zhang
    return guest_rtc;
106 56038ef6 Yang Zhang
}
107 dff38e7b bellard
108 93b66569 aliguori
#ifdef TARGET_I386
109 93b66569 aliguori
static void rtc_coalesced_timer_update(RTCState *s)
110 93b66569 aliguori
{
111 93b66569 aliguori
    if (s->irq_coalesced == 0) {
112 93b66569 aliguori
        qemu_del_timer(s->coalesced_timer);
113 93b66569 aliguori
    } else {
114 93b66569 aliguori
        /* divide each RTC interval to 2 - 8 smaller intervals */
115 93b66569 aliguori
        int c = MIN(s->irq_coalesced, 7) + 1; 
116 74475455 Paolo Bonzini
        int64_t next_clock = qemu_get_clock_ns(rtc_clock) +
117 e46deaba Paolo Bonzini
            muldiv64(s->period / c, get_ticks_per_sec(), RTC_CLOCK_RATE);
118 93b66569 aliguori
        qemu_mod_timer(s->coalesced_timer, next_clock);
119 93b66569 aliguori
    }
120 93b66569 aliguori
}
121 93b66569 aliguori
122 93b66569 aliguori
static void rtc_coalesced_timer(void *opaque)
123 93b66569 aliguori
{
124 93b66569 aliguori
    RTCState *s = opaque;
125 93b66569 aliguori
126 93b66569 aliguori
    if (s->irq_coalesced != 0) {
127 93b66569 aliguori
        apic_reset_irq_delivered();
128 93b66569 aliguori
        s->cmos_data[RTC_REG_C] |= 0xc0;
129 aa6f63ff Blue Swirl
        DPRINTF_C("cmos: injecting from timer\n");
130 7d932dfd Jan Kiszka
        qemu_irq_raise(s->irq);
131 93b66569 aliguori
        if (apic_get_irq_delivered()) {
132 93b66569 aliguori
            s->irq_coalesced--;
133 aa6f63ff Blue Swirl
            DPRINTF_C("cmos: coalesced irqs decreased to %d\n",
134 aa6f63ff Blue Swirl
                      s->irq_coalesced);
135 93b66569 aliguori
        }
136 93b66569 aliguori
    }
137 93b66569 aliguori
138 93b66569 aliguori
    rtc_coalesced_timer_update(s);
139 93b66569 aliguori
}
140 93b66569 aliguori
#endif
141 93b66569 aliguori
142 56038ef6 Yang Zhang
/* handle periodic timer */
143 c4c18e24 Yang Zhang
static void periodic_timer_update(RTCState *s, int64_t current_time)
144 dff38e7b bellard
{
145 dff38e7b bellard
    int period_code, period;
146 dff38e7b bellard
    int64_t cur_clock, next_irq_clock;
147 dff38e7b bellard
148 dff38e7b bellard
    period_code = s->cmos_data[RTC_REG_A] & 0x0f;
149 100d9891 aurel32
    if (period_code != 0
150 7d932dfd Jan Kiszka
        && ((s->cmos_data[RTC_REG_B] & REG_B_PIE)
151 100d9891 aurel32
            || ((s->cmos_data[RTC_REG_B] & REG_B_SQWE) && s->sqw_irq))) {
152 dff38e7b bellard
        if (period_code <= 2)
153 dff38e7b bellard
            period_code += 7;
154 dff38e7b bellard
        /* period in 32 Khz cycles */
155 dff38e7b bellard
        period = 1 << (period_code - 1);
156 73822ec8 aliguori
#ifdef TARGET_I386
157 aa6f63ff Blue Swirl
        if (period != s->period) {
158 73822ec8 aliguori
            s->irq_coalesced = (s->irq_coalesced * s->period) / period;
159 aa6f63ff Blue Swirl
            DPRINTF_C("cmos: coalesced irqs scaled to %d\n", s->irq_coalesced);
160 aa6f63ff Blue Swirl
        }
161 73822ec8 aliguori
        s->period = period;
162 73822ec8 aliguori
#endif
163 dff38e7b bellard
        /* compute 32 khz clock */
164 e46deaba Paolo Bonzini
        cur_clock = muldiv64(current_time, RTC_CLOCK_RATE, get_ticks_per_sec());
165 dff38e7b bellard
        next_irq_clock = (cur_clock & ~(period - 1)) + period;
166 6875204c Jan Kiszka
        s->next_periodic_time =
167 e46deaba Paolo Bonzini
            muldiv64(next_irq_clock, get_ticks_per_sec(), RTC_CLOCK_RATE) + 1;
168 dff38e7b bellard
        qemu_mod_timer(s->periodic_timer, s->next_periodic_time);
169 dff38e7b bellard
    } else {
170 73822ec8 aliguori
#ifdef TARGET_I386
171 73822ec8 aliguori
        s->irq_coalesced = 0;
172 73822ec8 aliguori
#endif
173 dff38e7b bellard
        qemu_del_timer(s->periodic_timer);
174 dff38e7b bellard
    }
175 dff38e7b bellard
}
176 dff38e7b bellard
177 dff38e7b bellard
static void rtc_periodic_timer(void *opaque)
178 dff38e7b bellard
{
179 dff38e7b bellard
    RTCState *s = opaque;
180 dff38e7b bellard
181 c4c18e24 Yang Zhang
    periodic_timer_update(s, s->next_periodic_time);
182 663447d4 Paolo Bonzini
    s->cmos_data[RTC_REG_C] |= REG_C_PF;
183 100d9891 aurel32
    if (s->cmos_data[RTC_REG_B] & REG_B_PIE) {
184 663447d4 Paolo Bonzini
        s->cmos_data[RTC_REG_C] |= REG_C_IRQF;
185 93b66569 aliguori
#ifdef TARGET_I386
186 433acf0d Jan Kiszka
        if (s->lost_tick_policy == LOST_TICK_SLEW) {
187 ba32edab Gleb Natapov
            if (s->irq_reinject_on_ack_count >= RTC_REINJECT_ON_ACK_COUNT)
188 ba32edab Gleb Natapov
                s->irq_reinject_on_ack_count = 0;                
189 93b66569 aliguori
            apic_reset_irq_delivered();
190 7d932dfd Jan Kiszka
            qemu_irq_raise(s->irq);
191 93b66569 aliguori
            if (!apic_get_irq_delivered()) {
192 93b66569 aliguori
                s->irq_coalesced++;
193 93b66569 aliguori
                rtc_coalesced_timer_update(s);
194 aa6f63ff Blue Swirl
                DPRINTF_C("cmos: coalesced irqs increased to %d\n",
195 aa6f63ff Blue Swirl
                          s->irq_coalesced);
196 93b66569 aliguori
            }
197 93b66569 aliguori
        } else
198 93b66569 aliguori
#endif
199 7d932dfd Jan Kiszka
        qemu_irq_raise(s->irq);
200 100d9891 aurel32
    }
201 100d9891 aurel32
    if (s->cmos_data[RTC_REG_B] & REG_B_SQWE) {
202 100d9891 aurel32
        /* Not square wave at all but we don't want 2048Hz interrupts!
203 100d9891 aurel32
           Must be seen as a pulse.  */
204 100d9891 aurel32
        qemu_irq_raise(s->sqw_irq);
205 100d9891 aurel32
    }
206 dff38e7b bellard
}
207 80cabfad bellard
208 56038ef6 Yang Zhang
/* handle update-ended timer */
209 56038ef6 Yang Zhang
static void check_update_timer(RTCState *s)
210 56038ef6 Yang Zhang
{
211 56038ef6 Yang Zhang
    uint64_t next_update_time;
212 56038ef6 Yang Zhang
    uint64_t guest_nsec;
213 00cf5774 Paolo Bonzini
    int next_alarm_sec;
214 56038ef6 Yang Zhang
215 41a9b8b2 Yang Zhang
    /* From the data sheet: "Holding the dividers in reset prevents
216 41a9b8b2 Yang Zhang
     * interrupts from operating, while setting the SET bit allows"
217 41a9b8b2 Yang Zhang
     * them to occur.  However, it will prevent an alarm interrupt
218 41a9b8b2 Yang Zhang
     * from occurring, because the time of day is not updated.
219 56038ef6 Yang Zhang
     */
220 41a9b8b2 Yang Zhang
    if ((s->cmos_data[RTC_REG_A] & 0x60) == 0x60) {
221 41a9b8b2 Yang Zhang
        qemu_del_timer(s->update_timer);
222 41a9b8b2 Yang Zhang
        return;
223 41a9b8b2 Yang Zhang
    }
224 56038ef6 Yang Zhang
    if ((s->cmos_data[RTC_REG_C] & REG_C_UF) &&
225 56038ef6 Yang Zhang
        (s->cmos_data[RTC_REG_B] & REG_B_SET)) {
226 56038ef6 Yang Zhang
        qemu_del_timer(s->update_timer);
227 56038ef6 Yang Zhang
        return;
228 56038ef6 Yang Zhang
    }
229 56038ef6 Yang Zhang
    if ((s->cmos_data[RTC_REG_C] & REG_C_UF) &&
230 56038ef6 Yang Zhang
        (s->cmos_data[RTC_REG_C] & REG_C_AF)) {
231 56038ef6 Yang Zhang
        qemu_del_timer(s->update_timer);
232 56038ef6 Yang Zhang
        return;
233 56038ef6 Yang Zhang
    }
234 56038ef6 Yang Zhang
235 56038ef6 Yang Zhang
    guest_nsec = get_guest_rtc_ns(s) % NSEC_PER_SEC;
236 00cf5774 Paolo Bonzini
    /* if UF is clear, reprogram to next second */
237 56038ef6 Yang Zhang
    next_update_time = qemu_get_clock_ns(rtc_clock)
238 56038ef6 Yang Zhang
        + NSEC_PER_SEC - guest_nsec;
239 00cf5774 Paolo Bonzini
240 00cf5774 Paolo Bonzini
    /* Compute time of next alarm.  One second is already accounted
241 00cf5774 Paolo Bonzini
     * for in next_update_time.
242 00cf5774 Paolo Bonzini
     */
243 00cf5774 Paolo Bonzini
    next_alarm_sec = get_next_alarm(s);
244 00cf5774 Paolo Bonzini
    s->next_alarm_time = next_update_time + (next_alarm_sec - 1) * NSEC_PER_SEC;
245 00cf5774 Paolo Bonzini
246 00cf5774 Paolo Bonzini
    if (s->cmos_data[RTC_REG_C] & REG_C_UF) {
247 00cf5774 Paolo Bonzini
        /* UF is set, but AF is clear.  Program the timer to target
248 00cf5774 Paolo Bonzini
         * the alarm time.  */
249 00cf5774 Paolo Bonzini
        next_update_time = s->next_alarm_time;
250 00cf5774 Paolo Bonzini
    }
251 56038ef6 Yang Zhang
    if (next_update_time != qemu_timer_expire_time_ns(s->update_timer)) {
252 56038ef6 Yang Zhang
        qemu_mod_timer(s->update_timer, next_update_time);
253 56038ef6 Yang Zhang
    }
254 56038ef6 Yang Zhang
}
255 56038ef6 Yang Zhang
256 56038ef6 Yang Zhang
static inline uint8_t convert_hour(RTCState *s, uint8_t hour)
257 56038ef6 Yang Zhang
{
258 56038ef6 Yang Zhang
    if (!(s->cmos_data[RTC_REG_B] & REG_B_24H)) {
259 56038ef6 Yang Zhang
        hour %= 12;
260 56038ef6 Yang Zhang
        if (s->cmos_data[RTC_HOURS] & 0x80) {
261 56038ef6 Yang Zhang
            hour += 12;
262 56038ef6 Yang Zhang
        }
263 56038ef6 Yang Zhang
    }
264 56038ef6 Yang Zhang
    return hour;
265 56038ef6 Yang Zhang
}
266 56038ef6 Yang Zhang
267 00cf5774 Paolo Bonzini
static uint64_t get_next_alarm(RTCState *s)
268 56038ef6 Yang Zhang
{
269 00cf5774 Paolo Bonzini
    int32_t alarm_sec, alarm_min, alarm_hour, cur_hour, cur_min, cur_sec;
270 00cf5774 Paolo Bonzini
    int32_t hour, min, sec;
271 00cf5774 Paolo Bonzini
272 00cf5774 Paolo Bonzini
    rtc_update_time(s);
273 56038ef6 Yang Zhang
274 56038ef6 Yang Zhang
    alarm_sec = rtc_from_bcd(s, s->cmos_data[RTC_SECONDS_ALARM]);
275 56038ef6 Yang Zhang
    alarm_min = rtc_from_bcd(s, s->cmos_data[RTC_MINUTES_ALARM]);
276 56038ef6 Yang Zhang
    alarm_hour = rtc_from_bcd(s, s->cmos_data[RTC_HOURS_ALARM]);
277 00cf5774 Paolo Bonzini
    alarm_hour = alarm_hour == -1 ? -1 : convert_hour(s, alarm_hour);
278 56038ef6 Yang Zhang
279 56038ef6 Yang Zhang
    cur_sec = rtc_from_bcd(s, s->cmos_data[RTC_SECONDS]);
280 56038ef6 Yang Zhang
    cur_min = rtc_from_bcd(s, s->cmos_data[RTC_MINUTES]);
281 56038ef6 Yang Zhang
    cur_hour = rtc_from_bcd(s, s->cmos_data[RTC_HOURS]);
282 56038ef6 Yang Zhang
    cur_hour = convert_hour(s, cur_hour);
283 56038ef6 Yang Zhang
284 00cf5774 Paolo Bonzini
    if (alarm_hour == -1) {
285 00cf5774 Paolo Bonzini
        alarm_hour = cur_hour;
286 00cf5774 Paolo Bonzini
        if (alarm_min == -1) {
287 00cf5774 Paolo Bonzini
            alarm_min = cur_min;
288 00cf5774 Paolo Bonzini
            if (alarm_sec == -1) {
289 00cf5774 Paolo Bonzini
                alarm_sec = cur_sec + 1;
290 00cf5774 Paolo Bonzini
            } else if (cur_sec > alarm_sec) {
291 00cf5774 Paolo Bonzini
                alarm_min++;
292 00cf5774 Paolo Bonzini
            }
293 00cf5774 Paolo Bonzini
        } else if (cur_min == alarm_min) {
294 00cf5774 Paolo Bonzini
            if (alarm_sec == -1) {
295 00cf5774 Paolo Bonzini
                alarm_sec = cur_sec + 1;
296 00cf5774 Paolo Bonzini
            } else {
297 00cf5774 Paolo Bonzini
                if (cur_sec > alarm_sec) {
298 00cf5774 Paolo Bonzini
                    alarm_hour++;
299 00cf5774 Paolo Bonzini
                }
300 00cf5774 Paolo Bonzini
            }
301 00cf5774 Paolo Bonzini
            if (alarm_sec == SEC_PER_MIN) {
302 00cf5774 Paolo Bonzini
                /* wrap to next hour, minutes is not in don't care mode */
303 00cf5774 Paolo Bonzini
                alarm_sec = 0;
304 00cf5774 Paolo Bonzini
                alarm_hour++;
305 00cf5774 Paolo Bonzini
            }
306 00cf5774 Paolo Bonzini
        } else if (cur_min > alarm_min) {
307 00cf5774 Paolo Bonzini
            alarm_hour++;
308 00cf5774 Paolo Bonzini
        }
309 00cf5774 Paolo Bonzini
    } else if (cur_hour == alarm_hour) {
310 00cf5774 Paolo Bonzini
        if (alarm_min == -1) {
311 00cf5774 Paolo Bonzini
            alarm_min = cur_min;
312 00cf5774 Paolo Bonzini
            if (alarm_sec == -1) {
313 00cf5774 Paolo Bonzini
                alarm_sec = cur_sec + 1;
314 00cf5774 Paolo Bonzini
            } else if (cur_sec > alarm_sec) {
315 00cf5774 Paolo Bonzini
                alarm_min++;
316 00cf5774 Paolo Bonzini
            }
317 00cf5774 Paolo Bonzini
318 00cf5774 Paolo Bonzini
            if (alarm_sec == SEC_PER_MIN) {
319 00cf5774 Paolo Bonzini
                alarm_sec = 0;
320 00cf5774 Paolo Bonzini
                alarm_min++;
321 00cf5774 Paolo Bonzini
            }
322 00cf5774 Paolo Bonzini
            /* wrap to next day, hour is not in don't care mode */
323 00cf5774 Paolo Bonzini
            alarm_min %= MIN_PER_HOUR;
324 00cf5774 Paolo Bonzini
        } else if (cur_min == alarm_min) {
325 00cf5774 Paolo Bonzini
            if (alarm_sec == -1) {
326 00cf5774 Paolo Bonzini
                alarm_sec = cur_sec + 1;
327 00cf5774 Paolo Bonzini
            }
328 00cf5774 Paolo Bonzini
            /* wrap to next day, hours+minutes not in don't care mode */
329 00cf5774 Paolo Bonzini
            alarm_sec %= SEC_PER_MIN;
330 00cf5774 Paolo Bonzini
        }
331 56038ef6 Yang Zhang
    }
332 56038ef6 Yang Zhang
333 00cf5774 Paolo Bonzini
    /* values that are still don't care fire at the next min/sec */
334 00cf5774 Paolo Bonzini
    if (alarm_min == -1) {
335 00cf5774 Paolo Bonzini
        alarm_min = 0;
336 00cf5774 Paolo Bonzini
    }
337 00cf5774 Paolo Bonzini
    if (alarm_sec == -1) {
338 00cf5774 Paolo Bonzini
        alarm_sec = 0;
339 00cf5774 Paolo Bonzini
    }
340 00cf5774 Paolo Bonzini
341 00cf5774 Paolo Bonzini
    /* keep values in range */
342 00cf5774 Paolo Bonzini
    if (alarm_sec == SEC_PER_MIN) {
343 00cf5774 Paolo Bonzini
        alarm_sec = 0;
344 00cf5774 Paolo Bonzini
        alarm_min++;
345 00cf5774 Paolo Bonzini
    }
346 00cf5774 Paolo Bonzini
    if (alarm_min == MIN_PER_HOUR) {
347 00cf5774 Paolo Bonzini
        alarm_min = 0;
348 00cf5774 Paolo Bonzini
        alarm_hour++;
349 00cf5774 Paolo Bonzini
    }
350 00cf5774 Paolo Bonzini
    alarm_hour %= HOUR_PER_DAY;
351 00cf5774 Paolo Bonzini
352 00cf5774 Paolo Bonzini
    hour = alarm_hour - cur_hour;
353 00cf5774 Paolo Bonzini
    min = hour * MIN_PER_HOUR + alarm_min - cur_min;
354 00cf5774 Paolo Bonzini
    sec = min * SEC_PER_MIN + alarm_sec - cur_sec;
355 00cf5774 Paolo Bonzini
    return sec <= 0 ? sec + SEC_PER_DAY : sec;
356 56038ef6 Yang Zhang
}
357 56038ef6 Yang Zhang
358 56038ef6 Yang Zhang
static void rtc_update_timer(void *opaque)
359 56038ef6 Yang Zhang
{
360 56038ef6 Yang Zhang
    RTCState *s = opaque;
361 56038ef6 Yang Zhang
    int32_t irqs = REG_C_UF;
362 56038ef6 Yang Zhang
    int32_t new_irqs;
363 56038ef6 Yang Zhang
364 41a9b8b2 Yang Zhang
    assert((s->cmos_data[RTC_REG_A] & 0x60) != 0x60);
365 41a9b8b2 Yang Zhang
366 56038ef6 Yang Zhang
    /* UIP might have been latched, update time and clear it.  */
367 56038ef6 Yang Zhang
    rtc_update_time(s);
368 56038ef6 Yang Zhang
    s->cmos_data[RTC_REG_A] &= ~REG_A_UIP;
369 56038ef6 Yang Zhang
370 00cf5774 Paolo Bonzini
    if (qemu_get_clock_ns(rtc_clock) >= s->next_alarm_time) {
371 56038ef6 Yang Zhang
        irqs |= REG_C_AF;
372 56038ef6 Yang Zhang
        if (s->cmos_data[RTC_REG_B] & REG_B_AIE) {
373 56038ef6 Yang Zhang
            qemu_system_wakeup_request(QEMU_WAKEUP_REASON_RTC);
374 56038ef6 Yang Zhang
        }
375 56038ef6 Yang Zhang
    }
376 00cf5774 Paolo Bonzini
377 56038ef6 Yang Zhang
    new_irqs = irqs & ~s->cmos_data[RTC_REG_C];
378 56038ef6 Yang Zhang
    s->cmos_data[RTC_REG_C] |= irqs;
379 56038ef6 Yang Zhang
    if ((new_irqs & s->cmos_data[RTC_REG_B]) != 0) {
380 56038ef6 Yang Zhang
        s->cmos_data[RTC_REG_C] |= REG_C_IRQF;
381 56038ef6 Yang Zhang
        qemu_irq_raise(s->irq);
382 56038ef6 Yang Zhang
    }
383 56038ef6 Yang Zhang
    check_update_timer(s);
384 56038ef6 Yang Zhang
}
385 56038ef6 Yang Zhang
386 0da8c842 Alexander Graf
static void cmos_ioport_write(void *opaque, hwaddr addr,
387 0da8c842 Alexander Graf
                              uint64_t data, unsigned size)
388 80cabfad bellard
{
389 b41a2cd1 bellard
    RTCState *s = opaque;
390 80cabfad bellard
391 80cabfad bellard
    if ((addr & 1) == 0) {
392 80cabfad bellard
        s->cmos_index = data & 0x7f;
393 80cabfad bellard
    } else {
394 ec51e364 Isaku Yamahata
        CMOS_DPRINTF("cmos: write index=0x%02x val=0x%02x\n",
395 ec51e364 Isaku Yamahata
                     s->cmos_index, data);
396 dff38e7b bellard
        switch(s->cmos_index) {
397 80cabfad bellard
        case RTC_SECONDS_ALARM:
398 80cabfad bellard
        case RTC_MINUTES_ALARM:
399 80cabfad bellard
        case RTC_HOURS_ALARM:
400 80cabfad bellard
            s->cmos_data[s->cmos_index] = data;
401 56038ef6 Yang Zhang
            check_update_timer(s);
402 80cabfad bellard
            break;
403 e67edb94 Paolo Bonzini
        case RTC_IBM_PS2_CENTURY_BYTE:
404 e67edb94 Paolo Bonzini
            s->cmos_index = RTC_CENTURY;
405 e67edb94 Paolo Bonzini
            /* fall through */
406 e67edb94 Paolo Bonzini
        case RTC_CENTURY:
407 80cabfad bellard
        case RTC_SECONDS:
408 80cabfad bellard
        case RTC_MINUTES:
409 80cabfad bellard
        case RTC_HOURS:
410 80cabfad bellard
        case RTC_DAY_OF_WEEK:
411 80cabfad bellard
        case RTC_DAY_OF_MONTH:
412 80cabfad bellard
        case RTC_MONTH:
413 80cabfad bellard
        case RTC_YEAR:
414 80cabfad bellard
            s->cmos_data[s->cmos_index] = data;
415 dff38e7b bellard
            /* if in set mode, do not update the time */
416 41a9b8b2 Yang Zhang
            if (rtc_running(s)) {
417 dff38e7b bellard
                rtc_set_time(s);
418 56038ef6 Yang Zhang
                check_update_timer(s);
419 dff38e7b bellard
            }
420 80cabfad bellard
            break;
421 80cabfad bellard
        case RTC_REG_A:
422 41a9b8b2 Yang Zhang
            if ((data & 0x60) == 0x60) {
423 41a9b8b2 Yang Zhang
                if (rtc_running(s)) {
424 41a9b8b2 Yang Zhang
                    rtc_update_time(s);
425 41a9b8b2 Yang Zhang
                }
426 41a9b8b2 Yang Zhang
                /* What happens to UIP when divider reset is enabled is
427 41a9b8b2 Yang Zhang
                 * unclear from the datasheet.  Shouldn't matter much
428 41a9b8b2 Yang Zhang
                 * though.
429 41a9b8b2 Yang Zhang
                 */
430 41a9b8b2 Yang Zhang
                s->cmos_data[RTC_REG_A] &= ~REG_A_UIP;
431 41a9b8b2 Yang Zhang
            } else if (((s->cmos_data[RTC_REG_A] & 0x60) == 0x60) &&
432 41a9b8b2 Yang Zhang
                    (data & 0x70)  <= 0x20) {
433 41a9b8b2 Yang Zhang
                /* when the divider reset is removed, the first update cycle
434 41a9b8b2 Yang Zhang
                 * begins one-half second later*/
435 41a9b8b2 Yang Zhang
                if (!(s->cmos_data[RTC_REG_B] & REG_B_SET)) {
436 41a9b8b2 Yang Zhang
                    s->offset = 500000000;
437 41a9b8b2 Yang Zhang
                    rtc_set_time(s);
438 41a9b8b2 Yang Zhang
                }
439 41a9b8b2 Yang Zhang
                s->cmos_data[RTC_REG_A] &= ~REG_A_UIP;
440 41a9b8b2 Yang Zhang
            }
441 dff38e7b bellard
            /* UIP bit is read only */
442 dff38e7b bellard
            s->cmos_data[RTC_REG_A] = (data & ~REG_A_UIP) |
443 dff38e7b bellard
                (s->cmos_data[RTC_REG_A] & REG_A_UIP);
444 c4c18e24 Yang Zhang
            periodic_timer_update(s, qemu_get_clock_ns(rtc_clock));
445 56038ef6 Yang Zhang
            check_update_timer(s);
446 dff38e7b bellard
            break;
447 80cabfad bellard
        case RTC_REG_B:
448 dff38e7b bellard
            if (data & REG_B_SET) {
449 56038ef6 Yang Zhang
                /* update cmos to when the rtc was stopping */
450 41a9b8b2 Yang Zhang
                if (rtc_running(s)) {
451 56038ef6 Yang Zhang
                    rtc_update_time(s);
452 56038ef6 Yang Zhang
                }
453 dff38e7b bellard
                /* set mode: reset UIP mode */
454 dff38e7b bellard
                s->cmos_data[RTC_REG_A] &= ~REG_A_UIP;
455 dff38e7b bellard
                data &= ~REG_B_UIE;
456 dff38e7b bellard
            } else {
457 dff38e7b bellard
                /* if disabling set mode, update the time */
458 41a9b8b2 Yang Zhang
                if ((s->cmos_data[RTC_REG_B] & REG_B_SET) &&
459 41a9b8b2 Yang Zhang
                    (s->cmos_data[RTC_REG_A] & 0x70) <= 0x20) {
460 56038ef6 Yang Zhang
                    s->offset = get_guest_rtc_ns(s) % NSEC_PER_SEC;
461 dff38e7b bellard
                    rtc_set_time(s);
462 dff38e7b bellard
                }
463 dff38e7b bellard
            }
464 9324cc50 Yang Zhang
            /* if an interrupt flag is already set when the interrupt
465 9324cc50 Yang Zhang
             * becomes enabled, raise an interrupt immediately.  */
466 9324cc50 Yang Zhang
            if (data & s->cmos_data[RTC_REG_C] & REG_C_MASK) {
467 9324cc50 Yang Zhang
                s->cmos_data[RTC_REG_C] |= REG_C_IRQF;
468 9324cc50 Yang Zhang
                qemu_irq_raise(s->irq);
469 9324cc50 Yang Zhang
            } else {
470 9324cc50 Yang Zhang
                s->cmos_data[RTC_REG_C] &= ~REG_C_IRQF;
471 9324cc50 Yang Zhang
                qemu_irq_lower(s->irq);
472 9324cc50 Yang Zhang
            }
473 bedc572e Yang Zhang
            s->cmos_data[RTC_REG_B] = data;
474 c4c18e24 Yang Zhang
            periodic_timer_update(s, qemu_get_clock_ns(rtc_clock));
475 56038ef6 Yang Zhang
            check_update_timer(s);
476 80cabfad bellard
            break;
477 80cabfad bellard
        case RTC_REG_C:
478 80cabfad bellard
        case RTC_REG_D:
479 80cabfad bellard
            /* cannot write to them */
480 80cabfad bellard
            break;
481 80cabfad bellard
        default:
482 80cabfad bellard
            s->cmos_data[s->cmos_index] = data;
483 80cabfad bellard
            break;
484 80cabfad bellard
        }
485 80cabfad bellard
    }
486 80cabfad bellard
}
487 80cabfad bellard
488 abd0c6bd Paul Brook
static inline int rtc_to_bcd(RTCState *s, int a)
489 80cabfad bellard
{
490 6f1bf24d aurel32
    if (s->cmos_data[RTC_REG_B] & REG_B_DM) {
491 dff38e7b bellard
        return a;
492 dff38e7b bellard
    } else {
493 dff38e7b bellard
        return ((a / 10) << 4) | (a % 10);
494 dff38e7b bellard
    }
495 80cabfad bellard
}
496 80cabfad bellard
497 abd0c6bd Paul Brook
static inline int rtc_from_bcd(RTCState *s, int a)
498 80cabfad bellard
{
499 00cf5774 Paolo Bonzini
    if ((a & 0xc0) == 0xc0) {
500 00cf5774 Paolo Bonzini
        return -1;
501 00cf5774 Paolo Bonzini
    }
502 6f1bf24d aurel32
    if (s->cmos_data[RTC_REG_B] & REG_B_DM) {
503 dff38e7b bellard
        return a;
504 dff38e7b bellard
    } else {
505 dff38e7b bellard
        return ((a >> 4) * 10) + (a & 0x0f);
506 dff38e7b bellard
    }
507 dff38e7b bellard
}
508 dff38e7b bellard
509 e2826cf4 Paolo Bonzini
static void rtc_get_time(RTCState *s, struct tm *tm)
510 dff38e7b bellard
{
511 abd0c6bd Paul Brook
    tm->tm_sec = rtc_from_bcd(s, s->cmos_data[RTC_SECONDS]);
512 abd0c6bd Paul Brook
    tm->tm_min = rtc_from_bcd(s, s->cmos_data[RTC_MINUTES]);
513 abd0c6bd Paul Brook
    tm->tm_hour = rtc_from_bcd(s, s->cmos_data[RTC_HOURS] & 0x7f);
514 3b89eb43 Paolo Bonzini
    if (!(s->cmos_data[RTC_REG_B] & REG_B_24H)) {
515 3b89eb43 Paolo Bonzini
        tm->tm_hour %= 12;
516 3b89eb43 Paolo Bonzini
        if (s->cmos_data[RTC_HOURS] & 0x80) {
517 3b89eb43 Paolo Bonzini
            tm->tm_hour += 12;
518 3b89eb43 Paolo Bonzini
        }
519 43f493af bellard
    }
520 abd0c6bd Paul Brook
    tm->tm_wday = rtc_from_bcd(s, s->cmos_data[RTC_DAY_OF_WEEK]) - 1;
521 abd0c6bd Paul Brook
    tm->tm_mday = rtc_from_bcd(s, s->cmos_data[RTC_DAY_OF_MONTH]);
522 abd0c6bd Paul Brook
    tm->tm_mon = rtc_from_bcd(s, s->cmos_data[RTC_MONTH]) - 1;
523 b8994faf Paolo Bonzini
    tm->tm_year =
524 b8994faf Paolo Bonzini
        rtc_from_bcd(s, s->cmos_data[RTC_YEAR]) + s->base_year +
525 b8994faf Paolo Bonzini
        rtc_from_bcd(s, s->cmos_data[RTC_CENTURY]) * 100 - 1900;
526 e2826cf4 Paolo Bonzini
}
527 e2826cf4 Paolo Bonzini
528 e2826cf4 Paolo Bonzini
static void rtc_set_time(RTCState *s)
529 e2826cf4 Paolo Bonzini
{
530 e2826cf4 Paolo Bonzini
    struct tm tm;
531 80cd3478 Luiz Capitulino
532 e2826cf4 Paolo Bonzini
    rtc_get_time(s, &tm);
533 e2826cf4 Paolo Bonzini
    s->base_rtc = mktimegm(&tm);
534 56038ef6 Yang Zhang
    s->last_update = qemu_get_clock_ns(rtc_clock);
535 56038ef6 Yang Zhang
536 e2826cf4 Paolo Bonzini
    rtc_change_mon_event(&tm);
537 43f493af bellard
}
538 43f493af bellard
539 e2826cf4 Paolo Bonzini
static void rtc_set_cmos(RTCState *s, const struct tm *tm)
540 43f493af bellard
{
541 42fc73a1 aurel32
    int year;
542 dff38e7b bellard
543 abd0c6bd Paul Brook
    s->cmos_data[RTC_SECONDS] = rtc_to_bcd(s, tm->tm_sec);
544 abd0c6bd Paul Brook
    s->cmos_data[RTC_MINUTES] = rtc_to_bcd(s, tm->tm_min);
545 c29cd656 Aurelien Jarno
    if (s->cmos_data[RTC_REG_B] & REG_B_24H) {
546 43f493af bellard
        /* 24 hour format */
547 abd0c6bd Paul Brook
        s->cmos_data[RTC_HOURS] = rtc_to_bcd(s, tm->tm_hour);
548 43f493af bellard
    } else {
549 43f493af bellard
        /* 12 hour format */
550 3b89eb43 Paolo Bonzini
        int h = (tm->tm_hour % 12) ? tm->tm_hour % 12 : 12;
551 3b89eb43 Paolo Bonzini
        s->cmos_data[RTC_HOURS] = rtc_to_bcd(s, h);
552 43f493af bellard
        if (tm->tm_hour >= 12)
553 43f493af bellard
            s->cmos_data[RTC_HOURS] |= 0x80;
554 43f493af bellard
    }
555 abd0c6bd Paul Brook
    s->cmos_data[RTC_DAY_OF_WEEK] = rtc_to_bcd(s, tm->tm_wday + 1);
556 abd0c6bd Paul Brook
    s->cmos_data[RTC_DAY_OF_MONTH] = rtc_to_bcd(s, tm->tm_mday);
557 abd0c6bd Paul Brook
    s->cmos_data[RTC_MONTH] = rtc_to_bcd(s, tm->tm_mon + 1);
558 b8994faf Paolo Bonzini
    year = tm->tm_year + 1900 - s->base_year;
559 b8994faf Paolo Bonzini
    s->cmos_data[RTC_YEAR] = rtc_to_bcd(s, year % 100);
560 b8994faf Paolo Bonzini
    s->cmos_data[RTC_CENTURY] = rtc_to_bcd(s, year / 100);
561 43f493af bellard
}
562 43f493af bellard
563 56038ef6 Yang Zhang
static void rtc_update_time(RTCState *s)
564 43f493af bellard
{
565 56038ef6 Yang Zhang
    struct tm ret;
566 56038ef6 Yang Zhang
    time_t guest_sec;
567 56038ef6 Yang Zhang
    int64_t guest_nsec;
568 56038ef6 Yang Zhang
569 56038ef6 Yang Zhang
    guest_nsec = get_guest_rtc_ns(s);
570 56038ef6 Yang Zhang
    guest_sec = guest_nsec / NSEC_PER_SEC;
571 56038ef6 Yang Zhang
    gmtime_r(&guest_sec, &ret);
572 e2826cf4 Paolo Bonzini
    rtc_set_cmos(s, &ret);
573 43f493af bellard
}
574 43f493af bellard
575 56038ef6 Yang Zhang
static int update_in_progress(RTCState *s)
576 43f493af bellard
{
577 56038ef6 Yang Zhang
    int64_t guest_nsec;
578 3b46e624 ths
579 41a9b8b2 Yang Zhang
    if (!rtc_running(s)) {
580 56038ef6 Yang Zhang
        return 0;
581 dff38e7b bellard
    }
582 56038ef6 Yang Zhang
    if (qemu_timer_pending(s->update_timer)) {
583 56038ef6 Yang Zhang
        int64_t next_update_time = qemu_timer_expire_time_ns(s->update_timer);
584 56038ef6 Yang Zhang
        /* Latch UIP until the timer expires.  */
585 56038ef6 Yang Zhang
        if (qemu_get_clock_ns(rtc_clock) >= (next_update_time - UIP_HOLD_LENGTH)) {
586 56038ef6 Yang Zhang
            s->cmos_data[RTC_REG_A] |= REG_A_UIP;
587 56038ef6 Yang Zhang
            return 1;
588 dff38e7b bellard
        }
589 dff38e7b bellard
    }
590 dff38e7b bellard
591 56038ef6 Yang Zhang
    guest_nsec = get_guest_rtc_ns(s);
592 56038ef6 Yang Zhang
    /* UIP bit will be set at last 244us of every second. */
593 56038ef6 Yang Zhang
    if ((guest_nsec % NSEC_PER_SEC) >= (NSEC_PER_SEC - UIP_HOLD_LENGTH)) {
594 56038ef6 Yang Zhang
        return 1;
595 dff38e7b bellard
    }
596 56038ef6 Yang Zhang
    return 0;
597 80cabfad bellard
}
598 80cabfad bellard
599 0da8c842 Alexander Graf
static uint64_t cmos_ioport_read(void *opaque, hwaddr addr,
600 0da8c842 Alexander Graf
                                 unsigned size)
601 80cabfad bellard
{
602 b41a2cd1 bellard
    RTCState *s = opaque;
603 80cabfad bellard
    int ret;
604 80cabfad bellard
    if ((addr & 1) == 0) {
605 80cabfad bellard
        return 0xff;
606 80cabfad bellard
    } else {
607 80cabfad bellard
        switch(s->cmos_index) {
608 e67edb94 Paolo Bonzini
        case RTC_IBM_PS2_CENTURY_BYTE:
609 e67edb94 Paolo Bonzini
            s->cmos_index = RTC_CENTURY;
610 e67edb94 Paolo Bonzini
            /* fall through */
611 e67edb94 Paolo Bonzini
        case RTC_CENTURY:
612 80cabfad bellard
        case RTC_SECONDS:
613 80cabfad bellard
        case RTC_MINUTES:
614 80cabfad bellard
        case RTC_HOURS:
615 80cabfad bellard
        case RTC_DAY_OF_WEEK:
616 80cabfad bellard
        case RTC_DAY_OF_MONTH:
617 80cabfad bellard
        case RTC_MONTH:
618 80cabfad bellard
        case RTC_YEAR:
619 56038ef6 Yang Zhang
            /* if not in set mode, calibrate cmos before
620 56038ef6 Yang Zhang
             * reading*/
621 41a9b8b2 Yang Zhang
            if (rtc_running(s)) {
622 56038ef6 Yang Zhang
                rtc_update_time(s);
623 56038ef6 Yang Zhang
            }
624 80cabfad bellard
            ret = s->cmos_data[s->cmos_index];
625 80cabfad bellard
            break;
626 80cabfad bellard
        case RTC_REG_A:
627 56038ef6 Yang Zhang
            if (update_in_progress(s)) {
628 56038ef6 Yang Zhang
                s->cmos_data[s->cmos_index] |= REG_A_UIP;
629 56038ef6 Yang Zhang
            } else {
630 56038ef6 Yang Zhang
                s->cmos_data[s->cmos_index] &= ~REG_A_UIP;
631 56038ef6 Yang Zhang
            }
632 80cabfad bellard
            ret = s->cmos_data[s->cmos_index];
633 80cabfad bellard
            break;
634 80cabfad bellard
        case RTC_REG_C:
635 80cabfad bellard
            ret = s->cmos_data[s->cmos_index];
636 d537cf6c pbrook
            qemu_irq_lower(s->irq);
637 fbc15e27 Paolo Bonzini
            s->cmos_data[RTC_REG_C] = 0x00;
638 56038ef6 Yang Zhang
            if (ret & (REG_C_UF | REG_C_AF)) {
639 56038ef6 Yang Zhang
                check_update_timer(s);
640 56038ef6 Yang Zhang
            }
641 ba32edab Gleb Natapov
#ifdef TARGET_I386
642 ba32edab Gleb Natapov
            if(s->irq_coalesced &&
643 fbc15e27 Paolo Bonzini
                    (s->cmos_data[RTC_REG_B] & REG_B_PIE) &&
644 ba32edab Gleb Natapov
                    s->irq_reinject_on_ack_count < RTC_REINJECT_ON_ACK_COUNT) {
645 ba32edab Gleb Natapov
                s->irq_reinject_on_ack_count++;
646 fbc15e27 Paolo Bonzini
                s->cmos_data[RTC_REG_C] |= REG_C_IRQF | REG_C_PF;
647 ba32edab Gleb Natapov
                apic_reset_irq_delivered();
648 aa6f63ff Blue Swirl
                DPRINTF_C("cmos: injecting on ack\n");
649 ba32edab Gleb Natapov
                qemu_irq_raise(s->irq);
650 aa6f63ff Blue Swirl
                if (apic_get_irq_delivered()) {
651 ba32edab Gleb Natapov
                    s->irq_coalesced--;
652 aa6f63ff Blue Swirl
                    DPRINTF_C("cmos: coalesced irqs decreased to %d\n",
653 aa6f63ff Blue Swirl
                              s->irq_coalesced);
654 aa6f63ff Blue Swirl
                }
655 ba32edab Gleb Natapov
            }
656 ba32edab Gleb Natapov
#endif
657 80cabfad bellard
            break;
658 80cabfad bellard
        default:
659 80cabfad bellard
            ret = s->cmos_data[s->cmos_index];
660 80cabfad bellard
            break;
661 80cabfad bellard
        }
662 ec51e364 Isaku Yamahata
        CMOS_DPRINTF("cmos: read index=0x%02x val=0x%02x\n",
663 ec51e364 Isaku Yamahata
                     s->cmos_index, ret);
664 80cabfad bellard
        return ret;
665 80cabfad bellard
    }
666 80cabfad bellard
}
667 80cabfad bellard
668 1d914fa0 Isaku Yamahata
void rtc_set_memory(ISADevice *dev, int addr, int val)
669 dff38e7b bellard
{
670 1d914fa0 Isaku Yamahata
    RTCState *s = DO_UPCAST(RTCState, dev, dev);
671 dff38e7b bellard
    if (addr >= 0 && addr <= 127)
672 dff38e7b bellard
        s->cmos_data[addr] = val;
673 dff38e7b bellard
}
674 dff38e7b bellard
675 1d914fa0 Isaku Yamahata
static void rtc_set_date_from_host(ISADevice *dev)
676 ea55ffb3 ths
{
677 1d914fa0 Isaku Yamahata
    RTCState *s = DO_UPCAST(RTCState, dev, dev);
678 f6503059 balrog
    struct tm tm;
679 ea55ffb3 ths
680 f6503059 balrog
    qemu_get_timedate(&tm, 0);
681 56038ef6 Yang Zhang
682 56038ef6 Yang Zhang
    s->base_rtc = mktimegm(&tm);
683 56038ef6 Yang Zhang
    s->last_update = qemu_get_clock_ns(rtc_clock);
684 56038ef6 Yang Zhang
    s->offset = 0;
685 56038ef6 Yang Zhang
686 56038ef6 Yang Zhang
    /* set the CMOS date */
687 e2826cf4 Paolo Bonzini
    rtc_set_cmos(s, &tm);
688 ea55ffb3 ths
}
689 ea55ffb3 ths
690 6b075b8a Juan Quintela
static int rtc_post_load(void *opaque, int version_id)
691 80cabfad bellard
{
692 dff38e7b bellard
    RTCState *s = opaque;
693 dff38e7b bellard
694 56038ef6 Yang Zhang
    if (version_id <= 2) {
695 56038ef6 Yang Zhang
        rtc_set_time(s);
696 56038ef6 Yang Zhang
        s->offset = 0;
697 56038ef6 Yang Zhang
        check_update_timer(s);
698 56038ef6 Yang Zhang
    }
699 56038ef6 Yang Zhang
700 56038ef6 Yang Zhang
#ifdef TARGET_I386
701 048c74c4 Juan Quintela
    if (version_id >= 2) {
702 433acf0d Jan Kiszka
        if (s->lost_tick_policy == LOST_TICK_SLEW) {
703 048c74c4 Juan Quintela
            rtc_coalesced_timer_update(s);
704 048c74c4 Juan Quintela
        }
705 048c74c4 Juan Quintela
    }
706 6b075b8a Juan Quintela
#endif
707 73822ec8 aliguori
    return 0;
708 73822ec8 aliguori
}
709 73822ec8 aliguori
710 6b075b8a Juan Quintela
static const VMStateDescription vmstate_rtc = {
711 6b075b8a Juan Quintela
    .name = "mc146818rtc",
712 56038ef6 Yang Zhang
    .version_id = 3,
713 6b075b8a Juan Quintela
    .minimum_version_id = 1,
714 6b075b8a Juan Quintela
    .minimum_version_id_old = 1,
715 6b075b8a Juan Quintela
    .post_load = rtc_post_load,
716 6b075b8a Juan Quintela
    .fields      = (VMStateField []) {
717 6b075b8a Juan Quintela
        VMSTATE_BUFFER(cmos_data, RTCState),
718 6b075b8a Juan Quintela
        VMSTATE_UINT8(cmos_index, RTCState),
719 89166459 Paolo Bonzini
        VMSTATE_UNUSED(7*4),
720 6b075b8a Juan Quintela
        VMSTATE_TIMER(periodic_timer, RTCState),
721 6b075b8a Juan Quintela
        VMSTATE_INT64(next_periodic_time, RTCState),
722 56038ef6 Yang Zhang
        VMSTATE_UNUSED(3*8),
723 6b075b8a Juan Quintela
        VMSTATE_UINT32_V(irq_coalesced, RTCState, 2),
724 6b075b8a Juan Quintela
        VMSTATE_UINT32_V(period, RTCState, 2),
725 56038ef6 Yang Zhang
        VMSTATE_UINT64_V(base_rtc, RTCState, 3),
726 56038ef6 Yang Zhang
        VMSTATE_UINT64_V(last_update, RTCState, 3),
727 56038ef6 Yang Zhang
        VMSTATE_INT64_V(offset, RTCState, 3),
728 56038ef6 Yang Zhang
        VMSTATE_TIMER_V(update_timer, RTCState, 3),
729 00cf5774 Paolo Bonzini
        VMSTATE_UINT64_V(next_alarm_time, RTCState, 3),
730 6b075b8a Juan Quintela
        VMSTATE_END_OF_LIST()
731 6b075b8a Juan Quintela
    }
732 6b075b8a Juan Quintela
};
733 6b075b8a Juan Quintela
734 17604dac Jan Kiszka
static void rtc_notify_clock_reset(Notifier *notifier, void *data)
735 17604dac Jan Kiszka
{
736 17604dac Jan Kiszka
    RTCState *s = container_of(notifier, RTCState, clock_reset_notifier);
737 17604dac Jan Kiszka
    int64_t now = *(int64_t *)data;
738 17604dac Jan Kiszka
739 17604dac Jan Kiszka
    rtc_set_date_from_host(&s->dev);
740 c4c18e24 Yang Zhang
    periodic_timer_update(s, now);
741 56038ef6 Yang Zhang
    check_update_timer(s);
742 17604dac Jan Kiszka
#ifdef TARGET_I386
743 433acf0d Jan Kiszka
    if (s->lost_tick_policy == LOST_TICK_SLEW) {
744 17604dac Jan Kiszka
        rtc_coalesced_timer_update(s);
745 17604dac Jan Kiszka
    }
746 17604dac Jan Kiszka
#endif
747 17604dac Jan Kiszka
}
748 17604dac Jan Kiszka
749 da98c8eb Gerd Hoffmann
/* set CMOS shutdown status register (index 0xF) as S3_resume(0xFE)
750 da98c8eb Gerd Hoffmann
   BIOS will read it and start S3 resume at POST Entry */
751 da98c8eb Gerd Hoffmann
static void rtc_notify_suspend(Notifier *notifier, void *data)
752 da98c8eb Gerd Hoffmann
{
753 da98c8eb Gerd Hoffmann
    RTCState *s = container_of(notifier, RTCState, suspend_notifier);
754 da98c8eb Gerd Hoffmann
    rtc_set_memory(&s->dev, 0xF, 0xFE);
755 da98c8eb Gerd Hoffmann
}
756 da98c8eb Gerd Hoffmann
757 eeb7c03c Gleb Natapov
static void rtc_reset(void *opaque)
758 eeb7c03c Gleb Natapov
{
759 eeb7c03c Gleb Natapov
    RTCState *s = opaque;
760 eeb7c03c Gleb Natapov
761 72716184 Anthony Liguori
    s->cmos_data[RTC_REG_B] &= ~(REG_B_PIE | REG_B_AIE | REG_B_SQWE);
762 72716184 Anthony Liguori
    s->cmos_data[RTC_REG_C] &= ~(REG_C_UF | REG_C_IRQF | REG_C_PF | REG_C_AF);
763 56038ef6 Yang Zhang
    check_update_timer(s);
764 eeb7c03c Gleb Natapov
765 72716184 Anthony Liguori
    qemu_irq_lower(s->irq);
766 eeb7c03c Gleb Natapov
767 eeb7c03c Gleb Natapov
#ifdef TARGET_I386
768 433acf0d Jan Kiszka
    if (s->lost_tick_policy == LOST_TICK_SLEW) {
769 433acf0d Jan Kiszka
        s->irq_coalesced = 0;
770 433acf0d Jan Kiszka
    }
771 eeb7c03c Gleb Natapov
#endif
772 eeb7c03c Gleb Natapov
}
773 eeb7c03c Gleb Natapov
774 b2c5009b Richard Henderson
static const MemoryRegionOps cmos_ops = {
775 0da8c842 Alexander Graf
    .read = cmos_ioport_read,
776 0da8c842 Alexander Graf
    .write = cmos_ioport_write,
777 0da8c842 Alexander Graf
    .impl = {
778 0da8c842 Alexander Graf
        .min_access_size = 1,
779 0da8c842 Alexander Graf
        .max_access_size = 1,
780 0da8c842 Alexander Graf
    },
781 0da8c842 Alexander Graf
    .endianness = DEVICE_LITTLE_ENDIAN,
782 b2c5009b Richard Henderson
};
783 b2c5009b Richard Henderson
784 57c9fafe Anthony Liguori
static void rtc_get_date(Object *obj, Visitor *v, void *opaque,
785 18297050 Anthony Liguori
                         const char *name, Error **errp)
786 18297050 Anthony Liguori
{
787 57c9fafe Anthony Liguori
    ISADevice *isa = ISA_DEVICE(obj);
788 18297050 Anthony Liguori
    RTCState *s = DO_UPCAST(RTCState, dev, isa);
789 e2826cf4 Paolo Bonzini
    struct tm current_tm;
790 18297050 Anthony Liguori
791 56038ef6 Yang Zhang
    rtc_update_time(s);
792 e2826cf4 Paolo Bonzini
    rtc_get_time(s, &current_tm);
793 18297050 Anthony Liguori
    visit_start_struct(v, NULL, "struct tm", name, 0, errp);
794 e2826cf4 Paolo Bonzini
    visit_type_int32(v, &current_tm.tm_year, "tm_year", errp);
795 e2826cf4 Paolo Bonzini
    visit_type_int32(v, &current_tm.tm_mon, "tm_mon", errp);
796 e2826cf4 Paolo Bonzini
    visit_type_int32(v, &current_tm.tm_mday, "tm_mday", errp);
797 e2826cf4 Paolo Bonzini
    visit_type_int32(v, &current_tm.tm_hour, "tm_hour", errp);
798 e2826cf4 Paolo Bonzini
    visit_type_int32(v, &current_tm.tm_min, "tm_min", errp);
799 e2826cf4 Paolo Bonzini
    visit_type_int32(v, &current_tm.tm_sec, "tm_sec", errp);
800 18297050 Anthony Liguori
    visit_end_struct(v, errp);
801 18297050 Anthony Liguori
}
802 18297050 Anthony Liguori
803 32e0c826 Gerd Hoffmann
static int rtc_initfn(ISADevice *dev)
804 dff38e7b bellard
{
805 32e0c826 Gerd Hoffmann
    RTCState *s = DO_UPCAST(RTCState, dev, dev);
806 32e0c826 Gerd Hoffmann
    int base = 0x70;
807 80cabfad bellard
808 80cabfad bellard
    s->cmos_data[RTC_REG_A] = 0x26;
809 80cabfad bellard
    s->cmos_data[RTC_REG_B] = 0x02;
810 80cabfad bellard
    s->cmos_data[RTC_REG_C] = 0x00;
811 80cabfad bellard
    s->cmos_data[RTC_REG_D] = 0x80;
812 80cabfad bellard
813 b8994faf Paolo Bonzini
    /* This is for historical reasons.  The default base year qdev property
814 b8994faf Paolo Bonzini
     * was set to 2000 for most machine types before the century byte was
815 b8994faf Paolo Bonzini
     * implemented.
816 b8994faf Paolo Bonzini
     *
817 b8994faf Paolo Bonzini
     * This if statement means that the century byte will be always 0
818 b8994faf Paolo Bonzini
     * (at least until 2079...) for base_year = 1980, but will be set
819 b8994faf Paolo Bonzini
     * correctly for base_year = 2000.
820 b8994faf Paolo Bonzini
     */
821 b8994faf Paolo Bonzini
    if (s->base_year == 2000) {
822 b8994faf Paolo Bonzini
        s->base_year = 0;
823 b8994faf Paolo Bonzini
    }
824 b8994faf Paolo Bonzini
825 1d914fa0 Isaku Yamahata
    rtc_set_date_from_host(dev);
826 ea55ffb3 ths
827 93b66569 aliguori
#ifdef TARGET_I386
828 433acf0d Jan Kiszka
    switch (s->lost_tick_policy) {
829 433acf0d Jan Kiszka
    case LOST_TICK_SLEW:
830 6875204c Jan Kiszka
        s->coalesced_timer =
831 74475455 Paolo Bonzini
            qemu_new_timer_ns(rtc_clock, rtc_coalesced_timer, s);
832 433acf0d Jan Kiszka
        break;
833 433acf0d Jan Kiszka
    case LOST_TICK_DISCARD:
834 433acf0d Jan Kiszka
        break;
835 433acf0d Jan Kiszka
    default:
836 433acf0d Jan Kiszka
        return -EINVAL;
837 433acf0d Jan Kiszka
    }
838 93b66569 aliguori
#endif
839 433acf0d Jan Kiszka
840 433acf0d Jan Kiszka
    s->periodic_timer = qemu_new_timer_ns(rtc_clock, rtc_periodic_timer, s);
841 56038ef6 Yang Zhang
    s->update_timer = qemu_new_timer_ns(rtc_clock, rtc_update_timer, s);
842 56038ef6 Yang Zhang
    check_update_timer(s);
843 dff38e7b bellard
844 17604dac Jan Kiszka
    s->clock_reset_notifier.notify = rtc_notify_clock_reset;
845 17604dac Jan Kiszka
    qemu_register_clock_reset_notifier(rtc_clock, &s->clock_reset_notifier);
846 17604dac Jan Kiszka
847 da98c8eb Gerd Hoffmann
    s->suspend_notifier.notify = rtc_notify_suspend;
848 da98c8eb Gerd Hoffmann
    qemu_register_suspend_notifier(&s->suspend_notifier);
849 da98c8eb Gerd Hoffmann
850 b2c5009b Richard Henderson
    memory_region_init_io(&s->io, &cmos_ops, s, "rtc", 2);
851 b2c5009b Richard Henderson
    isa_register_ioport(dev, &s->io, base);
852 dff38e7b bellard
853 56038ef6 Yang Zhang
    qdev_set_legacy_instance_id(&dev->qdev, base, 3);
854 a08d4367 Jan Kiszka
    qemu_register_reset(rtc_reset, s);
855 18297050 Anthony Liguori
856 57c9fafe Anthony Liguori
    object_property_add(OBJECT(s), "date", "struct tm",
857 57c9fafe Anthony Liguori
                        rtc_get_date, NULL, NULL, s, NULL);
858 18297050 Anthony Liguori
859 32e0c826 Gerd Hoffmann
    return 0;
860 32e0c826 Gerd Hoffmann
}
861 32e0c826 Gerd Hoffmann
862 48a18b3c Hervé Poussineau
ISADevice *rtc_init(ISABus *bus, int base_year, qemu_irq intercept_irq)
863 32e0c826 Gerd Hoffmann
{
864 32e0c826 Gerd Hoffmann
    ISADevice *dev;
865 7d932dfd Jan Kiszka
    RTCState *s;
866 eeb7c03c Gleb Natapov
867 48a18b3c Hervé Poussineau
    dev = isa_create(bus, "mc146818rtc");
868 7d932dfd Jan Kiszka
    s = DO_UPCAST(RTCState, dev, dev);
869 32e0c826 Gerd Hoffmann
    qdev_prop_set_int32(&dev->qdev, "base_year", base_year);
870 e23a1b33 Markus Armbruster
    qdev_init_nofail(&dev->qdev);
871 7d932dfd Jan Kiszka
    if (intercept_irq) {
872 7d932dfd Jan Kiszka
        s->irq = intercept_irq;
873 7d932dfd Jan Kiszka
    } else {
874 7d932dfd Jan Kiszka
        isa_init_irq(dev, &s->irq, RTC_ISA_IRQ);
875 7d932dfd Jan Kiszka
    }
876 1d914fa0 Isaku Yamahata
    return dev;
877 80cabfad bellard
}
878 80cabfad bellard
879 39bffca2 Anthony Liguori
static Property mc146818rtc_properties[] = {
880 39bffca2 Anthony Liguori
    DEFINE_PROP_INT32("base_year", RTCState, base_year, 1980),
881 39bffca2 Anthony Liguori
    DEFINE_PROP_LOSTTICKPOLICY("lost_tick_policy", RTCState,
882 39bffca2 Anthony Liguori
                               lost_tick_policy, LOST_TICK_DISCARD),
883 39bffca2 Anthony Liguori
    DEFINE_PROP_END_OF_LIST(),
884 39bffca2 Anthony Liguori
};
885 39bffca2 Anthony Liguori
886 8f04ee08 Anthony Liguori
static void rtc_class_initfn(ObjectClass *klass, void *data)
887 8f04ee08 Anthony Liguori
{
888 39bffca2 Anthony Liguori
    DeviceClass *dc = DEVICE_CLASS(klass);
889 8f04ee08 Anthony Liguori
    ISADeviceClass *ic = ISA_DEVICE_CLASS(klass);
890 8f04ee08 Anthony Liguori
    ic->init = rtc_initfn;
891 39bffca2 Anthony Liguori
    dc->no_user = 1;
892 39bffca2 Anthony Liguori
    dc->vmsd = &vmstate_rtc;
893 39bffca2 Anthony Liguori
    dc->props = mc146818rtc_properties;
894 8f04ee08 Anthony Liguori
}
895 8f04ee08 Anthony Liguori
896 39bffca2 Anthony Liguori
static TypeInfo mc146818rtc_info = {
897 39bffca2 Anthony Liguori
    .name          = "mc146818rtc",
898 39bffca2 Anthony Liguori
    .parent        = TYPE_ISA_DEVICE,
899 39bffca2 Anthony Liguori
    .instance_size = sizeof(RTCState),
900 39bffca2 Anthony Liguori
    .class_init    = rtc_class_initfn,
901 32e0c826 Gerd Hoffmann
};
902 32e0c826 Gerd Hoffmann
903 83f7d43a Andreas Färber
static void mc146818rtc_register_types(void)
904 100d9891 aurel32
{
905 39bffca2 Anthony Liguori
    type_register_static(&mc146818rtc_info);
906 100d9891 aurel32
}
907 83f7d43a Andreas Färber
908 83f7d43a Andreas Färber
type_init(mc146818rtc_register_types)