mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-31 12:52:44 +00:00 
			
		
		
		
	 736092a087
			
		
	
	
		736092a087
		
	
	
	
	
		
			
			There's a ton of work that would need to be done before we could spin up on another architecture, but let's at least try to separate things out a bit.
		
			
				
	
	
		
			83 lines
		
	
	
	
		
			1.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			83 lines
		
	
	
	
		
			1.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "i8253.h"
 | |
| #include "IO.h"
 | |
| #include "PIC.h"
 | |
| #include "Scheduler.h"
 | |
| #include <Kernel/Arch/i386/CPU.h>
 | |
| 
 | |
| #define IRQ_TIMER 0
 | |
| 
 | |
| extern "C" void timer_interrupt_entry();
 | |
| extern "C" void timer_interrupt_handler(RegisterDump&);
 | |
| 
 | |
| asm(
 | |
|     ".globl timer_interrupt_entry \n"
 | |
|     "timer_interrupt_entry: \n"
 | |
|     "    pusha\n"
 | |
|     "    pushw %ds\n"
 | |
|     "    pushw %es\n"
 | |
|     "    pushw %fs\n"
 | |
|     "    pushw %gs\n"
 | |
|     "    pushw %ss\n"
 | |
|     "    pushw %ss\n"
 | |
|     "    pushw %ss\n"
 | |
|     "    pushw %ss\n"
 | |
|     "    pushw %ss\n"
 | |
|     "    popw %ds\n"
 | |
|     "    popw %es\n"
 | |
|     "    popw %fs\n"
 | |
|     "    popw %gs\n"
 | |
|     "    mov %esp, %eax\n"
 | |
|     "    call timer_interrupt_handler\n"
 | |
|     "    popw %gs\n"
 | |
|     "    popw %gs\n"
 | |
|     "    popw %fs\n"
 | |
|     "    popw %es\n"
 | |
|     "    popw %ds\n"
 | |
|     "    popa\n"
 | |
|     "    iret\n");
 | |
| 
 | |
| static dword s_ticks_this_second;
 | |
| static dword s_seconds_since_boot;
 | |
| 
 | |
| void timer_interrupt_handler(RegisterDump& regs)
 | |
| {
 | |
|     IRQHandlerScope scope(IRQ_TIMER);
 | |
|     if (++s_ticks_this_second >= TICKS_PER_SECOND) {
 | |
|         // FIXME: Synchronize with the RTC somehow to prevent drifting apart.
 | |
|         ++s_seconds_since_boot;
 | |
|         s_ticks_this_second = 0;
 | |
|     }
 | |
|     Scheduler::timer_tick(regs);
 | |
| }
 | |
| 
 | |
| namespace PIT {
 | |
| 
 | |
| dword ticks_this_second()
 | |
| {
 | |
|     return s_ticks_this_second;
 | |
| }
 | |
| 
 | |
| dword seconds_since_boot()
 | |
| {
 | |
|     return s_seconds_since_boot;
 | |
| }
 | |
| 
 | |
| void initialize()
 | |
| {
 | |
|     word timer_reload;
 | |
| 
 | |
|     IO::out8(PIT_CTL, TIMER0_SELECT | WRITE_WORD | MODE_SQUARE_WAVE);
 | |
| 
 | |
|     timer_reload = (BASE_FREQUENCY / TICKS_PER_SECOND);
 | |
| 
 | |
|     kprintf("PIT: %u Hz, square wave (%x)\n", TICKS_PER_SECOND, timer_reload);
 | |
| 
 | |
|     IO::out8(TIMER0_CTL, LSB(timer_reload));
 | |
|     IO::out8(TIMER0_CTL, MSB(timer_reload));
 | |
| 
 | |
|     register_interrupt_handler(IRQ_VECTOR_BASE + IRQ_TIMER, timer_interrupt_entry);
 | |
| 
 | |
|     PIC::enable(IRQ_TIMER);
 | |
| }
 | |
| 
 | |
| }
 |