andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 1 | /* |
| 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.org | 87c29b5 | 2013-07-12 10:03:52 | [diff] [blame] | 11 | #include "webrtc/modules/video_capture/android/video_capture_android.h" |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 12 | |
fischman@webrtc.org | d3d2598 | 2014-06-06 20:13:49 | [diff] [blame] | 13 | #include "webrtc/base/common.h" |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 14 | #include "webrtc/modules/utility/interface/helpers_android.h" |
| 15 | #include "webrtc/modules/video_capture/android/device_info_android.h" |
pbos@webrtc.org | 87c29b5 | 2013-07-12 10:03:52 | [diff] [blame] | 16 | #include "webrtc/system_wrappers/interface/critical_section_wrapper.h" |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 17 | #include "webrtc/system_wrappers/interface/logcat_trace_context.h" |
glaznev@webrtc.org | ab52e9a | 2014-06-20 20:55:54 | [diff] [blame] | 18 | #include "webrtc/system_wrappers/interface/logging.h" |
pbos@webrtc.org | 87c29b5 | 2013-07-12 10:03:52 | [diff] [blame] | 19 | #include "webrtc/system_wrappers/interface/ref_count.h" |
| 20 | #include "webrtc/system_wrappers/interface/trace.h" |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 21 | |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 22 | static JavaVM* g_jvm = NULL; |
| 23 | static jclass g_java_capturer_class = NULL; // VideoCaptureAndroid.class. |
fischman@webrtc.org | 5101f84 | 2014-06-06 18:40:44 | [diff] [blame] | 24 | static jobject g_context = NULL; // Owned android.content.Context. |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 25 | |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 26 | namespace webrtc { |
| 27 | |
fischman@webrtc.org | 5101f84 | 2014-06-06 18:40:44 | [diff] [blame] | 28 | // Called by Java to get the global application context. |
| 29 | jobject JNICALL GetContext(JNIEnv* env, jclass) { |
| 30 | assert(g_context); |
| 31 | return g_context; |
| 32 | } |
| 33 | |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 34 | // Called by Java when the camera has a new frame to deliver. |
| 35 | void JNICALL ProvideCameraFrame( |
| 36 | JNIEnv* env, |
| 37 | jobject, |
| 38 | jbyteArray javaCameraFrame, |
| 39 | jint length, |
glaznev@webrtc.org | aafd1e5 | 2014-10-17 16:25:06 | [diff] [blame] | 40 | jint rotation, |
glaznev@webrtc.org | ab52e9a | 2014-06-20 20:55:54 | [diff] [blame] | 41 | jlong timeStamp, |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 42 | 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.org | aafd1e5 | 2014-10-17 16:25:06 | [diff] [blame] | 48 | reinterpret_cast<uint8_t*>(cameraFrame), length, rotation, 0); |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 49 | env->ReleaseByteArrayElements(javaCameraFrame, cameraFrame, JNI_ABORT); |
| 50 | } |
| 51 | |
fischman@webrtc.org | 5101f84 | 2014-06-06 18:40:44 | [diff] [blame] | 52 | int32_t SetCaptureAndroidVM(JavaVM* javaVM, jobject context) { |
braveyao@webrtc.org | ca2c70f | 2014-05-15 03:18:15 | [diff] [blame] | 53 | if (javaVM) { |
| 54 | assert(!g_jvm); |
| 55 | g_jvm = javaVM; |
| 56 | AttachThreadScoped ats(g_jvm); |
fischman@webrtc.org | 5101f84 | 2014-06-06 18:40:44 | [diff] [blame] | 57 | g_context = ats.env()->NewGlobalRef(context); |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 58 | |
braveyao@webrtc.org | ca2c70f | 2014-05-15 03:18:15 | [diff] [blame] | 59 | videocapturemodule::DeviceInfoAndroid::Initialize(ats.env()); |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 60 | |
braveyao@webrtc.org | ca2c70f | 2014-05-15 03:18:15 | [diff] [blame] | 61 | 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.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 67 | |
fischman@webrtc.org | 5101f84 | 2014-06-06 18:40:44 | [diff] [blame] | 68 | JNINativeMethod native_methods[] = { |
| 69 | {"GetContext", |
| 70 | "()Landroid/content/Context;", |
| 71 | reinterpret_cast<void*>(&GetContext)}, |
fischman@webrtc.org | 5101f84 | 2014-06-06 18:40:44 | [diff] [blame] | 72 | {"ProvideCameraFrame", |
glaznev@webrtc.org | aafd1e5 | 2014-10-17 16:25:06 | [diff] [blame] | 73 | "([BIIJJ)V", |
fischman@webrtc.org | 5101f84 | 2014-06-06 18:40:44 | [diff] [blame] | 74 | reinterpret_cast<void*>(&ProvideCameraFrame)}}; |
braveyao@webrtc.org | ca2c70f | 2014-05-15 03:18:15 | [diff] [blame] | 75 | if (ats.env()->RegisterNatives(g_java_capturer_class, |
glaznev@webrtc.org | aafd1e5 | 2014-10-17 16:25:06 | [diff] [blame] | 76 | native_methods, 2) != 0) |
braveyao@webrtc.org | ca2c70f | 2014-05-15 03:18:15 | [diff] [blame] | 77 | 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.org | 5101f84 | 2014-06-06 18:40:44 | [diff] [blame] | 84 | ats.env()->DeleteGlobalRef(g_context); |
| 85 | g_context = NULL; |
braveyao@webrtc.org | ca2c70f | 2014-05-15 03:18:15 | [diff] [blame] | 86 | videocapturemodule::DeviceInfoAndroid::DeInitialize(); |
| 87 | g_jvm = NULL; |
| 88 | } |
| 89 | } |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 90 | |
| 91 | return 0; |
| 92 | } |
| 93 | |
| 94 | namespace videocapturemodule { |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 95 | |
| 96 | VideoCaptureModule* VideoCaptureImpl::Create( |
pbos@webrtc.org | f7e44d6 | 2013-04-10 08:23:13 | [diff] [blame] | 97 | const int32_t id, |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 98 | const char* deviceUniqueIdUTF8) { |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 99 | RefCountImpl<videocapturemodule::VideoCaptureAndroid>* implementation = |
| 100 | new RefCountImpl<videocapturemodule::VideoCaptureAndroid>(id); |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 101 | if (implementation->Init(id, deviceUniqueIdUTF8) != 0) { |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 102 | delete implementation; |
| 103 | implementation = NULL; |
| 104 | } |
| 105 | return implementation; |
| 106 | } |
| 107 | |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 108 | int32_t VideoCaptureAndroid::OnIncomingFrame(uint8_t* videoFrame, |
pkasting@chromium.org | 0ab923a | 2014-11-20 22:28:14 | [diff] [blame^] | 109 | size_t videoFrameLength, |
glaznev@webrtc.org | aafd1e5 | 2014-10-17 16:25:06 | [diff] [blame] | 110 | int32_t degrees, |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 111 | int64_t captureTime) { |
glaznev@webrtc.org | 82383d9 | 2014-07-14 17:01:53 | [diff] [blame] | 112 | if (!_captureStarted) |
| 113 | return 0; |
glaznev@webrtc.org | aafd1e5 | 2014-10-17 16:25:06 | [diff] [blame] | 114 | 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.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 127 | return IncomingFrame( |
| 128 | videoFrame, videoFrameLength, _captureCapability, captureTime); |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 129 | } |
| 130 | |
pbos@webrtc.org | f7e44d6 | 2013-04-10 08:23:13 | [diff] [blame] | 131 | VideoCaptureAndroid::VideoCaptureAndroid(const int32_t id) |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 132 | : VideoCaptureImpl(id), |
| 133 | _deviceInfo(id), |
| 134 | _jCapturer(NULL), |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 135 | _captureStarted(false) { |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 136 | } |
| 137 | |
pbos@webrtc.org | f7e44d6 | 2013-04-10 08:23:13 | [diff] [blame] | 138 | int32_t VideoCaptureAndroid::Init(const int32_t id, |
| 139 | const char* deviceUniqueIdUTF8) { |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 140 | const int nameLength = strlen(deviceUniqueIdUTF8); |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 141 | if (nameLength >= kVideoCaptureUniqueNameLength) |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 142 | return -1; |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 143 | |
| 144 | // Store the device name |
glaznev@webrtc.org | ab52e9a | 2014-06-20 20:55:54 | [diff] [blame] | 145 | LOG(LS_INFO) << "VideoCaptureAndroid::Init: " << deviceUniqueIdUTF8; |
| 146 | size_t camera_id = 0; |
| 147 | if (!_deviceInfo.FindCameraIndex(deviceUniqueIdUTF8, &camera_id)) |
| 148 | return -1; |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 149 | _deviceUniqueId = new char[nameLength + 1]; |
| 150 | memcpy(_deviceUniqueId, deviceUniqueIdUTF8, nameLength + 1); |
| 151 | |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 152 | AttachThreadScoped ats(g_jvm); |
| 153 | JNIEnv* env = ats.env(); |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 154 | 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.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 157 | _jCapturer = env->NewGlobalRef( |
| 158 | env->NewObject(g_java_capturer_class, ctor, camera_id, j_this)); |
| 159 | assert(_jCapturer); |
glaznev@webrtc.org | aafd1e5 | 2014-10-17 16:25:06 | [diff] [blame] | 160 | _rotation = kCameraRotate0; |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 161 | return 0; |
| 162 | } |
| 163 | |
| 164 | VideoCaptureAndroid::~VideoCaptureAndroid() { |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 165 | // 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.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 170 | } |
| 171 | |
pbos@webrtc.org | f7e44d6 | 2013-04-10 08:23:13 | [diff] [blame] | 172 | int32_t VideoCaptureAndroid::StartCapture( |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 173 | const VideoCaptureCapability& capability) { |
| 174 | CriticalSectionScoped cs(&_apiCs); |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 175 | AttachThreadScoped ats(g_jvm); |
| 176 | JNIEnv* env = ats.env(); |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 177 | |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 178 | if (_deviceInfo.GetBestMatchedCapability( |
| 179 | _deviceUniqueId, capability, _captureCapability) < 0) { |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 180 | WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceVideoCapture, -1, |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 181 | "%s: GetBestMatchedCapability failed: %dx%d", |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 182 | __FUNCTION__, capability.width, capability.height); |
| 183 | return -1; |
| 184 | } |
| 185 | |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 186 | _captureDelay = _captureCapability.expectedCaptureDelay; |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 187 | |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 188 | 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.org | b515322 | 2014-04-09 01:18:32 | [diff] [blame] | 193 | _deviceInfo.GetMFpsRange(_deviceUniqueId, _captureCapability.maxFPS, |
| 194 | &min_mfps, &max_mfps); |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 195 | bool started = env->CallBooleanMethod(_jCapturer, j_start, |
| 196 | _captureCapability.width, |
| 197 | _captureCapability.height, |
| 198 | min_mfps, max_mfps); |
| 199 | if (started) { |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 200 | _requestedCapability = capability; |
| 201 | _captureStarted = true; |
| 202 | } |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 203 | return started ? 0 : -1; |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 204 | } |
| 205 | |
pbos@webrtc.org | f7e44d6 | 2013-04-10 08:23:13 | [diff] [blame] | 206 | int32_t VideoCaptureAndroid::StopCapture() { |
glaznev@webrtc.org | 82383d9 | 2014-07-14 17:01:53 | [diff] [blame] | 207 | _apiCs.Enter(); |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 208 | AttachThreadScoped ats(g_jvm); |
| 209 | JNIEnv* env = ats.env(); |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 210 | |
| 211 | memset(&_requestedCapability, 0, sizeof(_requestedCapability)); |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 212 | memset(&_captureCapability, 0, sizeof(_captureCapability)); |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 213 | _captureStarted = false; |
glaznev@webrtc.org | 82383d9 | 2014-07-14 17:01:53 | [diff] [blame] | 214 | // Exit critical section to avoid blocking camera thread inside |
| 215 | // onIncomingFrame() call. |
| 216 | _apiCs.Leave(); |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 217 | |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 218 | jmethodID j_stop = |
| 219 | env->GetMethodID(g_java_capturer_class, "stopCapture", "()Z"); |
| 220 | return env->CallBooleanMethod(_jCapturer, j_stop) ? 0 : -1; |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 221 | } |
| 222 | |
| 223 | bool VideoCaptureAndroid::CaptureStarted() { |
| 224 | CriticalSectionScoped cs(&_apiCs); |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 225 | return _captureStarted; |
| 226 | } |
| 227 | |
pbos@webrtc.org | f7e44d6 | 2013-04-10 08:23:13 | [diff] [blame] | 228 | int32_t VideoCaptureAndroid::CaptureSettings( |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 229 | VideoCaptureCapability& settings) { |
| 230 | CriticalSectionScoped cs(&_apiCs); |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 231 | settings = _requestedCapability; |
| 232 | return 0; |
| 233 | } |
| 234 | |
pbos@webrtc.org | f7e44d6 | 2013-04-10 08:23:13 | [diff] [blame] | 235 | int32_t VideoCaptureAndroid::SetCaptureRotation( |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 236 | VideoCaptureRotation rotation) { |
fischman@webrtc.org | 5101f84 | 2014-06-06 18:40:44 | [diff] [blame] | 237 | int32_t status = VideoCaptureImpl::SetCaptureRotation(rotation); |
| 238 | if (status != 0) |
| 239 | return status; |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 240 | |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 241 | AttachThreadScoped ats(g_jvm); |
| 242 | JNIEnv* env = ats.env(); |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 243 | |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 244 | 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.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 250 | } |
fischman@webrtc.org | 81cd5ca | 2013-10-03 18:23:13 | [diff] [blame] | 251 | env->CallVoidMethod(_jCapturer, j_spr, rotation_degrees); |
andrew@webrtc.org | b015cbe | 2012-10-22 18:19:23 | [diff] [blame] | 252 | return 0; |
| 253 | } |
| 254 | |
| 255 | } // namespace videocapturemodule |
| 256 | } // namespace webrtc |