xref: /onnv-gate/usr/src/uts/sparc/os/iscsi_boot.c (revision 10822:2a6b5dc1374c)
18194SJack.Meng@Sun.COM /*
28194SJack.Meng@Sun.COM  * CDDL HEADER START
38194SJack.Meng@Sun.COM  *
48194SJack.Meng@Sun.COM  * The contents of this file are subject to the terms of the
58194SJack.Meng@Sun.COM  * Common Development and Distribution License (the "License").
68194SJack.Meng@Sun.COM  * You may not use this file except in compliance with the License.
78194SJack.Meng@Sun.COM  *
88194SJack.Meng@Sun.COM  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
98194SJack.Meng@Sun.COM  * or http://www.opensolaris.org/os/licensing.
108194SJack.Meng@Sun.COM  * See the License for the specific language governing permissions
118194SJack.Meng@Sun.COM  * and limitations under the License.
128194SJack.Meng@Sun.COM  *
138194SJack.Meng@Sun.COM  * When distributing Covered Code, include this CDDL HEADER in each
148194SJack.Meng@Sun.COM  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
158194SJack.Meng@Sun.COM  * If applicable, add the following below this CDDL HEADER, with the
168194SJack.Meng@Sun.COM  * fields enclosed by brackets "[]" replaced with your own identifying
178194SJack.Meng@Sun.COM  * information: Portions Copyright [yyyy] [name of copyright owner]
188194SJack.Meng@Sun.COM  *
198194SJack.Meng@Sun.COM  * CDDL HEADER END
208194SJack.Meng@Sun.COM  */
218194SJack.Meng@Sun.COM /*
22*10822SJack.Meng@Sun.COM  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
238194SJack.Meng@Sun.COM  * Use is subject to license terms.
248194SJack.Meng@Sun.COM  */
258194SJack.Meng@Sun.COM 
268194SJack.Meng@Sun.COM #include <sys/bootprops.h>
27*10822SJack.Meng@Sun.COM #include <sys/bootconf.h>
28*10822SJack.Meng@Sun.COM #include <sys/modctl.h>
29*10822SJack.Meng@Sun.COM #include <sys/mman.h>
30*10822SJack.Meng@Sun.COM #include <sys/kmem.h>
31*10822SJack.Meng@Sun.COM #include <sys/ddi.h>
32*10822SJack.Meng@Sun.COM #include <sys/sunddi.h>
33*10822SJack.Meng@Sun.COM #include <sys/types.h>
34*10822SJack.Meng@Sun.COM #include <sys/obpdefs.h>
35*10822SJack.Meng@Sun.COM #include <sys/promif.h>
36*10822SJack.Meng@Sun.COM #include <sys/iscsi_protocol.h>
378194SJack.Meng@Sun.COM 
38*10822SJack.Meng@Sun.COM #define	ISCSI_OBP_MAX_CHAP_USER_LEN	16
39*10822SJack.Meng@Sun.COM #define	ISCSI_OBP_MIN_CHAP_LEN		12
40*10822SJack.Meng@Sun.COM #define	ISCSI_OBP_MAX_CHAP_LEN		16
41*10822SJack.Meng@Sun.COM 
42*10822SJack.Meng@Sun.COM #define	OBP_GET_KEY_STATUS_OK		0
43*10822SJack.Meng@Sun.COM #define	OBP_GET_KEY_STATUS_NOT_EXIST	-3
44*10822SJack.Meng@Sun.COM 
45*10822SJack.Meng@Sun.COM ib_boot_prop_t boot_property;
468194SJack.Meng@Sun.COM extern ib_boot_prop_t *iscsiboot_prop;
47*10822SJack.Meng@Sun.COM static int inet_aton(char *ipstr, uchar_t *ip);
48*10822SJack.Meng@Sun.COM static boolean_t parse_lun_num(uchar_t *str_num, uchar_t *hex_num);
49*10822SJack.Meng@Sun.COM static void generate_iscsi_initiator_id(void);
508194SJack.Meng@Sun.COM 
51*10822SJack.Meng@Sun.COM static int
isdigit(int ch)52*10822SJack.Meng@Sun.COM isdigit(int ch)
53*10822SJack.Meng@Sun.COM {
54*10822SJack.Meng@Sun.COM 	return (ch >= '0' && ch <= '9');
55*10822SJack.Meng@Sun.COM }
56*10822SJack.Meng@Sun.COM 
57*10822SJack.Meng@Sun.COM static boolean_t
iscsiboot_tgt_prop_read(void)58*10822SJack.Meng@Sun.COM iscsiboot_tgt_prop_read(void)
59*10822SJack.Meng@Sun.COM {
60*10822SJack.Meng@Sun.COM 	int		proplen;
61*10822SJack.Meng@Sun.COM 	boolean_t	set		= B_FALSE;
62*10822SJack.Meng@Sun.COM 	char		iscsi_target_ip[INET6_ADDRSTRLEN];
63*10822SJack.Meng@Sun.COM 	uchar_t		iscsi_target_name[ISCSI_MAX_NAME_LEN];
64*10822SJack.Meng@Sun.COM 	uchar_t		iscsi_par[8];
65*10822SJack.Meng@Sun.COM 	char		chap_user[ISCSI_OBP_MAX_CHAP_USER_LEN]	= {0};
66*10822SJack.Meng@Sun.COM 	char		chap_password[ISCSI_OBP_MAX_CHAP_LEN]	= {0};
67*10822SJack.Meng@Sun.COM 	uchar_t		iscsi_port[8];
68*10822SJack.Meng@Sun.COM 	uchar_t		iscsi_lun[8];
69*10822SJack.Meng@Sun.COM 	uchar_t		iscsi_tpgt[8];
70*10822SJack.Meng@Sun.COM 	long		iscsi_tpgtl;
71*10822SJack.Meng@Sun.COM 	long		port;
72*10822SJack.Meng@Sun.COM 	int		ret		= 0;
73*10822SJack.Meng@Sun.COM 	int		status		= 0;
74*10822SJack.Meng@Sun.COM 	int		chap_user_len	= 0;
75*10822SJack.Meng@Sun.COM 	int		chap_pwd_len	= 0;
76*10822SJack.Meng@Sun.COM 
77*10822SJack.Meng@Sun.COM 	/* Get iscsi target IP address */
78*10822SJack.Meng@Sun.COM 	proplen = BOP_GETPROPLEN(bootops, BP_ISCSI_TARGET_IP);
79*10822SJack.Meng@Sun.COM 	if (proplen > 0) {
80*10822SJack.Meng@Sun.COM 		if (BOP_GETPROP(bootops, BP_ISCSI_TARGET_IP,
81*10822SJack.Meng@Sun.COM 		    iscsi_target_ip) > 0) {
82*10822SJack.Meng@Sun.COM 			if (inet_aton(iscsi_target_ip,
83*10822SJack.Meng@Sun.COM 			    (uchar_t *)&boot_property.boot_tgt.tgt_ip_u) ==
84*10822SJack.Meng@Sun.COM 			    0) {
85*10822SJack.Meng@Sun.COM 				boot_property.boot_tgt.sin_family = AF_INET;
86*10822SJack.Meng@Sun.COM 				set = B_TRUE;
87*10822SJack.Meng@Sun.COM 			}
88*10822SJack.Meng@Sun.COM 		}
89*10822SJack.Meng@Sun.COM 	}
90*10822SJack.Meng@Sun.COM 	if (set != B_TRUE) {
91*10822SJack.Meng@Sun.COM 		return (B_FALSE);
92*10822SJack.Meng@Sun.COM 	}
93*10822SJack.Meng@Sun.COM 
94*10822SJack.Meng@Sun.COM 	/* Get iscsi target port number */
95*10822SJack.Meng@Sun.COM 	set = B_FALSE;
96*10822SJack.Meng@Sun.COM 	proplen = BOP_GETPROPLEN(bootops, BP_ISCSI_PORT);
97*10822SJack.Meng@Sun.COM 	if (proplen > 0) {
98*10822SJack.Meng@Sun.COM 		if (BOP_GETPROP(bootops, BP_ISCSI_PORT,
99*10822SJack.Meng@Sun.COM 		    iscsi_port) > 0) {
100*10822SJack.Meng@Sun.COM 			if (ddi_strtol((const char *)iscsi_port, NULL,
101*10822SJack.Meng@Sun.COM 			    10, &port) == 0) {
102*10822SJack.Meng@Sun.COM 				boot_property.boot_tgt.tgt_port =
103*10822SJack.Meng@Sun.COM 				    (unsigned int)port;
104*10822SJack.Meng@Sun.COM 				set = B_TRUE;
105*10822SJack.Meng@Sun.COM 			}
106*10822SJack.Meng@Sun.COM 		}
107*10822SJack.Meng@Sun.COM 	}
108*10822SJack.Meng@Sun.COM 	if (set != B_TRUE) {
109*10822SJack.Meng@Sun.COM 		boot_property.boot_tgt.tgt_port = 3260;
110*10822SJack.Meng@Sun.COM 	}
111*10822SJack.Meng@Sun.COM 
112*10822SJack.Meng@Sun.COM 	/* Get iscsi target LUN number */
113*10822SJack.Meng@Sun.COM 	set = B_FALSE;
114*10822SJack.Meng@Sun.COM 	proplen = BOP_GETPROPLEN(bootops, BP_ISCSI_LUN);
115*10822SJack.Meng@Sun.COM 	if (proplen > 0) {
116*10822SJack.Meng@Sun.COM 		if (BOP_GETPROP(bootops, BP_ISCSI_LUN,
117*10822SJack.Meng@Sun.COM 		    iscsi_lun) > 0) {
118*10822SJack.Meng@Sun.COM 			if (parse_lun_num(iscsi_lun,
119*10822SJack.Meng@Sun.COM 			    (uchar_t *)
120*10822SJack.Meng@Sun.COM 			    (&boot_property.boot_tgt.tgt_boot_lun[0]))
121*10822SJack.Meng@Sun.COM 			    == B_TRUE) {
122*10822SJack.Meng@Sun.COM 				set = B_TRUE;
123*10822SJack.Meng@Sun.COM 			}
124*10822SJack.Meng@Sun.COM 		}
125*10822SJack.Meng@Sun.COM 	}
126*10822SJack.Meng@Sun.COM 	if (set != B_TRUE) {
127*10822SJack.Meng@Sun.COM 		bzero((void *)boot_property.boot_tgt.tgt_boot_lun, 8);
128*10822SJack.Meng@Sun.COM 	}
129*10822SJack.Meng@Sun.COM 
130*10822SJack.Meng@Sun.COM 	/* Get iscsi target portal group tag */
131*10822SJack.Meng@Sun.COM 	set = B_FALSE;
132*10822SJack.Meng@Sun.COM 	proplen = BOP_GETPROPLEN(bootops, BP_ISCSI_TPGT);
133*10822SJack.Meng@Sun.COM 	if (proplen > 0) {
134*10822SJack.Meng@Sun.COM 		if (BOP_GETPROP(bootops, BP_ISCSI_TPGT,
135*10822SJack.Meng@Sun.COM 		    iscsi_tpgt) > 0) {
136*10822SJack.Meng@Sun.COM 			if (ddi_strtol((const char *)iscsi_tpgt, NULL, 10,
137*10822SJack.Meng@Sun.COM 			    &iscsi_tpgtl) == 0) {
138*10822SJack.Meng@Sun.COM 				boot_property.boot_tgt.tgt_tpgt =
139*10822SJack.Meng@Sun.COM 				    (uint16_t)iscsi_tpgtl;
140*10822SJack.Meng@Sun.COM 				set = B_TRUE;
141*10822SJack.Meng@Sun.COM 			}
142*10822SJack.Meng@Sun.COM 		}
143*10822SJack.Meng@Sun.COM 	}
144*10822SJack.Meng@Sun.COM 	if (set != B_TRUE) {
145*10822SJack.Meng@Sun.COM 		boot_property.boot_tgt.tgt_tpgt = 1;
146*10822SJack.Meng@Sun.COM 	}
147*10822SJack.Meng@Sun.COM 
148*10822SJack.Meng@Sun.COM 	/* Get iscsi target node name */
149*10822SJack.Meng@Sun.COM 	set = B_FALSE;
150*10822SJack.Meng@Sun.COM 	boot_property.boot_tgt.tgt_name = NULL;
151*10822SJack.Meng@Sun.COM 	proplen = BOP_GETPROPLEN(bootops, BP_ISCSI_TARGET_NAME);
152*10822SJack.Meng@Sun.COM 	if (proplen > 0) {
153*10822SJack.Meng@Sun.COM 		if (BOP_GETPROP(bootops, BP_ISCSI_TARGET_NAME,
154*10822SJack.Meng@Sun.COM 		    iscsi_target_name) > 0) {
155*10822SJack.Meng@Sun.COM 			boot_property.boot_tgt.tgt_name =
156*10822SJack.Meng@Sun.COM 			    (uchar_t *)kmem_zalloc(proplen + 1, KM_SLEEP);
157*10822SJack.Meng@Sun.COM 			boot_property.boot_tgt.tgt_name_len = proplen + 1;
158*10822SJack.Meng@Sun.COM 			(void) snprintf((char *)boot_property.boot_tgt.tgt_name,
159*10822SJack.Meng@Sun.COM 			    proplen + 1, "%s", iscsi_target_name);
160*10822SJack.Meng@Sun.COM 			set = B_TRUE;
161*10822SJack.Meng@Sun.COM 		}
162*10822SJack.Meng@Sun.COM 	}
163*10822SJack.Meng@Sun.COM 	if (set != B_TRUE) {
164*10822SJack.Meng@Sun.COM 		if (boot_property.boot_tgt.tgt_name != NULL) {
165*10822SJack.Meng@Sun.COM 			kmem_free(boot_property.boot_tgt.tgt_name,
166*10822SJack.Meng@Sun.COM 			    boot_property.boot_tgt.tgt_name_len);
167*10822SJack.Meng@Sun.COM 			boot_property.boot_tgt.tgt_name = NULL;
168*10822SJack.Meng@Sun.COM 			boot_property.boot_tgt.tgt_name_len = 0;
169*10822SJack.Meng@Sun.COM 		}
170*10822SJack.Meng@Sun.COM 		return (B_FALSE);
171*10822SJack.Meng@Sun.COM 	}
172*10822SJack.Meng@Sun.COM 
173*10822SJack.Meng@Sun.COM 	/* Get iscsi target boot partition */
174*10822SJack.Meng@Sun.COM 	set = B_FALSE;
175*10822SJack.Meng@Sun.COM 	boot_property.boot_tgt.tgt_boot_par = NULL;
176*10822SJack.Meng@Sun.COM 	boot_property.boot_tgt.tgt_boot_par_len = 0;
177*10822SJack.Meng@Sun.COM 	proplen = BOP_GETPROPLEN(bootops, BP_ISCSI_PAR);
178*10822SJack.Meng@Sun.COM 	if (proplen > 0) {
179*10822SJack.Meng@Sun.COM 		if (BOP_GETPROP(bootops, BP_ISCSI_PAR, iscsi_par) > 0) {
180*10822SJack.Meng@Sun.COM 			boot_property.boot_tgt.tgt_boot_par =
181*10822SJack.Meng@Sun.COM 			    (uchar_t *)kmem_zalloc(proplen + 1, KM_SLEEP);
182*10822SJack.Meng@Sun.COM 			boot_property.boot_tgt.tgt_boot_par_len = proplen + 1;
183*10822SJack.Meng@Sun.COM 			(void) snprintf(
184*10822SJack.Meng@Sun.COM 			    (char *)boot_property.boot_tgt.tgt_boot_par,
185*10822SJack.Meng@Sun.COM 			    proplen + 1, "%s", iscsi_par);
186*10822SJack.Meng@Sun.COM 			set = B_TRUE;
187*10822SJack.Meng@Sun.COM 		}
188*10822SJack.Meng@Sun.COM 	}
189*10822SJack.Meng@Sun.COM 	if (set != B_TRUE) {
190*10822SJack.Meng@Sun.COM 		boot_property.boot_tgt.tgt_boot_par =
191*10822SJack.Meng@Sun.COM 		    (uchar_t *)kmem_zalloc(2, KM_SLEEP);
192*10822SJack.Meng@Sun.COM 		boot_property.boot_tgt.tgt_boot_par_len = 2;
193*10822SJack.Meng@Sun.COM 		boot_property.boot_tgt.tgt_boot_par[0] = 'a';
194*10822SJack.Meng@Sun.COM 	}
195*10822SJack.Meng@Sun.COM 
196*10822SJack.Meng@Sun.COM 	/* Get CHAP name and secret */
197*10822SJack.Meng@Sun.COM 	ret = prom_get_security_key(BP_CHAP_USER, chap_user,
198*10822SJack.Meng@Sun.COM 	    ISCSI_OBP_MAX_CHAP_USER_LEN, &chap_user_len, &status);
199*10822SJack.Meng@Sun.COM 	if (ret != 0) {
200*10822SJack.Meng@Sun.COM 		return (B_FALSE);
201*10822SJack.Meng@Sun.COM 	}
202*10822SJack.Meng@Sun.COM 	if (status == OBP_GET_KEY_STATUS_NOT_EXIST) {
203*10822SJack.Meng@Sun.COM 		/* No chap name */
204*10822SJack.Meng@Sun.COM 		return (B_TRUE);
205*10822SJack.Meng@Sun.COM 	}
206*10822SJack.Meng@Sun.COM 	if (status != OBP_GET_KEY_STATUS_OK ||
207*10822SJack.Meng@Sun.COM 	    chap_user_len > ISCSI_OBP_MAX_CHAP_USER_LEN ||
208*10822SJack.Meng@Sun.COM 	    chap_user_len <= 0) {
209*10822SJack.Meng@Sun.COM 		return (B_FALSE);
210*10822SJack.Meng@Sun.COM 	}
211*10822SJack.Meng@Sun.COM 
212*10822SJack.Meng@Sun.COM 	ret = prom_get_security_key(BP_CHAP_PASSWORD, chap_password,
213*10822SJack.Meng@Sun.COM 	    ISCSI_OBP_MAX_CHAP_LEN, &chap_pwd_len, &status);
214*10822SJack.Meng@Sun.COM 	if (ret != 0) {
215*10822SJack.Meng@Sun.COM 		return (B_FALSE);
216*10822SJack.Meng@Sun.COM 	}
217*10822SJack.Meng@Sun.COM 
218*10822SJack.Meng@Sun.COM 	if (status == OBP_GET_KEY_STATUS_NOT_EXIST) {
219*10822SJack.Meng@Sun.COM 		/* No chap secret */
220*10822SJack.Meng@Sun.COM 		return (B_TRUE);
221*10822SJack.Meng@Sun.COM 	}
222*10822SJack.Meng@Sun.COM 	if (status != OBP_GET_KEY_STATUS_OK ||
223*10822SJack.Meng@Sun.COM 	    chap_pwd_len > ISCSI_OBP_MAX_CHAP_LEN ||
224*10822SJack.Meng@Sun.COM 	    chap_pwd_len <= 0) {
225*10822SJack.Meng@Sun.COM 		return (B_FALSE);
226*10822SJack.Meng@Sun.COM 	}
227*10822SJack.Meng@Sun.COM 
228*10822SJack.Meng@Sun.COM 	boot_property.boot_init.ini_chap_name =
229*10822SJack.Meng@Sun.COM 	    (uchar_t *)kmem_zalloc(chap_user_len + 1, KM_SLEEP);
230*10822SJack.Meng@Sun.COM 	boot_property.boot_init.ini_chap_name_len = chap_user_len + 1;
231*10822SJack.Meng@Sun.COM 	(void) memcpy(boot_property.boot_init.ini_chap_name, chap_user,
232*10822SJack.Meng@Sun.COM 	    chap_user_len);
233*10822SJack.Meng@Sun.COM 
234*10822SJack.Meng@Sun.COM 	boot_property.boot_init.ini_chap_sec =
235*10822SJack.Meng@Sun.COM 	    (uchar_t *)kmem_zalloc(chap_pwd_len + 1, KM_SLEEP);
236*10822SJack.Meng@Sun.COM 	boot_property.boot_init.ini_chap_sec_len = chap_pwd_len + 1;
237*10822SJack.Meng@Sun.COM 	(void) memcpy(boot_property.boot_init.ini_chap_sec, chap_password,
238*10822SJack.Meng@Sun.COM 	    chap_pwd_len);
239*10822SJack.Meng@Sun.COM 
240*10822SJack.Meng@Sun.COM 	return (B_TRUE);
241*10822SJack.Meng@Sun.COM }
242*10822SJack.Meng@Sun.COM 
243*10822SJack.Meng@Sun.COM static boolean_t
iscsiboot_init_prop_read(void)244*10822SJack.Meng@Sun.COM iscsiboot_init_prop_read(void)
245*10822SJack.Meng@Sun.COM {
246*10822SJack.Meng@Sun.COM 	int	proplen;
247*10822SJack.Meng@Sun.COM 	uchar_t	iscsi_initiator_id[ISCSI_MAX_NAME_LEN];
248*10822SJack.Meng@Sun.COM 	boolean_t	set = B_FALSE;
249*10822SJack.Meng@Sun.COM 
250*10822SJack.Meng@Sun.COM 	/* Get initiator node name */
251*10822SJack.Meng@Sun.COM 	proplen = BOP_GETPROPLEN(bootops, BP_ISCSI_INITIATOR_ID);
252*10822SJack.Meng@Sun.COM 	if (proplen > 0) {
253*10822SJack.Meng@Sun.COM 		if (BOP_GETPROP(bootops, BP_ISCSI_INITIATOR_ID,
254*10822SJack.Meng@Sun.COM 		    iscsi_initiator_id) > 0) {
255*10822SJack.Meng@Sun.COM 			boot_property.boot_init.ini_name =
256*10822SJack.Meng@Sun.COM 			    (uchar_t *)kmem_zalloc(proplen + 1, KM_SLEEP);
257*10822SJack.Meng@Sun.COM 			boot_property.boot_init.ini_name_len = proplen + 1;
258*10822SJack.Meng@Sun.COM 			(void) snprintf(
259*10822SJack.Meng@Sun.COM 			    (char *)boot_property.boot_init.ini_name,
260*10822SJack.Meng@Sun.COM 			    proplen + 1, "%s", iscsi_initiator_id);
261*10822SJack.Meng@Sun.COM 			set = B_TRUE;
262*10822SJack.Meng@Sun.COM 		}
263*10822SJack.Meng@Sun.COM 	}
264*10822SJack.Meng@Sun.COM 	if (set != B_TRUE) {
265*10822SJack.Meng@Sun.COM 		generate_iscsi_initiator_id();
266*10822SJack.Meng@Sun.COM 	}
267*10822SJack.Meng@Sun.COM 	return (B_TRUE);
268*10822SJack.Meng@Sun.COM }
269*10822SJack.Meng@Sun.COM 
270*10822SJack.Meng@Sun.COM static boolean_t
iscsiboot_nic_prop_read(void)271*10822SJack.Meng@Sun.COM iscsiboot_nic_prop_read(void)
272*10822SJack.Meng@Sun.COM {
273*10822SJack.Meng@Sun.COM 	int	proplen;
274*10822SJack.Meng@Sun.COM 	char	host_ip[INET6_ADDRSTRLEN];
275*10822SJack.Meng@Sun.COM 	char	router_ip[INET6_ADDRSTRLEN];
276*10822SJack.Meng@Sun.COM 	char	subnet_mask[INET6_ADDRSTRLEN];
277*10822SJack.Meng@Sun.COM 	uchar_t	iscsi_network_path[MAXPATHLEN];
278*10822SJack.Meng@Sun.COM 	char	host_mac[6];
279*10822SJack.Meng@Sun.COM 	uchar_t	hex_netmask[4];
280*10822SJack.Meng@Sun.COM 	pnode_t	nodeid;
281*10822SJack.Meng@Sun.COM 	boolean_t	set = B_FALSE;
282*10822SJack.Meng@Sun.COM 
283*10822SJack.Meng@Sun.COM 	/* Get host IP address */
284*10822SJack.Meng@Sun.COM 	proplen = BOP_GETPROPLEN(bootops, BP_HOST_IP);
285*10822SJack.Meng@Sun.COM 	if (proplen > 0) {
286*10822SJack.Meng@Sun.COM 		if (BOP_GETPROP(bootops, BP_HOST_IP,
287*10822SJack.Meng@Sun.COM 		    host_ip) > 0) {
288*10822SJack.Meng@Sun.COM 			if (inet_aton(host_ip,
289*10822SJack.Meng@Sun.COM 			    (uchar_t *)&boot_property.boot_nic.nic_ip_u) ==
290*10822SJack.Meng@Sun.COM 			    0) {
291*10822SJack.Meng@Sun.COM 				boot_property.boot_nic.sin_family = AF_INET;
292*10822SJack.Meng@Sun.COM 				set = B_TRUE;
293*10822SJack.Meng@Sun.COM 			}
294*10822SJack.Meng@Sun.COM 		}
295*10822SJack.Meng@Sun.COM 	}
296*10822SJack.Meng@Sun.COM 	if (set != B_TRUE) {
297*10822SJack.Meng@Sun.COM 		return (B_FALSE);
298*10822SJack.Meng@Sun.COM 	}
299*10822SJack.Meng@Sun.COM 
300*10822SJack.Meng@Sun.COM 	/* Get router IP address */
301*10822SJack.Meng@Sun.COM 	proplen = BOP_GETPROPLEN(bootops, BP_ROUTER_IP);
302*10822SJack.Meng@Sun.COM 	if (proplen > 0) {
303*10822SJack.Meng@Sun.COM 		if (BOP_GETPROP(bootops, BP_ROUTER_IP,
304*10822SJack.Meng@Sun.COM 		    router_ip) > 0) {
305*10822SJack.Meng@Sun.COM 			(void) inet_aton(router_ip,
306*10822SJack.Meng@Sun.COM 			    (uchar_t *)&boot_property.boot_nic.nic_gw_u);
307*10822SJack.Meng@Sun.COM 		}
308*10822SJack.Meng@Sun.COM 	}
309*10822SJack.Meng@Sun.COM 
310*10822SJack.Meng@Sun.COM 	/* Get host netmask */
311*10822SJack.Meng@Sun.COM 	set = B_FALSE;
312*10822SJack.Meng@Sun.COM 	proplen = BOP_GETPROPLEN(bootops, BP_SUBNET_MASK);
313*10822SJack.Meng@Sun.COM 	if (proplen > 0) {
314*10822SJack.Meng@Sun.COM 		if (BOP_GETPROP(bootops, BP_SUBNET_MASK,
315*10822SJack.Meng@Sun.COM 		    subnet_mask) > 0) {
316*10822SJack.Meng@Sun.COM 			if (inet_aton(subnet_mask, hex_netmask) == 0) {
317*10822SJack.Meng@Sun.COM 				int i = 0;
318*10822SJack.Meng@Sun.COM 				uint32_t tmp = *((uint32_t *)hex_netmask);
319*10822SJack.Meng@Sun.COM 				while (tmp) {
320*10822SJack.Meng@Sun.COM 					i ++;
321*10822SJack.Meng@Sun.COM 					tmp = tmp << 1;
322*10822SJack.Meng@Sun.COM 				}
323*10822SJack.Meng@Sun.COM 				boot_property.boot_nic.sub_mask_prefix = i;
324*10822SJack.Meng@Sun.COM 				set = B_TRUE;
325*10822SJack.Meng@Sun.COM 			}
326*10822SJack.Meng@Sun.COM 		}
327*10822SJack.Meng@Sun.COM 	}
328*10822SJack.Meng@Sun.COM 	if (set != B_TRUE) {
329*10822SJack.Meng@Sun.COM 		boot_property.boot_nic.sub_mask_prefix = 24;
330*10822SJack.Meng@Sun.COM 	}
331*10822SJack.Meng@Sun.COM 
332*10822SJack.Meng@Sun.COM 	/* Get iscsi boot NIC path in OBP */
333*10822SJack.Meng@Sun.COM 	set = B_FALSE;
334*10822SJack.Meng@Sun.COM 	proplen = BOP_GETPROPLEN(bootops, BP_ISCSI_NETWORK_BOOTPATH);
335*10822SJack.Meng@Sun.COM 	if (proplen > 0) {
336*10822SJack.Meng@Sun.COM 		if (BOP_GETPROP(bootops, BP_ISCSI_NETWORK_BOOTPATH,
337*10822SJack.Meng@Sun.COM 		    iscsi_network_path) > 0) {
338*10822SJack.Meng@Sun.COM 			nodeid = prom_finddevice((char *)iscsi_network_path);
339*10822SJack.Meng@Sun.COM 			proplen = prom_getproplen(nodeid, BP_LOCAL_MAC_ADDRESS);
340*10822SJack.Meng@Sun.COM 			if (proplen > 0) {
341*10822SJack.Meng@Sun.COM 				if (prom_getprop(nodeid, BP_LOCAL_MAC_ADDRESS,
342*10822SJack.Meng@Sun.COM 				    host_mac) > 0) {
343*10822SJack.Meng@Sun.COM 					(void) memcpy(
344*10822SJack.Meng@Sun.COM 					    boot_property.boot_nic.nic_mac,
345*10822SJack.Meng@Sun.COM 					    host_mac, 6);
346*10822SJack.Meng@Sun.COM 					set = B_TRUE;
347*10822SJack.Meng@Sun.COM 				}
348*10822SJack.Meng@Sun.COM 			}
349*10822SJack.Meng@Sun.COM 		}
350*10822SJack.Meng@Sun.COM 	}
351*10822SJack.Meng@Sun.COM 	if (set != B_TRUE) {
352*10822SJack.Meng@Sun.COM 		return (B_FALSE);
353*10822SJack.Meng@Sun.COM 	}
354*10822SJack.Meng@Sun.COM 
355*10822SJack.Meng@Sun.COM 	return (B_TRUE);
356*10822SJack.Meng@Sun.COM }
357*10822SJack.Meng@Sun.COM 
358*10822SJack.Meng@Sun.COM /*
359*10822SJack.Meng@Sun.COM  * Manully construct iscsiboot_prop table based on
360*10822SJack.Meng@Sun.COM  * OBP '/chosen' properties related to iscsi boot
361*10822SJack.Meng@Sun.COM  */
3628194SJack.Meng@Sun.COM void
ld_ib_prop()363*10822SJack.Meng@Sun.COM ld_ib_prop()
364*10822SJack.Meng@Sun.COM {
365*10822SJack.Meng@Sun.COM 	if (iscsiboot_prop != NULL)
366*10822SJack.Meng@Sun.COM 		return;
367*10822SJack.Meng@Sun.COM 
368*10822SJack.Meng@Sun.COM 	if ((iscsiboot_tgt_prop_read() == B_TRUE) &&
369*10822SJack.Meng@Sun.COM 	    (iscsiboot_init_prop_read() == B_TRUE) &&
370*10822SJack.Meng@Sun.COM 	    (iscsiboot_nic_prop_read() == B_TRUE)) {
371*10822SJack.Meng@Sun.COM 		iscsiboot_prop = &boot_property;
372*10822SJack.Meng@Sun.COM 	} else {
373*10822SJack.Meng@Sun.COM 		iscsi_boot_prop_free();
374*10822SJack.Meng@Sun.COM 	}
375*10822SJack.Meng@Sun.COM }
376*10822SJack.Meng@Sun.COM 
377*10822SJack.Meng@Sun.COM static boolean_t
parse_lun_num(uchar_t * str_num,uchar_t * hex_num)378*10822SJack.Meng@Sun.COM parse_lun_num(uchar_t *str_num, uchar_t *hex_num)
379*10822SJack.Meng@Sun.COM {
380*10822SJack.Meng@Sun.COM 	char *p, *buf;
381*10822SJack.Meng@Sun.COM 	uint16_t *conv_num = (uint16_t *)hex_num;
382*10822SJack.Meng@Sun.COM 	long tmp;
383*10822SJack.Meng@Sun.COM 	int i = 0;
384*10822SJack.Meng@Sun.COM 
385*10822SJack.Meng@Sun.COM 	if ((str_num == NULL) || (hex_num == NULL)) {
386*10822SJack.Meng@Sun.COM 		return (B_FALSE);
387*10822SJack.Meng@Sun.COM 	}
388*10822SJack.Meng@Sun.COM 	bzero((void *)hex_num, 8);
389*10822SJack.Meng@Sun.COM 	buf = (char *)str_num;
390*10822SJack.Meng@Sun.COM 
391*10822SJack.Meng@Sun.COM 	for (i = 0; i < 4; i++) {
392*10822SJack.Meng@Sun.COM 		p = NULL;
393*10822SJack.Meng@Sun.COM 		p = strchr((const char *)buf, '-');
394*10822SJack.Meng@Sun.COM 		if (p != NULL) {
395*10822SJack.Meng@Sun.COM 			*p = '\0';
396*10822SJack.Meng@Sun.COM 		}
397*10822SJack.Meng@Sun.COM 		if (ddi_strtol((const char *)buf, NULL, 16, &tmp) != 0) {
398*10822SJack.Meng@Sun.COM 			return (B_FALSE);
399*10822SJack.Meng@Sun.COM 		}
400*10822SJack.Meng@Sun.COM 		conv_num[i] = (uint16_t)tmp;
401*10822SJack.Meng@Sun.COM 		if (p != NULL) {
402*10822SJack.Meng@Sun.COM 			buf = p + 1;
403*10822SJack.Meng@Sun.COM 		} else {
404*10822SJack.Meng@Sun.COM 			break;
405*10822SJack.Meng@Sun.COM 		}
406*10822SJack.Meng@Sun.COM 	}
407*10822SJack.Meng@Sun.COM 
408*10822SJack.Meng@Sun.COM 	return (B_TRUE);
4098194SJack.Meng@Sun.COM }
410*10822SJack.Meng@Sun.COM 
411*10822SJack.Meng@Sun.COM static void
generate_iscsi_initiator_id(void)412*10822SJack.Meng@Sun.COM generate_iscsi_initiator_id(void)
413*10822SJack.Meng@Sun.COM {
414*10822SJack.Meng@Sun.COM 	boot_property.boot_init.ini_name_len = 38;
415*10822SJack.Meng@Sun.COM 	boot_property.boot_init.ini_name =
416*10822SJack.Meng@Sun.COM 	    (uchar_t *)kmem_zalloc(boot_property.boot_init.ini_name_len,
417*10822SJack.Meng@Sun.COM 	    KM_SLEEP);
418*10822SJack.Meng@Sun.COM 	(void) snprintf((char *)boot_property.boot_init.ini_name,
419*10822SJack.Meng@Sun.COM 	    38, "iqn.1986-03.com.sun:boot.%02x%02x%02x%02x%02x%02x",
420*10822SJack.Meng@Sun.COM 	    boot_property.boot_nic.nic_mac[0],
421*10822SJack.Meng@Sun.COM 	    boot_property.boot_nic.nic_mac[1],
422*10822SJack.Meng@Sun.COM 	    boot_property.boot_nic.nic_mac[2],
423*10822SJack.Meng@Sun.COM 	    boot_property.boot_nic.nic_mac[3],
424*10822SJack.Meng@Sun.COM 	    boot_property.boot_nic.nic_mac[4],
425*10822SJack.Meng@Sun.COM 	    boot_property.boot_nic.nic_mac[5]);
426*10822SJack.Meng@Sun.COM }
427*10822SJack.Meng@Sun.COM 
428*10822SJack.Meng@Sun.COM 
429*10822SJack.Meng@Sun.COM /* We only deal with a.b.c.d decimal format. ip points to 4 byte storage */
430*10822SJack.Meng@Sun.COM static int
inet_aton(char * ipstr,uchar_t * ip)431*10822SJack.Meng@Sun.COM inet_aton(char *ipstr, uchar_t *ip)
432*10822SJack.Meng@Sun.COM {
433*10822SJack.Meng@Sun.COM 	int i = 0;
434*10822SJack.Meng@Sun.COM 	uchar_t val[4] = {0};
435*10822SJack.Meng@Sun.COM 	char c = *ipstr;
436*10822SJack.Meng@Sun.COM 
437*10822SJack.Meng@Sun.COM 	for (;;) {
438*10822SJack.Meng@Sun.COM 		if (!isdigit(c))
439*10822SJack.Meng@Sun.COM 			return (-1);
440*10822SJack.Meng@Sun.COM 		for (;;) {
441*10822SJack.Meng@Sun.COM 			if (!isdigit(c))
442*10822SJack.Meng@Sun.COM 				break;
443*10822SJack.Meng@Sun.COM 			val[i] = val[i] * 10 + (c - '0');
444*10822SJack.Meng@Sun.COM 			c = *++ipstr;
445*10822SJack.Meng@Sun.COM 		}
446*10822SJack.Meng@Sun.COM 		i++;
447*10822SJack.Meng@Sun.COM 		if (i == 4)
448*10822SJack.Meng@Sun.COM 			break;
449*10822SJack.Meng@Sun.COM 		if (c != '.')
450*10822SJack.Meng@Sun.COM 			return (-1);
451*10822SJack.Meng@Sun.COM 		c = *++ipstr;
452*10822SJack.Meng@Sun.COM 	}
453*10822SJack.Meng@Sun.COM 	if (c != 0)
454*10822SJack.Meng@Sun.COM 		return (-1);
455*10822SJack.Meng@Sun.COM 	bcopy(val, ip, 4);
456*10822SJack.Meng@Sun.COM 	return (0);
457*10822SJack.Meng@Sun.COM }
458