| // Copyright 2017 The Bazel Authors. All rights reserved. |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| #ifndef BAZEL_SRC_MAIN_NATIVE_WINDOWS_FILE_H_ |
| #define BAZEL_SRC_MAIN_NATIVE_WINDOWS_FILE_H_ |
| |
| #ifndef WIN32_LEAN_AND_MEAN |
| #define WIN32_LEAN_AND_MEAN |
| #endif |
| |
| #include <windows.h> |
| |
| #ifndef SYMBOLIC_LINK_FLAG_ALLOW_UNPRIVILEGED_CREATE |
| #define SYMBOLIC_LINK_FLAG_ALLOW_UNPRIVILEGED_CREATE 0x2 |
| #endif |
| |
| #include <memory> |
| #include <string> |
| |
| namespace bazel { |
| namespace windows { |
| |
| using std::unique_ptr; |
| using std::wstring; |
| |
| bool IsDeveloperModeEnabled(); |
| |
| DWORD DetermineSymlinkPrivilegeFlag(); |
| |
| // The flag SYMBOLIC_LINK_FLAG_ALLOW_UNPRIVILEGED_CREATE requires |
| // developer mode to be enabled. If it is not enabled, do not use the |
| // flag. The process will need to be run with elevated privileges. |
| const DWORD symlinkPrivilegeFlag = DetermineSymlinkPrivilegeFlag(); |
| |
| template <typename char_type> |
| bool HasUncPrefix(const char_type* path) { |
| // Return true iff `path` starts with "\\?\", "\\.\", or "\??\". |
| return path[0] == '\\' && |
| ((path[1] == '\\' && (path[2] == '?' || path[2] == '.')) || |
| (path[1] == '?' && path[2] == '?')) && |
| path[3] == '\\'; |
| } |
| |
| template <typename char_type> |
| bool IsDevNull(const char_type* path) { |
| return (path[0] == 'N' || path[0] == 'n') && |
| (path[1] == 'U' || path[1] == 'u') && |
| (path[2] == 'L' || path[2] == 'l'); |
| } |
| |
| template <typename char_type> |
| bool HasDriveSpecifierPrefix(const char_type* p) { |
| if (HasUncPrefix(p)) { |
| return iswalpha(p[4]) && p[5] == ':' && (p[6] == '\\' || p[6] == '/'); |
| } else { |
| return iswalpha(p[0]) && p[1] == ':' && (p[2] == '\\' || p[2] == '/'); |
| } |
| } |
| |
| std::wstring AddUncPrefixMaybe(const std::wstring& path); |
| |
| std::wstring RemoveUncPrefixMaybe(const std::wstring& path); |
| |
| bool IsAbsoluteNormalizedWindowsPath(const std::wstring& p); |
| |
| // Keep in sync with j.c.g.devtools.build.lib.windows.WindowsFileOperations |
| struct IsSymlinkOrJunctionResult { |
| enum { |
| kSuccess = 0, |
| kError = 1, |
| kDoesNotExist = 2, |
| }; |
| }; |
| |
| // Keep in sync with j.c.g.devtools.build.lib.windows.WindowsFileOperations |
| struct GetChangeTimeResult { |
| enum { |
| kSuccess = 0, |
| kError = 1, |
| kDoesNotExist = 2, |
| kAccessDenied = 3, |
| }; |
| }; |
| |
| // Keep in sync with j.c.g.devtools.build.lib.windows.WindowsFileOperations |
| struct DeletePathResult { |
| enum { |
| kSuccess = 0, |
| kError = 1, |
| kDoesNotExist = 2, |
| kDirectoryNotEmpty = 3, |
| kAccessDenied = 4, |
| }; |
| }; |
| |
| // Keep in sync with j.c.g.devtools.build.lib.windows.WindowsFileOperations |
| struct CreateJunctionResult { |
| enum { |
| kSuccess = 0, |
| kError = 1, |
| kTargetNameTooLong = 2, |
| kAlreadyExistsWithDifferentTarget = 3, |
| kAlreadyExistsButNotJunction = 4, |
| kAccessDenied = 5, |
| kDisappeared = 6, |
| }; |
| }; |
| |
| // Keep in sync with j.c.g.devtools.build.lib.windows.WindowsFileOperations |
| struct CreateSymlinkResult { |
| enum { |
| kSuccess = 0, |
| kError = 1, |
| kTargetIsDirectory = 2, |
| }; |
| }; |
| |
| // Keep in sync with j.c.g.devtools.build.lib.windows.WindowsFileOperations |
| struct ReadSymlinkOrJunctionResult { |
| enum { |
| kSuccess = 0, |
| kError = 1, |
| kAccessDenied = 2, |
| kDoesNotExist = 3, |
| kNotALink = 4, |
| kUnknownLinkType = 5, |
| }; |
| }; |
| |
| // Determines whether `path` is a junction (or directory symlink). |
| // |
| // `path` should be an absolute, normalized, Windows-style path, with "\\?\" |
| // prefix if it's longer than MAX_PATH. |
| // |
| // To read about differences between junctions and directory symlinks, |
| // see http://superuser.com/a/343079. In Bazel we only ever create junctions. |
| int IsSymlinkOrJunction(const WCHAR* path, bool* result, wstring* error); |
| |
| // Retrieves the FILETIME at which `path` was last changed, including metadata. |
| // |
| // `path` should be an absolute, normalized, Windows-style path, with "\\?\" |
| // prefix if it's longer than MAX_PATH. |
| int GetChangeTime(const WCHAR* path, bool follow_reparse_points, |
| int64_t* result, wstring* error); |
| |
| // Computes the long version of `path` if it has any 8dot3 style components. |
| // Returns the empty string upon success, or a human-readable error message upon |
| // failure. |
| // `path` must be an absolute, normalized, Windows style path, with a "\\?\" |
| // prefix if it's longer than MAX_PATH. The result will have a "\\?\" prefix if |
| // and only if `path` had one as well. (It's the caller's responsibility to keep |
| // or remove this prefix.) |
| // TODO(laszlocsomor): update GetLongPath so it succeeds even if the path does |
| // not (fully) exist. |
| wstring GetLongPath(const WCHAR* path, unique_ptr<WCHAR[]>* result); |
| |
| // Creates a junction at `name`, pointing to `target`. |
| // Returns CreateJunctionResult::kSuccess if it could create the junction, or if |
| // the junction already exists with the same target. |
| // If the junction's name already exists as an empty directory, this function |
| // will turn it into a junction and return kSuccess. |
| // Otherwise returns one of the other CreateJunctionResult::k* constants for |
| // known error cases, or CreateJunctionResult::kError for unknown error cases. |
| // When the function returns CreateJunctionResult::kError, and `error` is |
| // non-null, the function writes an error message into `error`. If the return |
| // value is anything other than CreateJunctionResult::kError, then this function |
| // ignores the `error` argument. |
| // |
| // Neither `junction_name` nor `junction_target` needs to have a "\\?\" prefix, |
| // not even if they are longer than MAX_PATH, though it's okay if they do. This |
| // function will add the right prefixes as necessary. |
| int CreateJunction(const wstring& junction_name, const wstring& junction_target, |
| wstring* error); |
| |
| // Creates a symlink at `symlink_name`, pointing to `symlink_target`. |
| // Returns CreateSymlinkResult::kSuccess if could create the symlink. |
| // If the target is a directory, this function will return |
| // CreateSymlinkResult::kTargetIsDirectory since a junction is preferred |
| // instead. When the function returns CreateSymlinkResult::kError and `error` |
| // is non-null then `error` receives an error message. |
| int CreateSymlink(const wstring& symlink_name, const wstring& symlink_target, |
| wstring* error); |
| |
| // Reads the symlink or junction into 'result'. |
| // Returns a value from 'ReadSymlinkOrJunctionResult'. |
| // When the method returns 'ReadSymlinkOrJunctionResult::kError' and 'error' is |
| // non-null then 'error' receives an error message. |
| int ReadSymlinkOrJunction(const wstring& path, wstring* result, wstring* error); |
| |
| // Deletes the file, junction, or empty directory at `path`. |
| // Returns DELETE_PATH_SUCCESS if it successfully deleted the path, otherwise |
| // returns one of the other DELETE_PATH_* constants (e.g. when the directory is |
| // not empty or the file is in use by another process). |
| // Returns DELETE_PATH_ERROR for unexpected errors. If `error` is not null, the |
| // function writes an error message into it. |
| int DeletePath(const wstring& path, wstring* error); |
| |
| // Returns a normalized form of the input `path`. |
| // |
| // Normalization: |
| // Normalization means removing "." references, resolving ".." references, |
| // and deduplicating "/" characters while converting them to "\\". For |
| // example if `path` is "foo/../bar/.//qux", the result is "bar\\qux". |
| // |
| // Uplevel references ("..") that cannot go any higher in the directory tree |
| // are preserved if the path is relative, and ignored if the path is |
| // absolute, e.g. "../../foo" is normalized to "..\\..\\foo" but "c:/.." is |
| // normalized to "c:\\". |
| // |
| // This method does not check the semantics of the `path` beyond checking if |
| // it starts with a directory separator. Illegal paths such as one with a |
| // drive specifier in the middle (e.g. "foo/c:/bar") are accepted -- it's the |
| // caller's responsibility to pass a path that, when normalized, will be |
| // semantically correct. |
| // |
| // Current directory references (".") are preserved if and only if they are |
| // the only path segment, so "./" becomes "." but "./foo" becomes "foo". |
| // |
| // Arguments: |
| // `path` must be a relative or absolute Windows path, it may use "/" instead |
| // of "\\". The path should not start with "/" or "\\". |
| // |
| // Result: |
| // Returns false if and only if the path starts with a directory separator. |
| // |
| // The result won't have a UNC prefix, even if `path` did. The result won't |
| // have a trailing "\\" except when and only when the path is normalized to |
| // just a drive specifier (e.g. when `path` is "c:/" or "c:/foo/.."). The |
| // result will preserve the casing of the input, so "D:/Bar" becomes |
| // "D:\\Bar". |
| std::string Normalize(const std::string& p); |
| std::wstring Normalize(const std::wstring& p); |
| |
| bool GetCwd(std::wstring* result, DWORD* err_code); |
| |
| } // namespace windows |
| } // namespace bazel |
| |
| #endif // BAZEL_SRC_MAIN_NATIVE_WINDOWS_FILE_H_ |