metaforce/hecl/lib/hecl.cpp

590 lines
18 KiB
C++
Raw Normal View History

2016-03-04 23:02:44 +00:00
#include "hecl/hecl.hpp"
#include <thread>
#include <mutex>
#include <unordered_map>
#ifdef WIN32
#include <windows.h>
#ifndef _WIN32_IE
#define _WIN32_IE 0x0400
#endif
#include <shlobj.h>
#endif
#ifdef __APPLE__
#include <Carbon/Carbon.h>
#endif
#ifdef __linux__
#include <mntent.h>
#endif
2016-03-04 23:02:44 +00:00
namespace hecl
{
2015-08-06 19:10:12 +00:00
unsigned VerbosityLevel = 0;
2016-03-04 23:02:44 +00:00
logvisor::Module LogModule("hecl");
static const std::string Illegals {"<>?\"|"};
2015-08-05 01:54:35 +00:00
void SanitizePath(std::string& path)
{
if (path.empty())
return;
2015-08-05 01:54:35 +00:00
path.erase(std::remove(path.begin(), path.end(), '\n'), path.end());
path.erase(std::remove(path.begin(), path.end(), '\r'), path.end());
std::string::iterator p1 = path.begin();
2015-11-11 04:01:36 +00:00
bool ic = false;
std::transform(path.begin(), path.end(), path.begin(), [&](const char a) -> char {
++p1;
if (Illegals.find_first_of(a) != std::string::npos)
2015-11-11 08:41:25 +00:00
{
ic = false;
return '_';
}
2015-11-11 04:01:36 +00:00
if (ic)
{
ic = false;
return a;
}
if (a == '\\' && (p1 == path.end() || *p1 != '\\'))
2015-11-11 04:01:36 +00:00
{
ic = true;
return '/';
2015-11-11 04:01:36 +00:00
}
2015-08-05 02:40:54 +00:00
return a;
});
2015-08-05 01:54:35 +00:00
}
static const std::wstring WIllegals {L"<>?*\"|"};
2015-08-05 01:54:35 +00:00
void SanitizePath(std::wstring& path)
{
if (path.empty())
return;
2015-08-05 01:54:35 +00:00
path.erase(std::remove(path.begin(), path.end(), L'\n'), path.end());
path.erase(std::remove(path.begin(), path.end(), L'\r'), path.end());
std::wstring::iterator p1 = path.begin();
2015-11-11 04:01:36 +00:00
bool ic = false;
std::transform(path.begin(), path.end(), path.begin(), [&](const wchar_t a) -> wchar_t {
++p1;
if (WIllegals.find_first_of(a) != std::wstring::npos)
2015-11-11 08:41:25 +00:00
{
ic = false;
return L'_';
}
2015-11-11 04:01:36 +00:00
if (ic)
{
ic = false;
return a;
}
if (a == L'\\' && (p1 == path.end() || *p1 != L'\\'))
2015-11-11 04:01:36 +00:00
{
ic = true;
return L'/';
2015-11-11 04:01:36 +00:00
}
2015-08-05 02:40:54 +00:00
return a;
});
2015-08-05 01:54:35 +00:00
}
static std::mutex PathsMutex;
static std::unordered_map<std::thread::id, ProjectPath> PathsInProgress;
bool ResourceLock::InProgress(const ProjectPath& path)
{
std::unique_lock<std::mutex> lk(PathsMutex);
for (const auto& p : PathsInProgress)
if (p.second == path)
return true;
return false;
}
bool ResourceLock::SetThreadRes(const ProjectPath& path)
{
std::unique_lock<std::mutex> lk(PathsMutex);
if (PathsInProgress.find(std::this_thread::get_id()) != PathsInProgress.cend())
LogModule.report(logvisor::Fatal, "multiple resource locks on thread");
for (const auto& p : PathsInProgress)
if (p.second == path)
return false;
PathsInProgress[std::this_thread::get_id()] = path;
return true;
}
void ResourceLock::ClearThreadRes()
{
std::unique_lock<std::mutex> lk(PathsMutex);
PathsInProgress.erase(std::this_thread::get_id());
}
2016-03-04 23:02:44 +00:00
bool IsPathPNG(const hecl::ProjectPath& path)
2015-10-01 00:40:06 +00:00
{
2016-03-04 23:02:44 +00:00
FILE* fp = hecl::Fopen(path.getAbsolutePath().c_str(), _S("rb"));
2015-10-01 00:40:06 +00:00
if (!fp)
return false;
uint32_t buf;
if (fread(&buf, 1, 4, fp) != 4)
{
fclose(fp);
return false;
}
fclose(fp);
2016-03-04 23:02:44 +00:00
buf = hecl::SBig(buf);
2015-10-01 00:40:06 +00:00
if (buf == 0x89504e47)
return true;
return false;
}
2016-03-04 23:02:44 +00:00
bool IsPathBlend(const hecl::ProjectPath& path)
2015-10-01 00:40:06 +00:00
{
2016-09-21 05:41:06 +00:00
const SystemChar* lastCompExt = path.getLastComponentExt();
2016-03-04 23:02:44 +00:00
if (!lastCompExt || hecl::StrCmp(lastCompExt, _S("blend")))
2015-10-07 01:16:54 +00:00
return false;
2016-09-21 05:41:06 +00:00
FILE* fp = hecl::Fopen(path.getAbsolutePath().c_str(), _S("rb"));
2015-10-01 00:40:06 +00:00
if (!fp)
return false;
uint32_t buf;
if (fread(&buf, 1, 4, fp) != 4)
{
fclose(fp);
return false;
}
fclose(fp);
2016-03-04 23:02:44 +00:00
buf = hecl::SLittle(buf);
2015-10-01 00:40:06 +00:00
if (buf == 0x4e454c42 || buf == 0x88b1f)
return true;
return false;
}
2016-03-04 23:02:44 +00:00
bool IsPathYAML(const hecl::ProjectPath& path)
2015-10-01 00:40:06 +00:00
{
const SystemChar* lastCompExt = path.getLastComponentExt();
if (!lastCompExt)
return false;
2016-03-04 23:02:44 +00:00
if (!hecl::StrCmp(lastCompExt, _S("yaml")) ||
!hecl::StrCmp(lastCompExt, _S("yml")))
2015-10-01 00:40:06 +00:00
return true;
return false;
}
2016-03-04 23:02:44 +00:00
hecl::DirectoryEnumerator::DirectoryEnumerator(const hecl::SystemChar* path, Mode mode,
bool sizeSort, bool reverse, bool noHidden)
2016-01-01 00:16:20 +00:00
{
2016-03-04 23:02:44 +00:00
hecl::Sstat theStat;
if (hecl::Stat(path, &theStat) || !S_ISDIR(theStat.st_mode))
2016-01-01 00:16:20 +00:00
return;
#if _WIN32
2016-03-04 23:02:44 +00:00
hecl::SystemString wc(path);
2016-01-01 00:16:20 +00:00
wc += _S("/*");
WIN32_FIND_DATAW d;
HANDLE dir = FindFirstFileW(wc.c_str(), &d);
if (dir == INVALID_HANDLE_VALUE)
return;
switch (mode)
{
case Mode::Native:
do
{
if (!wcscmp(d.cFileName, _S(".")) || !wcscmp(d.cFileName, _S("..")))
continue;
2016-01-02 04:16:20 +00:00
if (noHidden && (d.cFileName[0] == L'.' || (d.dwFileAttributes & FILE_ATTRIBUTE_HIDDEN) != 0))
continue;
2016-03-04 23:02:44 +00:00
hecl::SystemString fp(path);
2016-01-01 00:16:20 +00:00
fp += _S('/');
fp += d.cFileName;
2016-03-04 23:02:44 +00:00
hecl::Sstat st;
if (hecl::Stat(fp.c_str(), &st))
2016-01-01 00:16:20 +00:00
continue;
size_t sz = 0;
bool isDir = false;
if (S_ISDIR(st.st_mode))
isDir = true;
else if (S_ISREG(st.st_mode))
sz = st.st_size;
else
continue;
m_entries.push_back(std::move(Entry(std::move(fp), d.cFileName, sz, isDir)));
} while (FindNextFileW(dir, &d));
break;
case Mode::DirsThenFilesSorted:
case Mode::DirsSorted:
{
2016-03-04 23:02:44 +00:00
std::map<hecl::SystemString, Entry, CaseInsensitiveCompare> sort;
2016-01-01 00:16:20 +00:00
do
{
if (!wcscmp(d.cFileName, _S(".")) || !wcscmp(d.cFileName, _S("..")))
continue;
2016-01-02 04:16:20 +00:00
if (noHidden && (d.cFileName[0] == L'.' || (d.dwFileAttributes & FILE_ATTRIBUTE_HIDDEN) != 0))
continue;
2016-03-04 23:02:44 +00:00
hecl::SystemString fp(path);
2016-01-01 00:16:20 +00:00
fp +=_S('/');
fp += d.cFileName;
2016-03-04 23:02:44 +00:00
hecl::Sstat st;
if (hecl::Stat(fp.c_str(), &st) || !S_ISDIR(st.st_mode))
2016-01-01 00:16:20 +00:00
continue;
sort.emplace(std::make_pair(d.cFileName, Entry(std::move(fp), d.cFileName, 0, true)));
} while (FindNextFileW(dir, &d));
if (reverse)
for (auto it=sort.crbegin() ; it != sort.crend() ; ++it)
m_entries.push_back(std::move(it->second));
else
for (auto& e : sort)
m_entries.push_back(std::move(e.second));
if (mode == Mode::DirsSorted)
break;
FindClose(dir);
dir = FindFirstFileW(wc.c_str(), &d);
}
case Mode::FilesSorted:
{
if (mode == Mode::FilesSorted)
m_entries.clear();
if (sizeSort)
{
std::multimap<size_t, Entry> sort;
2016-01-01 00:16:20 +00:00
do
{
if (!wcscmp(d.cFileName, _S(".")) || !wcscmp(d.cFileName, _S("..")))
continue;
2016-01-02 04:16:20 +00:00
if (noHidden && (d.cFileName[0] == L'.' || (d.dwFileAttributes & FILE_ATTRIBUTE_HIDDEN) != 0))
continue;
2016-03-04 23:02:44 +00:00
hecl::SystemString fp(path);
2016-01-01 00:16:20 +00:00
fp += _S('/');
fp += d.cFileName;
2016-03-04 23:02:44 +00:00
hecl::Sstat st;
if (hecl::Stat(fp.c_str(), &st) || !S_ISREG(st.st_mode))
2016-01-01 00:16:20 +00:00
continue;
sort.emplace(std::make_pair(st.st_size, Entry(std::move(fp), d.cFileName, st.st_size, false)));
} while (FindNextFileW(dir, &d));
if (reverse)
for (auto it=sort.crbegin() ; it != sort.crend() ; ++it)
m_entries.push_back(std::move(it->second));
else
for (auto& e : sort)
m_entries.push_back(std::move(e.second));
}
else
{
2016-03-04 23:02:44 +00:00
std::map<hecl::SystemString, Entry, CaseInsensitiveCompare> sort;
2016-01-01 00:16:20 +00:00
do
{
if (!wcscmp(d.cFileName, _S(".")) || !wcscmp(d.cFileName, _S("..")))
continue;
2016-01-02 04:16:20 +00:00
if (noHidden && (d.cFileName[0] == L'.' || (d.dwFileAttributes & FILE_ATTRIBUTE_HIDDEN) != 0))
continue;
2016-03-04 23:02:44 +00:00
hecl::SystemString fp(path);
2016-01-01 00:16:20 +00:00
fp += _S('/');
fp += d.cFileName;
2016-03-04 23:02:44 +00:00
hecl::Sstat st;
if (hecl::Stat(fp.c_str(), &st) || !S_ISREG(st.st_mode))
2016-01-01 00:16:20 +00:00
continue;
sort.emplace(std::make_pair(d.cFileName, Entry(std::move(fp), d.cFileName, st.st_size, false)));
} while (FindNextFileW(dir, &d));
if (reverse)
for (auto it=sort.crbegin() ; it != sort.crend() ; ++it)
m_entries.push_back(std::move(it->second));
else
for (auto& e : sort)
m_entries.push_back(std::move(e.second));
}
break;
}
}
FindClose(dir);
#else
DIR* dir = opendir(path);
if (!dir)
return;
const dirent* d;
switch (mode)
{
case Mode::Native:
while ((d = readdir(dir)))
{
if (!strcmp(d->d_name, ".") || !strcmp(d->d_name, ".."))
continue;
if (noHidden && d->d_name[0] == '.')
continue;
2016-03-04 23:02:44 +00:00
hecl::SystemString fp(path);
2016-01-01 00:16:20 +00:00
fp += '/';
fp += d->d_name;
2016-03-04 23:02:44 +00:00
hecl::Sstat st;
if (hecl::Stat(fp.c_str(), &st))
2016-01-01 00:16:20 +00:00
continue;
size_t sz = 0;
bool isDir = false;
if (S_ISDIR(st.st_mode))
isDir = true;
else if (S_ISREG(st.st_mode))
sz = st.st_size;
else
continue;
m_entries.push_back(std::move(Entry(std::move(fp), d->d_name, sz, isDir)));
}
break;
case Mode::DirsThenFilesSorted:
case Mode::DirsSorted:
{
2016-03-04 23:02:44 +00:00
std::map<hecl::SystemString, Entry, CaseInsensitiveCompare> sort;
2016-01-01 00:16:20 +00:00
while ((d = readdir(dir)))
{
if (!strcmp(d->d_name, ".") || !strcmp(d->d_name, ".."))
continue;
if (noHidden && d->d_name[0] == '.')
continue;
2016-03-04 23:02:44 +00:00
hecl::SystemString fp(path);
2016-01-01 00:16:20 +00:00
fp += '/';
fp += d->d_name;
2016-03-04 23:02:44 +00:00
hecl::Sstat st;
if (hecl::Stat(fp.c_str(), &st) || !S_ISDIR(st.st_mode))
2016-01-01 00:16:20 +00:00
continue;
sort.emplace(std::make_pair(d->d_name, Entry(std::move(fp), d->d_name, 0, true)));
}
if (reverse)
for (auto it=sort.crbegin() ; it != sort.crend() ; ++it)
m_entries.push_back(std::move(it->second));
else
for (auto& e : sort)
m_entries.push_back(std::move(e.second));
if (mode == Mode::DirsSorted)
break;
rewinddir(dir);
}
case Mode::FilesSorted:
{
if (mode == Mode::FilesSorted)
m_entries.clear();
if (sizeSort)
{
std::multimap<size_t, Entry> sort;
2016-01-01 00:16:20 +00:00
while ((d = readdir(dir)))
{
if (!strcmp(d->d_name, ".") || !strcmp(d->d_name, ".."))
continue;
if (noHidden && d->d_name[0] == '.')
continue;
2016-03-04 23:02:44 +00:00
hecl::SystemString fp(path);
2016-01-01 00:16:20 +00:00
fp += '/';
fp += d->d_name;
2016-03-04 23:02:44 +00:00
hecl::Sstat st;
if (hecl::Stat(fp.c_str(), &st) || !S_ISREG(st.st_mode))
2016-01-01 00:16:20 +00:00
continue;
sort.emplace(std::make_pair(st.st_size, Entry(std::move(fp), d->d_name, st.st_size, false)));
}
if (reverse)
for (auto it=sort.crbegin() ; it != sort.crend() ; ++it)
m_entries.push_back(std::move(it->second));
else
for (auto& e : sort)
m_entries.push_back(std::move(e.second));
}
else
{
2016-03-04 23:02:44 +00:00
std::map<hecl::SystemString, Entry, CaseInsensitiveCompare> sort;
2016-01-01 00:16:20 +00:00
while ((d = readdir(dir)))
{
if (!strcmp(d->d_name, ".") || !strcmp(d->d_name, ".."))
continue;
if (noHidden && d->d_name[0] == '.')
continue;
2016-03-04 23:02:44 +00:00
hecl::SystemString fp(path);
2016-01-01 00:16:20 +00:00
fp += '/';
fp += d->d_name;
2016-03-04 23:02:44 +00:00
hecl::Sstat st;
if (hecl::Stat(fp.c_str(), &st) || !S_ISREG(st.st_mode))
2016-01-01 00:16:20 +00:00
continue;
sort.emplace(std::make_pair(d->d_name, Entry(std::move(fp), d->d_name, st.st_size, false)));
}
if (reverse)
for (auto it=sort.crbegin() ; it != sort.crend() ; ++it)
m_entries.push_back(std::move(it->second));
else
for (auto& e : sort)
m_entries.push_back(std::move(e.second));
}
break;
}
}
closedir(dir);
#endif
}
2016-01-02 04:16:20 +00:00
#define FILE_MAXDIR 768
2016-03-04 23:02:44 +00:00
static std::pair<hecl::SystemString, std::string>
NameFromPath(const hecl::SystemString& path)
2016-01-02 04:16:20 +00:00
{
2016-03-04 23:02:44 +00:00
hecl::SystemUTF8View utf8(path);
2016-01-02 04:16:20 +00:00
if (utf8.str().size() == 1 && utf8.str()[0] == '/')
return {path, "/"};
size_t lastSlash = utf8.str().rfind('/');
if (lastSlash != std::string::npos)
return {path, std::string(utf8.str().cbegin() + lastSlash + 1, utf8.str().cend())};
else
return {path, utf8.str()};
}
2016-03-04 23:02:44 +00:00
std::vector<std::pair<hecl::SystemString, std::string>> GetSystemLocations()
{
2016-03-04 23:02:44 +00:00
std::vector<std::pair<hecl::SystemString, std::string>> ret;
#ifdef WIN32
2016-01-02 04:16:20 +00:00
/* Add the drive names to the listing (as queried by blender) */
{
wchar_t wline[FILE_MAXDIR];
2016-01-02 04:16:20 +00:00
wchar_t* name;
__int64 tmp;
int i;
tmp = GetLogicalDrives();
2016-01-02 04:16:20 +00:00
for (i = 0; i < 26; i++)
{
if ((tmp >> i) & 1)
{
wline[0] = L'A' + i;
wline[1] = L':';
wline[2] = L'/';
wline[3] = L'\0';
name = nullptr;
/* Flee from horrible win querying hover floppy drives! */
2016-01-02 04:16:20 +00:00
if (i > 1)
{
/* Try to get volume label as well... */
2016-01-02 04:16:20 +00:00
if (GetVolumeInformationW(wline, wline + 4, FILE_MAXDIR - 4, nullptr, nullptr, nullptr, nullptr, 0))
{
size_t labelLen = wcslen(wline + 4);
_snwprintf(wline + 4 + labelLen, FILE_MAXDIR - 4 - labelLen, L" (%.2s)", wline);
name = wline + 4;
}
}
2016-01-02 04:16:20 +00:00
wline[2] = L'\0';
if (name)
2016-03-04 23:02:44 +00:00
ret.emplace_back(wline, hecl::WideToUTF8(name));
2016-01-02 04:16:20 +00:00
else
ret.push_back(NameFromPath(wline));
}
}
/* Adding Desktop and My Documents */
2016-01-02 04:16:20 +00:00
SystemString wpath;
SHGetSpecialFolderPathW(0, wline, CSIDL_PERSONAL, 0);
wpath.assign(wline);
SanitizePath(wpath);
ret.push_back(NameFromPath(wpath));
SHGetSpecialFolderPathW(0, wline, CSIDL_DESKTOPDIRECTORY, 0);
wpath.assign(wline);
SanitizePath(wpath);
ret.push_back(NameFromPath(wpath));
}
#else
#ifdef __APPLE__
{
2016-03-04 23:02:44 +00:00
hecl::Sstat theStat;
const char* home = getenv("HOME");
if (home)
{
ret.push_back(NameFromPath(home));
std::string desktop(home);
desktop += "/Desktop";
2016-03-04 23:02:44 +00:00
if (!hecl::Stat(desktop.c_str(), &theStat))
ret.push_back(NameFromPath(desktop));
}
/* Get mounted volumes better method OSX 10.6 and higher, see: */
/*https://developer.apple.com/library/mac/#documentation/CoreFOundation/Reference/CFURLRef/Reference/reference.html*/
/* we get all volumes sorted including network and do not relay on user-defined finder visibility, less confusing */
CFURLRef cfURL = NULL;
CFURLEnumeratorResult result = kCFURLEnumeratorSuccess;
CFURLEnumeratorRef volEnum = CFURLEnumeratorCreateForMountedVolumes(NULL, kCFURLEnumeratorSkipInvisibles, NULL);
2016-01-02 20:39:14 +00:00
while (result != kCFURLEnumeratorEnd)
{
char defPath[1024];
result = CFURLEnumeratorGetNextURL(volEnum, &cfURL, NULL);
if (result != kCFURLEnumeratorSuccess)
continue;
CFURLGetFileSystemRepresentation(cfURL, false, (UInt8 *)defPath, 1024);
2016-01-02 20:39:14 +00:00
ret.push_back(NameFromPath(defPath));
}
CFRelease(volEnum);
}
#else
/* unix */
{
2016-03-04 23:02:44 +00:00
hecl::Sstat theStat;
const char* home = getenv("HOME");
if (home)
{
2016-01-02 23:11:11 +00:00
ret.push_back(NameFromPath(home));
std::string desktop(home);
desktop += "/Desktop";
2016-03-04 23:02:44 +00:00
if (!hecl::Stat(desktop.c_str(), &theStat))
2016-01-02 23:11:11 +00:00
ret.push_back(NameFromPath(desktop));
}
{
bool found = false;
#ifdef __linux__
/* Loop over mount points */
struct mntent *mnt;
FILE* fp = setmntent(MOUNTED, "r");
if (fp)
{
while ((mnt = getmntent(fp)))
{
if (strlen(mnt->mnt_fsname) < 4 || strncmp(mnt->mnt_fsname, "/dev", 4))
continue;
std::string mntStr(mnt->mnt_dir);
if (mntStr.size() > 1 && mntStr.back() == '/')
mntStr.pop_back();
2016-01-02 23:11:11 +00:00
ret.push_back(NameFromPath(mntStr));
found = true;
}
endmntent(fp);
}
#endif
/* Fallback */
if (!found)
2016-01-02 23:11:11 +00:00
ret.push_back(NameFromPath("/"));
}
}
#endif
#endif
return ret;
}
}