blob: 2e1eb4b5421b0003f21e8c7632a41971e9ba17ff [file] [log] [blame]
/*
* The PCI Library -- Reading of Bus Dumps
*
* Copyright (c) 1997--2026 Martin Mares <mj@ucw.cz>
*
* Can be freely distributed and used under the terms of the GNU GPL v2+.
*
* SPDX-License-Identifier: GPL-2.0-or-later
*/
#include <stdio.h>
#include <ctype.h>
#include <string.h>
#include <errno.h>
#include "internal.h"
struct dump_data {
int len, allocated;
byte data[1];
};
static void
dump_config(struct pci_access *a)
{
pci_define_param(a, "dump.name", "", "Name of the bus dump file to read from");
}
static int
dump_detect(struct pci_access *a)
{
char *name = pci_get_param(a, "dump.name");
return name && name[0];
}
static void
dump_alloc_data(struct pci_dev *dev, int len)
{
struct dump_data *dd = pci_malloc(dev->access, sizeof(struct dump_data) + len - 1);
dd->allocated = len;
dd->len = 0;
memset(dd->data, 0xff, len);
dev->backend_data = dd;
}
static int
dump_validate(char *s, char *fmt)
{
while (*fmt)
{
if (*fmt == '#' ? !isxdigit(*s) : *fmt != *s)
return 0;
fmt++, s++;
}
return 1;
}
static const char *
dump_parse(struct pci_access *a, FILE *f)
{
char buf[256];
struct pci_dev *dev = NULL;
int len, mn, bn, dn, fn, i, j;
while (fgets(buf, sizeof(buf)-1, f))
{
char *z = strchr(buf, '\n');
if (!z)
return "Line too long or unterminated";
*z-- = 0;
if (z >= buf && *z == '\r')
*z-- = 0;
len = z - buf + 1;
mn = 0;
if (dump_validate(buf, "##:##.# ") && sscanf(buf, "%x:%x.%d", &bn, &dn, &fn) == 3 ||
dump_validate(buf, "####:##:##.# ") && sscanf(buf, "%x:%x:%x.%d", &mn, &bn, &dn, &fn) == 4 ||
dump_validate(buf, "#####:##:##.# ") && sscanf(buf, "%x:%x:%x.%d", &mn, &bn, &dn, &fn) == 4 ||
dump_validate(buf, "######:##:##.# ") && sscanf(buf, "%x:%x:%x.%d", &mn, &bn, &dn, &fn) == 4)
{
dev = pci_get_dev(a, mn, bn, dn, fn);
dump_alloc_data(dev, 256);
pci_link_dev(a, dev);
}
else if (!len)
dev = NULL;
else if (dev &&
(dump_validate(buf, "##: ") || dump_validate(buf, "###: ") || dump_validate(buf, "####: ") ||
dump_validate(buf, "#####: ") || dump_validate(buf, "######: ") ||
dump_validate(buf, "#######: ") || dump_validate(buf, "########: ")) &&
sscanf(buf, "%x: ", &i) == 1)
{
struct dump_data *dd = dev->backend_data;
z = strchr(buf, ' ') + 1;
while (isxdigit(z[0]) && isxdigit(z[1]) && (!z[2] || z[2] == ' ') &&
sscanf(z, "%x", &j) == 1 && j < 256)
{
if (i >= 4096)
return "At most 4096 bytes of config space are supported";
if (i >= dd->allocated) /* Need to re-allocate the buffer */
{
dump_alloc_data(dev, 4096);
memcpy(((struct dump_data *) dev->backend_data)->data, dd->data, 256);
pci_mfree(dd);
dd = dev->backend_data;
}
dd->data[i++] = j;
if (i > dd->len)
dd->len = i;
z += 2;
if (*z)
z++;
}
if (*z)
return "Malformed line";
}
}
return NULL;
}
static void
dump_init(struct pci_access *a)
{
char *name = pci_get_param(a, "dump.name");
const char *err;
if (!name)
a->error("dump: File name not given.");
if (!strcmp(name, "-"))
err = dump_parse(a, stdin);
else
{
FILE *f = fopen(name, "r");
if (!f)
a->error("dump: Cannot open %s: %s", name, strerror(errno));
err = dump_parse(a, f);
fclose(f);
}
if (err)
a->error("dump: %s", err);
}
static void
dump_cleanup(struct pci_access *a UNUSED)
{
}
static void
dump_scan(struct pci_access *a UNUSED)
{
}
static int
dump_read(struct pci_dev *d, int pos, byte *buf, int len)
{
struct dump_data *dd;
if (!(dd = d->backend_data))
{
struct pci_dev *e = d->access->devices;
while (e && (e->domain != d->domain || e->bus != d->bus || e->dev != d->dev || e->func != d->func))
e = e->next;
if (!e)
return 0;
dd = e->backend_data;
}
if (pos + len > dd->len)
return 0;
memcpy(buf, dd->data + pos, len);
return 1;
}
static int
dump_write(struct pci_dev *d UNUSED, int pos UNUSED, byte *buf UNUSED, int len UNUSED)
{
d->access->error("Writing to dump files is not supported.");
return 0;
}
static void
dump_cleanup_dev(struct pci_dev *d)
{
if (d->backend_data)
{
pci_mfree(d->backend_data);
d->backend_data = NULL;
}
}
struct pci_methods pm_dump = {
.name = "dump",
.help = "Reading of register dumps (set the `dump.name' parameter)",
.config = dump_config,
.detect = dump_detect,
.init = dump_init,
.cleanup = dump_cleanup,
.scan = dump_scan,
.fill_info = pci_generic_fill_info,
.read = dump_read,
.write = dump_write,
.cleanup_dev = dump_cleanup_dev,
};