xref: /qemu/include/hw/ptimer.h (revision 5580ea4576b60a4fa615c85e254fab1401149b45)
149d4d9b6SPaolo Bonzini /*
249d4d9b6SPaolo Bonzini  * General purpose implementation of a simple periodic countdown timer.
349d4d9b6SPaolo Bonzini  *
449d4d9b6SPaolo Bonzini  * Copyright (c) 2007 CodeSourcery.
549d4d9b6SPaolo Bonzini  *
649d4d9b6SPaolo Bonzini  * This code is licensed under the GNU LGPL.
749d4d9b6SPaolo Bonzini  */
849d4d9b6SPaolo Bonzini #ifndef PTIMER_H
949d4d9b6SPaolo Bonzini #define PTIMER_H
1049d4d9b6SPaolo Bonzini 
1149d4d9b6SPaolo Bonzini #include "qemu-common.h"
121de7afc9SPaolo Bonzini #include "qemu/timer.h"
13caf71f86SPaolo Bonzini #include "migration/vmstate.h"
1449d4d9b6SPaolo Bonzini 
15e7ea81c3SDmitry Osipenko /* The default ptimer policy retains backward compatibility with the legacy
16e7ea81c3SDmitry Osipenko  * timers. Custom policies are adjusting the default one. Consider providing
17e7ea81c3SDmitry Osipenko  * a correct policy for your timer.
18e7ea81c3SDmitry Osipenko  *
19e7ea81c3SDmitry Osipenko  * The rough edges of the default policy:
20e7ea81c3SDmitry Osipenko  *  - Starting to run with a period = 0 emits error message and stops the
21e7ea81c3SDmitry Osipenko  *    timer without a trigger.
22e7ea81c3SDmitry Osipenko  *
23e7ea81c3SDmitry Osipenko  *  - Setting period to 0 of the running timer emits error message and
24e7ea81c3SDmitry Osipenko  *    stops the timer without a trigger.
25e7ea81c3SDmitry Osipenko  *
26e7ea81c3SDmitry Osipenko  *  - Starting to run with counter = 0 or setting it to "0" while timer
27e7ea81c3SDmitry Osipenko  *    is running causes a trigger and reloads counter with a limit value.
28e7ea81c3SDmitry Osipenko  *    If limit = 0, ptimer emits error message and stops the timer.
29e7ea81c3SDmitry Osipenko  *
30e7ea81c3SDmitry Osipenko  *  - Counter value of the running timer is one less than the actual value.
31e7ea81c3SDmitry Osipenko  *
32e7ea81c3SDmitry Osipenko  *  - Changing period/frequency of the running timer loses time elapsed
33e7ea81c3SDmitry Osipenko  *    since the last period, effectively restarting the timer with a
34e7ea81c3SDmitry Osipenko  *    counter = counter value at the moment of change (.i.e. one less).
35e7ea81c3SDmitry Osipenko  */
36e7ea81c3SDmitry Osipenko #define PTIMER_POLICY_DEFAULT               0
37e7ea81c3SDmitry Osipenko 
382b5c0322SDmitry Osipenko /* Periodic timer counter stays with "0" for a one period before wrapping
392b5c0322SDmitry Osipenko  * around.  */
402b5c0322SDmitry Osipenko #define PTIMER_POLICY_WRAP_AFTER_ONE_PERIOD (1 << 0)
412b5c0322SDmitry Osipenko 
42ef0a9984SDmitry Osipenko /* Running periodic timer that has counter = limit = 0 would continuously
43ef0a9984SDmitry Osipenko  * re-trigger every period.  */
44ef0a9984SDmitry Osipenko #define PTIMER_POLICY_CONTINUOUS_TRIGGER    (1 << 1)
45ef0a9984SDmitry Osipenko 
4622471b8aSDmitry Osipenko /* Starting to run with/setting counter to "0" won't trigger immediately,
4722471b8aSDmitry Osipenko  * but after a one period for both oneshot and periodic modes.  */
4822471b8aSDmitry Osipenko #define PTIMER_POLICY_NO_IMMEDIATE_TRIGGER  (1 << 2)
4922471b8aSDmitry Osipenko 
503f6e6a13SDmitry Osipenko /* Starting to run with/setting counter to "0" won't re-load counter
513f6e6a13SDmitry Osipenko  * immediately, but after a one period.  */
523f6e6a13SDmitry Osipenko #define PTIMER_POLICY_NO_IMMEDIATE_RELOAD   (1 << 3)
533f6e6a13SDmitry Osipenko 
54*5580ea45SDmitry Osipenko /* Make counter value of the running timer represent the actual value and
55*5580ea45SDmitry Osipenko  * not the one less.  */
56*5580ea45SDmitry Osipenko #define PTIMER_POLICY_NO_COUNTER_ROUND_DOWN (1 << 4)
57*5580ea45SDmitry Osipenko 
5849d4d9b6SPaolo Bonzini /* ptimer.c */
5949d4d9b6SPaolo Bonzini typedef struct ptimer_state ptimer_state;
6049d4d9b6SPaolo Bonzini typedef void (*ptimer_cb)(void *opaque);
6149d4d9b6SPaolo Bonzini 
62e7ea81c3SDmitry Osipenko ptimer_state *ptimer_init(QEMUBH *bh, uint8_t policy_mask);
6349d4d9b6SPaolo Bonzini void ptimer_set_period(ptimer_state *s, int64_t period);
6449d4d9b6SPaolo Bonzini void ptimer_set_freq(ptimer_state *s, uint32_t freq);
65578c4b2fSDmitry Osipenko uint64_t ptimer_get_limit(ptimer_state *s);
6649d4d9b6SPaolo Bonzini void ptimer_set_limit(ptimer_state *s, uint64_t limit, int reload);
6749d4d9b6SPaolo Bonzini uint64_t ptimer_get_count(ptimer_state *s);
6849d4d9b6SPaolo Bonzini void ptimer_set_count(ptimer_state *s, uint64_t count);
6949d4d9b6SPaolo Bonzini void ptimer_run(ptimer_state *s, int oneshot);
7049d4d9b6SPaolo Bonzini void ptimer_stop(ptimer_state *s);
7149d4d9b6SPaolo Bonzini 
72701a8f76SPaolo Bonzini extern const VMStateDescription vmstate_ptimer;
73701a8f76SPaolo Bonzini 
7420bcf73fSPeter Maydell #define VMSTATE_PTIMER(_field, _state) \
7520bcf73fSPeter Maydell     VMSTATE_STRUCT_POINTER_V(_field, _state, 1, vmstate_ptimer, ptimer_state)
76701a8f76SPaolo Bonzini 
77a1f05e79SPeter Maydell #define VMSTATE_PTIMER_ARRAY(_f, _s, _n)                                \
78a1f05e79SPeter Maydell     VMSTATE_ARRAY_OF_POINTER_TO_STRUCT(_f, _s, _n, 0,                   \
79a1f05e79SPeter Maydell                                        vmstate_ptimer, ptimer_state)
80a1f05e79SPeter Maydell 
8149d4d9b6SPaolo Bonzini #endif
82