apple2ix/src/timing.h
2018-11-07 06:49:16 -08:00

149 lines
4.2 KiB
C

/*
* Apple // emulator for *ix
*
* This software package is subject to the GNU General Public License
* version 3 or later (your choice) as published by the Free Software
* Foundation.
*
* Copyright 2013-2015 Aaron Culliney
*
*/
/*
* 65c02 CPU Timing Support.
*
*/
#ifndef _TIMING_H_
#define _TIMING_H_
#include "common.h"
#if !defined(NANOSECONDS_PER_SECOND)
#define NANOSECONDS_PER_SECOND 1000000000UL
#endif
// At a rate of ~1000x/sec, the emulator will (1) determine the number X of 65c02 instructions to execute and then
// executes them, (2) perform post-instruction-churn bookkeeping, and (3) sleep for a calculated interval.
//
// * The actual wall clock time to perform the emulated churn and bookkeeping is used to determine the sleep interval
//
// * The speaker provides feedback to the calculation of X (the number of instructions to churn)
#define EXECUTION_CHURN_RATE 1000UL
#define EXECUTION_PERIOD_NSECS 1000000UL // NANOSECONDS_PER_SECOND / EXECUTION_CHURN_RATE
// timing values cribbed from AppleWin ... reference: Sather's _Understanding the Apple IIe_
// 14318181.81...
#define _M14 (157500000.0 / 11.0)
#define _M14_INT (157500000 / 11)
// 65 cycles per 912 14M clocks = 1020484.45 ...
#define CLK_6502 ((_M14 * 65.0) / 912.0)
#define CLK_6502_INT ((unsigned int)CLK_6502)
// HBL & VBL constants
// UtAIIe:3-13 "There are exactly 17030 (65 x 262) 6502 cycles in every television scan of an American Apple"
#define CYCLES_HBL 25
#define CYCLES_VIS_BEGIN CYCLES_HBL
#define CYCLES_VIS 40
#define CYCLES_SCANLINE (CYCLES_HBL + CYCLES_VIS) // 65
#define SCANLINES_VBL 70
#define SCANLINES_MIX (20*8) // 160
#define SCANLINES_VIS (64*3) // 192
#define SCANLINES_VBL_BEGIN SCANLINES_VIS
#define SCANLINES_FRAME (SCANLINES_VBL + SCANLINES_VIS) // 262
#define CYCLES_FRAME (CYCLES_SCANLINE * SCANLINES_FRAME) // 17030
#define CPU_SCALE_SLOWEST 0.25
#define CPU_SCALE_FASTEST_PIVOT 4.0
#define CPU_SCALE_FASTEST (CPU_SCALE_FASTEST_PIVOT + 0.0625)
#ifdef INTERFACE_CLASSIC
# define CPU_SCALE_STEP_DIV 0.01
# define CPU_SCALE_STEP 0.05
#endif
extern unsigned long cycles_count_total; // cumulative cycles count from machine reset
extern unsigned int cycles_video_frame; // unprocessed video frame cycles count
extern double cycles_persec_target; // CLK_6502 * current CPU scale
extern int cycles_speaker_feedback; // current -/+ speaker requested feedback
extern double cpu_scale_factor; // scale factor #1
extern double cpu_altscale_factor; // scale factor #2
extern bool is_fullspeed; // emulation in full native speed?
extern bool alt_speed_enabled;
extern READONLY pthread_t cpu_thread_id;
/*
* calculate the difference between two timespec structures
*/
struct timespec timespec_diff(struct timespec start, struct timespec end, bool *negative);
struct timespec timespec_add(struct timespec start, unsigned long nsecs);
/*
* True if current thread is the CPU thread.
*/
bool timing_isCPUThread(void);
/*
* start CPU thread
*/
void timing_startCPU(void);
/*
* stop CPU thread
*/
void timing_stopCPU(void);
/*
* toggles CPU speed between configured values
*/
void timing_toggleCPUSpeed(void);
/*
* check whether automatic adjusting of CPU speed is configured.
*/
bool timing_shouldAutoAdjustSpeed(void);
/*
* initialize timing
*/
void timing_initialize(void);
/*
* Pause timing/CPU thread.
*
* This may block for a short amount of time to grab the appropriate mutex. CPU thread is blocked upon function return,
* until call to cpu_resume() is made.
*/
void cpu_pause(void);
/*
* Resume timing/CPU thread
*/
void cpu_resume(void);
/*
* Is the CPU paused?
*/
bool cpu_isPaused(void);
// ----------------------------------------------------------------------------
// Video frame and IRQ fine-grained timing.
/*
* Checkpoints current cycle count and updates total (for timing-dependent I/O)
*/
void timing_checkpointCycles(void) CALL_ON_CPU_THREAD;
// ----------------------------------------------------------------------------
// save/restore state
bool timing_saveState(StateHelper_s *helper);
bool timing_loadState(StateHelper_s *helper);
#endif // whole file