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#include <inttypes.h>
32#include <stdio.h>
33#include <ctype.h>
34#include <stdlib.h>
35#include <string.h>
36#include <console.h>
37#include <sys/pci.h>
38#include <com32.h>
39#include <syslinux/boot.h>
40#include <syslinux/config.h>
41#include <dprintf.h>
42
43#define MAX_LINE 512
44
45
46static int looking_at(const char *line, const char *kwd)
47{
48 const char *p = line;
49 const char *q = kwd;
50
51 while (*p && *q && ((*p ^ *q) & ~0x20) == 0) {
52 p++;
53 q++;
54 }
55
56 if (*q)
57 return 0;
58
59 return *p <= ' ';
60}
61
62static char *get_did(char *p, uint32_t * idptr, uint32_t * maskptr)
63{
64 unsigned long vid, did, m1, m2;
65
66 *idptr = -1;
67 *maskptr = 0xffffffff;
68
69 vid = strtoul(p, &p, 16);
70 if (*p != ':')
71 return p;
72 did = strtoul(p + 1, &p, 16);
73
74 *idptr = (did << 16) + vid;
75
76 if (*p == '/') {
77 m1 = strtoul(p + 1, &p, 16);
78 if (*p != ':') {
79 *maskptr = (m1 << 16) | 0xffff;
80 } else {
81 m2 = strtoul(p + 1, &p, 16);
82 *maskptr = (m1 << 16) | m2;
83 }
84 }
85
86 return p;
87}
88
89static char *get_rid_range(char *p, uint8_t * rid_min, uint8_t * rid_max)
90{
91 unsigned long r0, r1;
92
93 p = skipspace(p + 3);
94
95 r0 = strtoul(p, &p, 16);
96 if (*p == '-') {
97 r1 = strtoul(p + 1, &p, 16);
98 } else {
99 r1 = r0;
100 }
101
102 *rid_min = r0;
103 *rid_max = r1;
104
105 return p;
106}
107
108static struct match *parse_config(const char *filename)
109{
110 char line[MAX_LINE], *p;
111 FILE *f;
112 struct match *list = NULL;
113 struct match **ep = &list;
114 struct match *m;
115
116 if (!filename)
117 filename = syslinux_config_file();
118
119 f = fopen(filename, "r");
120 if (!f)
121 return list;
122
123 while (fgets(line, sizeof line, f)) {
124 p = skipspace(line);
125
126 if (!looking_at(p, "#"))
127 continue;
128 p = skipspace(p + 1);
129
130 if (!looking_at(p, "dev"))
131 continue;
132 p = skipspace(p + 3);
133
134 m = malloc(sizeof(struct match));
135 if (!m)
136 continue;
137
138 memset(m, 0, sizeof *m);
139 m->rid_max = 0xff;
140
141 for (;;) {
142 p = skipspace(p);
143
144 if (looking_at(p, "did")) {
145 p = get_did(p + 3, &m->did, &m->did_mask);
146 } else if (looking_at(p, "sid")) {
147 p = get_did(p + 3, &m->sid, &m->sid_mask);
148 } else if (looking_at(p, "rid")) {
149 p = get_rid_range(p + 3, &m->rid_min, &m->rid_max);
150 } else {
151 char *e;
152
153 e = strchr(p, '\n');
154 if (*e)
155 *e = '\0';
156 e = strchr(p, '\r');
157 if (*e)
158 *e = '\0';
159
160 m->filename = strdup(p);
161 if (!m->filename)
162 m->did = -1;
163 break;
164 }
165 }
166
167 dprintf("DEV DID %08x/%08x SID %08x/%08x RID %02x-%02x CMD %s\n",
168 m->did, m->did_mask, m->sid, m->sid_mask,
169 m->rid_min, m->rid_max, m->filename);
170
171 *ep = m;
172 ep = &m->next;
173 }
174
175 return list;
176}
177
178int main(int argc, char *argv[])
179{
180 struct match *list, *match;
181 struct pci_domain *pci_domain;
182
183 openconsole(&dev_null_r, &dev_stdcon_w);
184 pci_domain = pci_scan();
185
186 if (pci_domain) {
187 list = parse_config(argc < 2 ? NULL : argv[1]);
188
189 match = find_pci_device(pci_domain, list);
190
191 if (match)
192 syslinux_run_command(match->filename);
193 }
194
195
196 fputs("Error: no recognized network card found!\n", stderr);
197 return 1;
198}
199