|  | /* | 
|  | * Copyright (C) 2015 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. | 
|  | */ | 
|  |  | 
|  | #pragma once | 
|  |  | 
|  | #include <condition_variable> | 
|  | #include <mutex> | 
|  | #include <string> | 
|  | #include <string_view> | 
|  | #include <type_traits> | 
|  | #include <vector> | 
|  |  | 
|  | #include <android-base/macros.h> | 
|  |  | 
|  | #include "adb.h" | 
|  | #include "adb_unique_fd.h" | 
|  |  | 
|  | void close_stdin(); | 
|  |  | 
|  | bool getcwd(std::string* cwd); | 
|  | bool directory_exists(const std::string& path); | 
|  |  | 
|  | // Return the user's home directory. | 
|  | std::string adb_get_homedir_path(); | 
|  |  | 
|  | // Return the adb user directory. | 
|  | std::string adb_get_android_dir_path(); | 
|  |  | 
|  | bool mkdirs(const std::string& path); | 
|  |  | 
|  | std::string escape_arg(const std::string& s); | 
|  |  | 
|  | std::string dump_hex(const void* ptr, size_t byte_count); | 
|  | std::string dump_header(const amessage* msg); | 
|  | std::string dump_packet(const char* name, const char* func, const apacket* p); | 
|  |  | 
|  | std::string perror_str(const char* msg); | 
|  |  | 
|  | [[noreturn]] void error_exit(const char* fmt, ...) __attribute__((__format__(__printf__, 1, 2))); | 
|  | [[noreturn]] void perror_exit(const char* fmt, ...) __attribute__((__format__(__printf__, 1, 2))); | 
|  |  | 
|  | bool set_file_block_mode(borrowed_fd fd, bool block); | 
|  |  | 
|  | // Given forward/reverse targets, returns true if they look sane. If an error is found, fills | 
|  | // |error| and returns false. | 
|  | // Currently this only checks "tcp:" targets. Additional checking could be added for other targets | 
|  | // if needed. | 
|  | bool forward_targets_are_valid(const std::string& source, const std::string& dest, | 
|  | std::string* error); | 
|  |  | 
|  | // A thread-safe blocking queue. | 
|  | template <typename T> | 
|  | class BlockingQueue { | 
|  | std::mutex mutex; | 
|  | std::condition_variable cv; | 
|  | std::vector<T> queue; | 
|  |  | 
|  | public: | 
|  | void Push(const T& t) { | 
|  | { | 
|  | std::unique_lock<std::mutex> lock(mutex); | 
|  | queue.push_back(t); | 
|  | } | 
|  | cv.notify_one(); | 
|  | } | 
|  |  | 
|  | template <typename Fn> | 
|  | void PopAll(Fn fn) { | 
|  | std::vector<T> popped; | 
|  |  | 
|  | { | 
|  | std::unique_lock<std::mutex> lock(mutex); | 
|  | cv.wait(lock, [this]() { return !queue.empty(); }); | 
|  | popped = std::move(queue); | 
|  | queue.clear(); | 
|  | } | 
|  |  | 
|  | for (const T& t : popped) { | 
|  | fn(t); | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | std::string GetLogFilePath(); | 
|  |  | 
|  | inline std::string_view StripTrailingNulls(std::string_view str) { | 
|  | size_t n = 0; | 
|  | for (auto it = str.rbegin(); it != str.rend(); ++it) { | 
|  | if (*it != '\0') { | 
|  | break; | 
|  | } | 
|  | ++n; | 
|  | } | 
|  |  | 
|  | str.remove_suffix(n); | 
|  | return str; | 
|  | } | 
|  |  | 
|  | // Base-10 stroll on a string_view. | 
|  | template <typename T> | 
|  | inline bool ParseUint(T* result, std::string_view str, std::string_view* remaining = nullptr) { | 
|  | if (str.empty() || !isdigit(str[0])) { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | T value = 0; | 
|  | std::string_view::iterator it; | 
|  | constexpr T max = std::numeric_limits<T>::max(); | 
|  | for (it = str.begin(); it != str.end() && isdigit(*it); ++it) { | 
|  | if (value > max / 10) { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | value *= 10; | 
|  |  | 
|  | T digit = *it - '0'; | 
|  | if (value > max - digit) { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | value += digit; | 
|  | } | 
|  | *result = value; | 
|  | if (remaining) { | 
|  | *remaining = str.substr(it - str.begin()); | 
|  | } else { | 
|  | return it == str.end(); | 
|  | } | 
|  |  | 
|  | return true; | 
|  | } |