Skip site navigation (1)Skip section navigation (2)
Date:      Tue, 26 Feb 2002 04:20:09 -0800 (PST)
From:      "Jose M. Alcaide" <jose@we.lc.ehu.es>
To:        freebsd-bugs@FreeBSD.org
Subject:   Re: bin/12244: realpath() fails when there is no permission to read the WD
Message-ID:  <200202261220.g1QCK9C67924@freefall.freebsd.org>

next in thread | raw e-mail | index | archive | help
The following reply was made to PR bin/12244; it has been noted by GNATS.

From: "Jose M. Alcaide" <jose@we.lc.ehu.es>
To: Sheldon Hearn <sheldonh@starjuice.net>
Cc: iedowse@FreeBSD.org, bug-followup@FreeBSD.org
Subject: Re: bin/12244: realpath() fails when there is no permission to read the WD
Date: Tue, 26 Feb 2002 13:19:09 +0100

 --UlVJffcvxoiEqYs2
 Content-Type: text/plain; charset=us-ascii
 Content-Disposition: inline
 
 On Thu, Jan 31, 2002 at 05:13:11PM +0200, Sheldon Hearn wrote:
 > On Mon, 21 Jan 2002 02:15:34 +0100, "Jose M. Alcaide" wrote:
 > > Yes, it is. Realpath(3) continues using open()/fchdir() for
 > > saving/restoring the current directory. I think that the getwd()/chdir()
 > > method should be used as fallback.
 > 
 > Could you submit a patch for that?
 
 I have written a patch (attached). I did not tested it, but it is pretty
 trivial. It is somewhat ugly, too ;-)
 
 -- 
 ****** Jose M. Alcaide  //  jose@we.lc.ehu.es  //  jmas@FreeBSD.org ******
 ** "Beware of Programmers who carry screwdrivers" --  Leonard Brandwein **
 
 --UlVJffcvxoiEqYs2
 Content-Type: text/plain; charset=us-ascii
 Content-Disposition: attachment; filename="realpath.c.patch"
 
 --- lib/libc/stdlib/realpath.c.orig	Fri Jan 28 00:06:50 2000
 +++ lib/libc/stdlib/realpath.c	Tue Feb 26 13:16:31 2002
 @@ -63,14 +63,19 @@
  {
  	struct stat sb;
  	int fd, n, rootd, serrno;
 -	char *p, *q, wbuf[MAXPATHLEN];
 +	char *p, *q, wbuf[MAXPATHLEN], scwd[MAXPATHLEN];
  	int symlinks = 0;
  
 -	/* Save the starting point. */
 -	if ((fd = _open(".", O_RDONLY)) < 0) {
 -		(void)strcpy(resolved, ".");
 -		return (NULL);
 -	}
 +	/*
 +	 * Save the starting point:
 +	 * First, try open()/fchdir() method and, if this fails,
 +	 * then try getcwd()/chdir().
 +	 */
 +	if ((fd = _open(".", O_RDONLY)) < 0)
 +		if (getcwd(scwd, MAXPATHLEN) == 0) {
 +			(void)strcpy(resolved, ".");
 +			return (NULL);
 +		}
  
  	/*
  	 * Find the dirname and basename from the path to be resolved.
 @@ -96,7 +101,7 @@
  			q = resolved;
  		}
  		if (chdir(q) < 0)
 -			goto err1;
 +			goto err;
  	} else
  		p = resolved;
  
 @@ -105,17 +110,17 @@
  		if (S_ISLNK(sb.st_mode)) {
  			if (++symlinks > MAXSYMLINKS) {
  				errno = ELOOP;
 -				goto err1;
 +				goto err;
  			}
  			n = readlink(p, resolved, MAXPATHLEN - 1);
  			if (n < 0)
 -				goto err1;
 +				goto err;
  			resolved[n] = '\0';
  			goto loop;
  		}
  		if (S_ISDIR(sb.st_mode)) {
  			if (chdir(p) < 0)
 -				goto err1;
 +				goto err;
  			p = "";
  		}
  	}
 @@ -126,7 +131,7 @@
  	 */
  	(void)strcpy(wbuf, p);
  	if (getcwd(resolved, MAXPATHLEN) == 0)
 -		goto err1;
 +		goto err;
  
  	/*
  	 * Join the two strings together, ensuring that the right thing
 @@ -140,26 +145,38 @@
  	if (*wbuf) {
  		if (strlen(resolved) + strlen(wbuf) + rootd + 1 > MAXPATHLEN) {
  			errno = ENAMETOOLONG;
 -			goto err1;
 +			goto err;
  		}
  		if (rootd == 0)
  			(void)strcat(resolved, "/");
  		(void)strcat(resolved, wbuf);
  	}
  
 -	/* Go back to where we came from. */
 -	if (fchdir(fd) < 0) {
 -		serrno = errno;
 -		goto err2;
 -	}
 +	/*
 +	 * Go back to where we came from. If fd < 0, we are using the
 +	 * getcwd()/chdir() method; else, we are using open()/fchdir().
 +	 */
 +	if (fd < 0) {
 +		if (chdir(scwd) < 0)
 +			return (NULL);
 +	} else  if (fchdir(fd) < 0) {
 +			serrno = errno;
 +			(void)_close(fd);
 +			errno = serrno;
 +			return (NULL);
 +		}
  
  	/* It's okay if the close fails, what's an fd more or less? */
  	(void)_close(fd);
  	return (resolved);
  
 -err1:	serrno = errno;
 -	(void)fchdir(fd);
 -err2:	(void)_close(fd);
 +err:	serrno = errno;
 +	if (fd < 0)
 +		(void)chdir(scwd);
 +	else {
 +		(void)fchdir(fd);
 +		(void)_close(fd);
 +	}
  	errno = serrno;
  	return (NULL);
  }
 
 --UlVJffcvxoiEqYs2--

To Unsubscribe: send mail to majordomo@FreeBSD.org
with "unsubscribe freebsd-bugs" in the body of the message




Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?200202261220.g1QCK9C67924>