#include "bochs.h" #define BX_USE_VIRTUAL_TIMERS 1 #define BX_VIRTUAL_TIMERS_REALTIME 1 //Important constant #defines: #define USEC_PER_SECOND (1000000) // define a macro to convert floating point numbers into 64-bit integers. // In MSVC++ you can convert a 64-bit float into a 64-bit signed integer, // but it will not convert a 64-bit float into a 64-bit unsigned integer. // This macro works around that. #define F2I(x) ((Bit64u)(Bit64s) (x)) #define I2F(x) ((double)(Bit64s) (x)) //CONFIGURATION #defines: //MAINLINE Configuration (For realtime PIT): //How much faster than real time we can go: #define MAX_MULT (1.25) //Minimum number of emulated useconds per second. // Now calculated using BX_MIN_IPS, the minimum number of // instructions per second. #define MIN_USEC_PER_SECOND (((((Bit64u)USEC_PER_SECOND)*((Bit64u)BX_MIN_IPS))/((Bit64u)(bx_options.Oips->get())))+(Bit64u)1) //DEBUG configuration: //Debug with printf options. #define DEBUG_REALTIME_WITH_PRINTF 1 //Use to test execution at multiples of real time. #define TIME_DIVIDER (1) #define TIME_MULTIPLIER (1) #define TIME_HEADSTART (0) #define GET_VIRT_REALTIME64_USEC() (((bx_get_realtime64_usec()*(Bit64u)TIME_MULTIPLIER/(Bit64u)TIME_DIVIDER))) //Set up Logging. #define LOG_THIS bx_virt_timer. //A single instance. bx_virt_timer_c bx_virt_timer; //Generic MAX and MIN Functions #define BX_MAX(a,b) ( ((a)>(b))?(a):(b) ) #define BX_MIN(a,b) ( ((a)>(b))?(b):(a) ) //USEC_ALPHA is multiplier for the past. //USEC_ALPHA_B is 1-USEC_ALPHA, or multiplier for the present. #define USEC_ALPHA ((double)(.8)) #define USEC_ALPHA_B ((double)(((double)1)-USEC_ALPHA)) #define USEC_ALPHA2 ((double)(.5)) #define USEC_ALPHA2_B ((double)(((double)1)-USEC_ALPHA2)) #define ALPHA_LOWER(old,new) ((Bit64u)((old0); BX_ASSERT(!in_timer_handler); if(last_sequential_time >= current_timers_time) { periodic(1); last_sequential_time = current_timers_time; } return current_timers_time; } //Register a timer handler to go off after a given interval. //Register a timer handler to go off with a periodic interval. int bx_virt_timer_c::register_timer( void *this_ptr, bx_timer_handler_t handler, Bit32u useconds, bx_bool continuous, bx_bool active, const char *id) { if(!use_virtual_timers) { return bx_pc_system.register_timer(this_ptr, handler, useconds, continuous, active, id); } //We don't like starting with a zero period timer. BX_ASSERT((!active) || (useconds>0)); //Search for an unused timer. int i; for (i=0; i < numTimers; i++) { if (timer[i].inUse == 0 || i==numTimers) break; } // If we didn't find a free slot, increment the bound, numTimers. if (i==numTimers) numTimers++; // One new timer installed. BX_ASSERT(numTimers= 0); BX_ASSERT(timerID < BX_MAX_VIRTUAL_TIMERS); //No need to prevent doing this to unused timers. timer[timerID].inUse = 0; } void bx_virt_timer_c::start_timers(void) { if(!use_virtual_timers) { return bx_pc_system.start_timers(); } //FIXME } //activate a deactivated but registered timer. void bx_virt_timer_c::activate_timer( unsigned timer_index, Bit32u useconds, bx_bool continuous ) { if(!use_virtual_timers) { return bx_pc_system.activate_timer(timer_index, useconds, continuous); } BX_ASSERT(timer_index >= 0); BX_ASSERT(timer_index < BX_MAX_VIRTUAL_TIMERS); BX_ASSERT(timer[timer_index].inUse); BX_ASSERT(useconds>0); timer[timer_index].period=useconds; timer[timer_index].timeToFire = current_timers_time + (Bit64u)useconds; timer[timer_index].active=1; timer[timer_index].continuous=continuous; if(useconds < timers_next_event_time) { timers_next_event_time = useconds; next_event_time_update(); //FIXME } } //deactivate (but don't unregister) a currently registered timer. void bx_virt_timer_c::deactivate_timer( unsigned timer_index ) { if(!use_virtual_timers) { return bx_pc_system.deactivate_timer(timer_index); } BX_ASSERT(timer_index >= 0); BX_ASSERT(timer_index < BX_MAX_VIRTUAL_TIMERS); //No need to prevent doing this to unused/inactive timers. timer[timer_index].active = 0; } void bx_virt_timer_c::advance_virtual_time(Bit64u time_passed) { BX_ASSERT(time_passed <= virtual_next_event_time); current_virtual_time += time_passed; virtual_next_event_time -= time_passed; if(current_virtual_time > current_timers_time) { periodic(current_virtual_time - current_timers_time); } } //Called when next_event_time changes. void bx_virt_timer_c::next_event_time_update(void) { virtual_next_event_time = timers_next_event_time + current_timers_time - current_virtual_time; if(init_done) { bx_pc_system.deactivate_timer(system_timer_id); BX_ASSERT(virtual_next_event_time); bx_pc_system.activate_timer(system_timer_id, BX_MAX(1,TICKS_TO_USEC(virtual_next_event_time)), 0); } } void bx_virt_timer_c::init(void) { virtual_timers_realtime = BX_VIRTUAL_TIMERS_REALTIME; register_timer(this, nullTimer, NullTimerInterval, 1, 1, "Null Timer"); system_timer_id = bx_pc_system.register_timer(this, pc_system_timer_handler,virtual_next_event_time , 0, 1, "Virtual Timer"); //Real time variables: last_real_time=GET_VIRT_REALTIME64_USEC()+(Bit64u)TIME_HEADSTART*(Bit64u)USEC_PER_SECOND; total_real_usec=0; last_realtime_delta=0; //System time variables: last_usec; usec_per_second = USEC_PER_SECOND; stored_delta=0; last_system_usec=0; em_last_realtime=0; //Virtual timer variables: total_ticks=0; last_realtime_ticks=0; ticks_per_second = USEC_PER_SECOND; init_done = 1; } void bx_virt_timer_c::timer_handler(void) { if(!virtual_timers_realtime) { Bit64u temp_final_time = bx_pc_system.time_usec(); temp_final_time-=current_virtual_time; while(temp_final_time) { if((temp_final_time)>(virtual_next_event_time)) { temp_final_time-=virtual_next_event_time; advance_virtual_time(virtual_next_event_time); } else { advance_virtual_time(temp_final_time); temp_final_time-=temp_final_time; } } bx_pc_system.activate_timer(system_timer_id, (virtual_next_event_time>2)?(virtual_next_event_time-2):1, 0); return; } Bit64u usec_delta = bx_pc_system.time_usec()-last_usec; if (usec_delta) { #if BX_HAVE_REALTIME_USEC Bit64u ticks_delta = 0; Bit64u real_time_delta = GET_VIRT_REALTIME64_USEC() - last_real_time; Bit64u real_time_total = real_time_delta + total_real_usec; Bit64u system_time_delta = (Bit64u)usec_delta + (Bit64u)stored_delta; if(real_time_delta) { last_realtime_delta = real_time_delta; last_realtime_ticks = total_ticks; } ticks_per_second = USEC_PER_SECOND; //Start out with the number of ticks we would like // to have to line up with real time. ticks_delta = real_time_total - total_ticks; if(real_time_total < total_ticks) { //This slows us down if we're already ahead. // probably only an issue on startup, but it solves some problems. ticks_delta = 0; } if(ticks_delta + total_ticks - last_realtime_ticks > (F2I(MAX_MULT * I2F(last_realtime_delta)))) { //This keeps us from going too fast in relation to real time. #if 0 ticks_delta = (F2I(MAX_MULT * I2F(last_realtime_delta))) + last_realtime_ticks - total_ticks; #endif ticks_per_second = F2I(MAX_MULT * I2F(USEC_PER_SECOND)); } if(ticks_delta > system_time_delta * USEC_PER_SECOND / MIN_USEC_PER_SECOND) { //This keeps us from having too few instructions between ticks. ticks_delta = system_time_delta * USEC_PER_SECOND / MIN_USEC_PER_SECOND; } if(ticks_delta > virtual_next_event_time) { //This keeps us from missing ticks. ticks_delta = virtual_next_event_time; } if(ticks_delta) { # if DEBUG_REALTIME_WITH_PRINTF //Every second print some info. if(((last_real_time + real_time_delta) / USEC_PER_SECOND) > (last_real_time / USEC_PER_SECOND)) { Bit64u temp1, temp2, temp3, temp4; temp1 = (Bit64u) total_real_usec; temp2 = (total_real_usec); temp3 = (Bit64u)total_ticks; temp4 = (Bit64u)((total_real_usec) - total_ticks); printf("useconds: %llu, ",temp1); printf("expect ticks: %llu, ",temp2); printf("ticks: %llu, ",temp3); printf("diff: %llu\n",temp4); } # endif last_real_time += real_time_delta; total_real_usec += real_time_delta; last_system_usec += system_time_delta; stored_delta = 0; total_ticks += ticks_delta; } else { stored_delta = system_time_delta; } Bit64u a,b; a=(usec_per_second); if(real_time_delta) { //FIXME Bit64u em_realtime_delta = last_system_usec + stored_delta - em_last_realtime; b=((Bit64u)USEC_PER_SECOND * em_realtime_delta / real_time_delta); em_last_realtime = last_system_usec + stored_delta; } else { b=a; } usec_per_second = ALPHA_LOWER(a,b); #else BX_ASSERT(0); #endif advance_virtual_time(ticks_delta); } last_usec=last_usec + usec_delta; bx_pc_system.deactivate_timer(system_timer_id); BX_ASSERT(virtual_next_event_time); bx_pc_system.activate_timer(system_timer_id, BX_MAX(1,TICKS_TO_USEC(virtual_next_event_time)), 0); } void bx_virt_timer_c::pc_system_timer_handler(void* this_ptr) { ((bx_virt_timer_c *)this_ptr)->timer_handler(); }