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