xref: /pciutils/lib/dump.c (revision b59b41bf)
1727ce158SMartin Mares /*
2727ce158SMartin Mares  *	The PCI Library -- Reading of Bus Dumps
3727ce158SMartin Mares  *
4cb6ee324SMartin Mares  *	Copyright (c) 1997--2008 Martin Mares <[email protected]>
5727ce158SMartin Mares  *
661829219SMartin Mares  *	Can be freely distributed and used under the terms of the GNU GPL v2+.
761829219SMartin Mares  *
861829219SMartin Mares  *	SPDX-License-Identifier: GPL-2.0-or-later
9727ce158SMartin Mares  */
10727ce158SMartin Mares 
11727ce158SMartin Mares #include <stdio.h>
12727ce158SMartin Mares #include <ctype.h>
13727ce158SMartin Mares #include <string.h>
14727ce158SMartin Mares #include <errno.h>
15727ce158SMartin Mares 
16727ce158SMartin Mares #include "internal.h"
17727ce158SMartin Mares 
1809817437SMartin Mares struct dump_data {
19f7821e53SMartin Mares   int len, allocated;
2009817437SMartin Mares   byte data[1];
2109817437SMartin Mares };
2209817437SMartin Mares 
23cb6ee324SMartin Mares static void
dump_config(struct pci_access * a)24cb6ee324SMartin Mares dump_config(struct pci_access *a)
25cb6ee324SMartin Mares {
26cb6ee324SMartin Mares   pci_define_param(a, "dump.name", "", "Name of the bus dump file to read from");
27cb6ee324SMartin Mares }
28cb6ee324SMartin Mares 
29727ce158SMartin Mares static int
dump_detect(struct pci_access * a)30727ce158SMartin Mares dump_detect(struct pci_access *a)
31727ce158SMartin Mares {
32cb6ee324SMartin Mares   char *name = pci_get_param(a, "dump.name");
33cb6ee324SMartin Mares   return name && name[0];
34727ce158SMartin Mares }
35727ce158SMartin Mares 
36727ce158SMartin Mares static void
dump_alloc_data(struct pci_dev * dev,int len)3709817437SMartin Mares dump_alloc_data(struct pci_dev *dev, int len)
3809817437SMartin Mares {
3909817437SMartin Mares   struct dump_data *dd = pci_malloc(dev->access, sizeof(struct dump_data) + len - 1);
40f7821e53SMartin Mares   dd->allocated = len;
41f7821e53SMartin Mares   dd->len = 0;
4209817437SMartin Mares   memset(dd->data, 0xff, len);
43a997ef13SMartin Mares   dev->backend_data = dd;
4409817437SMartin Mares }
4509817437SMartin Mares 
46d19394dbSMartin Mares static int
dump_validate(char * s,char * fmt)47d19394dbSMartin Mares dump_validate(char *s, char *fmt)
48d19394dbSMartin Mares {
49d19394dbSMartin Mares   while (*fmt)
50d19394dbSMartin Mares     {
51d19394dbSMartin Mares       if (*fmt == '#' ? !isxdigit(*s) : *fmt != *s)
52d19394dbSMartin Mares 	return 0;
53dc01dd60SMartin Mares       fmt++, s++;
54d19394dbSMartin Mares     }
55d19394dbSMartin Mares   return 1;
56d19394dbSMartin Mares }
57d19394dbSMartin Mares 
5809817437SMartin Mares static void
dump_init(struct pci_access * a)59727ce158SMartin Mares dump_init(struct pci_access *a)
60727ce158SMartin Mares {
61cb6ee324SMartin Mares   char *name = pci_get_param(a, "dump.name");
62727ce158SMartin Mares   FILE *f;
63727ce158SMartin Mares   char buf[256];
64727ce158SMartin Mares   struct pci_dev *dev = NULL;
651f7c91ccSMartin Mares   int len, mn, bn, dn, fn, i, j;
66727ce158SMartin Mares 
67364275f8SVille Skyttä   if (!name)
68727ce158SMartin Mares     a->error("dump: File name not given.");
69727ce158SMartin Mares   if (!(f = fopen(name, "r")))
70727ce158SMartin Mares     a->error("dump: Cannot open %s: %s", name, strerror(errno));
71727ce158SMartin Mares   while (fgets(buf, sizeof(buf)-1, f))
72727ce158SMartin Mares     {
73727ce158SMartin Mares       char *z = strchr(buf, '\n');
74727ce158SMartin Mares       if (!z)
75202a8f8aSMartin Mares 	{
76202a8f8aSMartin Mares 	  fclose(f);
77727ce158SMartin Mares 	  a->error("dump: line too long or unterminated");
78202a8f8aSMartin Mares 	}
79727ce158SMartin Mares       *z-- = 0;
80727ce158SMartin Mares       if (z >= buf && *z == '\r')
81727ce158SMartin Mares 	*z-- = 0;
82727ce158SMartin Mares       len = z - buf + 1;
831f7c91ccSMartin Mares       mn = 0;
84d19394dbSMartin Mares       if (dump_validate(buf, "##:##.# ") && sscanf(buf, "%x:%x.%d", &bn, &dn, &fn) == 3 ||
85ac459e26SRohit Mundra 	  dump_validate(buf, "####:##:##.# ") && sscanf(buf, "%x:%x:%x.%d", &mn, &bn, &dn, &fn) == 4 ||
86*b59b41bfSKonrad Sztyber 	  dump_validate(buf, "#####:##:##.# ") && sscanf(buf, "%x:%x:%x.%d", &mn, &bn, &dn, &fn) == 4 ||
87*b59b41bfSKonrad Sztyber 	  dump_validate(buf, "######:##:##.# ") && sscanf(buf, "%x:%x:%x.%d", &mn, &bn, &dn, &fn) == 4)
88727ce158SMartin Mares 	{
891f7c91ccSMartin Mares 	  dev = pci_get_dev(a, mn, bn, dn, fn);
9009817437SMartin Mares 	  dump_alloc_data(dev, 256);
91727ce158SMartin Mares 	  pci_link_dev(a, dev);
92727ce158SMartin Mares 	}
93727ce158SMartin Mares       else if (!len)
94727ce158SMartin Mares 	dev = NULL;
9509817437SMartin Mares       else if (dev &&
96006ca87fSPali Rohár 	       (dump_validate(buf, "##: ") || dump_validate(buf, "###: ") || dump_validate(buf, "####: ") ||
97006ca87fSPali Rohár 		dump_validate(buf, "#####: ") || dump_validate(buf, "######: ") ||
98006ca87fSPali Rohár 		dump_validate(buf, "#######: ") || dump_validate(buf, "########: ")) &&
99727ce158SMartin Mares 	       sscanf(buf, "%x: ", &i) == 1)
100727ce158SMartin Mares 	{
101a997ef13SMartin Mares 	  struct dump_data *dd = dev->backend_data;
102d19394dbSMartin Mares 	  z = strchr(buf, ' ') + 1;
103d19394dbSMartin Mares 	  while (isxdigit(z[0]) && isxdigit(z[1]) && (!z[2] || z[2] == ' ') &&
104d19394dbSMartin Mares 		 sscanf(z, "%x", &j) == 1 && j < 256)
105727ce158SMartin Mares 	    {
10609817437SMartin Mares 	      if (i >= 4096)
107202a8f8aSMartin Mares 		{
108202a8f8aSMartin Mares 		  fclose(f);
109d19394dbSMartin Mares 		  a->error("dump: At most 4096 bytes of config space are supported");
110202a8f8aSMartin Mares 		}
111d19394dbSMartin Mares 	      if (i >= dd->allocated)	/* Need to re-allocate the buffer */
11209817437SMartin Mares 		{
11309817437SMartin Mares 		  dump_alloc_data(dev, 4096);
114a997ef13SMartin Mares 		  memcpy(((struct dump_data *) dev->backend_data)->data, dd->data, 256);
11509817437SMartin Mares 		  pci_mfree(dd);
116a997ef13SMartin Mares 		  dd = dev->backend_data;
11709817437SMartin Mares 		}
11809817437SMartin Mares 	      dd->data[i++] = j;
119f7821e53SMartin Mares 	      if (i > dd->len)
120f7821e53SMartin Mares 		dd->len = i;
121727ce158SMartin Mares 	      z += 2;
122d19394dbSMartin Mares 	      if (*z)
123d19394dbSMartin Mares 		z++;
124727ce158SMartin Mares 	    }
125d19394dbSMartin Mares 	  if (*z)
126202a8f8aSMartin Mares 	    {
127202a8f8aSMartin Mares 	      fclose(f);
128d19394dbSMartin Mares 	      a->error("dump: Malformed line");
129727ce158SMartin Mares 	    }
130727ce158SMartin Mares 	}
131727ce158SMartin Mares     }
132202a8f8aSMartin Mares   fclose(f);
133202a8f8aSMartin Mares }
134727ce158SMartin Mares 
135727ce158SMartin Mares static void
dump_cleanup(struct pci_access * a UNUSED)136a832f6f1SMartin Mares dump_cleanup(struct pci_access *a UNUSED)
137727ce158SMartin Mares {
138727ce158SMartin Mares }
139727ce158SMartin Mares 
140727ce158SMartin Mares static void
dump_scan(struct pci_access * a UNUSED)141a832f6f1SMartin Mares dump_scan(struct pci_access *a UNUSED)
142727ce158SMartin Mares {
143727ce158SMartin Mares }
144727ce158SMartin Mares 
145727ce158SMartin Mares static int
dump_read(struct pci_dev * d,int pos,byte * buf,int len)146727ce158SMartin Mares dump_read(struct pci_dev *d, int pos, byte *buf, int len)
147727ce158SMartin Mares {
14809817437SMartin Mares   struct dump_data *dd;
149a997ef13SMartin Mares   if (!(dd = d->backend_data))
150727ce158SMartin Mares     {
151727ce158SMartin Mares       struct pci_dev *e = d->access->devices;
152d19394dbSMartin Mares       while (e && (e->domain != d->domain || e->bus != d->bus || e->dev != d->dev || e->func != d->func))
153727ce158SMartin Mares 	e = e->next;
15409817437SMartin Mares       if (!e)
155727ce158SMartin Mares 	return 0;
156a997ef13SMartin Mares       dd = e->backend_data;
157727ce158SMartin Mares     }
15809817437SMartin Mares   if (pos + len > dd->len)
15909817437SMartin Mares     return 0;
16009817437SMartin Mares   memcpy(buf, dd->data + pos, len);
161727ce158SMartin Mares   return 1;
162727ce158SMartin Mares }
163727ce158SMartin Mares 
164727ce158SMartin Mares static int
dump_write(struct pci_dev * d UNUSED,int pos UNUSED,byte * buf UNUSED,int len UNUSED)165a832f6f1SMartin Mares dump_write(struct pci_dev *d UNUSED, int pos UNUSED, byte *buf UNUSED, int len UNUSED)
166727ce158SMartin Mares {
167727ce158SMartin Mares   d->access->error("Writing to dump files is not supported.");
168727ce158SMartin Mares   return 0;
169727ce158SMartin Mares }
170727ce158SMartin Mares 
171727ce158SMartin Mares static void
dump_cleanup_dev(struct pci_dev * d)172727ce158SMartin Mares dump_cleanup_dev(struct pci_dev *d)
173727ce158SMartin Mares {
174a997ef13SMartin Mares   if (d->backend_data)
175727ce158SMartin Mares     {
176a997ef13SMartin Mares       pci_mfree(d->backend_data);
177a997ef13SMartin Mares       d->backend_data = NULL;
178727ce158SMartin Mares     }
179727ce158SMartin Mares }
180727ce158SMartin Mares 
181727ce158SMartin Mares struct pci_methods pm_dump = {
1821660c737SMartin Mares   .name = "dump",
1831660c737SMartin Mares   .help = "Reading of register dumps (set the `dump.name' parameter)",
1841660c737SMartin Mares   .config = dump_config,
1851660c737SMartin Mares   .detect = dump_detect,
1861660c737SMartin Mares   .init = dump_init,
1871660c737SMartin Mares   .cleanup = dump_cleanup,
1881660c737SMartin Mares   .scan = dump_scan,
1891660c737SMartin Mares   .fill_info = pci_generic_fill_info,
1901660c737SMartin Mares   .read = dump_read,
1911660c737SMartin Mares   .write = dump_write,
1921660c737SMartin Mares   .cleanup_dev = dump_cleanup_dev,
193727ce158SMartin Mares };
194