gmyth-stream/gmemcoder/src/gmencoder.c
author morphbr
Wed Jul 04 15:03:53 2007 +0100 (2007-07-04)
branchtrunk
changeset 778 d32867e8767b
parent 772 47de5c5976bf
child 780 7feaeeed26d5
permissions -rw-r--r--
[svn r784] - Bug fix on gmencoder (-d option)
renatofilho@588
     1
#ifdef HAVE_CONFIG_H
renatofilho@588
     2
#include "config.h"
renatofilho@588
     3
#endif
renatofilho@588
     4
renatofilho@600
     5
#include <sys/stat.h>
renatofilho@600
     6
#include <fcntl.h>
renatofilho@600
     7
#include <unistd.h>
renatofilho@588
     8
#include <glib.h>
renatofilho@588
     9
#include <gst/gst.h>
renatofilho@588
    10
#include <string.h>
renatofilho@762
    11
#include <sys/types.h>
renatofilho@762
    12
#include <sys/socket.h>
renatofilho@588
    13
renatofilho@588
    14
#include "gmencoder.h"
renatofilho@588
    15
renatofilho@588
    16
#define G_MENCODER_GET_PRIVATE(obj) \
renatofilho@588
    17
    (G_TYPE_INSTANCE_GET_PRIVATE ((obj), G_TYPE_MENCODER, GMencoderPrivate))
renatofilho@588
    18
renatofilho@754
    19
// #define SUPPORT_MULT_INPUT 0
renatofilho@761
    20
#define USE_MANUAL_SINK
renatofilho@762
    21
#define BUFFER_SIZE (1024 * 64)
renatofilho@588
    22
renatofilho@588
    23
typedef struct _GMencoderPrivate GMencoderPrivate;
renatofilho@600
    24
typedef struct _SetupInfo SetupInfo;
renatofilho@600
    25
renatofilho@754
    26
struct _SetupInfo {
renatofilho@754
    27
    gchar          *video_encode;
renatofilho@754
    28
    gchar          *mux_name;
renatofilho@754
    29
    gchar         **video_encode_prop;
renatofilho@754
    30
    gdouble         video_fps;
renatofilho@754
    31
    gdouble         video_rate;
renatofilho@754
    32
    guint           video_width;
renatofilho@754
    33
    guint           video_height;
renatofilho@754
    34
    gchar          *audio_encode;
renatofilho@754
    35
    gchar         **audio_encode_prop;
renatofilho@754
    36
    guint           audio_rate;
renatofilho@600
    37
};
renatofilho@600
    38
renatofilho@588
    39
renatofilho@754
    40
struct _GMencoderPrivate {
renatofilho@754
    41
    GstElement     *pipe;
renatofilho@754
    42
    GstElement     *abin;
renatofilho@754
    43
    GstElement     *vbin;
renatofilho@754
    44
    GstElement     *sink;
renatofilho@754
    45
    GstElement     *src;
renatofilho@754
    46
    gboolean        ready;
renatofilho@754
    47
    SetupInfo      *info;
renatofilho@754
    48
    GstClockTime    videot;
renatofilho@754
    49
    GstClockTime    audiot;
renatofilho@754
    50
    gint            fd;
renatofilho@754
    51
    gint            sources;
renatofilho@754
    52
    gint            tick_id;
renatofilho@754
    53
    gint64          duration;
renatofilho@768
    54
    gboolean        send_chunked;
renatofilho@762
    55
#ifdef USE_MANUAL_SINK
renatofilho@762
    56
    GByteArray	    *queue;
renatofilho@762
    57
#endif
renatofilho@588
    58
};
renatofilho@588
    59
renatofilho@754
    60
enum {
renatofilho@754
    61
    PAUSED,
renatofilho@754
    62
    PLAYING,
renatofilho@754
    63
    STOPED,
renatofilho@754
    64
    EOS,
renatofilho@754
    65
    ERROR,
renatofilho@754
    66
    LAST_SIGNAL
renatofilho@588
    67
};
renatofilho@588
    68
renatofilho@754
    69
static void     g_mencoder_class_init(GMencoderClass * klass);
renatofilho@754
    70
static void     g_mencoder_init(GMencoder * object);
renatofilho@754
    71
static void     g_mencoder_dispose(GObject * object);
renatofilho@754
    72
static void     g_mencoder_finalize(GObject * object);
renatofilho@752
    73
static GstElement *_create_audio_bin(const gchar * encode,
renatofilho@754
    74
                                     gchar ** encode_prop, gint rate);
renatofilho@752
    75
static GstElement *_create_video_bin(const gchar * encode,
renatofilho@754
    76
                                     gchar ** encode_prop,
renatofilho@754
    77
                                     gdouble fps,
renatofilho@777
    78
                                     gint rate, guint width, guint height,
renatofilho@777
    79
                                     gboolean use_deinterlace);
renatofilho@588
    80
renatofilho@754
    81
static          gboolean
renatofilho@752
    82
_pipeline_bus_cb(GstBus * bus, GstMessage * msg, gpointer user_data);
morphbr@748
    83
renatofilho@754
    84
static void     _decodebin_new_pad_cb(GstElement * object,
renatofilho@754
    85
                                      GstPad * pad,
renatofilho@754
    86
                                      gboolean flag, gpointer user_data);
morphbr@748
    87
renatofilho@754
    88
static void     _decodebin_unknown_type_cb(GstElement * object,
renatofilho@754
    89
                                           GstPad * pad,
renatofilho@754
    90
                                           GstCaps * caps,
renatofilho@754
    91
                                           gpointer user_data);
morphbr@748
    92
renatofilho@754
    93
static void     _close_output(GMencoder * self);
renatofilho@757
    94
static gboolean _open_output(GMencoder * self, const gchar * uri);
morphbr@748
    95
renatofilho@752
    96
static GstElement *_create_source(const gchar * uri);
renatofilho@752
    97
static GstElement *_create_pipeline(GMencoder * self,
renatofilho@754
    98
                                    const gchar * video_encode,
renatofilho@754
    99
                                    const gchar * mux_name,
renatofilho@754
   100
                                    gchar ** video_encode_prop,
renatofilho@754
   101
                                    gdouble video_fps,
renatofilho@754
   102
                                    gdouble video_rate,
renatofilho@754
   103
                                    guint video_width,
renatofilho@754
   104
                                    guint video_height,
renatofilho@754
   105
                                    const gchar * audio_encode,
renatofilho@754
   106
                                    gchar ** audio_encode_prop,
renatofilho@777
   107
                                    guint audio_rate,
renatofilho@777
   108
				    gboolean deinterlace);
renatofilho@761
   109
#ifdef USE_MANUAL_SINK
renatofilho@762
   110
static void _flush_queue 	    (GMencoder *self);
renatofilho@761
   111
static void _buffer_arrive_cb       (GstElement* object,
renatofilho@761
   112
                                     GstBuffer* buff,
renatofilho@761
   113
                                     GstPad* pad,
renatofilho@761
   114
                                     gpointer user_data);
renatofilho@761
   115
#endif
renatofilho@761
   116
renatofilho@654
   117
renatofilho@752
   118
static gboolean _tick_cb(gpointer data);
renatofilho@588
   119
renatofilho@754
   120
static guint    g_mencoder_signals[LAST_SIGNAL] = { 0 };
renatofilho@588
   121
renatofilho@588
   122
G_DEFINE_TYPE(GMencoder, g_mencoder, G_TYPE_OBJECT)
renatofilho@761
   123
renatofilho@761
   124
static void     g_mencoder_class_init(GMencoderClass * klass)
renatofilho@752
   125
{
renatofilho@754
   126
    GObjectClass   *object_class;
renatofilho@754
   127
    object_class = (GObjectClass *) klass;
renatofilho@754
   128
    g_type_class_add_private(klass, sizeof(GMencoderPrivate));
renatofilho@588
   129
renatofilho@754
   130
    object_class->dispose = g_mencoder_dispose;
renatofilho@754
   131
    object_class->finalize = g_mencoder_finalize;
renatofilho@588
   132
renatofilho@754
   133
    g_mencoder_signals[PAUSED] =
renatofilho@754
   134
        g_signal_new("paused",
renatofilho@754
   135
                     G_OBJECT_CLASS_TYPE(object_class),
renatofilho@754
   136
                     G_SIGNAL_RUN_FIRST,
renatofilho@754
   137
                     0, NULL, NULL,
renatofilho@754
   138
                     g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0);
renatofilho@588
   139
renatofilho@754
   140
    g_mencoder_signals[PLAYING] =
renatofilho@754
   141
        g_signal_new("playing",
renatofilho@754
   142
                     G_OBJECT_CLASS_TYPE(object_class),
renatofilho@754
   143
                     G_SIGNAL_RUN_FIRST,
renatofilho@754
   144
                     0, NULL, NULL,
renatofilho@754
   145
                     g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0);
morphbr@748
   146
renatofilho@754
   147
    g_mencoder_signals[STOPED] =
renatofilho@754
   148
        g_signal_new("stoped",
renatofilho@754
   149
                     G_OBJECT_CLASS_TYPE(object_class),
renatofilho@754
   150
                     G_SIGNAL_RUN_FIRST,
renatofilho@754
   151
                     0, NULL, NULL,
renatofilho@754
   152
                     g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0);
renatofilho@588
   153
renatofilho@754
   154
    g_mencoder_signals[EOS] =
renatofilho@754
   155
        g_signal_new("eos",
renatofilho@754
   156
                     G_OBJECT_CLASS_TYPE(object_class),
renatofilho@754
   157
                     G_SIGNAL_RUN_FIRST,
renatofilho@754
   158
                     0, NULL, NULL,
renatofilho@754
   159
                     g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0);
renatofilho@588
   160
renatofilho@754
   161
    g_mencoder_signals[ERROR] =
renatofilho@754
   162
        g_signal_new("error",
renatofilho@754
   163
                     G_OBJECT_CLASS_TYPE(object_class),
renatofilho@754
   164
                     G_SIGNAL_RUN_LAST,
renatofilho@754
   165
                     0, NULL, NULL,
renatofilho@754
   166
                     g_cclosure_marshal_VOID__STRING,
renatofilho@754
   167
                     G_TYPE_NONE, 1, G_TYPE_STRING);
renatofilho@588
   168
}
renatofilho@588
   169
renatofilho@588
   170
static void
renatofilho@752
   171
g_mencoder_init(GMencoder * self)
renatofilho@588
   172
{
renatofilho@754
   173
    GMencoderPrivate *priv = G_MENCODER_GET_PRIVATE(self);
renatofilho@754
   174
    priv->info = g_new0(SetupInfo, 1);
renatofilho@762
   175
#ifdef USE_MANUAL_SINK
renatofilho@762
   176
    priv->queue = g_byte_array_new ();
renatofilho@762
   177
#endif
renatofilho@588
   178
}
renatofilho@588
   179
morphbr@748
   180
static void
renatofilho@752
   181
g_mencoder_dispose(GObject * object)
renatofilho@588
   182
{
renatofilho@588
   183
}
renatofilho@588
   184
morphbr@748
   185
static void
renatofilho@752
   186
g_mencoder_finalize(GObject * object)
renatofilho@588
   187
{
renatofilho@762
   188
    GMencoderPrivate *priv = G_MENCODER_GET_PRIVATE(object);
renatofilho@762
   189
renatofilho@754
   190
    // TODO: clear vars
renatofilho@754
   191
    g_mencoder_close_stream(G_MENCODER(object));
renatofilho@762
   192
    g_free (priv->info);
renatofilho@762
   193
#ifdef USE_MANUAL_SINK
renatofilho@762
   194
    g_byte_array_free (priv->queue, TRUE);
renatofilho@762
   195
#endif
renatofilho@588
   196
}
renatofilho@588
   197
renatofilho@754
   198
GMencoder      *
renatofilho@752
   199
g_mencoder_new(void)
renatofilho@588
   200
{
renatofilho@754
   201
    return g_object_new(G_TYPE_MENCODER, NULL);
renatofilho@588
   202
}
renatofilho@588
   203
renatofilho@600
   204
renatofilho@588
   205
static void
renatofilho@754
   206
_obj_set_prop(GObject * obj, const gchar * prop_name,
renatofilho@754
   207
              const gchar * prop_val)
renatofilho@588
   208
{
renatofilho@754
   209
    GValue          p = { 0 };
renatofilho@754
   210
    GValue          v = { 0 };
renatofilho@754
   211
    GParamSpec     *s = NULL;
renatofilho@754
   212
    GObjectClass   *k = G_OBJECT_GET_CLASS(obj);
renatofilho@588
   213
renatofilho@588
   214
renatofilho@754
   215
    g_value_init(&v, G_TYPE_STRING);
renatofilho@754
   216
    g_value_set_string(&v, prop_val);
renatofilho@588
   217
renatofilho@754
   218
    s = g_object_class_find_property(k, prop_name);
renatofilho@754
   219
    if (s == NULL) {
renatofilho@754
   220
        g_print("Invalid property name: %s\n", prop_name);
renatofilho@754
   221
        return;
renatofilho@754
   222
    }
renatofilho@588
   223
renatofilho@754
   224
    g_value_init(&p, s->value_type);
renatofilho@754
   225
    switch (s->value_type) {
renatofilho@754
   226
    case G_TYPE_INT:
renatofilho@754
   227
        g_value_set_int(&p, atoi(prop_val));
renatofilho@754
   228
        break;
renatofilho@772
   229
    case G_TYPE_ULONG:
renatofilho@772
   230
        g_value_set_ulong (&p, atol(prop_val));
renatofilho@772
   231
        break;
renatofilho@754
   232
    case G_TYPE_STRING:
renatofilho@754
   233
        g_value_set_string(&p, prop_val);
renatofilho@754
   234
        break;
renatofilho@772
   235
    case G_TYPE_BOOLEAN:
renatofilho@772
   236
        g_value_set_boolean(&p, (gboolean) atoi (prop_val));
renatofilho@772
   237
        break;
renatofilho@772
   238
    case G_TYPE_DOUBLE:
renatofilho@772
   239
        g_value_set_double(&p, atof (prop_val));
renatofilho@772
   240
        break;
renatofilho@772
   241
    case G_TYPE_FLOAT:
renatofilho@772
   242
        g_value_set_float(&p, (float) atof (prop_val));
renatofilho@772
   243
        break;
renatofilho@754
   244
    default:
renatofilho@777
   245
        g_value_set_enum(&p, atoi(prop_val));
renatofilho@777
   246
        g_warning ("Property %s of type %s. Not supported using default enum", 
renatofilho@772
   247
                   prop_name, g_type_name (s->value_type));
renatofilho@754
   248
        return;
renatofilho@754
   249
    }
morphbr@748
   250
renatofilho@754
   251
    g_object_set_property(obj, prop_name, &p);
renatofilho@754
   252
    g_value_unset(&v);
renatofilho@754
   253
    g_value_unset(&p);
renatofilho@588
   254
}
renatofilho@588
   255
renatofilho@752
   256
static GstElement *
renatofilho@752
   257
_create_element_with_prop(const gchar * factory_name,
renatofilho@754
   258
                          const gchar * element_name, gchar ** prop)
renatofilho@588
   259
{
renatofilho@754
   260
    GstElement     *ret;
renatofilho@754
   261
    int             i;
renatofilho@588
   262
renatofilho@754
   263
    ret = gst_element_factory_make(factory_name, element_name);
renatofilho@754
   264
    if (ret == NULL)
renatofilho@754
   265
        return NULL;
renatofilho@588
   266
renatofilho@754
   267
    if (prop != NULL) {
renatofilho@754
   268
        for (i = 0; i < g_strv_length(prop); i++) {
renatofilho@754
   269
            if (prop[i] != NULL) {
renatofilho@754
   270
                char          **v = g_strsplit(prop[i], "=", 2);
renatofilho@754
   271
                if (g_strv_length(v) == 2) {
renatofilho@754
   272
                    _obj_set_prop(G_OBJECT(ret), v[0], v[1]);
renatofilho@754
   273
                }
renatofilho@754
   274
                g_strfreev(v);
renatofilho@754
   275
            }
renatofilho@754
   276
        }
renatofilho@754
   277
    }
renatofilho@588
   278
renatofilho@754
   279
    return ret;
renatofilho@588
   280
renatofilho@588
   281
}
renatofilho@588
   282
renatofilho@752
   283
static GstElement *
renatofilho@752
   284
_create_audio_bin(const gchar * encode, gchar ** encode_prop, gint rate)
renatofilho@588
   285
{
renatofilho@754
   286
    GstElement     *abin = NULL;
renatofilho@754
   287
    GstElement     *aqueue = NULL;
renatofilho@754
   288
    GstElement     *aconvert = NULL;
renatofilho@754
   289
    GstElement     *aencode = NULL;
renatofilho@754
   290
    GstElement     *aqueue_src = NULL;
renatofilho@754
   291
    GstPad         *apad = NULL;
renatofilho@588
   292
renatofilho@754
   293
    // audio/x-raw-int ! queue ! audioconvert ! faac ! rtpmp4gpay !
renatofilho@754
   294
    // udpsink name=upd_audio host=224.0.0.1 port=5002
renatofilho@754
   295
    abin = gst_bin_new("abin");
renatofilho@754
   296
    aqueue = gst_element_factory_make("queue", "aqueue");
renatofilho@754
   297
    aconvert = gst_element_factory_make("audioconvert", "aconvert");
renatofilho@754
   298
    aencode =
renatofilho@754
   299
        _create_element_with_prop((encode ? encode : "lame"), "aencode",
renatofilho@754
   300
                                  encode_prop);
renatofilho@754
   301
    aqueue_src = gst_element_factory_make("queue", "aqueue_src");
renatofilho@588
   302
renatofilho@754
   303
    if ((abin == NULL) || (aqueue == NULL) || (aconvert == NULL)
renatofilho@754
   304
        || (aencode == NULL) || (aqueue_src == NULL)) {
renatofilho@754
   305
        g_warning("Audio elements not found");
renatofilho@754
   306
        goto error;
renatofilho@754
   307
    }
renatofilho@752
   308
renatofilho@754
   309
    g_object_set(G_OBJECT(aencode), "bitrate", 32, NULL);
renatofilho@754
   310
    /*
renatofilho@754
   311
     * if (rate > 0) { g_object_set (G_OBJECT (aencode), "bitrate", 32,
renatofilho@754
   312
     * NULL); } 
renatofilho@754
   313
     */
renatofilho@754
   314
renatofilho@754
   315
    gst_bin_add_many(GST_BIN(abin), aqueue, aconvert, aencode, aqueue_src,
renatofilho@754
   316
                     NULL);
renatofilho@754
   317
    if (gst_element_link_many(aqueue, aconvert, aencode, aqueue_src, NULL)
renatofilho@754
   318
        == FALSE) {
renatofilho@754
   319
        g_warning("Not Link audio elements");
morphbr@748
   320
    }
renatofilho@754
   321
    // TODO: apply audio rate
renatofilho@588
   322
renatofilho@754
   323
    // ghost pad the audio bin
renatofilho@754
   324
    apad = gst_element_get_pad(aqueue, "sink");
renatofilho@754
   325
    gst_element_add_pad(abin, gst_ghost_pad_new("sink", apad));
renatofilho@754
   326
    gst_object_unref(apad);
renatofilho@600
   327
renatofilho@754
   328
    apad = gst_element_get_pad(aqueue_src, "src");
renatofilho@754
   329
    gst_element_add_pad(abin, gst_ghost_pad_new("src", apad));
renatofilho@754
   330
    gst_object_unref(apad);
renatofilho@588
   331
renatofilho@754
   332
    return abin;
renatofilho@754
   333
  error:
renatofilho@754
   334
    if (abin != NULL)
renatofilho@754
   335
        gst_object_unref(abin);
renatofilho@588
   336
renatofilho@754
   337
    if (aqueue != NULL)
renatofilho@754
   338
        gst_object_unref(aqueue);
morphbr@748
   339
renatofilho@754
   340
    if (aconvert != NULL)
renatofilho@754
   341
        gst_object_unref(aconvert);
renatofilho@588
   342
renatofilho@754
   343
    if (aencode != NULL)
renatofilho@754
   344
        gst_object_unref(aencode);
renatofilho@588
   345
renatofilho@754
   346
    if (aqueue_src != NULL)
renatofilho@754
   347
        gst_object_unref(aqueue_src);
renatofilho@588
   348
renatofilho@754
   349
    if (apad != NULL)
renatofilho@754
   350
        gst_object_unref(apad);
renatofilho@588
   351
renatofilho@754
   352
    return NULL;
renatofilho@588
   353
}
renatofilho@588
   354
renatofilho@588
   355
renatofilho@588
   356
renatofilho@588
   357
renatofilho@754
   358
// queue ! videoscale ! video/x-raw-yuv,width=240,height=144 ! colorspace
renatofilho@754
   359
// ! rate ! encode ! queue
renatofilho@752
   360
static GstElement *
renatofilho@752
   361
_create_video_bin(const gchar * encode,
renatofilho@754
   362
                  gchar ** encode_prop,
renatofilho@777
   363
                  gdouble fps, gint rate, guint width, guint height,
renatofilho@777
   364
                  gboolean use_deinterlace)
renatofilho@588
   365
{
renatofilho@754
   366
    GstElement     *vbin = NULL;
renatofilho@754
   367
    GstElement     *vqueue = NULL;
renatofilho@754
   368
    GstElement     *vqueue_src = NULL;
renatofilho@754
   369
    GstElement     *vcolorspace = NULL;
renatofilho@754
   370
    GstElement     *vencode = NULL;
renatofilho@754
   371
    GstElement     *vrate = NULL;
renatofilho@777
   372
    GstElement 	   *deinterlace = NULL;
renatofilho@777
   373
    GstElement     *walk = NULL;
renatofilho@754
   374
    GstPad         *vpad = NULL;
renatofilho@588
   375
renatofilho@754
   376
    vbin = gst_bin_new("vbin");
renatofilho@754
   377
    vqueue = gst_element_factory_make("queue", "vqueue");
renatofilho@754
   378
    vcolorspace =
renatofilho@754
   379
        gst_element_factory_make("ffmpegcolorspace", "colorspace");
renatofilho@616
   380
renatofilho@777
   381
    if (use_deinterlace) {
renatofilho@777
   382
	deinterlace = gst_element_factory_make ("ffdeinterlace", "deinterlace");
renatofilho@777
   383
	if (deinterlace == NULL) {
renatofilho@777
   384
	    g_warning ("Fail to create deinterlace element: Continue without deinterlace.");
renatofilho@777
   385
	} 
renatofilho@777
   386
    }
renatofilho@777
   387
renatofilho@777
   388
renatofilho@754
   389
    vencode = _create_element_with_prop((encode !=
renatofilho@754
   390
                                         NULL ? encode :
renatofilho@754
   391
                                         "ffenc_mpeg1video"), "vencode",
renatofilho@754
   392
                                        encode_prop);
renatofilho@754
   393
    vqueue_src = gst_element_factory_make("queue", "queue_src");
renatofilho@588
   394
renatofilho@754
   395
    if ((vbin == NULL) || (vqueue == NULL) || (vcolorspace == NULL)
renatofilho@754
   396
        || (vencode == NULL) || (vqueue_src == NULL)) {
renatofilho@754
   397
        g_warning("Video elements not found");
renatofilho@754
   398
        goto error;
renatofilho@754
   399
    }
renatofilho@588
   400
renatofilho@754
   401
    gst_bin_add_many(GST_BIN(vbin), vqueue, vcolorspace, vencode,
renatofilho@754
   402
                     vqueue_src, NULL);
morphbr@748
   403
renatofilho@777
   404
    if (deinterlace != NULL) {
renatofilho@777
   405
        gst_bin_add(GST_BIN(vbin), deinterlace);
renatofilho@777
   406
        gst_element_link (vqueue, deinterlace);
renatofilho@777
   407
        walk = deinterlace;
renatofilho@777
   408
    } else {
renatofilho@777
   409
        walk = vqueue;
renatofilho@777
   410
    }
renatofilho@777
   411
renatofilho@754
   412
    if ((width > 0) && (height > 0)) {
renatofilho@754
   413
        // Scalling video
renatofilho@754
   414
        GstCaps        *vcaps;
renatofilho@754
   415
        GstElement     *vscale =
renatofilho@754
   416
            gst_element_factory_make("videoscale", "vscale");
renatofilho@588
   417
renatofilho@777
   418
	g_object_set (G_OBJECT (vscale), "method", 1, NULL);
renatofilho@777
   419
renatofilho@754
   420
        gst_bin_add(GST_BIN(vbin), vscale);
renatofilho@588
   421
renatofilho@754
   422
        vcaps = gst_caps_new_simple("video/x-raw-yuv",
renatofilho@754
   423
                                    "width", G_TYPE_INT, width,
renatofilho@754
   424
                                    "height", G_TYPE_INT, height, NULL);
renatofilho@588
   425
renatofilho@777
   426
        gst_element_link(walk, vscale);
morphbr@748
   427
renatofilho@754
   428
        if (gst_element_link_filtered(vscale, vcolorspace, vcaps) == FALSE) {
renatofilho@754
   429
            g_warning("Fail to resize video");
renatofilho@754
   430
            gst_object_unref(vcaps);
renatofilho@754
   431
            gst_object_unref(vscale);
renatofilho@754
   432
            goto error;
renatofilho@754
   433
        }
renatofilho@754
   434
        gst_caps_unref(vcaps);
renatofilho@754
   435
    } else {
renatofilho@777
   436
        gst_element_link(walk, vcolorspace);
renatofilho@754
   437
    }
renatofilho@588
   438
renatofilho@754
   439
    if (fps > 0) {
renatofilho@754
   440
        // Changing the video fps
renatofilho@754
   441
        GstCaps        *vcaps;
renatofilho@754
   442
        vrate = gst_element_factory_make("videorate", "vrate");
renatofilho@588
   443
renatofilho@754
   444
        gst_bin_add(GST_BIN(vbin), vrate);
renatofilho@588
   445
renatofilho@754
   446
        if (gst_element_link(vcolorspace, vrate) == FALSE) {
renatofilho@754
   447
            g_warning("Fail to link video elements");
renatofilho@754
   448
            goto error;
renatofilho@754
   449
        }
renatofilho@588
   450
renatofilho@754
   451
        vcaps = gst_caps_new_simple("video/x-raw-yuv",
renatofilho@754
   452
                                    "framerate", GST_TYPE_FRACTION,
renatofilho@754
   453
                                    (int) (fps * 1000), 1000, NULL);
renatofilho@588
   454
renatofilho@754
   455
        if (gst_element_link_filtered(vrate, vencode, vcaps) == FALSE) {
renatofilho@754
   456
            g_warning("Fail to link vrate with vencode.");
renatofilho@754
   457
            goto error;
renatofilho@754
   458
        }
renatofilho@754
   459
        gst_caps_unref(vcaps);
renatofilho@754
   460
    } else {
renatofilho@754
   461
        if (gst_element_link(vcolorspace, vencode) == FALSE) {
renatofilho@754
   462
            g_warning("Fail to link colorspace and video encode element.");
renatofilho@754
   463
            goto error;
renatofilho@754
   464
        }
renatofilho@754
   465
    }
morphbr@748
   466
renatofilho@754
   467
    gst_element_link(vencode, vqueue_src);
renatofilho@588
   468
renatofilho@754
   469
    // ghost pad the video bin
renatofilho@754
   470
    vpad = gst_element_get_pad(vqueue, "sink");
renatofilho@754
   471
    gst_element_add_pad(vbin, gst_ghost_pad_new("sink", vpad));
renatofilho@754
   472
    gst_object_unref(vpad);
morphbr@748
   473
renatofilho@754
   474
    vpad = gst_element_get_pad(vqueue_src, "src");
renatofilho@754
   475
    gst_element_add_pad(vbin, gst_ghost_pad_new("src", vpad));
renatofilho@754
   476
    gst_object_unref(vpad);
renatofilho@588
   477
renatofilho@754
   478
    return vbin;
renatofilho@588
   479
renatofilho@754
   480
  error:
renatofilho@754
   481
    if (vpad != NULL)
renatofilho@754
   482
        gst_object_unref(vpad);
renatofilho@588
   483
renatofilho@754
   484
    if (vbin != NULL)
renatofilho@754
   485
        gst_object_unref(vbin);
renatofilho@588
   486
renatofilho@754
   487
    if (vqueue != NULL)
renatofilho@754
   488
        gst_object_unref(vqueue);
renatofilho@588
   489
renatofilho@754
   490
    if (vencode != NULL)
renatofilho@754
   491
        gst_object_unref(vencode);
renatofilho@588
   492
renatofilho@754
   493
    if (vqueue_src != NULL)
renatofilho@754
   494
        gst_object_unref(vqueue_src);
renatofilho@588
   495
renatofilho@754
   496
    if (vcolorspace != NULL)
renatofilho@754
   497
        gst_object_unref(vcolorspace);
renatofilho@588
   498
renatofilho@754
   499
    return NULL;
renatofilho@588
   500
}
renatofilho@588
   501
renatofilho@588
   502
renatofilho@600
   503
renatofilho@757
   504
gboolean 
renatofilho@752
   505
g_mencoder_setup_stream(GMencoder * self,
renatofilho@768
   506
			gboolean chunked,
renatofilho@777
   507
			gboolean deinterlace,
renatofilho@754
   508
                        const gchar * mux_name,
renatofilho@754
   509
                        const gchar * video_encode,
renatofilho@754
   510
                        gchar ** video_encode_prop,
renatofilho@754
   511
                        gdouble video_fps,
renatofilho@754
   512
                        gdouble video_rate,
renatofilho@754
   513
                        guint video_width,
renatofilho@754
   514
                        guint video_height,
renatofilho@754
   515
                        const gchar * audio_encode,
renatofilho@754
   516
                        gchar ** audio_encode_prop,
renatofilho@754
   517
                        guint audio_rate, const gchar * out_uri)
renatofilho@600
   518
{
renatofilho@754
   519
    GMencoderPrivate *priv = G_MENCODER_GET_PRIVATE(self);
renatofilho@754
   520
    if (priv->ready == TRUE) {
renatofilho@754
   521
        g_warning
renatofilho@754
   522
            ("Stream already configured. You need close stream first.");
renatofilho@757
   523
        return FALSE;
renatofilho@754
   524
    }
renatofilho@600
   525
renatofilho@754
   526
    _close_output(self);
renatofilho@757
   527
    if (_open_output(self, out_uri) == FALSE)
renatofilho@757
   528
        return FALSE;
renatofilho@634
   529
renatofilho@754
   530
    priv->sources = 0;
renatofilho@768
   531
    priv->send_chunked = chunked;
renatofilho@754
   532
    priv->pipe = _create_pipeline(self,
renatofilho@754
   533
                                  video_encode,
renatofilho@754
   534
                                  mux_name,
renatofilho@754
   535
                                  video_encode_prop,
renatofilho@754
   536
                                  video_fps,
renatofilho@754
   537
                                  video_rate,
renatofilho@754
   538
                                  video_width,
renatofilho@754
   539
                                  video_height,
renatofilho@754
   540
                                  audio_encode, audio_encode_prop,
renatofilho@777
   541
                                  audio_rate,
renatofilho@777
   542
                                  deinterlace);
renatofilho@600
   543
renatofilho@757
   544
    return (priv->pipe != NULL);
renatofilho@600
   545
}
renatofilho@600
   546
renatofilho@600
   547
renatofilho@600
   548
gboolean
renatofilho@752
   549
g_mencoder_append_uri(GMencoder * self, const gchar * uri)
renatofilho@600
   550
{
renatofilho@754
   551
    GstPad         *pad_src;
renatofilho@754
   552
    GstPad         *pad_sink;
renatofilho@754
   553
    GstElement     *src;
renatofilho@754
   554
    GMencoderPrivate *priv = G_MENCODER_GET_PRIVATE(self);
renatofilho@754
   555
    gboolean        ret = FALSE;
renatofilho@754
   556
    GstElement     *ap = NULL;
renatofilho@754
   557
    GstElement     *vp = NULL;
renatofilho@634
   558
renatofilho@600
   559
renatofilho@754
   560
    g_return_val_if_fail(priv->pipe != NULL, FALSE);
renatofilho@754
   561
    g_return_val_if_fail(priv->ready == FALSE, FALSE);
renatofilho@600
   562
renatofilho@634
   563
#ifndef SUPPORT_MULT_INPUT
renatofilho@754
   564
    g_return_val_if_fail(priv->sources < 1, FALSE);
renatofilho@634
   565
#endif
renatofilho@634
   566
renatofilho@754
   567
    src = _create_source(uri);
renatofilho@754
   568
    if (src == NULL)
renatofilho@754
   569
        return FALSE;
renatofilho@600
   570
renatofilho@754
   571
    priv->src = gst_bin_get_by_name(GST_BIN(src), "src");
renatofilho@671
   572
renatofilho@754
   573
    gst_bin_add(GST_BIN(priv->pipe), src);
renatofilho@600
   574
renatofilho@634
   575
#ifdef SUPPORT_MULT_INPUT
renatofilho@754
   576
    ap = gst_bin_get_by_name(GST_BIN(priv->pipe), "ap");
renatofilho@754
   577
    vp = gst_bin_get_by_name(GST_BIN(priv->pipe), "vp");
renatofilho@634
   578
#else
renatofilho@754
   579
    ap = gst_bin_get_by_name(GST_BIN(priv->pipe), "abin");
renatofilho@754
   580
    vp = gst_bin_get_by_name(GST_BIN(priv->pipe), "vbin");
renatofilho@634
   581
#endif
renatofilho@634
   582
renatofilho@754
   583
    if ((vp == NULL) || (ap == NULL)) {
renatofilho@754
   584
        g_warning("Fail to get output bin");
renatofilho@754
   585
        goto error;
renatofilho@754
   586
    }
renatofilho@600
   587
renatofilho@754
   588
    pad_src = gst_element_get_pad(src, "src_audio");
renatofilho@754
   589
    pad_sink = gst_element_get_compatible_pad(ap,
renatofilho@754
   590
                                              pad_src,
renatofilho@754
   591
                                              gst_pad_get_caps(pad_src));
renatofilho@600
   592
renatofilho@754
   593
    if ((pad_sink == NULL) || (pad_src == NULL))
renatofilho@754
   594
        goto error;
renatofilho@600
   595
renatofilho@754
   596
    GstPadLinkReturn lret = gst_pad_link(pad_src, pad_sink);
renatofilho@754
   597
    if (lret != GST_PAD_LINK_OK)
renatofilho@754
   598
        goto error;
renatofilho@600
   599
renatofilho@754
   600
    gst_object_unref(pad_src);
renatofilho@754
   601
    gst_object_unref(pad_sink);
renatofilho@600
   602
renatofilho@754
   603
    pad_src = gst_element_get_pad(src, "src_video");
renatofilho@754
   604
    pad_sink = gst_element_get_compatible_pad(vp,
renatofilho@754
   605
                                              pad_src,
renatofilho@754
   606
                                              gst_pad_get_caps(pad_src));
renatofilho@600
   607
renatofilho@754
   608
    if ((pad_src == NULL) || (pad_sink == NULL))
renatofilho@754
   609
        goto error;
renatofilho@600
   610
renatofilho@754
   611
    if (gst_pad_link(pad_src, pad_sink) != GST_PAD_LINK_OK) {
renatofilho@754
   612
        g_warning("invalid source. video");
renatofilho@754
   613
        goto error;
renatofilho@754
   614
    }
renatofilho@600
   615
renatofilho@754
   616
    priv->sources++;
renatofilho@754
   617
    ret = TRUE;
renatofilho@754
   618
  error:
renatofilho@600
   619
renatofilho@754
   620
    if ((src != NULL) && (ret == FALSE)) {
renatofilho@754
   621
        gst_bin_remove(GST_BIN(priv->pipe), src);
renatofilho@754
   622
        gst_object_unref(src);
renatofilho@754
   623
    }
renatofilho@600
   624
renatofilho@754
   625
    if (ap != NULL)
renatofilho@754
   626
        gst_object_unref(ap);
renatofilho@600
   627
renatofilho@754
   628
    if (vp != NULL)
renatofilho@754
   629
        gst_object_unref(vp);
renatofilho@600
   630
renatofilho@754
   631
    if (pad_src != NULL)
renatofilho@754
   632
        gst_object_unref(pad_src);
renatofilho@600
   633
renatofilho@754
   634
    if (pad_sink != NULL)
renatofilho@754
   635
        gst_object_unref(pad_sink);
renatofilho@600
   636
renatofilho@754
   637
    return ret;
renatofilho@600
   638
}
renatofilho@600
   639
renatofilho@600
   640
renatofilho@600
   641
morphbr@748
   642
void
renatofilho@752
   643
g_mencoder_remove_uri(GMencoder * self, const gchar * uri)
renatofilho@600
   644
{
renatofilho@754
   645
    // GMencoderPrivate *priv = G_MENCODER_GET_PRIVATE (self);
renatofilho@754
   646
    // TODO: remove src
renatofilho@600
   647
}
renatofilho@600
   648
renatofilho@600
   649
void
renatofilho@752
   650
g_mencoder_play_stream(GMencoder * self)
renatofilho@600
   651
{
renatofilho@754
   652
    GMencoderPrivate *priv = G_MENCODER_GET_PRIVATE(self);
renatofilho@754
   653
    g_return_if_fail(priv->ready == FALSE);
renatofilho@754
   654
    priv->ready = TRUE;
renatofilho@754
   655
    gst_element_set_state(priv->pipe, GST_STATE_PLAYING);
renatofilho@754
   656
    priv->tick_id = g_timeout_add(500, _tick_cb, self);
renatofilho@600
   657
}
renatofilho@600
   658
renatofilho@600
   659
void
renatofilho@752
   660
g_mencoder_pause_stream(GMencoder * self)
renatofilho@600
   661
{
renatofilho@754
   662
    GMencoderPrivate *priv = G_MENCODER_GET_PRIVATE(self);
renatofilho@754
   663
    g_return_if_fail(priv->ready == TRUE);
renatofilho@754
   664
    gst_element_set_state(priv->pipe, GST_STATE_PAUSED);
renatofilho@600
   665
}
renatofilho@600
   666
renatofilho@600
   667
void
renatofilho@752
   668
g_mencoder_close_stream(GMencoder * self)
renatofilho@600
   669
{
renatofilho@600
   670
renatofilho@754
   671
    GMencoderPrivate *priv = G_MENCODER_GET_PRIVATE(self);
renatofilho@754
   672
    if (priv->tick_id != 0) {
renatofilho@754
   673
        g_source_remove(priv->tick_id);
renatofilho@754
   674
        priv->tick_id = 0;
renatofilho@754
   675
    }
renatofilho@600
   676
renatofilho@754
   677
    if (priv->pipe != NULL) {
renatofilho@754
   678
        // TODO: fixe pipeline dispose
renatofilho@754
   679
        // gst_element_set_state (priv->pipe, GST_STATE_NULL);
renatofilho@754
   680
        // g_debug ("SETING STATE TO NULL: OK");
renatofilho@754
   681
        // gst_element_set_state (priv->pipe, GST_STATE_NULL);
renatofilho@754
   682
        // gst_object_unref (priv->pipe);
renatofilho@754
   683
        gst_object_unref(priv->src);
renatofilho@754
   684
        priv->src = NULL;
renatofilho@754
   685
        priv->pipe = NULL;
renatofilho@754
   686
        priv->abin = NULL;
renatofilho@754
   687
        priv->vbin = NULL;
renatofilho@754
   688
        priv->sink = NULL;
renatofilho@754
   689
    }
renatofilho@754
   690
    priv->ready = FALSE;
renatofilho@600
   691
}
renatofilho@600
   692
renatofilho@752
   693
static GstElement *
renatofilho@752
   694
_create_pipeline(GMencoder * self,
renatofilho@754
   695
                 const gchar * video_encode,
renatofilho@754
   696
                 const gchar * mux_name,
renatofilho@754
   697
                 gchar ** video_encode_prop,
renatofilho@754
   698
                 gdouble video_fps,
renatofilho@754
   699
                 gdouble video_rate,
renatofilho@754
   700
                 guint video_width,
renatofilho@754
   701
                 guint video_height,
renatofilho@754
   702
                 const gchar * audio_encode,
renatofilho@777
   703
                 gchar ** audio_encode_prop, guint audio_rate,
renatofilho@777
   704
	     	 gboolean deinterlace)
renatofilho@588
   705
{
renatofilho@754
   706
    GstBus         *bus = NULL;
renatofilho@754
   707
    GstElement     *pipe = NULL;
renatofilho@754
   708
    GstElement     *sink = NULL;
renatofilho@754
   709
    GstElement     *mux = NULL;
renatofilho@754
   710
    GstElement     *abin = NULL;
renatofilho@754
   711
    GstElement     *vbin = NULL;
renatofilho@754
   712
    GstElement     *queue = NULL;
renatofilho@754
   713
    GstPad         *aux_pad = NULL;
renatofilho@754
   714
    GstPad         *mux_pad = NULL;
renatofilho@634
   715
#ifdef SUPPORT_MULT_INPUT
renatofilho@754
   716
    GstElement     *ap = NULL;
renatofilho@754
   717
    GstElement     *vp = NULL;
renatofilho@634
   718
#endif
renatofilho@588
   719
renatofilho@754
   720
    pipe = gst_pipeline_new("pipe");
renatofilho@588
   721
morphbr@748
   722
#ifdef SUPPORT_MULT_INPUT
renatofilho@754
   723
    ap = gst_element_factory_make("concatmux", "ap");
renatofilho@754
   724
    vp = gst_element_factory_make("concatmux", "vp");
renatofilho@754
   725
    gst_bin_add_many(GST_BIN(pipe), ap, vp, NULL);
renatofilho@634
   726
#endif
renatofilho@762
   727
    GMencoderPrivate *priv = G_MENCODER_GET_PRIVATE(self);
renatofilho@588
   728
renatofilho@754
   729
    mux =
renatofilho@754
   730
        gst_element_factory_make((mux_name ? mux_name : "ffmux_mpeg"),
renatofilho@754
   731
                                 "mux");
renatofilho@754
   732
    if (mux == NULL)
renatofilho@754
   733
        goto error;
renatofilho@588
   734
renatofilho@754
   735
    queue = gst_element_factory_make("queue", "queueu_sink");
renatofilho@712
   736
renatofilho@634
   737
renatofilho@761
   738
#ifdef USE_MANUAL_SINK
renatofilho@761
   739
    sink = gst_element_factory_make("fakesink", "sink");
renatofilho@762
   740
    g_object_set (G_OBJECT (sink), "signal-handoffs", TRUE, NULL);
renatofilho@761
   741
    g_signal_connect (G_OBJECT (sink),
renatofilho@761
   742
                      "handoff",
renatofilho@761
   743
                      G_CALLBACK (_buffer_arrive_cb),
renatofilho@762
   744
                      self);
renatofilho@761
   745
#else    
renatofilho@754
   746
    sink = gst_element_factory_make("fdsink", "sink");
renatofilho@754
   747
    if (sink == NULL)
renatofilho@754
   748
        goto error;
renatofilho@588
   749
renatofilho@754
   750
    g_object_set(G_OBJECT(sink), "fd", priv->fd, "sync", FALSE, NULL);
renatofilho@761
   751
#endif
renatofilho@600
   752
renatofilho@754
   753
    abin = _create_audio_bin(audio_encode, audio_encode_prop, audio_rate);
renatofilho@754
   754
    if (abin == NULL)
renatofilho@754
   755
        goto error;
renatofilho@588
   756
renatofilho@754
   757
    vbin =
renatofilho@754
   758
        _create_video_bin(video_encode, video_encode_prop, video_fps,
renatofilho@777
   759
                          video_rate, video_width, video_height, deinterlace);
renatofilho@754
   760
    if (vbin == NULL)
renatofilho@754
   761
        goto error;
renatofilho@588
   762
renatofilho@754
   763
    // Finish Pipe
renatofilho@754
   764
    gst_bin_add_many(GST_BIN(pipe), abin, vbin, mux, queue, sink, NULL);
renatofilho@600
   765
renatofilho@600
   766
morphbr@748
   767
#ifdef SUPPORT_MULT_INPUT
renatofilho@754
   768
    if (gst_element_link(ap, abin) == FALSE) {
renatofilho@754
   769
        g_warning("Fail to link concat and abin");
renatofilho@754
   770
        goto error;
renatofilho@754
   771
    }
renatofilho@634
   772
renatofilho@754
   773
    if (gst_element_link(vp, vbin) == FALSE) {
renatofilho@754
   774
        g_warning("Fail to link concat and vbin");
renatofilho@754
   775
    }
morphbr@748
   776
#endif
renatofilho@588
   777
renatofilho@754
   778
    // Link bins with mux
renatofilho@754
   779
    aux_pad = gst_element_get_pad(abin, "src");
renatofilho@754
   780
    mux_pad =
renatofilho@754
   781
        gst_element_get_compatible_pad(mux, aux_pad,
renatofilho@754
   782
                                       GST_PAD_CAPS(aux_pad));
renatofilho@754
   783
    if (mux_pad == NULL) {
renatofilho@754
   784
        g_warning("Mux element no have audio PAD");
renatofilho@754
   785
        goto error;
renatofilho@754
   786
    }
renatofilho@754
   787
    GstPadLinkReturn ret = gst_pad_link(aux_pad, mux_pad);
renatofilho@754
   788
    if (ret != GST_PAD_LINK_OK) {
renatofilho@754
   789
        g_warning("Fail link audio and mux: %d", ret);
renatofilho@754
   790
        goto error;
morphbr@748
   791
renatofilho@754
   792
    }
renatofilho@754
   793
    gst_object_unref(aux_pad);
renatofilho@754
   794
    gst_object_unref(mux_pad);
renatofilho@588
   795
renatofilho@754
   796
    aux_pad = gst_element_get_pad(vbin, "src");
renatofilho@754
   797
    mux_pad =
renatofilho@754
   798
        gst_element_get_compatible_pad(mux, aux_pad,
renatofilho@754
   799
                                       GST_PAD_CAPS(aux_pad));
renatofilho@754
   800
    if (mux_pad == NULL) {
renatofilho@754
   801
        g_warning("Mux element no have video PAD");
renatofilho@754
   802
        goto error;
renatofilho@754
   803
    }
renatofilho@754
   804
    ret = gst_pad_link(aux_pad, mux_pad);
renatofilho@754
   805
    if (ret != GST_PAD_LINK_OK) {
renatofilho@754
   806
        g_warning("Fail link video and mux: %d", ret);
renatofilho@754
   807
        goto error;
renatofilho@754
   808
    }
renatofilho@754
   809
    gst_object_unref(aux_pad);
renatofilho@754
   810
    gst_object_unref(mux_pad);
renatofilho@754
   811
    aux_pad = NULL;
renatofilho@754
   812
    mux_pad = NULL;
renatofilho@588
   813
renatofilho@754
   814
    // Link mux with sink
renatofilho@754
   815
    gst_element_link_many(mux, queue, sink, NULL);
renatofilho@588
   816
renatofilho@754
   817
    bus = gst_pipeline_get_bus(GST_PIPELINE(pipe));
renatofilho@754
   818
    gst_bus_add_watch(bus, _pipeline_bus_cb, self);
renatofilho@754
   819
    gst_object_unref(bus);
renatofilho@754
   820
    return pipe;
renatofilho@588
   821
renatofilho@754
   822
  error:
renatofilho@754
   823
    g_warning("Invalid uri");
morphbr@748
   824
renatofilho@754
   825
    if (pipe != NULL) {
renatofilho@754
   826
        gst_object_unref(pipe);
renatofilho@754
   827
    }
renatofilho@588
   828
renatofilho@588
   829
renatofilho@754
   830
    if (mux != NULL) {
renatofilho@754
   831
        gst_object_unref(mux);
renatofilho@754
   832
    }
renatofilho@588
   833
renatofilho@754
   834
    if (mux_pad != NULL) {
renatofilho@754
   835
        gst_object_unref(mux_pad);
renatofilho@754
   836
    }
renatofilho@588
   837
renatofilho@754
   838
    if (aux_pad != NULL) {
renatofilho@754
   839
        gst_object_unref(mux_pad);
renatofilho@754
   840
    }
renatofilho@588
   841
renatofilho@754
   842
    if (sink != NULL) {
renatofilho@754
   843
        gst_object_unref(sink);
renatofilho@754
   844
    }
renatofilho@588
   845
renatofilho@754
   846
    if (abin != NULL) {
renatofilho@754
   847
        gst_object_unref(abin);
renatofilho@754
   848
    }
renatofilho@588
   849
renatofilho@754
   850
    if (vbin != NULL) {
renatofilho@754
   851
        gst_object_unref(vbin);
renatofilho@754
   852
    }
renatofilho@588
   853
renatofilho@754
   854
    return FALSE;
renatofilho@588
   855
}
renatofilho@588
   856
renatofilho@600
   857
renatofilho@600
   858
static void
renatofilho@752
   859
_close_output(GMencoder * self)
renatofilho@588
   860
{
renatofilho@600
   861
}
renatofilho@600
   862
renatofilho@752
   863
static GstElement *
renatofilho@752
   864
_create_source(const gchar * uri)
renatofilho@600
   865
{
renatofilho@600
   866
renatofilho@754
   867
    GstElement     *bsrc = NULL;
renatofilho@754
   868
    GstElement     *src = NULL;
renatofilho@754
   869
    GstElement     *queue = NULL;
renatofilho@754
   870
    GstElement     *aqueue = NULL;
renatofilho@754
   871
    GstElement     *vqueue = NULL;
renatofilho@754
   872
    GstElement     *decode = NULL;
renatofilho@754
   873
    GstPad         *src_pad = NULL;
renatofilho@600
   874
renatofilho@600
   875
renatofilho@754
   876
    bsrc = gst_bin_new(NULL);
renatofilho@600
   877
renatofilho@754
   878
    // src = gst_element_factory_make ("gnomevfssrc", "src");
renatofilho@754
   879
    // g_object_set (G_OBJECT (src), "location", uri, NULL);
renatofilho@754
   880
    src = gst_element_make_from_uri(GST_URI_SRC, uri, "src");
renatofilho@754
   881
    if (src == NULL)
renatofilho@754
   882
        goto error;
renatofilho@600
   883
renatofilho@754
   884
    decode = gst_element_factory_make("decodebin", "decode");
renatofilho@754
   885
    if (decode == NULL)
renatofilho@754
   886
        goto error;
renatofilho@600
   887
renatofilho@754
   888
    queue = gst_element_factory_make("queue", "queue_src");
renatofilho@754
   889
    aqueue = gst_element_factory_make("queue", "aqueue");
renatofilho@754
   890
    if (aqueue == NULL)
renatofilho@754
   891
        goto error;
renatofilho@600
   892
renatofilho@754
   893
    vqueue = gst_element_factory_make("queue", "vqueue");
renatofilho@754
   894
    if (vqueue == NULL)
renatofilho@754
   895
        goto error;
renatofilho@600
   896
renatofilho@754
   897
    gst_bin_add_many(GST_BIN(bsrc), src, queue, decode, aqueue, vqueue,
renatofilho@754
   898
                     NULL);
renatofilho@754
   899
    gst_element_link_many(src, queue, decode, NULL);
renatofilho@600
   900
renatofilho@754
   901
    g_signal_connect(G_OBJECT(decode),
renatofilho@754
   902
                     "new-decoded-pad",
renatofilho@754
   903
                     G_CALLBACK(_decodebin_new_pad_cb), bsrc);
renatofilho@600
   904
renatofilho@754
   905
    g_signal_connect(G_OBJECT(decode),
renatofilho@754
   906
                     "unknown-type",
renatofilho@754
   907
                     G_CALLBACK(_decodebin_unknown_type_cb), pipe);
renatofilho@600
   908
renatofilho@754
   909
    src_pad = gst_element_get_pad(aqueue, "src");
renatofilho@754
   910
    gst_element_add_pad(bsrc, gst_ghost_pad_new("src_audio", src_pad));
renatofilho@754
   911
    gst_object_unref(src_pad);
renatofilho@600
   912
renatofilho@754
   913
    src_pad = gst_element_get_pad(vqueue, "src");
renatofilho@754
   914
    gst_element_add_pad(bsrc, gst_ghost_pad_new("src_video", src_pad));
renatofilho@754
   915
    gst_object_unref(src_pad);
renatofilho@600
   916
renatofilho@754
   917
    return bsrc;
renatofilho@600
   918
renatofilho@754
   919
  error:
renatofilho@754
   920
    if (src != NULL) {
renatofilho@754
   921
        gst_object_unref(src);
renatofilho@754
   922
    }
renatofilho@600
   923
renatofilho@754
   924
    if (decode != NULL) {
renatofilho@754
   925
        gst_object_unref(decode);
renatofilho@754
   926
    }
renatofilho@600
   927
renatofilho@754
   928
    if (aqueue != NULL) {
renatofilho@754
   929
        gst_object_unref(aqueue);
renatofilho@754
   930
    }
renatofilho@600
   931
renatofilho@754
   932
    if (vqueue != NULL) {
renatofilho@754
   933
        gst_object_unref(vqueue);
renatofilho@754
   934
    }
renatofilho@600
   935
renatofilho@754
   936
    return NULL;
renatofilho@600
   937
}
renatofilho@600
   938
renatofilho@757
   939
static gboolean
renatofilho@752
   940
_open_output(GMencoder * self, const gchar * uri)
renatofilho@600
   941
{
renatofilho@757
   942
    gboolean ret = TRUE;
renatofilho@754
   943
    gchar         **i;
renatofilho@754
   944
    GMencoderPrivate *priv = G_MENCODER_GET_PRIVATE(self);
renatofilho@588
   945
renatofilho@754
   946
    i = g_strsplit(uri, "://", 0);
renatofilho@754
   947
    if (strcmp(i[0], "fd") == 0) {
renatofilho@754
   948
        priv->fd = atoi(i[1]);
renatofilho@768
   949
	if (priv->send_chunked)
renatofilho@768
   950
	    fcntl (priv->fd, F_SETFL, O_ASYNC);
renatofilho@754
   951
    } else if (strcmp(i[0], "file") == 0) {
renatofilho@756
   952
        if (g_file_test (i[1], G_FILE_TEST_EXISTS)) {
renatofilho@757
   953
            if (unlink (i[1]) != 0) {
renatofilho@757
   954
                g_warning ("Fail to write in : %s", uri);
renatofilho@757
   955
                ret = FALSE;
renatofilho@757
   956
                goto done;
renatofilho@757
   957
            }
renatofilho@756
   958
        }
renatofilho@754
   959
        priv->fd = open(i[1], O_WRONLY | O_CREAT | O_TRUNC,
renatofilho@754
   960
                        S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH);
renatofilho@757
   961
renatofilho@757
   962
        if (priv->fd == -1) {
renatofilho@757
   963
            g_warning ("Fail to open : %s", uri);
renatofilho@757
   964
            ret = FALSE;
renatofilho@757
   965
        }
renatofilho@754
   966
    } else {
renatofilho@754
   967
        g_warning("Output uri not supported");
renatofilho@757
   968
        ret = FALSE;
renatofilho@754
   969
    }
renatofilho@588
   970
renatofilho@757
   971
done:
renatofilho@754
   972
    g_strfreev(i);
renatofilho@757
   973
    return ret;
renatofilho@588
   974
}
renatofilho@588
   975
renatofilho@754
   976
static          gboolean
renatofilho@752
   977
_pipeline_bus_cb(GstBus * bus, GstMessage * msg, gpointer user_data)
renatofilho@588
   978
{
renatofilho@754
   979
    GMencoderPrivate *priv = G_MENCODER_GET_PRIVATE(user_data);
renatofilho@600
   980
renatofilho@754
   981
    switch (GST_MESSAGE_TYPE(msg)) {
renatofilho@588
   982
renatofilho@754
   983
    case GST_MESSAGE_STATE_CHANGED:
renatofilho@754
   984
        {
renatofilho@754
   985
            GstState        oldstate;
renatofilho@754
   986
            GstState        newstate;
renatofilho@754
   987
            GstState        pendingstate;
renatofilho@588
   988
morphbr@748
   989
renatofilho@754
   990
            gst_message_parse_state_changed(msg, &oldstate,
renatofilho@754
   991
                                            &newstate, &pendingstate);
renatofilho@588
   992
renatofilho@754
   993
            if (pendingstate != GST_STATE_VOID_PENDING)
renatofilho@754
   994
                break;
renatofilho@588
   995
renatofilho@754
   996
            if ((oldstate == GST_STATE_READY)
renatofilho@754
   997
                && (newstate == GST_STATE_PAUSED)) {
renatofilho@754
   998
                if (priv->ready)
renatofilho@754
   999
                    g_signal_emit(user_data, g_mencoder_signals[PAUSED],
renatofilho@754
  1000
                                  0);
renatofilho@754
  1001
            } else if ((oldstate == GST_STATE_PAUSED)
renatofilho@754
  1002
                       && (newstate == GST_STATE_PLAYING)) {
renatofilho@754
  1003
                g_signal_emit(user_data, g_mencoder_signals[PLAYING], 0);
renatofilho@754
  1004
            } else if ((oldstate == GST_STATE_READY) &&
renatofilho@754
  1005
                       (newstate == GST_STATE_NULL)) {
renatofilho@754
  1006
                g_signal_emit(user_data, g_mencoder_signals[STOPED], 0);
renatofilho@754
  1007
            }
renatofilho@754
  1008
            break;
renatofilho@754
  1009
        }
renatofilho@678
  1010
renatofilho@754
  1011
    case GST_MESSAGE_ERROR:
renatofilho@754
  1012
        {
renatofilho@754
  1013
            GError         *error;
renatofilho@754
  1014
            gchar          *debug;
renatofilho@754
  1015
            gchar          *err_str;
morphbr@748
  1016
renatofilho@754
  1017
            if (priv->tick_id != 0) {
renatofilho@754
  1018
                g_source_remove(priv->tick_id);
renatofilho@754
  1019
                priv->tick_id = 0;
renatofilho@754
  1020
            }
morphbr@748
  1021
renatofilho@754
  1022
            gst_message_parse_error(msg, &error, &debug);
renatofilho@754
  1023
            err_str = g_strdup_printf("Error [%d] %s (%s)", error->code,
renatofilho@754
  1024
                                      error->message, debug);
renatofilho@754
  1025
            priv->ready = FALSE;
renatofilho@754
  1026
            g_signal_emit(user_data, g_mencoder_signals[ERROR], 0,
renatofilho@754
  1027
                          err_str);
renatofilho@754
  1028
            g_free(err_str);
renatofilho@754
  1029
            g_clear_error(&error);
renatofilho@754
  1030
            g_free(debug);
renatofilho@754
  1031
            break;
renatofilho@754
  1032
        }
morphbr@748
  1033
renatofilho@754
  1034
    case GST_MESSAGE_EOS:
renatofilho@754
  1035
        priv->ready = FALSE;
renatofilho@762
  1036
#ifdef USE_MANUAL_SINK
renatofilho@762
  1037
	_flush_queue (G_MENCODER (user_data));
renatofilho@762
  1038
#endif
renatofilho@754
  1039
        g_signal_emit(user_data, g_mencoder_signals[EOS], 0);
renatofilho@754
  1040
        break;
morphbr@748
  1041
renatofilho@754
  1042
    case GST_MESSAGE_DURATION:
renatofilho@754
  1043
        {
renatofilho@754
  1044
            GstFormat       format;
renatofilho@754
  1045
            gint64          duration;
renatofilho@754
  1046
            gst_message_parse_duration(msg, &format, &duration);
renatofilho@754
  1047
            if (format == GST_FORMAT_BYTES)
renatofilho@754
  1048
                priv->duration = duration;
renatofilho@754
  1049
            break;
renatofilho@754
  1050
        }
renatofilho@754
  1051
    default:
renatofilho@754
  1052
        {
renatofilho@754
  1053
            break;
renatofilho@754
  1054
        }
renatofilho@754
  1055
    }
renatofilho@754
  1056
    return TRUE;
renatofilho@588
  1057
}
renatofilho@588
  1058
renatofilho@600
  1059
renatofilho@600
  1060
morphbr@748
  1061
static void
renatofilho@752
  1062
_decodebin_new_pad_cb(GstElement * object,
renatofilho@754
  1063
                      GstPad * pad, gboolean flag, gpointer user_data)
renatofilho@588
  1064
{
renatofilho@754
  1065
    GstCaps        *caps;
renatofilho@754
  1066
    gchar          *str_caps = NULL;
renatofilho@754
  1067
    GstElement     *sink_element;
renatofilho@754
  1068
    GstPad         *sink_pad;
renatofilho@588
  1069
renatofilho@754
  1070
    caps = gst_pad_get_caps(pad);
renatofilho@754
  1071
    str_caps = gst_caps_to_string(caps);
renatofilho@754
  1072
    if (strstr(str_caps, "audio") != NULL) {
renatofilho@754
  1073
        sink_element = gst_bin_get_by_name(GST_BIN(user_data), "aqueue");
renatofilho@754
  1074
    } else if (strstr(str_caps, "video") != NULL) {
renatofilho@754
  1075
        sink_element = gst_bin_get_by_name(GST_BIN(user_data), "vqueue");
renatofilho@754
  1076
    } else {
renatofilho@754
  1077
        g_warning("invalid caps %s", str_caps);
renatofilho@754
  1078
    }
renatofilho@588
  1079
renatofilho@754
  1080
    sink_pad = gst_element_get_pad(sink_element, "sink");
renatofilho@754
  1081
    gst_pad_link(pad, sink_pad);
renatofilho@600
  1082
renatofilho@754
  1083
    gst_object_unref(sink_element);
renatofilho@754
  1084
    gst_object_unref(sink_pad);
renatofilho@754
  1085
    g_free(str_caps);
renatofilho@754
  1086
    gst_caps_unref(caps);
renatofilho@588
  1087
}
renatofilho@588
  1088
morphbr@748
  1089
static void
renatofilho@752
  1090
_decodebin_unknown_type_cb(GstElement * object,
renatofilho@754
  1091
                           GstPad * pad, GstCaps * caps,
renatofilho@754
  1092
                           gpointer user_data)
renatofilho@588
  1093
{
renatofilho@754
  1094
    g_warning("Unknown Type");
renatofilho@754
  1095
    // priv->ready = FALSE;
renatofilho@588
  1096
}
renatofilho@654
  1097
renatofilho@754
  1098
static          gboolean
renatofilho@752
  1099
_tick_cb(gpointer user_data)
renatofilho@654
  1100
{
renatofilho@754
  1101
    GstFormat       format = GST_FORMAT_BYTES;
renatofilho@754
  1102
    gint64          cur = 0;
renatofilho@654
  1103
renatofilho@754
  1104
    GMencoderPrivate *priv = G_MENCODER_GET_PRIVATE(user_data);
renatofilho@654
  1105
renatofilho@754
  1106
    if (priv->duration == 0) {
renatofilho@754
  1107
        gint64          d = 0;
renatofilho@754
  1108
        if (gst_element_query_duration(priv->src, &format, &d))
renatofilho@754
  1109
            priv->duration = d;
renatofilho@754
  1110
    }
renatofilho@671
  1111
renatofilho@754
  1112
    if (priv->duration != 0) {
renatofilho@754
  1113
        gst_element_query_position(priv->src, &format, &cur);
renatofilho@754
  1114
        g_print("PROGRESS:%lli\n", (99 * cur) / priv->duration);
renatofilho@754
  1115
    }
renatofilho@654
  1116
renatofilho@754
  1117
    return TRUE;
renatofilho@654
  1118
}
renatofilho@761
  1119
renatofilho@761
  1120
renatofilho@761
  1121
#ifdef USE_MANUAL_SINK
renatofilho@762
  1122
static gboolean
renatofilho@762
  1123
_send_buffer (gint fd, gpointer buff, gint size)
renatofilho@762
  1124
{
renatofilho@762
  1125
    gboolean ret = TRUE;
renatofilho@762
  1126
    gchar *msg;
renatofilho@762
  1127
    GByteArray *b_send;
renatofilho@762
  1128
renatofilho@762
  1129
    b_send = g_byte_array_new ();
renatofilho@762
  1130
    msg = g_strdup_printf ("%x\r\n", size);
renatofilho@762
  1131
    b_send = g_byte_array_append (b_send, (const guint8*) msg, strlen (msg) * sizeof (gchar));
renatofilho@762
  1132
    g_free (msg);
renatofilho@762
  1133
renatofilho@762
  1134
    b_send = g_byte_array_append (b_send, buff, size);
renatofilho@762
  1135
renatofilho@762
  1136
    msg = g_strdup ("\r\n");
renatofilho@762
  1137
    b_send = g_byte_array_append (b_send, (const guint8*) msg, strlen (msg) * sizeof (gchar));
renatofilho@762
  1138
    g_free (msg);
renatofilho@762
  1139
renatofilho@762
  1140
    if (send (fd, b_send->data, b_send->len, MSG_MORE) <= 0) 
renatofilho@762
  1141
        ret = FALSE;
renatofilho@762
  1142
    g_byte_array_free (b_send, TRUE);
renatofilho@762
  1143
renatofilho@762
  1144
    return ret;
renatofilho@762
  1145
}
renatofilho@762
  1146
renatofilho@762
  1147
static void
renatofilho@762
  1148
_flush_queue (GMencoder *self)
renatofilho@762
  1149
{
renatofilho@762
  1150
    gchar *end_msg;
renatofilho@762
  1151
    GMencoderPrivate *priv = G_MENCODER_GET_PRIVATE(self);
renatofilho@762
  1152
renatofilho@762
  1153
    if (BUFFER_SIZE == 0)
renatofilho@762
  1154
        return;
renatofilho@762
  1155
renatofilho@762
  1156
    if (priv->queue->len > 0) {
renatofilho@762
  1157
        _send_buffer (priv->fd, priv->queue->data, priv->queue->len);
renatofilho@762
  1158
        priv->queue = g_byte_array_remove_range (priv->queue, 0, priv->queue->len);
renatofilho@762
  1159
    }
renatofilho@762
  1160
renatofilho@762
  1161
    end_msg = g_strdup ("0\r\n\r\n");
renatofilho@762
  1162
    write (priv->fd, (const guint8*) end_msg, strlen(end_msg) * sizeof(gchar));
renatofilho@762
  1163
    g_free (end_msg);
renatofilho@762
  1164
    return;
renatofilho@762
  1165
}
renatofilho@761
  1166
static void 
renatofilho@761
  1167
_buffer_arrive_cb (GstElement* object,
renatofilho@761
  1168
                   GstBuffer* buff,
renatofilho@761
  1169
                   GstPad* pad,
renatofilho@761
  1170
                   gpointer user_data)
renatofilho@761
  1171
{
renatofilho@762
  1172
    GMencoderPrivate *priv = G_MENCODER_GET_PRIVATE(user_data);
renatofilho@761
  1173
renatofilho@762
  1174
    if (BUFFER_SIZE == 0) {
renatofilho@762
  1175
	if (_send_buffer (priv->fd, GST_BUFFER_DATA (buff), GST_BUFFER_SIZE (buff)) == FALSE)
renatofilho@762
  1176
             goto error;
renatofilho@762
  1177
        return;
renatofilho@762
  1178
    }
renatofilho@761
  1179
renatofilho@762
  1180
    priv->queue = g_byte_array_append (priv->queue, GST_BUFFER_DATA (buff), GST_BUFFER_SIZE (buff));
renatofilho@762
  1181
    while (priv->queue->len >= BUFFER_SIZE) {
renatofilho@768
  1182
    	//g_usleep (0.2 * G_USEC_PER_SEC);
renatofilho@768
  1183
    	if (priv->send_chunked) {	
renatofilho@768
  1184
	    if (_send_buffer (priv->fd, priv->queue->data, BUFFER_SIZE) == FALSE)
renatofilho@768
  1185
                goto error;
renatofilho@768
  1186
	} else {
renatofilho@768
  1187
	    if (write (priv->fd, priv->queue->data, BUFFER_SIZE) == -1)
renatofilho@768
  1188
                goto error;
renatofilho@768
  1189
	}
renatofilho@762
  1190
        priv->queue = g_byte_array_remove_range (priv->queue, 0, BUFFER_SIZE);
renatofilho@762
  1191
    }
renatofilho@762
  1192
    return;
renatofilho@761
  1193
renatofilho@762
  1194
error:
renatofilho@762
  1195
    if (priv->tick_id != 0) {
renatofilho@762
  1196
        g_source_remove(priv->tick_id);
renatofilho@762
  1197
        priv->tick_id = 0;
renatofilho@761
  1198
    }
renatofilho@762
  1199
    priv->queue = g_byte_array_remove_range (priv->queue, 0, priv->queue->len);
renatofilho@762
  1200
    g_signal_emit(user_data, g_mencoder_signals[ERROR], 0, "Error on socket");
renatofilho@761
  1201
}
renatofilho@761
  1202
#endif 
renatofilho@761
  1203