Merge into main the new state of the operating system/kernel #1
11
Makefile
11
Makefile
@ -5,7 +5,7 @@ CC = ${HOME}/opt/cross/bin/i686-elf-gcc
|
|||||||
CPP = ${HOME}/opt/cross/bin/i686-elf-g++
|
CPP = ${HOME}/opt/cross/bin/i686-elf-g++
|
||||||
CFLAGS = -ffreestanding -O2 -Wall -Wextra
|
CFLAGS = -ffreestanding -O2 -Wall -Wextra
|
||||||
|
|
||||||
OFILES = $(BUILD_DIR)/boot.o $(BUILD_DIR)/kterm.o $(BUILD_DIR)/kernel.o $(BUILD_DIR)/pit.o $(BUILD_DIR)/keyboard.o $(BUILD_DIR)/PhysicalMemoryManager.o $(BUILD_DIR)/io.o $(BUILD_DIR)/PageDirectory.o $(BUILD_DIR)/gdtc.o $(BUILD_DIR)/idt.o $(BUILD_DIR)/pic.o $(BUILD_DIR)/string.o
|
OFILES =$(BUILD_DIR)/boot.o $(BUILD_DIR)/kterm.o $(BUILD_DIR)/kernel.o $(BUILD_DIR)/pit.o $(BUILD_DIR)/time.o $(BUILD_DIR)/keyboard.o $(BUILD_DIR)/PhysicalMemoryManager.o $(BUILD_DIR)/io.o $(BUILD_DIR)/PageDirectory.o $(BUILD_DIR)/gdtc.o $(BUILD_DIR)/idt.o $(BUILD_DIR)/pic.o $(BUILD_DIR)/sv-terminal.o $(BUILD_DIR)/string.o $(BUILD_DIR)/PageFrameAllocator.o
|
||||||
|
|
||||||
SRC_DIR = src
|
SRC_DIR = src
|
||||||
BUILD_DIR = build
|
BUILD_DIR = build
|
||||||
@ -33,7 +33,7 @@ iso: clean_iso clean build
|
|||||||
cp src/grub.cfg root/boot/grub/grub.cfg
|
cp src/grub.cfg root/boot/grub/grub.cfg
|
||||||
grub-mkrescue -o build/barinkOS.iso root
|
grub-mkrescue -o build/barinkOS.iso root
|
||||||
|
|
||||||
test:
|
run: all
|
||||||
$(EMULATOR) -kernel $(BUILD_DIR)/myos.bin -serial stdio -vga std -display gtk -m 2G -cpu core2duo
|
$(EMULATOR) -kernel $(BUILD_DIR)/myos.bin -serial stdio -vga std -display gtk -m 2G -cpu core2duo
|
||||||
|
|
||||||
build_kernel: $(OBJ_LINK_LIST)
|
build_kernel: $(OBJ_LINK_LIST)
|
||||||
@ -90,3 +90,10 @@ $(BUILD_DIR)/pit.o:
|
|||||||
|
|
||||||
$(BUILD_DIR)/keyboard.o:
|
$(BUILD_DIR)/keyboard.o:
|
||||||
$(CPP) -c $(SRC_DIR)/kernel/keyboard/keyboard.cpp -o $(BUILD_DIR)/keyboard.o $(CFLAGS) -fno-exceptions -fno-rtti
|
$(CPP) -c $(SRC_DIR)/kernel/keyboard/keyboard.cpp -o $(BUILD_DIR)/keyboard.o $(CFLAGS) -fno-exceptions -fno-rtti
|
||||||
|
|
||||||
|
|
||||||
|
$(BUILD_DIR)/time.o:
|
||||||
|
$(CPP) -c $(SRC_DIR)/kernel/time.cpp -o $(BUILD_DIR)/time.o $(CFLAGS) -fno-exceptions -fno-rtti
|
||||||
|
|
||||||
|
$(BUILD_DIR)/sv-terminal.o:
|
||||||
|
$(CPP) -c $(SRC_DIR)/kernel/sv-terminal/superVisorTerminal.cpp -o $(BUILD_DIR)/sv-terminal.o $(CFLAGS) -fno-exceptions -fno-rtti
|
||||||
|
@ -10,66 +10,81 @@ void CheckMBT ( multiboot_info_t* mbt ){
|
|||||||
/* Set MBI to the addresss of the multiboot information structure*/
|
/* Set MBI to the addresss of the multiboot information structure*/
|
||||||
multiboot_info_t * mbi = (multiboot_info_t *) mbt;
|
multiboot_info_t * mbi = (multiboot_info_t *) mbt;
|
||||||
|
|
||||||
|
#ifdef __VERBOSE__
|
||||||
/* Print out the flags */
|
/* Print out the flags */
|
||||||
printf("flags = 0x%x\n", (unsigned) mbi->flags);
|
printf("flags = 0x%x\n", (unsigned) mbi->flags);
|
||||||
|
#endif
|
||||||
/* Are mem_* valid? */
|
/* Are mem_* valid? */
|
||||||
if ( CHECK_FLAG(mbi->flags,0)){
|
if ( CHECK_FLAG(mbi->flags,0)){
|
||||||
printf("mem_lower = %uKB, mem_upper = %uKB\n");
|
// Do nothing
|
||||||
}
|
}
|
||||||
|
|
||||||
/* is boot device valid ? */
|
/* is boot device valid ? */
|
||||||
if (CHECK_FLAG (mbi->flags, 1)){
|
if (CHECK_FLAG (mbi->flags, 1))
|
||||||
|
{
|
||||||
|
#ifdef __VERBOSE__
|
||||||
printf("boot_device = 0x0%x\n", (unsigned) mbi->boot_device);
|
printf("boot_device = 0x0%x\n", (unsigned) mbi->boot_device);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
/* is the command line passed? */
|
/* is the command line passed? */
|
||||||
if (CHECK_FLAG ( mbi->flags,2)){
|
if (CHECK_FLAG ( mbi->flags,2))
|
||||||
|
{
|
||||||
|
#ifdef __VERBOSE__
|
||||||
printf("cmdline = %s\n", (char *) mbi->cmdline);
|
printf("cmdline = %s\n", (char *) mbi->cmdline);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Are mods_* valid? */
|
/* Are mods_* valid? */
|
||||||
if(CHECK_FLAG ( mbi->flags, 3)){
|
if(CHECK_FLAG ( mbi->flags, 3)){
|
||||||
multiboot_module_t *mod;
|
multiboot_module_t *mod;
|
||||||
uint32_t i;
|
uint32_t i;
|
||||||
|
#ifdef __VERBOSE__
|
||||||
printf("mods count = %d, mods_addr = 0x%x\n", (int) mbi->mods_count, (int) mbi->mods_addr);
|
printf("mods count = %d, mods_addr = 0x%x\n", (int) mbi->mods_count, (int) mbi->mods_addr);
|
||||||
|
|
||||||
for(i = 0, mod = (multiboot_module_t *) mbi->mods_addr; i < mbi->mods_count; i++ , mod++){
|
for(i = 0, mod = (multiboot_module_t *) mbi->mods_addr; i < mbi->mods_count; i++ , mod++){
|
||||||
printf(" mod start = 0x%x, mod_end = 0x%x, cmdline = %s\n", (unsigned) mod->mod_start, (unsigned) mod->mod_end, (char*) mod->cmdline);
|
printf(" mod start = 0x%x, mod_end = 0x%x, cmdline = %s\n", (unsigned) mod->mod_start, (unsigned) mod->mod_end, (char*) mod->cmdline);
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Bits 4 and 5 are mutually exclusive! */
|
/* Bits 4 and 5 are mutually exclusive! */
|
||||||
if (CHECK_FLAG (mbi->flags, 4) && CHECK_FLAG(mbi->flags, 5)){
|
if (CHECK_FLAG (mbi->flags, 4) && CHECK_FLAG(mbi->flags, 5))
|
||||||
|
{
|
||||||
|
#ifdef __VERBOSE__
|
||||||
printf("Both bits 4 and 5 are set.\n");
|
printf("Both bits 4 and 5 are set.\n");
|
||||||
|
#endif
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Is the symbol table of a.out valid? */
|
/* Is the symbol table of a.out valid? */
|
||||||
if (CHECK_FLAG(mbi->flags, 4)){
|
if (CHECK_FLAG(mbi->flags, 4)){
|
||||||
multiboot_aout_symbol_table_t *multiboot_aout_sym = &(mbi->u.aout_sym);
|
multiboot_aout_symbol_table_t *multiboot_aout_sym = &(mbi->u.aout_sym);
|
||||||
|
#ifdef __VERBOSE__
|
||||||
printf( "multiboot_aout_symbol_table: tabsize = 0x%0x, strsize = 0x%x, addr = 0x%x\n",
|
printf( "multiboot_aout_symbol_table: tabsize = 0x%0x, strsize = 0x%x, addr = 0x%x\n",
|
||||||
(unsigned) multiboot_aout_sym->tabsize,
|
(unsigned) multiboot_aout_sym->tabsize,
|
||||||
(unsigned) multiboot_aout_sym->strsize,
|
(unsigned) multiboot_aout_sym->strsize,
|
||||||
(unsigned) multiboot_aout_sym->addr);
|
(unsigned) multiboot_aout_sym->addr);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Is the section header table of ELF valid? */
|
/* Is the section header table of ELF valid? */
|
||||||
if (CHECK_FLAG(mbi->flags, 5)){
|
if (CHECK_FLAG(mbi->flags, 5)){
|
||||||
multiboot_elf_section_header_table_t *multiboot_elf_sec = &(mbi->u.elf_sec);
|
multiboot_elf_section_header_table_t *multiboot_elf_sec = &(mbi->u.elf_sec);
|
||||||
|
#ifdef __VERBOSE__
|
||||||
printf("multiboot_elf_sec: num = %u, size = 0x%x, addr = 0x%x, shnd = 0x%x\n",
|
printf("multiboot_elf_sec: num = %u, size = 0x%x, addr = 0x%x, shnd = 0x%x\n",
|
||||||
|
|
||||||
(unsigned) multiboot_elf_sec->num, (unsigned) multiboot_elf_sec->size,
|
(unsigned) multiboot_elf_sec->num, (unsigned) multiboot_elf_sec->size,
|
||||||
(unsigned) multiboot_elf_sec->addr, (unsigned) multiboot_elf_sec->shndx);
|
(unsigned) multiboot_elf_sec->addr, (unsigned) multiboot_elf_sec->shndx);
|
||||||
|
#endif
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Draw diagonal blue line */
|
/* Draw diagonal blue line */
|
||||||
if (CHECK_FLAG (mbt->flags, 12)){
|
if (CHECK_FLAG (mbt->flags, 12)){
|
||||||
printf("Can draw!");
|
#ifdef __VERBOSE__
|
||||||
|
printf("Can draw!\n");
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,17 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
// Let's write an ATA PIO | ATA driver for now. Mostly to show that I can in theory interact with a
|
|
||||||
// storage device
|
|
||||||
|
|
||||||
// PRIMARY_ATA_BUS
|
|
||||||
// 0x1F0 through 0x1F7
|
|
||||||
|
|
||||||
// SECONDARY_ATA_BUS
|
|
||||||
// 0x170 through 0x177
|
|
||||||
|
|
||||||
#define DEVICE_CONTROL_REGISTER 0x3F6
|
|
||||||
#define DEVICE_CONTROL_ALTERNATE 0x376
|
|
||||||
|
|
||||||
|
|
||||||
// IRQ14 Primary bus interrupt
|
|
||||||
// IRQ15 Secondary bus interrupt
|
|
@ -29,8 +29,9 @@ void add_descriptor(int which , unsigned long base, unsigned long limit, unsigne
|
|||||||
|
|
||||||
void initGDT(){
|
void initGDT(){
|
||||||
|
|
||||||
|
#ifdef __VERBOSE__
|
||||||
printf("Init GDT!\n");
|
printf("Init GDT!\n");
|
||||||
|
#endif
|
||||||
// NULL segment
|
// NULL segment
|
||||||
add_descriptor(NULL_SEGMENT, 0,0,0,0);
|
add_descriptor(NULL_SEGMENT, 0,0,0,0);
|
||||||
|
|
||||||
|
@ -254,8 +254,10 @@ void init_idt(){
|
|||||||
idt_ptr.length = sizeof(IDT_entry) * 255;
|
idt_ptr.length = sizeof(IDT_entry) * 255;
|
||||||
idt_ptr.base = (uint32_t)&idt_table;
|
idt_ptr.base = (uint32_t)&idt_table;
|
||||||
|
|
||||||
|
#ifdef __VERBOSE__
|
||||||
printf("Init IDT\n");
|
printf("Init IDT\n");
|
||||||
|
#endif
|
||||||
|
|
||||||
// TODO: Set everything to zero first
|
// TODO: Set everything to zero first
|
||||||
|
|
||||||
set_id_entry(0, (uint32_t) irs0 , 0x08, 0x8F);
|
set_id_entry(0, (uint32_t) irs0 , 0x08, 0x8F);
|
||||||
|
@ -4,28 +4,45 @@
|
|||||||
extern "C" void kernel_main (void);
|
extern "C" void kernel_main (void);
|
||||||
|
|
||||||
extern "C" void early_main(unsigned long magic, unsigned long addr){
|
extern "C" void early_main(unsigned long magic, unsigned long addr){
|
||||||
/** initialize terminal interface */
|
/**
|
||||||
|
* Initialize terminal interface
|
||||||
|
* NOTE: This should be done later on , the magic value should be checked first.
|
||||||
|
*/
|
||||||
kterm_init();
|
kterm_init();
|
||||||
|
|
||||||
// Check Multiboot magic number
|
/**
|
||||||
|
* Check Multiboot magic number
|
||||||
|
* NOTE: Printf call should not be a thing this early on ...
|
||||||
|
*/
|
||||||
if (magic != MULTIBOOT_BOOTLOADER_MAGIC){
|
if (magic != MULTIBOOT_BOOTLOADER_MAGIC){
|
||||||
printf("Invalid magic number: 0x%x\n", magic);
|
printf("Invalid magic number: 0x%x\n", magic);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
multiboot_info_t* mbt = (multiboot_info_t*) addr;
|
/**
|
||||||
|
* Show a little banner for cuteness
|
||||||
|
*/
|
||||||
|
printf("|=== BarinkOS ===|\n");
|
||||||
|
|
||||||
/* Are mmap_* valid? */
|
|
||||||
if (CHECK_FLAG(mbt->flags, 6)){
|
/**
|
||||||
|
* Use the address given as an argument as the pointer
|
||||||
|
* to a Multiboot information structure.
|
||||||
|
*/
|
||||||
|
multiboot_info_t* mbt = (multiboot_info_t*) addr;
|
||||||
|
|
||||||
|
/*
|
||||||
|
If we got a memory map from our bootloader we
|
||||||
|
should be parsing it to find out the memory regions available.
|
||||||
|
*/
|
||||||
|
if (CHECK_FLAG(mbt->flags, 6))
|
||||||
|
{
|
||||||
|
printf("Preliminary results mmap scan:\n");
|
||||||
|
mapMultibootMemoryMap(mbt);
|
||||||
|
|
||||||
PhysicalMemoryManager_initialise( mbt->mmap_addr, GB2/* Seriously dangerous hardcoded memory value*/);
|
PhysicalMemoryManager_initialise( mbt->mmap_addr, GB2/* Seriously dangerous hardcoded memory value*/);
|
||||||
PhysicalMemoryManager_initialise_available_regions(mbt->mmap_addr, mbt->mmap_addr + mbt->mmap_length);
|
PhysicalMemoryManager_initialise_available_regions(mbt->mmap_addr, mbt->mmap_addr + mbt->mmap_length);
|
||||||
PhysicalMemoryManager_deinitialise_kernel();
|
PhysicalMemoryManager_deinitialise_kernel();
|
||||||
extern uint8_t* kernel_begin;
|
|
||||||
extern uint8_t* kernel_end;
|
|
||||||
|
|
||||||
printf("Kernel MemoryMap:\n");
|
|
||||||
printf("kernel: 0x%x - 0x%x\n", &kernel_begin , &kernel_end);
|
|
||||||
printf("Frames used: 0x%x blocks of 4 KiB\n", used_blocks);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
initGDT();
|
initGDT();
|
||||||
|
@ -7,30 +7,21 @@ extern "C"{
|
|||||||
|
|
||||||
#include "./bootloader/multiboot.h"
|
#include "./bootloader/multiboot.h"
|
||||||
#include "bootcheck.h"
|
#include "bootcheck.h"
|
||||||
#include "memory/PhysicalMemoryManager.h"
|
#include "memory/physical/PhysicalMemoryManager.h"
|
||||||
|
#include "memory/frames/PageFrameAllocator.h"
|
||||||
|
|
||||||
#include "gdt/gdtc.h"
|
#include "gdt/gdtc.h"
|
||||||
#include "idt/idt.h"
|
#include "idt/idt.h"
|
||||||
|
|
||||||
#include "keyboard/keyboard.h"
|
|
||||||
|
|
||||||
#include "pit.h"
|
#include "pit.h"
|
||||||
#include "io.h"
|
#include "io.h"
|
||||||
#include "time.h"
|
|
||||||
#include "cpu.h"
|
#include "cpu.h"
|
||||||
#include "serial.h"
|
#include "serial.h"
|
||||||
|
|
||||||
|
#include "time.h"
|
||||||
|
#include "sv-terminal/superVisorTerminal.h"
|
||||||
|
|
||||||
#define CHECK_FLAG(flags, bit) ((flags) & (1 <<(bit)))
|
#define CHECK_FLAG(flags, bit) ((flags) & (1 <<(bit)))
|
||||||
#define PANIC(message) { return; }
|
#define PANIC(message) {return;}
|
||||||
|
|
||||||
|
|
||||||
/* This needs to be moved! */
|
|
||||||
/**
|
|
||||||
* simple delay function
|
|
||||||
**/
|
|
||||||
void delay(int t){
|
|
||||||
volatile int i,j;
|
|
||||||
for(i=0;i<t;i++)
|
|
||||||
for(j=0;j<25000;j++)
|
|
||||||
asm("NOP");
|
|
||||||
}
|
|
||||||
|
@ -7,8 +7,9 @@ void pit_initialise()
|
|||||||
{
|
{
|
||||||
asm volatile("CLI");
|
asm volatile("CLI");
|
||||||
|
|
||||||
printf("init PIT!\n");
|
#ifdef __VERBOSE__
|
||||||
|
printf("Init PIT!\n");
|
||||||
|
#endif
|
||||||
// clear mask for IRQ 0
|
// clear mask for IRQ 0
|
||||||
uint8_t value = inb(0x21) & ~(1<< 0);
|
uint8_t value = inb(0x21) & ~(1<< 0);
|
||||||
outb(0x21, value);
|
outb(0x21, value);
|
||||||
|
@ -4,7 +4,11 @@
|
|||||||
#include "io.h"
|
#include "io.h"
|
||||||
#define PORT 0x3f8
|
#define PORT 0x3f8
|
||||||
static int init_serial() {
|
static int init_serial() {
|
||||||
printf("Init serial\n");
|
|
||||||
|
#ifdef __VERBOSE__
|
||||||
|
printf("Init Serial\n");
|
||||||
|
#endif
|
||||||
|
|
||||||
outb(PORT + 1, 0x00); // Disable all interrupts
|
outb(PORT + 1, 0x00); // Disable all interrupts
|
||||||
outb(PORT + 3, 0x80); // Enable DLAB (set baud rate divisor)
|
outb(PORT + 3, 0x80); // Enable DLAB (set baud rate divisor)
|
||||||
outb(PORT + 0, 0x03); // Set divisor to 3 (lo byte) 38400 baud
|
outb(PORT + 0, 0x03); // Set divisor to 3 (lo byte) 38400 baud
|
||||||
|
74
src/kernel/sv-terminal/superVisorTerminal.cpp
Normal file
74
src/kernel/sv-terminal/superVisorTerminal.cpp
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
#include "superVisorTerminal.h"
|
||||||
|
|
||||||
|
void startSuperVisorTerminal(){
|
||||||
|
while (true){
|
||||||
|
|
||||||
|
printf("SUPERVISOR:>$ " );
|
||||||
|
int characterCount = 0;
|
||||||
|
char command[10] = "";
|
||||||
|
|
||||||
|
// NOTE: lets just show a kernel prompt
|
||||||
|
uint8_t ScanCode = getKey();
|
||||||
|
while( ScanCode != 0x1C )
|
||||||
|
{
|
||||||
|
char character = getASCIIKey();
|
||||||
|
kterm_put(character );
|
||||||
|
// wHAT THE HELL
|
||||||
|
|
||||||
|
if( characterCount < 10 ){
|
||||||
|
command[characterCount] = character;
|
||||||
|
characterCount++;
|
||||||
|
}
|
||||||
|
|
||||||
|
ScanCode = getKey();
|
||||||
|
}
|
||||||
|
printf("\n");
|
||||||
|
KeyHandled();
|
||||||
|
|
||||||
|
|
||||||
|
if ( strncmp("DATE", command , characterCount ) == 0 )
|
||||||
|
{
|
||||||
|
read_rtc();
|
||||||
|
printf("======= Time & Date ==========\n");
|
||||||
|
printf(" - Date: %02d-%02d-%02d\n",day, month, year);
|
||||||
|
printf(" - Time: %02d:%02d:%02d\n" , hour, minute, second);
|
||||||
|
printf(" - Ticks: %09d\n", pit_tick);
|
||||||
|
}
|
||||||
|
else if( strncmp ("MEMORY" , command , characterCount) == 0 )
|
||||||
|
{
|
||||||
|
// Show memory layout
|
||||||
|
printf("========= Memory ==========\n");
|
||||||
|
printf("Kernel MemoryMap:\n");
|
||||||
|
printf("kernel: 0x%x - 0x%x\n", &kernel_begin , &kernel_end);
|
||||||
|
printf("Frames used: 0x%x blocks of 4 KiB\n", used_blocks);
|
||||||
|
//printf("\n\n");
|
||||||
|
//PrintPhysicalMemoryAllocation( );
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
else if(strncmp("TEST", command, characterCount) == 0)
|
||||||
|
{
|
||||||
|
// TEST #DE exception
|
||||||
|
asm volatile ("MOV $4, %AX ; MOV $0, %BX ; DIV %BX"); // IRS 0
|
||||||
|
}
|
||||||
|
else if (strncmp("VERSION", command , characterCount) == 0)
|
||||||
|
{
|
||||||
|
// Show version information
|
||||||
|
printf("========= Version ========\n");
|
||||||
|
printf("Kernel v%d\n", 0);
|
||||||
|
|
||||||
|
}
|
||||||
|
else if(strncmp("CLEAR", command, characterCount) == 0)
|
||||||
|
{
|
||||||
|
kterm_init();
|
||||||
|
printf("|=== BarinkOS ===|\n");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
printf("Unknown command\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
delay(1000);
|
||||||
|
}
|
||||||
|
}
|
8
src/kernel/sv-terminal/superVisorTerminal.h
Normal file
8
src/kernel/sv-terminal/superVisorTerminal.h
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
#pragma once
|
||||||
|
#include "../tty/kterm.h"
|
||||||
|
#include "../time.h"
|
||||||
|
#include "../pit.h"
|
||||||
|
#include "../keyboard/keyboard.h"
|
||||||
|
#include "../memory/physical/PhysicalMemoryManager.h"
|
||||||
|
|
||||||
|
void startSuperVisorTerminal();
|
111
src/kernel/time.cpp
Normal file
111
src/kernel/time.cpp
Normal file
@ -0,0 +1,111 @@
|
|||||||
|
#include "time.h"
|
||||||
|
|
||||||
|
// Set by ACPI table parsing code if possible
|
||||||
|
int century_register = 0x00;
|
||||||
|
unsigned char second;
|
||||||
|
unsigned char minute;
|
||||||
|
unsigned char hour;
|
||||||
|
unsigned char day;
|
||||||
|
unsigned char month;
|
||||||
|
unsigned int year;
|
||||||
|
|
||||||
|
|
||||||
|
int get_update_in_progress_flag() {
|
||||||
|
outb(cmos_address, 0x0A);
|
||||||
|
return (inb(cmos_data) & 0x80);
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned char get_RTC_register(int reg) {
|
||||||
|
outb(cmos_address, reg);
|
||||||
|
return inb(cmos_data);
|
||||||
|
}
|
||||||
|
|
||||||
|
void read_rtc() {
|
||||||
|
unsigned char century;
|
||||||
|
unsigned char last_second;
|
||||||
|
unsigned char last_minute;
|
||||||
|
unsigned char last_hour;
|
||||||
|
unsigned char last_day;
|
||||||
|
unsigned char last_month;
|
||||||
|
unsigned char last_year;
|
||||||
|
unsigned char last_century;
|
||||||
|
unsigned char registerB;
|
||||||
|
|
||||||
|
// Note: This uses the "read registers until you get the same values twice in a row" technique
|
||||||
|
// to avoid getting dodgy/inconsistent values due to RTC updates
|
||||||
|
|
||||||
|
while (get_update_in_progress_flag()); // Make sure an update isn't in progress
|
||||||
|
second = get_RTC_register(0x00);
|
||||||
|
minute = get_RTC_register(0x02);
|
||||||
|
hour = get_RTC_register(0x04);
|
||||||
|
day = get_RTC_register(0x07);
|
||||||
|
month = get_RTC_register(0x08);
|
||||||
|
year = get_RTC_register(0x09);
|
||||||
|
if(century_register != 0) {
|
||||||
|
century = get_RTC_register(century_register);
|
||||||
|
} else {
|
||||||
|
century = 21;
|
||||||
|
}
|
||||||
|
|
||||||
|
do {
|
||||||
|
last_second = second;
|
||||||
|
last_minute = minute;
|
||||||
|
last_hour = hour;
|
||||||
|
last_day = day;
|
||||||
|
last_month = month;
|
||||||
|
last_year = year;
|
||||||
|
last_century = century;
|
||||||
|
|
||||||
|
while (get_update_in_progress_flag()); // Make sure an update isn't in progress
|
||||||
|
second = get_RTC_register(0x00);
|
||||||
|
minute = get_RTC_register(0x02);
|
||||||
|
hour = get_RTC_register(0x04);
|
||||||
|
day = get_RTC_register(0x07);
|
||||||
|
month = get_RTC_register(0x08);
|
||||||
|
year = get_RTC_register(0x09);
|
||||||
|
if(century_register != 0) {
|
||||||
|
century = get_RTC_register(century_register);
|
||||||
|
}
|
||||||
|
} while( (last_second != second) || (last_minute != minute) || (last_hour != hour) ||
|
||||||
|
(last_day != day) || (last_month != month) || (last_year != year) ||
|
||||||
|
(last_century != century) );
|
||||||
|
|
||||||
|
registerB = get_RTC_register(0x0B);
|
||||||
|
|
||||||
|
// Convert BCD to binary values if necessary
|
||||||
|
|
||||||
|
if (!(registerB & 0x04)) {
|
||||||
|
second = (second & 0x0F) + ((second / 16) * 10);
|
||||||
|
minute = (minute & 0x0F) + ((minute / 16) * 10);
|
||||||
|
hour = ( (hour & 0x0F) + (((hour & 0x70) / 16) * 10) ) | (hour & 0x80);
|
||||||
|
day = (day & 0x0F) + ((day / 16) * 10);
|
||||||
|
month = (month & 0x0F) + ((month / 16) * 10);
|
||||||
|
year = (year & 0x0F) + ((year / 16) * 10);
|
||||||
|
if(century_register != 0) {
|
||||||
|
century = (century & 0x0F) + ((century / 16) * 10);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Convert 12 hour clock to 24 hour clock if necessary
|
||||||
|
|
||||||
|
if (!(registerB & 0x02) && (hour & 0x80)) {
|
||||||
|
hour = ((hour & 0x7F) + 12) % 24;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Calculate the full (4-digit) year
|
||||||
|
|
||||||
|
if(century_register != 0) {
|
||||||
|
year += century * 100;
|
||||||
|
} else {
|
||||||
|
year += (CURRENT_YEAR / 100) * 100;
|
||||||
|
if(year < CURRENT_YEAR) year += 100;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void delay(int t){
|
||||||
|
volatile int i,j;
|
||||||
|
for(i=0;i<t;i++)
|
||||||
|
for(j=0;j<25000;j++)
|
||||||
|
asm("NOP");
|
||||||
|
}
|
@ -1,14 +1,15 @@
|
|||||||
#define CURRENT_YEAR 2021 // Change this each year!
|
#pragma once
|
||||||
|
#include "io.h"
|
||||||
int century_register = 0x00; // Set by ACPI table parsing code if possible
|
#define CURRENT_YEAR 2021
|
||||||
|
|
||||||
unsigned char second;
|
|
||||||
unsigned char minute;
|
|
||||||
unsigned char hour;
|
|
||||||
unsigned char day;
|
|
||||||
unsigned char month;
|
|
||||||
unsigned int year;
|
|
||||||
|
|
||||||
|
|
||||||
|
extern int century_register;
|
||||||
|
extern unsigned char second;
|
||||||
|
extern unsigned char minute;
|
||||||
|
extern unsigned char hour;
|
||||||
|
extern unsigned char day;
|
||||||
|
extern unsigned char month;
|
||||||
|
extern unsigned int year;
|
||||||
|
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
@ -16,138 +17,7 @@ enum {
|
|||||||
cmos_data = 0x71
|
cmos_data = 0x71
|
||||||
};
|
};
|
||||||
|
|
||||||
int get_update_in_progress_flag() {
|
int get_update_in_progress_flag();
|
||||||
outb(cmos_address, 0x0A);
|
unsigned char get_RTC_register();
|
||||||
return (inb(cmos_data) & 0x80);
|
void read_rtc();
|
||||||
}
|
void delay(int t);
|
||||||
|
|
||||||
unsigned char get_RTC_register(int reg) {
|
|
||||||
outb(cmos_address, reg);
|
|
||||||
return inb(cmos_data);
|
|
||||||
}
|
|
||||||
|
|
||||||
void read_rtc() {
|
|
||||||
unsigned char century;
|
|
||||||
unsigned char last_second;
|
|
||||||
unsigned char last_minute;
|
|
||||||
unsigned char last_hour;
|
|
||||||
unsigned char last_day;
|
|
||||||
unsigned char last_month;
|
|
||||||
unsigned char last_year;
|
|
||||||
unsigned char last_century;
|
|
||||||
unsigned char registerB;
|
|
||||||
|
|
||||||
// Note: This uses the "read registers until you get the same values twice in a row" technique
|
|
||||||
// to avoid getting dodgy/inconsistent values due to RTC updates
|
|
||||||
|
|
||||||
while (get_update_in_progress_flag()); // Make sure an update isn't in progress
|
|
||||||
second = get_RTC_register(0x00);
|
|
||||||
minute = get_RTC_register(0x02);
|
|
||||||
hour = get_RTC_register(0x04);
|
|
||||||
day = get_RTC_register(0x07);
|
|
||||||
month = get_RTC_register(0x08);
|
|
||||||
year = get_RTC_register(0x09);
|
|
||||||
if(century_register != 0) {
|
|
||||||
century = get_RTC_register(century_register);
|
|
||||||
} else {
|
|
||||||
century = 21;
|
|
||||||
}
|
|
||||||
|
|
||||||
do {
|
|
||||||
last_second = second;
|
|
||||||
last_minute = minute;
|
|
||||||
last_hour = hour;
|
|
||||||
last_day = day;
|
|
||||||
last_month = month;
|
|
||||||
last_year = year;
|
|
||||||
last_century = century;
|
|
||||||
|
|
||||||
while (get_update_in_progress_flag()); // Make sure an update isn't in progress
|
|
||||||
second = get_RTC_register(0x00);
|
|
||||||
minute = get_RTC_register(0x02);
|
|
||||||
hour = get_RTC_register(0x04);
|
|
||||||
day = get_RTC_register(0x07);
|
|
||||||
month = get_RTC_register(0x08);
|
|
||||||
year = get_RTC_register(0x09);
|
|
||||||
if(century_register != 0) {
|
|
||||||
century = get_RTC_register(century_register);
|
|
||||||
}
|
|
||||||
} while( (last_second != second) || (last_minute != minute) || (last_hour != hour) ||
|
|
||||||
(last_day != day) || (last_month != month) || (last_year != year) ||
|
|
||||||
(last_century != century) );
|
|
||||||
|
|
||||||
registerB = get_RTC_register(0x0B);
|
|
||||||
|
|
||||||
// Convert BCD to binary values if necessary
|
|
||||||
|
|
||||||
if (!(registerB & 0x04)) {
|
|
||||||
second = (second & 0x0F) + ((second / 16) * 10);
|
|
||||||
minute = (minute & 0x0F) + ((minute / 16) * 10);
|
|
||||||
hour = ( (hour & 0x0F) + (((hour & 0x70) / 16) * 10) ) | (hour & 0x80);
|
|
||||||
day = (day & 0x0F) + ((day / 16) * 10);
|
|
||||||
month = (month & 0x0F) + ((month / 16) * 10);
|
|
||||||
year = (year & 0x0F) + ((year / 16) * 10);
|
|
||||||
if(century_register != 0) {
|
|
||||||
century = (century & 0x0F) + ((century / 16) * 10);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Convert 12 hour clock to 24 hour clock if necessary
|
|
||||||
|
|
||||||
if (!(registerB & 0x02) && (hour & 0x80)) {
|
|
||||||
hour = ((hour & 0x7F) + 12) % 24;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Calculate the full (4-digit) year
|
|
||||||
|
|
||||||
if(century_register != 0) {
|
|
||||||
year += century * 100;
|
|
||||||
} else {
|
|
||||||
year += (CURRENT_YEAR / 100) * 100;
|
|
||||||
if(year < CURRENT_YEAR) year += 100;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
void ReadFromCMOS(unsigned char array[])
|
|
||||||
{
|
|
||||||
unsigned char tvalue, index;
|
|
||||||
|
|
||||||
for (index = 0; index < 128; index++)
|
|
||||||
{
|
|
||||||
asm(
|
|
||||||
"cli\n\t" // Disable interrupts
|
|
||||||
"mov al, index\n\t" // Move index address
|
|
||||||
// since the 0x80 bit of al is not set, NMI is active
|
|
||||||
"out 0x70,al\n\t" // Copy address to CMOS register
|
|
||||||
// some kind of real delay here is probably best
|
|
||||||
"in al,0x71\n\t" // Fetch 1 byte to al
|
|
||||||
"sti\n\t" // Enable interrupts
|
|
||||||
"mov tvalue,al\n\t");
|
|
||||||
|
|
||||||
array[index] = tvalue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
|
|
||||||
/*
|
|
||||||
void WriteTOCMOS(unsigned char array[])
|
|
||||||
{
|
|
||||||
unsigned char index;
|
|
||||||
|
|
||||||
for(index = 0; index < 128; index++)
|
|
||||||
{
|
|
||||||
unsigned char tvalue = array[index];
|
|
||||||
|
|
||||||
asm("cli\n\t" // Clear interrupts
|
|
||||||
"mov al,index\n\t" // move index address
|
|
||||||
"out 0x70,al\n\t" // copy address to CMOS register
|
|
||||||
// some kind of real delay here is probably best
|
|
||||||
"mov al,tvalue\n\t" // move value to al
|
|
||||||
"out 0x71,al\n\t" // write 1 byte to CMOS
|
|
||||||
"sti\n\\t" ); // Enable interrupts
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
*/
|
|
Loading…
x
Reference in New Issue
Block a user