/* Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The ASF licenses this file to You under the Apache License, Version 2.0 * (the "License"); you may not use this file except in compliance with * the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "fspr_arch_file_io.h" #include "fsio.h" #include "nks/dirio.h" #include "fspr_file_io.h" #include "fspr_general.h" #include "fspr_strings.h" #include "fspr_errno.h" #include "fspr_hash.h" #include "fspr_thread_rwlock.h" #ifdef HAVE_UTIME_H #include #endif #define APR_HAS_PSA static fspr_filetype_e filetype_from_mode(mode_t mode) { fspr_filetype_e type = APR_NOFILE; if (S_ISREG(mode)) type = APR_REG; else if (S_ISDIR(mode)) type = APR_DIR; else if (S_ISCHR(mode)) type = APR_CHR; else if (S_ISBLK(mode)) type = APR_BLK; else if (S_ISFIFO(mode)) type = APR_PIPE; else if (S_ISLNK(mode)) type = APR_LNK; else if (S_ISSOCK(mode)) type = APR_SOCK; else type = APR_UNKFILE; return type; } static void fill_out_finfo(fspr_finfo_t *finfo, struct stat *info, fspr_int32_t wanted) { finfo->valid = APR_FINFO_MIN | APR_FINFO_IDENT | APR_FINFO_NLINK | APR_FINFO_OWNER | APR_FINFO_PROT; finfo->protection = fspr_unix_mode2perms(info->st_mode); finfo->filetype = filetype_from_mode(info->st_mode); finfo->user = info->st_uid; finfo->group = info->st_gid; finfo->size = info->st_size; finfo->inode = info->st_ino; finfo->device = info->st_dev; finfo->nlink = info->st_nlink; fspr_time_ansi_put(&finfo->atime, info->st_atime.tv_sec); fspr_time_ansi_put(&finfo->mtime, info->st_mtime.tv_sec); fspr_time_ansi_put(&finfo->ctime, info->st_ctime.tv_sec); /* ### needs to be revisited * if (wanted & APR_FINFO_CSIZE) { * finfo->csize = info->st_blocks * 512; * finfo->valid |= APR_FINFO_CSIZE; * } */ } APR_DECLARE(fspr_status_t) fspr_file_info_get(fspr_finfo_t *finfo, fspr_int32_t wanted, fspr_file_t *thefile) { struct stat info; if (thefile->buffered) { fspr_status_t rv = fspr_file_flush(thefile); if (rv != APR_SUCCESS) return rv; } if (fstat(thefile->filedes, &info) == 0) { finfo->pool = thefile->pool; finfo->fname = thefile->fname; fill_out_finfo(finfo, &info, wanted); return (wanted & ~finfo->valid) ? APR_INCOMPLETE : APR_SUCCESS; } else { return errno; } } APR_DECLARE(fspr_status_t) fspr_file_perms_set(const char *fname, fspr_fileperms_t perms) { mode_t mode = fspr_unix_perms2mode(perms); if (chmod(fname, mode) == -1) return errno; return APR_SUCCESS; } APR_DECLARE(fspr_status_t) fspr_file_attrs_set(const char *fname, fspr_fileattrs_t attributes, fspr_fileattrs_t attr_mask, fspr_pool_t *pool) { fspr_status_t status; fspr_finfo_t finfo; /* Don't do anything if we can't handle the requested attributes */ if (!(attr_mask & (APR_FILE_ATTR_READONLY | APR_FILE_ATTR_EXECUTABLE))) return APR_SUCCESS; status = fspr_stat(&finfo, fname, APR_FINFO_PROT, pool); if (status) return status; /* ### TODO: should added bits be umask'd? */ if (attr_mask & APR_FILE_ATTR_READONLY) { if (attributes & APR_FILE_ATTR_READONLY) { finfo.protection &= ~APR_UWRITE; finfo.protection &= ~APR_GWRITE; finfo.protection &= ~APR_WWRITE; } else { /* ### umask this! */ finfo.protection |= APR_UWRITE; finfo.protection |= APR_GWRITE; finfo.protection |= APR_WWRITE; } } if (attr_mask & APR_FILE_ATTR_EXECUTABLE) { if (attributes & APR_FILE_ATTR_EXECUTABLE) { /* ### umask this! */ finfo.protection |= APR_UEXECUTE; finfo.protection |= APR_GEXECUTE; finfo.protection |= APR_WEXECUTE; } else { finfo.protection &= ~APR_UEXECUTE; finfo.protection &= ~APR_GEXECUTE; finfo.protection &= ~APR_WEXECUTE; } } return fspr_file_perms_set(fname, finfo.protection); } #ifndef APR_HAS_PSA static fspr_status_t stat_cache_cleanup(void *data) { fspr_pool_t *p = (fspr_pool_t *)getGlobalPool(); fspr_hash_index_t *hi; fspr_hash_t *statCache = (fspr_hash_t*)data; char *key; fspr_ssize_t keylen; NXPathCtx_t pathctx; for (hi = fspr_hash_first(p, statCache); hi; hi = fspr_hash_next(hi)) { fspr_hash_this(hi, (const void**)&key, &keylen, (void**)&pathctx); if (pathctx) { NXFreePathContext(pathctx); } } return APR_SUCCESS; } int cstat (NXPathCtx_t ctx, char *path, struct stat *buf, unsigned long requestmap, fspr_pool_t *p) { fspr_pool_t *gPool = (fspr_pool_t *)getGlobalPool(); fspr_hash_t *statCache = NULL; fspr_thread_rwlock_t *rwlock = NULL; NXPathCtx_t pathctx = 0; char *ptr = NULL, *tr; int len = 0, x; char *ppath; char *pinfo; if (ctx == 1) { /* If there isn't a global pool then just stat the file and return */ if (!gPool) { char poolname[50]; if (fspr_pool_create(&gPool, NULL) != APR_SUCCESS) { return getstat(ctx, path, buf, requestmap); } setGlobalPool(gPool); fspr_pool_tag(gPool, fspr_pstrdup(gPool, "cstat_mem_pool")); statCache = fspr_hash_make(gPool); fspr_pool_userdata_set ((void*)statCache, "STAT_CACHE", stat_cache_cleanup, gPool); fspr_thread_rwlock_create(&rwlock, gPool); fspr_pool_userdata_set ((void*)rwlock, "STAT_CACHE_LOCK", fspr_pool_cleanup_null, gPool); } else { fspr_pool_userdata_get((void**)&statCache, "STAT_CACHE", gPool); fspr_pool_userdata_get((void**)&rwlock, "STAT_CACHE_LOCK", gPool); } if (!gPool || !statCache || !rwlock) { return getstat(ctx, path, buf, requestmap); } for (x = 0,tr = path;*tr != '\0';tr++,x++) { if (*tr == '\\' || *tr == '/') { ptr = tr; len = x; } if (*tr == ':') { ptr = "\\"; len = x; } } if (ptr) { ppath = fspr_pstrndup (p, path, len); strlwr(ppath); if (ptr[1] != '\0') { ptr++; } /* If the path ended in a trailing slash then our result path will be a single slash. To avoid stat'ing the root with a slash, we need to make sure we stat the current directory with a dot */ if (((*ptr == '/') || (*ptr == '\\')) && (*(ptr+1) == '\0')) { pinfo = fspr_pstrdup (p, "."); } else { pinfo = fspr_pstrdup (p, ptr); } } /* If we have a statCache then try to pull the information from the cache. Otherwise just stat the file and return.*/ if (statCache) { fspr_thread_rwlock_rdlock(rwlock); pathctx = (NXPathCtx_t) fspr_hash_get(statCache, ppath, APR_HASH_KEY_STRING); fspr_thread_rwlock_unlock(rwlock); if (pathctx) { return getstat(pathctx, pinfo, buf, requestmap); } else { int err; err = NXCreatePathContext(0, ppath, 0, NULL, &pathctx); if (!err) { fspr_thread_rwlock_wrlock(rwlock); fspr_hash_set(statCache, fspr_pstrdup(gPool,ppath) , APR_HASH_KEY_STRING, (void*)pathctx); fspr_thread_rwlock_unlock(rwlock); return getstat(pathctx, pinfo, buf, requestmap); } } } } return getstat(ctx, path, buf, requestmap); } #endif APR_DECLARE(fspr_status_t) fspr_stat(fspr_finfo_t *finfo, const char *fname, fspr_int32_t wanted, fspr_pool_t *pool) { struct stat info; int srv; NXPathCtx_t pathCtx = 0; getcwdpath(NULL, &pathCtx, CTX_ACTUAL_CWD); #ifdef APR_HAS_PSA srv = getstat(pathCtx, (char*)fname, &info, ST_STAT_BITS|ST_NAME_BIT); #else srv = cstat(pathCtx, (char*)fname, &info, ST_STAT_BITS|ST_NAME_BIT, pool); #endif errno = srv; if (srv == 0) { finfo->pool = pool; finfo->fname = fname; fill_out_finfo(finfo, &info, wanted); if (wanted & APR_FINFO_LINK) wanted &= ~APR_FINFO_LINK; if (wanted & APR_FINFO_NAME) { finfo->name = fspr_pstrdup(pool, info.st_name); finfo->valid |= APR_FINFO_NAME; } return (wanted & ~finfo->valid) ? APR_INCOMPLETE : APR_SUCCESS; } else { #if !defined(ENOENT) || !defined(ENOTDIR) #error ENOENT || ENOTDIR not defined; please see the #error comments at this line in the source for a workaround. /* * If ENOENT || ENOTDIR is not defined in one of the your OS's * include files, APR cannot report a good reason why the stat() * of the file failed; there are cases where it can fail even though * the file exists. This opens holes in Apache, for example, because * it becomes possible for someone to get a directory listing of a * directory even though there is an index (eg. index.html) file in * it. If you do not have a problem with this, delete the above * #error lines and start the compile again. If you need to do this, * please submit a bug report to http://www.apache.org/bug_report.html * letting us know that you needed to do this. Please be sure to * include the operating system you are using. */ /* WARNING: All errors will be handled as not found */ #if !defined(ENOENT) return APR_ENOENT; #else /* WARNING: All errors but not found will be handled as not directory */ if (errno != ENOENT) return APR_ENOENT; else return errno; #endif #else /* All was defined well, report the usual: */ return errno; #endif } } APR_DECLARE(fspr_status_t) fspr_file_mtime_set(const char *fname, fspr_time_t mtime, fspr_pool_t *pool) { fspr_status_t status; fspr_finfo_t finfo; status = fspr_stat(&finfo, fname, APR_FINFO_ATIME, pool); if (status) { return status; } #ifdef HAVE_UTIMES { struct timeval tvp[2]; tvp[0].tv_sec = fspr_time_sec(finfo.atime); tvp[0].tv_usec = fspr_time_usec(finfo.atime); tvp[1].tv_sec = fspr_time_sec(mtime); tvp[1].tv_usec = fspr_time_usec(mtime); if (utimes(fname, tvp) == -1) { return errno; } } #elif defined(HAVE_UTIME) { struct utimbuf buf; buf.actime = (time_t) (finfo.atime / APR_USEC_PER_SEC); buf.modtime = (time_t) (mtime / APR_USEC_PER_SEC); if (utime(fname, &buf) == -1) { return errno; } } #else return APR_ENOTIMPL; #endif return APR_SUCCESS; }