2008-09-07 12:29:46 +02:00
|
|
|
// Copyright (C) 2003-2008 Dolphin Project.
|
|
|
|
|
|
|
|
// 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, version 2.0.
|
|
|
|
|
|
|
|
// 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 2.0 for more details.
|
|
|
|
|
|
|
|
// A copy of the GPL 2.0 should have been included with the program.
|
|
|
|
// If not, see http://www.gnu.org/licenses/
|
|
|
|
|
|
|
|
// Official SVN repository and contact information can be found at
|
|
|
|
// http://code.google.com/p/dolphin-emu/
|
|
|
|
|
2008-07-16 22:50:16 +02:00
|
|
|
#include "Common.h"
|
|
|
|
#include "FileUtil.h"
|
2008-09-07 12:29:46 +02:00
|
|
|
|
2008-07-21 03:27:13 +02:00
|
|
|
#ifdef _WIN32
|
2008-09-22 21:48:12 +02:00
|
|
|
#include <windows.h>
|
2008-09-07 12:29:46 +02:00
|
|
|
#include <shlobj.h> // for SHGetFolderPath
|
2008-07-20 14:26:32 +02:00
|
|
|
#include <shellapi.h>
|
2008-09-07 12:29:46 +02:00
|
|
|
#include <commdlg.h> // for GetSaveFileName
|
2008-09-24 01:27:38 +02:00
|
|
|
#include <io.h>
|
2008-07-27 22:51:02 +02:00
|
|
|
#else
|
2008-09-07 12:29:46 +02:00
|
|
|
|
2008-07-27 22:51:02 +02:00
|
|
|
#include <sys/stat.h>
|
2008-09-05 13:59:28 +02:00
|
|
|
#include <errno.h>
|
2008-09-05 15:12:47 +02:00
|
|
|
#include <stdlib.h>
|
2008-07-21 03:27:13 +02:00
|
|
|
#endif
|
2008-07-16 22:50:16 +02:00
|
|
|
|
2008-09-24 00:23:38 +02:00
|
|
|
namespace File
|
|
|
|
{
|
|
|
|
|
|
|
|
bool Exists(const char *filename)
|
2008-07-16 22:50:16 +02:00
|
|
|
{
|
|
|
|
#ifdef _WIN32
|
2008-09-22 21:48:12 +02:00
|
|
|
return GetFileAttributes(filename) != INVALID_FILE_ATTRIBUTES;
|
2008-07-16 22:50:16 +02:00
|
|
|
#else
|
2008-07-27 22:51:02 +02:00
|
|
|
struct stat file_info;
|
2008-09-22 21:48:12 +02:00
|
|
|
int result = stat(filename, &file_info);
|
2008-07-27 22:51:02 +02:00
|
|
|
return result == 0;
|
2008-07-16 22:50:16 +02:00
|
|
|
#endif
|
2008-07-20 14:26:32 +02:00
|
|
|
}
|
|
|
|
|
2008-09-24 00:23:38 +02:00
|
|
|
bool IsDirectory(const char *filename)
|
2008-09-22 21:48:12 +02:00
|
|
|
{
|
2008-07-20 17:12:12 +02:00
|
|
|
#ifdef _WIN32
|
2008-11-01 16:59:06 +01:00
|
|
|
DWORD Attribs = GetFileAttributes(filename);
|
|
|
|
if (Attribs == INVALID_FILE_ATTRIBUTES)
|
|
|
|
return false;
|
|
|
|
|
2008-09-22 21:48:12 +02:00
|
|
|
return (GetFileAttributes(filename) & FILE_ATTRIBUTE_DIRECTORY) != 0;
|
2008-07-20 17:12:12 +02:00
|
|
|
#else
|
2008-09-24 00:23:38 +02:00
|
|
|
struct stat file_info;
|
2008-09-22 21:48:12 +02:00
|
|
|
int result = stat(filename, &file_info);
|
2008-09-24 00:23:38 +02:00
|
|
|
if (result == 0)
|
|
|
|
return S_ISDIR(file_info.st_mode);
|
|
|
|
else
|
|
|
|
return false;
|
2008-07-20 17:12:12 +02:00
|
|
|
#endif
|
|
|
|
}
|
2008-07-20 14:26:32 +02:00
|
|
|
|
2008-09-24 00:23:38 +02:00
|
|
|
bool Delete(const char *filename)
|
2008-09-22 21:48:12 +02:00
|
|
|
{
|
2008-09-24 00:23:38 +02:00
|
|
|
if (!Exists(filename))
|
2008-09-22 21:48:12 +02:00
|
|
|
return false;
|
2008-09-24 00:23:38 +02:00
|
|
|
if (IsDirectory(filename))
|
2008-09-22 21:48:12 +02:00
|
|
|
return false;
|
|
|
|
#ifdef _WIN32
|
|
|
|
DeleteFile(filename);
|
|
|
|
#else
|
|
|
|
unlink(filename);
|
|
|
|
#endif
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string SanitizePath(const char *filename)
|
|
|
|
{
|
2008-07-20 14:26:32 +02:00
|
|
|
std::string copy = filename;
|
2008-08-17 21:28:24 +02:00
|
|
|
for (size_t i = 0; i < copy.size(); i++)
|
2008-07-27 22:51:02 +02:00
|
|
|
if (copy[i] == '/')
|
|
|
|
copy[i] = '\\';
|
2008-07-20 14:26:32 +02:00
|
|
|
return copy;
|
|
|
|
}
|
|
|
|
|
2008-09-24 00:23:38 +02:00
|
|
|
void Launch(const char *filename)
|
2008-07-20 14:26:32 +02:00
|
|
|
{
|
|
|
|
#ifdef _WIN32
|
|
|
|
std::string win_filename = SanitizePath(filename);
|
|
|
|
SHELLEXECUTEINFO shex = { sizeof(shex) };
|
|
|
|
shex.fMask = SEE_MASK_NO_CONSOLE; // | SEE_MASK_ASYNC_OK;
|
|
|
|
shex.lpVerb = "open";
|
|
|
|
shex.lpFile = win_filename.c_str();
|
|
|
|
shex.nShow = SW_SHOWNORMAL;
|
|
|
|
ShellExecuteEx(&shex);
|
2008-07-27 22:51:02 +02:00
|
|
|
#else
|
|
|
|
// TODO: Insert GNOME/KDE code here.
|
2008-07-20 14:26:32 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2008-09-24 00:23:38 +02:00
|
|
|
void Explore(const char *path)
|
2008-07-20 14:26:32 +02:00
|
|
|
{
|
|
|
|
#ifdef _WIN32
|
|
|
|
std::string win_path = SanitizePath(path);
|
|
|
|
SHELLEXECUTEINFO shex = { sizeof(shex) };
|
|
|
|
shex.fMask = SEE_MASK_NO_CONSOLE; // | SEE_MASK_ASYNC_OK;
|
|
|
|
shex.lpVerb = "explore";
|
|
|
|
shex.lpFile = win_path.c_str();
|
|
|
|
shex.nShow = SW_SHOWNORMAL;
|
|
|
|
ShellExecuteEx(&shex);
|
2008-07-27 22:51:02 +02:00
|
|
|
#else
|
|
|
|
// TODO: Insert GNOME/KDE code here.
|
2008-07-20 14:26:32 +02:00
|
|
|
#endif
|
|
|
|
}
|
2008-07-20 23:20:22 +02:00
|
|
|
|
|
|
|
// Returns true if successful, or path already exists.
|
2008-09-24 00:23:38 +02:00
|
|
|
bool CreateDir(const char *path)
|
2008-07-20 23:20:22 +02:00
|
|
|
{
|
|
|
|
#ifdef _WIN32
|
2008-09-22 21:48:12 +02:00
|
|
|
if (::CreateDirectory(path, NULL))
|
2008-07-20 23:20:22 +02:00
|
|
|
return true;
|
2008-07-27 22:51:02 +02:00
|
|
|
DWORD error = GetLastError();
|
|
|
|
if (error == ERROR_ALREADY_EXISTS)
|
2008-07-20 23:20:22 +02:00
|
|
|
{
|
2008-09-22 21:48:12 +02:00
|
|
|
PanicAlert("%s already exists", path);
|
2008-07-27 22:51:02 +02:00
|
|
|
return true;
|
2008-07-20 23:20:22 +02:00
|
|
|
}
|
2008-07-27 22:51:02 +02:00
|
|
|
PanicAlert("Error creating directory: %i", error);
|
|
|
|
return false;
|
|
|
|
#else
|
2008-09-24 00:23:38 +02:00
|
|
|
if (mkdir(path, 0644) == 0)
|
2008-09-07 12:29:46 +02:00
|
|
|
return true;
|
2008-09-05 13:59:28 +02:00
|
|
|
|
2008-09-24 00:23:38 +02:00
|
|
|
int err = errno;
|
2008-09-05 13:59:28 +02:00
|
|
|
|
2008-09-24 00:23:38 +02:00
|
|
|
if (err == EEXIST)
|
|
|
|
{
|
2008-09-22 21:48:12 +02:00
|
|
|
PanicAlert("%s already exists", path);
|
2008-09-07 12:29:46 +02:00
|
|
|
return true;
|
2008-09-24 00:23:38 +02:00
|
|
|
}
|
2008-09-05 13:59:28 +02:00
|
|
|
|
2008-09-24 00:23:38 +02:00
|
|
|
PanicAlert("Error creating directory: %s", strerror(err));
|
2008-09-05 13:59:28 +02:00
|
|
|
return false;
|
2008-09-24 00:23:38 +02:00
|
|
|
|
2008-07-20 23:20:22 +02:00
|
|
|
#endif
|
2008-08-17 21:28:24 +02:00
|
|
|
}
|
2008-09-05 15:12:47 +02:00
|
|
|
|
2008-09-24 00:23:38 +02:00
|
|
|
std::string GetUserDirectory()
|
|
|
|
{
|
2008-09-05 15:12:47 +02:00
|
|
|
#ifdef _WIN32
|
2008-09-07 12:29:46 +02:00
|
|
|
char path[MAX_PATH];
|
|
|
|
if (SUCCEEDED(SHGetFolderPath(NULL, CSIDL_COMMON_APPDATA, NULL, 0, path)))
|
|
|
|
{
|
|
|
|
return std::string(path);
|
|
|
|
}
|
|
|
|
return std::string("");
|
2008-09-05 15:12:47 +02:00
|
|
|
#else
|
2008-09-07 12:29:46 +02:00
|
|
|
char *dir = getenv("HOME");
|
|
|
|
if (!dir)
|
|
|
|
return std::string("");
|
2008-09-24 00:23:38 +02:00
|
|
|
return dir;
|
2008-09-05 15:12:47 +02:00
|
|
|
#endif
|
|
|
|
}
|
2008-09-22 21:48:12 +02:00
|
|
|
|
2008-09-24 00:23:38 +02:00
|
|
|
u64 GetSize(const char *filename)
|
2008-09-22 21:48:12 +02:00
|
|
|
{
|
2008-11-01 16:59:06 +01:00
|
|
|
FILE *pFile = fopen(filename, "rb");
|
|
|
|
if (pFile)
|
|
|
|
{
|
|
|
|
fseek(pFile, 0, SEEK_END);
|
|
|
|
u64 pos = ftell(pFile);
|
|
|
|
fclose(pFile);
|
|
|
|
return pos;
|
|
|
|
}
|
|
|
|
return 0;
|
2008-09-22 21:48:12 +02:00
|
|
|
}
|
2008-09-24 00:23:38 +02:00
|
|
|
|
2008-09-24 01:27:38 +02:00
|
|
|
#ifdef _WIN32
|
|
|
|
static bool ReadFoundFile(const WIN32_FIND_DATA& ffd, FSTEntry& entry)
|
|
|
|
{
|
|
|
|
// ignore files starting with a .
|
|
|
|
if(strncmp(ffd.cFileName, ".", 1) == 0)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
entry.virtualName = ffd.cFileName;
|
|
|
|
|
|
|
|
if(ffd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
|
|
|
|
{
|
|
|
|
entry.isDirectory = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
entry.isDirectory = false;
|
|
|
|
entry.size = ffd.nFileSizeLow;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 ScanDirectoryTree(const std::string& _Directory, FSTEntry& parentEntry)
|
|
|
|
{
|
|
|
|
// Find the first file in the directory.
|
|
|
|
WIN32_FIND_DATA ffd;
|
|
|
|
std::string searchName = _Directory + "\\*";
|
|
|
|
HANDLE hFind = FindFirstFile(searchName.c_str(), &ffd);
|
|
|
|
|
|
|
|
u32 foundEntries = 0;
|
|
|
|
|
|
|
|
if (hFind != INVALID_HANDLE_VALUE)
|
|
|
|
{
|
|
|
|
do
|
|
|
|
{
|
|
|
|
FSTEntry entry;
|
|
|
|
|
|
|
|
if(ReadFoundFile(ffd, entry))
|
|
|
|
{
|
|
|
|
entry.physicalName = _Directory + "\\" + entry.virtualName;
|
|
|
|
if(entry.isDirectory)
|
|
|
|
{
|
|
|
|
u32 childEntries = ScanDirectoryTree(entry.physicalName, entry);
|
|
|
|
entry.size = childEntries;
|
|
|
|
foundEntries += childEntries;
|
|
|
|
}
|
|
|
|
|
|
|
|
++foundEntries;
|
|
|
|
|
|
|
|
parentEntry.children.push_back(entry);
|
|
|
|
}
|
|
|
|
} while (FindNextFile(hFind, &ffd) != 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
FindClose(hFind);
|
|
|
|
|
|
|
|
return foundEntries;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
u32 ScanDirectoryTree(const std::string& _Directory, FSTEntry& parentEntry)
|
|
|
|
{
|
|
|
|
// TODO - Insert linux stuff here
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2008-11-01 16:59:06 +01:00
|
|
|
bool CreateEmptyFile(const char *filename)
|
|
|
|
{
|
|
|
|
FILE* pFile = fopen(filename, "wb");
|
|
|
|
if (pFile == NULL)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
fclose(pFile);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2008-09-24 00:23:38 +02:00
|
|
|
} // namespace
|