From c30a5549871727668a97080cdb886800cb5e0023 Mon Sep 17 00:00:00 2001 From: Ashish Karale Date: Mon, 14 Oct 2024 08:49:09 -0700 Subject: [PATCH 1/3] Support long paths on windows --- src/filesystem/implementations/local.h | 42 +++++++++++++++++++++++--- 1 file changed, 37 insertions(+), 5 deletions(-) diff --git a/src/filesystem/implementations/local.h b/src/filesystem/implementations/local.h index 8e60054a6..4fa6a3782 100644 --- a/src/filesystem/implementations/local.h +++ b/src/filesystem/implementations/local.h @@ -31,7 +31,7 @@ #include #include "../../constants.h" -#include "common.h" +#include "common.h"W namespace triton { namespace core { @@ -60,12 +60,44 @@ class LocalFileSystem : public FileSystem { Status MakeTemporaryDirectory( std::string dir_path, std::string* temp_dir) override; Status DeletePath(const std::string& path) override; +private: + inline std::string getOSValidPath(const std::string& _path); + const char* kWindowsLongPathPrefix = "\\\\?\\"; }; + +//! Converts incoming utf-8 path to an OS valid path +//! +//! On Linux there is not much to do but make sure correct slashes are used +//! On Windows we need to take care of the long paths and handle them correctly +//! to avoid legacy issues with MAX_PATH +//! +//! More details: +//! https://learn.microsoft.com/en-us/windows/win32/fileio/maximum-file-path-limitation?tabs=registry +//! +inline std::string +LocalFileSystem::getOSValidPath(const std::string& _path) +{ + std::string path(_path); +#ifdef _WIN32 + // On Windows long paths must be marked correctly otherwise, due to backwards + // compatibility, all paths are limited to MAX_PATH length + if (path.size() >= MAX_PATH) { + // Must be prefixed with "\\?\" to be considered long path + if (path.substr(0, 4) != (kWindowsLongPathPrefix)) { + // Long path but not "tagged" correctly + path = (kWindowsLongPathPrefix) + path; + } + } + std::replace(path.begin(), path.end(), '/', '\\'); +#endif + return path; +} + Status LocalFileSystem::FileExists(const std::string& path, bool* exists) { - *exists = (access(path.c_str(), F_OK) == 0); + *exists = (access(getOSValidPath(path).c_str(), F_OK) == 0); return Status::Success; } @@ -75,7 +107,7 @@ LocalFileSystem::IsDirectory(const std::string& path, bool* is_dir) *is_dir = false; struct stat st; - if (stat(path.c_str(), &st) != 0) { + if (stat(getOSValidPath(path).c_str(), &st) != 0) { return Status(Status::Code::INTERNAL, "failed to stat file " + path); } @@ -88,7 +120,7 @@ LocalFileSystem::FileModificationTime( const std::string& path, int64_t* mtime_ns) { struct stat st; - if (stat(path.c_str(), &st) != 0) { + if (stat(getOSValidPath(path).c_str(), &st) != 0) { return Status(Status::Code::INTERNAL, "failed to stat file " + path); } @@ -187,7 +219,7 @@ LocalFileSystem::GetDirectoryFiles( Status LocalFileSystem::ReadTextFile(const std::string& path, std::string* contents) { - std::ifstream in(path, std::ios::in | std::ios::binary); + std::ifstream in(getOSValidPath(path), std::ios::in | std::ios::binary); if (!in) { return Status( Status::Code::INTERNAL, From 1d57d3e21cb36bae3db80f5b24ae36fafb8925f3 Mon Sep 17 00:00:00 2001 From: Ashish Karale Date: Mon, 14 Oct 2024 08:55:52 -0700 Subject: [PATCH 2/3] Pre commit fixes --- src/filesystem/implementations/local.h | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/src/filesystem/implementations/local.h b/src/filesystem/implementations/local.h index 4fa6a3782..aa394f25d 100644 --- a/src/filesystem/implementations/local.h +++ b/src/filesystem/implementations/local.h @@ -31,7 +31,7 @@ #include #include "../../constants.h" -#include "common.h"W +#include "common.h" namespace triton { namespace core { @@ -60,7 +60,8 @@ class LocalFileSystem : public FileSystem { Status MakeTemporaryDirectory( std::string dir_path, std::string* temp_dir) override; Status DeletePath(const std::string& path) override; -private: + + private: inline std::string getOSValidPath(const std::string& _path); const char* kWindowsLongPathPrefix = "\\\\?\\"; }; From 8ff3bcaaf7132489e3851e826244d0518f1e3ac9 Mon Sep 17 00:00:00 2001 From: Ashish Karale Date: Wed, 30 Oct 2024 02:57:44 -0700 Subject: [PATCH 3/3] Addressing review comments --- src/filesystem/implementations/local.h | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/src/filesystem/implementations/local.h b/src/filesystem/implementations/local.h index aa394f25d..109a42c34 100644 --- a/src/filesystem/implementations/local.h +++ b/src/filesystem/implementations/local.h @@ -63,7 +63,7 @@ class LocalFileSystem : public FileSystem { private: inline std::string getOSValidPath(const std::string& _path); - const char* kWindowsLongPathPrefix = "\\\\?\\"; + const char* WindowsLongPathPrefix = "\\\\?\\"; }; @@ -77,22 +77,22 @@ class LocalFileSystem : public FileSystem { //! https://learn.microsoft.com/en-us/windows/win32/fileio/maximum-file-path-limitation?tabs=registry //! inline std::string -LocalFileSystem::getOSValidPath(const std::string& _path) +LocalFileSystem::getOSValidPath(const std::string& path) { - std::string path(_path); + std::string l_path(path); #ifdef _WIN32 // On Windows long paths must be marked correctly otherwise, due to backwards // compatibility, all paths are limited to MAX_PATH length - if (path.size() >= MAX_PATH) { + if (l_path.size() >= MAX_PATH) { // Must be prefixed with "\\?\" to be considered long path - if (path.substr(0, 4) != (kWindowsLongPathPrefix)) { + if (l_path.substr(0, 4) != (WindowsLongPathPrefix)) { // Long path but not "tagged" correctly - path = (kWindowsLongPathPrefix) + path; + l_path = (WindowsLongPathPrefix) + l_path; } } - std::replace(path.begin(), path.end(), '/', '\\'); + std::replace(l_path.begin(), l_path.end(), '/', '\\'); #endif - return path; + return l_path; } Status