1

-masm=intel我使用带有选项的 GCC 编译我的代码。我的内核是由像 GRUB 这样的 Multiboot 加载程序加载的。

我想加载我的 GDT 的地址,然后我重新加载所有段寄存器,但这会导致三重错误(虚拟机重新启动)。如果我在本机程序集中(在 .asm 文件中)使用此代码,则该代码有效。

gdt.c:

#include "gdt.h"

GDT gdtp;

uint64_t gdt[GDT_ENTRIES];

void set_gdt_entry(int x, unsigned int base, unsigned int limit, int flags) {
    gdt[x] = limit & 0xffffLL;
    gdt[x] |= (base & 0xffffffLL) << 16;
    gdt[x] |= ((flags >> 4) & 0xffLL) << 40;
    gdt[x] |= ((limit >> 16) & 0xfLL) << 48;
    gdt[x] |= (flags & 0xfLL) << 52;
    gdt[x] |= ((base >> 24) & 0xffLL) << 56;
}

void gdt_init() {
    gdtp.limit = GDT_ENTRIES * 8 - 1;
    gdtp.pointer = gdt;

    set_gdt_entry(0, 0, 0, 0);
    set_gdt_entry(1, 0, 0xFFFFFFFF, GDT_SIZE | GDT_EXECUTABLE | GDT_SEGMENT | GDT_PRESENT);
    set_gdt_entry(2, 0, 0xFFFFFFFF, GDT_SIZE | GDT_SEGMENT | GDT_PRESENT);
    set_gdt_entry(3, 0, 0xFFFFFFFF, GDT_SIZE | GDT_EXECUTABLE | GDT_SEGMENT | GDT_PRESENT | GDT_RING3);
    set_gdt_entry(4, 0, 0xFFFFFFFF, GDT_SIZE | GDT_SEGMENT | GDT_PRESENT | GDT_RING3);

    asm volatile(
        "lgdt %0\n"
        "mov eax, 0x10\n"
        "mov ss, eax\n"
        "mov es, eax\n"
        "mov ds, eax\n"
        "mov gs, eax\n"
        "mov fs, eax\n"
        "jmp 0x08:1f\n"
        "1:\n"
        : : "m" (gdtp) : "eax"
    );
}

这是我的 gdt.h:

#include <stdint.h>

#define GDT_ENTRIES 7

typedef enum {
    GDT_AVAILABLE = 0x1,
    GDT_LONG_MODE = 0x2,
    GDT_SIZE = 0x3,
    GDT_GRANULARITY = 0x8,
    GDT_ACCESSED = 0x010,
    GDT_READ_WRITE = 0x020,
    GDT_CONFORMING = 0x040,
    GDT_EXECUTABLE = 0x080,
    GDT_SEGMENT = 0x100,
    GDT_RING1 = 0x200,
    GDT_RING2 = 0x400,
    GDT_RING3 = 0x600,
    GDT_PRESENT = 0x800
} GDT_FLAGS;

typedef struct {
    uint16_t limit;
    void *pointer;
}__attribute__((packed)) GDT;

void set_gdt_entry(int, unsigned int, unsigned int, int);
void gdt_init();

我该怎么做才能让它工作?

4

1 回答 1

2

问题不在内联汇编代码中,但是我在您添加到问题的代码片段中看到了错误:

  • 此条GDT_FLAGS目:

    GDT_SIZE = 0x3
    

    应该:

    GDT_SIZE = 0x4
    
  • 您正在使用 Multiboot 加载程序,您将访问 0x100000 以上的内存。您的 GDT 条目没有GDT_GRANULARITY设置位,因此您受限于较低的 1MiB 内存。同样,您还没有用该GDT_READ_WRITE位标记任何描述符。GDT 初始化应该是:

    void gdt_init() {
        gdtp.limit = GDT_ENTRIES * 8 - 1;
        gdtp.pointer = gdt;
    
        set_gdt_entry(0, 0, 0, 0);
        set_gdt_entry(1, 0, 0xFFFFFFFF, GDT_GRANULARITY | GDT_READ_WRITE \
                      | GDT_SIZE | GDT_EXECUTABLE | GDT_SEGMENT | GDT_PRESENT);
        set_gdt_entry(2, 0, 0xFFFFFFFF, GDT_GRANULARITY | GDT_READ_WRITE \
                      | GDT_SIZE | GDT_SEGMENT | GDT_PRESENT);
        set_gdt_entry(3, 0, 0xFFFFFFFF, GDT_GRANULARITY | GDT_READ_WRITE \
                      | GDT_SIZE | GDT_EXECUTABLE | GDT_SEGMENT | GDT_PRESENT | GDT_RING3);
        set_gdt_entry(4, 0, 0xFFFFFFFF, GDT_GRANULARITY | GDT_READ_WRITE \
                      | GDT_SIZE | GDT_SEGMENT | GDT_PRESENT | GDT_RING3);
    
        asm volatile(
            "lgdt %0\n"
            "mov eax, 0x10\n"
            "mov ss, eax\n"
            "mov es, eax\n"
            "mov ds, eax\n"
            "mov gs, eax\n"
            "mov fs, eax\n"
            "jmp 0x08:1f\n"
            "1:\n"
            : : "m" (gdtp) : "eax"
        );
    }
    

建议

  • 在操作系统开发的早期调试 GDT 代码和中断时,我发现使用 BOCHS 仿真器很有用。当出现问题(即三重故障)时,它将转储处理器状态信息,并具有将这些表转储到控制台的命令info gdtinfo idt要使用 BOCHS 进行操作系统开发,您可以生成 ISO 映像并作为 CD-ROM 引导。

  • 您正确编码您set_gdt_entry的方式是只使用低 20 位而丢弃高 12 位。为了使内容更具可读性,我建议使用介于 0x00000 和 0xFFFFF(含)之间的值指定限制。使用时GDT_GRANULARITY,CPU 将限制值左移 12 位,并将低 12 位设置为 0xFFF。设置时GDT_GRANULARITY,限制被视为 4KiB 页面而不是字节。

    如果GDT_GRANULARITY未设置,则限制值只是 0x00000 和 0xFFFFF 之间的 20 位值,将限制指定为字节而不是 4KiB 页面。

于 2019-08-23T16:31:25.047 回答