xref: /illumos-gate/usr/src/cmd/cdrw/main.c (revision e688a0bc223983e7c7fedad44c92b8d0292a10f7)
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 2008 Sun Microsystems, Inc.  All rights reserved.
23  * Use is subject to license terms.
24  */
25 
26 #pragma ident	"%Z%%M%	%I%	%E% SMI"
27 
28 #include <stdio.h>
29 #include <stdlib.h>
30 #include <sys/types.h>
31 #include <limits.h>
32 #include <unistd.h>
33 #include <libintl.h>
34 #include <locale.h>
35 #include <dbus/dbus.h>
36 #include <hal/libhal.h>
37 
38 #include "msgs.h"
39 #include "device.h"
40 #include "util.h"
41 #include "main.h"
42 #include "options.h"
43 #include "mmc.h"
44 #include "misc_scsi.h"
45 
46 /*
47  * global flags
48  */
49 int	debug = 0;
50 int	use_media_stated_capacity = 0;
51 int	keep_disc_open = 0;
52 int	requested_speed = 0;
53 int	simulation = 0;
54 int	verbose = 0;
55 char	*image_file = NULL;
56 char	*blanking_type = NULL;
57 int	audio_type = AUDIO_TYPE_NONE;
58 int	extract_track_no = 0;
59 char	*extract_file = NULL;
60 char	*alt_tmp_dir = NULL;
61 char	*copy_src = NULL;
62 int	vol_running = 0;
63 int	cflag = 0;
64 int	tflag = 0;
65 uid_t	ruid, cur_uid;
66 
67 /*
68  * global variables
69  */
70 cd_device *target = NULL;		/* Default target device */
71 static char *tgtdev = NULL;
72 int device_type = CD_RW;		/* Default to CD/RW */
73 int write_mode = TAO_MODE;		/* Default to track at once */
74 
75 static void
76 print_usage(void)
77 {
78 	err_msg(gettext("USAGE:\n"));
79 	err_msg(gettext("\tcdrw -i [ -vSCO ] [ -d device ] [ -p speed ]"));
80 	err_msg(gettext(" [ image-file ]\n"));
81 	err_msg(gettext("\tcdrw -a [ -vSCO ] [ -d device ] [ -p speed ]"));
82 	err_msg(gettext(" [ -T audio-type ] audio-file1 audio-file2 ...\n"));
83 	err_msg(gettext("\tcdrw -x [ -v ] [ -d device ] [ -T audio-type ]"));
84 	err_msg(gettext(" track-number audio-file\n"));
85 	err_msg(gettext("\tcdrw -c [ -cSC ] [ -d device ] [ -p speed ]"));
86 	err_msg(gettext(" [ -m tmp-dir ] [ -s src-device ]\n"));
87 	err_msg(
88 	    gettext("\tcdrw -b [ -v ] [ -d device ] all | session | fast\n"));
89 	err_msg(gettext("\tcdrw -M [ -v ] [ -d device ]\n"));
90 	err_msg(gettext("\tcdrw -L [v] [ -d device ]\n"));
91 	err_msg(gettext("\tcdrw -l [ -v ]\n"));
92 	err_msg(gettext("\tcdrw -h\n"));
93 
94 	exit(2);
95 }
96 
97 static void
98 check_invalid_option(options *specified, char *opstr)
99 {
100 	options c_op;
101 	int ret;
102 
103 	set_options_mask(&c_op, opstr);
104 	if ((ret = compare_options_mask(&c_op, specified)) != 0) {
105 		err_msg(
106 		    gettext("Option %c is not defined for this operation.\n"),
107 		    (char)ret);
108 		print_usage();
109 	}
110 }
111 
112 LibHalContext *
113 attach_to_hald(void)
114 {
115 	LibHalContext *ctx = NULL;
116 	DBusConnection *con = NULL;
117 	DBusError error;
118 	hal_state_t state;
119 
120 	/* Initialize the dbus error states */
121 	dbus_error_init(&error);
122 
123 	if ((con = dbus_bus_get(DBUS_BUS_SYSTEM, &error)) == NULL) {
124 		return (NULL);
125 	}
126 	state = DBUS_CONNECTION;
127 
128 	/* Allocate a new hal context to work with the dbus */
129 	if ((ctx = libhal_ctx_new()) == NULL)
130 		return (NULL);
131 	state = HAL_CONTEXT;
132 
133 	/* Pair up the context with the connection */
134 	if (!libhal_ctx_set_dbus_connection(ctx, con))
135 		goto fail;
136 	state = HAL_PAIRED;
137 
138 	/* If libhal_ctx_init fails hald is not present */
139 	if (!libhal_ctx_init(ctx, &error)) {
140 		goto fail;
141 	}
142 	state = HAL_INITIALIZED;
143 
144 	return (ctx);
145 fail:
146 	if (dbus_error_is_set(&error))
147 		dbus_error_free(&error);
148 	detach_from_hald(ctx, state);
149 	return (NULL);
150 
151 }
152 
153 void
154 detach_from_hald(LibHalContext *ctx, hal_state_t state)
155 {
156 	DBusError error;
157 	DBusConnection *con = libhal_ctx_get_dbus_connection(ctx);
158 
159 	dbus_error_init(&error);
160 
161 	switch (state) {
162 	case HAL_INITIALIZED:
163 		if (libhal_ctx_shutdown(ctx, &error) == FALSE)
164 			if (dbus_error_is_set(&error))
165 				dbus_error_free(&error);
166 	/*FALLTHROUGH*/
167 	case HAL_PAIRED:
168 		(void) libhal_ctx_free(ctx);
169 		dbus_connection_unref(con);
170 		break;
171 	case HAL_CONTEXT:
172 		(void) libhal_ctx_free(ctx);
173 		break;
174 	case DBUS_CONNECTION:
175 	default:
176 		break;
177 	}
178 }
179 
180 /*
181  * This function returns one if hald is running and
182  * zero if hald is not running
183  */
184 int
185 hald_running(void)
186 {
187 	LibHalContext *ctx;
188 
189 	if ((ctx = attach_to_hald()) == NULL)
190 		return (0);
191 
192 	detach_from_hald(ctx, HAL_INITIALIZED);
193 	return (1);
194 }
195 
196 int
197 setup_target(int flag)
198 {
199 	char *devpath;
200 
201 	if (tgtdev != NULL) {
202 		devpath = (char *)my_zalloc(PATH_MAX);
203 		if (lookup_device(tgtdev, devpath)) {
204 			target = get_device(tgtdev, devpath);
205 		}
206 		free(devpath);
207 		if (target == NULL) {
208 			return (0);
209 		}
210 		return (1);
211 	}
212 	return (scan_for_cd_device(flag, &target));
213 }
214 
215 int
216 main(int argc, char **argv)
217 {
218 	int		c;
219 	int		operations;
220 	options		specified_ops;
221 	int		aflag, iflag, Mflag, Lflag, lflag, bflag, xflag;
222 	int		ret;
223 
224 	(void) setlocale(LC_ALL, "");
225 
226 #if !defined(TEXT_DOMAIN)
227 #define	TEXT_DOMAIN	"SYS_TEST"
228 #endif
229 
230 
231 	(void) textdomain(TEXT_DOMAIN);
232 
233 	ruid = getuid();
234 	cur_uid = geteuid();
235 
236 	if (check_auth(ruid) != 1)  {
237 		err_msg(gettext(
238 		    "Authorization failed, Cannot access disks.\n"));
239 		exit(1);
240 	}
241 
242 	if ((cur_uid == 0) && (ruid != 0)) {
243 		priv_change_needed = 1;
244 		lower_priv();
245 	}
246 
247 	vol_running = hald_running();
248 
249 	tgtdev = NULL;
250 	operations = 0;
251 	set_options_mask(&specified_ops, "");
252 	iflag = Mflag = Lflag = lflag = bflag = aflag = xflag = cflag = 0;
253 
254 	while ((c = getopt(argc, argv, "abcCd:hiLlm:MOp:s:ST:vVx")) != EOF) {
255 		add_option(&specified_ops, c);
256 		switch (c) {
257 		case 'a':
258 			aflag = 1;
259 			operations++;
260 			break;
261 		case 'b':
262 			bflag = 1;
263 			operations++;
264 			break;
265 		case 'c':
266 			cflag = 1;
267 			operations++;
268 			break;
269 		case 'C':
270 			use_media_stated_capacity = 1;
271 			break;
272 		case 'd':
273 			tgtdev = optarg;
274 			break;
275 		case 'h':
276 			print_usage(); /* will not return */
277 			break;
278 		case 'i':
279 			iflag = 1;
280 			operations++;
281 			break;
282 		case 'L':
283 			Lflag = 1;
284 			operations++;
285 			break;
286 		case 'l':
287 			lflag = 1;
288 			operations++;
289 			break;
290 		case 'm':
291 			alt_tmp_dir = optarg;
292 			break;
293 		case 'M':
294 			Mflag = 1;
295 			operations++;
296 			break;
297 		case 'O':
298 			keep_disc_open = 1;
299 			break;
300 		case 'p':
301 			requested_speed = atoi(optarg);
302 			break;
303 		case 's':
304 			copy_src = optarg;
305 			break;
306 		case 'S':
307 			simulation++;
308 			break;
309 		case 'T':
310 			audio_type = get_audio_type(optarg);
311 			if (audio_type == -1) {
312 				err_msg(gettext("Unknown audio type %s\n"),
313 				    optarg);
314 				exit(1);
315 			}
316 			break;
317 		case 'v':
318 			verbose++;
319 			break;
320 		case 'V':
321 			/*
322 			 * more verbose. this will print out debug comments
323 			 */
324 
325 			debug++;
326 			break;
327 		case 'x':
328 			xflag++;
329 			operations++;
330 			break;
331 		default:
332 			print_usage();
333 		}
334 	}
335 	if (operations == 0) {
336 		err_msg(gettext("No operation specified.\n"));
337 		exit(1);
338 	}
339 	if (operations != 1) {
340 		err_msg(gettext("More than one operation specified.\n"));
341 		exit(1);
342 	}
343 
344 	if (lflag) {
345 		check_invalid_option(&specified_ops, "lhvV");
346 		list();
347 	}
348 
349 	/*
350 	 * we'll allow the user to specify the source device (-s) when
351 	 *  extracting audio.
352 	 */
353 
354 	if (xflag && copy_src)
355 		tgtdev = copy_src;
356 
357 	/*
358 	 * This will scan for all CD devices when xflag or Mflag
359 	 * (extract audio, list toc) commands are used, providing
360 	 * no CD-RW devices are found. Since these commands can
361 	 * be used without a CD writer.
362 	 */
363 
364 	if (xflag || Mflag) {
365 		ret = setup_target(SCAN_ALL_CDS);
366 	} else {
367 		ret = setup_target(SCAN_WRITERS);
368 	}
369 
370 	if (ret == 0) {
371 
372 		if (tgtdev != NULL) {
373 			err_msg(gettext(
374 			    "Cannot find device %s.\n"), tgtdev);
375 
376 		}
377 
378 		if (vol_running) {
379 			err_msg(gettext(
380 			    "No CD writers found or no media in the drive.\n"));
381 		} else {
382 			if (cur_uid != 0) {
383 				err_msg(gettext(
384 				    "Volume manager is not running.\n"));
385 				err_msg(gettext(
386 "Please start volume manager or run cdrw as root to access all devices.\n"));
387 			} else {
388 				err_msg(gettext(
389 				    "No CD writers found.\n"));
390 			}
391 		}
392 		exit(1);
393 
394 	} else if (ret != 1) {
395 		err_msg(gettext("More than one CD device found.\n"));
396 		err_msg(gettext("Specify one using -d option.\n"));
397 		err_msg(gettext(
398 		    "Or use -l option to list all the CD devices found\n"));
399 		exit(1);
400 	}
401 	(void) check_device(target, CHECK_TYPE_NOT_CDROM|EXIT_IF_CHECK_FAILED);
402 
403 	if (check_device(target, CHECK_NO_MEDIA) == 0) {
404 		int retry;
405 		for (retry = 0; retry < 5; retry++) {
406 			if (check_device(target, CHECK_DEVICE_NOT_READY) == 0)
407 				break;
408 			(void) sleep(3);
409 		}
410 	}
411 
412 	if (aflag) {
413 		check_invalid_option(&specified_ops, "ahvSCOdpTV");
414 		if (optind == argc) {
415 			err_msg(gettext("No audio files specified.\n"));
416 			exit(1);
417 		}
418 		write_audio(argv, optind, argc);
419 	}
420 	if (Mflag) {
421 		check_invalid_option(&specified_ops, "MhvdV");
422 		info();
423 	}
424 	if (iflag) {
425 		check_invalid_option(&specified_ops, "ihvSCOdpV");
426 		if (optind == (argc - 1)) {
427 			image_file = argv[optind];
428 			write_image();
429 		}
430 		if (optind == argc)
431 			write_image();
432 		err_msg(gettext("Command line parsing error.\n"));
433 		err_msg(gettext("Only one image-file can be specified.\n"));
434 		exit(1);
435 	}
436 	if (bflag) {
437 		check_invalid_option(&specified_ops, "bhvdV");
438 		if (optind != (argc - 1)) {
439 			err_msg(gettext("Command line parsing error.\n"));
440 			print_usage();
441 		}
442 		blanking_type = argv[argc - 1];
443 		blank();
444 	}
445 	if (xflag) {
446 		check_invalid_option(&specified_ops, "xhpvdsTV");
447 		if (optind != (argc - 2)) {
448 			err_msg(gettext("Command line parsing error.\n"));
449 			print_usage();
450 		}
451 		extract_track_no = atoi(argv[argc - 2]);
452 		extract_file = argv[argc - 1];
453 		extract_audio();
454 	}
455 	if (cflag) {
456 		check_invalid_option(&specified_ops, "chvSCdpmsV");
457 		copy_cd();
458 	}
459 
460 	/*
461 	 * Open a closed disk, we do this by erasing the track tail
462 	 * and then re-finalizing with an open leadout.
463 	 */
464 	if (Lflag) {
465 		check_invalid_option(&specified_ops, "LvdV");
466 		(void) check_device(target, CHECK_NO_MEDIA |
467 		    CHECK_DEVICE_NOT_READY | EXIT_IF_CHECK_FAILED);
468 
469 		/* no need to erase blank media */
470 		if (!check_device(target, CHECK_MEDIA_IS_NOT_BLANK))
471 			exit(0);
472 
473 		blanking_type = "leadout";
474 		blank();
475 
476 		write_init(TRACK_MODE_DATA);
477 		(void) close_track(target->d_fd, 0, 1, 1);
478 		(void) finalize(target);
479 		(void) printf(gettext("done.\n"));
480 		exit(0);
481 	}
482 	return (0);
483 }
484