| OLD | NEW |
| 1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2010 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 "webkit/plugins/ppapi/ppb_context_3d_impl.h" | 5 #include "webkit/plugins/ppapi/ppb_context_3d_impl.h" |
| 6 | 6 |
| 7 #include "gpu/command_buffer/common/command_buffer.h" | 7 #include "gpu/command_buffer/common/command_buffer.h" |
| 8 #include "ppapi/c/dev/ppb_graphics_3d_dev.h" | |
| 9 #include "webkit/plugins/ppapi/common.h" | 8 #include "webkit/plugins/ppapi/common.h" |
| 10 #include "webkit/plugins/ppapi/ppapi_plugin_instance.h" | 9 #include "webkit/plugins/ppapi/ppapi_plugin_instance.h" |
| 10 #include "webkit/plugins/ppapi/ppb_surface_3d_impl.h" |
| 11 | 11 |
| 12 namespace webkit { | 12 namespace webkit { |
| 13 namespace ppapi { | 13 namespace ppapi { |
| 14 | 14 |
| 15 namespace { | 15 namespace { |
| 16 | 16 |
| 17 // Size of the transfer buffer. | 17 // Size of the transfer buffer. |
| 18 enum { kTransferBufferSize = 512 * 1024 }; | 18 enum { kTransferBufferSize = 512 * 1024 }; |
| 19 | 19 |
| 20 PP_Resource Create(PP_Instance instance_id, | 20 PP_Resource Create(PP_Instance instance_id, |
| 21 PP_Config3D_Dev config, | 21 PP_Config3D_Dev config, |
| 22 PP_Resource share_context, | 22 PP_Resource share_context, |
| 23 const int32_t* attrib_list) { | 23 const int32_t* attrib_list) { |
| 24 // TODO(alokp): Support shared context. | 24 // TODO(alokp): Support shared context. |
| 25 DCHECK_EQ(0, share_context); | 25 DCHECK_EQ(0, share_context); |
| 26 if (share_context != 0) | 26 if (share_context != 0) |
| 27 return 0; | 27 return 0; |
| 28 | 28 |
| 29 PluginInstance* instance = ResourceTracker::Get()->GetInstance(instance_id); | 29 PluginInstance* instance = ResourceTracker::Get()->GetInstance(instance_id); |
| 30 if (!instance) | 30 if (!instance) |
| 31 return 0; | 31 return 0; |
| 32 | 32 |
| 33 scoped_refptr<PPB_Context3D_Impl> context( | 33 scoped_refptr<PPB_Context3D_Impl> context( |
| 34 new PPB_Context3D_Impl(instance->module())); | 34 new PPB_Context3D_Impl(instance)); |
| 35 if (!context->Init(instance, config, share_context, attrib_list)) | 35 if (!context->Init(config, share_context, attrib_list)) |
| 36 return 0; | 36 return 0; |
| 37 | 37 |
| 38 return context->GetReference(); | 38 return context->GetReference(); |
| 39 } | 39 } |
| 40 | 40 |
| 41 PP_Bool IsContext3D(PP_Resource resource) { | 41 PP_Bool IsContext3D(PP_Resource resource) { |
| 42 return BoolToPPBool(!!Resource::GetAs<PPB_Context3D_Impl>(resource)); | 42 return BoolToPPBool(!!Resource::GetAs<PPB_Context3D_Impl>(resource)); |
| 43 } | 43 } |
| 44 | 44 |
| 45 int32_t GetAttrib(PP_Resource context, | 45 int32_t GetAttrib(PP_Resource context, |
| 46 int32_t attribute, | 46 int32_t attribute, |
| 47 int32_t* value) { | 47 int32_t* value) { |
| 48 // TODO(alokp): Implement me. | 48 // TODO(alokp): Implement me. |
| 49 return 0; | 49 return 0; |
| 50 } | 50 } |
| 51 | 51 |
| 52 int32_t BindSurfaces(PP_Resource context, | 52 int32_t BindSurfaces(PP_Resource context_id, |
| 53 PP_Resource draw, | 53 PP_Resource draw, |
| 54 PP_Resource read) { | 54 PP_Resource read) { |
| 55 // TODO(alokp): Implement me. | 55 scoped_refptr<PPB_Context3D_Impl> context( |
| 56 return 0; | 56 Resource::GetAs<PPB_Context3D_Impl>(context_id)); |
| 57 if (!context.get()) |
| 58 return PP_ERROR_BADRESOURCE; |
| 59 |
| 60 scoped_refptr<PPB_Surface3D_Impl> draw_surface( |
| 61 Resource::GetAs<PPB_Surface3D_Impl>(draw)); |
| 62 if (!draw_surface.get()) |
| 63 return PP_ERROR_BADRESOURCE; |
| 64 |
| 65 scoped_refptr<PPB_Surface3D_Impl> read_surface( |
| 66 Resource::GetAs<PPB_Surface3D_Impl>(read)); |
| 67 if (!read_surface.get()) |
| 68 return PP_ERROR_BADRESOURCE; |
| 69 |
| 70 return context->BindSurfaces(draw_surface.get(), read_surface.get()); |
| 57 } | 71 } |
| 58 | 72 |
| 59 int32_t GetBoundSurfaces(PP_Resource context, | 73 int32_t GetBoundSurfaces(PP_Resource context, |
| 60 PP_Resource* draw, | 74 PP_Resource* draw, |
| 61 PP_Resource* read) { | 75 PP_Resource* read) { |
| 62 // TODO(alokp): Implement me. | 76 // TODO(alokp): Implement me. |
| 63 return 0; | 77 return 0; |
| 64 } | 78 } |
| 65 | 79 |
| 66 int32_t SwapBuffers(PP_Resource context_id, | |
| 67 PP_CompletionCallback callback) { | |
| 68 scoped_refptr<PPB_Context3D_Impl> context( | |
| 69 Resource::GetAs<PPB_Context3D_Impl>(context_id)); | |
| 70 return context->SwapBuffers(); | |
| 71 } | |
| 72 | |
| 73 const PPB_Context3D_Dev ppb_context3d = { | 80 const PPB_Context3D_Dev ppb_context3d = { |
| 74 &Create, | 81 &Create, |
| 75 &IsContext3D, | 82 &IsContext3D, |
| 76 &GetAttrib, | 83 &GetAttrib, |
| 77 &BindSurfaces, | 84 &BindSurfaces, |
| 78 &GetBoundSurfaces, | 85 &GetBoundSurfaces, |
| 79 &SwapBuffers | |
| 80 }; | 86 }; |
| 81 | 87 |
| 82 } // namespace | 88 } // namespace |
| 83 | 89 |
| 84 PPB_Context3D_Impl::PPB_Context3D_Impl(PluginModule* module) | 90 PPB_Context3D_Impl::PPB_Context3D_Impl(PluginInstance* instance) |
| 85 : Resource(module), | 91 : Resource(instance->module()), |
| 86 bound_instance_(NULL), | 92 instance_(instance), |
| 87 gles2_impl_(NULL) { | 93 gles2_impl_(NULL), |
| 94 draw_surface_(NULL), |
| 95 read_surface_(NULL) { |
| 88 } | 96 } |
| 89 | 97 |
| 90 PPB_Context3D_Impl::~PPB_Context3D_Impl() { | 98 PPB_Context3D_Impl::~PPB_Context3D_Impl() { |
| 91 Destroy(); | 99 Destroy(); |
| 92 } | 100 } |
| 93 | 101 |
| 94 const PPB_Context3D_Dev* PPB_Context3D_Impl::GetInterface() { | 102 const PPB_Context3D_Dev* PPB_Context3D_Impl::GetInterface() { |
| 95 return &ppb_context3d; | 103 return &ppb_context3d; |
| 96 } | 104 } |
| 97 | 105 |
| 98 PPB_Context3D_Impl* PPB_Context3D_Impl::AsPPB_Context3D_Impl() { | 106 PPB_Context3D_Impl* PPB_Context3D_Impl::AsPPB_Context3D_Impl() { |
| 99 return this; | 107 return this; |
| 100 } | 108 } |
| 101 | 109 |
| 102 bool PPB_Context3D_Impl::Init(PluginInstance* instance, | 110 bool PPB_Context3D_Impl::Init(PP_Config3D_Dev config, |
| 103 PP_Config3D_Dev config, | |
| 104 PP_Resource share_context, | 111 PP_Resource share_context, |
| 105 const int32_t* attrib_list) { | 112 const int32_t* attrib_list) { |
| 106 DCHECK(instance); | |
| 107 // Create and initialize the objects required to issue GLES2 calls. | 113 // Create and initialize the objects required to issue GLES2 calls. |
| 108 platform_context_.reset(instance->delegate()->CreateContext3D()); | 114 platform_context_.reset(instance()->delegate()->CreateContext3D()); |
| 109 if (!platform_context_.get()) { | 115 if (!platform_context_.get()) { |
| 110 Destroy(); | 116 Destroy(); |
| 111 return false; | 117 return false; |
| 112 } | 118 } |
| 113 if (!platform_context_->Init()) { | 119 if (!platform_context_->Init()) { |
| 114 Destroy(); | 120 Destroy(); |
| 115 return false; | 121 return false; |
| 116 } | 122 } |
| 117 | 123 |
| 118 gles2_impl_ = platform_context_->GetGLES2Implementation(); | 124 gles2_impl_ = platform_context_->GetGLES2Implementation(); |
| 119 DCHECK(gles2_impl_); | 125 DCHECK(gles2_impl_); |
| 120 | 126 |
| 121 return true; | 127 return true; |
| 122 } | 128 } |
| 123 | 129 |
| 124 bool PPB_Context3D_Impl::BindToInstance(PluginInstance* new_instance) { | 130 int32_t PPB_Context3D_Impl::BindSurfaces(PPB_Surface3D_Impl* draw, |
| 125 if (bound_instance_ == new_instance) | 131 PPB_Surface3D_Impl* read) { |
| 126 return true; // Rebinding the same device, nothing to do. | 132 // TODO(alokp): Support separate draw-read surfaces. |
| 127 if (bound_instance_ && new_instance) | 133 DCHECK_EQ(draw, read); |
| 128 return false; // Can't change a bound device. | 134 if (draw != read) |
| 135 return PP_GRAPHICS3DERROR_BAD_MATCH; |
| 129 | 136 |
| 130 if (new_instance) { | 137 if (draw == draw_surface_) |
| 131 // Resize the backing texture to the size of the instance when it is bound. | 138 return PP_OK; |
| 132 platform_context_->ResizeBackingTexture(new_instance->position().size()); | |
| 133 | 139 |
| 134 // This is a temporary hack. The SwapBuffers is issued to force the resize | 140 if (draw && draw->context()) |
| 135 // to take place before any subsequent rendering. This might lead to a | 141 return PP_GRAPHICS3DERROR_BAD_ACCESS; |
| 136 // partially rendered frame being displayed. It is also not thread safe | |
| 137 // since the SwapBuffers is written to the command buffer and that command | |
| 138 // buffer might be written to by another thread. | |
| 139 // TODO(apatrick): Figure out the semantics of binding and resizing. | |
| 140 platform_context_->SwapBuffers(); | |
| 141 } | |
| 142 | 142 |
| 143 bound_instance_ = new_instance; | 143 if (draw_surface_) |
| 144 return true; | 144 draw_surface_->BindToContext(NULL); |
| 145 } | 145 if (draw && !draw->BindToContext(platform_context_.get())) |
| 146 return PP_ERROR_NOMEMORY; |
| 146 | 147 |
| 147 bool PPB_Context3D_Impl::SwapBuffers() { | 148 draw_surface_ = draw; |
| 148 if (!platform_context_.get()) | 149 read_surface_ = read; |
| 149 return false; | 150 return PP_OK; |
| 150 | |
| 151 return platform_context_->SwapBuffers(); | |
| 152 } | |
| 153 | |
| 154 void PPB_Context3D_Impl::SetSwapBuffersCallback(Callback0::Type* callback) { | |
| 155 if (!platform_context_.get()) | |
| 156 return; | |
| 157 | |
| 158 platform_context_->SetSwapBuffersCallback(callback); | |
| 159 } | |
| 160 | |
| 161 unsigned int PPB_Context3D_Impl::GetBackingTextureId() { | |
| 162 if (!platform_context_.get()) | |
| 163 return 0; | |
| 164 | |
| 165 return platform_context_->GetBackingTextureId(); | |
| 166 } | |
| 167 | |
| 168 void PPB_Context3D_Impl::ResizeBackingTexture(const gfx::Size& size) { | |
| 169 if (!platform_context_.get()) | |
| 170 return; | |
| 171 | |
| 172 platform_context_->ResizeBackingTexture(size); | |
| 173 } | 151 } |
| 174 | 152 |
| 175 void PPB_Context3D_Impl::Destroy() { | 153 void PPB_Context3D_Impl::Destroy() { |
| 154 if (draw_surface_) |
| 155 draw_surface_->BindToContext(NULL); |
| 156 |
| 176 gles2_impl_ = NULL; | 157 gles2_impl_ = NULL; |
| 177 platform_context_.reset(); | 158 platform_context_.reset(); |
| 178 } | 159 } |
| 179 | 160 |
| 180 } // namespace ppapi | 161 } // namespace ppapi |
| 181 } // namespace webkit | 162 } // namespace webkit |
| 182 | 163 |
| OLD | NEW |