/* GStreamer PNM decoder * Copyright (C) 2009 Lutz Mueller * * 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., 59 Temple Place - Suite 330, * Boston, MA 02111-1307, USA. */ /** * SECTION:element-pnmdec * * Decodes pnm images. * * * Example launch line * |[ * gst-launch filesrc location=test.pnm ! pnmdec ! videoconvert ! autovideosink * ]| The above pipeline reads a pnm file and renders it to the screen. * */ /* * FIXME: Port to GstVideoDecoder */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include "gstpnmdec.h" #include "gstpnmutils.h" #include #include #include static GstStaticPadTemplate gst_pnmdec_src_pad_template = GST_STATIC_PAD_TEMPLATE ("src", GST_PAD_SRC, GST_PAD_ALWAYS, GST_STATIC_CAPS (GST_VIDEO_CAPS_MAKE ("RGB") "; " GST_VIDEO_CAPS_MAKE ("GRAY8"))); static GstStaticPadTemplate gst_pnmdec_sink_pad_template = GST_STATIC_PAD_TEMPLATE ("sink", GST_PAD_SINK, GST_PAD_ALWAYS, GST_STATIC_CAPS (MIME_ALL)); G_DEFINE_TYPE (GstPnmdec, gst_pnmdec, GST_TYPE_ELEMENT); static GstFlowReturn gst_pnmdec_push (GstPnmdec * s, GstPad * src, GstBuffer * buf) { /* Need to convert from PNM rowstride to GStreamer rowstride */ if (s->mngr.info.width % 4 != 0) { guint i_rowstride; guint o_rowstride; GstBuffer *obuf; guint i; GstMapInfo imap, omap; if (s->mngr.info.type == GST_PNM_TYPE_PIXMAP) { i_rowstride = 3 * s->mngr.info.width; o_rowstride = GST_ROUND_UP_4 (i_rowstride); } else { i_rowstride = s->mngr.info.width; o_rowstride = GST_ROUND_UP_4 (i_rowstride); } obuf = gst_buffer_new_and_alloc (o_rowstride * s->mngr.info.height); gst_buffer_copy_into (obuf, buf, GST_BUFFER_COPY_METADATA, 0, 0); gst_buffer_map (obuf, &omap, GST_MAP_WRITE); gst_buffer_map (buf, &imap, GST_MAP_READ); for (i = 0; i < s->mngr.info.height; i++) memcpy (omap.data + i * o_rowstride, imap.data + i * i_rowstride, i_rowstride); gst_buffer_unmap (buf, &imap); gst_buffer_unmap (obuf, &omap); gst_buffer_unref (buf); return gst_pad_push (src, obuf); } else { return gst_pad_push (src, buf); } } static GstFlowReturn gst_pnmdec_chain_raw (GstPnmdec * s, GstPad * src, GstBuffer * buf) { GstFlowReturn r = GST_FLOW_OK; GstBuffer *out; /* If we got the whole image, just push the buffer. */ if (gst_buffer_get_size (buf) == s->size) { memset (&s->mngr, 0, sizeof (GstPnmInfoMngr)); s->size = 0; return gst_pnmdec_push (s, src, buf); } /* We didn't get the whole image. */ if (!s->buf) { s->buf = buf; } else { out = gst_buffer_append (s->buf, buf); s->buf = out; } if (!s->buf) return GST_FLOW_ERROR; /* Do we now have the full image? If yes, push. */ if (gst_buffer_get_size (s->buf) == s->size) { r = gst_pnmdec_push (s, src, s->buf); s->buf = NULL; memset (&s->mngr, 0, sizeof (GstPnmInfoMngr)); s->size = 0; } return r; } static GstFlowReturn gst_pnmdec_chain_ascii (GstPnmdec * s, GstPad * src, GstBuffer * buf) { GScanner *scanner; GstBuffer *out; guint i = 0; gchar *b; guint bs; guint target; GstMapInfo map; GstMapInfo outmap; gst_buffer_map (buf, &map, GST_MAP_READ); b = (gchar *) map.data; bs = map.size; target = s->size - (s->buf ? map.size : 0); if (!bs) goto drop_ok; if (s->last_byte) { while (*b >= '0' && *b <= '9') { s->last_byte = 10 * s->last_byte + *b - '0'; b++; if (!--bs) { goto drop_error; } } if (s->last_byte > 255) { GST_DEBUG_OBJECT (s, "Corrupt ASCII encoded PNM file."); goto drop_error; } } out = gst_buffer_new_and_alloc (target); gst_buffer_map (out, &outmap, GST_MAP_READWRITE); if (s->last_byte) { outmap.data[i++] = s->last_byte; s->last_byte = 0; } scanner = g_scanner_new (NULL); g_scanner_input_text (scanner, b, bs); while (!g_scanner_eof (scanner)) { switch (g_scanner_get_next_token (scanner)) { case G_TOKEN_INT: if (i == target) { GST_DEBUG_OBJECT (s, "PNM file contains too much data."); gst_buffer_unmap (out, &outmap); gst_buffer_unref (out); goto drop_error; } outmap.data[i++] = scanner->value.v_int; break; default: /* Should we care? */ ; } } g_scanner_destroy (scanner); /* If we didn't get the whole image, handle the last byte with care. */ if (i && i < target && b[bs - 1] > '0' && b[bs - 1] <= '9') s->last_byte = outmap.data[--i]; gst_buffer_unmap (buf, &map); gst_buffer_unref (buf); if (!i) { gst_buffer_unref (out); return GST_FLOW_OK; } gst_buffer_set_size (out, i); return gst_pnmdec_chain_raw (s, src, out); drop_ok: gst_buffer_unmap (buf, &map); gst_buffer_unref (buf); return GST_FLOW_OK; drop_error: gst_buffer_unmap (buf, &map); gst_buffer_unref (buf); return GST_FLOW_ERROR; } static GstFlowReturn gst_pnmdec_chain (GstPad * pad, GstObject * parent, GstBuffer * data) { GstPnmdec *s = GST_PNMDEC (parent); GstPad *src = gst_element_get_static_pad (GST_ELEMENT (s), "src"); GstCaps *caps = NULL; GstFlowReturn r = GST_FLOW_OK; guint offset = 0; if (s->mngr.info.fields != GST_PNM_INFO_FIELDS_ALL) { GstMapInfo map; GstPnmInfoMngrResult res; gst_buffer_map (data, &map, GST_MAP_READ); res = gst_pnm_info_mngr_scan (&s->mngr, map.data, map.size); gst_buffer_unmap (data, &map); switch (res) { case GST_PNM_INFO_MNGR_RESULT_FAILED: gst_buffer_unref (data); r = GST_FLOW_ERROR; goto out; case GST_PNM_INFO_MNGR_RESULT_READING: gst_buffer_unref (data); r = GST_FLOW_OK; goto out; case GST_PNM_INFO_MNGR_RESULT_FINISHED: offset = s->mngr.data_offset; caps = gst_caps_copy (gst_pad_get_pad_template_caps (src)); switch (s->mngr.info.type) { case GST_PNM_TYPE_BITMAP: GST_DEBUG_OBJECT (s, "FIXME: BITMAP format not implemented!"); gst_caps_unref (caps); gst_buffer_unref (data); r = GST_FLOW_ERROR; goto out; case GST_PNM_TYPE_GRAYMAP: gst_caps_remove_structure (caps, 0); s->size = s->mngr.info.width * s->mngr.info.height * 1; break; case GST_PNM_TYPE_PIXMAP: gst_caps_remove_structure (caps, 1); s->size = s->mngr.info.width * s->mngr.info.height * 3; break; } gst_caps_set_simple (caps, "width", G_TYPE_INT, s->mngr.info.width, "height", G_TYPE_INT, s->mngr.info.height, "framerate", GST_TYPE_FRACTION, 0, 1, NULL); if (!gst_pad_set_caps (src, caps)) { gst_caps_unref (caps); gst_buffer_unref (data); r = GST_FLOW_ERROR; goto out; } gst_caps_unref (caps); } } if (offset == gst_buffer_get_size (data)) { gst_buffer_unref (data); r = GST_FLOW_OK; goto out; } if (offset) { GstBuffer *buf = gst_buffer_copy_region (data, GST_BUFFER_COPY_ALL, offset, gst_buffer_get_size (data) - offset); gst_buffer_unref (data); data = buf; } if (s->mngr.info.encoding == GST_PNM_ENCODING_ASCII) r = gst_pnmdec_chain_ascii (s, src, data); else r = gst_pnmdec_chain_raw (s, src, data); out: gst_object_unref (src); return r; } static void gst_pnmdec_finalize (GObject * object) { GstPnmdec *dec = GST_PNMDEC (object); if (dec->buf) { gst_buffer_unref (dec->buf); dec->buf = NULL; } G_OBJECT_CLASS (gst_pnmdec_parent_class)->finalize (object); } static void gst_pnmdec_init (GstPnmdec * s) { GstPad *pad; pad = gst_pad_new_from_static_template (&gst_pnmdec_sink_pad_template, "sink"); gst_pad_set_chain_function (pad, gst_pnmdec_chain); gst_element_add_pad (GST_ELEMENT (s), pad); pad = gst_pad_new_from_static_template (&gst_pnmdec_src_pad_template, "src"); gst_element_add_pad (GST_ELEMENT (s), pad); } static void gst_pnmdec_class_init (GstPnmdecClass * klass) { GObjectClass *gobject_class = (GObjectClass *) klass; GstElementClass *element_class = GST_ELEMENT_CLASS (klass); gst_element_class_add_pad_template (element_class, gst_static_pad_template_get (&gst_pnmdec_sink_pad_template)); gst_element_class_add_pad_template (element_class, gst_static_pad_template_get (&gst_pnmdec_src_pad_template)); gst_element_class_set_metadata (element_class, "PNM image decoder", "Codec/Decoder/Image", "Decodes images in portable pixmap/graymap/bitmap/anymamp (PNM) format", "Lutz Mueller "); gobject_class->finalize = gst_pnmdec_finalize; }