| /* |
| * 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; |
| } |
| } |