xref: /titanic_44/usr/src/uts/common/pcmcia/nexus/pcmcia.c (revision 4745263a792e84bbd9e36b3ceb07d1275762cf9b)
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, Version 1.0 only
6  * (the "License").  You may not use this file except in compliance
7  * with the License.
8  *
9  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10  * or http://www.opensolaris.org/os/licensing.
11  * See the License for the specific language governing permissions
12  * and limitations under the License.
13  *
14  * When distributing Covered Code, include this CDDL HEADER in each
15  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16  * If applicable, add the following below this CDDL HEADER, with the
17  * fields enclosed by brackets "[]" replaced with your own identifying
18  * information: Portions Copyright [yyyy] [name of copyright owner]
19  *
20  * CDDL HEADER END
21  */
22 /*
23  * Copyright 2005 Sun Microsystems, Inc.  All rights reserved.
24  * Use is subject to license terms.
25  */
26 
27 #pragma ident	"%Z%%M%	%I%	%E% SMI"
28 
29 /*
30  * PCMCIA NEXUS
31  *	The PCMCIA module is a generalized interface for
32  *	implementing PCMCIA nexus drivers.  It preserves
33  *	the logical socket name space while allowing multiple
34  *	instances of the hardware to be properly represented
35  *	in the device tree.
36  *
37  *	The nexus also exports events to an event manager
38  *	driver if it has registered.
39  */
40 
41 #include <sys/types.h>
42 #include <sys/systm.h>
43 #include <sys/user.h>
44 #include <sys/buf.h>
45 #include <sys/file.h>
46 #include <sys/uio.h>
47 #include <sys/conf.h>
48 #include <sys/stat.h>
49 #include <sys/autoconf.h>
50 #include <sys/vtoc.h>
51 #include <sys/dkio.h>
52 #include <sys/ddi.h>
53 #include <sys/debug.h>
54 #include <sys/sunddi.h>
55 #include <sys/sunndi.h>
56 #include <sys/cred.h>
57 #include <sys/kstat.h>
58 #include <sys/kmem.h>
59 #include <sys/modctl.h>
60 #include <sys/kobj.h>
61 #include <sys/callb.h>
62 #include <sys/param.h>
63 #include <sys/thread.h>
64 #include <sys/proc.h>
65 
66 #include <sys/pctypes.h>
67 #include <sys/pcmcia.h>
68 #include <sys/sservice.h>
69 #include <pcmcia/sys/cs_types.h>
70 #include <pcmcia/sys/cis.h>
71 #include <pcmcia/sys/cis_handlers.h>
72 #include <pcmcia/sys/cs.h>
73 #include <pcmcia/sys/cs_priv.h>
74 
75 #ifdef sparc
76 #include <sys/ddi_subrdefs.h>
77 #endif
78 
79 #undef SocketServices
80 
81 /* some bus specific stuff */
82 
83 /* need PCI regspec size for worst case at present */
84 #include <sys/pci.h>
85 
86 typedef struct pcmcia_logical_socket {
87 	int			ls_socket; /* adapter's socket number */
88 	uint32_t		ls_flags;
89 	struct pcmcia_adapter	*ls_adapter;
90 	pcmcia_if_t		*ls_if;
91 	dev_info_t		*ls_sockdrv;
92 	dev_info_t		*ls_dip[PCMCIA_MAX_FUNCTIONS];
93 	dev_info_t		*ls_mfintr_dip;
94 	int			ls_functions;
95 	uint32_t		ls_cs_events;
96 	uint32_t		ls_intr_pri;
97 	uint32_t		ls_intr_vec;
98 	int			ls_intrrefs;
99 	struct intrspec		ls_intrspec; /* MFC intrspec */
100 	inthandler_t		*ls_inthandlers; /* for multifunction cards */
101 	ddi_iblock_cookie_t	ls_iblk;
102 	ddi_idevice_cookie_t	ls_idev;
103 	kmutex_t		ls_ilock;
104 	int			ls_error; /* error for CS return */
105 } pcmcia_logical_socket_t;
106 
107 /*
108  * entry points used by the true nexus
109  */
110 int pcmcia_detach(dev_info_t *, ddi_detach_cmd_t);
111 int pcmcia_ctlops(dev_info_t *, dev_info_t *, ddi_ctl_enum_t, void *, void *);
112 int pcmcia_prop_op(dev_t, dev_info_t *, dev_info_t *, ddi_prop_op_t,
113 			int, char *, caddr_t, int *);
114 void pcmcia_set_assigned(dev_info_t *, int, ra_return_t *);
115 int pcmcia_intr_ops(dev_info_t *dip, dev_info_t *rdip, ddi_intr_op_t intr_op,
116     ddi_intr_handle_impl_t *hdlp, void *result);
117 
118 /*
119  * prototypes used internally by the nexus and sometimes Card Services
120  */
121 int SocketServices(int function, ...);
122 
123 
124 void *CISParser(int function, ...);
125 extern void *(*cis_parser)(int, ...);
126 
127 struct regspec *pcmcia_cons_regspec(dev_info_t *, int, uchar_t *,
128 					ra_return_t *);
129 
130 static int (*pcmcia_card_services)(int, ...) = NULL;
131 
132 /*
133  * variables used in the logical/physical mappings
134  * that the nexus common code maintains.
135  */
136 struct pcmcia_adapter *pcmcia_adapters[PCMCIA_MAX_ADAPTERS];
137 int    pcmcia_num_adapters;
138 pcmcia_logical_socket_t *pcmcia_sockets[PCMCIA_MAX_SOCKETS];
139 int    pcmcia_num_sockets;
140 pcmcia_logical_window_t *pcmcia_windows[PCMCIA_MAX_WINDOWS];
141 int    pcmcia_num_windows;
142 struct power_entry pcmcia_power_table[PCMCIA_MAX_POWER];
143 int	pcmcia_num_power;
144 
145 struct pcmcia_mif *pcmcia_mif_handlers = NULL;
146 pcm_dev_node_t *pcmcia_devnodes = NULL;
147 
148 kmutex_t pcmcia_global_lock;
149 kcondvar_t pcmcia_condvar;
150 kmutex_t pcmcia_enum_lock;
151 
152 /*
153  * Mapping of the device "type" to names acceptable to
154  * the DDI
155  */
156 static char *pcmcia_dev_type[] = {
157 	"multifunction",
158 	"byte",
159 	"serial",
160 	"parallel",
161 	"block",
162 	"display",
163 	"network",
164 	"block",
165 	"byte"
166 };
167 
168 char *pcmcia_default_pm_mode = "parental-suspend-resume";
169 
170 /*
171  * generic names from the approved list:
172  *	disk tape pci sbus scsi token-ring isa keyboard display mouse
173  *	audio ethernet timer memory parallel serial rtc nvram scanner
174  *	floppy(controller) fddi isdn atm ide pccard video-in video-out
175  * in some cases there will need to be device class dependent names.
176  * network -> ethernet, token-ring, etc.
177  * this list is a first guess and is used when all else fails.
178  */
179 
180 char *pcmcia_generic_names[] = {
181 	"multifunction",
182 	"memory",
183 	"serial",
184 	"parallel",
185 	"disk",
186 	"video",		/* no spec for video-out yet */
187 	"network",
188 	"aims",
189 	"scsi",
190 	"security"
191 };
192 
193 #define	PCM_GENNAME_SIZE	(sizeof (pcmcia_generic_names) / \
194 					sizeof (char *))
195 #define	PCMCIA_MAP_IO	0x0
196 #define	PCMCIA_MAP_MEM	0x1
197 
198 /*
199  * The following should be 2^^n - 1
200  */
201 #define	PCMCIA_SOCKET_BITS	0x7f
202 
203 #ifdef PCMCIA_DEBUG
204 int pcmcia_debug = 0x0;
205 static void pcmcia_dump_minors(dev_info_t *);
206 #endif
207 
208 static f_tt *pcmcia_cs_event = NULL;
209 int pcmcia_timer_id;
210 dev_info_t	*pcmcia_dip;
211 /*
212  * XXX - See comments in cs.c
213  */
214 static f_tt *pcmcia_cis_parser = NULL;
215 
216 extern struct pc_socket_services pc_socket_services;
217 
218 /* some function declarations */
219 static int pcm_adapter_callback(dev_info_t *, int, int, int);
220 extern void pcmcia_init_adapter(anp_t *, dev_info_t *);
221 static void pcmcia_enum_thread(anp_t *);
222 extern void pcmcia_find_cards(anp_t *);
223 extern void pcmcia_merge_power(struct power_entry *);
224 extern void pcmcia_do_resume(int, pcmcia_logical_socket_t *);
225 extern void pcmcia_resume(int, pcmcia_logical_socket_t *);
226 extern void pcmcia_do_suspend(int, pcmcia_logical_socket_t *);
227 extern void pcm_event_manager(int, int, void *);
228 static void pcmcia_create_dev_info(int);
229 static int pcmcia_create_device(ss_make_device_node_t *);
230 static void pcmcia_init_devinfo(dev_info_t *, struct pcm_device_info *);
231 void pcmcia_fix_string(char *str);
232 dev_info_t *pcmcia_number_socket(dev_info_t *, int);
233 static int pcmcia_merge_conf(dev_info_t *);
234 static uint32_t pcmcia_mfc_intr(caddr_t, caddr_t);
235 void pcmcia_free_resources(dev_info_t *);
236 static void pcmcia_ppd_free(struct pcmcia_parent_private *ppd);
237 int pcmcia_get_intr(dev_info_t *, int);
238 int pcmcia_return_intr(dev_info_t *, int);
239 int pcmcia_ra_alloc(dev_info_t *, ndi_ra_request_t *, ra_return_t *, char *);
240 int pcmcia_ra_free(dev_info_t *, ra_return_t *, char *);
241 
242 extern int cs_init(void);
243 extern int cs_deinit(void);
244 extern void cisp_init(void);
245 extern void cis_deinit(void);
246 
247 /*
248  * non-DDI compliant functions are listed here
249  * some will be declared while others that have
250  * entries in .h files. All will be commented on.
251  *
252  * with declarations:
253  *	ddi_add_child
254  *	ddi_binding_name
255  *	ddi_bus_prop_op
256  *	ddi_ctlops
257  *	ddi_find_devinfo
258  *	ddi_get_name_addr
259  *	ddi_get_parent_data
260  *	ddi_hold_installed_driver
261  *	ddi_name_to_major
262  *	ddi_node_name
263  *	ddi_pathname
264  *	ddi_rele_driver
265  *	ddi_set_name_addr
266  *	ddi_set_parent_data
267  *	ddi_unorphan_devs
268  *	i_ddi_bind_node_to_driver
269  *	i_ddi_bind_node_to_driver
270  *	i_ddi_bus_map
271  *	i_ddi_map_fault
272  *	i_ddi_mem_alloc
273  *	i_ddi_mem_alloc
274  *	i_ddi_mem_free
275  *	i_ddi_mem_free
276  *	modload
277  *	modunload
278  */
279 
280 extern void ddi_unorphan_devs(major_t);
281 
282 /* Card&Socket Services entry points */
283 static int GetCookiesAndDip(sservice_t *);
284 static int SSGetAdapter(get_adapter_t *);
285 static int SSGetPage(get_page_t *);
286 static int SSGetSocket(get_socket_t *);
287 static int SSGetStatus(get_ss_status_t *);
288 static int SSGetWindow(get_window_t *);
289 static int SSInquireAdapter(inquire_adapter_t *);
290 static int SSInquireSocket(inquire_socket_t *);
291 static int SSInquireWindow(inquire_window_t *);
292 static int SSResetSocket(int, int);
293 static int SSSetPage(set_page_t *);
294 static int SSSetSocket(set_socket_t *);
295 static int SSSetWindow(set_window_t *);
296 static int SSSetIRQHandler(set_irq_handler_t *);
297 static int SSClearIRQHandler(clear_irq_handler_t *);
298 
299 static struct modldrv modlmisc = {
300 	&mod_miscops,		/* Type of module. This one is a driver */
301 	"PCMCIA Nexus Support %I%", /* Name of the module. */
302 };
303 
304 static struct modlinkage modlinkage = {
305 	MODREV_1, (void *)&modlmisc, NULL
306 };
307 
308 int
309 _init()
310 {
311 	int	ret;
312 
313 	cisp_init();
314 
315 	if (cs_init() != CS_SUCCESS) {
316 	    if (cs_deinit() != CS_SUCCESS)
317 		cmn_err(CE_CONT, "pcmcia: _init cs_deinit error\n");
318 	    return (-1);
319 	}
320 
321 	mutex_init(&pcmcia_global_lock, NULL, MUTEX_DEFAULT, NULL);
322 	cv_init(&pcmcia_condvar, NULL, CV_DRIVER, NULL);
323 	mutex_init(&pcmcia_enum_lock, NULL, MUTEX_DEFAULT, NULL);
324 
325 	if ((ret = mod_install(&modlinkage)) != 0) {
326 		mutex_destroy(&pcmcia_global_lock);
327 		cv_destroy(&pcmcia_condvar);
328 		mutex_destroy(&pcmcia_enum_lock);
329 	}
330 	return (ret);
331 }
332 
333 int
334 _fini()
335 {
336 	int	ret;
337 
338 	if ((ret = mod_remove(&modlinkage)) == 0) {
339 		mutex_destroy(&pcmcia_global_lock);
340 		cv_destroy(&pcmcia_condvar);
341 		mutex_destroy(&pcmcia_enum_lock);
342 		cis_deinit();
343 		if (cs_deinit() != CS_SUCCESS) {
344 			cmn_err(CE_CONT, "pcmcia: _fini cs_deinit error\n");
345 		}
346 	}
347 	return (ret);
348 }
349 
350 int
351 _info(struct modinfo *modinfop)
352 {
353 	return (mod_info(&modlinkage, modinfop));
354 }
355 
356 extern pri_t minclsyspri;
357 
358 /*
359  * pcmcia_attach()
360  *	the attach routine must make sure that everything needed is present
361  *	including real hardware.  The sequence of events is:
362  *		attempt to load all adapter drivers
363  *		attempt to load Card Services (which _depends_on pcmcia)
364  *		initialize logical sockets
365  *		report the nexus exists
366  */
367 
368 int
369 pcmcia_attach(dev_info_t *dip, anp_t *adapter)
370 {
371 	int count, done, i;
372 
373 #if defined(PCMCIA_DEBUG)
374 	if (pcmcia_debug) {
375 		cmn_err(CE_CONT, "pcmcia_attach: dip=0x%p adapter=0x%p\n",
376 		    (void *)dip, (void *)adapter);
377 	}
378 #endif
379 
380 	pcmcia_dip = dip;
381 
382 	mutex_enter(&pcmcia_enum_lock);
383 	mutex_enter(&pcmcia_global_lock);
384 	if (pcmcia_num_adapters == 0) {
385 		pcmcia_cis_parser = (f_tt *)CISParser;
386 		cis_parser = (void *(*)(int, ...)) CISParser;
387 		pcmcia_cs_event = (f_tt *)cs_event;
388 		cs_socket_services = SocketServices;
389 		/* tell CS we are up with basic init level */
390 		(void) cs_event(PCE_SS_INIT_STATE, PCE_SS_STATE_INIT, 0);
391 	}
392 
393 	(void) ddi_prop_update_string(DDI_DEV_T_NONE, dip,
394 	    PCM_DEVICETYPE, "pccard");
395 
396 	ddi_report_dev(dip);	/* directory/device naming */
397 
398 	/*
399 	 * now setup any power management stuff necessary.
400 	 * we do it here in order to ensure that all PC Card nexi
401 	 * implement it.
402 	 */
403 
404 	if (pm_create_components(dip, 1) != DDI_SUCCESS) {
405 		cmn_err(CE_WARN, "%s: not power managed\n",
406 			ddi_get_name_addr(dip));
407 	} else {
408 		pm_set_normal_power(dip, 0, 1);
409 	}
410 
411 	/*
412 	 * setup the info necessary for Card Services/SocketServices
413 	 * and notify CS when ready.
414 	 */
415 
416 	pcmcia_free_resources(dip);
417 	pcmcia_init_adapter(adapter, dip);
418 	/* exit mutex so CS can run for any cards found */
419 	mutex_exit(&pcmcia_global_lock);
420 
421 	/*
422 	 * schedule an asynchronous thread to enumerate the cards
423 	 * present in the adapter at boot time
424 	 */
425 	(void) thread_create(NULL, 0, pcmcia_enum_thread, adapter, 0, &p0,
426 	    TS_RUN, minclsyspri);
427 
428 	/*
429 	 * make sure the devices are identified before
430 	 * returning.  We do this by checking each socket to see if
431 	 * a card is present.  If there is one, and there isn't a dip,
432 	 * we can't be done.  We scan the list of sockets doing the
433 	 * check. if we aren't done, wait for a condition variable to
434 	 * wakeup.
435 	 * Because we can miss a wakeup and because things can
436 	 * take time, we do eventually give up and have a timeout.
437 	 */
438 
439 	for (count = 0, done = 0;
440 	    done == 0 && count < max(pcmcia_num_sockets, 16);
441 	    count++) {
442 		done = 1;
443 		/* block CS while checking so we don't miss anything */
444 		mutex_enter(&pcmcia_global_lock);
445 		for (i = 0; i < pcmcia_num_sockets; i++) {
446 			get_ss_status_t status;
447 			if (pcmcia_sockets[i] == NULL)
448 				continue;
449 			bzero(&status, sizeof (status));
450 			status.socket = i;
451 			if (SSGetStatus(&status) == SUCCESS) {
452 				if (status.CardState & SBM_CD &&
453 				    pcmcia_sockets[i]->ls_dip[0] == NULL) {
454 					done = 0;
455 				}
456 			}
457 		}
458 		/* only wait if we aren't done with this set */
459 		if (!done) {
460 			mutex_exit(&pcmcia_global_lock);
461 			delay(10); /* give up CPU for a time */
462 			mutex_enter(&pcmcia_global_lock);
463 		}
464 		mutex_exit(&pcmcia_global_lock);
465 	}
466 
467 	mutex_exit(&pcmcia_enum_lock);
468 	return (DDI_SUCCESS);
469 }
470 
471 /*
472  * pcmcia_detach
473  *	unload everything and then detach the nexus
474  */
475 /* ARGSUSED */
476 int
477 pcmcia_detach(dev_info_t *dip, ddi_detach_cmd_t cmd)
478 {
479 	switch (cmd) {
480 	case DDI_DETACH:
481 		pm_destroy_components(dip);
482 		return (DDI_SUCCESS);
483 
484 	/*
485 	 * resume from a checkpoint
486 	 * We don't do anything special here since the adapter
487 	 * driver will generate resume events that we intercept
488 	 * and convert to insert events.
489 	 */
490 	case DDI_SUSPEND:
491 	case DDI_PM_SUSPEND:
492 		return (DDI_SUCCESS);
493 
494 	default:
495 		return (DDI_FAILURE);
496 	}
497 }
498 
499 /*
500  * card_services_error()
501  *	used to make 2.4/2.5 drivers get an error when
502  *	they try to initialize.
503  */
504 static int
505 card_services_error()
506 {
507 	return (CS_BAD_VERSION);
508 }
509 static int (*cs_error_ptr)() = card_services_error;
510 
511 /*
512  * pcmcia_ctlops
513  *	handle the nexus control operations for the cases where
514  *	a PC Card driver gets called and we need to modify the
515  *	devinfo structure or otherwise do bus specific operations
516  */
517 int
518 pcmcia_ctlops(dev_info_t *dip, dev_info_t *rdip,
519 	ddi_ctl_enum_t ctlop, void *arg, void *result)
520 {
521 	int e;
522 	char name[64];
523 	struct pcmcia_parent_private *ppd;
524 	power_req_t *pm;
525 
526 #if defined(PCMCIA_DEBUG)
527 	if (pcmcia_debug) {
528 		cmn_err(CE_CONT, "pcmcia_ctlops(%p, %p, %d, %p, %p)\n",
529 			(void *)dip, (void *)rdip, ctlop, (void *)arg,
530 			(void *)result);
531 		if (rdip != NULL && ddi_get_name(rdip) != NULL)
532 			cmn_err(CE_CONT, "\t[%s]\n", ddi_get_name(rdip));
533 	}
534 #endif
535 
536 	switch (ctlop) {
537 	case DDI_CTLOPS_REPORTDEV:
538 		if (rdip == (dev_info_t *)0)
539 			return (DDI_FAILURE);
540 
541 		if (strcmp("pcs", ddi_node_name(rdip)) == 0)
542 			cmn_err(CE_CONT, "?PCCard socket %d at %s@%s\n",
543 				ddi_get_instance(rdip),
544 				ddi_driver_name(dip), ddi_get_name_addr(dip));
545 		else
546 			cmn_err(CE_CONT, "?%s%d at %s@%s in socket %d\n",
547 				ddi_driver_name(rdip),
548 				ddi_get_instance(rdip),
549 				ddi_driver_name(dip),
550 				ddi_get_name_addr(dip),
551 				CS_GET_SOCKET_NUMBER(
552 				    ddi_getprop(DDI_DEV_T_NONE, rdip,
553 				    DDI_PROP_DONTPASS,
554 				    PCM_DEV_SOCKET, -1)));
555 
556 		return (DDI_SUCCESS);
557 
558 	case DDI_CTLOPS_INITCHILD:
559 		/*
560 		 * we get control here before the child is called.
561 		 * we can change things if necessary.  This is where
562 		 * the CardServices hook gets planted.
563 		 */
564 #if defined(PCMCIA_DEBUG)
565 		if (pcmcia_debug) {
566 			cmn_err(CE_CONT, "pcmcia: init child: %s(%d) @%p\n",
567 				ddi_node_name(arg), ddi_get_instance(arg),
568 				(void *)arg);
569 			if (DEVI(arg)->devi_binding_name != NULL)
570 				cmn_err(CE_CONT, "\tbinding_name=%s\n",
571 					DEVI(arg)->devi_binding_name);
572 			if (DEVI(arg)->devi_node_name != NULL)
573 				cmn_err(CE_CONT, "\tnode_name=%s\n",
574 					DEVI(arg)->devi_node_name);
575 		}
576 #endif
577 
578 		ppd = (struct pcmcia_parent_private *)
579 			ddi_get_parent_data((dev_info_t *)arg);
580 		if (ppd == NULL)
581 			return (DDI_FAILURE);
582 
583 		if (strcmp("pcs", ddi_node_name((dev_info_t *)arg)) == 0) {
584 			if (ppd == NULL)
585 				return (DDI_FAILURE);
586 			(void) sprintf(name, "%x",
587 			    (int)ppd->ppd_reg[0].phys_hi);
588 			ddi_set_name_addr((dev_info_t *)arg, name);
589 			return (DDI_SUCCESS);
590 		}
591 
592 		/*
593 		 * We don't want driver.conf files that stay in
594 		 * pseudo device form.	It is acceptable to have
595 		 * .conf files add properties only.
596 		 */
597 		if (ndi_dev_is_persistent_node((dev_info_t *)arg) == 0) {
598 			(void) pcmcia_merge_conf((dev_info_t *)arg);
599 			cmn_err(CE_WARN, "%s%d: %s.conf invalid",
600 				ddi_get_name((dev_info_t *)arg),
601 				ddi_get_instance((dev_info_t *)arg),
602 				ddi_get_name((dev_info_t *)arg));
603 			return (DDI_FAILURE);
604 		}
605 
606 
607 #if defined(PCMCIA_DEBUG)
608 		if (pcmcia_debug && ppd != NULL) {
609 			cmn_err(CE_CONT, "\tnreg=%x, intr=%x, socket=%x,"
610 				" function=%x, active=%x, flags=%x\n",
611 				ppd->ppd_nreg, ppd->ppd_intr,
612 				ppd->ppd_socket, ppd->ppd_function,
613 				ppd->ppd_active, ppd->ppd_flags);
614 		}
615 #endif
616 
617 		/*
618 		 * make sure names are relative to socket number
619 		 */
620 		if (ppd->ppd_function > 0) {
621 			int sock;
622 			int func;
623 			sock = ppd->ppd_socket;
624 			func = ppd->ppd_function;
625 			(void) sprintf(name, "%x,%x", sock, func);
626 		} else {
627 			(void) sprintf(name, "%x", ppd->ppd_socket);
628 		}
629 		ddi_set_name_addr((dev_info_t *)arg, name);
630 
631 #if defined(PCMCIA_DEBUG)
632 		if (pcmcia_debug)
633 			cmn_err(CE_CONT, "pcmcia: system init done for %s [%s] "
634 				"nodeid: %x @%s\n",
635 				ddi_get_name(arg), ddi_get_name_addr(arg),
636 				DEVI(arg)->devi_nodeid, name);
637 		if (pcmcia_debug > 1)
638 			pcmcia_dump_minors((dev_info_t *)arg);
639 #endif
640 
641 		return (DDI_SUCCESS);
642 
643 	case DDI_CTLOPS_UNINITCHILD:
644 
645 #if defined(PCMCIA_DEBUG)
646 		if (pcmcia_debug) {
647 			cmn_err(CE_CONT, "pcmcia: uninit child: %s(%d) @%p\n",
648 				ddi_node_name(arg), ddi_get_instance(arg),
649 				(void *)arg);
650 			if (DEVI(arg)->devi_binding_name != NULL)
651 				cmn_err(CE_CONT, "\tbinding_name=%s\n",
652 					DEVI(arg)->devi_binding_name);
653 			if (DEVI(arg)->devi_node_name != NULL)
654 				cmn_err(CE_CONT, "\tnode_name=%s\n",
655 					DEVI(arg)->devi_node_name);
656 		}
657 #endif
658 
659 		ddi_set_name_addr((dev_info_t *)arg, NULL);
660 		ddi_remove_minor_node((dev_info_t *)arg, NULL);
661 		return (DDI_SUCCESS);
662 
663 	case DDI_CTLOPS_SLAVEONLY:
664 		/* PCMCIA devices can't ever be busmaster until CardBus */
665 		ppd = (struct pcmcia_parent_private *)
666 			ddi_get_parent_data(rdip);
667 		if (ppd != NULL && ppd->ppd_flags & PPD_CB_BUSMASTER)
668 			return (DDI_FAILURE); /* at most */
669 		return (DDI_SUCCESS);
670 
671 	case DDI_CTLOPS_SIDDEV:
672 		/* in general this is true. */
673 		return (DDI_SUCCESS);
674 
675 	case DDI_CTLOPS_NREGS:
676 		ppd = (struct pcmcia_parent_private *)
677 			ddi_get_parent_data(rdip);
678 		if (ppd != NULL)
679 			*((uint32_t *)result) = (ppd->ppd_nreg);
680 		else
681 			*((uint32_t *)result) = 0;
682 		return (DDI_SUCCESS);
683 
684 	case DDI_CTLOPS_REGSIZE:
685 		ppd = (struct pcmcia_parent_private *)
686 			ddi_get_parent_data(rdip);
687 		if (ppd != NULL && ppd->ppd_nreg > 0)
688 			*((off_t *)result) =  sizeof (struct pcm_regs);
689 		else
690 			*((off_t *)result) = 0;
691 		return (DDI_SUCCESS);
692 
693 	case DDI_CTLOPS_POWER:
694 		ppd = (struct pcmcia_parent_private *)
695 			ddi_get_parent_data(rdip);
696 
697 		if (ppd == NULL)
698 			return (DDI_FAILURE);
699 		/*
700 		 * if this is not present, don't bother (claim success)
701 		 * since it is already in the right state.  Don't
702 		 * do any resume either since the card insertion will
703 		 * happen independently.
704 		 */
705 		if (!ppd->ppd_active)
706 			return (DDI_SUCCESS);
707 		for (e = 0; e < pcmcia_num_adapters; e++)
708 			if (pcmcia_adapters[e] ==
709 			    pcmcia_sockets[ppd->ppd_socket]->ls_adapter)
710 				break;
711 		if (e == pcmcia_num_adapters)
712 			return (DDI_FAILURE);
713 		pm = (power_req_t *)arg;
714 #if defined(PCMCIA_DEBUG)
715 		if (pcmcia_debug) {
716 			cmn_err(CE_WARN, "power: %d: %p, %d, %d [%s]\n",
717 				pm->request_type,
718 				(void *)pm->req.set_power_req.who,
719 				pm->req.set_power_req.cmpt,
720 				pm->req.set_power_req.level,
721 				ddi_get_name_addr(rdip));
722 		}
723 #endif
724 		e = ppd->ppd_socket;
725 		switch (pm->request_type) {
726 		case PMR_SUSPEND:
727 			if (!(pcmcia_sockets[e]->ls_flags &
728 			    PCS_SUSPENDED)) {
729 				pcmcia_do_suspend(ppd->ppd_socket,
730 				    pcmcia_sockets[e]);
731 			}
732 			ppd->ppd_flags |= PPD_SUSPENDED;
733 			return (DDI_SUCCESS);
734 		case PMR_RESUME:
735 			/* for now, we just succeed since the rest is done */
736 			return (DDI_SUCCESS);
737 		case PMR_SET_POWER:
738 			/*
739 			 * not sure how to handle power control
740 			 * for now, we let the child handle it itself
741 			 */
742 			(void) pcmcia_power(pm->req.set_power_req.who,
743 				pm->req.set_power_req.cmpt,
744 				pm->req.set_power_req.level);
745 			break;
746 		default:
747 			break;
748 		}
749 		return (DDI_FAILURE);
750 		/* These CTLOPS will need to be implemented for new form */
751 		/* let CardServices know about this */
752 
753 	default:
754 		/* if we don't understand, pass up the tree */
755 		/* most things default to general ops */
756 		return (ddi_ctlops(dip, rdip, ctlop, arg, result));
757 	}
758 }
759 
760 struct pcmcia_props {
761 	char *name;
762 	int   len;
763 	int   prop;
764 } pcmcia_internal_props[] = {
765 	{ PCM_DEV_ACTIVE, 0, PCMCIA_PROP_ACTIVE },
766 	{ PCM_DEV_R2TYPE, 0, PCMCIA_PROP_R2TYPE },
767 	{ PCM_DEV_CARDBUS, 0, PCMCIA_PROP_CARDBUS },
768 	{ CS_PROP, sizeof (void *), PCMCIA_PROP_OLDCS },
769 	{ "reg", 0, PCMCIA_PROP_REG },
770 	{ "interrupts", sizeof (int), PCMCIA_PROP_INTR },
771 	{ "pm-hardware-state", 0, PCMCIA_PROP_DEFAULT_PM },
772 };
773 
774 /*
775  * pcmcia_prop_decode(name)
776  *	decode the name and determine if this is a property
777  *	we construct on the fly, one we have on the prop list
778  *	or one that requires calling the CIS code.
779  */
780 static int
781 pcmcia_prop_decode(char *name)
782 {
783 	int i;
784 	if (strncmp(name, "cistpl_", 7) == 0)
785 		return (PCMCIA_PROP_CIS);
786 
787 	for (i = 0; i < (sizeof (pcmcia_internal_props) /
788 	    sizeof (struct pcmcia_props)); i++) {
789 		if (strcmp(name, pcmcia_internal_props[i].name) == 0)
790 			return (i);
791 	}
792 
793 	return (PCMCIA_PROP_UNKNOWN);
794 }
795 
796 /*
797  * pcmcia_prop_op()
798  *	we don't have properties in PROM per se so look for them
799  *	only in the devinfo node.  Future may allow us to find
800  *	certain CIS tuples via this interface if a user asks for
801  *	a property of the form "cistpl-<tuplename>" but not yet.
802  *
803  *	The addition of 1275 properties adds to the necessity.
804  */
805 int
806 pcmcia_prop_op(dev_t dev, dev_info_t *dip, dev_info_t *ch_dip,
807     ddi_prop_op_t prop_op, int mod_flags,
808     char *name, caddr_t valuep, int *lengthp)
809 {
810 	int len, proplen, which, flags;
811 	caddr_t buff, propptr;
812 	struct pcmcia_parent_private *ppd;
813 
814 	len = *lengthp;
815 	ppd = (struct pcmcia_parent_private *)ddi_get_parent_data(ch_dip);
816 
817 	switch (which = pcmcia_prop_decode(name)) {
818 	default:
819 		if (ppd == NULL)
820 			return (DDI_PROP_NOT_FOUND);
821 
822 		/* note that proplen may get modified */
823 		proplen = pcmcia_internal_props[which].len;
824 		switch (pcmcia_internal_props[which].prop) {
825 		case PCMCIA_PROP_DEFAULT_PM:
826 			propptr = pcmcia_default_pm_mode;
827 			proplen = strlen(propptr) + 1;
828 			break;
829 		case PCMCIA_PROP_OLDCS:
830 			propptr = (caddr_t)&cs_error_ptr;
831 			break;
832 		case PCMCIA_PROP_REG:
833 			propptr = (caddr_t)ppd->ppd_reg;
834 			proplen = ppd->ppd_nreg * sizeof (struct pcm_regs);
835 			break;
836 		case PCMCIA_PROP_INTR:
837 			propptr = (caddr_t)&ppd->ppd_intr;
838 			break;
839 
840 		/* the next set are boolean values */
841 		case PCMCIA_PROP_ACTIVE:
842 			propptr = NULL;
843 			if (!ppd->ppd_active) {
844 				return (DDI_PROP_NOT_FOUND);
845 			}
846 			break;
847 		case PCMCIA_PROP_R2TYPE:
848 			propptr = NULL;
849 			if (ppd->ppd_flags & PPD_CARD_CARDBUS)
850 				return (DDI_PROP_NOT_FOUND);
851 			break;
852 		case PCMCIA_PROP_CARDBUS:
853 			propptr = NULL;
854 			if (!(ppd->ppd_flags * PPD_CARD_CARDBUS))
855 				return (DDI_PROP_NOT_FOUND);
856 			break;
857 		}
858 
859 		break;
860 
861 	case PCMCIA_PROP_CIS:
862 		/*
863 		 * once we have the lookup code in place
864 		 * it is sufficient to break out of the switch
865 		 * once proplen and propptr are set.
866 		 * The common prop_op code deals with the rest.
867 		 */
868 	case PCMCIA_PROP_UNKNOWN:
869 		return (ddi_bus_prop_op(dev, dip, ch_dip, prop_op,
870 		    mod_flags | DDI_PROP_NOTPROM,
871 		    name, valuep, lengthp));
872 	}
873 
874 	if (prop_op == PROP_LEN) {
875 		/* just the length */
876 		*lengthp = proplen;
877 		return (DDI_PROP_SUCCESS);
878 	}
879 	switch (prop_op) {
880 	case PROP_LEN_AND_VAL_ALLOC:
881 		if (mod_flags & DDI_PROP_CANSLEEP)
882 			flags = KM_SLEEP;
883 		else
884 			flags = KM_NOSLEEP;
885 		buff = kmem_alloc((size_t)proplen, flags);
886 		if (buff == NULL)
887 			return (DDI_PROP_NO_MEMORY);
888 		*(caddr_t *)valuep = (caddr_t)buff;
889 		break;
890 	case PROP_LEN_AND_VAL_BUF:
891 		buff = (caddr_t)valuep;
892 		if (len < proplen)
893 			return (DDI_PROP_BUF_TOO_SMALL);
894 		break;
895 	default:
896 		break;
897 	}
898 
899 	if (proplen > 0)
900 		bcopy(propptr, buff, proplen);
901 	*lengthp = proplen;
902 	return (DDI_PROP_SUCCESS);
903 }
904 
905 
906 struct regspec *
907 pcmcia_rnum_to_regspec(dev_info_t *dip, int rnumber)
908 {
909 	struct pcmcia_parent_private *ppd;
910 	ppd = (struct pcmcia_parent_private *)ddi_get_parent_data(dip);
911 	if (ppd->ppd_nreg < rnumber)
912 		return (NULL);
913 	return ((struct regspec *)&ppd->ppd_reg[rnumber]);
914 }
915 
916 struct regspec *
917 pcmcia_rnum_to_mapped(dev_info_t *dip, int rnumber)
918 {
919 	struct pcmcia_parent_private *ppd;
920 	ppd = (struct pcmcia_parent_private *)ddi_get_parent_data(dip);
921 	if (ppd->ppd_nreg < rnumber)
922 		return (NULL);
923 	if (ppd->ppd_assigned[rnumber].phys_len == 0)
924 		return (NULL);
925 	else
926 		return ((struct regspec *)&ppd->ppd_assigned[rnumber]);
927 }
928 
929 int
930 pcmcia_find_rnum(dev_info_t *dip, struct regspec *reg)
931 {
932 	struct pcmcia_parent_private *ppd;
933 	struct regspec *regp;
934 	int i;
935 
936 	ppd = (struct pcmcia_parent_private *)ddi_get_parent_data(dip);
937 	if (ppd == NULL)
938 		return (-1);
939 	for (regp = (struct regspec *)ppd->ppd_reg, i = 0;
940 	    i < ppd->ppd_nreg; i++, regp++) {
941 		if (bcmp(reg, regp, sizeof (struct regspec)) == 0)
942 			return (i);
943 	}
944 	for (regp = (struct regspec *)ppd->ppd_assigned, i = 0;
945 	    i < ppd->ppd_nreg; i++, regp++) {
946 		if (bcmp(reg, regp, sizeof (struct regspec)) == 0)
947 			return (i);
948 	}
949 
950 	return (-1);
951 }
952 
953 int
954 pcmcia_bus_map(dev_info_t *dip, dev_info_t *rdip, ddi_map_req_t *mp,
955 	off_t offset, off_t len, caddr_t *vaddrp)
956 {
957 	struct pcm_regs *regs, *mregs = NULL, tmp_reg;
958 	ddi_map_req_t mr = *mp;
959 	ra_return_t ret;
960 	int check, rnum = -1;
961 	uint32_t base;
962 	uchar_t regbuf[sizeof (pci_regspec_t)];
963 
964 	mp = &mr;		/* a copy of original request */
965 
966 	/* check for register number */
967 	switch (mp->map_type) {
968 	case DDI_MT_REGSPEC:
969 		regs = (struct pcm_regs *)mp->map_obj.rp;
970 		mregs = (struct pcm_regs *)mp->map_obj.rp;
971 		/*
972 		 * when using regspec, must not be relocatable
973 		 * and should be from assigned space.
974 		 */
975 		if (!PC_REG_RELOC(regs->phys_hi))
976 			return (DDI_FAILURE);
977 		rnum = pcmcia_find_rnum(rdip, (struct regspec *)mregs);
978 		break;
979 	case DDI_MT_RNUMBER:
980 		regs = (struct pcm_regs *)
981 			pcmcia_rnum_to_regspec(rdip, mp->map_obj.rnumber);
982 		mregs = (struct pcm_regs *)
983 			pcmcia_rnum_to_mapped(rdip, mp->map_obj.rnumber);
984 		rnum = mp->map_obj.rnumber;
985 		if (regs == NULL)
986 			return (DDI_FAILURE);
987 		mp->map_type = DDI_MT_REGSPEC;
988 		mp->map_obj.rp = (struct regspec *)mregs;
989 		break;
990 	default:
991 		return (DDI_ME_INVAL);
992 	}
993 
994 	/* basic sanity checks */
995 	switch (mp->map_op) {
996 	default:
997 		return (DDI_ME_UNIMPLEMENTED);
998 	case DDI_MO_UNMAP:
999 		if (mregs == NULL)
1000 			return (DDI_FAILURE);
1001 		regs = mregs;
1002 		break;
1003 	case DDI_MO_MAP_LOCKED:
1004 	case DDI_MO_MAP_HANDLE:
1005 		panic("unsupported bus operation");
1006 		/*NOTREACHED*/
1007 	}
1008 
1009 	/*
1010 	 * we need a private copy for manipulation and
1011 	 * calculation of the correct ranges
1012 	 */
1013 	tmp_reg = *regs;
1014 	mp->map_obj.rp = (struct regspec *)(regs = &tmp_reg);
1015 	base = regs->phys_lo;
1016 	if (base == 0 && offset != 0) {
1017 		/*
1018 		 * for now this is an error.  What does it really mean
1019 		 * to ask for an offset from an address that hasn't
1020 		 * been allocated yet.
1021 		 */
1022 		return (DDI_ME_INVAL);
1023 	}
1024 	regs->phys_lo += (uint32_t)offset;
1025 	if (len != 0) {
1026 		if (len > regs->phys_len) {
1027 			return (DDI_ME_INVAL);
1028 		}
1029 		regs->phys_len = len;
1030 	}
1031 
1032 	/*
1033 	 * basic sanity is checked so now make sure
1034 	 * we can actually allocate something for this
1035 	 * request and then convert to a "standard"
1036 	 * regspec for the next layer up (pci/isa/rootnex/etc.)
1037 	 */
1038 
1039 	switch (PC_GET_REG_TYPE(regs->phys_hi)) {
1040 	case PC_REG_SPACE_IO:
1041 		check = PCA_RES_NEED_IO;
1042 		break;
1043 	case PC_REG_SPACE_MEMORY:
1044 		check = PCA_RES_NEED_MEM;
1045 		break;
1046 	default:
1047 		/* not a valid register type */
1048 		return (DDI_FAILURE);
1049 	}
1050 
1051 	mr.map_type = DDI_MT_REGSPEC;
1052 	ret.ra_addr_hi = 0;
1053 	ret.ra_addr_lo = regs->phys_lo;
1054 	ret.ra_len = regs->phys_len;
1055 	mr.map_obj.rp = pcmcia_cons_regspec(dip,
1056 	    (check == PCA_RES_NEED_IO) ?
1057 	    PCMCIA_MAP_IO : PCMCIA_MAP_MEM,
1058 	    regbuf, &ret);
1059 	switch (mp->map_op) {
1060 	case DDI_MO_UNMAP:
1061 		pcmcia_set_assigned(rdip, rnum, NULL);
1062 		break;
1063 	default:
1064 		break;
1065 	}
1066 	return (ddi_map(dip, &mr, (off_t)0, (off_t)0, vaddrp));
1067 }
1068 
1069 /*
1070  * pcmcia_cons_regspec()
1071  * based on parent's bus type, construct a regspec that is usable
1072  * by that parent to map the resource into the system.
1073  */
1074 #define	PTYPE_PCI	1
1075 #define	PTYPE_ISA	0
1076 struct regspec *
1077 pcmcia_cons_regspec(dev_info_t *dip, int type, uchar_t *buff, ra_return_t *ret)
1078 {
1079 	int ptype = -1, len, bus;
1080 	char device_type[MODMAXNAMELEN + 1];
1081 	dev_info_t *pdip;
1082 	struct regspec *defreg;
1083 	pci_regspec_t *pcireg;
1084 
1085 	pdip = ddi_get_parent(dip);
1086 	if (pdip != ddi_root_node()) {
1087 		/* we're not a child of root so find out what */
1088 		len = sizeof (device_type);
1089 		if (ddi_prop_op(DDI_DEV_T_ANY, pdip, PROP_LEN_AND_VAL_BUF, 0,
1090 			"device_type", (caddr_t)device_type, &len) ==
1091 		    DDI_PROP_SUCCESS) {
1092 			/* check things out */
1093 			if (strcmp(device_type, "pci") == 0)
1094 				ptype = PTYPE_PCI;
1095 			else if (strcmp(device_type, "isa") == 0)
1096 				ptype = PTYPE_ISA;
1097 		}
1098 	}
1099 	switch (ptype) {
1100 	case PTYPE_PCI:
1101 		/* XXX need to look at carefully */
1102 		if (ddi_getlongprop(DDI_DEV_T_ANY, dip, DDI_PROP_DONTPASS,
1103 			"reg", (caddr_t)&pcireg, &len) == DDI_SUCCESS) {
1104 			bus = PCI_REG_BUS_G(pcireg->pci_phys_hi);
1105 			kmem_free(pcireg, len);
1106 		} else {
1107 			bus = 0;
1108 		}
1109 		pcireg = (pci_regspec_t *)buff;
1110 		pcireg->pci_phys_hi = (type == PCMCIA_MAP_IO ? PCI_ADDR_IO :
1111 			PCI_ADDR_MEM32) | PCI_RELOCAT_B | (bus << 16);
1112 		pcireg->pci_phys_mid = ret->ra_addr_hi;
1113 		pcireg->pci_phys_low = ret->ra_addr_lo;
1114 		if (type == PCMCIA_MAP_IO)
1115 			pcireg->pci_phys_low &= 0xFFFF;
1116 		pcireg->pci_size_hi = 0;
1117 		pcireg->pci_size_low = ret->ra_len;
1118 		break;
1119 	default:
1120 		/* default case is to use struct regspec */
1121 		defreg = (struct regspec *)buff;
1122 		defreg->regspec_bustype = type == PCMCIA_MAP_IO ? 1 : 0;
1123 		defreg->regspec_addr = ret->ra_addr_lo;
1124 		defreg->regspec_size = ret->ra_len;
1125 		break;
1126 	}
1127 	return ((struct regspec *)buff);
1128 }
1129 
1130 /*
1131  * pcmcia_init_adapter
1132  *	Initialize the per-adapter structures and check to see if
1133  *	there are possible other instances coming.
1134  */
1135 void
1136 pcmcia_init_adapter(anp_t *adapter, dev_info_t *dip)
1137 {
1138 	int i, n;
1139 	pcmcia_if_t *ls_if;
1140 
1141 	i = pcmcia_num_adapters++;
1142 	pcmcia_adapters[i] = kmem_zalloc(sizeof (struct pcmcia_adapter),
1143 	    KM_SLEEP);
1144 	pcmcia_adapters[i]->pca_dip = dip;
1145 	/* should this be pca_winshift??? */
1146 	pcmcia_adapters[i]->pca_module = ddi_name_to_major(ddi_get_name(dip));
1147 	pcmcia_adapters[i]->pca_unit = ddi_get_instance(dip);
1148 	pcmcia_adapters[i]->pca_iblock = adapter->an_iblock;
1149 	pcmcia_adapters[i]->pca_idev = adapter->an_idev;
1150 	pcmcia_adapters[i]->pca_if = ls_if = adapter->an_if;
1151 	pcmcia_adapters[i]->pca_number = i;
1152 	(void) strcpy(pcmcia_adapters[i]->pca_name, ddi_get_name(dip));
1153 	pcmcia_adapters[i]->
1154 		pca_name[sizeof (pcmcia_adapters[i]->pca_name) - 1] = NULL;
1155 
1156 	if (ls_if != NULL) {
1157 		inquire_adapter_t conf;
1158 		int sock, win;
1159 
1160 		if (ls_if->pcif_inquire_adapter != NULL)
1161 			GET_CONFIG(ls_if, dip, &conf);
1162 
1163 		/* resources - assume worst case and fix from there */
1164 		pcmcia_adapters[i]->pca_flags = PCA_RES_NEED_IRQ |
1165 			PCA_RES_NEED_IO | PCA_RES_NEED_MEM;
1166 		/* indicate first socket not initialized */
1167 		pcmcia_adapters[i]->pca_first_socket = -1;
1168 
1169 		if (conf.ResourceFlags & RES_OWN_IRQ)
1170 			pcmcia_adapters[i]->pca_flags &= ~PCA_RES_NEED_IRQ;
1171 		if (conf.ResourceFlags & RES_OWN_IO)
1172 			pcmcia_adapters[i]->pca_flags &= ~PCA_RES_NEED_IO;
1173 		if (conf.ResourceFlags & RES_OWN_MEM)
1174 			pcmcia_adapters[i]->pca_flags &= ~PCA_RES_NEED_MEM;
1175 		if (conf.ResourceFlags & RES_IRQ_SHAREABLE)
1176 			pcmcia_adapters[i]->pca_flags |= PCA_IRQ_SHAREABLE;
1177 		if (conf.ResourceFlags & RES_IRQ_NEXUS)
1178 			pcmcia_adapters[i]->pca_flags |= PCA_IRQ_SMI_SHARE;
1179 
1180 		/* need to know interrupt limitations */
1181 		if (conf.ActiveLow) {
1182 			pcmcia_adapters[i]->pca_avail_intr = conf.ActiveLow;
1183 			pcmcia_adapters[i]->pca_flags |= PCA_IRQ_ISA;
1184 		} else
1185 			pcmcia_adapters[i]->pca_avail_intr = conf.ActiveHigh;
1186 
1187 		/* power entries for adapter */
1188 		pcmcia_adapters[i]->pca_power =
1189 			conf.power_entry;
1190 		pcmcia_adapters[i]->pca_numpower =
1191 			conf.NumPower;
1192 
1193 		for (n = 0; n < conf.NumPower; n++)
1194 			pcmcia_merge_power(&conf.power_entry[n]);
1195 
1196 		/* now setup the per socket info */
1197 		for (sock = 0; sock < conf.NumSockets;
1198 		    sock++) {
1199 			dev_info_t *sockdrv = NULL;
1200 			sockdrv = pcmcia_number_socket(dip, sock);
1201 			if (sockdrv == NULL)
1202 				n = sock + pcmcia_num_sockets;
1203 			else {
1204 				n = ddi_get_instance(sockdrv);
1205 			}
1206 			/* make sure we know first socket on adapter */
1207 			if (pcmcia_adapters[i]->pca_first_socket == -1)
1208 				pcmcia_adapters[i]->pca_first_socket = n;
1209 
1210 			/*
1211 			 * the number of sockets is weird.
1212 			 * we might have only two sockets but
1213 			 * due to persistence of instances we
1214 			 * will need to call them something other
1215 			 * than 0 and 1.  So, we use the largest
1216 			 * instance number as the number and
1217 			 * have some that just don't get used.
1218 			 */
1219 			if (n >= pcmcia_num_sockets)
1220 				pcmcia_num_sockets = n + 1;
1221 #if defined(PCMCIA_DEBUG)
1222 			if (pcmcia_debug) {
1223 				cmn_err(CE_CONT,
1224 					"pcmcia_init: new socket added %d "
1225 					"(%d)\n",
1226 					n, pcmcia_num_sockets);
1227 			}
1228 #endif
1229 
1230 			pcmcia_sockets[n] =
1231 				kmem_zalloc(sizeof (pcmcia_logical_socket_t),
1232 				KM_SLEEP);
1233 			pcmcia_sockets[n]->ls_socket = sock;
1234 			pcmcia_sockets[n]->ls_if = ls_if;
1235 			pcmcia_sockets[n]->ls_adapter =
1236 				pcmcia_adapters[i];
1237 			pcmcia_sockets[n]->ls_cs_events = 0L;
1238 			pcmcia_sockets[n]->ls_sockdrv = sockdrv;
1239 			/* Prototype of intrspec */
1240 			pcmcia_sockets[n]->ls_intr_pri =
1241 				adapter->an_ipl;
1242 #if defined(PCMCIA_DEBUG)
1243 			if (pcmcia_debug)
1244 				cmn_err(CE_CONT,
1245 					"phys sock %d, log sock %d\n",
1246 					sock, n);
1247 #endif
1248 			mutex_init(&pcmcia_sockets[n]->ls_ilock, NULL,
1249 				MUTEX_DRIVER, *adapter->an_iblock);
1250 		}
1251 
1252 		pcmcia_adapters[i]->pca_numsockets = conf.NumSockets;
1253 		/* now setup the per window information */
1254 		for (win = 0; win < conf.NumWindows; win++) {
1255 			n = win + pcmcia_num_windows;
1256 			pcmcia_windows[n] =
1257 				kmem_zalloc(sizeof (pcmcia_logical_window_t),
1258 				    KM_SLEEP);
1259 			pcmcia_windows[n]->lw_window = win;
1260 			pcmcia_windows[n]->lw_if = ls_if;
1261 			pcmcia_windows[n]->lw_adapter =
1262 				pcmcia_adapters[i];
1263 		}
1264 		pcmcia_num_windows += conf.NumWindows;
1265 		SET_CALLBACK(ls_if, dip,
1266 		    pcm_adapter_callback, i);
1267 
1268 		/* now tell CS about each socket */
1269 		for (sock = 0; sock < pcmcia_num_sockets; sock++) {
1270 #if defined(PCMCIA_DEBUG)
1271 			if (pcmcia_debug) {
1272 				cmn_err(CE_CONT,
1273 					"pcmcia_init: notify CS socket %d "
1274 					"sockp=%p\n",
1275 					sock, (void *)pcmcia_sockets[sock]);
1276 			}
1277 #endif
1278 			if (pcmcia_sockets[sock] == NULL ||
1279 			    (pcmcia_sockets[sock]->ls_flags &
1280 				PCS_SOCKET_ADDED)) {
1281 				/* skip the ones that are done already */
1282 				continue;
1283 			}
1284 			pcmcia_sockets[sock]->ls_flags |= PCS_SOCKET_ADDED;
1285 			if (cs_event(PCE_ADD_SOCKET, sock, 0) !=
1286 			    CS_SUCCESS) {
1287 				/* flag socket as broken */
1288 				pcmcia_sockets[sock]->ls_flags = 0;
1289 			} else {
1290 				pcm_event_manager(PCE_ADD_SOCKET,
1291 				    sock, NULL);
1292 			}
1293 		}
1294 
1295 	}
1296 #if defined(PCMCIA_DEBUG)
1297 	if (pcmcia_debug) {
1298 		cmn_err(CE_CONT, "logical sockets:\n");
1299 		for (i = 0; i < pcmcia_num_sockets; i++) {
1300 			if (pcmcia_sockets[i] == NULL)
1301 				continue;
1302 			cmn_err(CE_CONT,
1303 				"\t%d: phys sock=%d, if=%p, adapt=%p\n",
1304 				i, pcmcia_sockets[i]->ls_socket,
1305 				(void *)pcmcia_sockets[i]->ls_if,
1306 				(void *)pcmcia_sockets[i]->ls_adapter);
1307 		}
1308 		cmn_err(CE_CONT, "logical windows:\n");
1309 		for (i = 0; i < pcmcia_num_windows; i++) {
1310 			cmn_err(CE_CONT,
1311 				"\t%d: phys_window=%d, if=%p, adapt=%p\n",
1312 				i, pcmcia_windows[i]->lw_window,
1313 				(void *)pcmcia_windows[i]->lw_if,
1314 				(void *)pcmcia_windows[i]->lw_adapter);
1315 		}
1316 		cmn_err(CE_CONT, "\tpcmcia_num_power=%d\n", pcmcia_num_power);
1317 		for (n = 0; n < pcmcia_num_power; n++)
1318 			cmn_err(CE_CONT,
1319 				"\t\tPowerLevel: %d\tValidSignals: %x\n",
1320 				pcmcia_power_table[n].PowerLevel,
1321 				pcmcia_power_table[n].ValidSignals);
1322 	}
1323 #endif
1324 }
1325 
1326 /*
1327  * pcmcia_enum_thread()
1328  *
1329  *	This thread is scheduled by pcmcia_attach() to enumerate the
1330  *	cards present in an adapter at boot time.
1331  *	pcmcia_enum_lock is used to serialize the execution of this
1332  *	thread with pcmcia_attach().
1333  */
1334 static void
1335 pcmcia_enum_thread(anp_t *adapter)
1336 {
1337 	mutex_enter(&pcmcia_enum_lock);
1338 	pcmcia_find_cards(adapter);
1339 	mutex_exit(&pcmcia_enum_lock);
1340 	thread_exit();
1341 }
1342 
1343 /*
1344  * pcmcia_find_cards()
1345  *	check the adapter to see if there are cards present at
1346  *	driver attach time.  If there are, generate an artificial
1347  *	card insertion event to get CS running and the PC Card ultimately
1348  *	identified.
1349  */
1350 void
1351 pcmcia_find_cards(anp_t *adapt)
1352 {
1353 	int i;
1354 	get_ss_status_t status;
1355 	for (i = 0; i < pcmcia_num_sockets; i++) {
1356 		if (pcmcia_sockets[i] &&
1357 		    pcmcia_sockets[i]->ls_if == adapt->an_if) {
1358 			/* check the status */
1359 			status.socket = i;
1360 			if (SSGetStatus(&status) == SUCCESS &&
1361 			    status.CardState & SBM_CD &&
1362 			    pcmcia_sockets[i]->ls_dip[0] == NULL) {
1363 				(void) cs_event(PCE_CARD_INSERT, i, 0);
1364 				delay(1);
1365 			}
1366 		}
1367 	}
1368 }
1369 
1370 /*
1371  * pcmcia_number_socket(dip, adapt)
1372  *	we determine socket number by creating a driver for each
1373  *	socket on the adapter and then forcing it to attach.  This
1374  *	results in an instance being assigned which becomes the
1375  *	logical socket number.	If it fails, then we are the first
1376  *	set of sockets and renumbering occurs later.  We do this
1377  *	one socket at a time and return the dev_info_t so the
1378  *	instance number can be used.
1379  */
1380 dev_info_t *
1381 pcmcia_number_socket(dev_info_t *dip, int localsocket)
1382 {
1383 	dev_info_t *child = NULL;
1384 	struct pcmcia_parent_private *ppd;
1385 
1386 	if (ndi_devi_alloc(dip, "pcs", (dnode_t)DEVI_SID_NODEID,
1387 	    &child) == NDI_SUCCESS) {
1388 		ppd = kmem_zalloc(sizeof (struct pcmcia_parent_private),
1389 		    KM_SLEEP);
1390 		ppd->ppd_reg = kmem_zalloc(sizeof (struct pcm_regs), KM_SLEEP);
1391 		ppd->ppd_nreg = 1;
1392 		ppd->ppd_reg[0].phys_hi = localsocket;
1393 		ddi_set_parent_data(child, (caddr_t)ppd);
1394 		if (ndi_devi_online(child, 0) != NDI_SUCCESS) {
1395 			(void) ndi_devi_free(child);
1396 			child = NULL;
1397 		}
1398 	}
1399 	return (child);
1400 }
1401 
1402 /*
1403  * pcm_phys_to_log_socket()
1404  *	from an adapter and socket number return the logical socket
1405  */
1406 int
1407 pcm_phys_to_log_socket(struct pcmcia_adapter *adapt, int socket)
1408 {
1409 	register pcmcia_logical_socket_t *sockp;
1410 	int i;
1411 
1412 	for (i = 0, sockp = pcmcia_sockets[0];
1413 		i < pcmcia_num_sockets; i++, sockp = pcmcia_sockets[i]) {
1414 		if (sockp == NULL)
1415 			continue;
1416 		if (sockp->ls_socket == socket && sockp->ls_adapter == adapt)
1417 			break;
1418 	}
1419 	if (i >= pcmcia_num_sockets) {
1420 #if defined(PCMCIA_DEBUG)
1421 		if (pcmcia_debug)
1422 			cmn_err(CE_CONT,
1423 				"\tbad socket/adapter: %x/%p != %x/%x\n",
1424 				socket, (void *)adapt, pcmcia_num_sockets,
1425 				pcmcia_num_adapters);
1426 #endif
1427 		return (-1);
1428 	}
1429 
1430 	return (i);		/* want logical socket */
1431 }
1432 
1433 /*
1434  * pcm_adapter_callback()
1435  *	this function is called back by the adapter driver at interrupt time.
1436  *	It is here that events should get generated for the event manager if it
1437  *	is present.  It would also be the time where a device information
1438  *	tree could be constructed for a card that was added in if we
1439  *	choose to create them dynamically.
1440  */
1441 
1442 #if defined(PCMCIA_DEBUG)
1443 char *cblist[] = {
1444 	"removal",
1445 	"insert",
1446 	"ready",
1447 	"battery-warn",
1448 	"battery-dead",
1449 	"status-change",
1450 	"write-protect", "reset", "unlock", "client-info", "eject-complete",
1451 	"eject-request", "erase-complete", "exclusive-complete",
1452 	"exclusive-request", "insert-complete", "insert-request",
1453 	"reset-complete", "reset-request", "timer-expired",
1454 	"resume", "suspend"
1455 };
1456 #endif
1457 
1458 /*ARGSUSED*/
1459 static int
1460 pcm_adapter_callback(dev_info_t *dip, int adapter, int event, int socket)
1461 {
1462 	pcmcia_logical_socket_t *sockp;
1463 
1464 #if defined(PCMCIA_DEBUG)
1465 	if (pcmcia_debug) {
1466 		cmn_err(CE_CONT, "pcm_adapter_callback: %p %x %x %x: ",
1467 			(void *)dip, adapter, event, socket);
1468 		cmn_err(CE_CONT, "[%s]\n", cblist[event]);
1469 	}
1470 #endif
1471 
1472 	if (adapter >= pcmcia_num_adapters || adapter < 0) {
1473 #if defined(PCMCIA_DEBUG)
1474 		if (pcmcia_debug)
1475 			cmn_err(CE_CONT, "\tbad adapter number: %d : %d\n",
1476 				adapter, pcmcia_num_adapters);
1477 #endif
1478 		return (1);
1479 	}
1480 
1481 	/* get the logical socket since that is what CS knows */
1482 	socket = pcm_phys_to_log_socket(pcmcia_adapters[adapter], socket);
1483 	if (socket == -1) {
1484 		cmn_err(CE_WARN, "pcmcia callback - bad logical socket\n");
1485 		return (0);
1486 	}
1487 	sockp = pcmcia_sockets[socket];
1488 	switch (event) {
1489 	case -1:		/* special case of adapter going away */
1490 	case PCE_CARD_INSERT:
1491 		sockp->ls_cs_events |= PCE_E2M(PCE_CARD_INSERT) |
1492 			PCE_E2M(PCE_CARD_REMOVAL);
1493 		break;
1494 	case PCE_CARD_REMOVAL:
1495 				/* disable interrupts at this point */
1496 		sockp->ls_cs_events |= PCE_E2M(PCE_CARD_INSERT) |
1497 			PCE_E2M(PCE_CARD_REMOVAL);
1498 		/* remove children that never attached */
1499 
1500 		break;
1501 	case PCE_PM_RESUME:
1502 		pcmcia_do_resume(socket, sockp);
1503 		/* event = PCE_CARD_INSERT; */
1504 		break;
1505 	case PCE_PM_SUSPEND:
1506 		pcmcia_do_suspend(socket, sockp);
1507 		/* event = PCE_CARD_REMOVAL; */
1508 		break;
1509 	default:
1510 		/* nothing to do */
1511 		break;
1512 	}
1513 
1514 #if defined(PCMCIA_DEBUG)
1515 	if (pcmcia_debug) {
1516 		cmn_err(CE_CONT,
1517 			"\tevent %d, event mask=%x, match=%x (log socket=%d)\n",
1518 			event,
1519 			(int)sockp->ls_cs_events,
1520 			(int)(sockp->ls_cs_events & PCE_E2M(event)), socket);
1521 	}
1522 #endif
1523 
1524 	if (pcmcia_cs_event && sockp->ls_cs_events & (1 << event)) {
1525 #if defined(PCMCIA_DEBUG)
1526 		if (pcmcia_debug)
1527 			cmn_err(CE_CONT, "\tcalling CS event handler (%p) "
1528 				"with event=%d\n",
1529 				(void *)pcmcia_cs_event, event);
1530 #endif
1531 		CS_EVENT(event, socket, 0);
1532 	}
1533 
1534 	/* let the event manager(s) know about the event */
1535 	pcm_event_manager(event, socket, NULL);
1536 
1537 	return (0);
1538 }
1539 
1540 /*
1541  * pcm_event_manager()
1542  *	checks for registered management driver callback handlers
1543  *	if there are any, call them if the event warrants it
1544  */
1545 void
1546 pcm_event_manager(int event, int socket, void *arg)
1547 {
1548 	struct pcmcia_mif *mif;
1549 
1550 	for (mif = pcmcia_mif_handlers; mif != NULL; mif = mif->mif_next) {
1551 #if defined(PCMCIA_DEBUG)
1552 		if (pcmcia_debug)
1553 			cmn_err(CE_CONT,
1554 				"pcm_event_manager: event=%d, mif_events=%x"
1555 				" (tst:%d)\n",
1556 				event, (int)*(uint32_t *)mif->mif_events,
1557 				PR_GET(mif->mif_events, event));
1558 #endif
1559 		if (PR_GET(mif->mif_events, event)) {
1560 			mif->mif_function(mif->mif_id, event, socket, arg);
1561 		}
1562 	}
1563 
1564 }
1565 
1566 /*
1567  * pcm_search_devinfo(dev_info_t *, pcm_device_info *, int)
1568  * search for an immediate child node to the nexus and not siblings of nexus
1569  * and not grandchildren.  We follow the same sequence that name binding
1570  * follows so we match same class of device (modem == modem) and don't
1571  * have to depend on features that might not exist.
1572  */
1573 dev_info_t *
1574 pcm_search_devinfo(dev_info_t *self, struct pcm_device_info *info, int socket)
1575 {
1576 	char bf[256];
1577 	struct pcmcia_parent_private *ppd;
1578 	dev_info_t *dip;
1579 	int circular;
1580 
1581 #if defined(PCMCIA_DEBUG)
1582 	if (pcmcia_debug)
1583 		cmn_err(CE_CONT,
1584 		    "pcm_search_devinfo: socket=%x [%s|%s|%s] pd_flags=%x\n",
1585 		    socket, info->pd_bind_name, info->pd_generic_name,
1586 		    info->pd_vers1_name, info->pd_flags);
1587 #endif
1588 
1589 	ndi_devi_enter(self, &circular);
1590 	/* do searches in compatible property order */
1591 	for (dip = (dev_info_t *)DEVI(self)->devi_child;
1592 	    dip != NULL;
1593 	    dip = (dev_info_t *)DEVI(dip)->devi_sibling) {
1594 		int ppd_socket;
1595 		ppd = (struct pcmcia_parent_private *)
1596 			ddi_get_parent_data(dip);
1597 		if (ppd == NULL) {
1598 #if defined(PCMCIA_DEBUG)
1599 			cmn_err(CE_WARN, "No parent private data\n");
1600 #endif
1601 			continue;
1602 		}
1603 		ppd_socket = CS_MAKE_SOCKET_NUMBER(ppd->ppd_socket,
1604 		    ppd->ppd_function);
1605 #if defined(PCMCIA_DEBUG)
1606 		if (pcmcia_debug) {
1607 			cmn_err(CE_CONT, "\tbind=[%s], node=[%s]\n",
1608 				DEVI(dip)->devi_binding_name,
1609 				DEVI(dip)->devi_node_name);
1610 		}
1611 #endif
1612 		if (info->pd_flags & PCM_NAME_VERS1) {
1613 			(void) strcpy(bf, info->pd_vers1_name);
1614 			pcmcia_fix_string(bf);
1615 			if (DEVI(dip)->devi_binding_name &&
1616 			    strcmp(DEVI(dip)->devi_binding_name, bf) == 0 &&
1617 			    socket == ppd_socket)
1618 				break;
1619 		}
1620 		if ((info->pd_flags & (PCM_NAME_1275 | PCM_MULTI_FUNCTION)) ==
1621 		    (PCM_NAME_1275 | PCM_MULTI_FUNCTION)) {
1622 			(void) sprintf(bf, "%s,%x", info->pd_bind_name,
1623 				info->pd_function);
1624 			if (strcmp(bf, DEVI(dip)->devi_binding_name) == 0 &&
1625 			    socket == ppd->ppd_socket)
1626 				break;
1627 		}
1628 		if (info->pd_flags & PCM_NAME_1275) {
1629 			if (DEVI(dip)->devi_binding_name &&
1630 			    strcmp(DEVI(dip)->devi_binding_name,
1631 				info->pd_bind_name) == 0 &&
1632 			    socket == ppd_socket)
1633 				break;
1634 		}
1635 		if (info->pd_flags & PCM_NAME_GENERIC) {
1636 			(void) sprintf(bf, "%s,%s", PCMDEV_NAMEPREF,
1637 				info->pd_generic_name);
1638 			if (DEVI(dip)->devi_binding_name &&
1639 			    strcmp(DEVI(dip)->devi_binding_name, bf) == 0 &&
1640 			    socket == ppd_socket)
1641 				break;
1642 		}
1643 		if (info->pd_flags & PCM_NAME_GENERIC) {
1644 			if (DEVI(dip)->devi_binding_name &&
1645 			    strcmp(DEVI(dip)->devi_binding_name,
1646 				info->pd_generic_name) == 0 &&
1647 			    socket == ppd_socket)
1648 				break;
1649 		}
1650 		if (info->pd_flags & PCM_NO_CONFIG) {
1651 			if (DEVI(dip)->devi_binding_name &&
1652 			    strcmp(DEVI(dip)->devi_binding_name,
1653 					"pccard,memory") == 0 &&
1654 			    socket == ppd_socket)
1655 				break;
1656 		}
1657 	}
1658 	ndi_devi_exit(self, circular);
1659 	return (dip);
1660 }
1661 
1662 /*
1663  * pcm_find_devinfo()
1664  *	this is a wrapper around DDI calls to "find" any
1665  *	devinfo node and then from there find the one associated
1666  *	with the socket
1667  */
1668 dev_info_t *
1669 pcm_find_devinfo(dev_info_t *pdip, struct pcm_device_info *info, int socket)
1670 {
1671 	dev_info_t *dip;
1672 
1673 	dip = pcm_search_devinfo(pdip, info, socket);
1674 	if (dip == NULL)
1675 		return (NULL);
1676 	/*
1677 	 * we have at least a base level dip
1678 	 * see if there is one (this or a sibling)
1679 	 * that has the correct socket number
1680 	 * if there is, return that one else
1681 	 * NULL so a new one is created
1682 	 */
1683 #if defined(PCMCIA_DEBUG)
1684 	if (pcmcia_debug)
1685 		cmn_err(CE_CONT, "find: initial dip = %p, socket=%d, name=%s "
1686 			"(instance=%d, socket=%d, name=%s)\n",
1687 			(void *)dip, socket, info->pd_bind_name,
1688 			ddi_get_instance(dip),
1689 			ddi_getprop(DDI_DEV_T_ANY, dip, DDI_PROP_DONTPASS,
1690 			PCM_DEV_SOCKET, -1),
1691 			ddi_get_name(dip));
1692 #endif
1693 
1694 #if defined(PCMCIA_DEBUG)
1695 	if (pcmcia_debug && dip != NULL)
1696 		cmn_err(CE_CONT, "\treturning non-NULL dip (%s)\n",
1697 			ddi_get_name(dip));
1698 #endif
1699 	return (dip);
1700 }
1701 
1702 /*
1703  * pcm_find_parent_dip(socket)
1704  *	find the correct parent dip for this logical socket
1705  */
1706 dev_info_t *
1707 pcm_find_parent_dip(int socket)
1708 {
1709 	if ((socket < 0 || socket >= pcmcia_num_sockets) ||
1710 	    pcmcia_sockets[socket] == NULL)
1711 		return (NULL);
1712 	return (pcmcia_sockets[socket]->ls_adapter->pca_dip);
1713 }
1714 
1715 /*
1716  * pcmcia_set_em_handler()
1717  *	This is called by the management and event driver to tell
1718  *	the nexus what to call.	 Multiple drivers are allowed
1719  *	but normally only one will exist.
1720  */
1721 int
1722 pcmcia_set_em_handler(int (*handler)(), caddr_t events, int elen,
1723 			uint32_t id, void **cs, void **ss)
1724 {
1725 	struct pcmcia_mif *mif, *tmp;
1726 
1727 	if (handler == NULL) {
1728 		/* NULL means remove the handler based on the ID */
1729 		if (pcmcia_mif_handlers == NULL)
1730 			return (0);
1731 		mutex_enter(&pcmcia_global_lock);
1732 		if (pcmcia_mif_handlers->mif_id == id) {
1733 			mif = pcmcia_mif_handlers;
1734 			pcmcia_mif_handlers = mif->mif_next;
1735 			kmem_free(mif, sizeof (struct pcmcia_mif));
1736 		} else {
1737 			for (mif = pcmcia_mif_handlers;
1738 			    mif->mif_next != NULL &&
1739 			    mif->mif_next->mif_id != id;
1740 			    mif = mif->mif_next)
1741 				;
1742 			if (mif->mif_next != NULL &&
1743 			    mif->mif_next->mif_id == id) {
1744 				tmp = mif->mif_next;
1745 				mif->mif_next = tmp->mif_next;
1746 				kmem_free(tmp, sizeof (struct pcmcia_mif));
1747 			}
1748 		}
1749 		mutex_exit(&pcmcia_global_lock);
1750 	} else {
1751 
1752 		if (pcmcia_num_adapters == 0) {
1753 			return (ENXIO);
1754 		}
1755 		if (elen > EM_EVENTSIZE)
1756 			return (EINVAL);
1757 
1758 		mif = (struct pcmcia_mif *)
1759 			kmem_zalloc(sizeof (struct pcmcia_mif),
1760 			    KM_NOSLEEP);
1761 		if (mif == NULL)
1762 			return (ENOSPC);
1763 
1764 		mif->mif_function = (void (*)())handler;
1765 		bcopy(events, mif->mif_events, elen);
1766 		mif->mif_id = id;
1767 		mutex_enter(&pcmcia_global_lock);
1768 		mif->mif_next = pcmcia_mif_handlers;
1769 		pcmcia_mif_handlers = mif;
1770 		if (cs != NULL)
1771 			*cs = (void *)pcmcia_card_services;
1772 		if (ss != NULL) {
1773 			*ss = (void *)SocketServices;
1774 		}
1775 
1776 		mutex_exit(&pcmcia_global_lock);
1777 	}
1778 	return (0);
1779 }
1780 
1781 /*
1782  * pcm_fix_bits(uchar_t *data, int num, int dir)
1783  *	shift socket bits left(0) or right(0)
1784  *	This is used when mapping logical and physical
1785  */
1786 void
1787 pcm_fix_bits(socket_enum_t src, socket_enum_t dst, int num, int dir)
1788 {
1789 	int i;
1790 
1791 	PR_ZERO(dst);
1792 
1793 	if (dir == 0) {
1794 				/* LEFT */
1795 		for (i = 0; i <= (sizeof (dst) * PR_WORDSIZE) - num; i++) {
1796 			if (PR_GET(src, i))
1797 				PR_SET(dst, i + num);
1798 		}
1799 	} else {
1800 				/* RIGHT */
1801 		for (i = num; i < sizeof (dst) * PR_WORDSIZE; i++) {
1802 			if (PR_GET(src, i))
1803 			    PR_SET(dst, i - num);
1804 		}
1805 	}
1806 }
1807 
1808 uint32_t
1809 genmask(int len)
1810 {
1811 	uint32_t mask;
1812 	for (mask = 0; len > 0; len--) {
1813 		mask |= 1 << (len - 1);
1814 	}
1815 	return (mask);
1816 }
1817 
1818 int
1819 genp2(int val)
1820 {
1821 	int i;
1822 	if (val == 0)
1823 		return (0);
1824 	for (i = 0; i < 32; i++)
1825 		if (val > (1 << i))
1826 			return (i);
1827 	return (0);
1828 }
1829 
1830 #if defined(PCMCIA_DEBUG)
1831 char *ssfuncs[128] = {
1832 	"GetAdapter", "GetPage", "GetSocket", "GetStatus", "GetWindow",
1833 	"InquireAdapter", "InquireSocket", "InquireWindow", "ResetSocket",
1834 	"SetPage", "SetAdapter", "SetSocket", "SetWindow", "SetIRQHandler",
1835 	"ClearIRQHandler",
1836 	/* 15 */ NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
1837 	/* 25 */ NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
1838 	/* 35 */ NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
1839 	/* 45 */ NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
1840 	/* 55 */ NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
1841 	/* 65 */ NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
1842 	/* 75 */ NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
1843 	/* 85 */ NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
1844 	/* 95 */ NULL, NULL, NULL,
1845 	"CSIsActiveDip",
1846 	"CSInitDev", "CSRegister", "CSCISInit", "CSUnregister",
1847 	"CISGetAddress", "CISSetAddress", "CSCardRemoved", "CSGetCookiesAndDip"
1848 };
1849 #endif
1850 
1851 /*
1852  * SocketServices
1853  *	general entrypoint for Card Services to find
1854  *	Socket Services.  Finding the entry requires
1855  *	a _depends_on[] relationship.
1856  *
1857  *	In some cases, the work is done locally but usually
1858  *	the parameters are adjusted and the adapter driver
1859  *	code asked to do the work.
1860  */
1861 int
1862 SocketServices(int function, ...)
1863 {
1864 	va_list arglist;
1865 	uint32_t args[16];
1866 	csregister_t *reg;
1867 	sservice_t *serv;
1868 	dev_info_t *dip;
1869 	int socket, func;
1870 	int error = SUCCESS;
1871 	pcmcia_logical_socket_t *sockp;
1872 
1873 	va_start(arglist, function);
1874 
1875 #if defined(PCMCIA_DEBUG)
1876 	if (pcmcia_debug > 1)
1877 		cmn_err(CE_CONT, "SocketServices called for function %d [%s]\n",
1878 			function,
1879 			((function < 128) && ssfuncs[function] != NULL) ?
1880 			ssfuncs[function] : "UNKNOWN");
1881 #endif
1882 	switch (function) {
1883 	case CSRegister:
1884 	case CISGetAddress:
1885 	case CISSetAddress:
1886 
1887 		reg = va_arg(arglist, csregister_t *);
1888 
1889 		if (reg->cs_magic != PCCS_MAGIC ||
1890 		    reg->cs_version != PCCS_VERSION) {
1891 			cmn_err(CE_WARN,
1892 				"pcmcia: CSRegister (%x, %x, %p, %p) *ERROR*",
1893 				reg->cs_magic, reg->cs_version,
1894 				(void *)reg->cs_card_services,
1895 				(void *)reg->cs_event);
1896 			error = BAD_FUNCTION;
1897 			break;
1898 		}
1899 
1900 		switch (function) {
1901 		case CISGetAddress:
1902 			reg->cs_event = pcmcia_cis_parser;
1903 			break;
1904 		case CISSetAddress:
1905 			pcmcia_cis_parser = reg->cs_event;
1906 			break;
1907 		case CSRegister:
1908 			break;
1909 		}
1910 		break;
1911 
1912 	case CSUnregister:
1913 		break;
1914 
1915 	case CSCISInit:
1916 		args[0] = va_arg(arglist, int);
1917 #if defined(PCMCIA_DEBUG)
1918 		if (pcmcia_debug)
1919 			cmn_err(CE_CONT,
1920 				"CSCISInit: CIS is initialized on socket %d\n",
1921 				(int)args[0]);
1922 #endif
1923 		/*
1924 		 * now that the CIS has been parsed (there may not
1925 		 * be one but the work is done) we can create the
1926 		 * device information structures.
1927 		 *
1928 		 * we serialize the node creation to avoid problems
1929 		 * with initial probe/attach of nexi.
1930 		 */
1931 
1932 		mutex_enter(&pcmcia_global_lock);
1933 		pcmcia_create_dev_info(args[0]);
1934 		cv_broadcast(&pcmcia_condvar); /* wakeup the nexus attach */
1935 		mutex_exit(&pcmcia_global_lock);
1936 		break;
1937 
1938 	case CSInitDev:
1939 #if defined(PCMCIA_DEBUG)
1940 		if (pcmcia_debug)
1941 			cmn_err(CE_CONT, "CSInitDev: initialize device\n");
1942 #endif
1943 		/*
1944 		 * this is where we create the /devices entries
1945 		 * that let us out into the world
1946 		 */
1947 
1948 		(void) pcmcia_create_device(va_arg(arglist,
1949 		    ss_make_device_node_t *));
1950 		break;
1951 
1952 	case CSCardRemoved:
1953 		args[0] = va_arg(arglist, uint32_t);
1954 		socket = CS_GET_SOCKET_NUMBER(args[0]);
1955 		func = CS_GET_FUNCTION_NUMBER(args[0]);
1956 #if defined(PCMCIA_DEBUG)
1957 		if (pcmcia_debug)
1958 			cmn_err(CE_CONT,
1959 				"CSCardRemoved! (socket=%d)\n", (int)args[0]);
1960 #endif
1961 		if (socket >= pcmcia_num_sockets)
1962 			break;
1963 
1964 		sockp = pcmcia_sockets[socket];
1965 		if (sockp == NULL) {
1966 			cmn_err(CE_WARN,
1967 			    "pcmcia: bad socket = %x", socket);
1968 			break;
1969 		}
1970 
1971 		for (func = 0; func < sockp->ls_functions; func++) {
1972 			/*
1973 			 * break the association of dip and socket
1974 			 * for all functions on that socket
1975 			 */
1976 			dip = sockp->ls_dip[func];
1977 			sockp->ls_dip[func] = NULL;
1978 			if (dip != NULL) {
1979 				struct pcmcia_parent_private *ppd;
1980 				ppd = (struct pcmcia_parent_private *)
1981 				    ddi_get_parent_data(dip);
1982 				ppd->ppd_active = 0;
1983 				(void) ndi_devi_offline(dip,
1984 				    NDI_DEVI_REMOVE);
1985 			}
1986 #if defined(PCMCIA_DEBUG)
1987 			else {
1988 				if (pcmcia_debug)
1989 					cmn_err(CE_CONT,
1990 					    "CardRemoved: no "
1991 					    "dip present "
1992 					    "on socket %d!\n",
1993 					    (int)args[0]);
1994 			}
1995 #endif
1996 		}
1997 		if (sockp->ls_flags & PCS_SUSPENDED) {
1998 			mutex_enter(&pcmcia_global_lock);
1999 			sockp->ls_flags &= ~PCS_SUSPENDED;
2000 			cv_broadcast(&pcmcia_condvar);
2001 			mutex_exit(&pcmcia_global_lock);
2002 		}
2003 		break;
2004 
2005 	case CSGetCookiesAndDip:
2006 		serv = va_arg(arglist, sservice_t *);
2007 		if (serv != NULL)
2008 			error = GetCookiesAndDip(serv);
2009 		else
2010 			error = BAD_SOCKET;
2011 		break;
2012 
2013 	case CSGetActiveDip:
2014 		/*
2015 		 * get the dip associated with the card currently
2016 		 * in the specified socket
2017 		 */
2018 		args[0] = va_arg(arglist, uint32_t);
2019 		socket = CS_GET_SOCKET_NUMBER(args[0]);
2020 		func = CS_GET_FUNCTION_NUMBER(args[0]);
2021 		error = (long)pcmcia_sockets[socket]->ls_dip[func];
2022 		break;
2023 
2024 		/*
2025 		 * the remaining entries are SocketServices calls
2026 		 */
2027 	case SS_GetAdapter:
2028 		error = SSGetAdapter(va_arg(arglist, get_adapter_t *));
2029 		break;
2030 	case SS_GetPage:
2031 		error = SSGetPage(va_arg(arglist, get_page_t *));
2032 		break;
2033 	case SS_GetSocket:
2034 		error = SSGetSocket(va_arg(arglist, get_socket_t *));
2035 		break;
2036 	case SS_GetStatus:
2037 		error = SSGetStatus(va_arg(arglist, get_ss_status_t *));
2038 		break;
2039 	case SS_GetWindow:
2040 		error = SSGetWindow(va_arg(arglist, get_window_t *));
2041 		break;
2042 	case SS_InquireAdapter:
2043 		error = SSInquireAdapter(va_arg(arglist, inquire_adapter_t *));
2044 		break;
2045 	case SS_InquireSocket:
2046 		error = SSInquireSocket(va_arg(arglist, inquire_socket_t *));
2047 		break;
2048 	case SS_InquireWindow:
2049 		error = SSInquireWindow(va_arg(arglist, inquire_window_t *));
2050 		break;
2051 	case SS_ResetSocket:
2052 		args[0] = va_arg(arglist, uint32_t);
2053 		args[1] = va_arg(arglist, int);
2054 		error = SSResetSocket(args[0], args[1]);
2055 		break;
2056 	case SS_SetPage:
2057 		error = SSSetPage(va_arg(arglist, set_page_t *));
2058 		break;
2059 	case SS_SetSocket:
2060 		error = SSSetSocket(va_arg(arglist, set_socket_t *));
2061 		break;
2062 	case SS_SetWindow:
2063 		error = SSSetWindow(va_arg(arglist, set_window_t *));
2064 		break;
2065 	case SS_SetIRQHandler:
2066 		error = SSSetIRQHandler(va_arg(arglist, set_irq_handler_t *));
2067 		break;
2068 	case SS_ClearIRQHandler:
2069 		error = SSClearIRQHandler(va_arg(arglist,
2070 		    clear_irq_handler_t *));
2071 		break;
2072 	default:
2073 		error = BAD_FUNCTION;
2074 		break;
2075 	}
2076 	va_end(arglist);
2077 	return (error);
2078 }
2079 
2080 /*
2081  * pcmcia_merge_power()
2082  *	The adapters may have different power tables so it
2083  *	is necessary to construct a single power table that
2084  *	can be used throughout the system.  The result is
2085  *	a merger of all capabilities.  The nexus adds
2086  *	power table entries one at a time.
2087  */
2088 void
2089 pcmcia_merge_power(struct power_entry *power)
2090 {
2091 	int i;
2092 	struct power_entry pwr;
2093 
2094 	pwr = *power;
2095 
2096 	for (i = 0; i < pcmcia_num_power; i++) {
2097 		if (pwr.PowerLevel == pcmcia_power_table[i].PowerLevel) {
2098 			if (pwr.ValidSignals ==
2099 				pcmcia_power_table[i].ValidSignals) {
2100 				return;
2101 			} else {
2102 				/* partial match */
2103 				pwr.ValidSignals &=
2104 					~pcmcia_power_table[i].ValidSignals;
2105 			}
2106 		}
2107 	}
2108 	/* what's left becomes a new entry */
2109 	if (pcmcia_num_power == PCMCIA_MAX_POWER)
2110 		return;
2111 	pcmcia_power_table[pcmcia_num_power++] = pwr;
2112 }
2113 
2114 /*
2115  * pcmcia_do_suspend()
2116  *	tell CS that a suspend has happened by passing a
2117  *	card removal event.  Then cleanup the socket state
2118  *	to fake the cards being removed so resume works
2119  */
2120 void
2121 pcmcia_do_suspend(int socket, pcmcia_logical_socket_t *sockp)
2122 {
2123 	get_ss_status_t stat;
2124 	struct pcmcia_adapter *adapt;
2125 	pcmcia_if_t *ls_if;
2126 	dev_info_t *dip;
2127 	int i;
2128 
2129 #ifdef	XXX
2130 	if (pcmcia_cs_event == NULL) {
2131 		return;
2132 	}
2133 #endif
2134 
2135 	ls_if = sockp->ls_if;
2136 	adapt = sockp->ls_adapter;
2137 
2138 	if (ls_if == NULL || ls_if->pcif_get_status == NULL) {
2139 		return;
2140 	}
2141 
2142 	stat.socket = socket;
2143 #if defined(PCMCIA_DEBUG)
2144 	if (pcmcia_debug) {
2145 		cmn_err(CE_CONT,
2146 			"pcmcia_do_suspend(%d, %p)\n", socket, (void *)sockp);
2147 	}
2148 #endif
2149 
2150 	if (GET_STATUS(ls_if, adapt->pca_dip, &stat) != SUCCESS)
2151 		return;
2152 
2153 	/*
2154 	 * If there is a card in the socket, then we need to send
2155 	 *	everyone a PCE_CARD_REMOVAL event, and remove the
2156 	 *	card active property.
2157 	 */
2158 
2159 	for (i = 0; i < sockp->ls_functions; i++) {
2160 		struct pcmcia_parent_private *ppd;
2161 		dip = sockp->ls_dip[i];
2162 		if (dip != NULL) {
2163 			ppd = (struct pcmcia_parent_private *)
2164 				ddi_get_parent_data(dip);
2165 			ppd->ppd_flags |= PPD_SUSPENDED;
2166 		}
2167 #if 0
2168 		sockp->ls_dip[i] = NULL;
2169 #endif
2170 	}
2171 	sockp->ls_flags |= PCS_SUSPENDED;
2172 
2173 	if (pcmcia_cs_event &&
2174 	    (sockp->ls_cs_events & (1 << PCE_PM_SUSPEND))) {
2175 		CS_EVENT(PCE_PM_SUSPEND, socket, 0);
2176 	}
2177 	pcm_event_manager(PCE_PM_SUSPEND, socket, NULL);
2178 }
2179 
2180 /*
2181  * pcmcia_do_resume()
2182  *	tell CS that a suspend has happened by passing a
2183  *	card removal event.  Then cleanup the socket state
2184  *	to fake the cards being removed so resume works
2185  */
2186 void
2187 pcmcia_do_resume(int socket, pcmcia_logical_socket_t *sockp)
2188 {
2189 	get_ss_status_t stat;
2190 	struct pcmcia_adapter *adapt;
2191 	pcmcia_if_t *ls_if;
2192 
2193 #ifdef	XXX
2194 	if (pcmcia_cs_event == NULL) {
2195 		return;
2196 	}
2197 #endif
2198 
2199 	ls_if = sockp->ls_if;
2200 	adapt = sockp->ls_adapter;
2201 
2202 	if (ls_if == NULL || ls_if->pcif_get_status == NULL) {
2203 		return;
2204 	}
2205 
2206 	stat.socket = socket;
2207 #if defined(PCMCIA_DEBUG)
2208 	if (pcmcia_debug) {
2209 		cmn_err(CE_CONT,
2210 			"pcmcia_do_resume(%d, %p)\n", socket, (void *)sockp);
2211 	}
2212 #endif
2213 	if (GET_STATUS(ls_if, adapt->pca_dip, &stat) ==
2214 	    SUCCESS) {
2215 
2216 #if defined(PCMCIA_DEBUG)
2217 		if (pcmcia_debug)
2218 			cmn_err(CE_CONT, "\tsocket=%x, CardState=%x\n",
2219 				socket, stat.CardState);
2220 #endif
2221 #if 0
2222 		/* now have socket info -- do we have events? */
2223 		if ((stat.CardState & SBM_CD) == SBM_CD) {
2224 			if (pcmcia_cs_event &&
2225 			    (sockp->ls_cs_events & (1 << PCE_CARD_INSERT))) {
2226 				CS_EVENT(PCE_CARD_INSERT, socket, 0);
2227 			}
2228 
2229 			/* we should have card removed from CS soon */
2230 			pcm_event_manager(PCE_CARD_INSERT, socket, NULL);
2231 		}
2232 #else
2233 		if (pcmcia_cs_event &&
2234 		    (sockp->ls_cs_events & (1 << PCE_PM_SUSPEND))) {
2235 			CS_EVENT(PCE_PM_RESUME, socket, 0);
2236 			CS_EVENT(PCE_CARD_REMOVAL, socket, 0);
2237 			if ((stat.CardState & SBM_CD) == SBM_CD)
2238 				CS_EVENT(PCE_CARD_INSERT, socket, 0);
2239 		}
2240 #endif
2241 	}
2242 }
2243 
2244 /*
2245  * pcmcia_map_power_set()
2246  *	Given a power table entry and level, find it in the
2247  *	master table and return the index in the adapter table.
2248  */
2249 static int
2250 pcmcia_map_power_set(struct pcmcia_adapter *adapt, int level, int which)
2251 {
2252 	int plevel, i;
2253 	struct power_entry *pwr = (struct power_entry *)adapt->pca_power;
2254 	plevel = pcmcia_power_table[level].PowerLevel;
2255 	/* mask = pcmcia_power_table[level].ValidSignals; */
2256 	for (i = 0; i < adapt->pca_numpower; i++)
2257 		if (plevel == pwr[i].PowerLevel &&
2258 		    pwr[i].ValidSignals & which)
2259 			return (i);
2260 	return (0);
2261 }
2262 
2263 /*
2264  * pcmcia_map_power_get()
2265  *	Given an adapter power entry, find the appropriate index
2266  *	in the master table.
2267  */
2268 static int
2269 pcmcia_map_power_get(struct pcmcia_adapter *adapt, int level, int which)
2270 {
2271 	int plevel, i;
2272 	struct power_entry *pwr = (struct power_entry *)adapt->pca_power;
2273 	plevel = pwr[level].PowerLevel;
2274 	/* mask = pwr[level].ValidSignals; */
2275 	for (i = 0; i < pcmcia_num_power; i++)
2276 		if (plevel == pcmcia_power_table[i].PowerLevel &&
2277 		    pcmcia_power_table[i].ValidSignals & which)
2278 			return (i);
2279 	return (0);
2280 }
2281 
2282 /*
2283  * XXX - SS really needs a way to allow the caller to express
2284  *	interest in PCE_CARD_STATUS_CHANGE events.
2285  */
2286 static uint32_t
2287 pcm_event_map[32] = {
2288 	PCE_E2M(PCE_CARD_WRITE_PROTECT)|PCE_E2M(PCE_CARD_STATUS_CHANGE),
2289 	PCE_E2M(PCE_CARD_UNLOCK)|PCE_E2M(PCE_CARD_STATUS_CHANGE),
2290 	PCE_E2M(PCE_EJECTION_REQUEST)|PCE_E2M(PCE_CARD_STATUS_CHANGE),
2291 	PCE_E2M(PCE_INSERTION_REQUEST)|PCE_E2M(PCE_CARD_STATUS_CHANGE),
2292 	PCE_E2M(PCE_CARD_BATTERY_WARN)|PCE_E2M(PCE_CARD_STATUS_CHANGE),
2293 	PCE_E2M(PCE_CARD_BATTERY_DEAD)|PCE_E2M(PCE_CARD_STATUS_CHANGE),
2294 	PCE_E2M(PCE_CARD_READY)|PCE_E2M(PCE_CARD_STATUS_CHANGE),
2295 	PCE_E2M(PCE_CARD_REMOVAL)|PCE_E2M(PCE_CARD_INSERT)|
2296 					PCE_E2M(PCE_CARD_STATUS_CHANGE),
2297 	PCE_E2M(PCE_PM_SUSPEND)|PCE_E2M(PCE_PM_RESUME),
2298 };
2299 
2300 static int
2301 pcm_mapevents(uint32_t eventmask)
2302 {
2303 	uint32_t mask;
2304 	int i;
2305 
2306 	for (i = 0, mask = 0; eventmask && i < 32; i++) {
2307 		if (eventmask & (1 << i)) {
2308 			mask |= pcm_event_map[i];
2309 			eventmask &= ~(1 << i);
2310 		}
2311 	}
2312 	return (mask);
2313 }
2314 
2315 
2316 /*
2317  * PCMCIA Generic Naming Support
2318  *
2319  * With 2.6, PCMCIA naming moves to the 1275 and generic naming model.
2320  * Consequently, the whole naming mechanism is to be changed.  This is
2321  * not backward compatible with the current names but that isn't a problem
2322  * due to so few drivers existing.
2323  *
2324  * For cards with a device_id tuple, a generic name will be used.
2325  * if there is no device_id, then the 1275 name will be used if possible.
2326  * The 1275 name is of the form pccardNNNN,MMMM from the manfid tuple.
2327  * if there is not manfid tuple, an attempt will be made to bind the
2328  * node to the version_1 strings.
2329  *
2330  * In all cases, a "compatible" property is created with a number
2331  * of names.  The most generic name will be last in the list.
2332  */
2333 
2334 /*
2335  * pcmcia_fix_string()
2336  * want to avoid special characters in alias strings so convert
2337  * to something innocuous
2338  */
2339 
2340 void
2341 pcmcia_fix_string(char *str)
2342 {
2343 	for (; str && *str; str++) {
2344 		switch (*str) {
2345 			case ' ':
2346 			case '\t':
2347 				*str = '_';
2348 				break;
2349 		}
2350 	}
2351 }
2352 
2353 void
2354 pcmcia_1275_name(int socket, struct pcm_device_info *info,
2355 			client_handle_t handle)
2356 {
2357 	cistpl_manfid_t manfid;
2358 	cistpl_jedec_t jedec;
2359 	tuple_t tuple;
2360 	int i;
2361 
2362 	tuple.Socket = socket;
2363 
2364 	/* get MANFID if it exists -- this is most important form */
2365 	tuple.DesiredTuple = CISTPL_MANFID;
2366 	tuple.Attributes = 0;
2367 	if ((i = csx_GetFirstTuple(handle, &tuple)) ==
2368 	    SUCCESS) {
2369 		i = csx_Parse_CISTPL_MANFID(handle, &tuple,
2370 		    &manfid);
2371 		if (i == SUCCESS) {
2372 			(void) sprintf(info->pd_bind_name, "%s%x,%x",
2373 				PCMDEV_NAMEPREF,
2374 				manfid.manf, manfid.card);
2375 			info->pd_flags |= PCM_NAME_1275;
2376 		}
2377 	} else {
2378 		tuple.Attributes = 0;
2379 		tuple.DesiredTuple = CISTPL_JEDEC_A;
2380 		if ((i = csx_GetFirstTuple(handle, &tuple)) ==
2381 		    SUCCESS) {
2382 			i = csx_Parse_CISTPL_JEDEC_A(handle, &tuple,
2383 			    &jedec);
2384 			if (i == SUCCESS) {
2385 				(void) sprintf(info->pd_bind_name, "%s%x,%x",
2386 					PCMDEV_NAMEPREF,
2387 					jedec.jid[0].id, jedec.jid[0].info);
2388 				info->pd_flags |= PCM_NAME_1275;
2389 			}
2390 		}
2391 	}
2392 }
2393 
2394 void
2395 pcmcia_vers1_name(int socket, struct pcm_device_info *info,
2396 			client_handle_t handle)
2397 {
2398 	cistpl_vers_1_t vers1;
2399 	tuple_t tuple;
2400 	int which = 0;
2401 	int i, len, space;
2402 
2403 	tuple.Socket = socket;
2404 	info->pd_vers1_name[0] = '\0';
2405 
2406 	/* Version 1 strings */
2407 	tuple.DesiredTuple = CISTPL_VERS_1;
2408 	tuple.Attributes = 0;
2409 	if (!which &&
2410 	    (i = csx_GetFirstTuple(handle, &tuple)) ==
2411 		SUCCESS) {
2412 		i = csx_Parse_CISTPL_VERS_1(handle, &tuple, &vers1);
2413 		if (i == SUCCESS) {
2414 			for (i = 0, len = 0, space = 0; i < vers1.ns; i++) {
2415 			    if ((space + len + strlen(info->pd_vers1_name)) >=
2416 				sizeof (info->pd_vers1_name))
2417 				    break;
2418 			    if (space) {
2419 				    info->pd_vers1_name[len++] = ',';
2420 			    }
2421 			    (void) strcpy(info->pd_vers1_name + len,
2422 				(char *)vers1.pi[i]);
2423 			    len += strlen((char *)vers1.pi[i]);
2424 			    /* strip trailing spaces off of string */
2425 			    while (info->pd_vers1_name[len - 1] == ' ' &&
2426 				    len > 0)
2427 				    len--;
2428 			    space = 1;
2429 			}
2430 			info->pd_vers1_name[len] = '\0';
2431 			info->pd_flags |= PCM_NAME_VERS1;
2432 		}
2433 	}
2434 }
2435 
2436 
2437 int
2438 pcmcia_get_funce(client_handle_t handle, tuple_t *tuple)
2439 {
2440 	int ret = 0;
2441 
2442 	tuple->Attributes = 0;
2443 	while (csx_GetNextTuple(handle, tuple) == SUCCESS) {
2444 		if (tuple->TupleCode == CISTPL_FUNCID) {
2445 			break;
2446 		}
2447 		if (tuple->TupleCode == CISTPL_FUNCE) {
2448 			ret = 1;
2449 			break;
2450 		}
2451 		tuple->Attributes = 0;
2452 	}
2453 	return (ret);
2454 }
2455 
2456 char *pcmcia_lan_types[] = {
2457 	"arcnet",
2458 	"ethernet",
2459 	"token-ring",
2460 	"localtalk",
2461 	"fddi",
2462 	"atm",
2463 	"wireless",
2464 	"reserved"
2465 };
2466 
2467 void
2468 pcmcia_generic_name(int socket, struct pcm_device_info *info,
2469 			client_handle_t handle)
2470 {
2471 	cistpl_funcid_t funcid;
2472 	cistpl_funce_t funce;
2473 	tuple_t tuple;
2474 	int which = 0;
2475 	int i;
2476 
2477 	tuple.Socket = socket;
2478 
2479 	tuple.DesiredTuple = CISTPL_FUNCID;
2480 	tuple.Attributes = 0;
2481 	if ((i = csx_GetFirstTuple(handle, &tuple)) ==
2482 	    SUCCESS) {
2483 		/*
2484 		 * need to make sure that CISTPL_FUNCID is not
2485 		 * present in both a global and local CIS for MF
2486 		 * cards.  3COM seems to do this erroneously
2487 		 */
2488 
2489 		if (info->pd_flags & PCM_MULTI_FUNCTION &&
2490 		    tuple.Flags & CISTPLF_GLOBAL_CIS) {
2491 			tuple_t ltuple;
2492 			ltuple = tuple;
2493 			ltuple.DesiredTuple = CISTPL_FUNCID;
2494 			ltuple.Attributes = 0;
2495 			if ((i = csx_GetNextTuple(handle, &ltuple)) ==
2496 			    SUCCESS) {
2497 				/* this is the per-function funcid */
2498 				tuple = ltuple;
2499 			}
2500 		}
2501 
2502 		i = csx_Parse_CISTPL_FUNCID(handle, &tuple, &funcid);
2503 		if (i == SUCCESS) {
2504 			/* in case no function extension */
2505 			if (funcid.function < PCM_GENNAME_SIZE)
2506 				(void) strcpy(info->pd_generic_name,
2507 				    pcmcia_generic_names[funcid.function]);
2508 			else
2509 				(void) sprintf(info->pd_generic_name,
2510 					"class,%x",
2511 					funcid.function);
2512 		}
2513 		info->pd_type = funcid.function;
2514 		switch (funcid.function) {
2515 		case TPLFUNC_LAN:
2516 			which = pcmcia_get_funce(handle, &tuple);
2517 			if (which) {
2518 				i = csx_Parse_CISTPL_FUNCE(handle,
2519 				    &tuple,
2520 				    &funce, TPLFUNC_LAN);
2521 				if (i == SUCCESS) {
2522 					i = funce.data.lan.tech;
2523 					if (i > sizeof (pcmcia_lan_types) /
2524 					    sizeof (char *)) {
2525 						break;
2526 					}
2527 					(void) strcpy(info->pd_generic_name,
2528 						pcmcia_lan_types[i]);
2529 				}
2530 			}
2531 			break;
2532 		case TPLFUNC_VIDEO:
2533 #ifdef future_pcmcia_spec
2534 			which = pcmcia_get_funce(handle, &tuple);
2535 			if (which) {
2536 				i = csx_Parse_CISTPL_FUNCE(handle,
2537 				    &tuple,
2538 				    &funce, TPLFUNC_VIDEO);
2539 				if (i == SUCCESS) {
2540 					i = funce.video.tech;
2541 					if (i > sizeof (pcmcia_lan_types) /
2542 					    sizeof (char *)) {
2543 						break;
2544 					}
2545 					(void) strcpy(info->pd_generic_names,
2546 						pcmcia_lan_types[i]);
2547 				}
2548 			}
2549 #endif
2550 			break;
2551 		}
2552 		info->pd_flags |= PCM_NAME_GENERIC;
2553 	} else {
2554 		/* if no FUNCID, do we have CONFIG */
2555 		tuple.DesiredTuple = CISTPL_CONFIG;
2556 		tuple.Attributes = 0;
2557 		if (csx_GetFirstTuple(handle, &tuple) != SUCCESS) {
2558 			info->pd_flags |= PCM_NO_CONFIG | PCM_NAME_GENERIC;
2559 			(void) strcpy(info->pd_generic_name,
2560 				pcmcia_generic_names[PCM_TYPE_MEMORY]);
2561 			info->pd_type = PCM_TYPE_MEMORY;
2562 		}
2563 	}
2564 }
2565 
2566 
2567 /*
2568  * pcmcia_add_compatible()
2569  * add the cached compatible property list.
2570  */
2571 void
2572 pcmcia_add_compatible(dev_info_t *dip, struct pcm_device_info *info)
2573 {
2574 	int length = 0, i;
2575 	char buff[MAXNAMELEN];
2576 	char *compat_name[8];
2577 	int ci = 0;
2578 
2579 	bzero(compat_name, sizeof (compat_name));
2580 
2581 	if (info->pd_flags & PCM_NAME_VERS1) {
2582 		(void) sprintf(buff, "%s,%s", PCMDEV_NAMEPREF,
2583 		    info->pd_vers1_name);
2584 		pcmcia_fix_string(buff); /* don't want spaces */
2585 		length = strlen(buff) + 1;
2586 		compat_name[ci] = kmem_alloc(length, KM_SLEEP);
2587 		(void) strcpy(compat_name[ci++], buff);
2588 	}
2589 
2590 	if ((info->pd_flags & (PCM_NAME_1275 | PCM_MULTI_FUNCTION)) ==
2591 	    (PCM_NAME_1275 | PCM_MULTI_FUNCTION)) {
2592 		(void) sprintf(buff, "%s,%x", info->pd_bind_name,
2593 		    info->pd_function);
2594 		length = strlen(buff) + 1;
2595 		compat_name[ci] = kmem_alloc(length, KM_SLEEP);
2596 		(void) strcpy(compat_name[ci++], buff);
2597 	}
2598 
2599 	if (info->pd_flags & PCM_NAME_1275) {
2600 		length = strlen(info->pd_bind_name) + 1;
2601 		compat_name[ci] = kmem_alloc(length, KM_SLEEP);
2602 		(void) strcpy(compat_name[ci++], info->pd_bind_name);
2603 	}
2604 
2605 	if (info->pd_flags & PCM_NAME_GENERIC) {
2606 		if (strncmp(info->pd_generic_name, "class,", 6) == 0) {
2607 			/* no generic without "pccard" */
2608 			(void) sprintf(buff, "%s%s", PCMDEV_NAMEPREF,
2609 				info->pd_generic_name);
2610 		} else {
2611 			/* first pccard,generic-name */
2612 			(void) sprintf(buff, "%s,%s", PCMDEV_NAMEPREF,
2613 				info->pd_generic_name);
2614 		}
2615 		length = strlen(buff) + 1;
2616 		compat_name[ci] = kmem_alloc(length, KM_SLEEP);
2617 		(void) strcpy(compat_name[ci++], buff);
2618 
2619 		/* now the simple generic name */
2620 		length = strlen(info->pd_generic_name) + 1;
2621 		compat_name[ci] = kmem_alloc(length, KM_SLEEP);
2622 		(void) strcpy(compat_name[ci++], info->pd_generic_name);
2623 	}
2624 
2625 	if (info->pd_flags & PCM_NO_CONFIG) {
2626 		char *mem = "pccard,memory";
2627 		/*
2628 		 * I/O cards are required to have a config tuple.
2629 		 * there are some that violate the spec and don't
2630 		 * but it is most likely that this is a memory card
2631 		 * so tag it as such.  "memory" is more general
2632 		 * than other things so needs to come last.
2633 		 */
2634 		length = strlen(mem) + 1;
2635 		compat_name[ci] = kmem_alloc(length, KM_SLEEP);
2636 		(void) strcpy(compat_name[ci++], mem);
2637 	}
2638 
2639 	if (ci == 0)
2640 		return;
2641 
2642 	if (ndi_prop_update_string_array(DDI_DEV_T_NONE, dip,
2643 	    "compatible", (char **)compat_name, ci) != DDI_PROP_SUCCESS)
2644 		cmn_err(CE_WARN, "pcmcia: unable to create compatible prop");
2645 
2646 	for (i = 0; i < ci; i++)
2647 		kmem_free(compat_name[i], strlen(compat_name[i]) + 1);
2648 }
2649 /*
2650  * CIS parsing and other PC Card specific code
2651  */
2652 
2653 /*
2654  * pcmcia_get_mem_regs()
2655  */
2656 static int
2657 pcmcia_get_mem_regs(struct pcm_regs *regs, struct pcm_device_info *info,
2658 			int type, int pctype)
2659 {
2660 	int num_regs = 0;
2661 	tuple_t tuple;
2662 	cistpl_device_t device;
2663 	uint32_t curr_base;
2664 	int ret, len;
2665 	int space;
2666 
2667 	/*
2668 	 * current plan for reg spec:
2669 	 * device_a will be accumulated to determine max size of
2670 	 * attribute memory.  device for common.  Then config
2671 	 * tuples to get a worst case I/O size.
2672 	 */
2673 	bzero(&tuple, sizeof (tuple));
2674 	tuple.Socket = info->pd_socket;
2675 
2676 	tuple.DesiredTuple = (cisdata_t)type;
2677 
2678 	space = (type == CISTPL_DEVICE_A) ? PC_REG_SPACE_ATTRIBUTE :
2679 		PC_REG_SPACE_MEMORY;
2680 	if ((ret = csx_GetFirstTuple(info->pd_handle, &tuple)) == CS_SUCCESS) {
2681 		bzero(&device, sizeof (device));
2682 
2683 		if (type == CISTPL_DEVICE)
2684 			ret = csx_Parse_CISTPL_DEVICE(info->pd_handle, &tuple,
2685 			    &device);
2686 		else
2687 			ret = csx_Parse_CISTPL_DEVICE_A(info->pd_handle, &tuple,
2688 			    &device);
2689 
2690 		if (ret == CS_SUCCESS) {
2691 		    curr_base = 0;
2692 		    for (ret = 0; ret < device.num_devices;
2693 			ret++) {
2694 			    /* need to order these for real mem first */
2695 			    if (device.devnode[ret].type !=
2696 				CISTPL_DEVICE_DTYPE_NULL) {
2697 				    /* how to represent types??? */
2698 				    regs[num_regs].phys_hi =
2699 					PC_REG_PHYS_HI(0, 0,
2700 					    pctype,
2701 					    space,
2702 					    info->pd_socket,
2703 					    info->pd_function,
2704 					    0);
2705 				    regs[num_regs].phys_lo = curr_base;
2706 				    len = device.devnode[ret].size_in_bytes;
2707 				    curr_base += len;
2708 				    regs[num_regs].phys_len = len;
2709 				    num_regs++;
2710 			    } else {
2711 				/*
2712 				 * NULL device is a "hole"
2713 				 */
2714 				    curr_base +=
2715 					    device.devnode[ret].size_in_bytes;
2716 			    }
2717 			}
2718 	    }
2719 	}
2720 	return (num_regs);
2721 }
2722 
2723 /*
2724  *
2725  */
2726 static int
2727 pcmcia_get_io_regs(struct pcm_regs *regs, struct pcm_device_info *info,
2728 		    int pctype)
2729 {
2730 	int num_regs = 0;
2731 	tuple_t tuple;
2732 	uint32_t curr_base;
2733 	int len, curr, i, curr_len;
2734 	cistpl_config_t config;
2735 	cistpl_cftable_entry_t cftable;
2736 	struct pcm_regs tmp[16];
2737 	int found = 0;
2738 
2739 	bzero(&tuple, sizeof (tuple));
2740 	tuple.DesiredTuple = CISTPL_CONFIG;
2741 	tuple.Socket = info->pd_socket;
2742 	tuple.Attributes = 0;
2743 	curr_base = 0;
2744 	len = 0;
2745 
2746 	if (csx_GetFirstTuple(info->pd_handle, &tuple) == CS_SUCCESS) {
2747 	    if (csx_Parse_CISTPL_CONFIG(info->pd_handle,
2748 					&tuple, &config) != CS_SUCCESS) {
2749 		info->pd_flags |= PCM_NO_CONFIG; /* must be memory */
2750 		return (0);
2751 	}
2752 	curr = 0;
2753 
2754 	tuple.DesiredTuple = CISTPL_CFTABLE_ENTRY;
2755 	tuple.Socket = info->pd_socket;
2756 	tuple.Attributes = 0;
2757 	bzero(tmp, sizeof (tmp));
2758 	while (csx_GetNextTuple(info->pd_handle,
2759 				&tuple) == CS_SUCCESS) {
2760 	    bzero(&cftable, sizeof (cftable));
2761 	    if (csx_Parse_CISTPL_CFTABLE_ENTRY(info->pd_handle,
2762 						&tuple, &cftable) ==
2763 		CS_SUCCESS) {
2764 		if (cftable.flags & CISTPL_CFTABLE_TPCE_FS_IO) {
2765 		    /* we have an I/O entry */
2766 		    if (cftable.io.flags &
2767 			CISTPL_CFTABLE_TPCE_FS_IO_RANGE) {
2768 			len = cftable.io.addr_lines;
2769 			if (len != 0)
2770 				len = 1 << len;
2771 			for (i = 0; i < cftable.io.ranges && curr < 16; i++) {
2772 			    curr_base = cftable.io.range[i].addr;
2773 			    curr_len = cftable.io.range[i].length;
2774 			    if (curr_len == 0)
2775 				    curr_len = len;
2776 			    if (len != 0 || cftable.io.addr_lines == 0) {
2777 				/* we have potential relocation */
2778 				int mask;
2779 				mask = cftable.io.addr_lines ?
2780 						cftable.io.addr_lines :
2781 							genp2(len);
2782 				mask = genmask(mask);
2783 				if ((mask & curr_base) == 0) {
2784 					/* more accurate length */
2785 					regs->phys_len = curr_len;
2786 					regs->phys_lo = 0;
2787 					regs->phys_hi =
2788 					    PC_REG_PHYS_HI(0,
2789 						0,
2790 						pctype,
2791 						PC_REG_SPACE_IO,
2792 						info->pd_socket,
2793 						info->pd_function,
2794 						0);
2795 					num_regs++;
2796 					found = 2;
2797 					break;
2798 				}
2799 			    }
2800 			    tmp[curr].phys_len = curr_len;
2801 			    tmp[curr].phys_lo = curr_base;
2802 			    curr++;
2803 			    found = 1;
2804 			}
2805 			if (found == 2)
2806 				break;
2807 		    } else {
2808 			/* no I/O range so just a mask */
2809 			regs->phys_len = 1 << cftable.io.addr_lines;
2810 			regs->phys_hi =
2811 				PC_REG_PHYS_HI(0,
2812 						0,
2813 						pctype,
2814 						PC_REG_SPACE_IO,
2815 						info->pd_socket,
2816 						info->pd_function,
2817 						0);
2818 			regs->phys_lo = 0;
2819 			num_regs++;
2820 			regs++;
2821 			/* quit on "good" entry */
2822 			break;
2823 		    }
2824 		    /* was this the last CFTABLE Entry? */
2825 		    if (config.last == cftable.index)
2826 			    break;
2827 		}
2828 	    }
2829 	}
2830 	if (found == 1) {
2831 		/*
2832 		 * have some non-relocatable values
2833 		 * so we include them all for now
2834 		 */
2835 		for (i = 0; i < curr && num_regs < 8; i++) {
2836 		    regs->phys_len = tmp[i].phys_len;
2837 		    regs->phys_lo = tmp[i].phys_lo;
2838 		    regs->phys_hi = PC_REG_PHYS_HI(1, 0, pctype,
2839 			    PC_REG_SPACE_IO, info->pd_socket,
2840 			    info->pd_function, 0);
2841 		    regs++;
2842 		    num_regs++;
2843 		}
2844 	    }
2845 	}
2846 	return (num_regs);
2847 }
2848 
2849 /*
2850  * pcmcia_create_regs()
2851  *	create a valid set of regspecs for the card
2852  *	The first one is always for CIS access and naming
2853  */
2854 /*ARGSUSED*/
2855 static void
2856 pcmcia_find_regs(dev_info_t *dip, struct pcm_device_info *info,
2857 			struct pcmcia_parent_private *ppd)
2858 {
2859 	struct pcm_regs regs[32]; /* assume worst case */
2860 	int num_regs = 0;
2861 	int len;
2862 	int bustype;
2863 
2864 	if (ppd->ppd_flags & PPD_CARD_CARDBUS) {
2865 		/* always have a CIS map */
2866 		regs[0].phys_hi = PC_REG_PHYS_HI(0, 0, PC_REG_TYPE_CARDBUS,
2867 						PC_REG_SPACE_CONFIG,
2868 						info->pd_socket,
2869 						info->pd_function, 0);
2870 		bustype = PC_REG_TYPE_CARDBUS;
2871 	} else {
2872 		/* always have a CIS map */
2873 		regs[0].phys_hi = PC_REG_PHYS_HI(0, 0, PC_REG_TYPE_16BIT,
2874 						PC_REG_SPACE_ATTRIBUTE,
2875 						info->pd_socket,
2876 						info->pd_function, 0);
2877 		bustype = PC_REG_TYPE_16BIT;
2878 	}
2879 	regs[0].phys_lo = 0;	/* always starts at zero */
2880 	regs[0].phys_len = 0;
2881 	num_regs++;
2882 	/*
2883 	 * need to search CIS for other memory instances
2884 	 */
2885 
2886 	if (info->pd_flags & PCM_OTHER_NOCIS) {
2887 		/* special case of memory only card without CIS */
2888 		regs[1].phys_hi = PC_REG_PHYS_HI(0, 0, PC_REG_TYPE_16BIT,
2889 						PC_REG_SPACE_MEMORY,
2890 						info->pd_socket,
2891 						info->pd_function, 0);
2892 		regs[1].phys_lo = 0;
2893 		regs[1].phys_len = PCM_MAX_R2_MEM;
2894 		num_regs++;
2895 	} else {
2896 		/*
2897 		 * want to get any other memory and/or I/O regions
2898 		 * on the card and represent them here.
2899 		 */
2900 		num_regs += pcmcia_get_mem_regs(&regs[num_regs], info,
2901 						CISTPL_DEVICE_A, bustype);
2902 		num_regs += pcmcia_get_mem_regs(&regs[num_regs], info,
2903 						CISTPL_DEVICE, bustype);
2904 
2905 		/* now look for an I/O space to configure */
2906 		num_regs += pcmcia_get_io_regs(&regs[num_regs], info,
2907 						bustype);
2908 
2909 	}
2910 
2911 	len = num_regs * sizeof (uint32_t) * 3;
2912 	ppd->ppd_nreg = num_regs;
2913 	ppd->ppd_reg = kmem_alloc(len, KM_SLEEP);
2914 	bcopy(regs, ppd->ppd_reg, len);
2915 	len = sizeof (struct pcm_regs) * ppd->ppd_nreg;
2916 	ppd->ppd_assigned = kmem_zalloc(len, KM_SLEEP);
2917 }
2918 
2919 
2920 /*
2921  * pcmcia_need_intr()
2922  *	check to see if an interrupt tuple exists.
2923  *	existence means we need one in the intrspec.
2924  */
2925 static int
2926 pcmcia_need_intr(int socket, struct pcm_device_info *info)
2927 {
2928 	cistpl_config_t config;
2929 	cistpl_cftable_entry_t cftable;
2930 	tuple_t tuple;
2931 	int i;
2932 
2933 	bzero(&tuple, sizeof (tuple));
2934 	tuple.DesiredTuple = CISTPL_CONFIG;
2935 	tuple.Socket = socket;
2936 	tuple.Attributes = 0;
2937 	if (csx_GetFirstTuple(info->pd_handle, &tuple) != CS_SUCCESS) {
2938 		return (0);
2939 	}
2940 #if defined(PCMCIA_DEBUG)
2941 	if (pcmcia_debug) {
2942 		cmn_err(CE_CONT, "pcmcia_need_intr: have config tuple\n");
2943 	}
2944 #endif
2945 	bzero(&config, sizeof (config));
2946 	if (csx_Parse_CISTPL_CONFIG(info->pd_handle,
2947 	    &tuple, &config) != CS_SUCCESS) {
2948 		cmn_err(CE_WARN, "pcmcia: config failed to parse\n");
2949 		return (0);
2950 	}
2951 
2952 	for (cftable.index = (int)-1, i = -1;
2953 		i != config.last; i = cftable.index) {
2954 		tuple.DesiredTuple = CISTPL_CFTABLE_ENTRY;
2955 		tuple.Attributes = 0;
2956 		if (csx_GetNextTuple(info->pd_handle,
2957 		    &tuple) != CS_SUCCESS) {
2958 			cmn_err(CE_WARN, "pcmcia: get cftable failed\n");
2959 			break;
2960 		}
2961 		bzero(&cftable, sizeof (cftable));
2962 		if (csx_Parse_CISTPL_CFTABLE_ENTRY(info->pd_handle,
2963 		    &tuple, &cftable) !=
2964 		    CS_SUCCESS) {
2965 			cmn_err(CE_WARN, "pcmcia: parse cftable failed\n");
2966 			break;
2967 		}
2968 #if defined(PCMCIA_DEBUG)
2969 		if (pcmcia_debug)
2970 			cmn_err(CE_CONT, "\t%x: flags=%x (%x)\n",
2971 				i, cftable.flags,
2972 				cftable.flags & CISTPL_CFTABLE_TPCE_FS_IRQ);
2973 #endif
2974 		if (cftable.flags & CISTPL_CFTABLE_TPCE_FS_IRQ)
2975 			return (1);
2976 	}
2977 	return (0);
2978 
2979 }
2980 
2981 /*
2982  * pcmcia_num_funcs()
2983  *	look for a CISTPL_LONGLINK_MFC
2984  *	if there is one, return the number of functions
2985  *	if there isn't one, then there is one function
2986  */
2987 static int
2988 pcmcia_num_funcs(int socket, client_handle_t handle)
2989 {
2990 	int count = 1;
2991 	cistpl_longlink_mfc_t mfc;
2992 	tuple_t tuple;
2993 
2994 	bzero(&tuple, sizeof (tuple_t));
2995 	tuple.DesiredTuple = CISTPL_LONGLINK_MFC;
2996 	tuple.Socket = socket;
2997 	tuple.Attributes = 0;
2998 	if (csx_GetFirstTuple(handle, &tuple) == CS_SUCCESS) {
2999 		/* this is a multifunction card */
3000 		if (csx_ParseTuple(handle, &tuple, (cisparse_t *)&mfc,
3001 		    CISTPL_LONGLINK_MFC) == CS_SUCCESS) {
3002 			count = mfc.nfuncs;
3003 		}
3004 	}
3005 	return (count);
3006 }
3007 
3008 client_handle_t pcmcia_cs_handle;
3009 
3010 /*
3011  * pcmcia_create_dev_info(socket)
3012  *	either find or create the device information structure
3013  *	for the card(s) just inserted.	We don't care about removal yet.
3014  *	In any case, we will only do this at CS request
3015  */
3016 static void
3017 pcmcia_create_dev_info(int socket)
3018 {
3019 	struct pcm_device_info card_info;
3020 	client_reg_t reg;
3021 	cisinfo_t cisinfo;
3022 	int i;
3023 	dev_info_t *pdip;
3024 	static int handle_def = 0;
3025 
3026 #if defined(PCMCIA_DEBUG)
3027 	if (pcmcia_debug)
3028 		cmn_err(CE_CONT, "create dev_info_t for device in socket %d\n",
3029 			socket);
3030 #endif
3031 
3032 	/*
3033 	 * before we can do anything else, we need the parent
3034 	 * devinfo of the socket.  This gets things in the right
3035 	 * place in the device tree.
3036 	 */
3037 
3038 	pdip = pcm_find_parent_dip(socket);
3039 	if (pdip == NULL)
3040 		return;
3041 
3042 	/* Card Services calls needed to get CIS info */
3043 	reg.dip = NULL;
3044 	reg.Attributes = INFO_SOCKET_SERVICES;
3045 	reg.EventMask = 0;
3046 	reg.event_handler = NULL;
3047 	reg.Version = CS_VERSION;
3048 
3049 	bzero(&card_info, sizeof (card_info));
3050 
3051 	if (handle_def == 0) {
3052 		if (csx_RegisterClient(&pcmcia_cs_handle,
3053 		    &reg) != CS_SUCCESS) {
3054 #if defined(PCMCIA_DEBUG)
3055 			if (pcmcia_debug)
3056 				cmn_err(CE_CONT,
3057 					"pcmcia: RegisterClient failed\n");
3058 #endif
3059 			return;
3060 		}
3061 		handle_def++;
3062 	}
3063 	card_info.pd_handle = pcmcia_cs_handle;
3064 
3065 #if defined(PCMCIA_DEBUG)
3066 	if (pcmcia_debug)
3067 		cmn_err(CE_CONT,
3068 			"pcmcia_create_dev_info: handle = %x\n",
3069 			(int)card_info.pd_handle);
3070 #endif
3071 	card_info.pd_type = -1; /* no type to start */
3072 	card_info.pd_socket = socket;
3073 	card_info.pd_function = 0;
3074 	pcmcia_sockets[socket]->ls_functions = 1; /* default */
3075 
3076 	cisinfo.Socket = socket;
3077 
3078 	if ((i = csx_ValidateCIS(card_info.pd_handle,
3079 	    &cisinfo)) != SUCCESS ||
3080 	    cisinfo.Tuples == 0) {
3081 		/* no CIS means memory */
3082 		(void) strcpy(card_info.pd_generic_name, "memory");
3083 		card_info.pd_flags |= PCM_NAME_GENERIC |
3084 			PCM_OTHER_NOCIS | PCM_NAME_1275;
3085 		(void) strcpy(card_info.pd_bind_name, "pccard,memory");
3086 		(void) strcpy(card_info.pd_generic_name, "memory");
3087 		card_info.pd_type = PCM_TYPE_MEMORY;
3088 	} else {
3089 		int functions, lsocket;
3090 		card_info.pd_tuples = cisinfo.Tuples;
3091 
3092 		/*
3093 		 * how many functions on the card?
3094 		 * we need to know and then we do one
3095 		 * child node for each function using
3096 		 * the function specific tuples.
3097 		 */
3098 		lsocket = CS_MAKE_SOCKET_NUMBER(socket, CS_GLOBAL_CIS);
3099 		functions = pcmcia_num_funcs(lsocket,
3100 		    card_info.pd_handle);
3101 		pcmcia_sockets[socket]->ls_functions = functions;
3102 		if (functions > 1) {
3103 			card_info.pd_flags |= PCM_MULTI_FUNCTION;
3104 		}
3105 		for (i = 0; i < functions; i++) {
3106 		    register int flags;
3107 		    lsocket = CS_MAKE_SOCKET_NUMBER(socket, i);
3108 		    card_info.pd_socket = socket;
3109 		    card_info.pd_function = i;
3110 			/*
3111 			 * new name construction
3112 			 */
3113 		    if (functions != 1) {
3114 			    /* need per function handle */
3115 			    card_info.pd_function = i;
3116 			    /* get new handle */
3117 		    }
3118 		    pcmcia_1275_name(lsocket, &card_info,
3119 			card_info.pd_handle);
3120 		    pcmcia_vers1_name(lsocket, &card_info,
3121 			card_info.pd_handle);
3122 		    pcmcia_generic_name(lsocket, &card_info,
3123 			card_info.pd_handle);
3124 		    flags = card_info.pd_flags;
3125 		    if (!(flags & PCM_NAME_1275)) {
3126 			if (flags & PCM_NAME_VERS1) {
3127 			    (void) strcpy(card_info.pd_bind_name,
3128 				PCMDEV_NAMEPREF);
3129 			    card_info.pd_bind_name[sizeof (PCMDEV_NAMEPREF)] =
3130 				',';
3131 			    (void) strncpy(card_info.pd_bind_name +
3132 				sizeof (PCMDEV_NAMEPREF),
3133 				card_info.pd_vers1_name,
3134 				MODMAXNAMELEN -
3135 				sizeof (PCMDEV_NAMEPREF));
3136 			    pcmcia_fix_string(card_info.pd_bind_name);
3137 			} else {
3138 				/*
3139 				 * have a CIS but not the right info
3140 				 * so treat as generic "pccard"
3141 				 */
3142 				(void) strcpy(card_info.pd_generic_name,
3143 					"pccard,memory");
3144 				card_info.pd_flags |= PCM_NAME_GENERIC;
3145 				(void) strcpy(card_info.pd_bind_name,
3146 					"pccard,memory");
3147 			}
3148 		    }
3149 		    pcmcia_init_devinfo(pdip, &card_info);
3150 		}
3151 		return;
3152 	}
3153 
3154 	pcmcia_init_devinfo(pdip, &card_info);
3155 }
3156 
3157 /*
3158  * pcmcia_init_devinfo()
3159  *	if there isn't a device info structure, create one
3160  *	if there is, we don't do much.
3161  *
3162  *	Note: this will need updating as 1275 finalizes their spec.
3163  */
3164 static void
3165 pcmcia_init_devinfo(dev_info_t *pdip, struct pcm_device_info *info)
3166 {
3167 	int unit;
3168 	dev_info_t *dip;
3169 	char *name;
3170 	struct pcmcia_parent_private *ppd;
3171 
3172 #if defined(PCMCIA_DEBUG)
3173 	if (pcmcia_debug)
3174 		cmn_err(CE_CONT, "init_devinfo(%s, %d)\n", info->pd_bind_name,
3175 			info->pd_socket);
3176 #endif
3177 
3178 	/*
3179 	 * find out if there is already an instance of this
3180 	 * device.  We don't want to create a new one unnecessarily
3181 	 */
3182 	unit = CS_MAKE_SOCKET_NUMBER(info->pd_socket, info->pd_function);
3183 
3184 	dip = pcm_find_devinfo(pdip, info, unit);
3185 	if ((dip != NULL) && (ddi_getprop(DDI_DEV_T_NONE, dip,
3186 	    DDI_PROP_DONTPASS, PCM_DEV_SOCKET, -1) != -1)) {
3187 		/* it already exist but isn't a .conf file */
3188 
3189 #if defined(PCMCIA_DEBUG)
3190 		if (pcmcia_debug)
3191 			cmn_err(CE_CONT, "\tfound existing device node (%s)\n",
3192 				ddi_get_name(dip));
3193 #endif
3194 		if (strlen(info->pd_vers1_name) > 0)
3195 			(void) ndi_prop_update_string(DDI_DEV_T_NONE,
3196 			    dip, PCM_DEV_MODEL, info->pd_vers1_name);
3197 
3198 		ppd = (struct pcmcia_parent_private *)
3199 			ddi_get_parent_data(dip);
3200 
3201 		pcmcia_sockets[info->pd_socket]->ls_dip[info->pd_function] =
3202 		    dip;
3203 
3204 		ppd->ppd_active = 1;
3205 
3206 		if (ndi_devi_online(dip, 0) == NDI_FAILURE) {
3207 			pcmcia_sockets[info->pd_socket]-> \
3208 			    ls_dip[info->pd_function] = NULL;
3209 			ppd->ppd_active = 0;
3210 		}
3211 	} else {
3212 
3213 		char *dtype;
3214 
3215 #if defined(PCMCIA_DEBUG)
3216 		if (pcmcia_debug)
3217 			cmn_err(CE_CONT, "pcmcia: create child [%s](%d): %s\n",
3218 			    info->pd_bind_name, info->pd_socket,
3219 			    info->pd_generic_name);
3220 #endif
3221 
3222 		if (info->pd_flags & PCM_NAME_GENERIC)
3223 			name = info->pd_generic_name;
3224 		else
3225 			name = info->pd_bind_name;
3226 
3227 		if (ndi_devi_alloc(pdip, name, (dnode_t)DEVI_SID_NODEID,
3228 		    &dip) !=
3229 		    NDI_SUCCESS) {
3230 			cmn_err(CE_WARN,
3231 			    "pcmcia: unable to create device [%s](%d)\n",
3232 			    name, info->pd_socket);
3233 			return;
3234 		}
3235 		/*
3236 		 * construct the "compatible" property if the device
3237 		 * has a generic name
3238 		 */
3239 		pcmcia_add_compatible(dip, info);
3240 
3241 		ppd = kmem_zalloc(sizeof (struct pcmcia_parent_private),
3242 			    KM_SLEEP);
3243 
3244 		ppd->ppd_socket = info->pd_socket;
3245 		ppd->ppd_function = info->pd_function;
3246 
3247 		/*
3248 		 * add the "socket" property
3249 		 * the value of this property contains the logical PCMCIA
3250 		 * socket number the device has been inserted in, along
3251 		 * with the function # if the device is part of a
3252 		 * multi-function device.
3253 		 */
3254 		(void) ndi_prop_update_int(DDI_DEV_T_NONE, dip,
3255 		    PCM_DEV_SOCKET, unit);
3256 
3257 		if (info->pd_flags & PCM_MULTI_FUNCTION)
3258 			ppd->ppd_flags |= PPD_CARD_MULTI;
3259 
3260 		/*
3261 		 * determine all the properties we need for PPD
3262 		 * then create the properties
3263 		 */
3264 		/* socket is unique */
3265 		pcmcia_find_regs(dip, info, ppd);
3266 
3267 		ppd->ppd_intr = pcmcia_need_intr(unit, info);
3268 
3269 		if (ppd->ppd_nreg > 0)
3270 			(void) ddi_prop_update_int_array(DDI_DEV_T_NONE, dip,
3271 			    "reg", (int *)ppd->ppd_reg, ppd->ppd_nreg *
3272 			    sizeof (struct pcm_regs) / sizeof (int));
3273 		if (ppd->ppd_intr) {
3274 			(void) ddi_prop_update_int(DDI_DEV_T_NONE, dip,
3275 			    "interrupts", ppd->ppd_intr);
3276 			ppd->ppd_intrspec =
3277 			    kmem_zalloc(sizeof (struct intrspec), KM_SLEEP);
3278 		}
3279 
3280 		/* set parent private - our own format */
3281 		ddi_set_parent_data(dip, (caddr_t)ppd);
3282 
3283 		/* init the device type */
3284 		if (info->pd_type >= 0 &&
3285 		    info->pd_type < (sizeof (pcmcia_dev_type) /
3286 		    (sizeof (char *))))
3287 			dtype = pcmcia_dev_type[info->pd_type];
3288 		else
3289 			dtype = "unknown";
3290 
3291 		if (strlen(info->pd_vers1_name) > 0)
3292 			(void) ndi_prop_update_string(DDI_DEV_T_NONE,
3293 			    dip, PCM_DEV_MODEL, info->pd_vers1_name);
3294 
3295 		(void) ndi_prop_update_string(DDI_DEV_T_NONE, dip,
3296 		    PCM_DEVICETYPE, dtype);
3297 
3298 		/* set PC Card as active and present in socket */
3299 		pcmcia_sockets[info->pd_socket]->ls_dip[info->pd_function] =
3300 			dip;
3301 
3302 		ppd->ppd_active = 1;
3303 
3304 		/*
3305 		 * We should not call ndi_devi_online here if
3306 		 * pcmcia attach is in progress. This causes a deadlock.
3307 		 */
3308 		if (pcmcia_dip != dip) {
3309 			if (ndi_devi_online_async(dip, 0)
3310 			    != NDI_SUCCESS) {
3311 				pcmcia_sockets[info->pd_socket]->\
3312 				ls_dip[info->pd_function] = NULL;
3313 				pcmcia_ppd_free(ppd);
3314 				(void) ndi_devi_free(dip);
3315 				return;
3316 			}
3317 		}
3318 
3319 #if defined(PCMCIA_DEBUG)
3320 	if (pcmcia_debug)
3321 		cmn_err(CE_CONT, "\tjust added \"active\" to %s in %d\n",
3322 			ddi_get_name(dip), info->pd_socket);
3323 #endif
3324 	}
3325 
3326 	/*
3327 	 * inform the event manager that a child was added
3328 	 * to the device tree.
3329 	 */
3330 	pcm_event_manager(PCE_DEV_IDENT, unit, ddi_get_name(dip));
3331 
3332 #if defined(PCMCIA_DEBUG)
3333 	if (pcmcia_debug > 1) {
3334 		pcmcia_dump_minors(dip);
3335 	}
3336 #endif
3337 }
3338 
3339 /*
3340  * free any allocated parent-private data
3341  */
3342 static void
3343 pcmcia_ppd_free(struct pcmcia_parent_private *ppd)
3344 {
3345 	size_t len;
3346 
3347 	if (ppd->ppd_nreg != 0) {
3348 		len = ppd->ppd_nreg * sizeof (uint32_t) * 3;
3349 		kmem_free(ppd->ppd_reg, len);
3350 		len = sizeof (struct pcm_regs) * ppd->ppd_nreg;
3351 		kmem_free(ppd->ppd_assigned, len);
3352 	}
3353 
3354 	/*
3355 	 * pcmcia only allocates 1 intrspec today
3356 	 */
3357 	if (ppd->ppd_intr != 0) {
3358 		len = sizeof (struct intrspec) * ppd->ppd_intr;
3359 		kmem_free(ppd->ppd_intrspec, len);
3360 	}
3361 
3362 	kmem_free(ppd, sizeof (*ppd));
3363 }
3364 
3365 
3366 /*
3367  * pcmcia_get_devinfo(socket)
3368  *	entry point to allow finding the device info structure
3369  *	for a given logical socket.  Used by event manager
3370  */
3371 dev_info_t *
3372 pcmcia_get_devinfo(int socket)
3373 {
3374 	int func = CS_GET_FUNCTION_NUMBER(socket);
3375 	socket = CS_GET_SOCKET_NUMBER(socket);
3376 	if (pcmcia_sockets[socket])
3377 		return (pcmcia_sockets[socket]->ls_dip[func]);
3378 	return ((dev_info_t *)NULL);
3379 }
3380 
3381 /*
3382  * CSGetCookiesAndDip()
3383  *	get info needed by CS to setup soft interrupt handler and provide
3384  *		socket-specific adapter information
3385  */
3386 static int
3387 GetCookiesAndDip(sservice_t *serv)
3388 {
3389 	pcmcia_logical_socket_t *socket;
3390 	csss_adapter_info_t *ai;
3391 	int sock;
3392 
3393 	sock = CS_GET_SOCKET_NUMBER(serv->get_cookies.socket);
3394 
3395 	if (sock >= pcmcia_num_sockets ||
3396 	    (int)serv->get_cookies.socket < 0)
3397 		return (BAD_SOCKET);
3398 
3399 	socket = pcmcia_sockets[sock];
3400 	ai = &serv->get_cookies.adapter_info;
3401 	serv->get_cookies.dip = socket->ls_adapter->pca_dip;
3402 	serv->get_cookies.iblock = socket->ls_adapter->pca_iblock;
3403 	serv->get_cookies.idevice = socket->ls_adapter->pca_idev;
3404 
3405 	/*
3406 	 * Setup the adapter info for Card Services
3407 	 */
3408 	(void) strcpy(ai->name, socket->ls_adapter->pca_name);
3409 	ai->major = socket->ls_adapter->pca_module;
3410 	ai->minor = socket->ls_adapter->pca_unit;
3411 	ai->number = socket->ls_adapter->pca_number;
3412 	ai->num_sockets = socket->ls_adapter->pca_numsockets;
3413 	ai->first_socket = socket->ls_adapter->pca_first_socket;
3414 
3415 	return (SUCCESS);
3416 }
3417 
3418 /*
3419  * Note:
3420  *	The following functions that start with 'SS'
3421  *	implement SocketServices interfaces.  They
3422  *	simply map the socket and/or window number to
3423  *	the adapter specific number based on the general
3424  *	value that CardServices uses.
3425  *
3426  *	See the descriptions in SocketServices for
3427  *	details.  Also refer to specific adapter drivers
3428  *	for implementation reference.
3429  */
3430 
3431 static int
3432 SSGetAdapter(get_adapter_t *adapter)
3433 {
3434 	int n;
3435 	get_adapter_t info;
3436 
3437 	adapter->state = (unsigned)0xFFFFFFFF;
3438 	adapter->SCRouting = 0xFFFFFFFF;
3439 
3440 	for (n = 0; n < pcmcia_num_adapters; n++) {
3441 		GET_ADAPTER(pcmcia_adapters[n]->pca_if,
3442 			pcmcia_adapters[n]->pca_dip, &info);
3443 		adapter->state &= info.state;
3444 		adapter->SCRouting &= info.SCRouting;
3445 	}
3446 
3447 	return (SUCCESS);
3448 }
3449 
3450 static int
3451 SSGetPage(get_page_t *page)
3452 {
3453 	pcmcia_logical_window_t *window;
3454 	get_page_t newpage;
3455 	int retval, win;
3456 
3457 	if (page->window > pcmcia_num_windows) {
3458 		return (BAD_WINDOW);
3459 	}
3460 
3461 	window = pcmcia_windows[page->window];
3462 	newpage = *page;
3463 	win = newpage.window = window->lw_window; /* real window */
3464 
3465 	retval = GET_PAGE(window->lw_if, window->lw_adapter->pca_dip,
3466 		&newpage);
3467 	if (retval == SUCCESS) {
3468 		*page = newpage;
3469 		page->window = win;
3470 	}
3471 	return (retval);
3472 }
3473 
3474 static int
3475 SSGetSocket(get_socket_t *socket)
3476 {
3477 	int retval, sock;
3478 	get_socket_t newsocket;
3479 	pcmcia_logical_socket_t *sockp;
3480 
3481 	sock = socket->socket;
3482 	if (sock > pcmcia_num_sockets ||
3483 		(sockp = pcmcia_sockets[sock]) == NULL) {
3484 		return (BAD_SOCKET);
3485 	}
3486 
3487 	newsocket = *socket;
3488 	newsocket.socket = sockp->ls_socket;
3489 	retval = GET_SOCKET(sockp->ls_if, sockp->ls_adapter->pca_dip,
3490 	    &newsocket);
3491 	if (retval == SUCCESS) {
3492 		newsocket.VccLevel = pcmcia_map_power_get(sockp->ls_adapter,
3493 		    newsocket.VccLevel,
3494 		    VCC);
3495 		newsocket.Vpp1Level = pcmcia_map_power_get(sockp->ls_adapter,
3496 		    newsocket.Vpp1Level,
3497 		    VPP1);
3498 		newsocket.Vpp2Level = pcmcia_map_power_get(sockp->ls_adapter,
3499 		    newsocket.Vpp2Level,
3500 		    VPP2);
3501 		*socket = newsocket;
3502 		socket->socket = sock;
3503 	}
3504 
3505 	return (retval);
3506 }
3507 
3508 static int
3509 SSGetStatus(get_ss_status_t *status)
3510 {
3511 	get_ss_status_t newstat;
3512 	int sock, retval;
3513 	pcmcia_logical_socket_t *sockp;
3514 
3515 	sock = status->socket;
3516 	if (sock > pcmcia_num_sockets ||
3517 		(sockp = pcmcia_sockets[sock]) == NULL) {
3518 		return (BAD_SOCKET);
3519 	}
3520 
3521 	newstat = *status;
3522 	newstat.socket = sockp->ls_socket;
3523 	retval = GET_STATUS(sockp->ls_if, sockp->ls_adapter->pca_dip,
3524 	    &newstat);
3525 	if (retval == SUCCESS) {
3526 		*status = newstat;
3527 		status->socket = sock;
3528 	}
3529 
3530 	return (retval);
3531 }
3532 
3533 static int
3534 SSGetWindow(get_window_t *window)
3535 {
3536 	int win, retval;
3537 	get_window_t newwin;
3538 	pcmcia_logical_window_t *winp;
3539 
3540 	win = window->window;
3541 	winp = pcmcia_windows[win];
3542 	newwin = *window;
3543 	newwin.window = winp->lw_window;
3544 
3545 	retval = GET_WINDOW(winp->lw_if, winp->lw_adapter->pca_dip,
3546 	    &newwin);
3547 	if (retval == SUCCESS) {
3548 		newwin.socket = winp->lw_socket;
3549 		newwin.window = win;
3550 		*window = newwin;
3551 	}
3552 	return (retval);
3553 }
3554 
3555 /*
3556  * SSInquireAdapter()
3557  *	Get the capabilities of the "generic" adapter
3558  *	we are exporting to CS.
3559  */
3560 static int
3561 SSInquireAdapter(inquire_adapter_t *adapter)
3562 {
3563 	adapter->NumSockets = pcmcia_num_sockets;
3564 	adapter->NumWindows = pcmcia_num_windows;
3565 	adapter->NumEDCs = 0;
3566 	/*
3567 	 * notes: Adapter Capabilities are going to be difficult to
3568 	 * determine with reliability.	Fortunately, most of them
3569 	 * don't matter under Solaris or can be handled transparently
3570 	 */
3571 	adapter->AdpCaps = 0;	/* need to fix these */
3572 	/*
3573 	 * interrupts need a little work.  For x86, the valid IRQs will
3574 	 * be restricted to those that the system has exported to the nexus.
3575 	 * for SPARC, it will be the DoRight values.
3576 	 */
3577 	adapter->ActiveHigh = 0;
3578 	adapter->ActiveLow = 0;
3579 	adapter->power_entry = pcmcia_power_table; /* until we resolve this */
3580 	adapter->NumPower = pcmcia_num_power;
3581 	return (SUCCESS);
3582 }
3583 
3584 static int
3585 SSInquireSocket(inquire_socket_t *socket)
3586 {
3587 	int retval, sock;
3588 	inquire_socket_t newsocket;
3589 	pcmcia_logical_socket_t *sockp;
3590 
3591 	sock = socket->socket;
3592 	if (sock > pcmcia_num_sockets ||
3593 		(sockp = pcmcia_sockets[sock]) == NULL)
3594 		return (BAD_SOCKET);
3595 	newsocket = *socket;
3596 	newsocket.socket = sockp->ls_socket;
3597 	retval = INQUIRE_SOCKET(sockp->ls_if, sockp->ls_adapter->pca_dip,
3598 	    &newsocket);
3599 	if (retval == SUCCESS) {
3600 		*socket = newsocket;
3601 		socket->socket = sock;
3602 	}
3603 	return (retval);
3604 }
3605 
3606 static int
3607 SSInquireWindow(inquire_window_t *window)
3608 {
3609 	int retval, win;
3610 	pcmcia_logical_window_t *winp;
3611 	inquire_window_t newwin;
3612 	int slide;
3613 
3614 	win = window->window;
3615 	if (win > pcmcia_num_windows)
3616 		return (BAD_WINDOW);
3617 
3618 	winp = pcmcia_windows[win];
3619 	newwin = *window;
3620 	newwin.window = winp->lw_window;
3621 	retval = INQUIRE_WINDOW(winp->lw_if, winp->lw_adapter->pca_dip,
3622 	    &newwin);
3623 #if defined(PCMCIA_DEBUG)
3624 		if (pcmcia_debug > 1)
3625 			cmn_err(CE_CONT, "SSInquireWindow: win=%d, pwin=%d\n",
3626 				win, newwin.window);
3627 #endif
3628 	if (retval == SUCCESS) {
3629 		*window = newwin;
3630 		/* just in case */
3631 		window->iowin_char.IOWndCaps &= ~WC_BASE;
3632 		slide = winp->lw_adapter->pca_first_socket;
3633 		/*
3634 		 * note that sockets are relative to the adapter.
3635 		 * we have to adjust the bits to show a logical
3636 		 * version.
3637 		 */
3638 
3639 		pcm_fix_bits(newwin.Sockets, window->Sockets, slide, 0);
3640 
3641 #if defined(PCMCIA_DEBUG)
3642 		if (pcmcia_debug > 1) {
3643 			cmn_err(CE_CONT, "iw: orig bits=%x, new bits=%x\n",
3644 				(int)*(uint32_t *)newwin.Sockets,
3645 				(int)*(uint32_t *)window->Sockets);
3646 			cmn_err(CE_CONT, "\t%x.%x.%x\n", window->WndCaps,
3647 				window->mem_win_char.MemWndCaps,
3648 				window->mem_win_char.MinSize);
3649 		}
3650 #endif
3651 		window->window = win;
3652 	}
3653 	return (retval);
3654 }
3655 
3656 static int
3657 SSResetSocket(int socket, int mode)
3658 {
3659 	pcmcia_logical_socket_t *sockp;
3660 
3661 	if (socket >= pcmcia_num_sockets ||
3662 		(sockp = pcmcia_sockets[socket]) == NULL)
3663 		return (BAD_SOCKET);
3664 
3665 	return (RESET_SOCKET(sockp->ls_if, sockp->ls_adapter->pca_dip,
3666 	    sockp->ls_socket, mode));
3667 }
3668 
3669 static int
3670 SSSetPage(set_page_t *page)
3671 {
3672 	int window, retval;
3673 	set_page_t newpage;
3674 	pcmcia_logical_window_t *winp;
3675 
3676 	window = page->window;
3677 	if (window > pcmcia_num_windows) {
3678 #if defined(PCMCIA_DEBUG)
3679 		if (pcmcia_debug > 1)
3680 			cmn_err(CE_CONT, "SSSetPage: window=%d (of %d)\n",
3681 				window, pcmcia_num_windows);
3682 #endif
3683 		return (BAD_WINDOW);
3684 	}
3685 
3686 	winp = pcmcia_windows[window];
3687 	newpage = *page;
3688 	newpage.window = winp->lw_window;
3689 	retval = SET_PAGE(winp->lw_if, winp->lw_adapter->pca_dip, &newpage);
3690 	if (retval == SUCCESS) {
3691 		newpage.window = window;
3692 		*page = newpage;
3693 	}
3694 #if defined(PCMCIA_DEBUG)
3695 	if ((pcmcia_debug > 1) && retval != SUCCESS)
3696 		cmn_err(CE_CONT, "\tSetPage: returning error %x\n", retval);
3697 #endif
3698 	return (retval);
3699 }
3700 
3701 static int
3702 SSSetWindow(set_window_t *win)
3703 {
3704 	int socket, window, retval, func;
3705 	set_window_t newwin;
3706 	pcmcia_logical_window_t *winp;
3707 	pcmcia_logical_socket_t *sockp;
3708 
3709 	window = win->window;
3710 	if (window > pcmcia_num_windows)
3711 		return (BAD_WINDOW);
3712 
3713 	socket = CS_GET_SOCKET_NUMBER(win->socket);
3714 	func = CS_GET_FUNCTION_NUMBER(win->socket);
3715 
3716 	if (socket > pcmcia_num_sockets ||
3717 		(sockp = pcmcia_sockets[socket]) == NULL) {
3718 		return (BAD_SOCKET);
3719 	}
3720 
3721 	winp = pcmcia_windows[window];
3722 	winp->lw_socket = win->socket; /* reverse map */
3723 	newwin = *win;
3724 	newwin.window = winp->lw_window;
3725 	newwin.socket = sockp->ls_socket;
3726 	newwin.child = sockp->ls_dip[func]; /* so we carry the dip around */
3727 
3728 	retval = SET_WINDOW(winp->lw_if, winp->lw_adapter->pca_dip, &newwin);
3729 	if (retval == SUCCESS) {
3730 		newwin.window = window;
3731 		newwin.socket = winp->lw_socket;
3732 		*win = newwin;
3733 	}
3734 	return (retval);
3735 }
3736 
3737 static int
3738 SSSetSocket(set_socket_t *socket)
3739 {
3740 	int sock, retval;
3741 	pcmcia_logical_socket_t *sockp;
3742 	set_socket_t newsock;
3743 
3744 	sock = socket->socket;
3745 	if (sock > pcmcia_num_sockets ||
3746 		(sockp = pcmcia_sockets[sock]) == NULL) {
3747 		return (BAD_SOCKET);
3748 	}
3749 
3750 	newsock = *socket;
3751 	/* note: we force CS to always get insert/removal events */
3752 	sockp->ls_cs_events = pcm_mapevents(newsock.SCIntMask) |
3753 		PCE_E2M(PCE_CARD_INSERT) | PCE_E2M(PCE_CARD_REMOVAL);
3754 #if defined(PCMCIA_DEBUG)
3755 	if (pcmcia_debug > 1)
3756 		cmn_err(CE_CONT,
3757 			"SetSocket: SCIntMask = %x\n", newsock.SCIntMask);
3758 #endif
3759 	newsock.socket = sockp->ls_socket;
3760 	newsock.VccLevel = pcmcia_map_power_set(sockp->ls_adapter,
3761 	    newsock.VccLevel, VCC);
3762 	newsock.Vpp1Level = pcmcia_map_power_set(sockp->ls_adapter,
3763 	    newsock.Vpp1Level, VPP1);
3764 	newsock.Vpp2Level = pcmcia_map_power_set(sockp->ls_adapter,
3765 	    newsock.Vpp2Level, VPP2);
3766 	retval = SET_SOCKET(sockp->ls_if, sockp->ls_adapter->pca_dip,
3767 	    &newsock);
3768 	if (retval == SUCCESS) {
3769 		newsock.socket = sock;
3770 		newsock.VccLevel = pcmcia_map_power_get(sockp->ls_adapter,
3771 		    newsock.VccLevel,
3772 		    VCC);
3773 		newsock.Vpp1Level = pcmcia_map_power_get(sockp->ls_adapter,
3774 		    newsock.Vpp1Level,
3775 		    VPP1);
3776 		newsock.Vpp2Level = pcmcia_map_power_get(sockp->ls_adapter,
3777 		    newsock.Vpp2Level,
3778 		    VPP2);
3779 		*socket = newsock;
3780 		if (socket->IREQRouting & IRQ_ENABLE) {
3781 			sockp->ls_flags |= PCS_IRQ_ENABLED;
3782 		} else {
3783 			sockp->ls_flags &= ~PCS_IRQ_ENABLED;
3784 		}
3785 	}
3786 	return (retval);
3787 }
3788 
3789 /*
3790  * SSSetIRQHandler()
3791  *	arrange for IRQ to be allocated if appropriate and always
3792  *	arrange that PC Card interrupt handlers get called.
3793  */
3794 static int
3795 SSSetIRQHandler(set_irq_handler_t *handler)
3796 {
3797 	int sock, retval, func;
3798 	pcmcia_logical_socket_t *sockp;
3799 	struct pcmcia_parent_private *ppd;
3800 	dev_info_t *dip;
3801 	ddi_iblock_cookie_t iblk;
3802 	ddi_idevice_cookie_t idev;
3803 
3804 	sock = CS_GET_SOCKET_NUMBER(handler->socket);
3805 	func = CS_GET_FUNCTION_NUMBER(handler->socket);
3806 	if (sock > pcmcia_num_sockets ||
3807 		(sockp = pcmcia_sockets[sock]) == NULL) {
3808 		return (BAD_SOCKET);
3809 	}
3810 #if defined(PCMCIA_DEBUG)
3811 	if (pcmcia_debug) {
3812 
3813 		cmn_err(CE_CONT, "SSSetIRQHandler: socket=%x, function=%x\n",
3814 			sock, func);
3815 		cmn_err(CE_CONT, "\thandler(%p): socket=%x, irq=%x, id=%x\n",
3816 			(void *)handler->handler, handler->socket, handler->irq,
3817 			handler->handler_id);
3818 	}
3819 #endif
3820 	dip = sockp->ls_dip[func];
3821 
3822 	ppd = (struct pcmcia_parent_private *)ddi_get_parent_data(dip);
3823 
3824 	handler->iblk_cookie = &iblk;
3825 	handler->idev_cookie = &idev;
3826 
3827 	retval = ddi_add_intr(dip, 0, handler->iblk_cookie,
3828 	    handler->idev_cookie,
3829 	    (uint32_t(*)(caddr_t)) handler->handler,
3830 	    handler->arg1);
3831 
3832 	if (retval == DDI_SUCCESS) {
3833 		handler->iblk_cookie = &sockp->ls_iblk;
3834 		handler->idev_cookie = &sockp->ls_idev;
3835 		handler->irq = ppd->ppd_intrspec->intrspec_vec;
3836 		retval = SUCCESS;
3837 	} else {
3838 		retval = sockp->ls_error;
3839 	}
3840 	return (retval);
3841 }
3842 
3843 /*
3844  * SSClearIRQHandler()
3845  *	Arrange to have the interrupt handler specified removed
3846  *	from the interrupt list.
3847  */
3848 static int
3849 SSClearIRQHandler(clear_irq_handler_t *handler)
3850 {
3851 	int sock, func;
3852 	pcmcia_logical_socket_t *sockp;
3853 	dev_info_t *dip;
3854 
3855 	sock = CS_GET_SOCKET_NUMBER(handler->socket);
3856 	func = CS_GET_FUNCTION_NUMBER(handler->socket);
3857 
3858 #if defined(PCMCIA_DEBUG)
3859 	if (pcmcia_debug) {
3860 
3861 		cmn_err(CE_CONT,
3862 			"SSClearIRQHandler: socket=%x, function=%x\n",
3863 			sock, func);
3864 		cmn_err(CE_CONT,
3865 			"\thandler(%p): socket=%x, id=%x\n",
3866 			(void *)handler, handler->socket,
3867 			handler->handler_id);
3868 	}
3869 #endif
3870 
3871 	if (sock > pcmcia_num_sockets ||
3872 		(sockp = pcmcia_sockets[sock]) == NULL) {
3873 		return (BAD_SOCKET);
3874 	}
3875 	dip = sockp->ls_dip[func];
3876 	if (dip) {
3877 		ddi_remove_intr(dip, 0, NULL);
3878 		return (SUCCESS);
3879 	}
3880 	return (BAD_SOCKET);
3881 }
3882 
3883 
3884 /*
3885  * pcm_pathname()
3886  *	make a partial path from dip.
3887  *	used to mknods relative to /devices/pcmcia/
3888  *
3889  * XXX - we now use ddi_get_name_addr to get the "address" portion
3890  *	of the name; that way, we only have to modify the name creation
3891  *	algorithm in one place
3892  */
3893 static void
3894 pcm_pathname(dev_info_t *dip, char *name, char *path)
3895 {
3896 	(void) sprintf(path, "%s@%s:%s", ddi_node_name(dip),
3897 	    ddi_get_name_addr(dip), name);
3898 }
3899 
3900 /*
3901  * pcmcia_create_device()
3902  *	create the /devices entries for the driver
3903  *	it is assumed that the PC Card driver will do a
3904  *	RegisterClient for each subdevice.
3905  *	The device type string is encoded here to match
3906  *	the standardized names when possible.
3907  * XXX - note that we may need to provide a way for the
3908  *	caller to specify the complete name string that
3909  *	we pass to ddi_set_name_addr
3910  */
3911 static int
3912 pcmcia_create_device(ss_make_device_node_t *init)
3913 {
3914 	int err = SUCCESS;
3915 	struct pcm_make_dev device;
3916 	struct dev_ops *ops;
3917 	major_t major;
3918 
3919 	/*
3920 	 * Now that we have the name, create it.
3921 	 */
3922 
3923 	bzero(&device, sizeof (device));
3924 	if (init->flags & SS_CSINITDEV_CREATE_DEVICE) {
3925 		if ((err = ddi_create_minor_node(init->dip,
3926 		    init->name,
3927 		    init->spec_type,
3928 		    init->minor_num,
3929 		    init->node_type,
3930 		    0)) != DDI_SUCCESS) {
3931 #if defined(PCMCIA_DEBUG)
3932 			if (pcmcia_debug)
3933 				cmn_err(CE_CONT,
3934 					"pcmcia_create_device: failed "
3935 					"create\n");
3936 #endif
3937 			return (BAD_ATTRIBUTE);
3938 		}
3939 
3940 		major = ddi_name_to_major(ddi_binding_name(init->dip));
3941 		ops = ddi_get_driver(init->dip);
3942 		LOCK_DEV_OPS(&devnamesp[major].dn_lock);
3943 		INCR_DEV_OPS_REF(ops);
3944 		(void) ddi_pathname(init->dip, device.path);
3945 		DECR_DEV_OPS_REF(ops);
3946 		UNLOCK_DEV_OPS(&devnamesp[major].dn_lock);
3947 		(void) sprintf(device.path + strlen(device.path), ":%s",
3948 		    init->name);
3949 
3950 		(void) strcpy(device.driver, ddi_binding_name(init->dip));
3951 #if defined(PCMCIA_DEBUG)
3952 		if (pcmcia_debug)
3953 			cmn_err(CE_CONT,
3954 				"pcmcia_create_device: created %s "
3955 				"from %s [%s]\n",
3956 				device.path, init->name, device.driver);
3957 #endif
3958 		device.dev =
3959 		    makedevice(ddi_name_to_major(ddi_get_name(init->dip)),
3960 		    init->minor_num);
3961 		device.flags |= (init->flags & SS_CSINITDEV_MORE_DEVICES) ?
3962 		    PCM_EVENT_MORE : 0;
3963 		device.type = init->spec_type;
3964 		device.op = SS_CSINITDEV_CREATE_DEVICE;
3965 		device.socket = ddi_getprop(DDI_DEV_T_ANY, init->dip,
3966 		    DDI_PROP_CANSLEEP, PCM_DEV_SOCKET,
3967 		    -1);
3968 	} else if (init->flags & SS_CSINITDEV_REMOVE_DEVICE) {
3969 		device.op = SS_CSINITDEV_REMOVE_DEVICE;
3970 		device.socket = ddi_getprop(DDI_DEV_T_ANY, init->dip,
3971 		    DDI_PROP_CANSLEEP, PCM_DEV_SOCKET,
3972 		    -1);
3973 		if (init->name != NULL)
3974 			(void) strcpy(device.path, init->name);
3975 		device.dev =
3976 		    makedevice(ddi_name_to_major(ddi_get_name(init->dip)),
3977 		    0);
3978 		ddi_remove_minor_node(init->dip, init->name);
3979 	}
3980 
3981 	/*
3982 	 *	we send an event for ALL devices created.
3983 	 *	To do otherwise ties us to using drvconfig
3984 	 *	forever.  There are relatively few devices
3985 	 *	ever created so no need to do otherwise.
3986 	 *	The existence of the event manager must never
3987 	 *	be visible to a PCMCIA device driver.
3988 	 */
3989 	pcm_event_manager(PCE_INIT_DEV, device.socket, &device);
3990 
3991 	return (err);
3992 }
3993 
3994 /*
3995  * pcmcia_get_minors()
3996  *	We need to traverse the minor node list of the
3997  *	dip if there are any.  This takes two passes;
3998  *	one to get the count and buffer size and the
3999  *	other to actually copy the data into the buffer.
4000  *	The framework requires that the dip be locked
4001  *	during this time to avoid breakage as well as the
4002  *	driver being locked.
4003  */
4004 int
4005 pcmcia_get_minors(dev_info_t *dip, struct pcm_make_dev **minors)
4006 {
4007 	int count = 0;
4008 	struct ddi_minor_data *dp;
4009 	struct pcm_make_dev *md;
4010 	int socket;
4011 	major_t major;
4012 	struct dev_ops *ops;
4013 
4014 	socket = ddi_getprop(DDI_DEV_T_ANY, dip, DDI_PROP_DONTPASS,
4015 	    PCM_DEV_SOCKET, -1);
4016 	mutex_enter(&(DEVI(dip)->devi_lock));
4017 	if (DEVI(dip)->devi_minor != (struct ddi_minor_data *)NULL) {
4018 		for (dp = DEVI(dip)->devi_minor;
4019 			dp != (struct ddi_minor_data *)NULL;
4020 			dp = dp->next) {
4021 			count++; /* have one more */
4022 		}
4023 		/* we now know how many nodes to allocate */
4024 		md = kmem_zalloc(count * sizeof (struct pcm_make_dev),
4025 		    KM_NOSLEEP);
4026 		if (md != NULL) {
4027 			*minors = md;
4028 			for (dp = DEVI(dip)->devi_minor;
4029 				dp != (struct ddi_minor_data *)NULL;
4030 				dp = dp->next, md++) {
4031 #if defined(PCMCIA_DEBUG)
4032 				if (pcmcia_debug > 1) {
4033 					cmn_err(CE_CONT,
4034 						"pcmcia_get_minors: name=%s,"
4035 						"socket=%d, stype=%x, "
4036 						"ntype=%s, dev_t=%x",
4037 						dp->ddm_name,
4038 						socket,
4039 						dp->ddm_spec_type,
4040 						dp->ddm_node_type,
4041 						(int)dp->ddm_dev);
4042 					cmn_err(CE_CONT,
4043 						"\tbind name = %s\n",
4044 						ddi_binding_name(dip));
4045 				}
4046 #endif
4047 				md->socket = socket;
4048 				md->op = SS_CSINITDEV_CREATE_DEVICE;
4049 				md->dev = dp->ddm_dev;
4050 				md->type = dp->ddm_spec_type;
4051 				(void) strcpy(md->driver,
4052 				    ddi_binding_name(dip));
4053 				major = ddi_name_to_major(md->driver);
4054 				ops = ddi_get_driver(dip);
4055 				LOCK_DEV_OPS(&devnamesp[major].dn_lock);
4056 				pcm_pathname(dip, dp->ddm_name, md->path);
4057 				INCR_DEV_OPS_REF(ops);
4058 				(void) ddi_pathname(dip, md->path);
4059 				DECR_DEV_OPS_REF(ops);
4060 				UNLOCK_DEV_OPS(&devnamesp[major].dn_lock);
4061 				(void) sprintf(md->path + strlen(md->path),
4062 					":%s", dp->ddm_name);
4063 				if (dp->next == NULL)
4064 					/* no more */
4065 					md->flags |= PCM_EVENT_MORE;
4066 			}
4067 		} else {
4068 			count = 0;
4069 		}
4070 	}
4071 	mutex_exit(&(DEVI(dip)->devi_lock));
4072 	return (count);
4073 }
4074 
4075 #if defined(PCMCIA_DEBUG)
4076 static char *ddmtypes[] = { "minor", "alias", "default", "internal" };
4077 
4078 static void
4079 pcmcia_dump_minors(dev_info_t *dip)
4080 {
4081 	int count = 0;
4082 	struct ddi_minor_data *dp;
4083 	int unit, major;
4084 	dev_info_t *np;
4085 
4086 	unit = ddi_getprop(DDI_DEV_T_ANY, dip, DDI_PROP_DONTPASS,
4087 	    PCM_DEV_SOCKET, -1);
4088 	cmn_err(CE_CONT,
4089 		"pcmcia_dump_minors: dip=%p, socket=%d\n", (void *)dip, unit);
4090 
4091 	major = ddi_driver_major(dip);
4092 	if (major != -1) {
4093 		for (np = devnamesp[major].dn_head; np != NULL;
4094 		    np = (dev_info_t *)DEVI(np)->devi_next) {
4095 			char *cf2 = "";
4096 			char *cur = "";
4097 			if (i_ddi_node_state(np) == DS_READY)
4098 				cf2 = "DS_READY";
4099 			if (np == dip)
4100 				cur = "CUR";
4101 			cmn_err(CE_CONT, "\tsibs: %s %s %s\n",
4102 				ddi_binding_name(np), cf2, cur);
4103 
4104 			mutex_enter(&(DEVI(np)->devi_lock));
4105 			if (DEVI(np)->devi_minor !=
4106 			    (struct ddi_minor_data *)NULL) {
4107 				for (dp = DEVI(np)->devi_minor;
4108 				    dp != (struct ddi_minor_data *)NULL;
4109 				    dp = dp->next) {
4110 					count++; /* have one more */
4111 				}
4112 				for (dp = DEVI(dip)->devi_minor;
4113 				    dp != (struct ddi_minor_data *)NULL;
4114 				    dp = dp->next) {
4115 					cmn_err(CE_CONT, "\ttype=%s, name=%s,"
4116 						"socket=%d, stype=%x, "
4117 						"ntype=%s, dev_t=%x",
4118 						ddmtypes[dp->type],
4119 						dp->ddm_name,
4120 						unit,
4121 						dp->ddm_spec_type,
4122 						dp->ddm_node_type,
4123 						(int)dp->ddm_dev);
4124 					cmn_err(CE_CONT, "\tbind name = %s\n",
4125 						ddi_binding_name(np));
4126 				}
4127 			}
4128 			mutex_exit(&(DEVI(np)->devi_lock));
4129 		}
4130 	}
4131 }
4132 #endif
4133 
4134 /*
4135  * experimental merging code
4136  * what are the things that we should merge on?
4137  *	match something by name in the "compatible" property
4138  *	restrict to a specific "socket"
4139  *	restrict to a specific "instance"
4140  */
4141 /*ARGSUSED*/
4142 static int
4143 pcmcia_merge_conf(dev_info_t *dip)
4144 {
4145 	return (0);		/* merge failed */
4146 }
4147 
4148 /*
4149  * pcmcia_mfc_intr()
4150  *	Multifunction Card interrupt handler
4151  *	While some adapters share interrupts at the lowest
4152  *	level, some can't.  In order to be consistent, we
4153  *	split multifunction cards out with this intercept and
4154  *	allow the low level to do what is best for it.
4155  *	the arg is a pcmcia_socket structure and all interrupts
4156  *	are per-socket in this case.  We also have the option
4157  *	to optimize if the cards support it.  It also means
4158  *	that we can use the INTRACK mode if it proves desirable
4159  */
4160 /*ARGSUSED*/
4161 static uint32_t
4162 pcmcia_mfc_intr(caddr_t arg1, caddr_t arg2)
4163 {
4164 	pcmcia_logical_socket_t *sockp;
4165 	inthandler_t *intr, *first;
4166 	int done, result;
4167 
4168 	sockp = (pcmcia_logical_socket_t *)arg1;
4169 
4170 #if defined(PCMCIA_DEBUG)
4171 	if (pcmcia_debug > 1) {
4172 		cmn_err(CE_CONT, "pcmcia_mfc_intr sockp=%p"
4173 		    " ls_inthandlers=%p\n"
4174 		    "\t ls_flags=0x%x PCS_IRQ_ENABLED=0x%x \n",
4175 		    (void *) sockp, (void *) sockp->ls_inthandlers,
4176 		    sockp->ls_flags, PCS_IRQ_ENABLED);
4177 	}
4178 #endif
4179 
4180 	if (sockp == NULL || sockp->ls_inthandlers == NULL ||
4181 	    !(sockp->ls_flags & PCS_IRQ_ENABLED))
4182 		return (DDI_INTR_UNCLAIMED);
4183 
4184 	mutex_enter(&sockp->ls_ilock);
4185 	for (done = 0, result = 0, first = intr = sockp->ls_inthandlers;
4186 		intr != NULL && !done; intr = intr->next) {
4187 		result |= intr->intr(intr->arg1, intr->arg2);
4188 		if (intr->next == first)
4189 			done++;
4190 	}
4191 	if (intr == NULL) {
4192 		cmn_err(CE_WARN, "pcmcia_mfc_intr: bad MFC handler list");
4193 	}
4194 	if (sockp->ls_inthandlers)
4195 		sockp->ls_inthandlers = sockp->ls_inthandlers->next;
4196 
4197 	mutex_exit(&sockp->ls_ilock);
4198 	return (result ? DDI_INTR_CLAIMED : DDI_INTR_UNCLAIMED);
4199 }
4200 
4201 /*
4202  * pcmcia_power(dip)
4203  *	control power for nexus and children
4204  */
4205 int
4206 pcmcia_power(dev_info_t *dip, int component, int level)
4207 {
4208 #if 0
4209 	anp_t *anp = (anp_t *)ddi_get_driver_private(dip);
4210 	int i;
4211 	/*
4212 	 * for now, we only have one component.  Should there be one per-socket?
4213 	 * the level is only one (power on or off)
4214 	 */
4215 	if (component != 0 || level > 1)
4216 		return (DDI_FAILURE);
4217 
4218 	for (i = 0; i < pcic->pc_numsockets; i++) {
4219 		if (pcic->pc_callback)
4220 			PC_CALLBACK(dip, pcic->pc_cb_arg,
4221 			    (level == 0) ? PCE_PM_SUSPEND :
4222 			    PCE_PM_RESUME,
4223 			    i);
4224 	}
4225 #else
4226 	cmn_err(CE_WARN, "pcmcia_power: component=%d, level=%d for %s",
4227 		component, level, ddi_get_name_addr(dip));
4228 	return (DDI_FAILURE);
4229 #endif
4230 }
4231 
4232 void
4233 pcmcia_begin_resume(dev_info_t *dip)
4234 {
4235 	int i;
4236 	struct pcmcia_adapter *adapt = NULL;
4237 	for (i = 0; i < pcmcia_num_adapters; i++) {
4238 		if (pcmcia_adapters[i]->pca_dip == dip) {
4239 			adapt = pcmcia_adapters[i];
4240 			break;
4241 		}
4242 	}
4243 	if (adapt == NULL)
4244 		return;
4245 
4246 	for (i = 0; i < adapt->pca_numsockets; i++) {
4247 		int s;
4248 		s = adapt->pca_first_socket + i;
4249 		if (pcmcia_sockets[s]->ls_flags & PCS_SUSPENDED) {
4250 			if (pcmcia_sockets[s]->ls_flags &
4251 			    (1 << PCE_PM_RESUME)) {
4252 				(void) cs_event(PCE_PM_RESUME, s, 0);
4253 				pcm_event_manager(PCE_PM_RESUME, s, NULL);
4254 			}
4255 			(void) cs_event(PCE_CARD_REMOVAL, s, 0);
4256 			pcm_event_manager(PCE_CARD_REMOVAL, s, NULL);
4257 		}
4258 	}
4259 }
4260 
4261 void
4262 pcmcia_wait_insert(dev_info_t *dip)
4263 {
4264 	int i, f, tries, done;
4265 	clock_t tm;
4266 	struct pcmcia_adapter *adapt = NULL;
4267 	anp_t *nexus;
4268 
4269 	for (i = 0; i < pcmcia_num_adapters; i++) {
4270 		if (pcmcia_adapters[i]->pca_dip == dip) {
4271 			adapt = pcmcia_adapters[i];
4272 			break;
4273 		}
4274 	}
4275 	if (adapt == NULL)
4276 		return;
4277 
4278 	for (tries = adapt->pca_numsockets * 10; tries > 0; tries--) {
4279 		done = 1;
4280 		mutex_enter(&pcmcia_global_lock);
4281 		for (i = 0; i < adapt->pca_numsockets; i++) {
4282 			int s;
4283 			s = adapt->pca_first_socket + i;
4284 			for (f = 0; f < PCMCIA_MAX_FUNCTIONS; f++)
4285 				if (pcmcia_sockets[s] &&
4286 				    pcmcia_sockets[s]->ls_flags &
4287 				    PCS_SUSPENDED) {
4288 					done = 0;
4289 					break;
4290 				}
4291 		}
4292 		if (!done) {
4293 			tm = ddi_get_lbolt();
4294 			(void) cv_timedwait(&pcmcia_condvar,
4295 			    &pcmcia_global_lock,
4296 			    tm + drv_usectohz(100000));
4297 		} else {
4298 			tries = 0;
4299 		}
4300 		mutex_exit(&pcmcia_global_lock);
4301 	}
4302 
4303 	nexus = (anp_t *)ddi_get_driver_private(dip);
4304 	pcmcia_find_cards(nexus);
4305 }
4306 
4307 int
4308 pcmcia_map_reg(dev_info_t *pdip, dev_info_t *dip, ra_return_t *ra,
4309 		uint32_t state, caddr_t *base,
4310 		ddi_acc_handle_t *handle, ddi_device_acc_attr_t *attrib,
4311 		uint32_t req_base)
4312 {
4313 	struct pcmcia_parent_private *ppd;
4314 	int rnum = 0, type = PCMCIA_MAP_MEM;
4315 	ddi_map_req_t mr;
4316 	ddi_acc_hdl_t *hp;
4317 	int result;
4318 	struct regspec *reg;
4319 	ddi_device_acc_attr_t attr;
4320 
4321 	if (dip != NULL) {
4322 		ppd = (struct pcmcia_parent_private *)ddi_get_parent_data(dip);
4323 		if (ppd == NULL)
4324 			return (DDI_FAILURE);
4325 		for (rnum = 1; rnum < ppd->ppd_nreg; rnum++) {
4326 			struct pcm_regs *p;
4327 			p = &ppd->ppd_reg[rnum];
4328 			if (state & WS_IO) {
4329 				/* need I/O */
4330 				type = PCMCIA_MAP_IO;
4331 				/*
4332 				 * We want to find an IO regspec. When we
4333 				 *	find one, it either has to match
4334 				 *	the caller's requested base address
4335 				 *	or it has to be relocatable.
4336 				 * We match on the requested base address
4337 				 *	rather than the allocated base
4338 				 *	address so that we handle the case
4339 				 *	of adapters that have IO window base
4340 				 *	relocation registers.
4341 				 */
4342 				if ((p->phys_hi &
4343 				    PC_REG_SPACE(PC_REG_SPACE_IO)) &&
4344 					((req_base == p->phys_lo) ||
4345 					!(p->phys_hi & PC_REG_RELOC(1))))
4346 				    break;
4347 			} else {
4348 				/* need memory */
4349 				type = PCMCIA_MAP_MEM;
4350 				if (p->phys_hi &
4351 				    PC_REG_SPACE(PC_REG_SPACE_MEMORY|
4352 				    PC_REG_SPACE_ATTRIBUTE))
4353 					break;
4354 			}
4355 		}
4356 		if (rnum >= ppd->ppd_nreg)
4357 			return (DDI_FAILURE);
4358 	} else if (state & WS_IO) {
4359 		return (DDI_FAILURE);
4360 	}
4361 
4362 	reg = kmem_zalloc(sizeof (pci_regspec_t), KM_SLEEP);
4363 	reg = pcmcia_cons_regspec(pdip, type, (uchar_t *)reg, ra);
4364 
4365 	if (attrib == NULL ||
4366 	    attrib->devacc_attr_version != DDI_DEVICE_ATTR_V0) {
4367 		attr.devacc_attr_version = DDI_DEVICE_ATTR_V0;
4368 		attr.devacc_attr_endian_flags = DDI_NEVERSWAP_ACC;
4369 		attr.devacc_attr_dataorder = DDI_STRICTORDER_ACC;
4370 	} else {
4371 		attr = *attrib;
4372 	}
4373 	/*
4374 	 * Allocate and initialize the common elements of data access handle.
4375 	 */
4376 	*handle = impl_acc_hdl_alloc(KM_SLEEP, NULL);
4377 	hp = impl_acc_hdl_get(*handle);
4378 	hp->ah_vers = VERS_ACCHDL;
4379 	hp->ah_dip = dip != NULL ? dip : pdip;
4380 	hp->ah_rnumber = rnum;
4381 	hp->ah_offset = 0;
4382 	hp->ah_len = ra->ra_len;
4383 	hp->ah_acc = attr;
4384 
4385 	/*
4386 	 * Set up the mapping request and call to parent.
4387 	 */
4388 	mr.map_op = DDI_MO_MAP_LOCKED;
4389 	mr.map_type = DDI_MT_REGSPEC;
4390 	mr.map_obj.rp = reg;
4391 	mr.map_prot = PROT_READ | PROT_WRITE;
4392 	mr.map_flags = DDI_MF_KERNEL_MAPPING;
4393 	mr.map_handlep = hp;
4394 	mr.map_vers = DDI_MAP_VERSION;
4395 
4396 	result = ddi_map(pdip, &mr, 0, ra->ra_len, base);
4397 	if (result != DDI_SUCCESS) {
4398 		impl_acc_hdl_free(*handle);
4399 		*handle = (ddi_acc_handle_t)NULL;
4400 		kmem_free(reg, sizeof (pci_regspec_t));
4401 	} else {
4402 		hp->ah_addr = *base;
4403 		if (mr.map_op == DDI_MO_UNMAP)
4404 			ra = NULL;
4405 		if (dip != NULL)
4406 			pcmcia_set_assigned(dip, rnum, ra);
4407 	}
4408 
4409 	return (result);
4410 }
4411 
4412 struct pcmcia_adapter *
4413 pcmcia_get_adapter(dev_info_t *dip)
4414 {
4415 	int i;
4416 
4417 	for (i = 0; i < pcmcia_num_adapters; i++) {
4418 		if (pcmcia_adapters[i] &&
4419 		    pcmcia_adapters[i]->pca_dip == dip) {
4420 			return (pcmcia_adapters[i]);
4421 		}
4422 	}
4423 	return (NULL);
4424 }
4425 
4426 
4427 void
4428 pcmcia_set_assigned(dev_info_t *dip, int rnum, ra_return_t *ret)
4429 {
4430 	struct pcmcia_parent_private *ppd;
4431 	struct pcm_regs *reg, *assign;
4432 
4433 	ppd = (struct pcmcia_parent_private *)ddi_get_parent_data(dip);
4434 	if (ppd) {
4435 		reg = &ppd->ppd_reg[rnum];
4436 		assign = &ppd->ppd_assigned[rnum];
4437 		if (ret) {
4438 			if (assign->phys_hi == 0) {
4439 				assign->phys_hi = reg->phys_hi;
4440 				assign->phys_lo = ret->ra_addr_lo;
4441 				assign->phys_len = ret->ra_len;
4442 			} else if (assign->phys_lo != ret->ra_addr_lo) {
4443 #ifdef PCMCIA_DEBUG
4444 				cmn_err(CE_WARN, "pcmcia: bad address:"
4445 					"%s=<%x,%x>",
4446 					ddi_get_name_addr(dip),
4447 					ret->ra_addr_lo, assign->phys_lo);
4448 #else
4449 				cmn_err(CE_WARN, "!pcmcia: bad address:"
4450 					"%s=<%x,%x>",
4451 					ddi_get_name_addr(dip),
4452 					ret->ra_addr_lo, (int)assign->phys_lo);
4453 #endif
4454 			}
4455 			assign->phys_hi = PC_INCR_REFCNT(assign->phys_hi);
4456 		} else {
4457 			int i;
4458 			assign->phys_hi = PC_DECR_REFCNT(assign->phys_hi);
4459 			i = PC_GET_REG_REFCNT(assign->phys_hi);
4460 			if (i == 0) {
4461 				assign->phys_hi = 0;
4462 				assign->phys_lo = 0;
4463 				assign->phys_len = 0;
4464 			}
4465 		}
4466 	}
4467 }
4468 
4469 int
4470 pcmcia_alloc_mem(dev_info_t *dip, ndi_ra_request_t *req, ra_return_t *ret)
4471 {
4472 	return (pcmcia_ra_alloc(dip, req, ret, NDI_RA_TYPE_MEM));
4473 }
4474 
4475 int
4476 pcmcia_alloc_io(dev_info_t *dip, ndi_ra_request_t *req, ra_return_t *ret)
4477 {
4478 	return (pcmcia_ra_alloc(dip, req, ret, NDI_RA_TYPE_IO));
4479 }
4480 
4481 int
4482 pcmcia_ra_alloc(dev_info_t *dip, ndi_ra_request_t *req, ra_return_t *ret,
4483 		char *type)
4484 {
4485 	int rval;
4486 	uint64_t base = 0;
4487 	uint64_t len = 0;
4488 
4489 	/*
4490 	 * Allocate space from busra resource list
4491 	 * should not return an address > 32 bits
4492 	 */
4493 
4494 	if ((ndi_ra_alloc(dip, req, &base, &len, type, NDI_RA_PASS)
4495 							== NDI_FAILURE) ||
4496 	    ((base >> 32) != 0)) {
4497 		ret->ra_addr_lo = 0;
4498 		ret->ra_len = 0;
4499 		rval = DDI_FAILURE;
4500 	} else {
4501 		ret->ra_addr_lo =  base & 0xffffffff;
4502 		ret->ra_len = len;
4503 		rval = DDI_SUCCESS;
4504 	}
4505 	ret->ra_addr_hi = 0;
4506 	return (rval);
4507 }
4508 
4509 int
4510 pcmcia_free_mem(dev_info_t *dip, ra_return_t *ret)
4511 {
4512 	return (pcmcia_ra_free(dip, ret, NDI_RA_TYPE_MEM));
4513 }
4514 
4515 int
4516 pcmcia_free_io(dev_info_t *dip, ra_return_t *ret)
4517 {
4518 	return (pcmcia_ra_free(dip, ret, NDI_RA_TYPE_IO));
4519 }
4520 
4521 int
4522 pcmcia_ra_free(dev_info_t *dip, ra_return_t *ret, char *type)
4523 {
4524 	if (ndi_ra_free(dip, (uint64_t)ret->ra_addr_lo, (uint64_t)ret->ra_len,
4525 	    type, NDI_RA_PASS) == NDI_SUCCESS) {
4526 		return (DDI_SUCCESS);
4527 	} else {
4528 		return (DDI_FAILURE);
4529 	}
4530 }
4531 
4532 
4533 /*
4534  * when the low level device configuration does resource assignment
4535  * (devconf) then free the allocated resources so we can reassign them
4536  * later.  Walk the child list to get them.
4537  */
4538 void
4539 pcmcia_free_resources(dev_info_t *self)
4540 {
4541 	struct regspec *assigned;
4542 	int len;
4543 	dev_info_t *dip;
4544 	int circular;
4545 
4546 	ndi_devi_enter(self, &circular);
4547 	/* do searches in compatible property order */
4548 	for (dip = (dev_info_t *)DEVI(self)->devi_child;
4549 	    dip != NULL;
4550 	    dip = (dev_info_t *)DEVI(dip)->devi_sibling) {
4551 		len = 0;
4552 		if (ddi_getlongprop(DDI_DEV_T_ANY, dip,
4553 		    DDI_PROP_DONTPASS|DDI_PROP_CANSLEEP,
4554 		    "assigned-addresses",
4555 		    (caddr_t)&assigned,
4556 		    &len) == DDI_PROP_SUCCESS) {
4557 			/*
4558 			 * if there are assigned resources at this point,
4559 			 * then the OBP or devconf have assigned them and
4560 			 * they need to be freed.
4561 			 */
4562 			kmem_free(assigned, len);
4563 		}
4564 	}
4565 	ndi_devi_exit(self, circular);
4566 }
4567 
4568 /*
4569  * this is the equivalent of pcm_get_intr using ra_allocs.
4570  * returns -1 if failed, otherwise returns the allocated irq.
4571  * The input request, if less than zero it means not a specific
4572  * irq requested. If larger then 0 then we are requesting that specific
4573  * irq
4574  */
4575 int
4576 pcmcia_get_intr(dev_info_t *dip, int request)
4577 {
4578 	ndi_ra_request_t req;
4579 	uint64_t base;
4580 	uint64_t len;
4581 	int err;
4582 
4583 	bzero(&req, sizeof (req));
4584 	base = 0;
4585 	len = 1;
4586 	if (request >= 0) {
4587 		req.ra_flags = NDI_RA_ALLOC_SPECIFIED;
4588 		req.ra_len = 1;
4589 		req.ra_addr = (uint64_t)request;
4590 	}
4591 
4592 	req.ra_boundbase = 0;
4593 	req.ra_boundlen = 0xffffffffUL;
4594 	req.ra_flags |= NDI_RA_ALLOC_BOUNDED;
4595 
4596 	err = ndi_ra_alloc(dip, &req, &base, &len, NDI_RA_TYPE_INTR,
4597 	    NDI_RA_PASS);
4598 
4599 	if (err == NDI_FAILURE) {
4600 		return (-1);
4601 	} else {
4602 		return ((int)base);
4603 	}
4604 }
4605 
4606 
4607 int
4608 pcmcia_return_intr(dev_info_t *dip, int request)
4609 {
4610 	if ((ndi_ra_free(dip, (uint64_t)request, 1, NDI_RA_TYPE_INTR,
4611 	    NDI_RA_PASS)) == NDI_SUCCESS) {
4612 		return (0);
4613 	} else
4614 		return (-1);
4615 
4616 }
4617 
4618 #ifdef sparc
4619 
4620 int
4621 pcmcia_add_intr_impl(dev_info_t *dip, dev_info_t *rdip,
4622     ddi_intr_handle_impl_t *hdlp)
4623 {
4624 
4625 	struct pcmcia_parent_private *ppd;
4626 	pcmcia_logical_socket_t *sockp;
4627 	int socket, ret;
4628 	struct pcmcia_adapter *adapt;
4629 	set_irq_handler_t handler;
4630 	struct intrspec *pispec;
4631 
4632 #if defined(PCMCIA_DEBUG)
4633 	if (pcmcia_debug) {
4634 		cmn_err(CE_CONT,
4635 		    "pcmcia_add_intr_impl() entered "
4636 		    "dip=%p rdip=%p hdlp=%p \n",
4637 		    (void *)dip, (void *)rdip, (void *)hdlp);
4638 	}
4639 #endif
4640 
4641 	ppd = (struct pcmcia_parent_private *)ddi_get_parent_data(rdip);
4642 	socket = ppd->ppd_socket;
4643 	sockp = pcmcia_sockets[socket];
4644 	adapt = sockp->ls_adapter;
4645 
4646 #if defined(PCMCIA_DEBUG)
4647 	if (pcmcia_debug) {
4648 		cmn_err(CE_CONT, "pcmcia_add_intr_impl()"
4649 		    " ppd_flags=0X%x PPD_CARD_MULTI=0X%x\n"
4650 		    " ppd_intrspec=%p ls_inthandlers=%p\n",
4651 		    ppd->ppd_flags, PPD_CARD_MULTI,
4652 		    (void *) ppd->ppd_intrspec,
4653 		    (void *)sockp->ls_inthandlers);
4654 	}
4655 #endif
4656 
4657 	/*
4658 	 * calculate IPL level when we support multiple levels
4659 	 */
4660 	pispec = ppd->ppd_intrspec;
4661 	if (pispec == NULL) {
4662 		sockp->ls_error = BAD_IRQ;
4663 		return (DDI_FAILURE);
4664 	}
4665 
4666 	handler.socket = sockp->ls_socket;
4667 	handler.irq = 0;	/* default case */
4668 	handler.handler = (f_tt *)hdlp->ih_cb_func;
4669 	handler.arg1 = hdlp->ih_cb_arg1;
4670 	handler.arg2 = hdlp->ih_cb_arg2;
4671 	handler.handler_id = (uint32_t)rdip;
4672 
4673 	/*
4674 	 * check if multifunction and do the right thing
4675 	 * we put an intercept in between the mfc handler and
4676 	 * us so we can catch and process.  We might be able
4677 	 * to optimize this depending on the card features
4678 	 * (a future option).
4679 	 */
4680 	if (ppd->ppd_flags & PPD_CARD_MULTI) {
4681 		inthandler_t *intr;
4682 		/*
4683 		 * note that the first function is a special
4684 		 * case since it sets things up.  We fall through
4685 		 * to the lower code and get the hardware set up.
4686 		 * subsequent times we just lock the list and insert
4687 		 * the handler and all is well.
4688 		 */
4689 		intr = kmem_zalloc(sizeof (inthandler_t), KM_NOSLEEP);
4690 		if (intr == NULL) {
4691 			sockp->ls_error = BAD_IRQ;
4692 			return (DDI_FAILURE);
4693 		}
4694 		intr->intr = hdlp->ih_cb_func;
4695 		intr->handler_id = (uint_t)rdip;
4696 		intr->arg1 = hdlp->ih_cb_arg1;
4697 		intr->arg2 = hdlp->ih_cb_arg2;
4698 		intr->socket = socket;
4699 
4700 		mutex_enter(&sockp->ls_ilock);
4701 		if (sockp->ls_inthandlers == NULL) {
4702 			intr->next = intr->prev = intr;
4703 			sockp->ls_inthandlers = intr;
4704 			sockp->ls_mfintr_dip = rdip;
4705 			mutex_exit(&sockp->ls_ilock);
4706 
4707 			/*
4708 			 * replace first function handler with
4709 			 * the mfc handler
4710 			 */
4711 			handler.handler =  (f_tt *)pcmcia_mfc_intr;
4712 			handler.arg1 = (caddr_t)sockp;
4713 			handler.arg2 = NULL;
4714 		} else {
4715 			insque(intr, sockp->ls_inthandlers);
4716 			mutex_exit(&sockp->ls_ilock);
4717 
4718 			pispec->intrspec_vec = sockp->ls_intr_vec;
4719 			pispec->intrspec_pri = sockp->ls_intr_pri;
4720 			hdlp->ih_pri = sockp->ls_intr_pri;
4721 
4722 			return (DDI_SUCCESS);
4723 		}
4724 	}
4725 
4726 #if defined(PCMCIA_DEBUG)
4727 	if (pcmcia_debug) {
4728 		cmn_err(CE_CONT, "pcmcia_add_intr_impl() let adapter do it\n");
4729 	}
4730 #endif
4731 	pispec->intrspec_func = (uint32_t (*)())handler.handler;
4732 
4733 	/* set default IPL then check for override */
4734 
4735 	pispec->intrspec_pri = sockp->ls_intr_pri;
4736 	hdlp->ih_pri = pispec->intrspec_pri;
4737 
4738 #if defined(PCMCIA_DEBUG)
4739 	if (pcmcia_debug) {
4740 		cmn_err(CE_CONT, "pcmcia_add_intr_impl() socket=%d irq=%d"
4741 		    " handler_id=0X%x handler=%p arg1=%p arg2=%p\n",
4742 		    handler.socket, handler.irq,
4743 		    handler.handler_id, (void *)handler.handler, handler.arg1,
4744 		    handler.arg2);
4745 	}
4746 #endif
4747 
4748 	if ((ret = SET_IRQ(sockp->ls_if, adapt->pca_dip, &handler)) !=
4749 	    SUCCESS) {
4750 		sockp->ls_error = ret;
4751 		return (DDI_FAILURE);
4752 	}
4753 
4754 #if defined(PCMCIA_DEBUG)
4755 	if (pcmcia_debug) {
4756 		cmn_err(CE_CONT, "pcmcia_add_intr_impl()"
4757 		    " iblk_cookie=%p idev_cookie=%p\n"
4758 		    " ls_flags=0X%x PCS_COOKIES_VALID=0X%x\n",
4759 		    (void *)handler.iblk_cookie,
4760 		    (void *)handler.idev_cookie,
4761 		    sockp->ls_flags, PCS_COOKIES_VALID);
4762 	}
4763 #endif
4764 
4765 	if (!(sockp->ls_flags & PCS_COOKIES_VALID)) {
4766 		hdlp->ih_pri = (uint_t)*handler.iblk_cookie;
4767 		sockp->ls_iblk = *handler.iblk_cookie;
4768 		sockp->ls_idev = *handler.idev_cookie;
4769 		sockp->ls_flags |= PCS_COOKIES_VALID;
4770 	}
4771 
4772 	return (DDI_SUCCESS);
4773 }
4774 
4775 void
4776 pcmcia_remove_intr_impl(dev_info_t *dip, dev_info_t *rdip,
4777     ddi_intr_handle_impl_t *hdlp)
4778 {
4779 
4780 	struct pcmcia_parent_private *ppd;
4781 	pcmcia_logical_socket_t *sockp;
4782 	clear_irq_handler_t handler;
4783 	struct intrspec *pispec;
4784 	int socket;
4785 
4786 #if defined(PCMCIA_DEBUG)
4787 	if (pcmcia_debug) {
4788 		cmn_err(CE_CONT, "pcmcia_remove_intr_impl() entered"
4789 		    " dip=%p rdip=%p hdlp=%p\n",
4790 		    (void *)dip, (void *)rdip, (void *)hdlp);
4791 	}
4792 #endif
4793 
4794 	ppd = (struct pcmcia_parent_private *)ddi_get_parent_data(rdip);
4795 	socket = ppd->ppd_socket;
4796 	sockp = pcmcia_sockets[socket];
4797 	pispec = ppd->ppd_intrspec;
4798 
4799 #if defined(PCMCIA_DEBUG)
4800 	if (pcmcia_debug) {
4801 		cmn_err(CE_CONT, "pcmcia_remove_intr_impl()"
4802 		    " ls_inthandlers=%p ls_intrspec=%p\n",
4803 		    (void *)sockp->ls_inthandlers,
4804 		    (void *)&sockp->ls_intrspec);
4805 	}
4806 #endif
4807 
4808 	/* first handle the multifunction case since it is simple */
4809 	mutex_enter(&sockp->ls_ilock);
4810 	if (sockp->ls_inthandlers != NULL) {
4811 		/* we must be MFC */
4812 		inthandler_t *intr;
4813 		int remhandler = 0;
4814 		intr = sockp->ls_inthandlers;
4815 
4816 		/* Check if there is only one handler left */
4817 		if ((intr->next == intr) && (intr->prev == intr)) {
4818 			if (intr->handler_id == (unsigned)rdip) {
4819 				sockp->ls_inthandlers = NULL;
4820 				remhandler++;
4821 				kmem_free(intr, sizeof (inthandler_t));
4822 			}
4823 		} else {
4824 			inthandler_t *first;
4825 			int done;
4826 
4827 			for (done = 0, first = intr; !done; intr = intr->next) {
4828 				if (intr->next == first)
4829 					done++;
4830 				if (intr->handler_id == (unsigned)rdip) {
4831 					done++;
4832 
4833 					/*
4834 					 * If we're about to remove the
4835 					 *	handler at the head of
4836 					 *	the list, make the next
4837 					 *	handler in line the head.
4838 					 */
4839 					if (sockp->ls_inthandlers == intr)
4840 						sockp->ls_inthandlers =
4841 						    intr->next;
4842 
4843 					remque(intr);
4844 					kmem_free(intr, sizeof (inthandler_t));
4845 					break;
4846 				} /* handler_id */
4847 			} /* for */
4848 		} /* intr->next */
4849 
4850 		if (!remhandler) {
4851 			mutex_exit(&sockp->ls_ilock);
4852 			return;
4853 		}
4854 
4855 		/* need to get the dip that was used to add the handler */
4856 		rdip = sockp->ls_mfintr_dip;
4857 	}
4858 
4859 	mutex_exit(&sockp->ls_ilock);
4860 
4861 #if defined(PCMCIA_DEBUG)
4862 	if (pcmcia_debug) {
4863 		cmn_err(CE_CONT, "pcmcia_remove_intr_impl()"
4864 		    " pispec=%p rdip=%p\n",
4865 		    (void *)pispec, (void *)rdip);
4866 	}
4867 #endif
4868 
4869 	handler.socket = sockp->ls_socket;
4870 	handler.handler_id = (uint32_t)rdip;
4871 	handler.handler = (f_tt *)pispec->intrspec_func;
4872 	CLEAR_IRQ(sockp->ls_if, dip, &handler);
4873 }
4874 
4875 
4876 /* Consolidated interrupt processing interface */
4877 /*ARGSUSED*/
4878 int
4879 pcmcia_intr_ops(dev_info_t *dip, dev_info_t *rdip, ddi_intr_op_t intr_op,
4880     ddi_intr_handle_impl_t *hdlp, void *result)
4881 {
4882 	int	ret = DDI_SUCCESS;
4883 
4884 #if defined(PCMCIA_DEBUG)
4885 	if (pcmcia_debug) {
4886 		cmn_err(CE_CONT, "pcmcia_intr_ops() intr_op=%d\n",
4887 		    (int)intr_op);
4888 	}
4889 #endif
4890 
4891 	switch (intr_op) {
4892 	case DDI_INTROP_GETCAP:
4893 		*(int *)result = DDI_INTR_FLAG_LEVEL;
4894 		break;
4895 	case DDI_INTROP_SETCAP:
4896 		ret = DDI_ENOTSUP;
4897 		break;
4898 	case DDI_INTROP_ALLOC:
4899 		*(int *)result = hdlp->ih_scratch1;
4900 		break;
4901 	case DDI_INTROP_FREE:
4902 		break;
4903 	case DDI_INTROP_GETPRI:
4904 		if (pcmcia_add_intr_impl(dip, rdip, hdlp) != DDI_SUCCESS)
4905 			return (DDI_FAILURE);
4906 		*(int *)result = hdlp->ih_pri;
4907 		pcmcia_remove_intr_impl(dip, rdip, hdlp);
4908 		break;
4909 	case DDI_INTROP_SETPRI:
4910 		break;
4911 	case DDI_INTROP_ADDISR:
4912 		ret = pcmcia_add_intr_impl(dip, rdip, hdlp);
4913 		break;
4914 	case DDI_INTROP_REMISR:
4915 		pcmcia_remove_intr_impl(dip, rdip, hdlp);
4916 		break;
4917 	case DDI_INTROP_ENABLE:
4918 	case DDI_INTROP_DISABLE:
4919 		break;
4920 	case DDI_INTROP_NINTRS:
4921 	case DDI_INTROP_NAVAIL:
4922 		*(int *)result = i_ddi_get_nintrs(rdip);
4923 		break;
4924 	case DDI_INTROP_SUPPORTED_TYPES:
4925 		/* PCI nexus driver supports only fixed interrupts */
4926 		*(int *)result = i_ddi_get_nintrs(rdip) ?
4927 		    DDI_INTR_TYPE_FIXED : 0;
4928 		break;
4929 	default:
4930 		ret = DDI_ENOTSUP;
4931 		break;
4932 	}
4933 
4934 	return (ret);
4935 }
4936 
4937 #elif defined(__x86) || defined(__amd64)
4938 
4939 static struct intrspec	*pcmcia_intr_get_ispec(dev_info_t *, int,
4940 			    pcmcia_logical_socket_t **);
4941 static struct intrspec	*pcmcia_intr_add_isr(dev_info_t *, dev_info_t *,
4942 			    ddi_intr_handle_impl_t *);
4943 static int		pcmcia_intr_enable_isr(dev_info_t *, dev_info_t *,
4944 			    ddi_intr_handle_impl_t *);
4945 static void		pcmcia_intr_remove_isr(dev_info_t *, dev_info_t *,
4946 			    ddi_intr_handle_impl_t *);
4947 static void		pcmcia_intr_disable_isr(dev_info_t *, dev_info_t *,
4948 			    ddi_intr_handle_impl_t *);
4949 
4950 /*
4951  * pcmcia_intr_get_ispec:
4952  *	This is mostly copied from older 'pcmcia_get_intrspec' function
4953  */
4954 static struct intrspec *
4955 pcmcia_intr_get_ispec(dev_info_t *rdip, int inum,
4956     pcmcia_logical_socket_t **sockp)
4957 {
4958 	int				socket;
4959 	struct intrspec			*intrspec;
4960 	struct pcmcia_parent_private	*ppd;
4961 
4962 	if ((int)inum > 0 || (ddi_getprop(DDI_DEV_T_ANY, rdip,
4963 	    DDI_PROP_DONTPASS, "interrupts", -1) < 0))
4964 		return (NULL);
4965 
4966 	ppd = (struct pcmcia_parent_private *)ddi_get_parent_data(rdip);
4967 	if (ppd == NULL || ppd->ppd_intrspec == NULL)
4968 		return (NULL);
4969 
4970 	if ((socket = ppd->ppd_socket) < 0)
4971 		return (NULL);
4972 
4973 	if ((*sockp = pcmcia_sockets[socket]) == NULL)
4974 		return (NULL);
4975 
4976 	intrspec = ppd->ppd_intrspec;
4977 	if (intrspec->intrspec_vec == 0 && (*sockp)->ls_intr_vec != 0)
4978 		intrspec->intrspec_vec = (*sockp)->ls_intr_vec;
4979 
4980 	return (intrspec);
4981 }
4982 
4983 static struct intrspec *
4984 pcmcia_intr_add_isr(dev_info_t *dip, dev_info_t *rdip,
4985     ddi_intr_handle_impl_t *hdlp)
4986 {
4987 	int				socket;
4988 	struct intrspec			*ispecp;
4989 	struct pcmcia_adapter		*adapt;
4990 	pcmcia_logical_socket_t		*sockp;
4991 	struct pcmcia_parent_private	*ppd;
4992 
4993 #if defined(PCMCIA_DEBUG)
4994 	if (pcmcia_debug)
4995 		cmn_err(CE_CONT, "pcmcia_intr_add_isr: "
4996 		    "dip=0x%p rdip=0x%p hdlp=0x%p\n",
4997 		    (void *)dip, (void *)rdip, (void *)hdlp);
4998 #endif	/* PCMCIA_DEBUG */
4999 
5000 	ppd = (struct pcmcia_parent_private *)ddi_get_parent_data(rdip);
5001 	socket = ppd->ppd_socket;
5002 	sockp = pcmcia_sockets[socket];
5003 	adapt = sockp->ls_adapter;
5004 
5005 	ispecp = ppd->ppd_intrspec;
5006 	if (ispecp == NULL) {
5007 		sockp->ls_error = BAD_IRQ;
5008 		return (ispecp);
5009 	}
5010 
5011 	/*
5012 	 * check if multifunction and do the right thing
5013 	 * we put an intercept in between the mfc handler and us so we can
5014 	 * catch and process. We might be able to optimize this depending
5015 	 * on the card features (a future option).
5016 	 */
5017 	if (ppd->ppd_flags & PPD_CARD_MULTI &&
5018 	    hdlp->ih_cb_func != pcmcia_mfc_intr) {
5019 		inthandler_t *intr;
5020 
5021 		/*
5022 		 * note that the first function is a special case since it
5023 		 * sets things up.  We fall through to the lower code and
5024 		 * get the hardware set up. Subsequent times we just lock
5025 		 * the list and insert the handler and all is well.
5026 		 */
5027 		intr = kmem_zalloc(sizeof (inthandler_t), KM_NOSLEEP);
5028 		if (intr == NULL) {
5029 			sockp->ls_error = BAD_IRQ;
5030 			return (NULL);
5031 		}
5032 
5033 		intr->intr = (uint32_t (*)())hdlp->ih_cb_func;
5034 		intr->handler_id = (uint32_t)(uintptr_t)rdip;
5035 		intr->arg1 = hdlp->ih_cb_arg1;
5036 		intr->arg2 = hdlp->ih_cb_arg2;
5037 		intr->socket = socket;
5038 		mutex_enter(&sockp->ls_ilock);
5039 		if (sockp->ls_inthandlers == NULL) {
5040 			intr->next = intr->prev = intr;
5041 			sockp->ls_inthandlers = intr;
5042 			sockp->ls_mfintr_dip = rdip;
5043 		} else {
5044 			insque(intr, sockp->ls_inthandlers);
5045 		}
5046 		mutex_exit(&sockp->ls_ilock);
5047 		return (ispecp);
5048 	}
5049 
5050 	/*
5051 	 * Do we need to allocate an IRQ at this point or not?
5052 	 */
5053 	if (adapt->pca_flags & PCA_RES_NEED_IRQ) {
5054 		int i, irq;
5055 
5056 		/*
5057 		 * this adapter needs IRQ allocations
5058 		 * this is only necessary if it is the first function on the
5059 		 * card being setup. The socket will keep the allocation info
5060 		 */
5061 		/* all functions use same intrspec except mfc handler */
5062 		if (hdlp->ih_cb_func == pcmcia_mfc_intr) {
5063 			/*
5064 			 * We treat this special in order to allow things to
5065 			 * work properly for MFC cards. The intrspec for the
5066 			 * mfc dispatcher is intercepted and taken from the
5067 			 * logical socket in order to not be trying to
5068 			 * multiplex the meaning when ENABLE is called.
5069 			 */
5070 			ispecp = &sockp->ls_intrspec;
5071 			hdlp->ih_private = (void *)ispecp;
5072 		}
5073 
5074 		if (adapt->pca_flags & PCA_IRQ_ISA) {
5075 			for (irq = -1, i = 1; irq == -1 && i < 16; i++) {
5076 				/* find available and usable IRQ level */
5077 				if (adapt->pca_avail_intr & (1 << i))
5078 					irq = pcmcia_get_intr(dip, i);
5079 			}
5080 		}
5081 		if (irq < 0) {
5082 			sockp->ls_error = NO_RESOURCE;
5083 			return (NULL);
5084 		}
5085 		hdlp->ih_vector = sockp->ls_intr_vec = irq;
5086 
5087 
5088 #if defined(PCMCIA_DEBUG)
5089 		if (pcmcia_debug)
5090 			cmn_err(CE_CONT, "allocated irq=%x\n", irq);
5091 #endif	/* PCMCIA_DEBUG */
5092 
5093 		ispecp->intrspec_vec = sockp->ls_intr_vec;
5094 		ispecp->intrspec_pri = sockp->ls_intr_pri;
5095 		return (ispecp);
5096 	}
5097 
5098 	if (ispecp->intrspec_func != NULL)
5099 		ispecp->intrspec_func = hdlp->ih_cb_func;
5100 
5101 	/* set default IPL then check for override */
5102 	ispecp->intrspec_pri = sockp->ls_intr_pri;
5103 	return (ispecp);
5104 }
5105 
5106 
5107 static int
5108 pcmcia_intr_enable_isr(dev_info_t *dip, dev_info_t *rdip,
5109     ddi_intr_handle_impl_t *hdlp)
5110 {
5111 	int				socket, ret;
5112 	int				irq = 0;	/* default case */
5113 	dev_info_t			*parent = ddi_root_node();
5114 	struct intrspec			*ispecp;
5115 	set_irq_handler_t		handler;
5116 	struct pcmcia_adapter		*adapt;
5117 	pcmcia_logical_socket_t		*sockp;
5118 	struct pcmcia_parent_private	*ppd;
5119 
5120 #if defined(PCMCIA_DEBUG)
5121 	if (pcmcia_debug)
5122 		cmn_err(CE_CONT, "pcmcia_intr_enable_isr: "
5123 		    "dip=0x%p rdip=0x%p hdlp=0x%p\n",
5124 		    (void *)dip, (void *)rdip, (void *)hdlp);
5125 #endif	/* PCMCIA_DEBUG */
5126 
5127 	ppd = (struct pcmcia_parent_private *)ddi_get_parent_data(rdip);
5128 	socket = ppd->ppd_socket;
5129 	sockp = pcmcia_sockets[socket];
5130 	adapt = sockp->ls_adapter;
5131 
5132 	ispecp = ppd->ppd_intrspec;
5133 	ASSERT(ispecp);
5134 
5135 	mutex_enter(&sockp->ls_ilock);
5136 	if ((sockp->ls_inthandlers != NULL) &&
5137 	    (struct intrspec *)hdlp->ih_private != &sockp->ls_intrspec) {
5138 		inthandler_t *intr = sockp->ls_inthandlers;
5139 
5140 		ASSERT(ppd->ppd_flags & PPD_CARD_MULTI);
5141 
5142 		/* Only one handler. So, call ddi_add_intr on it */
5143 		if ((intr->next == intr) && (intr->prev == intr)) {
5144 			hdlp->ih_cb_func = pcmcia_mfc_intr;
5145 			hdlp->ih_cb_arg1 = (caddr_t)sockp;
5146 			hdlp->ih_cb_arg2 = NULL;
5147 
5148 			ret = (*(DEVI(parent)->devi_ops->devo_bus_ops->
5149 			    bus_intr_op))(parent, rdip, DDI_INTROP_ENABLE,
5150 			    hdlp, NULL);
5151 
5152 			if (ret == DDI_FAILURE) {
5153 				sockp->ls_inthandlers = NULL;
5154 				kmem_free(intr, sizeof (inthandler_t));
5155 				sockp->ls_error = BAD_IRQ;
5156 				mutex_exit(&sockp->ls_ilock);
5157 				return (ret);
5158 			}
5159 		}
5160 		mutex_exit(&sockp->ls_ilock);
5161 		hdlp->ih_vector = ispecp->intrspec_vec = sockp->ls_intr_vec;
5162 		hdlp->ih_pri = sockp->ls_intr_pri;
5163 		sockp->ls_iblk = (ddi_iblock_cookie_t)(uintptr_t)
5164 		    sockp->ls_intr_pri;
5165 		sockp->ls_idev.idev_vector = (ushort_t)hdlp->ih_vector;
5166 		sockp->ls_idev.idev_priority = (ushort_t)sockp->ls_intr_pri;
5167 		return (DDI_SUCCESS);
5168 	}
5169 	mutex_exit(&sockp->ls_ilock);
5170 
5171 	if (adapt->pca_flags & PCA_RES_NEED_IRQ) {
5172 		if (hdlp->ih_cb_func == pcmcia_mfc_intr)
5173 			ispecp = (struct intrspec *)&sockp->ls_intrspec;
5174 
5175 		/* XXX: remove it later as this is done in _add_isr as well */
5176 		ispecp->intrspec_vec = sockp->ls_intr_vec;
5177 		ispecp->intrspec_pri = sockp->ls_intr_pri;
5178 
5179 		/* Enable interrupts */
5180 		ret = (*(DEVI(parent)->devi_ops->devo_bus_ops->bus_intr_op))(
5181 		    parent, rdip, DDI_INTROP_ENABLE, hdlp, NULL);
5182 
5183 		sockp->ls_iblk = (ddi_iblock_cookie_t)(uintptr_t)
5184 		    sockp->ls_intr_pri;
5185 		sockp->ls_idev.idev_vector = (ushort_t)sockp->ls_intr_vec;
5186 		sockp->ls_idev.idev_priority = (ushort_t)sockp->ls_intr_pri;
5187 
5188 		if (ret != DDI_SUCCESS)
5189 			sockp->ls_error = BAD_IRQ;
5190 		return (ret);
5191 	}
5192 
5193 #if defined(PCMCIA_DEBUG)
5194 	if (pcmcia_debug)
5195 		cmn_err(CE_CONT, "pcmcia_intr_enable_isr; let adapter do it\n");
5196 #endif	/* PCMCIA_DEBUG */
5197 
5198 	handler.socket = sockp->ls_socket;
5199 	handler.irq = irq;
5200 	handler.handler = (f_tt *)hdlp->ih_cb_func;
5201 	handler.arg1 = hdlp->ih_cb_arg1;
5202 	handler.arg2 = hdlp->ih_cb_arg2;
5203 	handler.handler_id = (uint32_t)(uintptr_t)rdip;
5204 	if (ispecp->intrspec_func != NULL)
5205 		ispecp->intrspec_func = hdlp->ih_cb_func;
5206 
5207 	/* set default IPL then check for override */
5208 	ispecp->intrspec_pri = sockp->ls_intr_pri;
5209 
5210 	if ((ret = SET_IRQ(sockp->ls_if, adapt->pca_dip, &handler)) !=
5211 	    SUCCESS) {
5212 		sockp->ls_error = ret;
5213 		return (DDI_FAILURE);
5214 	}
5215 	ispecp->intrspec_func = hdlp->ih_cb_func;
5216 	if (!(sockp->ls_flags & PCS_COOKIES_VALID)) {
5217 		sockp->ls_iblk = *handler.iblk_cookie;
5218 		sockp->ls_idev = *handler.idev_cookie;
5219 		sockp->ls_flags |= PCS_COOKIES_VALID;
5220 	}
5221 	return (DDI_SUCCESS);
5222 }
5223 
5224 /* ARGSUSED */
5225 static void
5226 pcmcia_intr_remove_isr(dev_info_t *dip, dev_info_t *rdip,
5227     ddi_intr_handle_impl_t *hdlp)
5228 {
5229 	int				done, remhandler = 0;
5230 	inthandler_t			*intr, *first;
5231 	struct intrspec			*ispecp;
5232 	pcmcia_logical_socket_t		*sockp;
5233 
5234 #if defined(PCMCIA_DEBUG)
5235 	if (pcmcia_debug)
5236 		cmn_err(CE_CONT, "pcmcia_intr_remove_isr: "
5237 		    "dip=0x%p rdip=0x%p hdlp=0x%p\n",
5238 		    (void *)dip, (void *)rdip, (void *)hdlp);
5239 #endif	/* PCMCIA_DEBUG */
5240 
5241 	ispecp = pcmcia_intr_get_ispec(rdip, hdlp->ih_inum, &sockp);
5242 	ASSERT(ispecp);
5243 
5244 	/* first handle the multifunction case since it is simple */
5245 	mutex_enter(&sockp->ls_ilock);
5246 	if (sockp->ls_inthandlers != NULL &&
5247 	    (struct intrspec *)hdlp->ih_private != &sockp->ls_intrspec) {
5248 
5249 		intr = sockp->ls_inthandlers;
5250 
5251 		/* Check if there is only one handler left */
5252 		if ((intr->next == intr) && (intr->prev == intr)) {
5253 			if (intr->handler_id == (uint32_t)(uintptr_t)rdip) {
5254 				sockp->ls_inthandlers = NULL;
5255 				remhandler++;
5256 				kmem_free(intr, sizeof (inthandler_t));
5257 			}
5258 
5259 		} else {
5260 			for (done = 0, first = intr; !done; intr = intr->next) {
5261 				if (intr->next == first)
5262 					done++;
5263 				if (intr->handler_id ==
5264 				    (uint32_t)(uintptr_t)rdip) {
5265 					done++;
5266 
5267 					/*
5268 					 * If we're about to remove the handler
5269 					 * at the head of the list, make the
5270 					 * next handler in line the head.
5271 					 */
5272 					if (sockp->ls_inthandlers == intr)
5273 					    sockp->ls_inthandlers = intr->next;
5274 
5275 					remque(intr);
5276 					kmem_free(intr, sizeof (inthandler_t));
5277 					break;
5278 				} /* handler_id */
5279 			} /* end of for */
5280 		} /* end of if intr->next */
5281 
5282 		if (!remhandler) {
5283 			mutex_exit(&sockp->ls_ilock);
5284 			return;
5285 		}
5286 	}
5287 	mutex_exit(&sockp->ls_ilock);
5288 
5289 	if (sockp->ls_adapter->pca_flags & PCA_RES_NEED_IRQ) {
5290 		sockp->ls_intr_vec = 0;
5291 		ispecp->intrspec_vec = 0;
5292 	}
5293 }
5294 
5295 
5296 static void
5297 pcmcia_intr_disable_isr(dev_info_t *dip, dev_info_t *rdip,
5298     ddi_intr_handle_impl_t *hdlp)
5299 {
5300 	int				socket, ret;
5301 	dev_info_t			*parent;
5302 	struct intrspec			*ispecp;
5303 	clear_irq_handler_t		handler;
5304 	struct pcmcia_adapter		*adapt;
5305 	pcmcia_logical_socket_t		*sockp;
5306 	struct pcmcia_parent_private	*ppd;
5307 
5308 #if defined(PCMCIA_DEBUG)
5309 	if (pcmcia_debug)
5310 		cmn_err(CE_CONT, "pcmcia_intr_disable_isr: "
5311 		    "dip=0x%p rdip=0x%p hdlp=0x%p\n",
5312 		    (void *)dip, (void *)rdip, (void *)hdlp);
5313 #endif	/* PCMCIA_DEBUG */
5314 
5315 	ppd = (struct pcmcia_parent_private *)ddi_get_parent_data(rdip);
5316 	socket = ppd->ppd_socket;
5317 	sockp = pcmcia_sockets[socket];
5318 	adapt = sockp->ls_adapter;
5319 	ispecp = ppd->ppd_intrspec;
5320 	ASSERT(ispecp);
5321 
5322 	mutex_enter(&sockp->ls_ilock);
5323 	if (sockp->ls_inthandlers != NULL &&
5324 	    (struct intrspec *)hdlp->ih_private != &sockp->ls_intrspec) {
5325 		inthandler_t	*intr = sockp->ls_inthandlers;
5326 
5327 		/* Check if there is only one handler left */
5328 		if ((intr->next == intr) && (intr->prev == intr)) {
5329 			if (intr->handler_id != (uint32_t)(uintptr_t)rdip)
5330 				/*
5331 				 * need to get the dip that was
5332 				 * used to add the handler
5333 				 */
5334 				rdip = sockp->ls_mfintr_dip;
5335 				ispecp = (struct intrspec *)&sockp->ls_intrspec;
5336 		} else {
5337 			/* Don't call cleanup if list still has members */
5338 			mutex_exit(&sockp->ls_ilock);
5339 			return;
5340 		}
5341 	}
5342 	mutex_exit(&sockp->ls_ilock);
5343 
5344 	if ((struct intrspec *)hdlp->ih_private ==
5345 	    (struct intrspec *)&sockp->ls_intrspec)
5346 		ispecp = (struct intrspec *)hdlp->ih_private;
5347 
5348 	if (adapt->pca_flags & PCA_RES_NEED_IRQ) {
5349 		ret = ispecp->intrspec_vec;
5350 		parent = ddi_root_node();
5351 		ret = (*(DEVI(parent)->devi_ops->devo_bus_ops->bus_intr_op))(
5352 		    parent, rdip, DDI_INTROP_DISABLE, hdlp, NULL);
5353 		(void) pcmcia_return_intr(dip, hdlp->ih_vector);
5354 #if defined(PCMCIA_DEBUG)
5355 		if (pcmcia_debug)
5356 			cmn_err(CE_CONT, "pcmcia_intr_disable_isr: "
5357 			    "INTROP_DISABLE returned %x\n", ret);
5358 #endif	/* PCMCIA_DEBUG */
5359 	} else {
5360 		handler.socket = sockp->ls_socket;
5361 		handler.handler_id = (uint32_t)(uintptr_t)rdip;
5362 		handler.handler = (f_tt *)ispecp->intrspec_func;
5363 		ret = CLEAR_IRQ(sockp->ls_if, dip, &handler);
5364 #if defined(PCMCIA_DEBUG)
5365 		if (pcmcia_debug)
5366 			cmn_err(CE_CONT, "pcmcia_intr_disable_isr: "
5367 			    "CLEAR_IRQ returned %x\n", ret);
5368 #endif	/* PCMCIA_DEBUG */
5369 	}
5370 }
5371 
5372 /* Consolidated interrupt processing interface */
5373 int
5374 pcmcia_intr_ops(dev_info_t *dip, dev_info_t *rdip, ddi_intr_op_t intr_op,
5375     ddi_intr_handle_impl_t *hdlp, void *result)
5376 {
5377 	struct intrspec		*ispecp;
5378 	pcmcia_logical_socket_t	*sockp;
5379 
5380 #if defined(PCMCIA_DEBUG)
5381 	if (pcmcia_debug)
5382 		cmn_err(CE_CONT, "pcmcia_intr_ops: "
5383 		    "dip=0x%p rdip=0x%p op=0x%x hdlp=0x%p\n",
5384 		    (void *)dip, (void *)rdip, intr_op, (void *)hdlp);
5385 #endif	/* PCMCIA_DEBUG */
5386 
5387 	switch (intr_op) {
5388 	case DDI_INTROP_SUPPORTED_TYPES:
5389 		if (ddi_get_parent_data(rdip) == NULL) {
5390 			*(int *)result = 0;
5391 			return (DDI_FAILURE);
5392 		}
5393 		*(int *)result = DDI_INTR_TYPE_FIXED;
5394 		break;
5395 	case DDI_INTROP_GETCAP:
5396 		*(int *)result = DDI_INTR_FLAG_LEVEL;
5397 		break;
5398 	case DDI_INTROP_NINTRS:
5399 	case DDI_INTROP_NAVAIL:
5400 		if (ddi_get_parent_data(rdip) == NULL) {
5401 			*(int *)result = 0;
5402 			return (DDI_FAILURE);
5403 		}
5404 		*(int *)result = 1;	/* for PCMCIA there is only one intr */
5405 		break;
5406 	case DDI_INTROP_ALLOC:
5407 		if ((ispecp = pcmcia_intr_get_ispec(rdip, hdlp->ih_inum,
5408 		    &sockp)) == NULL)
5409 			return (DDI_FAILURE);
5410 		*(int *)result = hdlp->ih_scratch1;
5411 		hdlp->ih_private = (void *)ispecp;
5412 		break;
5413 	case DDI_INTROP_FREE:
5414 		hdlp->ih_private = NULL;
5415 		break;
5416 	case DDI_INTROP_GETPRI:
5417 		ispecp = pcmcia_intr_get_ispec(rdip, hdlp->ih_inum, &sockp);
5418 		if (ispecp == NULL) {
5419 			*(int *)result = 0;
5420 			return (DDI_FAILURE);
5421 		}
5422 
5423 		*(int *)result = ispecp->intrspec_pri = sockp->ls_intr_pri;
5424 		break;
5425 	case DDI_INTROP_SETPRI:
5426 		if (*(int *)result > LOCK_LEVEL)
5427 			return (DDI_FAILURE);
5428 		ispecp = pcmcia_intr_get_ispec(rdip, hdlp->ih_inum, &sockp);
5429 		ASSERT(ispecp);
5430 		ispecp->intrspec_pri = sockp->ls_intr_pri = *(int *)result;
5431 		break;
5432 	case DDI_INTROP_ADDISR:
5433 		if ((ispecp = pcmcia_intr_add_isr(dip, rdip, hdlp)) == NULL)
5434 			return (DDI_FAILURE);
5435 		hdlp->ih_private = (void *)ispecp;
5436 		break;
5437 	case DDI_INTROP_REMISR:
5438 		pcmcia_intr_remove_isr(dip, rdip, hdlp);
5439 		break;
5440 	case DDI_INTROP_ENABLE:
5441 		if (pcmcia_intr_enable_isr(dip, rdip, hdlp) != DDI_SUCCESS)
5442 			return (DDI_FAILURE);
5443 		break;
5444 	case DDI_INTROP_DISABLE:
5445 		pcmcia_intr_disable_isr(dip, rdip, hdlp);
5446 		break;
5447 	default:
5448 		return (DDI_ENOTSUP);
5449 	}
5450 
5451 	return (DDI_SUCCESS);
5452 }
5453 #endif
5454