mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-12 18:35:35 +00:00
1015 lines
27 KiB
C
1015 lines
27 KiB
C
/*
|
|
* Copyright (C) 2012, Collabora Ltd.
|
|
* Copyright (C) 2012, Cisco Systems, Inc.
|
|
* Author: Youness Alaoui <youness.alaoui@collabora.co.uk>
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation
|
|
* version 2.1 of the License.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include "gst-dvm.h"
|
|
#include "gstahccallback.h"
|
|
#include "gst-android-hardware-camera.h"
|
|
#include "stdio.h"
|
|
|
|
static struct
|
|
{
|
|
jclass klass;
|
|
jmethodID addCallbackBuffer;
|
|
jmethodID getCameraInfo;
|
|
jmethodID getNumberOfCameras;
|
|
jmethodID getParameters;
|
|
jmethodID lock;
|
|
jmethodID open;
|
|
jmethodID reconnect;
|
|
jmethodID release;
|
|
jmethodID setErrorCallback;
|
|
jmethodID setParameters;
|
|
jmethodID setPreviewCallbackWithBuffer;
|
|
jmethodID setPreviewTexture;
|
|
jmethodID startPreview;
|
|
jmethodID startSmoothZoom;
|
|
jmethodID stopPreview;
|
|
jmethodID stopSmoothZoom;
|
|
jmethodID unlock;
|
|
} android_hardware_camera = {
|
|
0};
|
|
|
|
static struct
|
|
{
|
|
jclass klass;
|
|
jmethodID constructor;
|
|
jfieldID facing;
|
|
jfieldID orientation;
|
|
jint CAMERA_FACING_BACK;
|
|
jint CAMERA_FACING_FRONT;
|
|
} android_hardware_camera_camerainfo = {
|
|
0};
|
|
|
|
gint CameraInfo_CAMERA_FACING_BACK;
|
|
gint CameraInfo_CAMERA_FACING_FRONT;
|
|
|
|
static struct
|
|
{
|
|
jclass klass;
|
|
jfieldID width;
|
|
jfieldID height;
|
|
} android_hardware_camera_size = {
|
|
0};
|
|
|
|
/* TODO: Add other parameters */
|
|
static struct
|
|
{
|
|
jclass klass;
|
|
jmethodID flatten;
|
|
jmethodID getPreviewFormat;
|
|
jmethodID getPreviewFpsRange;
|
|
jmethodID getPreviewSize;
|
|
jmethodID getSupportedPreviewFormats;
|
|
jmethodID getSupportedPreviewFpsRange;
|
|
jmethodID getSupportedPreviewSizes;
|
|
jmethodID setPreviewFormat;
|
|
jmethodID setPreviewFpsRange;
|
|
jmethodID setPreviewSize;
|
|
jmethodID unflatten;
|
|
} android_hardware_camera_parameters = {
|
|
0};
|
|
|
|
static struct
|
|
{
|
|
jclass klass;
|
|
jmethodID iterator;
|
|
} java_util_list = {
|
|
0};
|
|
|
|
static struct
|
|
{
|
|
jclass klass;
|
|
jmethodID hasNext;
|
|
jmethodID next;
|
|
} java_util_iterator = {
|
|
0};
|
|
|
|
static struct
|
|
{
|
|
jclass klass;
|
|
jmethodID intValue;
|
|
} java_lang_integer = {
|
|
0};
|
|
|
|
static struct
|
|
{
|
|
jclass klass;
|
|
jmethodID constructor;
|
|
} com_gstreamer_gstahccallback = {
|
|
0};
|
|
|
|
static void
|
|
gst_ah_camera_on_preview_frame (JNIEnv * env, jclass klass, jbyteArray data,
|
|
jobject camera, jlong callback, jlong user_data)
|
|
{
|
|
GstAHCPreviewCallback cb = (GstAHCPreviewCallback) (gsize) callback;
|
|
|
|
if (cb)
|
|
cb (data, (gpointer) (gsize) user_data);
|
|
}
|
|
|
|
static void
|
|
gst_ah_camera_on_error (JNIEnv * env, jclass klass, jint error,
|
|
jobject camera, jlong callback, jlong user_data)
|
|
{
|
|
GstAHCErrorCallback cb = (GstAHCErrorCallback) (gsize) callback;
|
|
|
|
if (cb)
|
|
cb (error, (gpointer) (gsize) user_data);
|
|
}
|
|
|
|
static JNINativeMethod native_methods[] = {
|
|
{"gst_ah_camera_on_preview_frame", "([BLandroid/hardware/Camera;JJ)V",
|
|
(void *) gst_ah_camera_on_preview_frame},
|
|
{"gst_ah_camera_on_error", "(ILandroid/hardware/Camera;JJ)V",
|
|
(void *) gst_ah_camera_on_error}
|
|
};
|
|
|
|
static gboolean
|
|
_init_classes (void)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
|
|
/* android.hardware.Camera */
|
|
GST_DVM_GET_CLASS (android_hardware_camera, "android/hardware/Camera");
|
|
GST_DVM_GET_METHOD (android_hardware_camera, addCallbackBuffer, "([B)V");
|
|
GST_DVM_GET_STATIC_METHOD (android_hardware_camera, getCameraInfo,
|
|
"(ILandroid/hardware/Camera$CameraInfo;)V");
|
|
GST_DVM_GET_STATIC_METHOD (android_hardware_camera, getNumberOfCameras,
|
|
"()I");
|
|
GST_DVM_GET_METHOD (android_hardware_camera, getParameters,
|
|
"()Landroid/hardware/Camera$Parameters;");
|
|
GST_DVM_GET_METHOD (android_hardware_camera, lock, "()V");
|
|
GST_DVM_GET_STATIC_METHOD (android_hardware_camera, open,
|
|
"(I)Landroid/hardware/Camera;");
|
|
GST_DVM_GET_METHOD (android_hardware_camera, reconnect, "()V");
|
|
GST_DVM_GET_METHOD (android_hardware_camera, release, "()V");
|
|
GST_DVM_GET_METHOD (android_hardware_camera, setErrorCallback,
|
|
"(Landroid/hardware/Camera$ErrorCallback;)V");
|
|
GST_DVM_GET_METHOD (android_hardware_camera, setParameters,
|
|
"(Landroid/hardware/Camera$Parameters;)V");
|
|
GST_DVM_GET_METHOD (android_hardware_camera, setPreviewCallbackWithBuffer,
|
|
"(Landroid/hardware/Camera$PreviewCallback;)V");
|
|
GST_DVM_GET_METHOD (android_hardware_camera, setPreviewTexture,
|
|
"(Landroid/graphics/SurfaceTexture;)V");
|
|
GST_DVM_GET_METHOD (android_hardware_camera, startPreview, "()V");
|
|
GST_DVM_GET_METHOD (android_hardware_camera, startSmoothZoom, "(I)V");
|
|
GST_DVM_GET_METHOD (android_hardware_camera, stopPreview, "()V");
|
|
GST_DVM_GET_METHOD (android_hardware_camera, stopSmoothZoom, "()V");
|
|
GST_DVM_GET_METHOD (android_hardware_camera, unlock, "()V");
|
|
|
|
/* android.hardware.Camera.CameraInfo */
|
|
GST_DVM_GET_CLASS (android_hardware_camera_camerainfo,
|
|
"android/hardware/Camera$CameraInfo");
|
|
GST_DVM_GET_CONSTRUCTOR (android_hardware_camera_camerainfo,
|
|
constructor, "()V");
|
|
GST_DVM_GET_FIELD (android_hardware_camera_camerainfo, facing, "I");
|
|
GST_DVM_GET_FIELD (android_hardware_camera_camerainfo, orientation, "I");
|
|
GST_DVM_GET_CONSTANT (android_hardware_camera_camerainfo, CAMERA_FACING_BACK,
|
|
Int, "I");
|
|
CameraInfo_CAMERA_FACING_BACK =
|
|
android_hardware_camera_camerainfo.CAMERA_FACING_BACK;
|
|
GST_DVM_GET_CONSTANT (android_hardware_camera_camerainfo, CAMERA_FACING_FRONT,
|
|
Int, "I");
|
|
CameraInfo_CAMERA_FACING_FRONT =
|
|
android_hardware_camera_camerainfo.CAMERA_FACING_FRONT;
|
|
|
|
/* android.hardware.Camera.Size */
|
|
GST_DVM_GET_CLASS (android_hardware_camera_size,
|
|
"android/hardware/Camera$Size");
|
|
GST_DVM_GET_FIELD (android_hardware_camera_size, width, "I");
|
|
GST_DVM_GET_FIELD (android_hardware_camera_size, height, "I");
|
|
|
|
/* android.hardware.Camera.Parameters */
|
|
GST_DVM_GET_CLASS (android_hardware_camera_parameters,
|
|
"android/hardware/Camera$Parameters");
|
|
GST_DVM_GET_METHOD (android_hardware_camera_parameters, flatten,
|
|
"()Ljava/lang/String;");
|
|
GST_DVM_GET_METHOD (android_hardware_camera_parameters, getPreviewFormat,
|
|
"()I");
|
|
GST_DVM_GET_METHOD (android_hardware_camera_parameters, getPreviewFpsRange,
|
|
"([I)V");
|
|
GST_DVM_GET_METHOD (android_hardware_camera_parameters, getPreviewSize,
|
|
"()Landroid/hardware/Camera$Size;");
|
|
GST_DVM_GET_METHOD (android_hardware_camera_parameters,
|
|
getSupportedPreviewFormats, "()Ljava/util/List;");
|
|
GST_DVM_GET_METHOD (android_hardware_camera_parameters,
|
|
getSupportedPreviewFpsRange, "()Ljava/util/List;");
|
|
GST_DVM_GET_METHOD (android_hardware_camera_parameters,
|
|
getSupportedPreviewSizes, "()Ljava/util/List;");
|
|
GST_DVM_GET_METHOD (android_hardware_camera_parameters, setPreviewFormat,
|
|
"(I)V");
|
|
GST_DVM_GET_METHOD (android_hardware_camera_parameters, setPreviewFpsRange,
|
|
"(II)V");
|
|
GST_DVM_GET_METHOD (android_hardware_camera_parameters, setPreviewSize,
|
|
"(II)V");
|
|
GST_DVM_GET_METHOD (android_hardware_camera_parameters, unflatten,
|
|
"(Ljava/lang/String;)V");
|
|
|
|
/* java.util.List */
|
|
GST_DVM_GET_CLASS (java_util_list, "java/util/List");
|
|
GST_DVM_GET_METHOD (java_util_list, iterator, "()Ljava/util/Iterator;");
|
|
|
|
/* java.util.Iterator */
|
|
GST_DVM_GET_CLASS (java_util_iterator, "java/util/Iterator");
|
|
GST_DVM_GET_METHOD (java_util_iterator, hasNext, "()Z");
|
|
GST_DVM_GET_METHOD (java_util_iterator, next, "()Ljava/lang/Object;");
|
|
|
|
/* java.lang.Integer */
|
|
GST_DVM_GET_CLASS (java_lang_integer, "java/lang/Integer");
|
|
GST_DVM_GET_METHOD (java_lang_integer, intValue, "()I");
|
|
|
|
/* com.gstreamer.GstAhcCallback */
|
|
if (gst_ahc_callback_jar) {
|
|
jclass dex_loader = NULL;
|
|
gchar *path = g_strdup_printf ("%s/GstAhcCallback.jar", g_getenv ("TMP"));
|
|
FILE *fd = fopen (path, "wb");
|
|
|
|
GST_DEBUG ("Found embedded GstAhcCallback.jar, trying to load dynamically"
|
|
"from %s", path);
|
|
if (fd) {
|
|
if (fwrite (gst_ahc_callback_jar, gst_ahc_callback_jar_size, 1, fd) == 1) {
|
|
dex_loader = (*env)->FindClass (env, "dalvik/system/DexClassLoader");
|
|
(*env)->ExceptionClear (env);
|
|
}
|
|
fclose (fd);
|
|
}
|
|
|
|
if (dex_loader) {
|
|
jmethodID constructor;
|
|
jmethodID load_class;
|
|
|
|
constructor = (*env)->GetMethodID (env, dex_loader, "<init>",
|
|
"(Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;"
|
|
"Ljava/lang/ClassLoader;)V");
|
|
load_class = (*env)->GetMethodID (env, dex_loader, "loadClass",
|
|
"(Ljava/lang/String;)Ljava/lang/Class;");
|
|
(*env)->ExceptionClear (env);
|
|
if (constructor && load_class) {
|
|
jstring dex_path = NULL;
|
|
jstring optimized_directory = NULL;
|
|
|
|
dex_path = (*env)->NewStringUTF (env, path);
|
|
optimized_directory = (*env)->NewStringUTF (env, g_getenv ("TMP"));
|
|
(*env)->ExceptionClear (env);
|
|
if (dex_path && optimized_directory) {
|
|
jobject loader;
|
|
jobject parent = NULL;
|
|
jclass klass;
|
|
|
|
klass = (*env)->FindClass (env, "java/lang/Class");
|
|
(*env)->ExceptionClear (env);
|
|
if (klass) {
|
|
jmethodID get_class_loader;
|
|
|
|
get_class_loader = (*env)->GetMethodID (env, klass,
|
|
"getClassLoader", "()Ljava/lang/ClassLoader;");
|
|
(*env)->ExceptionClear (env);
|
|
if (get_class_loader) {
|
|
parent = (*env)->CallObjectMethod (env, klass, get_class_loader);
|
|
(*env)->ExceptionClear (env);
|
|
}
|
|
(*env)->DeleteLocalRef (env, klass);
|
|
}
|
|
loader = (*env)->NewObject (env, dex_loader, constructor, dex_path,
|
|
optimized_directory, NULL, parent);
|
|
(*env)->ExceptionClear (env);
|
|
if (loader) {
|
|
jstring class_name = NULL;
|
|
|
|
class_name = (*env)->NewStringUTF (env,
|
|
"com/gstreamer/GstAhcCallback");
|
|
(*env)->ExceptionClear (env);
|
|
if (class_name) {
|
|
jclass temp;
|
|
|
|
temp = (*env)->CallObjectMethod (env, loader, load_class,
|
|
class_name);
|
|
(*env)->ExceptionClear (env);
|
|
if (temp) {
|
|
|
|
GST_DEBUG ("Successfully loaded embedded GstAhcCallback");
|
|
com_gstreamer_gstahccallback.klass = (*env)->NewGlobalRef (env,
|
|
temp);
|
|
(*env)->DeleteLocalRef (env, temp);
|
|
}
|
|
(*env)->DeleteLocalRef (env, class_name);
|
|
}
|
|
(*env)->DeleteLocalRef (env, loader);
|
|
}
|
|
if (parent)
|
|
(*env)->DeleteLocalRef (env, parent);
|
|
}
|
|
if (dex_path)
|
|
(*env)->DeleteLocalRef (env, dex_path);
|
|
if (optimized_directory)
|
|
(*env)->DeleteLocalRef (env, optimized_directory);
|
|
}
|
|
(*env)->DeleteLocalRef (env, dex_loader);
|
|
g_free (path);
|
|
}
|
|
} else {
|
|
GST_WARNING ("Did not find embedded GstAhcCallback.jar, fallback to"
|
|
" FindClass");
|
|
}
|
|
if (!com_gstreamer_gstahccallback.klass) {
|
|
GST_DVM_GET_CLASS (com_gstreamer_gstahccallback,
|
|
"com/gstreamer/GstAhcCallback");
|
|
}
|
|
GST_DVM_GET_CONSTRUCTOR (com_gstreamer_gstahccallback, constructor, "(JJ)V");
|
|
|
|
if ((*env)->RegisterNatives (env, com_gstreamer_gstahccallback.klass,
|
|
native_methods, G_N_ELEMENTS (native_methods))) {
|
|
GST_ERROR ("Failed to register native methods for GstAhcCallback");
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
gboolean
|
|
gst_android_hardware_camera_init (void)
|
|
{
|
|
if (!_init_classes ()) {
|
|
gst_android_hardware_camera_deinit ();
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
gst_android_hardware_camera_deinit (void)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
|
|
if (android_hardware_camera.klass)
|
|
(*env)->DeleteGlobalRef (env, android_hardware_camera.klass);
|
|
android_hardware_camera.klass = NULL;
|
|
|
|
if (android_hardware_camera_camerainfo.klass)
|
|
(*env)->DeleteGlobalRef (env, android_hardware_camera_camerainfo.klass);
|
|
android_hardware_camera_camerainfo.klass = NULL;
|
|
|
|
if (android_hardware_camera_size.klass)
|
|
(*env)->DeleteGlobalRef (env, android_hardware_camera_size.klass);
|
|
android_hardware_camera_size.klass = NULL;
|
|
|
|
if (android_hardware_camera_parameters.klass)
|
|
(*env)->DeleteGlobalRef (env, android_hardware_camera_parameters.klass);
|
|
android_hardware_camera_parameters.klass = NULL;
|
|
|
|
if (java_util_list.klass)
|
|
(*env)->DeleteGlobalRef (env, java_util_list.klass);
|
|
java_util_list.klass = NULL;
|
|
|
|
if (java_util_iterator.klass)
|
|
(*env)->DeleteGlobalRef (env, java_util_iterator.klass);
|
|
java_util_iterator.klass = NULL;
|
|
|
|
if (java_lang_integer.klass)
|
|
(*env)->DeleteGlobalRef (env, java_lang_integer.klass);
|
|
java_lang_integer.klass = NULL;
|
|
|
|
if (com_gstreamer_gstahccallback.klass) {
|
|
(*env)->UnregisterNatives (env, com_gstreamer_gstahccallback.klass);
|
|
(*env)->DeleteGlobalRef (env, com_gstreamer_gstahccallback.klass);
|
|
}
|
|
com_gstreamer_gstahccallback.klass = NULL;
|
|
}
|
|
|
|
/* android.hardware.Camera */
|
|
#define AHC_CALL(error_statement, type, method, ...) \
|
|
GST_DVM_CALL (error_statement, self->object, type, android_hardware_camera, \
|
|
method, ## __VA_ARGS__);
|
|
#define AHC_STATIC_CALL(error_statement, type, method, ...) \
|
|
GST_DVM_STATIC_CALL (error_statement, type, android_hardware_camera, \
|
|
method, ## __VA_ARGS__);
|
|
|
|
void
|
|
gst_ah_camera_add_callback_buffer (GstAHCamera * self, jbyteArray buffer)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
|
|
AHC_CALL (, Void, addCallbackBuffer, buffer);
|
|
}
|
|
|
|
gboolean
|
|
gst_ah_camera_get_camera_info (gint camera_id, GstAHCCameraInfo * camera_info)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
jobject jcamera_info = NULL;
|
|
gboolean ret = FALSE;
|
|
|
|
jcamera_info = (*env)->NewObject (env,
|
|
android_hardware_camera_camerainfo.klass,
|
|
android_hardware_camera_camerainfo.constructor);
|
|
if (!jcamera_info) {
|
|
GST_ERROR ("Failed to call Java method");
|
|
(*env)->ExceptionClear (env);
|
|
goto done;
|
|
}
|
|
|
|
AHC_STATIC_CALL (goto done, Void, getCameraInfo, camera_id, jcamera_info);
|
|
|
|
camera_info->facing = (*env)->GetIntField (env, jcamera_info,
|
|
android_hardware_camera_camerainfo.facing);
|
|
if ((*env)->ExceptionCheck (env)) {
|
|
GST_ERROR ("Failed to get CameraInfo.facing field");
|
|
(*env)->ExceptionClear (env);
|
|
goto done;
|
|
}
|
|
|
|
camera_info->orientation = (*env)->GetIntField (env, jcamera_info,
|
|
android_hardware_camera_camerainfo.orientation);
|
|
if ((*env)->ExceptionCheck (env)) {
|
|
GST_ERROR ("Failed to get CameraInfo.orientation field");
|
|
(*env)->ExceptionClear (env);
|
|
goto done;
|
|
}
|
|
|
|
ret = TRUE;
|
|
done:
|
|
if (jcamera_info)
|
|
(*env)->DeleteLocalRef (env, jcamera_info);
|
|
|
|
return ret;
|
|
}
|
|
|
|
gint
|
|
gst_ah_camera_get_number_of_cameras (void)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
gint num_cameras;
|
|
|
|
num_cameras = AHC_STATIC_CALL (return -1, Int, getNumberOfCameras);
|
|
|
|
return num_cameras;
|
|
}
|
|
|
|
GstAHCParameters *
|
|
gst_ah_camera_get_parameters (GstAHCamera * self)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
jobject object = NULL;
|
|
GstAHCParameters *params = NULL;
|
|
|
|
object = AHC_CALL (return NULL, Object, getParameters);
|
|
|
|
params = g_slice_new0 (GstAHCParameters);
|
|
params->object = (*env)->NewGlobalRef (env, object);
|
|
(*env)->DeleteLocalRef (env, object);
|
|
if (!params->object) {
|
|
GST_ERROR ("Failed to create global reference");
|
|
(*env)->ExceptionClear (env);
|
|
g_slice_free (GstAHCParameters, params);
|
|
return NULL;
|
|
}
|
|
|
|
return params;
|
|
}
|
|
|
|
gboolean
|
|
gst_ah_camera_lock (GstAHCamera * self)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
|
|
AHC_CALL (return FALSE, Void, lock);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
GstAHCamera *
|
|
gst_ah_camera_open (gint camera_id)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
jobject object = NULL;
|
|
GstAHCamera *camera = NULL;
|
|
|
|
object = AHC_STATIC_CALL (goto done, Object, open, camera_id);
|
|
|
|
camera = g_slice_new0 (GstAHCamera);
|
|
camera->object = (*env)->NewGlobalRef (env, object);
|
|
(*env)->DeleteLocalRef (env, object);
|
|
if (!camera->object) {
|
|
GST_ERROR ("Failed to create global reference");
|
|
(*env)->ExceptionClear (env);
|
|
g_slice_free (GstAHCamera, camera);
|
|
goto done;
|
|
}
|
|
|
|
return camera;
|
|
done:
|
|
return NULL;
|
|
}
|
|
|
|
gboolean
|
|
gst_ah_camera_reconnect (GstAHCamera * self)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
|
|
AHC_CALL (return FALSE, Void, reconnect);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
gst_ah_camera_release (GstAHCamera * self)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
|
|
AHC_CALL (, Void, release);
|
|
|
|
(*env)->DeleteGlobalRef (env, self->object);
|
|
g_slice_free (GstAHCamera, self);
|
|
}
|
|
|
|
gboolean
|
|
gst_ah_camera_set_parameters (GstAHCamera * self, GstAHCParameters * params)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
|
|
AHC_CALL (return FALSE, Void, setParameters, params->object);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
gst_ah_camera_set_error_callback (GstAHCamera * self, GstAHCErrorCallback cb,
|
|
gpointer user_data)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
jobject object = NULL;
|
|
gboolean ret = FALSE;
|
|
|
|
if (cb) {
|
|
object = (*env)->NewObject (env,
|
|
com_gstreamer_gstahccallback.klass,
|
|
com_gstreamer_gstahccallback.constructor,
|
|
*((jlong *) & cb), *((jlong *) & user_data));
|
|
if (!object) {
|
|
GST_ERROR ("Failed to create callback object");
|
|
(*env)->ExceptionClear (env);
|
|
goto done;
|
|
}
|
|
}
|
|
|
|
AHC_CALL (goto done, Void, setErrorCallback, object);
|
|
|
|
ret = TRUE;
|
|
done:
|
|
if (object)
|
|
(*env)->DeleteLocalRef (env, object);
|
|
|
|
return ret;
|
|
}
|
|
|
|
gboolean
|
|
gst_ah_camera_set_preview_callback_with_buffer (GstAHCamera * self,
|
|
GstAHCPreviewCallback cb, gpointer user_data)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
jobject object = NULL;
|
|
gboolean ret = FALSE;
|
|
|
|
if (cb) {
|
|
object = (*env)->NewObject (env,
|
|
com_gstreamer_gstahccallback.klass,
|
|
com_gstreamer_gstahccallback.constructor,
|
|
*((jlong *) & cb), *((jlong *) & user_data));
|
|
if (!object) {
|
|
GST_ERROR ("Failed to create callback object");
|
|
(*env)->ExceptionClear (env);
|
|
goto done;
|
|
}
|
|
}
|
|
|
|
AHC_CALL (goto done, Void, setPreviewCallbackWithBuffer, object);
|
|
|
|
ret = TRUE;
|
|
done:
|
|
if (object)
|
|
(*env)->DeleteLocalRef (env, object);
|
|
|
|
return ret;
|
|
}
|
|
|
|
void
|
|
gst_ah_camera_set_preview_texture (GstAHCamera * self,
|
|
GstAGSurfaceTexture * surfaceTexture)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
|
|
AHC_CALL (, Void, setPreviewTexture, surfaceTexture->object);
|
|
}
|
|
|
|
gboolean
|
|
gst_ah_camera_start_preview (GstAHCamera * self)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
|
|
AHC_CALL (return FALSE, Void, startPreview);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
gst_ah_camera_start_smooth_zoom (GstAHCamera * self, gint value)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
|
|
AHC_CALL (return FALSE, Void, startSmoothZoom, value);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
gst_ah_camera_stop_preview (GstAHCamera * self)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
|
|
AHC_CALL (return FALSE, Void, stopPreview);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
gst_ah_camera_stop_smooth_zoom (GstAHCamera * self)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
|
|
AHC_CALL (return FALSE, Void, stopSmoothZoom);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
gst_ah_camera_unlock (GstAHCamera * self)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
|
|
AHC_CALL (return FALSE, Void, unlock);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
#undef AHC_CALL
|
|
#undef AHC_STATIC_CALL
|
|
|
|
/* android.hardware.Camera.Size */
|
|
GstAHCSize *
|
|
gst_ahc_size_new (gint width, gint height)
|
|
{
|
|
GstAHCSize *self = g_slice_new0 (GstAHCSize);
|
|
|
|
self->width = width;
|
|
self->height = height;
|
|
|
|
return self;
|
|
}
|
|
|
|
void
|
|
gst_ahc_size_free (GstAHCSize * self)
|
|
{
|
|
g_slice_free (GstAHCSize, self);
|
|
}
|
|
|
|
/* java.util.List */
|
|
static jobject
|
|
java_util_list_iterator (JNIEnv * env, jobject obj)
|
|
{
|
|
return (*env)->CallObjectMethod (env, obj, java_util_list.iterator);
|
|
}
|
|
|
|
static jobject
|
|
java_util_iterator_next (JNIEnv * env, jobject obj)
|
|
{
|
|
return (*env)->CallObjectMethod (env, obj, java_util_iterator.next);
|
|
}
|
|
|
|
static jboolean
|
|
java_util_iterator_has_next (JNIEnv * env, jobject obj)
|
|
{
|
|
return (*env)->CallBooleanMethod (env, obj, java_util_iterator.hasNext);
|
|
}
|
|
|
|
static jint
|
|
java_lang_integer_int_value (JNIEnv * env, jobject obj)
|
|
{
|
|
return (*env)->CallIntMethod (env, obj, java_lang_integer.intValue);
|
|
}
|
|
|
|
/* android.hardware.Camera.Parameters */
|
|
#define AHCP_CALL(error_statement, type, method, ...) \
|
|
GST_DVM_CALL (error_statement, self->object, type, \
|
|
android_hardware_camera_parameters, method, ## __VA_ARGS__);
|
|
#define AHCP_STATIC_CALL(error_statement, type, method, ...) \
|
|
GST_DVM_STATIC_CALL (error_statement, type, \
|
|
android_hardware_camera_parameters, method, ## __VA_ARGS__);
|
|
|
|
gchar *
|
|
gst_ahc_parameters_flatten (GstAHCParameters * self)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
jstring v_str = NULL;
|
|
const gchar *v = NULL;
|
|
gchar *ret = NULL;
|
|
|
|
v_str = AHCP_CALL (goto done, Object, flatten);
|
|
v = (*env)->GetStringUTFChars (env, v_str, NULL);
|
|
if (!v) {
|
|
GST_ERROR ("Failed to convert string to UTF8");
|
|
(*env)->ExceptionClear (env);
|
|
goto done;
|
|
}
|
|
|
|
ret = g_strdup (v);
|
|
done:
|
|
if (v)
|
|
(*env)->ReleaseStringUTFChars (env, v_str, v);
|
|
if (v_str)
|
|
(*env)->DeleteLocalRef (env, v_str);
|
|
|
|
return ret;
|
|
}
|
|
|
|
gint
|
|
gst_ahc_parameters_get_preview_format (GstAHCParameters * self)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
gint format;
|
|
|
|
format = AHCP_CALL (return 0, Int, getPreviewFormat);
|
|
|
|
return format;
|
|
}
|
|
|
|
gboolean
|
|
gst_ahc_parameters_get_preview_fps_range (GstAHCParameters * self,
|
|
gint * min, gint * max)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
gboolean ret = FALSE;
|
|
jintArray range = NULL;
|
|
jint *fps = NULL;
|
|
|
|
range = (*env)->NewIntArray (env, 2);
|
|
if (!fps) {
|
|
(*env)->ExceptionClear (env);
|
|
GST_ERROR ("Failed to create array");
|
|
goto done;
|
|
}
|
|
|
|
AHCP_CALL (goto done, Void, getPreviewFpsRange, range);
|
|
|
|
fps = (*env)->GetIntArrayElements (env, range, NULL);
|
|
if ((*env)->ExceptionCheck (env) || !fps) {
|
|
(*env)->ExceptionClear (env);
|
|
GST_ERROR ("Failed to get array elements");
|
|
goto done;
|
|
}
|
|
if (min)
|
|
*min = fps[0];
|
|
if (max)
|
|
*max = fps[1];
|
|
|
|
ret = TRUE;
|
|
done:
|
|
if (fps)
|
|
(*env)->ReleaseIntArrayElements (env, range, fps, JNI_ABORT);
|
|
if (range)
|
|
(*env)->DeleteLocalRef (env, range);
|
|
|
|
return ret;
|
|
}
|
|
|
|
GstAHCSize *
|
|
gst_ahc_parameters_get_preview_size (GstAHCParameters * self)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
jobject jsize = NULL;
|
|
GstAHCSize *size = NULL;
|
|
|
|
jsize = AHCP_CALL (goto done, Object, getPreviewSize);
|
|
|
|
size = g_slice_new0 (GstAHCSize);
|
|
|
|
size->width = (*env)->GetIntField (env, jsize,
|
|
android_hardware_camera_size.width);
|
|
if ((*env)->ExceptionCheck (env)) {
|
|
GST_ERROR ("Failed to get Camera.Size.width field");
|
|
(*env)->ExceptionClear (env);
|
|
g_slice_free (GstAHCSize, size);
|
|
size = NULL;
|
|
goto done;
|
|
}
|
|
|
|
size->height = (*env)->GetIntField (env, jsize,
|
|
android_hardware_camera_size.height);
|
|
if ((*env)->ExceptionCheck (env)) {
|
|
GST_ERROR ("Failed to get Camera.Size.height field");
|
|
(*env)->ExceptionClear (env);
|
|
g_slice_free (GstAHCSize, size);
|
|
size = NULL;
|
|
goto done;
|
|
}
|
|
|
|
done:
|
|
if (jsize)
|
|
(*env)->DeleteLocalRef (env, jsize);
|
|
|
|
return size;
|
|
}
|
|
|
|
GList *
|
|
gst_ahc_parameters_get_supported_preview_formats (GstAHCParameters * self)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
jobject list = NULL;
|
|
jobject iterator = NULL;
|
|
GList *ret = NULL;
|
|
|
|
list = AHCP_CALL (return NULL, Object, getSupportedPreviewFormats);
|
|
|
|
iterator = java_util_list_iterator (env, list);
|
|
if (iterator) {
|
|
while (java_util_iterator_has_next (env, iterator)) {
|
|
jobject integer = java_util_iterator_next (env, iterator);
|
|
|
|
if (integer) {
|
|
jint value = java_lang_integer_int_value (env, integer);
|
|
|
|
ret = g_list_append (ret, GINT_TO_POINTER (value));
|
|
(*env)->DeleteLocalRef (env, integer);
|
|
}
|
|
}
|
|
(*env)->DeleteLocalRef (env, iterator);
|
|
}
|
|
(*env)->DeleteLocalRef (env, list);
|
|
|
|
return ret;
|
|
}
|
|
|
|
void
|
|
gst_ahc_parameters_supported_preview_formats_free (GList * list)
|
|
{
|
|
g_list_free (list);
|
|
}
|
|
|
|
GList *
|
|
gst_ahc_parameters_get_supported_preview_fps_range (GstAHCParameters * self)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
jobject list = NULL;
|
|
jobject iterator = NULL;
|
|
GList *ret = NULL;
|
|
|
|
list = AHCP_CALL (return NULL, Object, getSupportedPreviewFpsRange);
|
|
|
|
iterator = java_util_list_iterator (env, list);
|
|
if (iterator) {
|
|
while (java_util_iterator_has_next (env, iterator)) {
|
|
jintArray range = java_util_iterator_next (env, iterator);
|
|
|
|
if (range) {
|
|
jint *fps = malloc (sizeof (jint) * 2);
|
|
|
|
(*env)->GetIntArrayRegion (env, range, 0, 2, fps);
|
|
ret = g_list_append (ret, fps);
|
|
(*env)->DeleteLocalRef (env, range);
|
|
}
|
|
}
|
|
(*env)->DeleteLocalRef (env, iterator);
|
|
}
|
|
(*env)->DeleteLocalRef (env, list);
|
|
|
|
return ret;
|
|
}
|
|
|
|
void
|
|
gst_ahc_parameters_supported_preview_fps_range_free (GList * list)
|
|
{
|
|
g_list_free_full (list, (GDestroyNotify) g_free);
|
|
}
|
|
|
|
GList *
|
|
gst_ahc_parameters_get_supported_preview_sizes (GstAHCParameters * self)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
jobject list = NULL;
|
|
jobject iterator = NULL;
|
|
GList *ret = NULL;
|
|
|
|
list = AHCP_CALL (return NULL, Object, getSupportedPreviewSizes);
|
|
|
|
iterator = java_util_list_iterator (env, list);
|
|
if (iterator) {
|
|
while (java_util_iterator_has_next (env, iterator)) {
|
|
jobject jsize = java_util_iterator_next (env, iterator);
|
|
|
|
if (jsize) {
|
|
jint width, height;
|
|
|
|
width = (*env)->GetIntField (env, jsize,
|
|
android_hardware_camera_size.width);
|
|
height = (*env)->GetIntField (env, jsize,
|
|
android_hardware_camera_size.height);
|
|
|
|
ret = g_list_append (ret, gst_ahc_size_new (width, height));
|
|
(*env)->DeleteLocalRef (env, jsize);
|
|
}
|
|
}
|
|
(*env)->DeleteLocalRef (env, iterator);
|
|
}
|
|
(*env)->DeleteLocalRef (env, list);
|
|
|
|
return ret;
|
|
}
|
|
|
|
void
|
|
gst_ahc_parameters_supported_preview_sizes_free (GList * list)
|
|
{
|
|
g_list_free_full (list, (GDestroyNotify) gst_ahc_size_free);
|
|
}
|
|
|
|
gboolean
|
|
gst_ahc_parameters_set_preview_format (GstAHCParameters * self, gint format)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
|
|
AHCP_CALL (return FALSE, Void, setPreviewFormat, format);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
gst_ahc_parameters_set_preview_fps_range (GstAHCParameters * self,
|
|
gint min, gint max)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
|
|
AHCP_CALL (return FALSE, Void, setPreviewFpsRange, min, max);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
gst_ahc_parameters_set_preview_size (GstAHCParameters * self,
|
|
gint width, gint height)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
|
|
AHCP_CALL (return FALSE, Void, setPreviewSize, width, height);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
gst_ahc_parameters_unflatten (GstAHCParameters * self, gchar * flattened)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
jstring v_str = NULL;
|
|
gboolean ret = TRUE;
|
|
|
|
v_str = (*env)->NewStringUTF (env, flattened);
|
|
if (v_str == NULL)
|
|
return FALSE;
|
|
|
|
AHCP_CALL (ret = FALSE, Void, unflatten, v_str);
|
|
|
|
(*env)->DeleteLocalRef (env, v_str);
|
|
|
|
return ret;
|
|
}
|
|
|
|
void
|
|
gst_ahc_parameters_free (GstAHCParameters * self)
|
|
{
|
|
JNIEnv *env = gst_dvm_get_env ();
|
|
|
|
(*env)->DeleteGlobalRef (env, self->object);
|
|
g_slice_free (GstAHCParameters, self);
|
|
}
|