1392 lines
		
	
	
		
			32 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1392 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') {
 | 
						|
    fclose(ufopen(filename, "a"));
 | 
						|
  }
 | 
						|
 | 
						|
  /* 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
 |