|  | /* | 
|  | * 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. | 
|  | */ | 
|  |  | 
|  | #include "android-base/logging.h" | 
|  | #include "android-base/test_utils.h" | 
|  |  | 
|  | #include <fcntl.h> | 
|  | #include <stdio.h> | 
|  | #include <stdlib.h> | 
|  | #include <sys/stat.h> | 
|  | #include <unistd.h> | 
|  |  | 
|  | #if defined(_WIN32) | 
|  | #include <windows.h> | 
|  | #include <direct.h> | 
|  | #define OS_PATH_SEPARATOR '\\' | 
|  | #else | 
|  | #define OS_PATH_SEPARATOR '/' | 
|  | #endif | 
|  |  | 
|  | #include <string> | 
|  |  | 
|  | #ifdef _WIN32 | 
|  | int mkstemp(char* template_name) { | 
|  | if (_mktemp(template_name) == nullptr) { | 
|  | return -1; | 
|  | } | 
|  | // Use open() to match the close() that TemporaryFile's destructor does. | 
|  | // Use O_BINARY to match base file APIs. | 
|  | return open(template_name, O_CREAT | O_EXCL | O_RDWR | O_BINARY, | 
|  | S_IRUSR | S_IWUSR); | 
|  | } | 
|  |  | 
|  | char* mkdtemp(char* template_name) { | 
|  | if (_mktemp(template_name) == nullptr) { | 
|  | return nullptr; | 
|  | } | 
|  | if (_mkdir(template_name) == -1) { | 
|  | return nullptr; | 
|  | } | 
|  | return template_name; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | static std::string GetSystemTempDir() { | 
|  | #if defined(__ANDROID__) | 
|  | const char* tmpdir = "/data/local/tmp"; | 
|  | if (access(tmpdir, R_OK | W_OK | X_OK) == 0) { | 
|  | return tmpdir; | 
|  | } | 
|  | // Tests running in app context can't access /data/local/tmp, | 
|  | // so try current directory if /data/local/tmp is not accessible. | 
|  | return "."; | 
|  | #elif defined(_WIN32) | 
|  | char tmp_dir[MAX_PATH]; | 
|  | DWORD result = GetTempPathA(sizeof(tmp_dir), tmp_dir); | 
|  | CHECK_NE(result, 0ul) << "GetTempPathA failed, error: " << GetLastError(); | 
|  | CHECK_LT(result, sizeof(tmp_dir)) << "path truncated to: " << result; | 
|  |  | 
|  | // GetTempPath() returns a path with a trailing slash, but init() | 
|  | // does not expect that, so remove it. | 
|  | CHECK_EQ(tmp_dir[result - 1], '\\'); | 
|  | tmp_dir[result - 1] = '\0'; | 
|  | return tmp_dir; | 
|  | #else | 
|  | return "/tmp"; | 
|  | #endif | 
|  | } | 
|  |  | 
|  | TemporaryFile::TemporaryFile() { | 
|  | init(GetSystemTempDir()); | 
|  | } | 
|  |  | 
|  | TemporaryFile::~TemporaryFile() { | 
|  | if (fd != -1) { | 
|  | close(fd); | 
|  | } | 
|  | unlink(path); | 
|  | } | 
|  |  | 
|  | int TemporaryFile::release() { | 
|  | int result = fd; | 
|  | fd = -1; | 
|  | return result; | 
|  | } | 
|  |  | 
|  | void TemporaryFile::init(const std::string& tmp_dir) { | 
|  | snprintf(path, sizeof(path), "%s%cTemporaryFile-XXXXXX", tmp_dir.c_str(), | 
|  | OS_PATH_SEPARATOR); | 
|  | fd = mkstemp(path); | 
|  | } | 
|  |  | 
|  | TemporaryDir::TemporaryDir() { | 
|  | init(GetSystemTempDir()); | 
|  | } | 
|  |  | 
|  | TemporaryDir::~TemporaryDir() { | 
|  | rmdir(path); | 
|  | } | 
|  |  | 
|  | bool TemporaryDir::init(const std::string& tmp_dir) { | 
|  | snprintf(path, sizeof(path), "%s%cTemporaryDir-XXXXXX", tmp_dir.c_str(), | 
|  | OS_PATH_SEPARATOR); | 
|  | return (mkdtemp(path) != nullptr); | 
|  | } | 
|  |  | 
|  | CapturedStderr::CapturedStderr() : old_stderr_(-1) { | 
|  | init(); | 
|  | } | 
|  |  | 
|  | CapturedStderr::~CapturedStderr() { | 
|  | reset(); | 
|  | } | 
|  |  | 
|  | int CapturedStderr::fd() const { | 
|  | return temp_file_.fd; | 
|  | } | 
|  |  | 
|  | void CapturedStderr::init() { | 
|  | #if defined(_WIN32) | 
|  | // On Windows, stderr is often buffered, so make sure it is unbuffered so | 
|  | // that we can immediately read back what was written to stderr. | 
|  | CHECK_EQ(0, setvbuf(stderr, NULL, _IONBF, 0)); | 
|  | #endif | 
|  | old_stderr_ = dup(STDERR_FILENO); | 
|  | CHECK_NE(-1, old_stderr_); | 
|  | CHECK_NE(-1, dup2(fd(), STDERR_FILENO)); | 
|  | } | 
|  |  | 
|  | void CapturedStderr::reset() { | 
|  | CHECK_NE(-1, dup2(old_stderr_, STDERR_FILENO)); | 
|  | CHECK_EQ(0, close(old_stderr_)); | 
|  | // Note: cannot restore prior setvbuf() setting. | 
|  | } |