Merge branch 'Dani-Hub-develop' into develop

This commit is contained in:
Beman 2017-04-02 16:44:25 -04:00
commit 12bcf2e6ea

View File

@ -101,6 +101,7 @@ using std::wstring;
// See MinGW's windef.h
# define WINVER 0x501
# endif
# include <cwchar>
# include <io.h>
# include <windows.h>
# include <winnt.h>
@ -516,25 +517,115 @@ namespace
|| errval == ERROR_BAD_PATHNAME // "//nosuch" on Win64
|| errval == ERROR_BAD_NETPATH; // "//nosuch" on Win32
}
// File name case-insensitive comparison needs to be locale- and collation-independent.
// The approach used below follows a combined strategy described in the following
// articles:
// http://archives.miloush.net/michkap/archive/2005/10/17/481600.html
// http://archives.miloush.net/michkap/archive/2007/09/14/4900107.html
// http://archives.miloush.net/michkap/archive/2007/10/12/5396685.html
// CompareStringOrdinal is only available on newer systems and is just a wrapper of
// RtlCompareUnicodeString, but measurements showed that RtlEqualUnicodeString has better
// performance. Therefore we use RtlEqualUnicodeString, and if that does not exist
// we perform the equivalent characterwise comparsion using LCMapString and uppercase
// binary equality. Instead of calling RtlInitUnicodeString we use wcslen directly
// because that results in better performance as well.
// some distributions of mingw as early as GLIBCXX__ 20110325 have _stricmp, but the
// offical 4.6.2 release with __GLIBCXX__ 20111026 doesn't. Play it safe for now, and
// only use _stricmp if _MSC_VER is defined
#if defined(_MSC_VER) // || (defined(__GLIBCXX__) && __GLIBCXX__ >= 20110325)
# define BOOST_FILESYSTEM_STRICMP _stricmp
#else
# define BOOST_FILESYSTEM_STRICMP strcmp
// Windows ntdll.dll functions that may or may not be present
// must be accessed through pointers
typedef struct _UNICODE_STRING {
USHORT Length;
USHORT MaximumLength;
PWSTR Buffer;
} UNICODE_STRING;
typedef const UNICODE_STRING *PCUNICODE_STRING;
typedef BOOLEAN (WINAPI *PtrRtlEqualUnicodeString)(
/*_In_*/ PCUNICODE_STRING String1,
/*_In_*/ PCUNICODE_STRING String2,
/*_In_*/ BOOLEAN CaseInSensitive
);
PtrRtlEqualUnicodeString rtl_equal_unicode_string_api = PtrRtlEqualUnicodeString(
::GetProcAddress(
::GetModuleHandleW(L"ntdll.dll"), "RtlEqualUnicodeString"));
#ifndef LOCALE_INVARIANT
# define LOCALE_INVARIANT (MAKELCID(MAKELANGID(LANG_INVARIANT, SUBLANG_NEUTRAL), SORT_DEFAULT))
#endif
bool equal_string_ordinal_ic_1(const wchar_t* s1, const wchar_t* s2)
{
std::size_t len1 = std::wcslen(s1);
UNICODE_STRING us1;
us1.Buffer = const_cast<wchar_t*>(s1);
us1.Length = static_cast<USHORT>(sizeof(*s1) * len1);
us1.MaximumLength = us1.Length + sizeof(*s1);
std::size_t len2 = std::wcslen(s2);
UNICODE_STRING us2;
us2.Buffer = const_cast<wchar_t*>(s2);
us2.Length = static_cast<USHORT>(sizeof(*s2) * len2);
us2.MaximumLength = us2.Length + sizeof(*s2);
BOOLEAN res = rtl_equal_unicode_string_api(&us1, &us2, TRUE);
return res != FALSE;
}
inline
wchar_t to_upper_invariant(wchar_t input)
{
wchar_t result;
// According to
// https://msdn.microsoft.com/en-us/library/windows/desktop/dd318144(v=vs.85).aspx
// "When transforming between uppercase and lowercase, the function always maps a
// single character to a single character."
int res = ::LCMapStringW(LOCALE_INVARIANT, LCMAP_UPPERCASE, &input, 1, &result, 1);
if (res != 0)
return result;
assert(!"LCMapStringW failed to convert a character to upper case");
return input; // Should never happen, but this is a safe fallback.
}
bool equal_string_ordinal_ic_2(const wchar_t* s1, const wchar_t* s2)
{
for (;; ++s1, ++s2)
{
const wchar_t c1 = *s1;
const wchar_t c2 = *s2;
if (c1 == c2)
{
if (!c1)
return true; // We have reached the end of both strings, no difference found.
}
else
{
if (!c1 || !c2)
return false; // We have reached the end of one string
// This needs to be upper case to match the behavior of the operating system,
// see http://archives.miloush.net/michkap/archive/2005/10/17/481600.html
const wchar_t u1 = to_upper_invariant(c1);
const wchar_t u2 = to_upper_invariant(c2);
if (u1 != u2)
return false; // strings are different
}
}
}
typedef bool (*Ptr_equal_string_ordinal_ic)(const wchar_t*, const wchar_t*);
Ptr_equal_string_ordinal_ic equal_string_ordinal_ic =
rtl_equal_unicode_string_api ? equal_string_ordinal_ic_1 : equal_string_ordinal_ic_2;
perms make_permissions(const path& p, DWORD attr)
{
perms prms = fs::owner_read | fs::group_read | fs::others_read;
if ((attr & FILE_ATTRIBUTE_READONLY) == 0)
prms |= fs::owner_write | fs::group_write | fs::others_write;
if (BOOST_FILESYSTEM_STRICMP(p.extension().string().c_str(), ".exe") == 0
|| BOOST_FILESYSTEM_STRICMP(p.extension().string().c_str(), ".com") == 0
|| BOOST_FILESYSTEM_STRICMP(p.extension().string().c_str(), ".bat") == 0
|| BOOST_FILESYSTEM_STRICMP(p.extension().string().c_str(), ".cmd") == 0)
path ext = p.extension();
if (equal_string_ordinal_ic(ext.c_str(), L".exe")
|| equal_string_ordinal_ic(ext.c_str(), L".com")
|| equal_string_ordinal_ic(ext.c_str(), L".bat")
|| equal_string_ordinal_ic(ext.c_str(), L".cmd"))
prms |= fs::owner_exe | fs::group_exe | fs::others_exe;
return prms;
}
@ -696,7 +787,7 @@ namespace
PtrCreateHardLinkW create_hard_link_api = PtrCreateHardLinkW(
::GetProcAddress(
::GetModuleHandle(TEXT("kernel32.dll")), "CreateHardLinkW"));
::GetModuleHandleW(L"kernel32.dll"), "CreateHardLinkW"));
typedef BOOLEAN (WINAPI *PtrCreateSymbolicLinkW)(
/*__in*/ LPCWSTR lpSymlinkFileName,
@ -706,7 +797,7 @@ namespace
PtrCreateSymbolicLinkW create_symbolic_link_api = PtrCreateSymbolicLinkW(
::GetProcAddress(
::GetModuleHandle(TEXT("kernel32.dll")), "CreateSymbolicLinkW"));
::GetModuleHandleW(L"kernel32.dll"), "CreateSymbolicLinkW"));
#endif