how to get device io port or PCI numbers

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
zq
Posts: 9
Joined: Wed May 10, 2017 7:00 pm
Libera.chat IRC: si

how to get device io port or PCI numbers

Post by zq »

In real mode, I read/write a disk via int 13h, but in protected mode, we can only access it by io, of cource I can enum all the storage devices and check out which is the disk I boot from.

However, I would like to have a easier way to config out the io port or PCI bus&device$fuction number of the disk in real mode, so that after I enter protected mode, I can access it directly by it.
Octocontrabass
Member
Member
Posts: 5586
Joined: Mon Mar 25, 2013 7:01 pm

Re: how to get device io port or PCI numbers

Post by Octocontrabass »

If the BIOS supports it, you can find that information using INT 0x13 AH=0x48. Here's a link to the specification.

If the BIOS doesn't support it, you have to enumerate all of the storage devices to find the boot disk.
zq
Posts: 9
Joined: Wed May 10, 2017 7:00 pm
Libera.chat IRC: si

Re: how to get device io port or PCI numbers

Post by zq »

Octocontrabass wrote:If the BIOS supports it, you can find that information using INT 0x13 AH=0x48. Here's a link to the specification.

If the BIOS doesn't support it, you have to enumerate all of the storage devices to find the boot disk.
Thanks for your help, I tried with qemu and unfourtuantly it does not support EDD, maybe I need to deal with the enumeration.
User avatar
Brendan
Member
Member
Posts: 8561
Joined: Sat Jan 15, 2005 12:00 am
Location: At his keyboard!
Contact:

Re: how to get device io port or PCI numbers

Post by Brendan »

Hi,
zq wrote:
Octocontrabass wrote:If the BIOS doesn't support it, you have to enumerate all of the storage devices to find the boot disk.
Thanks for your help, I tried with qemu and unfourtuantly it does not support EDD, maybe I need to deal with the enumeration.
In this case there's multiple scenarios where it's impossible for the OS to figure out what it booted from.

For a simple example, assume the user has eight USB flash devices plugged into a USB hub where all of the flash devices are identical clones and have a "write protect" switch set.

For a more bizarre example (that you can expect to fail even if EDD is fully supported), imagine if the user boots a hard disk image from network using Syslinux MEMDISK (which downloads a disk image into RAM, then hooks "int 0x13" to make the RAM look like an actual disk).


Cheers,

Brendan
For all things; perfection is, and will always remain, impossible to achieve in practice. However; by striving for perfection we create things that are as perfect as practically possible. Let the pursuit of perfection be our guide.
User avatar
BrightLight
Member
Member
Posts: 901
Joined: Sat Dec 27, 2014 9:11 am
Location: Maadi, Cairo, Egypt
Contact:

Re: how to get device io port or PCI numbers

Post by BrightLight »

Just to give a more complete answer, there are mainly two ways of detecting the boot drive. First, you can have a kernel parameter that instructs it which drive to use. Or next, you can scan for all devices and search for boot files or other "magic" combinations that mark your boot device.

Scanning for all devices is not that hard. One way or another, you will have to detect all drives anyway. So you can scan the PCI bus for IDE/AHCI controllers, and then scan each IDE/AHCI controller for ATA/SATA/ATAPI/SATAPI devices. The same goes for the USB; you'd find USB controllers, scan each controller for USB hubs, and scan each USB hub for mass storage devices.

For Syslinux Memdisk, you're sure that if it exists, it is already the boot device. To detect Syslinux Memdisk (as I do in my code), you use BIOS INT 0x13 function 0x48 to get drive size. Then use the BIOS E820 memory map, and scan for a memory range that is the same size as the boot device, and is marked as "reserved." The start of that memory area represents the start of the memory-mapped disk. This works for Syslinux 4.07 at least, I haven't tried other versions, but probably works too.
You know your OS is advanced when you stop using the Intel programming guide as a reference.
Post Reply