⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 umount.c

📁 Util-linux 软件包包含许多工具。其中比较重要的是加载、卸载、格式化、分区和管理硬盘驱动器
💻 C
📖 第 1 页 / 共 2 页
字号:
/* * umount(8) for Linux 0.99 - jrs, 1993 */#include <stdio.h>#include <unistd.h>#include <getopt.h>#include <string.h>#include <errno.h>#include <ctype.h>#include <sys/stat.h>#include <sys/wait.h>#include <sys/mount.h>#include "mount_constants.h"#include "sundries.h"#include "getusername.h"#include "lomount.h"#include "loop.h"#include "fstab.h"#include "env.h"#include "nls.h"#ifdef HAVE_NFS#include <sys/socket.h>#include <sys/time.h>#include <netdb.h>#include <rpc/rpc.h>#include <rpc/pmap_clnt.h>#include <rpc/pmap_prot.h>#include "nfsmount.h"#include <arpa/inet.h>#endif#if defined(MNT_FORCE) && !defined(__sparc__) && !defined(__arm__)/* Interesting ... it seems libc knows about MNT_FORCE and presumably   about umount2 as well -- need not do anything */#else /* MNT_FORCE *//* Does the present kernel source know about umount2? */#include <linux/unistd.h>#ifdef __NR_umount2static int umount2(const char *path, int flags);_syscall2(int, umount2, const char *, path, int, flags);#else /* __NR_umount2 */static intumount2(const char *path, int flags) {	fprintf(stderr, _("umount: compiled without support for -f\n"));	errno = ENOSYS;	return -1;}#endif /* __NR_umount2 */#if !defined(MNT_FORCE)/* dare not try to include <linux/mount.h> -- lots of errors */#define MNT_FORCE 1#endif#endif /* MNT_FORCE */#if !defined(MNT_DETACH)#define MNT_DETACH 2#endif/* True if we are allowed to call /sbin/umount.${FSTYPE} */int external_allowed = 1;/* Nonzero for force umount (-f).  There is kernel support since 2.1.116.  */int force = 0;/* Nonzero for lazy umount (-l).  There is kernel support since 2.4.11.  */int lazy = 0;/* When umount fails, attempt a read-only remount (-r). */int remount = 0;/* Don't write a entry in /etc/mtab (-n).  */int nomtab = 0;/* Call losetup -d for each unmounted loop device. */int delloop = 0;/* Nonzero for chatty (-v). */int verbose = 0;/* True if ruid != euid.  */int suid = 0;/* * check_special_umountprog() *	If there is a special umount program for this type, exec it. * returns: 0: no exec was done, 1: exec was done, status has result */static intcheck_special_umountprog(const char *spec, const char *node,			 const char *type, int *status) {	char umountprog[120];	struct stat statbuf;	int res;	if (!external_allowed)		return 0;	if (type && strlen(type) < 100) {		sprintf(umountprog, "/sbin/umount.%s", type);		if (stat(umountprog, &statbuf) == 0) {			res = fork();			if (res == 0) {				char *umountargs[8];				int i = 0;				setuid(getuid());				setgid(getgid());				umountargs[i++] = umountprog;				umountargs[i++] = xstrdup(node);				if (nomtab)					umountargs[i++] = "-n";				if (lazy)					umountargs[i++] = "-l";				if (force)					umountargs[i++] = "-f";				if (verbose)					umountargs[i++] = "-v";				if (remount)					umountargs[i++] = "-r";				umountargs[i] = NULL;				execv(umountprog, umountargs);				exit(1);	/* exec failed */			} else if (res != -1) {				int st;				wait(&st);				*status = (WIFEXITED(st) ? WEXITSTATUS(st)					   : EX_SYSERR);				return 1;			} else {				int errsv = errno;				error(_("umount: cannot fork: %s"),				      strerror(errsv));			}		}	}	return 0;}#ifdef HAVE_NFSstatic int xdr_dir(XDR *xdrsp, char *dirp){      return (xdr_string(xdrsp, &dirp, MNTPATHLEN));}static intnfs_umount_rpc_call(const char *spec, const char *opts){      register CLIENT *clp;      struct sockaddr_in saddr;      struct timeval pertry, try;      enum clnt_stat clnt_stat;      int port = 0;      int so = RPC_ANYSOCK;      struct hostent *hostp;      char *hostname;      char *dirname;      char *p;      if (spec == NULL || (p = strchr(spec,':')) == NULL)		return 0;      hostname = xstrndup(spec, p-spec);      dirname = xstrdup(p+1);#ifdef DEBUG      printf(_("host: %s, directory: %s\n"), hostname, dirname);#endif      if (opts && (p = strstr(opts, "addr="))) {	   char *q;	   free(hostname);	   p += 5;	   q = p;	   while (*q && *q != ',') q++;	   hostname = xstrndup(p,q-p);      }      if (opts && (p = strstr(opts, "mountport=")) && isdigit(*(p+10)))	   port = atoi(p+10);      if (hostname[0] >= '0' && hostname[0] <= '9')	   saddr.sin_addr.s_addr = inet_addr(hostname);      else {	   if ((hostp = gethostbyname(hostname)) == NULL) {		fprintf(stderr, _("umount: can't get address for %s\n"),			hostname);		return 1;	   }	   if (hostp->h_length > sizeof(struct in_addr)) {		fprintf(stderr, _("umount: got bad hostp->h_length\n"));		hostp->h_length = sizeof(struct in_addr);	   }	   memcpy(&saddr.sin_addr, hostp->h_addr, hostp->h_length);      }      saddr.sin_family = AF_INET;      saddr.sin_port = htons(port);      pertry.tv_sec = 3;      pertry.tv_usec = 0;      if (opts && (p = strstr(opts, "tcp"))) {	   /* possibly: make sure option is not "notcp"	      possibly: try udp if tcp fails */	   if ((clp = clnttcp_create(&saddr, MOUNTPROG, MOUNTVERS,				     &so, 0, 0)) == NULL) {		clnt_pcreateerror("Cannot MOUNTPROG RPC (tcp)");		return 1;	   }      } else {           if ((clp = clntudp_create(&saddr, MOUNTPROG, MOUNTVERS,				     pertry, &so)) == NULL) {		clnt_pcreateerror("Cannot MOUNTPROG RPC");		return 1;	   }      }      clp->cl_auth = authunix_create_default();      try.tv_sec = 20;      try.tv_usec = 0;      clnt_stat = clnt_call(clp, MOUNTPROC_UMNT,			    (xdrproc_t) xdr_dir, dirname,			    (xdrproc_t) xdr_void, (caddr_t) 0,			    try);      if (clnt_stat != RPC_SUCCESS) {	   clnt_perror(clp, "Bad UMNT RPC");	   return 1;      }      auth_destroy(clp->cl_auth);      clnt_destroy(clp);      return 0;}#endif /* HAVE_NFS *//* complain about a failed umount */static void complain(int err, const char *dev) {  switch (err) {    case ENXIO:      error (_("umount: %s: invalid block device"), dev); break;    case EINVAL:      error (_("umount: %s: not mounted"), dev); break;    case EIO:      error (_("umount: %s: can't write superblock"), dev); break;    case EBUSY:     /* Let us hope fstab has a line "proc /proc ..."	and not "none /proc ..."*/      error (_("umount: %s: device is busy"), dev); break;    case ENOENT:      error (_("umount: %s: not found"), dev); break;    case EPERM:      error (_("umount: %s: must be superuser to umount"), dev); break;    case EACCES:      error (_("umount: %s: block devices not permitted on fs"), dev); break;    default:      error (_("umount: %s: %s"), dev, strerror (err)); break;  }}/* Umount a single device.  Return a status code, so don't exit   on a non-fatal error.  We lock/unlock around each umount.  */static intumount_one (const char *spec, const char *node, const char *type,	    const char *opts, struct mntentchn *mc) {	int umnt_err, umnt_err2;	int isroot;	int res;	int status;	const char *loopdev;	/* Special case for root.  As of 0.99pl10 we can (almost) unmount root;	   the kernel will remount it readonly so that we can carry on running	   afterwards.  The readonly remount is illegal if any files are opened	   for writing at the time, so we can't update mtab for an unmount of	   root.  As it is only really a remount, this doesn't matter too	   much.  [sct May 29, 1993] */	isroot = (streq (node, "/") || streq (node, "root")		  || streq (node, "rootfs"));	if (isroot)		nomtab++;		/*	 * Call umount.TYPE for types that require a separate umount program.	 * All such special things must occur isolated in the types string.	 */	if (check_special_umountprog(spec, node, type, &status))		return status;#ifdef HAVE_NFS	/* Ignore any RPC errors, so that you can umount the filesystem	   if the server is down.  */	if (strcasecmp(type, "nfs") == 0)		nfs_umount_rpc_call(spec, opts);#endif 	umnt_err = umnt_err2 = 0;	if (lazy) {		res = umount2 (node, MNT_DETACH);		if (res < 0)			umnt_err = errno;		goto writemtab;	}	if (force) {		/* only supported for NFS */		res = umount2 (node, MNT_FORCE);		if (res == -1) {			int errsv = errno;			perror("umount2");			errno = errsv;			if (errno == ENOSYS) {				if (verbose)					printf(_("no umount2, trying umount...\n"));				res = umount (node);			}		}	} else		res = umount (node);	if (res < 0) {		umnt_err = errno;		/* A device might have been mounted on a node that has since		   been deleted or renamed, so if node fails, also try spec. */		/* Note that this is incorrect in case spec was mounted		   several times. */		/* if (umnt_err == ENOENT || umnt_err == EINVAL) */		if (umnt_err != EBUSY && strcmp(node, spec)) {			if (verbose)				printf (_("could not umount %s - trying %s instead\n"),					node, spec);			res = umount (spec);			if (res < 0)				umnt_err2 = errno;			/* Do not complain about remote NFS mount points */			if (errno == ENOENT && index(spec, ':'))				umnt_err2 = 0;		}	}	if (res < 0 && remount && (umnt_err == EBUSY || umnt_err2 == EBUSY)) {		/* Umount failed - let us try a remount */		res = mount(spec, node, NULL,			    MS_MGC_VAL | MS_REMOUNT | MS_RDONLY, NULL);		if (res == 0) {			struct my_mntent remnt;			fprintf(stderr,				_("umount: %s busy - remounted read-only\n"),				spec);			remnt.mnt_type = remnt.mnt_fsname = NULL;			remnt.mnt_dir = xstrdup(node);			remnt.mnt_opts = xstrdup("ro");			update_mtab(node, &remnt);			return 0;		} else if (errno != EBUSY) { 	/* hmm ... */			perror("remount");			fprintf(stderr,				_("umount: could not remount %s read-only\n"),				spec);

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -