2017-04-17 10:31:08 +03:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
// $Id$
|
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// High Precision Event Timer emulation ported from Qemu
|
|
|
|
//
|
|
|
|
// Copyright (c) 2007 Alexander Graf
|
|
|
|
// Copyright (c) 2008 IBM Corporation
|
|
|
|
//
|
|
|
|
// Authors: Beth Kon <bkon@us.ibm.com>
|
|
|
|
//
|
2021-01-31 13:50:53 +03:00
|
|
|
// Copyright (C) 2017-2021 The Bochs Project
|
2017-04-17 10:31:08 +03:00
|
|
|
//
|
|
|
|
// This library is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU Lesser General Public
|
|
|
|
// License as published by the Free Software Foundation; either
|
|
|
|
// version 2 of the License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This library is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
// Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
|
|
// License along with this library; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
//
|
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
#ifndef BX_IODEV_HPET_H
|
|
|
|
#define BX_IODEV_HPET_H
|
|
|
|
|
|
|
|
#if BX_SUPPORT_PCI
|
|
|
|
|
|
|
|
#define HPET_MIN_TIMERS 3
|
|
|
|
#define HPET_MAX_TIMERS 32
|
|
|
|
|
|
|
|
#define HPET_CFG_ENABLE 0x001
|
|
|
|
#define HPET_CFG_LEGACY 0x002
|
|
|
|
|
|
|
|
#define HPET_ID 0x000
|
|
|
|
#define HPET_PERIOD 0x004
|
|
|
|
#define HPET_CFG 0x010
|
|
|
|
#define HPET_STATUS 0x020
|
|
|
|
#define HPET_COUNTER 0x0f0
|
|
|
|
#define HPET_TN_CFG 0x000
|
|
|
|
#define HPET_TN_CMP 0x008
|
|
|
|
#define HPET_TN_ROUTE 0x010
|
|
|
|
#define HPET_CFG_WRITE_MASK 0x3
|
|
|
|
|
|
|
|
#define HPET_TN_TYPE_LEVEL 0x002
|
|
|
|
#define HPET_TN_ENABLE 0x004
|
|
|
|
#define HPET_TN_PERIODIC 0x008
|
|
|
|
#define HPET_TN_PERIODIC_CAP 0x010
|
|
|
|
#define HPET_TN_SIZE_CAP 0x020
|
|
|
|
#define HPET_TN_SETVAL 0x040
|
|
|
|
#define HPET_TN_32BIT 0x100
|
|
|
|
#define HPET_TN_INT_ROUTE_MASK 0x3e00
|
|
|
|
#define HPET_TN_FSB_ENABLE 0x4000
|
|
|
|
#define HPET_TN_CFG_WRITE_MASK 0x7f4e
|
|
|
|
#define HPET_TN_INT_ROUTE_SHIFT 9
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
Bit8u tn;
|
2019-12-17 20:40:30 +03:00
|
|
|
int timer_id;
|
2017-04-17 10:31:08 +03:00
|
|
|
Bit64u config;
|
|
|
|
Bit64u cmp;
|
|
|
|
Bit64u fsb;
|
|
|
|
Bit64u period;
|
2019-12-17 20:40:30 +03:00
|
|
|
Bit64u last_checked;
|
2017-04-17 10:31:08 +03:00
|
|
|
} HPETTimer;
|
|
|
|
|
|
|
|
class bx_hpet_c : public bx_devmodel_c {
|
|
|
|
public:
|
|
|
|
bx_hpet_c();
|
|
|
|
virtual ~bx_hpet_c();
|
|
|
|
virtual void init();
|
|
|
|
virtual void reset(unsigned type);
|
|
|
|
virtual void register_state(void);
|
|
|
|
#if BX_DEBUGGER
|
|
|
|
virtual void debug_dump(int argc, char **argv);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
Bit32u read_aligned(bx_phy_address address);
|
2023-02-16 15:17:24 +03:00
|
|
|
void write_aligned(bx_phy_address address, Bit32u data, bool trailing_write);
|
2017-04-17 10:31:08 +03:00
|
|
|
|
|
|
|
private:
|
2023-11-05 16:35:50 +03:00
|
|
|
Bit32u hpet_in_legacy_mode(void) const { return s.config & HPET_CFG_LEGACY; }
|
|
|
|
Bit32u timer_int_route(const HPETTimer *timer) const
|
2017-04-17 10:31:08 +03:00
|
|
|
{
|
|
|
|
return (timer->config & HPET_TN_INT_ROUTE_MASK) >> HPET_TN_INT_ROUTE_SHIFT;
|
|
|
|
}
|
2023-11-05 16:35:50 +03:00
|
|
|
Bit32u timer_fsb_route(const HPETTimer *t) const { return t->config & HPET_TN_FSB_ENABLE; }
|
|
|
|
Bit32u hpet_enabled(void) const { return s.config & HPET_CFG_ENABLE; }
|
|
|
|
Bit32u timer_is_periodic(const HPETTimer *t) const { return t->config & HPET_TN_PERIODIC; }
|
|
|
|
Bit32u timer_enabled(const HPETTimer *t) const { return t->config & HPET_TN_ENABLE; }
|
2017-04-17 10:31:08 +03:00
|
|
|
Bit64u hpet_get_ticks(void);
|
2023-11-05 16:35:50 +03:00
|
|
|
Bit64u hpet_calculate_diff(const HPETTimer *t, Bit64u current) const;
|
2021-01-31 13:50:53 +03:00
|
|
|
void update_irq(HPETTimer *timer, bool set);
|
2017-04-17 10:31:08 +03:00
|
|
|
void hpet_set_timer(HPETTimer *t);
|
|
|
|
void hpet_del_timer(HPETTimer *t);
|
|
|
|
|
|
|
|
static void timer_handler(void *);
|
|
|
|
void hpet_timer(void);
|
|
|
|
|
|
|
|
struct {
|
|
|
|
Bit8u num_timers;
|
2019-12-17 20:40:30 +03:00
|
|
|
Bit64u hpet_reference_value;
|
|
|
|
Bit64u hpet_reference_time;
|
2017-04-17 10:31:08 +03:00
|
|
|
Bit64u capability;
|
|
|
|
Bit64u config;
|
|
|
|
Bit64u isr;
|
|
|
|
Bit64u hpet_counter;
|
|
|
|
HPETTimer timer[HPET_MAX_TIMERS];
|
|
|
|
} s;
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif
|