lots of new stuff. new memory manager, some stuff on file plumbing

This commit is contained in:
Benji Dial 2019-12-27 17:19:51 -05:00
parent 3f3f9a806d
commit 58b5c8ba41
10 changed files with 364 additions and 110 deletions

View file

@ -14,14 +14,16 @@ kernel: obj out
nasm src/kernel/keyb.asm -o obj/kkeyba.o -f elf32
nasm src/kernel/serial.asm -o obj/kserial.o -f elf32
nasm src/kernel/stub.asm -o obj/kstub.o -f elf32
gcc -c src/kernel/diskio.c -o obj/kdiskio.o -ffreestanding -nostdlib -m32
gcc -c src/kernel/files.c -o obj/kfiles.o -ffreestanding -nostdlib -m32
gcc -c src/kernel/gpt.c -o obj/kgpt.o -ffreestanding -nostdlib -m32
gcc -c src/kernel/keyb.c -o obj/kkeybc.o -ffreestanding -nostdlib -m32
gcc -c src/kernel/main.c -o obj/kmain.o -ffreestanding -nostdlib -m32
gcc -c src/kernel/mem.c -o obj/kmem.o -ffreestanding -nostdlib -m32
gcc -c src/kernel/proc.c -o obj/kproc.o -ffreestanding -nostdlib -m32
gcc -c src/kernel/vga.c -o obj/kvga.o -ffreestanding -nostdlib -m32
ld obj/kkeyba.o obj/kstub.o obj/kfiles.o obj/kkeybc.o obj/kmain.o \
obj/kmem.o obj/kproc.o obj/kserial.o obj/kvga.o \
ld obj/kkeyba.o obj/kserial.o obj/kstub.o obj/kdiskio.o obj/kfiles.o \
obj/kgpt.o obj/kkeybc.o obj/kmain.o obj/kmem.o obj/kproc.o obj/kvga.o \
-o out/kernel.elf -T src/kernel/link.ld -s --orphan-handling=discard -melf_i386
clean:

36
src/kernel/diskio.c Normal file
View file

@ -0,0 +1,36 @@
/*
Copyright 2019 Benji Dial
Permission to use, copy, modify, and/or distribute this
software for any purpose with or without fee is hereby
granted, provided that the above copyright notice and this
permission notice appear in all copies.
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS
ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO
EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER
RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
OF THIS SOFTWARE.
*/
#include "diskio.h"
uint32_t read_sectors(uint8_t drive, uint32_t start, uint32_t count, void *buffer) {
if ((drive >= 26) || !(drives[drive].flags & DISK_IN))
return 0;
return 0;//TODO
}
uint32_t write_sectors(uint8_t drive, uint32_t start, uint32_t count, void *buffer) {
if ((drive >= 26) || !(drives[drive].flags & DISK_IN))
return 0;
return 0;//TODO
}
void update_status(uint8_t drive) {
//TODO
}

52
src/kernel/diskio.h Normal file
View file

@ -0,0 +1,52 @@
/*
Copyright 2019 Benji Dial
Permission to use, copy, modify, and/or distribute this
software for any purpose with or without fee is hereby
granted, provided that the above copyright notice and this
permission notice appear in all copies.
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS
ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO
EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER
RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
OF THIS SOFTWARE.
*/
#include <stdint.h>
#include <stdbool.h>
#ifndef DISKIO_H
#define DISKIO_H
enum interface {
FLOPPY,
IDE,
ATA
};
enum drive_flags {
SPUN = 0x01,
DISK_IN = 0x02,
PRESENT = 0x04
};
struct drive_info {
uint8_t interface;
uint8_t number;
uint8_t flags;
uint8_t ssize;//log_2 of B
uint32_t length;//in kiB/2
};
struct drive_info drives[26];
uint32_t read_sectors(uint8_t drive, uint32_t start, uint32_t count, void *buffer);
uint32_t write_sectors(uint8_t drive, uint32_t start, uint32_t count, void *buffer);
void update_status(uint8_t drive);
#endif

View file

@ -18,6 +18,23 @@ OF THIS SOFTWARE.
*/
#include "files.h"
#include "diskio.h"
#include "gpt.h"
uint8_t working_drive;
uint8_t working_part;
uint8_t buffer[1024];
void set_working_drive(uint8_t name, uint8_t part) {
if ((name >= (uint8_t)'a') && (name <= (uint8_t)'z') &&
(drives[name - (uint8_t)'a'].flags & PRESENT))
working_drive = name - (uint8_t)'a';
if (part_info[working_drive].format & ~DIRECT)
working_part =
(part >= (uint8_t)'0') && (name <= (uint8_t)'9') &&
(part_info[working_drive].n_partitions > part) ?
(part & 0x0f) : 0;
}
uint16_t open_file(uint8_t *name) {
return 0;//TODO
@ -53,3 +70,21 @@ uint32_t read_line_file(uint16_t handle, uint32_t max, void *buffer) {
++i;
}
}
void detect_disk_parts(uint8_t drive) {
if ((drive >= (uint8_t)'a') && (drive <= (uint8_t)'z')) {
uint8_t dn = (drive & 0x1f) - 1;
update_status(dn);
if (drives[dn].flags & (PRESENT | DISK_IN)) {
read_sectors(dn, 0, 2, buffer);
if ((*(uint32_t *)&buffer[0x200] == 0x20494645) &&//EFI
(*(uint32_t *)&buffer[0x204] == 0x54524150)) //PART
part_info[dn].format = gpt_parse_pt(dn, buffer) ? GPT : UNKNOWN;
;
}
else {
part_info[dn].format = NOT_PRESENT;
part_info[dn].n_partitions = 0;
}
}
}

View file

@ -29,7 +29,30 @@ struct file_info {
uint32_t position;
};
enum fs_format {
ISO9660
};
enum pt_format {
NOT_PRESENT = 0,
UNKNOWN = 1,
DIRECT = 2,
MBR = 3,
GPT = 4
};
struct drive_parts {
uint8_t format;
uint8_t n_partitions;
uint8_t partition_types[10];
uint32_t partition_offsets[10];
uint32_t partition_sizes[10];
void *partition_cache[10];
};
struct drive_parts part_info[26];
struct file_info file_table[65536];
void set_working_drive(uint8_t drive, uint8_t part);
uint16_t open_file(uint8_t *name);
void close_file(uint16_t handle);
uint32_t read_file(uint16_t handle, uint32_t length, void *buffer);
@ -37,5 +60,6 @@ void write_file(uint16_t handle, uint32_t length, void *buffer);
void seek_file(uint16_t handle, int32_t by);
uint32_t get_size(uint16_t handle);
uint32_t read_line_file(uint16_t handle, uint32_t max, void *buffer);
void detect_disk_parts(uint8_t drive);
#endif

40
src/kernel/gpt.c Normal file
View file

@ -0,0 +1,40 @@
/*
Copyright 2019 Benji Dial
Permission to use, copy, modify, and/or distribute this
software for any purpose with or without fee is hereby
granted, provided that the above copyright notice and this
permission notice appear in all copies.
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS
ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO
EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER
RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
OF THIS SOFTWARE.
*/
#include "gpt.h"
#include "files.h"
#include "diskio.h"
bool gpt_parse_pt(uint8_t dn, uint8_t *buffer) {
struct gpt_header *head = (struct gpt_header *)&buffer[0x200];
struct drive_parts *pi = part_info + dn;
pi->format = GPT;
pi->n_partitions = (head->n_parts & 0xffffff00 ? 0x000000ff : head->n_parts);
uint32_t part_entry_size = head->part_size;
struct gpt_entry *as_entries = (struct gpt_entry *)buffer;
for (uint8_t i = 0; i < pi->n_partitions; ++i) {
if (!(i % 8))
read_sectors(dn, 2 + i / 4, 2, buffer);
pi->partition_sizes[i] = ((as_entries[i % 8].last_sector + 1) >> 1) - (pi->partition_offsets[i] = ((as_entries[i % 8].sector) >> 1));
switch (as_entries[i % 8].id.le) {
//TODO
}
}
return true;
}

69
src/kernel/gpt.h Normal file
View file

@ -0,0 +1,69 @@
/*
Copyright 2019 Benji Dial
Permission to use, copy, modify, and/or distribute this
software for any purpose with or without fee is hereby
granted, provided that the above copyright notice and this
permission notice appear in all copies.
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS
ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO
EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER
RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
OF THIS SOFTWARE.
*/
#include <stdint.h>
#include <stdbool.h>
#ifndef GPT_H
#define GPT_H
struct uuid {
uint64_t le;
uint64_t be;
} __attribute__ ((__packed__));
struct gpt_header {
uint8_t signature[8];
uint32_t version;
uint32_t header_length;
uint32_t check;//crc
uint32_t spacing;
uint64_t head_sector;
uint64_t head_backup_sector;
uint64_t data_start;
uint64_t data_last;
struct uuid id;
uint64_t pt_sector;
uint32_t n_parts;
uint32_t part_size;
uint32_t pt_check;
} __attribute__ ((__packed__));
struct gpt_entry {
struct uuid type;
struct uuid id;
uint64_t sector;
uint64_t last_sector;
uint64_t flags;
uint8_t name[72];
} __attribute__ ((__packed__));
enum type_le {
GPT_ESP_LE = 0xc12a7328f81f11d2,
GPT_BOOT_LE = 0x2168614864496e6f
};
enum type_be {
GPT_ESP_BE = 0x3bc93ec9a0004bba,
GPT_BOOT_BE = 0x4946456465654e74
};
bool gpt_parse_pt(uint8_t dn, uint8_t *buffer);
#endif

View file

@ -155,7 +155,8 @@ enum error_codes {
INSUFF_MEMORY = 0x00000002,
MMAP_TOO_SMALL = 0x00000003,
MISSING_RC = 0x00000004,
BAD_RC_LINE = 0x00000005
BAD_RC_LINE = 0x00000005,
BAD_MMAP = 0x00000006
};
struct tag_start *tag_pointer;
@ -172,7 +173,7 @@ uint32_t main(void) {
case BOOT_COMMAND:
put_sz(": boot arguments\n ");
put_sz(*(uint8_t *)(tag_pointer + 1) ?
(uint8_t *)(tag_pointer + 1) : "No arguments");
(uint8_t *)(tag_pointer + 1) : (uint8_t *)"No arguments");
break;
case LOADER_NAME:
@ -219,20 +220,22 @@ uint32_t main(void) {
break;
case MEMORY_MAP: {
mmap_start = (struct mmap_entry *)0;
uint32_t size = *(uint32_t *)(tag_pointer + 1);
struct mmap_tag_entry *tag = (struct mmap_tag_entry *)(tag_pointer + 2);
struct mmap_tag_entry *end = (struct mmap_tag_entry *)((uint32_t)tag_pointer + tag_pointer->size);
struct mmap_entry *entry = mmap_bss;
uint32_t bitmask = 0x00000001;
uint32_t byte = 0;
struct mmap_entry **last_next = &mmap_start;
uint32_t usable = 0;
put_sz(": memory map\n Regions:\n");
while (tag != end) {
if (!(tag->base & 0xffffffff00000000)) {
if (!(tag->base & 0xffffffff00000000) && tag->length) {
entry->base = (uint32_t)tag->base;
entry->length = (tag->base + tag->length - 1) & 0xffffffff00000000 ?
1 + ~(uint32_t)tag->base : (uint32_t)tag->length;
if (!entry->base)
mmap_start = entry;
put_sz(" 0x");
put_32_hex(entry->base);
put_sz(" - 0x");
@ -263,8 +266,6 @@ uint32_t main(void) {
entry->whose = HARDWARE;
break;
}
*last_next = entry;
last_next = &entry->next;
mmap_bitmap[byte] |= bitmask;
++entry;
if (!(bitmask <<= 1)) {
@ -279,29 +280,25 @@ uint32_t main(void) {
put_sz(" Total usable memory: 0x");
put_32_hex(usable);
bool dirty;
do {
dirty = false;
for (struct mmap_entry *current = mmap_start; current; current = current->next)
for (struct mmap_entry *against = mmap_start; against; against = against->next)
if ((current->base + current->length == against->base) &&
(current->whose == against->whose)) {
current->length += against->length;
unmark_entry(against);
if (against == mmap_start)
mmap_start = against->next;
else {
for (current = mmap_start; current->next != against; current = current->next)
;
current->next = against->next;
}
dirty = true;
}
} while (dirty);
for (struct mmap_entry *b = mmap_bss; b < entry; ++b)
for (struct mmap_entry *a = mmap_bss; a < entry; ++a)
if (b->base + b->length == a->base)
(b->after = a)->before = b;
for (struct mmap_entry *p = mmap_start; p->after; p = p->after)
while (p->whose == p->after->whose) {
p->length += p->after->whose;
unmark_entry(p->after);
if (p->after = p->after->after)
p->after->before = p;
}
if (mmap_start) {
have_mmap = true;
break;
}
return BAD_MMAP;
}
case VBE_INFO: {
struct vbe_tag *vbe_info = (struct vbe_tag *)(tag_pointer + 1);
@ -408,26 +405,33 @@ uint32_t main(void) {
if (!have_mmap)
return NO_MMAP;
put_sz("\n\nImportant locations:");
put_sz("\n\nInfo:");
if (have_text_base) {
put_sz("\n Text section: 0x");
put_32_hex(text_base);
}
put_sz("\n Memory map: 0x");
put_32_hex((uint32_t)mmap_bss);
put_sz(" - 0x");
put_32_hex((uint32_t)mmap_bss - 1 +
sizeof(struct mmap_entry) * MMAP_SIZE);
put_sz("\n Process table: 0x");
put_32_hex((uint32_t)proc_table);
put_sz(" - 0x");
put_32_hex((uint32_t)proc_table - 1 +
sizeof(struct proc_info) * 65536);
put_sz("\n File table: 0x");
put_32_hex((uint32_t)file_table);
put_sz(" - 0x");
put_32_hex((uint32_t)file_table - 1 +
sizeof(struct file_info) * 65536);
put_sz("\n Memory map size: 0x");
put_32_hex(sizeof(struct mmap_entry) * MMAP_SIZE);
put_sz("\n Process table size: 0x");
put_32_hex(sizeof(struct proc_info) * 65536);
put_sz("\n File table size: 0x");
put_32_hex(sizeof(struct file_info) * 65536);
put_sz("\n\nDisks:");
for (uint8_t i = 0; i < 26; ++i) {
detect_disk_parts((uint8_t)'a' + i);
if (part_info[i].format) {
put_sz("\n ");
put_char((uint8_t)'a' + i);
if (part_info[i].n_partitions) {
put_char('0');
uint8_t str[] = "\n 1";
for (uint8_t m = part_info[i].n_partitions + (uint8_t)'0'; str[4] < m; ++(str[4]))
put_sz(str);
}
}
}
put_sz("\n\n");
put_sz("No file support yet. Halting.");

View file

@ -23,75 +23,66 @@ uint32_t last_byte = 0;
uint32_t last_mask = 0x00000001;
void *allocate_block(uint32_t size, uint16_t proc_n) {
struct mmap_entry *i = mmap_start;
while (i->whose || (i->length < size))
if (i->next)
i = i->next;
else
return (void *)0;
if (i->length == size) {
i->whose = proc_n;
return (void *)i->base;
for (struct mmap_entry *find = mmap_start; find; find = find->after)
if (!find->whose) {
if (find->length == size) {
find->whose = proc_n;
return (void *)find->base;
}
uint32_t remaining = i->length - size;
i->whose = proc_n;
i->length = size;
else if (find->length > size) {
uint32_t byte = last_byte;
uint32_t mask = last_mask;
do {
while (mmap_bitmap[byte] & mask) {
if ((byte == last_byte) && (mask == last_mask)) {
while (find = find->after)
if (find->length == size) {
find->whose = proc_n;
return (void *)find->base;
}
return (void *)0;
}
if (!(mask <<= 1)) {
mask = 0x00000001;
if (++byte == MMAP_SIZE)
byte = 0;
}
if ((byte == last_byte) && (mask == last_mask))
return 0;
} while (mmap_bitmap[byte] & mask);
last_byte = byte;
last_mask = mask;
mmap_bitmap[byte] |= mask;
uint8_t bit = 0;
while (!(mask & 0x00000001)) {
++bit;
mask >>= 1;
}
struct mmap_entry *new = mmap_bss + (byte << 5) + bit;
new->base = i->base + size;
new->length = remaining;
mmap_bitmap[last_byte = byte] |= last_mask = mask;
struct mmap_entry *new = mmap_bss + (byte << 5);
while (mask >>= 1)
++new;
((((new->after = find->after)->before = new)->before = find)->after = new)->base = find->base + size;
new->length = find->length - size;
find->length = size;
find->whose = proc_n;
new->whose = FREE;
new->next = i->next;
i->next = new;
return (void *)i->base;
return (void *)find->base;
}
}
return (void *)0;
}
void deallocate_block(void *start) {
struct mmap_entry *find = mmap_start;
while (find->base != (uint32_t)start)
if (find->next)
find = find->next;
if (find->after)
find = find->after;
else
return;
find->whose = FREE;
for (struct mmap_entry *find_after = mmap_start; find_after; find_after = find_after->next)
if (find_after->base == find->base + find->length) {
if (!find_after->whose) {
find->length += find_after->length;
find->next = find_after->next;
unmark_entry(find_after);
}
break;
}
for (struct mmap_entry *find_before = mmap_start; find_before; find_before = find_before->next)
if (find->base == find_before->base + find_before->length) {
if (!find_before->whose) {
find_before->length += find->length;
find_before->next = find->next;
while (find->before && !find->before->whose) {
find->before->length += find->length;
if (find->before->after = find->after)
find->after->before = find->before;
unmark_entry(find);
find = find->before;
}
break;
if (!find->before)
mmap_start = find;
while (find->after && !find->after->whose) {
find->length += find->after->length;
unmark_entry(find->after);
if (find->after = find->after->after)
find->after->before = find;
}
}

View file

@ -32,14 +32,15 @@ struct mmap_entry {
uint32_t base;
uint32_t length;
uint16_t whose;
struct mmap_entry *next;
struct mmap_entry *after;
struct mmap_entry *before;
};
#define MMAP_SIZE 65536
struct mmap_entry *mmap_start;
struct mmap_entry mmap_bss[MMAP_SIZE];
uint32_t mmap_bitmap[MMAP_SIZE / 32];
#define unmark_entry(entry) mmap_bitmap[((entry) - mmap_bss) >> 5] ^= 1 << (((entry) - mmap_bss) & 0x0000001f);
#define unmark_entry(entry) (mmap_bitmap[((entry) - mmap_bss) >> 5] ^= 1 << (((entry) - mmap_bss) & 0x0000001f))
void *allocate_block(uint32_t size, uint16_t proc_n);
void deallocate_block(void *start);