blob: 1d1eed8e4fe90b34c23b567abe735a4fbd66f0a5 [file] [log] [blame]
/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
file Copyright.txt or https://cmake.org/licensing#kwsys for details. */
#ifdef __osf__
# define _OSF_SOURCE
# define _POSIX_C_SOURCE 199506L
# define _XOPEN_SOURCE_EXTENDED
#endif
#include "kwsysPrivate.h"
#include KWSYS_HEADER(Encoding.hxx)
#include KWSYS_HEADER(Encoding.h)
#include KWSYS_HEADER(String.h)
// Work-around CMake dependency scanning limitation. This must
// duplicate the above list of headers.
#if 0
# include "Encoding.h.in"
# include "Encoding.hxx.in"
# include "String.h"
#endif
#include <algorithm>
#include <cstdlib>
#include <cstring>
#include <vector>
#ifdef _MSC_VER
# pragma warning(disable : 4786)
#endif
// Windows API.
#if defined(_WIN32)
# include <windows.h>
# include <shellapi.h>
#endif
namespace KWSYS_NAMESPACE {
Encoding::CommandLineArguments Encoding::CommandLineArguments::Main(
int argc, char const* const* argv)
{
#ifdef _WIN32
(void)argc;
(void)argv;
int ac;
LPWSTR* w_av = CommandLineToArgvW(GetCommandLineW(), &ac);
std::vector<std::string> av1(ac);
std::vector<char const*> av2(ac);
for (int i = 0; i < ac; i++) {
av1[i] = ToNarrow(w_av[i]);
av2[i] = av1[i].c_str();
}
LocalFree(w_av);
return CommandLineArguments(ac, &av2[0]);
#else
return CommandLineArguments(argc, argv);
#endif
}
Encoding::CommandLineArguments::CommandLineArguments(int ac,
char const* const* av)
{
this->argv_.resize(ac + 1);
for (int i = 0; i < ac; i++) {
this->argv_[i] = strdup(av[i]);
}
this->argv_[ac] = nullptr;
}
Encoding::CommandLineArguments::CommandLineArguments(int ac,
wchar_t const* const* av)
{
this->argv_.resize(ac + 1);
for (int i = 0; i < ac; i++) {
this->argv_[i] = kwsysEncoding_DupToNarrow(av[i]);
}
this->argv_[ac] = nullptr;
}
Encoding::CommandLineArguments::~CommandLineArguments()
{
for (size_t i = 0; i < this->argv_.size(); i++) {
free(argv_[i]);
}
}
Encoding::CommandLineArguments::CommandLineArguments(
CommandLineArguments const& other)
{
this->argv_.resize(other.argv_.size());
for (size_t i = 0; i < this->argv_.size(); i++) {
this->argv_[i] = other.argv_[i] ? strdup(other.argv_[i]) : nullptr;
}
}
Encoding::CommandLineArguments& Encoding::CommandLineArguments::operator=(
CommandLineArguments const& other)
{
if (this != &other) {
size_t i;
for (i = 0; i < this->argv_.size(); i++) {
free(this->argv_[i]);
}
this->argv_.resize(other.argv_.size());
for (i = 0; i < this->argv_.size(); i++) {
this->argv_[i] = other.argv_[i] ? strdup(other.argv_[i]) : nullptr;
}
}
return *this;
}
int Encoding::CommandLineArguments::argc() const
{
return static_cast<int>(this->argv_.size() - 1);
}
char const* const* Encoding::CommandLineArguments::argv() const
{
return &this->argv_[0];
}
std::wstring Encoding::ToWide(std::string const& str)
{
std::wstring wstr;
#if defined(_WIN32)
int const wlength =
MultiByteToWideChar(KWSYS_ENCODING_DEFAULT_CODEPAGE, 0, str.data(),
int(str.size()), nullptr, 0);
if (wlength > 0) {
wstr.resize(wlength);
int r = MultiByteToWideChar(KWSYS_ENCODING_DEFAULT_CODEPAGE, 0, str.data(),
int(str.size()), &wstr[0], wlength);
wstr.resize(static_cast<size_t>((std::max)(0, r)));
}
#else
size_t pos = 0;
size_t nullPos = 0;
do {
if (pos < str.size() && str.at(pos) != '\0') {
wstr += ToWide(str.c_str() + pos);
}
nullPos = str.find('\0', pos);
if (nullPos != std::string::npos) {
pos = nullPos + 1;
wstr += wchar_t('\0');
}
} while (nullPos != std::string::npos);
#endif
return wstr;
}
std::string Encoding::ToNarrow(std::wstring const& str)
{
std::string nstr;
#if defined(_WIN32)
int length =
WideCharToMultiByte(KWSYS_ENCODING_DEFAULT_CODEPAGE, 0, str.c_str(),
int(str.size()), nullptr, 0, nullptr, nullptr);
if (length > 0) {
nstr.resize(length);
int r =
WideCharToMultiByte(KWSYS_ENCODING_DEFAULT_CODEPAGE, 0, str.c_str(),
int(str.size()), &nstr[0], length, nullptr, nullptr);
nstr.resize(static_cast<size_t>((std::max)(0, r)));
}
#else
size_t pos = 0;
size_t nullPos = 0;
do {
if (pos < str.size() && str.at(pos) != '\0') {
nstr += ToNarrow(str.c_str() + pos);
}
nullPos = str.find(wchar_t('\0'), pos);
if (nullPos != std::string::npos) {
pos = nullPos + 1;
nstr += '\0';
}
} while (nullPos != std::string::npos);
#endif
return nstr;
}
std::wstring Encoding::ToWide(char const* cstr)
{
std::wstring wstr;
size_t length = kwsysEncoding_mbstowcs(nullptr, cstr, 0);
if (length == 0 || length == static_cast<size_t>(-1)) {
return wstr;
}
++length;
wstr.resize(length);
length = kwsysEncoding_mbstowcs(&wstr[0], cstr, length);
wstr.resize(length);
return wstr;
}
std::string Encoding::ToNarrow(wchar_t const* wcstr)
{
std::string str;
size_t length = kwsysEncoding_wcstombs(nullptr, wcstr, 0);
if (length == 0 || length == static_cast<size_t>(-1)) {
return str;
}
++length;
str.resize(length);
length = kwsysEncoding_wcstombs(&str[0], wcstr, length);
str.resize(length);
return str;
}
#if defined(_WIN32)
// Convert local paths to UNC style paths:
// * https://web.archive.org/web/20250329050419/
// https://cpp.arsenmk.com/2015/12/handling-long-paths-on-windows.html
namespace {
// UNC prefix for local paths.
std::wstring const unc_local_prefix = L"\\\\?\\";
// UNC prefix for remote paths.
std::wstring const unc_remote_prefix = L"\\\\?\\UNC\\";
// The longest UNC prefix replaces two leading backslashes.
size_t const unc_max_space = unc_remote_prefix.length() - 2;
}
std::wstring Encoding::ToWindowsExtendedPath(std::string const& source)
{
return ToWindowsExtendedPath(ToWide(source));
}
std::wstring Encoding::ToWindowsExtendedPath(char const* source)
{
return ToWindowsExtendedPath(ToWide(source));
}
std::wstring Encoding::ToWindowsExtendedPath(std::wstring const& wsource)
{
// Resolve any relative paths
std::wstring wfull;
DWORD wfull_len;
// Try with a stack-allocated buffer first.
wchar_t local_buf[512];
size_t const local_buf_len = sizeof(local_buf) / sizeof(local_buf[0]);
wfull_len =
GetFullPathNameW(wsource.c_str(), local_buf_len, local_buf, nullptr);
if (wfull_len <= local_buf_len) {
// The stack-allocated buffer was large enough. It holds the result.
// Reserve room for the prefix added below.
wfull.reserve(wfull_len + unc_max_space);
// Store the absolute path to be returned.
wfull.assign(local_buf, wfull_len);
} else {
// The stack-allocated buffer was too small, but now we know how
// much to allocate on the heap.
// Some versions of GetFullPathNameW return a slightly too-small size.
wfull_len += 3;
// Reserve room for the prefix added below.
wfull.reserve(wfull_len + unc_max_space);
// Try again with a heap-allocated buffer.
wfull.resize(wfull_len, L'\0');
wfull_len =
GetFullPathNameW(wsource.c_str(), wfull_len, &wfull[0], nullptr);
wfull.resize(wfull_len);
}
if (wfull_len >= 2 && kwsysString_isalpha(wfull[0]) &&
wfull[1] == L':') { /* C:\Foo\bar\FooBar.txt */
wfull.insert(0, unc_local_prefix);
return wfull;
} else if (wfull_len >= 2 && wfull[0] == L'\\' &&
wfull[1] == L'\\') { /* Starts with \\ */
if (wfull_len >= 4 && wfull[2] == L'?' &&
wfull[3] == L'\\') { /* Starts with \\?\ */
if (wfull_len >= 8 && wfull[4] == L'U' && wfull[5] == L'N' &&
wfull[6] == L'C' &&
wfull[7] == L'\\') { /* \\?\UNC\Foo\bar\FooBar.txt */
return wfull;
} else if (wfull_len >= 6 && kwsysString_isalpha(wfull[4]) &&
wfull[5] == L':') { /* \\?\C:\Foo\bar\FooBar.txt */
return wfull;
} else if (wfull_len >= 5) { /* \\?\Foo\bar\FooBar.txt */
wfull.replace(0, 4, unc_remote_prefix);
return wfull;
}
} else if (wfull_len >= 4 && wfull[2] == L'.' &&
wfull[3] == L'\\') { /* Starts with \\.\ a device name */
if (wfull_len >= 6 && kwsysString_isalpha(wfull[4]) &&
wfull[5] == L':') { /* \\.\C:\Foo\bar\FooBar.txt */
wfull.replace(0, 4, unc_local_prefix);
return wfull;
} else if (wfull_len >=
5) { /* \\.\Foo\bar\ Device name is left unchanged */
return wfull;
}
} else if (wfull_len >= 3) { /* \\Foo\bar\FooBar.txt */
wfull.replace(0, 2, unc_remote_prefix);
return wfull;
}
}
// If this case has been reached, then the path is invalid. Leave it
// unchanged
return wsource;
}
#endif
} // namespace KWSYS_NAMESPACE