blob: fa4dcdaa60005d3a0b668e5886aff76a7cbffffc [file] [log] [blame]
andrew@webrtc.orgb015cbe2012-10-22 18:19:231/*
2 * Copyright (c) 2012 The WebRTC project authors. All Rights Reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
pbos@webrtc.org87c29b52013-07-12 10:03:5211#include "webrtc/modules/video_capture/android/video_capture_android.h"
andrew@webrtc.orgb015cbe2012-10-22 18:19:2312
fischman@webrtc.orgd3d25982014-06-06 20:13:4913#include "webrtc/base/common.h"
fischman@webrtc.org81cd5ca2013-10-03 18:23:1314#include "webrtc/modules/utility/interface/helpers_android.h"
15#include "webrtc/modules/video_capture/android/device_info_android.h"
pbos@webrtc.org87c29b52013-07-12 10:03:5216#include "webrtc/system_wrappers/interface/critical_section_wrapper.h"
fischman@webrtc.org81cd5ca2013-10-03 18:23:1317#include "webrtc/system_wrappers/interface/logcat_trace_context.h"
glaznev@webrtc.orgab52e9a2014-06-20 20:55:5418#include "webrtc/system_wrappers/interface/logging.h"
pbos@webrtc.org87c29b52013-07-12 10:03:5219#include "webrtc/system_wrappers/interface/ref_count.h"
20#include "webrtc/system_wrappers/interface/trace.h"
andrew@webrtc.orgb015cbe2012-10-22 18:19:2321
fischman@webrtc.org81cd5ca2013-10-03 18:23:1322static JavaVM* g_jvm = NULL;
23static jclass g_java_capturer_class = NULL; // VideoCaptureAndroid.class.
fischman@webrtc.org5101f842014-06-06 18:40:4424static jobject g_context = NULL; // Owned android.content.Context.
andrew@webrtc.orgb015cbe2012-10-22 18:19:2325
fischman@webrtc.org81cd5ca2013-10-03 18:23:1326namespace webrtc {
27
fischman@webrtc.org5101f842014-06-06 18:40:4428// Called by Java to get the global application context.
29jobject JNICALL GetContext(JNIEnv* env, jclass) {
30 assert(g_context);
31 return g_context;
32}
33
fischman@webrtc.org81cd5ca2013-10-03 18:23:1334// Called by Java when the camera has a new frame to deliver.
35void JNICALL ProvideCameraFrame(
36 JNIEnv* env,
37 jobject,
38 jbyteArray javaCameraFrame,
39 jint length,
glaznev@webrtc.orgaafd1e52014-10-17 16:25:0640 jint rotation,
glaznev@webrtc.orgab52e9a2014-06-20 20:55:5441 jlong timeStamp,
fischman@webrtc.org81cd5ca2013-10-03 18:23:1342 jlong context) {
43 webrtc::videocapturemodule::VideoCaptureAndroid* captureModule =
44 reinterpret_cast<webrtc::videocapturemodule::VideoCaptureAndroid*>(
45 context);
46 jbyte* cameraFrame = env->GetByteArrayElements(javaCameraFrame, NULL);
47 captureModule->OnIncomingFrame(
glaznev@webrtc.orgaafd1e52014-10-17 16:25:0648 reinterpret_cast<uint8_t*>(cameraFrame), length, rotation, 0);
fischman@webrtc.org81cd5ca2013-10-03 18:23:1349 env->ReleaseByteArrayElements(javaCameraFrame, cameraFrame, JNI_ABORT);
50}
51
fischman@webrtc.org5101f842014-06-06 18:40:4452int32_t SetCaptureAndroidVM(JavaVM* javaVM, jobject context) {
braveyao@webrtc.orgca2c70f2014-05-15 03:18:1553 if (javaVM) {
54 assert(!g_jvm);
55 g_jvm = javaVM;
56 AttachThreadScoped ats(g_jvm);
fischman@webrtc.org5101f842014-06-06 18:40:4457 g_context = ats.env()->NewGlobalRef(context);
fischman@webrtc.org81cd5ca2013-10-03 18:23:1358
braveyao@webrtc.orgca2c70f2014-05-15 03:18:1559 videocapturemodule::DeviceInfoAndroid::Initialize(ats.env());
fischman@webrtc.org81cd5ca2013-10-03 18:23:1360
braveyao@webrtc.orgca2c70f2014-05-15 03:18:1561 jclass j_capture_class =
62 ats.env()->FindClass("org/webrtc/videoengine/VideoCaptureAndroid");
63 assert(j_capture_class);
64 g_java_capturer_class =
65 reinterpret_cast<jclass>(ats.env()->NewGlobalRef(j_capture_class));
66 assert(g_java_capturer_class);
fischman@webrtc.org81cd5ca2013-10-03 18:23:1367
fischman@webrtc.org5101f842014-06-06 18:40:4468 JNINativeMethod native_methods[] = {
69 {"GetContext",
70 "()Landroid/content/Context;",
71 reinterpret_cast<void*>(&GetContext)},
fischman@webrtc.org5101f842014-06-06 18:40:4472 {"ProvideCameraFrame",
glaznev@webrtc.orgaafd1e52014-10-17 16:25:0673 "([BIIJJ)V",
fischman@webrtc.org5101f842014-06-06 18:40:4474 reinterpret_cast<void*>(&ProvideCameraFrame)}};
braveyao@webrtc.orgca2c70f2014-05-15 03:18:1575 if (ats.env()->RegisterNatives(g_java_capturer_class,
glaznev@webrtc.orgaafd1e52014-10-17 16:25:0676 native_methods, 2) != 0)
braveyao@webrtc.orgca2c70f2014-05-15 03:18:1577 assert(false);
78 } else {
79 if (g_jvm) {
80 AttachThreadScoped ats(g_jvm);
81 ats.env()->UnregisterNatives(g_java_capturer_class);
82 ats.env()->DeleteGlobalRef(g_java_capturer_class);
83 g_java_capturer_class = NULL;
fischman@webrtc.org5101f842014-06-06 18:40:4484 ats.env()->DeleteGlobalRef(g_context);
85 g_context = NULL;
braveyao@webrtc.orgca2c70f2014-05-15 03:18:1586 videocapturemodule::DeviceInfoAndroid::DeInitialize();
87 g_jvm = NULL;
88 }
89 }
fischman@webrtc.org81cd5ca2013-10-03 18:23:1390
91 return 0;
92}
93
94namespace videocapturemodule {
andrew@webrtc.orgb015cbe2012-10-22 18:19:2395
96VideoCaptureModule* VideoCaptureImpl::Create(
pbos@webrtc.orgf7e44d62013-04-10 08:23:1397 const int32_t id,
andrew@webrtc.orgb015cbe2012-10-22 18:19:2398 const char* deviceUniqueIdUTF8) {
andrew@webrtc.orgb015cbe2012-10-22 18:19:2399 RefCountImpl<videocapturemodule::VideoCaptureAndroid>* implementation =
100 new RefCountImpl<videocapturemodule::VideoCaptureAndroid>(id);
fischman@webrtc.org81cd5ca2013-10-03 18:23:13101 if (implementation->Init(id, deviceUniqueIdUTF8) != 0) {
andrew@webrtc.orgb015cbe2012-10-22 18:19:23102 delete implementation;
103 implementation = NULL;
104 }
105 return implementation;
106}
107
fischman@webrtc.org81cd5ca2013-10-03 18:23:13108int32_t VideoCaptureAndroid::OnIncomingFrame(uint8_t* videoFrame,
pkasting@chromium.org0ab923a2014-11-20 22:28:14109 size_t videoFrameLength,
glaznev@webrtc.orgaafd1e52014-10-17 16:25:06110 int32_t degrees,
fischman@webrtc.org81cd5ca2013-10-03 18:23:13111 int64_t captureTime) {
glaznev@webrtc.org82383d92014-07-14 17:01:53112 if (!_captureStarted)
113 return 0;
glaznev@webrtc.orgaafd1e52014-10-17 16:25:06114 VideoCaptureRotation current_rotation =
115 (degrees <= 45 || degrees > 315) ? kCameraRotate0 :
116 (degrees > 45 && degrees <= 135) ? kCameraRotate90 :
117 (degrees > 135 && degrees <= 225) ? kCameraRotate180 :
118 (degrees > 225 && degrees <= 315) ? kCameraRotate270 :
119 kCameraRotate0; // Impossible.
120 if (_rotation != current_rotation) {
121 LOG(LS_INFO) << "New camera rotation: " << degrees;
122 _rotation = current_rotation;
123 int32_t status = VideoCaptureImpl::SetCaptureRotation(_rotation);
124 if (status != 0)
125 return status;
126 }
fischman@webrtc.org81cd5ca2013-10-03 18:23:13127 return IncomingFrame(
128 videoFrame, videoFrameLength, _captureCapability, captureTime);
andrew@webrtc.orgb015cbe2012-10-22 18:19:23129}
130
pbos@webrtc.orgf7e44d62013-04-10 08:23:13131VideoCaptureAndroid::VideoCaptureAndroid(const int32_t id)
fischman@webrtc.org81cd5ca2013-10-03 18:23:13132 : VideoCaptureImpl(id),
133 _deviceInfo(id),
134 _jCapturer(NULL),
andrew@webrtc.orgb015cbe2012-10-22 18:19:23135 _captureStarted(false) {
andrew@webrtc.orgb015cbe2012-10-22 18:19:23136}
137
pbos@webrtc.orgf7e44d62013-04-10 08:23:13138int32_t VideoCaptureAndroid::Init(const int32_t id,
139 const char* deviceUniqueIdUTF8) {
andrew@webrtc.orgb015cbe2012-10-22 18:19:23140 const int nameLength = strlen(deviceUniqueIdUTF8);
fischman@webrtc.org81cd5ca2013-10-03 18:23:13141 if (nameLength >= kVideoCaptureUniqueNameLength)
andrew@webrtc.orgb015cbe2012-10-22 18:19:23142 return -1;
andrew@webrtc.orgb015cbe2012-10-22 18:19:23143
144 // Store the device name
glaznev@webrtc.orgab52e9a2014-06-20 20:55:54145 LOG(LS_INFO) << "VideoCaptureAndroid::Init: " << deviceUniqueIdUTF8;
146 size_t camera_id = 0;
147 if (!_deviceInfo.FindCameraIndex(deviceUniqueIdUTF8, &camera_id))
148 return -1;
andrew@webrtc.orgb015cbe2012-10-22 18:19:23149 _deviceUniqueId = new char[nameLength + 1];
150 memcpy(_deviceUniqueId, deviceUniqueIdUTF8, nameLength + 1);
151
fischman@webrtc.org81cd5ca2013-10-03 18:23:13152 AttachThreadScoped ats(g_jvm);
153 JNIEnv* env = ats.env();
fischman@webrtc.org81cd5ca2013-10-03 18:23:13154 jmethodID ctor = env->GetMethodID(g_java_capturer_class, "<init>", "(IJ)V");
155 assert(ctor);
156 jlong j_this = reinterpret_cast<intptr_t>(this);
fischman@webrtc.org81cd5ca2013-10-03 18:23:13157 _jCapturer = env->NewGlobalRef(
158 env->NewObject(g_java_capturer_class, ctor, camera_id, j_this));
159 assert(_jCapturer);
glaznev@webrtc.orgaafd1e52014-10-17 16:25:06160 _rotation = kCameraRotate0;
andrew@webrtc.orgb015cbe2012-10-22 18:19:23161 return 0;
162}
163
164VideoCaptureAndroid::~VideoCaptureAndroid() {
fischman@webrtc.org81cd5ca2013-10-03 18:23:13165 // Ensure Java camera is released even if our caller didn't explicitly Stop.
166 if (_captureStarted)
167 StopCapture();
168 AttachThreadScoped ats(g_jvm);
169 ats.env()->DeleteGlobalRef(_jCapturer);
andrew@webrtc.orgb015cbe2012-10-22 18:19:23170}
171
pbos@webrtc.orgf7e44d62013-04-10 08:23:13172int32_t VideoCaptureAndroid::StartCapture(
andrew@webrtc.orgb015cbe2012-10-22 18:19:23173 const VideoCaptureCapability& capability) {
174 CriticalSectionScoped cs(&_apiCs);
fischman@webrtc.org81cd5ca2013-10-03 18:23:13175 AttachThreadScoped ats(g_jvm);
176 JNIEnv* env = ats.env();
andrew@webrtc.orgb015cbe2012-10-22 18:19:23177
fischman@webrtc.org81cd5ca2013-10-03 18:23:13178 if (_deviceInfo.GetBestMatchedCapability(
179 _deviceUniqueId, capability, _captureCapability) < 0) {
andrew@webrtc.orgb015cbe2012-10-22 18:19:23180 WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceVideoCapture, -1,
fischman@webrtc.org81cd5ca2013-10-03 18:23:13181 "%s: GetBestMatchedCapability failed: %dx%d",
andrew@webrtc.orgb015cbe2012-10-22 18:19:23182 __FUNCTION__, capability.width, capability.height);
183 return -1;
184 }
185
fischman@webrtc.org81cd5ca2013-10-03 18:23:13186 _captureDelay = _captureCapability.expectedCaptureDelay;
andrew@webrtc.orgb015cbe2012-10-22 18:19:23187
fischman@webrtc.org81cd5ca2013-10-03 18:23:13188 jmethodID j_start =
189 env->GetMethodID(g_java_capturer_class, "startCapture", "(IIII)Z");
190 assert(j_start);
191 int min_mfps = 0;
192 int max_mfps = 0;
fischman@webrtc.orgb5153222014-04-09 01:18:32193 _deviceInfo.GetMFpsRange(_deviceUniqueId, _captureCapability.maxFPS,
194 &min_mfps, &max_mfps);
fischman@webrtc.org81cd5ca2013-10-03 18:23:13195 bool started = env->CallBooleanMethod(_jCapturer, j_start,
196 _captureCapability.width,
197 _captureCapability.height,
198 min_mfps, max_mfps);
199 if (started) {
andrew@webrtc.orgb015cbe2012-10-22 18:19:23200 _requestedCapability = capability;
201 _captureStarted = true;
202 }
fischman@webrtc.org81cd5ca2013-10-03 18:23:13203 return started ? 0 : -1;
andrew@webrtc.orgb015cbe2012-10-22 18:19:23204}
205
pbos@webrtc.orgf7e44d62013-04-10 08:23:13206int32_t VideoCaptureAndroid::StopCapture() {
glaznev@webrtc.org82383d92014-07-14 17:01:53207 _apiCs.Enter();
fischman@webrtc.org81cd5ca2013-10-03 18:23:13208 AttachThreadScoped ats(g_jvm);
209 JNIEnv* env = ats.env();
andrew@webrtc.orgb015cbe2012-10-22 18:19:23210
211 memset(&_requestedCapability, 0, sizeof(_requestedCapability));
fischman@webrtc.org81cd5ca2013-10-03 18:23:13212 memset(&_captureCapability, 0, sizeof(_captureCapability));
andrew@webrtc.orgb015cbe2012-10-22 18:19:23213 _captureStarted = false;
glaznev@webrtc.org82383d92014-07-14 17:01:53214 // Exit critical section to avoid blocking camera thread inside
215 // onIncomingFrame() call.
216 _apiCs.Leave();
andrew@webrtc.orgb015cbe2012-10-22 18:19:23217
fischman@webrtc.org81cd5ca2013-10-03 18:23:13218 jmethodID j_stop =
219 env->GetMethodID(g_java_capturer_class, "stopCapture", "()Z");
220 return env->CallBooleanMethod(_jCapturer, j_stop) ? 0 : -1;
andrew@webrtc.orgb015cbe2012-10-22 18:19:23221}
222
223bool VideoCaptureAndroid::CaptureStarted() {
224 CriticalSectionScoped cs(&_apiCs);
andrew@webrtc.orgb015cbe2012-10-22 18:19:23225 return _captureStarted;
226}
227
pbos@webrtc.orgf7e44d62013-04-10 08:23:13228int32_t VideoCaptureAndroid::CaptureSettings(
andrew@webrtc.orgb015cbe2012-10-22 18:19:23229 VideoCaptureCapability& settings) {
230 CriticalSectionScoped cs(&_apiCs);
andrew@webrtc.orgb015cbe2012-10-22 18:19:23231 settings = _requestedCapability;
232 return 0;
233}
234
pbos@webrtc.orgf7e44d62013-04-10 08:23:13235int32_t VideoCaptureAndroid::SetCaptureRotation(
andrew@webrtc.orgb015cbe2012-10-22 18:19:23236 VideoCaptureRotation rotation) {
fischman@webrtc.org5101f842014-06-06 18:40:44237 int32_t status = VideoCaptureImpl::SetCaptureRotation(rotation);
238 if (status != 0)
239 return status;
andrew@webrtc.orgb015cbe2012-10-22 18:19:23240
fischman@webrtc.org81cd5ca2013-10-03 18:23:13241 AttachThreadScoped ats(g_jvm);
242 JNIEnv* env = ats.env();
andrew@webrtc.orgb015cbe2012-10-22 18:19:23243
fischman@webrtc.org81cd5ca2013-10-03 18:23:13244 jmethodID j_spr =
245 env->GetMethodID(g_java_capturer_class, "setPreviewRotation", "(I)V");
246 assert(j_spr);
247 int rotation_degrees;
248 if (RotationInDegrees(rotation, &rotation_degrees) != 0) {
249 assert(false);
andrew@webrtc.orgb015cbe2012-10-22 18:19:23250 }
fischman@webrtc.org81cd5ca2013-10-03 18:23:13251 env->CallVoidMethod(_jCapturer, j_spr, rotation_degrees);
andrew@webrtc.orgb015cbe2012-10-22 18:19:23252 return 0;
253}
254
255} // namespace videocapturemodule
256} // namespace webrtc