mirror of
https://github.com/hrydgard/ppsspp.git
synced 2024-11-23 13:30:02 +00:00
267 lines
6.8 KiB
C++
267 lines
6.8 KiB
C++
#include "ppsspp_config.h"
|
|
|
|
#ifdef _WIN32
|
|
#define WIN32_LEAN_AND_MEAN
|
|
#include <Windows.h>
|
|
#include <direct.h>
|
|
#else
|
|
#include <strings.h>
|
|
#include <dirent.h>
|
|
#include <unistd.h>
|
|
#include <errno.h>
|
|
#endif
|
|
#include <cstring>
|
|
#include <string>
|
|
#include <set>
|
|
#include <algorithm>
|
|
#include <cstdio>
|
|
#include <sys/stat.h>
|
|
#include <ctype.h>
|
|
|
|
#include "Common/Data/Encoding/Utf8.h"
|
|
#include "Common/StringUtils.h"
|
|
#include "Common/File/DirListing.h"
|
|
#include "Common/File/FileUtil.h"
|
|
|
|
#if !defined(__linux__) && !defined(_WIN32) && !defined(__QNX__)
|
|
#define stat64 stat
|
|
#endif
|
|
|
|
#ifdef HAVE_LIBNX
|
|
// Far from optimal, but I guess it works...
|
|
#define fseeko fseek
|
|
#define ftello ftell
|
|
#define fileno
|
|
#endif // HAVE_LIBNX
|
|
|
|
namespace File {
|
|
|
|
bool GetFileInfo(const char *path, FileInfo * fileInfo) {
|
|
// TODO: Expand relative paths?
|
|
fileInfo->fullName = path;
|
|
|
|
#ifdef _WIN32
|
|
auto FiletimeToStatTime = [](FILETIME ft) {
|
|
const int windowsTickResolution = 10000000;
|
|
const int64_t secToUnixEpoch = 11644473600LL;
|
|
int64_t ticks = ((uint64_t)ft.dwHighDateTime << 32) | ft.dwLowDateTime;
|
|
return (int64_t)(ticks / windowsTickResolution - secToUnixEpoch);
|
|
};
|
|
|
|
WIN32_FILE_ATTRIBUTE_DATA attrs;
|
|
if (!GetFileAttributesExW(ConvertUTF8ToWString(path).c_str(), GetFileExInfoStandard, &attrs)) {
|
|
fileInfo->size = 0;
|
|
fileInfo->isDirectory = false;
|
|
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;
|
|
fileInfo->exists = true;
|
|
fileInfo->atime = FiletimeToStatTime(attrs.ftLastAccessTime);
|
|
fileInfo->mtime = FiletimeToStatTime(attrs.ftLastWriteTime);
|
|
fileInfo->ctime = FiletimeToStatTime(attrs.ftCreationTime);
|
|
if (attrs.dwFileAttributes & FILE_ATTRIBUTE_READONLY) {
|
|
fileInfo->access = 0444; // Read
|
|
} else {
|
|
fileInfo->access = 0666; // Read/Write
|
|
}
|
|
if (attrs.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) {
|
|
fileInfo->access |= 0111; // Execute
|
|
}
|
|
#else
|
|
|
|
#if (defined __ANDROID__) && (__ANDROID_API__ < 21)
|
|
struct stat file_info;
|
|
int result = stat(path, &file_info);
|
|
#else
|
|
struct stat64 file_info;
|
|
int result = stat64(path, &file_info);
|
|
#endif
|
|
if (result < 0) {
|
|
fileInfo->exists = false;
|
|
return false;
|
|
}
|
|
|
|
fileInfo->isDirectory = S_ISDIR(file_info.st_mode);
|
|
fileInfo->isWritable = false;
|
|
fileInfo->size = file_info.st_size;
|
|
fileInfo->exists = true;
|
|
fileInfo->atime = file_info.st_atime;
|
|
fileInfo->mtime = file_info.st_mtime;
|
|
fileInfo->ctime = file_info.st_ctime;
|
|
fileInfo->access = file_info.st_mode & 0x1ff;
|
|
// HACK: approximation
|
|
if (file_info.st_mode & 0200)
|
|
fileInfo->isWritable = true;
|
|
#endif
|
|
return true;
|
|
}
|
|
|
|
bool GetModifTime(const std::string & filename, tm & return_time) {
|
|
memset(&return_time, 0, sizeof(return_time));
|
|
FileInfo info;
|
|
if (GetFileInfo(filename.c_str(), &info)) {
|
|
time_t t = info.mtime;
|
|
localtime_r((time_t*)&t, &return_time);
|
|
return true;
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
bool FileInfo::operator <(const FileInfo & other) const {
|
|
if (isDirectory && !other.isDirectory)
|
|
return true;
|
|
else if (!isDirectory && other.isDirectory)
|
|
return false;
|
|
if (strcasecmp(name.c_str(), other.name.c_str()) < 0)
|
|
return true;
|
|
else
|
|
return false;
|
|
}
|
|
|
|
size_t GetFilesInDir(const char *directory, std::vector<FileInfo> * files, const char *filter, int flags) {
|
|
size_t foundEntries = 0;
|
|
std::set<std::string> filters;
|
|
if (filter) {
|
|
std::string tmp;
|
|
while (*filter) {
|
|
if (*filter == ':') {
|
|
filters.insert(std::move(tmp));
|
|
} else {
|
|
tmp.push_back(*filter);
|
|
}
|
|
filter++;
|
|
}
|
|
if (!tmp.empty())
|
|
filters.insert(std::move(tmp));
|
|
}
|
|
#ifdef _WIN32
|
|
// Find the first file in the directory.
|
|
WIN32_FIND_DATA ffd;
|
|
HANDLE hFind = FindFirstFileEx((ConvertUTF8ToWString(directory) + L"\\*").c_str(), FindExInfoStandard, &ffd, FindExSearchNameMatch, NULL, 0);
|
|
if (hFind == INVALID_HANDLE_VALUE) {
|
|
return 0;
|
|
}
|
|
// windows loop
|
|
do
|
|
{
|
|
const std::string virtualName = ConvertWStringToUTF8(ffd.cFileName);
|
|
#else
|
|
struct dirent *result = NULL;
|
|
|
|
//std::string directoryWithSlash = directory;
|
|
//if (directoryWithSlash.back() != '/')
|
|
// directoryWithSlash += "/";
|
|
|
|
DIR *dirp = opendir(directory);
|
|
if (!dirp)
|
|
return 0;
|
|
// non windows loop
|
|
while ((result = readdir(dirp)))
|
|
{
|
|
const std::string virtualName(result->d_name);
|
|
#endif
|
|
// check for "." and ".."
|
|
if (virtualName == "." || virtualName == "..")
|
|
continue;
|
|
|
|
// Remove dotfiles (optional with flag.)
|
|
if (!(flags & GETFILES_GETHIDDEN))
|
|
{
|
|
#ifdef _WIN32
|
|
if ((ffd.dwFileAttributes & FILE_ATTRIBUTE_HIDDEN) != 0)
|
|
continue;
|
|
#else
|
|
if (virtualName[0] == '.')
|
|
continue;
|
|
#endif
|
|
}
|
|
|
|
FileInfo info;
|
|
info.name = virtualName;
|
|
std::string dir = directory;
|
|
|
|
// Only append a slash if there isn't one on the end.
|
|
size_t lastSlash = dir.find_last_of("/");
|
|
if (lastSlash != (dir.length() - 1))
|
|
dir.append("/");
|
|
|
|
info.fullName = dir + virtualName;
|
|
info.isDirectory = IsDirectory(info.fullName);
|
|
info.exists = true;
|
|
info.size = 0;
|
|
info.isWritable = false; // TODO - implement some kind of check
|
|
if (!info.isDirectory) {
|
|
std::string ext = GetFileExtension(info.fullName);
|
|
if (filter) {
|
|
if (filters.find(ext) == filters.end())
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if (files)
|
|
files->push_back(std::move(info));
|
|
foundEntries++;
|
|
#ifdef _WIN32
|
|
} while (FindNextFile(hFind, &ffd) != 0);
|
|
FindClose(hFind);
|
|
#else
|
|
}
|
|
closedir(dirp);
|
|
#endif
|
|
if (files)
|
|
std::sort(files->begin(), files->end());
|
|
return foundEntries;
|
|
}
|
|
|
|
int64_t GetDirectoryRecursiveSize(const std::string & path, const char *filter, int flags) {
|
|
std::vector<FileInfo> fileInfo;
|
|
GetFilesInDir(path.c_str(), &fileInfo, filter, flags);
|
|
int64_t sizeSum = 0;
|
|
// Note: GetFilesInDir does not fill in fileSize properly.
|
|
for (size_t i = 0; i < fileInfo.size(); i++) {
|
|
FileInfo finfo;
|
|
GetFileInfo(fileInfo[i].fullName.c_str(), &finfo);
|
|
if (!finfo.isDirectory)
|
|
sizeSum += finfo.size;
|
|
else
|
|
sizeSum += GetDirectoryRecursiveSize(finfo.fullName, filter, flags);
|
|
}
|
|
return sizeSum;
|
|
}
|
|
|
|
#ifdef _WIN32
|
|
// Returns a vector with the device names
|
|
std::vector<std::string> GetWindowsDrives()
|
|
{
|
|
#if PPSSPP_PLATFORM(UWP)
|
|
return std::vector<std::string>(); // TODO UWP http://stackoverflow.com/questions/37404405/how-to-get-logical-drives-names-in-windows-10
|
|
#else
|
|
std::vector<std::string> drives;
|
|
|
|
const DWORD buffsize = GetLogicalDriveStrings(0, NULL);
|
|
std::vector<wchar_t> buff(buffsize);
|
|
if (GetLogicalDriveStrings(buffsize, buff.data()) == buffsize - 1)
|
|
{
|
|
auto drive = buff.data();
|
|
while (*drive)
|
|
{
|
|
std::string str(ConvertWStringToUTF8(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
|
|
}
|
|
#endif
|
|
|
|
} // namespace File
|