xref: /titanic_41/usr/src/lib/lvm/libmeta/common/meta_med.c (revision 3a8ad3333e0bc7ad2934d6fcdb575f3499633aff)
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 /*
23  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
24  * Use is subject to license terms.
25  */
26 
27 /*
28  * Just in case we're not in a build environment, make sure that
29  * TEXT_DOMAIN gets set to something.
30  */
31 #if !defined(TEXT_DOMAIN)
32 #define	TEXT_DOMAIN "SYS_TEST"
33 #endif
34 
35 /*
36  * Mediator functions
37  */
38 
39 #include <meta.h>
40 #include <metamed.h>
41 #include <dlfcn.h>
42 #include <sdssc.h>
43 
44 /*
45  * There are too many external factors that affect the timing of the
46  * operations, so we set the timeout to a very large value, in this
47  * case 1 day, which should handle HW timeouts, large configurations,
48  * and other potential delays.
49  */
50 #define	CL_LONG_TMO	86400L			/* 1 day */
51 #define	CL_MEDIUM_TMO	3600L			/* 1 hour */
52 #define	CL_SHORT_TMO	600L			/* 10 minutes */
53 #define	CL_DEF_TMO	10L			/* 10 seconds */
54 
55 static	md_timeval32_t def_rpcb_timeout =  { MD_CLNT_CREATE_TOUT, 0 };
56 
57 /*
58  * RPC handle
59  */
60 typedef struct {
61 	char	*hostname;
62 	CLIENT	*clntp;
63 } med_handle_t;
64 
65 /*
66  * Data to be sent from med_clnt_create_timed to med_create_helper via
67  * meta_client_create_retry.
68  */
69 typedef struct {
70 	rpcprog_t	mcd_program;	/* RPC program designation */
71 	rpcvers_t	mcd_version;	/* RPC version */
72 	char		*mcd_nettype;	/* Type of network to use for RPC */
73 } med_create_data_t;
74 
75 /*
76  * Perform the work of actually doing the clnt_create for
77  * meta_client_create_retry.
78  */
79 static CLIENT *
80 med_create_helper(char *hostname, void *private, struct timeval *time_out)
81 {
82 	med_create_data_t	*cd = (med_create_data_t *)private;
83 
84 	return (clnt_create_timed(hostname, cd->mcd_program, cd->mcd_version,
85 	    cd->mcd_nettype, time_out));
86 }
87 
88 static
89 CLIENT *med_clnt_create_timed(
90 	char *hostname,
91 	const ulong_t prog,
92 	const ulong_t vers,
93 	char *nettype,
94 	const md_timeval32_t *tp
95 )
96 {
97 	med_create_data_t	cd;	/* Create data. */
98 
99 	cd.mcd_program = prog;
100 	cd.mcd_version = vers;
101 	cd.mcd_nettype = nettype;
102 	return (meta_client_create_retry(hostname, med_create_helper,
103 	    (void *)&cd, (time_t)tp->tv_sec, NULL));
104 }
105 
106 /*
107  * Set the timeout value for this client handle.
108  */
109 static int
110 cl_sto_medd(
111 	CLIENT		*clntp,
112 	char		*hostname,
113 	long		time_out,
114 	md_error_t	*ep
115 )
116 {
117 	md_timeval32_t	nto;
118 
119 	(void) memset(&nto, '\0', sizeof (nto));
120 
121 	nto.tv_sec = time_out;
122 
123 	if (clnt_control(clntp, CLSET_TIMEOUT, (char *)&nto) != TRUE)
124 		return (mdrpcerror(ep, clntp, hostname,
125 		    dgettext(TEXT_DOMAIN, "metad client set timeout")));
126 
127 	return (0);
128 }
129 
130 /*
131  * close RPC connection
132  */
133 static void
134 close_medd(
135 	med_handle_t	*hp
136 )
137 {
138 	assert(hp != NULL);
139 	if (hp->hostname != NULL) {
140 		Free(hp->hostname);
141 	}
142 	if (hp->clntp != NULL) {
143 		auth_destroy(hp->clntp->cl_auth);
144 		clnt_destroy(hp->clntp);
145 	}
146 	Free(hp);
147 }
148 
149 /*
150  * open RPC connection to rpc.medd
151  */
152 static med_handle_t *
153 open_medd(
154 	char		*hostname,
155 	long		time_out,
156 	md_error_t	*ep
157 )
158 {
159 	CLIENT		*clntp;
160 	med_handle_t	*hp;
161 
162 	/* default to local host */
163 	if ((hostname == NULL) || (*hostname == '\0'))
164 		hostname = mynode();
165 
166 	/* open RPC connection */
167 	assert(hostname != NULL);
168 	if ((clntp = med_clnt_create_timed(hostname, MED_PROG, MED_VERS,
169 	    "tcp", &def_rpcb_timeout)) == NULL) {
170 		if (rpc_createerr.cf_stat != RPC_PROGNOTREGISTERED)
171 			clnt_pcreateerror(hostname);
172 		(void) mdrpccreateerror(ep, hostname,
173 		    "medd med_clnt_create_timed");
174 		return (NULL);
175 	} else {
176 		auth_destroy(clntp->cl_auth);
177 		clntp->cl_auth = authsys_create_default();
178 		assert(clntp->cl_auth != NULL);
179 	}
180 
181 	if (cl_sto_medd(clntp, hostname, time_out, ep) != 0)
182 		return (NULL);
183 
184 	/* return connection */
185 	hp = Zalloc(sizeof (*hp));
186 	hp->hostname = Strdup(hostname);
187 	hp->clntp = clntp;
188 
189 	return (hp);
190 }
191 
192 /*
193  * steal and convert med_err_t
194  */
195 int
196 meddstealerror(
197 	md_error_t	*ep,
198 	med_err_t	*medep
199 )
200 {
201 	char		buf[BUFSIZ];
202 	char		*p = buf;
203 	size_t		psize = BUFSIZ;
204 	char		*emsg;
205 	int		rval = -1;
206 
207 	/* no error */
208 	if (medep->med_errno == 0) {
209 		/* assert(medep->name == NULL); */
210 		rval = 0;
211 		goto out;
212 	}
213 
214 	/* steal error */
215 	if ((medep->med_node != NULL) && (medep->med_node[0] != '\0')) {
216 		(void) snprintf(p, psize, "%s: ", medep->med_node);
217 		p = &buf[strlen(buf)];
218 		psize = buf + BUFSIZ - p;
219 	}
220 
221 	if ((medep->med_misc != NULL) && (medep->med_misc[0] != '\0')) {
222 		(void) snprintf(p, psize, "%s: ", medep->med_misc);
223 		p = &buf[strlen(buf)];
224 		psize = buf + BUFSIZ - p;
225 	}
226 
227 	if (medep->med_errno < 0) {
228 		if ((emsg = med_errnum_to_str(medep->med_errno)) != NULL)
229 			(void) snprintf(p, psize, "%s", emsg);
230 		else
231 			(void) snprintf(p, psize, dgettext(TEXT_DOMAIN,
232 			    "unknown mediator errno %d\n"), medep->med_errno);
233 	} else {
234 		if ((emsg = strerror(medep->med_errno)) != NULL)
235 			(void) snprintf(p, psize, "%s", emsg);
236 		else
237 			(void) snprintf(p, psize, dgettext(TEXT_DOMAIN,
238 			    "errno %d out of range"), medep->med_errno);
239 	}
240 	(void) mderror(ep, MDE_MED_ERROR, buf);
241 
242 	/* cleanup, return success */
243 out:
244 	if (medep->med_node != NULL)
245 		Free(medep->med_node);
246 	if (medep->med_misc != NULL)
247 		Free(medep->med_misc);
248 	(void) memset(medep, 0, sizeof (*medep));
249 	return (rval);
250 }
251 
252 static med_handle_t *
253 open_medd_wrap(
254 	md_h_t		*mdhp,
255 	long		time_out,
256 	md_error_t	*ep
257 )
258 {
259 	med_handle_t		*hp = NULL;
260 	int			i;
261 	char    		*hnm;
262 
263 	assert(mdhp && mdhp->a_cnt > 0);
264 
265 	/* Loop through the hosts listed */
266 	i = min(mdhp->a_cnt, MAX_HOST_ADDRS) - 1;
267 	for (; i >= 0; i--) {
268 		hnm = mdhp->a_nm[i];
269 
270 		if ((hp = open_medd(hnm, time_out, ep)) == NULL) {
271 			if (mdanyrpcerror(ep) && i != 0) {
272 				mdclrerror(ep);
273 				continue;
274 			}
275 		}
276 		return (hp);
277 	}
278 
279 	rpc_createerr.cf_stat = RPC_CANTSEND;
280 	rpc_createerr.cf_error.re_status = 0;
281 	(void) mdrpccreateerror(ep, mdhp->a_nm[0],
282 	    dgettext(TEXT_DOMAIN, "medd open wrap"));
283 
284 	return (NULL);
285 }
286 
287 static int
288 setup_med_transtab(md_error_t *ep)
289 {
290 	mddb_med_t_parm_t	*tp = NULL;
291 	struct	stat		statb;
292 	int			i;
293 	size_t			alloc_size = 0;
294 	int			err = 0;
295 
296 
297 	if ((tp = Zalloc(sizeof (mddb_med_t_parm_t))) == NULL)
298 		return (mdsyserror(ep, ENOMEM, "setup_med_transtab"));
299 
300 	if (metaioctl(MD_MED_GET_TLEN, tp, &tp->med_tp_mde, NULL) != 0) {
301 		err = mdstealerror(ep, &tp->med_tp_mde);
302 		goto out;
303 	}
304 
305 	if (tp->med_tp_setup == 1)
306 		goto out;
307 
308 	alloc_size = (sizeof (mddb_med_t_parm_t) - sizeof (mddb_med_t_ent_t)) +
309 	    (sizeof (mddb_med_t_ent_t) * tp->med_tp_nents);
310 
311 	if ((tp = Realloc(tp, alloc_size)) == NULL) {
312 		err = mdsyserror(ep, ENOMEM, "setup_med_transtab");
313 		goto out;
314 	}
315 
316 	if (metaioctl(MD_MED_GET_T, tp, &tp->med_tp_mde, NULL) != 0) {
317 		err = mdstealerror(ep, &tp->med_tp_mde);
318 		goto out;
319 	}
320 
321 	for (i = 0; i < tp->med_tp_nents; i++) {
322 		if (meta_stat(tp->med_tp_ents[i].med_te_nm, &statb) == -1) {
323 			md_perror("setup_med_transtab(): stat():");
324 			tp->med_tp_ents[i].med_te_dev = NODEV64;
325 		} else {
326 			tp->med_tp_ents[i].med_te_dev =
327 			    meta_expldev(statb.st_rdev);
328 		}
329 	}
330 
331 	if (metaioctl(MD_MED_SET_T, tp, &tp->med_tp_mde, NULL) != 0)
332 		err = mdstealerror(ep, &tp->med_tp_mde);
333 
334 out:
335 	Free(tp);
336 	return (err);
337 }
338 
339 /*
340  * Externals
341  */
342 
343 /*
344  * NULLPROC - just returns a response
345  */
346 int
347 clnt_med_null(
348 	char			*hostname,
349 	md_error_t		*ep
350 )
351 {
352 	med_handle_t		*hp;
353 	med_err_t		res;
354 
355 	/* initialize */
356 	mdclrerror(ep);
357 
358 	/* do it */
359 	if ((hp = open_medd(hostname, CL_DEF_TMO, ep)) == NULL)
360 		return (-1);
361 
362 	if (med_null_1(NULL, &res, hp->clntp) != RPC_SUCCESS)
363 		(void) mdrpcerror(ep, hp->clntp, hostname,
364 		    dgettext(TEXT_DOMAIN, "medd nullproc"));
365 
366 	close_medd(hp);
367 
368 	xdr_free(xdr_med_err_t, (char *)&res);
369 
370 	if (! mdisok(ep))
371 		return (-1);
372 
373 	return (0);
374 }
375 
376 /*
377  * Update the mediator information on the mediator.
378  * This function does the same functionality as
379  * clnt_med_upd_data() except that it takes different
380  * argument so that host which is just a mediator, can
381  * still update its mediator record.
382  */
383 int
384 clnt_user_med_upd_data(
385 	md_h_t	*mdhp,
386 	bool_t	obandiskset,
387 	char	*setname,
388 	uint_t	setnum,
389 	med_data_t	*meddp,
390 	md_error_t	*ep
391 )
392 {
393 	med_handle_t    	*hp;
394 	med_upd_data_args_t	args;
395 	med_err_t		res;
396 
397 	/* Initialize */
398 	mdclrerror(ep);
399 	(void) memset(&args, 0, sizeof (args));
400 	(void) memset(&res, 0, sizeof (res));
401 
402 	/* Build args */
403 	if (obandiskset)
404 		args.med.med_caller = Strdup(MED_MN_CALLER);
405 	else
406 		args.med.med_caller = Strdup(mynode());
407 
408 	args.med.med_setname = Strdup(setname);
409 	args.med.med_setno = setnum;
410 	args.med_data = *meddp;
411 
412 	if ((hp = open_medd_wrap(mdhp, CL_DEF_TMO, ep)) == NULL)
413 		return (-1);
414 
415 	if (med_upd_data_1(&args, &res, hp->clntp) != RPC_SUCCESS)
416 		(void) mdrpcerror(ep, hp->clntp, hp->hostname,
417 		    dgettext(TEXT_DOMAIN, "medd get record"));
418 	else
419 		(void) meddstealerror(ep, &res);
420 
421 	close_medd(hp);
422 
423 	xdr_free(xdr_med_upd_data_args_t, (char *)&args);
424 	xdr_free(xdr_med_err_t, (char *)&res);
425 
426 	if (! mdisok(ep))
427 		return (-1);
428 
429 	return (0);
430 }
431 
432 /*
433  * Get the mediator information from the client.
434  * The code does same functinality as clnt_med_get_data()
435  * except that it takes different arguments so that
436  * host which doesn't have set information, can still
437  * get access to mediator information
438  */
439 int
440 clnt_user_med_get_data(
441 	md_h_t	*mdhp,
442 	bool_t	obandiskset,
443 	char	*setname,
444 	uint_t	setnum,
445 	med_data_t	*meddp,
446 	md_error_t	*ep
447 )
448 {
449 	int			rval = -1;
450 	med_handle_t		*hp;
451 	med_args_t		args;
452 	med_get_data_res_t	res;
453 
454 	/* Initialize */
455 	mdclrerror(ep);
456 	(void) memset(&args, 0, sizeof (args));
457 	(void) memset(&res, 0, sizeof (res));
458 
459 	/* Build args */
460 	if (obandiskset)
461 		args.med.med_caller = Strdup(MED_MN_CALLER);
462 	else
463 		args.med.med_caller = Strdup(mynode());
464 
465 	args.med.med_setname = Strdup(setname);
466 	args.med.med_setno = setnum;
467 
468 	if ((hp = open_medd_wrap(mdhp, CL_DEF_TMO, ep)) == NULL)
469 		return (-1);
470 
471 	if (med_get_data_1(&args, &res, hp->clntp) != RPC_SUCCESS)
472 		(void) mdrpcerror(ep, hp->clntp, hp->hostname,
473 		    dgettext(TEXT_DOMAIN, "medd get record"));
474 	else
475 		(void) meddstealerror(ep, &res.med_status);
476 
477 	close_medd(hp);
478 
479 	if (mdisok(ep)) {
480 		/* copy the mediator data in meddp */
481 		(void) memmove(meddp, &res.med_data, sizeof (med_data_t));
482 		rval = 0;
483 	}
484 
485 	xdr_free(xdr_med_args_t, (char *)&args);
486 	xdr_free(xdr_med_get_data_res_t, (char *)&res);
487 
488 	return (rval);
489 }
490 
491 
492 /*
493  * Update the mediator information on the mediator.
494  * *** This is not normally called from user code, the kernel does this! ***
495  */
496 int
497 clnt_med_upd_data(
498 	md_h_t			*mdhp,
499 	mdsetname_t		*sp,
500 	med_data_t		*meddp,
501 	md_error_t		*ep
502 )
503 {
504 	med_handle_t		*hp;
505 	med_upd_data_args_t	args;
506 	med_err_t		res;
507 	md_set_desc		*sd;
508 
509 	/* initialize */
510 	mdclrerror(ep);
511 	(void) memset(&args, 0, sizeof (args));
512 	(void) memset(&res, 0, sizeof (res));
513 
514 	/* build args */
515 	if ((sd = metaget_setdesc(sp, ep)) == NULL)
516 		return (-1);
517 
518 	if (MD_MNSET_DESC(sd))
519 		/*
520 		 * In the MN diskset, use a generic nodename, multiowner, as
521 		 * the node initiating the RPC request.  This allows
522 		 * any node to access mediator information.
523 		 *
524 		 * MN diskset reconfig cycle forces consistent
525 		 * view of set/node/drive/mediator information across all nodes
526 		 * in the MN diskset.  This allows the relaxation of
527 		 * node name checking in rpc.metamedd for MN disksets.
528 		 *
529 		 * In the traditional diskset, only a calling node that is
530 		 * in the mediator record's diskset nodelist can access
531 		 * mediator data.
532 		 */
533 		args.med.med_caller = Strdup(MED_MN_CALLER);
534 	else
535 		args.med.med_caller = Strdup(mynode());
536 	args.med.med_setname = Strdup(sp->setname);
537 	args.med.med_setno = sp->setno;
538 	args.med_data = *meddp;
539 
540 	/* do it */
541 	if ((hp = open_medd_wrap(mdhp, CL_DEF_TMO, ep)) == NULL)
542 		return (-1);
543 
544 	if (med_upd_data_1(&args, &res, hp->clntp) != RPC_SUCCESS)
545 		(void) mdrpcerror(ep, hp->clntp, hp->hostname,
546 		    dgettext(TEXT_DOMAIN, "medd update data"));
547 	else
548 		(void) meddstealerror(ep, &res);
549 
550 	close_medd(hp);
551 
552 	xdr_free(xdr_med_upd_data_args_t, (char *)&args);
553 	xdr_free(xdr_med_err_t, (char *)&res);
554 
555 	if (! mdisok(ep))
556 		return (-1);
557 
558 	return (0);
559 }
560 
561 /*
562  * Get the mediator data for this client from the mediator
563  */
564 int
565 clnt_med_get_data(
566 	md_h_t			*mdhp,
567 	mdsetname_t		*sp,
568 	med_data_t		*meddp,
569 	md_error_t		*ep
570 )
571 {
572 	med_handle_t		*hp;
573 	med_args_t		args;
574 	med_get_data_res_t	res;
575 	int			rval = -1;
576 	md_set_desc		*sd;
577 
578 	/* initialize */
579 	mdclrerror(ep);
580 	(void) memset(&args, 0, sizeof (args));
581 	(void) memset(&res, 0, sizeof (res));
582 
583 	/* build args */
584 	if ((sd = metaget_setdesc(sp, ep)) == NULL)
585 		return (-1);
586 
587 	if (MD_MNSET_DESC(sd))
588 		/*
589 		 * In the MN diskset, use a generic nodename, multiowner, as
590 		 * the node initiating the RPC request.  This allows
591 		 * any node to access mediator information.
592 		 *
593 		 * MN diskset reconfig cycle forces consistent
594 		 * view of set/node/drive/mediator information across all nodes
595 		 * in the MN diskset.  This allows the relaxation of
596 		 * node name checking in rpc.metamedd for MN disksets.
597 		 *
598 		 * In the traditional diskset, only a calling node that is
599 		 * in the mediator record's diskset nodelist can access
600 		 * mediator data.
601 		 */
602 		args.med.med_caller = Strdup(MED_MN_CALLER);
603 	else
604 		args.med.med_caller = Strdup(mynode());
605 	args.med.med_setname = Strdup(sp->setname);
606 	args.med.med_setno = sp->setno;
607 
608 	/* do it */
609 	if ((hp = open_medd_wrap(mdhp, CL_DEF_TMO, ep)) == NULL)
610 		return (-1);
611 
612 	if (med_get_data_1(&args, &res, hp->clntp) != RPC_SUCCESS)
613 		(void) mdrpcerror(ep, hp->clntp, hp->hostname,
614 		    dgettext(TEXT_DOMAIN, "medd get data"));
615 	else
616 		(void) meddstealerror(ep, &res.med_status);
617 
618 	close_medd(hp);
619 
620 	if (mdisok(ep)) {
621 		/* do something with the results */
622 		(void) memmove(meddp, &res.med_data, sizeof (med_data_t));
623 		rval = 0;
624 	}
625 
626 	xdr_free(xdr_med_args_t, (char *)&args);
627 	xdr_free(xdr_med_get_data_res_t, (char *)&res);
628 
629 	return (rval);
630 }
631 
632 /*
633  * Update the mediator record on the mediator.
634  */
635 int
636 clnt_med_upd_rec(
637 	md_h_t			*mdhp,
638 	mdsetname_t		*sp,
639 	med_rec_t		*medrp,
640 	md_error_t		*ep
641 )
642 {
643 	med_handle_t		*hp;
644 	med_upd_rec_args_t	args;
645 	med_err_t		res;
646 	md_set_desc		*sd;
647 
648 	/* initialize */
649 	mdclrerror(ep);
650 	(void) memset(&args, 0, sizeof (args));
651 	(void) memset(&res, 0, sizeof (res));
652 
653 	/* build args */
654 	if ((sd = metaget_setdesc(sp, ep)) == NULL)
655 		return (-1);
656 
657 	if (MD_MNSET_DESC(sd))
658 		/*
659 		 * In the MN diskset, use a generic nodename, multiowner, as
660 		 * the node initiating the RPC request.  This allows
661 		 * any node to access mediator information.
662 		 *
663 		 * MN diskset reconfig cycle forces consistent
664 		 * view of set/node/drive/mediator information across all nodes
665 		 * in the MN diskset.  This allows the relaxation of
666 		 * node name checking in rpc.metamedd for MN disksets.
667 		 *
668 		 * In the traditional diskset, only a calling node that is
669 		 * in the mediator record's diskset nodelist can access
670 		 * mediator data.
671 		 */
672 		args.med.med_caller = Strdup(MED_MN_CALLER);
673 	else
674 		args.med.med_caller = Strdup(mynode());
675 	args.med.med_setname = Strdup(sp->setname);
676 	args.med.med_setno = sp->setno;
677 	args.med_flags = 0;
678 	args.med_rec = *medrp;			/* structure assignment */
679 
680 	/* do it */
681 	if ((hp = open_medd_wrap(mdhp, CL_DEF_TMO, ep)) == NULL)
682 		return (-1);
683 
684 	if (med_upd_rec_1(&args, &res, hp->clntp) != RPC_SUCCESS)
685 		(void) mdrpcerror(ep, hp->clntp, hp->hostname,
686 		    dgettext(TEXT_DOMAIN, "medd update record"));
687 	else
688 		(void) meddstealerror(ep, &res);
689 
690 	close_medd(hp);
691 
692 	xdr_free(xdr_med_upd_rec_args_t, (char *)&args);
693 	xdr_free(xdr_med_err_t, (char *)&res);
694 
695 	if (! mdisok(ep))
696 		return (-1);
697 
698 	return (0);
699 }
700 
701 /*
702  * Get the mediator record for this client from the mediator
703  */
704 int
705 clnt_med_get_rec(
706 	md_h_t			*mdhp,
707 	mdsetname_t		*sp,
708 	med_rec_t		*medrp,
709 	md_error_t		*ep
710 )
711 {
712 	med_handle_t		*hp;
713 	med_args_t		args;
714 	med_get_rec_res_t	res;
715 	int			rval = -1;
716 	md_set_desc		*sd;
717 
718 	/* initialize */
719 	mdclrerror(ep);
720 	(void) memset(&args, 0, sizeof (args));
721 	(void) memset(&res, 0, sizeof (res));
722 
723 	/* build args */
724 	if ((sd = metaget_setdesc(sp, ep)) == NULL)
725 		return (-1);
726 
727 	if (MD_MNSET_DESC(sd))
728 		/*
729 		 * In the MN diskset, use a generic nodename, multiowner, as
730 		 * the node initiating the RPC request.  This allows
731 		 * any node to access mediator information.
732 		 *
733 		 * MN diskset reconfig cycle forces consistent
734 		 * view of set/node/drive/mediator information across all nodes
735 		 * in the MN diskset.  This allows the relaxation of
736 		 * node name checking in rpc.metamedd for MN disksets.
737 		 *
738 		 * In the traditional diskset, only a calling node that is
739 		 * in the mediator record's diskset nodelist can access
740 		 * mediator data.
741 		 */
742 		args.med.med_caller = Strdup(MED_MN_CALLER);
743 	else
744 		args.med.med_caller = Strdup(mynode());
745 	args.med.med_setname = Strdup(sp->setname);
746 	args.med.med_setno = sp->setno;
747 
748 	/* do it */
749 	if ((hp = open_medd_wrap(mdhp, CL_DEF_TMO, ep)) == NULL)
750 		return (-1);
751 
752 	if (med_get_rec_1(&args, &res, hp->clntp) != RPC_SUCCESS)
753 		(void) mdrpcerror(ep, hp->clntp, hp->hostname,
754 		    dgettext(TEXT_DOMAIN, "medd get record"));
755 	else
756 		(void) meddstealerror(ep, &res.med_status);
757 
758 	close_medd(hp);
759 
760 	if (mdisok(ep)) {
761 		/* do something with the results */
762 		(void) memmove(medrp, &res.med_rec, sizeof (med_rec_t));
763 		rval = 0;
764 	}
765 
766 	xdr_free(xdr_med_args_t, (char *)&args);
767 	xdr_free(xdr_med_get_rec_res_t, (char *)&res);
768 
769 	return (rval);
770 }
771 
772 /*
773  * Get the name of the host from the mediator daemon.
774  */
775 int
776 clnt_med_hostname(
777 	char			*hostname,
778 	char			**ret_hostname,
779 	md_error_t		*ep
780 )
781 {
782 	med_handle_t		*hp;
783 	med_hnm_res_t		res;
784 	int			rval = -1;
785 
786 	/* initialize */
787 	mdclrerror(ep);
788 	(void) memset(&res, 0, sizeof (res));
789 
790 	/* No args */
791 
792 	/* do it */
793 	if ((hp = open_medd(hostname, CL_DEF_TMO, ep)) == NULL)
794 		return (-1);
795 
796 	if (med_hostname_1(NULL, &res, hp->clntp) != RPC_SUCCESS)
797 		(void) mdrpcerror(ep, hp->clntp, hostname,
798 		    dgettext(TEXT_DOMAIN, "medd hostname"));
799 	else
800 		(void) meddstealerror(ep, &res.med_status);
801 
802 	close_medd(hp);
803 
804 	if (mdisok(ep)) {
805 		/* do something with the results */
806 		rval = 0;
807 
808 		if (ret_hostname != NULL)
809 			*ret_hostname = Strdup(res.med_hnm);
810 	}
811 
812 	xdr_free(xdr_med_hnm_res_t, (char *)&res);
813 
814 	return (rval);
815 }
816 
817 int
818 meta_med_hnm2ip(md_hi_arr_t *mp, md_error_t *ep)
819 {
820 	int		i, j;
821 	int		max_meds;
822 
823 	if ((max_meds = get_max_meds(ep)) == 0)
824 		return (-1);
825 
826 	for (i = 0; i < max_meds; i++) {
827 		mp->n_lst[i].a_flg = 0;
828 		/* See if this is the local host */
829 		if (mp->n_lst[i].a_cnt > 0 &&
830 		    strcmp(mp->n_lst[i].a_nm[0], mynode()) == NULL)
831 			mp->n_lst[i].a_flg |= NMIP_F_LOCAL;
832 
833 		for (j = 0; j < mp->n_lst[i].a_cnt; j++) {
834 			struct hostent	*hp;
835 			char		*hnm = mp->n_lst[i].a_nm[j];
836 
837 			/*
838 			 * Cluster nodename support
839 			 *
840 			 * See if the clustering code can give us an IP addr
841 			 * for the stored name. If not, find it the old way
842 			 * which will use the public interface.
843 			 */
844 			if (sdssc_get_priv_ipaddr(mp->n_lst[i].a_nm[j],
845 			    (struct in_addr *)&mp->n_lst[i].a_ip[j]) !=
846 			    SDSSC_OKAY) {
847 				if ((hp = gethostbyname(hnm)) == NULL)
848 					return (mdsyserror(ep, EADDRNOTAVAIL,
849 					    hnm));
850 
851 				/* We only do INET addresses */
852 				if (hp->h_addrtype != AF_INET)
853 					return (mdsyserror(ep, EPFNOSUPPORT,
854 					    hnm));
855 
856 				/* We take the first address only */
857 				if (*hp->h_addr_list) {
858 					(void) memmove(&mp->n_lst[i].a_ip[j],
859 					    *hp->h_addr_list,
860 					    sizeof (struct in_addr));
861 				} else
862 					return (mdsyserror(ep, EADDRNOTAVAIL,
863 					    hnm));
864 			}
865 
866 		}
867 	}
868 	return (0);
869 }
870 
871 int
872 meta_h2hi(md_h_arr_t *mdhp, md_hi_arr_t *mdhip, md_error_t *ep)
873 {
874 	int			i, j;
875 	int			max_meds;
876 
877 	if ((max_meds = get_max_meds(ep)) == 0)
878 		return (-1);
879 
880 	mdhip->n_cnt = mdhp->n_cnt;
881 
882 	for (i = 0; i < max_meds; i++) {
883 		mdhip->n_lst[i].a_flg = 0;
884 		mdhip->n_lst[i].a_cnt = mdhp->n_lst[i].a_cnt;
885 		if (mdhp->n_lst[i].a_cnt == 0)
886 			continue;
887 		for (j = 0; j < mdhp->n_lst[i].a_cnt; j++)
888 			(void) strcpy(mdhip->n_lst[i].a_nm[j],
889 			    mdhp->n_lst[i].a_nm[j]);
890 	}
891 	return (0);
892 }
893 
894 int
895 meta_hi2h(md_hi_arr_t *mdhip, md_h_arr_t *mdhp, md_error_t *ep)
896 {
897 	int			i, j;
898 	int			max_meds;
899 
900 	if ((max_meds = get_max_meds(ep)) == 0)
901 		return (-1);
902 
903 	mdhp->n_cnt = mdhip->n_cnt;
904 	for (i = 0; i < max_meds; i++) {
905 		mdhp->n_lst[i].a_cnt = mdhip->n_lst[i].a_cnt;
906 		if (mdhip->n_lst[i].a_cnt == 0)
907 			continue;
908 		for (j = 0; j < mdhip->n_lst[i].a_cnt; j++)
909 			(void) strcpy(mdhp->n_lst[i].a_nm[j],
910 			    mdhip->n_lst[i].a_nm[j]);
911 	}
912 	return (0);
913 }
914 
915 int
916 setup_med_cfg(
917 	mdsetname_t		*sp,
918 	mddb_config_t		*cp,
919 	int			force,
920 	md_error_t		*ep
921 )
922 {
923 	md_set_desc		*sd;
924 	int			i;
925 	int			max_meds;
926 
927 	if (metaislocalset(sp))
928 		return (0);
929 
930 	if ((sd = metaget_setdesc(sp, ep)) == NULL)
931 		return (-1);
932 
933 	if (setup_med_transtab(ep))
934 		return (-1);
935 
936 	if (meta_h2hi(&sd->sd_med, &cp->c_med, ep))
937 		return (-1);
938 
939 	/* Make sure the ip addresses are current */
940 	if (meta_med_hnm2ip(&cp->c_med, ep))
941 		return (-1);
942 
943 	if (force)
944 		return (0);
945 
946 	if ((max_meds = get_max_meds(ep)) == 0)
947 		return (-1);
948 
949 	/* Make sure metamedd still running on host - only chk nodename */
950 	for (i = 0; i < max_meds; i++) {
951 		char		*hostname;
952 		char		*hnm;
953 
954 		if (sd->sd_med.n_lst[i].a_cnt == 0)
955 			continue;
956 
957 		hnm = sd->sd_med.n_lst[i].a_nm[0];
958 
959 		if (clnt_med_hostname(hnm, &hostname, ep))
960 			return (mddserror(ep, MDE_DS_NOMEDONHOST, sp->setno,
961 			    hnm, NULL, sp->setname));
962 		Free(hostname);
963 	}
964 	return (0);
965 }
966 
967 /*
968  * This is a general routine to get mediator information from
969  * file /etc/lvm/meddb. Commands medstat and metainit use this
970  * routine to get mediator information from all mediator hosts or update
971  * its mediator record respectively.
972  */
973 int
974 meta_mediator_info_from_file(char *sname, int verbose, md_error_t *ep)
975 {
976 	uint_t		c;
977 	int		i;
978 	int		j;
979 	int		fd;
980 	int		rec_size;
981 	char		*setname;
982 	uint_t		setnum;
983 	med_rec_t	*rec_buf = NULL;
984 	med_db_hdr_t	*dbhbr;
985 	med_rec_t	*medrecp;
986 	med_data_t	medd;
987 	med_data_t	save_medd;
988 	md_h_t		mdh;
989 	uint_t		latest_med_dat_cc = 0;
990 	int		retval = 0;
991 	int		medok = 0;
992 	int		golden = 0;
993 	bool_t		obandiskset;
994 	int		isSetFound = 0;
995 
996 	/* Open the meddb file */
997 	if ((fd = open(MED_DB_FILE, O_RDONLY, 0)) == -1) {
998 
999 		/*
1000 		 * During the start up of the SVM services, this function
1001 		 * will be called with an empty sname. In that case it is
1002 		 * entirely possible for the MED_DB_FILE not to exist.
1003 		 * If so, then no need to report an error.
1004 		 */
1005 		if (sname != NULL) {
1006 			(void) mdsyserror(ep, errno, MED_DB_FILE);
1007 			mde_perror(ep, dgettext(TEXT_DOMAIN,
1008 			    "Error in opening meddb file"));
1009 			return (1);
1010 		}
1011 		return (0);
1012 	}
1013 
1014 	/* Initialize rec_size */
1015 	rec_size = roundup(sizeof (med_rec_t), DEV_BSIZE);
1016 
1017 	/* Allocate a record buffer */
1018 	if ((rec_buf = malloc(rec_size)) == NULL) {
1019 		(void) mdsyserror(ep, errno, MED_DB_FILE);
1020 		mde_perror(ep, dgettext(TEXT_DOMAIN,
1021 		    "Error in allocating memory"));
1022 		goto out;
1023 	}
1024 
1025 	/* read the file header */
1026 	if ((read(fd, rec_buf, rec_size)) != rec_size) {
1027 		(void) mdsyserror(ep, EINVAL, MED_DB_FILE);
1028 		mde_perror(ep, dgettext(TEXT_DOMAIN,
1029 		    "Error in reading mediator record"));
1030 		goto out;
1031 	}
1032 
1033 	dbhbr = (med_db_hdr_t *)rec_buf;
1034 
1035 	/* Number of records in the mediator file */
1036 	c = dbhbr->med_dbh_nm;
1037 
1038 	for (i = 0; i < c; i++) {
1039 		(void) memset(rec_buf, 0, rec_size);
1040 
1041 		if (read(fd, rec_buf, rec_size) == -1) {
1042 			(void) mdsyserror(ep, errno, MED_DB_FILE);
1043 			mde_perror(ep, dgettext(TEXT_DOMAIN,
1044 			    "Error in reading mediator record"));
1045 			goto out;
1046 		}
1047 
1048 		medrecp = (med_rec_t *)rec_buf;
1049 
1050 		/*
1051 		 * For oban diskset first entry in the rec_nodes field is
1052 		 * "multiowner" and all other entries are empty.
1053 		 * Check if this is really multiowner diskset.
1054 		 */
1055 
1056 		if ((strcmp(medrecp->med_rec_nodes[0], MED_MN_CALLER) == 0) &&
1057 		    (medrecp->med_rec_nodes[1][0] == '\0'))
1058 			obandiskset = TRUE;
1059 		else
1060 			obandiskset = FALSE;
1061 
1062 		if (sname != NULL) {
1063 			/*
1064 			 * Continue if the set name is not in our interest.
1065 			 * This is required when this routine is called
1066 			 * from medstat
1067 			 */
1068 
1069 			if (strcmp(sname, medrecp->med_rec_snm) != 0) {
1070 				continue;
1071 			}
1072 
1073 			if (verbose)
1074 				(void) printf("%8.8s\t\t%6.6s\t%6.6s\n",
1075 				    gettext("Mediator"), gettext("Status"),
1076 				    gettext("Golden"));
1077 
1078 			isSetFound = 1;
1079 			setname = sname;
1080 		} else {
1081 			setname = medrecp->med_rec_snm;
1082 		}
1083 		setnum = medrecp->med_rec_sn;
1084 		(void) memset(&medd, '\0', sizeof (medd));
1085 		(void) memset(&mdh, '\0', sizeof (mdh));
1086 		(void) memset(&save_medd, '\0', sizeof (save_medd));
1087 		latest_med_dat_cc = 0;
1088 
1089 		for (j = 0; j < MED_MAX_HOSTS; j++) {
1090 			/*
1091 			 * It is possible for the n_lst[j] slot to be empty
1092 			 * if the mediator node has already been removed so
1093 			 * go to the next slot.
1094 			 */
1095 			if (medrecp->med_rec_meds.n_lst[j].a_cnt == 0)
1096 				continue;
1097 			mdh = medrecp->med_rec_meds.n_lst[j];
1098 
1099 			if ((sname != NULL) && (verbose))
1100 				(void) printf("%-17.17s\t",
1101 				    medrecp->med_rec_meds.n_lst[j].a_nm[0]);
1102 
1103 			if (clnt_user_med_get_data(&mdh, obandiskset,
1104 			    setname, setnum, &medd, ep) == -1) {
1105 				if (sname == NULL) {
1106 					continue;
1107 				} else {
1108 					if (mdanyrpcerror(ep)) {
1109 						if (verbose)
1110 							(void) printf("%s\n",
1111 							    gettext("Unreach"
1112 							    "able"));
1113 						continue;
1114 					} else if (mdiserror(ep,
1115 					    MDE_MED_ERROR)) {
1116 						if (verbose)
1117 							(void) printf("%s\n",
1118 							    gettext("Bad"));
1119 					} else {
1120 						if (verbose)
1121 							(void) printf("%s\n",
1122 							    gettext("Fatal"));
1123 					}
1124 					mde_perror(ep, "");
1125 					if (mdiserror(ep, MDE_MED_ERROR))
1126 						continue;
1127 					goto out;
1128 				}
1129 			} else {
1130 				/*
1131 				 * Make sure this node has the correct value
1132 				 * for the mediator record. If not we want the
1133 				 * highest value from the other nodes. Save it
1134 				 * for updating once the loop through all the
1135 				 * mediator nodes has completed.
1136 				 */
1137 				if (sname == NULL) {
1138 					if (latest_med_dat_cc <
1139 					    medd.med_dat_cc) {
1140 						latest_med_dat_cc =
1141 						    medd.med_dat_cc;
1142 						(void) memcpy(&save_medd, &medd,
1143 						    sizeof (medd));
1144 					}
1145 				} else {
1146 					if (verbose)
1147 						(void) printf("%s",
1148 						    gettext("Ok"));
1149 					if (medd.med_dat_fl & MED_DFL_GOLDEN) {
1150 						if (verbose)
1151 							(void) printf("\t%s",
1152 							    gettext("Yes"));
1153 						golden++;
1154 					} else {
1155 						if (verbose)
1156 							(void) printf("\t%s",
1157 							    gettext("No"));
1158 					}
1159 					if (verbose)
1160 						(void) printf("\n");
1161 						medok++;
1162 				}
1163 			}
1164 		}
1165 		if (sname == NULL) {
1166 
1167 			/*
1168 			 * Mediators only become active when there are
1169 			 * replica updates to the sets and this can only
1170 			 * occur when there is a disk in the set.
1171 			 * If there are no disks found then the save_medd
1172 			 * structure will be empty. If save_medd is empty,
1173 			 * do not update the set.
1174 			 */
1175 			if (save_medd.med_dat_sn == 0)
1176 				continue;
1177 			/*
1178 			 * Update the latest mediator information
1179 			 * on this node
1180 			 */
1181 			(void) strlcpy(mdh.a_nm[0], mynode(),
1182 			    sizeof (mdh.a_nm[0]));
1183 			mdh.a_cnt = 1;
1184 			if (clnt_user_med_upd_data(&mdh, obandiskset,
1185 			    setname, setnum, &save_medd, ep) == -1) {
1186 				/*
1187 				 * We had some errors while updaing the
1188 				 * record. This means this metaset is
1189 				 * not updated with latest mediator
1190 				 * information.
1191 				 */
1192 				mde_perror(ep, "");
1193 			}
1194 
1195 		} else {
1196 			if (golden) {
1197 				retval = 0;
1198 				goto out;
1199 			}
1200 			if (medok < ((medrecp->med_rec_meds.n_cnt / 2) + 1))
1201 				retval = 1;
1202 		}
1203 	}
1204 
1205 out:
1206 	if ((sname != NULL) && (isSetFound == 0)) {
1207 		(void) mderror(ep, MDE_NO_SET, sname);
1208 		mde_perror(ep, "");
1209 		retval = 1;
1210 	}
1211 	if (rec_buf != NULL)
1212 		Free(rec_buf);
1213 	if (close(fd) < 0) {
1214 		(void) mdsyserror(ep, errno, MED_DB_FILE);
1215 		mde_perror(ep, dgettext(TEXT_DOMAIN,
1216 		    "Error in closing meddb file"));
1217 		return (1);
1218 	}
1219 	return (retval);
1220 }
1221