| OLD | NEW |
| 1 // Copyright 2015 The Chromium Authors. All rights reserved. | 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 | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "cc/raster/gpu_rasterizer.h" | 5 #include "cc/raster/gpu_rasterizer.h" |
| 6 | 6 |
| 7 #include <algorithm> | 7 #include <algorithm> |
| 8 | 8 |
| 9 #include "base/bind.h" | 9 #include "base/bind.h" |
| 10 #include "base/metrics/histogram.h" | 10 #include "base/metrics/histogram.h" |
| 11 #include "base/trace_event/trace_event.h" | 11 #include "base/trace_event/trace_event.h" |
| 12 #include "cc/debug/devtools_instrumentation.h" | 12 #include "cc/debug/devtools_instrumentation.h" |
| 13 #include "cc/debug/frame_viewer_instrumentation.h" | 13 #include "cc/debug/frame_viewer_instrumentation.h" |
| 14 #include "cc/output/context_provider.h" | 14 #include "cc/output/context_provider.h" |
| 15 #include "cc/playback/raster_source.h" | 15 #include "cc/playback/raster_source.h" |
| 16 #include "cc/raster/raster_buffer.h" | 16 #include "cc/raster/raster_buffer.h" |
| 17 #include "cc/raster/scoped_gpu_raster.h" | 17 #include "cc/raster/scoped_gpu_raster.h" |
| 18 #include "cc/resources/resource.h" | 18 #include "cc/resources/resource.h" |
| 19 #include "cc/tiles/tile_manager.h" | 19 #include "cc/tiles/tile_manager.h" |
| 20 #include "gpu/command_buffer/client/gles2_interface.h" | 20 #include "gpu/command_buffer/client/gles2_interface.h" |
| 21 #include "third_party/skia/include/core/SkMultiPictureDraw.h" | 21 #include "third_party/skia/include/core/SkMultiPictureDraw.h" |
| 22 #include "third_party/skia/include/core/SkPictureRecorder.h" | 22 #include "third_party/skia/include/core/SkPictureRecorder.h" |
| 23 #include "third_party/skia/include/core/SkSurface.h" | 23 #include "third_party/skia/include/core/SkSurface.h" |
| 24 #include "third_party/skia/include/gpu/GrContext.h" | 24 #include "third_party/skia/include/gpu/GrContext.h" |
| 25 | 25 |
| 26 namespace cc { | 26 namespace cc { |
| 27 | 27 |
| 28 namespace { |
| 29 SkColorType ToSkColorType(ResourceFormat format) { |
| 30 switch (format) { |
| 31 case RGBA_8888: |
| 32 return kRGBA_8888_SkColorType; |
| 33 case BGRA_8888: |
| 34 return kBGRA_8888_SkColorType; |
| 35 case RGBA_4444: |
| 36 return kARGB_4444_SkColorType; |
| 37 default: |
| 38 break; |
| 39 } |
| 40 DCHECK(false) << "Unsupported resource format."; |
| 41 return kN32_SkColorType; |
| 42 } |
| 43 } // namespace |
| 28 GpuRasterizer::GpuRasterizer(ContextProvider* worker_context_provider, | 44 GpuRasterizer::GpuRasterizer(ContextProvider* worker_context_provider, |
| 29 ResourceProvider* resource_provider, | 45 ResourceProvider* resource_provider, |
| 30 bool use_distance_field_text, | 46 bool use_device_independent_fonts, |
| 31 int msaa_sample_count) | 47 int msaa_sample_count) |
| 32 : worker_context_provider_(worker_context_provider), | 48 : worker_context_provider_(worker_context_provider), |
| 33 resource_provider_(resource_provider), | 49 resource_provider_(resource_provider), |
| 34 use_distance_field_text_(use_distance_field_text), | 50 use_device_independent_fonts_(use_device_independent_fonts), |
| 35 msaa_sample_count_(msaa_sample_count) { | 51 msaa_sample_count_(msaa_sample_count), |
| 36 DCHECK(worker_context_provider_); | 52 msaa_surface_use_device_independent_fonts_(false), |
| 37 } | 53 msaa_surface_can_use_lcd_text_(false) {} |
| 38 | 54 |
| 39 GpuRasterizer::~GpuRasterizer() { | 55 GpuRasterizer::~GpuRasterizer() { |
| 40 } | 56 } |
| 41 | 57 |
| 42 void GpuRasterizer::RasterizeSource( | 58 void GpuRasterizer::RasterizeSource( |
| 43 ResourceProvider::ScopedWriteLockGr* write_lock, | 59 ResourceProvider::ScopedWriteLockGr* write_lock, |
| 44 const RasterSource* raster_source, | 60 const RasterSource* raster_source, |
| 45 const gfx::Rect& raster_full_rect, | 61 const gfx::Rect& raster_full_rect, |
| 46 const gfx::Rect& playback_rect, | 62 const gfx::Rect& playback_rect, |
| 47 float scale, | 63 float scale, |
| 48 const RasterSource::PlaybackSettings& playback_settings) { | 64 const RasterSource::PlaybackSettings& playback_settings) { |
| 49 // Play back raster_source into temp SkPicture. | 65 // Play back raster_source into temp SkPicture. |
| 50 SkPictureRecorder recorder; | 66 SkPictureRecorder recorder; |
| 51 const gfx::Size size = write_lock->GetResourceSize(); | 67 const gfx::Size size = write_lock->GetResourceSize(); |
| 52 const int flags = SkPictureRecorder::kComputeSaveLayerInfo_RecordFlag; | 68 const int flags = SkPictureRecorder::kComputeSaveLayerInfo_RecordFlag; |
| 53 sk_sp<SkCanvas> canvas = sk_ref_sp( | 69 sk_sp<SkCanvas> canvas = sk_ref_sp( |
| 54 recorder.beginRecording(size.width(), size.height(), NULL, flags)); | 70 recorder.beginRecording(size.width(), size.height(), NULL, flags)); |
| 55 canvas->save(); | 71 canvas->save(); |
| 56 raster_source->PlaybackToCanvas(canvas.get(), raster_full_rect, playback_rect, | 72 raster_source->PlaybackToCanvas(canvas.get(), raster_full_rect, playback_rect, |
| 57 scale, playback_settings); | 73 scale, playback_settings); |
| 58 canvas->restore(); | 74 canvas->restore(); |
| 59 sk_sp<SkPicture> picture = recorder.finishRecordingAsPicture(); | 75 sk_sp<SkPicture> picture = recorder.finishRecordingAsPicture(); |
| 60 | 76 |
| 61 // Turn on distance fields for layers that have ever animated. | 77 // Turn on device independent fonts for layers that have ever animated. |
| 62 bool use_distance_field_text = | 78 bool use_device_independent_fonts = |
| 63 use_distance_field_text_ || | 79 use_device_independent_fonts_ || |
| 64 raster_source->ShouldAttemptToUseDistanceFieldText(); | 80 raster_source->ShouldAttemptToUseDistanceFieldText(); |
| 65 | 81 |
| 66 // Playback picture into resource. | 82 // Playback picture into resource. |
| 67 { | 83 { |
| 68 ScopedGpuRaster gpu_raster(worker_context_provider_); | 84 ScopedGpuRaster gpu_raster(worker_context_provider_); |
| 69 write_lock->InitSkSurface( | 85 SkSurface* sk_surface = |
| 70 worker_context_provider_->GrContext(), use_distance_field_text, | 86 AcquireSurface(gpu_raster, write_lock, use_device_independent_fonts, |
| 71 raster_source->CanUseLCDText(), msaa_sample_count_); | 87 raster_source->CanUseLCDText()); |
| 72 | |
| 73 SkSurface* sk_surface = write_lock->sk_surface(); | |
| 74 | 88 |
| 75 // Allocating an SkSurface will fail after a lost context. Pretend we | 89 // Allocating an SkSurface will fail after a lost context. Pretend we |
| 76 // rasterized, as the contents of the resource don't matter anymore. | 90 // rasterized, as the contents of the resource don't matter anymore. |
| 77 if (!sk_surface) | 91 if (!sk_surface) |
| 78 return; | 92 return; |
| 79 | 93 |
| 80 SkMultiPictureDraw multi_picture_draw; | 94 SkMultiPictureDraw multi_picture_draw; |
| 81 multi_picture_draw.add(sk_surface->getCanvas(), picture.get()); | 95 multi_picture_draw.add(sk_surface->getCanvas(), picture.get()); |
| 82 multi_picture_draw.draw(false); | 96 multi_picture_draw.draw(false); |
| 83 write_lock->ReleaseSkSurface(); | 97 ReleaseSurface(gpu_raster, write_lock); |
| 84 } | 98 } |
| 85 } | 99 } |
| 86 | 100 |
| 101 // ScopedGpuRaster is an argument so that we know caller holds it, proves |
| 102 // that Skia context has been reset. |
| 103 SkSurface* GpuRasterizer::AcquireSurface( |
| 104 const ScopedGpuRaster&, |
| 105 ResourceProvider::ScopedWriteLockGr* write_lock, |
| 106 bool use_device_independent_fonts, |
| 107 bool can_use_lcd_text) { |
| 108 bool uses_separate_msaa_backing_store = |
| 109 msaa_sample_count_ > 0 && |
| 110 !worker_context_provider_->ContextCapabilities() |
| 111 .multisampled_render_to_texture && |
| 112 !worker_context_provider_->ContextCapabilities() |
| 113 .chromium_framebuffer_mixed_samples; |
| 114 if (uses_separate_msaa_backing_store) { |
| 115 // Creating SkSurface, rendering to it and then destroying the surface is |
| 116 // slow due to need to allocate the backing store, e.g. GL renderbuffers. |
| 117 // Use intermediate longlived MSAA surface for rendering and then copy this |
| 118 // to the actual tile. |
| 119 gfx::Size size = write_lock->GetResourceSize(); |
| 120 SkColorType color_type = ToSkColorType(write_lock->GetResourceFormat()); |
| 121 SkImageInfo surface_image_info = SkImageInfo::Make( |
| 122 size.width(), size.height(), color_type, kPremul_SkAlphaType); |
| 123 AllocateMSAASurfaceIfNeeded(surface_image_info, |
| 124 use_device_independent_fonts, can_use_lcd_text); |
| 125 if (msaa_surface_) |
| 126 return msaa_surface_.get(); |
| 127 } |
| 128 |
| 129 msaa_surface_.reset(); |
| 130 |
| 131 write_lock->InitSkSurface(worker_context_provider_->GrContext(), |
| 132 use_device_independent_fonts, can_use_lcd_text, |
| 133 msaa_sample_count_); |
| 134 |
| 135 return write_lock->sk_surface(); |
| 136 } |
| 137 void GpuRasterizer::ReleaseSurface( |
| 138 const ScopedGpuRaster&, |
| 139 ResourceProvider::ScopedWriteLockGr* write_lock) { |
| 140 if (msaa_sample_count_ > 0 && msaa_surface_) { |
| 141 write_lock->InitSkSurface(worker_context_provider_->GrContext(), false, |
| 142 false, 0); |
| 143 SkSurface* target_surface = write_lock->sk_surface(); |
| 144 if (!target_surface) |
| 145 return; |
| 146 msaa_surface_->draw(target_surface->getCanvas(), 0, 0, nullptr); |
| 147 } |
| 148 write_lock->ReleaseSkSurface(); |
| 149 } |
| 150 |
| 151 void GpuRasterizer::AllocateMSAASurfaceIfNeeded( |
| 152 const SkImageInfo& image_info, |
| 153 bool use_device_independent_fonts, |
| 154 bool can_use_lcd_text) { |
| 155 if (msaa_surface_ && msaa_surface_image_info_ == image_info && |
| 156 msaa_surface_use_device_independent_fonts_ == |
| 157 use_device_independent_fonts && |
| 158 msaa_surface_can_use_lcd_text_ == can_use_lcd_text) |
| 159 return; |
| 160 |
| 161 msaa_surface_.reset(); |
| 162 |
| 163 uint32_t flags = use_device_independent_fonts |
| 164 ? SkSurfaceProps::kUseDeviceIndependentFonts_Flag |
| 165 : 0; |
| 166 SkSurfaceProps surface_props(flags, kUnknown_SkPixelGeometry); |
| 167 if (can_use_lcd_text) |
| 168 surface_props = |
| 169 SkSurfaceProps(flags, SkSurfaceProps::kLegacyFontHost_InitType); |
| 170 |
| 171 msaa_surface_ = SkSurface::MakeRenderTarget( |
| 172 worker_context_provider_->GrContext(), SkBudgeted::kYes, image_info, |
| 173 msaa_sample_count_, &surface_props); |
| 174 msaa_surface_image_info_ = image_info; |
| 175 msaa_surface_use_device_independent_fonts_ = use_device_independent_fonts; |
| 176 msaa_surface_can_use_lcd_text_ = can_use_lcd_text; |
| 177 } |
| 178 |
| 87 } // namespace cc | 179 } // namespace cc |
| OLD | NEW |