2006-07-03 22:23:07 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2004-2005 The Trustees of Indiana University and Indiana
|
|
|
|
* University Research and Technology
|
|
|
|
* Corporation. All rights reserved.
|
|
|
|
* Copyright (c) 2004-2005 The University of Tennessee and The University
|
|
|
|
* of Tennessee Research Foundation. All rights
|
|
|
|
* reserved.
|
|
|
|
* Copyright (c) 2004-2005 High Performance Computing Center Stuttgart,
|
|
|
|
* University of Stuttgart. All rights reserved.
|
|
|
|
* Copyright (c) 2004-2005 The Regents of the University of California.
|
|
|
|
* All rights reserved.
|
|
|
|
* $COPYRIGHT$
|
|
|
|
*
|
|
|
|
* Additional copyrights may follow
|
|
|
|
*
|
|
|
|
* $HEADER$
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include "opal_config.h"
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
#include <unistd.h>
|
2007-04-01 16:16:54 +00:00
|
|
|
#endif /* HAVE_UNISTD_H */
|
2006-07-03 22:23:07 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#ifdef HAVE_SYS_TYPES_H
|
|
|
|
#include <sys/types.h>
|
2007-04-01 16:16:54 +00:00
|
|
|
#endif /* HAVE_SYS_TYPES_H */
|
2006-07-03 22:23:07 +00:00
|
|
|
#ifdef HAVE_DIRENT_H
|
|
|
|
#include <dirent.h>
|
2007-04-01 16:16:54 +00:00
|
|
|
#endif /* HAVE_DIRENT_H */
|
2006-07-03 22:23:07 +00:00
|
|
|
|
|
|
|
#include "opal/util/os_dirpath.h"
|
|
|
|
#include "opal/util/argv.h"
|
|
|
|
#include "opal/util/os_path.h"
|
|
|
|
#include "opal/constants.h"
|
|
|
|
|
2006-08-22 23:25:13 +00:00
|
|
|
static const char path_sep[] = OPAL_PATH_SEP;
|
2006-07-03 22:23:07 +00:00
|
|
|
|
|
|
|
int opal_os_dirpath_create(const char *path, const mode_t mode)
|
|
|
|
{
|
|
|
|
struct stat buf;
|
|
|
|
char **parts, *tmp;
|
|
|
|
int i, len;
|
|
|
|
|
|
|
|
if (NULL == path) { /* protect ourselves from errors */
|
|
|
|
return(OPAL_ERROR);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (0 == stat(path, &buf)) { /* already exists */
|
|
|
|
if (mode == (mode & buf.st_mode)) { /* has correct mode */
|
|
|
|
return(OPAL_SUCCESS);
|
|
|
|
}
|
|
|
|
if (0 == chmod(path, (buf.st_mode | mode))) { /* successfully change mode */
|
|
|
|
return(OPAL_SUCCESS);
|
|
|
|
}
|
|
|
|
return(OPAL_ERROR); /* can't set correct mode */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* quick -- try to make directory */
|
|
|
|
if (0 == mkdir(path, mode)) {
|
|
|
|
return(OPAL_SUCCESS);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* didnt work, so now have to build our way down the tree */
|
|
|
|
/* Split the requested path up into its individual parts */
|
|
|
|
|
|
|
|
parts = opal_argv_split(path, path_sep[0]);
|
|
|
|
|
|
|
|
/* Ensure to allocate enough space for tmp: the strlen of the
|
|
|
|
incoming path + 1 (for \0) */
|
|
|
|
|
2006-08-22 20:07:42 +00:00
|
|
|
tmp = (char*)malloc(strlen(path) + 1);
|
2006-07-03 22:23:07 +00:00
|
|
|
tmp[0] = '\0';
|
|
|
|
|
|
|
|
/* Iterate through all the subdirectory names in the path,
|
|
|
|
building up a directory name. Check to see if that dirname
|
|
|
|
exists. If it doesn't, create it. */
|
|
|
|
|
|
|
|
/* Notes about stat(): Windows has funny definitions of what will
|
|
|
|
return 0 from stat(). "C:" will return failure, while "C:\"
|
|
|
|
will return success. Similarly, "C:\foo" will return success,
|
|
|
|
while "C:\foo\" will return failure (assuming that a folder
|
|
|
|
named "foo" exists under C:\).
|
|
|
|
|
|
|
|
POSIX implementations of stat() are generally a bit more
|
|
|
|
forgiving; most will return true for "/foo" and "/foo/"
|
|
|
|
(assuming /foo exists). But we might as well abide by the same
|
|
|
|
rules as Windows and generally disallow checking for names
|
|
|
|
ending with path_sep (the only possible allowable one is
|
|
|
|
checking for "/", which is the root directory, and is
|
|
|
|
guaranteed to exist on valid POSIX filesystems, and is
|
|
|
|
therefore not worth checking for). */
|
|
|
|
|
|
|
|
len = opal_argv_count(parts);
|
|
|
|
for (i = 0; i < len; ++i) {
|
|
|
|
if (i == 0) {
|
|
|
|
|
|
|
|
#ifdef __WINDOWS__
|
|
|
|
/* In the Windows case, check for "<drive>:" case (i.e.,
|
|
|
|
an absolute pathname). If this is the case, ensure
|
|
|
|
that it ends in a path_sep. */
|
|
|
|
|
|
|
|
if (2 == strlen(parts[0]) && isalpha(parts[0][0]) &&
|
|
|
|
':' == parts[0][1]) {
|
|
|
|
strcat(tmp, parts[i]);
|
|
|
|
strcat(tmp, path_sep);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Otherwise, it's a relative path. Per the comment
|
|
|
|
above, we don't want a '\' at the end, so just append
|
|
|
|
this part. */
|
|
|
|
|
|
|
|
else {
|
|
|
|
strcat(tmp, parts[i]);
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
/* If in POSIX-land, ensure that we never end a directory
|
|
|
|
name with path_sep */
|
|
|
|
|
|
|
|
if ('/' == path[0]) {
|
|
|
|
strcat(tmp, path_sep);
|
|
|
|
}
|
|
|
|
strcat(tmp, parts[i]);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If it's not the first part, ensure that there's a
|
|
|
|
preceeding path_sep and then append this part */
|
|
|
|
|
|
|
|
else {
|
|
|
|
if (path_sep[0] != tmp[strlen(tmp) - 1]) {
|
|
|
|
strcat(tmp, path_sep);
|
|
|
|
}
|
|
|
|
strcat(tmp, parts[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Now that we finally have the name to check, check it.
|
|
|
|
Create it if it doesn't exist. */
|
|
|
|
|
|
|
|
if (0 != stat(tmp, &buf)) {
|
|
|
|
if (0 != mkdir(tmp, mode) && 0 != stat(tmp, &buf)) {
|
|
|
|
opal_argv_free(parts);
|
|
|
|
free(tmp);
|
|
|
|
return OPAL_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* All done */
|
|
|
|
|
|
|
|
opal_argv_free(parts);
|
|
|
|
free(tmp);
|
|
|
|
return OPAL_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2006-12-14 18:27:27 +00:00
|
|
|
/**
|
|
|
|
* This function attempts to remove a directory along with all the
|
|
|
|
* files in it. If the recursive variable is non-zero, then it will
|
|
|
|
* try to recursively remove all directories. If provided, the
|
|
|
|
* callback function is executed prior to the directory or file being
|
|
|
|
* removed. If the callback returns non-zero, then no removal is
|
2007-01-09 16:31:08 +00:00
|
|
|
* done.
|
2006-12-14 18:27:27 +00:00
|
|
|
*/
|
2006-07-03 22:23:07 +00:00
|
|
|
int opal_os_dirpath_destroy(const char *path,
|
|
|
|
bool recursive,
|
|
|
|
opal_os_dirpath_destroy_callback_fn_t cbfunc)
|
|
|
|
{
|
|
|
|
int rc, exit_status = OPAL_SUCCESS;
|
|
|
|
bool is_dir = false;
|
|
|
|
|
2006-08-22 23:25:13 +00:00
|
|
|
#ifndef __WINDOWS__
|
2006-07-03 22:23:07 +00:00
|
|
|
DIR *dp;
|
|
|
|
struct dirent *ep;
|
|
|
|
char *filenm;
|
|
|
|
#ifndef HAVE_STRUCT_DIRENT_D_TYPE
|
|
|
|
struct stat buf;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (NULL == path) { /* protect against error */
|
|
|
|
return OPAL_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Make sure we have access to the the base directory
|
|
|
|
*/
|
|
|
|
if( OPAL_SUCCESS != (rc = opal_os_dirpath_access(path, 0) ) ) {
|
|
|
|
exit_status = rc;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Open up the directory
|
|
|
|
*/
|
|
|
|
dp = opendir(path);
|
|
|
|
if (NULL == dp) {
|
|
|
|
return OPAL_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (NULL != (ep = readdir(dp)) ) {
|
|
|
|
/* skip:
|
|
|
|
* - . and ..
|
|
|
|
*/
|
|
|
|
if ((0 == strcmp(ep->d_name, ".")) ||
|
|
|
|
(0 == strcmp(ep->d_name, "..")) ) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check to see if it is a directory */
|
2006-07-11 17:31:05 +00:00
|
|
|
is_dir = false;
|
2006-12-14 18:27:27 +00:00
|
|
|
|
|
|
|
/* Create a pathname. This is not always needed, but it makes
|
|
|
|
* for cleaner code just to create it here. Note that we are
|
|
|
|
* allocating memory here, so we need to free it later on.
|
|
|
|
*/
|
|
|
|
filenm = opal_os_path(false, path, ep->d_name, NULL);
|
2006-07-03 22:23:07 +00:00
|
|
|
#ifdef HAVE_STRUCT_DIRENT_D_TYPE
|
|
|
|
if (DT_DIR == ep->d_type) {
|
|
|
|
is_dir = true;
|
|
|
|
}
|
|
|
|
#else /* have dirent.d_type */
|
|
|
|
rc = stat(filenm, &buf);
|
|
|
|
if (rc < 0 || S_ISDIR(buf.st_mode)) {
|
|
|
|
is_dir = true;
|
|
|
|
}
|
|
|
|
#endif /* have dirent.d_type */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If not recursively decending, then if we find a directory then fail
|
|
|
|
* since we were not told to remove it.
|
|
|
|
*/
|
|
|
|
if( is_dir && !recursive) {
|
|
|
|
/* Set the error indicating that we found a directory,
|
|
|
|
* but continue removing files
|
|
|
|
*/
|
|
|
|
exit_status = OPAL_ERROR;
|
2006-12-14 18:27:27 +00:00
|
|
|
free(filenm);
|
2006-07-03 22:23:07 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Will the caller allow us to remove this file/directory? */
|
|
|
|
if(NULL != cbfunc) {
|
|
|
|
/*
|
|
|
|
* Caller does not wish to remove this file/directory,
|
|
|
|
* continue with the rest of the entries
|
|
|
|
*/
|
|
|
|
if( ! (cbfunc(path, ep->d_name)) ) {
|
2006-12-14 18:27:27 +00:00
|
|
|
free(filenm);
|
2006-07-03 22:23:07 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Directories are recursively destroyed */
|
|
|
|
if(is_dir) {
|
2006-12-14 18:27:27 +00:00
|
|
|
rc = opal_os_dirpath_destroy(filenm, recursive, cbfunc);
|
|
|
|
free(filenm);
|
|
|
|
if (OPAL_SUCCESS != rc) {
|
2006-07-03 22:23:07 +00:00
|
|
|
exit_status = rc;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Files are removed right here */
|
|
|
|
else {
|
|
|
|
if( 0 != (rc = unlink(filenm) ) ) {
|
|
|
|
exit_status = OPAL_ERROR;
|
|
|
|
}
|
|
|
|
free(filenm);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Done with this directory
|
|
|
|
*/
|
|
|
|
closedir(dp);
|
|
|
|
#else
|
|
|
|
char search_path[MAX_PATH];
|
|
|
|
HANDLE file;
|
|
|
|
WIN32_FIND_DATA file_data;
|
|
|
|
TCHAR *file_name;
|
|
|
|
|
|
|
|
if (NULL == path) {
|
|
|
|
return OPAL_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
strncpy(search_path, path, strlen(path)+1);
|
2006-08-22 23:25:13 +00:00
|
|
|
strncat (search_path, OPAL_PATH_SEP"*", 3);
|
2006-07-03 22:23:07 +00:00
|
|
|
file = FindFirstFile(search_path, &file_data);
|
|
|
|
|
|
|
|
if (INVALID_HANDLE_VALUE == file) {
|
2006-08-22 23:25:13 +00:00
|
|
|
FindClose(file);
|
2006-08-22 20:07:42 +00:00
|
|
|
return OPAL_ERROR;
|
2006-07-03 22:23:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
do {
|
2007-01-09 16:31:08 +00:00
|
|
|
/* Skip . and .. */
|
2006-07-03 22:23:07 +00:00
|
|
|
if ((0 == strcmp(file_data.cFileName, ".")) ||
|
|
|
|
(0 == strcmp(file_data.cFileName, "..")) ) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2006-07-11 17:31:05 +00:00
|
|
|
is_dir = false;
|
2006-07-03 22:23:07 +00:00
|
|
|
if(file_data.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) {
|
|
|
|
is_dir = true;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* If not recursively decending, then if we find a directory then fail
|
|
|
|
* since we were not told to remove it.
|
|
|
|
*/
|
|
|
|
if( is_dir && !recursive) {
|
|
|
|
/* Set the error indicating that we found a directory,
|
|
|
|
* but continue removing files
|
|
|
|
*/
|
|
|
|
exit_status = OPAL_ERROR;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Will the caller allow us to remove this file/directory */
|
|
|
|
if( NULL != cbfunc) {
|
2007-01-09 16:31:08 +00:00
|
|
|
/*
|
|
|
|
* Caller does not wish to remove this file/directory,
|
|
|
|
* continue with the rest of the entries
|
|
|
|
*/
|
2006-07-03 22:23:07 +00:00
|
|
|
if( !cbfunc(path, file_data.cFileName) ) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
file_name = opal_os_path(false, path, file_data.cFileName, NULL);
|
|
|
|
if( is_dir ) {
|
|
|
|
if( OPAL_SUCCESS != (rc = opal_os_dirpath_destroy(file_name,
|
|
|
|
recursive,
|
|
|
|
cbfunc) ) ) {
|
|
|
|
exit_status = rc;
|
2007-01-09 16:31:08 +00:00
|
|
|
free(file_name);
|
2006-07-03 22:23:07 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
DeleteFile(file_name);
|
|
|
|
}
|
2007-01-09 16:31:08 +00:00
|
|
|
free(file_name);
|
2006-08-22 23:25:13 +00:00
|
|
|
} while( 0 != FindNextFile(file, &file_data) );
|
2006-07-03 22:23:07 +00:00
|
|
|
|
2006-08-22 23:25:13 +00:00
|
|
|
FindClose(file);
|
2006-07-03 22:23:07 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If the directory is empty, them remove it
|
|
|
|
*/
|
|
|
|
if(opal_os_dirpath_is_empty(path)) {
|
|
|
|
rmdir(path);
|
|
|
|
}
|
|
|
|
|
|
|
|
return exit_status;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool opal_os_dirpath_is_empty(const char *path ) {
|
|
|
|
#ifndef __WINDOWS__
|
|
|
|
DIR *dp;
|
|
|
|
struct dirent *ep;
|
|
|
|
|
|
|
|
if (NULL != path) { /* protect against error */
|
|
|
|
dp = opendir(path);
|
|
|
|
if (NULL != dp) {
|
|
|
|
while ((ep = readdir(dp))) {
|
|
|
|
if ((0 != strcmp(ep->d_name, ".")) &&
|
|
|
|
(0 != strcmp(ep->d_name, ".."))) {
|
|
|
|
closedir(dp);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
closedir(dp);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
#else
|
|
|
|
char search_path[MAX_PATH];
|
|
|
|
HANDLE file;
|
|
|
|
WIN32_FIND_DATA file_data;
|
2006-08-22 23:25:13 +00:00
|
|
|
bool found = false;
|
2006-07-03 22:23:07 +00:00
|
|
|
|
|
|
|
if (NULL != path) {
|
|
|
|
strncpy(search_path, path, strlen(path)+1);
|
|
|
|
strncat (search_path, "\\*", 3);
|
|
|
|
|
|
|
|
file = FindFirstFile(search_path, &file_data);
|
|
|
|
if (INVALID_HANDLE_VALUE == file) {
|
2006-08-22 23:25:13 +00:00
|
|
|
goto cleanup;
|
2006-07-03 22:23:07 +00:00
|
|
|
}
|
|
|
|
|
2006-08-22 23:25:13 +00:00
|
|
|
do {
|
2006-07-03 22:23:07 +00:00
|
|
|
if (0 != strcmp(file_data.cFileName, ".") || 0 != strcmp(file_data.cFileName, "..")) {
|
2006-08-22 23:25:13 +00:00
|
|
|
found = true;
|
|
|
|
goto cleanup;
|
2006-07-03 22:23:07 +00:00
|
|
|
}
|
2006-08-22 23:25:13 +00:00
|
|
|
} while (0 != FindNextFile(file, &file_data));
|
2006-07-03 22:23:07 +00:00
|
|
|
}
|
2006-08-22 23:25:13 +00:00
|
|
|
cleanup:
|
|
|
|
FindClose(file);
|
|
|
|
return found;
|
2006-07-03 22:23:07 +00:00
|
|
|
#endif /* ifndef __WINDOWS__ */
|
|
|
|
}
|
|
|
|
|
|
|
|
int opal_os_dirpath_access(const char *path, const mode_t in_mode ) {
|
|
|
|
#ifndef __WINDOWS__
|
|
|
|
struct stat buf;
|
|
|
|
#else
|
|
|
|
struct __stat64 buf;
|
|
|
|
#endif
|
2007-04-01 16:16:54 +00:00
|
|
|
mode_t loc_mode = S_IRWXU; /* looking for full rights */
|
2006-07-03 22:23:07 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* If there was no mode specified, use the default mode
|
|
|
|
*/
|
|
|
|
if( 0 != in_mode ) {
|
|
|
|
loc_mode = in_mode;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef __WINDOWS__
|
|
|
|
if (0 == stat(path, &buf)) { /* exists - check access */
|
|
|
|
#else
|
|
|
|
if (0 == _stat64(path, &buf)) { /* exist -- check */
|
|
|
|
#endif
|
|
|
|
if ((buf.st_mode & loc_mode) == loc_mode) { /* okay, I can work here */
|
|
|
|
return(OPAL_SUCCESS);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* Don't have access rights to the existing path */
|
|
|
|
return(OPAL_ERROR);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* We could not find the path */
|
|
|
|
return( OPAL_ERR_NOT_FOUND );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|