aboutsummaryrefslogtreecommitdiff
path: root/boot
diff options
context:
space:
mode:
authorMarvin Borner2021-04-05 00:16:11 +0200
committerMarvin Borner2021-04-05 00:16:11 +0200
commit614df32883bd988a59ce1a2a201ce2dd8211ab3a (patch)
tree9df26abeb4815dd5f3275ca99ef8980d20f78950 /boot
parentacffb01f7a8474d4adf82049690380d63faba782 (diff)
Added ELF and disk detection support to bootloader
Diffstat (limited to 'boot')
-rw-r--r--boot/Makefile2
-rw-r--r--boot/entry.asm16
-rw-r--r--boot/load.c861
3 files changed, 699 insertions, 180 deletions
diff --git a/boot/Makefile b/boot/Makefile
index 7ef6cab..d3a1f2d 100644
--- a/boot/Makefile
+++ b/boot/Makefile
@@ -14,5 +14,5 @@ all: compile
compile:
@mkdir -p ../build/
@$(CC) -c $(CFLAGS) load.c -o load.o
- @$(LD) -N -emain -Ttext 0x00040000 -o ../build/load.bin load.o --oformat binary
+ @$(LD) -N -emain -Ttext 0x00009000 -o ../build/load.bin load.o --oformat binary
@$(AS) -f bin entry.asm -o ../build/boot.bin
diff --git a/boot/entry.asm b/boot/entry.asm
index 92a8406..8f12dff 100644
--- a/boot/entry.asm
+++ b/boot/entry.asm
@@ -93,9 +93,9 @@
%define GDT_SIZE (0x40 | 0x00) ; Use 32 bit selectors
%define GDT_DATA_OFFSET 0x10 ; Offset to GDT data segment
-; Kernel constants
+; Kernel loader constants
%define STACK_POINTER 0x00500000 ; The initial stack pointer in kernel mode
-%define KERNEL_POSITION 0x00040000 ; Loaded kernel position in protected mode (* 0x10)
+%define LOADER_POSITION 0x00009000 ; Kernel loader position in protected mode
; ENOUGH, let's go!
@@ -322,11 +322,11 @@ lba:
times SECTOR_SIZE - ($ - $$) db 0
dw SECTOR_END
-; This is the second stage. It tries to load the kernel into memory.
+; This is the second stage. It tries to load the kernel loader into memory.
; To do this, it first checks the integrity of the ext2 fs. Then it has to find
; the address of the root inode (2), find the filename in it and load its contents into memory.
; After this is finished, the stage can jump into the protected mode, enable the
-; A20 line and finally jump to the kernel! ez
+; A20 line and finally jump to the kernel loader! ez
stage_two:
; Verify signature
mov ax, [superblock + EXT2_SIG_OFFSET]
@@ -377,7 +377,7 @@ stage_two:
cmp cx, 256 + 12 ; BLOCK_SIZE / sizeof(u32) = 256
jge disk_error
lea di, [bx + EXT2_POINTER_OFFSET] ; Address of first block pointer
- mov bx, 0x4000 ; Load to this address
+ mov bx, 0x900 ; Load to this address (LOADER_POSITION >> 4)
mov [dest + 2], bx
mov bx, 0 ; Inode location = 0xF0000
mov [dest], bx
@@ -454,13 +454,13 @@ protected_mode:
mov ax, (gdt_tss - gdt) | 0b11 ; Load TSS in ring 3
ltr ax
- mov eax, vid_info ; Pass VBE struct to kernel
+ mov eax, vid_info ; Pass VBE struct to kernel loader
push eax ; Push as second kernel parameter
- mov eax, mem_info ; Pass meminfo to kernel
+ mov eax, mem_info ; Pass meminfo to kernel loader
push eax ; Push as first kernel parameter
- mov edx, KERNEL_POSITION
+ mov edx, LOADER_POSITION
lea eax, [edx]
call eax
diff --git a/boot/load.c b/boot/load.c
index 4a6f3db..a3566f4 100644
--- a/boot/load.c
+++ b/boot/load.c
@@ -1,5 +1,9 @@
-// MIT License, Copyright (c) 2020 Marvin Borner
-// Independent ext2 loader
+// MIT License, Copyright (c) 2021 Marvin Borner
+// Independent ext2 loader - mostly copied from kernel
+
+/**
+ * Some general definitions
+ */
typedef signed char s8;
typedef unsigned char u8;
@@ -7,44 +11,258 @@ typedef unsigned char u8;
typedef signed short s16;
typedef unsigned short u16;
-typedef signed long s32;
-typedef unsigned long u32;
+typedef signed int s32;
+typedef unsigned int u32;
+
+#define NULL ((void *)0)
+#define MIN(a, b) (((a) < (b)) ? (a) : (b))
+
+#define PACKED __attribute__((packed))
-typedef signed long long s64;
-typedef unsigned long long u64;
+#define assert(exp) \
+ if (!(exp)) { \
+ print(__FILE__); \
+ print(": "); \
+ print(__func__); \
+ print(": Bootloader assertion '"); \
+ print(#exp); \
+ print("' failed.\n"); \
+ __asm__ volatile("cli\nhlt"); \
+ }
+
+/**
+ * ATA numbers
+ */
#define BLOCK_SIZE 1024
#define BLOCK_COUNT 256 // BLOCK_SIZE / sizeof(u32)
#define SECTOR_SIZE 512
-#define IDE_BUSY (1 << 7)
-#define IDE_READY (1 << 6)
-#define IDE_DRIVE_FAULT (1 << 5)
-#define IDE_ERROR (1 << 0)
-#define IDE_IO 0x1F0
-#define IDE_DATA 0x0
-#define IDE_FEATURES 0x1
-#define IDE_SECTOR_COUNT 0x2
-#define IDE_LOW 0x3
-#define IDE_MID 0x4
-#define IDE_HIGH 0x5
-#define IDE_HEAD 0x6
-#define IDE_CMD 0x7
-#define IDE_ALTERNATE 0x3F6
-#define LBA_LOW(c) ((u8)(c & 0xFF))
-#define LBA_MID(c) ((u8)(c >> 8) & 0xFF)
-#define LBA_HIGH(c) ((u8)(c >> 16) & 0xFF)
-#define LBA_LAST(c) ((u8)(c >> 24) & 0xF)
-#define IDE_CMD_READ (BLOCK_SIZE / SECTOR_SIZE == 1) ? 0x20 : 0xC4
-#define IDE_CMD_WRITE (BLOCK_SIZE / SECTOR_SIZE == 1) ? 0x30 : 0xC5
-#define IDE_CMD_READ_MUL 0xC4
-#define IDE_CMD_WRITE_MUL 0xC5
+#define SECTOR_COUNT (BLOCK_SIZE / SECTOR_SIZE)
+
+#define ATA_PRIMARY_IO 0x1f0
+#define ATA_SECONDARY_IO 0x170
+
+#define ATA_PRIMARY 0x00
+#define ATA_SECONDARY 0x01
+#define ATA_READ 0x00
+#define ATA_WRITE 0x013
+#define ATA_MASTER 0x00
+#define ATA_SLAVE 0x01
+#define ATA_SR_BSY 0x80
+#define ATA_SR_DRDY 0x40
+#define ATA_SR_DF 0x20
+#define ATA_SR_DSC 0x10
+#define ATA_SR_DRQ 0x08
+#define ATA_SR_CORR 0x04
+#define ATA_SR_IDX 0x02
+#define ATA_SR_ERR 0x01
+#define ATA_REG_DATA 0x00
+#define ATA_REG_ERROR 0x01
+#define ATA_REG_FEATURES 0x01
+#define ATA_REG_SECCOUNT0 0x02
+#define ATA_REG_LBA0 0x03
+#define ATA_REG_LBA1 0x04
+#define ATA_REG_LBA2 0x05
+#define ATA_REG_HDDEVSEL 0x06
+#define ATA_REG_COMMAND 0x07
+#define ATA_REG_STATUS 0x07
+#define ATA_REG_SECCOUNT1 0x08
+#define ATA_REG_LBA3 0x09
+#define ATA_REG_LBA4 0x0a
+#define ATA_REG_LBA5 0x0b
+#define ATA_REG_CONTROL 0x0c
+#define ATA_REG_ALTSTATUS 0x0c
+#define ATA_REG_DEVADDRESS 0x0d
+#define ATA_CMD_READ_PIO 0x20
+#define ATA_CMD_READ_PIO_EXT 0x24
+#define ATA_CMD_READ_DMA 0xc8
+#define ATA_CMD_READ_DMA_EXT 0x25
+#define ATA_CMD_WRITE_PIO 0x30
+#define ATA_CMD_WRITE_PIO_EXT 0x34
+#define ATA_CMD_WRITE_DMA 0xca
+#define ATA_CMD_WRITE_DMA_EXT 0x35
+#define ATA_CMD_CACHE_FLUSH 0xe7
+#define ATA_CMD_CACHE_FLUSH_EXT 0xea
+#define ATA_CMD_PACKET 0xa0
+#define ATA_CMD_IDENTIFY_PACKET 0xa1
+#define ATA_CMD_IDENTIFY 0xec
+#define ATA_IDENT_DEVICETYPE 0
+#define ATA_IDENT_CYLINDERS 2
+#define ATA_IDENT_HEADS 6
+#define ATA_IDENT_SECTORS 12
+#define ATA_IDENT_SERIAL 20
+#define ATA_IDENT_MODEL 54
+#define ATA_IDENT_CAPABILITIES 98
+#define ATA_IDENT_FIELDVALID 106
+#define ATA_IDENT_MAX_LBA 120
+#define ATA_IDENT_COMMANDSETS 164
+#define ATA_IDENT_MAX_LBA_EXT 200
+
+/**
+ * ELF stuff
+ */
+
+#define ELF_MAG0 0x7F
+#define ELF_MAG1 'E'
+#define ELF_MAG2 'L'
+#define ELF_MAG3 'F'
+
+#define ELF_IDENT_COUNT 16
+#define ELF_IDENT_MAG0 0
+#define ELF_IDENT_MAG1 1
+#define ELF_IDENT_MAG2 2
+#define ELF_IDENT_MAG3 3
+
+#define ELF_IDENT_CLASS 4
+#define ELF_IDENT_CLASS_NONE 0
+#define ELF_IDENT_CLASS_32 1
+#define ELF_IDENT_CLASS_64 2
+
+#define ELF_IDENT_DATA 5
+#define ELF_IDENT_DATA_NONE 0
+#define ELF_IDENT_DATA_LSB 1
+#define ELF_IDENT_DATA_MSB 2
+
+#define ELF_IDENT_VERSION 6
+#define ELF_IDENT_OSABI 7
+#define ELF_IDENT_ABIVERSION 8
+#define ELF_IDENT_PAD 9
+
+#define ELF_ETYPE_NONE 0
+#define ELF_ETYPE_REL 1
+#define ELF_ETYPE_EXEC 2
+#define ELF_ETYPE_DYN 3
+#define ELF_ETYPE_CORE 4
+#define ELF_ETYPE_NUM 5
+
+#define ELF_MACHINE_NONE 0
+#define ELF_MACHINE_SPARC 2
+#define ELF_MACHINE_386 3
+#define ELF_MACHINE_SPARC32PLUS 18
+#define ELF_MACHINE_SPARCV9 43
+#define ELF_MACHINE_AMD64 62
+
+#define ELF_PROGRAM_TYPE_NULL 0
+#define ELF_PROGRAM_TYPE_LOAD 1
+#define ELF_PROGRAM_TYPE_DYNAMIC 2
+#define ELF_PROGRAM_TYPE_INTERP 3
+#define ELF_PROGRAM_TYPE_NOTE 4
+#define ELF_PROGRAM_TYPE_SHLIB 5
+#define ELF_PROGRAM_TYPE_PHDR 6
+#define ELF_PROGRAM_TYPE_TLS 7
+
+#define ELF_PROGRAM_FLAG_X 0x1
+#define ELF_PROGRAM_FLAG_W 0x2
+#define ELF_PROGRAM_FLAG_R 0x4
+
+#define ELF_SECTION_TYPE_NULL 0
+#define ELF_SECTION_TYPE_PROGBITS 1
+#define ELF_SECTION_TYPE_SYMTAB 2
+#define ELF_SECTION_TYPE_STRTAB 3
+#define ELF_SECTION_TYPE_RELA 4
+#define ELF_SECTION_TYPE_HASH 5
+#define ELF_SECTION_TYPE_DYNAMIC 6
+#define ELF_SECTION_TYPE_NOTE 7
+#define ELF_SECTION_TYPE_NOBITS 8
+#define ELF_SECTION_TYPE_REL 9
+#define ELF_SECTION_TYPE_SHLIB 10
+#define ELF_SECTION_TYPE_DYNSYM 11
+#define ELF_SECTION_TYPE_COUNT 12
+
+#define ELF_SECTION_FLAG_WRITE 0x1
+#define ELF_SECTION_FLAG_ALLOC 0x2
+#define ELF_SECTION_FLAG_EXEC 0x3
+#define ELF_SECTION_FLAG_MERGE 0x10
+#define ELF_SECTION_FLAG_STRINGS 0x20
+#define ELF_SECTION_FLAG_INFO_LINK 0x40
+#define ELF_SECTION_FLAG_LINK_ORDER 0x80
+#define ELF_SECTION_FLAG_OS_SPECIAL 0x100
+#define ELF_SECTION_FLAG_GROUP 0x200
+#define ELF_SECTION_FLAG_TLS 0x400
+#define ELF_SECTION_FLAG_COMPRESSED 0x800
+
+#define ELF_BSS ".bss"
+#define ELF_DATA ".data"
+#define ELF_DEBUG ".debug"
+#define ELF_DYNAMIC ".dynamic"
+#define ELF_DYNSTR ".dynstr"
+#define ELF_DYNSYM ".dynsym"
+#define ELF_FINI ".fini"
+#define ELF_GOT ".got"
+#define ELF_HASH ".hash"
+#define ELF_INIT ".init"
+#define ELF_REL_DATA ".rel.data"
+#define ELF_REL_FINI ".rel.fini"
+#define ELF_REL_INIT ".rel.init"
+#define ELF_REL_DYN ".rel.dyn"
+#define ELF_REL_RODATA ".rel.rodata"
+#define ELF_REL_TEXT ".rel.text"
+#define ELF_RODATA ".rodata"
+#define ELF_SHSTRTAB ".shstrtab"
+#define ELF_STRTAB ".strtab"
+#define ELF_SYMTAB ".symtab"
+#define ELF_TEXT ".text"
+
+struct PACKED elf_header {
+ u8 ident[ELF_IDENT_COUNT];
+ u16 type;
+ u16 machine;
+ u32 version;
+ u32 entry;
+ u32 phoff;
+ u32 shoff;
+ u32 flags;
+ u16 ehsize;
+ u16 phentsize;
+ u16 phnum;
+ u16 shentsize;
+ u16 shnum;
+ u16 shstrndx;
+};
+
+struct PACKED elf_program {
+ u32 type;
+ u32 offset;
+ u32 vaddr;
+ u32 paddr;
+ u32 filesz;
+ u32 memsz;
+ u32 flags;
+ u32 align;
+};
+
+struct PACKED elf_section {
+ u32 name;
+ u32 type;
+ u32 flags;
+ u32 addr;
+ u32 offset;
+ u32 size;
+ u32 link;
+ u32 info;
+ u32 addralign;
+ u32 entsize;
+};
+
+struct PACKED elf_symbol {
+ u32 name;
+ u32 value;
+ u32 size;
+ u8 info;
+ u8 other;
+ u16 shndx;
+};
+
+/**
+ * EXT2 numbers/structs
+ */
#define EXT2_BOOT 0
#define EXT2_SUPER 1
#define EXT2_ROOT 2
#define EXT2_MAGIC 0x0000EF53
-struct superblock {
+struct ext2_superblock {
u32 total_inodes;
u32 total_blocks;
u32 su_res_blocks; // Superuser reserved
@@ -72,7 +290,7 @@ struct superblock {
u16 res_block_gid;
};
-struct bgd {
+struct ext2_bgd {
u32 block_bitmap;
u32 inode_bitmap;
u32 inode_table;
@@ -83,7 +301,7 @@ struct bgd {
u8 bg_reserved[12];
};
-struct inode {
+struct ext2_inode {
u16 mode;
u16 uid;
u32 size;
@@ -108,9 +326,9 @@ struct inode {
u8 os_specific_val2[12];
};
-#define INODE_SIZE (sizeof(struct inode))
+#define EXT2_INODE_SIZE (sizeof(struct ext2_inode))
-struct dirent {
+struct ext2_dirent {
u32 inode_num;
u16 total_len;
u8 name_len;
@@ -118,77 +336,93 @@ struct dirent {
u8 name[];
};
-struct file {
- struct inode inode;
+struct ext2_file {
+ struct ext2_inode inode;
u32 pos;
u8 block_index;
u8 *buf;
u32 curr_block_pos;
};
-static u32 heap;
-void *read_inode(struct inode *in);
-struct inode *get_inode(int i);
-int find_inode(const char *name, int dir_inode);
-void serial_install(void);
-void serial_print(const char *data);
+/**
+ * Memory
+ */
+
+static u32 heap = 0x0000c000;
-int main(void *data1, void *data2)
+static void *memcpy(void *dest, const void *src, u32 n)
{
- serial_install();
- serial_print("Loaded bootloader!\n");
- heap = 0xf00000;
- void (*entry)(void *, void *);
- *(void **)(&entry) = read_inode(get_inode(find_inode("kernel.bin", 2)));
- if (entry) {
- serial_print("Loaded kernel!\n");
- entry(data1, data2);
- } else {
- serial_print("Couldn't find kernel!\n");
- }
- while (1) {
- };
- return 0;
+ // Inspired by jgraef at osdev
+ u32 num_dwords = n / 4;
+ u32 num_bytes = n % 4;
+ u32 *dest32 = (u32 *)dest;
+ const u32 *src32 = (const u32 *)src;
+ u8 *dest8 = ((u8 *)dest) + num_dwords * 4;
+ const u8 *src8 = ((const u8 *)src) + num_dwords * 4;
+
+ __asm__ volatile("rep movsl\n"
+ : "=S"(src32), "=D"(dest32), "=c"(num_dwords)
+ : "S"(src32), "D"(dest32), "c"(num_dwords)
+ : "memory");
+
+ for (u32 i = 0; i < num_bytes; i++)
+ dest8[i] = src8[i];
+
+ return dest;
}
-static u8 inb(u16 port)
+static void *memset(void *dest, u32 val, u32 n)
{
- u8 value;
- __asm__ volatile("inb %1, %0" : "=a"(value) : "Nd"(port));
- return value;
+ // Inspired by jgraef at osdev
+ u32 uval = val;
+ u32 num_dwords = n / 4;
+ u32 num_bytes = n % 4;
+ u32 *dest32 = (u32 *)dest;
+ u8 *dest8 = ((u8 *)dest) + num_dwords * 4;
+ u8 val8 = (u8)val;
+ u32 val32 = uval | (uval << 8) | (uval << 16) | (uval << 24);
+
+ __asm__ volatile("rep stosl\n"
+ : "=D"(dest32), "=c"(num_dwords)
+ : "D"(dest32), "c"(num_dwords), "a"(val32)
+ : "memory");
+
+ for (u32 i = 0; i < num_bytes; i++)
+ dest8[i] = val8;
+
+ return dest;
}
-static void insl(u16 port, void *addr, int n)
+static void *malloc(u32 size)
{
- __asm__ volatile("cld; rep insl"
- : "=D"(addr), "=c"(n)
- : "d"(port), "0"(addr), "1"(n)
- : "memory", "cc");
+ return (u32 *)(heap += size);
}
-static void outb(u16 port, u8 data)
+static void *zalloc(u32 size)
{
- __asm__ volatile("outb %0, %1" ::"a"(data), "Nd"(port));
+ void *ret = malloc(size);
+ memset(ret, 0, size);
+ return ret;
}
-static void *memcpy(void *dst, const void *src, u32 n)
+static void free(void *ptr)
{
- const char *sp = (const char *)src;
- char *dp = (char *)dst;
- for (; n != 0; n--)
- *dp++ = *sp++;
- return dst;
+ (void)ptr;
}
-static void *memset(void *dst, char val, u32 n)
+/**
+ * String
+ */
+
+static u32 strlen(const char *str)
{
- char *temp = (char *)dst;
- for (; n != 0; n--)
- *temp++ = val;
- return dst;
+ const char *s = str;
+ while (*s)
+ s++;
+ return s - str;
}
-static int strncmp(const char *s1, const char *s2, u32 n)
+static s32 strncmp(const char *s1, const char *s2, u32 n)
{
const u8 *c1 = (const u8 *)s1;
const u8 *c2 = (const u8 *)s2;
@@ -204,15 +438,44 @@ static int strncmp(const char *s1, const char *s2, u32 n)
return d;
}
-static u32 strlen(const char *s)
+static char *strdup(const char *s)
{
- const char *ss = s;
- while (*ss)
- ss++;
- return ss - s;
+ int l = strlen(s) + 1;
+ char *d = malloc(l);
+
+ memcpy(d, s, l);
+
+ return d;
}
-void serial_install(void)
+/**
+ * CPU IO
+ */
+
+static u8 inb(u16 port)
+{
+ u8 value;
+ __asm__ volatile("inb %1, %0" : "=a"(value) : "Nd"(port));
+ return value;
+}
+
+static u16 inw(u16 port)
+{
+ u16 value;
+ __asm__ volatile("inw %1, %0" : "=a"(value) : "Nd"(port));
+ return value;
+}
+
+static void outb(u16 port, u8 data)
+{
+ __asm__ volatile("outb %0, %1" ::"a"(data), "Nd"(port));
+}
+
+/**
+ * Serial
+ */
+
+static void serial_install(void)
{
outb(0x3f8 + 1, 0x00);
outb(0x3f8 + 3, 0x80);
@@ -235,158 +498,414 @@ static void serial_put(char ch)
outb(0x3f8, (u8)ch);
}
-void serial_print(const char *data)
+static void print(const char *data)
{
for (u32 i = 0; i < strlen(data); i++)
serial_put(data[i]);
}
-static void *malloc(u32 size)
+/**
+ * IDE/ATA
+ */
+
+static u8 *ide_buf = NULL;
+
+static void ide_select_drive(u8 bus, u8 drive)
{
- return (u32 *)(heap += size);
+ if (bus == ATA_PRIMARY) {
+ if (drive == ATA_MASTER)
+ outb(ATA_PRIMARY_IO + ATA_REG_HDDEVSEL, 0xa0);
+ else
+ outb(ATA_PRIMARY_IO + ATA_REG_HDDEVSEL, 0xb0);
+ } else {
+ if (drive == ATA_MASTER)
+ outb(ATA_SECONDARY_IO + ATA_REG_HDDEVSEL, 0xa0);
+ else
+ outb(ATA_SECONDARY_IO + ATA_REG_HDDEVSEL, 0xb0);
+ }
}
-static int ide_wait(int check)
+static u8 ide_find(u8 bus, u8 drive)
{
- char r;
+ u16 io = bus == ATA_PRIMARY ? ATA_PRIMARY_IO : ATA_SECONDARY_IO;
+ ide_select_drive(bus, drive);
+
+ // Reset
+ outb(io + ATA_REG_SECCOUNT0, 0);
+ outb(io + ATA_REG_LBA0, 0);
+ outb(io + ATA_REG_LBA1, 0);
+ outb(io + ATA_REG_LBA2, 0);
+
+ // Identify
+ outb(io + ATA_REG_COMMAND, ATA_CMD_IDENTIFY);
+ u8 status = inb(io + ATA_REG_STATUS);
+ if (!status)
+ return 0;
- // Wait while drive is busy. Once just ready is set, exit the loop
- while (((r = (char)inb(IDE_IO | IDE_CMD)) & (IDE_BUSY | IDE_READY)) != IDE_READY)
+ while ((inb(io + ATA_REG_STATUS) & ATA_SR_BSY) != 0)
;
- // Check for errors
- if (check && (r & (IDE_DRIVE_FAULT | IDE_ERROR)) != 0)
- return 0xF;
- return 0;
+ do {
+ status = inb(io + ATA_REG_STATUS);
+ if (status & ATA_SR_ERR)
+ return 0;
+ } while ((status & ATA_SR_DRQ) == 0);
+
+ for (int i = 0; i < BLOCK_COUNT; i++)
+ *(u16 *)(ide_buf + i * 2) = inw(io + ATA_REG_DATA);
+
+ return 1;
}
-static void *ide_read(void *b, u32 block)
+static void ide_delay(u16 io) // 400ns
{
- int sector_per_block = BLOCK_SIZE / SECTOR_SIZE; // 2
- int sector = block * sector_per_block;
+ for (int i = 0; i < 4; i++)
+ inb(io + ATA_REG_ALTSTATUS);
+}
- ide_wait(0);
- outb(IDE_IO | IDE_SECTOR_COUNT, sector_per_block); // Number of sectors
- outb(IDE_IO | IDE_LOW, LBA_LOW(sector));
- outb(IDE_IO | IDE_MID, LBA_MID(sector));
- outb(IDE_IO | IDE_HIGH, LBA_HIGH(sector));
+static void ide_poll(u16 io)
+{
+ for (int i = 0; i < 4; i++)
+ inb(io + ATA_REG_ALTSTATUS);
- // Slave/Master << 4 and last 4 bits
- outb(IDE_IO | IDE_HEAD, 0xE0 | (1 << 4) | LBA_LAST(sector));
- outb(IDE_IO | IDE_CMD, IDE_CMD_READ);
- ide_wait(0);
+ u8 status;
+ do {
+ status = inb(io + ATA_REG_STATUS);
+ } while (status & ATA_SR_BSY);
- // Read-only
- insl(IDE_IO, b, BLOCK_SIZE / 4);
+ do {
+ status = inb(io + ATA_REG_STATUS);
+ /* assert(!(status & ATA_SR_ERR)) */
+ } while (!(status & ATA_SR_DRQ));
+}
- return b;
+static u8 ata_read_one(u8 *buf, u32 lba, u8 drive)
+{
+ u16 io = (drive & ATA_PRIMARY << 1) == ATA_PRIMARY ? ATA_PRIMARY_IO : ATA_SECONDARY_IO;
+ drive = (drive & ATA_SLAVE) == ATA_SLAVE ? ATA_SLAVE : ATA_MASTER;
+ u8 cmd = drive == ATA_MASTER ? 0xe0 : 0xf0;
+ outb(io + ATA_REG_HDDEVSEL, (cmd | (u8)((lba >> 24 & 0x0f))));
+ outb(io + 1, 0x00);
+ outb(io + ATA_REG_SECCOUNT0, 1);
+ outb(io + ATA_REG_LBA0, (u8)lba);
+ outb(io + ATA_REG_LBA1, (u8)(lba >> 8));
+ outb(io + ATA_REG_LBA2, (u8)(lba >> 16));
+ outb(io + ATA_REG_COMMAND, ATA_CMD_READ_PIO);
+ ide_poll(io);
+
+ for (int i = 0; i < BLOCK_COUNT; i++) {
+ u16 data = inw(io + ATA_REG_DATA);
+ *(u16 *)(buf + i * 2) = data;
+ }
+ ide_delay(io);
+ return 1;
}
-static void *buffer_read(int block)
+static u32 ata_read(void *buf, u32 lba, u32 sector_count, u8 drive)
{
- return ide_read(malloc(BLOCK_SIZE), block);
+ u8 *b = buf; // I love bytes, yk
+ for (u32 i = 0; i < sector_count; i++) {
+ ata_read_one(b, lba + i, drive);
+ b += SECTOR_SIZE;
+ }
+ return sector_count;
}
-static struct superblock *get_superblock(void)
+static u8 ata_probe(void)
{
- struct superblock *sb = buffer_read(EXT2_SUPER);
- if (sb->magic != EXT2_MAGIC)
- return 0;
- return sb;
+ for (u8 i = 0; i < 4; i++) {
+ u32 bus = i < 2 ? ATA_PRIMARY : ATA_SECONDARY;
+ u32 drive = i % 2 ? ATA_MASTER : ATA_SLAVE;
+
+ if (!ide_find(bus, drive))
+ continue;
+
+ u8 found_drive = (bus << 1) | drive;
+ // TODO: What about the other drives?
+ return found_drive;
+ }
+
+ return 0;
+}
+
+/**
+ * EXT2
+ */
+
+static void *buffer_read(u32 block, u8 drive)
+{
+ void *buf = zalloc(BLOCK_SIZE);
+ ata_read(buf, block * SECTOR_COUNT, SECTOR_COUNT, drive);
+ return buf;
}
-static struct bgd *get_bgd(void)
+static struct ext2_superblock *get_superblock(u8 drive)
{
- return buffer_read(EXT2_SUPER + 1);
+ struct ext2_superblock *sb = buffer_read(EXT2_SUPER, drive);
+
+ assert(sb->magic == EXT2_MAGIC);
+ return sb;
}
-struct inode *get_inode(int i)
+static struct ext2_bgd *get_bgd(u8 drive)
{
- struct superblock *s = get_superblock();
- //assert(s);
- struct bgd *b = get_bgd();
- //assert(b);
+ return buffer_read(EXT2_SUPER + 1, drive);
+}
- int block_group = (i - 1) / s->inodes_per_group;
- int index = (i - 1) % s->inodes_per_group;
- int block = (index * INODE_SIZE) / BLOCK_SIZE;
+static struct ext2_inode *get_inode(u32 i, struct ext2_inode *in_buf, u8 drive)
+{
+ struct ext2_superblock *s = get_superblock(drive);
+ assert(s);
+ struct ext2_bgd *b = get_bgd(drive);
+ assert(b);
+
+ u32 block_group = (i - 1) / s->inodes_per_group;
+ u32 index = (i - 1) % s->inodes_per_group;
+ u32 block = (index * EXT2_INODE_SIZE) / BLOCK_SIZE;
b += block_group;
- u32 *data = buffer_read(b->inode_table + block);
- struct inode *in =
- (struct inode *)((u32)data + (index % (BLOCK_SIZE / INODE_SIZE)) * INODE_SIZE);
- return in;
+ u32 *buf = buffer_read(b->inode_table + block, drive);
+ struct ext2_inode *in =
+ (struct ext2_inode *)((u32)buf +
+ (index % (BLOCK_SIZE / EXT2_INODE_SIZE)) * EXT2_INODE_SIZE);
+
+ memcpy(in_buf, in, sizeof(*in_buf));
+ free(buf);
+ free(s);
+ free(b - block_group);
+
+ return in_buf;
}
-static u32 read_indirect(u32 indirect, u32 block_num)
+static u32 find_inode(const char *name, u32 dir_inode, u8 drive)
{
- char *data = buffer_read(indirect);
- return *(u32 *)((u32)data + block_num * sizeof(u32));
+ if (!dir_inode)
+ return (unsigned)-1;
+
+ struct ext2_inode i = { 0 };
+ get_inode(dir_inode, &i, drive);
+
+ char *buf = malloc(BLOCK_SIZE * i.blocks / 2);
+ memset(buf, 0, BLOCK_SIZE * i.blocks / 2);
+
+ for (u32 q = 0; q < i.blocks / 2; q++) {
+ char *data = buffer_read(i.block[q], drive);
+ memcpy((u32 *)((u32)buf + q * BLOCK_SIZE), data, BLOCK_SIZE);
+ free(data);
+ }
+
+ struct ext2_dirent *d = (struct ext2_dirent *)buf;
+
+ u32 sum = 0;
+ do {
+ // Calculate the 4byte aligned size of each entry
+ sum += d->total_len;
+ if (strlen(name) == d->name_len &&
+ strncmp((void *)d->name, name, d->name_len) == 0) {
+ free(buf);
+ return d->inode_num;
+ }
+ d = (struct ext2_dirent *)((u32)d + d->total_len);
+
+ } while (sum < (1024 * i.blocks / 2));
+ free(buf);
+ return (unsigned)-1;
}
-void *read_inode(struct inode *in)
+static u32 read_indirect(u32 indirect, u32 block_num, u8 drive)
{
- //assert(in);
- if (!in)
+ void *data = buffer_read(indirect, drive);
+ u32 ind = *(u32 *)((u32)data + block_num * sizeof(u32));
+ free(data);
+ return ind;
+}
+
+static s32 read_inode(struct ext2_inode *in, void *buf, u32 offset, u32 count, u8 drive)
+{
+ if (!in || !buf)
+ return -1;
+
+ if (in->size == 0)
return 0;
- u32 num_blocks = in->blocks / (BLOCK_SIZE / SECTOR_SIZE);
+ u32 num_blocks = in->blocks / (BLOCK_SIZE / SECTOR_SIZE) + 1;
- //assert(num_blocks != 0);
if (!num_blocks)
- return 0;
+ return -1;
- void *buf = (void *)0x9000;
- //assert(buf != 0);
+ u32 first_block = offset / BLOCK_SIZE;
+ u32 last_block = (offset + count) / BLOCK_SIZE;
+ if (last_block >= num_blocks)
+ last_block = num_blocks - 1;
+ u32 first_block_offset = offset % BLOCK_SIZE;
- int indirect;
+ u32 remaining = MIN(count, in->size - offset);
+ u32 copied = 0;
- int blocknum;
- for (u32 i = 0; i < num_blocks; i++) {
+ u32 indirect = 0;
+ u32 blocknum = 0;
+
+ // TODO: Support triply indirect pointers
+ for (u32 i = first_block; i <= last_block; i++) {
if (i < 12) {
blocknum = in->block[i];
} else if (i < BLOCK_COUNT + 12) {
indirect = in->block[12];
- blocknum = read_indirect(indirect, i - 12);
+ blocknum = read_indirect(indirect, i - 12, drive);
} else {
indirect = in->block[13];
- blocknum = read_indirect(indirect, (i - (BLOCK_COUNT + 12)) / BLOCK_COUNT);
- blocknum = read_indirect(blocknum, (i - (BLOCK_COUNT + 12)) % BLOCK_COUNT);
+ blocknum = read_indirect(indirect, (i - (BLOCK_COUNT + 12)) / BLOCK_COUNT,
+ drive);
+ blocknum = read_indirect(blocknum, (i - (BLOCK_COUNT + 12)) % BLOCK_COUNT,
+ drive);
}
- char *data = buffer_read(blocknum);
- memcpy((u32 *)((u32)buf + i * BLOCK_SIZE), data, BLOCK_SIZE);
+ char *data = buffer_read(blocknum, drive);
+ u32 block_offset = (i == first_block) ? first_block_offset : 0;
+ u32 byte_count = MIN(BLOCK_SIZE - block_offset, remaining);
+
+ memcpy((u8 *)buf + copied, data + block_offset, byte_count);
+
+ copied += byte_count;
+ remaining -= byte_count;
+
+ free(data);
}
- return buf;
+ return copied;
}
-int find_inode(const char *name, int dir_inode)
+static struct ext2_inode *find_inode_by_path(const char *path, struct ext2_inode *in_buf, u8 drive)
{
- if (!dir_inode)
+ char *path_cp = strdup(path);
+ char *init = path_cp; // For freeing
+
+ if (path_cp[0] != '/') {
+ free(init);
+ return NULL;
+ }
+
+ path_cp++;
+ u32 current_inode = EXT2_ROOT;
+
+ u32 i = 0;
+ while (1) {
+ for (i = 0; path_cp[i] != '/' && path_cp[i] != '\0'; i++)
+ ;
+
+ if (path_cp[i] == '\0')
+ break;
+
+ path_cp[i] = '\0';
+ current_inode = find_inode(path_cp, current_inode, drive);
+ path_cp[i] = '/';
+
+ if (current_inode == 0) {
+ free(init);
+ return NULL;
+ }
+
+ path_cp += i + 1;
+ }
+
+ u32 inode = find_inode(path_cp, current_inode, drive);
+ free(init);
+ if ((signed)inode <= 0)
+ return NULL;
+
+ return get_inode(inode, in_buf, drive);
+}
+
+static s32 read(const char *path, void *buf, u32 offset, u32 count, u8 drive)
+{
+ struct ext2_inode in = { 0 };
+ if (find_inode_by_path(path, &in, drive) == &in) {
+ return read_inode(&in, buf, offset, count, drive);
+ } else {
+ print("Couldn't find kernel!\n");
return -1;
+ }
+}
- struct inode *i = get_inode(dir_inode);
+/**
+ * ELF
+ */
- char *buf = malloc(BLOCK_SIZE * i->blocks / 2);
- memset(buf, 0, BLOCK_SIZE * i->blocks / 2);
+static s32 elf_load(const char *path, u8 drive)
+{
+ struct elf_header header = { 0 };
+ s32 rd = read(path, &header, 0, sizeof(header), drive);
+ if (rd < 0)
+ return rd;
+ if (rd != sizeof(header))
+ return -1;
- for (u32 q = 0; q < i->blocks / 2; q++) {
- char *data = buffer_read(i->block[q]);
- memcpy((u32 *)((u32)buf + q * BLOCK_SIZE), data, BLOCK_SIZE);
+ // Valid?
+ u8 *magic = header.ident;
+ u8 valid_magic = magic[ELF_IDENT_MAG0] == ELF_MAG0 && magic[ELF_IDENT_MAG1] == ELF_MAG1 &&
+ magic[ELF_IDENT_MAG2] == ELF_MAG2 && magic[ELF_IDENT_MAG3] == ELF_MAG3 &&
+ magic[ELF_IDENT_CLASS] == ELF_IDENT_CLASS_32 &&
+ magic[ELF_IDENT_DATA] == ELF_IDENT_DATA_LSB;
+ if (!valid_magic || (header.type != ELF_ETYPE_REL && header.type != ELF_ETYPE_EXEC) ||
+ header.version != 1 || header.machine != ELF_MACHINE_386)
+ return -1;
+
+ // Loop through programs
+ for (u32 i = 0; i < header.phnum; i++) {
+ struct elf_program program = { 0 };
+
+ if (read(path, &program, header.phoff + header.phentsize * i, sizeof(program),
+ drive) != sizeof(program))
+ return -1;
+
+ if (program.type == ELF_PROGRAM_TYPE_INTERP)
+ return -1;
+
+ if (program.vaddr == 0 || program.type != ELF_PROGRAM_TYPE_LOAD)
+ continue;
+
+ if ((u32)read(path, (void *)program.vaddr, program.offset, program.filesz, drive) !=
+ program.filesz)
+ return -1;
}
- struct dirent *d = (struct dirent *)buf;
+ // Find section string table
+ struct elf_section section_strings = { 0 };
+ if (read(path, &section_strings, header.shoff + header.shentsize * header.shstrndx,
+ sizeof(section_strings), drive) != sizeof(section_strings))
+ return -1;
+
+ if (section_strings.type != ELF_SECTION_TYPE_STRTAB)
+ return -1;
- u32 sum = 0;
- do {
- // Calculate the 4byte aligned size of each entry
- sum += d->total_len;
- if (strncmp((void *)d->name, name, d->name_len) == 0) {
- return d->inode_num;
- }
- d = (struct dirent *)((u32)d + d->total_len);
+ return header.entry;
+}
- } while (sum < (1024 * i->blocks / 2));
- return -1;
+/**
+ * Let's go!
+ */
+
+int main(void *first, void *second)
+{
+ serial_install();
+ print("Loaded bootloader!\n");
+
+ u8 drive = ata_probe();
+ assert(drive);
+
+ s32 elf = elf_load("/bin/kernel", drive);
+ assert(elf > 0);
+
+ void (*kernel)(void *, void *);
+ *(void **)(&kernel) = (void *)elf;
+
+ print("Loaded kernel!\n");
+ kernel(first, second);
+
+ print("WTF, kernel returned!\n");
+
+ while (1)
+ ;
+
+ return 0;
}