| /* | 
 |  * Copyright (C) 2023 The Android Open Source Project | 
 |  * | 
 |  * 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. | 
 |  */ | 
 |  | 
 | #ifdef _WIN32 | 
 | #include <android-base/utf8.h> | 
 | #include <direct.h> | 
 | #include <shlobj.h> | 
 | #else | 
 | #include <pwd.h> | 
 | #endif | 
 |  | 
 | #include <android-base/logging.h> | 
 | #include <android-base/parseint.h> | 
 | #include <sys/file.h> | 
 | #include <sys/stat.h> | 
 | #include <unistd.h> | 
 |  | 
 | #include <vector> | 
 |  | 
 | #include "filesystem.h" | 
 |  | 
 | namespace { | 
 |  | 
 | int LockFile(int fd) { | 
 | #ifdef _WIN32 | 
 |     HANDLE handle = reinterpret_cast<HANDLE>(_get_osfhandle(fd)); | 
 |     OVERLAPPED overlapped = {}; | 
 |     const BOOL locked = | 
 |             LockFileEx(handle, LOCKFILE_EXCLUSIVE_LOCK, 0, MAXDWORD, MAXDWORD, &overlapped); | 
 |     return locked ? 0 : -1; | 
 | #else | 
 |     return flock(fd, LOCK_EX); | 
 | #endif | 
 | } | 
 |  | 
 | }  // namespace | 
 |  | 
 | // inspired by adb implementation: | 
 | // cs.android.com/android/platform/superproject/+/master:packages/modules/adb/adb_utils.cpp;l=275 | 
 | std::string GetHomeDirPath() { | 
 | #ifdef _WIN32 | 
 |     WCHAR path[MAX_PATH]; | 
 |     const HRESULT hr = SHGetFolderPathW(NULL, CSIDL_PROFILE, NULL, 0, path); | 
 |     if (FAILED(hr)) { | 
 |         return {}; | 
 |     } | 
 |     std::string home_str; | 
 |     if (!android::base::WideToUTF8(path, &home_str)) { | 
 |         return {}; | 
 |     } | 
 |     return home_str; | 
 | #else | 
 |     if (const char* const home = getenv("HOME")) { | 
 |         return home; | 
 |     } | 
 |  | 
 |     struct passwd pwent; | 
 |     struct passwd* result; | 
 |     int pwent_max = sysconf(_SC_GETPW_R_SIZE_MAX); | 
 |     if (pwent_max == -1) { | 
 |         pwent_max = 16384; | 
 |     } | 
 |     std::vector<char> buf(pwent_max); | 
 |     int rc = getpwuid_r(getuid(), &pwent, buf.data(), buf.size(), &result); | 
 |     if (rc == 0 && result) { | 
 |         return result->pw_dir; | 
 |     } | 
 | #endif | 
 |  | 
 |     return {}; | 
 | } | 
 |  | 
 | bool FileExists(const std::string& path) { | 
 |     return access(path.c_str(), F_OK) == 0; | 
 | } | 
 |  | 
 | bool EnsureDirectoryExists(const std::string& directory_path) { | 
 |     const int result = | 
 | #ifdef _WIN32 | 
 |             _mkdir(directory_path.c_str()); | 
 | #else | 
 |             mkdir(directory_path.c_str(), S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH); | 
 | #endif | 
 |  | 
 |     return result == 0 || errno == EEXIST; | 
 | } | 
 |  | 
 | FileLock::FileLock(const std::string& path) : fd_(open(path.c_str(), O_CREAT | O_WRONLY, 0644)) { | 
 |     if (LockFile(fd_.get()) != 0) { | 
 |         LOG(FATAL) << "Failed to acquire a lock on " << path; | 
 |     } | 
 | } |