/*
- * $Id: lspci.c,v 1.13 1998/07/15 20:37:12 mj Exp $
+ * $Id: lspci.c,v 1.14 1998/07/17 08:57:14 mj Exp $
*
* Linux PCI Utilities -- List All PCI Devices
*
else
irq = d->kernel_irq;
+ if (verbose && subsys_v && subsys_v != 0xffff)
+ printf("\tSubsystem: %s\n", lookup_subsys_device_full(subsys_v, subsys_d));
+
if (verbose > 1)
{
- if (subsys_v)
- printf("\tSubsystem ID: %04x:%04x\n", subsys_v, subsys_d);
printf("\tControl: I/O%c Mem%c BusMaster%c SpecCycle%c MemWINV%c VGASnoop%c ParErr%c Stepping%c SERR%c FastB2B%c\n",
(cmd & PCI_COMMAND_IO) ? '+' : '-',
(cmd & PCI_COMMAND_MEMORY) ? '+' : '-',
show_machine(struct device *d)
{
int c;
+ word sv_id=0, sd_id=0;
+
+ switch (get_conf_byte(d, PCI_HEADER_TYPE) & 0x7f)
+ {
+ case PCI_HEADER_TYPE_NORMAL:
+ sv_id = get_conf_word(d, PCI_SUBSYSTEM_VENDOR_ID);
+ sd_id = get_conf_word(d, PCI_SUBSYSTEM_ID);
+ break;
+ case PCI_HEADER_TYPE_CARDBUS:
+ sv_id = get_conf_word(d, PCI_CB_SUBSYSTEM_VENDOR_ID);
+ sd_id = get_conf_word(d, PCI_CB_SUBSYSTEM_ID);
+ break;
+ }
if (verbose)
{
printf("Class:\t%s\n", lookup_class(get_conf_word(d, PCI_CLASS_DEVICE)));
printf("Vendor:\t%s\n", lookup_vendor(d->vendid));
printf("Device:\t%s\n", lookup_device(d->vendid, d->devid));
+ if (sv_id && sv_id != 0xffff)
+ {
+ printf("SVendor:\t%s\n", lookup_subsys_vendor(sv_id));
+ printf("SDevice:\t%s\n", lookup_subsys_device(sv_id, sd_id));
+ }
if (c = get_conf_byte(d, PCI_REVISION_ID))
printf("Rev:\t%02x\n", c);
if (c = get_conf_byte(d, PCI_CLASS_PROG))
printf(" -r%02x", c);
if (c = get_conf_byte(d, PCI_CLASS_PROG))
printf(" -p%02x", c);
+ if (sv_id && sv_id != 0xffff)
+ printf(" \"%s\" \"%s\"", lookup_subsys_vendor(sv_id), lookup_subsys_device(sv_id, sd_id));
+ else
+ printf(" \"\" \"\"");
putchar('\n');
}
}
/*
- * $Id: names.c,v 1.5 1998/06/09 19:16:45 mj Exp $
+ * $Id: names.c,v 1.6 1998/07/17 08:57:16 mj Exp $
*
* Linux PCI Utilities -- Device ID to Name Translation
*
struct nl_entry {
struct nl_entry *next;
- int id1, id2;
+ word id1, id2;
+ int cat;
byte *name;
};
-#define ID1_VENDOR -1
-#define ID1_CLASS -2
-#define ID1_SUBCLASS -3
-#define ID1_ERROR -4
+#define NL_VENDOR 0
+#define NL_DEVICE 1
+#define NL_CLASS 2
+#define NL_SUBCLASS 3
+#define NL_SUBSYSTEM_VENDOR 4
+#define NL_SUBSYSTEM_DEVICE 5
#define HASH_SIZE 1024
static struct nl_entry *nl_hash[HASH_SIZE];
-static inline unsigned int nl_calc_hash(int id1, int id2)
+static inline unsigned int nl_calc_hash(int cat, int id1, int id2)
{
unsigned int h;
- h = id1 ^ id2;
- h ^= (h >> 6);
+ h = id1 ^ id2 ^ (cat << 5);
+ h += (h >> 6);
return h & (HASH_SIZE-1);
}
-static struct nl_entry *nl_lookup(int id1, int id2)
+static struct nl_entry *nl_lookup(int cat, int id1, int id2)
{
- unsigned int h = nl_calc_hash(id1, id2);
+ unsigned int h = nl_calc_hash(cat, id1, id2);
struct nl_entry *n = nl_hash[h];
- while (n && (n->id1 != id1 || n->id2 != id2))
+ while (n && (n->id1 != id1 || n->id2 != id2 || n->cat != cat))
n = n->next;
return n;
}
-static int nl_add(int id1, int id2, byte *text)
+static int nl_add(int cat, int id1, int id2, byte *text)
{
- unsigned int h = nl_calc_hash(id1, id2);
+ unsigned int h = nl_calc_hash(cat, id1, id2);
struct nl_entry *n = nl_hash[h];
- while (n && (n->id1 != id1 || n->id2 != id2))
+ while (n && (n->id1 != id1 || n->id2 != id2 || n->cat != cat))
n = n->next;
if (n)
return 1;
n = xmalloc(sizeof(struct nl_entry));
n->id1 = id1;
n->id2 = id2;
+ n->cat = cat;
n->name = text;
n->next = nl_hash[h];
nl_hash[h] = n;
byte *p = name_list;
byte *q, *r;
int lino = 0;
- int id1 = ID1_ERROR;
- int id2 = 0;
- int i, j;
+ unsigned int id1=0, id2=0;
+ int cat, last_cat = -1;
while (*p)
{
r = q;
while (*q == ' ')
q++;
- if (strlen(q) < 5 || q[4] != ' ')
- goto parserr;
if (r == q)
{
if (q[0] == 'C' && q[1] == ' ')
{
- if (sscanf(q+2, "%x", &j) != 1)
+ if (strlen(q+2) < 3 ||
+ q[4] != ' ' ||
+ sscanf(q+2, "%x", &id1) != 1)
goto parserr;
- i = ID1_CLASS;
+ cat = last_cat = NL_CLASS;
+ }
+ else if (q[0] == 'S' && q[1] == ' ')
+ {
+ if (strlen(q+2) < 5 ||
+ q[6] != ' ' ||
+ sscanf(q+2, "%x", &id1) != 1)
+ goto parserr;
+ cat = last_cat = NL_SUBSYSTEM_VENDOR;
+ q += 2;
}
else
{
- if (sscanf(q, "%x", &j) != 1)
+ if (strlen(q) < 5 ||
+ q[4] != ' ' ||
+ sscanf(q, "%x", &id1) != 1)
goto parserr;
- i = ID1_VENDOR;
+ cat = last_cat = NL_VENDOR;
}
- id1 = i;
- id2 = j;
+ id2 = 0;
}
else
{
- if (sscanf(q, "%x", &j) != 1)
+ if (sscanf(q, "%x", &id2) != 1)
goto parserr;
- if (id1 == ID1_ERROR)
+ if (last_cat < 0)
goto parserr;
- if (id1 == ID1_CLASS)
- {
- i = ID1_SUBCLASS;
- j |= (id2 << 8);
- }
+ if (last_cat == NL_CLASS)
+ cat = NL_SUBCLASS;
else
- i = id2;
+ cat = last_cat+1;
}
q += 4;
while (*q == ' ')
q++;
if (!*q)
goto parserr;
- if (nl_add(i, j, q))
+ if (nl_add(cat, id1, id2, q))
{
fprintf(stderr, "%s, line %d: duplicate entry\n", pci_ids, lino);
exit(1);
}
char *
-lookup_vendor(word i)
+do_lookup_vendor(int cat, word i)
{
static char vendbuf[6];
{
struct nl_entry *e;
- e = nl_lookup(ID1_VENDOR, i);
+ e = nl_lookup(cat, i, 0);
if (e)
return e->name;
}
}
char *
-lookup_device(word v, word i)
+do_lookup_device(int cat, word v, word i)
{
static char devbuf[6];
{
struct nl_entry *e;
- e = nl_lookup(v, i);
+ e = nl_lookup(cat, v, i);
if (e)
return e->name;
}
}
char *
-lookup_device_full(word v, word i)
+do_lookup_device_full(int cat, word v, word i)
{
static char fullbuf[256];
{
struct nl_entry *e, *e2;
- e = nl_lookup(ID1_VENDOR, v);
- e2 = nl_lookup(v, i);
+ e = nl_lookup(cat, v, 0);
+ e2 = nl_lookup(cat+1, v, i);
if (!e)
sprintf(fullbuf, "Unknown device %04x:%04x", v, i);
else if (!e2)
return fullbuf;
}
+char *
+lookup_vendor(word i)
+{
+ return do_lookup_vendor(NL_VENDOR, i);
+}
+
+char *
+lookup_subsys_vendor(word i)
+{
+ return do_lookup_vendor(NL_SUBSYSTEM_VENDOR, i);
+}
+
+char *
+lookup_device(word i, word v)
+{
+ return do_lookup_device(NL_DEVICE, v, i);
+}
+
+char *
+lookup_subsys_device(word v, word i)
+{
+ return do_lookup_device(NL_SUBSYSTEM_DEVICE, v, i);
+}
+
+char *
+lookup_device_full(word v, word i)
+{
+ return do_lookup_device_full(NL_VENDOR, v, i);
+}
+
+char *
+lookup_subsys_device_full(word v, word i)
+{
+ return do_lookup_device_full(NL_SUBSYSTEM_VENDOR, v, i);
+}
+
char *
lookup_class(word c)
{
{
struct nl_entry *e;
- e = nl_lookup(ID1_SUBCLASS, c);
+ e = nl_lookup(NL_SUBCLASS, c >> 8, c & 0xff);
if (e)
return e->name;
- e = nl_lookup(ID1_CLASS, c);
+ e = nl_lookup(NL_CLASS, c, 0);
if (e)
sprintf(classbuf, "%s [%04x]", e->name, c);
else