245 lines
5.5 KiB
C
245 lines
5.5 KiB
C
/*
|
|
* Guillaume Cottenceau (gc@mandrakesoft.com)
|
|
*
|
|
* Copyright 2000 MandrakeSoft
|
|
*
|
|
* This software may be freely redistributed under the terms of the GNU
|
|
* public license.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
*
|
|
*/
|
|
|
|
/*
|
|
* Portions from Erik Troan (ewt@redhat.com)
|
|
*
|
|
* Copyright 1996 Red Hat Software
|
|
*
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
#include <limits.h>
|
|
#include <unistd.h>
|
|
#include <string.h>
|
|
#include <sys/mount.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/types.h>
|
|
#include <sys/wait.h>
|
|
|
|
#ifndef DISABLE_NETWORK
|
|
#include <sys/socket.h>
|
|
#include <netinet/in.h>
|
|
#include <arpa/inet.h>
|
|
#endif
|
|
|
|
#include "log.h"
|
|
#include "modules.h"
|
|
|
|
#include "mount.h"
|
|
#include "dns.h"
|
|
|
|
|
|
#ifndef DISABLE_MEDIAS
|
|
/* WARNING: this won't work if the argument is not /dev/ based */
|
|
int ensure_dev_exists(char *dev)
|
|
{
|
|
int major, minor;
|
|
int type = S_IFBLK; /* my default type is block. don't forget to change for chars */
|
|
char * name;
|
|
struct stat buf;
|
|
char * ptr;
|
|
|
|
name = &dev[5]; /* we really need that dev be passed as /dev/something.. */
|
|
|
|
if (!stat(dev, &buf))
|
|
return 0; /* if the file already exists, we assume it's correct */
|
|
|
|
if (name[0] == 's' && name[1] == 'd') {
|
|
/* SCSI disks */
|
|
major = 8;
|
|
minor = (name[2] - 'a') << 4;
|
|
if (name[3] && name[4])
|
|
minor += 10 + (name[4] - '0');
|
|
else if (name[3])
|
|
minor += (name[3] - '0');
|
|
} else if (name[0] == 'h' && name[1] == 'd') {
|
|
/* IDE disks/cd's */
|
|
if (name[2] == 'a')
|
|
major = 3, minor = 0;
|
|
else if (name[2] == 'b')
|
|
major = 3, minor = 64;
|
|
else if (name[2] == 'c')
|
|
major = 22, minor = 0;
|
|
else if (name[2] == 'd')
|
|
major = 22, minor = 64;
|
|
else if (name[2] == 'e')
|
|
major = 33, minor = 0;
|
|
else if (name[2] == 'f')
|
|
major = 33, minor = 64;
|
|
else if (name[2] == 'g')
|
|
major = 34, minor = 0;
|
|
else if (name[2] == 'h')
|
|
major = 34, minor = 64;
|
|
else
|
|
return -1;
|
|
|
|
if (name[3] && name[4])
|
|
minor += 10 + (name[4] - '0');
|
|
else if (name[3])
|
|
minor += (name[3] - '0');
|
|
} else if (name[0] == 's' && name[1] == 'r') {
|
|
/* SCSI cd's */
|
|
major = 11;
|
|
minor = name[2] - '0';
|
|
} else if (ptr_begins_static_str(name, "ida/") ||
|
|
ptr_begins_static_str(name, "cciss/")) {
|
|
/* Compaq Smart Array "ida/c0d0{p1}" */
|
|
ptr = strchr(name, '/');
|
|
mkdir("/dev/ida", 0755);
|
|
mkdir("/dev/cciss", 0755);
|
|
major = ptr_begins_static_str(name, "ida/") ? 72 : 104 + charstar_to_int(ptr+2);
|
|
ptr = strchr(ptr, 'd');
|
|
minor = 16 * charstar_to_int(ptr+1);
|
|
ptr = strchr(ptr, 'p');
|
|
minor += charstar_to_int(ptr+1);
|
|
} else if (ptr_begins_static_str(name, "rd/")) {
|
|
/* DAC960 "rd/cXdXXpX" */
|
|
mkdir("/dev/rd", 0755);
|
|
major = 48 + charstar_to_int(name+4);
|
|
ptr = strchr(name+4, 'd');
|
|
minor = 8 * charstar_to_int(ptr+1);
|
|
ptr = strchr(ptr, 'p');
|
|
minor += charstar_to_int(ptr+1);
|
|
} else {
|
|
log_message("I don't know how to create device %s, please post bugreport to me!", dev);
|
|
return -1;
|
|
}
|
|
|
|
if (mknod(dev, type | 0600, makedev(major, minor))) {
|
|
log_perror(dev);
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
#endif /* DISABLE_MEDIAS */
|
|
|
|
#ifndef DISABLE_NETWORK
|
|
|
|
static int nfsmount(char *dev, char *location)
|
|
{
|
|
char spec[PATH_MAX + 17], *sep;
|
|
struct sockaddr_in saddr;
|
|
int n, pid, status;
|
|
|
|
if ((sep = strchr(dev, ':'))) {
|
|
*sep = '\0';
|
|
} else {
|
|
log_message("nfsmount: directory to mount not in host:dir format");
|
|
return -1;
|
|
}
|
|
|
|
saddr.sin_family = AF_INET;
|
|
if (!inet_aton(dev, &saddr.sin_addr) &&
|
|
mygethostbyname(dev, &saddr.sin_addr)) {
|
|
log_message("nfsmount: can't get address for %s", dev);
|
|
*sep = ':';
|
|
return -1;
|
|
}
|
|
|
|
*sep = ':';
|
|
strcpy(spec, inet_ntoa(saddr.sin_addr));
|
|
n = strlen(spec);
|
|
strncpy(spec + n, sep, sizeof(spec) - n);
|
|
log_message("nfsmount %s %s", spec, location);
|
|
|
|
if (!(pid = fork())) {
|
|
char * argv[] = {"/bin/nfsmount", spec, location, NULL};
|
|
close(0);
|
|
close(1);
|
|
close(2);
|
|
execve(argv[0], argv, NULL);
|
|
exit(1);
|
|
}
|
|
|
|
waitpid(pid, &status, 0);
|
|
return (WIFEXITED(status) && !WEXITSTATUS(status)) ? 0 : -1;
|
|
}
|
|
#endif
|
|
|
|
/* mounts, creating the device if needed+possible */
|
|
int my_mount(char *dev, char *location, char *fs, int force_rw)
|
|
{
|
|
unsigned long flags = MS_MGC_VAL | (force_rw ? 0 : MS_RDONLY);
|
|
char * opts = NULL;
|
|
struct stat buf;
|
|
int rc;
|
|
|
|
#ifndef DISABLE_MEDIAS
|
|
if (strcmp(fs, "nfs")) {
|
|
rc = ensure_dev_exists(dev);
|
|
if (rc != 0) {
|
|
log_message("could not create required device file");
|
|
return -1;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
log_message("mounting %s on %s as type %s", dev, location, fs);
|
|
|
|
if (stat(location, &buf)) {
|
|
if (mkdir(location, 0755)) {
|
|
log_perror("could not create location dir");
|
|
return -1;
|
|
}
|
|
} else if (!S_ISDIR(buf.st_mode)) {
|
|
log_message("not a dir %s, will unlink and mkdir", location);
|
|
if (unlink(location)) {
|
|
log_perror("could not unlink");
|
|
return -1;
|
|
}
|
|
if (mkdir(location, 0755)) {
|
|
log_perror("could not create location dir");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
#ifndef DISABLE_NETWORK
|
|
if (!strcmp(fs, "nfs")) {
|
|
return nfsmount(dev, location);
|
|
}
|
|
#endif
|
|
|
|
#ifndef DISABLE_MEDIAS
|
|
if (!strcmp(fs, "squashfs"))
|
|
my_insmod("squashfs", NULL);
|
|
|
|
if (!strcmp(fs, "ext2"))
|
|
my_insmod("ext2", NULL);
|
|
|
|
if (!strcmp(fs, "vfat")) {
|
|
my_insmod("vfat", NULL);
|
|
opts = "check=relaxed";
|
|
}
|
|
|
|
if (!strcmp(fs, "ntfs"))
|
|
my_insmod("ntfs", NULL);
|
|
|
|
if (!strcmp(fs, "reiserfs"))
|
|
my_insmod("reiserfs", NULL);
|
|
|
|
if (!strcmp(fs, "iso9660"))
|
|
my_insmod("isofs", NULL);
|
|
#endif
|
|
|
|
rc = mount(dev, location, fs, flags, opts);
|
|
if (rc != 0) {
|
|
log_perror("mount failed");
|
|
rmdir(location);
|
|
}
|
|
|
|
return rc;
|
|
}
|