118c2aff7Sartem /***************************************************************************
218c2aff7Sartem  *
318c2aff7Sartem  * hal-storage-eject.c : Eject method handler
418c2aff7Sartem  *
518c2aff7Sartem  * Copyright (C) 2006 David Zeuthen, <david@fubar.dk>
618c2aff7Sartem  *
718c2aff7Sartem  * This program is free software; you can redistribute it and/or modify
818c2aff7Sartem  * it under the terms of the GNU General Public License as published by
918c2aff7Sartem  * the Free Software Foundation; either version 2 of the License, or
1018c2aff7Sartem  * (at your option) any later version.
1118c2aff7Sartem  *
1218c2aff7Sartem  * This program is distributed in the hope that it will be useful,
1318c2aff7Sartem  * but WITHOUT ANY WARRANTY; without even the implied warranty of
1418c2aff7Sartem  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
1518c2aff7Sartem  * GNU General Public License for more details.
1618c2aff7Sartem  *
1718c2aff7Sartem  * You should have received a copy of the GNU General Public License
1818c2aff7Sartem  * along with this program; if not, write to the Free Software
1918c2aff7Sartem  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
2018c2aff7Sartem  *
2118c2aff7Sartem  **************************************************************************/
2218c2aff7Sartem 
2318c2aff7Sartem 
2418c2aff7Sartem #ifdef HAVE_CONFIG_H
2518c2aff7Sartem #  include <config.h>
2618c2aff7Sartem #endif
2718c2aff7Sartem 
2818c2aff7Sartem #include <stdio.h>
2918c2aff7Sartem #include <stdlib.h>
3018c2aff7Sartem #include <string.h>
3118c2aff7Sartem #include <glib.h>
3218c2aff7Sartem #include <glib/gstdio.h>
3318c2aff7Sartem #include <sys/types.h>
3418c2aff7Sartem #include <unistd.h>
3518c2aff7Sartem 
3618c2aff7Sartem #include <libhal.h>
3718c2aff7Sartem #include <libhal-storage.h>
3818c2aff7Sartem #ifdef HAVE_POLKIT
3918c2aff7Sartem #include <libpolkit.h>
4018c2aff7Sartem #endif
4118c2aff7Sartem 
4218c2aff7Sartem #include "hal-storage-shared.h"
4318c2aff7Sartem 
4418c2aff7Sartem /* possible values: "Volume", "Storage" */
4518c2aff7Sartem static char *devtype = "Volume";
4618c2aff7Sartem 
4718c2aff7Sartem 
4818c2aff7Sartem static void
usage(void)4918c2aff7Sartem usage (void)
5018c2aff7Sartem {
5118c2aff7Sartem 	fprintf (stderr, "This program should only be started by hald.\n");
5218c2aff7Sartem 	exit (1);
5318c2aff7Sartem }
5418c2aff7Sartem 
5518c2aff7Sartem 
5618c2aff7Sartem void static
unknown_eject_error(const char * detail)5718c2aff7Sartem unknown_eject_error (const char *detail)
5818c2aff7Sartem {
5918c2aff7Sartem 	fprintf (stderr, "org.freedesktop.Hal.Device.%s.UnknownFailure\n", devtype);
6018c2aff7Sartem 	fprintf (stderr, "%s\n", detail);
6118c2aff7Sartem 	exit (1);
6218c2aff7Sartem }
6318c2aff7Sartem 
6418c2aff7Sartem 
6518c2aff7Sartem static void
invalid_eject_option(const char * option,const char * uid)6618c2aff7Sartem invalid_eject_option (const char *option, const char *uid)
6718c2aff7Sartem {
6818c2aff7Sartem 	fprintf (stderr, "org.freedesktop.Hal.Device.Volume.InvalidEjectOption\n");
6918c2aff7Sartem 	fprintf (stderr, "The option '%s' is not allowed for uid=%s\n", option, uid);
7018c2aff7Sartem 	exit (1);
7118c2aff7Sartem }
7218c2aff7Sartem 
7318c2aff7Sartem 
7418c2aff7Sartem int
main(int argc,char * argv[])7518c2aff7Sartem main (int argc, char *argv[])
7618c2aff7Sartem {
7718c2aff7Sartem 	char *udi;
7818c2aff7Sartem 	char *device;
7918c2aff7Sartem 	const char *drive_udi;
8018c2aff7Sartem 	LibHalDrive *drive;
8118c2aff7Sartem 	LibHalVolume *volume;
8218c2aff7Sartem 	DBusError error;
8318c2aff7Sartem 	LibHalContext *hal_ctx = NULL;
8418c2aff7Sartem 	DBusConnection *system_bus = NULL;
8518c2aff7Sartem #ifdef HAVE_POLKIT
8618c2aff7Sartem 	LibPolKitContext *pol_ctx = NULL;
8718c2aff7Sartem #endif
8818c2aff7Sartem 	char *invoked_by_uid;
8918c2aff7Sartem 	char *invoked_by_syscon_name;
9018c2aff7Sartem 	char **volume_udis;
9118c2aff7Sartem 	int num_volumes;
9218c2aff7Sartem 	int i;
9318c2aff7Sartem 	char eject_options[1024];
9418c2aff7Sartem 	char **given_options;
9518c2aff7Sartem 	const char *end;
9618c2aff7Sartem 
9718c2aff7Sartem 	device = getenv ("HAL_PROP_BLOCK_DEVICE");
9818c2aff7Sartem 	if (device == NULL)
9918c2aff7Sartem 		usage ();
10018c2aff7Sartem 
10118c2aff7Sartem 	udi = getenv ("HAL_PROP_INFO_UDI");
10218c2aff7Sartem 	if (udi == NULL)
10318c2aff7Sartem 		usage ();
10418c2aff7Sartem 
10518c2aff7Sartem 	invoked_by_uid = getenv ("HAL_METHOD_INVOKED_BY_UID");
10618c2aff7Sartem 
10718c2aff7Sartem 	invoked_by_syscon_name = getenv ("HAL_METHOD_INVOKED_BY_SYSTEMBUS_CONNECTION_NAME");
10818c2aff7Sartem 
10918c2aff7Sartem 	dbus_error_init (&error);
11018c2aff7Sartem 	if ((hal_ctx = libhal_ctx_init_direct (&error)) == NULL) {
11118c2aff7Sartem 		printf ("Cannot connect to hald\n");
11218c2aff7Sartem 		LIBHAL_FREE_DBUS_ERROR (&error);
11318c2aff7Sartem 		usage ();
11418c2aff7Sartem 	}
11518c2aff7Sartem 
11618c2aff7Sartem 	dbus_error_init (&error);
11718c2aff7Sartem 	system_bus = dbus_bus_get (DBUS_BUS_SYSTEM, &error);
11818c2aff7Sartem 	if (system_bus == NULL) {
11918c2aff7Sartem 		printf ("Cannot connect to the system bus\n");
12018c2aff7Sartem 		LIBHAL_FREE_DBUS_ERROR (&error);
12118c2aff7Sartem 		usage ();
12218c2aff7Sartem 	}
12318c2aff7Sartem #ifdef HAVE_POLKIT
12418c2aff7Sartem 	pol_ctx = libpolkit_new_context (system_bus);
12518c2aff7Sartem 	if (pol_ctx == NULL) {
12618c2aff7Sartem 		printf ("Cannot get libpolkit context\n");
12718c2aff7Sartem 		unknown_eject_error ("Cannot get libpolkit context");
12818c2aff7Sartem 	}
12918c2aff7Sartem #endif
13018c2aff7Sartem 
13118c2aff7Sartem 	/* read from stdin */
13218c2aff7Sartem 	if (strlen (fgets (eject_options, sizeof (eject_options), stdin)) > 0)
13318c2aff7Sartem 		eject_options [strlen (eject_options) - 1] = '\0';
13418c2aff7Sartem 	/* validate that input from stdin is UTF-8 */
13518c2aff7Sartem 	if (!g_utf8_validate (eject_options, -1, &end))
13618c2aff7Sartem 		unknown_eject_error ("Error validating eject_options as UTF-8");
13718c2aff7Sartem #ifdef DEBUG
13818c2aff7Sartem 	printf ("eject_options  = '%s'\n", eject_options);
13918c2aff7Sartem #endif
14018c2aff7Sartem 
14118c2aff7Sartem 	/* delete any trailing whitespace options from splitting the string */
14218c2aff7Sartem 	given_options = g_strsplit (eject_options, "\t", 0);
14318c2aff7Sartem 	for (i = g_strv_length (given_options) - 1; i >= 0; --i) {
14418c2aff7Sartem 		if (strlen (given_options[i]) > 0)
14518c2aff7Sartem 			break;
14618c2aff7Sartem 		given_options[i] = NULL;
14718c2aff7Sartem 	}
14818c2aff7Sartem 
14918c2aff7Sartem 	/* check eject options */
15018c2aff7Sartem 	for (i = 0; given_options[i] != NULL; i++) {
15118c2aff7Sartem 		char *given = given_options[i];
15218c2aff7Sartem 
15318c2aff7Sartem 		/* none supported right now */
15418c2aff7Sartem 
15518c2aff7Sartem 		invalid_eject_option (given, invoked_by_uid);
15618c2aff7Sartem 	}
15718c2aff7Sartem 	g_strfreev (given_options);
15818c2aff7Sartem 
15918c2aff7Sartem 	/* should be either volume or storage */
16018c2aff7Sartem 	if ((volume = libhal_volume_from_udi (hal_ctx, udi)) != NULL) {
16118c2aff7Sartem 		drive_udi = libhal_volume_get_storage_device_udi (volume);
16218c2aff7Sartem 	} else {
16318c2aff7Sartem 		drive_udi = g_strdup (udi);
16418c2aff7Sartem 		devtype = "Storage";
16518c2aff7Sartem 	}
16618c2aff7Sartem 	if (drive_udi == NULL) {
16718c2aff7Sartem 		unknown_eject_error ("Cannot get drive udi");
16818c2aff7Sartem 	}
16918c2aff7Sartem 	if ((drive = libhal_drive_from_udi (hal_ctx, drive_udi)) == NULL) {
17018c2aff7Sartem 		unknown_eject_error ("Cannot get drive from udi");
17118c2aff7Sartem 	}
17218c2aff7Sartem 
17318c2aff7Sartem 	/* first, unmount all volumes */
17418c2aff7Sartem 	volume_udis = libhal_drive_find_all_volumes (hal_ctx, drive, &num_volumes);
17518c2aff7Sartem 	if (volume_udis == NULL)
17618c2aff7Sartem 		unknown_eject_error ("Cannot get all enclosed volumes");
17718c2aff7Sartem 	for (i = 0; i < num_volumes; i++) {
17818c2aff7Sartem 		char *volume_udi;
17918c2aff7Sartem 		LibHalVolume *volume_to_unmount;
18018c2aff7Sartem 
18118c2aff7Sartem 		volume_udi = volume_udis[i];
182*55fea89dSDan Cross 
18318c2aff7Sartem #ifdef DEBUG
18418c2aff7Sartem 		printf ("processing drive's volume %s (%d of %d)\n", volume_udi, i + 1, num_volumes);
18518c2aff7Sartem #endif
18618c2aff7Sartem 		volume_to_unmount = libhal_volume_from_udi (hal_ctx, volume_udi);
18718c2aff7Sartem 		if (volume_to_unmount == NULL) {
18818c2aff7Sartem 			unknown_eject_error ("Cannot get volume object");
18918c2aff7Sartem 		}
19018c2aff7Sartem 
19118c2aff7Sartem 		if (libhal_volume_is_mounted (volume_to_unmount)) {
19218c2aff7Sartem #ifdef DEBUG
19318c2aff7Sartem 			printf (" unmounting\n");
19418c2aff7Sartem #endif
19518c2aff7Sartem 			/* only lock around unmount call because hald's /proc/mounts handler
19618c2aff7Sartem 			 * will also want to lock the /media/.hal-mtab-lock file for peeking
19718c2aff7Sartem 			 */
19818c2aff7Sartem 			if (!lock_hal_mtab ()) {
19918c2aff7Sartem 				unknown_eject_error ("Cannot obtain lock on /media/.hal-mtab");
20018c2aff7Sartem 			}
201*55fea89dSDan Cross 			handle_unmount (hal_ctx,
20218c2aff7Sartem #ifdef HAVE_POLKIT
203*55fea89dSDan Cross 					pol_ctx,
20418c2aff7Sartem #endif
20518c2aff7Sartem 					volume_udi, volume_to_unmount, drive,
206*55fea89dSDan Cross 					libhal_volume_get_device_file (volume_to_unmount),
20718c2aff7Sartem 					invoked_by_uid, invoked_by_syscon_name,
20818c2aff7Sartem 					FALSE, FALSE, system_bus); /* use neither lazy nor force */
20918c2aff7Sartem 			unlock_hal_mtab ();
21018c2aff7Sartem 		} else {
21118c2aff7Sartem #ifdef DEBUG
21218c2aff7Sartem 			printf (" not mounted\n");
21318c2aff7Sartem #endif
21418c2aff7Sartem 		}
21518c2aff7Sartem 
21618c2aff7Sartem 		libhal_volume_free (volume_to_unmount);
21718c2aff7Sartem 
21818c2aff7Sartem 	}
21918c2aff7Sartem 	libhal_free_string_array (volume_udis);
22018c2aff7Sartem 
22118c2aff7Sartem 	/* now attempt the eject */
222*55fea89dSDan Cross 	handle_eject (hal_ctx,
22318c2aff7Sartem #ifdef HAVE_POLKIT
224*55fea89dSDan Cross 		      pol_ctx,
22518c2aff7Sartem #endif
22618c2aff7Sartem 		      libhal_drive_get_udi (drive),
22718c2aff7Sartem 		      drive,
22818c2aff7Sartem 		      libhal_drive_get_device_file (drive),
229*55fea89dSDan Cross 		      invoked_by_uid,
23018c2aff7Sartem 		      invoked_by_syscon_name,
23118c2aff7Sartem 		      FALSE, system_bus);
23218c2aff7Sartem 
23318c2aff7Sartem 	return 0;
23418c2aff7Sartem }
23518c2aff7Sartem 
23618c2aff7Sartem 
237