Version 4.0.0.1, tag libreoffice-4.0.0.1
[LibreOffice.git] / avmedia / source / gstreamer / gstframegrabber.cxx
blobed998ddd7d77771e5ee1f4e673b29a5818d2b8cf
1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 /*
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
9 * This file incorporates work covered by the following license notice:
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
20 #include "gstframegrabber.hxx"
21 #include "gstplayer.hxx"
23 #include <gst/gstbuffer.h>
24 #include <gst/video/video.h>
25 #include <gst/video/gstvideosink.h>
27 #include <vcl/graph.hxx>
28 #include <vcl/bmpacc.hxx>
30 #include <string>
32 #ifdef AVMEDIA_GST_0_10
33 # define AVMEDIA_GST_FRAMEGRABBER_IMPLEMENTATIONNAME "com.sun.star.comp.avmedia.FrameGrabber_GStreamer_0_10"
34 # define AVMEDIA_GST_FRAMEGRABBER_SERVICENAME "com.sun.star.media.FrameGrabber_GStreamer_0_10"
35 #else
36 # define AVMEDIA_GST_FRAMEGRABBER_IMPLEMENTATIONNAME "com.sun.star.comp.avmedia.FrameGrabber_GStreamer"
37 # define AVMEDIA_GST_FRAMEGRABBER_SERVICENAME "com.sun.star.media.FrameGrabber_GStreamer"
38 #endif
40 using namespace ::com::sun::star;
42 namespace avmedia { namespace gstreamer {
44 void FrameGrabber::disposePipeline()
46 if( mpPipeline != NULL )
48 gst_element_set_state( mpPipeline, GST_STATE_NULL );
49 g_object_unref( G_OBJECT( mpPipeline ) );
50 mpPipeline = NULL;
54 FrameGrabber::FrameGrabber( const OUString &rURL ) :
55 FrameGrabber_BASE()
57 gchar *pPipelineStr;
58 pPipelineStr = g_strdup_printf(
59 #ifdef AVMEDIA_GST_0_10
60 "uridecodebin uri=%s ! ffmpegcolorspace ! videoscale ! appsink "
61 "name=sink caps=\"video/x-raw-rgb,format=RGB,pixel-aspect-ratio=1/1,"
62 "bpp=(int)24,depth=(int)24,endianness=(int)4321,"
63 "red_mask=(int)0xff0000, green_mask=(int)0x00ff00, blue_mask=(int)0x0000ff\"",
64 #else
65 "uridecodebin uri=%s ! videoconvert ! videoscale ! appsink "
66 "name=sink caps=\"video/x-raw,format=RGB,pixel-aspect-ratio=1/1\"",
67 #endif
68 rtl::OUStringToOString( rURL, RTL_TEXTENCODING_UTF8 ).getStr() );
70 GError *pError = NULL;
71 mpPipeline = gst_parse_launch( pPipelineStr, &pError );
72 if( pError != NULL) {
73 g_warning( "Failed to construct frame-grabber pipeline '%s'\n", pError->message );
74 g_error_free( pError );
75 disposePipeline();
78 if( mpPipeline ) {
79 // pre-roll
80 switch( gst_element_set_state( mpPipeline, GST_STATE_PAUSED ) ) {
81 case GST_STATE_CHANGE_FAILURE:
82 case GST_STATE_CHANGE_NO_PREROLL:
83 g_warning( "failure pre-rolling media" );
84 disposePipeline();
85 break;
86 default:
87 break;
90 if( mpPipeline &&
91 gst_element_get_state( mpPipeline, NULL, NULL, 5 * GST_SECOND ) == GST_STATE_CHANGE_FAILURE )
92 disposePipeline();
95 FrameGrabber::~FrameGrabber()
97 disposePipeline();
100 FrameGrabber* FrameGrabber::create( const OUString &rURL )
102 return new FrameGrabber( rURL );
105 uno::Reference< graphic::XGraphic > SAL_CALL FrameGrabber::grabFrame( double fMediaTime )
106 throw (uno::RuntimeException)
108 uno::Reference< graphic::XGraphic > xRet;
110 if( !mpPipeline )
111 return xRet;
113 gint64 gst_position = llround( fMediaTime * 1E9 );
114 gst_element_seek_simple(
115 mpPipeline, GST_FORMAT_TIME,
116 (GstSeekFlags)(GST_SEEK_FLAG_KEY_UNIT | GST_SEEK_FLAG_FLUSH),
117 gst_position );
119 GstElement *pSink = gst_bin_get_by_name( GST_BIN( mpPipeline ), "sink" );
120 if( !pSink )
121 return xRet;
123 GstBuffer *pBuf = NULL;
124 GstCaps *pCaps = NULL;
126 // synchronously fetch the frame
127 #ifdef AVMEDIA_GST_0_10
128 g_signal_emit_by_name( pSink, "pull-preroll", &pBuf, NULL );
129 if( pBuf )
130 pCaps = GST_BUFFER_CAPS( pBuf );
131 #else
132 GstSample *pSample = NULL;
133 g_signal_emit_by_name( pSink, "pull-preroll", &pSample, NULL );
135 if( pSample )
137 pBuf = gst_sample_get_buffer( pSample );
138 pCaps = gst_sample_get_caps( pSample );
140 #endif
142 // get geometry
143 int nWidth = 0, nHeight = 0;
144 if( !pCaps )
145 g_warning( "could not get snapshot format\n" );
146 else
148 GstStructure *pStruct = gst_caps_get_structure( pCaps, 0 );
150 /* we need to get the final caps on the buffer to get the size */
151 if( !gst_structure_get_int( pStruct, "width", &nWidth ) ||
152 !gst_structure_get_int( pStruct, "height", &nHeight ) )
153 nWidth = nHeight = 0;
156 if( pBuf && nWidth > 0 && nHeight > 0 &&
157 // sanity check the size
158 #ifdef AVMEDIA_GST_0_10
159 GST_BUFFER_SIZE( pBuf ) >= static_cast<unsigned>( nWidth * nHeight * 3 )
160 #else
161 gst_buffer_get_size( pBuf ) >= ( nWidth * nHeight * 3 )
162 #endif
165 sal_uInt8 *pData = NULL;
166 #ifdef AVMEDIA_GST_0_10
167 pData = GST_BUFFER_DATA( pBuf );
168 #else
169 GstMapInfo aMapInfo;
170 gst_buffer_map( pBuf, &aMapInfo, GST_MAP_READ );
171 pData = aMapInfo.data;
172 #endif
174 int nStride = GST_ROUND_UP_4( nWidth * 3 );
175 Bitmap aBmp( Size( nWidth, nHeight ), 24 );
177 BitmapWriteAccess *pWrite = aBmp.AcquireWriteAccess();
178 if( pWrite )
180 // yet another cheesy pixel copying loop
181 for( int y = 0; y < nHeight; ++y )
183 sal_uInt8 *p = pData + y * nStride;
184 for( int x = 0; x < nWidth; ++x )
186 BitmapColor col( p[0], p[1], p[2] );
187 pWrite->SetPixel( y, x, col );
188 p += 3;
192 aBmp.ReleaseAccess( pWrite );
194 #ifndef AVMEDIA_GST_0_10
195 gst_buffer_unmap( pBuf, &aMapInfo );
196 #endif
198 xRet = Graphic( aBmp ).GetXGraphic();
201 return xRet;
204 OUString SAL_CALL FrameGrabber::getImplementationName( )
205 throw (uno::RuntimeException)
207 return OUString( AVMEDIA_GST_FRAMEGRABBER_IMPLEMENTATIONNAME );
210 sal_Bool SAL_CALL FrameGrabber::supportsService( const OUString& ServiceName )
211 throw (uno::RuntimeException)
213 return ServiceName == AVMEDIA_GST_FRAMEGRABBER_SERVICENAME;
216 uno::Sequence< OUString > SAL_CALL FrameGrabber::getSupportedServiceNames()
217 throw (uno::RuntimeException)
219 uno::Sequence< OUString > aRet(1);
220 aRet[0] = AVMEDIA_GST_FRAMEGRABBER_SERVICENAME;
222 return aRet;
225 } // namespace gstreamer
226 } // namespace avmedia
228 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */