[SOLVED] GUI for an ELF32 kernel booted from GRUB2 UEFI

Question about which tools to use, bugs, the best way to implement a function, etc should go here. Don't forget to see if your question is answered in the wiki first! When in doubt post here.
Post Reply
User avatar
prajwal
Member
Member
Posts: 154
Joined: Sat Oct 23, 2004 11:00 pm
Contact:

[SOLVED] GUI for an ELF32 kernel booted from GRUB2 UEFI

Post by prajwal »

Hi,

I have developed (and developing) a ELF32 bit OS which I boot from legacy grub in typical 80x25 text console (in QEMU and on my old laptop)

However, as most new computers come with UEFI firmware, to boot from UEFI firmware, I built grub2-efi and was able to boot my elf32 bit kernel under QEMU (using OVMF) - with debug information logged to serial port

But because my kernel doesn't support graphics mode - I am stuck without a display as GRUB2 EFI would land my kernel in graphics mode (legacy 80x25 text console is not supported as I understand). So, I am thinking to develop graphics support in my elf32 bit kernel that's booted from GRUB2 EFI (which uses GOP)

I have got the framebuffer information from GRUB2 as part of multiboot header where the framebuffer_addr is 0x8000000 (2GB) and have valid values for width, height, bpp, pitch etc.. - but I am not sure if this can be used or how to use this framebuffer information returned by GRUB2 EFI in my elf32 bit kernel ? OR What are the possible ways to build graphics support in my case (i.e. GRUB2 EFI booting ELF32 bit kernel)

PS:
1. The RAM size setup in QEMU is 128 MB
2. I am just fine with 80x25 text console but I wonder if GRUB2 EFI would allow that ?

Regards,
- Prajwala
Last edited by prajwal on Sun Jan 10, 2016 9:01 pm, edited 2 times in total.
complexity is the core of simplicity
User avatar
Combuster
Member
Member
Posts: 9301
Joined: Wed Oct 18, 2006 3:45 am
Libera.chat IRC: [com]buster
Location: On the balcony, where I can actually keep 1½m distance
Contact:

Re: GUI for an ELF32 kernel booted from GRUB2 UEFI

Post by Combuster »

where the framebuffer_addr is 0x8000000 (2GB) and have valid values for width, height, bpp, pitch etc.. - but I am not sure if this can be used
Have you tried?
"Certainly avoid yourself. He is a newbie and might not realize it. You'll hate his code deeply a few years down the road." - Sortie
[ My OS ] [ VDisk/SFS ]
User avatar
prajwal
Member
Member
Posts: 154
Joined: Sat Oct 23, 2004 11:00 pm
Contact:

Re: GUI for an ELF32 kernel booted from GRUB2 UEFI

Post by prajwal »

Information from multiboot header (as logged via serial port):-
FRAMEBUFFER_ADDR: 0x80000000
FRAMEBUFFER_PITCH: 3200
FRAMEBUFFER_WIDTH: 800
FRAMEBUFFER_HEIGHT: 600
FRAMEBUFFER_BPP: 32
FRAMEBUFFER_RED_POS: 0
FRAMEBUFFER_RED_SIZE: 0
FRAMEBUFFER_GREEN_POS: 16
FRAMEBUFFER_GREEN_SIZE: 8
FRAMEBUFFER_BLUE_POS: 8
FRAMEBUFFER_BLUE_SIZE: 8
I mapped framebuffer_addr 0x80000000 (2GB) in the page table and then ran below code to fill the screen:-

Code: Select all

   unsigned* framebuffer = (unsigned*)(0x80000000);
    for(int y = 0; y < 600; ++y)
    {
      for(int x = 0; x < 800; ++x)
      {
        //unsigned* ptr = (unsigned*)(0x4000000 + y * 3200 + x * 4);
        //*ptr = 0xFFAAAAAA;
        framebuffer[y * 800 + x] = 0xFFAAAA00; //yellow color
      }
    }
and I have attached output (QEMU screenshot)
Attachments
gui.jpg
complexity is the core of simplicity
alexfru
Member
Member
Posts: 1112
Joined: Tue Mar 04, 2014 5:27 am

Re: GUI for an ELF32 kernel booted from GRUB2 UEFI

Post by alexfru »

prajwala wrote: FRAMEBUFFER_PITCH: 3200
AFAIR, that's the distance between two consecutive horizontal lines in the frame buffer (I believe, in pixels). You're ignoring it. You need to multiply Y by it and not by 800.

Also this looks odd:
prajwala wrote: FRAMEBUFFER_RED_SIZE: 0
User avatar
Combuster
Member
Member
Posts: 9301
Joined: Wed Oct 18, 2006 3:45 am
Libera.chat IRC: [com]buster
Location: On the balcony, where I can actually keep 1½m distance
Contact:

Re: GUI for an ELF32 kernel booted from GRUB2 UEFI

Post by Combuster »

I find it especially strange that your horizontal lines increase in distance so I don't trust that your code is actually running as expected. Your compiler and bootloader might not agree on architecture or you have broken the binary or environment in other ways.

Try to draw a 10x10 pixel square and see how that behaves. If it still doesn't you'll at least have reduced the number of memory accesses to the point you can verify the instructions, addresses and values by hand using a debugger.
"Certainly avoid yourself. He is a newbie and might not realize it. You'll hate his code deeply a few years down the road." - Sortie
[ My OS ] [ VDisk/SFS ]
User avatar
BrightLight
Member
Member
Posts: 901
Joined: Sat Dec 27, 2014 9:11 am
Location: Maadi, Cairo, Egypt
Contact:

Re: GUI for an ELF32 kernel booted from GRUB2 UEFI

Post by BrightLight »

alexfru wrote:AFAIR, that's the distance between two consecutive horizontal lines in the frame buffer (I believe, in pixels).
Bytes, not pixels.
In general getting the pixel location in the framebuffer really is just:

Code: Select all

(y * pitch) + (x * (bpp/8)) + framebuffer_base
You know your OS is advanced when you stop using the Intel programming guide as a reference.
User avatar
prajwal
Member
Member
Posts: 154
Joined: Sat Oct 23, 2004 11:00 pm
Contact:

Re: GUI for an ELF32 kernel booted from GRUB2 UEFI

Post by prajwal »

Thanks Combuster.. that approach to debug helped - it turned out to be a silly mistake in the code which was mapping LFB into page table. It works now after fixing page table mapping

Thanks everyone else for your time and suggestion

Regards,
- Prajwala
complexity is the core of simplicity
Post Reply