xref: /titanic_41/usr/src/cmd/avs/rdc/sndrboot.c (revision c9a6ea2e938727c95af7108c5e00eee4c890c7ae)
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 2010 Sun Microsystems, Inc.  All rights reserved.
24  * Use is subject to license terms.
25  */
26 
27 
28 #include <sys/types.h>
29 #include <sys/wait.h>
30 #include <stdio.h>
31 #include <sys/mnttab.h>
32 #include <errno.h>
33 #include <limits.h>
34 #include <fcntl.h>
35 #include <strings.h>
36 #include <stdlib.h>
37 #include <unistd.h>
38 
39 #include <locale.h>
40 #include <langinfo.h>
41 #include <libintl.h>
42 #include <stdarg.h>
43 #include <netdb.h>
44 #include <ctype.h>
45 #include <sys/utsname.h>
46 
47 #include <sys/nsctl/rdc_io.h>
48 #include <sys/nsctl/rdc_ioctl.h>
49 #include <sys/nsctl/rdc_prot.h>
50 
51 #include <sys/nsctl/cfg.h>
52 #include <sys/nsctl/cfg_cluster.h>
53 
54 #include <sys/unistat/spcs_s.h>
55 #include <sys/unistat/spcs_s_u.h>
56 #include <sys/unistat/spcs_errors.h>
57 
58 #include "rdcadm.h"
59 
60 /*
61  * Special re-use of sndrboot to fix SNDR set IDs during post-patch processing
62  */
63 #define	RDC_CMD_FIXSETIDS	0xFEEDFACE
64 
65 /*
66  * config file user level Dual copy pair structure
67  */
68 typedef struct _sd_dual_pair {
69 	char fhost[MAX_RDC_HOST_SIZE];	/* Hostname for primary device */
70 	char fnetaddr[RDC_MAXADDR];	/* Host netaddr for primary device */
71 	char ffile[NSC_MAXPATH];	/* Primary device */
72 	char fbitmap[NSC_MAXPATH];	/* Primary bitmap device */
73 	char thost[MAX_RDC_HOST_SIZE];	/* Hostname for secondary device */
74 	char tnetaddr[RDC_MAXADDR];	/* Host netaddr for secondary device */
75 	char tfile[NSC_MAXPATH];	/* Secondary device */
76 	char tbitmap[NSC_MAXPATH];	/* Secondary bitmap device */
77 	char directfile[NSC_MAXPATH];	/* Local FCAL direct IO volume */
78 	char diskqueue[NSC_MAXPATH];	/* Disk Queue volume */
79 	char group[NSC_MAXPATH];	/* Group name */
80 	char lhost[MAX_RDC_HOST_SIZE];  /* Logical hostname for cluster */
81 	int  doasync;			/* Device is in sync/async mode */
82 	int  setid;			/* unique setid of this set */
83 } _sd_dual_pair_t;
84 
85 #include <sys/socket.h>
86 #include <netinet/in.h>
87 #include <arpa/inet.h>
88 #include <netinet/tcp.h>
89 #include <rpc/rpc_com.h>
90 #include <rpc/rpc.h>
91 
92 #include <sys/nsctl/librdc.h>
93 
94 char *ctag = NULL;
95 
96 int parseopts(int, char **, int *);
97 static int rdc_operation(char *, char *, char *, char *, char *, char *, int,
98     char *, char *, char *, int, char *, int setid);
99 static int read_libcfg(int);
100 static void usage(void);
101 
102 extern char *basename(char *);
103 
104 int rdc_maxsets;
105 static _sd_dual_pair_t *pair_list;
106 char *program;
107 
108 struct netbuf svaddr;
109 struct netbuf *svp;
110 struct netconfig nconf;
111 struct netconfig *conf;
112 struct knetconfig knconf;
113 static int clustered = 0;
114 static int proto_test = 0;
115 
116 #ifdef lint
117 int
118 sndrboot_lintmain(int argc, char *argv[])
119 #else
120 int
121 main(int argc, char *argv[])
122 #endif
123 {
124 	char fromhost[MAX_RDC_HOST_SIZE];
125 	char tohost[MAX_RDC_HOST_SIZE];
126 	char fromfile[NSC_MAXPATH];
127 	char tofile[NSC_MAXPATH];
128 	char frombitmap[NSC_MAXPATH];
129 	char tobitmap[NSC_MAXPATH];
130 	char directfile[NSC_MAXPATH];
131 	char diskqueue[NSC_MAXPATH];
132 	char group[NSC_MAXPATH];
133 	char lhost[MAX_RDC_HOST_SIZE];
134 	int pairs;
135 	int pid;
136 	int flag = 0;
137 	int doasync;
138 	int rc;
139 	char *required;
140 	int setid;
141 
142 	(void) setlocale(LC_ALL, "");
143 	(void) textdomain("rdc");
144 
145 	program = basename(argv[0]);
146 
147 	rc = rdc_check_release(&required);
148 	if (rc < 0) {
149 		rdc_err(NULL,
150 		    gettext("unable to determine the current "
151 		    "Solaris release: %s\n"), strerror(errno));
152 	} else if (rc == FALSE) {
153 		rdc_err(NULL,
154 		    gettext("incorrect Solaris release (requires %s)\n"),
155 		    required);
156 	}
157 
158 	rdc_maxsets = rdc_get_maxsets();
159 	if (rdc_maxsets == -1) {
160 		spcs_log("sndr", NULL,
161 		    gettext("%s unable to get maxsets value from kernel"),
162 		    program);
163 
164 		rdc_err(NULL,
165 		    gettext("unable to get maxsets value from kernel"));
166 	}
167 
168 	pair_list = calloc(rdc_maxsets, sizeof (*pair_list));
169 	if (pair_list == NULL) {
170 		rdc_err(NULL,
171 		    gettext(
172 			"unable to allocate pair_list"
173 			" array for %d sets"),
174 			rdc_maxsets);
175 	}
176 
177 	if (parseopts(argc, argv, &flag))
178 		return (1);
179 	pairs = read_libcfg(flag);
180 
181 	if (flag == RDC_CMD_FIXSETIDS) {
182 		if (pairs) {
183 			spcs_log("sndr", NULL, gettext("Fixed %d Remote Mirror"
184 				    " set IDs"), pairs);
185 #ifdef DEBUG
186 			rdc_warn(NULL, gettext("Fixed %d Remote Mirror set "
187 				    "IDs"), pairs);
188 #endif
189 		}
190 		return (0);
191 	}
192 
193 	if (pairs == 0) {
194 #ifdef DEBUG
195 		rdc_err(NULL,
196 		    gettext("Config contains no dual copy sets"));
197 #else
198 		return (0);
199 #endif
200 	}
201 
202 	while (pairs--) {
203 		pid = fork();
204 		if (pid == -1) {		/* error forking */
205 			perror("fork");
206 			continue;
207 		}
208 
209 		if (pid > 0)		/* this is parent process */
210 			continue;
211 
212 /*
213  * At this point, this is the child process.  Do the operation
214  */
215 
216 		(void) strncpy(fromfile,
217 		    pair_list[pairs].ffile, NSC_MAXPATH);
218 		(void) strncpy(tofile,
219 		    pair_list[pairs].tfile, NSC_MAXPATH);
220 		(void) strncpy(frombitmap,
221 		    pair_list[pairs].fbitmap, NSC_MAXPATH);
222 		(void) strncpy(fromhost,
223 		    pair_list[pairs].fhost, MAX_RDC_HOST_SIZE);
224 		(void) strncpy(tohost,
225 		    pair_list[pairs].thost, MAX_RDC_HOST_SIZE);
226 		(void) strncpy(tobitmap,
227 		    pair_list[pairs].tbitmap, NSC_MAXPATH);
228 		(void) strncpy(directfile,
229 		    pair_list[pairs].directfile, NSC_MAXPATH);
230 		(void) strncpy(diskqueue,
231 		    pair_list[pairs].diskqueue, NSC_MAXPATH);
232 		(void) strncpy(group,
233 		    pair_list[pairs].group, NSC_MAXPATH);
234 		(void) strncpy(lhost,
235 		    pair_list[pairs].lhost, MAX_RDC_HOST_SIZE);
236 
237 		doasync = pair_list[pairs].doasync;
238 		setid = pair_list[pairs].setid;
239 		if (rdc_operation(fromhost, fromfile, frombitmap,
240 		    tohost, tofile, tobitmap, flag, directfile, group,
241 		    diskqueue, doasync, lhost, setid)
242 		    < 0) {
243 			exit(255);
244 		}
245 
246 		exit(0);
247 	}
248 
249 	while ((wait((int *)0) > 0))
250 		;
251 	return (0);
252 }
253 
254 static int
255 rdc_operation(fromhost, fromfile, frombitmap, tohost, tofile,
256     tobitmap, flag, directfile, group, diskqueue, doasync,
257     lhost, setid)
258 char *fromhost, *fromfile, *frombitmap;
259 char *tohost, *tofile, *tobitmap;
260 int flag, doasync;
261 char *directfile;
262 char *group, *diskqueue;
263 int setid;
264 char *lhost;
265 {
266 	const int getaddr = (flag == RDC_CMD_RESUME);
267 	const int rpcbind = !getaddr;
268 	rdc_config_t parms;
269 	int ret;
270 	spcs_s_info_t ustatus;
271 	struct hostent *hp;
272 	char fromname[MAXHOSTNAMELEN], toname[MAXHOSTNAMELEN];
273 	struct t_info tinfo;
274 	int i;
275 
276 	conf = &nconf;
277 	bzero(&fromname, MAXHOSTNAMELEN);
278 	bzero(&toname, MAXHOSTNAMELEN);
279 
280 	hp = gethost_byname(fromhost);
281 	if (hp == NULL) {
282 		spcs_log("sndr", NULL,
283 		    gettext("%s gethost_byname failed for %s"),
284 		    program, fromhost);
285 	}
286 	if (strcmp(hp->h_name, fromhost) == 0)
287 		(void) strncpy(fromname, hp->h_name, MAXHOSTNAMELEN);
288 	else {
289 	for (i = 0; hp->h_aliases[i] != NULL; i++) {
290 		if (strcmp(hp->h_aliases[i], fromhost) == 0)
291 			(void) strncpy(fromname, hp->h_aliases[i],
292 			    MAXHOSTNAMELEN);
293 		}
294 	}
295 	if (fromname[0] == '\0') {
296 		spcs_log("sndr", NULL,
297 		    gettext("%s host %s is not local"),
298 		    program, fromhost);
299 		rdc_err(NULL, gettext("Host %s is not local"),
300 		    fromhost);
301 	}
302 	hp = gethost_byname(tohost);
303 	if (hp == NULL) {
304 		spcs_log("sndr", NULL,
305 		    gettext("%s gethost_byname failed for %s"),
306 		    program, tohost);
307 	}
308 	if (strcmp(hp->h_name, tohost) == 0)
309 		(void) strncpy(toname, hp->h_name, MAXHOSTNAMELEN);
310 	else {
311 	for (i = 0; hp->h_aliases[i] != NULL; i++) {
312 		if (strcmp(hp->h_aliases[i], tohost) == 0)
313 			(void) strncpy(toname, hp->h_aliases[i],
314 			    MAXHOSTNAMELEN);
315 		}
316 	}
317 	if (toname[0] == '\0') {
318 		spcs_log("sndr", NULL,
319 		    gettext("%s host %s is not local"),
320 		    program, tohost);
321 		rdc_err(NULL, gettext("Host %s is not local"),
322 		    tohost);
323 	}
324 
325 	if (self_check(fromname) && self_check(toname)) {
326 		spcs_log("sndr", NULL,
327 		    gettext("%s Both %s and %s are local"),
328 		    program, fromhost, tohost);
329 		rdc_err(NULL, gettext("Both %s and %s are local"),
330 		    fromhost, tohost);
331 	}
332 
333 	/*
334 	 * Now build up the address for each host including port and transport
335 	 */
336 	if (getaddr) {
337 		svp = get_addr(toname, RDC_PROGRAM, RDC_VERS_MIN,
338 			&conf, proto_test?NC_UDP: NULL, "rdc", &tinfo, rpcbind);
339 
340 		if (svp == NULL) {
341 #ifdef DEBUG
342 			(void) printf("get_addr failed for Ver 4 %s\n", toname);
343 #endif
344 			spcs_log("sndr", NULL,
345 			    gettext("%s get_addr failed for Ver 4"),
346 			    program);
347 			return (-1);
348 		}
349 		svaddr = *svp;
350 	} else {
351 		bzero(&svaddr, sizeof (svaddr));
352 	}
353 
354 	parms.rdc_set->secondary.addr.len = svaddr.len;
355 	parms.rdc_set->secondary.addr.maxlen = svaddr.maxlen;
356 	parms.rdc_set->secondary.addr.buf = (void *)svaddr.buf;
357 
358 #ifdef DEBUG_ADDR
359 	(void) fprintf(stderr,
360 		"secondary buf %x len %d\n", svaddr.buf, svaddr.len);
361 
362 	for (i = 0; i < svaddr.len; i++)
363 		(void) printf("%u ", svaddr.buf[i]);
364 	(void) printf("\n");
365 #endif
366 
367 	if (getaddr) {
368 		svp = get_addr(fromname, RDC_PROGRAM, RDC_VERS_MIN,
369 			&conf, proto_test?NC_UDP: NULL, "rdc", &tinfo, rpcbind);
370 		if (svp == NULL) {
371 #ifdef DEBUG
372 			(void) printf("get_addr failed for Ver 4 %s\n",
373 			    fromname);
374 #endif
375 			return (-1);
376 		}
377 		svaddr = *svp;
378 	} else {
379 		;
380 		/*EMPTY*/
381 	}
382 	parms.rdc_set->primary.addr.len = svaddr.len;
383 	parms.rdc_set->primary.addr.maxlen = svaddr.maxlen;
384 	parms.rdc_set->primary.addr.buf =
385 				(void *)svaddr.buf;
386 
387 #ifdef DEBUG_ADDR
388 	(void) fprintf(stderr, "primary buf %x len %d\n",
389 	    svaddr.buf, svaddr.len);
390 	for (i = 0; i < svaddr.len; i++)
391 		(void) printf("%u ", svaddr.buf[i]);
392 	(void) printf("\n");
393 #endif
394 
395 	if (getaddr) {
396 		(void) convert_nconf_to_knconf(conf, &knconf);
397 #ifdef DEBUG_ADDR
398 		(void) printf("knconf %x %s %s %x\n", knconf.knc_semantics,
399 		    knconf.knc_protofmly, knconf.knc_proto, knconf.knc_rdev);
400 #endif
401 		parms.rdc_set->netconfig = &knconf;
402 	} else {
403 		parms.rdc_set->netconfig = NULL;
404 	}
405 	if (!clustered && !self_check(fromname) && !self_check(toname)) {
406 		spcs_log("sndr", NULL,
407 		    gettext("%s Neither %s nor %s is local"),
408 		    program, fromhost, tohost);
409 		rdc_err(NULL, gettext("Neither %s nor %s is local"),
410 		    fromhost, tohost);
411 	}
412 	(void) strncpy(parms.rdc_set->primary.intf, fromhost,
413 	    MAX_RDC_HOST_SIZE);
414 	(void) strncpy(parms.rdc_set->primary.file, fromfile, NSC_MAXPATH);
415 	(void) strncpy(parms.rdc_set->primary.bitmap, frombitmap, NSC_MAXPATH);
416 
417 	(void) strncpy(parms.rdc_set->secondary.intf, tohost,
418 	    MAX_RDC_HOST_SIZE);
419 	(void) strncpy(parms.rdc_set->secondary.file, tofile, NSC_MAXPATH);
420 	(void) strncpy(parms.rdc_set->secondary.bitmap, tobitmap, NSC_MAXPATH);
421 
422 	(void) strncpy(parms.rdc_set->group_name, group, NSC_MAXPATH);
423 	(void) strncpy(parms.rdc_set->disk_queue, diskqueue, NSC_MAXPATH);
424 
425 	parms.rdc_set->maxqfbas = maxqfbas;
426 	parms.rdc_set->maxqitems = maxqitems;
427 	parms.rdc_set->autosync = autosync;
428 	parms.rdc_set->asyncthr = asyncthr;
429 	parms.rdc_set->setid = setid;
430 
431 	/* gethostid(3c) is defined to return a 32bit value */
432 	parms.rdc_set->syshostid = (int32_t)gethostid();
433 
434 	parms.command = 0;
435 	parms.options = 0;
436 	parms.command = flag;
437 
438 	if (flag == RDC_CMD_RESUME) {
439 		if (doasync)
440 			parms.options |= RDC_OPT_ASYNC;
441 		else
442 			parms.options |= RDC_OPT_SYNC;
443 	}
444 	if (clustered) {
445 		if (!ctag)
446 			goto noconfig;
447 		if (strcmp(ctag, "-") == 0)
448 			goto noconfig;
449 
450 #ifdef DEBUG
451 		(void) fprintf(stderr, "logical hostname: %s\n", lhost);
452 #endif
453 
454 		if (strcmp(lhost, fromname) == 0) {
455 			parms.options |= RDC_OPT_PRIMARY;
456 			(void) strncpy(parms.rdc_set->direct_file, directfile,
457 			    NSC_MAXPATH);
458 
459 		} else {
460 			parms.options |= RDC_OPT_SECONDARY;
461 			parms.rdc_set->direct_file[0] = 0; /* no fcal direct */
462 		}
463 	} else {
464 noconfig:
465 		/*
466 		 * If not clustered, don't resume sndr sets with lhost
467 		 */
468 		if ((flag == RDC_CMD_RESUME) && lhost && strlen(lhost))
469 			return (0);
470 
471 		if (self_check(fromname)) {
472 			parms.options |= RDC_OPT_PRIMARY;
473 			(void) strncpy(parms.rdc_set->direct_file, directfile,
474 			    NSC_MAXPATH);
475 		} else {
476 			parms.options |= RDC_OPT_SECONDARY;
477 			parms.rdc_set->direct_file[0] = 0; /* no fcal direct */
478 		}
479 	}
480 
481 	ustatus = spcs_s_ucreate();
482 
483 	errno = 0;
484 	ret = RDC_IOCTL(RDC_CONFIG, &parms, NULL, 0, 0, 0, ustatus);
485 	if (ret != SPCS_S_OK) {
486 
487 		/* Surpress error messages for suspend on cluster elements */
488 		if ((flag == RDC_CMD_SUSPEND) && (errno == RDC_EALREADY) &&
489 			!clustered && lhost && strlen(lhost)) {
490 			spcs_s_ufree(&ustatus);
491 			return (0);
492 		}
493 
494 		(void) fprintf(stderr,
495 			gettext("Remote Mirror: %s %s %s %s %s %s\n"),
496 			fromhost, fromfile,
497 			frombitmap, tohost, tofile, tobitmap);
498 
499 		if (errno == RDC_EEINVAL) {
500 			spcs_log("sndr", NULL,
501 			    gettext("%s %s %s %s %s %s %s %s\n%s"),
502 			    program, rdc_decode_flag(flag, parms.options),
503 			    fromhost, fromfile, frombitmap,
504 			    tohost, tofile, tobitmap,
505 			    gettext("invalid command option"));
506 			rdc_err(&ustatus,
507 			    gettext("Remote Mirror: invalid command option "
508 				    "'%s'"), rdc_decode_flag(flag,
509 				    parms.options));
510 		} else {
511 			spcs_log("sndr", &ustatus,
512 			    gettext("%s %s %s %s %s %s %s %s"),
513 			    program, rdc_decode_flag(flag, parms.options),
514 			    fromhost, fromfile, frombitmap,
515 			    tohost, tofile, tobitmap);
516 			rdc_err(&ustatus, 0);
517 		}
518 	}
519 
520 	spcs_log("sndr", NULL,
521 	    gettext("%s %s %s %s %s %s %s %s\nSuccessful"),
522 	    program, rdc_decode_flag(flag, parms.options),
523 	    fromhost, fromfile, frombitmap, tohost, tofile, tobitmap);
524 
525 	spcs_s_ufree(&ustatus);
526 	return (0);
527 }
528 /*
529  * assign setid's to any existing
530  * sets without setids, making sure of course NOT to re-use a setid
531  */
532 int
533 update_setids(CFGFILE *cfg, int *no_id, int highest)
534 {
535 	int setid;
536 	char buf[CFG_MAX_BUF];
537 	char key[CFG_MAX_KEY];
538 	char *ctag;
539 
540 	/* If in a Sun Cluster, SetIDs need to have a ctag */
541 	if ((ctag = cfg_get_resource(cfg)) != NULL) {
542 		ctag = strdup(ctag);
543 		cfg_resource(cfg, "setid-ctag");
544 	}
545 
546 	/*
547 	 * Paranoia. IF there are any sets with setids, we don't
548 	 * want to re-use their number.
549 	 */
550 	if (highest > get_new_cfg_setid(cfg)) {
551 		bzero(&buf, sizeof (buf));
552 		(void) sprintf(buf, "%d", highest);
553 		if (cfg_put_cstring(cfg, "setid.set1.value", buf,
554 		    sizeof (buf)) < 0)
555 			rdc_warn(NULL, gettext("sndrboot: Unable to store "
556 			    "new setid"));
557 	}
558 
559 	for (setid = 0; no_id[setid]; setid++) {
560 		bzero(&buf, sizeof (buf));
561 		bzero(&key, sizeof (key));
562 		(void) sprintf(buf, "%d", get_new_cfg_setid(cfg));
563 		(void) sprintf(key, "sndr.set%d.options", no_id[setid]);
564 		if (cfg_put_options(cfg, CFG_SEC_CONF, key, "setid", buf) < 0)
565 			rdc_warn(NULL, gettext("sndrboot: Unable to store "
566 			    "unique setid"));
567 
568 		pair_list[no_id[setid] - 1].setid = atoi(buf);
569 	}
570 
571 	/* Restore old ctag if in a Sun Cluster */
572 	if (ctag) {
573 		cfg_resource(cfg, ctag);
574 		free(ctag);
575 	}
576 
577 	if (cfg_commit(cfg) < 0)
578 		rdc_err(NULL, gettext("sndrboot: Failed to commit setids"));
579 
580 	return (setid);
581 }
582 
583 /*
584  * this is called when the option lghn is no available in libdscfg
585  * that should only happen on an upgrade.
586  * cfg write lock must be held across this function
587  */
588 char *
589 get_lghn(CFGFILE *cfg, char *ctag, int setnum, int flag)
590 {
591 	FILE *pipe;
592 	char rsgrp[SCCONF_MAXSTRINGLEN];
593 	char cmd[SCCONF_MAXSTRINGLEN];
594 	static char lhostname[MAX_RDC_HOST_SIZE];
595 	char key[CFG_MAX_KEY];
596 	int rc;
597 
598 	if (ctag == NULL)
599 		goto fail;
600 
601 	bzero(&lhostname, sizeof (lhostname));
602 
603 	(void) sprintf(rsgrp, "%s-stor-rg", ctag);
604 /* BEGIN CSTYLED */
605 	rc = snprintf(cmd, SCCONF_MAXSTRINGLEN,
606 	    "/usr/cluster/bin/scrgadm -pvv | fgrep HostnameList \
607 | fgrep %s | fgrep value | awk -F: '{ print $4 }'", rsgrp);
608 /* END CSTYLED */
609 
610 	if (rc < 0) {
611 		rdc_err(NULL, gettext("Error getting scrgadm output"));
612 	}
613 
614 	pipe = popen(cmd, "r");
615 
616 	if (pipe == NULL) {
617 		rdc_err(NULL, gettext("Error opening pipe"));
618 	}
619 	rc = fscanf(pipe, "%s", lhostname);
620 	(void) pclose(pipe);
621 
622 	if (rc != 1) {
623 		rdc_err(NULL, gettext("Unable to get logical host"));
624 	}
625 
626 	/* not really failing, but suspend does not have the config lock */
627 	if (flag == RDC_CMD_SUSPEND)
628 		goto fail;
629 
630 	bzero(&key, sizeof (key));
631 	(void) snprintf(key, sizeof (key), "sndr.set%d.options", setnum);
632 	if (cfg_put_options(cfg, CFG_SEC_CONF, key, "lghn", lhostname) < 0)
633 		rdc_warn(NULL, gettext("sndrboot: Unable to store logical "
634 		    "host name in configuration database"));
635 
636 	if (cfg_commit(cfg) < 0)
637 		rdc_err(NULL,
638 		    gettext("sndrboot: Failed to commit logical host name"));
639 
640 fail:
641 	return (lhostname);
642 
643 }
644 
645 /*
646  * read_libcfg()
647  *
648  * DESCRIPTION: Read the relevant config info via libcfg
649  *
650  * Outputs:
651  *	int i			Number of pairs of devices
652  *
653  * Side Effects: The 0 to i-1 entries in the pair_list are filled.
654  *
655  */
656 static int
657 read_libcfg(int flag)
658 {
659 	char fromhost[MAX_RDC_HOST_SIZE];
660 	char fromfile[NSC_MAXPATH];
661 	char frombitmap[NSC_MAXPATH];
662 	char tohost[MAX_RDC_HOST_SIZE];
663 	char tofile[NSC_MAXPATH];
664 	char tobitmap[NSC_MAXPATH];
665 	char directfile[NSC_MAXPATH];
666 	char diskqueue[NSC_MAXPATH];
667 	char group[NSC_MAXPATH];
668 	char lhost[MAX_RDC_HOST_SIZE];
669 	char sync[16];
670 	char setid[64];
671 	int doasync;
672 	CFGFILE *cfg;
673 	int i, j = 0;
674 	int rc;
675 	char buf[CFG_MAX_BUF];
676 	char key[CFG_MAX_KEY];
677 	char dummy[NSC_MAXPATH];
678 	int setnumber;
679 	int numsets;
680 	int highest = 0;
681 	char lghn[5];
682 	int *no_id;
683 
684 
685 	if ((cfg = cfg_open("")) == NULL)
686 		rdc_err(NULL, gettext("Error opening config"));
687 
688 	/*
689 	 * If RDC_CMD_FIXSETIDS, we were called during post-patch install
690 	 * Acquire a write-lock on the cfg_lock(), so the code can attempt
691 	 * to fix setIDs
692 	 */
693 	if (flag == RDC_CMD_FIXSETIDS) {
694 		if (!cfg_lock(cfg, CFG_WRLOCK))
695 			rdc_err(NULL, gettext("Error write locking config"));
696 		cfg_resource(cfg, NULL);
697 	} else {
698 		if (!cfg_lock(cfg, CFG_RDLOCK))
699 			rdc_err(NULL, gettext("Error locking config"));
700 		cfg_resource(cfg, ctag);
701 	}
702 
703 	if ((numsets = cfg_get_num_entries(cfg, "sndr")) < 0)
704 		rdc_err(NULL, gettext("Unable to get set info from config"));
705 
706 	no_id = (int *)calloc(numsets + 1, sizeof (int));
707 	if (!no_id)
708 		rdc_err(NULL, gettext("No memory"));
709 
710 
711 	(void) snprintf(lghn, sizeof (lghn), "lghn");
712 
713 	for (i = 0; i < rdc_maxsets; i++) {
714 		setnumber = i + 1;
715 
716 		bzero(buf, CFG_MAX_BUF);
717 		(void) snprintf(key, sizeof (key), "sndr.set%d", setnumber);
718 		if (cfg_get_cstring(cfg, key, buf, CFG_MAX_BUF) < 0)
719 			break;
720 
721 		rc = sscanf(buf, "%s %s %s %s %s %s %s %s %s %s %s %s",
722 		    fromhost, fromfile, frombitmap, tohost, tofile, tobitmap,
723 			directfile, sync, group, dummy, dummy, diskqueue);
724 		if (rc != 12)
725 			rdc_err(NULL, gettext("cfg input error (%d)"), rc);
726 
727 		if (strcmp(directfile, "ip") == 0)
728 			(void) strcpy(directfile, "");
729 
730 		if (strcmp(group, "-") == 0)
731 			(void) strcpy(group, "");
732 
733 		if (strcmp(diskqueue, "-") == 0)
734 			(void) strcpy(diskqueue, "");
735 
736 		(void) snprintf(key, sizeof (key),
737 			"sndr.set%d.options", setnumber);
738 
739 		if (cfg_get_single_option(cfg, CFG_SEC_CONF, key,
740 			lghn, lhost, MAX_RDC_HOST_SIZE) < 0)
741 			(void) strcpy(lhost,
742 			    get_lghn(cfg, ctag, setnumber, flag));
743 
744 		if (strcmp(sync, "sync") == 0)
745 			doasync = 0;
746 		else if (strcmp(sync, "async") == 0)
747 			doasync = 1;
748 		else {
749 			cfg_close(cfg);
750 			rdc_err(NULL,
751 			    gettext("Set %s:%s neither sync nor async"),
752 			    tohost, tofile);
753 		}
754 
755 		(void) strncpy(pair_list[i].fhost, fromhost,
756 		    MAX_RDC_HOST_SIZE);
757 		(void) strncpy(pair_list[i].ffile, fromfile, NSC_MAXPATH);
758 		(void) strncpy(pair_list[i].fbitmap, frombitmap, NSC_MAXPATH);
759 		(void) strncpy(pair_list[i].thost, tohost, MAX_RDC_HOST_SIZE);
760 		(void) strncpy(pair_list[i].tfile, tofile, NSC_MAXPATH);
761 		(void) strncpy(pair_list[i].tbitmap, tobitmap, NSC_MAXPATH);
762 		(void) strncpy(pair_list[i].directfile, directfile,
763 		    NSC_MAXPATH);
764 		(void) strncpy(pair_list[i].diskqueue, diskqueue,
765 		    NSC_MAXPATH);
766 		(void) strncpy(pair_list[i].group, group, NSC_MAXPATH);
767 		(void) strncpy(pair_list[i].lhost, lhost, MAX_RDC_HOST_SIZE);
768 		pair_list[i].doasync = doasync;
769 
770 		if (cfg_get_single_option(cfg, CFG_SEC_CONF, key, "setid",
771 		    setid, sizeof (setid)) < 0) {
772 			no_id[j++] = setnumber;
773 		}
774 		pair_list[i].setid = atoi(setid);
775 
776 		if (pair_list[i].setid > highest)
777 			highest = pair_list[i].setid;
778 
779 		if (gethost_netaddrs(fromhost, tohost,
780 		    (char *)pair_list[i].fnetaddr,
781 		    (char *)pair_list[i].tnetaddr) < 0) {
782 			cfg_close(cfg);
783 			spcs_log("sndr", NULL,
784 			    gettext("%s unable to determine IP addresses "
785 			    "for hosts %s %s"), program, fromhost, tohost);
786 			rdc_err(NULL, gettext("unable to determine IP "
787 			    "addresses for hosts %s, %s"), fromhost, tohost);
788 		}
789 	}
790 	/*
791 	 * fix any lost set ids if possible, also deal with upgrade
792 	 */
793 	if (j > 0 && flag == RDC_CMD_FIXSETIDS) {
794 		(void) update_setids(cfg, no_id, highest);
795 		i = j;	/* Set number of fixups */
796 	}
797 	free(no_id);
798 	cfg_close(cfg);
799 	return (i);
800 }
801 
802 
803 int
804 parseopts(argc, argv, flag)
805 int argc;
806 char **argv;
807 int *flag;
808 {
809 	int  errflag = 0;
810 	char c;
811 	char inval = 0;
812 #ifdef DEBUG
813 	while ((c = getopt(argc, argv, "C:Urs")) != -1) {
814 #else
815 	while ((c = getopt(argc, argv, "C:rs")) != -1) {
816 #endif
817 		switch (c) {
818 		case 'C':
819 			clustered = TRUE;
820 			ctag = optarg;
821 			break;
822 #ifdef DEBUG
823 		case 'U':
824 			proto_test = 1;
825 			break;
826 #endif
827 		case 'r':
828 			if (*flag)
829 				inval = 1;
830 			*flag = RDC_CMD_RESUME;
831 			break;
832 		case 's':
833 			if (*flag)
834 				inval = 1;
835 			*flag = RDC_CMD_SUSPEND;
836 			break;
837 		case '?':
838 			errflag++;
839 		}
840 	}
841 
842 	/*
843 	 * Special fix to address no SetIds in AVS 3.1 to 3.2 install + patch
844 	 * Adjust set IDs, if someone invokes the following invalid command
845 	 *
846 	 *	/use/sbin/sndrboot -C post-patch-setids -r -s
847 	 *
848 	 * Command will be called in post-install of the patch containing fix
849 	 *
850 	 */
851 	if (clustered && (strcmp(ctag, "post-patch-setids") == 0) &&
852 	    *flag && inval) {
853 		*flag = RDC_CMD_FIXSETIDS;
854 		return (0);
855 	}
856 
857 	if (inval) {
858 		rdc_warn(NULL, gettext("Invalid argument combination"));
859 		errflag = 1;
860 	}
861 
862 	if (!*flag || errflag) {
863 		usage();
864 		return (-1);
865 	}
866 
867 	return (0);
868 }
869 
870 static void
871 usage()
872 {
873 	(void) fprintf(stderr, gettext("usage:\n"));
874 	(void) fprintf(stderr,
875 		gettext("\t%s -r [-C tag]\t\t"
876 			"resume\n"), program);
877 
878 	(void) fprintf(stderr,
879 		gettext("\t%s -s [-C tag]\t\t"
880 			"suspend\n"), program);
881 }
882