xref: /titanic_50/usr/src/cmd/avs/dsw/iicpbmp.c (revision 3c112a2b34403220c06c3e2fcac403358cfba168)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 /*
22  * Copyright 2010 Sun Microsystems, Inc.  All rights reserved.
23  * Use is subject to license terms.
24  */
25 
26 #include <sys/types.h>
27 #include <sys/time.h>
28 #include <errno.h>
29 #include <signal.h>
30 #include <stdio.h>
31 #include <string.h>
32 #include <fcntl.h>
33 #include <stdlib.h>
34 #include <unistd.h>
35 #include <values.h>
36 #include <locale.h>
37 #include <sys/stat.h>
38 #include <strings.h>
39 #include <stdarg.h>
40 #include <sys/param.h>
41 #include <nsctl.h>
42 
43 #include <sys/nsctl/cfg.h>
44 #include <sys/unistat/spcs_s.h>
45 #include <sys/unistat/spcs_s_u.h>
46 #include <sys/unistat/spcs_errors.h>
47 #include <sys/nsctl/dsw.h>
48 #include <sys/nsctl/dsw_dev.h>
49 
50 #define	DSW_TEXT_DOMAIN	"II"
51 
52 void iicpbmp_usage();
53 void copybmp(char *, char *);
54 int find_bitmap_cfg(char *);
55 
56 extern int optind;
57 
58 char	*cmdnam;
59 
60 extern char *optarg;
61 extern int optind, opterr, optopt;
62 int	update_cfg = 1;
63 CFGFILE *cfg;
64 char shadow[DSW_NAMELEN];
65 char buf[CFG_MAX_BUF];
66 char key[CFG_MAX_KEY];
67 int setnumber;
68 
69 #ifdef lint
70 int
71 iicpbmp_lintmain(int argc, char *argv[])
72 #else
73 int
74 main(int argc, char *argv[])
75 #endif
76 {
77 	cmdnam = argv[0];
78 
79 	if (argc > 1) {
80 		if (strcmp(argv[1], "-c") == 0) {
81 			/* don't update cfg information */
82 			update_cfg = 0;
83 			argc--;
84 			argv++;
85 		}
86 	}
87 
88 	if (argc == 1 || (argc%2) == 0)	/* must have pairs of filenames */
89 		iicpbmp_usage();
90 
91 	if (update_cfg) {
92 		if ((cfg = cfg_open(NULL)) == NULL) {
93 			(void) fprintf(stderr,
94 			    gettext("Error opening config\n"));
95 			exit(1);
96 		}
97 
98 		if (!cfg_lock(cfg, CFG_WRLOCK)) {
99 			spcs_log("ii", NULL,
100 			    "iicpbmp CFG_WRLOCK failed, errno %d", errno);
101 			(void) fprintf(stderr,
102 			    gettext("Error locking config\n"));
103 			exit(1);
104 		}
105 	}
106 
107 	for (argv++; *argv != NULL; argv += 2)
108 		copybmp(argv[0], argv[1]);
109 	if (update_cfg)
110 		cfg_close(cfg);
111 	exit(0);
112 	return (0);
113 }
114 
115 void
116 iicpbmp_usage()
117 {
118 	(void) fprintf(stderr, gettext("Usage:\n"));
119 	(void) fprintf(stderr,
120 	    gettext("\tiicpbmp [-c] old_bitmap new_bitmap\n"));
121 	exit(1);
122 }
123 
124 void
125 copybmp(char *old_bitmap, char *new_bitmap)
126 {
127 	int i;
128 	int dsw_fd;
129 	FILE *ifp, *ofp;
130 	ii_header_t header;
131 	char cp_buffer[256];
132 	dsw_stat_t args;
133 
134 	dsw_fd = open(DSWDEV, O_RDONLY);
135 	if (dsw_fd < 0) {
136 		perror(DSWDEV);
137 		exit(1);
138 	}
139 	if (*old_bitmap != '/' || *new_bitmap != '/') {
140 		(void) fprintf(stderr, gettext("Both old and new bitmap "
141 		    "file names must begin with a /.\n"));
142 		exit(1);
143 	}
144 
145 	if (strlen(new_bitmap) > DSW_NAMELEN) {
146 		(void) fprintf(stderr,
147 		    gettext("New bitmap name is too long.\n"));
148 		exit(1);
149 	}
150 
151 	if (update_cfg && find_bitmap_cfg(old_bitmap) == 0) {
152 		perror(old_bitmap);
153 		(void) fprintf(stderr,
154 		    gettext("Old bitmap not in existing cfg\n"));
155 		exit(1);
156 	}
157 
158 	(void) strncpy(args.shadow_vol, shadow, DSW_NAMELEN);
159 	args.shadow_vol[DSW_NAMELEN-1] = '\0';
160 
161 	args.status = spcs_s_ucreate();
162 	if (ioctl(dsw_fd, DSWIOC_STAT, &args) != -1) {
163 		(void) fprintf(stderr, gettext("Suspend the Point-in-Time Copy "
164 		    "set first\n"));
165 		(void) close(dsw_fd);
166 		exit(1);
167 	}
168 
169 	if ((ifp = fopen(old_bitmap, "r")) == NULL) {
170 		perror(old_bitmap);
171 		(void) fprintf(stderr, gettext("Can't open old bitmap file\n"));
172 		exit(1);
173 	}
174 
175 	/* Check old header looks like an Point-in-Time Copy bitmap header */
176 
177 	if (fread(&header, sizeof (header), 1, ifp) != 1) {
178 		(void) fprintf(stderr, gettext("Can't read old bitmap file\n"));
179 		exit(1);
180 	}
181 
182 	if (header.ii_magic != DSW_CLEAN && header.ii_magic != DSW_DIRTY) {
183 		(void) fprintf(stderr, gettext("%s is not a Point-in-Time Copy "
184 		    "bitmap.\n"), old_bitmap);
185 		exit(1);
186 	}
187 
188 	if (strncmp(header.bitmap_vol, old_bitmap, DSW_NAMELEN) != 0) {
189 		(void) fprintf(stderr, gettext(
190 		    "%s has Point-in-Time Copy bitmap magic number,\n"
191 		    "but does not contain correct data.\n"), old_bitmap);
192 		exit(1);
193 	}
194 
195 	if ((ofp = fopen(new_bitmap, "w")) == NULL) {
196 		perror(new_bitmap);
197 		(void) fprintf(stderr, gettext("Can't open new bitmap file\n"));
198 		exit(1);
199 	}
200 
201 	/* Set up new header */
202 
203 	(void) memset(header.bitmap_vol, 0, DSW_NAMELEN);
204 	(void) strncpy(header.bitmap_vol, new_bitmap, DSW_NAMELEN);
205 
206 	if (fwrite(&header, sizeof (header), 1, ofp) != 1) {
207 		perror(new_bitmap);
208 		(void) fprintf(stderr,
209 		    gettext("Can't write new bitmap header\n"));
210 		exit(1);
211 	}
212 
213 	/* Copy the bitmap itself */
214 
215 	while ((i = fread(cp_buffer, sizeof (char), sizeof (cp_buffer), ifp))
216 	    > 0) {
217 		if (fwrite(cp_buffer, sizeof (char), i, ofp) != i) {
218 			perror(gettext("Write new bitmap failed"));
219 			break;
220 		}
221 	}
222 	(void) fclose(ofp);
223 	(void) fclose(ifp);
224 	(void) close(dsw_fd);
225 	if (update_cfg) {
226 		(void) sprintf(key, "ii.set%d.bitmap", setnumber);
227 		if (cfg_put_cstring(cfg, key, new_bitmap, strlen(new_bitmap))
228 		    < 0) {
229 				perror("cfg_put_cstring");
230 		}
231 		(void) cfg_commit(cfg);
232 		spcs_log("ii", NULL,
233 		    "iicpbmp copy bit map for %s from %s to %s",
234 		    shadow, old_bitmap, new_bitmap);
235 	}
236 }
237 
238 /*
239  * find_bitmap_cfg()
240  *
241  */
242 
243 int
244 find_bitmap_cfg(char *bitmap)
245 {
246 	for (setnumber = 1; ; setnumber++) {
247 		bzero(buf, CFG_MAX_BUF);
248 		(void) snprintf(key, sizeof (key), "ii.set%d.bitmap",
249 		    setnumber);
250 
251 		if (cfg_get_cstring(cfg, key, buf, DSW_NAMELEN) < 0)
252 			return (0);
253 		if (strcmp(buf, bitmap) == 0) {
254 			(void) snprintf(key, sizeof (key), "ii.set%d.shadow",
255 			    setnumber);
256 			(void) cfg_get_cstring(cfg, key, shadow, DSW_NAMELEN);
257 			return (setnumber);
258 		}
259 	}
260 }
261