| // Copyright (c) 2011 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "ipc/ipc_message_attachment_set.h" |
| |
| #include <stddef.h> |
| |
| #include <algorithm> |
| |
| #include "base/logging.h" |
| #include "base/posix/eintr_wrapper.h" |
| #include "build/build_config.h" |
| #include "ipc/brokerable_attachment.h" |
| #include "ipc/ipc_message_attachment.h" |
| |
| #if defined(OS_POSIX) |
| #include <sys/stat.h> |
| #include <sys/types.h> |
| #include <unistd.h> |
| #include "ipc/ipc_platform_file_attachment_posix.h" |
| #endif // OS_POSIX |
| |
| namespace IPC { |
| |
| namespace { |
| |
| unsigned count_attachments_of_type( |
| const std::vector<scoped_refptr<MessageAttachment>>& attachments, |
| MessageAttachment::Type type) { |
| unsigned count = 0; |
| for (const scoped_refptr<MessageAttachment>& attachment : attachments) { |
| if (attachment->GetType() == type) |
| ++count; |
| } |
| return count; |
| } |
| |
| } // namespace |
| |
| MessageAttachmentSet::MessageAttachmentSet() |
| : consumed_descriptor_highwater_(0) { |
| } |
| |
| MessageAttachmentSet::~MessageAttachmentSet() { |
| if (consumed_descriptor_highwater_ == num_non_brokerable_attachments()) |
| return; |
| |
| // We close all the owning descriptors. If this message should have |
| // been transmitted, then closing those with close flags set mirrors |
| // the expected behaviour. |
| // |
| // If this message was received with more descriptors than expected |
| // (which could a DOS against the browser by a rogue renderer) then all |
| // the descriptors have their close flag set and we free all the extra |
| // kernel resources. |
| LOG(WARNING) << "MessageAttachmentSet destroyed with unconsumed descriptors: " |
| << consumed_descriptor_highwater_ << "/" << num_descriptors(); |
| } |
| |
| unsigned MessageAttachmentSet::num_descriptors() const { |
| return count_attachments_of_type(attachments_, |
| MessageAttachment::TYPE_PLATFORM_FILE); |
| } |
| |
| unsigned MessageAttachmentSet::num_mojo_handles() const { |
| return count_attachments_of_type(attachments_, |
| MessageAttachment::TYPE_MOJO_HANDLE); |
| } |
| |
| unsigned MessageAttachmentSet::num_brokerable_attachments() const { |
| return static_cast<unsigned>(brokerable_attachments_.size()); |
| } |
| |
| unsigned MessageAttachmentSet::num_non_brokerable_attachments() const { |
| return static_cast<unsigned>(attachments_.size()); |
| } |
| |
| unsigned MessageAttachmentSet::size() const { |
| return static_cast<unsigned>(attachments_.size() + |
| brokerable_attachments_.size()); |
| } |
| |
| bool MessageAttachmentSet::AddAttachment( |
| scoped_refptr<MessageAttachment> attachment, |
| size_t* index, |
| bool* brokerable) { |
| #if defined(OS_POSIX) |
| if (attachment->GetType() == MessageAttachment::TYPE_PLATFORM_FILE && |
| num_descriptors() == kMaxDescriptorsPerMessage) { |
| DLOG(WARNING) << "Cannot add file descriptor. MessageAttachmentSet full."; |
| return false; |
| } |
| #endif |
| |
| switch (attachment->GetType()) { |
| case MessageAttachment::TYPE_PLATFORM_FILE: |
| case MessageAttachment::TYPE_MOJO_HANDLE: |
| attachments_.push_back(attachment); |
| *index = attachments_.size() - 1; |
| *brokerable = false; |
| return true; |
| case MessageAttachment::TYPE_BROKERABLE_ATTACHMENT: |
| BrokerableAttachment* brokerable_attachment = |
| static_cast<BrokerableAttachment*>(attachment.get()); |
| scoped_refptr<BrokerableAttachment> a(brokerable_attachment); |
| brokerable_attachments_.push_back(a); |
| *index = brokerable_attachments_.size() - 1; |
| *brokerable = true; |
| return true; |
| } |
| return false; |
| } |
| |
| bool MessageAttachmentSet::AddAttachment( |
| scoped_refptr<MessageAttachment> attachment) { |
| bool brokerable; |
| size_t index; |
| return AddAttachment(attachment, &index, &brokerable); |
| } |
| |
| scoped_refptr<MessageAttachment> |
| MessageAttachmentSet::GetNonBrokerableAttachmentAt(unsigned index) { |
| if (index >= num_non_brokerable_attachments()) { |
| DLOG(WARNING) << "Accessing out of bound index:" << index << "/" |
| << num_non_brokerable_attachments(); |
| return scoped_refptr<MessageAttachment>(); |
| } |
| |
| // We should always walk the descriptors in order, so it's reasonable to |
| // enforce this. Consider the case where a compromised renderer sends us |
| // the following message: |
| // |
| // ExampleMsg: |
| // num_fds:2 msg:FD(index = 1) control:SCM_RIGHTS {n, m} |
| // |
| // Here the renderer sent us a message which should have a descriptor, but |
| // actually sent two in an attempt to fill our fd table and kill us. By |
| // setting the index of the descriptor in the message to 1 (it should be |
| // 0), we would record a highwater of 1 and then consider all the |
| // descriptors to have been used. |
| // |
| // So we can either track of the use of each descriptor in a bitset, or we |
| // can enforce that we walk the indexes strictly in order. |
| // |
| // There's one more wrinkle: When logging messages, we may reparse them. So |
| // we have an exception: When the consumed_descriptor_highwater_ is at the |
| // end of the array and index 0 is requested, we reset the highwater value. |
| // TODO(morrita): This is absurd. This "wringle" disallow to introduce clearer |
| // ownership model. Only client is NaclIPCAdapter. See crbug.com/415294 |
| if (index == 0 && |
| consumed_descriptor_highwater_ == num_non_brokerable_attachments()) { |
| consumed_descriptor_highwater_ = 0; |
| } |
| |
| if (index != consumed_descriptor_highwater_) |
| return scoped_refptr<MessageAttachment>(); |
| |
| consumed_descriptor_highwater_ = index + 1; |
| |
| return attachments_[index]; |
| } |
| |
| scoped_refptr<MessageAttachment> |
| MessageAttachmentSet::GetBrokerableAttachmentAt(unsigned index) { |
| if (index >= num_brokerable_attachments()) { |
| DLOG(WARNING) << "Accessing out of bound index:" << index << "/" |
| << num_brokerable_attachments(); |
| return scoped_refptr<MessageAttachment>(); |
| } |
| |
| scoped_refptr<BrokerableAttachment> brokerable_attachment( |
| brokerable_attachments_[index]); |
| return scoped_refptr<MessageAttachment>(brokerable_attachment.get()); |
| } |
| |
| void MessageAttachmentSet::CommitAllDescriptors() { |
| attachments_.clear(); |
| consumed_descriptor_highwater_ = 0; |
| } |
| |
| std::vector<scoped_refptr<IPC::BrokerableAttachment>> |
| MessageAttachmentSet::GetBrokerableAttachments() const { |
| return brokerable_attachments_; |
| } |
| |
| void MessageAttachmentSet::ReplacePlaceholderWithAttachment( |
| const scoped_refptr<BrokerableAttachment>& attachment) { |
| DCHECK_NE(BrokerableAttachment::PLACEHOLDER, attachment->GetBrokerableType()); |
| for (auto it = brokerable_attachments_.begin(); |
| it != brokerable_attachments_.end(); ++it) { |
| if ((*it)->GetBrokerableType() == BrokerableAttachment::PLACEHOLDER && |
| (*it)->GetIdentifier() == attachment->GetIdentifier()) { |
| *it = attachment; |
| return; |
| } |
| } |
| |
| // This function should only be called if there is a placeholder ready to be |
| // replaced. |
| NOTREACHED(); |
| } |
| |
| #if defined(OS_POSIX) |
| |
| void MessageAttachmentSet::PeekDescriptors(base::PlatformFile* buffer) const { |
| for (size_t i = 0; i != attachments_.size(); ++i) |
| buffer[i] = internal::GetPlatformFile(attachments_[i]); |
| } |
| |
| bool MessageAttachmentSet::ContainsDirectoryDescriptor() const { |
| struct stat st; |
| |
| for (auto i = attachments_.begin(); i != attachments_.end(); ++i) { |
| if (fstat(internal::GetPlatformFile(*i), &st) == 0 && S_ISDIR(st.st_mode)) |
| return true; |
| } |
| |
| return false; |
| } |
| |
| void MessageAttachmentSet::ReleaseFDsToClose( |
| std::vector<base::PlatformFile>* fds) { |
| for (size_t i = 0; i < attachments_.size(); ++i) { |
| internal::PlatformFileAttachment* file = |
| static_cast<internal::PlatformFileAttachment*>(attachments_[i].get()); |
| if (file->Owns()) |
| fds->push_back(file->TakePlatformFile()); |
| } |
| |
| CommitAllDescriptors(); |
| } |
| |
| void MessageAttachmentSet::AddDescriptorsToOwn(const base::PlatformFile* buffer, |
| unsigned count) { |
| DCHECK(count <= kMaxDescriptorsPerMessage); |
| DCHECK_EQ(num_descriptors(), 0u); |
| DCHECK_EQ(consumed_descriptor_highwater_, 0u); |
| |
| attachments_.reserve(count); |
| for (unsigned i = 0; i < count; ++i) |
| AddAttachment( |
| new internal::PlatformFileAttachment(base::ScopedFD(buffer[i]))); |
| } |
| |
| #endif // OS_POSIX |
| |
| } // namespace IPC |
| |
| |