/* * 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 #include #include #include #include #include #include #include #include #include #ifndef DISABLE_NETWORK #include #include #include #endif #include "log.h" #include "modules.h" #include "mount.h" #include "dns.h" #include "common.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, '/'); mkdirs_dev("ida", "cciss", NULL); 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"); 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; pid_t pid; int n, status, ret = -1; if ((sep = strchr(dev, ':'))) { *sep = '\0'; } else { log_message("nfsmount: directory to mount not in host:dir format"); goto out; } 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 = ':'; goto out; } *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); pid = fork(); if (pid < 0) { log_message("%s: error: fork: %s", __func__, strerror(errno)); goto out; } else if (pid == 0) { char * argv[] = {"/bin/nfsmount", spec, location, NULL}; redirect2log(STDOUT_FILENO); redirect2log(STDERR_FILENO); redirect2null(STDIN_FILENO); execve(argv[0], argv, NULL); exit(1); } else { if (waitpid(pid, &status, 0) < 0) { log_message("%s: error: waitpid: %s", __func__, strerror(errno)); goto out; } ret = (WIFEXITED(status) && !WEXITSTATUS(status)) ? 0 : -1; } out: return ret; } #ifdef ENABLE_CIFS static int cifsmount3(char *dev, char *location, const char *extra_opts) { char spec[PATH_MAX + 19], *sep, *ptr = dev; struct sockaddr_in saddr; pid_t pid; int n, status, ret = -1; const char *default_opts = "guest"; char *opts = NULL; if (extra_opts) { if (asprintf(&opts, "%s,%s", default_opts, extra_opts) < 0) { log_message("%s: error: asprintf", __func__); return -1; } } else { opts = strdup(default_opts); if (!opts) { log_message("%s: error: strdup", __func__); return -1; } } while (*ptr == '/') ptr++; if ((sep = strchr(ptr, '/'))) { *sep = '\0'; } else { log_message("cifsmount: directory to mount not in //host/dir format"); goto out; } saddr.sin_family = AF_INET; if (!inet_aton(ptr, &saddr.sin_addr) && mygethostbyname(ptr, &saddr.sin_addr)) { log_message("cifsmount: can't get address for %s", ptr); *sep = '/'; goto out; } *sep = '/'; *spec = '\0'; strcpy(spec, "//"); strcat(spec, inet_ntoa(saddr.sin_addr)); n = strlen(spec); strncpy(spec + n, sep, sizeof(spec) - n); log_message("/sbin/mount.cifs %s %s --verbose -o %s", spec, location, opts); pid = fork(); if (pid < 0) { log_message("%s: error: fork: %s", __func__, strerror(errno)); goto out; } else if (pid == 0) { char * argv[] = {"/sbin/mount.cifs", spec, location, "--verbose", "-o", opts, NULL}; redirect2log(STDOUT_FILENO); redirect2log(STDERR_FILENO); redirect2null(STDIN_FILENO); execve(argv[0], argv, NULL); exit(1); } else { if (waitpid(pid, &status, 0) < 0) { log_message("%s: error: waitpid: %s", __func__, strerror(errno)); goto out; } ret = (WIFEXITED(status) && !WEXITSTATUS(status)) ? 0 : -1; } out: free(opts); return ret; } static int cifsmount(char *dev, char *location) { int rc = 0; static const char *fallbacks[] = { "vers=2.1,sec=none", "vers=2.0,sec=none", "vers=1.0,sec=none", NULL }; const char **opts = NULL; log_message("%s: attempting to mount %s with default protocol version", __func__, dev); rc = cifsmount3(dev, location, NULL); for (opts = fallbacks; rc < 0 && *opts; opts++) { log_message("%s: failed, retrying with %s", __func__, *opts); rc = cifsmount3(dev, location, *opts); } if (rc < 0) { log_message("%s: sorry, nothing worked", __func__); } return rc; } #endif #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 #ifdef ENABLE_CIFS if (strcmp(fs, "nfs") && strcmp(fs, "cifs")) { #else if (strcmp(fs, "nfs")) { #endif 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); } #ifdef ENABLE_CIFS if (!strcmp(fs, "cifs")) { return cifsmount(dev, location); } #endif #endif #ifndef DISABLE_MEDIAS if (!strcmp(fs, "squashfs")) my_insmod("squashfs", NULL); if (!strcmp(fs, "ext2")) my_insmod("ext2", NULL); if (!strcmp(fs, "ext3")) my_insmod("ext3", NULL); if (!strcmp(fs, "ext4")) my_insmod("ext4", NULL); if (!strcmp(fs, "vfat")) { my_insmod("vfat", NULL); opts = "check=relaxed"; } if (!strcmp(fs, "exfat")) { my_insmod("exfat", 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; }