#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);
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);
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
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) {
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;
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;
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;
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;
}
* 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;
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;
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;
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;
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);
}
__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");
}
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;
/*
* 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;
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();
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);
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);
/*
* 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();
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();
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;
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;
}
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);
}
}
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;
}
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;
}
}
}
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();
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)
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);
}
}
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;
}
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;
}
}
-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,
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,
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,
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,
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,
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
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;
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,
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)
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:
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");
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");
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:
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:
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;
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) {
}
/* 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);
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 */
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();
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;
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;
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;
/* 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;
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);*/
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);
}
}
+#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);
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);
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;
default:
break;
}
- LOGD("input packet format : %s", format);
+
+ LOGD("input packet format : %s", format ? format : "(null)" );
+
return format;
}
* will not be invoked
*/
gst_buffer_unref(mcbuffer->buffer);
- free(mcbuffer);
+ g_free(mcbuffer);
return NULL;
}
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;
}
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;
}
return ret;
ERROR:
if (packet)
- media_packet_destroy(packet);
+ media_packet_unref(packet);
MEDIACODEC_FLEAVE();
return ret;
}
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.");
{
int ret = MEDIA_PACKET_ERROR_NONE;
tbm_surface_h surface = NULL;
- GstVideoInfo vinfo;
GstMemory *mem = NULL;
if (!mcbuffer->packet) {
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;
}
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;
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();
}
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;
*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;
+}