 8b815c7ce5
			
		
	
	8b815c7ce5
	
	
	
		
			
			Previously the return value of `ufopen` wasn't checked and if it failed, `NULL` was passed into `fclose()` which resulted in a crash. This patch avoids this by returning from `BLI_gzopen` when the file cannot be created. Reviewed By: sebbas, iss Differential Revision: https://developer.blender.org/D9576
		
			
				
	
	
		
			1397 lines
		
	
	
		
			32 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1397 lines
		
	
	
		
			32 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * This program is free software; you can redistribute it and/or
 | |
|  * modify it under the terms of the GNU General Public License
 | |
|  * as published by the Free Software Foundation; either version 2
 | |
|  * of the License, or (at your option) any later version.
 | |
|  *
 | |
|  * This program is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  * GNU General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU General Public License
 | |
|  * along with this program; if not, write to the Free Software Foundation,
 | |
|  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
 | |
|  *
 | |
|  * The Original Code is Copyright (C) 2001-2002 by NaN Holding BV.
 | |
|  * All rights reserved.
 | |
|  */
 | |
| 
 | |
| /** \file
 | |
|  * \ingroup bli
 | |
|  */
 | |
| 
 | |
| #include <stdlib.h> /* malloc */
 | |
| #include <string.h>
 | |
| 
 | |
| #include <fcntl.h>
 | |
| #include <sys/stat.h>
 | |
| #include <sys/types.h>
 | |
| 
 | |
| #include <errno.h>
 | |
| 
 | |
| #include "zlib.h"
 | |
| 
 | |
| #ifdef WIN32
 | |
| #  include "BLI_fileops_types.h"
 | |
| #  include "BLI_winstuff.h"
 | |
| #  include "utf_winfunc.h"
 | |
| #  include "utfconv.h"
 | |
| #  include <io.h>
 | |
| #  include <shellapi.h>
 | |
| #  include <shobjidl.h>
 | |
| #  include <windows.h>
 | |
| #else
 | |
| #  if defined(__APPLE__)
 | |
| #    include <CoreFoundation/CoreFoundation.h>
 | |
| #    include <objc/message.h>
 | |
| #    include <objc/runtime.h>
 | |
| #  endif
 | |
| #  include <dirent.h>
 | |
| #  include <sys/param.h>
 | |
| #  include <sys/wait.h>
 | |
| #  include <unistd.h>
 | |
| #endif
 | |
| 
 | |
| #include "MEM_guardedalloc.h"
 | |
| 
 | |
| #include "BLI_fileops.h"
 | |
| #include "BLI_path_util.h"
 | |
| #include "BLI_string.h"
 | |
| #include "BLI_sys_types.h" /* for intptr_t support */
 | |
| #include "BLI_utildefines.h"
 | |
| 
 | |
| #if 0 /* UNUSED */
 | |
| /* gzip the file in from and write it to "to".
 | |
|  * return -1 if zlib fails, -2 if the originating file does not exist
 | |
|  * note: will remove the "from" file
 | |
|  */
 | |
| int BLI_file_gzip(const char *from, const char *to)
 | |
| {
 | |
|   char buffer[10240];
 | |
|   int file;
 | |
|   int readsize = 0;
 | |
|   int rval = 0, err;
 | |
|   gzFile gzfile;
 | |
| 
 | |
|   /* level 1 is very close to 3 (the default) in terms of file size,
 | |
|    * but about twice as fast, best use for speedy saving - campbell */
 | |
|   gzfile = BLI_gzopen(to, "wb1");
 | |
|   if (gzfile == NULL) {
 | |
|     return -1;
 | |
|   }
 | |
|   file = BLI_open(from, O_BINARY | O_RDONLY, 0);
 | |
|   if (file == -1) {
 | |
|     return -2;
 | |
|   }
 | |
| 
 | |
|   while (1) {
 | |
|     readsize = read(file, buffer, sizeof(buffer));
 | |
| 
 | |
|     if (readsize < 0) {
 | |
|       rval = -2; /* error happened in reading */
 | |
|       fprintf(stderr, "Error reading file %s: %s.\n", from, strerror(errno));
 | |
|       break;
 | |
|     }
 | |
|     else if (readsize == 0) {
 | |
|       break; /* done reading */
 | |
|     }
 | |
| 
 | |
|     if (gzwrite(gzfile, buffer, readsize) <= 0) {
 | |
|       rval = -1; /* error happened in writing */
 | |
|       fprintf(stderr, "Error writing gz file %s: %s.\n", to, gzerror(gzfile, &err));
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   gzclose(gzfile);
 | |
|   close(file);
 | |
| 
 | |
|   return rval;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| /* gzip the file in from_file and write it to memory to_mem, at most size bytes.
 | |
|  * return the unzipped size
 | |
|  */
 | |
| char *BLI_file_ungzip_to_mem(const char *from_file, int *r_size)
 | |
| {
 | |
|   gzFile gzfile;
 | |
|   int readsize, size, alloc_size = 0;
 | |
|   char *mem = NULL;
 | |
|   const int chunk_size = 512 * 1024;
 | |
| 
 | |
|   size = 0;
 | |
| 
 | |
|   gzfile = BLI_gzopen(from_file, "rb");
 | |
|   for (;;) {
 | |
|     if (mem == NULL) {
 | |
|       mem = MEM_callocN(chunk_size, "BLI_ungzip_to_mem");
 | |
|       alloc_size = chunk_size;
 | |
|     }
 | |
|     else {
 | |
|       mem = MEM_reallocN(mem, size + chunk_size);
 | |
|       alloc_size += chunk_size;
 | |
|     }
 | |
| 
 | |
|     readsize = gzread(gzfile, mem + size, chunk_size);
 | |
|     if (readsize > 0) {
 | |
|       size += readsize;
 | |
|     }
 | |
|     else {
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   gzclose(gzfile);
 | |
| 
 | |
|   if (size == 0) {
 | |
|     MEM_freeN(mem);
 | |
|     mem = NULL;
 | |
|   }
 | |
|   else if (alloc_size != size) {
 | |
|     mem = MEM_reallocN(mem, size);
 | |
|   }
 | |
| 
 | |
|   *r_size = size;
 | |
| 
 | |
|   return mem;
 | |
| }
 | |
| 
 | |
| #define CHUNK 256 * 1024
 | |
| 
 | |
| /* gzip byte array from memory and write it to file at certain position.
 | |
|  * return size of gzip stream.
 | |
|  */
 | |
| size_t BLI_gzip_mem_to_file_at_pos(
 | |
|     void *buf, size_t len, FILE *file, size_t gz_stream_offset, int compression_level)
 | |
| {
 | |
|   int ret, flush;
 | |
|   unsigned have;
 | |
|   z_stream strm;
 | |
|   unsigned char out[CHUNK];
 | |
| 
 | |
|   fseek(file, gz_stream_offset, 0);
 | |
| 
 | |
|   strm.zalloc = Z_NULL;
 | |
|   strm.zfree = Z_NULL;
 | |
|   strm.opaque = Z_NULL;
 | |
|   ret = deflateInit(&strm, compression_level);
 | |
|   if (ret != Z_OK) {
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   strm.avail_in = len;
 | |
|   strm.next_in = (Bytef *)buf;
 | |
|   flush = Z_FINISH;
 | |
| 
 | |
|   do {
 | |
|     strm.avail_out = CHUNK;
 | |
|     strm.next_out = out;
 | |
|     ret = deflate(&strm, flush);
 | |
|     if (ret == Z_STREAM_ERROR) {
 | |
|       return 0;
 | |
|     }
 | |
|     have = CHUNK - strm.avail_out;
 | |
|     if (fwrite(out, 1, have, file) != have || ferror(file)) {
 | |
|       deflateEnd(&strm);
 | |
|       return 0;
 | |
|     }
 | |
|   } while (strm.avail_out == 0);
 | |
| 
 | |
|   if (strm.avail_in != 0 || ret != Z_STREAM_END) {
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   deflateEnd(&strm);
 | |
|   return (size_t)strm.total_out;
 | |
| }
 | |
| 
 | |
| /* read and decompress gzip stream from file at certain position to buffer.
 | |
|  * return size of decompressed data.
 | |
|  */
 | |
| size_t BLI_ungzip_file_to_mem_at_pos(void *buf, size_t len, FILE *file, size_t gz_stream_offset)
 | |
| {
 | |
|   int ret;
 | |
|   z_stream strm;
 | |
|   size_t chunk = 256 * 1024;
 | |
|   unsigned char in[CHUNK];
 | |
| 
 | |
|   fseek(file, gz_stream_offset, 0);
 | |
| 
 | |
|   strm.zalloc = Z_NULL;
 | |
|   strm.zfree = Z_NULL;
 | |
|   strm.opaque = Z_NULL;
 | |
|   strm.avail_in = 0;
 | |
|   strm.next_in = Z_NULL;
 | |
|   ret = inflateInit(&strm);
 | |
|   if (ret != Z_OK) {
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   do {
 | |
|     strm.avail_in = fread(in, 1, chunk, file);
 | |
|     strm.next_in = in;
 | |
|     if (ferror(file)) {
 | |
|       inflateEnd(&strm);
 | |
|       return 0;
 | |
|     }
 | |
| 
 | |
|     do {
 | |
|       strm.avail_out = len;
 | |
|       strm.next_out = (Bytef *)buf + strm.total_out;
 | |
| 
 | |
|       ret = inflate(&strm, Z_NO_FLUSH);
 | |
|       if (ret == Z_STREAM_ERROR) {
 | |
|         return 0;
 | |
|       }
 | |
|     } while (strm.avail_out == 0);
 | |
| 
 | |
|   } while (ret != Z_STREAM_END);
 | |
| 
 | |
|   inflateEnd(&strm);
 | |
|   return (size_t)strm.total_out;
 | |
| }
 | |
| 
 | |
| #undef CHUNK
 | |
| 
 | |
| /**
 | |
|  * Returns true if the file with the specified name can be written.
 | |
|  * This implementation uses access(2), which makes the check according
 | |
|  * to the real UID and GID of the process, not its effective UID and GID.
 | |
|  * This shouldn't matter for Blender, which is not going to run privileged
 | |
|  * anyway.
 | |
|  */
 | |
| bool BLI_file_is_writable(const char *filename)
 | |
| {
 | |
|   bool writable;
 | |
|   if (BLI_access(filename, W_OK) == 0) {
 | |
|     /* file exists and I can write to it */
 | |
|     writable = true;
 | |
|   }
 | |
|   else if (errno != ENOENT) {
 | |
|     /* most likely file or containing directory cannot be accessed */
 | |
|     writable = false;
 | |
|   }
 | |
|   else {
 | |
|     /* file doesn't exist -- check I can create it in parent directory */
 | |
|     char parent[FILE_MAX];
 | |
|     BLI_split_dirfile(filename, parent, NULL, sizeof(parent), 0);
 | |
| #ifdef WIN32
 | |
|     /* windows does not have X_OK */
 | |
|     writable = BLI_access(parent, W_OK) == 0;
 | |
| #else
 | |
|     writable = BLI_access(parent, X_OK | W_OK) == 0;
 | |
| #endif
 | |
|   }
 | |
|   return writable;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Creates the file with nothing in it, or updates its last-modified date if it already exists.
 | |
|  * Returns true if successful (like the unix touch command).
 | |
|  */
 | |
| bool BLI_file_touch(const char *file)
 | |
| {
 | |
|   FILE *f = BLI_fopen(file, "r+b");
 | |
| 
 | |
|   if (f != NULL) {
 | |
|     int c = getc(f);
 | |
| 
 | |
|     if (c == EOF) {
 | |
|       /* Empty file, reopen in truncate write mode... */
 | |
|       fclose(f);
 | |
|       f = BLI_fopen(file, "w+b");
 | |
|     }
 | |
|     else {
 | |
|       /* Otherwise, rewrite first byte. */
 | |
|       rewind(f);
 | |
|       putc(c, f);
 | |
|     }
 | |
|   }
 | |
|   else {
 | |
|     f = BLI_fopen(file, "wb");
 | |
|   }
 | |
|   if (f) {
 | |
|     fclose(f);
 | |
|     return true;
 | |
|   }
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| #ifdef WIN32
 | |
| 
 | |
| static void callLocalErrorCallBack(const char *err)
 | |
| {
 | |
|   printf("%s\n", err);
 | |
| }
 | |
| 
 | |
| FILE *BLI_fopen(const char *filename, const char *mode)
 | |
| {
 | |
|   BLI_assert(!BLI_path_is_rel(filename));
 | |
| 
 | |
|   return ufopen(filename, mode);
 | |
| }
 | |
| 
 | |
| void BLI_get_short_name(char short_name[256], const char *filename)
 | |
| {
 | |
|   wchar_t short_name_16[256];
 | |
|   int i = 0;
 | |
| 
 | |
|   UTF16_ENCODE(filename);
 | |
| 
 | |
|   GetShortPathNameW(filename_16, short_name_16, 256);
 | |
| 
 | |
|   for (i = 0; i < 256; i++) {
 | |
|     short_name[i] = (char)short_name_16[i];
 | |
|   }
 | |
| 
 | |
|   UTF16_UN_ENCODE(filename);
 | |
| }
 | |
| 
 | |
| void *BLI_gzopen(const char *filename, const char *mode)
 | |
| {
 | |
|   gzFile gzfile;
 | |
| 
 | |
|   BLI_assert(!BLI_path_is_rel(filename));
 | |
| 
 | |
|   /* xxx Creates file before transcribing the path */
 | |
|   if (mode[0] == 'w') {
 | |
|     FILE *file = ufopen(filename, "a");
 | |
|     if (file == NULL) {
 | |
|       /* File couldn't be opened, e.g. due to permission error. */
 | |
|       return NULL;
 | |
|     }
 | |
|     fclose(file);
 | |
|   }
 | |
| 
 | |
|   /* temporary #if until we update all libraries to 1.2.7
 | |
|    * for correct wide char path handling */
 | |
| #  if ZLIB_VERNUM >= 0x1270
 | |
|   UTF16_ENCODE(filename);
 | |
| 
 | |
|   gzfile = gzopen_w(filename_16, mode);
 | |
| 
 | |
|   UTF16_UN_ENCODE(filename);
 | |
| #  else
 | |
|   {
 | |
|     char short_name[256];
 | |
|     BLI_get_short_name(short_name, filename);
 | |
|     gzfile = gzopen(short_name, mode);
 | |
|   }
 | |
| #  endif
 | |
| 
 | |
|   return gzfile;
 | |
| }
 | |
| 
 | |
| int BLI_open(const char *filename, int oflag, int pmode)
 | |
| {
 | |
|   BLI_assert(!BLI_path_is_rel(filename));
 | |
| 
 | |
|   return uopen(filename, oflag, pmode);
 | |
| }
 | |
| 
 | |
| int BLI_access(const char *filename, int mode)
 | |
| {
 | |
|   BLI_assert(!BLI_path_is_rel(filename));
 | |
| 
 | |
|   return uaccess(filename, mode);
 | |
| }
 | |
| 
 | |
| static bool delete_soft(const wchar_t *path_16, const char **error_message)
 | |
| {
 | |
|   /* Deletes file or directory to recycling bin. The latter moves all contained files and
 | |
|    * directories recursively to the recycling bin as well. */
 | |
|   IFileOperation *pfo;
 | |
|   IShellItem *pSI;
 | |
| 
 | |
|   HRESULT hr = CoInitializeEx(NULL, COINIT_APARTMENTTHREADED | COINIT_DISABLE_OLE1DDE);
 | |
| 
 | |
|   if (FAILED(hr)) {
 | |
|     *error_message = "Failed to initialize COM";
 | |
|     goto error_1;
 | |
|   }
 | |
| 
 | |
|   hr = CoCreateInstance(
 | |
|       &CLSID_FileOperation, NULL, CLSCTX_ALL, &IID_IFileOperation, (void **)&pfo);
 | |
|   if (FAILED(hr)) {
 | |
|     *error_message = "Failed to create FileOperation instance";
 | |
|     goto error_2;
 | |
|   }
 | |
| 
 | |
|   /* Flags for deletion:
 | |
|    * FOF_ALLOWUNDO: Enables moving file to recycling bin.
 | |
|    * FOF_SILENT: Don't show progress dialog box.
 | |
|    * FOF_WANTNUKEWARNING: Show dialog box if file can't be moved to recycling bin. */
 | |
|   hr = pfo->lpVtbl->SetOperationFlags(pfo, FOF_ALLOWUNDO | FOF_SILENT | FOF_WANTNUKEWARNING);
 | |
| 
 | |
|   if (FAILED(hr)) {
 | |
|     *error_message = "Failed to set operation flags";
 | |
|     goto error_2;
 | |
|   }
 | |
| 
 | |
|   hr = SHCreateItemFromParsingName(path_16, NULL, &IID_IShellItem, (void **)&pSI);
 | |
|   if (FAILED(hr)) {
 | |
|     *error_message = "Failed to parse path";
 | |
|     goto error_2;
 | |
|   }
 | |
| 
 | |
|   hr = pfo->lpVtbl->DeleteItem(pfo, pSI, NULL);
 | |
|   if (FAILED(hr)) {
 | |
|     *error_message = "Failed to prepare delete operation";
 | |
|     goto error_2;
 | |
|   }
 | |
| 
 | |
|   hr = pfo->lpVtbl->PerformOperations(pfo);
 | |
| 
 | |
|   if (FAILED(hr)) {
 | |
|     *error_message = "Failed to delete file or directory";
 | |
|   }
 | |
| 
 | |
| error_2:
 | |
|   pfo->lpVtbl->Release(pfo);
 | |
|   CoUninitialize(); /* Has to be uninitialized when CoInitializeEx returns either S_OK or S_FALSE
 | |
|                      */
 | |
| error_1:
 | |
|   return FAILED(hr);
 | |
| }
 | |
| 
 | |
| static bool delete_unique(const char *path, const bool dir)
 | |
| {
 | |
|   bool err;
 | |
| 
 | |
|   UTF16_ENCODE(path);
 | |
| 
 | |
|   if (dir) {
 | |
|     err = !RemoveDirectoryW(path_16);
 | |
|     if (err) {
 | |
|       printf("Unable to remove directory\n");
 | |
|     }
 | |
|   }
 | |
|   else {
 | |
|     err = !DeleteFileW(path_16);
 | |
|     if (err) {
 | |
|       callLocalErrorCallBack("Unable to delete file");
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   UTF16_UN_ENCODE(path);
 | |
| 
 | |
|   return err;
 | |
| }
 | |
| 
 | |
| static bool delete_recursive(const char *dir)
 | |
| {
 | |
|   struct direntry *filelist, *fl;
 | |
|   bool err = false;
 | |
|   uint nbr, i;
 | |
| 
 | |
|   i = nbr = BLI_filelist_dir_contents(dir, &filelist);
 | |
|   fl = filelist;
 | |
|   while (i--) {
 | |
|     const char *file = BLI_path_basename(fl->path);
 | |
| 
 | |
|     if (FILENAME_IS_CURRPAR(file)) {
 | |
|       /* Skip! */
 | |
|     }
 | |
|     else if (S_ISDIR(fl->type)) {
 | |
|       char path[FILE_MAXDIR];
 | |
| 
 | |
|       /* dir listing produces dir path without trailing slash... */
 | |
|       BLI_strncpy(path, fl->path, sizeof(path));
 | |
|       BLI_path_slash_ensure(path);
 | |
| 
 | |
|       if (delete_recursive(path)) {
 | |
|         err = true;
 | |
|       }
 | |
|     }
 | |
|     else {
 | |
|       if (delete_unique(fl->path, false)) {
 | |
|         err = true;
 | |
|       }
 | |
|     }
 | |
|     fl++;
 | |
|   }
 | |
| 
 | |
|   if (!err && delete_unique(dir, true)) {
 | |
|     err = true;
 | |
|   }
 | |
| 
 | |
|   BLI_filelist_free(filelist, nbr);
 | |
| 
 | |
|   return err;
 | |
| }
 | |
| 
 | |
| int BLI_delete(const char *file, bool dir, bool recursive)
 | |
| {
 | |
|   int err;
 | |
| 
 | |
|   BLI_assert(!BLI_path_is_rel(file));
 | |
| 
 | |
|   if (recursive) {
 | |
|     err = delete_recursive(file);
 | |
|   }
 | |
|   else {
 | |
|     err = delete_unique(file, dir);
 | |
|   }
 | |
| 
 | |
|   return err;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Moves the files or directories to the recycling bin.
 | |
|  */
 | |
| int BLI_delete_soft(const char *file, const char **error_message)
 | |
| {
 | |
|   int err;
 | |
| 
 | |
|   BLI_assert(!BLI_path_is_rel(file));
 | |
| 
 | |
|   UTF16_ENCODE(file);
 | |
| 
 | |
|   err = delete_soft(file_16, error_message);
 | |
| 
 | |
|   UTF16_UN_ENCODE(file);
 | |
| 
 | |
|   return err;
 | |
| }
 | |
| 
 | |
| /* Not used anywhere! */
 | |
| #  if 0
 | |
| int BLI_move(const char *file, const char *to)
 | |
| {
 | |
|   char str[MAXPATHLEN + 12];
 | |
|   int err;
 | |
| 
 | |
|   /* windows doesn't support moving to a directory
 | |
|    * it has to be 'mv filename filename' and not
 | |
|    * 'mv filename destination_directory' */
 | |
| 
 | |
|   BLI_strncpy(str, to, sizeof(str));
 | |
|   /* points 'to' to a directory ? */
 | |
|   if (BLI_path_slash_rfind(str) == (str + strlen(str) - 1)) {
 | |
|     if (BLI_path_slash_rfind(file) != NULL) {
 | |
|       strcat(str, BLI_path_slash_rfind(file) + 1);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   UTF16_ENCODE(file);
 | |
|   UTF16_ENCODE(str);
 | |
|   err = !MoveFileW(file_16, str_16);
 | |
|   UTF16_UN_ENCODE(str);
 | |
|   UTF16_UN_ENCODE(file);
 | |
| 
 | |
|   if (err) {
 | |
|     callLocalErrorCallBack("Unable to move file");
 | |
|     printf(" Move from '%s' to '%s' failed\n", file, str);
 | |
|   }
 | |
| 
 | |
|   return err;
 | |
| }
 | |
| #  endif
 | |
| 
 | |
| int BLI_copy(const char *file, const char *to)
 | |
| {
 | |
|   char str[MAXPATHLEN + 12];
 | |
|   int err;
 | |
| 
 | |
|   /* windows doesn't support copying to a directory
 | |
|    * it has to be 'cp filename filename' and not
 | |
|    * 'cp filename destdir' */
 | |
| 
 | |
|   BLI_strncpy(str, to, sizeof(str));
 | |
|   /* points 'to' to a directory ? */
 | |
|   if (BLI_path_slash_rfind(str) == (str + strlen(str) - 1)) {
 | |
|     if (BLI_path_slash_rfind(file) != NULL) {
 | |
|       strcat(str, BLI_path_slash_rfind(file) + 1);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   UTF16_ENCODE(file);
 | |
|   UTF16_ENCODE(str);
 | |
|   err = !CopyFileW(file_16, str_16, false);
 | |
|   UTF16_UN_ENCODE(str);
 | |
|   UTF16_UN_ENCODE(file);
 | |
| 
 | |
|   if (err) {
 | |
|     callLocalErrorCallBack("Unable to copy file!");
 | |
|     printf(" Copy from '%s' to '%s' failed\n", file, str);
 | |
|   }
 | |
| 
 | |
|   return err;
 | |
| }
 | |
| 
 | |
| #  if 0
 | |
| int BLI_create_symlink(const char *file, const char *to)
 | |
| {
 | |
|   /* See patch from T30870, should this ever become needed. */
 | |
|   callLocalErrorCallBack("Linking files is unsupported on Windows");
 | |
|   (void)file;
 | |
|   (void)to;
 | |
|   return 1;
 | |
| }
 | |
| #  endif
 | |
| 
 | |
| /** \return true on success (i.e. given path now exists on FS), false otherwise. */
 | |
| bool BLI_dir_create_recursive(const char *dirname)
 | |
| {
 | |
|   char *lslash;
 | |
|   char tmp[MAXPATHLEN];
 | |
|   bool ret = true;
 | |
| 
 | |
|   /* First remove possible slash at the end of the dirname.
 | |
|    * This routine otherwise tries to create
 | |
|    * blah1/blah2/ (with slash) after creating
 | |
|    * blah1/blah2 (without slash) */
 | |
| 
 | |
|   BLI_strncpy(tmp, dirname, sizeof(tmp));
 | |
|   BLI_path_slash_rstrip(tmp);
 | |
| 
 | |
|   /* check special case "c:\foo", don't try create "c:", harmless but prints an error below */
 | |
|   if (isalpha(tmp[0]) && (tmp[1] == ':') && tmp[2] == '\0') {
 | |
|     return true;
 | |
|   }
 | |
| 
 | |
|   if (BLI_is_dir(tmp)) {
 | |
|     return true;
 | |
|   }
 | |
|   else if (BLI_exists(tmp)) {
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   lslash = (char *)BLI_path_slash_rfind(tmp);
 | |
| 
 | |
|   if (lslash) {
 | |
|     /* Split about the last slash and recurse */
 | |
|     *lslash = 0;
 | |
|     if (!BLI_dir_create_recursive(tmp)) {
 | |
|       ret = false;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (ret && dirname[0]) { /* patch, this recursive loop tries to create a nameless directory */
 | |
|     if (umkdir(dirname) == -1) {
 | |
|       printf("Unable to create directory %s\n", dirname);
 | |
|       ret = false;
 | |
|     }
 | |
|   }
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| int BLI_rename(const char *from, const char *to)
 | |
| {
 | |
|   if (!BLI_exists(from)) {
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   /* make sure the filenames are different (case insensitive) before removing */
 | |
|   if (BLI_exists(to) && BLI_strcasecmp(from, to)) {
 | |
|     if (BLI_delete(to, false, false)) {
 | |
|       return 1;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return urename(from, to);
 | |
| }
 | |
| 
 | |
| #else /* The UNIX world */
 | |
| 
 | |
| /* results from recursive_operation and its callbacks */
 | |
| enum {
 | |
|   /* operation succeeded */
 | |
|   RecursiveOp_Callback_OK = 0,
 | |
| 
 | |
|   /* operation requested not to perform recursive digging for current path */
 | |
|   RecursiveOp_Callback_StopRecurs = 1,
 | |
| 
 | |
|   /* error occurred in callback and recursive walking should stop immediately */
 | |
|   RecursiveOp_Callback_Error = 2,
 | |
| };
 | |
| 
 | |
| typedef int (*RecursiveOp_Callback)(const char *from, const char *to);
 | |
| 
 | |
| /* appending of filename to dir (ensures for buffer size before appending) */
 | |
| static void join_dirfile_alloc(char **dst, size_t *alloc_len, const char *dir, const char *file)
 | |
| {
 | |
|   size_t len = strlen(dir) + strlen(file) + 1;
 | |
| 
 | |
|   if (*dst == NULL) {
 | |
|     *dst = MEM_mallocN(len + 1, "join_dirfile_alloc path");
 | |
|   }
 | |
|   else if (*alloc_len < len) {
 | |
|     *dst = MEM_reallocN(*dst, len + 1);
 | |
|   }
 | |
| 
 | |
|   *alloc_len = len;
 | |
| 
 | |
|   BLI_join_dirfile(*dst, len + 1, dir, file);
 | |
| }
 | |
| 
 | |
| static char *strip_last_slash(const char *dir)
 | |
| {
 | |
|   char *result = BLI_strdup(dir);
 | |
|   BLI_path_slash_rstrip(result);
 | |
| 
 | |
|   return result;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Scans \a startfrom, generating a corresponding destination name for each item found by
 | |
|  * prefixing it with startto, recursively scanning subdirectories, and invoking the specified
 | |
|  * callbacks for files and subdirectories found as appropriate.
 | |
|  *
 | |
|  * \param startfrom: Top-level source path.
 | |
|  * \param startto: Top-level destination path.
 | |
|  * \param callback_dir_pre: Optional, to be invoked before entering a subdirectory, can return
 | |
|  *                          RecursiveOp_Callback_StopRecurs to skip the subdirectory.
 | |
|  * \param callback_file: Optional, to be invoked on each file found.
 | |
|  * \param callback_dir_post: optional, to be invoked after leaving a subdirectory.
 | |
|  * \return
 | |
|  */
 | |
| static int recursive_operation(const char *startfrom,
 | |
|                                const char *startto,
 | |
|                                RecursiveOp_Callback callback_dir_pre,
 | |
|                                RecursiveOp_Callback callback_file,
 | |
|                                RecursiveOp_Callback callback_dir_post)
 | |
| {
 | |
|   struct stat st;
 | |
|   char *from = NULL, *to = NULL;
 | |
|   char *from_path = NULL, *to_path = NULL;
 | |
|   struct dirent **dirlist = NULL;
 | |
|   size_t from_alloc_len = -1, to_alloc_len = -1;
 | |
|   int i, n = 0, ret = 0;
 | |
| 
 | |
|   do { /* once */
 | |
|     /* ensure there's no trailing slash in file path */
 | |
|     from = strip_last_slash(startfrom);
 | |
|     if (startto) {
 | |
|       to = strip_last_slash(startto);
 | |
|     }
 | |
| 
 | |
|     ret = lstat(from, &st);
 | |
|     if (ret < 0) {
 | |
|       /* source wasn't found, nothing to operate with */
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     if (!S_ISDIR(st.st_mode)) {
 | |
|       /* source isn't a directory, can't do recursive walking for it,
 | |
|        * so just call file callback and leave */
 | |
|       if (callback_file != NULL) {
 | |
|         ret = callback_file(from, to);
 | |
|         if (ret != RecursiveOp_Callback_OK) {
 | |
|           ret = -1;
 | |
|         }
 | |
|       }
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     n = scandir(startfrom, &dirlist, NULL, alphasort);
 | |
|     if (n < 0) {
 | |
|       /* error opening directory for listing */
 | |
|       perror("scandir");
 | |
|       ret = -1;
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     if (callback_dir_pre != NULL) {
 | |
|       ret = callback_dir_pre(from, to);
 | |
|       if (ret != RecursiveOp_Callback_OK) {
 | |
|         if (ret == RecursiveOp_Callback_StopRecurs) {
 | |
|           /* callback requested not to perform recursive walking, not an error */
 | |
|           ret = 0;
 | |
|         }
 | |
|         else {
 | |
|           ret = -1;
 | |
|         }
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     for (i = 0; i < n; i++) {
 | |
|       const struct dirent *const dirent = dirlist[i];
 | |
| 
 | |
|       if (FILENAME_IS_CURRPAR(dirent->d_name)) {
 | |
|         continue;
 | |
|       }
 | |
| 
 | |
|       join_dirfile_alloc(&from_path, &from_alloc_len, from, dirent->d_name);
 | |
|       if (to) {
 | |
|         join_dirfile_alloc(&to_path, &to_alloc_len, to, dirent->d_name);
 | |
|       }
 | |
| 
 | |
|       bool is_dir;
 | |
| 
 | |
| #  ifdef __HAIKU__
 | |
|       {
 | |
|         struct stat st_dir;
 | |
|         char filename[FILE_MAX];
 | |
|         BLI_path_join(filename, sizeof(filename), startfrom, dirent->d_name, NULL);
 | |
|         lstat(filename, &st_dir);
 | |
|         is_dir = S_ISDIR(st_dir.st_mode);
 | |
|       }
 | |
| #  else
 | |
|       is_dir = (dirent->d_type == DT_DIR);
 | |
| #  endif
 | |
| 
 | |
|       if (is_dir) {
 | |
|         /* recursively dig into a subfolder */
 | |
|         ret = recursive_operation(
 | |
|             from_path, to_path, callback_dir_pre, callback_file, callback_dir_post);
 | |
|       }
 | |
|       else if (callback_file != NULL) {
 | |
|         ret = callback_file(from_path, to_path);
 | |
|         if (ret != RecursiveOp_Callback_OK) {
 | |
|           ret = -1;
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       if (ret != 0) {
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
|     if (ret != 0) {
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     if (callback_dir_post != NULL) {
 | |
|       ret = callback_dir_post(from, to);
 | |
|       if (ret != RecursiveOp_Callback_OK) {
 | |
|         ret = -1;
 | |
|       }
 | |
|     }
 | |
|   } while (false);
 | |
| 
 | |
|   if (dirlist != NULL) {
 | |
|     for (i = 0; i < n; i++) {
 | |
|       free(dirlist[i]);
 | |
|     }
 | |
|     free(dirlist);
 | |
|   }
 | |
|   if (from_path != NULL) {
 | |
|     MEM_freeN(from_path);
 | |
|   }
 | |
|   if (to_path != NULL) {
 | |
|     MEM_freeN(to_path);
 | |
|   }
 | |
|   if (from != NULL) {
 | |
|     MEM_freeN(from);
 | |
|   }
 | |
|   if (to != NULL) {
 | |
|     MEM_freeN(to);
 | |
|   }
 | |
| 
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| static int delete_callback_post(const char *from, const char *UNUSED(to))
 | |
| {
 | |
|   if (rmdir(from)) {
 | |
|     perror("rmdir");
 | |
| 
 | |
|     return RecursiveOp_Callback_Error;
 | |
|   }
 | |
| 
 | |
|   return RecursiveOp_Callback_OK;
 | |
| }
 | |
| 
 | |
| static int delete_single_file(const char *from, const char *UNUSED(to))
 | |
| {
 | |
|   if (unlink(from)) {
 | |
|     perror("unlink");
 | |
| 
 | |
|     return RecursiveOp_Callback_Error;
 | |
|   }
 | |
| 
 | |
|   return RecursiveOp_Callback_OK;
 | |
| }
 | |
| 
 | |
| #  ifdef __APPLE__
 | |
| static int delete_soft(const char *file, const char **error_message)
 | |
| {
 | |
|   int ret = -1;
 | |
| 
 | |
|   Class NSAutoreleasePoolClass = objc_getClass("NSAutoreleasePool");
 | |
|   SEL allocSel = sel_registerName("alloc");
 | |
|   SEL initSel = sel_registerName("init");
 | |
|   id poolAlloc = ((id(*)(Class, SEL))objc_msgSend)(NSAutoreleasePoolClass, allocSel);
 | |
|   id pool = ((id(*)(id, SEL))objc_msgSend)(poolAlloc, initSel);
 | |
| 
 | |
|   Class NSStringClass = objc_getClass("NSString");
 | |
|   SEL stringWithUTF8StringSel = sel_registerName("stringWithUTF8String:");
 | |
|   id pathString = ((id(*)(Class, SEL, const char *))objc_msgSend)(
 | |
|       NSStringClass, stringWithUTF8StringSel, file);
 | |
| 
 | |
|   Class NSFileManagerClass = objc_getClass("NSFileManager");
 | |
|   SEL defaultManagerSel = sel_registerName("defaultManager");
 | |
|   id fileManager = ((id(*)(Class, SEL))objc_msgSend)(NSFileManagerClass, defaultManagerSel);
 | |
| 
 | |
|   Class NSURLClass = objc_getClass("NSURL");
 | |
|   SEL fileURLWithPathSel = sel_registerName("fileURLWithPath:");
 | |
|   id nsurl = ((id(*)(Class, SEL, id))objc_msgSend)(NSURLClass, fileURLWithPathSel, pathString);
 | |
| 
 | |
|   SEL trashItemAtURLSel = sel_registerName("trashItemAtURL:resultingItemURL:error:");
 | |
|   BOOL deleteSuccessful = ((BOOL(*)(id, SEL, id, id, id))objc_msgSend)(
 | |
|       fileManager, trashItemAtURLSel, nsurl, nil, nil);
 | |
| 
 | |
|   if (deleteSuccessful) {
 | |
|     ret = 0;
 | |
|   }
 | |
|   else {
 | |
|     *error_message = "The Cocoa API call to delete file or directory failed";
 | |
|   }
 | |
| 
 | |
|   SEL drainSel = sel_registerName("drain");
 | |
|   ((void (*)(id, SEL))objc_msgSend)(pool, drainSel);
 | |
| 
 | |
|   return ret;
 | |
| }
 | |
| #  else
 | |
| static int delete_soft(const char *file, const char **error_message)
 | |
| {
 | |
|   const char *args[5];
 | |
|   const char *process_failed;
 | |
| 
 | |
|   char *xdg_current_desktop = getenv("XDG_CURRENT_DESKTOP");
 | |
|   char *xdg_session_desktop = getenv("XDG_SESSION_DESKTOP");
 | |
| 
 | |
|   if ((xdg_current_desktop != NULL && STREQ(xdg_current_desktop, "KDE")) ||
 | |
|       (xdg_session_desktop != NULL && STREQ(xdg_session_desktop, "KDE"))) {
 | |
|     args[0] = "kioclient5";
 | |
|     args[1] = "move";
 | |
|     args[2] = file;
 | |
|     args[3] = "trash:/";
 | |
|     args[4] = NULL;
 | |
|     process_failed = "kioclient5 reported failure";
 | |
|   }
 | |
|   else {
 | |
|     args[0] = "gio";
 | |
|     args[1] = "trash";
 | |
|     args[2] = file;
 | |
|     args[3] = NULL;
 | |
|     process_failed = "gio reported failure";
 | |
|   }
 | |
| 
 | |
|   int pid = fork();
 | |
| 
 | |
|   if (pid != 0) {
 | |
|     /* Parent process */
 | |
|     int wstatus = 0;
 | |
| 
 | |
|     waitpid(pid, &wstatus, 0);
 | |
| 
 | |
|     if (!WIFEXITED(wstatus)) {
 | |
|       *error_message =
 | |
|           "Blender may not support moving files or directories to trash on your system.";
 | |
|       return -1;
 | |
|     }
 | |
|     if (WIFEXITED(wstatus) && WEXITSTATUS(wstatus)) {
 | |
|       *error_message = process_failed;
 | |
|       return -1;
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   execvp(args[0], (char **)args);
 | |
| 
 | |
|   *error_message = "Forking process failed.";
 | |
|   return -1; /* This should only be reached if execvp fails and stack isn't replaced. */
 | |
| }
 | |
| #  endif
 | |
| 
 | |
| FILE *BLI_fopen(const char *filename, const char *mode)
 | |
| {
 | |
|   BLI_assert(!BLI_path_is_rel(filename));
 | |
| 
 | |
|   return fopen(filename, mode);
 | |
| }
 | |
| 
 | |
| void *BLI_gzopen(const char *filename, const char *mode)
 | |
| {
 | |
|   BLI_assert(!BLI_path_is_rel(filename));
 | |
| 
 | |
|   return gzopen(filename, mode);
 | |
| }
 | |
| 
 | |
| int BLI_open(const char *filename, int oflag, int pmode)
 | |
| {
 | |
|   BLI_assert(!BLI_path_is_rel(filename));
 | |
| 
 | |
|   return open(filename, oflag, pmode);
 | |
| }
 | |
| 
 | |
| int BLI_access(const char *filename, int mode)
 | |
| {
 | |
|   BLI_assert(!BLI_path_is_rel(filename));
 | |
| 
 | |
|   return access(filename, mode);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Deletes the specified file or directory (depending on dir), optionally
 | |
|  * doing recursive delete of directory contents.
 | |
|  *
 | |
|  * \return zero on success (matching 'remove' behavior).
 | |
|  */
 | |
| int BLI_delete(const char *file, bool dir, bool recursive)
 | |
| {
 | |
|   BLI_assert(!BLI_path_is_rel(file));
 | |
| 
 | |
|   if (recursive) {
 | |
|     return recursive_operation(file, NULL, NULL, delete_single_file, delete_callback_post);
 | |
|   }
 | |
|   if (dir) {
 | |
|     return rmdir(file);
 | |
|   }
 | |
|   return remove(file);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Soft deletes the specified file or directory (depending on dir) by moving the files to the
 | |
|  * recycling bin, optionally doing recursive delete of directory contents.
 | |
|  *
 | |
|  * \return zero on success (matching 'remove' behavior).
 | |
|  */
 | |
| int BLI_delete_soft(const char *file, const char **error_message)
 | |
| {
 | |
|   BLI_assert(!BLI_path_is_rel(file));
 | |
| 
 | |
|   return delete_soft(file, error_message);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Do the two paths denote the same file-system object?
 | |
|  */
 | |
| static bool check_the_same(const char *path_a, const char *path_b)
 | |
| {
 | |
|   struct stat st_a, st_b;
 | |
| 
 | |
|   if (lstat(path_a, &st_a)) {
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   if (lstat(path_b, &st_b)) {
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   return st_a.st_dev == st_b.st_dev && st_a.st_ino == st_b.st_ino;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Sets the mode and ownership of file to the values from st.
 | |
|  */
 | |
| static int set_permissions(const char *file, const struct stat *st)
 | |
| {
 | |
|   if (chown(file, st->st_uid, st->st_gid)) {
 | |
|     perror("chown");
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   if (chmod(file, st->st_mode)) {
 | |
|     perror("chmod");
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| /* pre-recursive callback for copying operation
 | |
|  * creates a destination directory where all source content fill be copied to */
 | |
| static int copy_callback_pre(const char *from, const char *to)
 | |
| {
 | |
|   struct stat st;
 | |
| 
 | |
|   if (check_the_same(from, to)) {
 | |
|     fprintf(stderr, "%s: '%s' is the same as '%s'\n", __func__, from, to);
 | |
|     return RecursiveOp_Callback_Error;
 | |
|   }
 | |
| 
 | |
|   if (lstat(from, &st)) {
 | |
|     perror("stat");
 | |
|     return RecursiveOp_Callback_Error;
 | |
|   }
 | |
| 
 | |
|   /* create a directory */
 | |
|   if (mkdir(to, st.st_mode)) {
 | |
|     perror("mkdir");
 | |
|     return RecursiveOp_Callback_Error;
 | |
|   }
 | |
| 
 | |
|   /* set proper owner and group on new directory */
 | |
|   if (chown(to, st.st_uid, st.st_gid)) {
 | |
|     perror("chown");
 | |
|     return RecursiveOp_Callback_Error;
 | |
|   }
 | |
| 
 | |
|   return RecursiveOp_Callback_OK;
 | |
| }
 | |
| 
 | |
| static int copy_single_file(const char *from, const char *to)
 | |
| {
 | |
|   FILE *from_stream, *to_stream;
 | |
|   struct stat st;
 | |
|   char buf[4096];
 | |
|   size_t len;
 | |
| 
 | |
|   if (check_the_same(from, to)) {
 | |
|     fprintf(stderr, "%s: '%s' is the same as '%s'\n", __func__, from, to);
 | |
|     return RecursiveOp_Callback_Error;
 | |
|   }
 | |
| 
 | |
|   if (lstat(from, &st)) {
 | |
|     perror("lstat");
 | |
|     return RecursiveOp_Callback_Error;
 | |
|   }
 | |
| 
 | |
|   if (S_ISLNK(st.st_mode)) {
 | |
|     /* symbolic links should be copied in special way */
 | |
|     char *link_buffer;
 | |
|     int need_free;
 | |
|     ssize_t link_len;
 | |
| 
 | |
|     /* get large enough buffer to read link content */
 | |
|     if ((st.st_size + 1) < sizeof(buf)) {
 | |
|       link_buffer = buf;
 | |
|       need_free = 0;
 | |
|     }
 | |
|     else {
 | |
|       link_buffer = MEM_callocN(st.st_size + 2, "copy_single_file link_buffer");
 | |
|       need_free = 1;
 | |
|     }
 | |
| 
 | |
|     link_len = readlink(from, link_buffer, st.st_size + 1);
 | |
|     if (link_len < 0) {
 | |
|       perror("readlink");
 | |
| 
 | |
|       if (need_free) {
 | |
|         MEM_freeN(link_buffer);
 | |
|       }
 | |
| 
 | |
|       return RecursiveOp_Callback_Error;
 | |
|     }
 | |
| 
 | |
|     link_buffer[link_len] = '\0';
 | |
| 
 | |
|     if (symlink(link_buffer, to)) {
 | |
|       perror("symlink");
 | |
|       if (need_free) {
 | |
|         MEM_freeN(link_buffer);
 | |
|       }
 | |
|       return RecursiveOp_Callback_Error;
 | |
|     }
 | |
| 
 | |
|     if (need_free) {
 | |
|       MEM_freeN(link_buffer);
 | |
|     }
 | |
| 
 | |
|     return RecursiveOp_Callback_OK;
 | |
|   }
 | |
|   if (S_ISCHR(st.st_mode) || S_ISBLK(st.st_mode) || S_ISFIFO(st.st_mode) || S_ISSOCK(st.st_mode)) {
 | |
|     /* copy special type of file */
 | |
|     if (mknod(to, st.st_mode, st.st_rdev)) {
 | |
|       perror("mknod");
 | |
|       return RecursiveOp_Callback_Error;
 | |
|     }
 | |
| 
 | |
|     if (set_permissions(to, &st)) {
 | |
|       return RecursiveOp_Callback_Error;
 | |
|     }
 | |
| 
 | |
|     return RecursiveOp_Callback_OK;
 | |
|   }
 | |
|   if (!S_ISREG(st.st_mode)) {
 | |
|     fprintf(stderr, "Copying of this kind of files isn't supported yet\n");
 | |
|     return RecursiveOp_Callback_Error;
 | |
|   }
 | |
| 
 | |
|   from_stream = fopen(from, "rb");
 | |
|   if (!from_stream) {
 | |
|     perror("fopen");
 | |
|     return RecursiveOp_Callback_Error;
 | |
|   }
 | |
| 
 | |
|   to_stream = fopen(to, "wb");
 | |
|   if (!to_stream) {
 | |
|     perror("fopen");
 | |
|     fclose(from_stream);
 | |
|     return RecursiveOp_Callback_Error;
 | |
|   }
 | |
| 
 | |
|   while ((len = fread(buf, 1, sizeof(buf), from_stream)) > 0) {
 | |
|     fwrite(buf, 1, len, to_stream);
 | |
|   }
 | |
| 
 | |
|   fclose(to_stream);
 | |
|   fclose(from_stream);
 | |
| 
 | |
|   if (set_permissions(to, &st)) {
 | |
|     return RecursiveOp_Callback_Error;
 | |
|   }
 | |
| 
 | |
|   return RecursiveOp_Callback_OK;
 | |
| }
 | |
| 
 | |
| /* Not used anywhere! */
 | |
| #  if 0
 | |
| static int move_callback_pre(const char *from, const char *to)
 | |
| {
 | |
|   int ret = rename(from, to);
 | |
| 
 | |
|   if (ret) {
 | |
|     return copy_callback_pre(from, to);
 | |
|   }
 | |
| 
 | |
|   return RecursiveOp_Callback_StopRecurs;
 | |
| }
 | |
| 
 | |
| static int move_single_file(const char *from, const char *to)
 | |
| {
 | |
|   int ret = rename(from, to);
 | |
| 
 | |
|   if (ret) {
 | |
|     return copy_single_file(from, to);
 | |
|   }
 | |
| 
 | |
|   return RecursiveOp_Callback_OK;
 | |
| }
 | |
| 
 | |
| /* if *file represents a directory, moves all its contents into *to, else renames
 | |
|  * file itself to *to. */
 | |
| int BLI_move(const char *file, const char *to)
 | |
| {
 | |
|   int ret = recursive_operation(file, to, move_callback_pre, move_single_file, NULL);
 | |
| 
 | |
|   if (ret && ret != -1) {
 | |
|     return recursive_operation(file, NULL, NULL, delete_single_file, delete_callback_post);
 | |
|   }
 | |
| 
 | |
|   return ret;
 | |
| }
 | |
| #  endif
 | |
| 
 | |
| static const char *check_destination(const char *file, const char *to)
 | |
| {
 | |
|   struct stat st;
 | |
| 
 | |
|   if (!stat(to, &st)) {
 | |
|     if (S_ISDIR(st.st_mode)) {
 | |
|       char *str, *path;
 | |
|       const char *filename;
 | |
|       size_t len = 0;
 | |
| 
 | |
|       str = strip_last_slash(file);
 | |
|       filename = BLI_path_slash_rfind(str);
 | |
| 
 | |
|       if (!filename) {
 | |
|         MEM_freeN(str);
 | |
|         return (char *)to;
 | |
|       }
 | |
| 
 | |
|       /* skip slash */
 | |
|       filename += 1;
 | |
| 
 | |
|       len = strlen(to) + strlen(filename) + 1;
 | |
|       path = MEM_callocN(len + 1, "check_destination path");
 | |
|       BLI_join_dirfile(path, len + 1, to, filename);
 | |
| 
 | |
|       MEM_freeN(str);
 | |
| 
 | |
|       return path;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return to;
 | |
| }
 | |
| 
 | |
| int BLI_copy(const char *file, const char *to)
 | |
| {
 | |
|   const char *actual_to = check_destination(file, to);
 | |
|   int ret;
 | |
| 
 | |
|   ret = recursive_operation(file, actual_to, copy_callback_pre, copy_single_file, NULL);
 | |
| 
 | |
|   if (actual_to != to) {
 | |
|     MEM_freeN((void *)actual_to);
 | |
|   }
 | |
| 
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| #  if 0
 | |
| int BLI_create_symlink(const char *file, const char *to)
 | |
| {
 | |
|   return symlink(to, file);
 | |
| }
 | |
| #  endif
 | |
| 
 | |
| /** \return true on success (i.e. given path now exists on FS), false otherwise. */
 | |
| bool BLI_dir_create_recursive(const char *dirname)
 | |
| {
 | |
|   char *lslash;
 | |
|   size_t size;
 | |
| #  ifdef MAXPATHLEN
 | |
|   char static_buf[MAXPATHLEN];
 | |
| #  endif
 | |
|   char *tmp;
 | |
|   bool ret = true;
 | |
| 
 | |
|   if (BLI_is_dir(dirname)) {
 | |
|     return true;
 | |
|   }
 | |
|   if (BLI_exists(dirname)) {
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
| #  ifdef MAXPATHLEN
 | |
|   size = MAXPATHLEN;
 | |
|   tmp = static_buf;
 | |
| #  else
 | |
|   size = strlen(dirname) + 1;
 | |
|   tmp = MEM_callocN(size, __func__);
 | |
| #  endif
 | |
| 
 | |
|   BLI_strncpy(tmp, dirname, size);
 | |
| 
 | |
|   /* Avoids one useless recursion in case of '/foo/bar/' path... */
 | |
|   BLI_path_slash_rstrip(tmp);
 | |
| 
 | |
|   lslash = (char *)BLI_path_slash_rfind(tmp);
 | |
|   if (lslash) {
 | |
|     /* Split about the last slash and recurse */
 | |
|     *lslash = 0;
 | |
|     if (!BLI_dir_create_recursive(tmp)) {
 | |
|       ret = false;
 | |
|     }
 | |
|   }
 | |
| 
 | |
| #  ifndef MAXPATHLEN
 | |
|   MEM_freeN(tmp);
 | |
| #  endif
 | |
| 
 | |
|   if (ret) {
 | |
|     ret = (mkdir(dirname, 0777) == 0);
 | |
|   }
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * \return zero on success (matching 'rename' behavior).
 | |
|  */
 | |
| int BLI_rename(const char *from, const char *to)
 | |
| {
 | |
|   if (!BLI_exists(from)) {
 | |
|     return 1;
 | |
|   }
 | |
| 
 | |
|   if (BLI_exists(to)) {
 | |
|     if (BLI_delete(to, false, false)) {
 | |
|       return 1;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return rename(from, to);
 | |
| }
 | |
| 
 | |
| #endif
 |