viddec: some cleanups
[glsdk/gst-plugin-ducati.git] / src / gstducatividdec.c
1 /*
2  * GStreamer
3  * Copyright (c) 2010, Texas Instruments Incorporated
4  *
5  * This library is free software; you can redistribute it and/or
6  * modify it under the terms of the GNU Lesser General Public
7  * License as published by the Free Software Foundation
8  * version 2.1 of the License.
9  *
10  * This library is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13  * Lesser General Public License for more details.
14  *
15  * You should have received a copy of the GNU Lesser General Public
16  * License along with this library; if not, write to the Free Software
17  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
18  */
20 #ifdef HAVE_CONFIG_H
21 #  include <config.h>
22 #endif
24 #include <gst/gst.h>
25 #include <gst/video/video.h>
27 #include "gstducatividdec.h"
29 GST_BOILERPLATE (GstDucatiVidDec, gst_ducati_viddec, GstElement,
30     GST_TYPE_ELEMENT);
32 static GstStaticPadTemplate src_factory = GST_STATIC_PAD_TEMPLATE ("src",
33     GST_PAD_SRC,
34     GST_PAD_ALWAYS,
35     GST_STATIC_CAPS (GST_VIDEO_CAPS_YUV_STRIDED ("NV12", "[ 0, max ]"))
36     );
38 /* helper functions */
40 static void
41 engine_close (GstDucatiVidDec * self)
42 {
43   if (self->engine) {
44     Engine_close (self->engine);
45     self->engine = NULL;
46   }
48   if (self->params) {
49     dce_free (self->params);
50     self->params = NULL;
51   }
53   if (self->dynParams) {
54     dce_free (self->dynParams);
55     self->dynParams = NULL;
56   }
58   if (self->status) {
59     dce_free (self->status);
60     self->status = NULL;
61   }
63   if (self->inBufs) {
64     dce_free (self->inBufs);
65     self->inBufs = NULL;
66   }
68   if (self->outBufs) {
69     dce_free (self->outBufs);
70     self->outBufs = NULL;
71   }
73   if (self->inArgs) {
74     dce_free (self->inArgs);
75     self->inArgs = NULL;
76   }
78   if (self->outArgs) {
79     dce_free (self->outArgs);
80     self->outArgs = NULL;
81   }
82 }
84 static gboolean
85 engine_open (GstDucatiVidDec * self)
86 {
87   gboolean ret;
89   if (G_UNLIKELY (self->engine)) {
90     return TRUE;
91   }
93   GST_DEBUG_OBJECT (self, "opening engine");
95   self->engine = Engine_open ("ivahd_vidsvr", NULL, NULL);
96   if (G_UNLIKELY (!self->engine)) {
97     GST_ERROR_OBJECT (self, "could not create engine");
98     return FALSE;
99   }
101   ret = GST_DUCATIVIDDEC_GET_CLASS (self)->allocate_params (self,
102       sizeof (IVIDDEC3_Params), sizeof (IVIDDEC3_DynamicParams),
103       sizeof (IVIDDEC3_Status), sizeof (IVIDDEC3_InArgs),
104       sizeof (IVIDDEC3_OutArgs));
106   return ret;
109 static void
110 codec_delete (GstDucatiVidDec * self)
112   if (self->pool) {
113     gst_ducati_bufferpool_destroy (self->pool);
114     self->pool = NULL;
115   }
117   if (self->codec) {
118     VIDDEC3_delete(self->codec);
119     self->codec = NULL;
120   }
122   if (self->input) {
123     MemMgr_Free (self->input);
124     self->input = NULL;
125   }
128 static gboolean
129 codec_create (GstDucatiVidDec * self)
131   gint err;
132   const gchar *codec_name;
134   codec_delete (self);
136   if (G_UNLIKELY (!self->engine)) {
137     GST_ERROR_OBJECT (self, "no engine");
138     return FALSE;
139   }
141   /* these need to be set before VIDDEC3_create */
142   self->params->maxWidth = (self->width + 15) & ~0xf;   /* round up to MB */
143   self->params->maxHeight = (self->height + 15) & ~0xf; /* round up to MB */
145   codec_name = GST_DUCATIVIDDEC_GET_CLASS (self)->codec_name;
147   /* create codec: */
148   GST_DEBUG_OBJECT (self, "creating codec: %s", codec_name);
149   self->codec = VIDDEC3_create (self->engine, (char *)codec_name, self->params);
151   if (!self->codec) {
152     return FALSE;
153   }
155   err = VIDDEC3_control (self->codec, XDM_SETPARAMS, self->dynParams, self->status);
156   if (err) {
157     GST_ERROR_OBJECT (self, "failed XDM_SETPARAMS");
158     return FALSE;
159   }
161   self->first_in_buffer = TRUE;
162   self->first_out_buffer = TRUE;
164   /* allocate input buffer and initialize inBufs: */
165   self->inBufs->numBufs = 1;
166   self->input = gst_ducati_alloc_1d (self->width * self->height);
167   self->inBufs->descs[0].buf = (XDAS_Int8 *) TilerMem_VirtToPhys (self->input);
168   self->inBufs->descs[0].memType = XDM_MEMTYPE_RAW;
170   return TRUE;
173 static inline GstBuffer *
174 codec_bufferpool_get (GstDucatiVidDec * self, GstBuffer * buf)
176   if (G_UNLIKELY (!self->pool)) {
177     GST_DEBUG_OBJECT (self, "creating bufferpool");
178     self->pool = gst_ducati_bufferpool_new (GST_ELEMENT (self),
179         GST_PAD_CAPS (self->srcpad));
180   }
181   return GST_BUFFER (gst_ducati_bufferpool_get (self->pool, buf));
184 static XDAS_Int32
185 codec_prepare_outbuf (GstDucatiVidDec * self, GstBuffer * buf)
187   XDAS_Int16 y_type, uv_type;
188   guint8 *y_vaddr, *uv_vaddr;
189   SSPtr y_paddr, uv_paddr;
191   y_vaddr = GST_BUFFER_DATA (buf);
192   uv_vaddr = y_vaddr + self->stride * self->padded_height;
194   y_paddr = TilerMem_VirtToPhys (y_vaddr);
195   uv_paddr = TilerMem_VirtToPhys (uv_vaddr);
197   y_type = gst_ducati_get_mem_type (y_paddr);
198   uv_type = gst_ducati_get_mem_type (uv_paddr);
200   if ((y_type < 0) || (uv_type < 0)) {
201     GST_DEBUG_OBJECT (self, "non TILER buffer, fallback to bufferpool");
202     return codec_prepare_outbuf (self, codec_bufferpool_get (self, buf));
203   }
205   if (!self->outBufs->numBufs) {
206     /* initialize output buffer type */
207     self->outBufs->numBufs = 2;
208     self->outBufs->descs[0].memType = y_type;
209     self->outBufs->descs[0].bufSize.tileMem.width = self->padded_width;
210     self->outBufs->descs[0].bufSize.tileMem.height = self->padded_height;
211     self->outBufs->descs[1].memType = uv_type;
212     /* note that UV interleaved width is same a Y: */
213     self->outBufs->descs[1].bufSize.tileMem.width = self->padded_width;
214     self->outBufs->descs[1].bufSize.tileMem.height = self->padded_height / 2;
215   } else {
216     /* verify output buffer type matches what we've already given
217      * to the codec
218      */
219     if ((self->outBufs->descs[0].memType != y_type) ||
220         (self->outBufs->descs[1].memType != uv_type)) {
221       GST_DEBUG_OBJECT (self, "buffer mismatch, fallback to bufferpool");
222       return codec_prepare_outbuf (self, codec_bufferpool_get (self, buf));
223     }
224   }
226   self->outBufs->descs[0].buf = (XDAS_Int8 *) y_paddr;
227   self->outBufs->descs[1].buf = (XDAS_Int8 *) uv_paddr;
229   return (XDAS_Int32) buf;      // XXX use lookup table
232 static GstBuffer *
233 codec_get_outbuf (GstDucatiVidDec * self, XDAS_Int32 id)
235   GstBuffer *buf = (GstBuffer *) id;    // XXX use lookup table
236   if (buf) {
237     gst_buffer_ref (buf);
238   }
239   return buf;
242 static void
243 codec_unlock_outbuf (GstDucatiVidDec * self, XDAS_Int32 id)
245   GstBuffer *buf = (GstBuffer *) id;    // XXX use lookup table
246   if (buf) {
247     GST_DEBUG_OBJECT (self, "free buffer: %d %p", id, buf);
248     gst_buffer_unref (buf);
249   }
252 static gint
253 codec_process (GstDucatiVidDec * self, gboolean send, gboolean flush)
255   gint err;
256   GstClockTime t;
257   GstBuffer *outbuf = NULL;
258   gint i;
260   self->outArgs->outputID[0] = 0;
261   self->outArgs->freeBufID[0] = 0;
263   t = gst_util_get_timestamp ();
264   err = VIDDEC3_process (self->codec,
265       self->inBufs, self->outBufs, self->inArgs, self->outArgs);
266   GST_INFO_OBJECT (self, "%10dns", (gint) (gst_util_get_timestamp () - t));
268   if (err) {
269     GST_WARNING_OBJECT (self, "err=%d, extendedError=%08x",
270         err, self->outArgs->extendedError);
272     err = VIDDEC3_control (self->codec, XDM_GETSTATUS,
273         self->dynParams, self->status);
275     GST_WARNING_OBJECT (self, "XDM_GETSTATUS: err=%d, extendedError=%08x",
276         err, self->status->extendedError);
278     if (XDM_ISFATALERROR (self->outArgs->extendedError) || flush) {
279       /* we are processing for display and it is a non-fatal error, so lets
280        * try to recover.. otherwise return the error
281        */
282       err = XDM_EFAIL;
283     }
284   }
286   for (i = 0; self->outArgs->outputID[i]; i++) {
287     if (G_UNLIKELY (self->first_out_buffer) && send) {
288       /* send region of interest to sink on first buffer: */
289       XDM_Rect *r = &(self->outArgs->displayBufs.bufDesc[0].activeFrameRegion);
291       GST_DEBUG_OBJECT (self, "setting crop to %d, %d, %d, %d",
292           r->topLeft.x, r->topLeft.y, r->bottomRight.x, r->bottomRight.y);
294       gst_pad_push_event (self->srcpad,
295           gst_event_new_crop (r->topLeft.y, r->topLeft.x,
296               r->bottomRight.x - r->topLeft.x,
297               r->bottomRight.y - r->topLeft.y));
299       self->first_out_buffer = FALSE;
300     }
302     outbuf = codec_get_outbuf (self, self->outArgs->outputID[i]);
303     if (send) {
304       if (GST_IS_DUCATIBUFFER (outbuf)) {
305         outbuf = gst_ducati_buffer_get (GST_DUCATIBUFFER (outbuf));
306       }
307       GST_DEBUG_OBJECT (self, "got buffer: %d %p (%" GST_TIME_FORMAT ")",
308           i, outbuf, GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (outbuf)));
309       gst_pad_push (self->srcpad, outbuf);
310     } else {
311       GST_DEBUG_OBJECT (self, "free buffer: %d %p", i, outbuf);
312       gst_buffer_unref (outbuf);
313     }
314   }
316   for (i = 0; self->outArgs->freeBufID[i]; i++) {
317     codec_unlock_outbuf (self, self->outArgs->freeBufID[i]);
318   }
320   return err;
323 /** call control(FLUSH), and then process() to pop out all buffers */
324 static gboolean
325 codec_flush (GstDucatiVidDec * self, gboolean eos)
327   gint err;
329   GST_DEBUG_OBJECT (self, "flush: eos=%d", eos);
331   /* note: flush is synchronized against _chain() to avoid calling
332    * the codec from multiple threads
333    */
334   GST_PAD_STREAM_LOCK (self->sinkpad);
336   if (G_UNLIKELY (self->first_in_buffer)) {
337     return TRUE;
338   }
340   if (G_UNLIKELY (!self->codec)) {
341     GST_WARNING_OBJECT (self, "no codec");
342     return TRUE;
343   }
345   err = VIDDEC3_control (self->codec, XDM_FLUSH,
346       self->dynParams, self->status);
347   if (err) {
348     GST_ERROR_OBJECT (self, "failed XDM_FLUSH");
349     goto out;
350   }
352   self->inBufs->descs[0].bufSize.bytes = 0;
353   self->inArgs->numBytes = 0;
354   self->inArgs->inputID = 0;
356   do {
357     err = codec_process (self, eos, TRUE);
358   } while (err != XDM_EFAIL);
360   /* on a flush, it is normal (and not an error) for the last _process() call
361    * to return an error..
362    */
363   err = XDM_EOK;
365 out:
366   GST_PAD_STREAM_UNLOCK (self->sinkpad);
367   GST_DEBUG_OBJECT (self, "done");
369   return !err;
372 /* GstDucatiVidDec vmethod default implementations */
374 static gboolean
375 gst_ducati_viddec_parse_caps (GstDucatiVidDec * self, GstStructure * s)
377   const GValue *codec_data;
379   if (gst_structure_get_int (s, "width", &self->width) &&
380       gst_structure_get_int (s, "height", &self->height)) {
382     const GValue *codec_data = gst_structure_get_value (s, "codec_data");
384     if (codec_data) {
385       GstBuffer *buffer = gst_value_get_buffer (codec_data);
386       GST_DEBUG_OBJECT (self, "codec_data: %" GST_PTR_FORMAT, buffer);
387       self->codec_data = gst_buffer_ref (buffer);
388     }
390     return TRUE;
391   }
393   return FALSE;
396 static gboolean
397 gst_ducati_viddec_allocate_params (GstDucatiVidDec * self, gint params_sz,
398     gint dynparams_sz, gint status_sz, gint inargs_sz, gint outargs_sz)
401   /* allocate params: */
402   self->params = dce_alloc (params_sz);
403   if (G_UNLIKELY (!self->params)) {
404     return FALSE;
405   }
406   self->params->size = params_sz;
407   self->params->maxFrameRate = 30000;
408   self->params->maxBitRate = 10000000;
410   self->params->dataEndianness = XDM_BYTE;
411   self->params->forceChromaFormat = XDM_YUV_420SP;
412   self->params->operatingMode = IVIDEO_DECODE_ONLY;
414   self->params->displayBufsMode = IVIDDEC3_DISPLAYBUFS_EMBEDDED;
415   self->params->inputDataMode = IVIDEO_ENTIREFRAME;
416   self->params->outputDataMode = IVIDEO_ENTIREFRAME;
417   self->params->numInputDataUnits = 0;
418   self->params->numOutputDataUnits = 0;
420   self->params->metadataType[0] = IVIDEO_METADATAPLANE_NONE;
421   self->params->metadataType[1] = IVIDEO_METADATAPLANE_NONE;
422   self->params->metadataType[2] = IVIDEO_METADATAPLANE_NONE;
423   self->params->errorInfoMode = IVIDEO_ERRORINFO_OFF;
425   /* allocate dynParams: */
426   self->dynParams = dce_alloc (dynparams_sz);
427   if (G_UNLIKELY (!self->dynParams)) {
428     return FALSE;
429   }
430   self->dynParams->size = dynparams_sz;
431   self->dynParams->decodeHeader = XDM_DECODE_AU;
432   self->dynParams->displayWidth = 0;
433   self->dynParams->frameSkipMode = IVIDEO_NO_SKIP;
434   self->dynParams->newFrameFlag = XDAS_TRUE;
436   /* allocate status: */
437   self->status = dce_alloc (status_sz);
438   if (G_UNLIKELY (!self->status)) {
439     return FALSE;
440   }
441   self->status->size = status_sz;
443   /* allocate inBufs/outBufs: */
444   self->inBufs = dce_alloc (sizeof (XDM2_BufDesc));
445   self->outBufs = dce_alloc (sizeof (XDM2_BufDesc));
446   if (G_UNLIKELY (!self->inBufs) || G_UNLIKELY (!self->outBufs)) {
447     return FALSE;
448   }
450   /* allocate inArgs/outArgs: */
451   self->inArgs = dce_alloc (inargs_sz);
452   self->outArgs = dce_alloc (outargs_sz);
453   if (G_UNLIKELY (!self->inArgs) || G_UNLIKELY (!self->outArgs)) {
454     return FALSE;
455   }
456   self->inArgs->size = inargs_sz;
457   self->outArgs->size = outargs_sz;
460 static GstBuffer *
461 gst_ducati_viddec_push_input (GstDucatiVidDec * self, GstBuffer * buf)
463   if (G_UNLIKELY (self->first_in_buffer) && self->codec_data) {
464     push_input (self, GST_BUFFER_DATA (self->codec_data),
465         GST_BUFFER_SIZE (self->codec_data));
466   }
468   /* just copy entire buffer */
469   push_input (self, GST_BUFFER_DATA (buf), GST_BUFFER_SIZE (buf));
470   gst_buffer_unref (buf);
472   return NULL;
475 /* GstElement vmethod implementations */
477 static gboolean
478 gst_ducati_viddec_set_caps (GstPad * pad, GstCaps * caps)
480   gboolean ret = TRUE;
481   GstDucatiVidDec *self = GST_DUCATIVIDDEC (gst_pad_get_parent (pad));
482   GstDucatiVidDecClass *klass = GST_DUCATIVIDDEC_GET_CLASS (self);
483   GstStructure *s;
485   g_return_val_if_fail (caps, FALSE);
486   g_return_val_if_fail (gst_caps_is_fixed (caps), FALSE);
488   s = gst_caps_get_structure (caps, 0);
490   if (pad == self->sinkpad) {
491     gint frn = 0, frd = 1;
492     GST_INFO_OBJECT (self, "setcaps (sink): %" GST_PTR_FORMAT, caps);
494     if (klass->parse_caps (self, s)) {
495       GstCaps *outcaps;
496       gboolean interlaced = FALSE;
498       gst_structure_get_fraction (s, "framerate", &frn, &frd);
500       self->stride = 4096;      /* TODO: don't hardcode */
502       gst_structure_get_boolean (s, "interlaced", &interlaced);
504       /* update output/padded sizes:
505        */
506       klass->update_buffer_size (self);
508       self->outsize =
509           GST_ROUND_UP_2 (self->stride * self->padded_height * 3) / 2;
511       outcaps = gst_caps_new_simple ("video/x-raw-yuv-strided",
512           "rowstride", G_TYPE_INT, self->stride,
513           "format", GST_TYPE_FOURCC, GST_MAKE_FOURCC ('N','V','1','2'),
514           "width", G_TYPE_INT, self->padded_width,
515           "height", G_TYPE_INT, self->padded_height,
516           "framerate", GST_TYPE_FRACTION, frn, frd,
517           NULL);
519       if (interlaced) {
520         gst_caps_set_simple (outcaps, "interlaced", G_TYPE_BOOLEAN, TRUE, NULL);
521       }
523       GST_DEBUG_OBJECT (self, "outcaps: %" GST_PTR_FORMAT, outcaps);
525       ret = gst_pad_set_caps (self->srcpad, outcaps);
526       gst_caps_unref (outcaps);
528       if (!ret) {
529         GST_WARNING_OBJECT (self, "failed to set caps");
530         return FALSE;
531       }
532     } else {
533       GST_WARNING_OBJECT (self, "missing required fields");
534       return FALSE;
535     }
536   } else {
537     GST_INFO_OBJECT (self, "setcaps (src): %" GST_PTR_FORMAT, caps);
538     // XXX check to make sure caps are ok.. keep track if we
539     // XXX need to handle unstrided buffers..
540     GST_WARNING_OBJECT (self, "TODO");
541   }
543   gst_object_unref (self);
545   return gst_pad_set_caps (pad, caps);
548 static gboolean
549 gst_ducati_viddec_query (GstPad * pad, GstQuery * query)
551   GstDucatiVidDec *self = GST_DUCATIVIDDEC (GST_OBJECT_PARENT (pad));
553   switch (GST_QUERY_TYPE (query)) {
554     case GST_QUERY_BUFFERS:
555       GST_DEBUG_OBJECT (self, "min buffers: %d", self->min_buffers);
556       gst_query_set_buffers_count (query, self->min_buffers);
558       GST_DEBUG_OBJECT (self, "min dimensions: %dx%d",
559           self->padded_width, self->padded_height);
560       gst_query_set_buffers_dimensions (query,
561           self->padded_width, self->padded_height);
562       return TRUE;
563     default:
564       return FALSE;
565   }
568 static GstFlowReturn
569 gst_ducati_viddec_chain (GstPad * pad, GstBuffer * buf)
571   GstDucatiVidDec *self = GST_DUCATIVIDDEC (GST_OBJECT_PARENT (pad));
572   GstFlowReturn ret;
573   Int32 err;
574   GstBuffer *outbuf = NULL;
576   if (G_UNLIKELY (!self->engine)) {
577     GST_ERROR_OBJECT (self, "no engine");
578     return GST_FLOW_ERROR;
579   }
581   /* do this before creating codec to ensure reverse caps negotiation
582    * happens first:
583    */
584   ret = gst_pad_alloc_buffer_and_set_caps (self->srcpad, 0, self->outsize,
585       GST_PAD_CAPS (self->srcpad), &outbuf);
587   if (ret != GST_FLOW_OK) {
588     outbuf = codec_bufferpool_get (self, NULL);
589     ret = GST_FLOW_OK;
590   }
592   if (G_UNLIKELY (!self->codec)) {
593     if (!codec_create (self)) {
594       GST_ERROR_OBJECT (self, "could not create codec");
595       return GST_FLOW_ERROR;
596     }
597   }
599   GST_BUFFER_TIMESTAMP (outbuf) = GST_BUFFER_TIMESTAMP (buf);
600   GST_BUFFER_DURATION (outbuf) = GST_BUFFER_DURATION (buf);
602   /* pass new output buffer as to the decoder to decode into: */
603   self->inArgs->inputID = codec_prepare_outbuf (self, outbuf);
604   if (!self->inArgs->inputID) {
605     GST_ERROR_OBJECT (self, "could not prepare output buffer");
606     return GST_FLOW_ERROR;
607   }
609   self->in_size = 0;
610   buf = GST_DUCATIVIDDEC_GET_CLASS (self)->push_input (self, buf);
612   if (self->in_size == 0) {
613     GST_DEBUG_OBJECT (self, "no input, skipping process");
614     gst_buffer_unref (outbuf);
615     return GST_FLOW_OK;
616   }
618   self->inArgs->numBytes = self->in_size;
619   self->inBufs->descs[0].bufSize.bytes = self->in_size;
621   if (buf) {
622     // XXX
623     GST_WARNING_OBJECT (self, "TODO.. can't push more than one.. need loop");
624     gst_buffer_unref (buf);
625     buf = NULL;
626   }
628   err = codec_process (self, TRUE, FALSE);
629   if (err) {
630     GST_ERROR_OBJECT (self, "process returned error: %d %08x",
631         err, self->outArgs->extendedError);
632     return GST_FLOW_ERROR;
633   }
635   self->first_in_buffer = FALSE;
637   if (self->outArgs->outBufsInUseFlag) {
638     GST_WARNING_OBJECT (self, "TODO... outBufsInUseFlag");      // XXX
639   }
641   return GST_FLOW_OK;
644 static gboolean
645 gst_ducati_viddec_event (GstPad * pad, GstEvent * event)
647   GstDucatiVidDec *self = GST_DUCATIVIDDEC (GST_OBJECT_PARENT (pad));
648   gboolean ret = TRUE;
649   gboolean eos = FALSE;
651   GST_INFO_OBJECT (self, "begin: event=%s", GST_EVENT_TYPE_NAME (event));
653   switch (GST_EVENT_TYPE (event)) {
654     case GST_EVENT_EOS:
655       eos = TRUE;
656       /* fall-through */
657     case GST_EVENT_FLUSH_STOP:
658       if (!codec_flush (self, eos)) {
659         GST_ERROR_OBJECT (self, "could not flush");
660         return FALSE;
661       }
662       /* fall-through */
663     default:
664       ret = gst_pad_push_event (self->srcpad, event);
665       break;
666   }
668   GST_LOG_OBJECT (self, "end");
670   return ret;
673 static GstStateChangeReturn
674 gst_ducati_viddec_change_state (GstElement * element,
675     GstStateChange transition)
677   GstStateChangeReturn ret = GST_STATE_CHANGE_SUCCESS;
678   GstDucatiVidDec *self = GST_DUCATIVIDDEC (element);
680   GST_INFO_OBJECT (self, "begin: changing state %s -> %s",
681       gst_element_state_get_name (GST_STATE_TRANSITION_CURRENT (transition)),
682       gst_element_state_get_name (GST_STATE_TRANSITION_NEXT (transition)));
684   switch (transition) {
685     case GST_STATE_CHANGE_NULL_TO_READY:
686       if (!engine_open (self)) {
687         GST_ERROR_OBJECT (self, "could not open");
688         return GST_STATE_CHANGE_FAILURE;
689       }
690       break;
691     default:
692       break;
693   }
695   ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
697   if (ret == GST_STATE_CHANGE_FAILURE)
698     goto leave;
700   switch (transition) {
701     case GST_STATE_CHANGE_READY_TO_NULL:
702       codec_delete (self);
703       engine_close (self);
704       break;
705     default:
706       break;
707   }
709 leave:
710   GST_LOG_OBJECT (self, "end");
712   return ret;
715 /* GObject vmethod implementations */
717 static void
718 gst_ducati_viddec_finalize (GObject * obj)
720   GstDucatiVidDec *self = GST_DUCATIVIDDEC (obj);
722   codec_delete (self);
723   engine_close (self);
725   if (self->codec_data) {
726       gst_buffer_unref (self->codec_data);
727       self->codec_data = NULL;
728   }
730   G_OBJECT_CLASS (parent_class)->finalize (obj);
733 static void
734 gst_ducati_viddec_base_init (gpointer gclass)
736   GstElementClass *element_class = GST_ELEMENT_CLASS (gclass);
738   gst_element_class_add_pad_template (element_class,
739       gst_static_pad_template_get (&src_factory));
742 static void
743 gst_ducati_viddec_class_init (GstDucatiVidDecClass * klass)
745   GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
746   GstElementClass *gstelement_class = GST_ELEMENT_CLASS (klass);
748   gobject_class->finalize = gst_ducati_viddec_finalize;
749   gstelement_class->change_state = gst_ducati_viddec_change_state;
751   klass->parse_caps =
752       GST_DEBUG_FUNCPTR (gst_ducati_viddec_parse_caps);
753   klass->allocate_params =
754       GST_DEBUG_FUNCPTR (gst_ducati_viddec_allocate_params);
755   klass->push_input =
756       GST_DEBUG_FUNCPTR (gst_ducati_viddec_push_input);
759 static void
760 gst_ducati_viddec_init (GstDucatiVidDec * self, GstDucatiVidDecClass * klass)
762   GstElementClass *gstelement_class = GST_ELEMENT_CLASS (klass);
764   self->sinkpad = gst_pad_new_from_template (
765       gst_element_class_get_pad_template (gstelement_class, "sink"), "sink");
766   gst_pad_set_setcaps_function (self->sinkpad,
767       GST_DEBUG_FUNCPTR (gst_ducati_viddec_set_caps));
768   gst_pad_set_chain_function (self->sinkpad,
769       GST_DEBUG_FUNCPTR (gst_ducati_viddec_chain));
770   gst_pad_set_event_function (self->sinkpad,
771       GST_DEBUG_FUNCPTR (gst_ducati_viddec_event));
773   self->srcpad = gst_pad_new_from_static_template (&src_factory, "src");
774   gst_pad_set_setcaps_function (self->srcpad,
775       GST_DEBUG_FUNCPTR (gst_ducati_viddec_set_caps));
776   gst_pad_set_query_function (self->srcpad,
777           GST_DEBUG_FUNCPTR (gst_ducati_viddec_query));
779   gst_element_add_pad (GST_ELEMENT (self), self->sinkpad);
780   gst_element_add_pad (GST_ELEMENT (self), self->srcpad);