// Copyright 2019 The Crashpad Authors // // 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 "util/stream/log_output_stream.h" #include #include #include "base/check.h" #include "base/logging.h" namespace crashpad { LogOutputStream::LogOutputStream(std::unique_ptr delegate) : delegate_(std::move(delegate)), output_count_(0), flush_needed_(false), flushed_(false) { buffer_.reserve(delegate_->LineWidth()); } LogOutputStream::~LogOutputStream() { DCHECK(!flush_needed_); } bool LogOutputStream::Write(const uint8_t* data, size_t size) { DCHECK(!flushed_); static constexpr char kBeginMessage[] = "-----BEGIN CRASHPAD MINIDUMP-----"; if (output_count_ == 0 && WriteToLog(kBeginMessage) < 0) { return false; } flush_needed_ = true; while (size > 0) { size_t m = std::min(delegate_->LineWidth() - buffer_.size(), size); buffer_.append(reinterpret_cast(data), m); data += m; size -= m; if (buffer_.size() == delegate_->LineWidth() && !WriteBuffer()) { return false; } } return true; } bool LogOutputStream::WriteBuffer() { if (buffer_.empty()) return true; static constexpr char kAbortMessage[] = "-----ABORT CRASHPAD MINIDUMP-----"; output_count_ += buffer_.size(); if (output_count_ > delegate_->OutputCap()) { WriteToLog(kAbortMessage); flush_needed_ = false; return false; } int result = WriteToLog(buffer_.c_str()); if (result < 0) { if (result == -EAGAIN) { WriteToLog(kAbortMessage); } flush_needed_ = false; return false; } buffer_.clear(); return true; } int LogOutputStream::WriteToLog(const char* buf) { return delegate_->Log(buf); } bool LogOutputStream::Flush() { bool result = true; if (flush_needed_) { flush_needed_ = false; flushed_ = true; static constexpr char kEndMessage[] = "-----END CRASHPAD MINIDUMP-----"; if (!WriteBuffer() || WriteToLog(kEndMessage) < 0) { result = false; } } return result; } } // namespace crashpad