diff options
Diffstat (limited to 'biosdecode.c')
-rw-r--r-- | biosdecode.c | 666 |
1 files changed, 0 insertions, 666 deletions
diff --git a/biosdecode.c b/biosdecode.c deleted file mode 100644 index 354234e..0000000 --- a/biosdecode.c +++ /dev/null @@ -1,666 +0,0 @@ -/* - * BIOS Decode - * - * (C) 2000-2002 Alan Cox <alan@redhat.com> - * (C) 2002-2007 Jean Delvare <khali@linux-fr.org> - * - * 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA - * - * For the avoidance of doubt the "preferred form" of this code is one which - * is in an open unpatent encumbered format. Where cryptographic key signing - * forms part of the process of creating an executable the information - * including keys needed to generate an equivalently functional executable - * are deemed to be part of the source code. - * - * References: - * - DMTF "System Management BIOS Reference Specification" - * Version 2.3.4 - * http://www.dmtf.org/standards/smbios - * - Intel "Preboot Execution Environment (PXE) Specification" - * Version 2.1 - * http://www.intel.com/labs/manage/wfm/wfmspecs.htm - * - ACPI "Advanced Configuration and Power Interface Specification" - * Revision 2.0 - * http://www.acpi.info/spec20.htm - * - Phoenix "BIOS32 Service Directory" - * Revision 0.4 - * http://www.phoenix.com/en/support/white+papers-specs/ - * - Microsoft "Plug and Play BIOS Specification" - * Version 1.0A - * http://www.microsoft.com/hwdev/tech/PnP/ - * - Microsoft "PCI IRQ Routing Table Specification" - * Version 1.0 - * http://www.microsoft.com/hwdev/archive/BUSBIOS/pciirq.asp - * - Compaq "Technical Reference Guide for Compaq Deskpro 4000 and 6000" - * First Edition - * http://h18000.www1.hp.com/support/techpubs/technical_reference_guides/113a1097.html - * - IBM "Using the BIOS Build ID to identify Thinkpad systems" - * Revision 2005-09-19 - * http://www-307.ibm.com/pc/support/site.wss/MIGR-45120.html - * - Fujitsu application panel technical details - * As of July 23rd, 2004 - * http://apanel.sourceforge.net/tech.php - */ - -#include <stdio.h> -#include <string.h> -#include <stdlib.h> -#include <unistd.h> -#include <getopt.h> - -#include "version.h" -#include "config.h" -#include "types.h" -#include "util.h" - -/* Options are global */ -struct opt -{ - const char* devmem; - unsigned int flags; -}; -static struct opt opt; - -#define FLAG_VERSION (1<<0) -#define FLAG_HELP (1<<1) - -struct bios_entry { - const char *anchor; - size_t anchor_len; /* computed */ - off_t low_address; - off_t high_address; - size_t (*length)(const u8 *); - int (*decode)(const u8*, size_t); -}; - - -/* - * SMBIOS - */ - -static size_t smbios_length(const u8 *p) -{ - return(p[0x05]==0x1E?0x1F:p[0x05]); -} - -static int smbios_decode(const u8 *p, size_t len) -{ - if(len<0x1F || !checksum(p, p[0x05]) - || memcmp("_DMI_", p+0x10, 5)!=0 - || !checksum(p+0x10, 0x0F)) - return 0; - - printf("SMBIOS %u.%u present.\n", - p[0x06], p[0x07]); - printf("\tStructure Table Length: %u bytes\n", - WORD(p+0x16)); - printf("\tStructure Table Address: 0x%08X\n", - DWORD(p+0x18)); - printf("\tNumber Of Structures: %u\n", - WORD(p+0x1C)); - printf("\tMaximum Structure Size: %u bytes\n", - WORD(p+0x08)); - - return 1; -} - -static size_t dmi_length(const u8 *p) -{ - (void) p; - - return(0x0F); -} - -static int dmi_decode(const u8 *p, size_t len) -{ - if(len<0x0F || !checksum(p, len)) - return 0; - - printf("Legacy DMI %u.%u present.\n", - p[0x0E]>>4, p[0x0E]&0x0F); - printf("\tStructure Table Length: %u bytes\n", - WORD(p+0x06)); - printf("\tStructure Table Address: 0x%08X\n", - DWORD(p+0x08)); - printf("\tNumber Of Structures: %u\n", - WORD(p+0x0C)); - - return 1; -} - -/* - * SYSID - */ - -static size_t sysid_length(const u8 *p) -{ - return WORD(p+0x08); -} - -static int sysid_decode(const u8 *p, size_t len) -{ - if(len<0x11 || !checksum(p, WORD(p+0x08))) - return 0; - - printf("SYSID present.\n"); - printf("\tRevision: %u\n", - p[0x10]); - printf("\tStructure Table Address: 0x%08X\n", - DWORD(p+0x0A)); - printf("\tNumber Of Structures: %u\n", - WORD(p+0x0E)); - - return 1; -} - -/* - * PnP - */ - -static size_t pnp_length(const u8 *p) -{ - return(p[0x05]); -} - -static const char *pnp_event_notification(u8 code) -{ - static const char *notification[]={ - "Not Supported", /* 0x0 */ - "Polling", - "Asynchronous", - "Unknown" /* 0x3 */ - }; - - return notification[code]; -} - -static int pnp_decode(const u8 *p, size_t len) -{ - if(len<0x21 || !checksum(p, p[0x05])) - return 0; - - printf("PNP BIOS %u.%u present.\n", - p[0x04]>>4, p[0x04]&0x0F); - printf("\tEvent Notification: %s\n", - pnp_event_notification(WORD(p+0x06)&0x03)); - if((WORD(p+0x06)&0x03)==0x01) - printf("\tEvent Notification Flag Address: 0x%08X\n", - DWORD(p+0x09)); - printf("\tReal Mode 16-bit Code Address: %04X:%04X\n", - WORD(p+0x0F), WORD(p+0x0D)); - printf("\tReal Mode 16-bit Data Address: %04X:0000\n", - WORD(p+0x1B)); - printf("\t16-bit Protected Mode Code Address: 0x%08X\n", - DWORD(p+0x13)+WORD(p+0x11)); - printf("\t16-bit Protected Mode Data Address: 0x%08X\n", - DWORD(p+0x1D)); - if(DWORD(p+0x17)!=0) - printf("\tOEM Device Identifier: %c%c%c%02X%02X\n", - 0x40+((p[0x17]>>2)&0x1F), - 0x40+((p[0x17]&0x03)<<3)+((p[0x18]>>5)&0x07), - 0x40+(p[0x18]&0x1F), p[0x19], p[0x20]); - - return 1; -} - -/* - * ACPI - */ - -static size_t acpi_length(const u8 *p) -{ - return(p[15]==2?36:20); -} - -static const char *acpi_revision(u8 code) -{ - switch(code) - { - case 0: - return " 1.0"; - case 2: - return " 2.0"; - default: - return ""; - } -} - -static int acpi_decode(const u8 *p, size_t len) -{ - if(len<20 || !checksum(p, 20)) - return 0; - - printf("ACPI%s present.\n", - acpi_revision(p[15])); - printf("\tOEM Identifier: %c%c%c%c%c%c\n", - p[9], p[10], p[11], p[12], p[13], p[14]); - printf("\tRSD Table 32-bit Address: 0x%08X\n", - DWORD(p+16)); - - if(len<36) - return 1; - - if(DWORD(p+20)>len || !checksum(p, DWORD(p+20))) - return 0; - - if(DWORD(p+20)<32) return 1; - - printf("\tXSD Table 64-bit Address: 0x%08X%08X\n", - QWORD(p+24).h, QWORD(p+24).l); - - return 1; -} - -/* - * Sony - */ - -static size_t sony_length(const u8 *p) -{ - return(p[0x05]); -} - -static int sony_decode(const u8 *p, size_t len) -{ - if(!checksum(p, len)) - return 0; - - printf("Sony system detected.\n"); - - return 1; -} - -/* - * BIOS32 - */ - -static size_t bios32_length(const u8 *p) -{ - return p[0x09]<<4; -} - -static int bios32_decode(const u8 *p, size_t len) -{ - if(len<0x0A || !checksum(p, p[0x09]<<4)) - return 0; - - printf("BIOS32 Service Directory present.\n"); - printf("\tRevision: %u\n", - p[0x08]); - printf("\tCalling Interface Address: 0x%08X\n", - DWORD(p+0x04)); - - return 1; -} - -/* - * PIR - */ - -static void pir_irqs(u16 code) -{ - if(code==0) - printf(" None"); - else - { - u8 i; - - for(i=0; i<16; i++) - if(code&(1<<i)) - printf(" %u", i); - } -} - -static void pir_slot_number(u8 code) -{ - if(code==0) - printf(" on-board"); - else - printf(" slot number %u", code); -} - -static size_t pir_length(const u8 *p) -{ - return WORD(p+6); -} - -static int pir_decode(const u8 *p, size_t len) -{ - int i; - - if(len<32 || !checksum(p, WORD(p+6))) - return 0; - - printf("PCI Interrupt Routing %u.%u present.\n", - p[5], p[4]); - printf("\tRouter ID: %02x:%02x.%1x\n", - p[8], p[9]>>3, p[9]&0x07); - printf("\tExclusive IRQs:"); - pir_irqs(WORD(p+10)); - printf("\n"); - if(DWORD(p+12)!=0) - printf("\tCompatible Router: %04x:%04x\n", - WORD(p+12), WORD(p+14)); - if(DWORD(p+16)!=0) - printf("\tMiniport Data: 0x%08X\n", - DWORD(p+16)); - - for(i=1; i<=(WORD(p+6)-32)/16; i++) - { - printf("\tSlot Entry %u: ID %02x:%02x,", - i, p[(i+1)*16], p[(i+1)*16+1]>>3); - pir_slot_number(p[(i+1)*16+14]); - printf("\n"); -/* printf("\tSlot Entry %u\n", i); - printf("\t\tID: %02x:%02x\n", - p[(i+1)*16], p[(i+1)*16+1]>>3); - printf("\t\tLink Value for INTA#: %u\n", - p[(i+1)*16+2]); - printf("\t\tIRQ Bitmap for INTA#:"); - pir_irqs(WORD(p+(i+1)*16+3)); - printf("\n"); - printf("\t\tLink Value for INTB#: %u\n", - p[(i+1)*16+5]); - printf("\t\tIRQ Bitmap for INTB#:"); - pir_irqs(WORD(p+(i+1)*16+6)); - printf("\n"); - printf("\t\tLink Value for INTC#: %u\n", - p[(i+1)*16+8]); - printf("\t\tIRQ Bitmap for INTC#:"); - pir_irqs(WORD(p+(i+1)*16+9)); - printf("\n"); - printf("\t\tLink Value for INTD#: %u\n", - p[(i+1)*16+11]); - printf("\t\tIRQ Bitmap for INTD#:"); - pir_irqs(WORD(p+(i+1)*16+12)); - printf("\n"); - printf("\t\tSlot Number:"); - pir_slot_number(p[(i+1)*16+14]); - printf("\n");*/ - } - - return 1; -} - -/* - * Compaq-specific entries - */ - -static size_t compaq_length(const u8 *p) -{ - return (p[4]*10+5); -} - -static int compaq_decode(const u8 *p, size_t len) -{ - unsigned int i; - (void) len; - - printf("Compaq-specific entries present.\n"); - - /* integrity checking (lack of checksum) */ - for(i=0; i<p[4]; i++) - { - /* - * We do not check for truncated entries, because the length - * was computed from the number of records in compaq_length - * right above, so it can't be wrong. - */ - if(p[5+i*10]!='$' || !(p[6+i*10]>='A' && p[6+i*10]<='Z') - || !(p[7+i*10]>='A' && p[7+i*10]<='Z') - || !(p[8+i*10]>='A' && p[8+i*10]<='Z')) - { - printf("\t Abnormal entry! Please report. [%02X %02X " - "%02X %02X]\n", p[5+i*10], p[6+i*10], - p[7+i*10], p[8+i*10]); - return 0; - } - } - - for(i=0; i<p[4]; i++) - { - printf("\tEntry %u: %c%c%c%c at 0x%08X (%u bytes)\n", - i+1, p[5+i*10], p[6+i*10], p[7+i*10], p[8+i*10], - DWORD(p+9+i*10), WORD(p+13+i*10)); - } - - return 1; -} - -/* - * VPD (vital product data, IBM-specific) - */ - -static void vpd_print_entry(const char *name, const u8 *p, size_t len) -{ - size_t i; - - printf("\t%s: ", name); - for(i=0; i<len; i++) - if(p[i]>=32 && p[i]<127) - printf("%c", p[i]); - printf("\n"); -} - -static size_t vpd_length(const u8 *p) -{ - return (p[5]); -} - -static int vpd_decode(const u8 *p, size_t len) -{ - if(len<0x30) - return 0; - - /* XSeries have longer records. */ - if(!(len>=0x45 && checksum(p, len)) - /* Some Netvista seem to work with this. */ - && !checksum(p, 0x30) - /* The Thinkpad checksum does *not* include the first 13 bytes. */ - && !checksum(p+0x0D, 0x30-0x0D)) - return 0; - - printf("VPD present.\n"); - - vpd_print_entry("BIOS Build ID", p+0x0D, 9); - vpd_print_entry("Box Serial Number", p+0x16, 7); - vpd_print_entry("Motherboard Serial Number", p+0x1D, 11); - vpd_print_entry("Machine Type/Model", p+0x28, 7); - - if(len<0x45) - return 1; - - vpd_print_entry("BIOS Release Date", p+0x30, 8); - - return 1; -} - -/* - * Fujitsu application panel - */ - -static size_t fjkeyinf_length(const u8 *p) -{ - (void) p; - /* - * We don't know at this point, it's somewhere between 12 and 32. - * So we return the max, it shouldn't hurt. - */ - return 32; -} - -static int fjkeyinf_decode(const u8 *p, size_t len) -{ - (void) len; - int i; - - printf("Fujitsu application panel present.\n"); - - for (i = 0; i < 6; i++) - { - if (*(p+8+i*4)==0) - return 1; - printf("\tDevice %d: type %u, chip %u", i+1, - *(p+8+i*4), *(p+8+i*4+2)); - if (*(p+8+i*4+1)) /* Access method */ - printf(", SMBus address 0x%x", *(p+8+i*4+3) >> 1); - printf("\n"); - } - - return 1; -} - -/* - * Main - */ - -static struct bios_entry bios_entries[]={ - { "_SM_", 0, 0xF0000, 0xFFFFF, smbios_length, smbios_decode }, - { "_DMI_", 0, 0xF0000, 0xFFFFF, dmi_length, dmi_decode }, - { "_SYSID_", 0, 0xE0000, 0xFFFFF, sysid_length, sysid_decode }, - { "$PnP", 0, 0xF0000, 0xFFFFF, pnp_length, pnp_decode }, - { "RSD PTR ", 0, 0xE0000, 0xFFFFF, acpi_length, acpi_decode }, - { "$SNY", 0, 0xE0000, 0xFFFFF, sony_length, sony_decode }, - { "_32_", 0, 0xE0000, 0xFFFFF, bios32_length, bios32_decode }, - { "$PIR", 0, 0xF0000, 0xFFFFF, pir_length, pir_decode }, - { "32OS", 0, 0xE0000, 0xFFFFF, compaq_length, compaq_decode }, - { "\252\125VPD", 0, 0xF0000, 0xFFFFF, vpd_length, vpd_decode }, - { "FJKEYINF", 0, 0xF0000, 0xFFFFF, fjkeyinf_length, fjkeyinf_decode }, - { NULL, 0, 0, 0, NULL, NULL } -}; - -/* Believe it or not, this is significantly faster than memcmp and strncmp */ -static inline int anchor_match(const struct bios_entry *entry, const char *p) -{ - size_t i; - - for(i=0; i<entry->anchor_len; i++) - if(entry->anchor[i]!=p[i]) - return 0; - - return 1; -} - -/* Return -1 on error, 0 on success */ -static int parse_command_line(int argc, char * const argv[]) -{ - int option; - const char *optstring = "d:hV"; - struct option longopts[]={ - { "dev-mem", required_argument, NULL, 'd' }, - { "help", no_argument, NULL, 'h' }, - { "version", no_argument, NULL, 'V' }, - { 0, 0, 0, 0 } - }; - - while((option=getopt_long(argc, argv, optstring, longopts, NULL))!=-1) - switch(option) - { - case 'd': - opt.devmem=optarg; - break; - case 'h': - opt.flags|=FLAG_HELP; - break; - case 'V': - opt.flags|=FLAG_VERSION; - break; - case '?': - return -1; - } - - return 0; -} - -static void print_help(void) -{ - static const char *help= - "Usage: biosdecode [OPTIONS]\n" - "Options are:\n" - " -d, --dev-mem FILE Read memory from device FILE (default: " DEFAULT_MEM_DEV ")\n" - " -h, --help Display this help text and exit\n" - " -V, --version Display the version and exit\n"; - - printf("%s", help); -} - -int main(int argc, char * const argv[]) -{ - u8 *buf; - off_t fp; - int i; - - if(sizeof(u8)!=1 || sizeof(u16)!=2 || sizeof(u32)!=4) - { - fprintf(stderr, "%s: compiler incompatibility\n", argv[0]); - exit(255); - } - - /* Set default option values */ - opt.devmem=DEFAULT_MEM_DEV; - opt.flags=0; - - if(parse_command_line(argc, argv)<0) - exit(2); - - if(opt.flags & FLAG_HELP) - { - print_help(); - return 0; - } - - if(opt.flags & FLAG_VERSION) - { - printf("%s\n", VERSION); - return 0; - } - - printf("# biosdecode %s\n", VERSION); - - if((buf=mem_chunk(0xE0000, 0x20000, opt.devmem))==NULL) - exit(1); - - /* Compute anchor lengths once and for all */ - for(i=0; bios_entries[i].anchor!=NULL; i++) - bios_entries[i].anchor_len = strlen(bios_entries[i].anchor); - - for(fp=0xE0000; fp<=0xFFFF0; fp+=16) - { - u8 *p=buf+fp-0xE0000; - - for(i=0; bios_entries[i].anchor!=NULL; i++) - { - if(anchor_match(&bios_entries[i], (char *)p) - && fp>=bios_entries[i].low_address - && fp<bios_entries[i].high_address) - { - off_t len=bios_entries[i].length(p); - - if(fp+len-1<=bios_entries[i].high_address) - { - if(bios_entries[i].decode(p, len)) - { - fp+=(((len-1)>>4)<<4); - break; - } - } - } - } - } - - free(buf); - - return 0; -} |