xref: /illumos-gate/usr/src/uts/i86pc/io/psm/psm_common.c (revision 3ce33fb052b375020ea4249290d33b834d9f9e75)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 /*
22  * Copyright (c) 2004, 2010, Oracle and/or its affiliates. All rights reserved.
23  */
24 
25 #include <sys/types.h>
26 #include <sys/param.h>
27 #include <sys/cmn_err.h>
28 #include <sys/promif.h>
29 #include <sys/acpi/acpi.h>
30 #include <sys/acpica.h>
31 #include <sys/sunddi.h>
32 #include <sys/ddi.h>
33 #include <sys/ddi_impldefs.h>
34 #include <sys/pci.h>
35 #include <sys/debug.h>
36 #include <sys/psm_common.h>
37 #include <sys/sunndi.h>
38 #include <sys/ksynch.h>
39 
40 /* Global configurables */
41 
42 char *psm_module_name;	/* used to store name of psm module */
43 
44 /*
45  * acpi_irq_check_elcr: when set elcr will also be consulted for building
46  * the reserved irq list.  When 0 (false), the existing state of the ELCR
47  * is ignored when selecting a vector during IRQ translation, and the ELCR
48  * is programmed to the proper setting for the type of bus (level-triggered
49  * for PCI, edge-triggered for non-PCI).  When non-zero (true), vectors
50  * set to edge-mode will not be used when in PIC-mode.  The default value
51  * is 0 (false).  Note that ACPI's SCI vector is always set to conform to
52  * ACPI-specification regardless of this.
53  *
54  */
55 int acpi_irq_check_elcr = 0;
56 
57 int psm_verbose = 0;
58 
59 #define	PSM_VERBOSE_IRQ(fmt)	\
60 		if (psm_verbose & PSM_VERBOSE_IRQ_FLAG) \
61 			cmn_err fmt;
62 
63 #define	PSM_VERBOSE_POWEROFF(fmt)  \
64 		if (psm_verbose & PSM_VERBOSE_POWEROFF_FLAG || \
65 		    psm_verbose & PSM_VERBOSE_POWEROFF_PAUSE_FLAG) \
66 			prom_printf fmt;
67 
68 #define	PSM_VERBOSE_POWEROFF_PAUSE(fmt) \
69 		if (psm_verbose & PSM_VERBOSE_POWEROFF_FLAG || \
70 		    psm_verbose & PSM_VERBOSE_POWEROFF_PAUSE_FLAG) {\
71 			prom_printf fmt; \
72 			if (psm_verbose & PSM_VERBOSE_POWEROFF_PAUSE_FLAG) \
73 				(void) goany(); \
74 		}
75 
76 
77 /* Local storage */
78 static ACPI_HANDLE acpi_sbobj = NULL;
79 static kmutex_t acpi_irq_cache_mutex;
80 
81 /*
82  * irq_cache_table is a list that serves a two-key cache. It is used
83  * as a pci busid/devid/ipin <-> irq cache and also as a acpi
84  * interrupt lnk <-> irq cache.
85  */
86 static irq_cache_t *irq_cache_table;
87 
88 #define	IRQ_CACHE_INITLEN	20
89 static int irq_cache_len = 0;
90 static int irq_cache_valid = 0;
91 
92 static int acpi_get_gsiv(dev_info_t *dip, ACPI_HANDLE pciobj, int devno,
93 	int ipin, int *pci_irqp, iflag_t *iflagp,  acpi_psm_lnk_t *acpipsmlnkp);
94 
95 static int acpi_eval_lnk(dev_info_t *dip, char *lnkname,
96     int *pci_irqp, iflag_t *intr_flagp, acpi_psm_lnk_t *acpipsmlnkp);
97 
98 static int acpi_get_irq_lnk_cache_ent(ACPI_HANDLE lnkobj, int *pci_irqp,
99     iflag_t *intr_flagp);
100 
101 extern int goany(void);
102 
103 
104 #define	NEXT_PRT_ITEM(p)	\
105 		(void *)(((char *)(p)) + (p)->Length)
106 
107 static int
108 acpi_get_gsiv(dev_info_t *dip, ACPI_HANDLE pciobj, int devno, int ipin,
109     int *pci_irqp, iflag_t *intr_flagp, acpi_psm_lnk_t *acpipsmlnkp)
110 {
111 	ACPI_BUFFER rb;
112 	ACPI_PCI_ROUTING_TABLE *prtp;
113 	int status;
114 	int dev_adr;
115 
116 	/*
117 	 * Get the IRQ routing table
118 	 */
119 	rb.Pointer = NULL;
120 	rb.Length = ACPI_ALLOCATE_BUFFER;
121 	if (AcpiGetIrqRoutingTable(pciobj, &rb) != AE_OK) {
122 		return (ACPI_PSM_FAILURE);
123 	}
124 
125 	status = ACPI_PSM_FAILURE;
126 	dev_adr = (devno << 16 | 0xffff);
127 	for (prtp = rb.Pointer; prtp->Length != 0; prtp = NEXT_PRT_ITEM(prtp)) {
128 		/* look until a matching dev/pin is found */
129 		if (dev_adr != prtp->Address || ipin != prtp->Pin)
130 			continue;
131 
132 		/* NULL Source name means index is GSIV */
133 		if (*prtp->Source == 0) {
134 			intr_flagp->intr_el = INTR_EL_LEVEL;
135 			intr_flagp->intr_po = INTR_PO_ACTIVE_LOW;
136 			ASSERT(pci_irqp != NULL);
137 			*pci_irqp = prtp->SourceIndex;
138 			status = ACPI_PSM_SUCCESS;
139 		} else
140 			status = acpi_eval_lnk(dip, prtp->Source, pci_irqp,
141 			    intr_flagp, acpipsmlnkp);
142 
143 		break;
144 
145 	}
146 
147 	AcpiOsFree(rb.Pointer);
148 	return (status);
149 }
150 
151 /*
152  *
153  * If the interrupt link device is already configured,
154  * stores polarity and sensitivity in the structure pointed to by
155  * intr_flagp, and irqno in the value pointed to by pci_irqp.
156  *
157  * Returns:
158  *	ACPI_PSM_SUCCESS if the interrupt link device is already configured.
159  *	ACPI_PSM_PARTIAL if configuration is needed.
160  * 	ACPI_PSM_FAILURE in case of error.
161  *
162  * When two devices share the same interrupt link device, and the
163  * link device is already configured (i.e. found in the irq cache)
164  * we need to use the already configured irq instead of reconfiguring
165  * the link device.
166  */
167 static int
168 acpi_eval_lnk(dev_info_t *dip, char *lnkname, int *pci_irqp,
169 iflag_t *intr_flagp, acpi_psm_lnk_t *acpipsmlnkp)
170 {
171 	ACPI_HANDLE	tmpobj;
172 	ACPI_HANDLE	lnkobj;
173 	int status;
174 
175 	/*
176 	 * Convert the passed-in link device name to a handle
177 	 */
178 	if (AcpiGetHandle(NULL, lnkname, &lnkobj) != AE_OK) {
179 		return (ACPI_PSM_FAILURE);
180 	}
181 
182 	/*
183 	 * Assume that the link device is invalid if no _CRS method
184 	 * exists, since _CRS method is a required method
185 	 */
186 	if (AcpiGetHandle(lnkobj, "_CRS", &tmpobj) != AE_OK) {
187 		return (ACPI_PSM_FAILURE);
188 	}
189 
190 	ASSERT(acpipsmlnkp != NULL);
191 	acpipsmlnkp->lnkobj = lnkobj;
192 	if ((acpi_get_irq_lnk_cache_ent(lnkobj, pci_irqp, intr_flagp)) ==
193 	    ACPI_PSM_SUCCESS) {
194 		PSM_VERBOSE_IRQ((CE_CONT, "!psm: link object found from cache "
195 		    " for device %s, instance #%d, irq no %d\n",
196 		    ddi_get_name(dip), ddi_get_instance(dip), *pci_irqp));
197 		return (ACPI_PSM_SUCCESS);
198 	} else {
199 		if (acpica_eval_int(lnkobj, "_STA", &status) == AE_OK) {
200 			acpipsmlnkp->device_status = (uchar_t)status;
201 		}
202 
203 		return (ACPI_PSM_PARTIAL);
204 	}
205 }
206 
207 int
208 acpi_psm_init(char *module_name, int verbose_flags)
209 {
210 	psm_module_name = module_name;
211 
212 	psm_verbose = verbose_flags;
213 
214 	if (AcpiGetHandle(NULL, "\\_SB", &acpi_sbobj) != AE_OK) {
215 		cmn_err(CE_WARN, "!psm: get _SB failed");
216 		return (ACPI_PSM_FAILURE);
217 	}
218 
219 	mutex_init(&acpi_irq_cache_mutex, NULL, MUTEX_DEFAULT, NULL);
220 
221 	return (ACPI_PSM_SUCCESS);
222 
223 }
224 
225 /*
226  * Return bus/dev/fn for PCI dip (note: not the parent "pci" node).
227  */
228 
229 int
230 get_bdf(dev_info_t *dip, int *bus, int *device, int *func)
231 {
232 	pci_regspec_t *pci_rp;
233 	int len;
234 
235 	if (ddi_prop_lookup_int_array(DDI_DEV_T_ANY, dip, DDI_PROP_DONTPASS,
236 	    "reg", (int **)&pci_rp, (uint_t *)&len) != DDI_SUCCESS)
237 		return (-1);
238 
239 	if (len < (sizeof (pci_regspec_t) / sizeof (int))) {
240 		ddi_prop_free(pci_rp);
241 		return (-1);
242 	}
243 	if (bus != NULL)
244 		*bus = (int)PCI_REG_BUS_G(pci_rp->pci_phys_hi);
245 	if (device != NULL)
246 		*device = (int)PCI_REG_DEV_G(pci_rp->pci_phys_hi);
247 	if (func != NULL)
248 		*func = (int)PCI_REG_FUNC_G(pci_rp->pci_phys_hi);
249 	ddi_prop_free(pci_rp);
250 	return (0);
251 }
252 
253 
254 /*
255  * Build the reserved ISA irq list, and store it in the table pointed to by
256  * reserved_irqs_table. The caller is responsible for allocating this table
257  * with a minimum of MAX_ISA_IRQ + 1 entries.
258  *
259  * The routine looks in the device tree at the subtree rooted at /isa
260  * for each of the devices under that node, if an interrupts property
261  * is present, its values are used to "reserve" irqs so that later ACPI
262  * configuration won't choose those irqs.
263  *
264  * In addition, if acpi_irq_check_elcr is set, will use ELCR register
265  * to identify reserved IRQs.
266  */
267 void
268 build_reserved_irqlist(uchar_t *reserved_irqs_table)
269 {
270 	dev_info_t *isanode = ddi_find_devinfo("isa", -1, 0);
271 	dev_info_t *isa_child = 0;
272 	int i;
273 	uint_t	elcrval;
274 
275 	/* Initialize the reserved ISA IRQs: */
276 	for (i = 0; i <= MAX_ISA_IRQ; i++)
277 		reserved_irqs_table[i] = 0;
278 
279 	if (acpi_irq_check_elcr) {
280 
281 		elcrval = (inb(ELCR_PORT2) << 8) | (inb(ELCR_PORT1));
282 		if (ELCR_EDGE(elcrval, 0) && ELCR_EDGE(elcrval, 1) &&
283 		    ELCR_EDGE(elcrval, 2) && ELCR_EDGE(elcrval, 8) &&
284 		    ELCR_EDGE(elcrval, 13)) {
285 			/* valid ELCR */
286 			for (i = 0; i <= MAX_ISA_IRQ; i++)
287 				if (!ELCR_LEVEL(elcrval, i))
288 					reserved_irqs_table[i] = 1;
289 		}
290 	}
291 
292 	/* always check the isa devinfo nodes */
293 
294 	if (isanode != 0) { /* Found ISA */
295 		uint_t intcnt;		/* Interrupt count */
296 		int *intrs;		/* Interrupt values */
297 
298 		/* Load first child: */
299 		isa_child = ddi_get_child(isanode);
300 		while (isa_child != 0) { /* Iterate over /isa children */
301 			/* if child has any interrupts, save them */
302 			if (ddi_prop_lookup_int_array(DDI_DEV_T_ANY, isa_child,
303 			    DDI_PROP_DONTPASS, "interrupts", &intrs, &intcnt)
304 			    == DDI_PROP_SUCCESS) {
305 				/*
306 				 * iterate over child interrupt list, adding
307 				 * them to the reserved irq list
308 				 */
309 				while (intcnt-- > 0) {
310 					/*
311 					 * Each value MUST be <= MAX_ISA_IRQ
312 					 */
313 
314 					if ((intrs[intcnt] > MAX_ISA_IRQ) ||
315 					    (intrs[intcnt] < 0))
316 						continue;
317 
318 					reserved_irqs_table[intrs[intcnt]] = 1;
319 				}
320 				ddi_prop_free(intrs);
321 			}
322 			isa_child = ddi_get_next_sibling(isa_child);
323 		}
324 		/* The isa node was held by ddi_find_devinfo, so release it */
325 		ndi_rele_devi(isanode);
326 	}
327 
328 	/*
329 	 * Reserve IRQ14 & IRQ15 for IDE.  It shouldn't be hard-coded
330 	 * here but there's no other way to find the irqs for
331 	 * legacy-mode ata (since it's hard-coded in pci-ide also).
332 	 */
333 	reserved_irqs_table[14] = 1;
334 	reserved_irqs_table[15] = 1;
335 }
336 
337 /*
338  * Examine devinfo node to determine if it is a PCI-PCI bridge
339  *
340  * Returns:
341  *	0 if not a bridge or error
342  *	1 if a bridge
343  */
344 static int
345 psm_is_pci_bridge(dev_info_t *dip)
346 {
347 	ddi_acc_handle_t cfg_handle;
348 	int rv = 0;
349 
350 	if (pci_config_setup(dip, &cfg_handle) == DDI_SUCCESS) {
351 		rv = ((pci_config_get8(cfg_handle, PCI_CONF_BASCLASS) ==
352 		    PCI_CLASS_BRIDGE) && (pci_config_get8(cfg_handle,
353 		    PCI_CONF_SUBCLASS) == PCI_BRIDGE_PCI));
354 		pci_config_teardown(&cfg_handle);
355 	}
356 
357 	return (rv);
358 }
359 
360 /*
361  * Examines ACPI node for presence of _PRT object
362  * Check _STA to make sure node is present and/or enabled
363  *
364  * Returns:
365  *	0 if no _PRT or error
366  *	1 if _PRT is present
367  */
368 static int
369 psm_node_has_prt(ACPI_HANDLE *ah)
370 {
371 	ACPI_HANDLE rh;
372 	int sta;
373 
374 	/*
375 	 * Return 0 for "no _PRT" if device does not exist
376 	 * According to ACPI Spec,
377 	 * 1) setting either bit 0 or bit 3 means that device exists.
378 	 * 2) Absence of _STA method means all status bits set.
379 	 */
380 	if (ACPI_SUCCESS(acpica_eval_int(ah, "_STA", &sta)) &&
381 	    !(sta & (ACPI_STA_DEVICE_PRESENT | ACPI_STA_DEVICE_FUNCTIONING)))
382 		return (0);
383 
384 	return (AcpiGetHandle(ah, "_PRT", &rh) == AE_OK);
385 }
386 
387 
388 /*
389  * Look first for an ACPI PCI bus node matching busid, then for a _PRT on the
390  * parent node; then drop into the bridge-chasing code (which will also
391  * look for _PRTs on the way up the tree of bridges)
392  *
393  * Stores polarity and sensitivity in the structure pointed to by
394  * intr_flagp, and irqno in the value pointed to by pci_irqp.  *
395  * Returns:
396  *  	ACPI_PSM_SUCCESS on success.
397  *	ACPI_PSM_PARTIAL to indicate need to configure the interrupt
398  *	link device.
399  * 	ACPI_PSM_FAILURE  if an error prevented the system from
400  *	obtaining irq information for dip.
401  */
402 int
403 acpi_translate_pci_irq(dev_info_t *dip, int ipin, int *pci_irqp,
404     iflag_t *intr_flagp, acpi_psm_lnk_t *acpipsmlnkp)
405 {
406 	ACPI_HANDLE pciobj;
407 	int status = AE_ERROR;
408 	dev_info_t *curdip, *parentdip;
409 	int curpin, curbus, curdev;
410 
411 
412 	curpin = ipin;
413 	curdip = dip;
414 	while (curdip != ddi_root_node()) {
415 		parentdip = ddi_get_parent(curdip);
416 		ASSERT(parentdip != NULL);
417 
418 		if (get_bdf(curdip, &curbus, &curdev, NULL) != 0)
419 			break;
420 
421 		status = acpica_get_handle(parentdip, &pciobj);
422 		if ((status == AE_OK) && psm_node_has_prt(pciobj)) {
423 			return (acpi_get_gsiv(curdip, pciobj, curdev, curpin,
424 			    pci_irqp, intr_flagp, acpipsmlnkp));
425 		}
426 
427 		/* if we got here, we need to traverse a bridge upwards */
428 		if (!psm_is_pci_bridge(parentdip))
429 			break;
430 
431 		/*
432 		 * This is the rotating scheme that Compaq is using
433 		 * and documented in the PCI-PCI spec.  Also, if the
434 		 * PCI-PCI bridge is behind another PCI-PCI bridge,
435 		 * then it needs to keep ascending until an interrupt
436 		 * entry is found or the top is reached
437 		 */
438 		curpin = (curdev + curpin) % PCI_INTD;
439 		curdip = parentdip;
440 	}
441 
442 	/*
443 	 * We should never, ever get here; didn't find a _PRT
444 	 */
445 	return (ACPI_PSM_FAILURE);
446 }
447 
448 /*
449  * Sets the irq resource of the lnk object to the requested irq value.
450  *
451  * Returns ACPI_PSM_SUCCESS on success, ACPI_PSM_FAILURE upon failure.
452  */
453 int
454 acpi_set_irq_resource(acpi_psm_lnk_t *acpipsmlnkp, int irq)
455 {
456 	ACPI_BUFFER	rsb;
457 	ACPI_RESOURCE	*resp;
458 	ACPI_RESOURCE	*srsp;
459 	ACPI_HANDLE lnkobj;
460 	int srs_len, status;
461 
462 	ASSERT(acpipsmlnkp != NULL);
463 
464 	lnkobj = acpipsmlnkp->lnkobj;
465 
466 	/*
467 	 * Fetch the possible resources for the link
468 	 */
469 
470 	rsb.Pointer = NULL;
471 	rsb.Length = ACPI_ALLOCATE_BUFFER;
472 	status = AcpiGetPossibleResources(lnkobj, &rsb);
473 	if (status != AE_OK) {
474 		cmn_err(CE_WARN, "!psm: set_irq: _PRS failed");
475 		return (ACPI_PSM_FAILURE);
476 	}
477 
478 	/*
479 	 * Find an IRQ resource descriptor to use as template
480 	 */
481 	srsp = NULL;
482 	for (resp = rsb.Pointer; resp->Type != ACPI_RESOURCE_TYPE_END_TAG;
483 	    resp = ACPI_NEXT_RESOURCE(resp)) {
484 		if ((resp->Type == ACPI_RESOURCE_TYPE_IRQ) ||
485 		    (resp->Type == ACPI_RESOURCE_TYPE_EXTENDED_IRQ)) {
486 			ACPI_RESOURCE *endtag;
487 			/*
488 			 * Allocate enough room for this resource entry
489 			 * and one end tag following it
490 			 */
491 			srs_len = resp->Length + sizeof (*endtag);
492 			srsp = kmem_zalloc(srs_len, KM_SLEEP);
493 			bcopy(resp, srsp, resp->Length);
494 			endtag = ACPI_NEXT_RESOURCE(srsp);
495 			endtag->Type = ACPI_RESOURCE_TYPE_END_TAG;
496 			endtag->Length = 0;
497 			break;	/* drop out of the loop */
498 		}
499 	}
500 
501 	/*
502 	 * We're done with the PRS values, toss 'em lest we forget
503 	 */
504 	AcpiOsFree(rsb.Pointer);
505 
506 	if (srsp == NULL)
507 		return (ACPI_PSM_FAILURE);
508 
509 	/*
510 	 * The Interrupts[] array is always at least one entry
511 	 * long; see the definition of ACPI_RESOURCE.
512 	 */
513 	switch (srsp->Type) {
514 	case ACPI_RESOURCE_TYPE_IRQ:
515 		srsp->Data.Irq.InterruptCount = 1;
516 		srsp->Data.Irq.Interrupts[0] = (uint8_t)irq;
517 		break;
518 	case ACPI_RESOURCE_TYPE_EXTENDED_IRQ:
519 		srsp->Data.ExtendedIrq.InterruptCount = 1;
520 		srsp->Data.ExtendedIrq.Interrupts[0] = irq;
521 		break;
522 	}
523 
524 	rsb.Pointer = srsp;
525 	rsb.Length = srs_len;
526 	status = AcpiSetCurrentResources(lnkobj, &rsb);
527 	kmem_free(srsp, srs_len);
528 	if (status != AE_OK) {
529 		cmn_err(CE_WARN, "!psm: set_irq: _SRS failed");
530 		return (ACPI_PSM_FAILURE);
531 	}
532 
533 	if (acpica_eval_int(lnkobj, "_STA", &status) == AE_OK) {
534 		acpipsmlnkp->device_status = (uchar_t)status;
535 		return (ACPI_PSM_SUCCESS);
536 	} else
537 		return (ACPI_PSM_FAILURE);
538 }
539 
540 
541 /*
542  *
543  */
544 static int
545 psm_acpi_edgelevel(UINT32 el)
546 {
547 	switch (el) {
548 	case ACPI_EDGE_SENSITIVE:
549 		return (INTR_EL_EDGE);
550 	case ACPI_LEVEL_SENSITIVE:
551 		return (INTR_EL_LEVEL);
552 	default:
553 		/* el is a single bit; should never reach here */
554 		return (INTR_EL_CONFORM);
555 	}
556 }
557 
558 
559 /*
560  *
561  */
562 static int
563 psm_acpi_po(UINT32 po)
564 {
565 	switch (po) {
566 	case ACPI_ACTIVE_HIGH:
567 		return (INTR_PO_ACTIVE_HIGH);
568 	case ACPI_ACTIVE_LOW:
569 		return (INTR_PO_ACTIVE_LOW);
570 	default:
571 		/* po is a single bit; should never reach here */
572 		return (INTR_PO_CONFORM);
573 	}
574 }
575 
576 
577 /*
578  * Retrieves the current irq setting for the interrrupt link device.
579  *
580  * Stores polarity and sensitivity in the structure pointed to by
581  * intr_flagp, and irqno in the value pointed to by pci_irqp.
582  *
583  * Returns ACPI_PSM_SUCCESS on success, ACPI_PSM_FAILURE upon failure.
584  */
585 int
586 acpi_get_current_irq_resource(acpi_psm_lnk_t *acpipsmlnkp, int *pci_irqp,
587     iflag_t *intr_flagp)
588 {
589 	ACPI_HANDLE lnkobj;
590 	ACPI_BUFFER rb;
591 	ACPI_RESOURCE *rp;
592 	int irq;
593 	int status = ACPI_PSM_FAILURE;
594 
595 	ASSERT(acpipsmlnkp != NULL);
596 	lnkobj = acpipsmlnkp->lnkobj;
597 
598 	if (!(acpipsmlnkp->device_status & STA_PRESENT) ||
599 	    !(acpipsmlnkp->device_status & STA_ENABLE)) {
600 		PSM_VERBOSE_IRQ((CE_WARN, "!psm: crs device either not "
601 		    "present or disabled, status 0x%x",
602 		    acpipsmlnkp->device_status));
603 		return (ACPI_PSM_FAILURE);
604 	}
605 
606 	rb.Pointer = NULL;
607 	rb.Length = ACPI_ALLOCATE_BUFFER;
608 	if (AcpiGetCurrentResources(lnkobj, &rb) != AE_OK) {
609 		PSM_VERBOSE_IRQ((CE_WARN, "!psm: no crs object found or"
610 		" evaluation failed"));
611 		return (ACPI_PSM_FAILURE);
612 	}
613 
614 	irq = -1;
615 	for (rp = rb.Pointer; rp->Type != ACPI_RESOURCE_TYPE_END_TAG;
616 	    rp = ACPI_NEXT_RESOURCE(rp)) {
617 		if (rp->Type == ACPI_RESOURCE_TYPE_IRQ) {
618 			if (irq > 0) {
619 				PSM_VERBOSE_IRQ((CE_WARN, "!psm: multiple IRQ"
620 				" from _CRS "));
621 				status = ACPI_PSM_FAILURE;
622 				break;
623 			}
624 
625 			if (rp->Data.Irq.InterruptCount != 1) {
626 				PSM_VERBOSE_IRQ((CE_WARN, "!psm: <>1 interrupt"
627 				" from _CRS "));
628 				status = ACPI_PSM_FAILURE;
629 				break;
630 			}
631 
632 			intr_flagp->intr_el = psm_acpi_edgelevel(
633 			    rp->Data.Irq.Triggering);
634 			intr_flagp->intr_po = psm_acpi_po(
635 			    rp->Data.Irq.Polarity);
636 			irq = rp->Data.Irq.Interrupts[0];
637 			status = ACPI_PSM_SUCCESS;
638 		} else if (rp->Type == ACPI_RESOURCE_TYPE_EXTENDED_IRQ) {
639 			if (irq > 0) {
640 				PSM_VERBOSE_IRQ((CE_WARN, "!psm: multiple IRQ"
641 				" from _CRS "));
642 				status = ACPI_PSM_FAILURE;
643 				break;
644 			}
645 
646 			if (rp->Data.ExtendedIrq.InterruptCount != 1) {
647 				PSM_VERBOSE_IRQ((CE_WARN, "!psm: <>1 interrupt"
648 				" from _CRS "));
649 				status = ACPI_PSM_FAILURE;
650 				break;
651 			}
652 
653 			intr_flagp->intr_el = psm_acpi_edgelevel(
654 			    rp->Data.ExtendedIrq.Triggering);
655 			intr_flagp->intr_po = psm_acpi_po(
656 			    rp->Data.ExtendedIrq.Polarity);
657 			irq = rp->Data.ExtendedIrq.Interrupts[0];
658 			status = ACPI_PSM_SUCCESS;
659 		}
660 	}
661 
662 	AcpiOsFree(rb.Pointer);
663 	if (status == ACPI_PSM_SUCCESS) {
664 		*pci_irqp =  irq;
665 	}
666 
667 	return (status);
668 }
669 
670 /*
671  * Searches for the given IRQ in the irqlist passed in.
672  *
673  * If multiple matches exist, this returns true on the first match.
674  * Returns the interrupt flags, if a match was found, in `intr_flagp' if
675  * it's passed in non-NULL
676  */
677 int
678 acpi_irqlist_find_irq(acpi_irqlist_t *irqlistp, int irq, iflag_t *intr_flagp)
679 {
680 	int found = 0;
681 	int i;
682 
683 	while (irqlistp != NULL && !found) {
684 		for (i = 0; i < irqlistp->num_irqs; i++) {
685 			if (irqlistp->irqs[i] == irq) {
686 				if (intr_flagp)
687 					*intr_flagp = irqlistp->intr_flags;
688 				found = 1;
689 				break;	/* out of for() */
690 			}
691 		}
692 	}
693 
694 	return (found ? ACPI_PSM_SUCCESS : ACPI_PSM_FAILURE);
695 }
696 
697 /*
698  * Frees the irqlist allocated by acpi_get_possible_irq_resource.
699  * It takes a count of number of entries in the list.
700  */
701 void
702 acpi_free_irqlist(acpi_irqlist_t *irqlistp)
703 {
704 	acpi_irqlist_t *freednode;
705 
706 	while (irqlistp != NULL) {
707 		/* Free the irq list */
708 		kmem_free(irqlistp->irqs, irqlistp->num_irqs *
709 		    sizeof (int32_t));
710 
711 		freednode = irqlistp;
712 		irqlistp = irqlistp->next;
713 		kmem_free(freednode, sizeof (acpi_irqlist_t));
714 	}
715 }
716 
717 /*
718  * Creates a new entry in the given irqlist with the information passed in.
719  */
720 static void
721 acpi_add_irqlist_entry(acpi_irqlist_t **irqlistp, uint32_t *irqlist,
722     int irqlist_len, iflag_t *intr_flagp)
723 {
724 	acpi_irqlist_t *newent;
725 
726 	ASSERT(irqlist != NULL);
727 	ASSERT(intr_flagp != NULL);
728 
729 	newent = kmem_alloc(sizeof (acpi_irqlist_t), KM_SLEEP);
730 	newent->intr_flags = *intr_flagp;
731 	newent->irqs = irqlist;
732 	newent->num_irqs = irqlist_len;
733 	newent->next = *irqlistp;
734 
735 	*irqlistp = newent;
736 }
737 
738 
739 /*
740  * Retrieves a list of possible interrupt settings for the interrupt link
741  * device.
742  *
743  * Stores polarity and sensitivity in the structure pointed to by intr_flagp.
744  * Updates value pointed to by irqlistp with the address of a table it
745  * allocates. where interrupt numbers are stored. Stores the number of entries
746  * in this table in the value pointed to by num_entriesp;
747  *
748  * Each element in this table is of type int32_t. The table should be later
749  * freed by caller via acpi_free_irq_list().
750  *
751  * Returns ACPI_PSM_SUCCESS on success and ACPI_PSM_FAILURE upon failure
752  */
753 int
754 acpi_get_possible_irq_resources(acpi_psm_lnk_t *acpipsmlnkp,
755     acpi_irqlist_t **irqlistp)
756 {
757 	ACPI_HANDLE lnkobj;
758 	ACPI_BUFFER rsb;
759 	ACPI_RESOURCE *resp;
760 	int status;
761 
762 	int i, el, po, irqlist_len;
763 	uint32_t *irqlist;
764 	void *tmplist;
765 	iflag_t intr_flags;
766 
767 	ASSERT(acpipsmlnkp != NULL);
768 	lnkobj = acpipsmlnkp->lnkobj;
769 
770 	rsb.Pointer = NULL;
771 	rsb.Length = ACPI_ALLOCATE_BUFFER;
772 	status = AcpiGetPossibleResources(lnkobj, &rsb);
773 	if (status != AE_OK) {
774 		cmn_err(CE_WARN, "!psm: get_irq: _PRS failed");
775 		return (ACPI_PSM_FAILURE);
776 	}
777 
778 	/*
779 	 * Scan the resources looking for an interrupt resource
780 	 */
781 	*irqlistp = 0;
782 	for (resp = rsb.Pointer; resp->Type != ACPI_RESOURCE_TYPE_END_TAG;
783 	    resp = ACPI_NEXT_RESOURCE(resp)) {
784 		switch (resp->Type) {
785 		case ACPI_RESOURCE_TYPE_IRQ:
786 			irqlist_len = resp->Data.Irq.InterruptCount;
787 			tmplist = resp->Data.Irq.Interrupts;
788 			el = resp->Data.Irq.Triggering;
789 			po = resp->Data.Irq.Polarity;
790 			break;
791 		case ACPI_RESOURCE_TYPE_EXTENDED_IRQ:
792 			irqlist_len = resp->Data.ExtendedIrq.InterruptCount;
793 			tmplist = resp->Data.ExtendedIrq.Interrupts;
794 			el = resp->Data.ExtendedIrq.Triggering;
795 			po = resp->Data.ExtendedIrq.Polarity;
796 			break;
797 		default:
798 			continue;
799 		}
800 
801 		if (resp->Type != ACPI_RESOURCE_TYPE_IRQ &&
802 		    resp->Type != ACPI_RESOURCE_TYPE_EXTENDED_IRQ) {
803 			cmn_err(CE_WARN, "!psm: get_irq: no IRQ resource");
804 			return (ACPI_PSM_FAILURE);
805 		}
806 
807 		/* NEEDSWORK: move this into add_irqlist_entry someday */
808 		irqlist = kmem_zalloc(irqlist_len * sizeof (*irqlist),
809 		    KM_SLEEP);
810 		for (i = 0; i < irqlist_len; i++)
811 			if (resp->Type == ACPI_RESOURCE_TYPE_IRQ)
812 				irqlist[i] = ((uint8_t *)tmplist)[i];
813 			else
814 				irqlist[i] = ((uint32_t *)tmplist)[i];
815 		intr_flags.intr_el = psm_acpi_edgelevel(el);
816 		intr_flags.intr_po = psm_acpi_po(po);
817 		acpi_add_irqlist_entry(irqlistp, irqlist, irqlist_len,
818 		    &intr_flags);
819 	}
820 
821 	AcpiOsFree(rsb.Pointer);
822 	return (irqlistp == NULL ? ACPI_PSM_FAILURE : ACPI_PSM_SUCCESS);
823 }
824 
825 /*
826  * Adds a new cache entry to the irq cache which maps an irq and
827  * its attributes to PCI bus/dev/ipin and optionally to its associated ACPI
828  * interrupt link device object.
829  */
830 void
831 acpi_new_irq_cache_ent(int bus, int dev, int ipin, int pci_irq,
832     iflag_t *intr_flagp, acpi_psm_lnk_t *acpipsmlnkp)
833 {
834 	int newsize;
835 	irq_cache_t *new_arr, *ep;
836 
837 	mutex_enter(&acpi_irq_cache_mutex);
838 	if (irq_cache_valid >= irq_cache_len) {
839 		/* initially, or re-, allocate array */
840 
841 		newsize = (irq_cache_len ?
842 		    irq_cache_len * 2 : IRQ_CACHE_INITLEN);
843 		new_arr = kmem_zalloc(newsize * sizeof (irq_cache_t), KM_SLEEP);
844 		if (irq_cache_len != 0) {
845 			/* realloc: copy data, free old */
846 			bcopy(irq_cache_table, new_arr,
847 			    irq_cache_len * sizeof (irq_cache_t));
848 			kmem_free(irq_cache_table,
849 			    irq_cache_len * sizeof (irq_cache_t));
850 		}
851 		irq_cache_len = newsize;
852 		irq_cache_table = new_arr;
853 	}
854 	ep = &irq_cache_table[irq_cache_valid++];
855 	ep->bus = (uchar_t)bus;
856 	ep->dev = (uchar_t)dev;
857 	ep->ipin = (uchar_t)ipin;
858 	ep->flags = *intr_flagp;
859 	ep->irq = (uchar_t)pci_irq;
860 	ASSERT(acpipsmlnkp != NULL);
861 	ep->lnkobj = acpipsmlnkp->lnkobj;
862 	mutex_exit(&acpi_irq_cache_mutex);
863 }
864 
865 
866 /*
867  * Searches the irq caches for the given bus/dev/ipin.
868  *
869  * If info is found, stores polarity and sensitivity in the structure
870  * pointed to by intr_flagp, and irqno in the value pointed to by pci_irqp,
871  * and returns ACPI_PSM_SUCCESS.
872  * Otherwise, ACPI_PSM_FAILURE is returned.
873  */
874 int
875 acpi_get_irq_cache_ent(uchar_t bus, uchar_t dev, int ipin,
876     int *pci_irqp, iflag_t *intr_flagp)
877 {
878 
879 	irq_cache_t *irqcachep;
880 	int i;
881 	int ret = ACPI_PSM_FAILURE;
882 
883 	mutex_enter(&acpi_irq_cache_mutex);
884 	for (irqcachep = irq_cache_table, i = 0; i < irq_cache_valid;
885 	    irqcachep++, i++)
886 		if ((irqcachep->bus == bus) &&
887 		    (irqcachep->dev == dev) &&
888 		    (irqcachep->ipin == ipin)) {
889 			ASSERT(pci_irqp != NULL && intr_flagp != NULL);
890 			*pci_irqp = irqcachep->irq;
891 			*intr_flagp = irqcachep->flags;
892 			ret = ACPI_PSM_SUCCESS;
893 			break;
894 		}
895 
896 	mutex_exit(&acpi_irq_cache_mutex);
897 	return (ret);
898 }
899 
900 /*
901  * Searches the irq caches for the given interrupt lnk device object.
902  *
903  * If info is found, stores polarity and sensitivity in the structure
904  * pointed to by intr_flagp, and irqno in the value pointed to by pci_irqp,
905  * and returns ACPI_PSM_SUCCESS.
906  * Otherwise, ACPI_PSM_FAILURE is returned.
907  */
908 int
909 acpi_get_irq_lnk_cache_ent(ACPI_HANDLE lnkobj, int *pci_irqp,
910     iflag_t *intr_flagp)
911 {
912 
913 	irq_cache_t *irqcachep;
914 	int i;
915 	int ret = ACPI_PSM_FAILURE;
916 
917 	if (lnkobj == NULL)
918 		return (ACPI_PSM_FAILURE);
919 
920 	mutex_enter(&acpi_irq_cache_mutex);
921 	for (irqcachep = irq_cache_table, i = 0; i < irq_cache_valid;
922 	    irqcachep++, i++)
923 		if (irqcachep->lnkobj == lnkobj) {
924 			ASSERT(pci_irqp != NULL);
925 			*pci_irqp = irqcachep->irq;
926 			ASSERT(intr_flagp != NULL);
927 			*intr_flagp = irqcachep->flags;
928 			ret = ACPI_PSM_SUCCESS;
929 			break;
930 		}
931 	mutex_exit(&acpi_irq_cache_mutex);
932 	return (ret);
933 }
934 
935 /*
936  * Walk the irq_cache_table and re-configure the link device to
937  * the saved state.
938  */
939 void
940 acpi_restore_link_devices(void)
941 {
942 	irq_cache_t *irqcachep;
943 	acpi_psm_lnk_t psmlnk;
944 	int i, status;
945 
946 	/* XXX: may not need to hold this mutex */
947 	mutex_enter(&acpi_irq_cache_mutex);
948 	for (irqcachep = irq_cache_table, i = 0; i < irq_cache_valid;
949 	    irqcachep++, i++) {
950 		if (irqcachep->lnkobj != NULL) {
951 			/* only field used from psmlnk in set_irq is lnkobj */
952 			psmlnk.lnkobj = irqcachep->lnkobj;
953 			status = acpi_set_irq_resource(&psmlnk, irqcachep->irq);
954 			/* warn if set_irq failed; soldier on */
955 			if (status != ACPI_PSM_SUCCESS)
956 				cmn_err(CE_WARN, "Could not restore interrupt "
957 				    "link device for IRQ 0x%x: Devices using "
958 				    "this IRQ may no longer function properly."
959 				    "\n", irqcachep->irq);
960 		}
961 	}
962 	mutex_exit(&acpi_irq_cache_mutex);
963 }
964 
965 int
966 acpi_poweroff(void)
967 {
968 	extern int acpica_use_safe_delay;
969 	ACPI_STATUS status;
970 
971 	PSM_VERBOSE_POWEROFF(("acpi_poweroff: starting poweroff\n"));
972 
973 	acpica_use_safe_delay = 1;
974 
975 	status = AcpiEnterSleepStatePrep(5);
976 	if (status != AE_OK) {
977 		PSM_VERBOSE_POWEROFF(("acpi_poweroff: failed to prepare for "
978 		    "poweroff, status=0x%x\n", status));
979 		return (1);
980 	}
981 	ACPI_DISABLE_IRQS();
982 	status = AcpiEnterSleepState(5);
983 	ACPI_ENABLE_IRQS();
984 
985 	/* we should be off; if we get here it's an error */
986 	PSM_VERBOSE_POWEROFF(("acpi_poweroff: failed to actually power "
987 	    "off, status=0x%x\n", status));
988 	return (1);
989 }
990 
991 
992 /*
993  * psm_set_elcr() sets ELCR bit for specified vector
994  */
995 void
996 psm_set_elcr(int vecno, int val)
997 {
998 	int elcr_port = ELCR_PORT1 + (vecno >> 3);
999 	int elcr_bit = 1 << (vecno & 0x07);
1000 
1001 	ASSERT((vecno >= 0) && (vecno < 16));
1002 
1003 	if (val) {
1004 		/* set bit to force level-triggered mode */
1005 		outb(elcr_port, inb(elcr_port) | elcr_bit);
1006 	} else {
1007 		/* clear bit to force edge-triggered mode */
1008 		outb(elcr_port, inb(elcr_port) & ~elcr_bit);
1009 	}
1010 }
1011 
1012 /*
1013  * psm_get_elcr() returns status of ELCR bit for specific vector
1014  */
1015 int
1016 psm_get_elcr(int vecno)
1017 {
1018 	int elcr_port = ELCR_PORT1 + (vecno >> 3);
1019 	int elcr_bit = 1 << (vecno & 0x07);
1020 
1021 	ASSERT((vecno >= 0) && (vecno < 16));
1022 
1023 	return ((inb(elcr_port) & elcr_bit) ? 1 : 0);
1024 }
1025