James Robinson | 646469d | 2014-10-03 15:33:28 -0700 | [diff] [blame] | 1 | // Copyright 2014 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 "cc/surfaces/surface.h" |
| 6 | |
James Robinson | 7f48021 | 2014-10-31 10:28:08 -0700 | [diff] [blame] | 7 | #include <algorithm> |
| 8 | |
James Robinson | 646469d | 2014-10-03 15:33:28 -0700 | [diff] [blame] | 9 | #include "cc/output/compositor_frame.h" |
| 10 | #include "cc/output/copy_output_request.h" |
| 11 | #include "cc/surfaces/surface_factory.h" |
James Robinson | 1ae030a | 2014-11-07 08:32:47 -0800 | [diff] [blame] | 12 | #include "cc/surfaces/surface_id_allocator.h" |
James Robinson | e2ac7e8 | 2014-10-15 13:21:59 -0700 | [diff] [blame] | 13 | #include "cc/surfaces/surface_manager.h" |
James Robinson | 646469d | 2014-10-03 15:33:28 -0700 | [diff] [blame] | 14 | |
| 15 | namespace cc { |
| 16 | |
| 17 | // The frame index starts at 2 so that empty frames will be treated as |
| 18 | // completely damaged the first time they're drawn from. |
| 19 | static const int kFrameIndexStart = 2; |
| 20 | |
James Robinson | 6a64b81 | 2014-12-03 13:38:42 -0800 | [diff] [blame] | 21 | Surface::Surface(SurfaceId id, SurfaceFactory* factory) |
James Robinson | 646469d | 2014-10-03 15:33:28 -0700 | [diff] [blame] | 22 | : surface_id_(id), |
James Robinson | e2ac7e8 | 2014-10-15 13:21:59 -0700 | [diff] [blame] | 23 | factory_(factory->AsWeakPtr()), |
James Robinson | 646469d | 2014-10-03 15:33:28 -0700 | [diff] [blame] | 24 | frame_index_(kFrameIndexStart) { |
| 25 | } |
| 26 | |
| 27 | Surface::~Surface() { |
James Robinson | 80d418c | 2014-10-16 16:00:02 -0700 | [diff] [blame] | 28 | ClearCopyRequests(); |
James Robinson | e2ac7e8 | 2014-10-15 13:21:59 -0700 | [diff] [blame] | 29 | if (current_frame_ && factory_) { |
James Robinson | 646469d | 2014-10-03 15:33:28 -0700 | [diff] [blame] | 30 | ReturnedResourceArray current_resources; |
| 31 | TransferableResource::ReturnResources( |
| 32 | current_frame_->delegated_frame_data->resource_list, |
| 33 | ¤t_resources); |
| 34 | factory_->UnrefResources(current_resources); |
| 35 | } |
Etienne Membrives | b1556b3 | 2014-12-16 13:56:09 +0100 | [diff] [blame] | 36 | if (!draw_callback_.is_null()) |
Elliot Glaysher | eae4929 | 2015-01-28 10:47:32 -0800 | [diff] [blame] | 37 | draw_callback_.Run(SurfaceDrawStatus::DRAW_SKIPPED); |
James Robinson | 646469d | 2014-10-03 15:33:28 -0700 | [diff] [blame] | 38 | } |
| 39 | |
| 40 | void Surface::QueueFrame(scoped_ptr<CompositorFrame> frame, |
Etienne Membrives | b1556b3 | 2014-12-16 13:56:09 +0100 | [diff] [blame] | 41 | const DrawCallback& callback) { |
James Robinson | e2ac7e8 | 2014-10-15 13:21:59 -0700 | [diff] [blame] | 42 | DCHECK(factory_); |
James Robinson | 80d418c | 2014-10-16 16:00:02 -0700 | [diff] [blame] | 43 | ClearCopyRequests(); |
James Robinson | 646469d | 2014-10-03 15:33:28 -0700 | [diff] [blame] | 44 | TakeLatencyInfo(&frame->metadata.latency_info); |
| 45 | scoped_ptr<CompositorFrame> previous_frame = current_frame_.Pass(); |
| 46 | current_frame_ = frame.Pass(); |
| 47 | factory_->ReceiveFromChild( |
| 48 | current_frame_->delegated_frame_data->resource_list); |
James Robinson | 7b766f4 | 2015-02-06 15:14:04 -0800 | [diff] [blame] | 49 | // Empty frames shouldn't be drawn and shouldn't contribute damage, so don't |
| 50 | // increment frame index for them. |
| 51 | if (!current_frame_ || |
| 52 | !current_frame_->delegated_frame_data->render_pass_list.empty()) |
| 53 | ++frame_index_; |
James Robinson | 646469d | 2014-10-03 15:33:28 -0700 | [diff] [blame] | 54 | |
| 55 | if (previous_frame) { |
| 56 | ReturnedResourceArray previous_resources; |
| 57 | TransferableResource::ReturnResources( |
| 58 | previous_frame->delegated_frame_data->resource_list, |
| 59 | &previous_resources); |
| 60 | factory_->UnrefResources(previous_resources); |
| 61 | } |
| 62 | if (!draw_callback_.is_null()) |
Elliot Glaysher | eae4929 | 2015-01-28 10:47:32 -0800 | [diff] [blame] | 63 | draw_callback_.Run(SurfaceDrawStatus::DRAW_SKIPPED); |
James Robinson | 646469d | 2014-10-03 15:33:28 -0700 | [diff] [blame] | 64 | draw_callback_ = callback; |
James Robinson | e2ac7e8 | 2014-10-15 13:21:59 -0700 | [diff] [blame] | 65 | factory_->manager()->DidSatisfySequences( |
James Robinson | 1ae030a | 2014-11-07 08:32:47 -0800 | [diff] [blame] | 66 | SurfaceIdAllocator::NamespaceForId(surface_id_), |
| 67 | ¤t_frame_->metadata.satisfies_sequences); |
James Robinson | 646469d | 2014-10-03 15:33:28 -0700 | [diff] [blame] | 68 | } |
| 69 | |
| 70 | void Surface::RequestCopyOfOutput(scoped_ptr<CopyOutputRequest> copy_request) { |
James Robinson | 80d418c | 2014-10-16 16:00:02 -0700 | [diff] [blame] | 71 | if (current_frame_ && |
| 72 | !current_frame_->delegated_frame_data->render_pass_list.empty()) |
| 73 | current_frame_->delegated_frame_data->render_pass_list.back() |
| 74 | ->copy_requests.push_back(copy_request.Pass()); |
| 75 | else |
| 76 | copy_request->SendEmptyResult(); |
James Robinson | 646469d | 2014-10-03 15:33:28 -0700 | [diff] [blame] | 77 | } |
| 78 | |
| 79 | void Surface::TakeCopyOutputRequests( |
James Robinson | 80d418c | 2014-10-16 16:00:02 -0700 | [diff] [blame] | 80 | std::multimap<RenderPassId, CopyOutputRequest*>* copy_requests) { |
James Robinson | 646469d | 2014-10-03 15:33:28 -0700 | [diff] [blame] | 81 | DCHECK(copy_requests->empty()); |
James Robinson | 80d418c | 2014-10-16 16:00:02 -0700 | [diff] [blame] | 82 | if (current_frame_) { |
James Robinson | 7f48021 | 2014-10-31 10:28:08 -0700 | [diff] [blame] | 83 | for (const auto& render_pass : |
James Robinson | 80d418c | 2014-10-16 16:00:02 -0700 | [diff] [blame] | 84 | current_frame_->delegated_frame_data->render_pass_list) { |
| 85 | while (!render_pass->copy_requests.empty()) { |
| 86 | scoped_ptr<CopyOutputRequest> request = |
| 87 | render_pass->copy_requests.take_back(); |
| 88 | render_pass->copy_requests.pop_back(); |
| 89 | copy_requests->insert( |
| 90 | std::make_pair(render_pass->id, request.release())); |
| 91 | } |
| 92 | } |
| 93 | } |
James Robinson | 646469d | 2014-10-03 15:33:28 -0700 | [diff] [blame] | 94 | } |
| 95 | |
| 96 | const CompositorFrame* Surface::GetEligibleFrame() { |
| 97 | return current_frame_.get(); |
| 98 | } |
| 99 | |
| 100 | void Surface::TakeLatencyInfo(std::vector<ui::LatencyInfo>* latency_info) { |
| 101 | if (!current_frame_) |
| 102 | return; |
| 103 | if (latency_info->empty()) { |
| 104 | current_frame_->metadata.latency_info.swap(*latency_info); |
| 105 | return; |
| 106 | } |
| 107 | std::copy(current_frame_->metadata.latency_info.begin(), |
| 108 | current_frame_->metadata.latency_info.end(), |
| 109 | std::back_inserter(*latency_info)); |
| 110 | current_frame_->metadata.latency_info.clear(); |
| 111 | } |
| 112 | |
Elliot Glaysher | eae4929 | 2015-01-28 10:47:32 -0800 | [diff] [blame] | 113 | void Surface::RunDrawCallbacks(SurfaceDrawStatus drawn) { |
James Robinson | 646469d | 2014-10-03 15:33:28 -0700 | [diff] [blame] | 114 | if (!draw_callback_.is_null()) { |
Etienne Membrives | b1556b3 | 2014-12-16 13:56:09 +0100 | [diff] [blame] | 115 | DrawCallback callback = draw_callback_; |
| 116 | draw_callback_ = DrawCallback(); |
Elliot Glaysher | eae4929 | 2015-01-28 10:47:32 -0800 | [diff] [blame] | 117 | callback.Run(drawn); |
James Robinson | 646469d | 2014-10-03 15:33:28 -0700 | [diff] [blame] | 118 | } |
| 119 | } |
| 120 | |
James Robinson | 1ae030a | 2014-11-07 08:32:47 -0800 | [diff] [blame] | 121 | void Surface::AddDestructionDependency(SurfaceSequence sequence) { |
| 122 | destruction_dependencies_.push_back(sequence); |
| 123 | } |
| 124 | |
| 125 | void Surface::SatisfyDestructionDependencies( |
| 126 | base::hash_set<SurfaceSequence>* sequences) { |
| 127 | destruction_dependencies_.erase( |
| 128 | std::remove_if( |
| 129 | destruction_dependencies_.begin(), destruction_dependencies_.end(), |
| 130 | [sequences](SurfaceSequence seq) { return !!sequences->erase(seq); }), |
| 131 | destruction_dependencies_.end()); |
| 132 | } |
| 133 | |
James Robinson | 80d418c | 2014-10-16 16:00:02 -0700 | [diff] [blame] | 134 | void Surface::ClearCopyRequests() { |
| 135 | if (current_frame_) { |
James Robinson | 7f48021 | 2014-10-31 10:28:08 -0700 | [diff] [blame] | 136 | for (const auto& render_pass : |
James Robinson | 80d418c | 2014-10-16 16:00:02 -0700 | [diff] [blame] | 137 | current_frame_->delegated_frame_data->render_pass_list) { |
James Robinson | 7f48021 | 2014-10-31 10:28:08 -0700 | [diff] [blame] | 138 | for (const auto& copy_request : render_pass->copy_requests) |
James Robinson | 80d418c | 2014-10-16 16:00:02 -0700 | [diff] [blame] | 139 | copy_request->SendEmptyResult(); |
| 140 | } |
| 141 | } |
| 142 | } |
| 143 | |
James Robinson | 646469d | 2014-10-03 15:33:28 -0700 | [diff] [blame] | 144 | } // namespace cc |