Add new internal API to enable extra video convert element
[platform/core/api/mediacodec.git] / src / media_codec_port_gst.c
old mode 100755 (executable)
new mode 100644 (file)
index 5d09f4b..e2954ec
 #include <gst/gst.h>
 #include <gst/gstelement.h>
 #include <gst/app/gstappsrc.h>
+#include <gst/pbutils/pbutils.h>
+#include <gst/tag/tag.h>
 
 /*
  * Internal Implementation
  */
 static gpointer feed_task(gpointer data);
+static void _mc_gst_feed_task_new(mc_gst_core_t *core);
+static void _mc_gst_feed_task_free(mc_gst_core_t *core);
 static void __mc_gst_stop_feed(GstElement *pipeline, gpointer data);
 static void __mc_gst_start_feed(GstElement *pipeline, guint size, gpointer data);
 static media_packet_h _mc_get_input_buffer(mc_gst_core_t *core);
 
 static gboolean __mc_gst_init_gstreamer();
 static int _mc_output_media_packet_new(mc_gst_core_t *core, bool video, bool encoder, media_format_mimetype_e out_mime);
-static mc_ret_e _mc_gst_create_pipeline(mc_gst_core_t *core, gchar *factory_name);
+static mc_ret_e _mc_gst_create_pipeline(mc_gst_core_t *core, const gchar *factory_name);
 static mc_ret_e _mc_gst_destroy_pipeline(mc_gst_core_t *core);
 static void __mc_gst_buffer_add(GstElement *element, GstBuffer *buffer, GstPad *pad, gpointer data);
 static int __mc_output_buffer_finalize_cb(media_packet_h packet, int error_code, void *user_data);
@@ -56,6 +60,9 @@ static int __mc_set_caps_codecdata(mc_gst_core_t *core, GstMCBuffer *mcbuffer, g
 static void __mc_push_output_to_queue(mc_gst_core_t *core, GstMCBuffer *mcbuffer);
 static int __mc_gst_create_eos_packet(media_format_h fmt, GstMCBuffer *mcbuffer);
 static void _mc_gst_handle_input_buffer_used(mc_gst_core_t *core, media_packet_h packet);
+static gboolean _mc_gst_video_info_check_and_new(mc_gst_core_t *core, GstMCBuffer *mcbuffer);
+static void _mc_gst_video_info_free(mc_gst_core_t *core);
+static void _mc_gst_video_meta_add(mc_gst_core_t *core, GstBuffer *buffer);
 
 static gint __gst_handle_stream_error(mc_gst_core_t *core, GError *error, GstMessage *message);
 static gint __gst_transform_gsterror(mc_gst_core_t *core, GstMessage *message, GError *error);
@@ -81,80 +88,112 @@ static void _mc_send_eos_signal(mc_gst_core_t *core);
 static void _mc_wait_for_eos(mc_gst_core_t *core);
 static int _mc_get_mime(mc_gst_core_t *core);
 static int _mc_set_codec_data(mc_gst_core_t *core, GstMCBuffer *mcbuffer, bool codec_config);
-GstCaps *_mc_gst_vid_caps_new(mc_gst_core_t *core, mediacodec_codec_type_e codec_id, gint index);
-GstCaps *_mc_gst_aud_caps_new(mc_gst_core_t *core, mediacodec_codec_type_e codec_id, gint index);
+static gboolean _mc_gst_vid_caps_new(mc_gst_core_t *core, mediacodec_codec_type_e codec_id, gint index);
+static gboolean _mc_gst_aud_caps_new(mc_gst_core_t *core, mediacodec_codec_type_e codec_id, gint index);
 
+static int __mc_fill_input_buffer(mc_gst_core_t *core, media_packet_h packet, GstMCBuffer *mcbuffer);
+static int __mc_fill_output_buffer(mc_gst_core_t *core, void *data, int size, GstMCBuffer *mcbuffer);
+
+static int __mc_fill_input_buffer_with_packet(mc_gst_core_t *core, media_packet_h packet, GstMCBuffer *mcbuffer);
+static int __mc_fill_input_buffer_with_adec_packet(mc_gst_core_t *core, media_packet_h packet, GstMCBuffer *mcbuffer);
+static int __mc_fill_input_buffer_with_venc_packet(mc_gst_core_t *core, media_packet_h packet, GstMCBuffer *mcbuffer);
+
+static int __mc_fill_packet_with_output_buffer(mc_gst_core_t *core, void *data, int size, GstMCBuffer *mcbuffer);
+static int __mc_fill_aenc_packet_with_output_buffer(mc_gst_core_t *core, void *data, int size, GstMCBuffer *mcbuffer);
+static int __mc_fill_venc_packet_with_output_buffer(mc_gst_core_t *core, void *data, int size, GstMCBuffer *mcbuffer);
+static int __mc_fill_vdec_packet_with_output_buffer(mc_gst_core_t *core, void *data, int size, GstMCBuffer *mcbuffer);
+
+static mc_gst_core_t *mc_gst_core_new();
+static void mc_gst_core_free(mc_gst_core_t *core);
+
+static mc_gst_port_t *mc_gst_port_new(mc_gst_core_t *core);
+static void mc_gst_port_free(mc_gst_port_t *port);
 /* video vtable */
-int(*vdec_vtable[])() = {&__mc_fill_input_buffer_with_packet, &__mc_fill_packet_with_output_buffer};
-int(*venc_vtable[])() = {&__mc_fill_input_buffer_with_packet, &__mc_fill_packet_with_output_buffer};
+__attribute__((unused))
+static int(*vdec_vtable[])() = {&__mc_fill_input_buffer_with_packet, &__mc_fill_packet_with_output_buffer};
 
+static int(*venc_vtable[])() = {&__mc_fill_input_buffer_with_packet, &__mc_fill_packet_with_output_buffer};
 
-int(*vdec_h264_sw_vtable[])() = {&__mc_fill_input_buffer_with_packet,                  /* S/W H.264 Decoder Vtable */
+static int(*vdec_h264_sw_vtable[])() = {&__mc_fill_input_buffer_with_packet,                   /* S/W H.264 Decoder Vtable */
                                                                &__mc_fill_vdec_packet_with_output_buffer};
 
-int(*vdec_mpeg4_sw_vtable[])() = {&__mc_fill_input_buffer_with_packet,                 /* S/W MPEG4 Decoder Vtable */
+static int(*vdec_mpeg4_sw_vtable[])() = {&__mc_fill_input_buffer_with_packet,                  /* S/W MPEG4 Decoder Vtable */
                                                                &__mc_fill_vdec_packet_with_output_buffer};
 
-int(*vdec_h263_sw_vtable[])() = {&__mc_fill_input_buffer_with_packet,                  /* S/W MPEG4 Decoder Vtable */
+static int(*vdec_h263_sw_vtable[])() = {&__mc_fill_input_buffer_with_packet,                   /* S/W MPEG4 Decoder Vtable */
                                                                &__mc_fill_vdec_packet_with_output_buffer};
 
-int(*vdec_h264_hw_vtable[])() = {&__mc_fill_input_buffer_with_packet,                  /* H/W H.264 Decoder Vtable */
+static int(*vdec_h264_hw_vtable[])() = {&__mc_fill_input_buffer_with_packet,                   /* H/W H.264 Decoder Vtable */
                                                                &__mc_fill_vdec_packet_with_output_buffer};
 
-int(*vdec_mpeg4_hw_vtable[])() = {&__mc_fill_input_buffer_with_packet,                 /* H/W MPEG4 Decoder Vtable */
+static int(*vdec_mpeg4_hw_vtable[])() = {&__mc_fill_input_buffer_with_packet,                  /* H/W MPEG4 Decoder Vtable */
                                                                &__mc_fill_vdec_packet_with_output_buffer};
 
-int(*vdec_h263_hw_vtable[])() = {&__mc_fill_input_buffer_with_packet,                  /* H/W MPEG4 Decoder Vtable */
+static int(*vdec_h263_hw_vtable[])() = {&__mc_fill_input_buffer_with_packet,                   /* H/W MPEG4 Decoder Vtable */
                                                                &__mc_fill_vdec_packet_with_output_buffer};
 
-int(*venc_mpeg4_sw_vtable[])() = {&__mc_fill_input_buffer_with_venc_packet,            /* S/W MPEG4 Encoder Vtable */
+static int(*venc_mpeg4_sw_vtable[])() = {&__mc_fill_input_buffer_with_venc_packet,             /* S/W MPEG4 Encoder Vtable */
                                                                &__mc_fill_venc_packet_with_output_buffer};
 
-int(*venc_h263_sw_vtable[])() = {&__mc_fill_input_buffer_with_venc_packet,             /* S/W MPEG4 Encoder Vtable */
+static int(*venc_h263_sw_vtable[])() = {&__mc_fill_input_buffer_with_venc_packet,              /* S/W MPEG4 Encoder Vtable */
                                                                &__mc_fill_venc_packet_with_output_buffer};
 
-int(*venc_h264_hw_vtable[])() = {&__mc_fill_input_buffer_with_venc_packet,             /* H/W H.264 Encoder Vtable */
+static int(*venc_h264_hw_vtable[])() = {&__mc_fill_input_buffer_with_venc_packet,              /* H/W H.264 Encoder Vtable */
                                                                &__mc_fill_venc_packet_with_output_buffer};
 
-int(*venc_mpeg4_hw_vtable[])() = {&__mc_fill_input_buffer_with_venc_packet,            /* H/W MPEG4 Encoder Vtable */
+static int(*venc_mpeg4_hw_vtable[])() = {&__mc_fill_input_buffer_with_venc_packet,             /* H/W MPEG4 Encoder Vtable */
                                                                &__mc_fill_venc_packet_with_output_buffer};
 
-int(*venc_h263_hw_vtable[])() = {&__mc_fill_input_buffer_with_venc_packet,             /* H/W MPEG4 Encoder Vtable */
+static int(*venc_h263_hw_vtable[])() = {&__mc_fill_input_buffer_with_venc_packet,              /* H/W MPEG4 Encoder Vtable */
                                                                &__mc_fill_venc_packet_with_output_buffer};
 
-int(*aenc_vtable[])() = {&__mc_fill_input_buffer_with_packet, &__mc_fill_packet_with_output_buffer};
-int(*adec_vtable[])() = {&__mc_fill_input_buffer_with_packet, &__mc_fill_packet_with_output_buffer};
+__attribute__((unused))
+static int(*aenc_vtable[])() = {&__mc_fill_input_buffer_with_packet, &__mc_fill_packet_with_output_buffer};
+
+static int(*adec_vtable[])() = {&__mc_fill_input_buffer_with_packet, &__mc_fill_packet_with_output_buffer};
 
-int(*aenc_aac_vtable[])() = {&__mc_fill_input_buffer_with_packet,                    /* AAC LC Encoder vtable */
+static int(*aenc_aac_vtable[])() = {&__mc_fill_input_buffer_with_packet,                    /* AAC LC Encoder vtable */
                                                        &__mc_fill_aenc_packet_with_output_buffer};
 
-int(*adec_aac_vtable[])() = {&__mc_fill_input_buffer_with_adec_packet,               /* AAC LC Decoder Vtable */
+static int(*adec_aac_vtable[])() = {&__mc_fill_input_buffer_with_adec_packet,               /* AAC LC Decoder Vtable */
                                                        &__mc_fill_packet_with_output_buffer};
 
-int(*adec_aacv12_vtable[])() = {&__mc_fill_input_buffer_with_adec_packet,            /* AAC HE Decoder Vtable */
+static int(*adec_aacv12_vtable[])() = {&__mc_fill_input_buffer_with_adec_packet,            /* AAC HE Decoder Vtable */
                                                        &__mc_fill_packet_with_output_buffer};
 
-int(*adec_mp3_vtable[])() = {&__mc_fill_input_buffer_with_packet,                    /* MP3 Decoder Vtable */
+static int(*adec_mp3_vtable[])() = {&__mc_fill_input_buffer_with_packet,                    /* MP3 Decoder Vtable */
                                                        &__mc_fill_packet_with_output_buffer};
 
-int(*adec_amrnb_vtable[])() = {&__mc_fill_input_buffer_with_packet,                  /* AMR-NB Decoder Vtable */
+static int(*adec_amrnb_vtable[])() = {&__mc_fill_input_buffer_with_packet,                  /* AMR-NB Decoder Vtable */
                                                        &__mc_fill_packet_with_output_buffer};
 
-int(*adec_amrwb_vtable[])() = {&__mc_fill_input_buffer_with_packet,                  /* AMR-WB Decoder Vtable */
+static int(*adec_amrwb_vtable[])() = {&__mc_fill_input_buffer_with_packet,                  /* AMR-WB Decoder Vtable */
                                                        &__mc_fill_packet_with_output_buffer};
 
-int(*aenc_amrnb_vtable[])() = {&__mc_fill_input_buffer_with_packet,                  /* AMR-NB Encoder Vtable */
+static int(*aenc_amrnb_vtable[])() = {&__mc_fill_input_buffer_with_packet,                  /* AMR-NB Encoder Vtable */
                                                        &__mc_fill_packet_with_output_buffer};
 
-int(*adec_vorbis_vtable[])() = {&__mc_fill_input_buffer_with_packet,                 /* VORBIS Decoder Vtable */
+static int(*adec_vorbis_vtable[])() = {&__mc_fill_input_buffer_with_packet,                 /* VORBIS Decoder Vtable */
                                                        &__mc_fill_packet_with_output_buffer};
 
-int(*adec_flac_vtable[])() = {&__mc_fill_input_buffer_with_packet,                   /* FLAC Decoder Vtable */
+static int(*adec_flac_vtable[])() = {&__mc_fill_input_buffer_with_packet,                   /* FLAC Decoder Vtable */
                                                        &__mc_fill_packet_with_output_buffer};
 
-int(*adec_wma_vtable[])() = {&__mc_fill_input_buffer_with_packet,                    /* WMA Decoder Vtable */
+static int(*adec_wma_vtable[])() = {&__mc_fill_input_buffer_with_packet,                    /* WMA Decoder Vtable */
                                                        &__mc_fill_packet_with_output_buffer};
 
+static int(*aenc_opus_vtable[])() =  {&__mc_fill_input_buffer_with_packet,                  /* Opus Encoder Vtable */
+                                                       &__mc_fill_aenc_packet_with_output_buffer};
+
+#define MEDIACODEC_CROP_ARRAY_LENGTH 4
+#define MEDIACODEC_ELEMENT_SET_STATE(x_element, x_state)                                          \
+       do {                                                                                            \
+               LOGD("setting state [%s:%d] to [%s]\n", #x_state, x_state, GST_ELEMENT_NAME(x_element)); \
+               if (GST_STATE_CHANGE_FAILURE == gst_element_set_state(x_element, x_state)) {                    \
+                       LOGE("failed to set state %s to %s\n", #x_state, GST_ELEMENT_NAME(x_element));        \
+                       goto STATE_CHANGE_FAILED;                                                               \
+               }                                                                                           \
+       } while (0)
 
 /*
  * fill_inbuf virtual functions
@@ -164,12 +203,11 @@ int __mc_fill_input_buffer(mc_gst_core_t *core, media_packet_h packet, GstMCBuff
        return core->vtable[fill_inbuf](core, packet, mcbuffer);
 }
 
-int __mc_fill_inbuf_with_mm_video_buffer(mc_gst_core_t *core, media_packet_h packet, GstMCBuffer *mcbuffer)
+static int __mc_fill_input_buffer_with_packet(mc_gst_core_t *core, media_packet_h packet, GstMCBuffer *mcbuffer)
 {
        gint ret = MC_ERROR_NONE;
-       uint64_t buf_size = 0;
-       GstMemory *mem;
        void *buf_data = NULL;
+       uint64_t buf_size = 0;
 
        ret = media_packet_get_buffer_size(packet, &buf_size);
        if (ret != MEDIA_PACKET_ERROR_NONE) {
@@ -183,47 +221,18 @@ int __mc_fill_inbuf_with_mm_video_buffer(mc_gst_core_t *core, media_packet_h pac
                return MC_ERROR;
        }
 
-       mem = __mc_gst_make_tbm_buffer(core, mcbuffer);
-       if (mem == NULL) {
-               LOGW("Failed to make tizen memory");
-               return MC_ERROR;
-       }
-
-       gst_buffer_append_memory(mcbuffer->buffer, mem);
-       LOGD("tizen memory appended");
-
-       return ret;
-}
-
-int __mc_fill_input_buffer_with_packet(mc_gst_core_t *core, media_packet_h packet, GstMCBuffer *mcbuffer)
-{
-       gint ret = MC_ERROR_NONE;
-       void *buf_data = NULL;
-       uint64_t buf_size = 0;
-
-       ret = media_packet_get_buffer_size(packet, &buf_size);
-       if (ret != MEDIA_PACKET_ERROR_NONE) {
-               LOGW("buffer size get failed");
-               return MC_ERROR;
-       }
-
-       ret = media_packet_get_buffer_data_ptr(packet, &buf_data);
-       if (ret != MEDIA_PACKET_ERROR_NONE) {
-               LOGW("buffer data get failed");
+       if (buf_data == NULL)
                return MC_ERROR;
-       }
 
-       if (buf_data != NULL) {
-               gst_buffer_append_memory(mcbuffer->buffer,
-                               gst_memory_new_wrapped(GST_MEMORY_FLAG_READONLY, buf_data, buf_size, 0,
-                                       buf_size, mcbuffer, (GDestroyNotify)__mc_input_buffer_finalize_cb));
-               LOGD("packet data apended");
-       }
+       gst_buffer_append_memory(mcbuffer->buffer,
+               gst_memory_new_wrapped(GST_MEMORY_FLAG_READONLY, buf_data, buf_size, 0,
+                       buf_size, mcbuffer, (GDestroyNotify)__mc_input_buffer_finalize_cb));
+       LOGD("packet data apended");
 
        return ret;
 }
 
-int __mc_fill_input_buffer_with_adec_packet(mc_gst_core_t *core, media_packet_h packet, GstMCBuffer *mcbuffer)
+static int __mc_fill_input_buffer_with_adec_packet(mc_gst_core_t *core, media_packet_h packet, GstMCBuffer *mcbuffer)
 {
        gint ret = MC_ERROR_NONE;
        uint64_t buf_size = 0;
@@ -263,12 +272,15 @@ int __mc_fill_input_buffer_with_adec_packet(mc_gst_core_t *core, media_packet_h
                                                buf_size, mcbuffer, (GDestroyNotify)__mc_input_buffer_finalize_cb));
                        LOGD("packet data apended");
                }
+
+               if (codec_data)
+                       free(codec_data);
        }
 
        return ret;
 }
 
-int __mc_fill_input_buffer_with_venc_packet(mc_gst_core_t *core, media_packet_h packet, GstMCBuffer *mcbuffer)
+static int __mc_fill_input_buffer_with_venc_packet(mc_gst_core_t *core, media_packet_h packet, GstMCBuffer *mcbuffer)
 {
        gint ret = MC_ERROR_NONE;
        gint i;
@@ -281,6 +293,7 @@ int __mc_fill_input_buffer_with_venc_packet(mc_gst_core_t *core, media_packet_h
        gint height;
        uint32_t plane_num;
        guint8 *planes[2];
+       GstMemory *mem = NULL;
 
        mc_gst_port_def_t *port_def = &core->ports[in_port_index]->port_def;
 
@@ -352,18 +365,25 @@ int __mc_fill_input_buffer_with_venc_packet(mc_gst_core_t *core, media_packet_h
                                mcbuffer->buf_size += buf_size;
                        }
                }
-       } else {
-               GstMemory *mem = NULL;
 
+               mem = gst_memory_new_wrapped(GST_MEMORY_FLAG_READONLY,
+                               planes[0], mcbuffer->buf_size, 0,
+                               mcbuffer->buf_size, mcbuffer,
+                               (GDestroyNotify)__mc_input_buffer_finalize_cb);
+       } else {
                mem = __mc_gst_make_tbm_buffer(core, mcbuffer);
-               if (mem == NULL) {
-                       LOGW("Failed to make tizen memory");
-                       return MC_ERROR;
-               }
+               LOGD("tizen memory[%p]", mem);
+       }
 
-               gst_buffer_append_memory(mcbuffer->buffer, mem);
-               LOGD("tizen memory appended");
+       if (mem == NULL) {
+               LOGE("gstmemory failed[is_hw:%d]", core->is_hw);
+               return MC_ERROR;
        }
+
+       gst_buffer_append_memory(mcbuffer->buffer, mem);
+
+       _mc_gst_video_meta_add(core, mcbuffer->buffer);
+
        return ret;
 }
 
@@ -372,12 +392,12 @@ int __mc_fill_input_buffer_with_venc_packet(mc_gst_core_t *core, media_packet_h
  * fill_outbuf virtual functions
  */
 
-int __mc_fill_output_buffer(mc_gst_core_t *core, void *data, int size, GstMCBuffer *mcbuffer)
+static int __mc_fill_output_buffer(mc_gst_core_t *core, void *data, int size, GstMCBuffer *mcbuffer)
 {
        return core->vtable[fill_outbuf](core, data, size, mcbuffer);
 }
 
-int __mc_fill_vdec_packet_with_output_buffer(mc_gst_core_t *core, void *data, int size, GstMCBuffer *mcbuffer)
+static int __mc_fill_vdec_packet_with_output_buffer(mc_gst_core_t *core, void *data, int size, GstMCBuffer *mcbuffer)
 {
        int i;
        int stride_width;
@@ -449,7 +469,7 @@ int __mc_fill_vdec_packet_with_output_buffer(mc_gst_core_t *core, void *data, in
        return MC_ERROR_NONE;
 }
 
-int __mc_fill_packet_with_output_buffer(mc_gst_core_t *core, void *data, int size, GstMCBuffer *mcbuffer)
+static int __mc_fill_packet_with_output_buffer(mc_gst_core_t *core, void *data, int size, GstMCBuffer *mcbuffer)
 {
        gint ret = MC_ERROR_NONE;
        gchar *ext_mem = NULL;
@@ -481,7 +501,7 @@ int __mc_fill_packet_with_output_buffer(mc_gst_core_t *core, void *data, int siz
        return MC_ERROR_NONE;
 }
 
-int __mc_fill_venc_packet_with_output_buffer(mc_gst_core_t *core, void *data, int size, GstMCBuffer *mcbuffer)
+static int __mc_fill_venc_packet_with_output_buffer(mc_gst_core_t *core, void *data, int size, GstMCBuffer *mcbuffer)
 {
        gint ret = MC_ERROR_NONE;
        bool codec_config = FALSE;
@@ -542,7 +562,7 @@ int __mc_fill_venc_packet_with_output_buffer(mc_gst_core_t *core, void *data, in
        return ret;
 }
 
-int __mc_fill_aenc_packet_with_output_buffer(mc_gst_core_t *core, void *data, int size, GstMCBuffer *mcbuffer)
+static int __mc_fill_aenc_packet_with_output_buffer(mc_gst_core_t *core, void *data, int size, GstMCBuffer *mcbuffer)
 {
        int ret = MC_ERROR_NONE;
        gint mem_size = 0;
@@ -591,6 +611,9 @@ int __mc_fill_aenc_packet_with_output_buffer(mc_gst_core_t *core, void *data, in
 
                        s = gst_caps_get_structure(peercaps, 0);
                        codec_data = gst_structure_get_value(s, "codec_data");
+                       if (codec_data  == NULL && gst_structure_has_field(s, "streamheader"))
+                               codec_data = gst_structure_get_value(s, "streamheader");
+
                        core->codec_data = GST_BUFFER(g_value_dup_boxed(codec_data));
                        gst_caps_unref(peercaps);
                }
@@ -792,6 +815,31 @@ int __mc_set_caps_streamheader(mc_gst_core_t *core, GstMCBuffer *mcbuffer, guint
                __mc_gst_caps_set_buffer_array(core->caps, "streamheader", header1, header2, NULL);
                gst_buffer_unref(header1);
                gst_buffer_unref(header2);
+       } else if (core->codec_id == MEDIACODEC_OPUS) {
+               const GstTagList *tags;
+               GstTagList *empty_tags = NULL;
+               GstBuffer *header, *comments;
+               guint8 *decoding_channel_mapping = NULL;
+               mc_gst_port_def_t *port_def = &core->ports[in_port_index]->port_def;
+
+               header = gst_codec_utils_opus_create_header(port_def->info.audio.samplerate,
+                       port_def->info.audio.channel,
+                       0, /* channel mapping family */
+                       1, /* stream count */
+                       0, /* coupled count */
+                       decoding_channel_mapping,
+                       0, /* FIXME: look ahead configured value */
+                       0);
+               tags = gst_tag_setter_get_tag_list(GST_TAG_SETTER(core->codec));
+               if (!tags)
+                       tags = empty_tags = gst_tag_list_new_empty();
+               comments = gst_tag_list_to_vorbiscomment_buffer(tags, (const guint8 *) "OpusTags",
+                       8, "Encoded with GStreamer opusenc");
+               core->caps = gst_codec_utils_opus_create_caps_from_header(header, comments);
+               if (empty_tags)
+                       gst_tag_list_unref(empty_tags);
+               gst_buffer_unref(header);
+               gst_buffer_unref(comments);
        } else {
                LOGE("Not support case of Stream header Caps");
        }
@@ -818,10 +866,17 @@ int __mc_set_caps_codecdata(mc_gst_core_t *core, GstMCBuffer *mcbuffer, guint fi
 
        if (codec_data != NULL) {               /* get the codec data from media_packet_get_codec_data() */
                data = g_malloc0(codec_data_size);
+
                memcpy(data, codec_data, codec_data_size);
+
+               free(codec_data);
+               codec_data = NULL;
+
                codecdata_buffer = gst_buffer_new_wrapped(data, codec_data_size);
                gst_caps_set_simple(core->caps, "codec_data", GST_TYPE_BUFFER, codecdata_buffer, NULL);
+
                LOGD("set codec data : %" G_GSIZE_FORMAT "", gst_buffer_get_size(codecdata_buffer));
+
                gst_buffer_unref(codecdata_buffer);
        } else {                                        /* get the codec data from media_packet_get_buffer_data_ptr() */
                uint64_t buffer_size = 0;
@@ -882,7 +937,7 @@ int _mc_output_media_packet_new(mc_gst_core_t *core, bool video, bool encoder, m
 /*
  * mc_gst_core functions
  */
-mc_gst_core_t *mc_gst_core_new()
+static mc_gst_core_t *mc_gst_core_new()
 {
        mc_gst_core_t *core;
 
@@ -915,13 +970,11 @@ mc_gst_core_t *mc_gst_core_new()
        core->prepare_count = 0;
        core->etb_count = 0;
        core->codec_data = NULL;
-
-       g_atomic_int_set(&core->available_queue->running, 1);
-       core->available_queue->thread = g_thread_new("feed thread", &feed_task, core);
        core->allocator = gst_tizen_allocator_new();
 
        core->bufmgr = NULL;
        core->drm_fd = -1;
+
        LOGD("gst_core(%p) is created", core);
 
        MEDIACODEC_FLEAVE();
@@ -929,19 +982,10 @@ mc_gst_core_t *mc_gst_core_new()
        return core;
 }
 
-void mc_gst_core_free(mc_gst_core_t *core)
+static void mc_gst_core_free(mc_gst_core_t *core)
 {
        MEDIACODEC_FENTER();
 
-       mc_aqueue_t *async_queue;
-
-       async_queue = core->available_queue;
-
-       mc_async_queue_disable(async_queue->input);
-
-       g_atomic_int_set(&async_queue->running, 0);
-       g_thread_join(async_queue->thread);
-
        g_mutex_clear(&core->eos_mutex);
        g_mutex_clear(&core->prepare_lock);
        g_mutex_clear(&core->drain_lock);
@@ -949,8 +993,9 @@ void mc_gst_core_free(mc_gst_core_t *core)
        g_cond_clear(&core->buffer_cond);
        g_cond_clear(&core->out_buffer_cond);
 
-       mc_async_queue_free(async_queue->input);
-       g_free(async_queue);
+       mc_async_queue_free(core->available_queue->input);
+       g_free(core->available_queue);
+       core->available_queue = NULL;
 
        if (core->allocator) {
                gst_object_unref(core->allocator);
@@ -979,7 +1024,7 @@ void mc_gst_core_free(mc_gst_core_t *core)
 /*
  * mc_gst_port functions
  */
-mc_gst_port_t *mc_gst_port_new(mc_gst_core_t *core)
+static mc_gst_port_t *mc_gst_port_new(mc_gst_core_t *core)
 {
        MEDIACODEC_FENTER();
 
@@ -1000,7 +1045,7 @@ mc_gst_port_t *mc_gst_port_new(mc_gst_core_t *core)
        return port;
 }
 
-void mc_gst_port_free(mc_gst_port_t *port)
+static void mc_gst_port_free(mc_gst_port_t *port)
 {
        MEDIACODEC_FENTER();
 
@@ -1017,6 +1062,8 @@ void mc_gst_port_free(mc_gst_port_t *port)
 
 gboolean _mc_update_packet_info(mc_gst_core_t *core, media_format_h format)
 {
+       int ret = MEDIA_FORMAT_ERROR_NONE;
+       gint bitrate = 0;
        gboolean is_format_change = FALSE;
 
        mc_gst_port_def_t *input_port_def = &core->ports[in_port_index]->port_def;
@@ -1024,13 +1071,16 @@ gboolean _mc_update_packet_info(mc_gst_core_t *core, media_format_h format)
        if (core->video) {
                gint width = 0;
                gint height = 0;
-               gint bitrate = 0;
                gint framerate = 0;
                gchar *sformat = NULL;
                media_format_mimetype_e mimetype = 0;
 
-               media_format_get_video_info(format, &mimetype, &width, &height, &bitrate, NULL);
-               media_format_get_video_frame_rate(format, &framerate);
+               ret = media_format_get_video_info(format, &mimetype, &width, &height, &bitrate, NULL);
+               ret |= media_format_get_video_frame_rate(format, &framerate);
+               if (ret != MEDIA_FORMAT_ERROR_NONE) {
+                       LOGE("get video info/frame_rate failed[0x%x]", ret);
+                       return FALSE;
+               }
 
                is_format_change |= input_port_def->info.video.width != width;
                is_format_change |= input_port_def->info.video.height != height;
@@ -1044,13 +1094,16 @@ gboolean _mc_update_packet_info(mc_gst_core_t *core, media_format_h format)
                }
 
                if (core->encoder) {
+                       bitrate *= 1000;
                        sformat = __mc_get_gst_input_format(mimetype, core->is_hw);
-                       is_format_change |= ((input_port_def->info.video.bitrate != bitrate * 1000) && (bitrate != 0));
+                       is_format_change |= ((input_port_def->info.video.bitrate != bitrate) && (bitrate != 0));
                        is_format_change |= g_strcmp0(input_port_def->info.video.format, sformat);
                        if (is_format_change) {
-                               LOGD("Bitrate changed : %d -> %d", input_port_def->info.video.bitrate, bitrate);
+                               if (bitrate != 0) {
+                                       LOGD("Bitrate changed : %d -> %d", input_port_def->info.video.bitrate, bitrate);
+                                       input_port_def->info.video.bitrate = bitrate;
+                               }
                                LOGD("Format changed : %s -> %s", input_port_def->info.video.format, sformat);
-                               input_port_def->info.video.bitrate = bitrate * 1000;
                                input_port_def->info.video.format = g_strdup(sformat);
                        }
                }
@@ -1058,9 +1111,12 @@ gboolean _mc_update_packet_info(mc_gst_core_t *core, media_format_h format)
                gint channel;
                gint samplerate;
                gint bit;
-               gint bitrate;
 
-               media_format_get_audio_info(format, NULL, &channel, &samplerate, &bit, &bitrate);
+               ret = media_format_get_audio_info(format, NULL, &channel, &samplerate, &bit, &bitrate);
+               if (ret != MEDIA_FORMAT_ERROR_NONE) {
+                       LOGE("get audio info failed[0x%x]", ret);
+                       return FALSE;
+               }
 
                is_format_change |= input_port_def->info.audio.channel != channel;
                is_format_change |= input_port_def->info.audio.samplerate != samplerate;
@@ -1076,10 +1132,11 @@ gboolean _mc_update_packet_info(mc_gst_core_t *core, media_format_h format)
                }
 
                if (core->encoder) {
-                       is_format_change |= ((input_port_def->info.audio.bitrate != bitrate * 1000) && (bitrate != 0));
+                       bitrate *= 1000;
+                       is_format_change |= ((input_port_def->info.audio.bitrate != bitrate) && (bitrate != 0));
                        if (is_format_change) {
                                LOGD("Bitrate changed : %d -> %d", input_port_def->info.audio.bitrate, bitrate);
-                               input_port_def->info.audio.bitrate = bitrate * 1000;
+                               input_port_def->info.audio.bitrate = bitrate;
                        }
                }
        }
@@ -1087,6 +1144,94 @@ gboolean _mc_update_packet_info(mc_gst_core_t *core, media_format_h format)
        return is_format_change;
 }
 
+static gboolean _mc_gst_video_info_check_and_new(mc_gst_core_t *core, GstMCBuffer *mcbuffer)
+{
+       int i = 0;
+       gchar *caps_str = NULL;
+       tbm_surface_h surface = NULL;
+       tbm_surface_info_s sinfo = {0, };
+
+       if (!core || !mcbuffer) {
+               LOGE("NULL core[%p] or mcbuffer[%p]", core, mcbuffer);
+               return FALSE;
+       }
+
+       if (!core->video_info) {
+               caps_str = gst_caps_to_string(core->caps);
+
+               LOGI("caps[%s] to video info", caps_str);
+
+               g_free(caps_str);
+
+               core->video_info = gst_video_info_new_from_caps(core->caps);
+               if (!core->video_info) {
+                       LOGE("video info failed");
+                       return FALSE;
+               }
+
+               if (media_packet_get_tbm_surface(mcbuffer->packet, &surface) != MEDIA_PACKET_ERROR_NONE) {
+                       LOGE("get surface failed");
+                       return FALSE;
+               }
+
+               if (tbm_surface_get_info(surface, &sinfo) != TBM_SURFACE_ERROR_NONE) {
+                       LOGE("get surface info failed");
+                       return FALSE;
+               }
+
+               for (i = 0 ; i < sinfo.num_planes ; i++) {
+                       GST_VIDEO_INFO_PLANE_OFFSET(core->video_info, i) = sinfo.planes[i].offset;
+                       GST_VIDEO_INFO_PLANE_STRIDE(core->video_info, i) = sinfo.planes[i].stride;
+
+                       LOGI("[%d] offset[%zu], stride[%d]", i,
+                               GST_VIDEO_INFO_PLANE_OFFSET(core->video_info, i),
+                               GST_VIDEO_INFO_PLANE_STRIDE(core->video_info, i));
+               }
+
+               LOGI("new video info[%p]", core->video_info);
+       }
+
+       return TRUE;
+}
+
+static void _mc_gst_video_info_free(mc_gst_core_t *core)
+{
+       if (!core)
+               return;
+
+       if (core->video_info) {
+               LOGI("free video info[%p]", core->video_info);
+               gst_video_info_free(core->video_info);
+               core->video_info = NULL;
+       }
+}
+
+static void _mc_gst_video_meta_add(mc_gst_core_t *core, GstBuffer *buffer)
+{
+       GstVideoMeta *vmeta = NULL;
+
+       if (!core || !core->video_info || !buffer)
+               return;
+
+       vmeta = gst_buffer_add_video_meta_full(buffer, GST_VIDEO_FRAME_FLAG_NONE,
+               GST_VIDEO_INFO_FORMAT(core->video_info),
+               GST_VIDEO_INFO_WIDTH(core->video_info),
+               GST_VIDEO_INFO_HEIGHT(core->video_info),
+               GST_VIDEO_INFO_N_PLANES(core->video_info),
+               core->video_info->offset,
+               core->video_info->stride);
+
+       if (!vmeta) {
+               LOGE("No video meta");
+               return;
+       }
+
+       if (core->is_hw) {
+               vmeta->map = gst_tizen_video_meta_map;
+               vmeta->unmap = gst_tizen_video_meta_unmap;
+       }
+}
+
 static int _mc_set_codec_data(mc_gst_core_t *core, GstMCBuffer *mcbuffer, bool codec_config)
 {
        MEDIACODEC_FENTER();
@@ -1118,6 +1263,7 @@ static int _mc_set_codec_data(mc_gst_core_t *core, GstMCBuffer *mcbuffer, bool c
                break;
        case MEDIACODEC_VORBIS:
        case MEDIACODEC_FLAC:
+       case MEDIACODEC_OPUS:
                if (codec_config) {
                        ret = __mc_set_caps_streamheader(core, mcbuffer, VORBIS_CODECDATA_SIZE);
                        if (ret != MC_ERROR_NONE)
@@ -1164,8 +1310,13 @@ static gpointer feed_task(gpointer data)
 
                media_packet_get_format(input_buffer,  &format);
                if (format) {
-                       if (_mc_update_packet_info(core, format))
+                       if (_mc_update_packet_info(core, format)) {
                                is_format_change = TRUE;
+                               if (!core->mc_caps_new(core, core->codec_id, in_port_index)) {
+                                       LOGE("caps new failed");
+                                       goto ERROR;
+                               }
+                       }
                        media_format_unref(format);
                }
 
@@ -1219,11 +1370,19 @@ static gpointer feed_task(gpointer data)
                }
 
                if (is_format_change) {
-                       GstCaps *caps;
-                       caps = core->mc_caps_new(core, core->codec_id, in_port_index);
-                       g_object_set(core->appsrc, "caps", caps, NULL);
-                       gst_caps_unref(caps);
-                       LOGD("caps updated");
+                       g_object_set(core->appsrc, "caps", core->caps, NULL);
+
+                       if (core->encoder) {
+                               int bitrate = core->video ? \
+                                       core->ports[in_port_index]->port_def.info.video.bitrate : \
+                                       core->ports[in_port_index]->port_def.info.audio.bitrate;
+
+                               g_object_set(core->codec, core->codec_type->property->bitrate_name, bitrate, NULL);
+
+                               LOGI("bitrate[property:%s,v:%d] %d updated",
+                                       core->codec_type->property->bitrate_name, core->video, bitrate);
+                       }
+
                        is_format_change = FALSE;
                }
 
@@ -1263,6 +1422,28 @@ LEAVE:
        return NULL;
 }
 
+static void _mc_gst_feed_task_new(mc_gst_core_t *core)
+{
+       g_return_if_fail(core && core->available_queue);
+
+       g_atomic_int_set(&core->available_queue->running, 1);
+       core->available_queue->thread = g_thread_new("feed thread", &feed_task, core);
+
+       LOGI("new thread for feed[%p]", core->available_queue->thread);
+}
+
+static void _mc_gst_feed_task_free(mc_gst_core_t *core)
+{
+       g_return_if_fail(core && core->available_queue);
+
+       mc_async_queue_disable(core->available_queue->input);
+
+       g_atomic_int_set(&core->available_queue->running, 0);
+       g_thread_join(core->available_queue->thread);
+
+       LOGI("feed thread[%p] is released", core->available_queue->thread);
+}
+
 static void __mc_gst_stop_feed(GstElement *pipeline, gpointer data)
 {
        mc_gst_core_t *core = (mc_gst_core_t *)data;
@@ -1286,9 +1467,10 @@ static void __mc_gst_start_feed(GstElement *pipeline, guint size, gpointer data)
        }
 }
 
-GstCaps *_mc_gst_aud_caps_new(mc_gst_core_t *core, mediacodec_codec_type_e codec_id, gint index)
+static gboolean _mc_gst_aud_caps_new(mc_gst_core_t *core, mediacodec_codec_type_e codec_id, gint index)
 {
-       GstCaps *caps = NULL;
+       gchar *caps_string = NULL;
+       g_autoptr(GstCaps) caps = NULL;
        mc_gst_port_def_t *port_def = &core->ports[index]->port_def;
 
        caps = gst_caps_new_simple(core->mime,
@@ -1315,7 +1497,7 @@ GstCaps *_mc_gst_aud_caps_new(mc_gst_core_t *core, mediacodec_codec_type_e codec
        case MEDIACODEC_AAC_HE_PS:
                if (core->encoder) {
                        LOGD("[MC_NOT_SUPPORTED] he-aac-v12 encoder is not supported yet!!!");
-                       return NULL;
+                       return FALSE;
                } else {
                        gst_caps_set_simple(caps,
                                "mpegversion", G_TYPE_INT, 4,
@@ -1328,7 +1510,7 @@ GstCaps *_mc_gst_aud_caps_new(mc_gst_core_t *core, mediacodec_codec_type_e codec
        case MEDIACODEC_MP3:
                if (core->encoder) {
                        LOGD("[MC_NOT_SUPPORTED] mp3 encoder is not supported yet!!!");
-                       return NULL;
+                       return FALSE;
                } else {
                        gst_caps_set_simple(caps,
                                "framed", G_TYPE_BOOLEAN, TRUE,
@@ -1355,7 +1537,7 @@ GstCaps *_mc_gst_aud_caps_new(mc_gst_core_t *core, mediacodec_codec_type_e codec
        case MEDIACODEC_AMR_WB:
                if (core->encoder) {
                        LOGD("[MC_NOT_SUPPORTED] amr-wb encoder is not supported yet!!!");
-                       return NULL;
+                       return FALSE;
                } else {
                        gst_caps_set_simple(caps,
                                "rate", G_TYPE_INT, 16000,
@@ -1365,14 +1547,14 @@ GstCaps *_mc_gst_aud_caps_new(mc_gst_core_t *core, mediacodec_codec_type_e codec
        case MEDIACODEC_VORBIS:
                if (core->encoder) {
                        LOGD("[MC_NOT_SUPPORTED] vorbis encoder is not supported yet!!!");
-                       return NULL;
+                       return FALSE;
                } else {
                }
                break;
        case MEDIACODEC_FLAC:
                if (core->encoder) {
                        LOGD("[MC_NOT_SUPPORTED] flac encoder is not supported yet!!!");
-                       return NULL;
+                       return FALSE;
                } else {
                        gst_caps_set_simple(caps,
                                "framed", G_TYPE_BOOLEAN, TRUE,
@@ -1386,7 +1568,7 @@ GstCaps *_mc_gst_aud_caps_new(mc_gst_core_t *core, mediacodec_codec_type_e codec
        case MEDIACODEC_WMALSL:
                if (core->encoder) {
                        LOGD("[MC_NOT_SUPPORTED] wma encoder is not supported yet!!!");
-                       return NULL;
+                       return FALSE;
                } else {
                        /*
                         * Need to extract from Stream Type Specific ... or
@@ -1424,14 +1606,39 @@ GstCaps *_mc_gst_aud_caps_new(mc_gst_core_t *core, mediacodec_codec_type_e codec
                                NULL);
                }
                break;
+       case MEDIACODEC_OPUS:
+               if (core->encoder) {
+                       gst_caps_set_simple(caps,
+                               "format", G_TYPE_STRING, _mc_bit_to_string(port_def->info.audio.bit_depth),
+                               "layout", G_TYPE_STRING, "interleaved", NULL);
+               } else {
+                       LOGD("[MC_NOT_SUPPORTED] opus decoder is not supported yet!!!");
+                       return FALSE;
+               }
+               break;
        default:
                break;
        }
-       return caps;
+
+       caps_string = gst_caps_to_string(caps);
+       if (caps_string) {
+               LOGI("new caps[%p:%s]", caps, caps_string);
+               g_free(caps_string);
+       }
+
+       if (core->caps)
+               gst_caps_unref(core->caps);
+
+       _mc_gst_video_info_free(core);
+
+       core->caps = g_steal_pointer(&caps);
+
+       return TRUE;
 }
 
-GstCaps *_mc_gst_vid_caps_new(mc_gst_core_t *core, mediacodec_codec_type_e codec_id, gint index)
+static gboolean _mc_gst_vid_caps_new(mc_gst_core_t *core, mediacodec_codec_type_e codec_id, gint index)
 {
+       gchar *caps_string = NULL;
        GstCaps *caps = NULL;
        mc_gst_port_def_t *port_def = &core->ports[index]->port_def;
 
@@ -1477,7 +1684,7 @@ GstCaps *_mc_gst_vid_caps_new(mc_gst_core_t *core, mediacodec_codec_type_e codec
                        LOGE("format : %s", port_def->info.video.format);
                } else {
                        gst_caps_set_simple(caps,
-                               "parsed", G_TYPE_BOOLEAN, TRUE,                                         /* FIXME different from sw */
+                               "parsed", G_TYPE_BOOLEAN, TRUE,  /* FIXME different from sw */
                                "alignment", G_TYPE_STRING, "au",
                                "stream-format", G_TYPE_STRING, "byte-stream",
                                "framerate", GST_TYPE_FRACTION, 30, 1,
@@ -1488,7 +1695,19 @@ GstCaps *_mc_gst_vid_caps_new(mc_gst_core_t *core, mediacodec_codec_type_e codec
                break;
        }
 
-       return caps;
+       caps_string = gst_caps_to_string(caps);
+       if (caps_string) {
+               LOGI("new caps[%p:%s]", caps, caps_string);
+               g_free(caps_string);
+       }
+
+       if (core->caps)
+               gst_caps_unref(core->caps);
+       _mc_gst_video_info_free(core);
+
+       core->caps = caps;
+
+       return TRUE;
 }
 
 static int _mc_link_vtable(mc_gst_core_t *core, mediacodec_codec_type_e id, gboolean encoder, gboolean is_hw)
@@ -1499,11 +1718,10 @@ static int _mc_link_vtable(mc_gst_core_t *core, mediacodec_codec_type_e id, gboo
 
        switch (id) {
        case MEDIACODEC_AAC:
-               if (encoder) {
+               if (encoder)
                        core->vtable = aenc_aac_vtable;
-               } else {
+               else
                        core->vtable = adec_aac_vtable;
-               }
                break;
        case MEDIACODEC_AAC_HE:
        case MEDIACODEC_AAC_HE_PS:
@@ -1524,11 +1742,10 @@ static int _mc_link_vtable(mc_gst_core_t *core, mediacodec_codec_type_e id, gboo
                break;
        case MEDIACODEC_AMR_NB:
                LOGD("amrnb vtable");
-               if (encoder) {
+               if (encoder)
                        core->vtable = aenc_amrnb_vtable;
-               } else {
+               else
                        core->vtable = adec_amrnb_vtable;
-               }
                break;
        case MEDIACODEC_AMR_WB:
                LOGD("amrwb vtable - Only support decoder");
@@ -1541,12 +1758,10 @@ static int _mc_link_vtable(mc_gst_core_t *core, mediacodec_codec_type_e id, gboo
                break;
        case MEDIACODEC_VORBIS:
                LOGD("vorbis vtable");
-               if (encoder) {
-                       LOGD("[MC_NOT_SUPPORTED] vorbis encoder is not supported yet!!!");
+               if (encoder)
                        return MC_NOT_SUPPORTED;
-               } else {
+               else
                        core->vtable = adec_vorbis_vtable;
-               }
                break;
        case MEDIACODEC_FLAC:
                LOGD("flac vtable");
@@ -1557,6 +1772,13 @@ static int _mc_link_vtable(mc_gst_core_t *core, mediacodec_codec_type_e id, gboo
                        core->vtable = adec_flac_vtable;
                }
                break;
+       case MEDIACODEC_OPUS:
+               LOGD("opus vtable");
+               if (encoder)
+                       core->vtable = aenc_opus_vtable;
+               else
+                       core->vtable = adec_vtable;
+               break;
        case MEDIACODEC_WMAV1:
        case MEDIACODEC_WMAV2:
        case MEDIACODEC_WMAPRO:
@@ -1574,31 +1796,28 @@ static int _mc_link_vtable(mc_gst_core_t *core, mediacodec_codec_type_e id, gboo
                if (encoder) {
                        core->vtable = is_hw ? venc_h263_hw_vtable : venc_h263_sw_vtable;
                } else {
-                       if (is_hw) {
+                       if (is_hw)
                                core->vtable = vdec_h263_hw_vtable;
-                       } else {
+                       else
                                core->vtable = vdec_h263_sw_vtable;
-                       }
                }
                break;
        case MEDIACODEC_MPEG4:
                LOGD("mpeg4 vtable");
-               if (encoder) {
+               if (encoder)
                        core->vtable = is_hw ? venc_mpeg4_hw_vtable : venc_mpeg4_sw_vtable;
-               } else {
+               else
                        core->vtable = is_hw ? vdec_mpeg4_hw_vtable : vdec_mpeg4_sw_vtable;
-               }
                break;
        case MEDIACODEC_H264:
                LOGD("h264 vtable");
                if (encoder) {
                        core->vtable = is_hw ? venc_h264_hw_vtable : venc_vtable;
                } else {
-                       if (is_hw) {
+                       if (is_hw)
                                core->vtable = vdec_h264_hw_vtable;
-                       } else {
+                       else
                                core->vtable = vdec_h264_sw_vtable;
-                       }
                }
                break;
        default:
@@ -1623,24 +1842,22 @@ media_packet_h _mc_get_input_buffer(mc_gst_core_t *core)
 
 mc_ret_e mc_gst_prepare(mc_handle_t *mc_handle)
 {
-       MEDIACODEC_FENTER();
-
        int ret = MC_ERROR_NONE;
-       media_format_mimetype_e out_mime;
        int num_supported_codec = 0;
        int i = 0;
-       GstCaps *caps = NULL;
-
-       if (!mc_handle)
-               return MC_PARAM_ERROR;
-
-       mediacodec_codec_type_e id;
        bool video;
        bool encoder;
        bool hardware;
        gchar *factory_name = NULL;
+       media_format_mimetype_e out_mime;
+       mediacodec_codec_type_e id;
        mc_codec_map_t *codec_map;
 
+       MEDIACODEC_FENTER();
+
+       if (!mc_handle)
+               return MC_PARAM_ERROR;
+
        id = mc_handle->codec_id;
        video = mc_handle->is_video;
        encoder = mc_handle->is_encoder;
@@ -1671,7 +1888,9 @@ mc_ret_e mc_gst_prepare(mc_handle_t *mc_handle)
        memcpy(&new_core->ports[out_port_index]->port_def.info, &mc_handle->info, sizeof(mc_handle->info));
        new_core->ports[out_port_index]->port_def.coding_type = out_mime;
        new_core->codec_id = id;
+       new_core->codec_type = &codec_map[i].type;
        new_core->mc_caps_new = video ? &_mc_gst_vid_caps_new : &_mc_gst_aud_caps_new;
+       new_core->extra_converter = &mc_handle->extra_converter;
 
        new_core->bufmgr = tbm_bufmgr_init(new_core->drm_fd);
        if (new_core->bufmgr == NULL) {
@@ -1713,20 +1932,21 @@ mc_ret_e mc_gst_prepare(mc_handle_t *mc_handle)
        }
 
        /* set caps in capsfilter as default*/
-       caps = new_core->mc_caps_new(new_core, id, out_port_index);
+       if (!new_core->mc_caps_new(new_core, id, out_port_index)) {
+               LOGE("caps new failed");
+               return MC_INTERNAL_ERROR;
+       }
 
-       new_core->caps = caps;
        /* FIXME will parse input format from ini. format is needed when linking elements*/
        if (new_core->video && new_core->encoder) {
-               gchar *format = NULL;
-
-               format = new_core->is_hw ? "SN12" : "I420";
-               gst_caps_set_simple(caps,
-                       "format", G_TYPE_STRING, format,
+               gst_caps_set_simple(new_core->caps,
+                       "format", G_TYPE_STRING, (new_core->is_hw ? "SN12" : "I420"),
                        NULL);
        }
-       g_object_set(new_core->capsfilter, "caps", caps, NULL);
-       gst_caps_unref(caps);
+
+       g_object_set(new_core->capsfilter, "caps", new_core->caps, NULL);
+
+       _mc_gst_feed_task_new(new_core);
 
        mc_handle->core = new_core;
        LOGD("initialized... %d", ret);
@@ -1758,8 +1978,12 @@ mc_ret_e mc_gst_unprepare(mc_handle_t *mc_handle)
                if (core->eos)
                        _mc_send_eos_signal(core);
 
+               _mc_gst_feed_task_free(core);
+
                _mc_gst_set_flush_input(core);
 
+               _mc_gst_video_info_free(core);
+
                ret = _mc_gst_destroy_pipeline(core);
 
                /* unset callback */
@@ -1795,7 +2019,7 @@ mc_ret_e mc_gst_unprepare(mc_handle_t *mc_handle)
        return ret;
 }
 
-mc_ret_e mc_gst_process_input(mc_handle_t *mc_handle, media_packet_h inbuf, uint64_t timeOutUs)
+mc_ret_e mc_gst_process_input(mc_handle_t *mc_handle, media_packet_h inbuf, uint64_t timeout_us)
 {
        MEDIACODEC_FENTER();
 
@@ -1839,7 +2063,7 @@ mc_ret_e mc_gst_process_input(mc_handle_t *mc_handle, media_packet_h inbuf, uint
        return ret;
 }
 
-mc_ret_e mc_gst_get_output(mc_handle_t *mc_handle, media_packet_h *outbuf, uint64_t timeOutUs)
+mc_ret_e mc_gst_get_output(mc_handle_t *mc_handle, media_packet_h *outbuf, uint64_t timeout_us)
 {
        gint ret = MC_ERROR_NONE;
        gint64 end_time = -1;
@@ -1848,7 +2072,7 @@ mc_ret_e mc_gst_get_output(mc_handle_t *mc_handle, media_packet_h *outbuf, uint6
        media_packet_h out_packet = NULL;
 
        MEDIACODEC_FENTER();
-       end_time = g_get_monotonic_time() + timeOutUs;
+       end_time = g_get_monotonic_time() + timeout_us;
 
        if (!mc_handle)
                return MC_PARAM_ERROR;
@@ -1992,22 +2216,115 @@ ERROR:
        return FALSE;
 }
 
-mc_ret_e _mc_gst_create_pipeline(mc_gst_core_t *core, gchar *factory_name)
+static GstElement *__mc_gst_extra_converter_new(mc_extra_converter_t *extra_converter)
+{
+       GstElement *element = NULL;
+       gint array[MEDIACODEC_CROP_ARRAY_LENGTH] = {0, };
+
+       if (!extra_converter) {
+               LOGE("NULL extra converter");
+               return NULL;
+       }
+
+       LOGI("extra converter[%s]", extra_converter->name);
+
+       element = gst_element_factory_make(extra_converter->name, NULL);
+       if (!element) {
+               LOGE("create element[%s] failed", extra_converter->name);
+               return NULL;
+       }
+
+       array[0] = extra_converter->crop_x;
+       array[1] = extra_converter->crop_y;
+       array[2] = extra_converter->crop_w;
+       array[3] = extra_converter->crop_h;
+
+       if (mc_gst_set_property_array_int(element,
+               "crop", array, MEDIACODEC_CROP_ARRAY_LENGTH) != MC_ERROR_NONE) {
+               gst_object_unref(element);
+               return NULL;
+       }
+
+       LOGI("set crop to extra converter[%d,%d,%dx%d]",
+               array[0], array[1], array[2], array[3] );
+
+       return element;
+}
+
+
+#ifdef TIZEN_FEATURE_ENABLE_DUMP_SOURCE
+static GstPadProbeReturn __dump_probe(GstPad *pad, GstPadProbeInfo *info, gpointer u_data)
 {
+       static int count = 0;
+       int i = 0;
+       const gchar *dump_path = (const gchar *)u_data;
+       GstBuffer *buffer = GST_PAD_PROBE_INFO_BUFFER(info);
+       GstMemory *memory = NULL;
+       FILE *fp = NULL;
+       tbm_surface_h surface = NULL;
+       tbm_surface_info_s sinfo = {0, };
+
+       if (!dump_path) {
+               LOGW("NULL path");
+               return GST_PAD_PROBE_OK;
+       }
+
+       if (count++ > 10) {
+               LOGW("[%s] SKIP DUMP", dump_path);
+               return GST_PAD_PROBE_OK;
+       }
+
+       memory = gst_buffer_peek_memory(buffer, 0);
+
+       if (!gst_is_tizen_memory(memory)) {
+               LOGW("Not tizen memory");
+               return GST_PAD_PROBE_OK;
+       }
+
+       surface = gst_tizen_memory_get_surface(memory);
+       if (!surface) {
+               LOGW("No surface");
+               return GST_PAD_PROBE_OK;
+       }
+
+       fp = fopen(dump_path, "a");
+       if (!fp) {
+               LOGW("file open failed[%s]", dump_path);
+               return GST_PAD_PROBE_OK;
+       }
+
+       tbm_surface_get_info(surface, &sinfo);
+
+       for (i = 0 ; i < sinfo.num_planes ; i++) {
+               LOGI("[dump:%s] write plane[%d] size[%u]", dump_path, i, sinfo.planes[i].size);
+               fwrite(sinfo.planes[i].ptr, 1, sinfo.planes[i].size, fp);
+       }
+
+       fclose(fp);
+
+       return GST_PAD_PROBE_OK;
+}
+#endif /* TIZEN_FEATURE_ENABLE_DUMP_SOURCE */
+
+
+mc_ret_e _mc_gst_create_pipeline(mc_gst_core_t *core, const gchar *factory_name)
+{
+       int i = 0;
+       int bitrate = 0;
        GstBus *bus = NULL;
 
        MEDIACODEC_FENTER();
 
        g_mutex_lock(&core->prepare_lock);
-       if (core->prepare_count == 0) {
 
+       if (core->prepare_count == 0) {
                if (!__mc_gst_init_gstreamer()) {
                        LOGE("gstreamer initialize fail");
                        g_mutex_unlock(&core->prepare_lock);
                        return MC_NOT_INITIALIZED;
                }
-               core->codec = gst_element_factory_make(factory_name, NULL);
 
+               core->codec = gst_element_factory_make(factory_name, NULL);
                if (!core->codec) {
                        LOGE("codec element create fail");
                        goto ERROR;
@@ -2018,14 +2335,12 @@ mc_ret_e _mc_gst_create_pipeline(mc_gst_core_t *core, gchar *factory_name)
 
                /* create common elements */
                core->pipeline = gst_pipeline_new(NULL);
-
                if (!core->pipeline) {
                        LOGE("pipeline create fail");
                        goto ERROR;
                }
 
                core->appsrc = gst_element_factory_make("appsrc", NULL);
-
                if (!core->appsrc) {
                        LOGE("appsrc can't create");
                        goto ERROR;
@@ -2033,48 +2348,38 @@ mc_ret_e _mc_gst_create_pipeline(mc_gst_core_t *core, gchar *factory_name)
 
                if (!core->video && core->encoder) {
                        core->audioconvert = gst_element_factory_make("audioconvert", NULL);
-
                        if (!core->audioconvert) {
-                               LOGE("capsfilter can't create");
+                               LOGE("audioconvert can't create");
                                goto ERROR;
                        }
 
                        core->audioresample = gst_element_factory_make("audioresample", NULL);
-
                        if (!core->audioresample) {
-                               LOGE("capsfilter can't create");
+                               LOGE("audioresample can't create");
                                goto ERROR;
                        }
                }
 
-               if (core->video && core->encoder) {
-                       core->videoconvert = gst_element_factory_make("videoconvert", NULL);
-
+               if (core->video && core->encoder && core->extra_converter->enable) {
+                       core->videoconvert = __mc_gst_extra_converter_new(core->extra_converter);
                        if (!core->videoconvert) {
-                               LOGE("videoconvert can't create");
-                               goto ERROR;
-                       }
-
-                       core->videoscale = gst_element_factory_make("videoscale", NULL);
-                       if (!core->videoscale) {
-                               LOGE("videoscale can't create");
+                               LOGE("create[%s] failed", core->extra_converter->name);
                                goto ERROR;
                        }
                }
 
                core->capsfilter = gst_element_factory_make("capsfilter", NULL);
-
                if (!core->capsfilter) {
                        LOGE("capsfilter can't create");
                        goto ERROR;
                }
 
                core->fakesink = gst_element_factory_make("fakesink", NULL);
-
                if (!core->fakesink) {
                        LOGE("fakesink create fail");
                        goto ERROR;
                }
+
                g_object_set(core->fakesink, "enable-last-sample", FALSE, NULL);
 
                /*__mc_link_elements(core);*/
@@ -2087,12 +2392,22 @@ mc_ret_e _mc_gst_create_pipeline(mc_gst_core_t *core, gchar *factory_name)
                                goto ERROR;
                        }
                } else if (core->video && core->encoder) {
-                       gst_bin_add_many(GST_BIN(core->pipeline), core->appsrc, core->capsfilter, core->codec, core->fakesink, NULL);
+                       if (core->extra_converter->enable) {
+                               gst_bin_add_many(GST_BIN(core->pipeline), core->appsrc, core->capsfilter, core->videoconvert, core->codec, core->fakesink, NULL);
 
-                       /* link elements */
-                       if (!(gst_element_link_many(core->appsrc, core->codec, core->fakesink, NULL))) {
-                               LOGE("gst_element_link_many is failed");
-                               goto ERROR;
+                               /* link elements */
+                               if (!(gst_element_link_many(core->appsrc, core->videoconvert, core->codec, core->fakesink, NULL))) {
+                                       LOGE("gst_element_link_many is failed");
+                                       goto ERROR;
+                               }
+                       } else {
+                               gst_bin_add_many(GST_BIN(core->pipeline), core->appsrc, core->capsfilter, core->codec, core->fakesink, NULL);
+
+                               /* link elements */
+                               if (!(gst_element_link_many(core->appsrc, core->codec, core->fakesink, NULL))) {
+                                       LOGE("gst_element_link_many is failed");
+                                       goto ERROR;
+                               }
                        }
                } else {
                        gst_bin_add_many(GST_BIN(core->pipeline), core->appsrc, core->capsfilter, core->codec, core->fakesink, NULL);
@@ -2104,6 +2419,26 @@ mc_ret_e _mc_gst_create_pipeline(mc_gst_core_t *core, gchar *factory_name)
                        }
                }
 
+#ifdef TIZEN_FEATURE_ENABLE_DUMP_SOURCE
+               {
+                       GstPad *pad = NULL;
+
+                       pad = gst_element_get_static_pad(core->videoconvert, "sink");
+                       if (pad) {
+                               gst_pad_add_probe(pad, GST_PAD_PROBE_TYPE_BUFFER, __dump_probe, "/tmp/input.dump", NULL);
+                               gst_object_unref(pad);
+                               pad = NULL;
+                       }
+
+                       pad = gst_element_get_static_pad(core->videoconvert, "src");
+                       if (pad) {
+                               gst_pad_add_probe(pad, GST_PAD_PROBE_TYPE_BUFFER, __dump_probe, "/tmp/output.dump", NULL);
+                               gst_object_unref(pad);
+                               pad = NULL;
+                       }
+               }
+#endif /* TIZEN_FEATURE_ENABLE_DUMP_SOURCE */
+
                /* connect signals, bus watcher */
                bus = gst_pipeline_get_bus(GST_PIPELINE(core->pipeline));
                core->bus_watch_id = gst_bus_add_watch(bus, __mc_gst_bus_callback, core);
@@ -2121,11 +2456,35 @@ mc_ret_e _mc_gst_create_pipeline(mc_gst_core_t *core, gchar *factory_name)
                g_object_set(GST_OBJECT(core->fakesink), "signal-handoffs", TRUE, NULL);
                core->signal_handoff = g_signal_connect(core->fakesink, "handoff", G_CALLBACK(__mc_gst_buffer_add), core);
 
-               /* set state PLAYING */
-               MEDIACODEC_ELEMENT_SET_STATE(GST_ELEMENT_CAST(core->pipeline), GST_STATE_PLAYING);
+               /* set bitrate and properties */
+               if (core->encoder) {
+                       if (core->video)
+                               bitrate = core->ports[in_port_index]->port_def.info.video.bitrate;
+                       else
+                               bitrate = core->ports[in_port_index]->port_def.info.audio.bitrate;
 
+                       LOGI("set encoder bitrate[property:%s,v:%d] %d",
+                               core->codec_type->property->bitrate_name, core->video, bitrate);
 
+                       g_object_set(core->codec,
+                               core->codec_type->property->bitrate_name, bitrate,
+                               NULL);
+               }
+
+               for (i = 0 ; i < core->codec_type->property->ext_num ; i++) {
+                       LOGI("set property [%s] -> [%s]",
+                               core->codec_type->property->ext_name[i],
+                               core->codec_type->property->ext_value[i]);
+
+                       gst_util_set_object_arg((GObject *)core->codec,
+                               core->codec_type->property->ext_name[i],
+                               core->codec_type->property->ext_value[i]);
+               }
+
+               /* set state PLAYING */
+               MEDIACODEC_ELEMENT_SET_STATE(GST_ELEMENT_CAST(core->pipeline), GST_STATE_PLAYING);
        }
+
        core->prepare_count++;
        g_mutex_unlock(&core->prepare_lock);
 
@@ -2197,6 +2556,7 @@ mc_ret_e _mc_gst_destroy_pipeline(mc_gst_core_t *core)
                        MEDIACODEC_ELEMENT_SET_STATE(core->pipeline, GST_STATE_NULL);
 
                        gst_object_unref(GST_OBJECT(core->pipeline));
+
                        if (core->caps) {
                                gst_caps_unref(core->caps);
                                core->caps = NULL;
@@ -2326,7 +2686,9 @@ gchar *__mc_get_gst_input_format(media_format_mimetype_e mimetype, bool is_hw)
        default:
                break;
        }
-       LOGD("input packet format : %s", format);
+
+       LOGD("input packet format : %s", format ? format : "(null)" );
+
        return format;
 }
 
@@ -2368,7 +2730,7 @@ GstMCBuffer *_mc_gst_media_packet_to_gstbuffer(mc_gst_core_t *core, media_packet
                 * will not be invoked
                 */
                gst_buffer_unref(mcbuffer->buffer);
-               free(mcbuffer);
+               g_free(mcbuffer);
                return NULL;
        }
 
@@ -2401,33 +2763,35 @@ GstMCBuffer *__mc_gst_make_media_packet(mc_gst_core_t *core, GstBuffer *buffer)
        void *data = NULL;
        gint size = 0;
 
-
-       mcbuffer = (GstMCBuffer *)g_malloc0(sizeof(*mcbuffer));
-
-       if (mcbuffer == NULL) {
-               LOGE("malloc fail");
-               return NULL;
-       }
-
        mem = gst_buffer_peek_memory(buffer, 0);
 
        if (gst_is_tizen_memory(mem)) {
                data = (void *)gst_tizen_memory_get_surface(mem);
                size = tbm_surface_internal_get_size((tbm_surface_h)data);
        } else {
-               gst_memory_map(mem, &map, GST_MAP_READ);
+               if (!gst_memory_map(mem, &map, GST_MAP_READ)) {
+                       LOGE("gst_memory_map failed for buffer[%p]", buffer);
+                       return NULL;
+               }
+
                data = map.data;
                size = map.size;
-               gst_memory_unmap(mem, &map);
        }
 
+       mcbuffer = g_new0(GstMCBuffer, 1);
+
        mcbuffer->buffer = buffer;
        mcbuffer->core = core;
        mcbuffer->buf_size = size;
 
        ret = __mc_fill_output_buffer(core, data, mcbuffer->buf_size,  mcbuffer);
+
+       if (!gst_is_tizen_memory(mem))
+               gst_memory_unmap(mem, &map);
+
        if (ret != MC_ERROR_NONE) {
                LOGW("failed to fill outbuf: %s (ox%08x)", _mc_error_to_string(ret), ret);
+               g_free(mcbuffer);
                return NULL;
        }
 
@@ -2440,8 +2804,8 @@ int __mc_gst_create_eos_packet(media_format_h fmt, GstMCBuffer *mcbuffer)
        media_packet_h packet = NULL;
 
        MEDIACODEC_FENTER();
-       if (media_packet_create_alloc(fmt, __mc_output_buffer_finalize_cb, mcbuffer, &packet)) {
-               LOGE("media_packet_create_alloc failed\n");
+       if (media_packet_create(fmt, __mc_output_buffer_finalize_cb, mcbuffer, &packet)) {
+               LOGE("media_packet_create failed\n");
                ret = MC_ERROR;
                goto ERROR;
        }
@@ -2461,7 +2825,7 @@ int __mc_gst_create_eos_packet(media_format_h fmt, GstMCBuffer *mcbuffer)
        return ret;
 ERROR:
        if (packet)
-               media_packet_destroy(packet);
+               media_packet_unref(packet);
        MEDIACODEC_FLEAVE();
        return ret;
 }
@@ -2501,7 +2865,7 @@ gboolean __mc_gst_bus_callback(GstBus *bus, GstMessage *msg, gpointer data)
        case GST_MESSAGE_EOS:
                _mc_send_eos_signal(core);
 
-               mcbuffer = (GstMCBuffer *)g_malloc0(sizeof(GstMCBuffer));
+               mcbuffer = g_new0(GstMCBuffer, 1);
                if (__mc_gst_create_eos_packet(core->output_fmt, mcbuffer) == MC_ERROR_NONE) {
                        __mc_push_output_to_queue(core, mcbuffer);
                        LOGD("send eos packet.");
@@ -2631,7 +2995,6 @@ static GstMemory *__mc_gst_make_tbm_buffer(mc_gst_core_t *core, GstMCBuffer *mcb
 {
        int ret = MEDIA_PACKET_ERROR_NONE;
        tbm_surface_h surface = NULL;
-       GstVideoInfo vinfo;
        GstMemory *mem = NULL;
 
        if (!mcbuffer->packet) {
@@ -2639,19 +3002,19 @@ static GstMemory *__mc_gst_make_tbm_buffer(mc_gst_core_t *core, GstMCBuffer *mcb
                return NULL;
        }
 
-       ret = media_packet_get_tbm_surface(mcbuffer->packet, &surface);
-       if (ret != MEDIA_PACKET_ERROR_NONE) {
-               LOGE("Failed to get tbm surface");
+       if (!_mc_gst_video_info_check_and_new(core, mcbuffer)) {
+               LOGE("video info failed");
                return NULL;
        }
 
-       if (!gst_video_info_from_caps(&vinfo, core->caps)) {
-               LOGE("Failed to get video info");
+       ret = media_packet_get_tbm_surface(mcbuffer->packet, &surface);
+       if (ret != MEDIA_PACKET_ERROR_NONE) {
+               LOGE("Failed to get tbm surface");
                return NULL;
        }
 
        mem = gst_tizen_allocator_alloc_surface(core->allocator,
-                       &vinfo, surface, mcbuffer, (GDestroyNotify)__mc_input_buffer_finalize_cb);
+                       core->video_info, surface, mcbuffer, (GDestroyNotify)__mc_input_buffer_finalize_cb);
 
        return mem;
 }
@@ -2664,28 +3027,19 @@ static void __mc_input_buffer_finalize_cb(GstMCBuffer *mcbuffer)
                return;
 
        LOGD("============>>>>>  _finalize_cb  : %p, %p", mcbuffer, mcbuffer->packet);
-       mc_gst_core_t *core = (mc_gst_core_t *)mcbuffer->core;
 
-       _mc_gst_handle_input_buffer_used(core, mcbuffer->packet);
+       _mc_gst_handle_input_buffer_used(mcbuffer->core, mcbuffer->packet);
 
        g_free(mcbuffer);
-       mcbuffer = NULL;
 
        MEDIACODEC_FLEAVE();
-
-       return;
 }
 
 static GstMCBuffer *gst_mediacodec_buffer_new(mc_gst_core_t *core, media_packet_h packet, uint64_t size)
 {
        GstMCBuffer *mcbuffer = NULL;
 
-       mcbuffer = (GstMCBuffer *)g_malloc0(sizeof(GstMCBuffer));
-
-       if (mcbuffer == NULL) {
-               LOGE("malloc fail");
-               return NULL;
-       }
+       mcbuffer = g_new0(GstMCBuffer, 1);
 
        mcbuffer->buffer = gst_buffer_new();
        mcbuffer->buf_size = 0;
@@ -2925,18 +3279,16 @@ static void _mc_gst_set_flush_output(mc_gst_core_t *core)
        media_packet_h packet = NULL;
 
        MEDIACODEC_FENTER();
+
        g_mutex_lock(&core->ports[1]->mutex);
 
-       while (!g_queue_is_empty(core->ports[1]->queue)) {
-               packet = g_queue_pop_head(core->ports[1]->queue);
-               LOGD("%p poped", packet);
-               if (packet) {
-                       media_packet_destroy(packet);
-                       LOGD("outpacket destroyed");
-                       packet = NULL;
-               }
+       while ((packet = g_queue_pop_head(core->ports[1]->queue))) {
+               LOGD("release packet[%p]", packet);
+               media_packet_unref(packet);
        }
+
        g_mutex_unlock(&core->ports[1]->mutex);
+
        MEDIACODEC_FLEAVE();
 }
 
@@ -3339,8 +3691,14 @@ int _mc_get_mime(mc_gst_core_t *core)
        case MEDIACODEC_AMR_WB:
                mime = MEDIA_FORMAT_AMR_WB;
                break;
+       case MEDIACODEC_OPUS:
+               if (core->encoder)
+                       mime = MEDIA_FORMAT_OPUS;
+               else
+                       mime = MEDIA_FORMAT_PCM_S16LE;
+               break;
        default:
-               LOGE("NOT SUPPORTED!!!!");
+               LOGE("NOT SUPPORTED!!!! 0x%x ", core->codec_id);
                break;
        }
        return mime;
@@ -3421,3 +3779,36 @@ mc_ret_e mc_gst_get_packet_pool(mc_handle_t *mc_handle, media_packet_pool_h *pac
        *packet_pool = pool;
        return MC_ERROR_NONE;
 }
+
+mc_ret_e mc_gst_set_property_array_int(GstElement *element, const gchar *property_name, gint *array, guint array_length)
+{
+       guint i = 0;
+       GValue value = G_VALUE_INIT;
+       GValue tmp = G_VALUE_INIT;
+
+       if (!element || !property_name || !array) {
+               LOGE("NULL param[%p,%p,%p]", element, property_name, array);
+               return MC_PARAM_ERROR;
+       }
+
+       g_value_init(&value, GST_TYPE_ARRAY);
+       g_value_init(&tmp, G_TYPE_INT);
+
+       LOGI("set element[%s], property[%s], array_length[%d]",
+               GST_ELEMENT_NAME(element), property_name, array_length);
+
+       gst_value_array_init(&value, array_length);
+
+       for (i = 0 ; i < array_length ; i++) {
+               LOGI("    array[%u] %d", i, array[i]);
+               g_value_set_int(&tmp, array[i]);
+               gst_value_array_append_value(&value, &tmp);
+       }
+
+       g_object_set_property(G_OBJECT(element), property_name, &value);
+
+       g_value_unset(&tmp);
+       g_value_unset(&value);
+
+       return MC_ERROR_NONE;
+}