| /* |
| * Copyright (C) 2010 Ole André Vadla Ravnås <oleavr@soundrop.com> |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Library General Public |
| * License as published by the Free Software Foundation; either |
| * version 2 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Library General Public License for more details. |
| * |
| * You should have received a copy of the GNU Library General Public |
| * License along with this library; if not, write to the |
| * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, |
| * Boston, MA 02110-1301, USA. |
| */ |
| |
| #include "corevideobuffer.h" |
| |
| static gboolean |
| gst_core_video_meta_init (GstCoreVideoMeta * meta, gpointer params, |
| GstBuffer * buf) |
| { |
| meta->ctx = NULL; |
| meta->cvbuf = NULL; |
| meta->pixbuf = NULL; |
| |
| return TRUE; |
| } |
| |
| static void |
| gst_core_video_meta_free (GstCoreVideoMeta * meta, GstBuffer * buf) |
| { |
| GstCVApi *cv = meta->ctx->cv; |
| |
| if (meta->pixbuf != NULL) { |
| cv->CVPixelBufferUnlockBaseAddress (meta->pixbuf, |
| kCVPixelBufferLock_ReadOnly); |
| } |
| |
| cv->CVBufferRelease (meta->cvbuf); |
| g_object_unref (meta->ctx); |
| } |
| |
| GType |
| gst_core_video_meta_api_get_type (void) |
| { |
| static volatile GType type; |
| static const gchar *tags[] = { "memory", NULL }; |
| |
| if (g_once_init_enter (&type)) { |
| GType _type = gst_meta_api_type_register ("GstCoreVideoMetaAPI", tags); |
| g_once_init_leave (&type, _type); |
| } |
| return type; |
| } |
| |
| static const GstMetaInfo * |
| gst_core_video_meta_get_info (void) |
| { |
| static const GstMetaInfo *core_video_meta_info = NULL; |
| |
| if (g_once_init_enter (&core_video_meta_info)) { |
| const GstMetaInfo *meta = gst_meta_register (GST_CORE_VIDEO_META_API_TYPE, |
| "GstCoreVideoMeta", sizeof (GstCoreVideoMeta), |
| (GstMetaInitFunction) gst_core_video_meta_init, |
| (GstMetaFreeFunction) gst_core_video_meta_free, |
| (GstMetaTransformFunction) NULL); |
| g_once_init_leave (&core_video_meta_info, meta); |
| } |
| return core_video_meta_info; |
| } |
| |
| GstBuffer * |
| gst_core_video_buffer_new (GstCoreMediaCtx * ctx, CVBufferRef cvbuf, |
| GstVideoInfo * vinfo) |
| { |
| GstCVApi *cv = ctx->cv; |
| void *data; |
| size_t size; |
| CVPixelBufferRef pixbuf = NULL; |
| GstBuffer *buf; |
| GstCoreVideoMeta *meta; |
| guint width, height, n_planes, i; |
| gsize offset[GST_VIDEO_MAX_PLANES]; |
| gint stride[GST_VIDEO_MAX_PLANES]; |
| |
| if (CFGetTypeID (cvbuf) != cv->CVPixelBufferGetTypeID ()) |
| /* TODO: Do we need to handle other buffer types? */ |
| goto error; |
| |
| pixbuf = (CVPixelBufferRef) cvbuf; |
| |
| if (cv->CVPixelBufferLockBaseAddress (pixbuf, |
| kCVPixelBufferLock_ReadOnly) != kCVReturnSuccess) { |
| goto error; |
| } |
| |
| buf = gst_buffer_new (); |
| |
| /* add the corevideo meta to free the underlying corevideo buffer */ |
| meta = (GstCoreVideoMeta *) gst_buffer_add_meta (buf, |
| gst_core_video_meta_get_info (), NULL); |
| meta->ctx = g_object_ref (ctx); |
| meta->cvbuf = cv->CVBufferRetain (cvbuf); |
| meta->pixbuf = pixbuf; |
| |
| /* set stride, offset and size */ |
| memset (&offset, 0, sizeof (offset)); |
| memset (&stride, 0, sizeof (stride)); |
| |
| data = cv->CVPixelBufferGetBaseAddress (pixbuf); |
| height = cv->CVPixelBufferGetHeight (pixbuf); |
| if (cv->CVPixelBufferIsPlanar (pixbuf)) { |
| GstVideoInfo tmp_vinfo; |
| |
| n_planes = cv->CVPixelBufferGetPlaneCount (pixbuf); |
| for (i = 0; i < n_planes; ++i) |
| stride[i] = cv->CVPixelBufferGetBytesPerRowOfPlane (pixbuf, i); |
| |
| /* FIXME: don't hardcode NV12 */ |
| gst_video_info_init (&tmp_vinfo); |
| gst_video_info_set_format (&tmp_vinfo, |
| GST_VIDEO_FORMAT_NV12, stride[0], height); |
| offset[1] = tmp_vinfo.offset[1]; |
| size = tmp_vinfo.size; |
| } else { |
| n_planes = 1; |
| size = cv->CVPixelBufferGetBytesPerRow (pixbuf) * height; |
| } |
| |
| gst_buffer_append_memory (buf, |
| gst_memory_new_wrapped (GST_MEMORY_FLAG_NO_SHARE, data, |
| size, 0, size, NULL, NULL)); |
| |
| if (vinfo) { |
| GstVideoMeta *video_meta; |
| |
| width = vinfo->width; |
| video_meta = |
| gst_buffer_add_video_meta_full (buf, GST_VIDEO_FRAME_FLAG_NONE, |
| GST_VIDEO_FORMAT_NV12, width, height, n_planes, offset, stride); |
| } |
| |
| return buf; |
| |
| error: |
| return NULL; |
| } |