Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright 2017 The Chromium Authors. All rights reserved. | 1 // Copyright 2017 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 #if !defined(__has_feature) || !__has_feature(objc_arc) | 5 #if !defined(__has_feature) || !__has_feature(objc_arc) |
| 6 #error "This file requires ARC support." | 6 #error "This file requires ARC support." |
| 7 #endif | 7 #endif |
| 8 | 8 |
| 9 #include <array> | 9 #include <array> |
| 10 | 10 |
| 11 #import "remoting/client/ios/display/gl_display_handler.h" | 11 #import "remoting/client/ios/display/gl_display_handler.h" |
| 12 | 12 |
| 13 #import "base/mac/bind_objc_block.h" | 13 #import "base/mac/bind_objc_block.h" |
| 14 #import "remoting/client/display/sys_opengl.h" | 14 #import "remoting/client/display/sys_opengl.h" |
| 15 #import "remoting/client/ios/display/gl_demo_screen.h" | 15 #import "remoting/client/ios/display/gl_demo_screen.h" |
| 16 | 16 |
| 17 #include "base/bind.h" | 17 #include "base/bind.h" |
| 18 #include "base/macros.h" | 18 #include "base/macros.h" |
| 19 #include "base/memory/ptr_util.h" | 19 #include "base/memory/ptr_util.h" |
| 20 #include "base/memory/weak_ptr.h" | 20 #include "base/memory/weak_ptr.h" |
| 21 #include "remoting/client/chromoting_client_runtime.h" | 21 #include "remoting/client/chromoting_client_runtime.h" |
| 22 #include "remoting/client/cursor_shape_stub_proxy.h" | 22 #include "remoting/client/cursor_shape_stub_proxy.h" |
| 23 #include "remoting/client/display/gl_canvas.h" | 23 #include "remoting/client/display/gl_canvas.h" |
| 24 #include "remoting/client/display/gl_renderer.h" | 24 #include "remoting/client/display/gl_renderer.h" |
| 25 #include "remoting/client/display/gl_renderer_delegate.h" | 25 #include "remoting/client/display/gl_renderer_delegate.h" |
| 26 #include "remoting/client/dual_buffer_frame_consumer.h" | 26 #include "remoting/client/dual_buffer_frame_consumer.h" |
| 27 #include "remoting/client/queued_task_poster.h" | |
| 27 #include "remoting/client/software_video_renderer.h" | 28 #include "remoting/client/software_video_renderer.h" |
| 28 | 29 |
| 29 namespace remoting { | 30 namespace remoting { |
| 30 namespace GlDisplayHandler { | 31 namespace GlDisplayHandler { |
| 31 | 32 |
| 32 // The core that lives on the display thread. | 33 // The core that lives on the display thread. |
| 33 class Core : public protocol::CursorShapeStub, public GlRendererDelegate { | 34 class Core : public protocol::CursorShapeStub, public GlRendererDelegate { |
| 34 public: | 35 public: |
| 35 Core(); | 36 Core(); |
| 36 ~Core() override; | 37 ~Core() override; |
| 37 | 38 |
| 38 void Initialize(); | 39 void Initialize(); |
| 39 | 40 |
| 41 void SetHandlerDelegate(id<GlDisplayHandlerDelegate> delegate); | |
| 42 | |
| 40 // CursorShapeStub interface. | 43 // CursorShapeStub interface. |
| 41 void SetCursorShape(const protocol::CursorShapeInfo& cursor_shape) override; | 44 void SetCursorShape(const protocol::CursorShapeInfo& cursor_shape) override; |
| 42 | 45 |
| 43 // GlRendererDelegate interface. | 46 // GlRendererDelegate interface. |
| 44 bool CanRenderFrame() override; | 47 bool CanRenderFrame() override; |
| 45 void OnFrameRendered() override; | 48 void OnFrameRendered() override; |
| 46 void OnSizeChanged(int width, int height) override; | 49 void OnSizeChanged(int width, int height) override; |
| 47 | 50 |
| 48 void OnFrameReceived(std::unique_ptr<webrtc::DesktopFrame> frame, | 51 void OnFrameReceived(std::unique_ptr<webrtc::DesktopFrame> frame, |
| 49 const base::Closure& done); | 52 const base::Closure& done); |
| 50 void Stop(); | 53 void Stop(); |
| 51 void SurfaceCreated(GLKView* view); | 54 void SurfaceCreated(GLKView* view); |
| 52 void SurfaceChanged(int width, int height); | 55 void SurfaceChanged(int width, int height); |
| 56 void SetTransformation(const remoting::ViewMatrix& matrix); | |
| 53 std::unique_ptr<protocol::FrameConsumer> GrabFrameConsumer(); | 57 std::unique_ptr<protocol::FrameConsumer> GrabFrameConsumer(); |
| 54 EAGLContext* GetEAGLContext(); | 58 EAGLContext* GetEAGLContext(); |
| 55 base::WeakPtr<Core> GetWeakPtr(); | 59 base::WeakPtr<Core> GetWeakPtr(); |
| 56 | 60 |
| 57 private: | 61 private: |
| 58 remoting::ChromotingClientRuntime* runtime_; | 62 remoting::ChromotingClientRuntime* runtime_; |
| 59 | 63 |
| 60 // Will be std::move'd when GrabFrameConsumer() is called. | 64 // Will be std::move'd when GrabFrameConsumer() is called. |
| 61 std::unique_ptr<DualBufferFrameConsumer> owned_frame_consumer_; | 65 std::unique_ptr<DualBufferFrameConsumer> owned_frame_consumer_; |
| 62 base::WeakPtr<DualBufferFrameConsumer> frame_consumer_; | 66 base::WeakPtr<DualBufferFrameConsumer> frame_consumer_; |
| 63 | 67 |
| 68 // TODO(yuweih): Release references once the surface is destroyed. | |
| 64 GLKView* gl_view_; | 69 GLKView* gl_view_; |
| 65 EAGLContext* eagl_context_; | 70 EAGLContext* eagl_context_; |
| 66 std::unique_ptr<GlRenderer> renderer_; | 71 std::unique_ptr<GlRenderer> renderer_; |
| 67 // GlDemoScreen *demo_screen_; | 72 // GlDemoScreen *demo_screen_; |
| 73 id<GlDisplayHandlerDelegate> handler_delegate_; | |
| 68 | 74 |
| 69 // Used on display thread. | 75 // Used on display thread. |
| 70 base::WeakPtr<Core> weak_ptr_; | 76 base::WeakPtr<Core> weak_ptr_; |
| 71 base::WeakPtrFactory<Core> weak_factory_; | 77 base::WeakPtrFactory<Core> weak_factory_; |
| 72 | 78 |
| 73 DISALLOW_COPY_AND_ASSIGN(Core); | 79 DISALLOW_COPY_AND_ASSIGN(Core); |
| 74 }; | 80 }; |
| 75 | 81 |
| 76 Core::Core() : weak_factory_(this) { | 82 Core::Core() : weak_factory_(this) { |
| 77 runtime_ = ChromotingClientRuntime::GetInstance(); | 83 runtime_ = ChromotingClientRuntime::GetInstance(); |
| (...skipping 30 matching lines...) Expand all Loading... | |
| 108 | 114 |
| 109 renderer_ = remoting::GlRenderer::CreateGlRendererWithDesktop(); | 115 renderer_ = remoting::GlRenderer::CreateGlRendererWithDesktop(); |
| 110 | 116 |
| 111 // renderer_.RequestCanvasSize(); | 117 // renderer_.RequestCanvasSize(); |
| 112 | 118 |
| 113 // demo_screen_ = new GlDemoScreen(); | 119 // demo_screen_ = new GlDemoScreen(); |
| 114 // renderer_->AddDrawable(demo_screen_->GetWeakPtr()); | 120 // renderer_->AddDrawable(demo_screen_->GetWeakPtr()); |
| 115 renderer_->SetDelegate(weak_ptr_); | 121 renderer_->SetDelegate(weak_ptr_); |
| 116 } | 122 } |
| 117 | 123 |
| 124 void Core::SetHandlerDelegate(id<GlDisplayHandlerDelegate> delegate) { | |
| 125 DCHECK(runtime_->display_task_runner()->BelongsToCurrentThread()); | |
| 126 handler_delegate_ = delegate; | |
| 127 } | |
| 128 | |
| 118 void Core::SetCursorShape(const protocol::CursorShapeInfo& cursor_shape) { | 129 void Core::SetCursorShape(const protocol::CursorShapeInfo& cursor_shape) { |
| 119 DCHECK(runtime_->display_task_runner()->BelongsToCurrentThread()); | 130 DCHECK(runtime_->display_task_runner()->BelongsToCurrentThread()); |
| 120 renderer_->OnCursorShapeChanged(cursor_shape); | 131 renderer_->OnCursorShapeChanged(cursor_shape); |
| 121 } | 132 } |
| 122 | 133 |
| 123 bool Core::CanRenderFrame() { | 134 bool Core::CanRenderFrame() { |
| 124 DCHECK(runtime_->display_task_runner()->BelongsToCurrentThread()); | 135 DCHECK(runtime_->display_task_runner()->BelongsToCurrentThread()); |
| 125 return gl_view_ != NULL && eagl_context_ != NULL; | 136 return gl_view_ != NULL && eagl_context_ != NULL; |
| 126 } | 137 } |
| 127 | 138 |
| 128 std::unique_ptr<protocol::FrameConsumer> Core::GrabFrameConsumer() { | 139 std::unique_ptr<protocol::FrameConsumer> Core::GrabFrameConsumer() { |
| 129 DCHECK(owned_frame_consumer_) << "The frame consumer is already grabbed."; | 140 DCHECK(owned_frame_consumer_) << "The frame consumer is already grabbed."; |
| 130 return std::move(owned_frame_consumer_); | 141 return std::move(owned_frame_consumer_); |
| 131 } | 142 } |
| 132 | 143 |
| 133 void Core::OnFrameReceived(std::unique_ptr<webrtc::DesktopFrame> frame, | 144 void Core::OnFrameReceived(std::unique_ptr<webrtc::DesktopFrame> frame, |
| 134 const base::Closure& done) { | 145 const base::Closure& done) { |
| 135 DCHECK(runtime_->display_task_runner()->BelongsToCurrentThread()); | 146 DCHECK(runtime_->display_task_runner()->BelongsToCurrentThread()); |
| 136 renderer_->OnFrameReceived(std::move(frame), done); | 147 renderer_->OnFrameReceived(std::move(frame), done); |
| 137 } | 148 } |
| 138 | 149 |
| 139 void Core::OnFrameRendered() { | 150 void Core::OnFrameRendered() { |
| 140 [gl_view_ setNeedsDisplay]; | 151 [gl_view_ display]; |
| 141 } | 152 } |
| 142 | 153 |
| 143 void Core::OnSizeChanged(int width, int height) { | 154 void Core::OnSizeChanged(int width, int height) { |
| 144 // Nothing to do. | 155 DCHECK(runtime_->display_task_runner()->BelongsToCurrentThread()); |
| 156 runtime_->ui_task_runner()->PostTask( | |
| 157 FROM_HERE, base::BindBlockArc(^() { | |
| 158 [handler_delegate_ canvasSizeChanged:CGSizeMake(width, height)]; | |
| 159 })); | |
| 145 } | 160 } |
| 146 | 161 |
| 147 void Core::Stop() { | 162 void Core::Stop() { |
| 148 DCHECK(runtime_->display_task_runner()->BelongsToCurrentThread()); | 163 DCHECK(runtime_->display_task_runner()->BelongsToCurrentThread()); |
| 149 | 164 |
| 150 eagl_context_ = nil; | 165 eagl_context_ = nil; |
| 151 // demo_screen_ = nil; | 166 // demo_screen_ = nil; |
| 152 } | 167 } |
| 153 | 168 |
| 154 void Core::SurfaceCreated(GLKView* view) { | 169 void Core::SurfaceCreated(GLKView* view) { |
| 155 DCHECK(runtime_->display_task_runner()->BelongsToCurrentThread()); | 170 DCHECK(runtime_->display_task_runner()->BelongsToCurrentThread()); |
| 156 gl_view_ = view; | 171 gl_view_ = view; |
| 157 | 172 |
| 158 renderer_->OnSurfaceCreated( | 173 renderer_->OnSurfaceCreated( |
| 159 base::MakeUnique<GlCanvas>(static_cast<int>([eagl_context_ API]))); | 174 base::MakeUnique<GlCanvas>(static_cast<int>([eagl_context_ API]))); |
| 160 | 175 |
| 176 renderer_->RequestCanvasSize(); | |
| 177 | |
| 161 runtime_->network_task_runner()->PostTask( | 178 runtime_->network_task_runner()->PostTask( |
| 162 FROM_HERE, base::Bind(&DualBufferFrameConsumer::RequestFullDesktopFrame, | 179 FROM_HERE, base::Bind(&DualBufferFrameConsumer::RequestFullDesktopFrame, |
| 163 frame_consumer_)); | 180 frame_consumer_)); |
| 164 } | 181 } |
| 165 | 182 |
| 166 void Core::SurfaceChanged(int width, int height) { | 183 void Core::SurfaceChanged(int width, int height) { |
| 167 DCHECK(runtime_->display_task_runner()->BelongsToCurrentThread()); | 184 DCHECK(runtime_->display_task_runner()->BelongsToCurrentThread()); |
| 168 renderer_->OnSurfaceChanged(width, height); | 185 renderer_->OnSurfaceChanged(width, height); |
| 186 } | |
| 169 | 187 |
| 170 // TODO(nicholss): This are wrong values but it lets us get something on the | 188 void Core::SetTransformation(const remoting::ViewMatrix& matrix) { |
| 171 // screen. | 189 DCHECK(runtime_->display_task_runner()->BelongsToCurrentThread()); |
| 172 std::array<float, 9> matrix = {{1, 0, 0, // Row 1 | 190 renderer_->OnPixelTransformationChanged(matrix.ToMatrixArray()); |
| 173 0, 1, 0, // Row 2 | |
| 174 0, 0, 1}}; | |
| 175 | |
| 176 renderer_->OnPixelTransformationChanged(matrix); | |
| 177 } | 191 } |
| 178 | 192 |
| 179 EAGLContext* Core::GetEAGLContext() { | 193 EAGLContext* Core::GetEAGLContext() { |
| 180 return eagl_context_; | 194 return eagl_context_; |
| 181 } | 195 } |
| 182 | 196 |
| 183 base::WeakPtr<remoting::GlDisplayHandler::Core> Core::GetWeakPtr() { | 197 base::WeakPtr<remoting::GlDisplayHandler::Core> Core::GetWeakPtr() { |
| 184 return weak_ptr_; | 198 return weak_ptr_; |
| 185 } | 199 } |
| 186 | 200 |
| 187 } // namespace GlDisplayHandler | 201 } // namespace GlDisplayHandler |
| 188 } // namespace remoting | 202 } // namespace remoting |
| 189 | 203 |
| 190 @interface GlDisplayHandler () { | 204 @interface GlDisplayHandler () { |
| 191 remoting::GlDisplayHandler::Core* _core; | 205 remoting::GlDisplayHandler::Core* _core; |
| 192 remoting::ChromotingClientRuntime* _runtime; | 206 remoting::ChromotingClientRuntime* _runtime; |
| 207 std::unique_ptr<remoting::QueuedTaskPoster> _uiTaskPoster; | |
| 193 } | 208 } |
| 194 @end | 209 @end |
| 195 | 210 |
| 196 @implementation GlDisplayHandler | 211 @implementation GlDisplayHandler |
| 197 | 212 |
| 198 - (id)init { | 213 - (id)init { |
| 199 self = [super init]; | 214 self = [super init]; |
| 200 if (self) { | 215 if (self) { |
| 201 _runtime = remoting::ChromotingClientRuntime::GetInstance(); | 216 _runtime = remoting::ChromotingClientRuntime::GetInstance(); |
| 202 _core = new remoting::GlDisplayHandler::Core(); | 217 _core = new remoting::GlDisplayHandler::Core(); |
| 218 _uiTaskPoster.reset( | |
| 219 new remoting::QueuedTaskPoster(_runtime->display_task_runner())); | |
| 203 } | 220 } |
| 204 return self; | 221 return self; |
| 205 } | 222 } |
| 206 | 223 |
| 207 #pragma mark - Public | 224 #pragma mark - Public |
| 208 | 225 |
| 209 - (void)stop { | 226 - (void)stop { |
| 210 _runtime->display_task_runner()->PostTask( | 227 _runtime->display_task_runner()->PostTask( |
| 211 FROM_HERE, | 228 FROM_HERE, |
| 212 base::Bind(&remoting::GlDisplayHandler::Core::Stop, _core->GetWeakPtr())); | 229 base::Bind(&remoting::GlDisplayHandler::Core::Stop, _core->GetWeakPtr())); |
| 213 } | 230 } |
| 214 | 231 |
| 232 - (void)setDelegate:(id<GlDisplayHandlerDelegate>)delegate { | |
|
nicholss
2017/05/03 22:47:35
You can keep this here, this will override the aut
Yuwei
2017/05/04 00:28:07
Done. Moved below.
| |
| 233 _runtime->display_task_runner()->PostTask( | |
| 234 FROM_HERE, | |
| 235 base::Bind(&remoting::GlDisplayHandler::Core::SetHandlerDelegate, | |
| 236 _core->GetWeakPtr(), delegate)); | |
| 237 } | |
| 238 | |
| 215 - (std::unique_ptr<remoting::protocol::VideoRenderer>)CreateVideoRenderer { | 239 - (std::unique_ptr<remoting::protocol::VideoRenderer>)CreateVideoRenderer { |
| 216 return base::MakeUnique<remoting::SoftwareVideoRenderer>( | 240 return base::MakeUnique<remoting::SoftwareVideoRenderer>( |
| 217 _core->GrabFrameConsumer()); | 241 _core->GrabFrameConsumer()); |
| 218 } | 242 } |
| 219 | 243 |
| 220 - (std::unique_ptr<remoting::protocol::CursorShapeStub>)CreateCursorShapeStub { | 244 - (std::unique_ptr<remoting::protocol::CursorShapeStub>)CreateCursorShapeStub { |
| 221 return base::MakeUnique<remoting::CursorShapeStubProxy>( | 245 return base::MakeUnique<remoting::CursorShapeStubProxy>( |
| 222 _core->GetWeakPtr(), _runtime->display_task_runner()); | 246 _core->GetWeakPtr(), _runtime->display_task_runner()); |
| 223 } | 247 } |
| 224 | 248 |
| 225 - (EAGLContext*)GetEAGLContext { | 249 - (EAGLContext*)GetEAGLContext { |
| 226 return _core->GetEAGLContext(); | 250 return _core->GetEAGLContext(); |
| 227 } | 251 } |
| 228 | 252 |
| 229 - (void)onSurfaceCreated:(GLKView*)view { | 253 - (void)onSurfaceCreated:(GLKView*)view { |
| 230 _runtime->display_task_runner()->PostTask( | 254 _runtime->display_task_runner()->PostTask( |
| 231 FROM_HERE, base::Bind(&remoting::GlDisplayHandler::Core::SurfaceCreated, | 255 FROM_HERE, base::Bind(&remoting::GlDisplayHandler::Core::SurfaceCreated, |
| 232 _core->GetWeakPtr(), view)); | 256 _core->GetWeakPtr(), view)); |
| 233 } | 257 } |
| 234 | 258 |
| 235 - (void)onSurfaceChanged:(const CGRect&)frame { | 259 - (void)onSurfaceChanged:(const CGRect&)frame { |
| 236 _runtime->display_task_runner()->PostTask( | 260 _runtime->display_task_runner()->PostTask( |
| 237 FROM_HERE, | 261 FROM_HERE, |
| 238 base::Bind(&remoting::GlDisplayHandler::Core::SurfaceChanged, | 262 base::Bind(&remoting::GlDisplayHandler::Core::SurfaceChanged, |
| 239 _core->GetWeakPtr(), frame.size.width, frame.size.height)); | 263 _core->GetWeakPtr(), frame.size.width, frame.size.height)); |
| 240 } | 264 } |
| 241 | 265 |
| 266 - (void)onPixelTransformationChanged:(const remoting::ViewMatrix&)matrix { | |
| 267 _uiTaskPoster->AddTask( | |
|
nicholss
2017/05/03 22:47:35
How do you choose to AddTask on the task poster vs
Yuwei
2017/05/04 00:28:07
UITaskPoster is used to synchronize UI events that
| |
| 268 base::Bind(&remoting::GlDisplayHandler::Core::SetTransformation, | |
| 269 _core->GetWeakPtr(), matrix)); | |
| 270 } | |
| 271 | |
| 242 // TODO(nicholss): Remove this function, it is not used in the final impl, | 272 // TODO(nicholss): Remove this function, it is not used in the final impl, |
| 243 // or it should call RequestRender. | 273 // or it should call RequestRender. |
| 244 - (void)glkView:(GLKView*)view drawInRect:(CGRect)rect { | 274 - (void)glkView:(GLKView*)view drawInRect:(CGRect)rect { |
|
nicholss
2017/05/03 22:47:35
Can you delete this? It is not used, just cruft I
Yuwei
2017/05/04 00:28:07
Done.
| |
| 245 if (_core) { | 275 if (_core) { |
| 246 _runtime->display_task_runner()->PostTask( | 276 _runtime->display_task_runner()->PostTask( |
| 247 FROM_HERE, | 277 FROM_HERE, |
| 248 base::Bind(&remoting::GlDisplayHandler::Core::SurfaceChanged, | 278 base::Bind(&remoting::GlDisplayHandler::Core::SurfaceChanged, |
| 249 _core->GetWeakPtr(), rect.size.width, rect.size.height)); | 279 _core->GetWeakPtr(), rect.size.width, rect.size.height)); |
| 250 } | 280 } |
| 251 } | 281 } |
| 252 | 282 |
| 253 @end | 283 @end |
| OLD | NEW |