Blame tests/examples/gtk/glliveshader.c

Packit 8ff292
/*
Packit 8ff292
 * GStreamer
Packit 8ff292
 * Copyright (C) 2015 Matthew Waters <matthew@centricular.com>
Packit 8ff292
 *
Packit 8ff292
 * This library is free software; you can redistribute it and/or
Packit 8ff292
 * modify it under the terms of the GNU Library General Public
Packit 8ff292
 * License as published by the Free Software Foundation; either
Packit 8ff292
 * version 2 of the License, or (at your option) any later version.
Packit 8ff292
 *
Packit 8ff292
 * This library is distributed in the hope that it will be useful,
Packit 8ff292
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 8ff292
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 8ff292
 * Library General Public License for more details.
Packit 8ff292
 *
Packit 8ff292
 * You should have received a copy of the GNU Library General Public
Packit 8ff292
 * License along with this library; if not, write to the
Packit 8ff292
 * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
Packit 8ff292
 * Boston, MA 02110-1301, USA.
Packit 8ff292
 */
Packit 8ff292
Packit 8ff292
#include <gst/gst.h>
Packit 8ff292
#include <gst/gl/gl.h>
Packit 8ff292
#include <gst/gl/gstglfuncs.h>
Packit 8ff292
#include <gtk/gtk.h>
Packit 8ff292
#if GST_GL_HAVE_WINDOW_X11
Packit 8ff292
#include <X11/Xlib.h>
Packit 8ff292
#endif
Packit 8ff292
Packit 8ff292
#ifndef GL_GEOMETRY_SHADER
Packit 8ff292
#define GL_GEOMETRY_SHADER 0x8DD9
Packit 8ff292
#endif
Packit 8ff292
Packit 8ff292
static GMainLoop *loop;
Packit 8ff292
Packit 8ff292
static const gchar *vert = "#version 330\n\
Packit 8ff292
in vec4 a_position;\n\
Packit 8ff292
in vec2 a_texcoord;\n\
Packit 8ff292
out vec2 v_texcoord;\n\
Packit 8ff292
uniform float time;\n\
Packit 8ff292
uniform float width;\n\
Packit 8ff292
uniform float height;\n\
Packit 8ff292
void main()\n\
Packit 8ff292
{\n\
Packit 8ff292
  gl_Position = a_position;\n\
Packit 8ff292
  v_texcoord = a_texcoord;\n\
Packit 8ff292
}\n";
Packit 8ff292
Packit 8ff292
static const gchar *geom = "#version 330\n\
Packit 8ff292
\n\
Packit 8ff292
layout(triangles) in;\n\
Packit 8ff292
layout(triangle_strip, max_vertices = 3) out;\n\
Packit 8ff292
in vec2 v_texcoord[];\n\
Packit 8ff292
out vec2 g_texcoord;\n\
Packit 8ff292
\n\
Packit 8ff292
void main() {\n\
Packit 8ff292
  for(int i = 0; i < 3; i++) {\n\
Packit 8ff292
    gl_Position = gl_in[i].gl_Position;\n\
Packit 8ff292
    g_texcoord = v_texcoord[i];\n\
Packit 8ff292
    EmitVertex();\n\
Packit 8ff292
  }\n\
Packit 8ff292
  EndPrimitive();\n\
Packit 8ff292
}\n";
Packit 8ff292
Packit 8ff292
static const gchar *frag = "#version 330\n\
Packit 8ff292
in vec2 g_texcoord;\n\
Packit 8ff292
uniform sampler2D tex;\n\
Packit 8ff292
uniform float time;\n\
Packit 8ff292
uniform float width;\n\
Packit 8ff292
uniform float height;\n\
Packit 8ff292
void main()\n\
Packit 8ff292
{\n\
Packit 8ff292
  gl_FragColor = texture2D(tex, g_texcoord);\n\
Packit 8ff292
}\n";
Packit 8ff292
Packit 8ff292
#define MAX_SHADER_STAGES 8
Packit 8ff292
struct shader_state;
Packit 8ff292
Packit 8ff292
struct text_view_state
Packit 8ff292
{
Packit 8ff292
  struct shader_state *state;
Packit 8ff292
Packit 8ff292
  GLenum type;
Packit 8ff292
  gchar *str;
Packit 8ff292
};
Packit 8ff292
Packit 8ff292
struct shader_state
Packit 8ff292
{
Packit 8ff292
  GstGLContext *context;
Packit 8ff292
  GstElement *shader;
Packit 8ff292
  gboolean shader_linked;
Packit 8ff292
  GtkWidget *label;
Packit 8ff292
  struct text_view_state text_states[MAX_SHADER_STAGES];
Packit 8ff292
  gint n_stages;
Packit 8ff292
};
Packit 8ff292
Packit 8ff292
static gboolean
Packit 8ff292
bus_call (GstBus * bus, GstMessage * msg, gpointer data)
Packit 8ff292
{
Packit 8ff292
  switch (GST_MESSAGE_TYPE (msg)) {
Packit 8ff292
    case GST_MESSAGE_EOS:
Packit 8ff292
      g_print ("End of stream\n");
Packit 8ff292
      g_main_loop_quit (loop);
Packit 8ff292
      break;
Packit 8ff292
    case GST_MESSAGE_ERROR:{
Packit 8ff292
      gchar *debug;
Packit 8ff292
      GError *error;
Packit 8ff292
Packit 8ff292
      gst_message_parse_error (msg, &error, &debug);
Packit 8ff292
      g_free (debug);
Packit 8ff292
Packit 8ff292
      g_printerr ("Error: %s\n", error->message);
Packit 8ff292
      g_error_free (error);
Packit 8ff292
Packit 8ff292
      g_main_loop_quit (loop);
Packit 8ff292
      break;
Packit 8ff292
    }
Packit 8ff292
    default:
Packit 8ff292
      break;
Packit 8ff292
  }
Packit 8ff292
Packit 8ff292
  return TRUE;
Packit 8ff292
}
Packit 8ff292
Packit 8ff292
static gchar *
Packit 8ff292
_find_source_for_shader_type (struct shader_state *state, GLenum type)
Packit 8ff292
{
Packit 8ff292
  int i = 0;
Packit 8ff292
Packit 8ff292
  for (i = 0; i < state->n_stages; i++) {
Packit 8ff292
    if (state->text_states[i].type == type)
Packit 8ff292
      return state->text_states[i].str;
Packit 8ff292
  }
Packit 8ff292
Packit 8ff292
  return NULL;
Packit 8ff292
}
Packit 8ff292
Packit 8ff292
static gboolean
Packit 8ff292
_add_stage_to_shader (GstGLShader * shader, struct shader_state *state,
Packit 8ff292
    GLenum type, const gchar * default_src)
Packit 8ff292
{
Packit 8ff292
  GError *error = NULL;
Packit 8ff292
  GstGLSLVersion version;
Packit 8ff292
  GstGLSLProfile profile;
Packit 8ff292
  GstGLSLStage *stage;
Packit 8ff292
  const gchar *src;
Packit 8ff292
Packit 8ff292
  src = _find_source_for_shader_type (state, type);
Packit 8ff292
  if (!src)
Packit 8ff292
    src = default_src;
Packit 8ff292
  if (!src)
Packit 8ff292
    /* FIXME: assume this stage is not needed */
Packit 8ff292
    return TRUE;
Packit 8ff292
Packit 8ff292
  if (!gst_glsl_string_get_version_profile (src, &version, &profile)) {
Packit 8ff292
    g_print ("Warning: failed to retreive GLSL version and profile for "
Packit 8ff292
        "shader type 0x%x\nsrc:\n%s\n", type, src);
Packit 8ff292
  }
Packit 8ff292
Packit 8ff292
  if (!(stage = gst_glsl_stage_new_with_string (shader->context, type,
Packit 8ff292
              version, profile, src))) {
Packit 8ff292
    g_print ("Error: Failed to create GLSL Stage from src:\n%s\n", src);
Packit 8ff292
    return FALSE;
Packit 8ff292
  }
Packit 8ff292
Packit 8ff292
  if (!gst_gl_shader_compile_attach_stage (shader, stage, &error)) {
Packit 8ff292
    /* ignore failed shader compilations */
Packit 8ff292
    g_print ("%s", error->message);
Packit 8ff292
    return FALSE;
Packit 8ff292
  }
Packit 8ff292
Packit 8ff292
  return TRUE;
Packit 8ff292
}
Packit 8ff292
Packit 8ff292
static GstGLShader *
Packit 8ff292
_new_shader (GstGLContext * context, struct shader_state *state)
Packit 8ff292
{
Packit 8ff292
  GstGLShader *shader = gst_gl_shader_new (context);
Packit 8ff292
  GError *error = NULL;
Packit 8ff292
Packit 8ff292
  if (!_add_stage_to_shader (shader, state, GL_VERTEX_SHADER, vert)) {
Packit 8ff292
    gst_object_unref (shader);
Packit 8ff292
    return NULL;
Packit 8ff292
  }
Packit 8ff292
  if (!_add_stage_to_shader (shader, state, GL_GEOMETRY_SHADER, geom)) {
Packit 8ff292
    gst_object_unref (shader);
Packit 8ff292
    return NULL;
Packit 8ff292
  }
Packit 8ff292
  if (!_add_stage_to_shader (shader, state, GL_FRAGMENT_SHADER, frag)) {
Packit 8ff292
    gst_object_unref (shader);
Packit 8ff292
    return NULL;
Packit 8ff292
  }
Packit 8ff292
Packit 8ff292
  if (!gst_gl_shader_link (shader, &error)) {
Packit 8ff292
    /* ignore failed shader compilations */
Packit 8ff292
    g_print ("%s", error->message);
Packit 8ff292
    gst_object_unref (shader);
Packit 8ff292
    return NULL;
Packit 8ff292
  }
Packit 8ff292
Packit 8ff292
  return shader;
Packit 8ff292
}
Packit 8ff292
Packit 8ff292
static gboolean
Packit 8ff292
_set_compilation_state (struct shader_state *state)
Packit 8ff292
{
Packit 8ff292
  gtk_label_set_text (GTK_LABEL (state->label),
Packit 8ff292
      state->shader_linked ? "Success" : "Failure");
Packit 8ff292
Packit 8ff292
  return G_SOURCE_REMOVE;
Packit 8ff292
}
Packit 8ff292
Packit 8ff292
static GstGLShader *
Packit 8ff292
_create_shader (GstElement * element, struct shader_state *state)
Packit 8ff292
{
Packit 8ff292
  GstGLContext *context;
Packit 8ff292
  GstGLShader *shader, *new_shader;
Packit 8ff292
Packit 8ff292
  g_object_get (G_OBJECT (element), "context", &context, "shader", &shader,
Packit 8ff292
      NULL);
Packit 8ff292
Packit 8ff292
  new_shader = _new_shader (context, state);
Packit 8ff292
  if (!shader && !new_shader)
Packit 8ff292
    g_warning ("Failed to create a shader!");
Packit 8ff292
  state->shader_linked = new_shader != NULL;
Packit 8ff292
Packit 8ff292
  if (shader)
Packit 8ff292
    gst_object_unref (shader);
Packit 8ff292
  gst_object_unref (context);
Packit 8ff292
Packit 8ff292
  g_main_context_invoke (NULL, (GSourceFunc) _set_compilation_state, state);
Packit 8ff292
Packit 8ff292
  return new_shader;
Packit 8ff292
}
Packit 8ff292
Packit 8ff292
static void
Packit 8ff292
_on_text_changed (GtkTextBuffer * text, struct text_view_state *state)
Packit 8ff292
{
Packit 8ff292
  GtkTextIter start, end;
Packit 8ff292
Packit 8ff292
  gtk_text_buffer_get_bounds (text, &start, &end;;
Packit 8ff292
  g_free (state->str);
Packit 8ff292
  state->str = gtk_text_buffer_get_text (text, &start, &end, FALSE);
Packit 8ff292
  g_object_set (state->state->shader, "update-shader", TRUE, NULL);
Packit 8ff292
}
Packit 8ff292
Packit 8ff292
static GtkWidget *
Packit 8ff292
_new_source_view (struct shader_state *state, GLenum type, const gchar * templ)
Packit 8ff292
{
Packit 8ff292
  static int i = 0;
Packit 8ff292
  GtkWidget *scroll, *text_view;
Packit 8ff292
  GtkTextBuffer *text;
Packit 8ff292
Packit 8ff292
  g_return_val_if_fail (i < MAX_SHADER_STAGES, NULL);
Packit 8ff292
Packit 8ff292
  state->text_states[i].state = state;
Packit 8ff292
  state->text_states[i].type = type;
Packit 8ff292
  state->text_states[i].str = g_strdup (templ);
Packit 8ff292
Packit 8ff292
  scroll = gtk_scrolled_window_new (NULL, NULL);
Packit 8ff292
  gtk_widget_set_size_request (scroll, 20, 20);
Packit 8ff292
  text_view = gtk_text_view_new ();
Packit 8ff292
  gtk_container_add (GTK_CONTAINER (scroll), text_view);
Packit 8ff292
  text = gtk_text_view_get_buffer (GTK_TEXT_VIEW (text_view));
Packit 8ff292
  if (state->text_states[i].str)
Packit 8ff292
    gtk_text_buffer_set_text (text, state->text_states[i].str, -1);
Packit 8ff292
  g_signal_connect (text, "changed", G_CALLBACK (_on_text_changed),
Packit 8ff292
      &state->text_states[i]);
Packit 8ff292
  state->n_stages++;
Packit 8ff292
  i++;
Packit 8ff292
Packit 8ff292
  return scroll;
Packit 8ff292
}
Packit 8ff292
Packit 8ff292
int
Packit 8ff292
main (int argc, char *argv[])
Packit 8ff292
{
Packit 8ff292
  GstElement *pipeline, *src, *upload, *shader, *sink;
Packit 8ff292
  GtkWidget *window, *paned, *video, *right_box, *book;
Packit 8ff292
  struct shader_state state = { 0, };
Packit 8ff292
  GstBus *bus;
Packit 8ff292
Packit 8ff292
#if GST_GL_HAVE_WINDOW_X11
Packit 8ff292
  XInitThreads ();
Packit 8ff292
#endif
Packit 8ff292
Packit 8ff292
  gst_init (&argc, &argv);
Packit 8ff292
  gtk_init (&argc, &argv);
Packit 8ff292
Packit 8ff292
  loop = g_main_loop_new (NULL, FALSE);
Packit 8ff292
Packit 8ff292
  pipeline = gst_pipeline_new (NULL);
Packit 8ff292
  src = gst_element_factory_make ("videotestsrc", NULL);
Packit 8ff292
  upload = gst_element_factory_make ("glupload", NULL);
Packit 8ff292
  shader = gst_element_factory_make ("glshader", NULL);
Packit 8ff292
  sink = gst_element_factory_make ("gtkglsink", NULL);
Packit 8ff292
  g_object_get (sink, "widget", &video, NULL);
Packit 8ff292
Packit 8ff292
  g_assert (src && shader && sink);
Packit 8ff292
  gst_bin_add_many (GST_BIN (pipeline), src, upload, shader, sink, NULL);
Packit 8ff292
  g_assert (gst_element_link_many (src, upload, shader, sink, NULL));
Packit 8ff292
Packit 8ff292
  bus = gst_pipeline_get_bus (GST_PIPELINE (pipeline));
Packit 8ff292
  gst_bus_add_watch (bus, bus_call, loop);
Packit 8ff292
  gst_object_unref (bus);
Packit 8ff292
Packit 8ff292
  state.shader = gst_object_ref (shader);
Packit 8ff292
  g_signal_connect (shader, "create-shader", G_CALLBACK (_create_shader),
Packit 8ff292
      &state);
Packit 8ff292
Packit 8ff292
  book = gtk_notebook_new ();
Packit 8ff292
  /* text view inside a scroll view */
Packit 8ff292
  gtk_notebook_append_page (GTK_NOTEBOOK (book), _new_source_view (&state,
Packit 8ff292
          GL_VERTEX_SHADER, vert), gtk_label_new ("Vertex"));
Packit 8ff292
  gtk_notebook_append_page (GTK_NOTEBOOK (book), _new_source_view (&state,
Packit 8ff292
          GL_GEOMETRY_SHADER, geom), gtk_label_new ("Geometry"));
Packit 8ff292
  gtk_notebook_append_page (GTK_NOTEBOOK (book), _new_source_view (&state,
Packit 8ff292
          GL_FRAGMENT_SHADER, frag), gtk_label_new ("Fragment"));
Packit 8ff292
  /* status label */
Packit 8ff292
  state.label = gtk_label_new ("Success");
Packit 8ff292
Packit 8ff292
  /* right side source code editor */
Packit 8ff292
  right_box = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
Packit 8ff292
  gtk_box_pack_start (GTK_BOX (right_box), book, TRUE, TRUE, 0);
Packit 8ff292
  gtk_box_pack_start (GTK_BOX (right_box), state.label, FALSE, TRUE, 0);
Packit 8ff292
Packit 8ff292
  paned = gtk_paned_new (GTK_ORIENTATION_HORIZONTAL);
Packit 8ff292
  gtk_paned_pack1 (GTK_PANED (paned), video, TRUE, FALSE);
Packit 8ff292
  gtk_widget_set_size_request (video, 20, 20);
Packit 8ff292
  gtk_paned_pack2 (GTK_PANED (paned), right_box, TRUE, FALSE);
Packit 8ff292
Packit 8ff292
  window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
Packit 8ff292
  gtk_window_set_default_size (GTK_WINDOW (window), 640, 480);
Packit 8ff292
  gtk_container_add (GTK_CONTAINER (window), paned);
Packit 8ff292
Packit 8ff292
  gtk_widget_show_all (window);
Packit 8ff292
Packit 8ff292
  gst_element_set_state (pipeline, GST_STATE_PLAYING);
Packit 8ff292
Packit 8ff292
  g_main_loop_run (loop);
Packit 8ff292
Packit 8ff292
  gst_element_set_state (pipeline, GST_STATE_NULL);
Packit 8ff292
Packit 8ff292
  /*shader strings leaked here */
Packit 8ff292
  /*g_free (state.str); */
Packit 8ff292
  gst_object_unref (state.shader);
Packit 8ff292
Packit 8ff292
  gst_object_unref (pipeline);
Packit 8ff292
Packit 8ff292
  return 0;
Packit 8ff292
}