|
Packit |
8480eb |
/* ----------------------------------------------------------------------- *
|
|
Packit |
8480eb |
*
|
|
Packit |
8480eb |
* mount_changer.c - module for Linux automountd to mount filesystems
|
|
Packit |
8480eb |
* from cd changers
|
|
Packit |
8480eb |
*
|
|
Packit |
8480eb |
* Copyright 1999 Toby Jaffey - All Rights Reserved
|
|
Packit |
8480eb |
* CD swapping code from linux kernel in Documentation/cdrom/ide-cd
|
|
Packit |
8480eb |
* Based on code originally from Gerhard Zuber <zuber@berlin.snafu.de>.
|
|
Packit |
8480eb |
* Changer status information, and rewrite for the new Uniform CDROM driver
|
|
Packit |
8480eb |
* interface by Erik Andersen <andersee@debian.org>.
|
|
Packit |
8480eb |
*
|
|
Packit |
8480eb |
* This program is free software; you can redistribute it and/or modify
|
|
Packit |
8480eb |
* it under the terms of the GNU General Public License as published by
|
|
Packit |
8480eb |
* the Free Software Foundation, Inc., 675 Mass Ave, Cambridge MA 02139,
|
|
Packit |
8480eb |
* USA; either version 2 of the License, or (at your option) any later
|
|
Packit |
8480eb |
* version; incorporated herein by reference.
|
|
Packit |
8480eb |
*
|
|
Packit |
8480eb |
* ----------------------------------------------------------------------- */
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
#include <stdio.h>
|
|
Packit |
8480eb |
#include <malloc.h>
|
|
Packit |
8480eb |
#include <string.h>
|
|
Packit |
8480eb |
#include <sys/param.h>
|
|
Packit |
8480eb |
#include <sys/types.h>
|
|
Packit |
8480eb |
#include <sys/stat.h>
|
|
Packit |
8480eb |
#include <stdlib.h>
|
|
Packit |
8480eb |
#include <sys/ioctl.h>
|
|
Packit |
8480eb |
#include <linux/cdrom.h>
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
#define MODULE_MOUNT
|
|
Packit |
8480eb |
#include "automount.h"
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
#define MODPREFIX "mount(changer): "
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
int mount_version = AUTOFS_MOUNT_VERSION; /* Required by protocol */
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
int swapCD(const char *device, const char *slotName);
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
int mount_init(void **context)
|
|
Packit |
8480eb |
{
|
|
Packit |
8480eb |
return 0;
|
|
Packit |
8480eb |
}
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
int mount_reinit(void **context)
|
|
Packit |
8480eb |
{
|
|
Packit |
8480eb |
return 0;
|
|
Packit |
8480eb |
}
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
int mount_mount(struct autofs_point *ap, const char *root, const char *name, int name_len,
|
|
Packit |
8480eb |
const char *what, const char *fstype, const char *options, void *context)
|
|
Packit |
8480eb |
{
|
|
Packit |
8480eb |
char fullpath[PATH_MAX];
|
|
Packit |
8480eb |
char buf[MAX_ERR_BUF];
|
|
Packit |
8480eb |
int err;
|
|
Packit |
8480eb |
int len, status, existed = 1;
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
if (ap->flags & MOUNT_FLAG_REMOUNT)
|
|
Packit |
8480eb |
return 0;
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
fstype = "iso9660";
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
/* Root offset of multi-mount */
|
|
Packit |
8480eb |
len = strlen(root);
|
|
Packit |
8480eb |
if (root[len - 1] == '/') {
|
|
Packit |
8480eb |
len = snprintf(fullpath, len, "%s", root);
|
|
Packit |
8480eb |
} else if (*name == '/') {
|
|
Packit |
8480eb |
/*
|
|
Packit |
8480eb |
* Direct or offset mount, name is absolute path so
|
|
Packit |
8480eb |
* don't use root (but with move mount changes root
|
|
Packit |
8480eb |
* is now the same as name).
|
|
Packit |
8480eb |
*/
|
|
Packit |
8480eb |
len = sprintf(fullpath, "%s", root);
|
|
Packit |
8480eb |
} else {
|
|
Packit |
8480eb |
len = sprintf(fullpath, "%s/%s", root, name);
|
|
Packit |
8480eb |
}
|
|
Packit |
8480eb |
fullpath[len] = '\0';
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
debug(ap->logopt, MODPREFIX "calling umount %s", what);
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
err = spawn_umount(ap->logopt, what, NULL);
|
|
Packit |
8480eb |
if (err) {
|
|
Packit |
8480eb |
error(ap->logopt,
|
|
Packit |
8480eb |
MODPREFIX
|
|
Packit |
8480eb |
"umount of %s failed (all may be unmounted)",
|
|
Packit |
8480eb |
what);
|
|
Packit |
8480eb |
}
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
debug(ap->logopt, MODPREFIX "calling mkdir_path %s", fullpath);
|
|
Packit |
8480eb |
|
|
Packit Service |
2f7109 |
status = mkdir_path(fullpath, mp_mode);
|
|
Packit |
8480eb |
if (status && errno != EEXIST) {
|
|
Packit |
8480eb |
char *estr = strerror_r(errno, buf, MAX_ERR_BUF);
|
|
Packit |
8480eb |
error(ap->logopt,
|
|
Packit |
8480eb |
MODPREFIX "mkdir_path %s failed: %s", fullpath, estr);
|
|
Packit |
8480eb |
return 1;
|
|
Packit |
8480eb |
}
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
if (!status)
|
|
Packit |
8480eb |
existed = 0;
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
debug(ap->logopt, MODPREFIX "Swapping CD to slot %s", name);
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
err = swapCD(what, name);
|
|
Packit |
8480eb |
if (err) {
|
|
Packit |
8480eb |
error(ap->logopt,
|
|
Packit |
8480eb |
MODPREFIX "failed to swap CD to slot %s", name);
|
|
Packit |
8480eb |
return 1;
|
|
Packit |
8480eb |
}
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
if (options && options[0]) {
|
|
Packit |
8480eb |
debug(ap->logopt, MODPREFIX
|
|
Packit |
8480eb |
"calling mount -t %s -o %s %s %s",
|
|
Packit |
8480eb |
fstype, options, what, fullpath);
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
err = spawn_mount(ap->logopt, "-t", fstype,
|
|
Packit |
8480eb |
"-o", options, what, fullpath, NULL);
|
|
Packit |
8480eb |
} else {
|
|
Packit |
8480eb |
debug(ap->logopt,
|
|
Packit |
8480eb |
MODPREFIX "calling mount -t %s %s %s",
|
|
Packit |
8480eb |
fstype, what, fullpath);
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
err = spawn_mount(ap->logopt, "-t", fstype, what, fullpath, NULL);
|
|
Packit |
8480eb |
}
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
if (err) {
|
|
Packit |
8480eb |
info(ap->logopt, MODPREFIX "failed to mount %s (type %s) on %s",
|
|
Packit |
8480eb |
what, fstype, fullpath);
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
if (ap->type != LKP_INDIRECT)
|
|
Packit |
8480eb |
return 1;
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
if ((!(ap->flags & MOUNT_FLAG_GHOST) && name_len) || !existed)
|
|
Packit |
8480eb |
rmdir_path(ap, fullpath, ap->dev);
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
return 1;
|
|
Packit |
8480eb |
} else {
|
|
Packit |
8480eb |
debug(ap->logopt, MODPREFIX "mounted %s type %s on %s",
|
|
Packit |
8480eb |
what, fstype, fullpath);
|
|
Packit |
8480eb |
return 0;
|
|
Packit |
8480eb |
}
|
|
Packit |
8480eb |
}
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
int mount_done(void *context)
|
|
Packit |
8480eb |
{
|
|
Packit |
8480eb |
return 0;
|
|
Packit |
8480eb |
}
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
int swapCD(const char *device, const char *slotName)
|
|
Packit |
8480eb |
{
|
|
Packit |
8480eb |
int fd; /* file descriptor for CD-ROM device */
|
|
Packit |
8480eb |
int status; /* return status for system calls */
|
|
Packit |
8480eb |
int slot = -1;
|
|
Packit |
8480eb |
int total_slots_available;
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
slot = atoi(slotName) - 1;
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
/* open device */
|
|
Packit |
8480eb |
fd = open_fd(device, O_RDONLY | O_NONBLOCK);
|
|
Packit |
8480eb |
if (fd < 0) {
|
|
Packit |
8480eb |
logerr(MODPREFIX "Opening device %s failed : %s",
|
|
Packit |
8480eb |
device, strerror(errno));
|
|
Packit |
8480eb |
return 1;
|
|
Packit |
8480eb |
}
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
/* Check CD player status */
|
|
Packit |
8480eb |
total_slots_available = ioctl(fd, CDROM_CHANGER_NSLOTS);
|
|
Packit |
8480eb |
if (total_slots_available <= 1) {
|
|
Packit |
8480eb |
logerr(MODPREFIX
|
|
Packit |
8480eb |
"Device %s is not an ATAPI compliant CD changer.",
|
|
Packit |
8480eb |
device);
|
|
Packit |
8480eb |
close(fd);
|
|
Packit |
8480eb |
return 1;
|
|
Packit |
8480eb |
}
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
/* load */
|
|
Packit |
8480eb |
slot = ioctl(fd, CDROM_SELECT_DISC, slot);
|
|
Packit |
8480eb |
if (slot < 0) {
|
|
Packit |
8480eb |
logerr(MODPREFIX "CDROM_SELECT_DISC failed");
|
|
Packit |
8480eb |
close(fd);
|
|
Packit |
8480eb |
return 1;
|
|
Packit |
8480eb |
}
|
|
Packit |
8480eb |
|
|
Packit |
8480eb |
/* close device */
|
|
Packit |
8480eb |
status = close(fd);
|
|
Packit |
8480eb |
if (status != 0) {
|
|
Packit |
8480eb |
logerr(MODPREFIX "close failed for `%s': %s",
|
|
Packit |
8480eb |
device, strerror(errno));
|
|
Packit |
8480eb |
return 1;
|
|
Packit |
8480eb |
}
|
|
Packit |
8480eb |
return 0;
|
|
Packit |
8480eb |
}
|