Index: ui/ozone/platform/drm/gpu/drm_buffer.cc |
diff --git a/ui/ozone/platform/drm/gpu/drm_buffer.cc b/ui/ozone/platform/drm/gpu/drm_buffer.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..0ab1f61aa3fd53f80b621da6566fd504fce94e9e |
--- /dev/null |
+++ b/ui/ozone/platform/drm/gpu/drm_buffer.cc |
@@ -0,0 +1,118 @@ |
+// Copyright 2014 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 "ui/ozone/platform/drm/gpu/drm_buffer.h" |
+ |
+#include "base/logging.h" |
+#include "ui/ozone/platform/drm/gpu/drm_device.h" |
+ |
+namespace ui { |
+ |
+namespace { |
+ |
+// Modesetting cannot happen from a buffer with transparencies. Return the size |
+// of a pixel without alpha. |
+uint8_t GetColorDepth(SkColorType type) { |
+ switch (type) { |
+ case kUnknown_SkColorType: |
+ case kAlpha_8_SkColorType: |
+ return 0; |
+ case kIndex_8_SkColorType: |
+ return 8; |
+ case kRGB_565_SkColorType: |
+ return 16; |
+ case kARGB_4444_SkColorType: |
+ return 12; |
+ case kN32_SkColorType: |
+ return 24; |
+ default: |
+ NOTREACHED(); |
+ return 0; |
+ } |
+} |
+ |
+} // namespace |
+ |
+DrmBuffer::DrmBuffer(const scoped_refptr<DrmDevice>& drm) : drm_(drm) { |
+} |
+ |
+DrmBuffer::~DrmBuffer() { |
+ surface_.clear(); |
+ |
+ if (framebuffer_ && !drm_->RemoveFramebuffer(framebuffer_)) |
+ PLOG(ERROR) << "DrmBuffer: RemoveFramebuffer: fb " << framebuffer_; |
+ |
+ if (mmap_base_ && !drm_->UnmapDumbBuffer(mmap_base_, mmap_size_)) |
+ PLOG(ERROR) << "DrmBuffer: UnmapDumbBuffer: handle " << handle_; |
+ |
+ if (handle_ && !drm_->DestroyDumbBuffer(handle_)) |
+ PLOG(ERROR) << "DrmBuffer: DestroyDumbBuffer: handle " << handle_; |
+} |
+ |
+bool DrmBuffer::Initialize(const SkImageInfo& info, |
+ bool should_register_framebuffer) { |
+ if (!drm_->CreateDumbBuffer(info, &handle_, &stride_)) { |
+ PLOG(ERROR) << "DrmBuffer: CreateDumbBuffer: width " << info.width() |
+ << " height " << info.height(); |
+ return false; |
+ } |
+ |
+ mmap_size_ = info.getSafeSize(stride_); |
+ if (!drm_->MapDumbBuffer(handle_, mmap_size_, &mmap_base_)) { |
+ PLOG(ERROR) << "DrmBuffer: MapDumbBuffer: handle " << handle_; |
+ return false; |
+ } |
+ |
+ if (should_register_framebuffer && |
+ !drm_->AddFramebuffer( |
+ info.width(), info.height(), GetColorDepth(info.colorType()), |
+ info.bytesPerPixel() << 3, stride_, handle_, &framebuffer_)) { |
+ PLOG(ERROR) << "DrmBuffer: AddFramebuffer: handle " << handle_; |
+ return false; |
+ } |
+ |
+ surface_ = |
+ skia::AdoptRef(SkSurface::NewRasterDirect(info, mmap_base_, stride_)); |
+ if (!surface_) { |
+ LOG(ERROR) << "DrmBuffer: Failed to create SkSurface: handle " << handle_; |
+ return false; |
+ } |
+ |
+ return true; |
+} |
+ |
+SkCanvas* DrmBuffer::GetCanvas() const { |
+ return surface_->getCanvas(); |
+} |
+ |
+uint32_t DrmBuffer::GetFramebufferId() const { |
+ return framebuffer_; |
+} |
+ |
+uint32_t DrmBuffer::GetHandle() const { |
+ return handle_; |
+} |
+ |
+gfx::Size DrmBuffer::GetSize() const { |
+ return gfx::Size(surface_->width(), surface_->height()); |
+} |
+ |
+DrmBufferGenerator::DrmBufferGenerator() { |
+} |
+ |
+DrmBufferGenerator::~DrmBufferGenerator() { |
+} |
+ |
+scoped_refptr<ScanoutBuffer> DrmBufferGenerator::Create( |
+ const scoped_refptr<DrmDevice>& drm, |
+ const gfx::Size& size) { |
+ scoped_refptr<DrmBuffer> buffer(new DrmBuffer(drm)); |
+ SkImageInfo info = SkImageInfo::MakeN32Premul(size.width(), size.height()); |
+ if (!buffer->Initialize(info, true /* should_register_framebuffer */)) |
+ return NULL; |
+ |
+ return buffer; |
+} |
+ |
+} // namespace ui |