xref: /titanic_50/usr/src/uts/common/io/1394/t1394.c (revision 2449e17f82f6097fd2c665b64723e31ceecbeca6)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 /*
22  * Copyright 2006 Sun Microsystems, Inc.  All rights reserved.
23  * Use is subject to license terms.
24  */
25 
26 #pragma ident	"%Z%%M%	%I%	%E% SMI"
27 
28 /*
29  * t1394.c
30  *    1394 Target Driver Interface
31  *    This file contains all of the 1394 Software Framework routines called
32  *    by target drivers
33  */
34 
35 #include <sys/conf.h>
36 #include <sys/ddi.h>
37 #include <sys/sunddi.h>
38 #include <sys/types.h>
39 #include <sys/kmem.h>
40 #include <sys/disp.h>
41 #include <sys/tnf_probe.h>
42 
43 #include <sys/1394/t1394.h>
44 #include <sys/1394/s1394.h>
45 #include <sys/1394/h1394.h>
46 #include <sys/1394/ieee1394.h>
47 
48 static int s1394_allow_detach = 0;
49 
50 /*
51  * Function:    t1394_attach()
52  * Input(s):    dip			The dip given to the target driver
53  *					    in it's attach() routine
54  *		version			The version of the target driver -
55  *					    T1394_VERSION_V1
56  *		flags			The flags parameter is unused (for now)
57  *
58  * Output(s):	attachinfo		Used to pass info back to target,
59  *					    including bus generation, local
60  *					    node ID, dma attribute, etc.
61  *		t1394_hdl		The target "handle" to be used for
62  *					    all subsequent calls into the
63  *					    1394 Software Framework
64  *
65  * Description:	t1394_attach() registers the target (based on its dip) with
66  *		the 1394 Software Framework.  It returns the bus_generation,
67  *		local_nodeID, iblock_cookie and other useful information to
68  *		the target, as well as a handle (t1394_hdl) that will be used
69  *		in all subsequent calls into this framework.
70  */
71 /* ARGSUSED */
72 int
73 t1394_attach(dev_info_t *dip, int version, uint_t flags,
74     t1394_attachinfo_t *attachinfo, t1394_handle_t *t1394_hdl)
75 {
76 	s1394_hal_t	*hal;
77 	s1394_target_t	*target;
78 	uint_t		dev;
79 	uint_t		curr;
80 	uint_t		unit_dir;
81 	int		hp_node = 0;
82 
83 	ASSERT(t1394_hdl != NULL);
84 	ASSERT(attachinfo != NULL);
85 
86 	TNF_PROBE_0_DEBUG(t1394_attach_enter, S1394_TNF_SL_HOTPLUG_STACK, "");
87 
88 	*t1394_hdl = NULL;
89 
90 	if (version != T1394_VERSION_V1) {
91 		TNF_PROBE_1(t1394_attach_error, S1394_TNF_SL_HOTPLUG_ERROR, "",
92 		    tnf_string, msg, "Invalid version");
93 		TNF_PROBE_0_DEBUG(t1394_attach_exit,
94 		    S1394_TNF_SL_HOTPLUG_STACK, "");
95 		return (DDI_FAILURE);
96 	}
97 
98 	hal = s1394_dip_to_hal(ddi_get_parent(dip));
99 	if (hal == NULL) {
100 		TNF_PROBE_1(t1394_attach_error, S1394_TNF_SL_HOTPLUG_ERROR, "",
101 		    tnf_string, msg, "No parent dip found for target");
102 		TNF_PROBE_0_DEBUG(t1394_attach_exit,
103 		    S1394_TNF_SL_HOTPLUG_STACK, "");
104 		return (DDI_FAILURE);
105 	}
106 
107 	ASSERT(MUTEX_NOT_HELD(&hal->topology_tree_mutex));
108 
109 	hp_node = ddi_prop_exists(DDI_DEV_T_ANY, dip, DDI_PROP_DONTPASS,
110 	    "hp-node");
111 
112 	/* Allocate space for s1394_target_t */
113 	target = kmem_zalloc(sizeof (s1394_target_t), KM_SLEEP);
114 
115 	mutex_enter(&hal->topology_tree_mutex);
116 
117 	target->target_version = version;
118 
119 	/* Copy in the params */
120 	target->target_dip = dip;
121 	target->on_hal	   = hal;
122 
123 	/* Place the target on the appropriate node */
124 	target->on_node	= NULL;
125 
126 	rw_enter(&target->on_hal->target_list_rwlock, RW_WRITER);
127 	if (hp_node != 0) {
128 		s1394_add_target_to_node(target);
129 		/*
130 		 * on_node can be NULL if the node got unplugged
131 		 * while the target driver is in its attach routine.
132 		 */
133 		if (target->on_node == NULL) {
134 			s1394_remove_target_from_node(target);
135 			rw_exit(&target->on_hal->target_list_rwlock);
136 			mutex_exit(&hal->topology_tree_mutex);
137 			kmem_free(target, sizeof (s1394_target_t));
138 			TNF_PROBE_1(t1394_attach_error,
139 			    S1394_TNF_SL_HOTPLUG_ERROR, "", tnf_string, msg,
140 			    "on_node == NULL");
141 			TNF_PROBE_0_DEBUG(t1394_attach_exit,
142 			    S1394_TNF_SL_HOTPLUG_STACK, "");
143 			return (DDI_FAILURE);
144 		}
145 
146 		target->target_state = S1394_TARG_HP_NODE;
147 		if (S1394_NODE_BUS_PWR_CONSUMER(target->on_node) == B_TRUE)
148 			target->target_state |= S1394_TARG_BUS_PWR_CONSUMER;
149 	}
150 
151 	/* Return the current generation */
152 	attachinfo->localinfo.bus_generation = target->on_hal->generation_count;
153 
154 	/* Fill in hal node id */
155 	attachinfo->localinfo.local_nodeID = target->on_hal->node_id;
156 
157 	/* Give the target driver the iblock_cookie */
158 	attachinfo->iblock_cookie = target->on_hal->halinfo.hw_interrupt;
159 
160 	/* Give the target driver the attributes */
161 	attachinfo->acc_attr	= target->on_hal->halinfo.acc_attr;
162 	attachinfo->dma_attr	= target->on_hal->halinfo.dma_attr;
163 
164 	unit_dir = ddi_prop_get_int(DDI_DEV_T_ANY, dip,
165 		DDI_PROP_DONTPASS, "unit-dir-offset", 0);
166 	target->unit_dir = unit_dir;
167 
168 	/* By default, disable all physical AR requests */
169 	target->physical_arreq_enabled = 0;
170 
171 
172 	/* Get dev_max_payload & current_max_payload */
173 	s1394_get_maxpayload(target, &dev, &curr);
174 	target->dev_max_payload		= dev;
175 	target->current_max_payload	= curr;
176 
177 	/* Add into linked list */
178 	if ((target->on_hal->target_head == NULL) &&
179 	    (target->on_hal->target_tail == NULL)) {
180 		target->on_hal->target_head = target;
181 		target->on_hal->target_tail = target;
182 	} else {
183 		target->on_hal->target_tail->target_next = target;
184 		target->target_prev = target->on_hal->target_tail;
185 		target->on_hal->target_tail = target;
186 	}
187 	rw_exit(&target->on_hal->target_list_rwlock);
188 
189 	/* Fill in services layer private info */
190 	*t1394_hdl = (t1394_handle_t)target;
191 
192 	mutex_exit(&hal->topology_tree_mutex);
193 
194 	TNF_PROBE_0_DEBUG(t1394_attach_exit, S1394_TNF_SL_HOTPLUG_STACK, "");
195 	return (DDI_SUCCESS);
196 }
197 
198 /*
199  * Function:    t1394_detach()
200  * Input(s):    t1394_hdl		The target "handle" returned by
201  *					    t1394_attach()
202  *		flags			The flags parameter is unused (for now)
203  *
204  * Output(s):	DDI_SUCCESS		Target successfully detached
205  *		DDI_FAILURE		Target failed to detach
206  *
207  * Description:	t1394_detach() unregisters the target from the 1394 Software
208  *		Framework.  t1394_detach() can fail if the target has any
209  *		allocated commands that haven't been freed.
210  */
211 /* ARGSUSED */
212 int
213 t1394_detach(t1394_handle_t *t1394_hdl, uint_t flags)
214 {
215 	s1394_target_t	*target;
216 	uint_t		num_cmds;
217 
218 	TNF_PROBE_0_DEBUG(t1394_detach_enter, S1394_TNF_SL_HOTPLUG_STACK, "");
219 
220 	ASSERT(t1394_hdl != NULL);
221 
222 	target = (s1394_target_t *)(*t1394_hdl);
223 
224 	ASSERT(target->on_hal);
225 
226 	mutex_enter(&target->on_hal->topology_tree_mutex);
227 	rw_enter(&target->on_hal->target_list_rwlock, RW_WRITER);
228 
229 	/* How many cmds has this target allocated? */
230 	num_cmds = target->target_num_cmds;
231 
232 	if (num_cmds != 0) {
233 		rw_exit(&target->on_hal->target_list_rwlock);
234 		mutex_exit(&target->on_hal->topology_tree_mutex);
235 		TNF_PROBE_1(t1394_detach_error, S1394_TNF_SL_HOTPLUG_ERROR, "",
236 		    tnf_string, msg, "Must free all commands before detach()");
237 		TNF_PROBE_0_DEBUG(t1394_detach_exit,
238 		    S1394_TNF_SL_HOTPLUG_STACK, "");
239 		return (DDI_FAILURE);
240 	}
241 
242 	/*
243 	 * Remove from linked lists. Topology tree is already locked
244 	 * so that the node won't go away while we are looking at it.
245 	 */
246 	if ((target->on_hal->target_head == target) &&
247 	    (target->on_hal->target_tail == target)) {
248 		target->on_hal->target_head = NULL;
249 		target->on_hal->target_tail = NULL;
250 	} else {
251 		if (target->target_prev)
252 			target->target_prev->target_next = target->target_next;
253 		if (target->target_next)
254 			target->target_next->target_prev = target->target_prev;
255 		if (target->on_hal->target_head == target)
256 			target->on_hal->target_head = target->target_next;
257 		if (target->on_hal->target_tail == target)
258 			target->on_hal->target_tail = target->target_prev;
259 	}
260 
261 	s1394_remove_target_from_node(target);
262 	rw_exit(&target->on_hal->target_list_rwlock);
263 
264 	mutex_exit(&target->on_hal->topology_tree_mutex);
265 
266 	/* Free memory */
267 	kmem_free(target, sizeof (s1394_target_t));
268 
269 	*t1394_hdl = NULL;
270 
271 	TNF_PROBE_0_DEBUG(t1394_detach_exit, S1394_TNF_SL_HOTPLUG_STACK, "");
272 	return (DDI_SUCCESS);
273 }
274 
275 /*
276  * Function:    t1394_alloc_cmd()
277  * Input(s):    t1394_hdl		The target "handle" returned by
278  *					    t1394_attach()
279  *		flags			The flags parameter is described below
280  *
281  * Output(s):	cmdp			Pointer to the newly allocated command
282  *
283  * Description:	t1394_alloc_cmd() allocates a command for use with the
284  *		t1394_read(), t1394_write(), or t1394_lock() interfaces
285  *		of the 1394 Software Framework.  By default, t1394_alloc_cmd()
286  *		may sleep while allocating memory for the command structure.
287  *		If this is undesirable, the target may set the
288  *		T1394_ALLOC_CMD_NOSLEEP bit in the flags parameter.  Also,
289  *		this call may fail because a target driver has already
290  *		allocated MAX_NUMBER_ALLOC_CMDS commands.
291  */
292 int
293 t1394_alloc_cmd(t1394_handle_t t1394_hdl, uint_t flags, cmd1394_cmd_t **cmdp)
294 {
295 	s1394_hal_t	 *hal;
296 	s1394_target_t	 *target;
297 	s1394_cmd_priv_t *s_priv;
298 	uint_t		 num_cmds;
299 
300 	TNF_PROBE_0_DEBUG(t1394_alloc_cmd_enter, S1394_TNF_SL_ATREQ_STACK, "");
301 
302 	ASSERT(t1394_hdl != NULL);
303 
304 	target = (s1394_target_t *)t1394_hdl;
305 
306 	/* Find the HAL this target resides on */
307 	hal = target->on_hal;
308 
309 	rw_enter(&hal->target_list_rwlock, RW_WRITER);
310 
311 	/* How many cmds has this target allocated? */
312 	num_cmds = target->target_num_cmds;
313 
314 	if (num_cmds >= MAX_NUMBER_ALLOC_CMDS) {
315 		rw_exit(&hal->target_list_rwlock);
316 		TNF_PROBE_1(t1394_alloc_cmd_error, S1394_TNF_SL_ATREQ_ERROR,
317 		    "", tnf_string, msg, "Attempted to alloc > "
318 		    "MAX_NUMBER_ALLOC_CMDS");
319 		TNF_PROBE_0_DEBUG(t1394_alloc_cmd_exit,
320 		    S1394_TNF_SL_ATREQ_STACK, "");
321 		/* kstats - cmd alloc failures */
322 		hal->hal_kstats->cmd_alloc_fail++;
323 		return (DDI_FAILURE);
324 	}
325 
326 	/* Increment the number of cmds this target has allocated? */
327 	target->target_num_cmds = num_cmds + 1;
328 
329 	if (s1394_alloc_cmd(hal, flags, cmdp) != DDI_SUCCESS) {
330 		target->target_num_cmds = num_cmds;	/* Undo increment */
331 		rw_exit(&hal->target_list_rwlock);
332 		TNF_PROBE_1(t1394_alloc_cmd_error, S1394_TNF_SL_ATREQ_ERROR, "",
333 		    tnf_string, msg, "Failed to allocate command structure");
334 		TNF_PROBE_0_DEBUG(t1394_alloc_cmd_exit,
335 		    S1394_TNF_SL_ATREQ_STACK, "");
336 		/* kstats - cmd alloc failures */
337 		hal->hal_kstats->cmd_alloc_fail++;
338 		return (DDI_FAILURE);
339 	}
340 
341 	rw_exit(&hal->target_list_rwlock);
342 
343 	/* Get the Services Layer private area */
344 	s_priv = S1394_GET_CMD_PRIV(*cmdp);
345 
346 	/* Initialize the command's blocking mutex */
347 	mutex_init(&s_priv->blocking_mutex, NULL, MUTEX_DRIVER,
348 	    hal->halinfo.hw_interrupt);
349 
350 	/* Initialize the command's blocking condition variable */
351 	cv_init(&s_priv->blocking_cv, NULL, CV_DRIVER, NULL);
352 
353 	TNF_PROBE_0_DEBUG(t1394_alloc_cmd_exit, S1394_TNF_SL_ATREQ_STACK, "");
354 	return (DDI_SUCCESS);
355 }
356 
357 /*
358  * Function:    t1394_free_cmd()
359  * Input(s):    t1394_hdl		The target "handle" returned by
360  *					    t1394_attach()
361  *		flags			The flags parameter is unused (for now)
362  *		cmdp			Pointer to the command to be freed
363  *
364  * Output(s):	DDI_SUCCESS		Target successfully freed command
365  *		DDI_FAILURE		Target failed to free command
366  *
367  * Description:	t1394_free_cmd() attempts to free a command that has previously
368  *		been allocated by the target driver.  It is possible for
369  *		t1394_free_cmd() to fail because the command is currently
370  *		in-use by the 1394 Software Framework.
371  */
372 /* ARGSUSED */
373 int
374 t1394_free_cmd(t1394_handle_t t1394_hdl, uint_t flags, cmd1394_cmd_t **cmdp)
375 {
376 	s1394_hal_t	 *hal;
377 	s1394_target_t	 *target;
378 	s1394_cmd_priv_t *s_priv;
379 	uint_t		 num_cmds;
380 
381 	TNF_PROBE_0_DEBUG(t1394_free_cmd_enter, S1394_TNF_SL_ATREQ_STACK, "");
382 
383 	ASSERT(t1394_hdl != NULL);
384 
385 	target = (s1394_target_t *)t1394_hdl;
386 
387 	/* Find the HAL this target resides on */
388 	hal = target->on_hal;
389 
390 	rw_enter(&hal->target_list_rwlock, RW_WRITER);
391 
392 	/* How many cmds has this target allocated? */
393 	num_cmds = target->target_num_cmds;
394 
395 	if (num_cmds == 0) {
396 		rw_exit(&hal->target_list_rwlock);
397 		TNF_PROBE_2(t1394_free_cmd_error, S1394_TNF_SL_ATREQ_ERROR, "",
398 		    tnf_string, msg, "No commands left to be freed "
399 		    "(num_cmds <= 0)", tnf_uint, num_cmds, num_cmds);
400 		TNF_PROBE_0_DEBUG(t1394_free_cmd_exit,
401 		    S1394_TNF_SL_ATREQ_STACK, "");
402 		ASSERT(num_cmds != 0);
403 		return (DDI_FAILURE);
404 	}
405 
406 	/* Get the Services Layer private area */
407 	s_priv = S1394_GET_CMD_PRIV(*cmdp);
408 
409 	/* Check that command isn't in use */
410 	if (s_priv->cmd_in_use == B_TRUE) {
411 		rw_exit(&hal->target_list_rwlock);
412 		TNF_PROBE_1(t1394_free_cmd_error, S1394_TNF_SL_ATREQ_ERROR, "",
413 		    tnf_string, msg, "Attempted to free an in-use command");
414 		TNF_PROBE_0_DEBUG(t1394_free_cmd_exit,
415 		    S1394_TNF_SL_ATREQ_STACK, "");
416 		ASSERT(s_priv->cmd_in_use == B_FALSE);
417 		return (DDI_FAILURE);
418 	}
419 
420 	/* Decrement the number of cmds this target has allocated */
421 	target->target_num_cmds--;
422 
423 	rw_exit(&hal->target_list_rwlock);
424 
425 	/* Destroy the command's blocking condition variable */
426 	cv_destroy(&s_priv->blocking_cv);
427 
428 	/* Destroy the command's blocking mutex */
429 	mutex_destroy(&s_priv->blocking_mutex);
430 
431 	kmem_cache_free(hal->hal_kmem_cachep, *cmdp);
432 
433 	/* Command pointer is set to NULL before returning */
434 	*cmdp = NULL;
435 
436 	/* kstats - number of cmd frees */
437 	hal->hal_kstats->cmd_free++;
438 
439 	TNF_PROBE_0_DEBUG(t1394_free_cmd_exit, S1394_TNF_SL_ATREQ_STACK, "");
440 	return (DDI_SUCCESS);
441 }
442 
443 /*
444  * Function:    t1394_read()
445  * Input(s):    t1394_hdl		The target "handle" returned by
446  *					    t1394_attach()
447  *		cmd			Pointer to the command to send
448  *
449  * Output(s):	DDI_SUCCESS		Target successful sent the command
450  *		DDI_FAILURE		Target failed to send command
451  *
452  * Description:	t1394_read() attempts to send an asynchronous read request
453  *		onto the 1394 bus.
454  */
455 int
456 t1394_read(t1394_handle_t t1394_hdl, cmd1394_cmd_t *cmd)
457 {
458 	s1394_hal_t	  *to_hal;
459 	s1394_target_t	  *target;
460 	s1394_cmd_priv_t  *s_priv;
461 	s1394_hal_state_t state;
462 	int		  ret;
463 	int		  err;
464 
465 	TNF_PROBE_0_DEBUG(t1394_read_enter, S1394_TNF_SL_ATREQ_STACK, "");
466 
467 	ASSERT(t1394_hdl != NULL);
468 	ASSERT(cmd != NULL);
469 
470 	/* Get the Services Layer private area */
471 	s_priv = S1394_GET_CMD_PRIV(cmd);
472 
473 	/* Is this command currently in use? */
474 	if (s_priv->cmd_in_use == B_TRUE) {
475 		TNF_PROBE_1(t1394_read_error, S1394_TNF_SL_ATREQ_ERROR, "",
476 		    tnf_string, msg, "Attempted to resend an in-use command");
477 		TNF_PROBE_0_DEBUG(t1394_read_exit, S1394_TNF_SL_ATREQ_STACK,
478 		    "");
479 		ASSERT(s_priv->cmd_in_use == B_FALSE);
480 		return (DDI_FAILURE);
481 	}
482 
483 	target = (s1394_target_t *)t1394_hdl;
484 
485 	/* Set-up the destination of the command */
486 	to_hal = target->on_hal;
487 
488 	/* No status (default) */
489 	cmd->cmd_result = CMD1394_NOSTATUS;
490 
491 	/* Check for proper command type */
492 	if ((cmd->cmd_type != CMD1394_ASYNCH_RD_QUAD) &&
493 	    (cmd->cmd_type != CMD1394_ASYNCH_RD_BLOCK)) {
494 		cmd->cmd_result = CMD1394_EINVALID_COMMAND;
495 		TNF_PROBE_1(t1394_read_error, S1394_TNF_SL_ATREQ_ERROR, "",
496 		    tnf_string, msg, "Invalid command type specified");
497 		TNF_PROBE_0_DEBUG(t1394_read_exit,
498 		    S1394_TNF_SL_ATREQ_STACK, "");
499 		return (DDI_FAILURE);
500 	}
501 
502 	/* Is this a blocking command on interrupt stack? */
503 	if ((cmd->cmd_options & CMD1394_BLOCKING) &&
504 	    (servicing_interrupt())) {
505 		cmd->cmd_result = CMD1394_EINVALID_CONTEXT;
506 		TNF_PROBE_1(t1394_read_error, S1394_TNF_SL_ATREQ_ERROR, "",
507 		    tnf_string, msg, "Tried to use CMD1394_BLOCKING in "
508 		    "intr context");
509 		TNF_PROBE_0_DEBUG(t1394_read_exit,
510 		    S1394_TNF_SL_ATREQ_STACK, "");
511 		return (DDI_FAILURE);
512 	}
513 
514 	mutex_enter(&to_hal->topology_tree_mutex);
515 	state = to_hal->hal_state;
516 	if (state != S1394_HAL_NORMAL) {
517 		ret = s1394_HAL_asynch_error(to_hal, cmd, state);
518 		if (ret != CMD1394_CMDSUCCESS) {
519 			cmd->cmd_result = ret;
520 			mutex_exit(&to_hal->topology_tree_mutex);
521 			return (DDI_FAILURE);
522 		}
523 	}
524 
525 	ret = s1394_setup_asynch_command(to_hal, target, cmd,
526 	    S1394_CMD_READ, &err);
527 
528 	/* Command has now been put onto the queue! */
529 	if (ret != DDI_SUCCESS) {
530 		/* Copy error code into result */
531 		cmd->cmd_result = err;
532 		mutex_exit(&to_hal->topology_tree_mutex);
533 		TNF_PROBE_1(t1394_read_error, S1394_TNF_SL_ATREQ_ERROR, "",
534 		    tnf_string, msg, "Failed in s1394_setup_asynch_command()");
535 		TNF_PROBE_0_DEBUG(t1394_read_exit,
536 		    S1394_TNF_SL_ATREQ_STACK, "");
537 		return (DDI_FAILURE);
538 	}
539 
540 	/*
541 	 * If this command was sent during a bus reset,
542 	 * then put it onto the pending Q.
543 	 */
544 	if (state == S1394_HAL_RESET) {
545 		/* Remove cmd from outstanding request Q */
546 		s1394_remove_q_asynch_cmd(to_hal, cmd);
547 		/* Are we on the bus reset event stack? */
548 		if (s1394_on_br_thread(to_hal) == B_TRUE) {
549 			/* Blocking commands are not allowed */
550 			if (cmd->cmd_options & CMD1394_BLOCKING) {
551 				mutex_exit(&to_hal->topology_tree_mutex);
552 				s_priv->cmd_in_use = B_FALSE;
553 				cmd->cmd_result	   = CMD1394_EINVALID_CONTEXT;
554 				TNF_PROBE_1(t1394_read_error,
555 				    S1394_TNF_SL_ATREQ_ERROR, "", tnf_string,
556 				    msg, "CMD1394_BLOCKING in bus reset "
557 				    "context");
558 				TNF_PROBE_0_DEBUG(t1394_read_exit,
559 				    S1394_TNF_SL_ATREQ_STACK, "");
560 				return (DDI_FAILURE);
561 			}
562 		}
563 
564 		s1394_pending_q_insert(to_hal, cmd, S1394_PENDING_Q_FRONT);
565 		mutex_exit(&to_hal->topology_tree_mutex);
566 
567 		/* Block (if necessary) */
568 		goto block_on_asynch_cmd;
569 	}
570 	mutex_exit(&to_hal->topology_tree_mutex);
571 
572 	/* Send the command out */
573 	ret = s1394_xfer_asynch_command(to_hal, cmd, &err);
574 
575 	if (ret != DDI_SUCCESS) {
576 		if (err == CMD1394_ESTALE_GENERATION) {
577 			/* Remove cmd from outstanding request Q */
578 			s1394_remove_q_asynch_cmd(to_hal, cmd);
579 			s1394_pending_q_insert(to_hal, cmd,
580 			    S1394_PENDING_Q_FRONT);
581 
582 			/* Block (if necessary) */
583 			goto block_on_asynch_cmd;
584 
585 		} else {
586 			/* Remove cmd from outstanding request Q */
587 			s1394_remove_q_asynch_cmd(to_hal, cmd);
588 
589 			s_priv->cmd_in_use = B_FALSE;
590 
591 			/* Copy error code into result */
592 			cmd->cmd_result    = err;
593 
594 			TNF_PROBE_1(t1394_read_error, S1394_TNF_SL_ATREQ_ERROR,
595 			    "", tnf_string, msg, "Failed in "
596 			    "s1394_xfer_asynch_command()");
597 			TNF_PROBE_0_DEBUG(t1394_read_exit,
598 			    S1394_TNF_SL_ATREQ_STACK, "");
599 			return (DDI_FAILURE);
600 		}
601 	} else {
602 		/* Block (if necessary) */
603 		goto block_on_asynch_cmd;
604 	}
605 
606 block_on_asynch_cmd:
607 	s1394_block_on_asynch_cmd(cmd);
608 
609 	TNF_PROBE_0_DEBUG(t1394_read_exit,
610 	    S1394_TNF_SL_ATREQ_STACK, "");
611 	return (DDI_SUCCESS);
612 }
613 
614 /*
615  * Function:    t1394_write()
616  * Input(s):    t1394_hdl		The target "handle" returned by
617  *					    t1394_attach()
618  *		cmd			Pointer to the command to send
619  *
620  * Output(s):	DDI_SUCCESS		Target successful sent the command
621  *		DDI_FAILURE		Target failed to send command
622  *
623  * Description:	t1394_write() attempts to send an asynchronous write request
624  *		onto the 1394 bus.
625  */
626 int
627 t1394_write(t1394_handle_t t1394_hdl, cmd1394_cmd_t *cmd)
628 {
629 	s1394_hal_t	  *to_hal;
630 	s1394_target_t	  *target;
631 	s1394_cmd_priv_t  *s_priv;
632 	s1394_hal_state_t state;
633 	int		  ret;
634 	int		  err;
635 
636 	TNF_PROBE_0_DEBUG(t1394_write_enter, S1394_TNF_SL_ATREQ_STACK, "");
637 
638 	ASSERT(t1394_hdl != NULL);
639 	ASSERT(cmd != NULL);
640 
641 	/* Get the Services Layer private area */
642 	s_priv = S1394_GET_CMD_PRIV(cmd);
643 
644 	/* Is this command currently in use? */
645 	if (s_priv->cmd_in_use == B_TRUE) {
646 		TNF_PROBE_1(t1394_write_error, S1394_TNF_SL_ATREQ_ERROR, "",
647 		    tnf_string, msg, "Attempted to resend an in-use command");
648 		TNF_PROBE_0_DEBUG(t1394_write_exit, S1394_TNF_SL_ATREQ_STACK,
649 		    "");
650 		ASSERT(s_priv->cmd_in_use == B_FALSE);
651 		return (DDI_FAILURE);
652 	}
653 
654 	target = (s1394_target_t *)t1394_hdl;
655 
656 	/* Set-up the destination of the command */
657 	to_hal = target->on_hal;
658 
659 	/* Is this an FA request? */
660 	if (s_priv->cmd_ext_type == S1394_CMD_EXT_FA) {
661 		if (S1394_IS_CMD_FCP(s_priv) &&
662 		    (s1394_fcp_write_check_cmd(cmd) != DDI_SUCCESS)) {
663 			TNF_PROBE_0_DEBUG(t1394_write_exit,
664 			    S1394_TNF_SL_ATREQ_STACK, "");
665 			return (DDI_FAILURE);
666 		}
667 		s1394_fa_convert_cmd(to_hal, cmd);
668 	}
669 
670 	/* No status (default) */
671 	cmd->cmd_result = CMD1394_NOSTATUS;
672 
673 	/* Check for proper command type */
674 	if ((cmd->cmd_type != CMD1394_ASYNCH_WR_QUAD) &&
675 	    (cmd->cmd_type != CMD1394_ASYNCH_WR_BLOCK)) {
676 		cmd->cmd_result = CMD1394_EINVALID_COMMAND;
677 		s1394_fa_check_restore_cmd(to_hal, cmd);
678 		TNF_PROBE_1(t1394_write_error, S1394_TNF_SL_ATREQ_ERROR, "",
679 		    tnf_string, msg, "Invalid command type specified");
680 		TNF_PROBE_0_DEBUG(t1394_write_exit, S1394_TNF_SL_ATREQ_STACK,
681 		    "");
682 		return (DDI_FAILURE);
683 	}
684 
685 	/* Is this a blocking command on interrupt stack? */
686 	if ((cmd->cmd_options & CMD1394_BLOCKING) &&
687 	    (servicing_interrupt())) {
688 		cmd->cmd_result = CMD1394_EINVALID_CONTEXT;
689 		s1394_fa_check_restore_cmd(to_hal, cmd);
690 		TNF_PROBE_1(t1394_write_error, S1394_TNF_SL_ATREQ_ERROR, "",
691 		    tnf_string, msg, "Tried to use CMD1394_BLOCKING in intr "
692 		    "context");
693 		TNF_PROBE_0_DEBUG(t1394_write_exit, S1394_TNF_SL_ATREQ_STACK,
694 		    "");
695 		return (DDI_FAILURE);
696 	}
697 
698 	mutex_enter(&to_hal->topology_tree_mutex);
699 	state = to_hal->hal_state;
700 	if (state != S1394_HAL_NORMAL) {
701 		ret = s1394_HAL_asynch_error(to_hal, cmd, state);
702 		if (ret != CMD1394_CMDSUCCESS) {
703 			cmd->cmd_result = ret;
704 			mutex_exit(&to_hal->topology_tree_mutex);
705 			s1394_fa_check_restore_cmd(to_hal, cmd);
706 			return (DDI_FAILURE);
707 		}
708 	}
709 
710 	ret = s1394_setup_asynch_command(to_hal, target, cmd,
711 	    S1394_CMD_WRITE, &err);
712 
713 	/* Command has now been put onto the queue! */
714 	if (ret != DDI_SUCCESS) {
715 		/* Copy error code into result */
716 		cmd->cmd_result = err;
717 		mutex_exit(&to_hal->topology_tree_mutex);
718 		s1394_fa_check_restore_cmd(to_hal, cmd);
719 		TNF_PROBE_1(t1394_write_error, S1394_TNF_SL_ATREQ_ERROR, "",
720 		    tnf_string, msg, "Failed in s1394_setup_asynch_command()");
721 		TNF_PROBE_0_DEBUG(t1394_write_exit, S1394_TNF_SL_ATREQ_STACK,
722 		    "");
723 		return (DDI_FAILURE);
724 	}
725 
726 	/*
727 	 * If this command was sent during a bus reset,
728 	 * then put it onto the pending Q.
729 	 */
730 	if (state == S1394_HAL_RESET) {
731 		/* Remove cmd from outstanding request Q */
732 		s1394_remove_q_asynch_cmd(to_hal, cmd);
733 		/* Are we on the bus reset event stack? */
734 		if (s1394_on_br_thread(to_hal) == B_TRUE) {
735 			/* Blocking commands are not allowed */
736 			if (cmd->cmd_options & CMD1394_BLOCKING) {
737 				mutex_exit(&to_hal->topology_tree_mutex);
738 				s_priv->cmd_in_use = B_FALSE;
739 				cmd->cmd_result    = CMD1394_EINVALID_CONTEXT;
740 				s1394_fa_check_restore_cmd(to_hal, cmd);
741 				TNF_PROBE_1(t1394_write_error,
742 				    S1394_TNF_SL_ATREQ_ERROR, "", tnf_string,
743 				    msg, "CMD1394_BLOCKING in bus reset cntxt");
744 				TNF_PROBE_0_DEBUG(t1394_write_exit,
745 				    S1394_TNF_SL_ATREQ_STACK, "");
746 				return (DDI_FAILURE);
747 			}
748 		}
749 
750 		s1394_pending_q_insert(to_hal, cmd, S1394_PENDING_Q_FRONT);
751 		mutex_exit(&to_hal->topology_tree_mutex);
752 
753 		/* Block (if necessary) */
754 		s1394_block_on_asynch_cmd(cmd);
755 
756 		TNF_PROBE_0_DEBUG(t1394_write_exit, S1394_TNF_SL_ATREQ_STACK,
757 		    "");
758 		return (DDI_SUCCESS);
759 	}
760 	mutex_exit(&to_hal->topology_tree_mutex);
761 
762 	/* Send the command out */
763 	ret = s1394_xfer_asynch_command(to_hal, cmd, &err);
764 
765 	if (ret != DDI_SUCCESS) {
766 		if (err == CMD1394_ESTALE_GENERATION) {
767 			/* Remove cmd from outstanding request Q */
768 			s1394_remove_q_asynch_cmd(to_hal, cmd);
769 			s1394_pending_q_insert(to_hal, cmd,
770 			    S1394_PENDING_Q_FRONT);
771 
772 			/* Block (if necessary) */
773 			s1394_block_on_asynch_cmd(cmd);
774 
775 			TNF_PROBE_0_DEBUG(t1394_write_exit,
776 			    S1394_TNF_SL_ATREQ_STACK, "");
777 			return (DDI_SUCCESS);
778 		} else {
779 			/* Remove cmd from outstanding request Q */
780 			s1394_remove_q_asynch_cmd(to_hal, cmd);
781 
782 			s_priv->cmd_in_use = B_FALSE;
783 
784 			/* Copy error code into result */
785 			cmd->cmd_result = err;
786 
787 			s1394_fa_check_restore_cmd(to_hal, cmd);
788 			TNF_PROBE_1(t1394_write_error,
789 			    S1394_TNF_SL_ATREQ_ERROR, "", tnf_string, msg,
790 			    "Failed in s1394_xfer_asynch_command()");
791 			TNF_PROBE_0_DEBUG(t1394_write_exit,
792 			    S1394_TNF_SL_ATREQ_STACK, "");
793 			return (DDI_FAILURE);
794 		}
795 	} else {
796 		/* Block (if necessary) */
797 		s1394_block_on_asynch_cmd(cmd);
798 
799 		TNF_PROBE_0_DEBUG(t1394_write_exit, S1394_TNF_SL_ATREQ_STACK,
800 		    "");
801 		return (DDI_SUCCESS);
802 	}
803 }
804 
805 /*
806  * Function:    t1394_lock()
807  * Input(s):    t1394_hdl		The target "handle" returned by
808  *					    t1394_attach()
809  *		cmd			Pointer to the command to send
810  *
811  * Output(s):	DDI_SUCCESS		Target successful sent the command
812  *		DDI_FAILURE		Target failed to send command
813  *
814  * Description:	t1394_lock() attempts to send an asynchronous lock request
815  *		onto the 1394 bus.
816  */
817 int
818 t1394_lock(t1394_handle_t t1394_hdl, cmd1394_cmd_t *cmd)
819 {
820 	s1394_hal_t	    *to_hal;
821 	s1394_target_t	    *target;
822 	s1394_cmd_priv_t    *s_priv;
823 	s1394_hal_state_t   state;
824 	cmd1394_lock_type_t lock_type;
825 	uint_t		    num_retries;
826 	int		    ret;
827 
828 	TNF_PROBE_0_DEBUG(t1394_lock_enter, S1394_TNF_SL_ATREQ_STACK, "");
829 
830 	ASSERT(t1394_hdl != NULL);
831 	ASSERT(cmd != NULL);
832 
833 	/* Get the Services Layer private area */
834 	s_priv = S1394_GET_CMD_PRIV(cmd);
835 
836 	/* Is this command currently in use? */
837 	if (s_priv->cmd_in_use == B_TRUE) {
838 		TNF_PROBE_1(t1394_lock_error, S1394_TNF_SL_ATREQ_ERROR, "",
839 		    tnf_string, msg, "Attempted to resend an in-use command");
840 		TNF_PROBE_0_DEBUG(t1394_lock_exit, S1394_TNF_SL_ATREQ_STACK,
841 		    "");
842 		ASSERT(s_priv->cmd_in_use == B_FALSE);
843 		return (DDI_FAILURE);
844 	}
845 
846 	target = (s1394_target_t *)t1394_hdl;
847 
848 	/* Set-up the destination of the command */
849 	to_hal = target->on_hal;
850 
851 	mutex_enter(&to_hal->topology_tree_mutex);
852 	state = to_hal->hal_state;
853 	if (state != S1394_HAL_NORMAL) {
854 		ret = s1394_HAL_asynch_error(to_hal, cmd, state);
855 		if (ret != CMD1394_CMDSUCCESS) {
856 			cmd->cmd_result = ret;
857 			mutex_exit(&to_hal->topology_tree_mutex);
858 			return (DDI_FAILURE);
859 		}
860 	}
861 	mutex_exit(&to_hal->topology_tree_mutex);
862 
863 	/* Check for proper command type */
864 	if ((cmd->cmd_type != CMD1394_ASYNCH_LOCK_32) &&
865 	    (cmd->cmd_type != CMD1394_ASYNCH_LOCK_64)) {
866 		cmd->cmd_result = CMD1394_EINVALID_COMMAND;
867 		TNF_PROBE_1(t1394_lock_error, S1394_TNF_SL_ATREQ_ERROR, "",
868 		    tnf_string, msg, "Invalid command type sent to "
869 		    "t1394_lock()");
870 		TNF_PROBE_0_DEBUG(t1394_lock_exit, S1394_TNF_SL_ATREQ_STACK,
871 		    "");
872 		return (DDI_FAILURE);
873 	}
874 
875 	/* No status (default) */
876 	cmd->cmd_result = CMD1394_NOSTATUS;
877 
878 	/* Is this a blocking command on interrupt stack? */
879 	if ((cmd->cmd_options & CMD1394_BLOCKING) &&
880 	    (servicing_interrupt())) {
881 		cmd->cmd_result = CMD1394_EINVALID_CONTEXT;
882 		TNF_PROBE_1(t1394_lock_error, S1394_TNF_SL_ATREQ_ERROR, "",
883 		    tnf_string, msg, "Tried to use CMD1394_BLOCKING in intr "
884 		    "context");
885 		TNF_PROBE_0_DEBUG(t1394_lock_exit, S1394_TNF_SL_ATREQ_STACK,
886 		    "");
887 		return (DDI_FAILURE);
888 	}
889 
890 	if (cmd->cmd_type == CMD1394_ASYNCH_LOCK_32) {
891 		lock_type	= cmd->cmd_u.l32.lock_type;
892 		num_retries	= cmd->cmd_u.l32.num_retries;
893 	} else {	/* (cmd->cmd_type == CMD1394_ASYNCH_LOCK_64) */
894 		lock_type	= cmd->cmd_u.l64.lock_type;
895 		num_retries	= cmd->cmd_u.l64.num_retries;
896 	}
897 
898 	/* Make sure num_retries is reasonable */
899 	ASSERT(num_retries <= MAX_NUMBER_OF_LOCK_RETRIES);
900 
901 	switch (lock_type) {
902 	case CMD1394_LOCK_MASK_SWAP:
903 	case CMD1394_LOCK_FETCH_ADD:
904 	case CMD1394_LOCK_LITTLE_ADD:
905 	case CMD1394_LOCK_BOUNDED_ADD:
906 	case CMD1394_LOCK_WRAP_ADD:
907 	case CMD1394_LOCK_COMPARE_SWAP:
908 		ret = s1394_compare_swap(to_hal, target, cmd);
909 		break;
910 
911 	case CMD1394_LOCK_BIT_AND:
912 	case CMD1394_LOCK_BIT_OR:
913 	case CMD1394_LOCK_BIT_XOR:
914 	case CMD1394_LOCK_INCREMENT:
915 	case CMD1394_LOCK_DECREMENT:
916 	case CMD1394_LOCK_ADD:
917 	case CMD1394_LOCK_SUBTRACT:
918 	case CMD1394_LOCK_THRESH_ADD:
919 	case CMD1394_LOCK_THRESH_SUBTRACT:
920 	case CMD1394_LOCK_CLIP_ADD:
921 	case CMD1394_LOCK_CLIP_SUBTRACT:
922 		ret = s1394_split_lock_req(to_hal, target, cmd);
923 		break;
924 
925 	default:
926 		TNF_PROBE_1(t1394_lock_error, S1394_TNF_SL_ATREQ_ERROR, "",
927 		    tnf_string, msg, "Invalid lock_type in command");
928 		cmd->cmd_result = CMD1394_EINVALID_COMMAND;
929 		ret = DDI_FAILURE;
930 		break;
931 	}
932 
933 	TNF_PROBE_0_DEBUG(t1394_lock_exit, S1394_TNF_SL_ATREQ_STACK, "");
934 	return (ret);
935 }
936 
937 /*
938  * Function:    t1394_alloc_addr()
939  * Input(s):    t1394_hdl		The target "handle" returned by
940  *					    t1394_attach()
941  *		addr_allocp		The structure used to specify the type,
942  *					    size, permissions, and callbacks
943  *					    (if any) for the requested block
944  *					    of 1394 address space
945  *		flags			The flags parameter is unused (for now)
946  *
947  * Output(s):	result			Used to pass more specific info back
948  *					    to target
949  *
950  * Description:	t1394_alloc_addr() requests that part of the 1394 Address Space
951  *		on the local node be set aside for this target driver, and
952  *		associated with this address space should be some permissions
953  *		and callbacks.  If the request is unable to be fulfilled,
954  *		t1394_alloc_addr() will return DDI_FAILURE and result will
955  *		indicate the reason.  T1394_EINVALID_PARAM indicates that the
956  *		combination of flags given is invalid, and T1394_EALLOC_ADDR
957  *		indicates that the requested type of address space is
958  *		unavailable.
959  */
960 /* ARGSUSED */
961 int
962 t1394_alloc_addr(t1394_handle_t t1394_hdl, t1394_alloc_addr_t *addr_allocp,
963     uint_t flags, int *result)
964 {
965 	s1394_hal_t	*hal;
966 	s1394_target_t	*target;
967 	uint64_t	addr_lo;
968 	uint64_t	addr_hi;
969 	int		err;
970 
971 	TNF_PROBE_0_DEBUG(t1394_alloc_addr_enter, S1394_TNF_SL_ARREQ_STACK,
972 	    "");
973 
974 	ASSERT(t1394_hdl != NULL);
975 	ASSERT(addr_allocp != NULL);
976 
977 	target = (s1394_target_t *)t1394_hdl;
978 
979 	/* Find the HAL this target resides on */
980 	hal = target->on_hal;
981 
982 	/* Get the bounds of the request */
983 	addr_lo = addr_allocp->aa_address;
984 	addr_hi = addr_lo + addr_allocp->aa_length;
985 
986 	/* Check combination of flags */
987 	if ((addr_allocp->aa_enable & T1394_ADDR_RDENBL) &&
988 	    (addr_allocp->aa_evts.recv_read_request == NULL) &&
989 	    (addr_allocp->aa_kmem_bufp == NULL)) {
990 		if ((addr_allocp->aa_type != T1394_ADDR_FIXED)	||
991 		    (addr_lo < hal->physical_addr_lo)		||
992 		    (addr_hi > hal->physical_addr_hi)) {
993 
994 			/*
995 			 * Reads are enabled, but target doesn't want to
996 			 * be notified and hasn't given backing store
997 			 */
998 			*result = T1394_EINVALID_PARAM;
999 
1000 			TNF_PROBE_1(t1394_alloc_addr_error,
1001 			    S1394_TNF_SL_ARREQ_ERROR, "", tnf_string, msg,
1002 			    "Invalid flags "
1003 			    "(RDs on, notify off, no backing store)");
1004 			TNF_PROBE_0_DEBUG(t1394_alloc_addr_exit,
1005 			    S1394_TNF_SL_ARREQ_STACK, "");
1006 
1007 			/* kstats - addr alloc failures */
1008 			hal->hal_kstats->addr_alloc_fail++;
1009 			return (DDI_FAILURE);
1010 		} else {
1011 			addr_allocp->aa_enable &= ~T1394_ADDR_RDENBL;
1012 		}
1013 	}
1014 
1015 	if ((addr_allocp->aa_enable & T1394_ADDR_WRENBL) &&
1016 	    (addr_allocp->aa_evts.recv_write_request == NULL) &&
1017 	    (addr_allocp->aa_kmem_bufp == NULL)) {
1018 		if ((addr_allocp->aa_type != T1394_ADDR_FIXED)	||
1019 		    (addr_lo < hal->physical_addr_lo)		||
1020 		    (addr_hi > hal->physical_addr_hi)) {
1021 
1022 			/*
1023 			 * Writes are enabled, but target doesn't want to
1024 			 * be notified and hasn't given backing store
1025 			 */
1026 			*result = T1394_EINVALID_PARAM;
1027 
1028 			TNF_PROBE_1(t1394_alloc_addr_error,
1029 			    S1394_TNF_SL_ARREQ_ERROR, "", tnf_string, msg,
1030 			    "Invalid flags "
1031 			    "(WRs on, notify off, no backing store)");
1032 			TNF_PROBE_0_DEBUG(t1394_alloc_addr_exit,
1033 			    S1394_TNF_SL_ARREQ_STACK, "");
1034 
1035 			/* kstats - addr alloc failures */
1036 			hal->hal_kstats->addr_alloc_fail++;
1037 			return (DDI_FAILURE);
1038 		} else {
1039 			addr_allocp->aa_enable &= ~T1394_ADDR_WRENBL;
1040 		}
1041 	}
1042 
1043 	if ((addr_allocp->aa_enable & T1394_ADDR_LKENBL) &&
1044 	    (addr_allocp->aa_evts.recv_lock_request == NULL) &&
1045 	    (addr_allocp->aa_kmem_bufp == NULL)) {
1046 		if ((addr_allocp->aa_type != T1394_ADDR_FIXED)	||
1047 		    (addr_lo < hal->physical_addr_lo)		||
1048 		    (addr_hi > hal->physical_addr_hi)) {
1049 
1050 			/*
1051 			 * Locks are enabled, but target doesn't want to
1052 			 * be notified and hasn't given backing store
1053 			 */
1054 			*result = T1394_EINVALID_PARAM;
1055 
1056 			TNF_PROBE_1(t1394_alloc_addr_error,
1057 			    S1394_TNF_SL_ARREQ_ERROR, "", tnf_string, msg,
1058 			    "Invalid flags "
1059 			    "(LKs on, notify off, no backing store)");
1060 			TNF_PROBE_0_DEBUG(t1394_alloc_addr_exit,
1061 			    S1394_TNF_SL_ARREQ_STACK, "");
1062 
1063 			/* kstats - addr alloc failures */
1064 			hal->hal_kstats->addr_alloc_fail++;
1065 			return (DDI_FAILURE);
1066 		} else {
1067 			addr_allocp->aa_enable &= ~T1394_ADDR_LKENBL;
1068 		}
1069 	}
1070 
1071 	/* If not T1394_ADDR_FIXED, then allocate a block */
1072 	if (addr_allocp->aa_type != T1394_ADDR_FIXED) {
1073 		err = s1394_request_addr_blk((s1394_hal_t *)target->on_hal,
1074 					addr_allocp);
1075 		if (err != DDI_SUCCESS) {
1076 			*result = T1394_EALLOC_ADDR;
1077 			/* kstats - addr alloc failures */
1078 			hal->hal_kstats->addr_alloc_fail++;
1079 		} else {
1080 			*result = T1394_NOERROR;
1081 		}
1082 		TNF_PROBE_0_DEBUG(t1394_alloc_addr_exit,
1083 		    S1394_TNF_SL_ARREQ_STACK, "");
1084 		return (err);
1085 	} else {
1086 		err = s1394_claim_addr_blk((s1394_hal_t *)target->on_hal,
1087 					addr_allocp);
1088 		if (err != DDI_SUCCESS) {
1089 			*result = T1394_EALLOC_ADDR;
1090 			/* kstats - addr alloc failures */
1091 			hal->hal_kstats->addr_alloc_fail++;
1092 		} else {
1093 			*result = T1394_NOERROR;
1094 			/* If physical, update the AR request counter */
1095 			if ((addr_lo >= hal->physical_addr_lo) &&
1096 			    (addr_hi <= hal->physical_addr_hi)) {
1097 				rw_enter(&hal->target_list_rwlock, RW_WRITER);
1098 				target->physical_arreq_enabled++;
1099 				rw_exit(&hal->target_list_rwlock);
1100 
1101 				s1394_physical_arreq_set_one(target);
1102 			}
1103 		}
1104 		TNF_PROBE_0_DEBUG(t1394_alloc_addr_exit,
1105 		    S1394_TNF_SL_ARREQ_STACK, "");
1106 		return (err);
1107 	}
1108 }
1109 
1110 /*
1111  * Function:    t1394_free_addr()
1112  * Input(s):    t1394_hdl		The target "handle" returned by
1113  *					    t1394_attach()
1114  *		addr_hdl		The address "handle" returned by the
1115  *					   the t1394_alloc_addr() routine
1116  *		flags			The flags parameter is unused (for now)
1117  *
1118  * Output(s):	DDI_SUCCESS		Target successfully freed memory
1119  *		DDI_FAILURE		Target failed to free the memory block
1120  *
1121  * Description:	t1394_free_addr() attempts to free up memory that has been
1122  *		allocated by the target using t1394_alloc_addr().
1123  */
1124 /* ARGSUSED */
1125 int
1126 t1394_free_addr(t1394_handle_t t1394_hdl, t1394_addr_handle_t *addr_hdl,
1127     uint_t flags)
1128 {
1129 	s1394_addr_space_blk_t	*curr_blk;
1130 	s1394_hal_t		*hal;
1131 	s1394_target_t		*target;
1132 
1133 	TNF_PROBE_0_DEBUG(t1394_free_addr_enter, S1394_TNF_SL_ARREQ_STACK, "");
1134 
1135 	ASSERT(t1394_hdl != NULL);
1136 	ASSERT(addr_hdl != NULL);
1137 
1138 	target = (s1394_target_t *)t1394_hdl;
1139 
1140 	/* Find the HAL this target resides on */
1141 	hal = target->on_hal;
1142 
1143 	curr_blk = (s1394_addr_space_blk_t *)(*addr_hdl);
1144 
1145 	if (s1394_free_addr_blk(hal, curr_blk) != DDI_SUCCESS) {
1146 		TNF_PROBE_0_DEBUG(t1394_free_addr_exit,
1147 		    S1394_TNF_SL_ARREQ_STACK, "");
1148 		return (DDI_FAILURE);
1149 	}
1150 
1151 	/* If physical, update the AR request counter */
1152 	if (curr_blk->addr_type == T1394_ADDR_FIXED) {
1153 		target->physical_arreq_enabled--;
1154 		s1394_physical_arreq_clear_one(target);
1155 	}
1156 
1157 	*addr_hdl = NULL;
1158 
1159 	/* kstats - number of addr frees */
1160 	hal->hal_kstats->addr_space_free++;
1161 
1162 	TNF_PROBE_0_DEBUG(t1394_free_addr_exit, S1394_TNF_SL_ARREQ_STACK, "");
1163 	return (DDI_SUCCESS);
1164 }
1165 
1166 /*
1167  * Function:    t1394_recv_request_done()
1168  * Input(s):    t1394_hdl		The target "handle" returned by
1169  *					    t1394_attach()
1170  *		resp			Pointer to the command which the
1171  *					    target received in it's callback
1172  *		flags			The flags parameter is unused (for now)
1173  *
1174  * Output(s):	DDI_SUCCESS		Target successfully returned command
1175  *					    to the 1394 Software Framework,
1176  *					    and, if necessary, sent response
1177  *		DDI_FAILURE		Target failed to return the command to
1178  *					    the 1394 Software Framework
1179  *
1180  * Description:	t1394_recv_request_done() takes the command that is given and
1181  *		determines whether that command requires a response to be
1182  *		sent on the 1394 bus.  If it is necessary and it's response
1183  *		code (cmd_result) has been set appropriately, then a response
1184  *		will be sent.  If no response is necessary (broadcast or
1185  *		posted write), then the command resources are reclaimed.
1186  */
1187 /* ARGSUSED */
1188 int
1189 t1394_recv_request_done(t1394_handle_t t1394_hdl, cmd1394_cmd_t *resp,
1190     uint_t flags)
1191 {
1192 	s1394_hal_t	 *hal;
1193 	s1394_cmd_priv_t *s_priv;
1194 	h1394_cmd_priv_t *h_priv;
1195 	mblk_t		 *curr_blk;
1196 	size_t		 msgb_len;
1197 	size_t		 size;
1198 	int		 ret;
1199 	boolean_t	 response = B_TRUE;
1200 	boolean_t	 posted_write = B_FALSE;
1201 	boolean_t	 write_cmd = B_FALSE;
1202 	boolean_t	 mblk_too_small;
1203 
1204 	TNF_PROBE_0_DEBUG(t1394_recv_request_done_enter,
1205 	    S1394_TNF_SL_ARREQ_STACK, "");
1206 
1207 	ASSERT(t1394_hdl != NULL);
1208 	ASSERT(resp != NULL);
1209 
1210 	/* Find the HAL this target resides on */
1211 	hal = ((s1394_target_t *)t1394_hdl)->on_hal;
1212 
1213 	/* Get the Services Layer private area */
1214 	s_priv = S1394_GET_CMD_PRIV(resp);
1215 
1216 	/* Get a pointer to the HAL private struct */
1217 	h_priv = (h1394_cmd_priv_t *)&s_priv->hal_cmd_private;
1218 
1219 	/* Is this an FA request? */
1220 	if (s_priv->cmd_ext_type == S1394_CMD_EXT_FA) {
1221 		s1394_fa_convert_cmd(hal, resp);
1222 	}
1223 
1224 	/* Is this a write request? */
1225 	if ((resp->cmd_type == CMD1394_ASYNCH_WR_QUAD) ||
1226 	    (resp->cmd_type == CMD1394_ASYNCH_WR_BLOCK)) {
1227 		write_cmd = B_TRUE;
1228 		/* Is this a posted write request? */
1229 		posted_write = s_priv->posted_write;
1230 	}
1231 
1232 	/* If broadcast or posted write cmd, don't send response */
1233 	if ((resp->broadcast == 1) ||
1234 	    ((write_cmd == B_TRUE) && (posted_write == B_TRUE)))
1235 		response = B_FALSE;
1236 
1237 	if (response == B_FALSE) {
1238 		if ((write_cmd == B_TRUE) && (posted_write == B_TRUE)) {
1239 			/* kstats - Posted Write error */
1240 			hal->hal_kstats->arreq_posted_write_error++;
1241 		}
1242 
1243 		/* Free the command - Pass it back to the HAL */
1244 		HAL_CALL(hal).response_complete(hal->halinfo.hal_private, resp,
1245 		    h_priv);
1246 		TNF_PROBE_0_DEBUG(t1394_recv_request_done_exit,
1247 		    S1394_TNF_SL_ARREQ_STACK, "");
1248 		return (DDI_SUCCESS);
1249 	}
1250 
1251 	ASSERT(response == B_TRUE);
1252 
1253 	/* Verify valid response code */
1254 	switch (resp->cmd_result) {
1255 	case IEEE1394_RESP_COMPLETE:
1256 		/* Is the mblk_t too small? */
1257 		if (resp->cmd_type == CMD1394_ASYNCH_RD_BLOCK) {
1258 			curr_blk = resp->cmd_u.b.data_block;
1259 			size	 = resp->cmd_u.b.blk_length;
1260 			msgb_len = 0;
1261 			mblk_too_small = B_TRUE;
1262 
1263 			if (curr_blk == NULL) {
1264 				TNF_PROBE_1(t1394_recv_request_done_error,
1265 				    S1394_TNF_SL_ARREQ_ERROR, "", tnf_string,
1266 				    msg, "mblk_t is NULL in response");
1267 				TNF_PROBE_0_DEBUG(t1394_recv_request_done_exit,
1268 				    S1394_TNF_SL_ARREQ_STACK, "");
1269 				/*
1270 				 * Free the command - Pass it back
1271 				 * to the HAL
1272 				 */
1273 				HAL_CALL(hal).response_complete(
1274 				    hal->halinfo.hal_private, resp, h_priv);
1275 				ASSERT(curr_blk != NULL);
1276 				return (DDI_FAILURE);
1277 			}
1278 
1279 			while (curr_blk != NULL) {
1280 				msgb_len +=
1281 				    (curr_blk->b_wptr - curr_blk->b_rptr);
1282 
1283 				if (msgb_len >= size) {
1284 					mblk_too_small = B_FALSE;
1285 					break;
1286 				}
1287 				curr_blk = curr_blk->b_cont;
1288 			}
1289 
1290 			if (mblk_too_small == B_TRUE) {
1291 				TNF_PROBE_1(t1394_recv_request_done_error,
1292 				    S1394_TNF_SL_ARREQ_ERROR, "", tnf_string,
1293 				    msg, "mblk_t too small in response");
1294 				TNF_PROBE_0_DEBUG(t1394_recv_request_done_exit,
1295 				    S1394_TNF_SL_ARREQ_STACK, "");
1296 				/*
1297 				 * Free the command - Pass it back
1298 				 * to the HAL
1299 				 */
1300 				HAL_CALL(hal).response_complete(
1301 				    hal->halinfo.hal_private, resp, h_priv);
1302 				ASSERT(mblk_too_small != B_TRUE);
1303 				return (DDI_FAILURE);
1304 			}
1305 		}
1306 		/* FALLTHROUGH */
1307 	case IEEE1394_RESP_CONFLICT_ERROR:
1308 	case IEEE1394_RESP_DATA_ERROR:
1309 	case IEEE1394_RESP_TYPE_ERROR:
1310 	case IEEE1394_RESP_ADDRESS_ERROR:
1311 		ret = s1394_send_response(hal, resp);
1312 		TNF_PROBE_0_DEBUG(t1394_recv_request_done_exit,
1313 		    S1394_TNF_SL_ARREQ_STACK, "");
1314 		return (ret);
1315 
1316 	default:
1317 		TNF_PROBE_1(t1394_recv_request_done_error,
1318 		    S1394_TNF_SL_ARREQ_ERROR, "", tnf_string, msg,
1319 		    "Invalid response code");
1320 		TNF_PROBE_0_DEBUG(t1394_recv_request_done_exit,
1321 		    S1394_TNF_SL_ARREQ_STACK, "");
1322 		return (DDI_FAILURE);
1323 	}
1324 }
1325 
1326 
1327 /*
1328  * Function:    t1394_fcp_register_controller()
1329  * Input(s):    t1394_hdl		The target "handle" returned by
1330  *					    t1394_attach()
1331  *		evts			The structure in which the target
1332  *					    specifies its callback routines
1333  *
1334  *		flags			The flags parameter is unused (for now)
1335  *
1336  * Output(s):	DDI_SUCCESS		Successfully registered.
1337  *
1338  *		DDI_FAILURE		Not registered due to failure.
1339  *
1340  * Description:	Used to register the target within the Framework as an FCP
1341  *		controller.
1342  */
1343 /* ARGSUSED */
1344 int
1345 t1394_fcp_register_controller(t1394_handle_t t1394_hdl, t1394_fcp_evts_t *evts,
1346     uint_t flags)
1347 {
1348 	int		result;
1349 
1350 	TNF_PROBE_0_DEBUG(t1394_fcp_register_controller_enter,
1351 	    S1394_TNF_SL_FCP_STACK, "");
1352 
1353 	ASSERT(t1394_hdl != NULL);
1354 
1355 	result = s1394_fcp_register_ctl((s1394_target_t *)t1394_hdl, evts);
1356 
1357 	TNF_PROBE_0_DEBUG(t1394_fcp_register_controller_exit,
1358 	    S1394_TNF_SL_FCP_STACK, "");
1359 	return (result);
1360 }
1361 
1362 /*
1363  * Function:    t1394_fcp_unregister_controller()
1364  * Input(s):    t1394_hdl		The target "handle" returned by
1365  *					    t1394_attach()
1366  *
1367  * Output(s):	DDI_SUCCESS		Successfully unregistered.
1368  *
1369  *		DDI_FAILURE		Not unregistered due to failure.
1370  *
1371  * Description:	Used to unregister the target within the Framework as an FCP
1372  *		controller.
1373  */
1374 int
1375 t1394_fcp_unregister_controller(t1394_handle_t t1394_hdl)
1376 {
1377 	int		result;
1378 
1379 	TNF_PROBE_0_DEBUG(t1394_fcp_unregister_controller_enter,
1380 	    S1394_TNF_SL_FCP_STACK, "");
1381 
1382 	ASSERT(t1394_hdl != NULL);
1383 
1384 	result = s1394_fcp_unregister_ctl((s1394_target_t *)t1394_hdl);
1385 
1386 	TNF_PROBE_0_DEBUG(t1394_fcp_unregister_controller_exit,
1387 	    S1394_TNF_SL_FCP_STACK, "");
1388 	return (result);
1389 }
1390 
1391 /*
1392  * Function:    t1394_fcp_register_target()
1393  * Input(s):    t1394_hdl		The target "handle" returned by
1394  *					    t1394_attach()
1395  *		evts			The structure in which the target
1396  *					    specifies its callback routines
1397  *
1398  *		flags			The flags parameter is unused (for now)
1399  *
1400  * Output(s):	DDI_SUCCESS		Successfully registered.
1401  *
1402  *		DDI_FAILURE		Not registered due to failure.
1403  *
1404  * Description:	Used to register the target within the Framework as an FCP
1405  *		target.
1406  */
1407 /* ARGSUSED */
1408 int
1409 t1394_fcp_register_target(t1394_handle_t t1394_hdl, t1394_fcp_evts_t *evts,
1410     uint_t flags)
1411 {
1412 	int		result;
1413 
1414 	TNF_PROBE_0_DEBUG(t1394_fcp_register_target_enter,
1415 	    S1394_TNF_SL_FCP_STACK, "");
1416 
1417 	ASSERT(t1394_hdl != NULL);
1418 
1419 	result = s1394_fcp_register_tgt((s1394_target_t *)t1394_hdl, evts);
1420 
1421 	TNF_PROBE_0_DEBUG(t1394_fcp_register_target_exit,
1422 	    S1394_TNF_SL_FCP_STACK, "");
1423 	return (result);
1424 }
1425 
1426 /*
1427  * Function:    t1394_fcp_unregister_target()
1428  * Input(s):    t1394_hdl		The target "handle" returned by
1429  *					    t1394_attach()
1430  *
1431  * Output(s):	DDI_SUCCESS		Successfully unregistered.
1432  *
1433  *		DDI_FAILURE		Not unregistered due to failure.
1434  *
1435  * Description:	Used to unregister the target within the Framework as an FCP
1436  *		target.
1437  */
1438 int
1439 t1394_fcp_unregister_target(t1394_handle_t t1394_hdl)
1440 {
1441 	int		result;
1442 
1443 	TNF_PROBE_0_DEBUG(t1394_fcp_unregister_target_enter,
1444 	    S1394_TNF_SL_FCP_STACK, "");
1445 
1446 	ASSERT(t1394_hdl != NULL);
1447 
1448 	result = s1394_fcp_unregister_tgt((s1394_target_t *)t1394_hdl);
1449 
1450 	TNF_PROBE_0_DEBUG(t1394_fcp_unregister_target_exit,
1451 	    S1394_TNF_SL_FCP_STACK, "");
1452 	return (result);
1453 }
1454 
1455 /*
1456  * Function:    t1394_cmp_register()
1457  * Input(s):    t1394_hdl		The target "handle" returned by
1458  *					    t1394_attach()
1459  *		evts			The structure in which the target
1460  *					    specifies its callback routines
1461  *
1462  * Output(s):	DDI_SUCCESS		Successfully registered.
1463  *
1464  *		DDI_FAILURE		Not registered due to failure.
1465  *
1466  * Description:	Used to register the target within the Framework as a CMP
1467  *		device.
1468  */
1469 /* ARGSUSED */
1470 int
1471 t1394_cmp_register(t1394_handle_t t1394_hdl, t1394_cmp_evts_t *evts,
1472     uint_t flags)
1473 {
1474 	int		result;
1475 
1476 	TNF_PROBE_0_DEBUG(t1394_cmp_register_enter, S1394_TNF_SL_CMP_STACK, "");
1477 
1478 	ASSERT(t1394_hdl != NULL);
1479 
1480 	result = s1394_cmp_register((s1394_target_t *)t1394_hdl, evts);
1481 
1482 	TNF_PROBE_0_DEBUG(t1394_cmp_register_exit, S1394_TNF_SL_CMP_STACK, "");
1483 	return (result);
1484 }
1485 
1486 /*
1487  * Function:    t1394_cmp_unregister()
1488  * Input(s):    t1394_hdl		The target "handle" returned by
1489  *					    t1394_attach()
1490  *		evts			The structure in which the target
1491  *					    specifies its callback routines
1492  *
1493  * Output(s):	DDI_SUCCESS		Successfully registered.
1494  *
1495  *		DDI_FAILURE		Not registered due to failure.
1496  *
1497  * Description:	Used to unregister the target within the Framework as a CMP
1498  *		device.
1499  */
1500 int
1501 t1394_cmp_unregister(t1394_handle_t t1394_hdl)
1502 {
1503 	int		result;
1504 
1505 	TNF_PROBE_0_DEBUG(t1394_cmp_unregister_enter, S1394_TNF_SL_CMP_STACK,
1506 	    "");
1507 
1508 	ASSERT(t1394_hdl != NULL);
1509 
1510 	result = s1394_cmp_unregister((s1394_target_t *)t1394_hdl);
1511 
1512 	TNF_PROBE_0_DEBUG(t1394_cmp_unregister_exit, S1394_TNF_SL_CMP_STACK,
1513 	    "");
1514 	return (result);
1515 }
1516 
1517 /*
1518  * Function:    t1394_cmp_read()
1519  * Input(s):    t1394_hdl		The target "handle" returned by
1520  *					    t1394_attach()
1521  *		reg			Register type.
1522  *		valp			Returned register value.
1523  *
1524  * Output(s):	DDI_SUCCESS		Successfully registered.
1525  *
1526  *		DDI_FAILURE		Not registered due to failure.
1527  *
1528  * Description:	Used to read a CMP register value.
1529  */
1530 int
1531 t1394_cmp_read(t1394_handle_t t1394_hdl, t1394_cmp_reg_t reg, uint32_t *valp)
1532 {
1533 	int		result;
1534 
1535 	TNF_PROBE_0_DEBUG(t1394_cmp_read_enter, S1394_TNF_SL_CMP_STACK, "");
1536 
1537 	ASSERT(t1394_hdl != NULL);
1538 
1539 	result = s1394_cmp_read((s1394_target_t *)t1394_hdl, reg, valp);
1540 
1541 	TNF_PROBE_0_DEBUG(t1394_cmp_read_exit, S1394_TNF_SL_CMP_STACK, "");
1542 	return (result);
1543 }
1544 
1545 /*
1546  * Function:    t1394_cmp_cas()
1547  * Input(s):    t1394_hdl		The target "handle" returned by
1548  *					    t1394_attach()
1549  *		reg			Register type.
1550  *		arg_val			Compare argument.
1551  *		new_val			New register value.
1552  *		old_valp		Returned original register value.
1553  *
1554  * Output(s):	DDI_SUCCESS		Successfully registered.
1555  *
1556  *		DDI_FAILURE		Not registered due to failure.
1557  *
1558  * Description:	Used to compare-swap a CMP register value.
1559  */
1560 int
1561 t1394_cmp_cas(t1394_handle_t t1394_hdl, t1394_cmp_reg_t reg, uint32_t arg_val,
1562     uint32_t new_val, uint32_t *old_valp)
1563 {
1564 	int		result;
1565 
1566 	TNF_PROBE_0_DEBUG(t1394_cmp_read_enter, S1394_TNF_SL_CMP_STACK, "");
1567 
1568 	ASSERT(t1394_hdl != NULL);
1569 
1570 	result = s1394_cmp_cas((s1394_target_t *)t1394_hdl, reg, arg_val,
1571 				new_val, old_valp);
1572 
1573 	TNF_PROBE_0_DEBUG(t1394_cmp_read_exit, S1394_TNF_SL_CMP_STACK, "");
1574 	return (result);
1575 }
1576 
1577 /*
1578  * Function:    t1394_alloc_isoch_single()
1579  * Input(s):    t1394_hdl		The target "handle" returned by
1580  *					    t1394_attach()
1581  *		sii			The structure used to set up the
1582  *					    overall characteristics of the
1583  *					    isochronous stream
1584  *		flags			The flags parameter is unused (for now)
1585  *
1586  * Output(s):	setup_args		Contains the channel number that was
1587  *					    allocated
1588  *		t1394_single_hdl	This in the isoch "handle" used in
1589  *					    t1394_free_isoch_single()
1590  *		result			Used to pass more specific info back
1591  *					    to target
1592  *
1593  * Description:	t1394_alloc_isoch_single() is used to direct the 1394 Software
1594  *		Framework to allocate an isochronous channel and bandwidth
1595  *		from the Isochronous Resource Manager (IRM).  If a bus reset
1596  *		occurs, the 1394 Software Framework attempts to reallocate the
1597  *		same resources, calling the rsrc_fail_target() callback if
1598  *		it is unsuccessful.
1599  */
1600 /* ARGSUSED */
1601 int
1602 t1394_alloc_isoch_single(t1394_handle_t t1394_hdl,
1603     t1394_isoch_singleinfo_t *sii, uint_t flags,
1604     t1394_isoch_single_out_t *output_args,
1605     t1394_isoch_single_handle_t	*t1394_single_hdl, int *result)
1606 {
1607 	s1394_hal_t		*hal;
1608 	s1394_isoch_cec_t	*cec_new;
1609 	t1394_join_isochinfo_t	jii;
1610 	int			ret;
1611 	int			err;
1612 
1613 	TNF_PROBE_0_DEBUG(t1394_alloc_isoch_single_enter,
1614 	    S1394_TNF_SL_ISOCH_STACK, "");
1615 
1616 	ASSERT(t1394_hdl != NULL);
1617 	ASSERT(t1394_single_hdl != NULL);
1618 	ASSERT(sii != NULL);
1619 
1620 	hal = ((s1394_target_t *)t1394_hdl)->on_hal;
1621 
1622 	/* Check for invalid channel_mask */
1623 	if (sii->si_channel_mask == 0) {
1624 		TNF_PROBE_1(t1394_alloc_isoch_single_error,
1625 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
1626 		    "Invalid channel mask");
1627 		TNF_PROBE_0_DEBUG(t1394_alloc_isoch_single_exit,
1628 		    S1394_TNF_SL_ISOCH_STACK, "");
1629 		return (DDI_FAILURE);
1630 	}
1631 
1632 	/* Check for invalid bandwidth */
1633 	if ((sii->si_bandwidth <= IEEE1394_BANDWIDTH_MIN) ||
1634 	    (sii->si_bandwidth > IEEE1394_BANDWIDTH_MAX)) {
1635 		TNF_PROBE_1(t1394_alloc_isoch_single_error,
1636 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
1637 		    "Invalid bandwidth requirements");
1638 		TNF_PROBE_0_DEBUG(t1394_alloc_isoch_single_exit,
1639 		    S1394_TNF_SL_ISOCH_STACK, "");
1640 		return (DDI_FAILURE);
1641 	}
1642 
1643 	/* Verify that rsrc_fail_target() callback is non-NULL */
1644 	if (sii->rsrc_fail_target == NULL) {
1645 		TNF_PROBE_1(t1394_alloc_isoch_single_error,
1646 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
1647 		    "Invalid callback specified");
1648 		TNF_PROBE_0_DEBUG(t1394_alloc_isoch_single_exit,
1649 		    S1394_TNF_SL_ISOCH_STACK, "");
1650 		return (DDI_FAILURE);
1651 	}
1652 
1653 	/*
1654 	 * Allocate an Isoch CEC of type S1394_SINGLE
1655 	 */
1656 
1657 	/* Allocate the Isoch CEC structure */
1658 	cec_new = kmem_zalloc(sizeof (s1394_isoch_cec_t), KM_SLEEP);
1659 
1660 	/* Initialize the structure type */
1661 	cec_new->cec_type = S1394_SINGLE;
1662 
1663 	/* Create the mutex and "in_callbacks" cv */
1664 	mutex_init(&cec_new->isoch_cec_mutex, NULL, MUTEX_DRIVER,
1665 	    hal->halinfo.hw_interrupt);
1666 	cv_init(&cec_new->in_callbacks_cv, NULL, CV_DRIVER,
1667 	    hal->halinfo.hw_interrupt);
1668 
1669 	/* Initialize the Isoch CEC's member list */
1670 	cec_new->cec_member_list_head = NULL;
1671 	cec_new->cec_member_list_tail = NULL;
1672 
1673 	/* Initialize the filters */
1674 	cec_new->filter_min_speed	= sii->si_speed;
1675 	cec_new->filter_max_speed	= sii->si_speed;
1676 	cec_new->filter_current_speed	= cec_new->filter_max_speed;
1677 	cec_new->filter_channel_mask	= sii->si_channel_mask;
1678 	cec_new->bandwidth		= sii->si_bandwidth;
1679 	cec_new->state_transitions	= ISOCH_CEC_FREE | ISOCH_CEC_JOIN |
1680 					    ISOCH_CEC_SETUP;
1681 
1682 	mutex_enter(&hal->isoch_cec_list_mutex);
1683 
1684 	/* Insert Isoch CEC into the HAL's list */
1685 	s1394_isoch_cec_list_insert(hal, cec_new);
1686 
1687 	mutex_exit(&hal->isoch_cec_list_mutex);
1688 
1689 	/*
1690 	 * Join the newly created Isoch CEC
1691 	 */
1692 	jii.req_channel_mask	= sii->si_channel_mask;
1693 	jii.req_max_speed	= sii->si_speed;
1694 	jii.jii_options		= T1394_TALKER;
1695 	jii.isoch_cec_evts_arg	= sii->single_evt_arg;
1696 
1697 	/* All events are NULL except rsrc_fail_target() */
1698 	jii.isoch_cec_evts.setup_target	    = NULL;
1699 	jii.isoch_cec_evts.start_target	    = NULL;
1700 	jii.isoch_cec_evts.stop_target	    = NULL;
1701 	jii.isoch_cec_evts.stop_target	    = NULL;
1702 	jii.isoch_cec_evts.teardown_target  = NULL;
1703 	jii.isoch_cec_evts.rsrc_fail_target = sii->rsrc_fail_target;
1704 
1705 	ret = t1394_join_isoch_cec(t1394_hdl,
1706 	    (t1394_isoch_cec_handle_t)cec_new, 0, &jii);
1707 
1708 	if (ret != DDI_SUCCESS) {
1709 		TNF_PROBE_1(t1394_alloc_isoch_single_error,
1710 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
1711 		    "Unexpected error from t1394_join_isoch_cec()");
1712 
1713 		ret = t1394_free_isoch_cec(t1394_hdl, flags,
1714 		    (t1394_isoch_cec_handle_t *)&cec_new);
1715 		if (ret != DDI_SUCCESS) {
1716 			/* Unable to free the Isoch CEC */
1717 			TNF_PROBE_1(t1394_alloc_isoch_single_error,
1718 			    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
1719 			    "Unexpected error from t1394_free_isoch_cec()");
1720 			ASSERT(0);
1721 		}
1722 
1723 		/* Handle is nulled out before returning */
1724 		*t1394_single_hdl = NULL;
1725 
1726 		TNF_PROBE_0_DEBUG(t1394_alloc_isoch_single_exit,
1727 		    S1394_TNF_SL_ISOCH_STACK, "");
1728 		return (DDI_FAILURE);
1729 	}
1730 
1731 	/*
1732 	 * Setup the isoch resources, etc.
1733 	 */
1734 	ret = t1394_setup_isoch_cec(t1394_hdl,
1735 	    (t1394_isoch_cec_handle_t)cec_new, 0, &err);
1736 
1737 	if (ret != DDI_SUCCESS) {
1738 		TNF_PROBE_1(t1394_alloc_isoch_single_error,
1739 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
1740 		    "Unexpected error from t1394_setup_isoch_cec()");
1741 
1742 		*result = err;
1743 
1744 		/* Leave the Isoch CEC */
1745 		ret = t1394_leave_isoch_cec(t1394_hdl,
1746 		    (t1394_isoch_cec_handle_t)cec_new, 0);
1747 		if (ret != DDI_SUCCESS) {
1748 			/* Unable to leave the Isoch CEC */
1749 			TNF_PROBE_1(t1394_alloc_isoch_single_error,
1750 			    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
1751 			    "Unexpected error from t1394_leave_isoch_cec()");
1752 			ASSERT(0);
1753 		}
1754 
1755 		/* Free up the Isoch CEC */
1756 		ret = t1394_free_isoch_cec(t1394_hdl, flags,
1757 		    (t1394_isoch_cec_handle_t *)&cec_new);
1758 		if (ret != DDI_SUCCESS) {
1759 			/* Unable to free the Isoch CEC */
1760 			TNF_PROBE_1(t1394_alloc_isoch_single_error,
1761 			    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
1762 			    "Unexpected error from t1394_free_isoch_cec()");
1763 			ASSERT(0);
1764 		}
1765 
1766 		/* Handle is nulled out before returning */
1767 		*t1394_single_hdl = NULL;
1768 
1769 		TNF_PROBE_0_DEBUG(t1394_alloc_isoch_single_exit,
1770 		    S1394_TNF_SL_ISOCH_STACK, "");
1771 		return (DDI_FAILURE);
1772 	}
1773 
1774 	/* Return the setup_args - channel num and speed */
1775 	mutex_enter(&cec_new->isoch_cec_mutex);
1776 	output_args->channel_num  = cec_new->realloc_chnl_num;
1777 	mutex_exit(&cec_new->isoch_cec_mutex);
1778 
1779 	/* Update the handle */
1780 	*t1394_single_hdl = (t1394_isoch_single_handle_t)cec_new;
1781 
1782 	TNF_PROBE_0_DEBUG(t1394_alloc_isoch_single_exit,
1783 	    S1394_TNF_SL_ISOCH_STACK, "");
1784 	return (DDI_SUCCESS);
1785 }
1786 
1787 /*
1788  * Function:    t1394_free_isoch_single()
1789  * Input(s):    t1394_hdl		The target "handle" returned by
1790  *					    t1394_attach()
1791  *		t1394_single_hdl	The isoch "handle" return by
1792  *					    t1394_alloc_isoch_single()
1793  *		flags			The flags parameter is unused (for now)
1794  *
1795  * Output(s):	None
1796  *
1797  * Description:	t1394_free_isoch_single() frees the isochronous resources
1798  *		and the handle that were allocated during the call to
1799  *		t1394_alloc_isoch_single().
1800  */
1801 /* ARGSUSED */
1802 void
1803 t1394_free_isoch_single(t1394_handle_t t1394_hdl,
1804     t1394_isoch_single_handle_t *t1394_single_hdl, uint_t flags)
1805 {
1806 	s1394_isoch_cec_t *cec_curr;
1807 	int		  ret;
1808 
1809 	TNF_PROBE_0_DEBUG(t1394_free_isoch_single_enter,
1810 	    S1394_TNF_SL_ISOCH_STACK, "");
1811 
1812 	ASSERT(t1394_hdl != NULL);
1813 	ASSERT(t1394_single_hdl != NULL);
1814 
1815 	/* Convert the handle to an Isoch CEC pointer */
1816 	cec_curr = (s1394_isoch_cec_t *)(*t1394_single_hdl);
1817 
1818 	/*
1819 	 * Teardown the isoch resources, etc.
1820 	 */
1821 	ret = t1394_teardown_isoch_cec(t1394_hdl,
1822 	    (t1394_isoch_cec_handle_t)cec_curr, 0);
1823 	if (ret != DDI_SUCCESS) {
1824 		/* Unable to teardown the Isoch CEC */
1825 		TNF_PROBE_1(t1394_free_isoch_single_error,
1826 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
1827 		    "Unexpected error from t1394_teardown_isoch_cec()");
1828 		ASSERT(0);
1829 	}
1830 
1831 	/*
1832 	 * Leave the Isoch CEC
1833 	 */
1834 	ret = t1394_leave_isoch_cec(t1394_hdl,
1835 	    (t1394_isoch_cec_handle_t)cec_curr, 0);
1836 	if (ret != DDI_SUCCESS) {
1837 		/* Unable to leave the Isoch CEC */
1838 		TNF_PROBE_1(t1394_free_isoch_single_error,
1839 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
1840 		    "Unexpected error from t1394_leave_isoch_cec()");
1841 		ASSERT(0);
1842 	}
1843 
1844 	/*
1845 	 * Free the Isoch CEC
1846 	 */
1847 	ret = t1394_free_isoch_cec(t1394_hdl, flags,
1848 	    (t1394_isoch_cec_handle_t *)&cec_curr);
1849 	if (ret != DDI_SUCCESS) {
1850 		/* Unable to free the Isoch CEC */
1851 		TNF_PROBE_1(t1394_free_isoch_single_error,
1852 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
1853 		    "Unexpected error from t1394_free_isoch_cec()");
1854 		ASSERT(0);
1855 	}
1856 
1857 	/* Handle is nulled out before returning */
1858 	*t1394_single_hdl = NULL;
1859 
1860 	TNF_PROBE_0_DEBUG(t1394_free_isoch_single_exit,
1861 	    S1394_TNF_SL_ISOCH_STACK, "");
1862 }
1863 
1864 /*
1865  * Function:    t1394_alloc_isoch_cec()
1866  * Input(s):    t1394_hdl		The target "handle" returned by
1867  *					    t1394_attach()
1868  *		props			The structure used to set up the
1869  *					    overall characteristics of for
1870  *					    the Isoch CEC.
1871  *		flags			The flags parameter is unused (for now)
1872  *
1873  * Output(s):	t1394_isoch_cec_hdl	The Isoch CEC "handle" used in all
1874  *					    subsequent isoch_cec() calls
1875  *
1876  * Description:	t1394_alloc_isoch_cec() allocates and initializes an
1877  *		isochronous channel event coordinator (Isoch CEC) for use
1878  *		in managing and coordinating activity for an isoch channel
1879  */
1880 /* ARGSUSED */
1881 int
1882 t1394_alloc_isoch_cec(t1394_handle_t t1394_hdl, t1394_isoch_cec_props_t *props,
1883     uint_t flags, t1394_isoch_cec_handle_t *t1394_isoch_cec_hdl)
1884 {
1885 	s1394_hal_t	  *hal;
1886 	s1394_isoch_cec_t *cec_new;
1887 	uint64_t	  temp;
1888 
1889 	TNF_PROBE_0_DEBUG(t1394_alloc_isoch_cec_enter,
1890 	    S1394_TNF_SL_ISOCH_STACK, "");
1891 
1892 	ASSERT(t1394_hdl != NULL);
1893 	ASSERT(t1394_isoch_cec_hdl != NULL);
1894 	ASSERT(props != NULL);
1895 
1896 	hal = ((s1394_target_t *)t1394_hdl)->on_hal;
1897 
1898 	/* Check for invalid channel_mask */
1899 	if (props->cec_channel_mask == 0) {
1900 		TNF_PROBE_1(t1394_alloc_isoch_cec_error,
1901 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
1902 		    "Invalid channel mask");
1903 		TNF_PROBE_0_DEBUG(t1394_alloc_isoch_cec_exit,
1904 		    S1394_TNF_SL_ISOCH_STACK, "");
1905 		return (DDI_FAILURE);
1906 	}
1907 
1908 	/* Test conditions specific to T1394_NO_IRM_ALLOC */
1909 	temp = props->cec_channel_mask;
1910 	if (props->cec_options & T1394_NO_IRM_ALLOC) {
1911 		/* If T1394_NO_IRM_ALLOC, then only one bit should be set */
1912 		if ((temp & (temp - 1)) != 0) {
1913 			TNF_PROBE_1(t1394_alloc_isoch_cec_error,
1914 			    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
1915 			    "Invalid channel mask");
1916 			TNF_PROBE_0_DEBUG(t1394_alloc_isoch_cec_exit,
1917 			    S1394_TNF_SL_ISOCH_STACK, "");
1918 			return (DDI_FAILURE);
1919 		}
1920 
1921 		/* If T1394_NO_IRM_ALLOC, then speeds should be equal */
1922 		if (props->cec_min_speed != props->cec_max_speed) {
1923 			TNF_PROBE_1(t1394_alloc_isoch_cec_error,
1924 			    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
1925 			    "Invalid speeds (min != max)");
1926 			TNF_PROBE_0_DEBUG(t1394_alloc_isoch_cec_exit,
1927 			    S1394_TNF_SL_ISOCH_STACK, "");
1928 			return (DDI_FAILURE);
1929 		}
1930 	}
1931 
1932 	/* Check for invalid bandwidth */
1933 	if ((props->cec_bandwidth <= IEEE1394_BANDWIDTH_MIN) ||
1934 	    (props->cec_bandwidth > IEEE1394_BANDWIDTH_MAX)) {
1935 		TNF_PROBE_1(t1394_alloc_isoch_cec_error,
1936 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
1937 		    "Invalid bandwidth requirements");
1938 		TNF_PROBE_0_DEBUG(t1394_alloc_isoch_cec_exit,
1939 		    S1394_TNF_SL_ISOCH_STACK, "");
1940 		return (DDI_FAILURE);
1941 	}
1942 
1943 	/* Allocate the Isoch CEC structure */
1944 	cec_new = kmem_zalloc(sizeof (s1394_isoch_cec_t), KM_SLEEP);
1945 
1946 	/* Initialize the structure type */
1947 	cec_new->cec_type = S1394_PEER_TO_PEER;
1948 
1949 	/* Create the mutex and "in_callbacks" cv */
1950 	mutex_init(&cec_new->isoch_cec_mutex, NULL, MUTEX_DRIVER,
1951 	    hal->halinfo.hw_interrupt);
1952 	cv_init(&cec_new->in_callbacks_cv, NULL, CV_DRIVER,
1953 	    hal->halinfo.hw_interrupt);
1954 
1955 	/* Initialize the Isoch CEC's member list */
1956 	cec_new->cec_member_list_head	= NULL;
1957 	cec_new->cec_member_list_tail	= NULL;
1958 
1959 	/* Initialize the filters */
1960 	cec_new->filter_min_speed	= props->cec_min_speed;
1961 	cec_new->filter_max_speed	= props->cec_max_speed;
1962 	cec_new->filter_current_speed	= cec_new->filter_max_speed;
1963 	cec_new->filter_channel_mask	= props->cec_channel_mask;
1964 	cec_new->bandwidth		= props->cec_bandwidth;
1965 	cec_new->cec_options		= props->cec_options;
1966 	cec_new->state_transitions	= ISOCH_CEC_FREE | ISOCH_CEC_JOIN |
1967 					    ISOCH_CEC_SETUP;
1968 
1969 	mutex_enter(&hal->isoch_cec_list_mutex);
1970 
1971 	/* Insert Isoch CEC into the HAL's list */
1972 	s1394_isoch_cec_list_insert(hal, cec_new);
1973 
1974 	mutex_exit(&hal->isoch_cec_list_mutex);
1975 
1976 	/* Update the handle and return */
1977 	*t1394_isoch_cec_hdl = (t1394_isoch_cec_handle_t)cec_new;
1978 
1979 	TNF_PROBE_0_DEBUG(t1394_alloc_isoch_cec_exit,
1980 	    S1394_TNF_SL_ISOCH_STACK, "");
1981 	return (DDI_SUCCESS);
1982 }
1983 
1984 /*
1985  * Function:    t1394_free_isoch_cec()
1986  * Input(s):    t1394_hdl		The target "handle" returned by
1987  *					    t1394_attach()
1988  *		flags			The flags parameter is unused (for now)
1989  *		t1394_isoch_cec_hdl	The Isoch CEC "handle" returned by
1990  *					    t1394_alloc_isoch_cec()
1991  *
1992  * Output(s):	DDI_SUCCESS		Target successfully freed the Isoch CEC
1993  *		DDI_FAILURE		Target failed to free the Isoch CEC
1994  *
1995  * Description:	t1394_free_isoch_cec() attempts to free the Isoch CEC
1996  *		structure.  It will fail (DDI_FAILURE) if there are any
1997  *		remaining members who have not yet left.
1998  */
1999 /* ARGSUSED */
2000 int
2001 t1394_free_isoch_cec(t1394_handle_t t1394_hdl, uint_t flags,
2002     t1394_isoch_cec_handle_t *t1394_isoch_cec_hdl)
2003 {
2004 	s1394_hal_t	  *hal;
2005 	s1394_isoch_cec_t *cec_curr;
2006 
2007 	TNF_PROBE_0_DEBUG(t1394_free_isoch_cec_enter,
2008 	    S1394_TNF_SL_ISOCH_STACK, "");
2009 
2010 	ASSERT(t1394_hdl != NULL);
2011 	ASSERT(t1394_isoch_cec_hdl != NULL);
2012 
2013 	hal = ((s1394_target_t *)t1394_hdl)->on_hal;
2014 
2015 	/* Convert the handle to an Isoch CEC pointer */
2016 	cec_curr = (s1394_isoch_cec_t *)(*t1394_isoch_cec_hdl);
2017 
2018 	/* Lock the Isoch CEC member list */
2019 	mutex_enter(&cec_curr->isoch_cec_mutex);
2020 
2021 	/* Are we in any callbacks? */
2022 	if (CEC_IN_ANY_CALLBACKS(cec_curr)) {
2023 		/* Unlock the Isoch CEC member list */
2024 		mutex_exit(&cec_curr->isoch_cec_mutex);
2025 		TNF_PROBE_1(t1394_free_isoch_cec_error,
2026 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
2027 		    "Not allowed to free Isoch CEC (in callbacks)");
2028 		TNF_PROBE_0_DEBUG(t1394_free_isoch_cec_exit,
2029 		    S1394_TNF_SL_ISOCH_STACK, "");
2030 		return (DDI_FAILURE);
2031 	}
2032 
2033 	/* Is "free" a legal state transition? */
2034 	if (CEC_TRANSITION_LEGAL(cec_curr, ISOCH_CEC_FREE) == 0) {
2035 		/* Unlock the Isoch CEC member list */
2036 		mutex_exit(&cec_curr->isoch_cec_mutex);
2037 		TNF_PROBE_1(t1394_free_isoch_cec_error,
2038 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
2039 		    "Not allowed to free Isoch CEC");
2040 		TNF_PROBE_0_DEBUG(t1394_free_isoch_cec_exit,
2041 		    S1394_TNF_SL_ISOCH_STACK, "");
2042 		return (DDI_FAILURE);
2043 	}
2044 	mutex_exit(&cec_curr->isoch_cec_mutex);
2045 
2046 	mutex_enter(&hal->isoch_cec_list_mutex);
2047 
2048 	/* Remove Isoch CEC from HAL's list */
2049 	s1394_isoch_cec_list_remove(hal, cec_curr);
2050 
2051 	mutex_exit(&hal->isoch_cec_list_mutex);
2052 
2053 	/* Destroy the Isoch CEC's mutex and cv */
2054 	cv_destroy(&cec_curr->in_callbacks_cv);
2055 	mutex_destroy(&cec_curr->isoch_cec_mutex);
2056 
2057 	/* Free up the memory for the Isoch CEC struct */
2058 	kmem_free(cec_curr, sizeof (s1394_isoch_cec_t));
2059 
2060 	/* Update the handle and return */
2061 	*t1394_isoch_cec_hdl = NULL;
2062 
2063 	TNF_PROBE_0_DEBUG(t1394_free_isoch_cec_exit,
2064 	    S1394_TNF_SL_ISOCH_STACK, "");
2065 	return (DDI_SUCCESS);
2066 }
2067 
2068 /*
2069  * Function:    t1394_join_isoch_cec()
2070  * Input(s):    t1394_hdl		The target "handle" returned by
2071  *					    t1394_attach()
2072  *		t1394_isoch_cec_hdl	The Isoch CEC "handle" returned by
2073  *					    t1394_alloc_isoch_cec()
2074  *		flags			The flags parameter is unused (for now)
2075  *		join_isoch_info		This structure provides infomation
2076  *					    about a target that wishes to join
2077  *					    the given Isoch CEC.  It gives
2078  *					    max_speed, channel_mask, etc.
2079  *
2080  * Output(s):	DDI_SUCCESS		Target successfully joined the
2081  *					    Isoch CEC
2082  *		DDI_FAILURE		Target failed to join the Isoch CEC
2083  *
2084  * Description:	t1394_join_isoch_cec() determines, based on the information
2085  *		given in the join_isoch_info structure, if the target may
2086  *		join the Isoch CEC.  If it is determined that the target may
2087  *		join, the specified callback routines are stored away for
2088  *		later use in the coordination tasks.
2089  */
2090 /* ARGSUSED */
2091 int
2092 t1394_join_isoch_cec(t1394_handle_t t1394_hdl,
2093     t1394_isoch_cec_handle_t t1394_isoch_cec_hdl, uint_t flags,
2094     t1394_join_isochinfo_t *join_isoch_info)
2095 {
2096 	s1394_hal_t		 *hal;
2097 	s1394_isoch_cec_t	 *cec_curr;
2098 	s1394_isoch_cec_member_t *member_new;
2099 	uint64_t		 check_mask;
2100 	uint_t			 curr_max_speed;
2101 
2102 	TNF_PROBE_0_DEBUG(t1394_join_isoch_cec_enter,
2103 	    S1394_TNF_SL_ISOCH_STACK, "");
2104 
2105 	ASSERT(t1394_hdl != NULL);
2106 	ASSERT(t1394_isoch_cec_hdl != NULL);
2107 
2108 	hal = ((s1394_target_t *)t1394_hdl)->on_hal;
2109 
2110 	/* Convert the handle to an Isoch CEC pointer */
2111 	cec_curr = (s1394_isoch_cec_t *)t1394_isoch_cec_hdl;
2112 
2113 	/* Allocate a new Isoch CEC member structure */
2114 	member_new = kmem_zalloc(sizeof (s1394_isoch_cec_member_t), KM_SLEEP);
2115 
2116 	/* Lock the Isoch CEC member list */
2117 	mutex_enter(&cec_curr->isoch_cec_mutex);
2118 
2119 	/* Are we in any callbacks? (Wait for them to finish) */
2120 	while (CEC_IN_ANY_CALLBACKS(cec_curr)) {
2121 		cec_curr->cec_want_wakeup = B_TRUE;
2122 		cv_wait(&cec_curr->in_callbacks_cv,
2123 		    &cec_curr->isoch_cec_mutex);
2124 	}
2125 
2126 	/* Is "join" a legal state transition? */
2127 	if (CEC_TRANSITION_LEGAL(cec_curr, ISOCH_CEC_JOIN) == 0) {
2128 		kmem_free(member_new, sizeof (s1394_isoch_cec_member_t));
2129 		/* Unlock the Isoch CEC member list */
2130 		mutex_exit(&cec_curr->isoch_cec_mutex);
2131 		TNF_PROBE_1(t1394_join_isoch_cec_error,
2132 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
2133 		    "Not allowed to join Isoch CEC");
2134 		TNF_PROBE_0_DEBUG(t1394_join_isoch_cec_exit,
2135 		    S1394_TNF_SL_ISOCH_STACK, "");
2136 		return (DDI_FAILURE);
2137 	}
2138 
2139 	/* Check the channel mask for consistency */
2140 	check_mask = join_isoch_info->req_channel_mask &
2141 	    cec_curr->filter_channel_mask;
2142 	if (check_mask == 0) {
2143 		kmem_free(member_new, sizeof (s1394_isoch_cec_member_t));
2144 		/* Unlock the Isoch CEC member list */
2145 		mutex_exit(&cec_curr->isoch_cec_mutex);
2146 		TNF_PROBE_1(t1394_join_isoch_cec_error,
2147 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
2148 		    "Inconsistent channel mask specified");
2149 		TNF_PROBE_0_DEBUG(t1394_join_isoch_cec_exit,
2150 		    S1394_TNF_SL_ISOCH_STACK, "");
2151 		return (DDI_FAILURE);
2152 	}
2153 
2154 	/* Check for consistent speeds */
2155 	if (join_isoch_info->req_max_speed < cec_curr->filter_min_speed) {
2156 		kmem_free(member_new, sizeof (s1394_isoch_cec_member_t));
2157 		/* Unlock the Isoch CEC member list */
2158 		mutex_exit(&cec_curr->isoch_cec_mutex);
2159 		TNF_PROBE_1(t1394_join_isoch_cec_error,
2160 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
2161 		    "Inconsistent speed specified");
2162 		TNF_PROBE_0_DEBUG(t1394_join_isoch_cec_exit,
2163 		    S1394_TNF_SL_ISOCH_STACK, "");
2164 		return (DDI_FAILURE);
2165 	} else if (join_isoch_info->req_max_speed <
2166 	    cec_curr->filter_current_speed) {
2167 		curr_max_speed = join_isoch_info->req_max_speed;
2168 	} else {
2169 		curr_max_speed = cec_curr->filter_current_speed;
2170 	}
2171 
2172 	/* Check for no more than one talker */
2173 	if ((join_isoch_info->jii_options & T1394_TALKER) &&
2174 	    (cec_curr->cec_member_talker != NULL)) {
2175 		kmem_free(member_new, sizeof (s1394_isoch_cec_member_t));
2176 		/* Unlock the Isoch CEC member list */
2177 		mutex_exit(&cec_curr->isoch_cec_mutex);
2178 		TNF_PROBE_1(t1394_join_isoch_cec_error,
2179 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
2180 		    "Multiple talkers specified");
2181 		TNF_PROBE_0_DEBUG(t1394_join_isoch_cec_exit,
2182 		    S1394_TNF_SL_ISOCH_STACK, "");
2183 		return (DDI_FAILURE);
2184 	}
2185 
2186 	/* Verify that all callbacks are non-NULL (for PEER_TO_PEER) */
2187 	if ((cec_curr->cec_type == S1394_PEER_TO_PEER) &&
2188 	    ((join_isoch_info->isoch_cec_evts.setup_target	== NULL) ||
2189 	    (join_isoch_info->isoch_cec_evts.start_target	== NULL) ||
2190 	    (join_isoch_info->isoch_cec_evts.stop_target	== NULL) ||
2191 	    (join_isoch_info->isoch_cec_evts.rsrc_fail_target	== NULL) ||
2192 	    (join_isoch_info->isoch_cec_evts.teardown_target	== NULL))) {
2193 		/* Unlock the Isoch CEC member list */
2194 		mutex_exit(&cec_curr->isoch_cec_mutex);
2195 		TNF_PROBE_1(t1394_join_isoch_cec_error,
2196 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
2197 		    "Invalid callbacks specified");
2198 		TNF_PROBE_0_DEBUG(t1394_join_isoch_cec_exit,
2199 		    S1394_TNF_SL_ISOCH_STACK, "");
2200 		return (DDI_FAILURE);
2201 	}
2202 
2203 	/* Copy the events information into the struct */
2204 	member_new->isoch_cec_evts	= join_isoch_info->isoch_cec_evts;
2205 	member_new->isoch_cec_evts_arg	= join_isoch_info->isoch_cec_evts_arg;
2206 	member_new->cec_mem_options	= join_isoch_info->jii_options;
2207 	member_new->cec_mem_target	= (s1394_target_t *)t1394_hdl;
2208 
2209 	/* Insert new member into Isoch CEC's member list */
2210 	s1394_isoch_cec_member_list_insert(hal, cec_curr, member_new);
2211 
2212 	/* Update the channel mask filter */
2213 	cec_curr->filter_channel_mask	= check_mask;
2214 
2215 	/* Update the speed filter */
2216 	cec_curr->filter_current_speed	= curr_max_speed;
2217 
2218 	/* Update the talker pointer (if necessary) */
2219 	if (join_isoch_info->jii_options & T1394_TALKER)
2220 		cec_curr->cec_member_talker = cec_curr->cec_member_list_head;
2221 
2222 	/*
2223 	 * Now "leave" is a legal state transition
2224 	 * and "free" is an illegal state transition
2225 	 */
2226 	CEC_SET_LEGAL(cec_curr, ISOCH_CEC_LEAVE);
2227 	CEC_SET_ILLEGAL(cec_curr, ISOCH_CEC_FREE);
2228 
2229 	/* Unlock the Isoch CEC member list */
2230 	mutex_exit(&cec_curr->isoch_cec_mutex);
2231 
2232 	TNF_PROBE_0_DEBUG(t1394_join_isoch_cec_exit,
2233 	    S1394_TNF_SL_ISOCH_STACK, "");
2234 	return (DDI_SUCCESS);
2235 }
2236 
2237 /*
2238  * Function:    t1394_leave_isoch_cec()
2239  * Input(s):    t1394_hdl		The target "handle" returned by
2240  *					    t1394_attach()
2241  *		t1394_isoch_cec_hdl	The Isoch CEC "handle" returned by
2242  *					    t1394_alloc_isoch_cec()
2243  *		flags			The flags parameter is unused (for now)
2244  *
2245  * Output(s):	DDI_SUCCESS		Target successfully left the
2246  *					    Isoch CEC
2247  *		DDI_FAILURE		Target failed to leave the Isoch CEC
2248  *
2249  * Description:	t1394_leave_isoch_cec() is used by a target driver to remove
2250  *		itself from the Isoch CEC's member list.  It is possible
2251  *		for this call to fail because the target is not found in
2252  *		the current member list, or because it is not an appropriate
2253  *		time for a target to leave.
2254  */
2255 /* ARGSUSED */
2256 int
2257 t1394_leave_isoch_cec(t1394_handle_t t1394_hdl,
2258     t1394_isoch_cec_handle_t t1394_isoch_cec_hdl, uint_t flags)
2259 {
2260 	s1394_hal_t		 *hal;
2261 	s1394_isoch_cec_t	 *cec_curr;
2262 	s1394_isoch_cec_member_t *member_curr;
2263 	s1394_isoch_cec_member_t *member_temp;
2264 	boolean_t		 found;
2265 	uint64_t		 temp_channel_mask;
2266 	uint_t			 temp_max_speed;
2267 
2268 	TNF_PROBE_0_DEBUG(t1394_leave_isoch_cec_enter,
2269 	    S1394_TNF_SL_ISOCH_STACK, "");
2270 
2271 	ASSERT(t1394_hdl != NULL);
2272 	ASSERT(t1394_isoch_cec_hdl != NULL);
2273 
2274 	hal = ((s1394_target_t *)t1394_hdl)->on_hal;
2275 
2276 	/* Convert the handle to an Isoch CEC pointer */
2277 	cec_curr = (s1394_isoch_cec_t *)t1394_isoch_cec_hdl;
2278 
2279 	/* Lock the Isoch CEC member list */
2280 	mutex_enter(&cec_curr->isoch_cec_mutex);
2281 
2282 	/* Are we in any callbacks? (Wait for them to finish) */
2283 	while (CEC_IN_ANY_CALLBACKS(cec_curr)) {
2284 		cec_curr->cec_want_wakeup = B_TRUE;
2285 		cv_wait(&cec_curr->in_callbacks_cv,
2286 		    &cec_curr->isoch_cec_mutex);
2287 	}
2288 
2289 	/* Is "leave" a legal state transition? */
2290 	if (CEC_TRANSITION_LEGAL(cec_curr, ISOCH_CEC_LEAVE) == 0) {
2291 		/* Unlock the Isoch CEC member list */
2292 		mutex_exit(&cec_curr->isoch_cec_mutex);
2293 		TNF_PROBE_1(t1394_leave_isoch_cec_error,
2294 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
2295 		    "Not allowed to leave Isoch CEC");
2296 		TNF_PROBE_0_DEBUG(t1394_leave_isoch_cec_exit,
2297 		    S1394_TNF_SL_ISOCH_STACK, "");
2298 		return (DDI_FAILURE);
2299 	}
2300 
2301 	/* Find the Target on the CEC's member list */
2302 	found = B_FALSE;
2303 	temp_channel_mask = cec_curr->cec_alloc_props.cec_channel_mask;
2304 	temp_max_speed	  = cec_curr->cec_alloc_props.cec_max_speed;
2305 	member_curr	  = cec_curr->cec_member_list_head;
2306 	while (member_curr != NULL) {
2307 		if (member_curr->cec_mem_target ==
2308 		    (s1394_target_t *)t1394_hdl) {
2309 			member_temp = member_curr;
2310 			found	    = B_TRUE;
2311 		} else {
2312 			/* Keep track of channel mask and max speed info */
2313 			temp_channel_mask &= member_curr->req_channel_mask;
2314 			if (member_curr->req_max_speed < temp_max_speed)
2315 				temp_max_speed = member_curr->req_max_speed;
2316 		}
2317 		member_curr = member_curr->cec_mem_next;
2318 	}
2319 
2320 	/* Target not found on this Isoch CEC */
2321 	if (found == B_FALSE) {
2322 		/* Unlock the Isoch CEC member list */
2323 		mutex_exit(&cec_curr->isoch_cec_mutex);
2324 		TNF_PROBE_1(t1394_leave_isoch_cec_error,
2325 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
2326 		    "Target not found in Isoch CEC member list");
2327 		TNF_PROBE_0_DEBUG(t1394_leave_isoch_cec_exit,
2328 		    S1394_TNF_SL_ISOCH_STACK, "");
2329 		return (DDI_FAILURE);
2330 	} else {
2331 		/* This member's departure may change filter constraints */
2332 		cec_curr->filter_current_speed  = temp_max_speed;
2333 		cec_curr->filter_channel_mask   = temp_channel_mask;
2334 	}
2335 
2336 	/* Remove member from Isoch CEC's member list */
2337 	s1394_isoch_cec_member_list_remove(hal, cec_curr, member_temp);
2338 
2339 	/* If we are removing the talker, then update the pointer */
2340 	if (cec_curr->cec_member_talker == member_temp)
2341 		cec_curr->cec_member_talker = NULL;
2342 
2343 	/* Is the Isoch CEC's member list empty? */
2344 	if ((cec_curr->cec_member_list_head == NULL) &&
2345 	    (cec_curr->cec_member_list_tail == NULL)) {
2346 		/*
2347 		 * Now "free" _might_ be a legal state transition
2348 		 * if we aren't in setup or start phases and "leave"
2349 		 * is definitely an illegal state transition
2350 		 */
2351 		if (CEC_TRANSITION_LEGAL(cec_curr, ISOCH_CEC_JOIN) != 0)
2352 			CEC_SET_LEGAL(cec_curr, ISOCH_CEC_FREE);
2353 		CEC_SET_ILLEGAL(cec_curr, ISOCH_CEC_LEAVE);
2354 	}
2355 
2356 	/* Unlock the Isoch CEC member list */
2357 	mutex_exit(&cec_curr->isoch_cec_mutex);
2358 
2359 	/* Free the Isoch CEC member structure */
2360 	kmem_free(member_temp, sizeof (s1394_isoch_cec_member_t));
2361 
2362 	TNF_PROBE_0_DEBUG(t1394_leave_isoch_cec_exit,
2363 	    S1394_TNF_SL_ISOCH_STACK, "");
2364 	return (DDI_SUCCESS);
2365 }
2366 
2367 /*
2368  * Function:    t1394_setup_isoch_cec()
2369  * Input(s):    t1394_hdl		The target "handle" returned by
2370  *					    t1394_attach()
2371  *		t1394_isoch_cec_hdl	The Isoch CEC "handle" returned by
2372  *					    t1394_alloc_isoch_cec()
2373  *		flags			The flags parameter is unused (for now)
2374  *
2375  * Output(s):	result			Used to pass more specific info back
2376  *					    to target
2377  *
2378  * Description:	t1394_setup_isoch_cec() directs the 1394 Software Framework
2379  *		to allocate isochronous resources and invoke the setup_target()
2380  *		callback for each member of the Isoch CEC.  This call may
2381  *		fail because bandwidth was unavailable (T1394_ENO_BANDWIDTH),
2382  *		channels were unavailable (T1394_ENO_CHANNEL), or one of the
2383  *		member targets returned failure from its setup_target()
2384  *		callback.
2385  */
2386 /* ARGSUSED */
2387 int
2388 t1394_setup_isoch_cec(t1394_handle_t t1394_hdl,
2389     t1394_isoch_cec_handle_t t1394_isoch_cec_hdl, uint_t flags, int *result)
2390 {
2391 	s1394_hal_t			*hal;
2392 	s1394_isoch_cec_t		*cec_curr;
2393 	s1394_isoch_cec_member_t	*member_curr;
2394 	t1394_setup_target_args_t	target_args;
2395 	uint64_t			temp_chnl_mask;
2396 	uint32_t			old_chnl;
2397 	uint32_t			try_chnl;
2398 	uint_t				bw_alloc_units;
2399 	uint_t				generation;
2400 	int				chnl_num;
2401 	int				err;
2402 	int				ret;
2403 	int				j;
2404 	int	(*setup_callback)(t1394_isoch_cec_handle_t, opaque_t,
2405 			    t1394_setup_target_args_t *);
2406 
2407 	TNF_PROBE_0_DEBUG(t1394_setup_isoch_cec_enter,
2408 	    S1394_TNF_SL_ISOCH_STACK, "");
2409 
2410 	ASSERT(t1394_hdl != NULL);
2411 	ASSERT(t1394_isoch_cec_hdl != NULL);
2412 
2413 	hal = ((s1394_target_t *)t1394_hdl)->on_hal;
2414 
2415 	/* Convert the handle to an Isoch CEC pointer */
2416 	cec_curr = (s1394_isoch_cec_t *)t1394_isoch_cec_hdl;
2417 
2418 	/* Lock the Isoch CEC member list */
2419 	mutex_enter(&cec_curr->isoch_cec_mutex);
2420 
2421 	/* Are we in any callbacks? */
2422 	if (CEC_IN_ANY_CALLBACKS(cec_curr)) {
2423 		/* Unlock the Isoch CEC member list */
2424 		mutex_exit(&cec_curr->isoch_cec_mutex);
2425 		TNF_PROBE_1(t1394_setup_isoch_cec_error,
2426 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
2427 		    "Not allowed to setup Isoch CEC (in callbacks)");
2428 		TNF_PROBE_0_DEBUG(t1394_setup_isoch_cec_exit,
2429 		    S1394_TNF_SL_ISOCH_STACK, "");
2430 		return (DDI_FAILURE);
2431 	}
2432 
2433 	/* Is "setup" a legal state transition? */
2434 	if (CEC_TRANSITION_LEGAL(cec_curr, ISOCH_CEC_SETUP) == 0) {
2435 		/* Unlock the Isoch CEC member list */
2436 		mutex_exit(&cec_curr->isoch_cec_mutex);
2437 		TNF_PROBE_1(t1394_setup_isoch_cec_error,
2438 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
2439 		    "Not allowed to setup Isoch CEC");
2440 		TNF_PROBE_0_DEBUG(t1394_setup_isoch_cec_exit,
2441 		    S1394_TNF_SL_ISOCH_STACK, "");
2442 		return (DDI_FAILURE);
2443 	}
2444 
2445 	/* If T1394_NO_IRM_ALLOC is set then don't allocate... do callbacks */
2446 	if (cec_curr->cec_options & T1394_NO_IRM_ALLOC) {
2447 		goto setup_do_callbacks;
2448 	}
2449 
2450 	/* Allocate bandwidth and channels */
2451 	for (j = 0; j < S1394_ISOCH_ALLOC_RETRIES; j++) {
2452 		/*
2453 		 * Get the current generation number - don't
2454 		 * need the lock because we are read only here
2455 		 */
2456 		generation = hal->generation_count;
2457 
2458 		/* Compute how much bandwidth is needed */
2459 		bw_alloc_units = s1394_compute_bw_alloc_units(hal,
2460 		    cec_curr->bandwidth, cec_curr->filter_current_speed);
2461 
2462 		/* Check that the generation has not changed - */
2463 		/* don't need the lock (read only) */
2464 		if (generation != hal->generation_count)
2465 			continue;
2466 
2467 		/* Unlock the Isoch CEC member list */
2468 		mutex_exit(&cec_curr->isoch_cec_mutex);
2469 
2470 		/* Try to allocate the bandwidth */
2471 		ret = s1394_bandwidth_alloc(hal, bw_alloc_units, generation,
2472 		    &err);
2473 
2474 		/* Lock the Isoch CEC member list */
2475 		mutex_enter(&cec_curr->isoch_cec_mutex);
2476 
2477 		/* If there was a bus reset, start over */
2478 		if (ret == DDI_FAILURE) {
2479 			if (err == CMD1394_EBUSRESET) {
2480 				continue; /* start over and try again */
2481 			} else {
2482 				*result = T1394_ENO_BANDWIDTH;
2483 				/* Unlock the Isoch CEC member list */
2484 				mutex_exit(&cec_curr->isoch_cec_mutex);
2485 				TNF_PROBE_1(t1394_setup_isoch_cec_error,
2486 				    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string,
2487 				    msg, "Unable to allocate isoch bandwidth");
2488 				TNF_PROBE_0_DEBUG(t1394_setup_isoch_cec_exit,
2489 				    S1394_TNF_SL_ISOCH_STACK, "");
2490 				return (DDI_FAILURE);
2491 			}
2492 		}
2493 
2494 		/* Check that the generation has not changed - */
2495 		/* don't need the lock (read only) */
2496 		if (generation != hal->generation_count)
2497 			continue;
2498 
2499 		/*
2500 		 * Allocate a channel
2501 		 *    From IEEE 1394-1995, Section 8.3.2.3.8: "Bits
2502 		 *    allocated in the CHANNELS_AVAILABLE_HI field of
2503 		 *    this register shall start at bit zero (channel
2504 		 *    number zero), and additional channel numbers shall
2505 		 *    be represented in a monotonically increasing sequence
2506 		 *    of bit numbers up to a maximum of bit 31 (channel
2507 		 *    number 31).  Bits allocated in the CHANNELS_AVAILABLE_LO
2508 		 *    field of this register shall start at bit zero
2509 		 *    (channel number 32), and additional channel numbers
2510 		 *    shall be represented in a monotonically increasing
2511 		 *    sequence of bit numbers up to a maximum of bit 31
2512 		 *    (channel number 63).
2513 		 */
2514 		temp_chnl_mask = cec_curr->filter_channel_mask;
2515 		for (chnl_num = 63; chnl_num >= 0; chnl_num--) {
2516 			if ((temp_chnl_mask & 1) == 1) {
2517 				try_chnl = (1 << ((63 - chnl_num) % 32));
2518 
2519 				/* Unlock the Isoch CEC member list */
2520 				mutex_exit(&cec_curr->isoch_cec_mutex);
2521 				if (chnl_num < 32) {
2522 					ret = s1394_channel_alloc(hal,
2523 					    try_chnl, generation,
2524 					    S1394_CHANNEL_ALLOC_HI, &old_chnl,
2525 					    &err);
2526 				} else {
2527 					ret = s1394_channel_alloc(hal,
2528 					    try_chnl, generation,
2529 					    S1394_CHANNEL_ALLOC_LO, &old_chnl,
2530 					    &err);
2531 				}
2532 				/* Lock the Isoch CEC member list */
2533 				mutex_enter(&cec_curr->isoch_cec_mutex);
2534 
2535 				/* Did we get a channel? (or a bus reset) */
2536 				if ((ret == DDI_SUCCESS) ||
2537 				    (err == CMD1394_EBUSRESET))
2538 					break;
2539 			}
2540 			temp_chnl_mask = temp_chnl_mask >> 1;
2541 		}
2542 
2543 		/* If we've tried all the possible channels, then fail */
2544 		if (chnl_num == 0) {
2545 			*result = T1394_ENO_CHANNEL;
2546 			/*
2547 			 * If we successfully allocate bandwidth, and
2548 			 * then fail getting a channel, we need to
2549 			 * free up the bandwidth
2550 			 */
2551 
2552 			/* Check that the generation has not changed */
2553 			/* lock not needed here (read only) */
2554 			if (generation != hal->generation_count)
2555 				continue;
2556 
2557 			/* Unlock the Isoch CEC member list */
2558 			mutex_exit(&cec_curr->isoch_cec_mutex);
2559 
2560 			/* Try to free up the bandwidth */
2561 			ret = s1394_bandwidth_free(hal, bw_alloc_units,
2562 			    generation, &err);
2563 
2564 			/* Lock the Isoch CEC member list */
2565 			mutex_enter(&cec_curr->isoch_cec_mutex);
2566 
2567 			if (ret == DDI_FAILURE) {
2568 				if (err == CMD1394_EBUSRESET) {
2569 					continue;
2570 				} else {
2571 					TNF_PROBE_1(t1394_setup_isoch_cec_error,
2572 					    S1394_TNF_SL_ISOCH_ERROR, "",
2573 					    tnf_string, msg,
2574 					    "Unable to free isoch bandwidth");
2575 				}
2576 			}
2577 
2578 			/* Unlock the Isoch CEC member list */
2579 			mutex_exit(&cec_curr->isoch_cec_mutex);
2580 			TNF_PROBE_1(t1394_setup_isoch_cec_error,
2581 			    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
2582 			    "Unable to allocate isoch channel");
2583 			TNF_PROBE_0_DEBUG(t1394_setup_isoch_cec_exit,
2584 			    S1394_TNF_SL_ISOCH_STACK, "");
2585 			return (DDI_FAILURE);
2586 		}
2587 
2588 		/* If we got a channel, we're done (else start over) */
2589 		if (ret == DDI_SUCCESS)
2590 			break;
2591 		else if (err == CMD1394_EBUSRESET)
2592 			continue;
2593 	}
2594 
2595 	/* Have we gotten too many bus resets? */
2596 	if (j == S1394_ISOCH_ALLOC_RETRIES) {
2597 		*result = T1394_ENO_BANDWIDTH;
2598 		/* Unlock the Isoch CEC member list */
2599 		mutex_exit(&cec_curr->isoch_cec_mutex);
2600 		TNF_PROBE_1(t1394_setup_isoch_cec_error,
2601 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
2602 		    "Unable to allocate isoch channel");
2603 		TNF_PROBE_0_DEBUG(t1394_setup_isoch_cec_exit,
2604 		    S1394_TNF_SL_ISOCH_STACK, "");
2605 		return (DDI_FAILURE);
2606 	}
2607 
2608 	cec_curr->realloc_valid	    = B_TRUE;
2609 	cec_curr->realloc_chnl_num  = chnl_num;
2610 	cec_curr->realloc_bandwidth = cec_curr->bandwidth;
2611 	cec_curr->realloc_speed	    = cec_curr->filter_current_speed;
2612 
2613 setup_do_callbacks:
2614 	/* Call all of the setup_target() callbacks */
2615 	target_args.channel_num	    = chnl_num;
2616 	target_args.channel_speed   = cec_curr->filter_current_speed;
2617 
2618 	/* Now we are going into the callbacks */
2619 	cec_curr->in_callbacks	    = B_TRUE;
2620 
2621 	/* Unlock the Isoch CEC member list */
2622 	mutex_exit(&cec_curr->isoch_cec_mutex);
2623 
2624 	member_curr = cec_curr->cec_member_list_head;
2625 	*result = 0;
2626 	while (member_curr != NULL) {
2627 		if (member_curr->isoch_cec_evts.setup_target != NULL) {
2628 			setup_callback =
2629 			    member_curr->isoch_cec_evts.setup_target;
2630 			ret = setup_callback(t1394_isoch_cec_hdl,
2631 			    member_curr->isoch_cec_evts_arg, &target_args);
2632 			if (ret != DDI_SUCCESS)
2633 				*result = T1394_ETARGET;
2634 		}
2635 		member_curr = member_curr->cec_mem_next;
2636 	}
2637 
2638 	/* Lock the Isoch CEC member list */
2639 	mutex_enter(&cec_curr->isoch_cec_mutex);
2640 
2641 	/* We are finished with the callbacks */
2642 	cec_curr->in_callbacks = B_FALSE;
2643 	if (cec_curr->cec_want_wakeup == B_TRUE) {
2644 		cec_curr->cec_want_wakeup = B_FALSE;
2645 		cv_broadcast(&cec_curr->in_callbacks_cv);
2646 	}
2647 
2648 	/*
2649 	 * Now "start" and "teardown" are legal state transitions
2650 	 * and "join", "free", and "setup" are illegal state transitions
2651 	 */
2652 	CEC_SET_LEGAL(cec_curr, (ISOCH_CEC_START | ISOCH_CEC_TEARDOWN));
2653 	CEC_SET_ILLEGAL(cec_curr, (ISOCH_CEC_JOIN | ISOCH_CEC_FREE |
2654 	    ISOCH_CEC_SETUP));
2655 
2656 	/* Unlock the Isoch CEC member list */
2657 	mutex_exit(&cec_curr->isoch_cec_mutex);
2658 
2659 	/* Return DDI_FAILURE if any targets failed setup */
2660 	if (*result != 0) {
2661 		TNF_PROBE_1(t1394_setup_isoch_cec_error,
2662 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
2663 		    "Target returned error in setup_target()");
2664 		TNF_PROBE_0_DEBUG(t1394_setup_isoch_cec_exit,
2665 		    S1394_TNF_SL_ISOCH_STACK, "");
2666 		return (DDI_FAILURE);
2667 	}
2668 
2669 	TNF_PROBE_0_DEBUG(t1394_setup_isoch_cec_exit,
2670 	    S1394_TNF_SL_ISOCH_STACK, "");
2671 	return (DDI_SUCCESS);
2672 }
2673 
2674 /*
2675  * Function:    t1394_start_isoch_cec()
2676  * Input(s):    t1394_hdl		The target "handle" returned by
2677  *					    t1394_attach()
2678  *		t1394_isoch_cec_hdl	The Isoch CEC "handle" returned by
2679  *					    t1394_alloc_isoch_cec()
2680  *		flags			The flags parameter is unused (for now)
2681  *
2682  * Output(s):	DDI_SUCCESS		All start_target() callbacks returned
2683  *					    successfully
2684  *		DDI_FAILURE		One or more start_target() callbacks
2685  *					    returned failure
2686  *
2687  * Description:	t1394_start_isoch_cec() directs the 1394 Software Framework
2688  *		to invoke each of the start_target() callbacks, first for
2689  *		each listener, then for the talker.
2690  */
2691 /* ARGSUSED */
2692 int
2693 t1394_start_isoch_cec(t1394_handle_t t1394_hdl,
2694     t1394_isoch_cec_handle_t t1394_isoch_cec_hdl, uint_t flags)
2695 {
2696 	s1394_isoch_cec_t	 *cec_curr;
2697 	s1394_isoch_cec_member_t *member_curr;
2698 	int			 ret;
2699 	boolean_t		 err;
2700 	int	(*start_callback)(t1394_isoch_cec_handle_t, opaque_t);
2701 
2702 	TNF_PROBE_0_DEBUG(t1394_start_isoch_cec_enter,
2703 	    S1394_TNF_SL_ISOCH_STACK, "");
2704 
2705 	ASSERT(t1394_hdl != NULL);
2706 	ASSERT(t1394_isoch_cec_hdl != NULL);
2707 
2708 	/* Convert the handle to an Isoch CEC pointer */
2709 	cec_curr = (s1394_isoch_cec_t *)t1394_isoch_cec_hdl;
2710 
2711 	/* Lock the Isoch CEC member list */
2712 	mutex_enter(&cec_curr->isoch_cec_mutex);
2713 
2714 	/* Are we in any callbacks? */
2715 	if (CEC_IN_ANY_CALLBACKS(cec_curr)) {
2716 		/* Unlock the Isoch CEC member list */
2717 		mutex_exit(&cec_curr->isoch_cec_mutex);
2718 		TNF_PROBE_1(t1394_start_isoch_cec_error,
2719 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
2720 		    "Not allowed to start Isoch CEC (in callbacks)");
2721 		TNF_PROBE_0_DEBUG(t1394_start_isoch_cec_exit,
2722 		    S1394_TNF_SL_ISOCH_STACK, "");
2723 		return (DDI_FAILURE);
2724 	}
2725 
2726 	/* Is "start" a legal state transition? */
2727 	if (CEC_TRANSITION_LEGAL(cec_curr, ISOCH_CEC_START) == 0) {
2728 		/* Unlock the Isoch CEC member list */
2729 		mutex_exit(&cec_curr->isoch_cec_mutex);
2730 		TNF_PROBE_1(t1394_start_isoch_cec_error,
2731 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
2732 		    "Not allowed to start Isoch CEC");
2733 		TNF_PROBE_0_DEBUG(t1394_start_isoch_cec_exit,
2734 		    S1394_TNF_SL_ISOCH_STACK, "");
2735 		return (DDI_FAILURE);
2736 	}
2737 
2738 	/* Now we are going into the callbacks */
2739 	cec_curr->in_callbacks = B_TRUE;
2740 
2741 	/* Unlock the Isoch CEC member list */
2742 	mutex_exit(&cec_curr->isoch_cec_mutex);
2743 
2744 	/*
2745 	 * Call all of the start_target() callbacks
2746 	 * Start at the tail (listeners first) and
2747 	 * go toward the head (talker last)
2748 	 */
2749 	member_curr = cec_curr->cec_member_list_tail;
2750 	err = B_FALSE;
2751 	while (member_curr != NULL) {
2752 		if (member_curr->isoch_cec_evts.start_target != NULL) {
2753 			start_callback =
2754 			    member_curr->isoch_cec_evts.start_target;
2755 			ret = start_callback(t1394_isoch_cec_hdl,
2756 			    member_curr->isoch_cec_evts_arg);
2757 		if (ret != DDI_SUCCESS)
2758 			err = B_TRUE;
2759 		}
2760 		member_curr = member_curr->cec_mem_prev;
2761 	}
2762 
2763 	/* Lock the Isoch CEC member list */
2764 	mutex_enter(&cec_curr->isoch_cec_mutex);
2765 
2766 	/* We are finished with the callbacks */
2767 	cec_curr->in_callbacks = B_FALSE;
2768 	if (cec_curr->cec_want_wakeup == B_TRUE) {
2769 		cec_curr->cec_want_wakeup = B_FALSE;
2770 		cv_broadcast(&cec_curr->in_callbacks_cv);
2771 	}
2772 
2773 	/*
2774 	 * Now "stop" is a legal state transitions
2775 	 * and "start" and "teardown" are illegal state transitions
2776 	 */
2777 	CEC_SET_LEGAL(cec_curr, ISOCH_CEC_STOP);
2778 	CEC_SET_ILLEGAL(cec_curr, (ISOCH_CEC_START | ISOCH_CEC_TEARDOWN));
2779 
2780 	/* Unlock the Isoch CEC member list */
2781 	mutex_exit(&cec_curr->isoch_cec_mutex);
2782 
2783 	/* Return DDI_FAILURE if any targets failed start */
2784 	if (err == B_TRUE) {
2785 		TNF_PROBE_1(t1394_start_isoch_cec_error,
2786 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
2787 		    "Target returned error in start_target()");
2788 		TNF_PROBE_0_DEBUG(t1394_start_isoch_cec_exit,
2789 		    S1394_TNF_SL_ISOCH_STACK, "");
2790 		return (DDI_FAILURE);
2791 	}
2792 
2793 	TNF_PROBE_0_DEBUG(t1394_start_isoch_cec_exit,
2794 		    S1394_TNF_SL_ISOCH_STACK, "");
2795 	return (DDI_SUCCESS);
2796 }
2797 
2798 /*
2799  * Function:    t1394_stop_isoch_cec()
2800  * Input(s):    t1394_hdl		The target "handle" returned by
2801  *					    t1394_attach()
2802  *		t1394_isoch_cec_hdl	The Isoch CEC "handle" returned by
2803  *					    t1394_alloc_isoch_cec()
2804  *		flags			The flags parameter is unused (for now)
2805  *
2806  * Output(s):	DDI_SUCCESS		Target successfully stopped the
2807  *					    Isoch CEC
2808  *		DDI_FAILURE		Target failed to stop the Isoch CEC
2809  *
2810  * Description:	t1394_stop_isoch_cec() directs the 1394 Software Framework
2811  *		to invoke each of the stop_target() callbacks, first for
2812  *		the talker, then for each listener.
2813  *		(This call will fail if it is called at an
2814  *		inappropriate time, i.e. before the t1394_start_isoch_cec()
2815  *		call, etc.)
2816  */
2817 /* ARGSUSED */
2818 int
2819 t1394_stop_isoch_cec(t1394_handle_t t1394_hdl,
2820     t1394_isoch_cec_handle_t t1394_isoch_cec_hdl, uint_t flags)
2821 {
2822 	s1394_isoch_cec_t	 *cec_curr;
2823 	s1394_isoch_cec_member_t *member_curr;
2824 	void	(*stop_callback)(t1394_isoch_cec_handle_t, opaque_t);
2825 
2826 	TNF_PROBE_0_DEBUG(t1394_stop_isoch_cec_enter,
2827 	    S1394_TNF_SL_ISOCH_STACK, "");
2828 
2829 	ASSERT(t1394_hdl != NULL);
2830 	ASSERT(t1394_isoch_cec_hdl != NULL);
2831 
2832 	/* Convert the handle to an Isoch CEC pointer */
2833 	cec_curr = (s1394_isoch_cec_t *)t1394_isoch_cec_hdl;
2834 
2835 	/* Lock the Isoch CEC member list */
2836 	mutex_enter(&cec_curr->isoch_cec_mutex);
2837 
2838 	/* Are we in any callbacks? */
2839 	if (CEC_IN_ANY_CALLBACKS(cec_curr)) {
2840 		/* Unlock the Isoch CEC member list */
2841 		mutex_exit(&cec_curr->isoch_cec_mutex);
2842 		TNF_PROBE_1(t1394_stop_isoch_cec_error,
2843 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
2844 		    "Not allowed to stop Isoch CEC (in callbacks)");
2845 		TNF_PROBE_0_DEBUG(t1394_stop_isoch_cec_exit,
2846 		    S1394_TNF_SL_ISOCH_STACK, "");
2847 		return (DDI_FAILURE);
2848 	}
2849 
2850 	/* Is "stop" a legal state transition? */
2851 	if (CEC_TRANSITION_LEGAL(cec_curr, ISOCH_CEC_STOP) == 0) {
2852 		/* Unlock the Isoch CEC member list */
2853 		mutex_exit(&cec_curr->isoch_cec_mutex);
2854 		TNF_PROBE_1(t1394_stop_isoch_cec_error,
2855 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
2856 		    "Not allowed to stop Isoch CEC");
2857 		TNF_PROBE_0_DEBUG(t1394_stop_isoch_cec_exit,
2858 		    S1394_TNF_SL_ISOCH_STACK, "");
2859 		return (DDI_FAILURE);
2860 	}
2861 
2862 	/* Now we are going into the callbacks */
2863 	cec_curr->in_callbacks = B_TRUE;
2864 
2865 	/* Unlock the Isoch CEC member list */
2866 	mutex_exit(&cec_curr->isoch_cec_mutex);
2867 
2868 	/*
2869 	 * Call all of the stop_target() callbacks
2870 	 * Start at the head (talker first) and
2871 	 * go toward the tail (listeners last)
2872 	 */
2873 	member_curr = cec_curr->cec_member_list_head;
2874 	while (member_curr != NULL) {
2875 		if (member_curr->isoch_cec_evts.stop_target != NULL) {
2876 			stop_callback =
2877 			    member_curr->isoch_cec_evts.stop_target;
2878 			stop_callback(t1394_isoch_cec_hdl,
2879 			    member_curr->isoch_cec_evts_arg);
2880 		}
2881 		member_curr = member_curr->cec_mem_next;
2882 	}
2883 
2884 	/* Lock the Isoch CEC member list */
2885 	mutex_enter(&cec_curr->isoch_cec_mutex);
2886 
2887 	/* We are finished with the callbacks */
2888 	cec_curr->in_callbacks = B_FALSE;
2889 	if (cec_curr->cec_want_wakeup == B_TRUE) {
2890 		cec_curr->cec_want_wakeup = B_FALSE;
2891 		cv_broadcast(&cec_curr->in_callbacks_cv);
2892 	}
2893 
2894 	/*
2895 	 * Now "start" and "teardown" are legal state transitions
2896 	 * and "stop" is an illegal state transitions
2897 	 */
2898 	CEC_SET_LEGAL(cec_curr, (ISOCH_CEC_START | ISOCH_CEC_TEARDOWN));
2899 	CEC_SET_ILLEGAL(cec_curr, ISOCH_CEC_STOP);
2900 
2901 	/* Unlock the Isoch CEC member list */
2902 	mutex_exit(&cec_curr->isoch_cec_mutex);
2903 
2904 	TNF_PROBE_0_DEBUG(t1394_stop_isoch_cec_exit,
2905 	    S1394_TNF_SL_ISOCH_STACK, "");
2906 	return (DDI_SUCCESS);
2907 }
2908 
2909 /*
2910  * Function:    t1394_teardown_isoch_cec()
2911  * Input(s):    t1394_hdl		The target "handle" returned by
2912  *					    t1394_attach()
2913  *		t1394_isoch_cec_hdl	The Isoch CEC "handle" returned by
2914  *					    t1394_alloc_isoch_cec()
2915  *		flags			The flags parameter is unused (for now)
2916  *
2917  * Output(s):	DDI_SUCCESS		Target successfully tore down the
2918  *					    Isoch CEC
2919  *		DDI_FAILURE		Target failed to tear down the
2920  *					    Isoch CEC
2921  *
2922  * Description:	t1394_teardown_isoch_cec() directs the 1394 Software Framework
2923  *		to free up any isochronous resources we might be holding and
2924  *		call all of the teardown_target() callbacks.
2925  *		(This call will fail if it is called at an
2926  *		inappropriate time, i.e. before the t1394_start_isoch_cec()
2927  *		call, before the t1394_stop_isoch_cec, etc.
2928  */
2929 /* ARGSUSED */
2930 int
2931 t1394_teardown_isoch_cec(t1394_handle_t t1394_hdl,
2932     t1394_isoch_cec_handle_t t1394_isoch_cec_hdl, uint_t flags)
2933 {
2934 	s1394_hal_t		 *hal;
2935 	s1394_isoch_cec_t	 *cec_curr;
2936 	s1394_isoch_cec_member_t *member_curr;
2937 	uint32_t		 chnl_mask;
2938 	uint32_t		 old_chnl_mask;
2939 	uint_t			 bw_alloc_units;
2940 	uint_t			 generation;
2941 	int			 ret;
2942 	int			 err;
2943 	void	(*teardown_callback)(t1394_isoch_cec_handle_t, opaque_t);
2944 
2945 	TNF_PROBE_0_DEBUG(t1394_teardown_isoch_cec_enter,
2946 	    S1394_TNF_SL_ISOCH_STACK, "");
2947 
2948 	ASSERT(t1394_hdl != NULL);
2949 	ASSERT(t1394_isoch_cec_hdl != NULL);
2950 
2951 	hal = ((s1394_target_t *)t1394_hdl)->on_hal;
2952 
2953 	/* Convert the handle to an Isoch CEC pointer */
2954 	cec_curr = (s1394_isoch_cec_t *)t1394_isoch_cec_hdl;
2955 
2956 	/* Lock the Isoch CEC member list */
2957 	mutex_enter(&cec_curr->isoch_cec_mutex);
2958 
2959 	/* Are we in any callbacks? */
2960 	if (CEC_IN_ANY_CALLBACKS(cec_curr)) {
2961 		/* Unlock the Isoch CEC member list */
2962 		mutex_exit(&cec_curr->isoch_cec_mutex);
2963 		TNF_PROBE_1(t1394_teardown_isoch_cec_error,
2964 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
2965 		    "Not allowed to teardown Isoch CEC (in callbacks)");
2966 		TNF_PROBE_0_DEBUG(t1394_teardown_isoch_cec_exit,
2967 		    S1394_TNF_SL_ISOCH_STACK, "");
2968 		return (DDI_FAILURE);
2969 	}
2970 
2971 	/* Is "teardown" a legal state transition? */
2972 	if (CEC_TRANSITION_LEGAL(cec_curr, ISOCH_CEC_TEARDOWN) == 0) {
2973 		/* Unlock the Isoch CEC member list */
2974 		mutex_exit(&cec_curr->isoch_cec_mutex);
2975 		TNF_PROBE_1(t1394_teardown_isoch_cec_error,
2976 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
2977 		    "Not allowed to teardown Isoch CEC");
2978 		TNF_PROBE_0_DEBUG(t1394_teardown_isoch_cec_exit,
2979 		    S1394_TNF_SL_ISOCH_STACK, "");
2980 		return (DDI_FAILURE);
2981 	}
2982 
2983 	/* If T1394_NO_IRM_ALLOC is set then don't free... do callbacks */
2984 	if (cec_curr->cec_options & T1394_NO_IRM_ALLOC) {
2985 		goto teardown_do_callbacks;
2986 	}
2987 
2988 	/* If nothing has been allocated or we failed to */
2989 	/* reallocate, then we are done... call the callbacks */
2990 	if ((cec_curr->realloc_valid == B_FALSE) ||
2991 	    (cec_curr->realloc_failed == B_TRUE)) {
2992 		goto teardown_do_callbacks;
2993 	}
2994 
2995 	/*
2996 	 * Get the current generation number - don't need the
2997 	 * topology tree mutex here because it is read-only, and
2998 	 * there is a race condition with or without it.
2999 	 */
3000 	generation = hal->generation_count;
3001 
3002 	/* Compute the amount bandwidth to free */
3003 	bw_alloc_units = s1394_compute_bw_alloc_units(hal,
3004 	    cec_curr->bandwidth, cec_curr->realloc_speed);
3005 
3006 	/* Check that the generation has not changed - */
3007 	/* don't need the lock (read only) */
3008 	if (generation != hal->generation_count)
3009 		goto teardown_do_callbacks;
3010 
3011 	/* Unlock the Isoch CEC member list */
3012 	mutex_exit(&cec_curr->isoch_cec_mutex);
3013 
3014 	/* Try to free up the bandwidth */
3015 	ret = s1394_bandwidth_free(hal, bw_alloc_units, generation, &err);
3016 
3017 	/* Lock the Isoch CEC member list */
3018 	mutex_enter(&cec_curr->isoch_cec_mutex);
3019 
3020 	if (ret == DDI_FAILURE) {
3021 		if (err == CMD1394_EBUSRESET) {
3022 			goto teardown_do_callbacks;
3023 		} else {
3024 			TNF_PROBE_1(t1394_teardown_isoch_cec_error,
3025 			    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
3026 			    "Unable to free allocated bandwidth");
3027 		}
3028 	}
3029 
3030 	/* Free the allocated channel */
3031 	chnl_mask = (1 << ((63 - cec_curr->realloc_chnl_num) % 32));
3032 
3033 	/* Unlock the Isoch CEC member list */
3034 	mutex_exit(&cec_curr->isoch_cec_mutex);
3035 	if (cec_curr->realloc_chnl_num < 32) {
3036 		ret = s1394_channel_free(hal, chnl_mask, generation,
3037 		    S1394_CHANNEL_ALLOC_HI, &old_chnl_mask, &err);
3038 	} else {
3039 		ret = s1394_channel_free(hal, chnl_mask, generation,
3040 		    S1394_CHANNEL_ALLOC_LO, &old_chnl_mask, &err);
3041 	}
3042 	/* Lock the Isoch CEC member list */
3043 	mutex_enter(&cec_curr->isoch_cec_mutex);
3044 
3045 	if (ret == DDI_FAILURE) {
3046 		if (err == CMD1394_EBUSRESET) {
3047 			goto teardown_do_callbacks;
3048 		} else {
3049 			TNF_PROBE_1(t1394_teardown_isoch_cec_error,
3050 			    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
3051 			    "Unable to free allocated bandwidth");
3052 		}
3053 	}
3054 
3055 teardown_do_callbacks:
3056 	/* From here on reallocation is unnecessary */
3057 	cec_curr->realloc_valid	    = B_FALSE;
3058 	cec_curr->realloc_chnl_num  = 0;
3059 	cec_curr->realloc_bandwidth = 0;
3060 
3061 	/* Now we are going into the callbacks */
3062 	cec_curr->in_callbacks	    = B_TRUE;
3063 
3064 	/* Unlock the Isoch CEC member list */
3065 	mutex_exit(&cec_curr->isoch_cec_mutex);
3066 
3067 	/* Call all of the teardown_target() callbacks */
3068 	member_curr = cec_curr->cec_member_list_head;
3069 	while (member_curr != NULL) {
3070 		if (member_curr->isoch_cec_evts.teardown_target != NULL) {
3071 			teardown_callback =
3072 			    member_curr->isoch_cec_evts.teardown_target;
3073 			teardown_callback(t1394_isoch_cec_hdl,
3074 			    member_curr->isoch_cec_evts_arg);
3075 		}
3076 		member_curr = member_curr->cec_mem_next;
3077 	}
3078 
3079 	/* Lock the Isoch CEC member list */
3080 	mutex_enter(&cec_curr->isoch_cec_mutex);
3081 
3082 	/* We are finished with the callbacks */
3083 	cec_curr->in_callbacks = B_FALSE;
3084 	if (cec_curr->cec_want_wakeup == B_TRUE) {
3085 		cec_curr->cec_want_wakeup = B_FALSE;
3086 		cv_broadcast(&cec_curr->in_callbacks_cv);
3087 	}
3088 
3089 	/*
3090 	 * Now "join" and "setup" are legal state transitions
3091 	 * and "start" and "teardown" are illegal state transitions
3092 	 */
3093 	CEC_SET_LEGAL(cec_curr, (ISOCH_CEC_JOIN | ISOCH_CEC_SETUP));
3094 	CEC_SET_ILLEGAL(cec_curr, (ISOCH_CEC_START | ISOCH_CEC_TEARDOWN));
3095 
3096 	/* And if the member list is empty, then "free" is legal too */
3097 	if ((cec_curr->cec_member_list_head == NULL) &&
3098 	    (cec_curr->cec_member_list_tail == NULL)) {
3099 		CEC_SET_LEGAL(cec_curr, ISOCH_CEC_FREE);
3100 	}
3101 
3102 	/* Unlock the Isoch CEC member list */
3103 	mutex_exit(&cec_curr->isoch_cec_mutex);
3104 	TNF_PROBE_0_DEBUG(t1394_teardown_isoch_cec_exit,
3105 	    S1394_TNF_SL_ISOCH_STACK, "");
3106 	return (DDI_SUCCESS);
3107 }
3108 
3109 /*
3110  * Function:    t1394_alloc_isoch_dma()
3111  * Input(s):    t1394_hdl		The target "handle" returned by
3112  *					    t1394_attach()
3113  *		idi			This structure contains information
3114  *					    for configuring the data flow for
3115  *					    isochronous DMA
3116  *		flags			The flags parameter is unused (for now)
3117  *
3118  * Output(s):	t1394_idma_hdl		The IDMA "handle" used in all
3119  *					    subsequent isoch_dma() calls
3120  *		result			Used to pass more specific info back
3121  *					    to target
3122  *
3123  * Description:	t1394_alloc_isoch_dma() allocates and initializes an
3124  *		isochronous DMA resource for transmitting or receiving
3125  *		isochronous data.  If it fails, result may hold
3126  *		T1394_EIDMA_NO_RESRCS, indicating that no isoch DMA resource
3127  *		are available.
3128  */
3129 /* ARGSUSED */
3130 int
3131 t1394_alloc_isoch_dma(t1394_handle_t t1394_hdl,
3132     id1394_isoch_dmainfo_t *idi, uint_t flags,
3133     t1394_isoch_dma_handle_t *t1394_idma_hdl, int *result)
3134 {
3135 	s1394_hal_t	*hal;
3136 	int		ret;
3137 
3138 	TNF_PROBE_0_DEBUG(t1394_alloc_isoch_dma_enter,
3139 	    S1394_TNF_SL_ISOCH_STACK, "");
3140 
3141 	ASSERT(t1394_hdl != NULL);
3142 	ASSERT(idi != NULL);
3143 	ASSERT(t1394_idma_hdl != NULL);
3144 
3145 	/* Find the HAL this target resides on */
3146 	hal = ((s1394_target_t *)t1394_hdl)->on_hal;
3147 
3148 	/* Sanity check dma options.  If talk enabled, listen should be off */
3149 	if ((idi->idma_options & ID1394_TALK) &&
3150 	    (idi->idma_options != ID1394_TALK)) {
3151 		TNF_PROBE_1(t1394_alloc_isoch_dma_talk_conflict_error,
3152 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
3153 		    "conflicting idma options; talker and listener");
3154 		TNF_PROBE_0_DEBUG(t1394_alloc_isoch_dma_exit,
3155 		    S1394_TNF_SL_ISOCH_STACK, "");
3156 
3157 		*result = T1394_EIDMA_CONFLICT;
3158 		return (DDI_FAILURE);
3159 	}
3160 
3161 	/* Only one listen mode allowed */
3162 	if ((idi->idma_options & ID1394_LISTEN_PKT_MODE) &&
3163 	    (idi->idma_options & ID1394_LISTEN_BUF_MODE)) {
3164 		TNF_PROBE_1(t1394_alloc_isoch_dma_listen_conflict_error,
3165 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
3166 		    "conflicting idma options; both listener modes set");
3167 		TNF_PROBE_0_DEBUG(t1394_alloc_isoch_dma_exit,
3168 		    S1394_TNF_SL_ISOCH_STACK, "");
3169 
3170 		*result = T1394_EIDMA_CONFLICT;
3171 		return (DDI_FAILURE);
3172 	}
3173 
3174 	/* Have HAL alloc a resource and compile ixl */
3175 	ret = HAL_CALL(hal).alloc_isoch_dma(hal->halinfo.hal_private, idi,
3176 	    (void **)t1394_idma_hdl, result);
3177 
3178 	if (ret != DDI_SUCCESS) {
3179 		TNF_PROBE_1(t1394_alloc_isoch_dma_hal_error,
3180 		    S1394_TNF_SL_ISOCH_ERROR, "", tnf_string, msg,
3181 		    "HAL alloc_isoch_dma error, maybe IXL compilation");
3182 		if (*result == IXL1394_ENO_DMA_RESRCS) {
3183 			*result = T1394_EIDMA_NO_RESRCS;
3184 		}
3185 	}
3186 
3187 	TNF_PROBE_0_DEBUG(t1394_alloc_isoch_dma_exit,
3188 	    S1394_TNF_SL_ISOCH_STACK, "");
3189 	return (ret);
3190 }
3191 
3192 /*
3193  * Function:    t1394_free_isoch_dma()
3194  * Input(s):    t1394_hdl		The target "handle" returned by
3195  *					    t1394_attach()
3196  *		flags			The flags parameter is unused (for now)
3197  *		t1394_idma_hdl		The IDMA "handle" returned by
3198  *					    t1394_alloc_isoch_dma()
3199  *
3200  * Output(s):	None
3201  *
3202  * Description:	t1394_free_isoch_dma() is used to free all DMA resources
3203  *		allocated for the isoch stream associated with t1394_idma_hdl.
3204  */
3205 /* ARGSUSED */
3206 void
3207 t1394_free_isoch_dma(t1394_handle_t t1394_hdl, uint_t flags,
3208     t1394_isoch_dma_handle_t *t1394_idma_hdl)
3209 {
3210 	s1394_hal_t	*hal;
3211 
3212 	TNF_PROBE_0_DEBUG(t1394_free_isoch_dma_enter,
3213 	    S1394_TNF_SL_ISOCH_STACK, "");
3214 
3215 	ASSERT(t1394_hdl != NULL);
3216 	ASSERT(*t1394_idma_hdl != NULL);
3217 
3218 	/* Find the HAL this target resides on */
3219 	hal = ((s1394_target_t *)t1394_hdl)->on_hal;
3220 
3221 	/* Tell HAL to release local isoch dma resources */
3222 	HAL_CALL(hal).free_isoch_dma(hal->halinfo.hal_private, *t1394_idma_hdl);
3223 
3224 	/* Null out isoch handle */
3225 	*t1394_idma_hdl = NULL;
3226 
3227 	TNF_PROBE_0_DEBUG(t1394_free_isoch_dma_exit,
3228 	    S1394_TNF_SL_ISOCH_STACK, "");
3229 }
3230 
3231 /*
3232  * Function:    t1394_start_isoch_dma()
3233  * Input(s):    t1394_hdl		The target "handle" returned by
3234  *					    t1394_attach()
3235  *		t1394_idma_hdl		The IDMA "handle" returned by
3236  *					    t1394_alloc_isoch_dma()
3237  *		idma_ctrlinfo		This structure contains control args
3238  *					    used when starting isoch DMA for
3239  *					    the allocated resource
3240  *		flags			One flag defined - ID1394_START_ON_CYCLE
3241  *
3242  * Output(s):	result			Used to pass more specific info back
3243  *					    to target
3244  *
3245  * Description:	t1394_start_isoch_dma() is used to start DMA for the isoch
3246  *		stream associated with t1394_idma_hdl.
3247  */
3248 /* ARGSUSED */
3249 int
3250 t1394_start_isoch_dma(t1394_handle_t t1394_hdl,
3251     t1394_isoch_dma_handle_t t1394_idma_hdl,
3252     id1394_isoch_dma_ctrlinfo_t *idma_ctrlinfo, uint_t flags,
3253     int *result)
3254 {
3255 	s1394_hal_t	*hal;
3256 	int		ret;
3257 
3258 	TNF_PROBE_0_DEBUG(t1394_start_isoch_dma_enter,
3259 	    S1394_TNF_SL_ISOCH_STACK, "");
3260 
3261 	ASSERT(t1394_hdl != NULL);
3262 	ASSERT(t1394_idma_hdl != NULL);
3263 	ASSERT(idma_ctrlinfo != NULL);
3264 
3265 	/* Find the HAL this target resides on */
3266 	hal = ((s1394_target_t *)t1394_hdl)->on_hal;
3267 
3268 	ret = HAL_CALL(hal).start_isoch_dma(hal->halinfo.hal_private,
3269 	    (void *)t1394_idma_hdl, idma_ctrlinfo, flags, result);
3270 
3271 	TNF_PROBE_0_DEBUG(t1394_start_isoch_dma_exit,
3272 	    S1394_TNF_SL_ISOCH_STACK, "");
3273 	return (ret);
3274 }
3275 
3276 /*
3277  * Function:    t1394_stop_isoch_dma()
3278  * Input(s):    t1394_hdl		The target "handle" returned by
3279  *					    t1394_attach()
3280  *		t1394_idma_hdl		The IDMA "handle" returned by
3281  *					    t1394_alloc_isoch_dma()
3282  *		flags			The flags parameter is unused (for now)
3283  *
3284  * Output(s):	None
3285  *
3286  * Description:	t1394_stop_isoch_dma() is used to stop DMA for the isoch
3287  *		stream associated with t1394_idma_hdl.
3288  */
3289 /* ARGSUSED */
3290 void
3291 t1394_stop_isoch_dma(t1394_handle_t t1394_hdl,
3292     t1394_isoch_dma_handle_t t1394_idma_hdl, uint_t flags)
3293 {
3294 	s1394_hal_t	*hal;
3295 	int		result;
3296 
3297 	TNF_PROBE_0_DEBUG(t1394_stop_isoch_dma_enter,
3298 	    S1394_TNF_SL_ISOCH_STACK, "");
3299 
3300 	ASSERT(t1394_hdl != NULL);
3301 	ASSERT(t1394_idma_hdl != NULL);
3302 
3303 	/* Find the HAL this target resides on */
3304 	hal = ((s1394_target_t *)t1394_hdl)->on_hal;
3305 
3306 	HAL_CALL(hal).stop_isoch_dma(hal->halinfo.hal_private,
3307 	    (void *)t1394_idma_hdl, &result);
3308 
3309 	TNF_PROBE_0_DEBUG(t1394_stop_isoch_dma_exit,
3310 	    S1394_TNF_SL_ISOCH_STACK, "");
3311 }
3312 
3313 /*
3314  * Function:    t1394_update_isoch_dma()
3315  * Input(s):    t1394_hdl		The target "handle" returned by
3316  *					    t1394_attach()
3317  *		t1394_idma_hdl		The IDMA "handle" returned by
3318  *					    t1394_alloc_isoch_dma()
3319  *		idma_updateinfo		This structure contains ixl command args
3320  *					    used when updating args in an
3321  *					    existing list of ixl commands with
3322  *					    args in a new list of ixl commands.
3323  *		flags			The flags parameter is unused (for now)
3324  *
3325  * Output(s):	result			Used to pass more specific info back
3326  *					    to target
3327  *
3328  * Description:	t1394_update_isoch_dma() is used to alter an IXL program that
3329  *		has already been built (compiled) by t1394_alloc_isoch_dma().
3330  */
3331 /* ARGSUSED */
3332 int
3333 t1394_update_isoch_dma(t1394_handle_t t1394_hdl,
3334     t1394_isoch_dma_handle_t t1394_idma_hdl,
3335     id1394_isoch_dma_updateinfo_t *idma_updateinfo, uint_t flags,
3336     int *result)
3337 {
3338 	s1394_hal_t	*hal;
3339 	int		ret;
3340 
3341 	TNF_PROBE_0_DEBUG(t1394_update_isoch_dma_enter,
3342 	    S1394_TNF_SL_ISOCH_STACK, "");
3343 
3344 	ASSERT(t1394_hdl != NULL);
3345 	ASSERT(t1394_idma_hdl != NULL);
3346 	ASSERT(idma_updateinfo != NULL);
3347 
3348 	/* Find the HAL this target resides on */
3349 	hal = ((s1394_target_t *)t1394_hdl)->on_hal;
3350 
3351 	ret = HAL_CALL(hal).update_isoch_dma(hal->halinfo.hal_private,
3352 	    (void *)t1394_idma_hdl, idma_updateinfo, flags, result);
3353 
3354 	TNF_PROBE_0_DEBUG(t1394_update_isoch_dma_exit,
3355 	    S1394_TNF_SL_ISOCH_STACK, "");
3356 	return (ret);
3357 }
3358 
3359 /*
3360  * Function:    t1394_initiate_bus_reset()
3361  * Input(s):    t1394_hdl		The target "handle" returned by
3362  *					    t1394_attach()
3363  *		flags			The flags parameter is unused (for now)
3364  *
3365  * Output(s):	None
3366  *
3367  * Description:	t1394_initiate_bus_reset() determines whether the local
3368  *		device has a P1394A PHY and will support the arbitrated
3369  *		short bus reset. If not, it will initiate a normal bus reset.
3370  */
3371 /* ARGSUSED */
3372 void
3373 t1394_initiate_bus_reset(t1394_handle_t t1394_hdl, uint_t flags)
3374 {
3375 	s1394_hal_t	*hal;
3376 	int		ret;
3377 
3378 	TNF_PROBE_0_DEBUG(t1394_initiate_bus_reset_enter,
3379 	    S1394_TNF_SL_BR_STACK, "");
3380 
3381 	ASSERT(t1394_hdl != NULL);
3382 
3383 	/* Find the HAL this target resides on */
3384 	hal = ((s1394_target_t *)t1394_hdl)->on_hal;
3385 
3386 	/* Reset the bus */
3387 	if (hal->halinfo.phy == H1394_PHY_1394A) {
3388 		ret = HAL_CALL(hal).short_bus_reset(hal->halinfo.hal_private);
3389 		if (ret != DDI_SUCCESS) {
3390 			TNF_PROBE_1(t1394_initiate_bus_reset_error,
3391 			    S1394_TNF_SL_ERROR, "", tnf_string, msg,
3392 			    "Error initiating short bus reset");
3393 		}
3394 	} else {
3395 		ret = HAL_CALL(hal).bus_reset(hal->halinfo.hal_private);
3396 		if (ret != DDI_SUCCESS) {
3397 			TNF_PROBE_1(t1394_initiate_bus_reset_error,
3398 			    S1394_TNF_SL_ERROR, "", tnf_string, msg,
3399 			    "Error initiating bus reset");
3400 		}
3401 	}
3402 
3403 	TNF_PROBE_0_DEBUG(t1394_initiate_bus_reset_exit,
3404 	    S1394_TNF_SL_BR_STACK, "");
3405 }
3406 
3407 /*
3408  * Function:    t1394_get_topology_map()
3409  * Input(s):    t1394_hdl		The target "handle" returned by
3410  *					    t1394_attach()
3411  *		bus_generation		The current generation
3412  *		tm_length		The size of the tm_buffer given
3413  *		flags			The flags parameter is unused (for now)
3414  *
3415  * Output(s):	tm_buffer		Filled in by the 1394 Software Framework
3416  *					    with the contents of the local
3417  *					    TOPOLOGY_MAP
3418  *
3419  * Description:	t1394_get_topology_map() returns the 1394 TOPLOGY_MAP.  See
3420  *		IEEE 1394-1995 Section 8.2.3.4.1 for format information.  This
3421  *		call can fail if there is a generation mismatch or the
3422  *		tm_buffer is too small to hold the TOPOLOGY_MAP.
3423  */
3424 /* ARGSUSED */
3425 int
3426 t1394_get_topology_map(t1394_handle_t t1394_hdl, uint_t bus_generation,
3427     size_t tm_length, uint_t flags, uint32_t *tm_buffer)
3428 {
3429 	s1394_hal_t	*hal;
3430 	uint32_t	*tm_ptr;
3431 	uint_t		length;
3432 
3433 	TNF_PROBE_0_DEBUG(t1394_get_topology_map_enter, S1394_TNF_SL_CSR_STACK,
3434 	    "");
3435 
3436 	ASSERT(t1394_hdl != NULL);
3437 
3438 	/* Find the HAL this target resides on */
3439 	hal = ((s1394_target_t *)t1394_hdl)->on_hal;
3440 
3441 	/* Lock the topology tree */
3442 	mutex_enter(&hal->topology_tree_mutex);
3443 
3444 	/* Check the bus_generation for the Topology Map */
3445 	if (bus_generation != hal->generation_count) {
3446 		/* Unlock the topology tree */
3447 		mutex_exit(&hal->topology_tree_mutex);
3448 		TNF_PROBE_1(t1394_get_topology_map_error,
3449 		    S1394_TNF_SL_CSR_ERROR, "", tnf_string, msg,
3450 		    "Generation mismatch");
3451 		TNF_PROBE_0_DEBUG(t1394_get_topology_map_exit,
3452 		    S1394_TNF_SL_CSR_STACK, "");
3453 		return (DDI_FAILURE);
3454 	}
3455 
3456 	tm_ptr	= (uint32_t *)hal->CSR_topology_map;
3457 	length	= tm_ptr[0] >> 16;
3458 	length  = length * 4;	/* Bytes instead of quadlets   */
3459 	length  = length + 4;   /* don't forget the first quad */
3460 
3461 	/* Check that the buffer is big enough */
3462 	if (length > (uint_t)tm_length) {
3463 		/* Unlock the topology tree */
3464 		mutex_exit(&hal->topology_tree_mutex);
3465 		TNF_PROBE_1(t1394_get_topology_map_error,
3466 		    S1394_TNF_SL_CSR_ERROR, "", tnf_string, msg,
3467 		    "Buffer size too small");
3468 		TNF_PROBE_0_DEBUG(t1394_get_topology_map_exit,
3469 		    S1394_TNF_SL_CSR_STACK, "");
3470 		return (DDI_FAILURE);
3471 	}
3472 
3473 	/* Do the copy */
3474 	bcopy(tm_ptr, tm_buffer, length);
3475 
3476 	/* Unlock the topology tree */
3477 	mutex_exit(&hal->topology_tree_mutex);
3478 	TNF_PROBE_0_DEBUG(t1394_get_topology_map_exit, S1394_TNF_SL_CSR_STACK,
3479 	    "");
3480 	return (DDI_SUCCESS);
3481 }
3482 
3483 /*
3484  * Function:    t1394_CRC16()
3485  * Input(s):    d			The data to compute the CRC-16 for
3486  *		crc_length		The length into the data to compute for
3487  *		flags			The flags parameter is unused (for now)
3488  *
3489  * Output(s):	CRC			The CRC-16 computed for the length
3490  *					    of data specified
3491  *
3492  * Description:	t1394_CRC16() implements ISO/IEC 13213:1994, ANSI/IEEE Std
3493  *		1212, 1994 - 8.1.5.
3494  */
3495 /* ARGSUSED */
3496 uint_t
3497 t1394_CRC16(uint32_t *d, size_t crc_length, uint_t flags)
3498 {
3499 	/* Implements ISO/IEC 13213:1994,	*/
3500 	/* ANSI/IEEE Std 1212, 1994 - 8.1.5	*/
3501 	uint_t	ret;
3502 
3503 	TNF_PROBE_0_DEBUG(t1394_CRC16_enter, S1394_TNF_SL_STACK, "");
3504 
3505 	ret = s1394_CRC16((uint_t *)d, (uint_t)crc_length);
3506 
3507 	TNF_PROBE_0_DEBUG(t1394_CRC16_exit, S1394_TNF_SL_STACK, "");
3508 	return (ret);
3509 }
3510 
3511 /*
3512  * Function:    t1394_add_cfgrom_entry()
3513  * Input(s):    t1394_hdl		The target "handle" returned by
3514  *					    t1394_attach()
3515  *		cfgrom_entryinfo	This structure holds the cfgrom key,
3516  *					    buffer, and size
3517  *		flags			The flags parameter is unused (for now)
3518  *
3519  * Output(s):	t1394_cfgrom_hdl	The ConfigROM "handle" used in
3520  *					    t1394_rem_cfgrom_entry()
3521  *		result			Used to pass more specific info back
3522  *					    to target
3523  *
3524  * Description:	t1394_add_cfgrom_entry() adds an entry to the local Config ROM,
3525  *		updating the directory entries as necessary.  This call could
3526  *		fail because there is no room for the new entry in Config ROM
3527  *		(T1394_ECFGROM_FULL), the key is invalid (T1394_EINVALID_PARAM),
3528  *		or it was called in interrupt context (T1394_EINVALID_CONTEXT).
3529  */
3530 /* ARGSUSED */
3531 int
3532 t1394_add_cfgrom_entry(t1394_handle_t t1394_hdl,
3533     t1394_cfgrom_entryinfo_t *cfgrom_entryinfo, uint_t flags,
3534     t1394_cfgrom_handle_t *t1394_cfgrom_hdl, int *result)
3535 {
3536 	s1394_hal_t	*hal;
3537 	s1394_target_t	*target;
3538 	int		ret;
3539 	uint_t		key;
3540 	uint_t		size;
3541 	uint32_t	*buffer;
3542 
3543 	TNF_PROBE_0_DEBUG(t1394_add_cfgrom_entry_enter,
3544 	    S1394_TNF_SL_CFGROM_STACK, "");
3545 
3546 	ASSERT(t1394_hdl != NULL);
3547 
3548 	target = (s1394_target_t *)t1394_hdl;
3549 
3550 	key = cfgrom_entryinfo->ce_key;
3551 	buffer = cfgrom_entryinfo->ce_buffer;
3552 	size = (uint_t)cfgrom_entryinfo->ce_size;
3553 
3554 	/* Check for a valid size */
3555 	if (size == 0) {
3556 		*result = T1394_EINVALID_PARAM;
3557 		TNF_PROBE_1_DEBUG(t1394_add_cfgrom_entry_error,
3558 		    S1394_TNF_SL_CFGROM_ERROR, "", tnf_string, msg,
3559 		    "Invalid size of Config ROM buffer (== 0)");
3560 		TNF_PROBE_0_DEBUG(t1394_add_cfgrom_entry_exit,
3561 		    S1394_TNF_SL_CFGROM_STACK, "");
3562 		return (DDI_FAILURE);
3563 	}
3564 
3565 	/* Check for a valid key type */
3566 	if (((key << IEEE1212_KEY_VALUE_SHIFT) & IEEE1212_KEY_TYPE_MASK) == 0) {
3567 		*result = T1394_EINVALID_PARAM;
3568 		TNF_PROBE_1_DEBUG(t1394_add_cfgrom_entry_error,
3569 		    S1394_TNF_SL_CFGROM_ERROR, "", tnf_string, msg,
3570 		    "Invalid key_type in Config ROM key");
3571 		TNF_PROBE_0_DEBUG(t1394_add_cfgrom_entry_exit,
3572 		    S1394_TNF_SL_CFGROM_STACK, "");
3573 		return (DDI_FAILURE);
3574 	}
3575 
3576 	/* Find the HAL this target resides on */
3577 	hal = target->on_hal;
3578 
3579 	/* Is this on the interrupt stack? */
3580 	if (servicing_interrupt()) {
3581 		*result = T1394_EINVALID_CONTEXT;
3582 		TNF_PROBE_0_DEBUG(t1394_add_cfgrom_entry_exit,
3583 		    S1394_TNF_SL_CFGROM_STACK, "");
3584 		return (DDI_FAILURE);
3585 	}
3586 
3587 	/* Lock the Config ROM buffer */
3588 	mutex_enter(&hal->local_config_rom_mutex);
3589 
3590 	ret = s1394_add_config_rom_entry(hal, key, buffer, size,
3591 	    (void **)t1394_cfgrom_hdl, result);
3592 	if (ret != DDI_SUCCESS) {
3593 		if (*result == CMD1394_ERSRC_CONFLICT)
3594 			*result = T1394_ECFGROM_FULL;
3595 		mutex_exit(&hal->local_config_rom_mutex);
3596 
3597 		TNF_PROBE_1(t1394_add_cfgrom_entry_error,
3598 		    S1394_TNF_SL_CFGROM_ERROR, "", tnf_string, msg,
3599 		    "Failed in s1394_add_cfgrom_entry()");
3600 		TNF_PROBE_0_DEBUG(t1394_add_cfgrom_entry_exit,
3601 		    "stacktrace 1394 s1394", "");
3602 		return (ret);
3603 	}
3604 
3605 	/* Setup the timeout function */
3606 	if (hal->config_rom_timer_set == B_FALSE) {
3607 		hal->config_rom_timer_set = B_TRUE;
3608 		mutex_exit(&hal->local_config_rom_mutex);
3609 		hal->config_rom_timer =
3610 		    timeout(s1394_update_config_rom_callback, hal,
3611 			drv_usectohz(CONFIG_ROM_UPDATE_DELAY * 1000));
3612 	} else {
3613 		mutex_exit(&hal->local_config_rom_mutex);
3614 	}
3615 
3616 	TNF_PROBE_0_DEBUG(t1394_add_cfgrom_entry_exit,
3617 	    S1394_TNF_SL_CFGROM_STACK, "");
3618 	return (ret);
3619 }
3620 
3621 /*
3622  * Function:    t1394_rem_cfgrom_entry()
3623  * Input(s):    t1394_hdl		The target "handle" returned by
3624  *					    t1394_attach()
3625  *		flags			The flags parameter is unused (for now)
3626  *		t1394_cfgrom_hdl	The ConfigROM "handle" returned by
3627  *					    t1394_add_cfgrom_entry()
3628  *
3629  * Output(s):	result			Used to pass more specific info back
3630  *					    to target
3631  *
3632  * Description:	t1394_rem_cfgrom_entry() is used to remove a previously added
3633  *		Config ROM entry (indicated by t1394_cfgrom_hdl).
3634  */
3635 /* ARGSUSED */
3636 int
3637 t1394_rem_cfgrom_entry(t1394_handle_t t1394_hdl, uint_t flags,
3638     t1394_cfgrom_handle_t *t1394_cfgrom_hdl, int *result)
3639 {
3640 	s1394_hal_t	*hal;
3641 	s1394_target_t	*target;
3642 	int		ret;
3643 
3644 	TNF_PROBE_0_DEBUG(t1394_rem_cfgrom_entry_enter,
3645 	    S1394_TNF_SL_CFGROM_STACK, "");
3646 
3647 	ASSERT(t1394_hdl != NULL);
3648 
3649 	target = (s1394_target_t *)t1394_hdl;
3650 
3651 	/* Find the HAL this target resides on */
3652 	hal = target->on_hal;
3653 
3654 	/* Is this on the interrupt stack? */
3655 	if (servicing_interrupt()) {
3656 		*result = T1394_EINVALID_CONTEXT;
3657 		TNF_PROBE_0_DEBUG(t1394_rem_cfgrom_entry_exit,
3658 		    S1394_TNF_SL_CFGROM_STACK, "");
3659 		return (DDI_FAILURE);
3660 	}
3661 
3662 	/* Lock the Config ROM buffer */
3663 	mutex_enter(&hal->local_config_rom_mutex);
3664 
3665 	ret = s1394_remove_config_rom_entry(hal, (void **)t1394_cfgrom_hdl,
3666 	    result);
3667 	if (ret != DDI_SUCCESS) {
3668 		mutex_exit(&hal->local_config_rom_mutex);
3669 		TNF_PROBE_1(t1394_rem_cfgrom_entry_error,
3670 		    S1394_TNF_SL_CFGROM_ERROR, "", tnf_string, msg,
3671 		    "Failed in s1394_remove_cfgrom_entry()");
3672 		TNF_PROBE_0_DEBUG(t1394_rem_cfgrom_entry_exit,
3673 		    "stacktrace 1394 s1394", "");
3674 		return (ret);
3675 	}
3676 
3677 	/* Setup the timeout function */
3678 	if (hal->config_rom_timer_set == B_FALSE) {
3679 		hal->config_rom_timer_set = B_TRUE;
3680 		mutex_exit(&hal->local_config_rom_mutex);
3681 		hal->config_rom_timer =
3682 		    timeout(s1394_update_config_rom_callback, hal,
3683 			drv_usectohz(CONFIG_ROM_UPDATE_DELAY * 1000));
3684 	} else {
3685 		mutex_exit(&hal->local_config_rom_mutex);
3686 	}
3687 
3688 	TNF_PROBE_0_DEBUG(t1394_rem_cfgrom_entry_exit,
3689 	    S1394_TNF_SL_CFGROM_STACK, "");
3690 	return (ret);
3691 }
3692 
3693 /*
3694  * Function:    t1394_get_targetinfo()
3695  * Input(s):    t1394_hdl		The target "handle" returned by
3696  *					    t1394_attach()
3697  *		bus_generation		The current generation
3698  *		flags			The flags parameter is unused (for now)
3699  *
3700  * Output(s):	targetinfo		Structure containing max_payload,
3701  *					    max_speed, and target node ID.
3702  *
3703  * Description:	t1394_get_targetinfo() is used to retrieve information specific
3704  *		to a target device.  It will fail if the generation given
3705  *		does not match the current generation.
3706  */
3707 /* ARGSUSED */
3708 int
3709 t1394_get_targetinfo(t1394_handle_t t1394_hdl, uint_t bus_generation,
3710     uint_t flags, t1394_targetinfo_t *targetinfo)
3711 {
3712 	s1394_hal_t	*hal;
3713 	s1394_target_t	*target;
3714 	uint_t		dev;
3715 	uint_t		curr;
3716 	uint_t		from_node;
3717 	uint_t		to_node;
3718 
3719 	TNF_PROBE_0_DEBUG(t1394_get_targetinfo_enter, S1394_TNF_SL_STACK, "");
3720 
3721 	ASSERT(t1394_hdl != NULL);
3722 
3723 	/* Find the HAL this target resides on */
3724 	hal = ((s1394_target_t *)t1394_hdl)->on_hal;
3725 
3726 	target = (s1394_target_t *)t1394_hdl;
3727 
3728 	/* Lock the topology tree */
3729 	mutex_enter(&hal->topology_tree_mutex);
3730 
3731 	/* Check the bus_generation */
3732 	if (bus_generation != hal->generation_count) {
3733 		/* Unlock the topology tree */
3734 		mutex_exit(&hal->topology_tree_mutex);
3735 		TNF_PROBE_3(t1394_get_targetinfo_error, S1394_TNF_SL_STACK, "",
3736 		    tnf_string, msg, "Generation mismatch",
3737 		    tnf_uint, gen, bus_generation,
3738 		    tnf_uint, current_gen, hal->generation_count);
3739 		return (DDI_FAILURE);
3740 	}
3741 
3742 	rw_enter(&hal->target_list_rwlock, RW_READER);
3743 	/*
3744 	 * If there is no node, report T1394_INVALID_NODEID for target_nodeID;
3745 	 * current_max_speed and current_max_payload are undefined for this
3746 	 * case.
3747 	 */
3748 	if (((target->target_state & S1394_TARG_GONE) != 0) ||
3749 	    (target->on_node == NULL)) {
3750 		targetinfo->target_nodeID = T1394_INVALID_NODEID;
3751 		TNF_PROBE_1_DEBUG(t1394_get_targetinfo_exit,
3752 		    S1394_TNF_SL_STACK, "", tnf_string, msg, "No device");
3753 	} else {
3754 		targetinfo->target_nodeID =
3755 		    (target->on_hal->node_id & IEEE1394_BUS_NUM_MASK) |
3756 		    target->on_node->node_num;
3757 
3758 		from_node = (target->on_hal->node_id) & IEEE1394_NODE_NUM_MASK;
3759 		to_node = target->on_node->node_num;
3760 
3761 		targetinfo->current_max_speed = (uint_t)s1394_speed_map_get(
3762 		    hal, from_node, to_node);
3763 
3764 		/* Get current_max_payload */
3765 		s1394_get_maxpayload(target, &dev, &curr);
3766 		targetinfo->current_max_payload	= curr;
3767 
3768 		TNF_PROBE_3_DEBUG(t1394_get_targetinfo_exit,
3769 		    S1394_TNF_SL_STACK, "",
3770 		    tnf_uint, payload, targetinfo->current_max_payload,
3771 		    tnf_uint, speed, targetinfo->current_max_speed,
3772 		    tnf_uint, nodeid, targetinfo->target_nodeID);
3773 	}
3774 
3775 	rw_exit(&hal->target_list_rwlock);
3776 	/* Unlock the topology tree */
3777 	mutex_exit(&hal->topology_tree_mutex);
3778 	return (DDI_SUCCESS);
3779 }
3780