| OLD | NEW |
| (Empty) | |
| 1 // Copyright 2015 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. |
| 4 |
| 5 #include "content/child/shared_memory_received_data_factory.h" |
| 6 |
| 7 #include <algorithm> |
| 8 |
| 9 #include "content/common/resource_messages.h" |
| 10 #include "ipc/ipc_sender.h" |
| 11 |
| 12 namespace content { |
| 13 |
| 14 class SharedMemoryReceivedDataFactory::SharedMemoryReceivedData final |
| 15 : public RequestPeer::ReceivedData { |
| 16 public: |
| 17 SharedMemoryReceivedData( |
| 18 const char* payload, |
| 19 int length, |
| 20 int encoded_length, |
| 21 scoped_refptr<SharedMemoryReceivedDataFactory> factory, |
| 22 SharedMemoryReceivedDataFactory::TicketId id) |
| 23 : payload_(payload), |
| 24 length_(length), |
| 25 encoded_length_(encoded_length), |
| 26 factory_(factory), |
| 27 id_(id) {} |
| 28 |
| 29 ~SharedMemoryReceivedData() override { factory_->Reclaim(id_); } |
| 30 |
| 31 const char* payload() const override { return payload_; } |
| 32 int length() const override { return length_; } |
| 33 int encoded_length() const override { return encoded_length_; } |
| 34 |
| 35 private: |
| 36 const char* const payload_; |
| 37 const int length_; |
| 38 const int encoded_length_; |
| 39 |
| 40 scoped_refptr<SharedMemoryReceivedDataFactory> factory_; |
| 41 SharedMemoryReceivedDataFactory::TicketId id_; |
| 42 |
| 43 DISALLOW_COPY_AND_ASSIGN(SharedMemoryReceivedData); |
| 44 }; |
| 45 |
| 46 SharedMemoryReceivedDataFactory::SharedMemoryReceivedDataFactory( |
| 47 IPC::Sender* message_sender, |
| 48 int request_id, |
| 49 linked_ptr<base::SharedMemory> memory) |
| 50 : id_(0), |
| 51 oldest_(0), |
| 52 message_sender_(message_sender), |
| 53 request_id_(request_id), |
| 54 is_stopped_(false), |
| 55 memory_(memory) { |
| 56 } |
| 57 |
| 58 SharedMemoryReceivedDataFactory::~SharedMemoryReceivedDataFactory() { |
| 59 if (!is_stopped_) |
| 60 SendAck(released_tickets_.size()); |
| 61 } |
| 62 |
| 63 scoped_ptr<RequestPeer::ReceivedData> SharedMemoryReceivedDataFactory::Create( |
| 64 int offset, |
| 65 int length, |
| 66 int encoded_length) { |
| 67 const char* start = static_cast<char*>(memory_->memory()); |
| 68 const char* payload = start + offset; |
| 69 TicketId id = id_++; |
| 70 |
| 71 return make_scoped_ptr( |
| 72 new SharedMemoryReceivedData(payload, length, encoded_length, this, id)); |
| 73 } |
| 74 |
| 75 void SharedMemoryReceivedDataFactory::Stop() { |
| 76 is_stopped_ = true; |
| 77 released_tickets_.clear(); |
| 78 message_sender_ = nullptr; |
| 79 } |
| 80 |
| 81 class SharedMemoryReceivedDataFactory::TicketComparator final { |
| 82 public: |
| 83 explicit TicketComparator(TicketId oldest) : oldest_(oldest) {} |
| 84 bool operator()(TicketId x, TicketId y) const { |
| 85 if ((oldest_ <= x) == (oldest_ <= y)) |
| 86 return x <= y; |
| 87 |
| 88 return (oldest_ <= x); |
| 89 } |
| 90 |
| 91 private: |
| 92 TicketId oldest_; |
| 93 }; |
| 94 |
| 95 void SharedMemoryReceivedDataFactory::Reclaim(TicketId id) { |
| 96 if (is_stopped_) |
| 97 return; |
| 98 if (oldest_ != id) { |
| 99 released_tickets_.push_back(id); |
| 100 return; |
| 101 } |
| 102 |
| 103 ++oldest_; |
| 104 SendAck(1); |
| 105 if (released_tickets_.empty()) { |
| 106 // Fast path: (hopefully) the most typical case. |
| 107 return; |
| 108 } |
| 109 std::sort(released_tickets_.begin(), released_tickets_.end(), |
| 110 TicketComparator(oldest_)); |
| 111 size_t count = 0; |
| 112 for (size_t i = 0;; ++i) { |
| 113 if (i == released_tickets_.size() || |
| 114 released_tickets_[i] != static_cast<TicketId>(id + i + 1)) { |
| 115 count = i; |
| 116 break; |
| 117 } |
| 118 } |
| 119 released_tickets_.erase(released_tickets_.begin(), |
| 120 released_tickets_.begin() + count); |
| 121 oldest_ += count; |
| 122 SendAck(count); |
| 123 } |
| 124 |
| 125 void SharedMemoryReceivedDataFactory::SendAck(size_t count) { |
| 126 for (size_t i = 0; i < count; ++i) { |
| 127 message_sender_->Send(new ResourceHostMsg_DataReceived_ACK(request_id_)); |
| 128 } |
| 129 } |
| 130 |
| 131 } // namespace content |
| OLD | NEW |