Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 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 "media/video/capture/screen/screen_capture_frame_queue.h" | 5 #include "media/video/capture/screen/screen_capture_frame_queue.h" |
| 6 | 6 |
| 7 #include <algorithm> | 7 #include <algorithm> |
| 8 | 8 |
| 9 #include "base/basictypes.h" | 9 #include "base/basictypes.h" |
| 10 #include "media/video/capture/screen/screen_capture_frame.h" | 10 #include "base/logging.h" |
| 11 #include "base/threading/non_thread_safe.h" | |
| 12 #include "third_party/webrtc/modules/desktop_capture/desktop_frame.h" | |
| 11 | 13 |
| 12 namespace media { | 14 namespace media { |
| 13 | 15 |
| 16 // EmittedFrame is a DesktopFrame wrapper that shares ownership of a | |
| 17 // DesktopFrame with the creating ScreenCaptureFrameQueue. If EmittedFrame is | |
| 18 // deleted before the queue is deleted then it returns the frame back to the | |
| 19 // queue. | |
| 20 class ScreenCaptureFrameQueue::EmittedFrame | |
|
alexeypa (please no reviews)
2013/05/08 22:24:59
How about allowing multiple EmittedFrame object to
Sergey Ulanov
2013/05/09 18:49:02
Problem is that if consumer tries to keep a frame
| |
| 21 : public webrtc::DesktopFrame, | |
| 22 public base::NonThreadSafe { | |
| 23 public: | |
| 24 EmittedFrame(webrtc::DesktopFrame* frame, | |
| 25 ScreenCaptureFrameQueue* queue, | |
| 26 int index) | |
| 27 : DesktopFrame(frame->size(), frame->stride(), frame->data(), | |
| 28 frame->shared_memory()), | |
| 29 frame_(frame), | |
| 30 queue_(queue), | |
| 31 index_(index) { | |
| 32 set_dpi(frame->dpi()); | |
| 33 set_capture_time_ms(frame->capture_time_ms()); | |
| 34 updated_region_ = frame->updated_region(); | |
| 35 } | |
| 36 | |
| 37 virtual ~EmittedFrame() { | |
| 38 if (queue_) | |
| 39 queue_->ReturnEmittedFrame(frame_.release(), index_); | |
|
alexeypa (please no reviews)
2013/05/08 22:24:59
Can the emitted frames be returned out of order? T
Sergey Ulanov
2013/05/09 18:49:02
Order in which frames are returned doesn't matter
| |
| 40 } | |
| 41 | |
| 42 void DetachFromQueue() { | |
|
alexeypa (please no reviews)
2013/05/08 22:24:59
nit: DetachFromQueue() is only called in a situati
Sergey Ulanov
2013/05/09 18:49:02
It's also called from queue destructor - for the c
| |
| 43 queue_ = NULL; | |
| 44 } | |
| 45 | |
| 46 private: | |
| 47 scoped_ptr<webrtc::DesktopFrame> frame_; | |
| 48 ScreenCaptureFrameQueue* queue_; | |
| 49 int index_; | |
| 50 }; | |
| 51 | |
| 14 ScreenCaptureFrameQueue::ScreenCaptureFrameQueue() | 52 ScreenCaptureFrameQueue::ScreenCaptureFrameQueue() |
| 15 : current_(0), | 53 : current_(0) { |
| 16 previous_(NULL) { | 54 memset(frames_, 0, sizeof(frames_)); |
| 17 SetAllFramesNeedUpdate(); | 55 memset(emitted_frames_, 0, sizeof(emitted_frames_)); |
| 18 } | 56 } |
| 19 | 57 |
| 20 ScreenCaptureFrameQueue::~ScreenCaptureFrameQueue() { | 58 ScreenCaptureFrameQueue::~ScreenCaptureFrameQueue() { |
| 59 for (int i = 0; i < kQueueLength; ++i) { | |
| 60 if (emitted_frames_[i]) { | |
| 61 emitted_frames_[i]->DetachFromQueue(); | |
| 62 } else { | |
| 63 delete frames_[i]; | |
| 64 } | |
| 65 } | |
| 21 } | 66 } |
| 22 | 67 |
| 23 void ScreenCaptureFrameQueue::DoneWithCurrentFrame() { | 68 void ScreenCaptureFrameQueue::MoveToNextFrame() { |
| 24 previous_ = current_frame(); | |
| 25 current_ = (current_ + 1) % kQueueLength; | 69 current_ = (current_ + 1) % kQueueLength; |
| 70 | |
| 71 // Verify that the frame has been released by the consumer before it tries to | |
| 72 // capture another one. | |
| 73 DCHECK(!emitted_frames_[current_]); | |
| 74 } | |
| 75 | |
| 76 webrtc::DesktopFrame* ScreenCaptureFrameQueue::EmitCurrentFrame() { | |
| 77 DCHECK(!emitted_frames_[current_]); | |
| 78 | |
| 79 // Wrap the frame with EmittedFrame. | |
| 80 EmittedFrame* result = new EmittedFrame(frames_[current_], this, current_); | |
| 81 emitted_frames_[current_] = result; | |
| 82 return result; | |
| 26 } | 83 } |
| 27 | 84 |
| 28 void ScreenCaptureFrameQueue::ReplaceCurrentFrame( | 85 void ScreenCaptureFrameQueue::ReplaceCurrentFrame( |
| 29 scoped_ptr<ScreenCaptureFrame> frame) { | 86 scoped_ptr<webrtc::DesktopFrame> frame) { |
| 30 frames_[current_] = frame.Pass(); | 87 DCHECK(!emitted_frames_[current_]); |
| 31 needs_update_[current_] = false; | 88 frames_[current_] = frame.release(); |
| 32 } | 89 } |
| 33 | 90 |
| 34 void ScreenCaptureFrameQueue::SetAllFramesNeedUpdate() { | 91 void ScreenCaptureFrameQueue::Reset() { |
| 35 std::fill(needs_update_, needs_update_ + arraysize(needs_update_), true); | 92 for (int i = 0; i < kQueueLength; ++i) { |
| 36 previous_ = NULL; | 93 if (emitted_frames_[i]) { |
| 94 emitted_frames_[i]->DetachFromQueue(); | |
| 95 emitted_frames_[i] = NULL; | |
| 96 } else { | |
| 97 delete frames_[i]; | |
| 98 } | |
| 99 frames_[i] = NULL; | |
| 100 } | |
| 101 } | |
| 102 | |
| 103 void ScreenCaptureFrameQueue::ReturnEmittedFrame(webrtc::DesktopFrame* frame, | |
| 104 int index) { | |
| 105 if (frame == frames_[index]) { | |
| 106 emitted_frames_[index] = NULL; | |
| 107 } else { | |
| 108 delete frame; | |
| 109 } | |
| 37 } | 110 } |
| 38 | 111 |
| 39 } // namespace media | 112 } // namespace media |
| OLD | NEW |