xref: /freebsd/sys/dev/pci/pci_user.c (revision 727de621c5f07681034eed2d6ce3662a8239e987)
1aad970f1SDavid E. O'Brien /*-
24d846d26SWarner Losh  * SPDX-License-Identifier: BSD-2-Clause
3718cf2ccSPedro F. Giffuni  *
4fce5d19dSStefan Eßer  * Copyright 1997, Stefan Esser <se@freebsd.org>
58983cfbfSMike Smith  *
68983cfbfSMike Smith  * Redistribution and use in source and binary forms, with or without
78983cfbfSMike Smith  * modification, are permitted provided that the following conditions
88983cfbfSMike Smith  * are met:
98983cfbfSMike Smith  * 1. Redistributions of source code must retain the above copyright
108983cfbfSMike Smith  *    notice unmodified, this list of conditions, and the following
118983cfbfSMike Smith  *    disclaimer.
128983cfbfSMike Smith  * 2. Redistributions in binary form must reproduce the above copyright
138983cfbfSMike Smith  *    notice, this list of conditions and the following disclaimer in the
148983cfbfSMike Smith  *    documentation and/or other materials provided with the distribution.
158983cfbfSMike Smith  *
168983cfbfSMike Smith  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
178983cfbfSMike Smith  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
188983cfbfSMike Smith  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
198983cfbfSMike Smith  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
208983cfbfSMike Smith  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
218983cfbfSMike Smith  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
228983cfbfSMike Smith  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
238983cfbfSMike Smith  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
248983cfbfSMike Smith  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
258983cfbfSMike Smith  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
268983cfbfSMike Smith  */
278983cfbfSMike Smith 
28aad970f1SDavid E. O'Brien #include <sys/cdefs.h>
298983cfbfSMike Smith #include "opt_bus.h"	/* XXX trim includes */
308983cfbfSMike Smith 
3187842989SKonstantin Belousov #include <sys/types.h>
328983cfbfSMike Smith #include <sys/param.h>
338983cfbfSMike Smith #include <sys/systm.h>
348983cfbfSMike Smith #include <sys/malloc.h>
358983cfbfSMike Smith #include <sys/module.h>
368983cfbfSMike Smith #include <sys/linker.h>
378983cfbfSMike Smith #include <sys/fcntl.h>
388983cfbfSMike Smith #include <sys/conf.h>
398983cfbfSMike Smith #include <sys/kernel.h>
4087842989SKonstantin Belousov #include <sys/mman.h>
418002488bSRobert Watson #include <sys/proc.h>
428983cfbfSMike Smith #include <sys/queue.h>
4387842989SKonstantin Belousov #include <sys/rwlock.h>
4487842989SKonstantin Belousov #include <sys/sglist.h>
458983cfbfSMike Smith 
468983cfbfSMike Smith #include <vm/vm.h>
478983cfbfSMike Smith #include <vm/pmap.h>
488983cfbfSMike Smith #include <vm/vm_extern.h>
4987842989SKonstantin Belousov #include <vm/vm_map.h>
5087842989SKonstantin Belousov #include <vm/vm_object.h>
5187842989SKonstantin Belousov #include <vm/vm_page.h>
5287842989SKonstantin Belousov #include <vm/vm_pager.h>
538983cfbfSMike Smith 
548983cfbfSMike Smith #include <sys/bus.h>
558983cfbfSMike Smith #include <machine/bus.h>
568983cfbfSMike Smith #include <sys/rman.h>
578983cfbfSMike Smith #include <machine/resource.h>
588983cfbfSMike Smith 
598983cfbfSMike Smith #include <sys/pciio.h>
6038d8c994SWarner Losh #include <dev/pci/pcireg.h>
6138d8c994SWarner Losh #include <dev/pci/pcivar.h>
628983cfbfSMike Smith 
638983cfbfSMike Smith #include "pcib_if.h"
648983cfbfSMike Smith #include "pci_if.h"
658983cfbfSMike Smith 
66b7edb6faSBrooks Davis #ifdef COMPAT_FREEBSD32
67b7edb6faSBrooks Davis struct pci_conf32 {
68b7edb6faSBrooks Davis 	struct pcisel	pc_sel;		/* domain+bus+slot+function */
69b7edb6faSBrooks Davis 	u_int8_t	pc_hdr;		/* PCI header type */
70b7edb6faSBrooks Davis 	u_int16_t	pc_subvendor;	/* card vendor ID */
71b7edb6faSBrooks Davis 	u_int16_t	pc_subdevice;	/* card device ID, assigned by
72b7edb6faSBrooks Davis 					   card vendor */
73b7edb6faSBrooks Davis 	u_int16_t	pc_vendor;	/* chip vendor ID */
74b7edb6faSBrooks Davis 	u_int16_t	pc_device;	/* chip device ID, assigned by
75b7edb6faSBrooks Davis 					   chip vendor */
76b7edb6faSBrooks Davis 	u_int8_t	pc_class;	/* chip PCI class */
77b7edb6faSBrooks Davis 	u_int8_t	pc_subclass;	/* chip PCI subclass */
78b7edb6faSBrooks Davis 	u_int8_t	pc_progif;	/* chip PCI programming interface */
79b7edb6faSBrooks Davis 	u_int8_t	pc_revid;	/* chip revision ID */
80b7edb6faSBrooks Davis 	char		pd_name[PCI_MAXNAMELEN + 1];  /* device name */
81b7edb6faSBrooks Davis 	u_int32_t	pd_unit;	/* device unit number */
82b7edb6faSBrooks Davis };
83b7edb6faSBrooks Davis 
84b7edb6faSBrooks Davis struct pci_match_conf32 {
85b7edb6faSBrooks Davis 	struct pcisel		pc_sel;		/* domain+bus+slot+function */
86b7edb6faSBrooks Davis 	char			pd_name[PCI_MAXNAMELEN + 1];  /* device name */
87b7edb6faSBrooks Davis 	u_int32_t		pd_unit;	/* Unit number */
88b7edb6faSBrooks Davis 	u_int16_t		pc_vendor;	/* PCI Vendor ID */
89b7edb6faSBrooks Davis 	u_int16_t		pc_device;	/* PCI Device ID */
90b7edb6faSBrooks Davis 	u_int8_t		pc_class;	/* PCI class */
91b7edb6faSBrooks Davis 	u_int32_t		flags;		/* Matching expression */
92b7edb6faSBrooks Davis };
93b7edb6faSBrooks Davis 
94b7edb6faSBrooks Davis struct pci_conf_io32 {
95b7edb6faSBrooks Davis 	u_int32_t		pat_buf_len;	/* pattern buffer length */
96b7edb6faSBrooks Davis 	u_int32_t		num_patterns;	/* number of patterns */
97b7edb6faSBrooks Davis 	u_int32_t		patterns;	/* struct pci_match_conf ptr */
98b7edb6faSBrooks Davis 	u_int32_t		match_buf_len;	/* match buffer length */
99b7edb6faSBrooks Davis 	u_int32_t		num_matches;	/* number of matches returned */
100b7edb6faSBrooks Davis 	u_int32_t		matches;	/* struct pci_conf ptr */
101b7edb6faSBrooks Davis 	u_int32_t		offset;		/* offset into device list */
102b7edb6faSBrooks Davis 	u_int32_t		generation;	/* device list generation */
103b7edb6faSBrooks Davis 	u_int32_t		status;		/* request status */
104b7edb6faSBrooks Davis };
105b7edb6faSBrooks Davis 
106b7edb6faSBrooks Davis #define	PCIOCGETCONF32	_IOC_NEWTYPE(PCIOCGETCONF, struct pci_conf_io32)
107b7edb6faSBrooks Davis #endif
108b7edb6faSBrooks Davis 
1098983cfbfSMike Smith /*
1108983cfbfSMike Smith  * This is the user interface to PCI configuration space.
1118983cfbfSMike Smith  */
1128983cfbfSMike Smith 
113b40ce416SJulian Elischer static d_open_t 	pci_open;
114b40ce416SJulian Elischer static d_close_t	pci_close;
115b40ce416SJulian Elischer static d_ioctl_t	pci_ioctl;
1168983cfbfSMike Smith 
1178983cfbfSMike Smith struct cdevsw pcicdev = {
118dc08ffecSPoul-Henning Kamp 	.d_version =	D_VERSION,
119dd615d09SWarner Losh 	.d_flags =	0,
1207ac40f5fSPoul-Henning Kamp 	.d_open =	pci_open,
1217ac40f5fSPoul-Henning Kamp 	.d_close =	pci_close,
1227ac40f5fSPoul-Henning Kamp 	.d_ioctl =	pci_ioctl,
1237ac40f5fSPoul-Henning Kamp 	.d_name =	"pci",
1248983cfbfSMike Smith };
1258983cfbfSMike Smith 
1268983cfbfSMike Smith static int
pci_open(struct cdev * dev,int oflags,int devtype,struct thread * td)12789c9c53dSPoul-Henning Kamp pci_open(struct cdev *dev, int oflags, int devtype, struct thread *td)
1288983cfbfSMike Smith {
1298002488bSRobert Watson 	int error;
1308002488bSRobert Watson 
1318002488bSRobert Watson 	if (oflags & FWRITE) {
132a854ed98SJohn Baldwin 		error = securelevel_gt(td->td_ucred, 0);
1338002488bSRobert Watson 		if (error)
1348002488bSRobert Watson 			return (error);
1358983cfbfSMike Smith 	}
1368002488bSRobert Watson 
1378002488bSRobert Watson 	return (0);
1388983cfbfSMike Smith }
1398983cfbfSMike Smith 
1408983cfbfSMike Smith static int
pci_close(struct cdev * dev,int flag,int devtype,struct thread * td)14189c9c53dSPoul-Henning Kamp pci_close(struct cdev *dev, int flag, int devtype, struct thread *td)
1428983cfbfSMike Smith {
1438983cfbfSMike Smith 	return 0;
1448983cfbfSMike Smith }
1458983cfbfSMike Smith 
1468983cfbfSMike Smith /*
1478983cfbfSMike Smith  * Match a single pci_conf structure against an array of pci_match_conf
1488983cfbfSMike Smith  * structures.  The first argument, 'matches', is an array of num_matches
1498983cfbfSMike Smith  * pci_match_conf structures.  match_buf is a pointer to the pci_conf
1508983cfbfSMike Smith  * structure that will be compared to every entry in the matches array.
1518983cfbfSMike Smith  * This function returns 1 on failure, 0 on success.
1528983cfbfSMike Smith  */
1538983cfbfSMike Smith static int
pci_conf_match_native(struct pci_match_conf * matches,int num_matches,struct pci_conf * match_buf)154e9ed3a70SBrooks Davis pci_conf_match_native(struct pci_match_conf *matches, int num_matches,
1558983cfbfSMike Smith 	       struct pci_conf *match_buf)
1568983cfbfSMike Smith {
1578983cfbfSMike Smith 	int i;
1588983cfbfSMike Smith 
1598983cfbfSMike Smith 	if ((matches == NULL) || (match_buf == NULL) || (num_matches <= 0))
1608983cfbfSMike Smith 		return(1);
1618983cfbfSMike Smith 
1628983cfbfSMike Smith 	for (i = 0; i < num_matches; i++) {
1638983cfbfSMike Smith 		/*
1648983cfbfSMike Smith 		 * I'm not sure why someone would do this...but...
1658983cfbfSMike Smith 		 */
1668983cfbfSMike Smith 		if (matches[i].flags == PCI_GETCONF_NO_MATCH)
1678983cfbfSMike Smith 			continue;
1688983cfbfSMike Smith 
1698983cfbfSMike Smith 		/*
1708983cfbfSMike Smith 		 * Look at each of the match flags.  If it's set, do the
1718983cfbfSMike Smith 		 * comparison.  If the comparison fails, we don't have a
1728983cfbfSMike Smith 		 * match, go on to the next item if there is one.
1738983cfbfSMike Smith 		 */
17455aaf894SMarius Strobl 		if (((matches[i].flags & PCI_GETCONF_MATCH_DOMAIN) != 0)
17555aaf894SMarius Strobl 		 && (match_buf->pc_sel.pc_domain !=
17655aaf894SMarius Strobl 		 matches[i].pc_sel.pc_domain))
17755aaf894SMarius Strobl 			continue;
17855aaf894SMarius Strobl 
1798983cfbfSMike Smith 		if (((matches[i].flags & PCI_GETCONF_MATCH_BUS) != 0)
1808983cfbfSMike Smith 		 && (match_buf->pc_sel.pc_bus != matches[i].pc_sel.pc_bus))
1818983cfbfSMike Smith 			continue;
1828983cfbfSMike Smith 
1838983cfbfSMike Smith 		if (((matches[i].flags & PCI_GETCONF_MATCH_DEV) != 0)
1848983cfbfSMike Smith 		 && (match_buf->pc_sel.pc_dev != matches[i].pc_sel.pc_dev))
1858983cfbfSMike Smith 			continue;
1868983cfbfSMike Smith 
1878983cfbfSMike Smith 		if (((matches[i].flags & PCI_GETCONF_MATCH_FUNC) != 0)
1888983cfbfSMike Smith 		 && (match_buf->pc_sel.pc_func != matches[i].pc_sel.pc_func))
1898983cfbfSMike Smith 			continue;
1908983cfbfSMike Smith 
1918983cfbfSMike Smith 		if (((matches[i].flags & PCI_GETCONF_MATCH_VENDOR) != 0)
1928983cfbfSMike Smith 		 && (match_buf->pc_vendor != matches[i].pc_vendor))
1938983cfbfSMike Smith 			continue;
1948983cfbfSMike Smith 
1958983cfbfSMike Smith 		if (((matches[i].flags & PCI_GETCONF_MATCH_DEVICE) != 0)
1968983cfbfSMike Smith 		 && (match_buf->pc_device != matches[i].pc_device))
1978983cfbfSMike Smith 			continue;
1988983cfbfSMike Smith 
1998983cfbfSMike Smith 		if (((matches[i].flags & PCI_GETCONF_MATCH_CLASS) != 0)
2008983cfbfSMike Smith 		 && (match_buf->pc_class != matches[i].pc_class))
2018983cfbfSMike Smith 			continue;
2028983cfbfSMike Smith 
2038983cfbfSMike Smith 		if (((matches[i].flags & PCI_GETCONF_MATCH_UNIT) != 0)
2048983cfbfSMike Smith 		 && (match_buf->pd_unit != matches[i].pd_unit))
2058983cfbfSMike Smith 			continue;
2068983cfbfSMike Smith 
2078983cfbfSMike Smith 		if (((matches[i].flags & PCI_GETCONF_MATCH_NAME) != 0)
2088983cfbfSMike Smith 		 && (strncmp(matches[i].pd_name, match_buf->pd_name,
2098983cfbfSMike Smith 			     sizeof(match_buf->pd_name)) != 0))
2108983cfbfSMike Smith 			continue;
2118983cfbfSMike Smith 
2128983cfbfSMike Smith 		return(0);
2138983cfbfSMike Smith 	}
2148983cfbfSMike Smith 
2158983cfbfSMike Smith 	return(1);
2168983cfbfSMike Smith }
2178983cfbfSMike Smith 
218b7edb6faSBrooks Davis #ifdef COMPAT_FREEBSD32
219b7edb6faSBrooks Davis static int
pci_conf_match32(struct pci_match_conf32 * matches,int num_matches,struct pci_conf * match_buf)220b7edb6faSBrooks Davis pci_conf_match32(struct pci_match_conf32 *matches, int num_matches,
221b7edb6faSBrooks Davis 	       struct pci_conf *match_buf)
222b7edb6faSBrooks Davis {
223b7edb6faSBrooks Davis 	int i;
224b7edb6faSBrooks Davis 
225b7edb6faSBrooks Davis 	if ((matches == NULL) || (match_buf == NULL) || (num_matches <= 0))
226b7edb6faSBrooks Davis 		return(1);
227b7edb6faSBrooks Davis 
228b7edb6faSBrooks Davis 	for (i = 0; i < num_matches; i++) {
229b7edb6faSBrooks Davis 		/*
230b7edb6faSBrooks Davis 		 * I'm not sure why someone would do this...but...
231b7edb6faSBrooks Davis 		 */
232b7edb6faSBrooks Davis 		if (matches[i].flags == PCI_GETCONF_NO_MATCH)
233b7edb6faSBrooks Davis 			continue;
234b7edb6faSBrooks Davis 
235b7edb6faSBrooks Davis 		/*
236b7edb6faSBrooks Davis 		 * Look at each of the match flags.  If it's set, do the
237b7edb6faSBrooks Davis 		 * comparison.  If the comparison fails, we don't have a
238b7edb6faSBrooks Davis 		 * match, go on to the next item if there is one.
239b7edb6faSBrooks Davis 		 */
240b7edb6faSBrooks Davis 		if (((matches[i].flags & PCI_GETCONF_MATCH_DOMAIN) != 0)
241b7edb6faSBrooks Davis 		 && (match_buf->pc_sel.pc_domain !=
242b7edb6faSBrooks Davis 		 matches[i].pc_sel.pc_domain))
243b7edb6faSBrooks Davis 			continue;
244b7edb6faSBrooks Davis 
245b7edb6faSBrooks Davis 		if (((matches[i].flags & PCI_GETCONF_MATCH_BUS) != 0)
246b7edb6faSBrooks Davis 		 && (match_buf->pc_sel.pc_bus != matches[i].pc_sel.pc_bus))
247b7edb6faSBrooks Davis 			continue;
248b7edb6faSBrooks Davis 
249b7edb6faSBrooks Davis 		if (((matches[i].flags & PCI_GETCONF_MATCH_DEV) != 0)
250b7edb6faSBrooks Davis 		 && (match_buf->pc_sel.pc_dev != matches[i].pc_sel.pc_dev))
251b7edb6faSBrooks Davis 			continue;
252b7edb6faSBrooks Davis 
253b7edb6faSBrooks Davis 		if (((matches[i].flags & PCI_GETCONF_MATCH_FUNC) != 0)
254b7edb6faSBrooks Davis 		 && (match_buf->pc_sel.pc_func != matches[i].pc_sel.pc_func))
255b7edb6faSBrooks Davis 			continue;
256b7edb6faSBrooks Davis 
257b7edb6faSBrooks Davis 		if (((matches[i].flags & PCI_GETCONF_MATCH_VENDOR) != 0)
258b7edb6faSBrooks Davis 		 && (match_buf->pc_vendor != matches[i].pc_vendor))
259b7edb6faSBrooks Davis 			continue;
260b7edb6faSBrooks Davis 
261b7edb6faSBrooks Davis 		if (((matches[i].flags & PCI_GETCONF_MATCH_DEVICE) != 0)
262b7edb6faSBrooks Davis 		 && (match_buf->pc_device != matches[i].pc_device))
263b7edb6faSBrooks Davis 			continue;
264b7edb6faSBrooks Davis 
265b7edb6faSBrooks Davis 		if (((matches[i].flags & PCI_GETCONF_MATCH_CLASS) != 0)
266b7edb6faSBrooks Davis 		 && (match_buf->pc_class != matches[i].pc_class))
267b7edb6faSBrooks Davis 			continue;
268b7edb6faSBrooks Davis 
269b7edb6faSBrooks Davis 		if (((matches[i].flags & PCI_GETCONF_MATCH_UNIT) != 0)
270b7edb6faSBrooks Davis 		 && (match_buf->pd_unit != matches[i].pd_unit))
271b7edb6faSBrooks Davis 			continue;
272b7edb6faSBrooks Davis 
273b7edb6faSBrooks Davis 		if (((matches[i].flags & PCI_GETCONF_MATCH_NAME) != 0)
274b7edb6faSBrooks Davis 		 && (strncmp(matches[i].pd_name, match_buf->pd_name,
275b7edb6faSBrooks Davis 			     sizeof(match_buf->pd_name)) != 0))
276b7edb6faSBrooks Davis 			continue;
277b7edb6faSBrooks Davis 
278b7edb6faSBrooks Davis 		return(0);
279b7edb6faSBrooks Davis 	}
280b7edb6faSBrooks Davis 
281b7edb6faSBrooks Davis 	return(1);
282b7edb6faSBrooks Davis }
283b7edb6faSBrooks Davis #endif	/* COMPAT_FREEBSD32 */
284b7edb6faSBrooks Davis 
28533d3fffaSMarius Strobl #if defined(COMPAT_FREEBSD4) || defined(COMPAT_FREEBSD5) || \
28633d3fffaSMarius Strobl     defined(COMPAT_FREEBSD6)
287b2068c0cSWarner Losh #define PRE7_COMPAT
28833d3fffaSMarius Strobl 
28933d3fffaSMarius Strobl typedef enum {
29056ecc8a9SWarner Losh 	PCI_GETCONF_NO_MATCH_FREEBSD6	= 0x00,
29156ecc8a9SWarner Losh 	PCI_GETCONF_MATCH_BUS_FREEBSD6	= 0x01,
29256ecc8a9SWarner Losh 	PCI_GETCONF_MATCH_DEV_FREEBSD6	= 0x02,
29356ecc8a9SWarner Losh 	PCI_GETCONF_MATCH_FUNC_FREEBSD6	= 0x04,
29456ecc8a9SWarner Losh 	PCI_GETCONF_MATCH_NAME_FREEBSD6	= 0x08,
29556ecc8a9SWarner Losh 	PCI_GETCONF_MATCH_UNIT_FREEBSD6	= 0x10,
29656ecc8a9SWarner Losh 	PCI_GETCONF_MATCH_VENDOR_FREEBSD6 = 0x20,
29756ecc8a9SWarner Losh 	PCI_GETCONF_MATCH_DEVICE_FREEBSD6 = 0x40,
29856ecc8a9SWarner Losh 	PCI_GETCONF_MATCH_CLASS_FREEBSD6 = 0x80
29956ecc8a9SWarner Losh } pci_getconf_flags_freebsd6;
30033d3fffaSMarius Strobl 
30156ecc8a9SWarner Losh struct pcisel_freebsd6 {
30233d3fffaSMarius Strobl 	u_int8_t	pc_bus;		/* bus number */
30333d3fffaSMarius Strobl 	u_int8_t	pc_dev;		/* device on this bus */
30433d3fffaSMarius Strobl 	u_int8_t	pc_func;	/* function on this device */
30533d3fffaSMarius Strobl };
30633d3fffaSMarius Strobl 
30756ecc8a9SWarner Losh struct pci_conf_freebsd6 {
30856ecc8a9SWarner Losh 	struct pcisel_freebsd6 pc_sel;	/* bus+slot+function */
30933d3fffaSMarius Strobl 	u_int8_t	pc_hdr;		/* PCI header type */
31033d3fffaSMarius Strobl 	u_int16_t	pc_subvendor;	/* card vendor ID */
31133d3fffaSMarius Strobl 	u_int16_t	pc_subdevice;	/* card device ID, assigned by
31233d3fffaSMarius Strobl 					   card vendor */
31333d3fffaSMarius Strobl 	u_int16_t	pc_vendor;	/* chip vendor ID */
31433d3fffaSMarius Strobl 	u_int16_t	pc_device;	/* chip device ID, assigned by
31533d3fffaSMarius Strobl 					   chip vendor */
31633d3fffaSMarius Strobl 	u_int8_t	pc_class;	/* chip PCI class */
31733d3fffaSMarius Strobl 	u_int8_t	pc_subclass;	/* chip PCI subclass */
31833d3fffaSMarius Strobl 	u_int8_t	pc_progif;	/* chip PCI programming interface */
31933d3fffaSMarius Strobl 	u_int8_t	pc_revid;	/* chip revision ID */
32033d3fffaSMarius Strobl 	char		pd_name[PCI_MAXNAMELEN + 1];  /* device name */
32133d3fffaSMarius Strobl 	u_long		pd_unit;	/* device unit number */
32233d3fffaSMarius Strobl };
32333d3fffaSMarius Strobl 
32456ecc8a9SWarner Losh struct pci_match_conf_freebsd6 {
32556ecc8a9SWarner Losh 	struct pcisel_freebsd6	pc_sel;		/* bus+slot+function */
32633d3fffaSMarius Strobl 	char			pd_name[PCI_MAXNAMELEN + 1];  /* device name */
32733d3fffaSMarius Strobl 	u_long			pd_unit;	/* Unit number */
32833d3fffaSMarius Strobl 	u_int16_t		pc_vendor;	/* PCI Vendor ID */
32933d3fffaSMarius Strobl 	u_int16_t		pc_device;	/* PCI Device ID */
33033d3fffaSMarius Strobl 	u_int8_t		pc_class;	/* PCI class */
33156ecc8a9SWarner Losh 	pci_getconf_flags_freebsd6 flags;	/* Matching expression */
33233d3fffaSMarius Strobl };
33333d3fffaSMarius Strobl 
33456ecc8a9SWarner Losh struct pci_io_freebsd6 {
33556ecc8a9SWarner Losh 	struct pcisel_freebsd6 pi_sel;	/* device to operate on */
33633d3fffaSMarius Strobl 	int		pi_reg;		/* configuration register to examine */
33733d3fffaSMarius Strobl 	int		pi_width;	/* width (in bytes) of read or write */
33833d3fffaSMarius Strobl 	u_int32_t	pi_data;	/* data to write or result of read */
33933d3fffaSMarius Strobl };
34033d3fffaSMarius Strobl 
341b01bf72bSMaxim Sobolev #ifdef COMPAT_FREEBSD32
34256ecc8a9SWarner Losh struct pci_conf_freebsd6_32 {
34356ecc8a9SWarner Losh 	struct pcisel_freebsd6 pc_sel;	/* bus+slot+function */
344e5280830SGleb Smirnoff 	uint8_t		pc_hdr;		/* PCI header type */
345e5280830SGleb Smirnoff 	uint16_t	pc_subvendor;	/* card vendor ID */
346e5280830SGleb Smirnoff 	uint16_t	pc_subdevice;	/* card device ID, assigned by
347b01bf72bSMaxim Sobolev 					   card vendor */
348e5280830SGleb Smirnoff 	uint16_t	pc_vendor;	/* chip vendor ID */
349e5280830SGleb Smirnoff 	uint16_t	pc_device;	/* chip device ID, assigned by
350b01bf72bSMaxim Sobolev 					   chip vendor */
351e5280830SGleb Smirnoff 	uint8_t		pc_class;	/* chip PCI class */
352e5280830SGleb Smirnoff 	uint8_t		pc_subclass;	/* chip PCI subclass */
353e5280830SGleb Smirnoff 	uint8_t		pc_progif;	/* chip PCI programming interface */
354e5280830SGleb Smirnoff 	uint8_t		pc_revid;	/* chip revision ID */
355b01bf72bSMaxim Sobolev 	char		pd_name[PCI_MAXNAMELEN + 1]; /* device name */
356e5280830SGleb Smirnoff 	uint32_t	pd_unit;	/* device unit number (u_long) */
357b01bf72bSMaxim Sobolev };
358b01bf72bSMaxim Sobolev 
35956ecc8a9SWarner Losh struct pci_match_conf_freebsd6_32 {
36056ecc8a9SWarner Losh 	struct pcisel_freebsd6 pc_sel;	/* bus+slot+function */
361b01bf72bSMaxim Sobolev 	char		pd_name[PCI_MAXNAMELEN + 1]; /* device name */
362e5280830SGleb Smirnoff 	uint32_t	pd_unit;	/* Unit number (u_long) */
363e5280830SGleb Smirnoff 	uint16_t	pc_vendor;	/* PCI Vendor ID */
364e5280830SGleb Smirnoff 	uint16_t	pc_device;	/* PCI Device ID */
365e5280830SGleb Smirnoff 	uint8_t		pc_class;	/* PCI class */
36656ecc8a9SWarner Losh 	pci_getconf_flags_freebsd6 flags; /* Matching expression */
367b01bf72bSMaxim Sobolev };
368b01bf72bSMaxim Sobolev 
36956ecc8a9SWarner Losh #define	PCIOCGETCONF_FREEBSD6_32	_IOWR('p', 1, struct pci_conf_io32)
370904c3909SGleb Smirnoff #endif	/* COMPAT_FREEBSD32 */
371b01bf72bSMaxim Sobolev 
37256ecc8a9SWarner Losh #define	PCIOCGETCONF_FREEBSD6	_IOWR('p', 1, struct pci_conf_io)
37356ecc8a9SWarner Losh #define	PCIOCREAD_FREEBSD6	_IOWR('p', 2, struct pci_io_freebsd6)
37456ecc8a9SWarner Losh #define	PCIOCWRITE_FREEBSD6	_IOWR('p', 3, struct pci_io_freebsd6)
37533d3fffaSMarius Strobl 
37633d3fffaSMarius Strobl static int
pci_conf_match_freebsd6(struct pci_match_conf_freebsd6 * matches,int num_matches,struct pci_conf * match_buf)37756ecc8a9SWarner Losh pci_conf_match_freebsd6(struct pci_match_conf_freebsd6 *matches, int num_matches,
37833d3fffaSMarius Strobl     struct pci_conf *match_buf)
37933d3fffaSMarius Strobl {
38033d3fffaSMarius Strobl 	int i;
38133d3fffaSMarius Strobl 
38233d3fffaSMarius Strobl 	if ((matches == NULL) || (match_buf == NULL) || (num_matches <= 0))
38333d3fffaSMarius Strobl 		return(1);
38433d3fffaSMarius Strobl 
38533d3fffaSMarius Strobl 	for (i = 0; i < num_matches; i++) {
38633d3fffaSMarius Strobl 		if (match_buf->pc_sel.pc_domain != 0)
38733d3fffaSMarius Strobl 			continue;
38833d3fffaSMarius Strobl 
38933d3fffaSMarius Strobl 		/*
39033d3fffaSMarius Strobl 		 * I'm not sure why someone would do this...but...
39133d3fffaSMarius Strobl 		 */
39256ecc8a9SWarner Losh 		if (matches[i].flags == PCI_GETCONF_NO_MATCH_FREEBSD6)
39333d3fffaSMarius Strobl 			continue;
39433d3fffaSMarius Strobl 
39533d3fffaSMarius Strobl 		/*
39633d3fffaSMarius Strobl 		 * Look at each of the match flags.  If it's set, do the
39733d3fffaSMarius Strobl 		 * comparison.  If the comparison fails, we don't have a
39833d3fffaSMarius Strobl 		 * match, go on to the next item if there is one.
39933d3fffaSMarius Strobl 		 */
40056ecc8a9SWarner Losh 		if (((matches[i].flags & PCI_GETCONF_MATCH_BUS_FREEBSD6) != 0)
40133d3fffaSMarius Strobl 		 && (match_buf->pc_sel.pc_bus != matches[i].pc_sel.pc_bus))
40233d3fffaSMarius Strobl 			continue;
40333d3fffaSMarius Strobl 
40456ecc8a9SWarner Losh 		if (((matches[i].flags & PCI_GETCONF_MATCH_DEV_FREEBSD6) != 0)
40533d3fffaSMarius Strobl 		 && (match_buf->pc_sel.pc_dev != matches[i].pc_sel.pc_dev))
40633d3fffaSMarius Strobl 			continue;
40733d3fffaSMarius Strobl 
40856ecc8a9SWarner Losh 		if (((matches[i].flags & PCI_GETCONF_MATCH_FUNC_FREEBSD6) != 0)
40933d3fffaSMarius Strobl 		 && (match_buf->pc_sel.pc_func != matches[i].pc_sel.pc_func))
41033d3fffaSMarius Strobl 			continue;
41133d3fffaSMarius Strobl 
41256ecc8a9SWarner Losh 		if (((matches[i].flags & PCI_GETCONF_MATCH_VENDOR_FREEBSD6) != 0)
41333d3fffaSMarius Strobl 		 && (match_buf->pc_vendor != matches[i].pc_vendor))
41433d3fffaSMarius Strobl 			continue;
41533d3fffaSMarius Strobl 
41656ecc8a9SWarner Losh 		if (((matches[i].flags & PCI_GETCONF_MATCH_DEVICE_FREEBSD6) != 0)
41733d3fffaSMarius Strobl 		 && (match_buf->pc_device != matches[i].pc_device))
41833d3fffaSMarius Strobl 			continue;
41933d3fffaSMarius Strobl 
42056ecc8a9SWarner Losh 		if (((matches[i].flags & PCI_GETCONF_MATCH_CLASS_FREEBSD6) != 0)
42133d3fffaSMarius Strobl 		 && (match_buf->pc_class != matches[i].pc_class))
42233d3fffaSMarius Strobl 			continue;
42333d3fffaSMarius Strobl 
42456ecc8a9SWarner Losh 		if (((matches[i].flags & PCI_GETCONF_MATCH_UNIT_FREEBSD6) != 0)
42533d3fffaSMarius Strobl 		 && (match_buf->pd_unit != matches[i].pd_unit))
42633d3fffaSMarius Strobl 			continue;
42733d3fffaSMarius Strobl 
42856ecc8a9SWarner Losh 		if (((matches[i].flags & PCI_GETCONF_MATCH_NAME_FREEBSD6) != 0)
42933d3fffaSMarius Strobl 		 && (strncmp(matches[i].pd_name, match_buf->pd_name,
43033d3fffaSMarius Strobl 			     sizeof(match_buf->pd_name)) != 0))
43133d3fffaSMarius Strobl 			continue;
43233d3fffaSMarius Strobl 
43333d3fffaSMarius Strobl 		return(0);
43433d3fffaSMarius Strobl 	}
43533d3fffaSMarius Strobl 
43633d3fffaSMarius Strobl 	return(1);
43733d3fffaSMarius Strobl }
43833d3fffaSMarius Strobl 
439904c3909SGleb Smirnoff #ifdef COMPAT_FREEBSD32
440b01bf72bSMaxim Sobolev static int
pci_conf_match_freebsd6_32(struct pci_match_conf_freebsd6_32 * matches,int num_matches,struct pci_conf * match_buf)44156ecc8a9SWarner Losh pci_conf_match_freebsd6_32(struct pci_match_conf_freebsd6_32 *matches, int num_matches,
442b01bf72bSMaxim Sobolev     struct pci_conf *match_buf)
443b01bf72bSMaxim Sobolev {
444b01bf72bSMaxim Sobolev 	int i;
445b01bf72bSMaxim Sobolev 
446b01bf72bSMaxim Sobolev 	if ((matches == NULL) || (match_buf == NULL) || (num_matches <= 0))
447b01bf72bSMaxim Sobolev 		return(1);
448b01bf72bSMaxim Sobolev 
449b01bf72bSMaxim Sobolev 	for (i = 0; i < num_matches; i++) {
450b01bf72bSMaxim Sobolev 		if (match_buf->pc_sel.pc_domain != 0)
451b01bf72bSMaxim Sobolev 			continue;
452b01bf72bSMaxim Sobolev 
453b01bf72bSMaxim Sobolev 		/*
454b01bf72bSMaxim Sobolev 		 * I'm not sure why someone would do this...but...
455b01bf72bSMaxim Sobolev 		 */
45656ecc8a9SWarner Losh 		if (matches[i].flags == PCI_GETCONF_NO_MATCH_FREEBSD6)
457b01bf72bSMaxim Sobolev 			continue;
458b01bf72bSMaxim Sobolev 
459b01bf72bSMaxim Sobolev 		/*
460b01bf72bSMaxim Sobolev 		 * Look at each of the match flags.  If it's set, do the
461b01bf72bSMaxim Sobolev 		 * comparison.  If the comparison fails, we don't have a
462b01bf72bSMaxim Sobolev 		 * match, go on to the next item if there is one.
463b01bf72bSMaxim Sobolev 		 */
46456ecc8a9SWarner Losh 		if (((matches[i].flags & PCI_GETCONF_MATCH_BUS_FREEBSD6) != 0) &&
465e5280830SGleb Smirnoff 		    (match_buf->pc_sel.pc_bus != matches[i].pc_sel.pc_bus))
466b01bf72bSMaxim Sobolev 			continue;
467b01bf72bSMaxim Sobolev 
46856ecc8a9SWarner Losh 		if (((matches[i].flags & PCI_GETCONF_MATCH_DEV_FREEBSD6) != 0) &&
469e5280830SGleb Smirnoff 		    (match_buf->pc_sel.pc_dev != matches[i].pc_sel.pc_dev))
470b01bf72bSMaxim Sobolev 			continue;
471b01bf72bSMaxim Sobolev 
47256ecc8a9SWarner Losh 		if (((matches[i].flags & PCI_GETCONF_MATCH_FUNC_FREEBSD6) != 0) &&
473e5280830SGleb Smirnoff 		    (match_buf->pc_sel.pc_func != matches[i].pc_sel.pc_func))
474b01bf72bSMaxim Sobolev 			continue;
475b01bf72bSMaxim Sobolev 
47656ecc8a9SWarner Losh 		if (((matches[i].flags & PCI_GETCONF_MATCH_VENDOR_FREEBSD6) != 0) &&
477e5280830SGleb Smirnoff 		    (match_buf->pc_vendor != matches[i].pc_vendor))
478b01bf72bSMaxim Sobolev 			continue;
479b01bf72bSMaxim Sobolev 
48056ecc8a9SWarner Losh 		if (((matches[i].flags & PCI_GETCONF_MATCH_DEVICE_FREEBSD6) != 0) &&
481e5280830SGleb Smirnoff 		    (match_buf->pc_device != matches[i].pc_device))
482b01bf72bSMaxim Sobolev 			continue;
483b01bf72bSMaxim Sobolev 
48456ecc8a9SWarner Losh 		if (((matches[i].flags & PCI_GETCONF_MATCH_CLASS_FREEBSD6) != 0) &&
485e5280830SGleb Smirnoff 		    (match_buf->pc_class != matches[i].pc_class))
486b01bf72bSMaxim Sobolev 			continue;
487b01bf72bSMaxim Sobolev 
48856ecc8a9SWarner Losh 		if (((matches[i].flags & PCI_GETCONF_MATCH_UNIT_FREEBSD6) != 0) &&
489e5280830SGleb Smirnoff 		    ((u_int32_t)match_buf->pd_unit != matches[i].pd_unit))
490b01bf72bSMaxim Sobolev 			continue;
491b01bf72bSMaxim Sobolev 
49256ecc8a9SWarner Losh 		if (((matches[i].flags & PCI_GETCONF_MATCH_NAME_FREEBSD6) != 0) &&
493e5280830SGleb Smirnoff 		    (strncmp(matches[i].pd_name, match_buf->pd_name,
494b01bf72bSMaxim Sobolev 		    sizeof(match_buf->pd_name)) != 0))
495b01bf72bSMaxim Sobolev 			continue;
496b01bf72bSMaxim Sobolev 
497b01bf72bSMaxim Sobolev 		return (0);
498b01bf72bSMaxim Sobolev 	}
499b01bf72bSMaxim Sobolev 
500b01bf72bSMaxim Sobolev 	return (1);
501b01bf72bSMaxim Sobolev }
502904c3909SGleb Smirnoff #endif	/* COMPAT_FREEBSD32 */
503e9ed3a70SBrooks Davis #endif	/* !PRE7_COMPAT */
504e9ed3a70SBrooks Davis 
505e9ed3a70SBrooks Davis union pci_conf_union {
506e9ed3a70SBrooks Davis 	struct pci_conf			pc;
507b7edb6faSBrooks Davis #ifdef COMPAT_FREEBSD32
508b7edb6faSBrooks Davis 	struct pci_conf32		pc32;
509b7edb6faSBrooks Davis #endif
510e9ed3a70SBrooks Davis #ifdef PRE7_COMPAT
51156ecc8a9SWarner Losh 	struct pci_conf_freebsd6	pco;
512e9ed3a70SBrooks Davis #ifdef COMPAT_FREEBSD32
51356ecc8a9SWarner Losh 	struct pci_conf_freebsd6_32	pco32;
514e9ed3a70SBrooks Davis #endif
515e9ed3a70SBrooks Davis #endif
516e9ed3a70SBrooks Davis };
517e9ed3a70SBrooks Davis 
518e9ed3a70SBrooks Davis static int
pci_conf_match(u_long cmd,struct pci_match_conf * matches,int num_matches,struct pci_conf * match_buf)519e9ed3a70SBrooks Davis pci_conf_match(u_long cmd, struct pci_match_conf *matches, int num_matches,
520e9ed3a70SBrooks Davis     struct pci_conf *match_buf)
521e9ed3a70SBrooks Davis {
522e9ed3a70SBrooks Davis 
523e9ed3a70SBrooks Davis 	switch (cmd) {
524e9ed3a70SBrooks Davis 	case PCIOCGETCONF:
525e9ed3a70SBrooks Davis 		return (pci_conf_match_native(
526e9ed3a70SBrooks Davis 		    (struct pci_match_conf *)matches, num_matches, match_buf));
527b7edb6faSBrooks Davis #ifdef COMPAT_FREEBSD32
528b7edb6faSBrooks Davis 	case PCIOCGETCONF32:
529b7edb6faSBrooks Davis 		return (pci_conf_match32((struct pci_match_conf32 *)matches,
530b7edb6faSBrooks Davis 		    num_matches, match_buf));
531b7edb6faSBrooks Davis #endif
532e9ed3a70SBrooks Davis #ifdef PRE7_COMPAT
53356ecc8a9SWarner Losh 	case PCIOCGETCONF_FREEBSD6:
53456ecc8a9SWarner Losh 		return (pci_conf_match_freebsd6(
53556ecc8a9SWarner Losh 		    (struct pci_match_conf_freebsd6 *)matches, num_matches,
536e9ed3a70SBrooks Davis 		    match_buf));
537e9ed3a70SBrooks Davis #ifdef COMPAT_FREEBSD32
53856ecc8a9SWarner Losh 	case PCIOCGETCONF_FREEBSD6_32:
53956ecc8a9SWarner Losh 		return (pci_conf_match_freebsd6_32(
54056ecc8a9SWarner Losh 		    (struct pci_match_conf_freebsd6_32 *)matches, num_matches,
541e9ed3a70SBrooks Davis 		    match_buf));
542e9ed3a70SBrooks Davis #endif
543e9ed3a70SBrooks Davis #endif
544e9ed3a70SBrooks Davis 	default:
545e9ed3a70SBrooks Davis 		/* programmer error */
546e9ed3a70SBrooks Davis 		return (0);
547e9ed3a70SBrooks Davis 	}
548e9ed3a70SBrooks Davis }
54933d3fffaSMarius Strobl 
55074aa2d49SJohn Baldwin /*
55174aa2d49SJohn Baldwin  * Like PVE_NEXT but takes an explicit length since 'pve' is a user
55274aa2d49SJohn Baldwin  * pointer that cannot be dereferenced.
55374aa2d49SJohn Baldwin  */
55474aa2d49SJohn Baldwin #define	PVE_NEXT_LEN(pve, datalen)					\
55574aa2d49SJohn Baldwin 	((struct pci_vpd_element *)((char *)(pve) +			\
55674aa2d49SJohn Baldwin 	    sizeof(struct pci_vpd_element) + (datalen)))
55774aa2d49SJohn Baldwin 
5588983cfbfSMike Smith static int
pci_list_vpd(device_t dev,struct pci_list_vpd_io * lvio)55984b755dfSJohn Baldwin pci_list_vpd(device_t dev, struct pci_list_vpd_io *lvio)
56084b755dfSJohn Baldwin {
56184b755dfSJohn Baldwin 	struct pci_vpd_element vpd_element, *vpd_user;
56284b755dfSJohn Baldwin 	struct pcicfg_vpd *vpd;
563f01c8633SStefan Eßer 	size_t len, datalen;
56484b755dfSJohn Baldwin 	int error, i;
56584b755dfSJohn Baldwin 
56684b755dfSJohn Baldwin 	vpd = pci_fetch_vpd_list(dev);
56784b755dfSJohn Baldwin 	if (vpd->vpd_reg == 0 || vpd->vpd_ident == NULL)
56884b755dfSJohn Baldwin 		return (ENXIO);
56984b755dfSJohn Baldwin 
57084b755dfSJohn Baldwin 	/*
57184b755dfSJohn Baldwin 	 * Calculate the amount of space needed in the data buffer.  An
57284b755dfSJohn Baldwin 	 * identifier element is always present followed by the read-only
57384b755dfSJohn Baldwin 	 * and read-write keywords.
57484b755dfSJohn Baldwin 	 */
57584b755dfSJohn Baldwin 	len = sizeof(struct pci_vpd_element) + strlen(vpd->vpd_ident);
57684b755dfSJohn Baldwin 	for (i = 0; i < vpd->vpd_rocnt; i++)
57784b755dfSJohn Baldwin 		len += sizeof(struct pci_vpd_element) + vpd->vpd_ros[i].len;
57884b755dfSJohn Baldwin 	for (i = 0; i < vpd->vpd_wcnt; i++)
57984b755dfSJohn Baldwin 		len += sizeof(struct pci_vpd_element) + vpd->vpd_w[i].len;
58084b755dfSJohn Baldwin 
58184b755dfSJohn Baldwin 	if (lvio->plvi_len == 0) {
58284b755dfSJohn Baldwin 		lvio->plvi_len = len;
58384b755dfSJohn Baldwin 		return (0);
58484b755dfSJohn Baldwin 	}
58584b755dfSJohn Baldwin 	if (lvio->plvi_len < len) {
58684b755dfSJohn Baldwin 		lvio->plvi_len = len;
58784b755dfSJohn Baldwin 		return (ENOMEM);
58884b755dfSJohn Baldwin 	}
58984b755dfSJohn Baldwin 
59084b755dfSJohn Baldwin 	/*
59184b755dfSJohn Baldwin 	 * Copyout the identifier string followed by each keyword and
59284b755dfSJohn Baldwin 	 * value.
59384b755dfSJohn Baldwin 	 */
594f01c8633SStefan Eßer 	datalen = strlen(vpd->vpd_ident);
595f01c8633SStefan Eßer 	KASSERT(datalen <= 255, ("invalid VPD ident length"));
59684b755dfSJohn Baldwin 	vpd_user = lvio->plvi_data;
59784b755dfSJohn Baldwin 	vpd_element.pve_keyword[0] = '\0';
59884b755dfSJohn Baldwin 	vpd_element.pve_keyword[1] = '\0';
59984b755dfSJohn Baldwin 	vpd_element.pve_flags = PVE_FLAG_IDENT;
600f01c8633SStefan Eßer 	vpd_element.pve_datalen = datalen;
60184b755dfSJohn Baldwin 	error = copyout(&vpd_element, vpd_user, sizeof(vpd_element));
60284b755dfSJohn Baldwin 	if (error)
60384b755dfSJohn Baldwin 		return (error);
604f01c8633SStefan Eßer 	error = copyout(vpd->vpd_ident, vpd_user->pve_data, datalen);
60584b755dfSJohn Baldwin 	if (error)
60684b755dfSJohn Baldwin 		return (error);
60774aa2d49SJohn Baldwin 	vpd_user = PVE_NEXT_LEN(vpd_user, vpd_element.pve_datalen);
60884b755dfSJohn Baldwin 	vpd_element.pve_flags = 0;
60984b755dfSJohn Baldwin 	for (i = 0; i < vpd->vpd_rocnt; i++) {
61084b755dfSJohn Baldwin 		vpd_element.pve_keyword[0] = vpd->vpd_ros[i].keyword[0];
61184b755dfSJohn Baldwin 		vpd_element.pve_keyword[1] = vpd->vpd_ros[i].keyword[1];
61284b755dfSJohn Baldwin 		vpd_element.pve_datalen = vpd->vpd_ros[i].len;
61384b755dfSJohn Baldwin 		error = copyout(&vpd_element, vpd_user, sizeof(vpd_element));
61484b755dfSJohn Baldwin 		if (error)
61584b755dfSJohn Baldwin 			return (error);
61684b755dfSJohn Baldwin 		error = copyout(vpd->vpd_ros[i].value, vpd_user->pve_data,
61784b755dfSJohn Baldwin 		    vpd->vpd_ros[i].len);
61884b755dfSJohn Baldwin 		if (error)
61984b755dfSJohn Baldwin 			return (error);
62074aa2d49SJohn Baldwin 		vpd_user = PVE_NEXT_LEN(vpd_user, vpd_element.pve_datalen);
62184b755dfSJohn Baldwin 	}
62284b755dfSJohn Baldwin 	vpd_element.pve_flags = PVE_FLAG_RW;
62384b755dfSJohn Baldwin 	for (i = 0; i < vpd->vpd_wcnt; i++) {
62484b755dfSJohn Baldwin 		vpd_element.pve_keyword[0] = vpd->vpd_w[i].keyword[0];
62584b755dfSJohn Baldwin 		vpd_element.pve_keyword[1] = vpd->vpd_w[i].keyword[1];
62684b755dfSJohn Baldwin 		vpd_element.pve_datalen = vpd->vpd_w[i].len;
62784b755dfSJohn Baldwin 		error = copyout(&vpd_element, vpd_user, sizeof(vpd_element));
62884b755dfSJohn Baldwin 		if (error)
62984b755dfSJohn Baldwin 			return (error);
63084b755dfSJohn Baldwin 		error = copyout(vpd->vpd_w[i].value, vpd_user->pve_data,
63184b755dfSJohn Baldwin 		    vpd->vpd_w[i].len);
63284b755dfSJohn Baldwin 		if (error)
63384b755dfSJohn Baldwin 			return (error);
63474aa2d49SJohn Baldwin 		vpd_user = PVE_NEXT_LEN(vpd_user, vpd_element.pve_datalen);
63584b755dfSJohn Baldwin 	}
63684b755dfSJohn Baldwin 	KASSERT((char *)vpd_user - (char *)lvio->plvi_data == len,
63784b755dfSJohn Baldwin 	    ("length mismatch"));
63884b755dfSJohn Baldwin 	lvio->plvi_len = len;
63984b755dfSJohn Baldwin 	return (0);
64084b755dfSJohn Baldwin }
64184b755dfSJohn Baldwin 
642e9ed3a70SBrooks Davis static size_t
pci_match_conf_size(u_long cmd)643e9ed3a70SBrooks Davis pci_match_conf_size(u_long cmd)
6448983cfbfSMike Smith {
6458983cfbfSMike Smith 
64684b755dfSJohn Baldwin 	switch (cmd) {
647e9ed3a70SBrooks Davis 	case PCIOCGETCONF:
648e9ed3a70SBrooks Davis 		return (sizeof(struct pci_match_conf));
649b7edb6faSBrooks Davis #ifdef COMPAT_FREEBSD32
650b7edb6faSBrooks Davis 	case PCIOCGETCONF32:
651b7edb6faSBrooks Davis 		return (sizeof(struct pci_match_conf32));
652b7edb6faSBrooks Davis #endif
65384b755dfSJohn Baldwin #ifdef PRE7_COMPAT
65456ecc8a9SWarner Losh 	case PCIOCGETCONF_FREEBSD6:
65556ecc8a9SWarner Losh 		return (sizeof(struct pci_match_conf_freebsd6));
65684b755dfSJohn Baldwin #ifdef COMPAT_FREEBSD32
65756ecc8a9SWarner Losh 	case PCIOCGETCONF_FREEBSD6_32:
65856ecc8a9SWarner Losh 		return (sizeof(struct pci_match_conf_freebsd6_32));
65984b755dfSJohn Baldwin #endif
660e9ed3a70SBrooks Davis #endif
661e9ed3a70SBrooks Davis 	default:
662e9ed3a70SBrooks Davis 		/* programmer error */
663e9ed3a70SBrooks Davis 		return (0);
664e9ed3a70SBrooks Davis 	}
665e9ed3a70SBrooks Davis }
666e9ed3a70SBrooks Davis 
667e9ed3a70SBrooks Davis static size_t
pci_conf_size(u_long cmd)668e9ed3a70SBrooks Davis pci_conf_size(u_long cmd)
669e9ed3a70SBrooks Davis {
670e9ed3a70SBrooks Davis 
671e9ed3a70SBrooks Davis 	switch (cmd) {
672e9ed3a70SBrooks Davis 	case PCIOCGETCONF:
673e9ed3a70SBrooks Davis 		return (sizeof(struct pci_conf));
674b7edb6faSBrooks Davis #ifdef COMPAT_FREEBSD32
675b7edb6faSBrooks Davis 	case PCIOCGETCONF32:
676b7edb6faSBrooks Davis 		return (sizeof(struct pci_conf32));
677b7edb6faSBrooks Davis #endif
678e9ed3a70SBrooks Davis #ifdef PRE7_COMPAT
67956ecc8a9SWarner Losh 	case PCIOCGETCONF_FREEBSD6:
68056ecc8a9SWarner Losh 		return (sizeof(struct pci_conf_freebsd6));
681e9ed3a70SBrooks Davis #ifdef COMPAT_FREEBSD32
68256ecc8a9SWarner Losh 	case PCIOCGETCONF_FREEBSD6_32:
68356ecc8a9SWarner Losh 		return (sizeof(struct pci_conf_freebsd6_32));
684e9ed3a70SBrooks Davis #endif
685e9ed3a70SBrooks Davis #endif
686e9ed3a70SBrooks Davis 	default:
687e9ed3a70SBrooks Davis 		/* programmer error */
688e9ed3a70SBrooks Davis 		return (0);
689e9ed3a70SBrooks Davis 	}
690e9ed3a70SBrooks Davis }
691e9ed3a70SBrooks Davis 
692e9ed3a70SBrooks Davis static void
pci_conf_io_init(struct pci_conf_io * cio,caddr_t data,u_long cmd)693e9ed3a70SBrooks Davis pci_conf_io_init(struct pci_conf_io *cio, caddr_t data, u_long cmd)
694e9ed3a70SBrooks Davis {
695*727de621SJohn Baldwin #ifdef COMPAT_FREEBSD32
696e9ed3a70SBrooks Davis 	struct pci_conf_io32 *cio32;
697e9ed3a70SBrooks Davis #endif
698e9ed3a70SBrooks Davis 
699e9ed3a70SBrooks Davis 	switch (cmd) {
700e9ed3a70SBrooks Davis 	case PCIOCGETCONF:
701e9ed3a70SBrooks Davis #ifdef PRE7_COMPAT
70256ecc8a9SWarner Losh 	case PCIOCGETCONF_FREEBSD6:
70384b755dfSJohn Baldwin #endif
704e9ed3a70SBrooks Davis 		*cio = *(struct pci_conf_io *)data;
705e9ed3a70SBrooks Davis 		return;
70684b755dfSJohn Baldwin 
707b7edb6faSBrooks Davis #ifdef COMPAT_FREEBSD32
708b7edb6faSBrooks Davis 	case PCIOCGETCONF32:
709b7edb6faSBrooks Davis #ifdef PRE7_COMPAT
71056ecc8a9SWarner Losh 	case PCIOCGETCONF_FREEBSD6_32:
711b7edb6faSBrooks Davis #endif
712b01bf72bSMaxim Sobolev 		cio32 = (struct pci_conf_io32 *)data;
713b01bf72bSMaxim Sobolev 		cio->pat_buf_len = cio32->pat_buf_len;
714b01bf72bSMaxim Sobolev 		cio->num_patterns = cio32->num_patterns;
715b01bf72bSMaxim Sobolev 		cio->patterns = (void *)(uintptr_t)cio32->patterns;
716b01bf72bSMaxim Sobolev 		cio->match_buf_len = cio32->match_buf_len;
717b01bf72bSMaxim Sobolev 		cio->num_matches = cio32->num_matches;
718b01bf72bSMaxim Sobolev 		cio->matches = (void *)(uintptr_t)cio32->matches;
719b01bf72bSMaxim Sobolev 		cio->offset = cio32->offset;
720b01bf72bSMaxim Sobolev 		cio->generation = cio32->generation;
721b01bf72bSMaxim Sobolev 		cio->status = cio32->status;
722e9ed3a70SBrooks Davis 		return;
723904c3909SGleb Smirnoff #endif
724e9ed3a70SBrooks Davis 
725e9ed3a70SBrooks Davis 	default:
726e9ed3a70SBrooks Davis 		/* programmer error */
727e9ed3a70SBrooks Davis 		return;
728e9ed3a70SBrooks Davis 	}
729904c3909SGleb Smirnoff }
730904c3909SGleb Smirnoff 
731e9ed3a70SBrooks Davis static void
pci_conf_io_update_data(const struct pci_conf_io * cio,caddr_t data,u_long cmd)732e9ed3a70SBrooks Davis pci_conf_io_update_data(const struct pci_conf_io *cio, caddr_t data,
733e9ed3a70SBrooks Davis     u_long cmd)
734e9ed3a70SBrooks Davis {
735e9ed3a70SBrooks Davis 	struct pci_conf_io *d_cio;
736*727de621SJohn Baldwin #ifdef COMPAT_FREEBSD32
737e9ed3a70SBrooks Davis 	struct pci_conf_io32 *cio32;
738e9ed3a70SBrooks Davis #endif
739e9ed3a70SBrooks Davis 
740904c3909SGleb Smirnoff 	switch (cmd) {
741e9ed3a70SBrooks Davis 	case PCIOCGETCONF:
742e9ed3a70SBrooks Davis #ifdef PRE7_COMPAT
74356ecc8a9SWarner Losh 	case PCIOCGETCONF_FREEBSD6:
744e9ed3a70SBrooks Davis #endif
745e9ed3a70SBrooks Davis 		d_cio = (struct pci_conf_io *)data;
746e9ed3a70SBrooks Davis 		d_cio->status = cio->status;
747e9ed3a70SBrooks Davis 		d_cio->generation = cio->generation;
748e9ed3a70SBrooks Davis 		d_cio->offset = cio->offset;
749e9ed3a70SBrooks Davis 		d_cio->num_matches = cio->num_matches;
750e9ed3a70SBrooks Davis 		return;
751e9ed3a70SBrooks Davis 
752b7edb6faSBrooks Davis #ifdef COMPAT_FREEBSD32
753b7edb6faSBrooks Davis 	case PCIOCGETCONF32:
754b7edb6faSBrooks Davis #ifdef PRE7_COMPAT
75556ecc8a9SWarner Losh 	case PCIOCGETCONF_FREEBSD6_32:
756b7edb6faSBrooks Davis #endif
757e9ed3a70SBrooks Davis 		cio32 = (struct pci_conf_io32 *)data;
758e9ed3a70SBrooks Davis 
759e9ed3a70SBrooks Davis 		cio32->status = cio->status;
760e9ed3a70SBrooks Davis 		cio32->generation = cio->generation;
761e9ed3a70SBrooks Davis 		cio32->offset = cio->offset;
762e9ed3a70SBrooks Davis 		cio32->num_matches = cio->num_matches;
763e9ed3a70SBrooks Davis 		return;
764e9ed3a70SBrooks Davis #endif
765e9ed3a70SBrooks Davis 
766e9ed3a70SBrooks Davis 	default:
767e9ed3a70SBrooks Davis 		/* programmer error */
768e9ed3a70SBrooks Davis 		return;
769e9ed3a70SBrooks Davis 	}
770e9ed3a70SBrooks Davis }
771e9ed3a70SBrooks Davis 
772e9ed3a70SBrooks Davis static void
pci_conf_for_copyout(const struct pci_conf * pcp,union pci_conf_union * pcup,u_long cmd)773e9ed3a70SBrooks Davis pci_conf_for_copyout(const struct pci_conf *pcp, union pci_conf_union *pcup,
774e9ed3a70SBrooks Davis     u_long cmd)
775e9ed3a70SBrooks Davis {
776e9ed3a70SBrooks Davis 
777e9ed3a70SBrooks Davis 	memset(pcup, 0, sizeof(*pcup));
778e9ed3a70SBrooks Davis 
779e9ed3a70SBrooks Davis 	switch (cmd) {
780e9ed3a70SBrooks Davis 	case PCIOCGETCONF:
781e9ed3a70SBrooks Davis 		pcup->pc = *pcp;
782e9ed3a70SBrooks Davis 		return;
783e9ed3a70SBrooks Davis 
784b7edb6faSBrooks Davis #ifdef COMPAT_FREEBSD32
785b7edb6faSBrooks Davis 	case PCIOCGETCONF32:
786b7edb6faSBrooks Davis 		pcup->pc32.pc_sel = pcp->pc_sel;
787b7edb6faSBrooks Davis 		pcup->pc32.pc_hdr = pcp->pc_hdr;
788b7edb6faSBrooks Davis 		pcup->pc32.pc_subvendor = pcp->pc_subvendor;
789b7edb6faSBrooks Davis 		pcup->pc32.pc_subdevice = pcp->pc_subdevice;
790b7edb6faSBrooks Davis 		pcup->pc32.pc_vendor = pcp->pc_vendor;
791b7edb6faSBrooks Davis 		pcup->pc32.pc_device = pcp->pc_device;
792b7edb6faSBrooks Davis 		pcup->pc32.pc_class = pcp->pc_class;
793b7edb6faSBrooks Davis 		pcup->pc32.pc_subclass = pcp->pc_subclass;
794b7edb6faSBrooks Davis 		pcup->pc32.pc_progif = pcp->pc_progif;
795b7edb6faSBrooks Davis 		pcup->pc32.pc_revid = pcp->pc_revid;
796b7edb6faSBrooks Davis 		strlcpy(pcup->pc32.pd_name, pcp->pd_name,
797b7edb6faSBrooks Davis 		    sizeof(pcup->pc32.pd_name));
798b7edb6faSBrooks Davis 		pcup->pc32.pd_unit = (uint32_t)pcp->pd_unit;
799b7edb6faSBrooks Davis 		return;
800b7edb6faSBrooks Davis #endif
801b7edb6faSBrooks Davis 
802904c3909SGleb Smirnoff #ifdef PRE7_COMPAT
803904c3909SGleb Smirnoff #ifdef COMPAT_FREEBSD32
80456ecc8a9SWarner Losh 	case PCIOCGETCONF_FREEBSD6_32:
805e9ed3a70SBrooks Davis 		pcup->pco32.pc_sel.pc_bus = pcp->pc_sel.pc_bus;
806e9ed3a70SBrooks Davis 		pcup->pco32.pc_sel.pc_dev = pcp->pc_sel.pc_dev;
807e9ed3a70SBrooks Davis 		pcup->pco32.pc_sel.pc_func = pcp->pc_sel.pc_func;
808e9ed3a70SBrooks Davis 		pcup->pco32.pc_hdr = pcp->pc_hdr;
809e9ed3a70SBrooks Davis 		pcup->pco32.pc_subvendor = pcp->pc_subvendor;
810e9ed3a70SBrooks Davis 		pcup->pco32.pc_subdevice = pcp->pc_subdevice;
811e9ed3a70SBrooks Davis 		pcup->pco32.pc_vendor = pcp->pc_vendor;
812e9ed3a70SBrooks Davis 		pcup->pco32.pc_device = pcp->pc_device;
813e9ed3a70SBrooks Davis 		pcup->pco32.pc_class = pcp->pc_class;
814e9ed3a70SBrooks Davis 		pcup->pco32.pc_subclass = pcp->pc_subclass;
815e9ed3a70SBrooks Davis 		pcup->pco32.pc_progif = pcp->pc_progif;
816e9ed3a70SBrooks Davis 		pcup->pco32.pc_revid = pcp->pc_revid;
817e9ed3a70SBrooks Davis 		strlcpy(pcup->pco32.pd_name, pcp->pd_name,
818e9ed3a70SBrooks Davis 		    sizeof(pcup->pco32.pd_name));
819e9ed3a70SBrooks Davis 		pcup->pco32.pd_unit = (uint32_t)pcp->pd_unit;
820e9ed3a70SBrooks Davis 		return;
8218983cfbfSMike Smith 
822e9ed3a70SBrooks Davis #endif /* COMPAT_FREEBSD32 */
82356ecc8a9SWarner Losh 	case PCIOCGETCONF_FREEBSD6:
824e9ed3a70SBrooks Davis 		pcup->pco.pc_sel.pc_bus = pcp->pc_sel.pc_bus;
825e9ed3a70SBrooks Davis 		pcup->pco.pc_sel.pc_dev = pcp->pc_sel.pc_dev;
826e9ed3a70SBrooks Davis 		pcup->pco.pc_sel.pc_func = pcp->pc_sel.pc_func;
827e9ed3a70SBrooks Davis 		pcup->pco.pc_hdr = pcp->pc_hdr;
828e9ed3a70SBrooks Davis 		pcup->pco.pc_subvendor = pcp->pc_subvendor;
829e9ed3a70SBrooks Davis 		pcup->pco.pc_subdevice = pcp->pc_subdevice;
830e9ed3a70SBrooks Davis 		pcup->pco.pc_vendor = pcp->pc_vendor;
831e9ed3a70SBrooks Davis 		pcup->pco.pc_device = pcp->pc_device;
832e9ed3a70SBrooks Davis 		pcup->pco.pc_class = pcp->pc_class;
833e9ed3a70SBrooks Davis 		pcup->pco.pc_subclass = pcp->pc_subclass;
834e9ed3a70SBrooks Davis 		pcup->pco.pc_progif = pcp->pc_progif;
835e9ed3a70SBrooks Davis 		pcup->pco.pc_revid = pcp->pc_revid;
836e9ed3a70SBrooks Davis 		strlcpy(pcup->pco.pd_name, pcp->pd_name,
837e9ed3a70SBrooks Davis 		    sizeof(pcup->pco.pd_name));
838e9ed3a70SBrooks Davis 		pcup->pco.pd_unit = pcp->pd_unit;
839e9ed3a70SBrooks Davis 		return;
840e9ed3a70SBrooks Davis #endif /* PRE7_COMPAT */
841e9ed3a70SBrooks Davis 
842e9ed3a70SBrooks Davis 	default:
843e9ed3a70SBrooks Davis 		/* programmer error */
844e9ed3a70SBrooks Davis 		return;
845e9ed3a70SBrooks Davis 	}
846e9ed3a70SBrooks Davis }
847e9ed3a70SBrooks Davis 
848e9ed3a70SBrooks Davis static int
pci_bar_mmap(device_t pcidev,struct pci_bar_mmap * pbm)84987842989SKonstantin Belousov pci_bar_mmap(device_t pcidev, struct pci_bar_mmap *pbm)
85087842989SKonstantin Belousov {
85187842989SKonstantin Belousov 	vm_map_t map;
85287842989SKonstantin Belousov 	vm_object_t obj;
85387842989SKonstantin Belousov 	struct thread *td;
85487842989SKonstantin Belousov 	struct sglist *sg;
85587842989SKonstantin Belousov 	struct pci_map *pm;
8569857e00aSMarcin Wojtas 	rman_res_t membase;
85787842989SKonstantin Belousov 	vm_paddr_t pbase;
85887842989SKonstantin Belousov 	vm_size_t plen;
85987842989SKonstantin Belousov 	vm_offset_t addr;
86087842989SKonstantin Belousov 	vm_prot_t prot;
86187842989SKonstantin Belousov 	int error, flags;
86287842989SKonstantin Belousov 
86387842989SKonstantin Belousov 	td = curthread;
86487842989SKonstantin Belousov 	map = &td->td_proc->p_vmspace->vm_map;
86587842989SKonstantin Belousov 	if ((pbm->pbm_flags & ~(PCIIO_BAR_MMAP_FIXED | PCIIO_BAR_MMAP_EXCL |
86687842989SKonstantin Belousov 	    PCIIO_BAR_MMAP_RW | PCIIO_BAR_MMAP_ACTIVATE)) != 0 ||
86787842989SKonstantin Belousov 	    pbm->pbm_memattr != (vm_memattr_t)pbm->pbm_memattr ||
86887842989SKonstantin Belousov 	    !pmap_is_valid_memattr(map->pmap, pbm->pbm_memattr))
86987842989SKonstantin Belousov 		return (EINVAL);
87087842989SKonstantin Belousov 
87187842989SKonstantin Belousov 	/* Fetch the BAR physical base and length. */
87287842989SKonstantin Belousov 	pm = pci_find_bar(pcidev, pbm->pbm_reg);
87387842989SKonstantin Belousov 	if (pm == NULL)
87487842989SKonstantin Belousov 		return (EINVAL);
87587842989SKonstantin Belousov 	if (!pci_bar_enabled(pcidev, pm))
87687842989SKonstantin Belousov 		return (EBUSY); /* XXXKIB enable if _ACTIVATE */
87787842989SKonstantin Belousov 	if (!PCI_BAR_MEM(pm->pm_value))
87887842989SKonstantin Belousov 		return (EIO);
8791fb99e97SMark Johnston 	error = bus_translate_resource(pcidev, SYS_RES_MEMORY,
8809857e00aSMarcin Wojtas 	    pm->pm_value & PCIM_BAR_MEM_BASE, &membase);
881f2f1ab39SMarcin Wojtas 	if (error != 0)
882f2f1ab39SMarcin Wojtas 		return (error);
883f2f1ab39SMarcin Wojtas 
884f48c41acSHans Petter Selasky 	pbase = trunc_page(membase);
885f48c41acSHans Petter Selasky 	plen = round_page(membase + ((pci_addr_t)1 << pm->pm_size)) -
88687842989SKonstantin Belousov 	    pbase;
88787842989SKonstantin Belousov 	prot = VM_PROT_READ | (((pbm->pbm_flags & PCIIO_BAR_MMAP_RW) != 0) ?
88887842989SKonstantin Belousov 	    VM_PROT_WRITE : 0);
88987842989SKonstantin Belousov 
89087842989SKonstantin Belousov 	/* Create vm structures and mmap. */
89187842989SKonstantin Belousov 	sg = sglist_alloc(1, M_WAITOK);
89287842989SKonstantin Belousov 	error = sglist_append_phys(sg, pbase, plen);
89387842989SKonstantin Belousov 	if (error != 0)
89487842989SKonstantin Belousov 		goto out;
89587842989SKonstantin Belousov 	obj = vm_pager_allocate(OBJT_SG, sg, plen, prot, 0, td->td_ucred);
89687842989SKonstantin Belousov 	if (obj == NULL) {
89787842989SKonstantin Belousov 		error = EIO;
89887842989SKonstantin Belousov 		goto out;
89987842989SKonstantin Belousov 	}
90087842989SKonstantin Belousov 	obj->memattr = pbm->pbm_memattr;
90187842989SKonstantin Belousov 	flags = MAP_SHARED;
90287842989SKonstantin Belousov 	addr = 0;
90387842989SKonstantin Belousov 	if ((pbm->pbm_flags & PCIIO_BAR_MMAP_FIXED) != 0) {
90487842989SKonstantin Belousov 		addr = (uintptr_t)pbm->pbm_map_base;
90587842989SKonstantin Belousov 		flags |= MAP_FIXED;
90687842989SKonstantin Belousov 	}
90787842989SKonstantin Belousov 	if ((pbm->pbm_flags & PCIIO_BAR_MMAP_EXCL) != 0)
90887842989SKonstantin Belousov 		flags |= MAP_CHECK_EXCL;
90987842989SKonstantin Belousov 	error = vm_mmap_object(map, &addr, plen, prot, prot, flags, obj, 0,
91087842989SKonstantin Belousov 	    FALSE, td);
91187842989SKonstantin Belousov 	if (error != 0) {
91287842989SKonstantin Belousov 		vm_object_deallocate(obj);
91387842989SKonstantin Belousov 		goto out;
91487842989SKonstantin Belousov 	}
91587842989SKonstantin Belousov 	pbm->pbm_map_base = (void *)addr;
91687842989SKonstantin Belousov 	pbm->pbm_map_length = plen;
917f48c41acSHans Petter Selasky 	pbm->pbm_bar_off = membase - pbase;
91887842989SKonstantin Belousov 	pbm->pbm_bar_length = (pci_addr_t)1 << pm->pm_size;
91987842989SKonstantin Belousov 
92087842989SKonstantin Belousov out:
92187842989SKonstantin Belousov 	sglist_free(sg);
92287842989SKonstantin Belousov 	return (error);
92387842989SKonstantin Belousov }
92487842989SKonstantin Belousov 
92587842989SKonstantin Belousov static int
pci_bar_io(device_t pcidev,struct pci_bar_ioreq * pbi)9267e14be0bSMark Johnston pci_bar_io(device_t pcidev, struct pci_bar_ioreq *pbi)
9277e14be0bSMark Johnston {
9287e14be0bSMark Johnston 	struct pci_map *pm;
9297e14be0bSMark Johnston 	struct resource *res;
9307e14be0bSMark Johnston 	uint32_t offset, width;
9317e14be0bSMark Johnston 	int bar, error, type;
9327e14be0bSMark Johnston 
9337e14be0bSMark Johnston 	if (pbi->pbi_op != PCIBARIO_READ &&
9347e14be0bSMark Johnston 	    pbi->pbi_op != PCIBARIO_WRITE)
9357e14be0bSMark Johnston 		return (EINVAL);
9367e14be0bSMark Johnston 
9377e14be0bSMark Johnston 	bar = PCIR_BAR(pbi->pbi_bar);
9387e14be0bSMark Johnston 	pm = pci_find_bar(pcidev, bar);
9397e14be0bSMark Johnston 	if (pm == NULL)
9407e14be0bSMark Johnston 		return (EINVAL);
9417e14be0bSMark Johnston 
9427e14be0bSMark Johnston 	offset = pbi->pbi_offset;
9437e14be0bSMark Johnston 	width = pbi->pbi_width;
9447e14be0bSMark Johnston 
9457e14be0bSMark Johnston 	if (offset + width < offset ||
9467e14be0bSMark Johnston 	    ((pci_addr_t)1 << pm->pm_size) < offset + width)
9477e14be0bSMark Johnston 		return (EINVAL);
9487e14be0bSMark Johnston 
9497e14be0bSMark Johnston 	type = PCI_BAR_MEM(pm->pm_value) ? SYS_RES_MEMORY : SYS_RES_IOPORT;
9507e14be0bSMark Johnston 
9517e14be0bSMark Johnston 	/*
9527e14be0bSMark Johnston 	 * This will fail if a driver has allocated the resource.  This could be
9537e14be0bSMark Johnston 	 * worked around by detecting that case and using bus_map_resource() to
9547e14be0bSMark Johnston 	 * populate the handle, but so far this is not needed.
9557e14be0bSMark Johnston 	 */
9567e14be0bSMark Johnston 	res = bus_alloc_resource_any(pcidev, type, &bar, RF_ACTIVE);
9577e14be0bSMark Johnston 	if (res == NULL)
9587e14be0bSMark Johnston 		return (ENOENT);
9597e14be0bSMark Johnston 
9607e14be0bSMark Johnston 	error = 0;
9617e14be0bSMark Johnston 	switch (pbi->pbi_op) {
9627e14be0bSMark Johnston 	case PCIBARIO_READ:
9637e14be0bSMark Johnston 		switch (pbi->pbi_width) {
9647e14be0bSMark Johnston 		case 1:
9657e14be0bSMark Johnston 			pbi->pbi_value = bus_read_1(res, offset);
9667e14be0bSMark Johnston 			break;
9677e14be0bSMark Johnston 		case 2:
9687e14be0bSMark Johnston 			pbi->pbi_value = bus_read_2(res, offset);
9697e14be0bSMark Johnston 			break;
9707e14be0bSMark Johnston 		case 4:
9717e14be0bSMark Johnston 			pbi->pbi_value = bus_read_4(res, offset);
9727e14be0bSMark Johnston 			break;
9737e14be0bSMark Johnston #ifndef __i386__
9747e14be0bSMark Johnston 		case 8:
9757e14be0bSMark Johnston 			pbi->pbi_value = bus_read_8(res, offset);
9767e14be0bSMark Johnston 			break;
9777e14be0bSMark Johnston #endif
9787e14be0bSMark Johnston 		default:
9797e14be0bSMark Johnston 			error = EINVAL;
9807e14be0bSMark Johnston 			break;
9817e14be0bSMark Johnston 		}
9827e14be0bSMark Johnston 		break;
9837e14be0bSMark Johnston 	case PCIBARIO_WRITE:
9847e14be0bSMark Johnston 		switch (pbi->pbi_width) {
9857e14be0bSMark Johnston 		case 1:
9867e14be0bSMark Johnston 			bus_write_1(res, offset, pbi->pbi_value);
9877e14be0bSMark Johnston 			break;
9887e14be0bSMark Johnston 		case 2:
9897e14be0bSMark Johnston 			bus_write_2(res, offset, pbi->pbi_value);
9907e14be0bSMark Johnston 			break;
9917e14be0bSMark Johnston 		case 4:
9927e14be0bSMark Johnston 			bus_write_4(res, offset, pbi->pbi_value);
9937e14be0bSMark Johnston 			break;
9947e14be0bSMark Johnston #ifndef __i386__
9957e14be0bSMark Johnston 		case 8:
9967e14be0bSMark Johnston 			bus_write_8(res, offset, pbi->pbi_value);
9977e14be0bSMark Johnston 			break;
9987e14be0bSMark Johnston #endif
9997e14be0bSMark Johnston 		default:
10007e14be0bSMark Johnston 			error = EINVAL;
10017e14be0bSMark Johnston 			break;
10027e14be0bSMark Johnston 		}
10037e14be0bSMark Johnston 		break;
10047e14be0bSMark Johnston 	}
10057e14be0bSMark Johnston 
10067e14be0bSMark Johnston 	bus_release_resource(pcidev, type, bar, res);
10077e14be0bSMark Johnston 
10087e14be0bSMark Johnston 	return (error);
10097e14be0bSMark Johnston }
10107e14be0bSMark Johnston 
10117e14be0bSMark Johnston static int
pci_ioctl(struct cdev * dev,u_long cmd,caddr_t data,int flag,struct thread * td)1012e9ed3a70SBrooks Davis pci_ioctl(struct cdev *dev, u_long cmd, caddr_t data, int flag, struct thread *td)
1013e9ed3a70SBrooks Davis {
1014e9ed3a70SBrooks Davis 	device_t pcidev;
1015e9ed3a70SBrooks Davis 	const char *name;
1016e9ed3a70SBrooks Davis 	struct devlist *devlist_head;
1017e9ed3a70SBrooks Davis 	struct pci_conf_io *cio = NULL;
1018e9ed3a70SBrooks Davis 	struct pci_devinfo *dinfo;
1019e9ed3a70SBrooks Davis 	struct pci_io *io;
10207e14be0bSMark Johnston 	struct pci_bar_ioreq *pbi;
1021e9ed3a70SBrooks Davis 	struct pci_bar_io *bio;
1022e9ed3a70SBrooks Davis 	struct pci_list_vpd_io *lvio;
1023e9ed3a70SBrooks Davis 	struct pci_match_conf *pattern_buf;
1024e9ed3a70SBrooks Davis 	struct pci_map *pm;
102587842989SKonstantin Belousov 	struct pci_bar_mmap *pbm;
1026e9ed3a70SBrooks Davis 	size_t confsz, iolen;
1027e9ed3a70SBrooks Davis 	int error, ionum, i, num_patterns;
1028e9ed3a70SBrooks Davis 	union pci_conf_union pcu;
1029e9ed3a70SBrooks Davis #ifdef PRE7_COMPAT
1030e9ed3a70SBrooks Davis 	struct pci_io iodata;
103156ecc8a9SWarner Losh 	struct pci_io_freebsd6 *io_freebsd6;
1032e9ed3a70SBrooks Davis 
103356ecc8a9SWarner Losh 	io_freebsd6 = NULL;
1034e9ed3a70SBrooks Davis #endif
1035e9ed3a70SBrooks Davis 
103685ae35efSKonstantin Belousov 	/*
103785ae35efSKonstantin Belousov 	 * Interpret read-only opened /dev/pci as a promise that no
103885ae35efSKonstantin Belousov 	 * operation of the file descriptor could modify system state,
103985ae35efSKonstantin Belousov 	 * including side-effects due to reading devices registers.
104085ae35efSKonstantin Belousov 	 */
104185ae35efSKonstantin Belousov 	if ((flag & FWRITE) == 0) {
1042e9ed3a70SBrooks Davis 		switch (cmd) {
1043e9ed3a70SBrooks Davis 		case PCIOCGETCONF:
1044b56f51f1SBrooks Davis #ifdef COMPAT_FREEBSD32
1045b56f51f1SBrooks Davis 		case PCIOCGETCONF32:
1046b56f51f1SBrooks Davis #endif
1047e9ed3a70SBrooks Davis #ifdef PRE7_COMPAT
104856ecc8a9SWarner Losh 		case PCIOCGETCONF_FREEBSD6:
1049e9ed3a70SBrooks Davis #ifdef COMPAT_FREEBSD32
105056ecc8a9SWarner Losh 		case PCIOCGETCONF_FREEBSD6_32:
1051e9ed3a70SBrooks Davis #endif
1052e9ed3a70SBrooks Davis #endif
1053e9ed3a70SBrooks Davis 		case PCIOCGETBAR:
1054e9ed3a70SBrooks Davis 		case PCIOCLISTVPD:
1055e9ed3a70SBrooks Davis 			break;
1056e9ed3a70SBrooks Davis 		default:
1057e9ed3a70SBrooks Davis 			return (EPERM);
1058e9ed3a70SBrooks Davis 		}
1059e9ed3a70SBrooks Davis 	}
1060e9ed3a70SBrooks Davis 
1061c6df6f53SWarner Losh 	/*
1062c6df6f53SWarner Losh 	 * Use bus topology lock to ensure that the pci list of devies doesn't
1063c6df6f53SWarner Losh 	 * change while we're traversing the list, in some cases multiple times.
1064c6df6f53SWarner Losh 	 */
1065c6df6f53SWarner Losh 	bus_topo_lock();
106696b506a5SWarner Losh 
1067e9ed3a70SBrooks Davis 	switch (cmd) {
1068e9ed3a70SBrooks Davis 	case PCIOCGETCONF:
1069b56f51f1SBrooks Davis #ifdef COMPAT_FREEBSD32
1070b56f51f1SBrooks Davis 	case PCIOCGETCONF32:
1071b56f51f1SBrooks Davis #endif
1072e9ed3a70SBrooks Davis #ifdef PRE7_COMPAT
107356ecc8a9SWarner Losh 	case PCIOCGETCONF_FREEBSD6:
1074e9ed3a70SBrooks Davis #ifdef COMPAT_FREEBSD32
107556ecc8a9SWarner Losh 	case PCIOCGETCONF_FREEBSD6_32:
1076e9ed3a70SBrooks Davis #endif
1077e9ed3a70SBrooks Davis #endif
1078e9ed3a70SBrooks Davis 		cio = malloc(sizeof(struct pci_conf_io), M_TEMP,
1079e9ed3a70SBrooks Davis 		    M_WAITOK | M_ZERO);
1080e9ed3a70SBrooks Davis 		pci_conf_io_init(cio, data, cmd);
108133d3fffaSMarius Strobl 		pattern_buf = NULL;
10828983cfbfSMike Smith 		num_patterns = 0;
10838983cfbfSMike Smith 		dinfo = NULL;
10848983cfbfSMike Smith 
108533d3fffaSMarius Strobl 		cio->num_matches = 0;
108633d3fffaSMarius Strobl 
10878983cfbfSMike Smith 		/*
10888983cfbfSMike Smith 		 * If the user specified an offset into the device list,
10898983cfbfSMike Smith 		 * but the list has changed since they last called this
10908983cfbfSMike Smith 		 * ioctl, tell them that the list has changed.  They will
10918983cfbfSMike Smith 		 * have to get the list from the beginning.
10928983cfbfSMike Smith 		 */
10938983cfbfSMike Smith 		if ((cio->offset != 0)
10948983cfbfSMike Smith 		 && (cio->generation != pci_generation)){
10958983cfbfSMike Smith 			cio->status = PCI_GETCONF_LIST_CHANGED;
10968983cfbfSMike Smith 			error = 0;
1097b01bf72bSMaxim Sobolev 			goto getconfexit;
10988983cfbfSMike Smith 		}
10998983cfbfSMike Smith 
11008983cfbfSMike Smith 		/*
11018983cfbfSMike Smith 		 * Check to see whether the user has asked for an offset
11028983cfbfSMike Smith 		 * past the end of our list.
11038983cfbfSMike Smith 		 */
11048983cfbfSMike Smith 		if (cio->offset >= pci_numdevs) {
11058983cfbfSMike Smith 			cio->status = PCI_GETCONF_LAST_DEVICE;
11068983cfbfSMike Smith 			error = 0;
1107b01bf72bSMaxim Sobolev 			goto getconfexit;
11088983cfbfSMike Smith 		}
11098983cfbfSMike Smith 
11108983cfbfSMike Smith 		/* get the head of the device queue */
11118983cfbfSMike Smith 		devlist_head = &pci_devq;
11128983cfbfSMike Smith 
11138983cfbfSMike Smith 		/*
11148983cfbfSMike Smith 		 * Determine how much room we have for pci_conf structures.
11158983cfbfSMike Smith 		 * Round the user's buffer size down to the nearest
11168983cfbfSMike Smith 		 * multiple of sizeof(struct pci_conf) in case the user
11178983cfbfSMike Smith 		 * didn't specify a multiple of that size.
11188983cfbfSMike Smith 		 */
1119e9ed3a70SBrooks Davis 		confsz = pci_conf_size(cmd);
112033d3fffaSMarius Strobl 		iolen = min(cio->match_buf_len - (cio->match_buf_len % confsz),
112133d3fffaSMarius Strobl 		    pci_numdevs * confsz);
11228983cfbfSMike Smith 
11238983cfbfSMike Smith 		/*
11248983cfbfSMike Smith 		 * Since we know that iolen is a multiple of the size of
11258983cfbfSMike Smith 		 * the pciconf union, it's okay to do this.
11268983cfbfSMike Smith 		 */
112733d3fffaSMarius Strobl 		ionum = iolen / confsz;
11288983cfbfSMike Smith 
11298983cfbfSMike Smith 		/*
11308983cfbfSMike Smith 		 * If this test is true, the user wants the pci_conf
11318983cfbfSMike Smith 		 * structures returned to match the supplied entries.
11328983cfbfSMike Smith 		 */
1133e3f932deSJohn-Mark Gurney 		if ((cio->num_patterns > 0) && (cio->num_patterns < pci_numdevs)
11348983cfbfSMike Smith 		 && (cio->pat_buf_len > 0)) {
11358983cfbfSMike Smith 			/*
11368983cfbfSMike Smith 			 * pat_buf_len needs to be:
11378983cfbfSMike Smith 			 * num_patterns * sizeof(struct pci_match_conf)
11388983cfbfSMike Smith 			 * While it is certainly possible the user just
11398983cfbfSMike Smith 			 * allocated a large buffer, but set the number of
11408983cfbfSMike Smith 			 * matches correctly, it is far more likely that
11418983cfbfSMike Smith 			 * their kernel doesn't match the userland utility
11428983cfbfSMike Smith 			 * they're using.  It's also possible that the user
11438983cfbfSMike Smith 			 * forgot to initialize some variables.  Yes, this
11448983cfbfSMike Smith 			 * may be overly picky, but I hazard to guess that
11458983cfbfSMike Smith 			 * it's far more likely to just catch folks that
11468983cfbfSMike Smith 			 * updated their kernel but not their userland.
11478983cfbfSMike Smith 			 */
1148e9ed3a70SBrooks Davis 			if (cio->num_patterns * pci_match_conf_size(cmd) !=
1149e9ed3a70SBrooks Davis 			    cio->pat_buf_len) {
115033d3fffaSMarius Strobl 				/* The user made a mistake, return an error. */
11518983cfbfSMike Smith 				cio->status = PCI_GETCONF_ERROR;
11528983cfbfSMike Smith 				error = EINVAL;
1153b01bf72bSMaxim Sobolev 				goto getconfexit;
11548983cfbfSMike Smith 			}
11558983cfbfSMike Smith 
11568983cfbfSMike Smith 			/*
11578983cfbfSMike Smith 			 * Allocate a buffer to hold the patterns.
11588983cfbfSMike Smith 			 */
11598983cfbfSMike Smith 			pattern_buf = malloc(cio->pat_buf_len, M_TEMP,
1160a163d034SWarner Losh 			    M_WAITOK);
11618983cfbfSMike Smith 			error = copyin(cio->patterns, pattern_buf,
11628983cfbfSMike Smith 			    cio->pat_buf_len);
1163d08239c1SJohn-Mark Gurney 			if (error != 0) {
1164d08239c1SJohn-Mark Gurney 				error = EINVAL;
1165d08239c1SJohn-Mark Gurney 				goto getconfexit;
1166d08239c1SJohn-Mark Gurney 			}
11678983cfbfSMike Smith 			num_patterns = cio->num_patterns;
11688983cfbfSMike Smith 		} else if ((cio->num_patterns > 0)
11698983cfbfSMike Smith 			|| (cio->pat_buf_len > 0)) {
11708983cfbfSMike Smith 			/*
11718983cfbfSMike Smith 			 * The user made a mistake, spit out an error.
11728983cfbfSMike Smith 			 */
11738983cfbfSMike Smith 			cio->status = PCI_GETCONF_ERROR;
11748983cfbfSMike Smith 			error = EINVAL;
1175b01bf72bSMaxim Sobolev 			goto getconfexit;
117633d3fffaSMarius Strobl 		}
11778983cfbfSMike Smith 
11788983cfbfSMike Smith 		/*
11798983cfbfSMike Smith 		 * Go through the list of devices and copy out the devices
11808983cfbfSMike Smith 		 * that match the user's criteria.
11818983cfbfSMike Smith 		 */
118210012d53SJohn Baldwin 		for (cio->num_matches = 0, i = 0,
11838983cfbfSMike Smith 				 dinfo = STAILQ_FIRST(devlist_head);
118410012d53SJohn Baldwin 		     dinfo != NULL;
11858983cfbfSMike Smith 		     dinfo = STAILQ_NEXT(dinfo, pci_links), i++) {
11868983cfbfSMike Smith 			if (i < cio->offset)
11878983cfbfSMike Smith 				continue;
11888983cfbfSMike Smith 
11898983cfbfSMike Smith 			/* Populate pd_name and pd_unit */
11908983cfbfSMike Smith 			name = NULL;
11910678f786SJohn Baldwin 			if (dinfo->cfg.dev)
11928983cfbfSMike Smith 				name = device_get_name(dinfo->cfg.dev);
11938983cfbfSMike Smith 			if (name) {
11948983cfbfSMike Smith 				strncpy(dinfo->conf.pd_name, name,
11958983cfbfSMike Smith 					sizeof(dinfo->conf.pd_name));
11968983cfbfSMike Smith 				dinfo->conf.pd_name[PCI_MAXNAMELEN] = 0;
11978983cfbfSMike Smith 				dinfo->conf.pd_unit =
11988983cfbfSMike Smith 					device_get_unit(dinfo->cfg.dev);
11990678f786SJohn Baldwin 			} else {
12000678f786SJohn Baldwin 				dinfo->conf.pd_name[0] = '\0';
12010678f786SJohn Baldwin 				dinfo->conf.pd_unit = 0;
12028983cfbfSMike Smith 			}
12038983cfbfSMike Smith 
120433d3fffaSMarius Strobl 			if (pattern_buf == NULL ||
1205e9ed3a70SBrooks Davis 			    pci_conf_match(cmd, pattern_buf, num_patterns,
120633d3fffaSMarius Strobl 			    &dinfo->conf) == 0) {
12078983cfbfSMike Smith 				/*
12088983cfbfSMike Smith 				 * If we've filled up the user's buffer,
12098983cfbfSMike Smith 				 * break out at this point.  Since we've
12108983cfbfSMike Smith 				 * got a match here, we'll pick right back
12118983cfbfSMike Smith 				 * up at the matching entry.  We can also
12128983cfbfSMike Smith 				 * tell the user that there are more matches
12138983cfbfSMike Smith 				 * left.
12148983cfbfSMike Smith 				 */
1215fadd3f8aSConrad Meyer 				if (cio->num_matches >= ionum) {
1216fadd3f8aSConrad Meyer 					error = 0;
12178983cfbfSMike Smith 					break;
1218fadd3f8aSConrad Meyer 				}
12198983cfbfSMike Smith 
1220e9ed3a70SBrooks Davis 				pci_conf_for_copyout(&dinfo->conf, &pcu, cmd);
1221e9ed3a70SBrooks Davis 				error = copyout(&pcu,
1222c5860546SMarius Strobl 				    (caddr_t)cio->matches +
122310012d53SJohn Baldwin 				    confsz * cio->num_matches, confsz);
122410012d53SJohn Baldwin 				if (error)
122510012d53SJohn Baldwin 					break;
122633d3fffaSMarius Strobl 				cio->num_matches++;
12278983cfbfSMike Smith 			}
1228d08239c1SJohn-Mark Gurney 		}
12298983cfbfSMike Smith 
12308983cfbfSMike Smith 		/*
12318983cfbfSMike Smith 		 * Set the pointer into the list, so if the user is getting
12328983cfbfSMike Smith 		 * n records at a time, where n < pci_numdevs,
12338983cfbfSMike Smith 		 */
12348983cfbfSMike Smith 		cio->offset = i;
12358983cfbfSMike Smith 
12368983cfbfSMike Smith 		/*
12378983cfbfSMike Smith 		 * Set the generation, the user will need this if they make
12388983cfbfSMike Smith 		 * another ioctl call with offset != 0.
12398983cfbfSMike Smith 		 */
12408983cfbfSMike Smith 		cio->generation = pci_generation;
12418983cfbfSMike Smith 
12428983cfbfSMike Smith 		/*
12438983cfbfSMike Smith 		 * If this is the last device, inform the user so he won't
12448983cfbfSMike Smith 		 * bother asking for more devices.  If dinfo isn't NULL, we
12458983cfbfSMike Smith 		 * know that there are more matches in the list because of
12468983cfbfSMike Smith 		 * the way the traversal is done.
12478983cfbfSMike Smith 		 */
12488983cfbfSMike Smith 		if (dinfo == NULL)
12498983cfbfSMike Smith 			cio->status = PCI_GETCONF_LAST_DEVICE;
12508983cfbfSMike Smith 		else
12518983cfbfSMike Smith 			cio->status = PCI_GETCONF_MORE_DEVS;
12528983cfbfSMike Smith 
1253d08239c1SJohn-Mark Gurney getconfexit:
1254e9ed3a70SBrooks Davis 		pci_conf_io_update_data(cio, data, cmd);
1255b01bf72bSMaxim Sobolev 		free(cio, M_TEMP);
1256904c3909SGleb Smirnoff 		free(pattern_buf, M_TEMP);
12578983cfbfSMike Smith 
12588983cfbfSMike Smith 		break;
12598983cfbfSMike Smith 
1260b2068c0cSWarner Losh #ifdef PRE7_COMPAT
126156ecc8a9SWarner Losh 	case PCIOCREAD_FREEBSD6:
126256ecc8a9SWarner Losh 	case PCIOCWRITE_FREEBSD6:
126356ecc8a9SWarner Losh 		io_freebsd6 = (struct pci_io_freebsd6 *)data;
126433d3fffaSMarius Strobl 		iodata.pi_sel.pc_domain = 0;
126556ecc8a9SWarner Losh 		iodata.pi_sel.pc_bus = io_freebsd6->pi_sel.pc_bus;
126656ecc8a9SWarner Losh 		iodata.pi_sel.pc_dev = io_freebsd6->pi_sel.pc_dev;
126756ecc8a9SWarner Losh 		iodata.pi_sel.pc_func = io_freebsd6->pi_sel.pc_func;
126856ecc8a9SWarner Losh 		iodata.pi_reg = io_freebsd6->pi_reg;
126956ecc8a9SWarner Losh 		iodata.pi_width = io_freebsd6->pi_width;
127056ecc8a9SWarner Losh 		iodata.pi_data = io_freebsd6->pi_data;
127133d3fffaSMarius Strobl 		data = (caddr_t)&iodata;
127233d3fffaSMarius Strobl 		/* FALLTHROUGH */
127333d3fffaSMarius Strobl #endif
127466f314b5SStefan Eßer 	case PCIOCREAD:
12758983cfbfSMike Smith 	case PCIOCWRITE:
12768983cfbfSMike Smith 		io = (struct pci_io *)data;
12778983cfbfSMike Smith 		switch(io->pi_width) {
12788983cfbfSMike Smith 		case 4:
12798983cfbfSMike Smith 		case 2:
12808983cfbfSMike Smith 		case 1:
1281d16d35fdSAndriy Gapon 			/* Make sure register is not negative and aligned. */
128233d3fffaSMarius Strobl 			if (io->pi_reg < 0 ||
128333d3fffaSMarius Strobl 			    io->pi_reg & (io->pi_width - 1)) {
128466f314b5SStefan Eßer 				error = EINVAL;
128540ed3f47SRuslan Ermilov 				break;
128640ed3f47SRuslan Ermilov 			}
12878983cfbfSMike Smith 			/*
12888983cfbfSMike Smith 			 * Assume that the user-level bus number is
128937ce43b7SBruce M Simpson 			 * in fact the physical PCI bus number.
129037ce43b7SBruce M Simpson 			 * Look up the grandparent, i.e. the bridge device,
129137ce43b7SBruce M Simpson 			 * so that we can issue configuration space cycles.
12928983cfbfSMike Smith 			 */
129355aaf894SMarius Strobl 			pcidev = pci_find_dbsf(io->pi_sel.pc_domain,
129455aaf894SMarius Strobl 			    io->pi_sel.pc_bus, io->pi_sel.pc_dev,
129555aaf894SMarius Strobl 			    io->pi_sel.pc_func);
129637ce43b7SBruce M Simpson 			if (pcidev) {
1297b2068c0cSWarner Losh #ifdef PRE7_COMPAT
129856ecc8a9SWarner Losh 				if (cmd == PCIOCWRITE || cmd == PCIOCWRITE_FREEBSD6)
129933d3fffaSMarius Strobl #else
130066f314b5SStefan Eßer 				if (cmd == PCIOCWRITE)
130133d3fffaSMarius Strobl #endif
13025060ec97SRyan Stone 					pci_write_config(pcidev,
13038983cfbfSMike Smith 							  io->pi_reg,
13048983cfbfSMike Smith 							  io->pi_data,
13058983cfbfSMike Smith 							  io->pi_width);
1306b2068c0cSWarner Losh #ifdef PRE7_COMPAT
130756ecc8a9SWarner Losh 				else if (cmd == PCIOCREAD_FREEBSD6)
130856ecc8a9SWarner Losh 					io_freebsd6->pi_data =
13095060ec97SRyan Stone 						pci_read_config(pcidev,
131033d3fffaSMarius Strobl 							  io->pi_reg,
131133d3fffaSMarius Strobl 							  io->pi_width);
131233d3fffaSMarius Strobl #endif
131366f314b5SStefan Eßer 				else
131466f314b5SStefan Eßer 					io->pi_data =
13155060ec97SRyan Stone 						pci_read_config(pcidev,
131666f314b5SStefan Eßer 							  io->pi_reg,
131766f314b5SStefan Eßer 							  io->pi_width);
13188983cfbfSMike Smith 				error = 0;
13198983cfbfSMike Smith 			} else {
13208910aa92SPaul Saab #ifdef COMPAT_FREEBSD4
132156ecc8a9SWarner Losh 				if (cmd == PCIOCREAD_FREEBSD6) {
132256ecc8a9SWarner Losh 					io_freebsd6->pi_data = -1;
13238910aa92SPaul Saab 					error = 0;
13248910aa92SPaul Saab 				} else
13258910aa92SPaul Saab #endif
13268983cfbfSMike Smith 					error = ENODEV;
13278983cfbfSMike Smith 			}
13288983cfbfSMike Smith 			break;
13298983cfbfSMike Smith 		default:
1330d08239c1SJohn-Mark Gurney 			error = EINVAL;
13318983cfbfSMike Smith 			break;
13328983cfbfSMike Smith 		}
13338983cfbfSMike Smith 		break;
13348983cfbfSMike Smith 
1335da1e0915SJohn Baldwin 	case PCIOCGETBAR:
1336da1e0915SJohn Baldwin 		bio = (struct pci_bar_io *)data;
1337da1e0915SJohn Baldwin 
1338da1e0915SJohn Baldwin 		/*
1339da1e0915SJohn Baldwin 		 * Assume that the user-level bus number is
1340da1e0915SJohn Baldwin 		 * in fact the physical PCI bus number.
1341da1e0915SJohn Baldwin 		 */
1342da1e0915SJohn Baldwin 		pcidev = pci_find_dbsf(bio->pbi_sel.pc_domain,
1343da1e0915SJohn Baldwin 		    bio->pbi_sel.pc_bus, bio->pbi_sel.pc_dev,
1344da1e0915SJohn Baldwin 		    bio->pbi_sel.pc_func);
1345da1e0915SJohn Baldwin 		if (pcidev == NULL) {
1346da1e0915SJohn Baldwin 			error = ENODEV;
1347da1e0915SJohn Baldwin 			break;
1348da1e0915SJohn Baldwin 		}
1349a90dd577SJohn Baldwin 		pm = pci_find_bar(pcidev, bio->pbi_reg);
1350a90dd577SJohn Baldwin 		if (pm == NULL) {
1351da1e0915SJohn Baldwin 			error = EINVAL;
1352da1e0915SJohn Baldwin 			break;
1353da1e0915SJohn Baldwin 		}
1354a90dd577SJohn Baldwin 		bio->pbi_base = pm->pm_value;
1355a90dd577SJohn Baldwin 		bio->pbi_length = (pci_addr_t)1 << pm->pm_size;
1356a90dd577SJohn Baldwin 		bio->pbi_enabled = pci_bar_enabled(pcidev, pm);
1357da1e0915SJohn Baldwin 		error = 0;
1358da1e0915SJohn Baldwin 		break;
1359762aad81SNeel Natu 	case PCIOCATTACHED:
1360762aad81SNeel Natu 		error = 0;
1361762aad81SNeel Natu 		io = (struct pci_io *)data;
1362762aad81SNeel Natu 		pcidev = pci_find_dbsf(io->pi_sel.pc_domain, io->pi_sel.pc_bus,
1363762aad81SNeel Natu 				       io->pi_sel.pc_dev, io->pi_sel.pc_func);
1364762aad81SNeel Natu 		if (pcidev != NULL)
1365762aad81SNeel Natu 			io->pi_data = device_is_attached(pcidev);
1366762aad81SNeel Natu 		else
1367762aad81SNeel Natu 			error = ENODEV;
1368762aad81SNeel Natu 		break;
136984b755dfSJohn Baldwin 	case PCIOCLISTVPD:
137084b755dfSJohn Baldwin 		lvio = (struct pci_list_vpd_io *)data;
137184b755dfSJohn Baldwin 
137284b755dfSJohn Baldwin 		/*
137384b755dfSJohn Baldwin 		 * Assume that the user-level bus number is
137484b755dfSJohn Baldwin 		 * in fact the physical PCI bus number.
137584b755dfSJohn Baldwin 		 */
137684b755dfSJohn Baldwin 		pcidev = pci_find_dbsf(lvio->plvi_sel.pc_domain,
137784b755dfSJohn Baldwin 		    lvio->plvi_sel.pc_bus, lvio->plvi_sel.pc_dev,
137884b755dfSJohn Baldwin 		    lvio->plvi_sel.pc_func);
137984b755dfSJohn Baldwin 		if (pcidev == NULL) {
138084b755dfSJohn Baldwin 			error = ENODEV;
138184b755dfSJohn Baldwin 			break;
138284b755dfSJohn Baldwin 		}
138384b755dfSJohn Baldwin 		error = pci_list_vpd(pcidev, lvio);
138484b755dfSJohn Baldwin 		break;
138587842989SKonstantin Belousov 
138687842989SKonstantin Belousov 	case PCIOCBARMMAP:
138787842989SKonstantin Belousov 		pbm = (struct pci_bar_mmap *)data;
138887842989SKonstantin Belousov 		if ((flag & FWRITE) == 0 &&
138996b506a5SWarner Losh 		    (pbm->pbm_flags & PCIIO_BAR_MMAP_RW) != 0) {
139096b506a5SWarner Losh 			error = EPERM;
139196b506a5SWarner Losh 			break;
139296b506a5SWarner Losh 		}
139387842989SKonstantin Belousov 		pcidev = pci_find_dbsf(pbm->pbm_sel.pc_domain,
139487842989SKonstantin Belousov 		    pbm->pbm_sel.pc_bus, pbm->pbm_sel.pc_dev,
139587842989SKonstantin Belousov 		    pbm->pbm_sel.pc_func);
139687842989SKonstantin Belousov 		error = pcidev == NULL ? ENODEV : pci_bar_mmap(pcidev, pbm);
139787842989SKonstantin Belousov 		break;
139887842989SKonstantin Belousov 
13997e14be0bSMark Johnston 	case PCIOCBARIO:
14007e14be0bSMark Johnston 		pbi = (struct pci_bar_ioreq *)data;
14017e14be0bSMark Johnston 
14027e14be0bSMark Johnston 		pcidev = pci_find_dbsf(pbi->pbi_sel.pc_domain,
14037e14be0bSMark Johnston 		    pbi->pbi_sel.pc_bus, pbi->pbi_sel.pc_dev,
14047e14be0bSMark Johnston 		    pbi->pbi_sel.pc_func);
14057e14be0bSMark Johnston 		if (pcidev == NULL) {
14067e14be0bSMark Johnston 			error = ENODEV;
14077e14be0bSMark Johnston 			break;
14087e14be0bSMark Johnston 		}
14097e14be0bSMark Johnston 		error = pci_bar_io(pcidev, pbi);
14107e14be0bSMark Johnston 		break;
14117e14be0bSMark Johnston 
14128983cfbfSMike Smith 	default:
14138983cfbfSMike Smith 		error = ENOTTY;
14148983cfbfSMike Smith 		break;
14158983cfbfSMike Smith 	}
14168983cfbfSMike Smith 
1417c6df6f53SWarner Losh 	bus_topo_unlock();
141896b506a5SWarner Losh 
14198983cfbfSMike Smith 	return (error);
14208983cfbfSMike Smith }
1421