biossums.c

来自「xen虚拟机源代码安装包」· C语言 代码 · 共 283 行

C
283
字号
/* biossums.c  --- written by Eike W. for the Bochs BIOS *//* adapted for the LGPL'd VGABIOS by vruppert *//*  This library is free software; you can redistribute it and/or *  modify it under the terms of the GNU Lesser General Public *  License as published by the Free Software Foundation; either *  version 2 of the License, or (at your option) any later version. * *  This library 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 *  Lesser General Public License for more details. * *  You should have received a copy of the GNU Lesser General Public *  License along with this library; if not, write to the Free Software *  Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301 USA */#include <stdlib.h>#include <stdio.h>#include <string.h>typedef unsigned char byte;void check( int value, char* message );#define MAX_BIOS_DATA 0x10000long chksum_bios_get_offset( byte* data, long offset );byte chksum_bios_calc_value( byte* data, long offset );byte chksum_bios_get_value(  byte* data, long offset );void chksum_bios_set_value(  byte* data, long offset, byte value );#define PMID_LEN        20#define PMID_CHKSUM     19long chksum_pmid_get_offset( byte* data, long offset );byte chksum_pmid_calc_value( byte* data, long offset );byte chksum_pmid_get_value(  byte* data, long offset );void chksum_pmid_set_value(  byte* data, long offset, byte value );#define PCIR_LEN        24long chksum_pcir_get_offset( byte* data, long offset );byte bios_data[MAX_BIOS_DATA];long bios_len;int main(int argc, char* argv[]){  FILE* stream;  long  offset, tmp_offset, pcir_offset;  byte  bios_len_byte, cur_val = 0, new_val = 0;  int   hits, modified;  if (argc != 2) {    printf( "Error. Need a file-name as an argument.\n" );    exit( EXIT_FAILURE );  }  if ((stream = fopen(argv[1], "rb")) == NULL) {    printf("Error opening %s for reading.\n", argv[1]);    exit(EXIT_FAILURE);  }  memset(bios_data, 0, MAX_BIOS_DATA);  bios_len = fread(bios_data, 1, MAX_BIOS_DATA, stream);  if (bios_len > MAX_BIOS_DATA) {    printf("Error reading max. 65536 Bytes from %s.\n", argv[1]);    fclose(stream);    exit(EXIT_FAILURE);  }  fclose(stream);  modified = 0;  if (bios_len < 0x8000) {    bios_len = 0x8000;    modified = 1;  } else if ((bios_len & 0x1FF) != 0) {    bios_len = (bios_len + 0x200) & ~0x1FF;    modified = 1;  }  bios_len_byte = (byte)(bios_len / 512);  if (bios_len_byte != bios_data[2]) {    if (modified == 0) {      bios_len += 0x200;    }    bios_data[2] = (byte)(bios_len / 512);    modified = 1;  }  hits   = 0;  offset = 0L;  while( (tmp_offset = chksum_pmid_get_offset( bios_data, offset )) != -1L ) {    offset  = tmp_offset;    cur_val = chksum_pmid_get_value(  bios_data, offset );    new_val = chksum_pmid_calc_value( bios_data, offset );    printf( "\nPMID entry at: 0x%4lX\n", offset  );    printf( "Current checksum:     0x%02X\n",   cur_val );    printf( "Calculated checksum:  0x%02X  ",   new_val );    hits++;  }  if ((hits == 1) && (cur_val != new_val)) {    printf("Setting checksum.");    chksum_pmid_set_value( bios_data, offset, new_val );    if (modified == 0) {      bios_len += 0x200;      bios_data[2]++;    }    modified = 1;  }  if (hits >= 2) {    printf( "Multiple PMID entries! No checksum set." );  }  if (hits) {    printf("\n");  }  offset = 0L;  pcir_offset = chksum_pcir_get_offset( bios_data, offset );  if (pcir_offset != -1L) {    if (bios_data[pcir_offset + 16] != bios_data[2]) {      bios_data[pcir_offset + 16] = bios_data[2];      if (modified == 0) {        bios_len += 0x200;        bios_data[2]++;        bios_data[pcir_offset + 16]++;      }      modified = 1;    }  }  offset  = 0L;  do {    offset  = chksum_bios_get_offset(bios_data, offset);    cur_val = chksum_bios_get_value(bios_data, offset);    new_val = chksum_bios_calc_value(bios_data, offset);    if ((cur_val != new_val) && (modified == 0)) {      bios_len += 0x200;      bios_data[2]++;      if (pcir_offset != -1L) {        bios_data[pcir_offset + 16]++;      }      modified = 1;    } else {      printf("\nBios checksum at:   0x%4lX\n", offset);      printf("Current checksum:     0x%02X\n", cur_val);      printf("Calculated checksum:  0x%02X  ", new_val);      if (cur_val != new_val) {        printf("Setting checksum.");        chksum_bios_set_value(bios_data, offset, new_val);        cur_val = new_val;        modified = 1;      }      printf( "\n" );    }  } while (cur_val != new_val);  if (modified == 1) {    if ((stream = fopen( argv[1], "wb")) == NULL) {      printf("Error opening %s for writing.\n", argv[1]);      exit(EXIT_FAILURE);    }    if (fwrite(bios_data, 1, bios_len, stream) < bios_len) {      printf("Error writing %d KBytes to %s.\n", bios_len / 1024, argv[1]);      fclose(stream);      exit(EXIT_FAILURE);    }    fclose(stream);  }  return (EXIT_SUCCESS);}void check( int okay, char* message ) {  if( !okay ) {    printf( "\n\nError. %s.\n", message );    exit( EXIT_FAILURE );  }}long chksum_bios_get_offset( byte* data, long offset ) {  return (bios_len - 1);}byte chksum_bios_calc_value( byte* data, long offset ) {  int   i;  byte  sum;  sum = 0;  for( i = 0; i < offset; i++ ) {    sum = sum + *( data + i );  }  sum = -sum;          /* iso ensures -s + s == 0 on unsigned types */  return( sum );}byte chksum_bios_get_value( byte* data, long offset ) {  return( *( data + offset ) );}void chksum_bios_set_value( byte* data, long offset, byte value ) {  *( data + offset ) = value;}byte chksum_pmid_calc_value( byte* data, long offset ) {  int           i;  int           len;  byte sum;  len = PMID_LEN;  check((offset + len) <= (bios_len - 1), "PMID entry length out of bounds" );  sum = 0;  for( i = 0; i < len; i++ ) {    if( i != PMID_CHKSUM ) {      sum = sum + *( data + offset + i );    }  }  sum = -sum;  return( sum );}long chksum_pmid_get_offset( byte* data, long offset ) {  long result = -1L;  while ((offset + PMID_LEN) < (bios_len - 1)) {    offset = offset + 1;    if( *( data + offset + 0 ) == 'P' && \        *( data + offset + 1 ) == 'M' && \        *( data + offset + 2 ) == 'I' && \        *( data + offset + 3 ) == 'D' ) {      result = offset;      break;    }  }  return( result );}byte chksum_pmid_get_value( byte* data, long offset ) {  check((offset + PMID_CHKSUM) <= (bios_len - 1), "PMID checksum out of bounds" );  return(  *( data + offset + PMID_CHKSUM ) );}void chksum_pmid_set_value( byte* data, long offset, byte value ) {  check((offset + PMID_CHKSUM) <= (bios_len - 1), "PMID checksum out of bounds" );  *( data + offset + PMID_CHKSUM ) = value;}long chksum_pcir_get_offset( byte* data, long offset ) {  long result = -1L;  while ((offset + PCIR_LEN) < (bios_len - 1)) {    offset = offset + 1;    if( *( data + offset + 0 ) == 'P' && \        *( data + offset + 1 ) == 'C' && \        *( data + offset + 2 ) == 'I' && \        *( data + offset + 3 ) == 'R' ) {      result = offset;      break;    }  }  return( result );}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?