153 lines
3.4 KiB
C
153 lines
3.4 KiB
C
/*
|
|
* Tool for reading QEMU fwcfg values.
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <stddef.h>
|
|
#include <stdint.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <getopt.h>
|
|
|
|
#define FW_CFG_PORT_OUT 0x510
|
|
#define FW_CFG_PORT_IN 0x511
|
|
#define FW_CFG_SELECT_QEMU 0x0000
|
|
#define FW_CFG_SELECT_LIST 0x0019
|
|
|
|
|
|
/* outw / inb helper functions */
|
|
static void outports(unsigned short _port, unsigned short _data) {
|
|
asm volatile ("outw %1, %0" : : "dN" (_port), "a" (_data));
|
|
}
|
|
|
|
static unsigned char inportb(unsigned short _port) {
|
|
unsigned char rv;
|
|
asm volatile ("inb %1, %0" : "=a" (rv) : "dN" (_port));
|
|
return rv;
|
|
}
|
|
|
|
/* Despite primarily emulating x86, these are all big-endian */
|
|
static void swap_bytes(void * in, int count) {
|
|
char * bytes = in;
|
|
if (count == 4) {
|
|
uint32_t * t = in;
|
|
*t = (bytes[0] << 24) | (bytes[1] << 12) | (bytes[2] << 8) | bytes[3];
|
|
} else if (count == 2) {
|
|
uint16_t * t = in;
|
|
*t = (bytes[0] << 8) | bytes[1];
|
|
}
|
|
}
|
|
|
|
/* Layout of the information returned from the fw_cfg port */
|
|
struct fw_cfg_file {
|
|
uint32_t size;
|
|
uint16_t select;
|
|
uint16_t reserved;
|
|
char name[56];
|
|
};
|
|
|
|
static int usage(char * argv[]) {
|
|
printf(
|
|
"Obtain QEMU fw_cfg values\n"
|
|
"\n"
|
|
"usage: %s [-?ln] [config name]\n"
|
|
"\n"
|
|
" -l \033[3mlist available config entries\033[0m\n"
|
|
" -n \033[3mdon't print a new line after image\033[0m\n"
|
|
" -? \033[3mshow this help text\033[0m\n"
|
|
"\n", argv[0]);
|
|
return 1;
|
|
}
|
|
|
|
int main(int argc, char * argv[]) {
|
|
|
|
uint32_t count = 0;
|
|
uint8_t * bytes = (uint8_t *)&count;
|
|
int found = 0;
|
|
struct fw_cfg_file file;
|
|
uint8_t * tmp = (uint8_t *)&file;
|
|
|
|
int opt = 0;
|
|
int list = 0;
|
|
int no_newline = 0;
|
|
|
|
while ((opt = getopt(argc, argv, "?ln")) != -1) {
|
|
switch (opt) {
|
|
case '?':
|
|
return usage(argv);
|
|
case 'n':
|
|
no_newline = 1;
|
|
break;
|
|
case 'l':
|
|
list = 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (optind >= argc && !list) {
|
|
return usage(argv);
|
|
}
|
|
|
|
/* First check for QEMU */
|
|
outports(FW_CFG_PORT_OUT, FW_CFG_SELECT_QEMU);
|
|
if (inportb(FW_CFG_PORT_IN) != 'Q' ||
|
|
inportb(FW_CFG_PORT_IN) != 'E' ||
|
|
inportb(FW_CFG_PORT_IN) != 'M' ||
|
|
inportb(FW_CFG_PORT_IN) != 'U') {
|
|
fprintf(stderr, "%s: this doesn't seem to be qemu\n", argv[0]);
|
|
return 1;
|
|
}
|
|
|
|
/* Then get the list of "files" so we can look at names */
|
|
outports(FW_CFG_PORT_OUT, FW_CFG_SELECT_LIST);
|
|
for (int i = 0; i < 4; ++i) {
|
|
bytes[i] = inportb(FW_CFG_PORT_IN);
|
|
}
|
|
swap_bytes(&count, sizeof(count));
|
|
|
|
for (unsigned int i = 0; i < count; ++i) {
|
|
|
|
/* read one file entry */
|
|
for (unsigned int j = 0; j < sizeof(struct fw_cfg_file); ++j) {
|
|
tmp[j] = inportb(FW_CFG_PORT_IN);
|
|
}
|
|
|
|
/* endian swap to get file size and selector ID */
|
|
swap_bytes(&file.size, sizeof(file.size));
|
|
swap_bytes(&file.select, sizeof(file.select));
|
|
|
|
if (list) {
|
|
/* 0x0020 org/whatever (1234 bytes) */
|
|
fprintf(stdout, "0x%04x %s (%d byte%s)\n", file.select, file.name, (int)file.size, file.size == 1 ? "" : "s");
|
|
} else {
|
|
if (!strcmp(file.name, argv[optind])) {
|
|
/* found the requested file */
|
|
found = 1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (found) {
|
|
/* if we found the requested file, read it from the port */
|
|
outports(FW_CFG_PORT_OUT, file.select);
|
|
|
|
for (unsigned int i = 0; i < file.size; ++i) {
|
|
fputc(inportb(FW_CFG_PORT_IN), stdout);
|
|
}
|
|
|
|
if (!no_newline) {
|
|
fprintf(stdout, "\n");
|
|
} else {
|
|
fflush(stdout);
|
|
}
|
|
|
|
} else if (!list) {
|
|
fprintf(stderr, "%s: config option not found\n", argv[0]);
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|