1*c2e0c6b5SAndroid Build Coastguard Worker /*
2*c2e0c6b5SAndroid Build Coastguard Worker * The PCI Utilities -- Show Kernel Drivers
3*c2e0c6b5SAndroid Build Coastguard Worker *
4*c2e0c6b5SAndroid Build Coastguard Worker * Copyright (c) 1997--2013 Martin Mares <[email protected]>
5*c2e0c6b5SAndroid Build Coastguard Worker *
6*c2e0c6b5SAndroid Build Coastguard Worker * Can be freely distributed and used under the terms of the GNU GPL v2+.
7*c2e0c6b5SAndroid Build Coastguard Worker *
8*c2e0c6b5SAndroid Build Coastguard Worker * SPDX-License-Identifier: GPL-2.0-or-later
9*c2e0c6b5SAndroid Build Coastguard Worker */
10*c2e0c6b5SAndroid Build Coastguard Worker
11*c2e0c6b5SAndroid Build Coastguard Worker #include <stdio.h>
12*c2e0c6b5SAndroid Build Coastguard Worker #include <string.h>
13*c2e0c6b5SAndroid Build Coastguard Worker
14*c2e0c6b5SAndroid Build Coastguard Worker #include "lspci.h"
15*c2e0c6b5SAndroid Build Coastguard Worker
16*c2e0c6b5SAndroid Build Coastguard Worker #ifdef PCI_OS_LINUX
17*c2e0c6b5SAndroid Build Coastguard Worker
18*c2e0c6b5SAndroid Build Coastguard Worker #include <sys/utsname.h>
19*c2e0c6b5SAndroid Build Coastguard Worker
20*c2e0c6b5SAndroid Build Coastguard Worker #ifdef PCI_USE_LIBKMOD
21*c2e0c6b5SAndroid Build Coastguard Worker
22*c2e0c6b5SAndroid Build Coastguard Worker #include <libkmod.h>
23*c2e0c6b5SAndroid Build Coastguard Worker
24*c2e0c6b5SAndroid Build Coastguard Worker static struct kmod_ctx *kmod_ctx;
25*c2e0c6b5SAndroid Build Coastguard Worker
26*c2e0c6b5SAndroid Build Coastguard Worker static int
show_kernel_init(void)27*c2e0c6b5SAndroid Build Coastguard Worker show_kernel_init(void)
28*c2e0c6b5SAndroid Build Coastguard Worker {
29*c2e0c6b5SAndroid Build Coastguard Worker static int show_kernel_inited = -1;
30*c2e0c6b5SAndroid Build Coastguard Worker if (show_kernel_inited >= 0)
31*c2e0c6b5SAndroid Build Coastguard Worker return show_kernel_inited;
32*c2e0c6b5SAndroid Build Coastguard Worker
33*c2e0c6b5SAndroid Build Coastguard Worker kmod_ctx = kmod_new(NULL, NULL);
34*c2e0c6b5SAndroid Build Coastguard Worker if (!kmod_ctx)
35*c2e0c6b5SAndroid Build Coastguard Worker {
36*c2e0c6b5SAndroid Build Coastguard Worker fprintf(stderr, "lspci: Unable to initialize libkmod context\n");
37*c2e0c6b5SAndroid Build Coastguard Worker goto failed;
38*c2e0c6b5SAndroid Build Coastguard Worker }
39*c2e0c6b5SAndroid Build Coastguard Worker
40*c2e0c6b5SAndroid Build Coastguard Worker int err;
41*c2e0c6b5SAndroid Build Coastguard Worker if ((err = kmod_load_resources(kmod_ctx)) < 0)
42*c2e0c6b5SAndroid Build Coastguard Worker {
43*c2e0c6b5SAndroid Build Coastguard Worker fprintf(stderr, "lspci: Unable to load libkmod resources: error %d\n", err);
44*c2e0c6b5SAndroid Build Coastguard Worker goto failed;
45*c2e0c6b5SAndroid Build Coastguard Worker }
46*c2e0c6b5SAndroid Build Coastguard Worker
47*c2e0c6b5SAndroid Build Coastguard Worker show_kernel_inited = 1;
48*c2e0c6b5SAndroid Build Coastguard Worker return 1;
49*c2e0c6b5SAndroid Build Coastguard Worker
50*c2e0c6b5SAndroid Build Coastguard Worker failed:
51*c2e0c6b5SAndroid Build Coastguard Worker show_kernel_inited = 0;
52*c2e0c6b5SAndroid Build Coastguard Worker return 0;
53*c2e0c6b5SAndroid Build Coastguard Worker }
54*c2e0c6b5SAndroid Build Coastguard Worker
55*c2e0c6b5SAndroid Build Coastguard Worker void
show_kernel_cleanup(void)56*c2e0c6b5SAndroid Build Coastguard Worker show_kernel_cleanup(void)
57*c2e0c6b5SAndroid Build Coastguard Worker {
58*c2e0c6b5SAndroid Build Coastguard Worker if (kmod_ctx)
59*c2e0c6b5SAndroid Build Coastguard Worker kmod_unref(kmod_ctx);
60*c2e0c6b5SAndroid Build Coastguard Worker }
61*c2e0c6b5SAndroid Build Coastguard Worker
next_module(struct device * d)62*c2e0c6b5SAndroid Build Coastguard Worker static const char *next_module(struct device *d)
63*c2e0c6b5SAndroid Build Coastguard Worker {
64*c2e0c6b5SAndroid Build Coastguard Worker static struct kmod_list *klist, *kcurrent;
65*c2e0c6b5SAndroid Build Coastguard Worker static struct kmod_module *kmodule;
66*c2e0c6b5SAndroid Build Coastguard Worker
67*c2e0c6b5SAndroid Build Coastguard Worker if (kmodule)
68*c2e0c6b5SAndroid Build Coastguard Worker {
69*c2e0c6b5SAndroid Build Coastguard Worker kmod_module_unref(kmodule);
70*c2e0c6b5SAndroid Build Coastguard Worker kmodule = NULL;
71*c2e0c6b5SAndroid Build Coastguard Worker }
72*c2e0c6b5SAndroid Build Coastguard Worker
73*c2e0c6b5SAndroid Build Coastguard Worker if (!klist)
74*c2e0c6b5SAndroid Build Coastguard Worker {
75*c2e0c6b5SAndroid Build Coastguard Worker pci_fill_info(d->dev, PCI_FILL_MODULE_ALIAS);
76*c2e0c6b5SAndroid Build Coastguard Worker if (!d->dev->module_alias)
77*c2e0c6b5SAndroid Build Coastguard Worker return NULL;
78*c2e0c6b5SAndroid Build Coastguard Worker int err = kmod_module_new_from_lookup(kmod_ctx, d->dev->module_alias, &klist);
79*c2e0c6b5SAndroid Build Coastguard Worker if (err < 0)
80*c2e0c6b5SAndroid Build Coastguard Worker {
81*c2e0c6b5SAndroid Build Coastguard Worker fprintf(stderr, "lspci: libkmod lookup failed: error %d\n", err);
82*c2e0c6b5SAndroid Build Coastguard Worker return NULL;
83*c2e0c6b5SAndroid Build Coastguard Worker }
84*c2e0c6b5SAndroid Build Coastguard Worker kcurrent = klist;
85*c2e0c6b5SAndroid Build Coastguard Worker }
86*c2e0c6b5SAndroid Build Coastguard Worker else
87*c2e0c6b5SAndroid Build Coastguard Worker kcurrent = kmod_list_next(klist, kcurrent);
88*c2e0c6b5SAndroid Build Coastguard Worker
89*c2e0c6b5SAndroid Build Coastguard Worker if (kcurrent)
90*c2e0c6b5SAndroid Build Coastguard Worker {
91*c2e0c6b5SAndroid Build Coastguard Worker kmodule = kmod_module_get_module(kcurrent);
92*c2e0c6b5SAndroid Build Coastguard Worker return kmod_module_get_name(kmodule);
93*c2e0c6b5SAndroid Build Coastguard Worker }
94*c2e0c6b5SAndroid Build Coastguard Worker
95*c2e0c6b5SAndroid Build Coastguard Worker kmod_module_unref_list(klist);
96*c2e0c6b5SAndroid Build Coastguard Worker klist = NULL;
97*c2e0c6b5SAndroid Build Coastguard Worker return NULL;
98*c2e0c6b5SAndroid Build Coastguard Worker }
99*c2e0c6b5SAndroid Build Coastguard Worker
100*c2e0c6b5SAndroid Build Coastguard Worker #else
101*c2e0c6b5SAndroid Build Coastguard Worker
102*c2e0c6b5SAndroid Build Coastguard Worker struct pcimap_entry {
103*c2e0c6b5SAndroid Build Coastguard Worker struct pcimap_entry *next;
104*c2e0c6b5SAndroid Build Coastguard Worker unsigned int vendor, device;
105*c2e0c6b5SAndroid Build Coastguard Worker unsigned int subvendor, subdevice;
106*c2e0c6b5SAndroid Build Coastguard Worker unsigned int class, class_mask;
107*c2e0c6b5SAndroid Build Coastguard Worker char module[1];
108*c2e0c6b5SAndroid Build Coastguard Worker };
109*c2e0c6b5SAndroid Build Coastguard Worker
110*c2e0c6b5SAndroid Build Coastguard Worker static struct pcimap_entry *pcimap_head;
111*c2e0c6b5SAndroid Build Coastguard Worker
112*c2e0c6b5SAndroid Build Coastguard Worker static int
show_kernel_init(void)113*c2e0c6b5SAndroid Build Coastguard Worker show_kernel_init(void)
114*c2e0c6b5SAndroid Build Coastguard Worker {
115*c2e0c6b5SAndroid Build Coastguard Worker static int tried_pcimap;
116*c2e0c6b5SAndroid Build Coastguard Worker struct utsname uts;
117*c2e0c6b5SAndroid Build Coastguard Worker char *name, line[1024];
118*c2e0c6b5SAndroid Build Coastguard Worker FILE *f;
119*c2e0c6b5SAndroid Build Coastguard Worker
120*c2e0c6b5SAndroid Build Coastguard Worker if (tried_pcimap)
121*c2e0c6b5SAndroid Build Coastguard Worker return 1;
122*c2e0c6b5SAndroid Build Coastguard Worker tried_pcimap = 1;
123*c2e0c6b5SAndroid Build Coastguard Worker
124*c2e0c6b5SAndroid Build Coastguard Worker if (name = opt_pcimap)
125*c2e0c6b5SAndroid Build Coastguard Worker {
126*c2e0c6b5SAndroid Build Coastguard Worker f = fopen(name, "r");
127*c2e0c6b5SAndroid Build Coastguard Worker if (!f)
128*c2e0c6b5SAndroid Build Coastguard Worker die("Cannot open pcimap file %s: %m", name);
129*c2e0c6b5SAndroid Build Coastguard Worker }
130*c2e0c6b5SAndroid Build Coastguard Worker else
131*c2e0c6b5SAndroid Build Coastguard Worker {
132*c2e0c6b5SAndroid Build Coastguard Worker if (uname(&uts) < 0)
133*c2e0c6b5SAndroid Build Coastguard Worker die("uname() failed: %m");
134*c2e0c6b5SAndroid Build Coastguard Worker name = alloca(64 + strlen(uts.release));
135*c2e0c6b5SAndroid Build Coastguard Worker sprintf(name, "/lib/modules/%s/modules.pcimap", uts.release);
136*c2e0c6b5SAndroid Build Coastguard Worker f = fopen(name, "r");
137*c2e0c6b5SAndroid Build Coastguard Worker if (!f)
138*c2e0c6b5SAndroid Build Coastguard Worker return 1;
139*c2e0c6b5SAndroid Build Coastguard Worker }
140*c2e0c6b5SAndroid Build Coastguard Worker
141*c2e0c6b5SAndroid Build Coastguard Worker while (fgets(line, sizeof(line), f))
142*c2e0c6b5SAndroid Build Coastguard Worker {
143*c2e0c6b5SAndroid Build Coastguard Worker char *c = strchr(line, '\n');
144*c2e0c6b5SAndroid Build Coastguard Worker struct pcimap_entry *e;
145*c2e0c6b5SAndroid Build Coastguard Worker
146*c2e0c6b5SAndroid Build Coastguard Worker if (!c)
147*c2e0c6b5SAndroid Build Coastguard Worker die("Unterminated or too long line in %s", name);
148*c2e0c6b5SAndroid Build Coastguard Worker *c = 0;
149*c2e0c6b5SAndroid Build Coastguard Worker if (!line[0] || line[0] == '#')
150*c2e0c6b5SAndroid Build Coastguard Worker continue;
151*c2e0c6b5SAndroid Build Coastguard Worker
152*c2e0c6b5SAndroid Build Coastguard Worker c = line;
153*c2e0c6b5SAndroid Build Coastguard Worker while (*c && *c != ' ' && *c != '\t')
154*c2e0c6b5SAndroid Build Coastguard Worker c++;
155*c2e0c6b5SAndroid Build Coastguard Worker if (!*c)
156*c2e0c6b5SAndroid Build Coastguard Worker continue; /* FIXME: Emit warnings! */
157*c2e0c6b5SAndroid Build Coastguard Worker *c++ = 0;
158*c2e0c6b5SAndroid Build Coastguard Worker
159*c2e0c6b5SAndroid Build Coastguard Worker e = xmalloc(sizeof(*e) + strlen(line));
160*c2e0c6b5SAndroid Build Coastguard Worker if (sscanf(c, "%i%i%i%i%i%i",
161*c2e0c6b5SAndroid Build Coastguard Worker &e->vendor, &e->device,
162*c2e0c6b5SAndroid Build Coastguard Worker &e->subvendor, &e->subdevice,
163*c2e0c6b5SAndroid Build Coastguard Worker &e->class, &e->class_mask) != 6)
164*c2e0c6b5SAndroid Build Coastguard Worker continue;
165*c2e0c6b5SAndroid Build Coastguard Worker e->next = pcimap_head;
166*c2e0c6b5SAndroid Build Coastguard Worker pcimap_head = e;
167*c2e0c6b5SAndroid Build Coastguard Worker strcpy(e->module, line);
168*c2e0c6b5SAndroid Build Coastguard Worker }
169*c2e0c6b5SAndroid Build Coastguard Worker fclose(f);
170*c2e0c6b5SAndroid Build Coastguard Worker
171*c2e0c6b5SAndroid Build Coastguard Worker return 1;
172*c2e0c6b5SAndroid Build Coastguard Worker }
173*c2e0c6b5SAndroid Build Coastguard Worker
174*c2e0c6b5SAndroid Build Coastguard Worker static int
match_pcimap(struct device * d,struct pcimap_entry * e)175*c2e0c6b5SAndroid Build Coastguard Worker match_pcimap(struct device *d, struct pcimap_entry *e)
176*c2e0c6b5SAndroid Build Coastguard Worker {
177*c2e0c6b5SAndroid Build Coastguard Worker struct pci_dev *dev = d->dev;
178*c2e0c6b5SAndroid Build Coastguard Worker unsigned int class = (((unsigned int)dev->device_class << 8) | dev->prog_if);
179*c2e0c6b5SAndroid Build Coastguard Worker
180*c2e0c6b5SAndroid Build Coastguard Worker #define MATCH(x, y) ((y) > 0xffff || (x) == (y))
181*c2e0c6b5SAndroid Build Coastguard Worker return
182*c2e0c6b5SAndroid Build Coastguard Worker MATCH(dev->vendor_id, e->vendor) &&
183*c2e0c6b5SAndroid Build Coastguard Worker MATCH(dev->device_id, e->device) &&
184*c2e0c6b5SAndroid Build Coastguard Worker MATCH(dev->subsys_vendor_id, e->subvendor) &&
185*c2e0c6b5SAndroid Build Coastguard Worker MATCH(dev->subsys_id, e->subdevice) &&
186*c2e0c6b5SAndroid Build Coastguard Worker (class & e->class_mask) == e->class;
187*c2e0c6b5SAndroid Build Coastguard Worker #undef MATCH
188*c2e0c6b5SAndroid Build Coastguard Worker }
189*c2e0c6b5SAndroid Build Coastguard Worker
next_module(struct device * d)190*c2e0c6b5SAndroid Build Coastguard Worker static const char *next_module(struct device *d)
191*c2e0c6b5SAndroid Build Coastguard Worker {
192*c2e0c6b5SAndroid Build Coastguard Worker static struct pcimap_entry *current;
193*c2e0c6b5SAndroid Build Coastguard Worker
194*c2e0c6b5SAndroid Build Coastguard Worker if (!current)
195*c2e0c6b5SAndroid Build Coastguard Worker current = pcimap_head;
196*c2e0c6b5SAndroid Build Coastguard Worker else
197*c2e0c6b5SAndroid Build Coastguard Worker current = current->next;
198*c2e0c6b5SAndroid Build Coastguard Worker
199*c2e0c6b5SAndroid Build Coastguard Worker while (current)
200*c2e0c6b5SAndroid Build Coastguard Worker {
201*c2e0c6b5SAndroid Build Coastguard Worker if (match_pcimap(d, current))
202*c2e0c6b5SAndroid Build Coastguard Worker return current->module;
203*c2e0c6b5SAndroid Build Coastguard Worker current = current->next;
204*c2e0c6b5SAndroid Build Coastguard Worker }
205*c2e0c6b5SAndroid Build Coastguard Worker
206*c2e0c6b5SAndroid Build Coastguard Worker return NULL;
207*c2e0c6b5SAndroid Build Coastguard Worker }
208*c2e0c6b5SAndroid Build Coastguard Worker
209*c2e0c6b5SAndroid Build Coastguard Worker void
show_kernel_cleanup(void)210*c2e0c6b5SAndroid Build Coastguard Worker show_kernel_cleanup(void)
211*c2e0c6b5SAndroid Build Coastguard Worker {
212*c2e0c6b5SAndroid Build Coastguard Worker }
213*c2e0c6b5SAndroid Build Coastguard Worker
214*c2e0c6b5SAndroid Build Coastguard Worker #endif
215*c2e0c6b5SAndroid Build Coastguard Worker
216*c2e0c6b5SAndroid Build Coastguard Worker static const char *
next_module_filtered(struct device * d)217*c2e0c6b5SAndroid Build Coastguard Worker next_module_filtered(struct device *d)
218*c2e0c6b5SAndroid Build Coastguard Worker {
219*c2e0c6b5SAndroid Build Coastguard Worker static char prev_module[256];
220*c2e0c6b5SAndroid Build Coastguard Worker const char *module;
221*c2e0c6b5SAndroid Build Coastguard Worker
222*c2e0c6b5SAndroid Build Coastguard Worker while (module = next_module(d))
223*c2e0c6b5SAndroid Build Coastguard Worker {
224*c2e0c6b5SAndroid Build Coastguard Worker if (strcmp(module, prev_module))
225*c2e0c6b5SAndroid Build Coastguard Worker {
226*c2e0c6b5SAndroid Build Coastguard Worker strncpy(prev_module, module, sizeof(prev_module));
227*c2e0c6b5SAndroid Build Coastguard Worker prev_module[sizeof(prev_module) - 1] = 0;
228*c2e0c6b5SAndroid Build Coastguard Worker return module;
229*c2e0c6b5SAndroid Build Coastguard Worker }
230*c2e0c6b5SAndroid Build Coastguard Worker }
231*c2e0c6b5SAndroid Build Coastguard Worker prev_module[0] = 0;
232*c2e0c6b5SAndroid Build Coastguard Worker return NULL;
233*c2e0c6b5SAndroid Build Coastguard Worker }
234*c2e0c6b5SAndroid Build Coastguard Worker
235*c2e0c6b5SAndroid Build Coastguard Worker void
show_kernel(struct device * d)236*c2e0c6b5SAndroid Build Coastguard Worker show_kernel(struct device *d)
237*c2e0c6b5SAndroid Build Coastguard Worker {
238*c2e0c6b5SAndroid Build Coastguard Worker const char *driver, *module;
239*c2e0c6b5SAndroid Build Coastguard Worker
240*c2e0c6b5SAndroid Build Coastguard Worker pci_fill_info(d->dev, PCI_FILL_DRIVER);
241*c2e0c6b5SAndroid Build Coastguard Worker if (driver = pci_get_string_property(d->dev, PCI_FILL_DRIVER))
242*c2e0c6b5SAndroid Build Coastguard Worker printf("\tKernel driver in use: %s\n", driver);
243*c2e0c6b5SAndroid Build Coastguard Worker
244*c2e0c6b5SAndroid Build Coastguard Worker if (!show_kernel_init())
245*c2e0c6b5SAndroid Build Coastguard Worker return;
246*c2e0c6b5SAndroid Build Coastguard Worker
247*c2e0c6b5SAndroid Build Coastguard Worker int cnt = 0;
248*c2e0c6b5SAndroid Build Coastguard Worker while (module = next_module_filtered(d))
249*c2e0c6b5SAndroid Build Coastguard Worker printf("%s %s", (cnt++ ? "," : "\tKernel modules:"), module);
250*c2e0c6b5SAndroid Build Coastguard Worker if (cnt)
251*c2e0c6b5SAndroid Build Coastguard Worker putchar('\n');
252*c2e0c6b5SAndroid Build Coastguard Worker }
253*c2e0c6b5SAndroid Build Coastguard Worker
254*c2e0c6b5SAndroid Build Coastguard Worker void
show_kernel_machine(struct device * d)255*c2e0c6b5SAndroid Build Coastguard Worker show_kernel_machine(struct device *d)
256*c2e0c6b5SAndroid Build Coastguard Worker {
257*c2e0c6b5SAndroid Build Coastguard Worker const char *driver, *module;
258*c2e0c6b5SAndroid Build Coastguard Worker
259*c2e0c6b5SAndroid Build Coastguard Worker pci_fill_info(d->dev, PCI_FILL_DRIVER);
260*c2e0c6b5SAndroid Build Coastguard Worker if (driver = pci_get_string_property(d->dev, PCI_FILL_DRIVER))
261*c2e0c6b5SAndroid Build Coastguard Worker printf("Driver:\t%s\n", driver);
262*c2e0c6b5SAndroid Build Coastguard Worker
263*c2e0c6b5SAndroid Build Coastguard Worker if (!show_kernel_init())
264*c2e0c6b5SAndroid Build Coastguard Worker return;
265*c2e0c6b5SAndroid Build Coastguard Worker
266*c2e0c6b5SAndroid Build Coastguard Worker while (module = next_module_filtered(d))
267*c2e0c6b5SAndroid Build Coastguard Worker printf("Module:\t%s\n", module);
268*c2e0c6b5SAndroid Build Coastguard Worker }
269*c2e0c6b5SAndroid Build Coastguard Worker
270*c2e0c6b5SAndroid Build Coastguard Worker #else
271*c2e0c6b5SAndroid Build Coastguard Worker
272*c2e0c6b5SAndroid Build Coastguard Worker void
show_kernel(struct device * d)273*c2e0c6b5SAndroid Build Coastguard Worker show_kernel(struct device *d)
274*c2e0c6b5SAndroid Build Coastguard Worker {
275*c2e0c6b5SAndroid Build Coastguard Worker const char *driver;
276*c2e0c6b5SAndroid Build Coastguard Worker
277*c2e0c6b5SAndroid Build Coastguard Worker pci_fill_info(d->dev, PCI_FILL_DRIVER);
278*c2e0c6b5SAndroid Build Coastguard Worker if (driver = pci_get_string_property(d->dev, PCI_FILL_DRIVER))
279*c2e0c6b5SAndroid Build Coastguard Worker printf("\tDriver in use: %s\n", driver);
280*c2e0c6b5SAndroid Build Coastguard Worker }
281*c2e0c6b5SAndroid Build Coastguard Worker
282*c2e0c6b5SAndroid Build Coastguard Worker void
show_kernel_machine(struct device * d)283*c2e0c6b5SAndroid Build Coastguard Worker show_kernel_machine(struct device *d)
284*c2e0c6b5SAndroid Build Coastguard Worker {
285*c2e0c6b5SAndroid Build Coastguard Worker const char *driver;
286*c2e0c6b5SAndroid Build Coastguard Worker
287*c2e0c6b5SAndroid Build Coastguard Worker pci_fill_info(d->dev, PCI_FILL_DRIVER);
288*c2e0c6b5SAndroid Build Coastguard Worker if (driver = pci_get_string_property(d->dev, PCI_FILL_DRIVER))
289*c2e0c6b5SAndroid Build Coastguard Worker printf("Driver:\t%s\n", driver);
290*c2e0c6b5SAndroid Build Coastguard Worker }
291*c2e0c6b5SAndroid Build Coastguard Worker
292*c2e0c6b5SAndroid Build Coastguard Worker void
show_kernel_cleanup(void)293*c2e0c6b5SAndroid Build Coastguard Worker show_kernel_cleanup(void)
294*c2e0c6b5SAndroid Build Coastguard Worker {
295*c2e0c6b5SAndroid Build Coastguard Worker }
296*c2e0c6b5SAndroid Build Coastguard Worker
297*c2e0c6b5SAndroid Build Coastguard Worker #endif
298*c2e0c6b5SAndroid Build Coastguard Worker
299