Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(176)

Side by Side Diff: media/capture/content/android/screen_capture_machine_android.cc

Issue 1917023003: ScreenCapture for Android phase1, part I (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: address review comments Created 4 years, 6 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« no previous file with comments | « media/capture/content/android/screen_capture_machine_android.h ('k') | media/media.gyp » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
(Empty)
1 // Copyright 2016 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "media/capture/content/android/screen_capture_machine_android.h"
6
7 #include "base/android/context_utils.h"
8 #include "base/android/jni_android.h"
9 #include "base/android/scoped_java_ref.h"
10 #include "jni/ScreenCapture_jni.h"
11 #include "media/base/video_capture_types.h"
12 #include "media/base/yuv_convert.h"
13 #include "media/capture/content/video_capture_oracle.h"
14 #include "third_party/libyuv/include/libyuv.h"
15
16 using base::android::AttachCurrentThread;
17
18 namespace media {
19
20 // static
21 bool ScreenCaptureMachineAndroid::RegisterScreenCaptureMachine(JNIEnv* env) {
22 return RegisterNativesImpl(env);
23 }
24
25 ScreenCaptureMachineAndroid::ScreenCaptureMachineAndroid() {}
26
27 ScreenCaptureMachineAndroid::~ScreenCaptureMachineAndroid() {}
28
29 // static
30 ScopedJavaLocalRef<jobject>
31 ScreenCaptureMachineAndroid::createScreenCaptureMachineAndroid(
32 jlong nativeScreenCaptureMachineAndroid) {
33 return (Java_ScreenCapture_createScreenCaptureMachine(
34 AttachCurrentThread(), base::android::GetApplicationContext(),
35 nativeScreenCaptureMachineAndroid));
36 }
37
38 void ScreenCaptureMachineAndroid::OnRGBAFrameAvailable(JNIEnv* env,
39 jobject obj,
40 jobject buf,
41 jint row_stride,
42 jint left,
43 jint top,
44 jint width,
45 jint height,
46 jlong timestamp) {
47 scoped_refptr<VideoFrame> i420_frame = VideoFrame::CreateFrame(
48 PIXEL_FORMAT_I420, gfx::Size(width, height), gfx::Rect(width, height),
49 gfx::Size(width, height), base::TimeDelta());
50
51 int offset = top * row_stride + left * 4;
mcasas 2016/06/12 10:11:51 const
braveyao 2016/06/13 22:30:32 Done.
52 // ABGR little endian (rgba in memory) to I420.
53 libyuv::ABGRToI420(
54 reinterpret_cast<uint8_t*>(env->GetDirectBufferAddress(buf)) + offset,
55 row_stride, i420_frame->visible_data(VideoFrame::kYPlane),
56 i420_frame->stride(VideoFrame::kYPlane),
57 i420_frame->visible_data(VideoFrame::kUPlane),
58 i420_frame->stride(VideoFrame::kUPlane),
59 i420_frame->visible_data(VideoFrame::kVPlane),
60 i420_frame->stride(VideoFrame::kVPlane),
61 i420_frame->visible_rect().width(), i420_frame->visible_rect().height());
62
63 OnIncomingFrameAvailable(i420_frame->visible_data(VideoFrame::kYPlane),
64 i420_frame->stride(VideoFrame::kYPlane),
65 i420_frame->visible_data(VideoFrame::kUPlane),
66 i420_frame->stride(VideoFrame::kUPlane),
67 i420_frame->visible_data(VideoFrame::kVPlane),
68 i420_frame->stride(VideoFrame::kVPlane),
69 i420_frame->visible_rect().width(),
70 i420_frame->visible_rect().height(),
71 static_cast<int64_t>(timestamp));
72 }
73
74 void ScreenCaptureMachineAndroid::OnI420FrameAvailable(JNIEnv* env,
75 jobject obj,
76 jobject y_buffer,
77 jint y_stride,
78 jobject u_buffer,
79 jobject v_buffer,
80 jint uv_row_stride,
81 jint uv_pixel_stride,
82 jint left,
83 jint top,
84 jint width,
85 jint height,
86 jlong timestamp) {
87 uint8_t* y_src =
mcasas 2016/06/12 10:11:51 uint8_t* const
braveyao 2016/06/13 22:30:32 Done.
mcasas 2016/06/15 15:19:13 I don't see the "Done"s being done. Did you forget
88 reinterpret_cast<uint8_t*>(env->GetDirectBufferAddress(y_buffer));
89 CHECK(y_src);
mcasas 2016/06/12 10:11:50 Why CHECK here and take it face-value in l.54?
braveyao 2016/06/13 22:30:32 Done.
90 uint8_t* u_src =
91 reinterpret_cast<uint8_t*>(env->GetDirectBufferAddress(u_buffer));
mcasas 2016/06/12 10:11:51 uint8_t* const
braveyao 2016/06/13 22:30:32 can't be const. It will be changed below.
92 CHECK(u_src);
93 uint8_t* v_src =
94 reinterpret_cast<uint8_t*>(env->GetDirectBufferAddress(v_buffer));
mcasas 2016/06/12 10:11:51 uint8_t* const
braveyao 2016/06/13 22:30:32 ditto
95 CHECK(v_src);
96
97 // De-interleave the U and V planes into temporary buffers, if needed.
98 int uv_stride = uv_row_stride;
99 std::unique_ptr<uint8_t[]> u_tmp, v_tmp;
100 if (uv_pixel_stride != 1) {
101 // U and V planes are actually interleaved, unpack them here.
102 int uv_plane_len = (int)env->GetDirectBufferCapacity(u_buffer);
103 u_tmp.reset(new uint8_t[(uv_plane_len + 1) / uv_pixel_stride]);
104 v_tmp.reset(new uint8_t[(uv_plane_len + 1) / uv_pixel_stride]);
105 for (int index = 0; index * uv_pixel_stride <= uv_plane_len; index++) {
106 u_tmp[index] = u_src[index * uv_pixel_stride];
107 v_tmp[index] = v_src[index * uv_pixel_stride];
108 }
109 u_src = u_tmp.get();
110 v_src = v_tmp.get();
111 uv_stride /= uv_pixel_stride;
112 }
113
114 int y_offset = top * y_stride + left;
115 int uv_offset = (top / 2) * uv_stride + left / 2;
mcasas 2016/06/15 15:19:13 l.114 and this line should be const.
braveyao 2016/06/15 21:56:36 Done.
116 OnIncomingFrameAvailable(y_src + y_offset, y_stride, u_src + uv_offset,
117 uv_stride, v_src + uv_offset, uv_stride, width,
118 height, static_cast<int64_t>(timestamp));
119 }
120
121 void ScreenCaptureMachineAndroid::OnActivityResult(JNIEnv* env,
122 jobject obj,
123 jboolean result) {
124 if (!result) {
125 oracle_proxy_->ReportError(FROM_HERE, "The user denied screen capture");
126 return;
127 }
128
129 Java_ScreenCapture_startCapture(env, obj);
130 }
131
132 void ScreenCaptureMachineAndroid::Start(
133 const scoped_refptr<ThreadSafeCaptureOracle>& oracle_proxy,
134 const VideoCaptureParams& params,
135 const base::Callback<void(bool)> callback) {
136 DCHECK(oracle_proxy.get());
137 oracle_proxy_ = oracle_proxy;
138
139 j_capture_.Reset(
140 createScreenCaptureMachineAndroid(reinterpret_cast<intptr_t>(this)));
141
142 if (j_capture_.obj() == nullptr) {
143 LOG(ERROR) << "Failed to createScreenCaptureAndroid,"
144 " Maybe android version is too low";
mcasas 2016/06/12 10:11:50 DLOG() and, I think this error message is not rea
braveyao 2016/06/13 22:30:32 Done.
145 callback.Run(false);
146 return;
147 }
148
149 DCHECK(params.requested_format.frame_size.GetArea());
150 DCHECK(!(params.requested_format.frame_size.width() % 2));
151 DCHECK(!(params.requested_format.frame_size.height() % 2));
152
153 const jboolean ret = Java_ScreenCapture_startPrompt(
154 AttachCurrentThread(), j_capture_.obj(),
155 params.requested_format.frame_size.width(),
156 params.requested_format.frame_size.height());
157
158 callback.Run(ret);
159 }
160
161 void ScreenCaptureMachineAndroid::Stop(const base::Closure& callback) {
162 Java_ScreenCapture_stopCapture(AttachCurrentThread(), j_capture_.obj());
163
164 callback.Run();
165 }
166
167 // ScreenCapture on Android works in passive way and there will be no captured
168 // frame when there is no update to the screen. When oracle asks capturing for
169 // refresh, the cached last frame will be delivered.
mcasas 2016/06/12 10:11:50 Strange reading. Suggestions: ... in _a_ passive
braveyao 2016/06/13 22:30:32 Done.
170 void ScreenCaptureMachineAndroid::MaybeCaptureForRefresh() {
171 if (lastFrame_.get() == nullptr)
172 return;
173
174 OnIncomingFrameAvailable(lastFrame_->visible_data(VideoFrame::kYPlane),
175 lastFrame_->stride(VideoFrame::kYPlane),
176 lastFrame_->visible_data(VideoFrame::kUPlane),
177 lastFrame_->stride(VideoFrame::kUPlane),
178 lastFrame_->visible_data(VideoFrame::kVPlane),
179 lastFrame_->stride(VideoFrame::kVPlane),
180 lastFrame_->visible_rect().width(),
181 lastFrame_->visible_rect().height(),
182 base::TimeTicks::Now().ToInternalValue());
183 }
184
185 void ScreenCaptureMachineAndroid::OnIncomingFrameAvailable(const uint8_t* y_src,
186 int y_stride,
187 const uint8_t* u_src,
188 int u_stride,
189 const uint8_t* v_src,
190 int v_stride,
191 int width,
192 int height,
193 int64_t timestamp) {
194 const VideoCaptureOracle::Event event = VideoCaptureOracle::kCompositorUpdate;
195 const uint64_t absolute_micro =
196 timestamp / base::Time::kNanosecondsPerMicrosecond;
197 const base::TimeTicks start_time =
198 base::TimeTicks() + base::TimeDelta::FromMicroseconds(absolute_micro);
199 scoped_refptr<VideoFrame> frame;
200 ThreadSafeCaptureOracle::CaptureFrameCallback capture_frame_cb;
201
202 if (!oracle_proxy_->ObserveEventAndDecideCapture(
203 event, gfx::Rect(), start_time, &frame, &capture_frame_cb)) {
204 return;
205 }
206
207 DCHECK(frame->format() == PIXEL_FORMAT_I420 ||
208 frame->format() == PIXEL_FORMAT_YV12);
209
210 libyuv::I420Scale(y_src, y_stride, u_src, u_stride, v_src, v_stride, width,
mcasas 2016/06/12 10:11:51 I don't like that we have a step of libyuv::ABGRTo
braveyao 2016/06/13 22:30:32 Not that wasteful. As to OnRGBAFrameAvailable, I
mcasas 2016/06/15 15:19:13 What do you mean "for most of time" ? Regardless..
braveyao 2016/06/15 21:56:37 Done.
211 height, frame->visible_data(VideoFrame::kYPlane),
212 frame->stride(VideoFrame::kYPlane),
213 frame->visible_data(VideoFrame::kUPlane),
214 frame->stride(VideoFrame::kUPlane),
215 frame->visible_data(VideoFrame::kVPlane),
216 frame->stride(VideoFrame::kVPlane),
217 frame->visible_rect().width(),
218 frame->visible_rect().height(), libyuv::kFilterBilinear);
219
220 // Deliver the populated VideoFrame.
mcasas 2016/06/12 10:11:50 nit: superfluous comment?
braveyao 2016/06/13 22:30:32 Done.
221 capture_frame_cb.Run(frame, start_time, true);
222
223 lastFrame_ = frame;
224 }
225
226 } // namespace media
OLDNEW
« no previous file with comments | « media/capture/content/android/screen_capture_machine_android.h ('k') | media/media.gyp » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698