/* * 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/test_utils.h" #include <fcntl.h> #include <stdio.h> #include <stdlib.h> #include <sys/stat.h> #include <unistd.h> #include <string> #include <android-base/file.h> #include <android-base/logging.h> CapturedStdFd::CapturedStdFd(int std_fd) : std_fd_(std_fd), old_fd_(-1) { Start(); } CapturedStdFd::~CapturedStdFd() { if (old_fd_ != -1) { Stop(); } } int CapturedStdFd::fd() const { return temp_file_.fd; } std::string CapturedStdFd::str() { std::string result; CHECK_EQ(0, TEMP_FAILURE_RETRY(lseek(fd(), 0, SEEK_SET))); android::base::ReadFdToString(fd(), &result); return result; } void CapturedStdFd::Reset() { // Do not reset while capturing. CHECK_EQ(-1, old_fd_); CHECK_EQ(0, TEMP_FAILURE_RETRY(lseek(fd(), 0, SEEK_SET))); CHECK_EQ(0, ftruncate(fd(), 0)); } void CapturedStdFd::Start() { #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. if (std_fd_ == STDERR_FILENO) CHECK_EQ(0, setvbuf(stderr, nullptr, _IONBF, 0)); #endif old_fd_ = dup(std_fd_); CHECK_NE(-1, old_fd_); CHECK_NE(-1, dup2(fd(), std_fd_)); } void CapturedStdFd::Stop() { CHECK_NE(-1, old_fd_); CHECK_NE(-1, dup2(old_fd_, std_fd_)); close(old_fd_); old_fd_ = -1; // Note: cannot restore prior setvbuf() setting. }