getcwd.c
上传用户:zibowangxu
上传日期:2007-01-04
资源大小:331k
文件大小:6k
源码类别:

Ftp客户端

开发平台:

Unix_Linux

  1. /****************************************************************************  
  2.  
  3.   Copyright (c) 1999 WU-FTPD Development Group.  
  4.   All rights reserved.
  5.   
  6.   Portions Copyright (c) 1980, 1985, 1988, 1989, 1990, 1991, 1993, 1994
  7.     The Regents of the University of California.
  8.   Portions Copyright (c) 1993, 1994 Washington University in Saint Louis.
  9.   Portions Copyright (c) 1996, 1998 Berkeley Software Design, Inc.
  10.   Portions Copyright (c) 1989 Massachusetts Institute of Technology.
  11.   Portions Copyright (c) 1998 Sendmail, Inc.
  12.   Portions Copyright (c) 1983, 1995, 1996, 1997 Eric P.  Allman.
  13.   Portions Copyright (c) 1997 by Stan Barber.
  14.   Portions Copyright (c) 1997 by Kent Landfield.
  15.   Portions Copyright (c) 1991, 1992, 1993, 1994, 1995, 1996, 1997
  16.     Free Software Foundation, Inc.  
  17.  
  18.   Use and distribution of this software and its source code are governed 
  19.   by the terms and conditions of the WU-FTPD Software License ("LICENSE").
  20.  
  21.   If you did not receive a copy of the license, it may be obtained online
  22.   at http://www.wu-ftpd.org/license.html.
  23.  
  24.   $Id: getcwd.c,v 1.7 1999/08/24 23:25:37 wuftpd Exp $
  25.  
  26. ****************************************************************************/
  27. /*
  28.  * portable version of getcwd()
  29.  */
  30. #include "../src/config.h"
  31. #include <string.h>
  32. #include <sys/param.h>
  33. #include <sys/types.h>
  34. #ifdef HAVE_DIRENT_H
  35. #include <dirent.h>
  36. #else
  37. #include <sys/dir.h>
  38. #define dirent direct
  39. #endif /* DIRENT */
  40. #include <sys/stat.h>
  41. #include <signal.h>
  42. #include <stdlib.h>
  43. #include <sys/errno.h>
  44. extern int errno;
  45. extern int snprintf();
  46. static char *strnrcpy();
  47. #ifndef HAVE_LSTAT
  48. #define lstat stat
  49. #endif
  50. #define CGETS(b, c, d)        d
  51. /* return true if dp is of the form "../xxx" or "/../xxx" */
  52. #define TRM(a) ((a) & TRIM)
  53. #define NTRM(a) (a)
  54. #define ISDOT(c) (NTRM((c)[0]) == '.' && ((NTRM((c)[1]) == '') || 
  55.                   (NTRM((c)[1]) == '/')))
  56. #define ISDOTDOT(c) (NTRM((c)[0]) == '.' && ISDOT(&((c)[1])))
  57. #ifndef DEV_DEV_COMPARE
  58. #define DEV_DEV_COMPARE(x,y)   ((x) == (y))
  59. #endif /* DEV_DEV_COMPARE */
  60. /* getcwd():
  61.  *    Return the pathname of the current directory, or return
  62.  *      an error message in pathname.
  63.  */
  64. #if (SYSVREL != 0 && !defined(d_fileno)) || defined(_VMS_POSIX) || 
  65. (defined(AIX) && !defined(d_fileno)) || 
  66. defined(WINNT)
  67. #define d_fileno d_ino
  68. #endif
  69. char *
  70.      getcwd(pathname, pathlen)
  71.      char *pathname;
  72.      size_t pathlen;
  73. {
  74.     DIR *dp;
  75.     struct dirent *d;
  76.     struct stat st_root, st_cur, st_next, st_dotdot;
  77.     char pathbuf[MAXPATHLEN], nextpathbuf[MAXPATHLEN * 2];
  78.     char *pathptr, *nextpathptr, *cur_name_add;
  79.     int save_errno = 0;
  80.     /* find the inode of root */
  81.     if (stat("/", &st_root) == -1) {
  82. (void) snprintf(pathname, pathlen, CGETS(23, 23,
  83.   "getcwd: Cannot stat "/" (%s)"),
  84. strerror(errno));
  85. return NULL;
  86.     }
  87.     pathbuf[MAXPATHLEN - 1] = '';
  88.     pathptr = &pathbuf[MAXPATHLEN - 1];
  89.     nextpathbuf[MAXPATHLEN - 1] = '';
  90.     cur_name_add = nextpathptr = &nextpathbuf[MAXPATHLEN - 1];
  91.     /* find the inode of the current directory */
  92.     if (lstat(".", &st_cur) == -1) {
  93. (void) snprintf(pathname, pathlen, CGETS(23, 24,
  94.   "getcwd: Cannot stat "." (%s)"),
  95. strerror(errno));
  96. return NULL;
  97.     }
  98.     nextpathptr = strnrcpy(nextpathptr, "../", nextpathptr - nextpathbuf);
  99.     /* Descend to root */
  100.     for (;;) {
  101. /* look if we found root yet */
  102. if (st_cur.st_ino == st_root.st_ino &&
  103.     DEV_DEV_COMPARE(st_cur.st_dev, st_root.st_dev)) {
  104.     (void) strncpy(pathname, *pathptr != '/' ? "/" : pathptr, pathlen);
  105.     pathname[pathlen - 1] = '';
  106.     return pathname;
  107. }
  108. /* open the parent directory */
  109. if (stat(nextpathptr, &st_dotdot) == -1) {
  110.     (void) snprintf(pathname, pathlen, CGETS(23, 25,
  111.        "getcwd: Cannot stat directory "%s" (%s)"),
  112.     nextpathptr, strerror(errno));
  113.     return NULL;
  114. }
  115. if ((dp = opendir(nextpathptr)) == NULL) {
  116.     (void) snprintf(pathname, pathlen, CGETS(23, 26,
  117.        "getcwd: Cannot open directory "%s" (%s)"),
  118.     nextpathptr, strerror(errno));
  119.     return NULL;
  120. }
  121. /* look in the parent for the entry with the same inode */
  122. if (DEV_DEV_COMPARE(st_dotdot.st_dev, st_cur.st_dev)) {
  123.     /* Parent has same device. No need to stat every member */
  124.     for (d = readdir(dp); d != NULL; d = readdir(dp)) {
  125. #ifdef __clipper__
  126. if (((unsigned long) d->d_fileno & 0xffff) == st_cur.st_ino)
  127.     break;
  128. #else
  129. if (d->d_fileno == st_cur.st_ino)
  130.     break;
  131. #endif
  132.     }
  133. }
  134. else {
  135.     /*
  136.      * Parent has a different device. This is a mount point so we
  137.      * need to stat every member
  138.      */
  139.     for (d = readdir(dp); d != NULL; d = readdir(dp)) {
  140. if (ISDOT(d->d_name) || ISDOTDOT(d->d_name))
  141.     continue;
  142. (void) strncpy(cur_name_add, d->d_name,
  143.       &nextpathbuf[sizeof(nextpathbuf) - 1] - cur_name_add);
  144. if (lstat(nextpathptr, &st_next) == -1) {
  145.     /*
  146.      * We might not be able to stat() some path components
  147.      * if we are using afs, but this is not an error as
  148.      * long as we find the one we need; we also save the
  149.      * first error to report it if we don't finally succeed.
  150.      */
  151.     if (save_errno == 0)
  152. save_errno = errno;
  153.     continue;
  154. }
  155. /* check if we found it yet */
  156. if (st_next.st_ino == st_cur.st_ino &&
  157.     DEV_DEV_COMPARE(st_next.st_dev, st_cur.st_dev))
  158.     break;
  159.     }
  160. }
  161. if (d == NULL) {
  162.     (void) snprintf(pathname, pathlen, CGETS(23, 27,
  163. "getcwd: Cannot find "." in ".." (%s)"),
  164.     strerror(save_errno ? save_errno : ENOENT));
  165.     (void) closedir(dp);
  166.     return NULL;
  167. }
  168. else
  169.     save_errno = 0;
  170. st_cur = st_dotdot;
  171. pathptr = strnrcpy(pathptr, d->d_name, pathptr - pathbuf);
  172. pathptr = strnrcpy(pathptr, "/", pathptr - pathbuf);
  173. nextpathptr = strnrcpy(nextpathptr, "../", nextpathptr - nextpathbuf);
  174. *cur_name_add = '';
  175. (void) closedir(dp);
  176.     }
  177. } /* end getcwd */
  178. /* strnrcpy():
  179.  *    Like strncpy, going backwards and returning the new pointer
  180.  */
  181. static char *
  182.      strnrcpy(ptr, str, siz)
  183.      register char *ptr, *str;
  184.      size_t siz;
  185. {
  186.     register int len = strlen(str);
  187.     if (siz == 0)
  188. return ptr;
  189.     while (len && siz--)
  190. *--ptr = str[--len];
  191.     return (ptr);
  192. } /* end strnrcpy */