osdev - VGA in protected mode initializing weirdly

187 Views Asked by At

So, I have been trying to make a simple OS, that enters mode 13h (using my previously used code that should work on protected mode (it atleast did on my previous OS using grub as bootloader)
The code is posted below.
When i try to use this new code, i get really, really weird output. The writing pixel to the screen doesn't work, but the change in screen resolution does indicate that some video mode was initiated..
The video mode seems to be around 640px on Y axis and about 1940pixels on X axis
The asm code for initiating the c code is also below, in case that is the issue.
ASM loader:

[org 0x7C00]
bits 16
_start:
    ;segment registers
    mov ax, 0x0000
    mov ss, ax
    mov sp, 0x7C00
    mov ax, 0x07E0
    mov ds, ax
    
    ;enter protected mode
    cli
    lgdt [gdtr]
    mov eax, cr0
    or eax, 1 
    mov cr0, eax
    jmp main
gdt_start:
    dq 0x0000000000000000
    dq 0x00CF9A000000FFFF
    dq 0x00CF92000000FFFF
gdt_end:
gdtr:
    dw gdt_end - gdt_start - 1
    dd gdt_start
main:
    ;Segment reload 2
    mov ax, DATA_SEG
    mov ds, ax
    mov es, ax
    mov fs, ax
    mov gs, ax
    mov ss, ax

C program (simpliest possible example):

#include <stdint.h>
unsigned char g_320x200x256[] =
{
/* MISC */
    0x63,
/* SEQ */
    0x03, 0x01, 0x0F, 0x00, 0x0E,
/* CRTC */
    0x5F, 0x4F, 0x50, 0x82, 0x54, 0x80, 0xBF, 0x1F,
    0x00, 0x41, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
    0x9C, 0x0E, 0x8F, 0x28, 0x40, 0x96, 0xB9, 0xA3,
    0xFF,
/* GC */
    0x00, 0x00, 0x00, 0x00, 0x00, 0x40, 0x05, 0x0F,
    0xFF,
/* AC */
    0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07,
    0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F,
    0x41, 0x00, 0x0F, 0x00, 0x00
};
#define VGA_AC_INDEX        0x3C0
#define VGA_AC_WRITE        0x3C0
#define VGA_AC_READ         0x3C1
#define VGA_INSTAT_READ     0x3DA
#define VGA_MISC_WRITE      0x3C2
#define VGA_MISC_READ       0x3CC
#define VGA_ADDRESS 0xA0000
/*          COLOR emulation     MONO emulation */
#define VGA_CRTC_INDEX      0x3D4       /* 0x3B4 */
#define VGA_CRTC_DATA       0x3D5       /* 0x3B5 */
#define VGA_GC_INDEX        0x3CE
#define VGA_GC_DATA         0x3CF
#define VGA_SEQ_INDEX       0x3C4
#define VGA_SEQ_DATA        0x3C5

#define VGA_NUM_AC_REGS     21
#define VGA_NUM_CRTC_REGS   25
#define VGA_NUM_GC_REGS     9
#define VGA_NUM_SEQ_REGS    5

uint8_t inb(uint16_t port) {
    uint8_t value;
    asm volatile ("inb %1, %0" : "=a"(value) : "Nd"(port));
    return value;
}
void outb(uint16_t port, uint8_t value) {
    asm volatile ("outb %0, %1" : : "a"(value), "Nd"(port));
}
int main(){
    unsigned char* regs = g_320x200x256;
    unsigned i;

   ; vga mode driver (originally from osdev wiki)

/* write MISCELLANEOUS reg */
    outb(VGA_MISC_WRITE, *regs);
    regs++;
/* write SEQUENCER regs */
    for(i = 0; i < VGA_NUM_SEQ_REGS; i++)
    {
        outb(VGA_SEQ_INDEX, i);
        outb(VGA_SEQ_DATA, *regs);
        regs++;
    }
/* unlock CRTC registers */
    outb(VGA_CRTC_INDEX, 0x03);
    outb(VGA_CRTC_DATA, inb(VGA_CRTC_DATA) | 0x80);
    outb(VGA_CRTC_INDEX, 0x11);
    outb(VGA_CRTC_DATA, inb(VGA_CRTC_DATA) & ~0x80);
/* make sure they remain unlocked */
    regs[0x03] |= 0x80;
    regs[0x11] &= ~0x80;
/* write CRTC regs */
    for(i = 0; i < VGA_NUM_CRTC_REGS; i++)
    {
        outb(VGA_CRTC_INDEX, i);
        outb(VGA_CRTC_DATA, *regs);
        regs++;
    }
/* write GRAPHICS CONTROLLER regs */
    for(i = 0; i < VGA_NUM_GC_REGS; i++)
    {
        outb(VGA_GC_INDEX, i);
        outb(VGA_GC_DATA, *regs);
        regs++;
    }
/* write ATTRIBUTE CONTROLLER regs */
    for(i = 0; i < VGA_NUM_AC_REGS; i++)
    {
        (void)inb(VGA_INSTAT_READ);
        outb(VGA_AC_INDEX, i);
        outb(VGA_AC_WRITE, *regs);
        regs++;
    }
/* lock 16-color palette and unblank display */
    (void)inb(VGA_INSTAT_READ);
    outb(VGA_AC_INDEX, 0x20);
    unsigned char *VGA = (unsigned char*) VGA_ADDRESS;
    VGA[0] = 0;
    VGA[1] = 1;
    VGA[2] = 2;
    ;Infinite loop at end
    while(1);
}

For now, i am copying the compiled binary code after the asm bootloader and 0x55AA at the 511'th and 512'th byte

Steps to reproduce:
nasm -f bin kernel.asm Compiles the bootloader
i686-elf-gcc -nostdlib -ffreestanding -O2 -Wall -Wextra -m32 -c vgacode.c -o kern_c.o Compiles the c file (i386-gcc should work as well)
objcopy -O binary kern_c.o kernel.o Convert from elf32 to bin
Then, you can copy the code using cat (cat kenrnel kernel.o > file2.bin) , and emulate using:
qemu-system-x86_64 -fda file2.bin

I have tried to recheck the code, use qemu in various modes, exporting memory, etc.

1

There are 1 best solutions below

1
On BEST ANSWER

I managed to init the video mode in real mode, so the vga code inside protected mode wasnt needed.
For anyone having same issue here is the code.

    mov ax, 0x4F02
    mov bx, 0x11B
    int 0x10

    or

    mov ah, 0
    mov al, 13h
    int 0x10 

Modes: https://web.archive.org/web/20030603170438/http://www.vesa.org/vbelink.html