2015-05-24 06:55:12 +02:00
|
|
|
// Copyright 2008 Dolphin Emulator Project
|
2015-05-18 01:08:10 +02:00
|
|
|
// Licensed under GPLv2+
|
2013-04-17 22:43:11 -04:00
|
|
|
// Refer to the license.txt file included.
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2014-02-10 13:54:46 -05:00
|
|
|
#pragma once
|
2008-12-08 04:46:09 +00:00
|
|
|
|
|
|
|
// This is a system to schedule events into the emulated machine's future. Time is measured
|
|
|
|
// in main CPU clock cycles.
|
|
|
|
|
|
|
|
// To schedule an event, you first have to register its type. This is where you pass in the
|
|
|
|
// callback. You then schedule events using the type id you get back.
|
|
|
|
|
|
|
|
// See HW/SystemTimers.cpp for the main part of Dolphin's usage of this scheduler.
|
|
|
|
|
|
|
|
// The int cyclesLate that the callbacks get is how many cycles late it was.
|
|
|
|
// So to schedule a new event on a regular basis:
|
|
|
|
// inside callback:
|
|
|
|
// ScheduleEvent(periodInCycles - cyclesLate, callback, "whatever")
|
|
|
|
|
|
|
|
#include <string>
|
2014-09-07 20:06:58 -05:00
|
|
|
#include "Common/CommonTypes.h"
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2014-07-27 13:37:09 -04:00
|
|
|
class PointerWrap;
|
|
|
|
|
2008-12-08 04:46:09 +00:00
|
|
|
namespace CoreTiming
|
|
|
|
{
|
2016-03-24 04:04:18 +13:00
|
|
|
// These really shouldn't be global, but jit64 accesses them directly
|
2017-04-05 10:13:30 +01:00
|
|
|
struct Globals
|
|
|
|
{
|
|
|
|
s64 global_timer;
|
|
|
|
u64 fake_TB_start_value;
|
|
|
|
u64 fake_TB_start_ticks;
|
|
|
|
int slice_length;
|
|
|
|
float last_OC_factor_inverted;
|
|
|
|
};
|
|
|
|
extern Globals g;
|
2014-09-04 21:50:55 -07:00
|
|
|
|
2016-09-01 11:39:05 +00:00
|
|
|
// CoreTiming begins at the boundary of timing slice -1. An initial call to Advance() is
|
|
|
|
// required to end slice -1 and start slice 0 before the first cycle of code is executed.
|
2008-12-08 04:46:09 +00:00
|
|
|
void Init();
|
|
|
|
void Shutdown();
|
|
|
|
|
2016-04-10 02:06:09 +12:00
|
|
|
typedef void (*TimedCallback)(u64 userdata, s64 cyclesLate);
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
// This should only be called from the CPU thread, if you are calling it any other thread, you are
|
|
|
|
// doing something evil
|
2008-12-08 04:46:09 +00:00
|
|
|
u64 GetTicks();
|
|
|
|
u64 GetIdleTicks();
|
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
void DoState(PointerWrap& p);
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2016-09-01 10:54:18 +00:00
|
|
|
struct EventType;
|
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
// Returns the event_type identifier. if name is not unique, an existing event_type will be
|
|
|
|
// discarded.
|
2016-09-01 10:54:18 +00:00
|
|
|
EventType* RegisterEvent(const std::string& name, TimedCallback callback);
|
2008-12-08 04:46:09 +00:00
|
|
|
void UnregisterAllEvents();
|
|
|
|
|
2016-07-07 15:52:08 +02:00
|
|
|
enum class FromThread
|
|
|
|
{
|
|
|
|
CPU,
|
|
|
|
NON_CPU,
|
|
|
|
// Don't use ANY unless you're sure you need to call from
|
|
|
|
// both the CPU thread and at least one other thread
|
|
|
|
ANY
|
|
|
|
};
|
|
|
|
|
2015-02-17 14:57:36 +01:00
|
|
|
// userdata MAY NOT CONTAIN POINTERS. userdata might get written and reloaded from savestates.
|
2016-09-01 11:39:05 +00:00
|
|
|
// After the first Advance, the slice lengths and the downcount will be reduced whenever an event
|
|
|
|
// is scheduled earlier than the current values (when scheduled from the CPU Thread only).
|
|
|
|
// Scheduling from a callback will not update the downcount until the Advance() completes.
|
2016-09-01 10:54:18 +00:00
|
|
|
void ScheduleEvent(s64 cycles_into_future, EventType* event_type, u64 userdata = 0,
|
2016-07-07 15:52:08 +02:00
|
|
|
FromThread from = FromThread::CPU);
|
2008-12-08 04:46:09 +00:00
|
|
|
|
|
|
|
// We only permit one event of each type in the queue at a time.
|
2016-09-01 10:54:18 +00:00
|
|
|
void RemoveEvent(EventType* event_type);
|
|
|
|
void RemoveAllEvents(EventType* event_type);
|
2016-09-01 11:39:05 +00:00
|
|
|
|
|
|
|
// Advance must be called at the beginning of dispatcher loops, not the end. Advance() ends
|
|
|
|
// the previous timing slice and begins the next one, you must Advance from the previous
|
|
|
|
// slice to the current one before executing any cycles. CoreTiming starts in slice -1 so an
|
|
|
|
// Advance() is required to initialize the slice length before the first cycle of emulated
|
|
|
|
// instructions is executed.
|
|
|
|
// NOTE: Advance updates the PowerPC downcount and performs a PPC external exception check.
|
2008-12-08 04:46:09 +00:00
|
|
|
void Advance();
|
2010-06-24 13:28:54 +00:00
|
|
|
void MoveEvents();
|
2008-12-08 04:46:09 +00:00
|
|
|
|
|
|
|
// Pretend that the main CPU has executed enough cycles to reach the next event.
|
|
|
|
void Idle();
|
|
|
|
|
|
|
|
// Clear all pending events. This should ONLY be done on exit or state load.
|
|
|
|
void ClearPendingEvents();
|
|
|
|
|
|
|
|
void LogPendingEvents();
|
|
|
|
|
|
|
|
std::string GetScheduledEventsSummary();
|
|
|
|
|
2017-02-08 16:22:27 +01:00
|
|
|
void AdjustEventQueueTimes(u32 new_ppc_clock, u32 old_ppc_clock);
|
|
|
|
|
2011-01-19 08:13:15 +00:00
|
|
|
u32 GetFakeDecStartValue();
|
|
|
|
void SetFakeDecStartValue(u32 val);
|
|
|
|
u64 GetFakeDecStartTicks();
|
|
|
|
void SetFakeDecStartTicks(u64 val);
|
|
|
|
u64 GetFakeTBStartValue();
|
|
|
|
void SetFakeTBStartValue(u64 val);
|
|
|
|
u64 GetFakeTBStartTicks();
|
|
|
|
void SetFakeTBStartTicks(u64 val);
|
|
|
|
|
2016-03-23 12:23:17 +13:00
|
|
|
void ForceExceptionCheck(s64 cycles);
|
2013-03-25 00:47:44 +11:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
} // end of namespace
|