xref: /freebsd/sys/powerpc/powernv/opal_pci.c (revision bf2166bd72ad691765ff5e081f00cbba6b12da61)
1ac9b4325SWojciech Macek /*-
2ac9b4325SWojciech Macek  * Copyright (c) 2015-2016 Nathan Whitehorn
35c3e53efSWojciech Macek  * Copyright (c) 2017-2018 Semihalf
4ac9b4325SWojciech Macek  * All rights reserved.
5ac9b4325SWojciech Macek  *
6ac9b4325SWojciech Macek  * Redistribution and use in source and binary forms, with or without
7ac9b4325SWojciech Macek  * modification, are permitted provided that the following conditions
8ac9b4325SWojciech Macek  * are met:
9ac9b4325SWojciech Macek  * 1. Redistributions of source code must retain the above copyright
10ac9b4325SWojciech Macek  *    notice, this list of conditions and the following disclaimer.
11ac9b4325SWojciech Macek  * 2. Redistributions in binary form must reproduce the above copyright
12ac9b4325SWojciech Macek  *    notice, this list of conditions and the following disclaimer in the
13ac9b4325SWojciech Macek  *    documentation and/or other materials provided with the distribution.
14ac9b4325SWojciech Macek  *
15ac9b4325SWojciech Macek  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
16ac9b4325SWojciech Macek  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
17ac9b4325SWojciech Macek  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
18ac9b4325SWojciech Macek  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
19ac9b4325SWojciech Macek  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
20ac9b4325SWojciech Macek  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
21ac9b4325SWojciech Macek  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
22ac9b4325SWojciech Macek  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
23ac9b4325SWojciech Macek  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
24ac9b4325SWojciech Macek  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
25ac9b4325SWojciech Macek  * SUCH DAMAGE.
26ac9b4325SWojciech Macek  */
27ac9b4325SWojciech Macek 
28ac9b4325SWojciech Macek #include <sys/param.h>
29ac9b4325SWojciech Macek #include <sys/systm.h>
30ac9b4325SWojciech Macek #include <sys/module.h>
31ac9b4325SWojciech Macek #include <sys/bus.h>
32ac9b4325SWojciech Macek #include <sys/conf.h>
33ac9b4325SWojciech Macek #include <sys/kernel.h>
34ac9b4325SWojciech Macek #include <sys/pciio.h>
35ac9b4325SWojciech Macek #include <sys/endian.h>
36ac9b4325SWojciech Macek #include <sys/rman.h>
37ac9b4325SWojciech Macek #include <sys/vmem.h>
38ac9b4325SWojciech Macek 
39ac9b4325SWojciech Macek #include <dev/ofw/openfirm.h>
40ac9b4325SWojciech Macek #include <dev/ofw/ofw_pci.h>
41ac9b4325SWojciech Macek #include <dev/ofw/ofw_bus.h>
42ac9b4325SWojciech Macek #include <dev/ofw/ofw_bus_subr.h>
43ac9b4325SWojciech Macek #include <dev/ofw/ofwpci.h>
44ac9b4325SWojciech Macek 
45ac9b4325SWojciech Macek #include <dev/pci/pcivar.h>
46ac9b4325SWojciech Macek #include <dev/pci/pcireg.h>
47ac9b4325SWojciech Macek 
48ac9b4325SWojciech Macek #include <machine/bus.h>
49ac9b4325SWojciech Macek #include <machine/intr_machdep.h>
50ac9b4325SWojciech Macek #include <machine/md_var.h>
51ac9b4325SWojciech Macek 
52ac9b4325SWojciech Macek #include <vm/vm.h>
53ac9b4325SWojciech Macek #include <vm/pmap.h>
54ac9b4325SWojciech Macek 
55ac9b4325SWojciech Macek #include "pcib_if.h"
56ac9b4325SWojciech Macek #include "pic_if.h"
57ac9b4325SWojciech Macek #include "iommu_if.h"
58ac9b4325SWojciech Macek #include "opal.h"
59ac9b4325SWojciech Macek 
605c3e53efSWojciech Macek #define	OPAL_PCI_TCE_MAX_ENTRIES	(1024*1024UL)
61dceea51eSJustin Hibbits #define	OPAL_PCI_TCE_DEFAULT_SEG_SIZE	(16*1024*1024UL)
625c3e53efSWojciech Macek #define	OPAL_PCI_TCE_R			(1UL << 0)
635c3e53efSWojciech Macek #define	OPAL_PCI_TCE_W			(1UL << 1)
645c3e53efSWojciech Macek #define	PHB3_TCE_KILL_INVAL_ALL		(1UL << 63)
655c3e53efSWojciech Macek 
66ac9b4325SWojciech Macek /*
67ac9b4325SWojciech Macek  * Device interface.
68ac9b4325SWojciech Macek  */
69ac9b4325SWojciech Macek static int		opalpci_probe(device_t);
70ac9b4325SWojciech Macek static int		opalpci_attach(device_t);
71ac9b4325SWojciech Macek 
72ac9b4325SWojciech Macek /*
73ac9b4325SWojciech Macek  * pcib interface.
74ac9b4325SWojciech Macek  */
75ac9b4325SWojciech Macek static uint32_t		opalpci_read_config(device_t, u_int, u_int, u_int,
76ac9b4325SWojciech Macek 			    u_int, int);
77ac9b4325SWojciech Macek static void		opalpci_write_config(device_t, u_int, u_int, u_int,
78ac9b4325SWojciech Macek 			    u_int, u_int32_t, int);
79ac9b4325SWojciech Macek static int		opalpci_alloc_msi(device_t dev, device_t child,
80ac9b4325SWojciech Macek 			    int count, int maxcount, int *irqs);
81ac9b4325SWojciech Macek static int		opalpci_release_msi(device_t dev, device_t child,
82ac9b4325SWojciech Macek 			    int count, int *irqs);
83ac9b4325SWojciech Macek static int		opalpci_alloc_msix(device_t dev, device_t child,
84ac9b4325SWojciech Macek 			    int *irq);
85ac9b4325SWojciech Macek static int		opalpci_release_msix(device_t dev, device_t child,
86ac9b4325SWojciech Macek 			    int irq);
87ac9b4325SWojciech Macek static int		opalpci_map_msi(device_t dev, device_t child,
88ac9b4325SWojciech Macek 			    int irq, uint64_t *addr, uint32_t *data);
89ac9b4325SWojciech Macek static int opalpci_route_interrupt(device_t bus, device_t dev, int pin);
90ac9b4325SWojciech Macek 
91ac9b4325SWojciech Macek /*
92ac9b4325SWojciech Macek  * MSI PIC interface.
93ac9b4325SWojciech Macek  */
9456505ec0SJustin Hibbits static void opalpic_pic_enable(device_t dev, u_int irq, u_int vector, void **);
9556505ec0SJustin Hibbits static void opalpic_pic_eoi(device_t dev, u_int irq, void *);
96ac9b4325SWojciech Macek 
97d93e635aSLeandro Lupori /* Bus interface */
98d93e635aSLeandro Lupori static bus_dma_tag_t opalpci_get_dma_tag(device_t dev, device_t child);
99d93e635aSLeandro Lupori 
100ac9b4325SWojciech Macek /*
101ac9b4325SWojciech Macek  * Commands
102ac9b4325SWojciech Macek  */
103ac9b4325SWojciech Macek #define	OPAL_M32_WINDOW_TYPE		1
104ac9b4325SWojciech Macek #define	OPAL_M64_WINDOW_TYPE		2
105ac9b4325SWojciech Macek #define	OPAL_IO_WINDOW_TYPE		3
106ac9b4325SWojciech Macek 
107ac9b4325SWojciech Macek #define	OPAL_RESET_PHB_COMPLETE		1
108ac9b4325SWojciech Macek #define	OPAL_RESET_PCI_IODA_TABLE	6
109ac9b4325SWojciech Macek 
110ac9b4325SWojciech Macek #define	OPAL_DISABLE_M64		0
111ac9b4325SWojciech Macek #define	OPAL_ENABLE_M64_SPLIT		1
112ac9b4325SWojciech Macek #define	OPAL_ENABLE_M64_NON_SPLIT	2
113ac9b4325SWojciech Macek 
114ac9b4325SWojciech Macek #define	OPAL_EEH_ACTION_CLEAR_FREEZE_MMIO	1
115ac9b4325SWojciech Macek #define	OPAL_EEH_ACTION_CLEAR_FREEZE_DMA	2
116ac9b4325SWojciech Macek #define	OPAL_EEH_ACTION_CLEAR_FREEZE_ALL	3
117ac9b4325SWojciech Macek 
1180effb2ccSJustin Hibbits #define	OPAL_EEH_STOPPED_NOT_FROZEN		0
1190effb2ccSJustin Hibbits 
120ac9b4325SWojciech Macek /*
121ac9b4325SWojciech Macek  * Constants
122ac9b4325SWojciech Macek  */
123ac9b4325SWojciech Macek #define OPAL_PCI_DEFAULT_PE			1
124ac9b4325SWojciech Macek 
125d93e635aSLeandro Lupori #define OPAL_PCI_BUS_SPACE_LOWADDR_32BIT	0x7FFFFFFFUL
126d93e635aSLeandro Lupori 
127ac9b4325SWojciech Macek /*
128ac9b4325SWojciech Macek  * Driver methods.
129ac9b4325SWojciech Macek  */
130ac9b4325SWojciech Macek static device_method_t	opalpci_methods[] = {
131ac9b4325SWojciech Macek 	/* Device interface */
132ac9b4325SWojciech Macek 	DEVMETHOD(device_probe,		opalpci_probe),
133ac9b4325SWojciech Macek 	DEVMETHOD(device_attach,	opalpci_attach),
134ac9b4325SWojciech Macek 
135ac9b4325SWojciech Macek 	/* pcib interface */
136ac9b4325SWojciech Macek 	DEVMETHOD(pcib_read_config,	opalpci_read_config),
137ac9b4325SWojciech Macek 	DEVMETHOD(pcib_write_config,	opalpci_write_config),
138ac9b4325SWojciech Macek 
139ac9b4325SWojciech Macek 	DEVMETHOD(pcib_alloc_msi,	opalpci_alloc_msi),
140ac9b4325SWojciech Macek 	DEVMETHOD(pcib_release_msi,	opalpci_release_msi),
141ac9b4325SWojciech Macek 	DEVMETHOD(pcib_alloc_msix,	opalpci_alloc_msix),
142ac9b4325SWojciech Macek 	DEVMETHOD(pcib_release_msix,	opalpci_release_msix),
143ac9b4325SWojciech Macek 	DEVMETHOD(pcib_map_msi,		opalpci_map_msi),
144ac9b4325SWojciech Macek 	DEVMETHOD(pcib_route_interrupt,	opalpci_route_interrupt),
145ac9b4325SWojciech Macek 
146ac9b4325SWojciech Macek 	/* PIC interface for MSIs */
147ac9b4325SWojciech Macek 	DEVMETHOD(pic_enable,		opalpic_pic_enable),
148ac9b4325SWojciech Macek 	DEVMETHOD(pic_eoi,		opalpic_pic_eoi),
149ac9b4325SWojciech Macek 
150d93e635aSLeandro Lupori 	/* Bus interface */
151d93e635aSLeandro Lupori 	DEVMETHOD(bus_get_dma_tag,	opalpci_get_dma_tag),
15249d9a597SJustin Hibbits 	DEVMETHOD(bus_get_cpus,		ofw_pcibus_get_cpus),
15349d9a597SJustin Hibbits 	DEVMETHOD(bus_get_domain,	ofw_pcibus_get_domain),
154d93e635aSLeandro Lupori 
155ac9b4325SWojciech Macek 	DEVMETHOD_END
156ac9b4325SWojciech Macek };
157ac9b4325SWojciech Macek 
158ac9b4325SWojciech Macek struct opalpci_softc {
159ac9b4325SWojciech Macek 	struct ofw_pci_softc ofw_sc;
160ac9b4325SWojciech Macek 	uint64_t phb_id;
161ac9b4325SWojciech Macek 	vmem_t *msi_vmem;
162ac9b4325SWojciech Macek 	int msi_base;		/* Base XIVE number */
163ac9b4325SWojciech Macek 	int base_msi_irq;	/* Base IRQ assigned by FreeBSD to this PIC */
1645c3e53efSWojciech Macek 	uint64_t *tce;		/* TCE table for 1:1 mapping */
1655c3e53efSWojciech Macek 	struct resource *r_reg;
166ac9b4325SWojciech Macek };
167ac9b4325SWojciech Macek 
168ac9b4325SWojciech Macek DEFINE_CLASS_1(pcib, opalpci_driver, opalpci_methods,
16924042910SMarcin Wojtas     sizeof(struct opalpci_softc), ofw_pcib_driver);
1705edf159fSJohn Baldwin EARLY_DRIVER_MODULE(opalpci, ofwbus, opalpci_driver, 0, 0, BUS_PASS_BUS);
171ac9b4325SWojciech Macek 
172ac9b4325SWojciech Macek static int
opalpci_probe(device_t dev)173ac9b4325SWojciech Macek opalpci_probe(device_t dev)
174ac9b4325SWojciech Macek {
175ac9b4325SWojciech Macek 	const char	*type;
176ac9b4325SWojciech Macek 
177ac9b4325SWojciech Macek 	if (opal_check() != 0)
178ac9b4325SWojciech Macek 		return (ENXIO);
179ac9b4325SWojciech Macek 
180ac9b4325SWojciech Macek 	type = ofw_bus_get_type(dev);
181ac9b4325SWojciech Macek 
182ac9b4325SWojciech Macek 	if (type == NULL || (strcmp(type, "pci") != 0 &&
183ac9b4325SWojciech Macek 	    strcmp(type, "pciex") != 0))
184ac9b4325SWojciech Macek 		return (ENXIO);
185ac9b4325SWojciech Macek 
186ac9b4325SWojciech Macek 	if (!OF_hasprop(ofw_bus_get_node(dev), "ibm,opal-phbid"))
187ac9b4325SWojciech Macek 		return (ENXIO);
188ac9b4325SWojciech Macek 
189ac9b4325SWojciech Macek 	device_set_desc(dev, "OPAL Host-PCI bridge");
190ac9b4325SWojciech Macek 	return (BUS_PROBE_GENERIC);
191ac9b4325SWojciech Macek }
192ac9b4325SWojciech Macek 
1935c3e53efSWojciech Macek static void
pci_phb3_tce_invalidate_entire(struct opalpci_softc * sc)1945c3e53efSWojciech Macek pci_phb3_tce_invalidate_entire(struct opalpci_softc *sc)
1955c3e53efSWojciech Macek {
1965c3e53efSWojciech Macek 
1975c3e53efSWojciech Macek 	mb();
1985c3e53efSWojciech Macek 	bus_write_8(sc->r_reg, 0x210, PHB3_TCE_KILL_INVAL_ALL);
1995c3e53efSWojciech Macek 	mb();
2005c3e53efSWojciech Macek }
2015c3e53efSWojciech Macek 
202dceea51eSJustin Hibbits /* Simple function to round to a power of 2 */
203dceea51eSJustin Hibbits static uint64_t
round_pow2(uint64_t val)204dceea51eSJustin Hibbits round_pow2(uint64_t val)
205dceea51eSJustin Hibbits {
206dceea51eSJustin Hibbits 
207dceea51eSJustin Hibbits 	return (1 << (flsl(val + (val - 1)) - 1));
208dceea51eSJustin Hibbits }
209dceea51eSJustin Hibbits 
210dceea51eSJustin Hibbits /*
211dceea51eSJustin Hibbits  * Starting with skiboot 5.10 PCIe nodes have a new property,
212dceea51eSJustin Hibbits  * "ibm,supported-tce-sizes", to denote the TCE sizes available.  This allows us
213dceea51eSJustin Hibbits  * to avoid hard-coding the maximum TCE size allowed, and instead provide a sane
214dceea51eSJustin Hibbits  * default (however, the "sane" default, which works for all targets, is 64k,
215dceea51eSJustin Hibbits  * limiting us to 64GB if we have 1M entries.
216dceea51eSJustin Hibbits  */
217dceea51eSJustin Hibbits static uint64_t
max_tce_size(device_t dev)218dceea51eSJustin Hibbits max_tce_size(device_t dev)
219dceea51eSJustin Hibbits {
220dceea51eSJustin Hibbits 	phandle_t node;
221dceea51eSJustin Hibbits 	cell_t sizes[64]; /* Property is a list of bit-widths, up to 64-bits */
222dceea51eSJustin Hibbits 	int count;
223dceea51eSJustin Hibbits 
224dceea51eSJustin Hibbits 	node = ofw_bus_get_node(dev);
225dceea51eSJustin Hibbits 
226dceea51eSJustin Hibbits 	count = OF_getencprop(node, "ibm,supported-tce-sizes",
227dceea51eSJustin Hibbits 	    sizes, sizeof(sizes));
2285ecc8c20SBreno Leitao 	if (count < (int) sizeof(cell_t))
229dceea51eSJustin Hibbits 		return OPAL_PCI_TCE_DEFAULT_SEG_SIZE;
230dceea51eSJustin Hibbits 
231dceea51eSJustin Hibbits 	count /= sizeof(cell_t);
232dceea51eSJustin Hibbits 
233dceea51eSJustin Hibbits 	return (1ULL << sizes[count - 1]);
234dceea51eSJustin Hibbits }
235dceea51eSJustin Hibbits 
236ac9b4325SWojciech Macek static int
opalpci_attach(device_t dev)237ac9b4325SWojciech Macek opalpci_attach(device_t dev)
238ac9b4325SWojciech Macek {
239ac9b4325SWojciech Macek 	struct opalpci_softc *sc;
240dceea51eSJustin Hibbits 	cell_t id[2], m64ranges[2], m64window[6], npe;
241f07ee2a7SJustin Hibbits 	phandle_t node;
242ac9b4325SWojciech Macek 	int i, err;
2435c3e53efSWojciech Macek 	uint64_t maxmem;
2445c3e53efSWojciech Macek 	uint64_t entries;
245dceea51eSJustin Hibbits 	uint64_t tce_size;
246dceea51eSJustin Hibbits 	uint64_t tce_tbl_size;
247dceea51eSJustin Hibbits 	int m64bar;
2485c3e53efSWojciech Macek 	int rid;
249ac9b4325SWojciech Macek 
250ac9b4325SWojciech Macek 	sc = device_get_softc(dev);
251f07ee2a7SJustin Hibbits 	node = ofw_bus_get_node(dev);
252ac9b4325SWojciech Macek 
253f07ee2a7SJustin Hibbits 	switch (OF_getproplen(node, "ibm,opal-phbid")) {
254ac9b4325SWojciech Macek 	case 8:
255f07ee2a7SJustin Hibbits 		OF_getencprop(node, "ibm,opal-phbid", id, 8);
256ac9b4325SWojciech Macek 		sc->phb_id = ((uint64_t)id[0] << 32) | id[1];
257ac9b4325SWojciech Macek 		break;
258ac9b4325SWojciech Macek 	case 4:
259f07ee2a7SJustin Hibbits 		OF_getencprop(node, "ibm,opal-phbid", id, 4);
260ac9b4325SWojciech Macek 		sc->phb_id = id[0];
261ac9b4325SWojciech Macek 		break;
262ac9b4325SWojciech Macek 	default:
263ac9b4325SWojciech Macek 		device_printf(dev, "PHB ID property had wrong length (%zd)\n",
264f07ee2a7SJustin Hibbits 		    OF_getproplen(node, "ibm,opal-phbid"));
265ac9b4325SWojciech Macek 		return (ENXIO);
266ac9b4325SWojciech Macek 	}
267ac9b4325SWojciech Macek 
268ac9b4325SWojciech Macek 	if (bootverbose)
269ac9b4325SWojciech Macek 		device_printf(dev, "OPAL ID %#lx\n", sc->phb_id);
270ac9b4325SWojciech Macek 
2715c3e53efSWojciech Macek 	rid = 0;
2725c3e53efSWojciech Macek 	sc->r_reg = bus_alloc_resource_any(dev, SYS_RES_MEMORY,
273*bf2166bdSJustin Hibbits 	    &rid, RF_ACTIVE | RF_SHAREABLE);
2745c3e53efSWojciech Macek 	if (sc->r_reg == NULL) {
2755c3e53efSWojciech Macek 		device_printf(dev, "Failed to allocate PHB[%jd] registers\n",
2765c3e53efSWojciech Macek 		    (uintmax_t)sc->phb_id);
2775c3e53efSWojciech Macek 		return (ENXIO);
2785c3e53efSWojciech Macek 	}
2795c3e53efSWojciech Macek 
280dceea51eSJustin Hibbits #if 0
281ac9b4325SWojciech Macek 	/*
282ac9b4325SWojciech Macek 	 * Reset PCI IODA table
283ac9b4325SWojciech Macek 	 */
284ac9b4325SWojciech Macek 	err = opal_call(OPAL_PCI_RESET, sc->phb_id, OPAL_RESET_PCI_IODA_TABLE,
285ac9b4325SWojciech Macek 	    1);
286ac9b4325SWojciech Macek 	if (err != 0) {
287ac9b4325SWojciech Macek 		device_printf(dev, "IODA table reset failed: %d\n", err);
288ac9b4325SWojciech Macek 		return (ENXIO);
289ac9b4325SWojciech Macek 	}
290dceea51eSJustin Hibbits 	err = opal_call(OPAL_PCI_RESET, sc->phb_id, OPAL_RESET_PHB_COMPLETE,
291dceea51eSJustin Hibbits 	    1);
292dceea51eSJustin Hibbits 	if (err < 0) {
293dceea51eSJustin Hibbits 		device_printf(dev, "PHB reset failed: %d\n", err);
294dceea51eSJustin Hibbits 		return (ENXIO);
295dceea51eSJustin Hibbits 	}
296dceea51eSJustin Hibbits 	if (err > 0) {
297dceea51eSJustin Hibbits 		while ((err = opal_call(OPAL_PCI_POLL, sc->phb_id)) > 0) {
2985c3e53efSWojciech Macek 			DELAY(1000*(err + 1)); /* Returns expected delay in ms */
299dceea51eSJustin Hibbits 		}
300dceea51eSJustin Hibbits 	}
301ac9b4325SWojciech Macek 	if (err < 0) {
3025c3e53efSWojciech Macek 		device_printf(dev, "WARNING: PHB IODA reset poll failed: %d\n", err);
303ac9b4325SWojciech Macek 	}
304dceea51eSJustin Hibbits 	err = opal_call(OPAL_PCI_RESET, sc->phb_id, OPAL_RESET_PHB_COMPLETE,
305dceea51eSJustin Hibbits 	    0);
306dceea51eSJustin Hibbits 	if (err < 0) {
307dceea51eSJustin Hibbits 		device_printf(dev, "PHB reset failed: %d\n", err);
308dceea51eSJustin Hibbits 		return (ENXIO);
309dceea51eSJustin Hibbits 	}
310dceea51eSJustin Hibbits 	if (err > 0) {
311dceea51eSJustin Hibbits 		while ((err = opal_call(OPAL_PCI_POLL, sc->phb_id)) > 0) {
312dceea51eSJustin Hibbits 			DELAY(1000*(err + 1)); /* Returns expected delay in ms */
313dceea51eSJustin Hibbits 		}
314dceea51eSJustin Hibbits 	}
315dceea51eSJustin Hibbits #endif
316ac9b4325SWojciech Macek 
317ac9b4325SWojciech Macek 	/*
318ac9b4325SWojciech Macek 	 * Map all devices on the bus to partitionable endpoint one until
319ac9b4325SWojciech Macek 	 * such time as we start wanting to do things like bhyve.
320ac9b4325SWojciech Macek 	 */
321ac9b4325SWojciech Macek 	err = opal_call(OPAL_PCI_SET_PE, sc->phb_id, OPAL_PCI_DEFAULT_PE,
32230f3b0f5SJustin Hibbits 	    0, OPAL_PCI_BUS_ANY, OPAL_IGNORE_RID_DEVICE_NUMBER,
32330f3b0f5SJustin Hibbits 	    OPAL_IGNORE_RID_FUNC_NUMBER, OPAL_MAP_PE);
324ac9b4325SWojciech Macek 	if (err != 0) {
325ac9b4325SWojciech Macek 		device_printf(dev, "PE mapping failed: %d\n", err);
326ac9b4325SWojciech Macek 		return (ENXIO);
327ac9b4325SWojciech Macek 	}
328ac9b4325SWojciech Macek 
329ac9b4325SWojciech Macek 	/*
330ac9b4325SWojciech Macek 	 * Turn on MMIO, mapped to PE 1
331ac9b4325SWojciech Macek 	 */
332f07ee2a7SJustin Hibbits 	if (OF_getencprop(node, "ibm,opal-num-pes", &npe, 4) != 4)
333ac9b4325SWojciech Macek 		npe = 1;
334ac9b4325SWojciech Macek 	for (i = 0; i < npe; i++) {
335ac9b4325SWojciech Macek 		err = opal_call(OPAL_PCI_MAP_PE_MMIO_WINDOW, sc->phb_id,
336ac9b4325SWojciech Macek 		    OPAL_PCI_DEFAULT_PE, OPAL_M32_WINDOW_TYPE, 0, i);
337ac9b4325SWojciech Macek 		if (err != 0)
338ac9b4325SWojciech Macek 			device_printf(dev, "MMIO %d map failed: %d\n", i, err);
339ac9b4325SWojciech Macek 	}
340ac9b4325SWojciech Macek 
341dceea51eSJustin Hibbits 	if (OF_getencprop(node, "ibm,opal-available-m64-ranges",
342dceea51eSJustin Hibbits 	    m64ranges, sizeof(m64ranges)) == sizeof(m64ranges))
343dceea51eSJustin Hibbits 		m64bar = m64ranges[0];
344dceea51eSJustin Hibbits 	else
345dceea51eSJustin Hibbits 	    m64bar = 0;
346dceea51eSJustin Hibbits 
347ac9b4325SWojciech Macek 	/* XXX: multiple M64 windows? */
348f07ee2a7SJustin Hibbits 	if (OF_getencprop(node, "ibm,opal-m64-window",
349ac9b4325SWojciech Macek 	    m64window, sizeof(m64window)) == sizeof(m64window)) {
3505c3e53efSWojciech Macek 		opal_call(OPAL_PCI_PHB_MMIO_ENABLE, sc->phb_id,
351dceea51eSJustin Hibbits 		    OPAL_M64_WINDOW_TYPE, m64bar, 0);
352ac9b4325SWojciech Macek 		opal_call(OPAL_PCI_SET_PHB_MEM_WINDOW, sc->phb_id,
353dceea51eSJustin Hibbits 		    OPAL_M64_WINDOW_TYPE, m64bar /* index */,
354ac9b4325SWojciech Macek 		    ((uint64_t)m64window[2] << 32) | m64window[3], 0,
355ac9b4325SWojciech Macek 		    ((uint64_t)m64window[4] << 32) | m64window[5]);
356ac9b4325SWojciech Macek 		opal_call(OPAL_PCI_MAP_PE_MMIO_WINDOW, sc->phb_id,
357ac9b4325SWojciech Macek 		    OPAL_PCI_DEFAULT_PE, OPAL_M64_WINDOW_TYPE,
358dceea51eSJustin Hibbits 		    m64bar /* index */, 0);
359ac9b4325SWojciech Macek 		opal_call(OPAL_PCI_PHB_MMIO_ENABLE, sc->phb_id,
360dceea51eSJustin Hibbits 		    OPAL_M64_WINDOW_TYPE, m64bar, OPAL_ENABLE_M64_NON_SPLIT);
361ac9b4325SWojciech Macek 	}
362ac9b4325SWojciech Macek 
363ac9b4325SWojciech Macek 	/*
3645c3e53efSWojciech Macek 	 * Enable IOMMU for PE1 - map everything 1:1 using
365dceea51eSJustin Hibbits 	 * segments of max_tce_size size
366ac9b4325SWojciech Macek 	 */
367dceea51eSJustin Hibbits 	tce_size = max_tce_size(dev);
368dceea51eSJustin Hibbits 	maxmem = roundup2(powerpc_ptob(Maxmem), tce_size);
369dceea51eSJustin Hibbits 	entries = round_pow2(maxmem / tce_size);
37049d9a597SJustin Hibbits 	tce_tbl_size = MAX(entries * sizeof(uint64_t), 4096);
3715c3e53efSWojciech Macek 	if (entries > OPAL_PCI_TCE_MAX_ENTRIES)
3725c3e53efSWojciech Macek 		panic("POWERNV supports only %jdGB of memory space\n",
373dceea51eSJustin Hibbits 		    (uintmax_t)((OPAL_PCI_TCE_MAX_ENTRIES * tce_size) >> 30));
374ac9b4325SWojciech Macek 	if (bootverbose)
3755c3e53efSWojciech Macek 		device_printf(dev, "Mapping 0-%#jx for DMA\n", (uintmax_t)maxmem);
376dceea51eSJustin Hibbits 	sc->tce = contigmalloc(tce_tbl_size,
3775c3e53efSWojciech Macek 	    M_DEVBUF, M_NOWAIT | M_ZERO, 0,
3782756851aSJustin Hibbits 	    BUS_SPACE_MAXADDR, tce_tbl_size, 0);
3795c3e53efSWojciech Macek 	if (sc->tce == NULL)
3805c3e53efSWojciech Macek 		panic("Failed to allocate TCE memory for PHB %jd\n",
3815c3e53efSWojciech Macek 		    (uintmax_t)sc->phb_id);
3825c3e53efSWojciech Macek 
3835c3e53efSWojciech Macek 	for (i = 0; i < entries; i++)
384f9acb7a8SBrandon Bergren 		sc->tce[i] = htobe64((i * tce_size) | OPAL_PCI_TCE_R | OPAL_PCI_TCE_W);
3855c3e53efSWojciech Macek 
3865c3e53efSWojciech Macek 	/* Map TCE for every PE. It seems necessary for Power8 */
3875c3e53efSWojciech Macek 	for (i = 0; i < npe; i++) {
3885c3e53efSWojciech Macek 		err = opal_call(OPAL_PCI_MAP_PE_DMA_WINDOW, sc->phb_id,
3895c3e53efSWojciech Macek 		    i, (i << 1),
3905c3e53efSWojciech Macek 		    1, pmap_kextract((uint64_t)&sc->tce[0]),
391dceea51eSJustin Hibbits 		    tce_tbl_size, tce_size);
392ac9b4325SWojciech Macek 		if (err != 0) {
3935c3e53efSWojciech Macek 			device_printf(dev, "DMA IOMMU mapping failed: %d\n", err);
394ac9b4325SWojciech Macek 			return (ENXIO);
395ac9b4325SWojciech Macek 		}
396ac9b4325SWojciech Macek 
3975c3e53efSWojciech Macek 		err = opal_call(OPAL_PCI_MAP_PE_DMA_WINDOW_REAL, sc->phb_id,
3985c3e53efSWojciech Macek 		    i, (i << 1) + 1,
3995c3e53efSWojciech Macek 		    (1UL << 59), maxmem);
4005c3e53efSWojciech Macek 		if (err != 0) {
4015c3e53efSWojciech Macek 			device_printf(dev, "DMA 64b bypass mapping failed: %d\n", err);
4025c3e53efSWojciech Macek 			return (ENXIO);
4035c3e53efSWojciech Macek 		}
4045c3e53efSWojciech Macek 	}
4055c3e53efSWojciech Macek 
4065c3e53efSWojciech Macek 	/*
4075c3e53efSWojciech Macek 	 * Invalidate all previous TCE entries.
4085c3e53efSWojciech Macek 	 */
4090b475340SJustin Hibbits 	if (ofw_bus_is_compatible(dev, "power8-pciex"))
4105c3e53efSWojciech Macek 		pci_phb3_tce_invalidate_entire(sc);
4110b475340SJustin Hibbits 	else
4120b475340SJustin Hibbits 		opal_call(OPAL_PCI_TCE_KILL, sc->phb_id, OPAL_PCI_TCE_KILL_ALL,
4130b475340SJustin Hibbits 		    OPAL_PCI_DEFAULT_PE, 0, 0, 0);
4145c3e53efSWojciech Macek 
415ac9b4325SWojciech Macek 	/*
416ac9b4325SWojciech Macek 	 * Get MSI properties
417ac9b4325SWojciech Macek 	 */
418ac9b4325SWojciech Macek 	sc->msi_vmem = NULL;
419f07ee2a7SJustin Hibbits 	if (OF_getproplen(node, "ibm,opal-msi-ranges") > 0) {
420ac9b4325SWojciech Macek 		cell_t msi_ranges[2];
421f07ee2a7SJustin Hibbits 		OF_getencprop(node, "ibm,opal-msi-ranges",
422ac9b4325SWojciech Macek 		    msi_ranges, sizeof(msi_ranges));
423ac9b4325SWojciech Macek 		sc->msi_base = msi_ranges[0];
424ac9b4325SWojciech Macek 
425ac9b4325SWojciech Macek 		sc->msi_vmem = vmem_create("OPAL MSI", msi_ranges[0],
42646e8ab5aSJustin Hibbits 		    msi_ranges[1], 1, 0, M_BESTFIT | M_WAITOK);
427ac9b4325SWojciech Macek 
428ac9b4325SWojciech Macek 		sc->base_msi_irq = powerpc_register_pic(dev,
429f07ee2a7SJustin Hibbits 		    OF_xref_from_node(node),
430ac9b4325SWojciech Macek 		    msi_ranges[0] + msi_ranges[1], 0, FALSE);
431ac9b4325SWojciech Macek 
432ac9b4325SWojciech Macek 		if (bootverbose)
433ac9b4325SWojciech Macek 			device_printf(dev, "Supports %d MSIs starting at %d\n",
434ac9b4325SWojciech Macek 			    msi_ranges[1], msi_ranges[0]);
435ac9b4325SWojciech Macek 	}
436ac9b4325SWojciech Macek 
437d93e635aSLeandro Lupori 	/* Create the parent DMA tag */
438ad39591aSJustin Hibbits 	/*
439ad39591aSJustin Hibbits 	 * Constrain it to POWER8 PHB (ioda2) for now.  It seems to mess up on
440ad39591aSJustin Hibbits 	 * POWER9 systems.
441ad39591aSJustin Hibbits 	 */
442ad39591aSJustin Hibbits 	if (ofw_bus_is_compatible(dev, "ibm,ioda2-phb")) {
443d93e635aSLeandro Lupori 		err = bus_dma_tag_create(bus_get_dma_tag(dev), /* parent */
444d93e635aSLeandro Lupori 		    1, 0,				/* alignment, bounds */
445d93e635aSLeandro Lupori 		    OPAL_PCI_BUS_SPACE_LOWADDR_32BIT,	/* lowaddr */
446d93e635aSLeandro Lupori 		    BUS_SPACE_MAXADDR_32BIT,		/* highaddr */
447d93e635aSLeandro Lupori 		    NULL, NULL,				/* filter, filterarg */
448d93e635aSLeandro Lupori 		    BUS_SPACE_MAXSIZE,			/* maxsize */
449d93e635aSLeandro Lupori 		    BUS_SPACE_UNRESTRICTED,		/* nsegments */
450d93e635aSLeandro Lupori 		    BUS_SPACE_MAXSIZE,			/* maxsegsize */
451d93e635aSLeandro Lupori 		    0,					/* flags */
452d93e635aSLeandro Lupori 		    NULL, NULL,				/* lockfunc, lockarg */
453d93e635aSLeandro Lupori 		    &sc->ofw_sc.sc_dmat);
454d93e635aSLeandro Lupori 		if (err != 0) {
455d93e635aSLeandro Lupori 			device_printf(dev, "Failed to create DMA tag\n");
456d93e635aSLeandro Lupori 			return (err);
457d93e635aSLeandro Lupori 		}
458ad39591aSJustin Hibbits 	}
459d93e635aSLeandro Lupori 
460ac9b4325SWojciech Macek 	/*
461ac9b4325SWojciech Macek 	 * General OFW PCI attach
462ac9b4325SWojciech Macek 	 */
46324042910SMarcin Wojtas 	err = ofw_pcib_init(dev);
464ac9b4325SWojciech Macek 	if (err != 0)
465ac9b4325SWojciech Macek 		return (err);
466ac9b4325SWojciech Macek 
467ac9b4325SWojciech Macek 	/*
468ac9b4325SWojciech Macek 	 * Unfreeze non-config-space PCI operations. Let this fail silently
469ac9b4325SWojciech Macek 	 * if e.g. there is no current freeze.
470ac9b4325SWojciech Macek 	 */
471ac9b4325SWojciech Macek 	opal_call(OPAL_PCI_EEH_FREEZE_CLEAR, sc->phb_id, OPAL_PCI_DEFAULT_PE,
472ac9b4325SWojciech Macek 	    OPAL_EEH_ACTION_CLEAR_FREEZE_ALL);
473ac9b4325SWojciech Macek 
474ac9b4325SWojciech Macek 	/*
475ac9b4325SWojciech Macek 	 * OPAL stores 64-bit BARs in a special property rather than "ranges"
476ac9b4325SWojciech Macek 	 */
477f07ee2a7SJustin Hibbits 	if (OF_getencprop(node, "ibm,opal-m64-window",
478ac9b4325SWojciech Macek 	    m64window, sizeof(m64window)) == sizeof(m64window)) {
479ac9b4325SWojciech Macek 		struct ofw_pci_range *rp;
480ac9b4325SWojciech Macek 
481ac9b4325SWojciech Macek 		sc->ofw_sc.sc_nrange++;
482ac9b4325SWojciech Macek 		sc->ofw_sc.sc_range = realloc(sc->ofw_sc.sc_range,
483ac9b4325SWojciech Macek 		    sc->ofw_sc.sc_nrange * sizeof(sc->ofw_sc.sc_range[0]),
484ac9b4325SWojciech Macek 		    M_DEVBUF, M_WAITOK);
485ac9b4325SWojciech Macek 		rp = &sc->ofw_sc.sc_range[sc->ofw_sc.sc_nrange-1];
486ac9b4325SWojciech Macek 		rp->pci_hi = OFW_PCI_PHYS_HI_SPACE_MEM64 |
487ac9b4325SWojciech Macek 		    OFW_PCI_PHYS_HI_PREFETCHABLE;
488ac9b4325SWojciech Macek 		rp->pci = ((uint64_t)m64window[0] << 32) | m64window[1];
489ac9b4325SWojciech Macek 		rp->host = ((uint64_t)m64window[2] << 32) | m64window[3];
490ac9b4325SWojciech Macek 		rp->size = ((uint64_t)m64window[4] << 32) | m64window[5];
491ac9b4325SWojciech Macek 		rman_manage_region(&sc->ofw_sc.sc_mem_rman, rp->pci,
492ac9b4325SWojciech Macek 		   rp->pci + rp->size - 1);
493ac9b4325SWojciech Macek 	}
494ac9b4325SWojciech Macek 
49524042910SMarcin Wojtas 	return (ofw_pcib_attach(dev));
496ac9b4325SWojciech Macek }
497ac9b4325SWojciech Macek 
498ac9b4325SWojciech Macek static uint32_t
opalpci_read_config(device_t dev,u_int bus,u_int slot,u_int func,u_int reg,int width)499ac9b4325SWojciech Macek opalpci_read_config(device_t dev, u_int bus, u_int slot, u_int func, u_int reg,
500ac9b4325SWojciech Macek     int width)
501ac9b4325SWojciech Macek {
502ac9b4325SWojciech Macek 	struct opalpci_softc *sc;
503ac9b4325SWojciech Macek 	uint64_t config_addr;
5040effb2ccSJustin Hibbits 	uint8_t byte, eeh_state;
505ac9b4325SWojciech Macek 	uint16_t half;
506ac9b4325SWojciech Macek 	uint32_t word;
507ac9b4325SWojciech Macek 	int error;
5080effb2ccSJustin Hibbits 	uint16_t err_type;
509ac9b4325SWojciech Macek 
510ac9b4325SWojciech Macek 	sc = device_get_softc(dev);
511ac9b4325SWojciech Macek 
512ac9b4325SWojciech Macek 	config_addr = (bus << 8) | ((slot & 0x1f) << 3) | (func & 0x7);
513ac9b4325SWojciech Macek 
514ac9b4325SWojciech Macek 	switch (width) {
515ac9b4325SWojciech Macek 	case 1:
516ac9b4325SWojciech Macek 		error = opal_call(OPAL_PCI_CONFIG_READ_BYTE, sc->phb_id,
517ac9b4325SWojciech Macek 		    config_addr, reg, vtophys(&byte));
518ac9b4325SWojciech Macek 		word = byte;
519ac9b4325SWojciech Macek 		break;
520ac9b4325SWojciech Macek 	case 2:
521ac9b4325SWojciech Macek 		error = opal_call(OPAL_PCI_CONFIG_READ_HALF_WORD, sc->phb_id,
522ac9b4325SWojciech Macek 		    config_addr, reg, vtophys(&half));
5239cbcb6ffSBrandon Bergren 		word = be16toh(half);
524ac9b4325SWojciech Macek 		break;
525ac9b4325SWojciech Macek 	case 4:
526ac9b4325SWojciech Macek 		error = opal_call(OPAL_PCI_CONFIG_READ_WORD, sc->phb_id,
527ac9b4325SWojciech Macek 		    config_addr, reg, vtophys(&word));
5289cbcb6ffSBrandon Bergren 		word = be32toh(word);
529ac9b4325SWojciech Macek 		break;
530ac9b4325SWojciech Macek 	default:
53172820025SNathan Whitehorn 		error = OPAL_SUCCESS;
532ac9b4325SWojciech Macek 		word = 0xffffffff;
53384ce4f03SJustin Hibbits 		width = 4;
534ac9b4325SWojciech Macek 	}
535ac9b4325SWojciech Macek 
536ac9b4325SWojciech Macek 	/*
537ac9b4325SWojciech Macek 	 * Poking config state for non-existant devices can make
538ac9b4325SWojciech Macek 	 * the host bridge hang up. Clear any errors.
539ac9b4325SWojciech Macek 	 */
540ac9b4325SWojciech Macek 
54184ce4f03SJustin Hibbits 	if (error != OPAL_SUCCESS ||
54284ce4f03SJustin Hibbits 	    (word == ((1UL << (8 * width)) - 1))) {
5430effb2ccSJustin Hibbits 		if (error != OPAL_HARDWARE) {
5440effb2ccSJustin Hibbits 			opal_call(OPAL_PCI_EEH_FREEZE_STATUS, sc->phb_id,
5450effb2ccSJustin Hibbits 			    OPAL_PCI_DEFAULT_PE, vtophys(&eeh_state),
5460effb2ccSJustin Hibbits 			    vtophys(&err_type), NULL);
5479cbcb6ffSBrandon Bergren 			err_type = be16toh(err_type); /* XXX unused */
5480effb2ccSJustin Hibbits 			if (eeh_state != OPAL_EEH_STOPPED_NOT_FROZEN)
5490effb2ccSJustin Hibbits 				opal_call(OPAL_PCI_EEH_FREEZE_CLEAR,
5500effb2ccSJustin Hibbits 				    sc->phb_id, OPAL_PCI_DEFAULT_PE,
5510effb2ccSJustin Hibbits 				    OPAL_EEH_ACTION_CLEAR_FREEZE_ALL);
5520effb2ccSJustin Hibbits 		}
55384ce4f03SJustin Hibbits 		if (error != OPAL_SUCCESS)
554ac9b4325SWojciech Macek 			word = 0xffffffff;
5550effb2ccSJustin Hibbits 	}
556ac9b4325SWojciech Macek 
557ac9b4325SWojciech Macek 	return (word);
558ac9b4325SWojciech Macek }
559ac9b4325SWojciech Macek 
560ac9b4325SWojciech Macek static void
opalpci_write_config(device_t dev,u_int bus,u_int slot,u_int func,u_int reg,uint32_t val,int width)561ac9b4325SWojciech Macek opalpci_write_config(device_t dev, u_int bus, u_int slot, u_int func,
562ac9b4325SWojciech Macek     u_int reg, uint32_t val, int width)
563ac9b4325SWojciech Macek {
564ac9b4325SWojciech Macek 	struct opalpci_softc *sc;
565ac9b4325SWojciech Macek 	uint64_t config_addr;
566ac9b4325SWojciech Macek 	int error = OPAL_SUCCESS;
567ac9b4325SWojciech Macek 
568ac9b4325SWojciech Macek 	sc = device_get_softc(dev);
569ac9b4325SWojciech Macek 
570ac9b4325SWojciech Macek 	config_addr = (bus << 8) | ((slot & 0x1f) << 3) | (func & 0x7);
571ac9b4325SWojciech Macek 
572ac9b4325SWojciech Macek 	switch (width) {
573ac9b4325SWojciech Macek 	case 1:
574ac9b4325SWojciech Macek 		error = opal_call(OPAL_PCI_CONFIG_WRITE_BYTE, sc->phb_id,
575ac9b4325SWojciech Macek 		    config_addr, reg, val);
576ac9b4325SWojciech Macek 		break;
577ac9b4325SWojciech Macek 	case 2:
578ac9b4325SWojciech Macek 		error = opal_call(OPAL_PCI_CONFIG_WRITE_HALF_WORD, sc->phb_id,
579ac9b4325SWojciech Macek 		    config_addr, reg, val);
580ac9b4325SWojciech Macek 		break;
581ac9b4325SWojciech Macek 	case 4:
582ac9b4325SWojciech Macek 		error = opal_call(OPAL_PCI_CONFIG_WRITE_WORD, sc->phb_id,
583ac9b4325SWojciech Macek 		    config_addr, reg, val);
584ac9b4325SWojciech Macek 		break;
585ac9b4325SWojciech Macek 	}
586ac9b4325SWojciech Macek 
587ac9b4325SWojciech Macek 	if (error != OPAL_SUCCESS) {
588ac9b4325SWojciech Macek 		/*
589ac9b4325SWojciech Macek 		 * Poking config state for non-existant devices can make
590ac9b4325SWojciech Macek 		 * the host bridge hang up. Clear any errors.
591ac9b4325SWojciech Macek 		 */
5920effb2ccSJustin Hibbits 		if (error != OPAL_HARDWARE) {
5930effb2ccSJustin Hibbits 			opal_call(OPAL_PCI_EEH_FREEZE_CLEAR,
5940effb2ccSJustin Hibbits 			    sc->phb_id, OPAL_PCI_DEFAULT_PE,
5950effb2ccSJustin Hibbits 			    OPAL_EEH_ACTION_CLEAR_FREEZE_ALL);
5960effb2ccSJustin Hibbits 		}
597ac9b4325SWojciech Macek 	}
598ac9b4325SWojciech Macek }
599ac9b4325SWojciech Macek 
600ac9b4325SWojciech Macek static int
opalpci_route_interrupt(device_t bus,device_t dev,int pin)601ac9b4325SWojciech Macek opalpci_route_interrupt(device_t bus, device_t dev, int pin)
602ac9b4325SWojciech Macek {
6035c3e53efSWojciech Macek 
604ac9b4325SWojciech Macek 	return (pin);
605ac9b4325SWojciech Macek }
606ac9b4325SWojciech Macek 
607ac9b4325SWojciech Macek static int
opalpci_alloc_msi(device_t dev,device_t child,int count,int maxcount,int * irqs)608ac9b4325SWojciech Macek opalpci_alloc_msi(device_t dev, device_t child, int count, int maxcount,
609ac9b4325SWojciech Macek     int *irqs)
610ac9b4325SWojciech Macek {
611ac9b4325SWojciech Macek 	struct opalpci_softc *sc;
612ac9b4325SWojciech Macek 	vmem_addr_t start;
613ac9b4325SWojciech Macek 	phandle_t xref;
614ac9b4325SWojciech Macek 	int err, i;
615ac9b4325SWojciech Macek 
616ac9b4325SWojciech Macek 	sc = device_get_softc(dev);
617ac9b4325SWojciech Macek 	if (sc->msi_vmem == NULL)
618ac9b4325SWojciech Macek 		return (ENODEV);
619ac9b4325SWojciech Macek 
620ac9b4325SWojciech Macek 	err = vmem_xalloc(sc->msi_vmem, count, powerof2(count), 0, 0,
621ac9b4325SWojciech Macek 	    VMEM_ADDR_MIN, VMEM_ADDR_MAX, M_BESTFIT | M_WAITOK, &start);
622ac9b4325SWojciech Macek 
623ac9b4325SWojciech Macek 	if (err)
624ac9b4325SWojciech Macek 		return (err);
625ac9b4325SWojciech Macek 
626ac9b4325SWojciech Macek 	xref = OF_xref_from_node(ofw_bus_get_node(dev));
627ac9b4325SWojciech Macek 	for (i = 0; i < count; i++)
628ac9b4325SWojciech Macek 		irqs[i] = MAP_IRQ(xref, start + i);
629ac9b4325SWojciech Macek 
630ac9b4325SWojciech Macek 	return (0);
631ac9b4325SWojciech Macek }
632ac9b4325SWojciech Macek 
633ac9b4325SWojciech Macek static int
opalpci_release_msi(device_t dev,device_t child,int count,int * irqs)634ac9b4325SWojciech Macek opalpci_release_msi(device_t dev, device_t child, int count, int *irqs)
635ac9b4325SWojciech Macek {
636ac9b4325SWojciech Macek 	struct opalpci_softc *sc;
637ac9b4325SWojciech Macek 
638ac9b4325SWojciech Macek 	sc = device_get_softc(dev);
639ac9b4325SWojciech Macek 	if (sc->msi_vmem == NULL)
640ac9b4325SWojciech Macek 		return (ENODEV);
641ac9b4325SWojciech Macek 
642ac9b4325SWojciech Macek 	vmem_xfree(sc->msi_vmem, irqs[0] - sc->base_msi_irq, count);
643ac9b4325SWojciech Macek 	return (0);
644ac9b4325SWojciech Macek }
645ac9b4325SWojciech Macek 
646ac9b4325SWojciech Macek static int
opalpci_alloc_msix(device_t dev,device_t child,int * irq)647ac9b4325SWojciech Macek opalpci_alloc_msix(device_t dev, device_t child, int *irq)
648ac9b4325SWojciech Macek {
649ac9b4325SWojciech Macek 	return (opalpci_alloc_msi(dev, child, 1, 1, irq));
650ac9b4325SWojciech Macek }
651ac9b4325SWojciech Macek 
652ac9b4325SWojciech Macek static int
opalpci_release_msix(device_t dev,device_t child,int irq)653ac9b4325SWojciech Macek opalpci_release_msix(device_t dev, device_t child, int irq)
654ac9b4325SWojciech Macek {
655ac9b4325SWojciech Macek 	return (opalpci_release_msi(dev, child, 1, &irq));
656ac9b4325SWojciech Macek }
657ac9b4325SWojciech Macek 
658ac9b4325SWojciech Macek static int
opalpci_map_msi(device_t dev,device_t child,int irq,uint64_t * addr,uint32_t * data)659ac9b4325SWojciech Macek opalpci_map_msi(device_t dev, device_t child, int irq, uint64_t *addr,
660ac9b4325SWojciech Macek     uint32_t *data)
661ac9b4325SWojciech Macek {
662ac9b4325SWojciech Macek 	struct opalpci_softc *sc;
663ac9b4325SWojciech Macek 	struct pci_devinfo *dinfo;
664ac9b4325SWojciech Macek 	int err, xive;
665ac9b4325SWojciech Macek 
666ac9b4325SWojciech Macek 	sc = device_get_softc(dev);
667ac9b4325SWojciech Macek 	if (sc->msi_vmem == NULL)
668ac9b4325SWojciech Macek 		return (ENODEV);
669ac9b4325SWojciech Macek 
670ac9b4325SWojciech Macek 	xive = irq - sc->base_msi_irq - sc->msi_base;
671ac9b4325SWojciech Macek 	opal_call(OPAL_PCI_SET_XIVE_PE, sc->phb_id, OPAL_PCI_DEFAULT_PE, xive);
672ac9b4325SWojciech Macek 
673ac9b4325SWojciech Macek 	dinfo = device_get_ivars(child);
674ac9b4325SWojciech Macek 	if (dinfo->cfg.msi.msi_alloc > 0 &&
675ac9b4325SWojciech Macek 	    (dinfo->cfg.msi.msi_ctrl & PCIM_MSICTRL_64BIT) == 0) {
676ac9b4325SWojciech Macek 		uint32_t msi32;
677ac9b4325SWojciech Macek 		err = opal_call(OPAL_GET_MSI_32, sc->phb_id,
678ac9b4325SWojciech Macek 		    OPAL_PCI_DEFAULT_PE, xive, 1, vtophys(&msi32),
679ac9b4325SWojciech Macek 		    vtophys(data));
680ac9b4325SWojciech Macek 		*addr = be32toh(msi32);
681ac9b4325SWojciech Macek 	} else {
682ac9b4325SWojciech Macek 		err = opal_call(OPAL_GET_MSI_64, sc->phb_id,
683ac9b4325SWojciech Macek 		    OPAL_PCI_DEFAULT_PE, xive, 1, vtophys(addr), vtophys(data));
684ac9b4325SWojciech Macek 		*addr = be64toh(*addr);
685ac9b4325SWojciech Macek 	}
686ac9b4325SWojciech Macek 	*data = be32toh(*data);
687ac9b4325SWojciech Macek 
688ac9b4325SWojciech Macek 	if (bootverbose && err != 0)
689ac9b4325SWojciech Macek 		device_printf(child, "OPAL MSI mapping error: %d\n", err);
690ac9b4325SWojciech Macek 
691ac9b4325SWojciech Macek 	return ((err == 0) ? 0 : ENXIO);
692ac9b4325SWojciech Macek }
693ac9b4325SWojciech Macek 
694ac9b4325SWojciech Macek static void
opalpic_pic_enable(device_t dev,u_int irq,u_int vector,void ** priv)69556505ec0SJustin Hibbits opalpic_pic_enable(device_t dev, u_int irq, u_int vector, void **priv)
696ac9b4325SWojciech Macek {
697013cc176SJustin Hibbits 	struct opalpci_softc *sc = device_get_softc(dev);
698013cc176SJustin Hibbits 
69956505ec0SJustin Hibbits 	PIC_ENABLE(root_pic, irq, vector, priv);
70056505ec0SJustin Hibbits 	opal_call(OPAL_PCI_MSI_EOI, sc->phb_id, irq, priv);
701ac9b4325SWojciech Macek }
702ac9b4325SWojciech Macek 
opalpic_pic_eoi(device_t dev,u_int irq,void * priv)70356505ec0SJustin Hibbits static void opalpic_pic_eoi(device_t dev, u_int irq, void *priv)
704ac9b4325SWojciech Macek {
705ac9b4325SWojciech Macek 	struct opalpci_softc *sc;
706ac9b4325SWojciech Macek 
707ac9b4325SWojciech Macek 	sc = device_get_softc(dev);
708ac9b4325SWojciech Macek 	opal_call(OPAL_PCI_MSI_EOI, sc->phb_id, irq);
709ac9b4325SWojciech Macek 
71056505ec0SJustin Hibbits 	PIC_EOI(root_pic, irq, priv);
711ac9b4325SWojciech Macek }
712d93e635aSLeandro Lupori 
713d93e635aSLeandro Lupori static bus_dma_tag_t
opalpci_get_dma_tag(device_t dev,device_t child)714d93e635aSLeandro Lupori opalpci_get_dma_tag(device_t dev, device_t child)
715d93e635aSLeandro Lupori {
716d93e635aSLeandro Lupori 	struct opalpci_softc *sc;
717d93e635aSLeandro Lupori 
718d93e635aSLeandro Lupori 	sc = device_get_softc(dev);
719d93e635aSLeandro Lupori 	return (sc->ofw_sc.sc_dmat);
720d93e635aSLeandro Lupori }
721