// PNP BIOS calls // // Copyright (C) 2008 Kevin O'Connor // // This file may be distributed under the terms of the GNU LGPLv3 license. #include "config.h" // BUILD_BIOS_ADDR #include "farptr.h" // SET_FARVAR #include "output.h" // dprintf #include "std/pnpbios.h" // PNP_SIGNATURE #include "string.h" // checksum #include "util.h" // pnp_init extern struct pnpheader PNPHEADER; extern char pnp_string[]; #if CONFIG_PNPBIOS struct pnpheader PNPHEADER __aligned(16) VARFSEG = { .signature = PNP_SIGNATURE, .version = 0x10, .length = sizeof(PNPHEADER), .real_cs = SEG_BIOS, .prot_base = BUILD_BIOS_ADDR, .real_ds = SEG_BIOS, .prot_database = BUILD_BIOS_ADDR, }; #else // We need a copy of this string in the 0xf000 segment, but we are not // actually a PnP BIOS, so make sure it is *not* aligned, so OSes will // not see it if they scan. char pnp_string[] __aligned(2) VARFSEG = " $PnP"; #endif // BBS - Get Version and Installation Check static u16 handle_pnp60(u16 *args) { u16 version_ptr = args[1]; u16 version_seg = args[2]; SET_FARVAR(version_seg, *(u16*)(version_ptr+0), 0x0101); return 0; } static u16 handle_pnpXX(u16 *args) { return FUNCTION_NOT_SUPPORTED; } u16 VISIBLE16 handle_pnp(u16 *args) { if (! CONFIG_PNPBIOS) return FUNCTION_NOT_SUPPORTED; u16 arg1 = args[0]; dprintf(DEBUG_HDL_pnp, "pnp call arg1=%x\n", arg1); switch (arg1) { case 0x60: return handle_pnp60(args); default: return handle_pnpXX(args); } } u16 get_pnp_offset(void) { if (! CONFIG_PNPBIOS) return (u32)pnp_string + 1 - BUILD_BIOS_ADDR; return (u32)&PNPHEADER - BUILD_BIOS_ADDR; } // romlayout.S extern void entry_pnp_real(void); extern void entry_pnp_prot(void); void pnp_init(void) { if (! CONFIG_PNPBIOS) return; dprintf(3, "init PNPBIOS table\n"); PNPHEADER.real_ip = (u32)entry_pnp_real - BUILD_BIOS_ADDR; PNPHEADER.prot_ip = (u32)entry_pnp_prot - BUILD_BIOS_ADDR; PNPHEADER.checksum -= checksum(&PNPHEADER, sizeof(PNPHEADER)); }