diff options
author | Josep Torra <n770galaxy@gmail.com> | 2013-04-11 17:35:19 +0200 |
---|---|---|
committer | Josep Torra <n770galaxy@gmail.com> | 2013-05-10 13:35:48 +0200 |
commit | c10031f371e5e26075b7ae79e9984c36d68c4231 (patch) | |
tree | 7058407f4276439444067ab7775c376d538671b7 | |
parent | 64ef1f0e11ee9f49c7b253f8d8affc561c394321 (diff) | |
download | gst-omx-c10031f371e5e26075b7ae79e9984c36d68c4231.tar.gz |
examples: add an example aplication based OpenGL ES + EGL
Application that shows how to integrate playbin with an OpenGL ES
scene through EGL. Renders a video on the surfaces of an animated cube.
The code is not ported to 1.x so it's not built by default.
-rw-r--r-- | Makefile.am | 4 | ||||
-rw-r--r-- | configure.ac | 4 | ||||
-rw-r--r-- | examples/Makefile.am | 5 | ||||
-rw-r--r-- | examples/egl/Makefile.am | 19 | ||||
-rw-r--r-- | examples/egl/cube_texture_and_coords.h | 100 | ||||
-rw-r--r-- | examples/egl/testegl.c | 1339 |
6 files changed, 1471 insertions, 0 deletions
diff --git a/Makefile.am b/Makefile.am index 86710b3..e993476 100644 --- a/Makefile.am +++ b/Makefile.am @@ -1,5 +1,9 @@ SUBDIRS = common omx tools config +# if BUILD_EXAMPLES +# SUBDIRS += examples +# endif + ACLOCAL_AMFLAGS = -I m4 -I common/m4 DISTCLEANFILES = _stdint.h diff --git a/configure.ac b/configure.ac index 21227c6..447e155 100644 --- a/configure.ac +++ b/configure.ac @@ -149,7 +149,9 @@ AG_GST_CHECK_GST_PLUGINS_BASE($GST_API_VERSION, [$GST_REQ], yes) AM_CONDITIONAL(HAVE_GST_CHECK, test "x$HAVE_GST_CHECK" = "xyes") PKG_CHECK_MODULES([GST_EGL], [gstreamer-egl-1.0], [ AC_DEFINE(HAVE_GST_EGL, 1, [Have gstreamer-egl]) + GST_EGL=yes ]) +AM_CONDITIONAL(HAVE_GST_EGL, test "x$GST_EGL" = "xyes") dnl Check for documentation xrefs GLIB_PREFIX="`$PKG_CONFIG --variable=prefix glib-2.0`" @@ -361,6 +363,8 @@ tools/Makefile config/Makefile config/bellagio/Makefile config/rpi/Makefile +examples/Makefile +examples/egl/Makefile ) AC_OUTPUT diff --git a/examples/Makefile.am b/examples/Makefile.am new file mode 100644 index 0000000..8094d86 --- /dev/null +++ b/examples/Makefile.am @@ -0,0 +1,5 @@ +if HAVE_GST_EGL +SUBDIRS = egl +endif + +DIST_SUBDIRS = egl diff --git a/examples/egl/Makefile.am b/examples/egl/Makefile.am new file mode 100644 index 0000000..0aaa2e6 --- /dev/null +++ b/examples/egl/Makefile.am @@ -0,0 +1,19 @@ +noinst_PROGRAMS = testegl + +testegl_SOURCES = testegl.c + +noinst_HEADERS = cube_texture_and_coords.h + +testegl_LDADD = \ + $(GST_PLUGINS_BASE_LIBS) \ + -lgstapp-@GST_MAJORMINOR@ \ + $(GST_BASE_LIBS) \ + $(GST_LIBS) \ + $(GST_EGL_LIBS) + +testegl_CFLAGS = \ + $(GST_PLUGINS_BASE_CFLAGS) \ + $(GST_BASE_CFLAGS) \ + $(GST_CFLAGS) \ + $(GST_EGL_CFLAGS) + diff --git a/examples/egl/cube_texture_and_coords.h b/examples/egl/cube_texture_and_coords.h new file mode 100644 index 0000000..663e23b --- /dev/null +++ b/examples/egl/cube_texture_and_coords.h @@ -0,0 +1,100 @@ +/* +Copyright (c) 2012, Broadcom Europe Ltd +All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are met: + * Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above copyright + notice, this list of conditions and the following disclaimer in the + documentation and/or other materials provided with the distribution. + * Neither the name of the copyright holder nor the + names of its contributors may be used to endorse or promote products + derived from this software without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE +DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY +DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES +(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; +LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND +ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS +SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +*/ + +// Spatial coordinates for the cube + +static const GLbyte quadx[6*4*3] = { + /* FRONT */ + -10, -10, 10, + 10, -10, 10, + -10, 10, 10, + 10, 10, 10, + + /* BACK */ + -10, -10, -10, + -10, 10, -10, + 10, -10, -10, + 10, 10, -10, + + /* LEFT */ + -10, -10, 10, + -10, 10, 10, + -10, -10, -10, + -10, 10, -10, + + /* RIGHT */ + 10, -10, -10, + 10, 10, -10, + 10, -10, 10, + 10, 10, 10, + + /* TOP */ + -10, 10, 10, + 10, 10, 10, + -10, 10, -10, + 10, 10, -10, + + /* BOTTOM */ + -10, -10, 10, + -10, -10, -10, + 10, -10, 10, + 10, -10, -10, +}; + +/** Texture coordinates for the quad. */ +static const GLfloat texCoords[6 * 4 * 2] = { + 0.f, 0.f, + 1.f, 0.f, + 0.f, 1.f, + 1.f, 1.f, + + 0.f, 0.f, + 1.f, 0.f, + 0.f, 1.f, + 1.f, 1.f, + + 0.f, 0.f, + 1.f, 0.f, + 0.f, 1.f, + 1.f, 1.f, + + 0.f, 0.f, + 1.f, 0.f, + 0.f, 1.f, + 1.f, 1.f, + + 0.f, 0.f, + 1.f, 0.f, + 0.f, 1.f, + 1.f, 1.f, + + 0.f, 0.f, + 1.f, 0.f, + 0.f, 1.f, + 1.f, 1.f, +}; + diff --git a/examples/egl/testegl.c b/examples/egl/testegl.c new file mode 100644 index 0000000..e482329 --- /dev/null +++ b/examples/egl/testegl.c @@ -0,0 +1,1339 @@ +/* +Copyright (c) 2012, Broadcom Europe Ltd +Copyright (c) 2012, OtherCrashOverride +Copyright (C) 2013, Fluendo S.A. + @author: Josep Torra <josep@fluendo.com> +All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are met: + * Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above copyright + notice, this list of conditions and the following disclaimer in the + documentation and/or other materials provided with the distribution. + * Neither the name of the copyright holder nor the + names of its contributors may be used to endorse or promote products + derived from this software without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE +DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY +DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES +(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; +LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND +ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS +SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +*/ + +/* A rotating cube rendered with OpenGL|ES and video played using GStreamer on + * the cube faces. + */ + +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif + +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#include <math.h> +#include <assert.h> +#include <unistd.h> + +#include <gst/gst.h> + +#if defined (USE_OMX_TARGET_RPI) && defined (HAVE_GST_EGL) && defined (__GNUC__) +#ifndef __VCCOREVER__ +#define __VCCOREVER__ 0x04000000 +#endif +#pragma GCC diagnostic push +#pragma GCC diagnostic ignored "-Wredundant-decls" +#pragma GCC optimize ("gnu89-inline") +#endif + +#include "bcm_host.h" + +#if defined (USE_OMX_TARGET_RPI) && defined (HAVE_GST_EGL) && defined (__GNUC__) +#pragma GCC reset_options +#pragma GCC diagnostic pop +#endif + +#include <GLES/gl.h> +#include <GLES/glext.h> + +#if defined (USE_OMX_TARGET_RPI) && defined (HAVE_GST_EGL) && defined (__GNUC__) +#ifndef __VCCOREVER__ +#define __VCCOREVER__ 0x04000000 +#endif +#pragma GCC diagnostic push +#pragma GCC diagnostic ignored "-Wredundant-decls" +#pragma GCC optimize ("gnu89-inline") +#endif + +#define EGL_EGLEXT_PROTOTYPES +#include <gst/egl/egl.h> + +#if defined (USE_OMX_TARGET_RPI) && defined (HAVE_GST_EGL) && defined (__GNUC__) +#pragma GCC reset_options +#pragma GCC diagnostic pop +#endif + +#include "cube_texture_and_coords.h" + +#ifndef M_PI +#define M_PI 3.141592654 +#endif + +#define TRACE_VC_MEMORY_ENABLED 0 + +#if TRACE_VC_MEMORY_ENABLED +#define TRACE_VC_MEMORY(str) \ + fprintf (stderr, "\n\n" str "\n"); \ + system ("vcdbg reloc >&2") + +#define TRACE_VC_MEMORY_DEFINE_ID(id) \ + static int id = 0 + +#define TRACE_VC_MEMORY_RESET_ID(id) \ + G_STMT_START { \ + id = 0; \ + } G_STMT_END + +#define TRACE_VC_MEMORY_ONCE_FOR_ID(str,id) \ + G_STMT_START { \ + if (id == 0) { \ + fprintf (stderr, "\n\n" str "\n"); \ + system ("vcdbg reloc >&2"); \ + id = 1; \ + } \ + } G_STMT_END + +#define TRACE_VC_MEMORY_ONCE(str,id) \ + G_STMT_START { \ + static int id = 0; \ + if (id == 0) { \ + fprintf (stderr, "\n\n" str "\n"); \ + system ("vcdbg reloc >&2"); \ + id = 1; \ + } \ + } G_STMT_END + +#else +#define TRACE_VC_MEMORY(str) while(0) +#define TRACE_VC_MEMORY_DEFINE_ID(id) +#define TRACE_VC_MEMORY_RESET_ID(id) while(0) +#define TRACE_VC_MEMORY_ONCE_FOR_ID(str,id) while(0) +#define TRACE_VC_MEMORY_ONCE(str,id) while(0) +#endif + + +typedef struct +{ + DISPMANX_DISPLAY_HANDLE_T dispman_display; + DISPMANX_ELEMENT_HANDLE_T dispman_element; + + uint32_t screen_width; + uint32_t screen_height; + gboolean animate; + gboolean sync_animation_with_video; + + /* OpenGL|ES objects */ + EGLDisplay display; + EGLSurface surface; + EGLContext context; + GLuint tex; + + /* model rotation vector and direction */ + GLfloat rot_angle_x_inc; + GLfloat rot_angle_y_inc; + GLfloat rot_angle_z_inc; + + /* current model rotation angles */ + GLfloat rot_angle_x; + GLfloat rot_angle_y; + GLfloat rot_angle_z; + + /* current distance from camera */ + GLfloat distance; + GLfloat distance_inc; + + /* GStreamer related resources */ + GstElement *pipeline; + GstEGLDisplay *gst_display; + + /* Interthread comunication */ + GAsyncQueue *queue; + GMutex *lock; + GCond *cond; + gboolean flushing; + GstMiniObject *popped_obj; + GstEGLImageMemory *current_mem; + + /* GLib mainloop */ + GMainLoop *main_loop; +} APP_STATE_T; + +static void init_ogl (APP_STATE_T * state); +static void init_model_proj (APP_STATE_T * state); +static void reset_model (APP_STATE_T * state); +static GLfloat inc_and_wrap_angle (GLfloat angle, GLfloat angle_inc); +static GLfloat inc_and_clip_distance (GLfloat distance, GLfloat distance_inc); +static void redraw_scene (APP_STATE_T * state); +static void update_model (APP_STATE_T * state); +static void init_textures (APP_STATE_T * state); +static APP_STATE_T _state, *state = &_state; + +TRACE_VC_MEMORY_DEFINE_ID (gid0); +TRACE_VC_MEMORY_DEFINE_ID (gid1); +TRACE_VC_MEMORY_DEFINE_ID (gid2); + +/*********************************************************** + * Name: init_ogl + * + * Arguments: + * APP_STATE_T *state - holds OGLES model info + * + * Description: Sets the display, OpenGL|ES context and screen stuff + * + * Returns: void + * + ***********************************************************/ +static void +init_ogl (APP_STATE_T * state) +{ + int32_t success = 0; + EGLBoolean result; + EGLint num_config; + + static EGL_DISPMANX_WINDOW_T nativewindow; + + DISPMANX_UPDATE_HANDLE_T dispman_update; + VC_RECT_T dst_rect; + VC_RECT_T src_rect; + + static const EGLint attribute_list[] = { + EGL_RED_SIZE, 8, + EGL_GREEN_SIZE, 8, + EGL_BLUE_SIZE, 8, + EGL_ALPHA_SIZE, 8, + EGL_DEPTH_SIZE, 16, + //EGL_SAMPLES, 4, + EGL_SURFACE_TYPE, EGL_WINDOW_BIT, + EGL_NONE + }; + + EGLConfig config; + + /* get an EGL display connection */ + state->display = eglGetDisplay (EGL_DEFAULT_DISPLAY); + assert (state->display != EGL_NO_DISPLAY); + + /* initialize the EGL display connection */ + result = eglInitialize (state->display, NULL, NULL); + assert (EGL_FALSE != result); + + /* get an appropriate EGL frame buffer configuration + * this uses a BRCM extension that gets the closest match, rather + * than standard which returns anything that matches. */ + result = + eglSaneChooseConfigBRCM (state->display, attribute_list, &config, 1, + &num_config); + assert (EGL_FALSE != result); + + /* create an EGL rendering context */ + state->context = + eglCreateContext (state->display, config, EGL_NO_CONTEXT, NULL); + assert (state->context != EGL_NO_CONTEXT); + + /* create an EGL window surface */ + success = graphics_get_display_size (0 /* LCD */ , &state->screen_width, + &state->screen_height); + assert (success >= 0); + + dst_rect.x = 0; + dst_rect.y = 0; + dst_rect.width = state->screen_width; + dst_rect.height = state->screen_height; + + src_rect.x = 0; + src_rect.y = 0; + src_rect.width = state->screen_width << 16; + src_rect.height = state->screen_height << 16; + + state->dispman_display = vc_dispmanx_display_open (0 /* LCD */ ); + dispman_update = vc_dispmanx_update_start (0); + + state->dispman_element = + vc_dispmanx_element_add (dispman_update, state->dispman_display, + 0 /*layer */ , &dst_rect, 0 /*src */ , + &src_rect, DISPMANX_PROTECTION_NONE, 0 /*alpha */ , 0 /*clamp */ , + 0 /*transform */ ); + + nativewindow.element = state->dispman_element; + nativewindow.width = state->screen_width; + nativewindow.height = state->screen_height; + vc_dispmanx_update_submit_sync (dispman_update); + + state->surface = + eglCreateWindowSurface (state->display, config, &nativewindow, NULL); + assert (state->surface != EGL_NO_SURFACE); + + /* connect the context to the surface */ + result = + eglMakeCurrent (state->display, state->surface, state->surface, + state->context); + assert (EGL_FALSE != result); + + /* Set background color and clear buffers */ + glClearColor (0.15f, 0.25f, 0.35f, 1.0f); + + /* Enable back face culling. */ + glEnable (GL_CULL_FACE); + + glMatrixMode (GL_MODELVIEW); +} + +/*********************************************************** + * Name: init_model_proj + * + * Arguments: + * APP_STATE_T *state - holds OGLES model info + * + * Description: Sets the OpenGL|ES model to default values + * + * Returns: void + * + ***********************************************************/ +static void +init_model_proj (APP_STATE_T * state) +{ + float nearp = 1.0f; + float farp = 500.0f; + float hht; + float hwd; + + glHint (GL_PERSPECTIVE_CORRECTION_HINT, GL_NICEST); + + glViewport (0, 0, (GLsizei) state->screen_width, + (GLsizei) state->screen_height); + + glMatrixMode (GL_PROJECTION); + glLoadIdentity (); + + hht = nearp * (float) tan (45.0 / 2.0 / 180.0 * M_PI); + hwd = hht * (float) state->screen_width / (float) state->screen_height; + + glFrustumf (-hwd, hwd, -hht, hht, nearp, farp); + + glEnableClientState (GL_VERTEX_ARRAY); + glVertexPointer (3, GL_BYTE, 0, quadx); + + reset_model (state); +} + +/*********************************************************** + * Name: reset_model + * + * Arguments: + * APP_STATE_T *state - holds OGLES model info + * + * Description: Resets the Model projection and rotation direction + * + * Returns: void + * + ***********************************************************/ +static void +reset_model (APP_STATE_T * state) +{ + /* reset model position */ + glMatrixMode (GL_MODELVIEW); + glLoadIdentity (); + glTranslatef (0.f, 0.f, -50.f); + + /* reset model rotation */ + state->rot_angle_x = 45.f; + state->rot_angle_y = 30.f; + state->rot_angle_z = 0.f; + state->rot_angle_x_inc = 0.5f; + state->rot_angle_y_inc = 0.5f; + state->rot_angle_z_inc = 0.f; + state->distance = 40.f; +} + +/*********************************************************** + * Name: update_model + * + * Arguments: + * APP_STATE_T *state - holds OGLES model info + * + * Description: Updates model projection to current position/rotation + * + * Returns: void + * + ***********************************************************/ +static void +update_model (APP_STATE_T * state) +{ + if (state->animate) { + /* update position */ + state->rot_angle_x = + inc_and_wrap_angle (state->rot_angle_x, state->rot_angle_x_inc); + state->rot_angle_y = + inc_and_wrap_angle (state->rot_angle_y, state->rot_angle_y_inc); + state->rot_angle_z = + inc_and_wrap_angle (state->rot_angle_z, state->rot_angle_z_inc); + state->distance = + inc_and_clip_distance (state->distance, state->distance_inc); + } + + glLoadIdentity (); + /* move camera back to see the cube */ + glTranslatef (0.f, 0.f, -state->distance); + + /* Rotate model to new position */ + glRotatef (state->rot_angle_x, 1.f, 0.f, 0.f); + glRotatef (state->rot_angle_y, 0.f, 1.f, 0.f); + glRotatef (state->rot_angle_z, 0.f, 0.f, 1.f); +} + +/*********************************************************** + * Name: inc_and_wrap_angle + * + * Arguments: + * GLfloat angle current angle + * GLfloat angle_inc angle increment + * + * Description: Increments or decrements angle by angle_inc degrees + * Wraps to 0 at 360 deg. + * + * Returns: new value of angle + * + ***********************************************************/ +static GLfloat +inc_and_wrap_angle (GLfloat angle, GLfloat angle_inc) +{ + angle += angle_inc; + + if (angle >= 360.0) + angle -= 360.f; + else if (angle <= 0) + angle += 360.f; + + return angle; +} + +/*********************************************************** + * Name: inc_and_clip_distance + * + * Arguments: + * GLfloat distance current distance + * GLfloat distance_inc distance increment + * + * Description: Increments or decrements distance by distance_inc units + * Clips to range + * + * Returns: new value of angle + * + ***********************************************************/ +static GLfloat +inc_and_clip_distance (GLfloat distance, GLfloat distance_inc) +{ + distance += distance_inc; + + if (distance >= 120.0f) + distance = 120.f; + else if (distance <= 40.0f) + distance = 40.0f; + + return distance; +} + +/*********************************************************** + * Name: redraw_scene + * + * Arguments: + * APP_STATE_T *state - holds OGLES model info + * + * Description: Draws the model and calls eglSwapBuffers + * to render to screen + * + * Returns: void + * + ***********************************************************/ +static void +redraw_scene (APP_STATE_T * state) +{ + /* Start with a clear screen */ + glClear (GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); + + /* Need to rotate textures - do this by rotating each cube face */ + glRotatef (270.f, 0.f, 0.f, 1.f); /* front face normal along z axis */ + + /* draw first 4 vertices */ + glDrawArrays (GL_TRIANGLE_STRIP, 0, 4); + + /* same pattern for other 5 faces - rotation chosen to make image orientation 'nice' */ + glRotatef (90.f, 0.f, 0.f, 1.f); /* back face normal along z axis */ + glDrawArrays (GL_TRIANGLE_STRIP, 4, 4); + + glRotatef (90.f, 1.f, 0.f, 0.f); /* left face normal along x axis */ + glDrawArrays (GL_TRIANGLE_STRIP, 8, 4); + + glRotatef (90.f, 1.f, 0.f, 0.f); /* right face normal along x axis */ + glDrawArrays (GL_TRIANGLE_STRIP, 12, 4); + + glRotatef (270.f, 0.f, 1.f, 0.f); /* top face normal along y axis */ + glDrawArrays (GL_TRIANGLE_STRIP, 16, 4); + + glRotatef (90.f, 0.f, 1.f, 0.f); /* bottom face normal along y axis */ + glDrawArrays (GL_TRIANGLE_STRIP, 20, 4); + + eglSwapBuffers (state->display, state->surface); +} + +/*********************************************************** + * Name: init_textures + * + * Arguments: + * APP_STATE_T *state - holds OGLES model info + * + * Description: Initialise OGL|ES texture surfaces to use image + * buffers + * + * Returns: void + * + ***********************************************************/ +static void +init_textures (APP_STATE_T * state) +{ + glGenTextures (1, &state->tex); + + glBindTexture (GL_TEXTURE_2D, state->tex); + +#if 0 + glTexParameteri (GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST); + glTexParameteri (GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST); +#else + glTexParameteri (GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); + glTexParameteri (GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); +#endif + + glTexParameteri (GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); + glTexParameteri (GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE); + + /* setup overall texture environment */ + glTexCoordPointer (2, GL_FLOAT, 0, texCoords); + glEnableClientState (GL_TEXTURE_COORD_ARRAY); + + glEnable (GL_TEXTURE_2D); + + /* Bind texture surface to current vertices */ + glBindTexture (GL_TEXTURE_2D, state->tex); +} + +static void +destroy_pool_resources (GstEGLImageMemoryPool * pool, gpointer user_data) +{ + APP_STATE_T *state = (APP_STATE_T *) user_data; + gint i, size = gst_egl_image_memory_pool_get_size (pool); + EGLClientBuffer client_buffer; + EGLImageKHR image; + EGLint error; + + TRACE_VC_MEMORY ("before pool destruction"); + for (i = 0; i < size; i++) { + if (gst_egl_image_memory_pool_get_resources (pool, i, &client_buffer, + &image)) { + GLuint tid = (GLuint) client_buffer; + error = EGL_SUCCESS; + + if (image != EGL_NO_IMAGE_KHR) { + eglDestroyImageKHR (state->display, image); + if ((error = eglGetError ()) != EGL_SUCCESS) { + g_print ("eglDestroyImageKHR failed %x\n", error); + } + } + + if (tid) { + error = GL_NO_ERROR; + glDeleteTextures (1, &tid); + if ((error = glGetError ()) != GL_NO_ERROR) { + g_print ("glDeleteTextures failed %x\n", error); + } + } + g_print ("destroyed texture %x image %p\n", tid, image); + } + } + TRACE_VC_MEMORY ("after pool destruction"); +} + +static GstEGLImageMemoryPool * +create_pool (APP_STATE_T * state, gint size, gint width, gint height) +{ + GstEGLImageMemoryPool *pool; + gint i; + EGLint error; + + TRACE_VC_MEMORY ("before pool creation"); + pool = gst_egl_image_memory_pool_new (size, state->gst_display, state, + destroy_pool_resources); + + for (i = 0; i < size; i++) { + GLuint tid; + EGLImageKHR image; + + error = GL_NO_ERROR; + glGenTextures (1, &tid); + if ((error = glGetError ()) != GL_NO_ERROR) { + g_print ("glGenTextures failed %x\n", error); + goto failed; + } + + glBindTexture (GL_TEXTURE_2D, tid); + glTexImage2D (GL_TEXTURE_2D, 0, GL_RGBA, width, height, 0, GL_RGBA, + GL_UNSIGNED_BYTE, NULL); + if ((error = glGetError ()) != GL_NO_ERROR) { + g_print ("glTexImage2D failed %x\n", error); + goto failed; + } + + /* Create EGL Image */ + error = EGL_SUCCESS; + image = eglCreateImageKHR (state->display, + state->context, EGL_GL_TEXTURE_2D_KHR, (EGLClientBuffer) tid, 0); + + if (image == EGL_NO_IMAGE_KHR) { + if ((error = eglGetError ()) != EGL_SUCCESS) { + g_print ("eglCreateImageKHR failed %x\n", error); + } else { + g_print ("eglCreateImageKHR failed.\n"); + } + goto failed; + } + g_print ("created texture %x image %p\n", tid, image); + gst_egl_image_memory_pool_set_resources (pool, i, (EGLClientBuffer) tid, + image); + } + + TRACE_VC_MEMORY ("after pool creation"); + TRACE_VC_MEMORY_RESET_ID (gid0); + TRACE_VC_MEMORY_RESET_ID (gid1); + TRACE_VC_MEMORY_RESET_ID (gid2); + + return pool; + +failed: + gst_egl_image_memory_pool_unref (pool); + return NULL; +} + +static gboolean +render_scene (APP_STATE_T * state) +{ + update_model (state); + redraw_scene (state); + TRACE_VC_MEMORY_ONCE_FOR_ID ("after render_scene", gid2); + + return !state->sync_animation_with_video; +} + +static void +update_image (APP_STATE_T * state, GstBuffer * buffer) +{ + GstEGLImageMemory *mem = (GstEGLImageMemory *) GST_BUFFER_DATA (buffer); + + g_mutex_lock (state->lock); + if (state->current_mem) { + gst_egl_image_memory_unref (state->current_mem); + } + state->current_mem = gst_egl_image_memory_ref (mem); + g_mutex_unlock (state->lock); + + TRACE_VC_MEMORY_ONCE_FOR_ID ("before glEGLImageTargetTexture2DOES", gid0); + glBindTexture (GL_TEXTURE_2D, state->tex); + glEGLImageTargetTexture2DOES (GL_TEXTURE_2D, + gst_egl_image_memory_get_image (mem)); + TRACE_VC_MEMORY_ONCE_FOR_ID ("after glEGLImageTargetTexture2DOES", gid1); + + if (state->sync_animation_with_video) { + render_scene (state); + } +} + +static void +init_intercom (APP_STATE_T * state) +{ + state->queue = + g_async_queue_new_full ((GDestroyNotify) gst_mini_object_unref); + state->lock = g_mutex_new (); + state->cond = g_cond_new (); +} + +static void +flush_internal (APP_STATE_T * state) +{ + if (state->current_mem) { + gst_egl_image_memory_unref (state->current_mem); + } + state->current_mem = NULL; +} + +static void +flush_start (APP_STATE_T * state) +{ + GstMiniObject *object = NULL; + + g_mutex_lock (state->lock); + state->flushing = TRUE; + g_cond_broadcast (state->cond); + g_mutex_unlock (state->lock); + + while ((object = g_async_queue_try_pop (state->queue))) { + gst_mini_object_unref (object); + } + + flush_internal (state); +} + +static void +flush_stop (APP_STATE_T * state) +{ + g_mutex_lock (state->lock); + state->popped_obj = NULL; + state->flushing = FALSE; + g_mutex_unlock (state->lock); +} + +static void +pipeline_pause (APP_STATE_T * state) +{ + flush_start (state); + gst_element_set_state (state->pipeline, GST_STATE_PAUSED); + flush_stop (state); +} + +static gint64 +pipeline_get_position (APP_STATE_T * state) +{ + gint64 position = -1; + + if (state->pipeline) { + gst_element_query_position (state->pipeline, GST_FORMAT_TIME, &position); + } + + return position; +} + +static gint64 +pipeline_get_duration (APP_STATE_T * state) +{ + gint64 duration = -1; + + if (state->pipeline) { + gst_element_query_duration (state->pipeline, GST_FORMAT_TIME, &duration); + } + + return duration; +} + +static void +pipeline_seek (APP_STATE_T * state, gint64 position) +{ + if (state->pipeline) { + GstEvent *event; + event = gst_event_new_seek (1.0, + GST_FORMAT_TIME, GST_SEEK_FLAG_FLUSH | GST_SEEK_FLAG_KEY_UNIT, + GST_SEEK_TYPE_SET, position, GST_SEEK_TYPE_SET, GST_CLOCK_TIME_NONE); + if (!gst_element_send_event (state->pipeline, event)) { + g_print ("seek failed\n"); + } + } +} + +static gboolean +handle_queued_objects (APP_STATE_T * state) +{ + GstMiniObject *object = NULL; + + if (g_async_queue_length (state->queue) == 0) { + return FALSE; + } + + while ((object = g_async_queue_try_pop (state->queue))) { + + g_mutex_lock (state->lock); + if (state->flushing) { + state->popped_obj = object; + gst_mini_object_unref (object); + g_cond_broadcast (state->cond); + g_mutex_unlock (state->lock); + continue; + } + g_mutex_unlock (state->lock); + + if (GST_IS_BUFFER (object)) { + GstBuffer *buffer = GST_BUFFER_CAST (object); + update_image (state, buffer); + gst_buffer_unref (buffer); + } else if (GST_IS_MESSAGE (object)) { + GstMessage *message = GST_MESSAGE_CAST (object); + g_print ("\nmessage %p ", message); + if (gst_structure_has_name (message->structure, "need-egl-pool")) { + GstElement *element = GST_ELEMENT (GST_MESSAGE_SRC (message)); + gint size, width, height; + + gst_message_parse_need_egl_pool (message, &size, &width, &height); + + g_print ("need-egl-pool, size %d width %d height %d\n", size, width, + height); + + if (g_object_class_find_property (G_OBJECT_GET_CLASS (element), "pool")) { + GstEGLImageMemoryPool *pool = NULL; + + if ((pool = create_pool (state, size, width, height))) { + g_object_set (element, "pool", pool, NULL); + } + } + } + gst_message_unref (message); + } else if (GST_IS_EVENT (object)) { + GstEvent *event = GST_EVENT_CAST (object); + g_print ("\nevent %p %s\n", event, + gst_event_type_get_name (GST_EVENT_TYPE (event))); + + g_mutex_lock (state->lock); + switch (GST_EVENT_TYPE (event)) { + case GST_EVENT_EOS: + flush_internal (state); + break; + default: + break; + } + g_mutex_unlock (state->lock); + gst_event_unref (event); + } + + g_mutex_lock (state->lock); + state->popped_obj = object; + g_cond_broadcast (state->cond); + g_mutex_unlock (state->lock); + } + + return FALSE; +} + +static gboolean +queue_object (APP_STATE_T * state, GstMiniObject * obj, gboolean synchronous) +{ + g_mutex_lock (state->lock); + if (state->flushing) { + g_mutex_unlock (state->lock); + gst_mini_object_unref (obj); + return FALSE; + } + + g_async_queue_push (state->queue, obj); + + + if (state->sync_animation_with_video) { + g_idle_add_full (G_PRIORITY_HIGH_IDLE, (GSourceFunc) handle_queued_objects, + state, NULL); + } + + if (synchronous) { + /* Waiting for object to be handled */ + do { + g_cond_wait (state->cond, state->lock); + } while (!state->flushing && state->popped_obj != obj); + } + g_mutex_unlock (state->lock); + + return TRUE; +} + +static gboolean +handle_msgs_and_render_scene (APP_STATE_T * state) +{ + handle_queued_objects (state); + return render_scene (state); +} + +static void +preroll_cb (GstElement * fakesink, GstBuffer * buffer, GstPad * pad, + gpointer user_data) +{ + APP_STATE_T *state = (APP_STATE_T *) user_data; + queue_object (state, GST_MINI_OBJECT_CAST (gst_buffer_ref (buffer)), FALSE); +} + +static void +buffers_cb (GstElement * fakesink, GstBuffer * buffer, GstPad * pad, + gpointer user_data) +{ + APP_STATE_T *state = (APP_STATE_T *) user_data; + queue_object (state, GST_MINI_OBJECT_CAST (gst_buffer_ref (buffer)), TRUE); +} + +static gboolean +events_cb (GstPad * pad, GstEvent * event, gpointer user_data) +{ + APP_STATE_T *state = (APP_STATE_T *) user_data; + + switch (GST_EVENT_TYPE (event)) { + case GST_EVENT_FLUSH_START: + flush_start (state); + break; + case GST_EVENT_FLUSH_STOP: + flush_stop (state); + break; + case GST_EVENT_EOS: + queue_object (state, GST_MINI_OBJECT_CAST (gst_event_ref (event)), TRUE); + break; + default: + break; + } + + return TRUE; +} + +static gboolean +init_playbin_player (APP_STATE_T * state, const gchar * uri) +{ + GstElement *asink; + GstElement *vsink; + + vsink = gst_element_factory_make ("fakesink", "vsink"); + g_object_set (vsink, "sync", TRUE, "silent", TRUE, + "enable-last-buffer", FALSE, + "max-lateness", 20 * GST_MSECOND, "signal-handoffs", TRUE, NULL); + + g_signal_connect (vsink, "preroll-handoff", G_CALLBACK (preroll_cb), state); + g_signal_connect (vsink, "handoff", G_CALLBACK (buffers_cb), state); + + gst_pad_add_event_probe (gst_element_get_static_pad (vsink, "sink"), + G_CALLBACK (events_cb), state); + +#if 0 + asink = gst_element_factory_make ("fakesink", "asink"); + g_object_set (asink, "sync", TRUE, "silent", TRUE, NULL); +#else + asink = gst_element_factory_make ("alsasink", "asink"); +#endif + + /* Instantiate and configure playbin */ + state->pipeline = gst_element_factory_make ("playbin", "player"); + g_object_set (state->pipeline, "uri", uri, + "video-sink", vsink, "audio-sink", asink, NULL); + + return TRUE; +} + +static gboolean +init_parse_launch_player (APP_STATE_T * state, const gchar * spipeline) +{ + GstElement *vsink; + GError *error = NULL; + + state->pipeline = gst_parse_launch (spipeline, &error); + + if (!state->pipeline) { + g_printerr ("Unable to instatiate pipeline '%s': %s\n", + spipeline, error->message); + return FALSE; + } + + vsink = gst_bin_get_by_name (GST_BIN (state->pipeline), "vsink"); + + if (!vsink) { + g_printerr ("Unable to find a fakesink named 'vsink'"); + return FALSE; + } + + g_object_set (vsink, "sync", TRUE, "silent", TRUE, + "enable-last-buffer", FALSE, + "max-lateness", 20 * GST_MSECOND, "signal-handoffs", TRUE, NULL); + + g_signal_connect (vsink, "preroll-handoff", G_CALLBACK (preroll_cb), state); + g_signal_connect (vsink, "handoff", G_CALLBACK (buffers_cb), state); + + gst_pad_add_event_probe (gst_element_get_static_pad (vsink, "sink"), + G_CALLBACK (events_cb), state); + + return TRUE; +} + +//------------------------------------------------------------------------------ + +static void +report_position_duration (APP_STATE_T * state) +{ + gint64 position, duration; + + duration = pipeline_get_duration (state); + position = pipeline_get_position (state); + + if (position != -1) { + g_print ("\n position / duration: %" GST_TIME_FORMAT, + GST_TIME_ARGS (position)); + } else { + g_print ("\n position / duration: unknown"); + } + + if (duration != -1) { + g_print (" / %" GST_TIME_FORMAT, GST_TIME_ARGS (duration)); + } else { + g_print (" / unknown"); + } + g_print ("\n"); +} + +static void +seek_forward (APP_STATE_T * state) +{ + gint64 position, duration; + + duration = pipeline_get_duration (state); + position = pipeline_get_position (state); + + if (position != -1) { + position += 30 * GST_SECOND; + if (duration != -1) { + position = MIN (position, duration); + } + pipeline_seek (state, position); + } +} + +static void +seek_backward (APP_STATE_T * state) +{ + gint64 position; + + position = pipeline_get_position (state); + + if (position != -1) { + position -= 30 * GST_SECOND; + position = MAX (position, 0); + pipeline_seek (state, position); + } +} + +#define SKIP(t) \ + while (*t) { \ + if ((*t == ' ') || (*t == '\n') || (*t == '\t') || (*t == '\r')) \ + t++; \ + else \ + break; \ + } + +/* Process keyboard input */ +static gboolean +handle_keyboard (GIOChannel * source, GIOCondition cond, APP_STATE_T * state) +{ + gchar *str = NULL; + char op; + + if (g_io_channel_read_line (source, &str, NULL, NULL, + NULL) == G_IO_STATUS_NORMAL) { + + gchar *cmd = str; + SKIP (cmd) + op = *cmd; + cmd++; + switch (op) { + case 'a': + if (state->animate) { + state->animate = FALSE; + } else { + state->animate = TRUE; + } + break; + case 'p': + pipeline_pause (state); + break; + case 'r': + gst_element_set_state (state->pipeline, GST_STATE_PLAYING); + break; + case 'l': + report_position_duration (state); + break; + case 'f': + seek_forward (state); + break; + case 'b': + seek_backward (state); + break; + case 'q': + flush_start (state); + gst_element_set_state (state->pipeline, GST_STATE_READY); + break; + case 'S': + if (state->sync_animation_with_video) { + state->sync_animation_with_video = FALSE; + /* Add the rendering task */ + g_idle_add_full (G_PRIORITY_HIGH_IDLE, + (GSourceFunc) handle_msgs_and_render_scene, state, NULL); + g_print ("\nanimation is not synchoronized with video\n"); + } else { + state->sync_animation_with_video = TRUE; + g_print ("\nanimation is synchoronized with video\n"); + } + break; + } + } + g_free (str); + return TRUE; +} + +static GstBusSyncReply +bus_sync_handler (GstBus * bus, GstMessage * message, GstPipeline * data) +{ + if ((GST_MESSAGE_TYPE (message) == GST_MESSAGE_ELEMENT) && + gst_structure_has_name (message->structure, "need-egl-pool")) { + queue_object (state, GST_MINI_OBJECT_CAST (gst_message_ref (message)), + TRUE); + } + return GST_BUS_PASS; +} + +/* on error print the error and quit the application */ +static void +error_cb (GstBus * bus, GstMessage * msg, APP_STATE_T * state) +{ + GError *err; + gchar *debug_info; + + gst_message_parse_error (msg, &err, &debug_info); + g_printerr ("Error received from element %s: %s\n", + GST_OBJECT_NAME (msg->src), err->message); + g_printerr ("Debugging information: %s\n", debug_info ? debug_info : "none"); + g_clear_error (&err); + g_free (debug_info); + flush_start (state); + gst_element_set_state (state->pipeline, GST_STATE_READY); +} + +/* buffering */ +static void +buffering_cb (GstBus * bus, GstMessage * msg, APP_STATE_T * state) +{ + gint percent; + + gst_message_parse_buffering (msg, &percent); + g_print ("Buffering %3d%%\r", percent); + if (percent < 100) + gst_element_set_state (state->pipeline, GST_STATE_PAUSED); + else { + g_print ("\n"); + gst_element_set_state (state->pipeline, GST_STATE_PLAYING); + } +} + +/* on EOS just quit the application */ +static void +eos_cb (GstBus * bus, GstMessage * msg, APP_STATE_T * state) +{ + if (GST_MESSAGE_SRC (msg) == GST_OBJECT (state->pipeline)) { + g_print ("End-Of-Stream reached.\n"); + flush_start (state); + gst_element_set_state (state->pipeline, GST_STATE_READY); + } +} + +static void +state_changed_cb (GstBus * bus, GstMessage * msg, APP_STATE_T * state) +{ + GstState old_state, new_state, pending_state; + if (GST_MESSAGE_SRC (msg) == GST_OBJECT (state->pipeline)) { + gst_message_parse_state_changed (msg, &old_state, &new_state, + &pending_state); + g_print ("State changed to %s\n", gst_element_state_get_name (new_state)); + if (old_state == GST_STATE_PAUSED && new_state == GST_STATE_READY) { + g_main_loop_quit (state->main_loop); + } + } +} + +//============================================================================== + +static void +close_ogl (void) +{ + DISPMANX_UPDATE_HANDLE_T dispman_update; + + /* clear screen */ + glClear (GL_COLOR_BUFFER_BIT); + eglSwapBuffers (state->display, state->surface); + + /* Release OpenGL resources */ + eglMakeCurrent (state->display, EGL_NO_SURFACE, EGL_NO_SURFACE, + EGL_NO_CONTEXT); + eglDestroySurface (state->display, state->surface); + eglDestroyContext (state->display, state->context); + gst_egl_display_unref (state->gst_display); + + dispman_update = vc_dispmanx_update_start (0); + vc_dispmanx_element_remove (dispman_update, state->dispman_element); + vc_dispmanx_update_submit_sync (dispman_update); + vc_dispmanx_display_close (state->dispman_display); +} + +//============================================================================== + +int +main (int argc, char **argv) +{ + GstBus *bus; + GOptionContext *ctx; + GIOChannel *io_stdin; + GError *err = NULL; + gboolean res; + GOptionEntry options[] = { + {NULL} + }; + + /* Clear application state */ + memset (state, 0, sizeof (*state)); + state->animate = TRUE; + state->sync_animation_with_video = TRUE; + + /* must initialise the threading system before using any other GLib funtion */ + if (!g_thread_supported ()) + g_thread_init (NULL); + + ctx = g_option_context_new ("[ADDITIONAL ARGUMENTS]"); + g_option_context_add_main_entries (ctx, options, NULL); + g_option_context_add_group (ctx, gst_init_get_option_group ()); + if (!g_option_context_parse (ctx, &argc, &argv, &err)) { + g_print ("Error initializing: %s\n", GST_STR_NULL (err->message)); + exit (1); + } + g_option_context_free (ctx); + + if (argc != 2) { + g_print ("Usage: %s <URI> or <PIPELINE-DESCRIPTION>\n", argv[0]); + exit (1); + } + + /* Initialize GStreamer */ + gst_init (&argc, &argv); + + TRACE_VC_MEMORY ("state 0"); + +#if defined (USE_OMX_TARGET_RPI) && defined (HAVE_GST_EGL) + bcm_host_init (); + TRACE_VC_MEMORY ("after bcm_host_init"); +#endif + + + /* Start OpenGLES */ + init_ogl (state); + TRACE_VC_MEMORY ("after init_ogl"); + + /* Wrap the EGL display */ + state->gst_display = gst_egl_display_new (state->display); + + /* Setup the model world */ + init_model_proj (state); + TRACE_VC_MEMORY ("after init_model_proj"); + + /* initialize the OGLES texture(s) */ + init_textures (state); + TRACE_VC_MEMORY ("after init_textures"); + + /* initialize inter thread comunnication */ + init_intercom (state); + + /* Initialize player */ + if (gst_uri_is_valid (argv[1])) { + res = init_playbin_player (state, argv[1]); + } else { + res = init_parse_launch_player (state, argv[1]); + } + + if (!res) + goto done; + + /* Create a GLib Main Loop and set it to run */ + state->main_loop = g_main_loop_new (NULL, FALSE); + + /* Add a keyboard watch so we get notified of keystrokes */ + io_stdin = g_io_channel_unix_new (fileno (stdin)); + g_io_add_watch (io_stdin, G_IO_IN, (GIOFunc) handle_keyboard, state); + g_io_channel_unref (io_stdin); + + /* *INDENT-OFF* */ + g_print ("Available commands: \n" + " a - Toggle animation \n" + " p - Pause playback \n" + " r - Resume playback \n" + " l - Query position/duration\n" + " f - Seek 30 seconds forward \n" + " b - Seek 30 seconds backward \n" + " S - Toggle synchronization of video and animation \n" + " q - Quit \n"); + /* *INDENT-ON* */ + + if (!state->sync_animation_with_video) { + /* Add the rendering task */ + g_idle_add_full (G_PRIORITY_HIGH_IDLE, + (GSourceFunc) handle_msgs_and_render_scene, state, NULL); + } + + /* Connect the bus handlers */ + bus = gst_element_get_bus (state->pipeline); + + gst_bus_set_sync_handler (bus, (GstBusSyncHandler) bus_sync_handler, state); + + gst_bus_add_signal_watch_full (bus, G_PRIORITY_HIGH); + gst_bus_enable_sync_message_emission (bus); + + g_signal_connect (G_OBJECT (bus), "message::error", (GCallback) error_cb, + state); + g_signal_connect (G_OBJECT (bus), "message::buffering", + (GCallback) buffering_cb, state); + g_signal_connect (G_OBJECT (bus), "message::eos", (GCallback) eos_cb, state); + g_signal_connect (G_OBJECT (bus), "message::state-changed", + (GCallback) state_changed_cb, state); + gst_object_unref (bus); + + /* Make player start playing */ + gst_element_set_state (state->pipeline, GST_STATE_PLAYING); + + /* Start the mainloop */ + state->main_loop = g_main_loop_new (NULL, FALSE); + g_main_loop_run (state->main_loop); + +done: + /* Release pipeline */ + if (state->pipeline) { + gst_element_set_state (state->pipeline, GST_STATE_NULL); + gst_object_unref (state->pipeline); + } + + /* Release intercom */ + if (state->queue) { + g_async_queue_unref (state->queue); + } + + if (state->lock) { + g_mutex_free (state->lock); + } + + if (state->cond) { + g_cond_free (state->cond); + } + + /* Unref the mainloop */ + if (state->main_loop) { + g_main_loop_unref (state->main_loop); + } + + close_ogl (); + + TRACE_VC_MEMORY ("at exit"); + return 0; +} |