|  | /* common.c - miscellaneous shared variables and routines */ | 
|  | /* | 
|  | *  GRUB  --  GRand Unified Bootloader | 
|  | *  Copyright (C) 1999,2000,2001,2002,2004  Free Software Foundation, Inc. | 
|  | * | 
|  | *  This program is free software; you can redistribute it and/or modify | 
|  | *  it under the terms of the GNU General Public License as published by | 
|  | *  the Free Software Foundation; either version 2 of the License, or | 
|  | *  (at your option) any later version. | 
|  | * | 
|  | *  This program is distributed in the hope that it will be useful, | 
|  | *  but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
|  | *  GNU General Public License for more details. | 
|  | * | 
|  | *  You should have received a copy of the GNU General Public License | 
|  | *  along with this program; if not, write to the Free Software | 
|  | *  Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. | 
|  | */ | 
|  |  | 
|  | #include <shared.h> | 
|  |  | 
|  | #ifdef SUPPORT_DISKLESS | 
|  | # define GRUB	1 | 
|  | # include <etherboot.h> | 
|  | #endif | 
|  |  | 
|  | /* | 
|  | *  Shared BIOS/boot data. | 
|  | */ | 
|  |  | 
|  | struct multiboot_info mbi; | 
|  | unsigned long saved_drive; | 
|  | unsigned long saved_partition; | 
|  | unsigned long cdrom_drive; | 
|  | #ifndef STAGE1_5 | 
|  | unsigned long saved_mem_upper; | 
|  |  | 
|  | /* This saves the maximum size of extended memory (in KB).  */ | 
|  | unsigned long extended_memory; | 
|  | #endif | 
|  |  | 
|  | /* | 
|  | *  Error code stuff. | 
|  | */ | 
|  |  | 
|  | grub_error_t errnum = ERR_NONE; | 
|  |  | 
|  | #ifndef STAGE1_5 | 
|  |  | 
|  | char *err_list[] = | 
|  | { | 
|  | [ERR_NONE] = 0, | 
|  | [ERR_BAD_ARGUMENT] = "Invalid argument", | 
|  | [ERR_BAD_FILENAME] = | 
|  | "Filename must be either an absolute pathname or blocklist", | 
|  | [ERR_BAD_FILETYPE] = "Bad file or directory type", | 
|  | [ERR_BAD_GZIP_DATA] = "Bad or corrupt data while decompressing file", | 
|  | [ERR_BAD_GZIP_HEADER] = "Bad or incompatible header in compressed file", | 
|  | [ERR_BAD_PART_TABLE] = "Partition table invalid or corrupt", | 
|  | [ERR_BAD_VERSION] = "Mismatched or corrupt version of stage1/stage2", | 
|  | [ERR_BELOW_1MB] = "Loading below 1MB is not supported", | 
|  | [ERR_BOOT_COMMAND] = "Kernel must be loaded before booting", | 
|  | [ERR_BOOT_FAILURE] = "Unknown boot failure", | 
|  | [ERR_BOOT_FEATURES] = "Unsupported Multiboot features requested", | 
|  | [ERR_DEV_FORMAT] = "Unrecognized device string", | 
|  | [ERR_DEV_NEED_INIT] = "Device not initialized yet", | 
|  | [ERR_DEV_VALUES] = "Invalid device requested", | 
|  | [ERR_EXEC_FORMAT] = "Invalid or unsupported executable format", | 
|  | [ERR_FILELENGTH] = | 
|  | "Filesystem compatibility error, cannot read whole file", | 
|  | [ERR_FILE_NOT_FOUND] = "File not found", | 
|  | [ERR_FSYS_CORRUPT] = "Inconsistent filesystem structure", | 
|  | [ERR_FSYS_MOUNT] = "Cannot mount selected partition", | 
|  | [ERR_GEOM] = "Selected cylinder exceeds maximum supported by BIOS", | 
|  | [ERR_NEED_LX_KERNEL] = "Linux kernel must be loaded before initrd", | 
|  | [ERR_NEED_MB_KERNEL] = "Multiboot kernel must be loaded before modules", | 
|  | [ERR_NO_DISK] = "Selected disk does not exist", | 
|  | [ERR_NO_DISK_SPACE] = "No spare sectors on the disk", | 
|  | [ERR_NO_PART] = "No such partition", | 
|  | [ERR_NUMBER_OVERFLOW] = "Overflow while parsing number", | 
|  | [ERR_NUMBER_PARSING] = "Error while parsing number", | 
|  | [ERR_OUTSIDE_PART] = "Attempt to access block outside partition", | 
|  | [ERR_PRIVILEGED] = "Must be authenticated", | 
|  | [ERR_READ] = "Disk read error", | 
|  | [ERR_SYMLINK_LOOP] = "Too many symbolic links", | 
|  | [ERR_UNALIGNED] = "File is not sector aligned", | 
|  | [ERR_UNRECOGNIZED] = "Unrecognized command", | 
|  | [ERR_WONT_FIT] = "Selected item cannot fit into memory", | 
|  | [ERR_WRITE] = "Disk write error", | 
|  | }; | 
|  |  | 
|  |  | 
|  | /* static for BIOS memory map fakery */ | 
|  | static struct AddrRangeDesc fakemap[3] = | 
|  | { | 
|  | {20, 0, 0, MB_ARD_MEMORY}, | 
|  | {20, 0x100000, 0, MB_ARD_MEMORY}, | 
|  | {20, 0x1000000, 0, MB_ARD_MEMORY} | 
|  | }; | 
|  |  | 
|  | /* A big problem is that the memory areas aren't guaranteed to be: | 
|  | (1) contiguous, (2) sorted in ascending order, or (3) non-overlapping. | 
|  | Thus this kludge.  */ | 
|  | static unsigned long | 
|  | mmap_avail_at (unsigned long bottom) | 
|  | { | 
|  | unsigned long long top; | 
|  | unsigned long addr; | 
|  | int cont; | 
|  |  | 
|  | top = bottom; | 
|  | do | 
|  | { | 
|  | for (cont = 0, addr = mbi.mmap_addr; | 
|  | addr < mbi.mmap_addr + mbi.mmap_length; | 
|  | addr += *((unsigned long *) addr) + 4) | 
|  | { | 
|  | struct AddrRangeDesc *desc = (struct AddrRangeDesc *) addr; | 
|  |  | 
|  | if (desc->Type == MB_ARD_MEMORY | 
|  | && desc->BaseAddr <= top | 
|  | && desc->BaseAddr + desc->Length > top) | 
|  | { | 
|  | top = desc->BaseAddr + desc->Length; | 
|  | cont++; | 
|  | } | 
|  | } | 
|  | } | 
|  | while (cont); | 
|  |  | 
|  | /* For now, GRUB assumes 32bits addresses, so...  */ | 
|  | if (top > 0xFFFFFFFF) | 
|  | top = 0xFFFFFFFF; | 
|  |  | 
|  | return (unsigned long) top - bottom; | 
|  | } | 
|  | #endif /* ! STAGE1_5 */ | 
|  |  | 
|  | /* This queries for BIOS information.  */ | 
|  | void | 
|  | init_bios_info (void) | 
|  | { | 
|  | #ifndef STAGE1_5 | 
|  | unsigned long cont, memtmp, addr; | 
|  | int drive; | 
|  | #endif | 
|  |  | 
|  | /* | 
|  | *  Get information from BIOS on installed RAM. | 
|  | */ | 
|  |  | 
|  | mbi.mem_lower = get_memsize (0); | 
|  | mbi.mem_upper = get_memsize (1); | 
|  |  | 
|  | #ifndef STAGE1_5 | 
|  | /* | 
|  | *  We need to call this somewhere before trying to put data | 
|  | *  above 1 MB, since without calling it, address line 20 will be wired | 
|  | *  to 0.  Not too desirable. | 
|  | */ | 
|  |  | 
|  | gateA20 (1); | 
|  |  | 
|  | /* Store the size of extended memory in EXTENDED_MEMORY, in order to | 
|  | tell it to non-Multiboot OSes.  */ | 
|  | extended_memory = mbi.mem_upper; | 
|  |  | 
|  | /* | 
|  | *  The "mbi.mem_upper" variable only recognizes upper memory in the | 
|  | *  first memory region.  If there are multiple memory regions, | 
|  | *  the rest are reported to a Multiboot-compliant OS, but otherwise | 
|  | *  unused by GRUB. | 
|  | */ | 
|  |  | 
|  | addr = get_code_end (); | 
|  | mbi.mmap_addr = addr; | 
|  | mbi.mmap_length = 0; | 
|  | cont = 0; | 
|  |  | 
|  | do | 
|  | { | 
|  | cont = get_mmap_entry ((void *) addr, cont); | 
|  |  | 
|  | /* If the returned buffer's length is zero, quit. */ | 
|  | if (! *((unsigned long *) addr)) | 
|  | break; | 
|  |  | 
|  | mbi.mmap_length += *((unsigned long *) addr) + 4; | 
|  | addr += *((unsigned long *) addr) + 4; | 
|  | } | 
|  | while (cont); | 
|  |  | 
|  | if (mbi.mmap_length) | 
|  | { | 
|  | unsigned long long max_addr; | 
|  |  | 
|  | /* | 
|  | *  This is to get the lower memory, and upper memory (up to the | 
|  | *  first memory hole), into the "mbi.mem_{lower,upper}" | 
|  | *  elements.  This is for OS's that don't care about the memory | 
|  | *  map, but might care about total RAM available. | 
|  | */ | 
|  | mbi.mem_lower = mmap_avail_at (0) >> 10; | 
|  | mbi.mem_upper = mmap_avail_at (0x100000) >> 10; | 
|  |  | 
|  | /* Find the maximum available address. Ignore any memory holes.  */ | 
|  | for (max_addr = 0, addr = mbi.mmap_addr; | 
|  | addr < mbi.mmap_addr + mbi.mmap_length; | 
|  | addr += *((unsigned long *) addr) + 4) | 
|  | { | 
|  | struct AddrRangeDesc *desc = (struct AddrRangeDesc *) addr; | 
|  |  | 
|  | if (desc->Type == MB_ARD_MEMORY && desc->Length > 0 | 
|  | && desc->BaseAddr + desc->Length > max_addr) | 
|  | max_addr = desc->BaseAddr + desc->Length; | 
|  | } | 
|  |  | 
|  | extended_memory = (max_addr - 0x100000) >> 10; | 
|  | } | 
|  | else if ((memtmp = get_eisamemsize ()) != -1) | 
|  | { | 
|  | cont = memtmp & ~0xFFFF; | 
|  | memtmp = memtmp & 0xFFFF; | 
|  |  | 
|  | if (cont != 0) | 
|  | extended_memory = (cont >> 10) + 0x3c00; | 
|  | else | 
|  | extended_memory = memtmp; | 
|  |  | 
|  | if (!cont || (memtmp == 0x3c00)) | 
|  | memtmp += (cont >> 10); | 
|  | else | 
|  | { | 
|  | /* XXX should I do this at all ??? */ | 
|  |  | 
|  | mbi.mmap_addr = (unsigned long) fakemap; | 
|  | mbi.mmap_length = sizeof (fakemap); | 
|  | fakemap[0].Length = (mbi.mem_lower << 10); | 
|  | fakemap[1].Length = (memtmp << 10); | 
|  | fakemap[2].Length = cont; | 
|  | } | 
|  |  | 
|  | mbi.mem_upper = memtmp; | 
|  | } | 
|  |  | 
|  | saved_mem_upper = mbi.mem_upper; | 
|  |  | 
|  | /* Get the drive info.  */ | 
|  | /* FIXME: This should be postponed until a Multiboot kernel actually | 
|  | requires it, because this could slow down the start-up | 
|  | unreasonably.  */ | 
|  | mbi.drives_length = 0; | 
|  | mbi.drives_addr = addr; | 
|  |  | 
|  | /* For now, GRUB doesn't probe floppies, since it is trivial to map | 
|  | floppy drives to BIOS drives.  */ | 
|  | for (drive = 0x80; drive < 0x88; drive++) | 
|  | { | 
|  | struct geometry geom; | 
|  | struct drive_info *info = (struct drive_info *) addr; | 
|  | unsigned short *port; | 
|  |  | 
|  | /* Get the geometry. This ensures that the drive is present.  */ | 
|  | if (get_diskinfo (drive, &geom)) | 
|  | break; | 
|  |  | 
|  | /* Clean out the I/O map.  */ | 
|  | grub_memset ((char *) io_map, 0, | 
|  | IO_MAP_SIZE * sizeof (unsigned short)); | 
|  |  | 
|  | /* Disable to probe I/O ports temporarily, because this doesn't | 
|  | work with some BIOSes (maybe they are too buggy).  */ | 
|  | #if 0 | 
|  | /* Track the int13 handler.  */ | 
|  | track_int13 (drive); | 
|  | #endif | 
|  |  | 
|  | /* Set the information.  */ | 
|  | info->drive_number = drive; | 
|  | info->drive_mode = ((geom.flags & BIOSDISK_FLAG_LBA_EXTENSION) | 
|  | ? MB_DI_LBA_MODE : MB_DI_CHS_MODE); | 
|  | info->drive_cylinders = geom.cylinders; | 
|  | info->drive_heads = geom.heads; | 
|  | info->drive_sectors = geom.sectors; | 
|  |  | 
|  | addr += sizeof (struct drive_info); | 
|  | for (port = io_map; *port; port++, addr += sizeof (unsigned short)) | 
|  | *((unsigned short *) addr) = *port; | 
|  |  | 
|  | info->size = addr - (unsigned long) info; | 
|  | mbi.drives_length += info->size; | 
|  | } | 
|  |  | 
|  | /* Get the ROM configuration table by INT 15, AH=C0h.  */ | 
|  | mbi.config_table = get_rom_config_table (); | 
|  |  | 
|  | /* Set the boot loader name.  */ | 
|  | mbi.boot_loader_name = (unsigned long) "GNU GRUB " VERSION; | 
|  |  | 
|  | /* Get the APM BIOS table.  */ | 
|  | get_apm_info (); | 
|  | if (apm_bios_info.version) | 
|  | mbi.apm_table = (unsigned long) &apm_bios_info; | 
|  |  | 
|  | /* | 
|  | *  Initialize other Multiboot Info flags. | 
|  | */ | 
|  |  | 
|  | mbi.flags = (MB_INFO_MEMORY | MB_INFO_CMDLINE | MB_INFO_BOOTDEV | 
|  | | MB_INFO_DRIVE_INFO | MB_INFO_CONFIG_TABLE | 
|  | | MB_INFO_BOOT_LOADER_NAME); | 
|  |  | 
|  | if (apm_bios_info.version) | 
|  | mbi.flags |= MB_INFO_APM_TABLE; | 
|  |  | 
|  | #endif /* STAGE1_5 */ | 
|  |  | 
|  | /* Set boot drive and partition.  */ | 
|  | saved_drive = boot_drive; | 
|  | saved_partition = install_partition; | 
|  |  | 
|  | /* Set cdrom drive.  */ | 
|  | { | 
|  | struct geometry geom; | 
|  |  | 
|  | /* Get the geometry.  */ | 
|  | if (get_diskinfo (boot_drive, &geom) | 
|  | || ! (geom.flags & BIOSDISK_FLAG_CDROM)) | 
|  | cdrom_drive = GRUB_INVALID_DRIVE; | 
|  | else | 
|  | cdrom_drive = boot_drive; | 
|  | } | 
|  |  | 
|  | /* Start main routine here.  */ | 
|  | cmain (); | 
|  | } |