1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
|
/*
* The PCI Library -- Looking up Names via UDEV and HWDB
*
* Copyright (c) 2013--2014 Tom Gundersen <teg@jklm.no>
* Copyright (c) 2014 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 <string.h>
#include "internal.h"
#include "names.h"
#ifdef PCI_HAVE_HWDB
#include <libudev.h>
#include <stdio.h>
#include <stdlib.h>
char *
pci_id_hwdb_lookup(struct pci_access *a, int cat, int id1, int id2, int id3, int id4 UNUSED)
{
char modalias[64];
const char *key = NULL;
const char *disabled = pci_get_param(a, "hwdb.disable");
if (disabled && atoi(disabled))
return NULL;
switch (cat)
{
case ID_VENDOR:
sprintf(modalias, "pci:v%08X*", id1);
key = "ID_VENDOR_FROM_DATABASE";
break;
case ID_DEVICE:
sprintf(modalias, "pci:v%08Xd%08X*", id1, id2);
key = "ID_MODEL_FROM_DATABASE";
break;
case ID_SUBSYSTEM:
/*
* There is no udev hwdb key which returns subsystem. Also note that query
* modalias "pci:v%08Xd%08Xsv%08Xsd%08X*" matches also hwdb device with
* modalias "pci:v%08Xd%08Xsv*sd*" (which is the default modalias), so
* there is no way to get information specific for the subsystem.
*/
return NULL;
case ID_GEN_SUBSYSTEM:
/* There is no udev hwdb key which returns generic subsystem. */
return NULL;
case ID_CLASS:
sprintf(modalias, "pci:v*d*sv*sd*bc%02X*", id1);
key = "ID_PCI_CLASS_FROM_DATABASE";
break;
case ID_SUBCLASS:
sprintf(modalias, "pci:v*d*sv*sd*bc%02Xsc%02X*", id1, id2);
key = "ID_PCI_SUBCLASS_FROM_DATABASE";
break;
case ID_PROGIF:
sprintf(modalias, "pci:v*d*sv*sd*bc%02Xsc%02Xi%02X*", id1, id2, id3);
key = "ID_PCI_INTERFACE_FROM_DATABASE";
break;
}
if (key)
{
if (!a->id_udev_hwdb)
{
a->debug("Initializing UDEV HWDB\n");
a->id_udev = udev_new();
a->id_udev_hwdb = udev_hwdb_new(a->id_udev);
}
struct udev_list_entry *entry;
udev_list_entry_foreach(entry, udev_hwdb_get_properties_list_entry(a->id_udev_hwdb, modalias, 0))
{
const char *entry_name = udev_list_entry_get_name(entry);
if (entry_name && !strcmp(entry_name, key))
{
const char *entry_value = udev_list_entry_get_value(entry);
if (entry_value)
return pci_strdup(a, entry_value);
}
}
}
return NULL;
}
void
pci_id_hwdb_free(struct pci_access *a)
{
if (a->id_udev_hwdb)
{
udev_hwdb_unref(a->id_udev_hwdb);
a->id_udev_hwdb = NULL;
}
if (a->id_udev)
{
udev_unref(a->id_udev);
a->id_udev = NULL;
}
}
#else
char *
pci_id_hwdb_lookup(struct pci_access *a UNUSED, int cat UNUSED, int id1 UNUSED, int id2 UNUSED, int id3 UNUSED, int id4 UNUSED)
{
return NULL;
}
void
pci_id_hwdb_free(struct pci_access *a UNUSED)
{
}
#endif
|