xref: /titanic_51/usr/src/uts/common/io/sundlpi.c (revision 7c478bd95313f5f23a4c958a745db2134aa03244)
1*7c478bd9Sstevel@tonic-gate /*
2*7c478bd9Sstevel@tonic-gate  * CDDL HEADER START
3*7c478bd9Sstevel@tonic-gate  *
4*7c478bd9Sstevel@tonic-gate  * The contents of this file are subject to the terms of the
5*7c478bd9Sstevel@tonic-gate  * Common Development and Distribution License, Version 1.0 only
6*7c478bd9Sstevel@tonic-gate  * (the "License").  You may not use this file except in compliance
7*7c478bd9Sstevel@tonic-gate  * with the License.
8*7c478bd9Sstevel@tonic-gate  *
9*7c478bd9Sstevel@tonic-gate  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10*7c478bd9Sstevel@tonic-gate  * or http://www.opensolaris.org/os/licensing.
11*7c478bd9Sstevel@tonic-gate  * See the License for the specific language governing permissions
12*7c478bd9Sstevel@tonic-gate  * and limitations under the License.
13*7c478bd9Sstevel@tonic-gate  *
14*7c478bd9Sstevel@tonic-gate  * When distributing Covered Code, include this CDDL HEADER in each
15*7c478bd9Sstevel@tonic-gate  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16*7c478bd9Sstevel@tonic-gate  * If applicable, add the following below this CDDL HEADER, with the
17*7c478bd9Sstevel@tonic-gate  * fields enclosed by brackets "[]" replaced with your own identifying
18*7c478bd9Sstevel@tonic-gate  * information: Portions Copyright [yyyy] [name of copyright owner]
19*7c478bd9Sstevel@tonic-gate  *
20*7c478bd9Sstevel@tonic-gate  * CDDL HEADER END
21*7c478bd9Sstevel@tonic-gate  */
22*7c478bd9Sstevel@tonic-gate /*
23*7c478bd9Sstevel@tonic-gate  * Copyright 2005 Sun Microsystems, Inc.  All rights reserved.
24*7c478bd9Sstevel@tonic-gate  * Use is subject to license terms.
25*7c478bd9Sstevel@tonic-gate  */
26*7c478bd9Sstevel@tonic-gate 
27*7c478bd9Sstevel@tonic-gate #pragma ident	"%Z%%M%	%I%	%E% SMI"
28*7c478bd9Sstevel@tonic-gate 
29*7c478bd9Sstevel@tonic-gate /*
30*7c478bd9Sstevel@tonic-gate  *  Common Sun DLPI routines.
31*7c478bd9Sstevel@tonic-gate  */
32*7c478bd9Sstevel@tonic-gate 
33*7c478bd9Sstevel@tonic-gate #include	<sys/types.h>
34*7c478bd9Sstevel@tonic-gate #include	<sys/sysmacros.h>
35*7c478bd9Sstevel@tonic-gate #include	<sys/byteorder.h>
36*7c478bd9Sstevel@tonic-gate #include	<sys/systm.h>
37*7c478bd9Sstevel@tonic-gate #include	<sys/stream.h>
38*7c478bd9Sstevel@tonic-gate #include	<sys/strsun.h>
39*7c478bd9Sstevel@tonic-gate #include	<sys/dlpi.h>
40*7c478bd9Sstevel@tonic-gate 
41*7c478bd9Sstevel@tonic-gate #define		DLADDRL		(80)
42*7c478bd9Sstevel@tonic-gate 
43*7c478bd9Sstevel@tonic-gate void
44*7c478bd9Sstevel@tonic-gate dlbindack(
45*7c478bd9Sstevel@tonic-gate 	queue_t		*wq,
46*7c478bd9Sstevel@tonic-gate 	mblk_t		*mp,
47*7c478bd9Sstevel@tonic-gate 	t_scalar_t	sap,
48*7c478bd9Sstevel@tonic-gate 	void		*addrp,
49*7c478bd9Sstevel@tonic-gate 	t_uscalar_t	addrlen,
50*7c478bd9Sstevel@tonic-gate 	t_uscalar_t	maxconind,
51*7c478bd9Sstevel@tonic-gate 	t_uscalar_t	xidtest)
52*7c478bd9Sstevel@tonic-gate {
53*7c478bd9Sstevel@tonic-gate 	union DL_primitives	*dlp;
54*7c478bd9Sstevel@tonic-gate 	size_t			size;
55*7c478bd9Sstevel@tonic-gate 
56*7c478bd9Sstevel@tonic-gate 	size = sizeof (dl_bind_ack_t) + addrlen;
57*7c478bd9Sstevel@tonic-gate 	if ((mp = mexchange(wq, mp, size, M_PCPROTO, DL_BIND_ACK)) == NULL)
58*7c478bd9Sstevel@tonic-gate 		return;
59*7c478bd9Sstevel@tonic-gate 
60*7c478bd9Sstevel@tonic-gate 	dlp = (union DL_primitives *)mp->b_rptr;
61*7c478bd9Sstevel@tonic-gate 	dlp->bind_ack.dl_sap = sap;
62*7c478bd9Sstevel@tonic-gate 	dlp->bind_ack.dl_addr_length = addrlen;
63*7c478bd9Sstevel@tonic-gate 	dlp->bind_ack.dl_addr_offset = sizeof (dl_bind_ack_t);
64*7c478bd9Sstevel@tonic-gate 	dlp->bind_ack.dl_max_conind = maxconind;
65*7c478bd9Sstevel@tonic-gate 	dlp->bind_ack.dl_xidtest_flg = xidtest;
66*7c478bd9Sstevel@tonic-gate 	if (addrlen != 0)
67*7c478bd9Sstevel@tonic-gate 		bcopy(addrp, mp->b_rptr + sizeof (dl_bind_ack_t), addrlen);
68*7c478bd9Sstevel@tonic-gate 
69*7c478bd9Sstevel@tonic-gate 	qreply(wq, mp);
70*7c478bd9Sstevel@tonic-gate }
71*7c478bd9Sstevel@tonic-gate 
72*7c478bd9Sstevel@tonic-gate void
73*7c478bd9Sstevel@tonic-gate dlokack(
74*7c478bd9Sstevel@tonic-gate 	queue_t		*wq,
75*7c478bd9Sstevel@tonic-gate 	mblk_t		*mp,
76*7c478bd9Sstevel@tonic-gate 	t_uscalar_t	correct_primitive)
77*7c478bd9Sstevel@tonic-gate {
78*7c478bd9Sstevel@tonic-gate 	union DL_primitives	*dlp;
79*7c478bd9Sstevel@tonic-gate 
80*7c478bd9Sstevel@tonic-gate 	if ((mp = mexchange(wq, mp, sizeof (dl_ok_ack_t), M_PCPROTO,
81*7c478bd9Sstevel@tonic-gate 	    DL_OK_ACK)) == NULL)
82*7c478bd9Sstevel@tonic-gate 		return;
83*7c478bd9Sstevel@tonic-gate 	dlp = (union DL_primitives *)mp->b_rptr;
84*7c478bd9Sstevel@tonic-gate 	dlp->ok_ack.dl_correct_primitive = correct_primitive;
85*7c478bd9Sstevel@tonic-gate 	qreply(wq, mp);
86*7c478bd9Sstevel@tonic-gate }
87*7c478bd9Sstevel@tonic-gate 
88*7c478bd9Sstevel@tonic-gate void
89*7c478bd9Sstevel@tonic-gate dlerrorack(
90*7c478bd9Sstevel@tonic-gate 	queue_t		*wq,
91*7c478bd9Sstevel@tonic-gate 	mblk_t		*mp,
92*7c478bd9Sstevel@tonic-gate 	t_uscalar_t	error_primitive,
93*7c478bd9Sstevel@tonic-gate 	t_uscalar_t	error,
94*7c478bd9Sstevel@tonic-gate 	t_uscalar_t	unix_errno)
95*7c478bd9Sstevel@tonic-gate {
96*7c478bd9Sstevel@tonic-gate 	union DL_primitives	*dlp;
97*7c478bd9Sstevel@tonic-gate 
98*7c478bd9Sstevel@tonic-gate 	if ((mp = mexchange(wq, mp, sizeof (dl_error_ack_t), M_PCPROTO,
99*7c478bd9Sstevel@tonic-gate 	    DL_ERROR_ACK)) == NULL)
100*7c478bd9Sstevel@tonic-gate 		return;
101*7c478bd9Sstevel@tonic-gate 	dlp = (union DL_primitives *)mp->b_rptr;
102*7c478bd9Sstevel@tonic-gate 	dlp->error_ack.dl_error_primitive = error_primitive;
103*7c478bd9Sstevel@tonic-gate 	dlp->error_ack.dl_errno = error;
104*7c478bd9Sstevel@tonic-gate 	dlp->error_ack.dl_unix_errno = unix_errno;
105*7c478bd9Sstevel@tonic-gate 	qreply(wq, mp);
106*7c478bd9Sstevel@tonic-gate }
107*7c478bd9Sstevel@tonic-gate 
108*7c478bd9Sstevel@tonic-gate void
109*7c478bd9Sstevel@tonic-gate dluderrorind(
110*7c478bd9Sstevel@tonic-gate 	queue_t		*wq,
111*7c478bd9Sstevel@tonic-gate 	mblk_t		*mp,
112*7c478bd9Sstevel@tonic-gate 	void		*addrp,
113*7c478bd9Sstevel@tonic-gate 	t_uscalar_t	addrlen,
114*7c478bd9Sstevel@tonic-gate 	t_uscalar_t	error,
115*7c478bd9Sstevel@tonic-gate 	t_uscalar_t	unix_errno)
116*7c478bd9Sstevel@tonic-gate {
117*7c478bd9Sstevel@tonic-gate 	union DL_primitives	*dlp;
118*7c478bd9Sstevel@tonic-gate 	char			buf[DLADDRL];
119*7c478bd9Sstevel@tonic-gate 	size_t			size;
120*7c478bd9Sstevel@tonic-gate 
121*7c478bd9Sstevel@tonic-gate 	if (addrlen > DLADDRL)
122*7c478bd9Sstevel@tonic-gate 		addrlen = DLADDRL;
123*7c478bd9Sstevel@tonic-gate 
124*7c478bd9Sstevel@tonic-gate 	bcopy(addrp, buf, addrlen);
125*7c478bd9Sstevel@tonic-gate 
126*7c478bd9Sstevel@tonic-gate 	size = sizeof (dl_uderror_ind_t) + addrlen;
127*7c478bd9Sstevel@tonic-gate 
128*7c478bd9Sstevel@tonic-gate 	if ((mp = mexchange(wq, mp, size, M_PCPROTO, DL_UDERROR_IND)) == NULL)
129*7c478bd9Sstevel@tonic-gate 		return;
130*7c478bd9Sstevel@tonic-gate 
131*7c478bd9Sstevel@tonic-gate 	dlp = (union DL_primitives *)mp->b_rptr;
132*7c478bd9Sstevel@tonic-gate 	dlp->uderror_ind.dl_dest_addr_length = addrlen;
133*7c478bd9Sstevel@tonic-gate 	dlp->uderror_ind.dl_dest_addr_offset = sizeof (dl_uderror_ind_t);
134*7c478bd9Sstevel@tonic-gate 	dlp->uderror_ind.dl_unix_errno = unix_errno;
135*7c478bd9Sstevel@tonic-gate 	dlp->uderror_ind.dl_errno = error;
136*7c478bd9Sstevel@tonic-gate 	bcopy((caddr_t)buf,
137*7c478bd9Sstevel@tonic-gate 		(caddr_t)(mp->b_rptr + sizeof (dl_uderror_ind_t)), addrlen);
138*7c478bd9Sstevel@tonic-gate 	qreply(wq, mp);
139*7c478bd9Sstevel@tonic-gate }
140*7c478bd9Sstevel@tonic-gate 
141*7c478bd9Sstevel@tonic-gate void
142*7c478bd9Sstevel@tonic-gate dlphysaddrack(
143*7c478bd9Sstevel@tonic-gate 	queue_t		*wq,
144*7c478bd9Sstevel@tonic-gate 	mblk_t		*mp,
145*7c478bd9Sstevel@tonic-gate 	void		*addrp,
146*7c478bd9Sstevel@tonic-gate 	t_uscalar_t	len)
147*7c478bd9Sstevel@tonic-gate {
148*7c478bd9Sstevel@tonic-gate 	union DL_primitives	*dlp;
149*7c478bd9Sstevel@tonic-gate 	size_t			size;
150*7c478bd9Sstevel@tonic-gate 
151*7c478bd9Sstevel@tonic-gate 	size = sizeof (dl_phys_addr_ack_t) + len;
152*7c478bd9Sstevel@tonic-gate 	if ((mp = mexchange(wq, mp, size, M_PCPROTO, DL_PHYS_ADDR_ACK)) == NULL)
153*7c478bd9Sstevel@tonic-gate 		return;
154*7c478bd9Sstevel@tonic-gate 	dlp = (union DL_primitives *)mp->b_rptr;
155*7c478bd9Sstevel@tonic-gate 	dlp->physaddr_ack.dl_addr_length = len;
156*7c478bd9Sstevel@tonic-gate 	dlp->physaddr_ack.dl_addr_offset = sizeof (dl_phys_addr_ack_t);
157*7c478bd9Sstevel@tonic-gate 	if (len != 0)
158*7c478bd9Sstevel@tonic-gate 		bcopy(addrp, mp->b_rptr + sizeof (dl_phys_addr_ack_t), len);
159*7c478bd9Sstevel@tonic-gate 	qreply(wq, mp);
160*7c478bd9Sstevel@tonic-gate }
161*7c478bd9Sstevel@tonic-gate 
162*7c478bd9Sstevel@tonic-gate void
163*7c478bd9Sstevel@tonic-gate dlcapabsetqid(dl_mid_t *idp, const queue_t *q)
164*7c478bd9Sstevel@tonic-gate {
165*7c478bd9Sstevel@tonic-gate #ifndef _LP64
166*7c478bd9Sstevel@tonic-gate 	idp->mid[0] = (t_uscalar_t)q;
167*7c478bd9Sstevel@tonic-gate #else
168*7c478bd9Sstevel@tonic-gate 	idp->mid[0] = (t_uscalar_t)BMASK_32((uint64_t)q);
169*7c478bd9Sstevel@tonic-gate 	idp->mid[1] = (t_uscalar_t)BMASK_32(((uint64_t)q) >> 32);
170*7c478bd9Sstevel@tonic-gate #endif
171*7c478bd9Sstevel@tonic-gate }
172*7c478bd9Sstevel@tonic-gate 
173*7c478bd9Sstevel@tonic-gate boolean_t
174*7c478bd9Sstevel@tonic-gate dlcapabcheckqid(const dl_mid_t *idp, const queue_t *q)
175*7c478bd9Sstevel@tonic-gate {
176*7c478bd9Sstevel@tonic-gate #ifndef _LP64
177*7c478bd9Sstevel@tonic-gate 	return ((queue_t *)(idp->mid[0]) == q);
178*7c478bd9Sstevel@tonic-gate #else
179*7c478bd9Sstevel@tonic-gate 	return ((queue_t *)
180*7c478bd9Sstevel@tonic-gate 	    ((uint64_t)idp->mid[0] | ((uint64_t)idp->mid[1] << 32)) == q);
181*7c478bd9Sstevel@tonic-gate #endif
182*7c478bd9Sstevel@tonic-gate }
183*7c478bd9Sstevel@tonic-gate 
184*7c478bd9Sstevel@tonic-gate void
185*7c478bd9Sstevel@tonic-gate dlnotifyack(
186*7c478bd9Sstevel@tonic-gate 	queue_t		*wq,
187*7c478bd9Sstevel@tonic-gate 	mblk_t		*mp,
188*7c478bd9Sstevel@tonic-gate 	uint32_t	notifications)
189*7c478bd9Sstevel@tonic-gate {
190*7c478bd9Sstevel@tonic-gate 	union DL_primitives	*dlp;
191*7c478bd9Sstevel@tonic-gate 
192*7c478bd9Sstevel@tonic-gate 	if ((mp = mexchange(wq, mp, sizeof (dl_ok_ack_t), M_PROTO,
193*7c478bd9Sstevel@tonic-gate 	    DL_NOTIFY_ACK)) == NULL)
194*7c478bd9Sstevel@tonic-gate 		return;
195*7c478bd9Sstevel@tonic-gate 	dlp = (union DL_primitives *)mp->b_rptr;
196*7c478bd9Sstevel@tonic-gate 	dlp->notify_ack.dl_notifications = notifications;
197*7c478bd9Sstevel@tonic-gate 	qreply(wq, mp);
198*7c478bd9Sstevel@tonic-gate }
199