// Copyright 2014 The Crashpad Authors. All rights reserved. // // 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 "minidump/minidump_simple_string_dictionary_writer.h" #include #include "base/logging.h" #include "base/memory/ptr_util.h" #include "base/stl_util.h" #include "util/file/file_writer.h" #include "util/numeric/safe_assignment.h" namespace crashpad { MinidumpSimpleStringDictionaryEntryWriter:: MinidumpSimpleStringDictionaryEntryWriter() : MinidumpWritable(), entry_(), key_(), value_() { } MinidumpSimpleStringDictionaryEntryWriter:: ~MinidumpSimpleStringDictionaryEntryWriter() { } const MinidumpSimpleStringDictionaryEntry* MinidumpSimpleStringDictionaryEntryWriter:: GetMinidumpSimpleStringDictionaryEntry() const { DCHECK_EQ(state(), kStateWritable); return &entry_; } void MinidumpSimpleStringDictionaryEntryWriter::SetKeyValue( const std::string& key, const std::string& value) { DCHECK_EQ(state(), kStateMutable); key_.SetUTF8(key); value_.SetUTF8(value); } bool MinidumpSimpleStringDictionaryEntryWriter::Freeze() { DCHECK_EQ(state(), kStateMutable); if (!MinidumpWritable::Freeze()) { return false; } key_.RegisterRVA(&entry_.key); value_.RegisterRVA(&entry_.value); return true; } size_t MinidumpSimpleStringDictionaryEntryWriter::SizeOfObject() { DCHECK_GE(state(), kStateFrozen); // This object doesn’t directly write anything itself. Its // MinidumpSimpleStringDictionaryEntry is written by its parent as part of a // MinidumpSimpleStringDictionary, and its children are responsible for // writing themselves. return 0; } std::vector MinidumpSimpleStringDictionaryEntryWriter::Children() { DCHECK_GE(state(), kStateFrozen); std::vector children(1, &key_); children.push_back(&value_); return children; } bool MinidumpSimpleStringDictionaryEntryWriter::WriteObject( FileWriterInterface* file_writer) { DCHECK_EQ(state(), kStateWritable); // This object doesn’t directly write anything itself. Its // MinidumpSimpleStringDictionaryEntry is written by its parent as part of a // MinidumpSimpleStringDictionary, and its children are responsible for // writing themselves. return true; } MinidumpSimpleStringDictionaryWriter::MinidumpSimpleStringDictionaryWriter() : MinidumpWritable(), entries_(), simple_string_dictionary_base_(new MinidumpSimpleStringDictionary()) { } MinidumpSimpleStringDictionaryWriter::~MinidumpSimpleStringDictionaryWriter() { base::STLDeleteContainerPairSecondPointers(entries_.begin(), entries_.end()); } void MinidumpSimpleStringDictionaryWriter::InitializeFromMap( const std::map& map) { DCHECK_EQ(state(), kStateMutable); DCHECK(entries_.empty()); for (const auto& iterator : map) { auto entry = base::WrapUnique(new MinidumpSimpleStringDictionaryEntryWriter()); entry->SetKeyValue(iterator.first, iterator.second); AddEntry(std::move(entry)); } } void MinidumpSimpleStringDictionaryWriter::AddEntry( std::unique_ptr entry) { DCHECK_EQ(state(), kStateMutable); const std::string& key = entry->Key(); auto iterator = entries_.find(key); if (iterator != entries_.end()) { delete iterator->second; iterator->second = entry.release(); } else { entries_[key] = entry.release(); } } bool MinidumpSimpleStringDictionaryWriter::IsUseful() const { return !entries_.empty(); } bool MinidumpSimpleStringDictionaryWriter::Freeze() { DCHECK_EQ(state(), kStateMutable); if (!MinidumpWritable::Freeze()) { return false; } size_t entry_count = entries_.size(); if (!AssignIfInRange(&simple_string_dictionary_base_->count, entry_count)) { LOG(ERROR) << "entry_count " << entry_count << " out of range"; return false; } return true; } size_t MinidumpSimpleStringDictionaryWriter::SizeOfObject() { DCHECK_GE(state(), kStateFrozen); return sizeof(*simple_string_dictionary_base_) + entries_.size() * sizeof(MinidumpSimpleStringDictionaryEntry); } std::vector MinidumpSimpleStringDictionaryWriter::Children() { DCHECK_GE(state(), kStateMutable); std::vector children; for (const auto& key_entry : entries_) { children.push_back(key_entry.second); } return children; } bool MinidumpSimpleStringDictionaryWriter::WriteObject( FileWriterInterface* file_writer) { DCHECK_GE(state(), kStateWritable); WritableIoVec iov; iov.iov_base = simple_string_dictionary_base_.get(); iov.iov_len = sizeof(*simple_string_dictionary_base_); std::vector iovecs(1, iov); for (const auto& key_entry : entries_) { iov.iov_base = key_entry.second->GetMinidumpSimpleStringDictionaryEntry(); iov.iov_len = sizeof(MinidumpSimpleStringDictionaryEntry); iovecs.push_back(iov); } return file_writer->WriteIoVec(&iovecs); } } // namespace crashpad