ppsspp/file/file_util.cpp

390 lines
8.2 KiB
C++
Raw Normal View History

2012-03-24 22:39:19 +00:00
#ifdef _WIN32
#define WIN32_LEAN_AND_MEAN
#include <Windows.h>
#include <direct.h>
2012-03-24 22:39:19 +00:00
#else
#include <dirent.h>
#include <unistd.h>
#include <errno.h>
2012-03-24 22:39:19 +00:00
#endif
#include <cstring>
2012-03-24 22:39:19 +00:00
#include <string>
2012-10-31 12:34:27 +00:00
#include <set>
2012-11-01 08:45:27 +00:00
#include <algorithm>
2012-03-24 22:39:19 +00:00
#include <stdio.h>
2012-10-31 12:34:27 +00:00
#include <sys/stat.h>
2012-10-31 19:42:43 +00:00
#include <ctype.h>
2012-10-31 12:34:27 +00:00
2012-03-24 22:39:19 +00:00
#include "base/logging.h"
#include "base/basictypes.h"
#include "file/file_util.h"
#if defined(__FreeBSD__) || defined(__APPLE__) || defined(__SYMBIAN32__)
#define stat64 stat
#endif
2012-12-13 06:43:08 +00:00
// Hack
#ifdef __SYMBIAN32__
static inline int readdir_r(DIR *dirp, struct dirent *entry, struct dirent **result) {
struct dirent *readdir_entry;
readdir_entry = readdir(dirp);
if (readdir_entry == NULL) {
*result = NULL;
return errno;
}
*entry = *readdir_entry;
*result = entry;
return 0;
}
#endif
2012-04-12 13:18:43 +00:00
bool writeStringToFile(bool text_file, const std::string &str, const char *filename)
2012-03-24 22:39:19 +00:00
{
2012-10-31 12:23:16 +00:00
FILE *f = fopen(filename, text_file ? "w" : "wb");
if (!f)
return false;
size_t len = str.size();
if (len != fwrite(str.data(), 1, str.size(), f))
{
fclose(f);
return false;
}
fclose(f);
return true;
2012-03-24 22:39:19 +00:00
}
2012-12-08 22:26:14 +00:00
bool writeDataToFile(bool text_file, const void* data, const unsigned int size, const char *filename)
{
FILE *f = fopen(filename, text_file ? "w" : "wb");
if (!f)
return false;
size_t len = size;
if (len != fwrite(data, 1, len, f))
{
fclose(f);
return false;
}
fclose(f);
return true;
}
2012-03-24 22:39:19 +00:00
uint64_t GetSize(FILE *f)
{
2012-10-31 12:23:16 +00:00
// can't use off_t here because it can be 32-bit
uint64_t pos = ftell(f);
if (fseek(f, 0, SEEK_END) != 0) {
return 0;
}
uint64_t size = ftell(f);
// Reset the seek position to where it was when we started.
if ((size != pos) && (fseek(f, pos, SEEK_SET) != 0)) {
// Should error here
return 0;
}
return size;
2012-03-24 22:39:19 +00:00
}
bool ReadFileToString(bool text_file, const char *filename, std::string &str)
{
2012-10-31 12:23:16 +00:00
FILE *f = fopen(filename, text_file ? "r" : "rb");
if (!f)
return false;
size_t len = (size_t)GetSize(f);
char *buf = new char[len + 1];
buf[fread(buf, 1, len, f)] = 0;
str = std::string(buf, len);
fclose(f);
delete [] buf;
return true;
2012-03-24 22:39:19 +00:00
}
2012-12-08 22:26:14 +00:00
bool readDataFromFile(bool text_file, unsigned char* &data, const unsigned int size, const char *filename)
{
FILE *f = fopen(filename, text_file ? "r" : "rb");
if (!f)
return false;
size_t len = (size_t)GetSize(f);
if(len < size)
{
fclose(f);
return false;
}
data[fread(data, 1, size, f)] = 0;
fclose(f);
return true;
}
2012-03-24 22:39:19 +00:00
#define DIR_SEP "/"
2012-10-28 10:37:10 +00:00
#define DIR_SEP_CHR '\\'
2012-03-24 22:39:19 +00:00
#ifndef METRO
2012-10-28 10:37:10 +00:00
// Remove any ending forward slashes from directory paths
// Modifies argument.
static void stripTailDirSlashes(std::string &fname)
{
if (fname.length() > 1)
{
size_t i = fname.length() - 1;
while (fname[i] == DIR_SEP_CHR)
fname[i--] = '\0';
}
return;
}
// Returns true if file filename exists
bool exists(const std::string &filename)
{
2012-10-30 15:23:08 +00:00
#ifdef _WIN32
return GetFileAttributes(filename.c_str()) != 0xFFFFFFFF;
#else
2012-10-28 10:37:10 +00:00
struct stat64 file_info;
std::string copy(filename);
stripTailDirSlashes(copy);
int result = stat64(copy.c_str(), &file_info);
return (result == 0);
2012-10-30 15:23:08 +00:00
#endif
2012-10-28 10:37:10 +00:00
}
// Returns true if filename is a directory
bool isDirectory(const std::string &filename)
{
FileInfo info;
getFileInfo(filename.c_str(), &info);
return info.isDirectory;
}
bool getFileInfo(const char *path, FileInfo *fileInfo)
{
// TODO: Expand relative paths?
fileInfo->fullName = path;
2012-10-30 15:23:08 +00:00
#ifdef _WIN32
WIN32_FILE_ATTRIBUTE_DATA attrs;
if (!GetFileAttributesExA(path, GetFileExInfoStandard, &attrs)) {
fileInfo->size = 0;
fileInfo->isDirectory = false;
2013-06-08 15:43:27 +00:00
fileInfo->exists = false;
return false;
}
fileInfo->size = (uint64_t)attrs.nFileSizeLow | ((uint64_t)attrs.nFileSizeHigh << 32);
fileInfo->isDirectory = (attrs.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) != 0;
fileInfo->isWritable = (attrs.dwFileAttributes & FILE_ATTRIBUTE_READONLY) == 0;
2013-06-08 15:43:27 +00:00
fileInfo->exists = true;
2012-10-30 15:23:08 +00:00
#else
2012-10-28 10:37:10 +00:00
struct stat64 file_info;
std::string copy(path);
2012-10-28 10:37:10 +00:00
stripTailDirSlashes(copy);
int result = stat64(copy.c_str(), &file_info);
if (result < 0) {
WLOG("IsDirectory: stat failed on %s", path);
2013-06-08 15:43:27 +00:00
fileInfo->exists = false;
2012-10-28 10:37:10 +00:00
return false;
}
fileInfo->isDirectory = S_ISDIR(file_info.st_mode);
fileInfo->isWritable = false;
fileInfo->size = file_info.st_size;
2013-06-08 15:43:27 +00:00
fileInfo->exists = true;
// HACK: approximation
if (file_info.st_mode & 0200)
fileInfo->isWritable = true;
2012-10-30 15:23:08 +00:00
#endif
return true;
2012-10-28 10:37:10 +00:00
}
2012-10-31 12:34:27 +00:00
std::string getFileExtension(const std::string &fn) {
int pos = fn.rfind(".");
if (pos < 0) return "";
std::string ext = fn.substr(pos+1);
for (size_t i = 0; i < ext.size(); i++) {
ext[i] = tolower(ext[i]);
}
return ext;
}
size_t getFilesInDir(const char *directory, std::vector<FileInfo> *files, const char *filter) {
2012-10-31 12:23:16 +00:00
size_t foundEntries = 0;
2012-10-31 12:34:27 +00:00
std::set<std::string> filters;
std::string tmp;
if (filter) {
while (*filter) {
if (*filter == ':') {
filters.insert(tmp);
tmp = "";
} else {
tmp.push_back(*filter);
}
filter++;
}
}
2013-04-18 21:39:22 +00:00
if (tmp.size())
filters.insert(tmp);
2012-03-24 22:39:19 +00:00
#ifdef _WIN32
2012-10-31 12:23:16 +00:00
// Find the first file in the directory.
WIN32_FIND_DATA ffd;
2012-03-24 22:39:19 +00:00
#ifdef UNICODE
2012-10-31 12:23:16 +00:00
HANDLE hFind = FindFirstFile((std::wstring(directory) + "\\*").c_str(), &ffd);
2012-03-24 22:39:19 +00:00
#else
2012-10-31 12:23:16 +00:00
HANDLE hFind = FindFirstFile((std::string(directory) + "\\*").c_str(), &ffd);
2012-03-24 22:39:19 +00:00
#endif
2012-10-31 12:23:16 +00:00
if (hFind == INVALID_HANDLE_VALUE) {
FindClose(hFind);
return 0;
}
// windows loop
do
{
const std::string virtualName(ffd.cFileName);
2012-03-24 22:39:19 +00:00
#else
2012-11-14 15:43:33 +00:00
struct dirent_large { struct dirent entry; char padding[FILENAME_MAX+1]; };
struct dirent_large diren;
struct dirent *result = NULL;
2012-03-24 22:39:19 +00:00
//std::string directoryWithSlash = directory;
//if (directoryWithSlash.back() != '/')
// directoryWithSlash += "/";
2012-10-31 12:23:16 +00:00
DIR *dirp = opendir(directory);
if (!dirp)
return 0;
// non windows loop
2012-11-14 15:43:33 +00:00
while (!readdir_r(dirp, (dirent*) &diren, &result) && result)
2012-10-31 12:23:16 +00:00
{
const std::string virtualName(result->d_name);
2012-03-24 22:39:19 +00:00
#endif
2012-10-31 12:23:16 +00:00
// check for "." and ".."
if (((virtualName[0] == '.') && (virtualName[1] == '\0')) ||
((virtualName[0] == '.') && (virtualName[1] == '.') &&
(virtualName[2] == '\0')))
continue;
2012-10-28 10:37:10 +00:00
// Remove dotfiles (should be made optional?)
if (virtualName[0] == '.')
continue;
2012-10-31 12:34:27 +00:00
2012-10-31 12:23:16 +00:00
FileInfo info;
info.name = virtualName;
info.fullName = std::string(directory) + "/" + virtualName;
info.isDirectory = isDirectory(info.fullName);
2012-11-27 15:38:24 +00:00
info.exists = true;
2013-06-09 10:40:53 +00:00
info.size = 0;
2012-10-31 12:34:27 +00:00
if (!info.isDirectory) {
std::string ext = getFileExtension(info.fullName);
if (filter) {
if (filters.find(ext) == filters.end())
continue;
}
}
2012-10-31 12:23:16 +00:00
files->push_back(info);
2012-10-31 12:34:27 +00:00
#ifdef _WIN32
2012-10-31 12:23:16 +00:00
} while (FindNextFile(hFind, &ffd) != 0);
FindClose(hFind);
2012-03-24 22:39:19 +00:00
#else
2012-10-31 12:23:16 +00:00
}
closedir(dirp);
2012-03-24 22:39:19 +00:00
#endif
2012-11-01 08:45:27 +00:00
std::sort(files->begin(), files->end());
2012-10-31 12:23:16 +00:00
return foundEntries;
2012-03-24 22:39:19 +00:00
}
void deleteFile(const char *file)
{
#ifdef _WIN32
2012-10-31 12:23:16 +00:00
if (!DeleteFile(file)) {
ELOG("Error deleting %s: %i", file, GetLastError());
}
2012-03-24 22:39:19 +00:00
#else
2012-10-31 12:23:16 +00:00
int err = unlink(file);
if (err) {
ELOG("Error unlinking %s: %i", file, err);
}
2012-03-24 22:39:19 +00:00
#endif
}
2013-06-08 15:43:27 +00:00
void deleteDir(const char *dir)
{
#ifdef _WIN32
if (!RemoveDirectory(dir)) {
ELOG("Error deleting directory %s: %i", dir, GetLastError());
}
#else
rmdir(dir);
#endif
}
2012-03-31 09:16:13 +00:00
#endif
2012-03-31 20:07:33 +00:00
std::string getDir(const std::string &path)
{
2012-10-28 10:37:10 +00:00
if (path == "/")
return path;
2013-06-08 15:43:27 +00:00
int n = (int)path.size() - 1;
2012-10-31 12:23:16 +00:00
while (n >= 0 && path[n] != '\\' && path[n] != '/')
n--;
2013-04-14 07:58:01 +00:00
std::string cutpath = n > 0 ? path.substr(0, n) : "";
2012-10-31 12:23:16 +00:00
for (size_t i = 0; i < cutpath.size(); i++)
{
if (cutpath[i] == '\\') cutpath[i] = '/';
}
2012-10-28 10:47:26 +00:00
#ifndef _WIN32
if (!cutpath.size()) {
return "/";
}
#endif
2012-10-31 12:23:16 +00:00
return cutpath;
}
std::string getFilename(std::string path) {
size_t off = getDir(path).size() + 1;
if (off < path.size())
return path.substr(off);
else
return path;
}
void mkDir(const std::string &path)
{
#ifdef _WIN32
mkdir(path.c_str());
#else
mkdir(path.c_str(), 0777);
#endif
}
#ifdef _WIN32
// Returns a vector with the device names
std::vector<std::string> getWindowsDrives()
{
std::vector<std::string> drives;
const DWORD buffsize = GetLogicalDriveStrings(0, NULL);
std::vector<TCHAR> buff(buffsize);
if (GetLogicalDriveStrings(buffsize, buff.data()) == buffsize - 1)
{
auto drive = buff.data();
while (*drive)
{
std::string str(drive);
str.pop_back(); // we don't want the final backslash
str += "/";
drives.push_back(str);
// advance to next drive
while (*drive++) {}
}
}
return drives;
}
#endif