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 50*3f6e6a13SDmitry Osipenko /* Starting to run with/setting counter to "0" won't re-load counter 51*3f6e6a13SDmitry Osipenko * immediately, but after a one period. */ 52*3f6e6a13SDmitry Osipenko #define PTIMER_POLICY_NO_IMMEDIATE_RELOAD (1 << 3) 53*3f6e6a13SDmitry Osipenko 5449d4d9b6SPaolo Bonzini /* ptimer.c */ 5549d4d9b6SPaolo Bonzini typedef struct ptimer_state ptimer_state; 5649d4d9b6SPaolo Bonzini typedef void (*ptimer_cb)(void *opaque); 5749d4d9b6SPaolo Bonzini 58e7ea81c3SDmitry Osipenko ptimer_state *ptimer_init(QEMUBH *bh, uint8_t policy_mask); 5949d4d9b6SPaolo Bonzini void ptimer_set_period(ptimer_state *s, int64_t period); 6049d4d9b6SPaolo Bonzini void ptimer_set_freq(ptimer_state *s, uint32_t freq); 61578c4b2fSDmitry Osipenko uint64_t ptimer_get_limit(ptimer_state *s); 6249d4d9b6SPaolo Bonzini void ptimer_set_limit(ptimer_state *s, uint64_t limit, int reload); 6349d4d9b6SPaolo Bonzini uint64_t ptimer_get_count(ptimer_state *s); 6449d4d9b6SPaolo Bonzini void ptimer_set_count(ptimer_state *s, uint64_t count); 6549d4d9b6SPaolo Bonzini void ptimer_run(ptimer_state *s, int oneshot); 6649d4d9b6SPaolo Bonzini void ptimer_stop(ptimer_state *s); 6749d4d9b6SPaolo Bonzini 68701a8f76SPaolo Bonzini extern const VMStateDescription vmstate_ptimer; 69701a8f76SPaolo Bonzini 7020bcf73fSPeter Maydell #define VMSTATE_PTIMER(_field, _state) \ 7120bcf73fSPeter Maydell VMSTATE_STRUCT_POINTER_V(_field, _state, 1, vmstate_ptimer, ptimer_state) 72701a8f76SPaolo Bonzini 73a1f05e79SPeter Maydell #define VMSTATE_PTIMER_ARRAY(_f, _s, _n) \ 74a1f05e79SPeter Maydell VMSTATE_ARRAY_OF_POINTER_TO_STRUCT(_f, _s, _n, 0, \ 75a1f05e79SPeter Maydell vmstate_ptimer, ptimer_state) 76a1f05e79SPeter Maydell 7749d4d9b6SPaolo Bonzini #endif 78