• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /* GStreamer
2  *
3  * Copyright (C) 2014-2015 Sebastian Dröge <sebastian@centricular.com>
4  * Copyright (C) 2015 Brijesh Singh <brijesh.ksingh@gmail.com>
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Library General Public
8  * License as published by the Free Software Foundation; either
9  * version 2 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Library General Public License for more details.
15  *
16  * You should have received a copy of the GNU Library General Public
17  * License along with this library; if not, write to the
18  * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
19  * Boston, MA 02110-1301, USA.
20  */
21 
22 /**
23  * SECTION:gstplayer
24  * @title: GstPlayer
25  * @short_description: Player
26  *
27  */
28 
29 /* TODO:
30  *
31  * - Equalizer
32  * - Gapless playback
33  * - Frame stepping
34  * - Subtitle font, connection speed
35  * - Deinterlacing
36  * - Buffering control (-> progressive downloading)
37  * - Playlist/queue object
38  * - Custom video sink (e.g. embed in GL scene)
39  *
40  */
41 
42 #ifdef HAVE_CONFIG_H
43 #include "config.h"
44 #endif
45 
46 #include "gstplayer.h"
47 #include "gstplayer-signal-dispatcher-private.h"
48 #include "gstplayer-video-renderer-private.h"
49 #include "gstplayer-media-info-private.h"
50 
51 #include <gst/gst.h>
52 #include <gst/video/video.h>
53 #include <gst/video/colorbalance.h>
54 #include <gst/tag/tag.h>
55 #include <gst/pbutils/descriptions.h>
56 
57 #include <string.h>
58 
59 GST_DEBUG_CATEGORY_STATIC (gst_player_debug);
60 #define GST_CAT_DEFAULT gst_player_debug
61 
62 #define DEFAULT_URI NULL
63 #define DEFAULT_POSITION GST_CLOCK_TIME_NONE
64 #define DEFAULT_DURATION GST_CLOCK_TIME_NONE
65 #define DEFAULT_VOLUME 1.0
66 #define DEFAULT_MUTE FALSE
67 #define DEFAULT_RATE 1.0
68 #define DEFAULT_POSITION_UPDATE_INTERVAL_MS 100
69 #define DEFAULT_AUDIO_VIDEO_OFFSET 0
70 #define DEFAULT_SUBTITLE_VIDEO_OFFSET 0
71 #define VIDEO_ERROR_MSG 1
72 #ifdef OHOS_EXT_FUNC
73 // ohos.ext.func.0007
74 #define DEFAULT_RING_BUFFER_MAX_SIZE 0
75 
76 // ohos.ext.func.0012
77 #define DEFAULT_BUFFER_DURATION   0
78 #define DEFAULT_LOW_PERCENT       10
79 #define DEFAULT_HIGH_PERCENT      99
80 #define DEFAULT_BUFFER_SIZE       -1
81 #define DEFAULT_TIMEOUT           15
82 #endif
83 
84 GQuark
gst_player_error_quark(void)85 gst_player_error_quark (void)
86 {
87   return g_quark_from_static_string ("gst-player-error-quark");
88 }
89 
90 static GQuark QUARK_CONFIG;
91 
92 /* Keep ConfigQuarkId and _config_quark_strings ordered and synced */
93 typedef enum
94 {
95   CONFIG_QUARK_USER_AGENT = 0,
96   CONFIG_QUARK_POSITION_INTERVAL_UPDATE,
97   CONFIG_QUARK_ACCURATE_SEEK,
98 
99   CONFIG_QUARK_MAX
100 } ConfigQuarkId;
101 
102 static const gchar *_config_quark_strings[] = {
103   "user-agent",
104   "position-interval-update",
105   "accurate-seek",
106 };
107 
108 GQuark _config_quark_table[CONFIG_QUARK_MAX];
109 
110 #define CONFIG_QUARK(q) _config_quark_table[CONFIG_QUARK_##q]
111 
112 enum
113 {
114   PROP_0,
115   PROP_VIDEO_RENDERER,
116   PROP_SIGNAL_DISPATCHER,
117   PROP_URI,
118   PROP_SUBURI,
119   PROP_POSITION,
120   PROP_DURATION,
121   PROP_MEDIA_INFO,
122   PROP_CURRENT_AUDIO_TRACK,
123   PROP_CURRENT_VIDEO_TRACK,
124   PROP_CURRENT_SUBTITLE_TRACK,
125   PROP_VOLUME,
126   PROP_MUTE,
127   PROP_RATE,
128 #ifdef OHOS_EXT_FUNC
129   // ohos.ext.func.0004
130   PROP_SEEK_MODE,
131   // ohos.ext.func.0007
132   PROP_RING_BUFFER_MAX_SIZE,
133 #endif
134   PROP_PIPELINE,
135   PROP_VIDEO_MULTIVIEW_MODE,
136   PROP_VIDEO_MULTIVIEW_FLAGS,
137   PROP_AUDIO_VIDEO_OFFSET,
138   PROP_SUBTITLE_VIDEO_OFFSET,
139 #ifdef OHOS_EXT_FUNC
140   // ohos.ext.func.0012
141   PROP_BUFFERING_FLAGS,
142   PROP_BUFFER_DURATION,
143   PROP_BUFFER_SIZE,
144   PROP_LOW_PERCENT,
145   PROP_HIGH_PERCENT,
146   PROP_EXIT_BLOCK,
147   PROP_TIMEOUT,
148 #endif
149   PROP_LAST
150 };
151 
152 enum
153 {
154   SIGNAL_URI_LOADED,
155   SIGNAL_POSITION_UPDATED,
156   SIGNAL_DURATION_CHANGED,
157   SIGNAL_STATE_CHANGED,
158   SIGNAL_BUFFERING,
159   SIGNAL_END_OF_STREAM,
160   SIGNAL_ERROR,
161 #ifdef OHOS_EXT_FUNC
162   // ohos.ext.func.0001
163   SIGNAL_ERROR_MSG,
164   // ohos.ext.func.0006
165   SIGNAL_SOURCE_SETUP,
166   // ohos.ext.func.0012
167   SIGNAL_BUFFERING_TIME,
168   // ohos.ext.func.0013
169   SIGNAL_MQ_NUM_USE_BUFFERING,
170   // ohos.ext.func.0014
171   SIGNAL_RESOLUTION_CHANGED,
172   // ohos.ext.func.0015
173   SIGNAL_RENDER_FIRST_VIDEO_FRAME,
174   /* ohos.ext.func.0017
175    * report the signal when add new pads
176    */
177   SIGNAL_ELEMENT_SETUP,
178 #endif
179   SIGNAL_WARNING,
180   SIGNAL_VIDEO_DIMENSIONS_CHANGED,
181   SIGNAL_MEDIA_INFO_UPDATED,
182   SIGNAL_VOLUME_CHANGED,
183   SIGNAL_MUTE_CHANGED,
184   SIGNAL_SEEK_DONE,
185   SIGNAL_LAST
186 };
187 
188 enum
189 {
190   GST_PLAY_FLAG_VIDEO = (1 << 0),
191   GST_PLAY_FLAG_AUDIO = (1 << 1),
192   GST_PLAY_FLAG_SUBTITLE = (1 << 2),
193   GST_PLAY_FLAG_VIS = (1 << 3)
194 };
195 
196 struct _GstPlayer
197 {
198   GstObject parent;
199 
200   GstPlayerVideoRenderer *video_renderer;
201   GstPlayerSignalDispatcher *signal_dispatcher;
202 
203   gchar *uri;
204   gchar *redirect_uri;
205   gchar *suburi;
206 
207   GThread *thread;
208   GMutex lock;
209   GCond cond;
210   GMainContext *context;
211   GMainLoop *loop;
212 
213   GstElement *playbin;
214   GstBus *bus;
215   GstState target_state, current_state;
216   gboolean is_live, is_eos;
217   GSource *tick_source, *ready_timeout_source;
218   GstClockTime cached_duration;
219 
220   gdouble rate;
221 #ifdef OHOS_EXT_FUNC
222   // ohos.ext.func.0004
223   gint seek_mode;
224   // ohos.ext.func.0012
225   guint timeout;
226 #endif
227 
228 #ifdef OHOS_OPT_COMPAT
229   // ohos.opt.compat.0010
230   gboolean isStateChange;
231 #endif
232 
233   GstPlayerState app_state;
234   gint buffering;
235 
236   GstTagList *global_tags;
237   GstPlayerMediaInfo *media_info;
238 
239   GstElement *current_vis_element;
240 
241   GstStructure *config;
242 
243   /* Protected by lock */
244   gboolean seek_pending;        /* Only set from main context */
245   GstClockTime last_seek_time;  /* Only set from main context */
246   GSource *seek_source;
247   GstClockTime seek_position;
248   /* If TRUE, all signals are inhibited except the
249    * state-changed:GST_PLAYER_STATE_STOPPED/PAUSED. This ensures that no signal
250    * is emitted after gst_player_stop/pause() has been called by the user. */
251   gboolean inhibit_sigs;
252 
253   /* For playbin3 */
254   gboolean use_playbin3;
255   GstStreamCollection *collection;
256   gchar *video_sid;
257   gchar *audio_sid;
258   gchar *subtitle_sid;
259   gulong stream_notify_id;
260 };
261 
262 struct _GstPlayerClass
263 {
264   GstObjectClass parent_class;
265 };
266 
267 #define parent_class gst_player_parent_class
268 G_DEFINE_TYPE (GstPlayer, gst_player, GST_TYPE_OBJECT);
269 
270 static guint signals[SIGNAL_LAST] = { 0, };
271 static GParamSpec *param_specs[PROP_LAST] = { NULL, };
272 
273 static void gst_player_dispose (GObject * object);
274 static void gst_player_finalize (GObject * object);
275 static void gst_player_set_property (GObject * object, guint prop_id,
276     const GValue * value, GParamSpec * pspec);
277 static void gst_player_get_property (GObject * object, guint prop_id,
278     GValue * value, GParamSpec * pspec);
279 static void gst_player_constructed (GObject * object);
280 
281 static gpointer gst_player_main (gpointer data);
282 
283 static void gst_player_seek_internal_locked (GstPlayer * self);
284 static void gst_player_stop_internal (GstPlayer * self, gboolean transient);
285 static gboolean gst_player_pause_internal (gpointer user_data);
286 static gboolean gst_player_play_internal (gpointer user_data);
287 static gboolean gst_player_seek_internal (gpointer user_data);
288 static void gst_player_set_rate_internal (GstPlayer * self);
289 static void change_state (GstPlayer * self, GstPlayerState state);
290 
291 static GstPlayerMediaInfo *gst_player_media_info_create (GstPlayer * self);
292 
293 static void gst_player_streams_info_create (GstPlayer * self,
294     GstPlayerMediaInfo * media_info, const gchar * prop, GType type);
295 static void gst_player_stream_info_update (GstPlayer * self,
296     GstPlayerStreamInfo * s);
297 static void gst_player_stream_info_update_tags_and_caps (GstPlayer * self,
298     GstPlayerStreamInfo * s);
299 static GstPlayerStreamInfo *gst_player_stream_info_find (GstPlayerMediaInfo *
300     media_info, GType type, gint stream_index);
301 static GstPlayerStreamInfo *gst_player_stream_info_get_current (GstPlayer *
302     self, const gchar * prop, GType type);
303 
304 static void gst_player_video_info_update (GstPlayer * self,
305     GstPlayerStreamInfo * stream_info);
306 static void gst_player_audio_info_update (GstPlayer * self,
307     GstPlayerStreamInfo * stream_info);
308 static void gst_player_subtitle_info_update (GstPlayer * self,
309     GstPlayerStreamInfo * stream_info);
310 
311 /* For playbin3 */
312 static void gst_player_streams_info_create_from_collection (GstPlayer * self,
313     GstPlayerMediaInfo * media_info, GstStreamCollection * collection);
314 static void gst_player_stream_info_update_from_stream (GstPlayer * self,
315     GstPlayerStreamInfo * s, GstStream * stream);
316 static GstPlayerStreamInfo *gst_player_stream_info_find_from_stream_id
317     (GstPlayerMediaInfo * media_info, const gchar * stream_id);
318 static GstPlayerStreamInfo *gst_player_stream_info_get_current_from_stream_id
319     (GstPlayer * self, const gchar * stream_id, GType type);
320 static void stream_notify_cb (GstStreamCollection * collection,
321     GstStream * stream, GParamSpec * pspec, GstPlayer * self);
322 
323 static void emit_media_info_updated_signal (GstPlayer * self);
324 
325 static void *get_title (GstTagList * tags);
326 static void *get_container_format (GstTagList * tags);
327 static void *get_from_tags (GstPlayer * self, GstPlayerMediaInfo * media_info,
328     void *(*func) (GstTagList *));
329 static void *get_cover_sample (GstTagList * tags);
330 
331 static void remove_seek_source (GstPlayer * self);
332 
333 static void
gst_player_init(GstPlayer * self)334 gst_player_init (GstPlayer * self)
335 {
336   GST_TRACE_OBJECT (self, "Initializing");
337 
338   self = gst_player_get_instance_private (self);
339 
340   g_mutex_init (&self->lock);
341   g_cond_init (&self->cond);
342 
343   self->context = g_main_context_new ();
344   self->loop = g_main_loop_new (self->context, FALSE);
345 
346   /* *INDENT-OFF* */
347   self->config = gst_structure_new_id (QUARK_CONFIG,
348       CONFIG_QUARK (POSITION_INTERVAL_UPDATE), G_TYPE_UINT, DEFAULT_POSITION_UPDATE_INTERVAL_MS,
349       CONFIG_QUARK (ACCURATE_SEEK), G_TYPE_BOOLEAN, FALSE,
350       NULL);
351   /* *INDENT-ON* */
352 
353   self->seek_pending = FALSE;
354   self->seek_position = GST_CLOCK_TIME_NONE;
355   self->last_seek_time = GST_CLOCK_TIME_NONE;
356   self->inhibit_sigs = FALSE;
357 
358 #ifdef OHOS_OPT_COMPAT
359   // ohos.opt.compat.0010
360   self->isStateChange = FALSE;
361 #endif
362 
363   GST_TRACE_OBJECT (self, "Initialized");
364 }
365 
366 static void
config_quark_initialize(void)367 config_quark_initialize (void)
368 {
369   gint i;
370 
371   QUARK_CONFIG = g_quark_from_static_string ("player-config");
372 
373   if (G_N_ELEMENTS (_config_quark_strings) != CONFIG_QUARK_MAX)
374     g_warning ("the quark table is not consistent! %d != %d",
375         (int) G_N_ELEMENTS (_config_quark_strings), CONFIG_QUARK_MAX);
376 
377   for (i = 0; i < CONFIG_QUARK_MAX; i++) {
378     _config_quark_table[i] =
379         g_quark_from_static_string (_config_quark_strings[i]);
380   }
381 }
382 
383 static void
gst_player_class_init(GstPlayerClass * klass)384 gst_player_class_init (GstPlayerClass * klass)
385 {
386   GObjectClass *gobject_class = (GObjectClass *) klass;
387 
388   gobject_class->set_property = gst_player_set_property;
389   gobject_class->get_property = gst_player_get_property;
390   gobject_class->dispose = gst_player_dispose;
391   gobject_class->finalize = gst_player_finalize;
392   gobject_class->constructed = gst_player_constructed;
393 
394   param_specs[PROP_VIDEO_RENDERER] =
395       g_param_spec_object ("video-renderer",
396       "Video Renderer", "Video renderer to use for rendering videos",
397       GST_TYPE_PLAYER_VIDEO_RENDERER,
398       G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS);
399 
400   param_specs[PROP_SIGNAL_DISPATCHER] =
401       g_param_spec_object ("signal-dispatcher",
402       "Signal Dispatcher", "Dispatcher for the signals to e.g. event loops",
403       GST_TYPE_PLAYER_SIGNAL_DISPATCHER,
404       G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS);
405 
406   param_specs[PROP_URI] = g_param_spec_string ("uri", "URI", "Current URI",
407       DEFAULT_URI, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
408 
409   param_specs[PROP_SUBURI] = g_param_spec_string ("suburi", "Subtitle URI",
410       "Current Subtitle URI", NULL, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
411 
412   param_specs[PROP_POSITION] =
413       g_param_spec_uint64 ("position", "Position", "Current Position",
414       0, G_MAXUINT64, DEFAULT_POSITION,
415       G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);
416 
417   param_specs[PROP_MEDIA_INFO] =
418       g_param_spec_object ("media-info", "Media Info",
419       "Current media information", GST_TYPE_PLAYER_MEDIA_INFO,
420       G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);
421 
422   param_specs[PROP_CURRENT_AUDIO_TRACK] =
423       g_param_spec_object ("current-audio-track", "Current Audio Track",
424       "Current audio track information", GST_TYPE_PLAYER_AUDIO_INFO,
425       G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);
426 
427   param_specs[PROP_CURRENT_VIDEO_TRACK] =
428       g_param_spec_object ("current-video-track", "Current Video Track",
429       "Current video track information", GST_TYPE_PLAYER_VIDEO_INFO,
430       G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);
431 
432   param_specs[PROP_CURRENT_SUBTITLE_TRACK] =
433       g_param_spec_object ("current-subtitle-track", "Current Subtitle Track",
434       "Current audio subtitle information", GST_TYPE_PLAYER_SUBTITLE_INFO,
435       G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);
436 
437   param_specs[PROP_DURATION] =
438       g_param_spec_uint64 ("duration", "Duration", "Duration",
439       0, G_MAXUINT64, DEFAULT_DURATION,
440       G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);
441 
442   param_specs[PROP_VOLUME] =
443       g_param_spec_double ("volume", "Volume", "Volume",
444       0, 10.0, DEFAULT_VOLUME, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
445 
446   param_specs[PROP_MUTE] =
447       g_param_spec_boolean ("mute", "Mute", "Mute",
448       DEFAULT_MUTE, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
449 
450   param_specs[PROP_PIPELINE] =
451       g_param_spec_object ("pipeline", "Pipeline",
452       "GStreamer pipeline that is used",
453       GST_TYPE_ELEMENT, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);
454 #ifdef OHOS_EXT_FUNC
455   // ohos.ext.func.0004
456   param_specs[PROP_SEEK_MODE] =
457       g_param_spec_int ("seek-mode", "seek-mode", "Playback seek-mode",
458       0, G_MAXINT, 0,
459       G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
460 
461   // ohos.ext.func.0007
462   param_specs[PROP_RING_BUFFER_MAX_SIZE] =
463       g_param_spec_uint64 ("ring-buffer-max-size",
464       "Max. ring buffer size (bytes)",
465       "Max. amount of data in the ring buffer (bytes, 0 = ring buffer disabled)",
466       0, G_MAXUINT, DEFAULT_RING_BUFFER_MAX_SIZE,
467       G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
468 
469   // ohos.ext.func.0012
470   param_specs[PROP_BUFFERING_FLAGS] =
471       g_param_spec_boolean ("buffering-flags", "Buffering Flags", "Flags to control behaviour",
472       FALSE, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
473 
474   param_specs[PROP_BUFFER_DURATION] =
475       g_param_spec_uint64 ("buffer-duration", "Buffer duration (ns)",
476         "Buffer duration when buffering network streams",
477         0, G_MAXUINT64 / 1000, DEFAULT_BUFFER_DURATION,
478         G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
479 
480   param_specs[PROP_BUFFER_SIZE] =
481       g_param_spec_int ("buffer-size", "Buffer size (bytes)",
482         "Buffer size when buffering network streams",
483         -1, G_MAXINT, DEFAULT_BUFFER_SIZE,
484         G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
485 
486   param_specs[PROP_LOW_PERCENT] =
487       g_param_spec_int ("low-percent", "Low percent",
488           "Low threshold for buffering to start", 0, 100,
489           DEFAULT_LOW_PERCENT, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
490 
491   param_specs[PROP_HIGH_PERCENT] =
492       g_param_spec_int ("high-percent", "High percent",
493           "High threshold for buffering to finish", 0, 100,
494           DEFAULT_HIGH_PERCENT, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
495 
496   param_specs[PROP_EXIT_BLOCK] =
497       g_param_spec_int ("exit-block", "EXIT BLOCK",
498           "souphttpsrc exit block", 0, (gint) (G_MAXINT32), 0,
499           G_PARAM_WRITABLE | G_PARAM_STATIC_STRINGS);
500 
501   param_specs[PROP_TIMEOUT] =
502       g_param_spec_uint ("timeout", "TIME OUT",
503           "souphttpsrc time out", 0, 3600, DEFAULT_TIMEOUT,
504           G_PARAM_WRITABLE | G_PARAM_STATIC_STRINGS);
505 #endif
506   param_specs[PROP_RATE] =
507       g_param_spec_double ("rate", "rate", "Playback rate",
508       -64.0, 64.0, DEFAULT_RATE, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
509 
510   param_specs[PROP_VIDEO_MULTIVIEW_MODE] =
511       g_param_spec_enum ("video-multiview-mode",
512       "Multiview Mode Override",
513       "Re-interpret a video stream as one of several frame-packed stereoscopic modes.",
514       GST_TYPE_VIDEO_MULTIVIEW_FRAME_PACKING,
515       GST_VIDEO_MULTIVIEW_FRAME_PACKING_NONE,
516       G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
517 
518   param_specs[PROP_VIDEO_MULTIVIEW_FLAGS] =
519       g_param_spec_flags ("video-multiview-flags",
520       "Multiview Flags Override",
521       "Override details of the multiview frame layout",
522       GST_TYPE_VIDEO_MULTIVIEW_FLAGS, GST_VIDEO_MULTIVIEW_FLAGS_NONE,
523       G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
524 
525   param_specs[PROP_AUDIO_VIDEO_OFFSET] =
526       g_param_spec_int64 ("audio-video-offset", "Audio Video Offset",
527       "The synchronisation offset between audio and video in nanoseconds",
528       G_MININT64, G_MAXINT64, 0, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
529 
530   param_specs[PROP_SUBTITLE_VIDEO_OFFSET] =
531       g_param_spec_int64 ("subtitle-video-offset", "Text Video Offset",
532       "The synchronisation offset between text and video in nanoseconds",
533       G_MININT64, G_MAXINT64, 0, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
534 
535   g_object_class_install_properties (gobject_class, PROP_LAST, param_specs);
536 
537   signals[SIGNAL_URI_LOADED] =
538       g_signal_new ("uri-loaded", G_TYPE_FROM_CLASS (klass),
539       G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, 0, NULL,
540       NULL, NULL, G_TYPE_NONE, 1, G_TYPE_STRING);
541 
542   signals[SIGNAL_POSITION_UPDATED] =
543       g_signal_new ("position-updated", G_TYPE_FROM_CLASS (klass),
544       G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, 0, NULL,
545       NULL, NULL, G_TYPE_NONE, 1, GST_TYPE_CLOCK_TIME);
546 
547   signals[SIGNAL_DURATION_CHANGED] =
548       g_signal_new ("duration-changed", G_TYPE_FROM_CLASS (klass),
549       G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, 0, NULL,
550       NULL, NULL, G_TYPE_NONE, 1, GST_TYPE_CLOCK_TIME);
551 
552   signals[SIGNAL_STATE_CHANGED] =
553       g_signal_new ("state-changed", G_TYPE_FROM_CLASS (klass),
554       G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, 0, NULL,
555       NULL, NULL, G_TYPE_NONE, 1, GST_TYPE_PLAYER_STATE);
556 
557   signals[SIGNAL_BUFFERING] =
558       g_signal_new ("buffering", G_TYPE_FROM_CLASS (klass),
559       G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, 0, NULL,
560       NULL, NULL, G_TYPE_NONE, 1, G_TYPE_INT);
561 
562   signals[SIGNAL_END_OF_STREAM] =
563       g_signal_new ("end-of-stream", G_TYPE_FROM_CLASS (klass),
564       G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, 0, NULL,
565       NULL, NULL, G_TYPE_NONE, 0, G_TYPE_INVALID);
566 
567   signals[SIGNAL_ERROR] =
568       g_signal_new ("error", G_TYPE_FROM_CLASS (klass),
569       G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, 0, NULL,
570       NULL, NULL, G_TYPE_NONE, 1, G_TYPE_ERROR);
571 // ohos.ext.func.0001: change error callback which engine can get all msg
572 #ifdef OHOS_EXT_FUNC
573   signals[SIGNAL_ERROR_MSG] =
574       g_signal_new ("error-msg", G_TYPE_FROM_CLASS (klass),
575       G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, 0, NULL,
576       NULL, NULL, G_TYPE_NONE, 1, GST_TYPE_MESSAGE);
577 #endif
578   signals[SIGNAL_VIDEO_DIMENSIONS_CHANGED] =
579       g_signal_new ("video-dimensions-changed", G_TYPE_FROM_CLASS (klass),
580       G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, 0, NULL,
581       NULL, NULL, G_TYPE_NONE, 2, G_TYPE_INT, G_TYPE_INT);
582 
583   signals[SIGNAL_MEDIA_INFO_UPDATED] =
584       g_signal_new ("media-info-updated", G_TYPE_FROM_CLASS (klass),
585       G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, 0, NULL,
586       NULL, NULL, G_TYPE_NONE, 1, GST_TYPE_PLAYER_MEDIA_INFO);
587 
588   signals[SIGNAL_VOLUME_CHANGED] =
589       g_signal_new ("volume-changed", G_TYPE_FROM_CLASS (klass),
590       G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, 0, NULL,
591       NULL, NULL, G_TYPE_NONE, 0, G_TYPE_INVALID);
592 
593   signals[SIGNAL_MUTE_CHANGED] =
594       g_signal_new ("mute-changed", G_TYPE_FROM_CLASS (klass),
595       G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, 0, NULL,
596       NULL, NULL, G_TYPE_NONE, 0, G_TYPE_INVALID);
597 
598   signals[SIGNAL_WARNING] =
599       g_signal_new ("warning", G_TYPE_FROM_CLASS (klass),
600       G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, 0, NULL,
601       NULL, NULL, G_TYPE_NONE, 1, G_TYPE_ERROR);
602 
603   signals[SIGNAL_SEEK_DONE] =
604       g_signal_new ("seek-done", G_TYPE_FROM_CLASS (klass),
605       G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, 0, NULL,
606       NULL, NULL, G_TYPE_NONE, 1, GST_TYPE_CLOCK_TIME);
607 #ifdef OHOS_EXT_FUNC
608   // ohos.ext.func.0006
609   signals[SIGNAL_SOURCE_SETUP] =
610       g_signal_new ("source-setup", G_TYPE_FROM_CLASS (klass),
611       G_SIGNAL_RUN_LAST, 0, NULL, NULL,
612       g_cclosure_marshal_generic, G_TYPE_NONE, 1, GST_TYPE_ELEMENT);
613 
614   // ohos.ext.func.0012
615   signals[SIGNAL_BUFFERING_TIME] =
616       g_signal_new ("buffering-time", G_TYPE_FROM_CLASS (klass),
617       G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, 0, NULL,
618       NULL, NULL, G_TYPE_NONE, 2, GST_TYPE_CLOCK_TIME, G_TYPE_UINT);
619 
620   // ohos.ext.func.0013
621   signals[SIGNAL_MQ_NUM_USE_BUFFERING] =
622       g_signal_new ("mq-num-use-buffering", G_TYPE_FROM_CLASS (klass),
623       G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, 0, NULL,
624       NULL, NULL, G_TYPE_NONE, 1, G_TYPE_UINT);
625 
626   // ohos.ext.func.0014
627   signals[SIGNAL_RESOLUTION_CHANGED] =
628       g_signal_new ("resolution-changed", G_TYPE_FROM_CLASS (klass),
629       G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, 0, NULL,
630       NULL, NULL, G_TYPE_NONE, 2, G_TYPE_INT, G_TYPE_INT);
631 
632   /* ohos.ext.func.0017
633    * report the signal when add new pads
634    */
635   signals[SIGNAL_ELEMENT_SETUP] =
636       g_signal_new ("element-setup", G_TYPE_FROM_CLASS (klass),
637       G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, 0, NULL,
638       NULL, NULL, G_TYPE_NONE, 1, GST_TYPE_ELEMENT);
639 #endif
640   config_quark_initialize ();
641 }
642 
643 static void
gst_player_dispose(GObject * object)644 gst_player_dispose (GObject * object)
645 {
646   GstPlayer *self = GST_PLAYER (object);
647 
648   GST_TRACE_OBJECT (self, "Stopping main thread");
649 
650   if (self->loop) {
651     g_main_loop_quit (self->loop);
652 
653     if (self->thread != g_thread_self ())
654       g_thread_join (self->thread);
655     else
656       g_thread_unref (self->thread);
657     self->thread = NULL;
658 
659     g_main_loop_unref (self->loop);
660     self->loop = NULL;
661 
662     g_main_context_unref (self->context);
663     self->context = NULL;
664   }
665 
666   G_OBJECT_CLASS (parent_class)->dispose (object);
667 }
668 
669 static void
gst_player_finalize(GObject * object)670 gst_player_finalize (GObject * object)
671 {
672   GstPlayer *self = GST_PLAYER (object);
673 
674   GST_TRACE_OBJECT (self, "Finalizing");
675 
676   g_free (self->uri);
677   g_free (self->redirect_uri);
678   g_free (self->suburi);
679   g_free (self->video_sid);
680   g_free (self->audio_sid);
681   g_free (self->subtitle_sid);
682   if (self->global_tags)
683     gst_tag_list_unref (self->global_tags);
684   if (self->video_renderer)
685     g_object_unref (self->video_renderer);
686   if (self->signal_dispatcher)
687     g_object_unref (self->signal_dispatcher);
688   if (self->current_vis_element)
689     gst_object_unref (self->current_vis_element);
690   if (self->config)
691     gst_structure_free (self->config);
692   if (self->collection)
693     gst_object_unref (self->collection);
694   g_mutex_clear (&self->lock);
695   g_cond_clear (&self->cond);
696 
697   G_OBJECT_CLASS (parent_class)->finalize (object);
698 }
699 
700 static void
gst_player_constructed(GObject * object)701 gst_player_constructed (GObject * object)
702 {
703   GstPlayer *self = GST_PLAYER (object);
704 
705   GST_TRACE_OBJECT (self, "Constructed");
706 
707   g_mutex_lock (&self->lock);
708   self->thread = g_thread_new ("GstPlayer", gst_player_main, self);
709   while (!self->loop || !g_main_loop_is_running (self->loop))
710     g_cond_wait (&self->cond, &self->lock);
711   g_mutex_unlock (&self->lock);
712 
713   G_OBJECT_CLASS (parent_class)->constructed (object);
714 }
715 
716 typedef struct
717 {
718   GstPlayer *player;
719   gchar *uri;
720 } UriLoadedSignalData;
721 
722 static void
uri_loaded_dispatch(gpointer user_data)723 uri_loaded_dispatch (gpointer user_data)
724 {
725   UriLoadedSignalData *data = user_data;
726 
727   g_signal_emit (data->player, signals[SIGNAL_URI_LOADED], 0, data->uri);
728 }
729 
730 static void
uri_loaded_signal_data_free(UriLoadedSignalData * data)731 uri_loaded_signal_data_free (UriLoadedSignalData * data)
732 {
733   g_object_unref (data->player);
734   g_free (data->uri);
735   g_free (data);
736 }
737 
738 static gboolean
gst_player_set_uri_internal(gpointer user_data)739 gst_player_set_uri_internal (gpointer user_data)
740 {
741   GstPlayer *self = user_data;
742 
743   gst_player_stop_internal (self, FALSE);
744 
745   g_mutex_lock (&self->lock);
746 
747   GST_DEBUG_OBJECT (self, "Changing URI to '%s'", GST_STR_NULL (self->uri));
748 
749   g_object_set (self->playbin, "uri", self->uri, NULL);
750 
751   if (g_signal_handler_find (self, G_SIGNAL_MATCH_ID,
752           signals[SIGNAL_URI_LOADED], 0, NULL, NULL, NULL) != 0) {
753     UriLoadedSignalData *data = g_new (UriLoadedSignalData, 1);
754 
755     data->player = g_object_ref (self);
756     data->uri = g_strdup (self->uri);
757     gst_player_signal_dispatcher_dispatch (self->signal_dispatcher, self,
758         uri_loaded_dispatch, data,
759         (GDestroyNotify) uri_loaded_signal_data_free);
760   }
761 
762   g_object_set (self->playbin, "suburi", NULL, NULL);
763 
764   g_mutex_unlock (&self->lock);
765 
766   return G_SOURCE_REMOVE;
767 }
768 
769 static gboolean
gst_player_set_suburi_internal(gpointer user_data)770 gst_player_set_suburi_internal (gpointer user_data)
771 {
772   GstPlayer *self = user_data;
773   GstClockTime position;
774   GstState target_state;
775 
776   /* save the state and position */
777   target_state = self->target_state;
778   position = gst_player_get_position (self);
779 
780   gst_player_stop_internal (self, TRUE);
781   g_mutex_lock (&self->lock);
782 
783   GST_DEBUG_OBJECT (self, "Changing SUBURI to '%s'",
784       GST_STR_NULL (self->suburi));
785 
786   g_object_set (self->playbin, "suburi", self->suburi, NULL);
787 
788   g_mutex_unlock (&self->lock);
789 
790   /* restore state and position */
791   if (position != GST_CLOCK_TIME_NONE)
792     gst_player_seek (self, position);
793   if (target_state == GST_STATE_PAUSED)
794     gst_player_pause_internal (self);
795   else if (target_state == GST_STATE_PLAYING)
796     gst_player_play_internal (self);
797 
798   return G_SOURCE_REMOVE;
799 }
800 
801 static void
gst_player_set_rate_internal(GstPlayer * self)802 gst_player_set_rate_internal (GstPlayer * self)
803 {
804   self->seek_position = gst_player_get_position (self);
805 
806   /* If there is no seek being dispatch to the main context currently do that,
807    * otherwise we just updated the rate so that it will be taken by
808    * the seek handler from the main context instead of the old one.
809    */
810   if (!self->seek_source) {
811     /* If no seek is pending then create new seek source */
812     if (!self->seek_pending) {
813       self->seek_source = g_idle_source_new ();
814       g_source_set_callback (self->seek_source,
815           (GSourceFunc) gst_player_seek_internal, self, NULL);
816       g_source_attach (self->seek_source, self->context);
817     }
818   }
819 }
820 
821 static void
gst_player_set_property(GObject * object,guint prop_id,const GValue * value,GParamSpec * pspec)822 gst_player_set_property (GObject * object, guint prop_id,
823     const GValue * value, GParamSpec * pspec)
824 {
825   GstPlayer *self = GST_PLAYER (object);
826 
827   switch (prop_id) {
828     case PROP_VIDEO_RENDERER:
829       self->video_renderer = g_value_dup_object (value);
830       break;
831     case PROP_SIGNAL_DISPATCHER:
832       self->signal_dispatcher = g_value_dup_object (value);
833       break;
834     case PROP_URI:{
835       g_mutex_lock (&self->lock);
836       g_free (self->uri);
837       g_free (self->redirect_uri);
838       self->redirect_uri = NULL;
839 
840       g_free (self->suburi);
841       self->suburi = NULL;
842 
843       self->uri = g_value_dup_string (value);
844       GST_DEBUG_OBJECT (self, "Set uri=%s", self->uri);
845       g_mutex_unlock (&self->lock);
846 
847       g_main_context_invoke_full (self->context, G_PRIORITY_DEFAULT,
848           gst_player_set_uri_internal, self, NULL);
849       break;
850     }
851     case PROP_SUBURI:{
852       g_mutex_lock (&self->lock);
853       g_free (self->suburi);
854 
855       self->suburi = g_value_dup_string (value);
856       GST_DEBUG_OBJECT (self, "Set suburi=%s", self->suburi);
857       g_mutex_unlock (&self->lock);
858 
859       g_main_context_invoke_full (self->context, G_PRIORITY_DEFAULT,
860           gst_player_set_suburi_internal, self, NULL);
861       break;
862     }
863     case PROP_VOLUME:
864       GST_DEBUG_OBJECT (self, "Set volume=%lf", g_value_get_double (value));
865       g_object_set_property (G_OBJECT (self->playbin), "volume", value);
866       break;
867     case PROP_RATE:
868       g_mutex_lock (&self->lock);
869       self->rate = g_value_get_double (value);
870       GST_DEBUG_OBJECT (self, "Set rate=%lf", g_value_get_double (value));
871       gst_player_set_rate_internal (self);
872       g_mutex_unlock (&self->lock);
873       break;
874 #ifdef OHOS_EXT_FUNC
875     /* ohos.ext.func.0004: The playback engine requires different seek modes.
876      * The capability of setting the seek mode needs to be added to gstplayer.
877      */
878     case PROP_SEEK_MODE:
879       g_mutex_lock (&self->lock);
880       self->seek_mode = g_value_get_int (value);
881       GST_DEBUG_OBJECT (self, "Set seek_mode=%d", g_value_get_int (value));
882       g_mutex_unlock (&self->lock);
883       break;
884     // ohos.ext.func.0007
885     case PROP_RING_BUFFER_MAX_SIZE:
886       GST_INFO_OBJECT (self, "set ring-buffer-max-size=%" G_GUINT64_FORMAT, g_value_get_uint64 (value));
887       g_object_set_property (G_OBJECT (self->playbin), "ring-buffer-max-size", value);
888       break;
889 
890     // ohos.ext.func.0012 : Control the behaviour of playbin
891     case PROP_BUFFERING_FLAGS:
892       GST_INFO_OBJECT (self, "buffering flag %d", g_value_get_boolean (value));
893       g_object_set_property (G_OBJECT (self->playbin), "buffering-flags", value);
894       break;
895     case PROP_BUFFER_DURATION:
896       GST_INFO_OBJECT (self, "set duration %" G_GUINT64_FORMAT, g_value_get_uint64 (value));
897       g_object_set_property (G_OBJECT (self->playbin), "buffer-duration", value);
898       break;
899     case PROP_BUFFER_SIZE:
900       GST_INFO_OBJECT (self, "set buffer size %d", g_value_get_int (value));
901       g_object_set_property (G_OBJECT (self->playbin), "buffer-size", value);
902       break;
903     case PROP_LOW_PERCENT:
904       GST_INFO_OBJECT (self, "set low-percent %d", g_value_get_int (value));
905       g_object_set_property (G_OBJECT (self->playbin), "low-percent", value);
906       break;
907     case PROP_HIGH_PERCENT:
908       GST_INFO_OBJECT (self, "set high-percent %d", g_value_get_int (value));
909       g_object_set_property (G_OBJECT (self->playbin), "high-percent", value);
910       break;
911     case PROP_EXIT_BLOCK:
912       g_object_set_property (G_OBJECT (self->playbin), "exit-block", value);
913       break;
914     case PROP_TIMEOUT:
915       g_mutex_lock (&self->lock);
916       self->timeout = g_value_get_uint (value);
917       GST_DEBUG_OBJECT (self, "Set timeout %u", g_value_get_uint (value));
918       g_mutex_unlock (&self->lock);
919       break;
920 #endif
921     case PROP_MUTE:
922       GST_DEBUG_OBJECT (self, "Set mute=%d", g_value_get_boolean (value));
923       g_object_set_property (G_OBJECT (self->playbin), "mute", value);
924       break;
925     case PROP_VIDEO_MULTIVIEW_MODE:
926       GST_DEBUG_OBJECT (self, "Set multiview mode=%u",
927           g_value_get_enum (value));
928       g_object_set_property (G_OBJECT (self->playbin), "video-multiview-mode",
929           value);
930       break;
931     case PROP_VIDEO_MULTIVIEW_FLAGS:
932       GST_DEBUG_OBJECT (self, "Set multiview flags=%x",
933           g_value_get_flags (value));
934       g_object_set_property (G_OBJECT (self->playbin), "video-multiview-flags",
935           value);
936       break;
937     case PROP_AUDIO_VIDEO_OFFSET:
938       g_object_set_property (G_OBJECT (self->playbin), "av-offset", value);
939       break;
940     case PROP_SUBTITLE_VIDEO_OFFSET:
941       g_object_set_property (G_OBJECT (self->playbin), "text-offset", value);
942       break;
943     default:
944       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
945       break;
946   }
947 }
948 
949 static void
gst_player_get_property(GObject * object,guint prop_id,GValue * value,GParamSpec * pspec)950 gst_player_get_property (GObject * object, guint prop_id,
951     GValue * value, GParamSpec * pspec)
952 {
953   GstPlayer *self = GST_PLAYER (object);
954 
955   switch (prop_id) {
956     case PROP_URI:
957       g_mutex_lock (&self->lock);
958       g_value_set_string (value, self->uri);
959       g_mutex_unlock (&self->lock);
960       break;
961     case PROP_SUBURI:
962       g_mutex_lock (&self->lock);
963       g_value_set_string (value, self->suburi);
964       g_mutex_unlock (&self->lock);
965       GST_DEBUG_OBJECT (self, "Returning suburi=%s",
966           g_value_get_string (value));
967       break;
968     case PROP_POSITION:{
969       gint64 position = GST_CLOCK_TIME_NONE;
970 
971       gst_element_query_position (self->playbin, GST_FORMAT_TIME, &position);
972       g_value_set_uint64 (value, position);
973       GST_TRACE_OBJECT (self, "Returning position=%" GST_TIME_FORMAT,
974           GST_TIME_ARGS (g_value_get_uint64 (value)));
975       break;
976     }
977     case PROP_DURATION:{
978       g_value_set_uint64 (value, self->cached_duration);
979       GST_TRACE_OBJECT (self, "Returning duration=%" GST_TIME_FORMAT,
980           GST_TIME_ARGS (g_value_get_uint64 (value)));
981       break;
982     }
983     case PROP_MEDIA_INFO:{
984       GstPlayerMediaInfo *media_info = gst_player_get_media_info (self);
985       g_value_take_object (value, media_info);
986       break;
987     }
988     case PROP_CURRENT_AUDIO_TRACK:{
989       GstPlayerAudioInfo *audio_info =
990           gst_player_get_current_audio_track (self);
991       g_value_take_object (value, audio_info);
992       break;
993     }
994     case PROP_CURRENT_VIDEO_TRACK:{
995       GstPlayerVideoInfo *video_info =
996           gst_player_get_current_video_track (self);
997       g_value_take_object (value, video_info);
998       break;
999     }
1000     case PROP_CURRENT_SUBTITLE_TRACK:{
1001       GstPlayerSubtitleInfo *subtitle_info =
1002           gst_player_get_current_subtitle_track (self);
1003       g_value_take_object (value, subtitle_info);
1004       break;
1005     }
1006     case PROP_VOLUME:
1007       g_object_get_property (G_OBJECT (self->playbin), "volume", value);
1008       GST_TRACE_OBJECT (self, "Returning volume=%lf",
1009           g_value_get_double (value));
1010       break;
1011     case PROP_RATE:
1012       g_mutex_lock (&self->lock);
1013       g_value_set_double (value, self->rate);
1014       g_mutex_unlock (&self->lock);
1015       break;
1016     case PROP_MUTE:
1017       g_object_get_property (G_OBJECT (self->playbin), "mute", value);
1018       GST_TRACE_OBJECT (self, "Returning mute=%d", g_value_get_boolean (value));
1019       break;
1020     case PROP_PIPELINE:
1021       g_value_set_object (value, self->playbin);
1022       break;
1023     case PROP_VIDEO_MULTIVIEW_MODE:{
1024       g_object_get_property (G_OBJECT (self->playbin), "video-multiview-mode",
1025           value);
1026       GST_TRACE_OBJECT (self, "Return multiview mode=%d",
1027           g_value_get_enum (value));
1028       break;
1029     }
1030     case PROP_VIDEO_MULTIVIEW_FLAGS:{
1031       g_object_get_property (G_OBJECT (self->playbin), "video-multiview-flags",
1032           value);
1033       GST_TRACE_OBJECT (self, "Return multiview flags=%x",
1034           g_value_get_flags (value));
1035       break;
1036     }
1037     case PROP_AUDIO_VIDEO_OFFSET:
1038       g_object_get_property (G_OBJECT (self->playbin), "av-offset", value);
1039       break;
1040     case PROP_SUBTITLE_VIDEO_OFFSET:
1041       g_object_get_property (G_OBJECT (self->playbin), "text-offset", value);
1042       break;
1043     default:
1044       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
1045       break;
1046   }
1047 }
1048 
1049 static gboolean
main_loop_running_cb(gpointer user_data)1050 main_loop_running_cb (gpointer user_data)
1051 {
1052   GstPlayer *self = GST_PLAYER (user_data);
1053 
1054   GST_TRACE_OBJECT (self, "Main loop running now");
1055 
1056   g_mutex_lock (&self->lock);
1057   g_cond_signal (&self->cond);
1058   g_mutex_unlock (&self->lock);
1059 
1060   return G_SOURCE_REMOVE;
1061 }
1062 
1063 typedef struct
1064 {
1065   GstPlayer *player;
1066   GstPlayerState state;
1067 } StateChangedSignalData;
1068 
1069 static void
state_changed_dispatch(gpointer user_data)1070 state_changed_dispatch (gpointer user_data)
1071 {
1072   StateChangedSignalData *data = user_data;
1073 
1074   if (data->player->inhibit_sigs && data->state != GST_PLAYER_STATE_STOPPED
1075       && data->state != GST_PLAYER_STATE_PAUSED)
1076     return;
1077 
1078   g_signal_emit (data->player, signals[SIGNAL_STATE_CHANGED], 0, data->state);
1079 }
1080 
1081 static void
state_changed_signal_data_free(StateChangedSignalData * data)1082 state_changed_signal_data_free (StateChangedSignalData * data)
1083 {
1084   g_object_unref (data->player);
1085   g_free (data);
1086 }
1087 
1088 static void
change_state(GstPlayer * self,GstPlayerState state)1089 change_state (GstPlayer * self, GstPlayerState state)
1090 {
1091 #ifdef OHOS_EXT_FUNC
1092 /* ohos.ext.func.0011
1093  * In non-seek mode of stream playback, stop is expected to be invoked in the EOS state,
1094  * but the gstplay does not call back when stop is invoked in the EOS state.
1095  */
1096   if (state == self->app_state && state != GST_PLAYER_STATE_STOPPED)
1097 #else
1098   if (state == self->app_state)
1099 #endif
1100     return;
1101 
1102 #ifdef OHOS_EXT_FUNC
1103   // ohos.ext.func.0012
1104   if (state == GST_PLAYER_STATE_PAUSED || state == GST_PLAYER_STATE_PLAYING) {
1105     g_object_set (self->playbin, "state-change", (guint)state, NULL);
1106   } else if (self->app_state == GST_PLAYER_STATE_PLAYING && state == GST_PLAYER_STATE_BUFFERING) {
1107     g_object_set (self->playbin, "timeout", self->timeout, NULL);
1108     g_object_set (self->playbin, "state-change", (guint)state, NULL);
1109   }
1110 #endif
1111 
1112   GST_DEBUG_OBJECT (self, "Changing app state from %s to %s",
1113       gst_player_state_get_name (self->app_state),
1114       gst_player_state_get_name (state));
1115   self->app_state = state;
1116 
1117   if (g_signal_handler_find (self, G_SIGNAL_MATCH_ID,
1118           signals[SIGNAL_STATE_CHANGED], 0, NULL, NULL, NULL) != 0) {
1119     StateChangedSignalData *data = g_new (StateChangedSignalData, 1);
1120 
1121     data->player = g_object_ref (self);
1122     data->state = state;
1123     gst_player_signal_dispatcher_dispatch (self->signal_dispatcher, self,
1124         state_changed_dispatch, data,
1125         (GDestroyNotify) state_changed_signal_data_free);
1126   }
1127 }
1128 
1129 typedef struct
1130 {
1131   GstPlayer *player;
1132   GstClockTime position;
1133 } PositionUpdatedSignalData;
1134 
1135 static void
position_updated_dispatch(gpointer user_data)1136 position_updated_dispatch (gpointer user_data)
1137 {
1138   PositionUpdatedSignalData *data = user_data;
1139 
1140   if (data->player->inhibit_sigs)
1141     return;
1142 
1143   if (data->player->target_state >= GST_STATE_PAUSED) {
1144     g_signal_emit (data->player, signals[SIGNAL_POSITION_UPDATED], 0,
1145         data->position);
1146     g_object_notify_by_pspec (G_OBJECT (data->player),
1147         param_specs[PROP_POSITION]);
1148   }
1149 }
1150 
1151 static void
position_updated_signal_data_free(PositionUpdatedSignalData * data)1152 position_updated_signal_data_free (PositionUpdatedSignalData * data)
1153 {
1154   g_object_unref (data->player);
1155   g_free (data);
1156 }
1157 
1158 static gboolean
tick_cb(gpointer user_data)1159 tick_cb (gpointer user_data)
1160 {
1161   GstPlayer *self = GST_PLAYER (user_data);
1162   gint64 position;
1163 
1164   if (self->target_state >= GST_STATE_PAUSED
1165       && gst_element_query_position (self->playbin, GST_FORMAT_TIME,
1166           &position)) {
1167     GST_LOG_OBJECT (self, "Position %" GST_TIME_FORMAT,
1168         GST_TIME_ARGS (position));
1169 
1170     if (g_signal_handler_find (self, G_SIGNAL_MATCH_ID,
1171             signals[SIGNAL_POSITION_UPDATED], 0, NULL, NULL, NULL) != 0) {
1172       PositionUpdatedSignalData *data = g_new (PositionUpdatedSignalData, 1);
1173 
1174       data->player = g_object_ref (self);
1175       data->position = position;
1176       gst_player_signal_dispatcher_dispatch (self->signal_dispatcher, self,
1177           position_updated_dispatch, data,
1178           (GDestroyNotify) position_updated_signal_data_free);
1179     }
1180   }
1181 
1182   return G_SOURCE_CONTINUE;
1183 }
1184 
1185 static void
add_tick_source(GstPlayer * self)1186 add_tick_source (GstPlayer * self)
1187 {
1188   guint position_update_interval_ms;
1189 
1190   if (self->tick_source)
1191     return;
1192 
1193   position_update_interval_ms =
1194       gst_player_config_get_position_update_interval (self->config);
1195   if (!position_update_interval_ms)
1196     return;
1197 
1198   self->tick_source = g_timeout_source_new (position_update_interval_ms);
1199   g_source_set_callback (self->tick_source, (GSourceFunc) tick_cb, self, NULL);
1200   g_source_attach (self->tick_source, self->context);
1201 }
1202 
1203 static void
remove_tick_source(GstPlayer * self)1204 remove_tick_source (GstPlayer * self)
1205 {
1206   if (!self->tick_source)
1207     return;
1208 
1209   g_source_destroy (self->tick_source);
1210   g_source_unref (self->tick_source);
1211   self->tick_source = NULL;
1212 }
1213 
1214 static gboolean
ready_timeout_cb(gpointer user_data)1215 ready_timeout_cb (gpointer user_data)
1216 {
1217   GstPlayer *self = user_data;
1218 
1219   if (self->target_state <= GST_STATE_READY) {
1220     GST_DEBUG_OBJECT (self, "Setting pipeline to NULL state");
1221     self->target_state = GST_STATE_NULL;
1222     self->current_state = GST_STATE_NULL;
1223     gst_element_set_state (self->playbin, GST_STATE_NULL);
1224   }
1225 
1226   return G_SOURCE_REMOVE;
1227 }
1228 
1229 static void
add_ready_timeout_source(GstPlayer * self)1230 add_ready_timeout_source (GstPlayer * self)
1231 {
1232   if (self->ready_timeout_source)
1233     return;
1234 
1235   self->ready_timeout_source = g_timeout_source_new_seconds (60);
1236   g_source_set_callback (self->ready_timeout_source,
1237       (GSourceFunc) ready_timeout_cb, self, NULL);
1238   g_source_attach (self->ready_timeout_source, self->context);
1239 }
1240 
1241 static void
remove_ready_timeout_source(GstPlayer * self)1242 remove_ready_timeout_source (GstPlayer * self)
1243 {
1244   if (!self->ready_timeout_source)
1245     return;
1246 
1247   g_source_destroy (self->ready_timeout_source);
1248   g_source_unref (self->ready_timeout_source);
1249   self->ready_timeout_source = NULL;
1250 }
1251 
1252 typedef struct
1253 {
1254   GstPlayer *player;
1255   GError *err;
1256 } ErrorSignalData;
1257 
1258 static void
error_dispatch(gpointer user_data)1259 error_dispatch (gpointer user_data)
1260 {
1261   ErrorSignalData *data = user_data;
1262 
1263   if (data->player->inhibit_sigs)
1264     return;
1265 
1266   g_signal_emit (data->player, signals[SIGNAL_ERROR], 0, data->err);
1267 }
1268 
1269 static void
free_error_signal_data(ErrorSignalData * data)1270 free_error_signal_data (ErrorSignalData * data)
1271 {
1272   g_object_unref (data->player);
1273   g_clear_error (&data->err);
1274   g_free (data);
1275 }
1276 
1277 static void
emit_error(GstPlayer * self,GError * err)1278 emit_error (GstPlayer * self, GError * err)
1279 {
1280   GST_ERROR_OBJECT (self, "Error: %s (%s, %d)", err->message,
1281       g_quark_to_string (err->domain), err->code);
1282 
1283   if (g_signal_handler_find (self, G_SIGNAL_MATCH_ID,
1284           signals[SIGNAL_ERROR], 0, NULL, NULL, NULL) != 0) {
1285     ErrorSignalData *data = g_new (ErrorSignalData, 1);
1286 
1287     data->player = g_object_ref (self);
1288     data->err = g_error_copy (err);
1289     gst_player_signal_dispatcher_dispatch (self->signal_dispatcher, self,
1290         error_dispatch, data, (GDestroyNotify) free_error_signal_data);
1291   }
1292 
1293   g_error_free (err);
1294 
1295   remove_tick_source (self);
1296   remove_ready_timeout_source (self);
1297 
1298   self->target_state = GST_STATE_NULL;
1299   self->current_state = GST_STATE_NULL;
1300   self->is_live = FALSE;
1301   self->is_eos = FALSE;
1302   gst_element_set_state (self->playbin, GST_STATE_NULL);
1303   change_state (self, GST_PLAYER_STATE_STOPPED);
1304   self->buffering = 100;
1305 
1306   g_mutex_lock (&self->lock);
1307   if (self->media_info) {
1308     g_object_unref (self->media_info);
1309     self->media_info = NULL;
1310   }
1311 
1312   if (self->global_tags) {
1313     gst_tag_list_unref (self->global_tags);
1314     self->global_tags = NULL;
1315   }
1316 
1317   self->seek_pending = FALSE;
1318   remove_seek_source (self);
1319   self->seek_position = GST_CLOCK_TIME_NONE;
1320   self->last_seek_time = GST_CLOCK_TIME_NONE;
1321   g_mutex_unlock (&self->lock);
1322 }
1323 
1324 static void
dump_dot_file(GstPlayer * self,const gchar * name)1325 dump_dot_file (GstPlayer * self, const gchar * name)
1326 {
1327   gchar *full_name;
1328 
1329   full_name = g_strdup_printf ("gst-player.%p.%s", self, name);
1330 
1331   GST_DEBUG_BIN_TO_DOT_FILE_WITH_TS (GST_BIN (self->playbin),
1332       GST_DEBUG_GRAPH_SHOW_ALL, full_name);
1333 
1334   g_free (full_name);
1335 }
1336 
1337 typedef struct
1338 {
1339   GstPlayer *player;
1340   GError *err;
1341 } WarningSignalData;
1342 
1343 static void
warning_dispatch(gpointer user_data)1344 warning_dispatch (gpointer user_data)
1345 {
1346   WarningSignalData *data = user_data;
1347 
1348   if (data->player->inhibit_sigs)
1349     return;
1350 
1351   g_signal_emit (data->player, signals[SIGNAL_WARNING], 0, data->err);
1352 }
1353 
1354 static void
free_warning_signal_data(WarningSignalData * data)1355 free_warning_signal_data (WarningSignalData * data)
1356 {
1357   g_object_unref (data->player);
1358   g_clear_error (&data->err);
1359   g_free (data);
1360 }
1361 
1362 static void
emit_warning(GstPlayer * self,GError * err)1363 emit_warning (GstPlayer * self, GError * err)
1364 {
1365   GST_ERROR_OBJECT (self, "Warning: %s (%s, %d)", err->message,
1366       g_quark_to_string (err->domain), err->code);
1367 
1368   if (g_signal_handler_find (self, G_SIGNAL_MATCH_ID,
1369           signals[SIGNAL_WARNING], 0, NULL, NULL, NULL) != 0) {
1370     WarningSignalData *data = g_new (WarningSignalData, 1);
1371 
1372     data->player = g_object_ref (self);
1373     data->err = g_error_copy (err);
1374     gst_player_signal_dispatcher_dispatch (self->signal_dispatcher, self,
1375         warning_dispatch, data, (GDestroyNotify) free_warning_signal_data);
1376   }
1377 
1378   g_error_free (err);
1379 }
1380 
1381 #ifdef OHOS_EXT_FUNC
1382 // ohos.ext.func.0001
1383 typedef struct
1384 {
1385   GstPlayer *player;
1386   GstMessage *msg;
1387 } MsgSignalData;
1388 
1389 static void
error_msg_dispatch(gpointer user_data)1390 error_msg_dispatch (gpointer user_data)
1391 {
1392   GST_ERROR_OBJECT (NULL, "dispatch error msg\n");
1393   MsgSignalData *data = user_data;
1394 
1395   if (data->player->inhibit_sigs)
1396     return;
1397   g_return_if_fail (data->msg != NULL);
1398   g_signal_emit (data->player, signals[SIGNAL_ERROR_MSG], 0, data->msg);
1399 }
1400 
1401 static void
free_error_msg_signal_data(MsgSignalData * data)1402 free_error_msg_signal_data (MsgSignalData * data)
1403 {
1404   GST_ERROR_OBJECT (NULL, "free error msg\n");
1405   g_object_unref (data->player);
1406   gst_message_unref (data->msg);
1407   g_free (data);
1408 }
1409 
1410 static void
emit_error_msg(GstPlayer * self,GstMessage * msg)1411 emit_error_msg (GstPlayer * self, GstMessage * msg)
1412 {
1413   g_return_if_fail (msg != NULL);
1414   if (g_signal_handler_find (self, G_SIGNAL_MATCH_ID,
1415           signals[SIGNAL_ERROR_MSG], 0, NULL, NULL, NULL) != 0) {
1416     MsgSignalData *data = g_new (MsgSignalData, 1);
1417 
1418     data->player = g_object_ref (self);
1419     data->msg = gst_message_ref (msg);
1420     gst_player_signal_dispatcher_dispatch (self->signal_dispatcher, self,
1421         error_msg_dispatch, data, (GDestroyNotify) free_error_msg_signal_data);
1422   }
1423 
1424   remove_tick_source (self);
1425   remove_ready_timeout_source (self);
1426 
1427   self->target_state = GST_STATE_NULL;
1428   self->current_state = GST_STATE_NULL;
1429   self->is_live = FALSE;
1430   self->is_eos = FALSE;
1431   gst_element_set_state (self->playbin, GST_STATE_NULL);
1432   change_state (self, GST_PLAYER_STATE_STOPPED);
1433   self->buffering = 100;
1434 
1435   g_mutex_lock (&self->lock);
1436   if (self->media_info) {
1437     g_object_unref (self->media_info);
1438     self->media_info = NULL;
1439   }
1440 
1441   if (self->global_tags) {
1442     gst_tag_list_unref (self->global_tags);
1443     self->global_tags = NULL;
1444   }
1445 
1446   self->seek_pending = FALSE;
1447   remove_seek_source (self);
1448   self->seek_position = GST_CLOCK_TIME_NONE;
1449   self->last_seek_time = GST_CLOCK_TIME_NONE;
1450   g_mutex_unlock (&self->lock);
1451 }
1452 
1453 static void
error_cb_msg(G_GNUC_UNUSED GstBus * bus,GstMessage * msg,gpointer user_data)1454 error_cb_msg (G_GNUC_UNUSED GstBus * bus, GstMessage * msg, gpointer user_data)
1455 {
1456   GstPlayer *self = GST_PLAYER (user_data);
1457   GST_ERROR_OBJECT (self, "deal with error msg cb\n");
1458 
1459   dump_dot_file (self, "error");
1460   if (msg == NULL) {
1461     GST_ERROR_OBJECT (self, "msg is null\n");
1462   }
1463   emit_error_msg (self, msg);
1464 }
1465 #else
1466 static void
error_cb(G_GNUC_UNUSED GstBus * bus,GstMessage * msg,gpointer user_data)1467 error_cb (G_GNUC_UNUSED GstBus * bus, GstMessage * msg, gpointer user_data)
1468 {
1469   GstPlayer *self = GST_PLAYER (user_data);
1470   GError *err, *player_err;
1471   gchar *name, *debug, *message, *full_message;
1472 
1473   dump_dot_file (self, "error");
1474 
1475   gst_message_parse_error (msg, &err, &debug);
1476 
1477   name = gst_object_get_path_string (msg->src);
1478   message = gst_error_get_message (err->domain, err->code);
1479 
1480   if (debug)
1481     full_message =
1482         g_strdup_printf ("Error from element %s: %s\n%s\n%s", name, message,
1483         err->message, debug);
1484   else
1485     full_message =
1486         g_strdup_printf ("Error from element %s: %s\n%s", name, message,
1487         err->message);
1488 
1489   GST_ERROR_OBJECT (self, "ERROR: from element %s: %s\n", name, err->message);
1490   if (debug != NULL)
1491     GST_ERROR_OBJECT (self, "Additional debug info:\n%s\n", debug);
1492 
1493   player_err =
1494       g_error_new_literal (GST_PLAYER_ERROR, GST_PLAYER_ERROR_FAILED,
1495       full_message);
1496   emit_error (self, player_err);
1497 
1498   g_clear_error (&err);
1499   g_free (debug);
1500   g_free (name);
1501   g_free (full_message);
1502   g_free (message);
1503 }
1504 #endif
1505 
1506 static void
warning_cb(G_GNUC_UNUSED GstBus * bus,GstMessage * msg,gpointer user_data)1507 warning_cb (G_GNUC_UNUSED GstBus * bus, GstMessage * msg, gpointer user_data)
1508 {
1509   GstPlayer *self = GST_PLAYER (user_data);
1510   GError *err, *player_err;
1511   gchar *name, *debug, *message, *full_message;
1512 
1513   dump_dot_file (self, "warning");
1514 
1515   gst_message_parse_warning (msg, &err, &debug);
1516 
1517   name = gst_object_get_path_string (msg->src);
1518   message = gst_error_get_message (err->domain, err->code);
1519 
1520   if (debug)
1521     full_message =
1522         g_strdup_printf ("Warning from element %s: %s\n%s\n%s", name, message,
1523         err->message, debug);
1524   else
1525     full_message =
1526         g_strdup_printf ("Warning from element %s: %s\n%s", name, message,
1527         err->message);
1528 
1529   GST_WARNING_OBJECT (self, "WARNING: from element %s: %s\n", name,
1530       err->message);
1531   if (debug != NULL)
1532     GST_WARNING_OBJECT (self, "Additional debug info:\n%s\n", debug);
1533 
1534   player_err =
1535       g_error_new_literal (GST_PLAYER_ERROR, GST_PLAYER_ERROR_FAILED,
1536       full_message);
1537   emit_warning (self, player_err);
1538 
1539   g_clear_error (&err);
1540   g_free (debug);
1541   g_free (name);
1542   g_free (full_message);
1543   g_free (message);
1544 }
1545 
1546 static void
eos_dispatch(gpointer user_data)1547 eos_dispatch (gpointer user_data)
1548 {
1549   GstPlayer *player = user_data;
1550 
1551   if (player->inhibit_sigs)
1552     return;
1553 
1554   g_signal_emit (player, signals[SIGNAL_END_OF_STREAM], 0);
1555 }
1556 
1557 static void
eos_cb(G_GNUC_UNUSED GstBus * bus,G_GNUC_UNUSED GstMessage * msg,gpointer user_data)1558 eos_cb (G_GNUC_UNUSED GstBus * bus, G_GNUC_UNUSED GstMessage * msg,
1559     gpointer user_data)
1560 {
1561   GstPlayer *self = GST_PLAYER (user_data);
1562 
1563   GST_DEBUG_OBJECT (self, "End of stream");
1564 
1565   tick_cb (self);
1566   remove_tick_source (self);
1567 
1568   if (g_signal_handler_find (self, G_SIGNAL_MATCH_ID,
1569           signals[SIGNAL_END_OF_STREAM], 0, NULL, NULL, NULL) != 0) {
1570     gst_player_signal_dispatcher_dispatch (self->signal_dispatcher, self,
1571         eos_dispatch, g_object_ref (self), (GDestroyNotify) g_object_unref);
1572   }
1573   change_state (self, GST_PLAYER_STATE_STOPPED);
1574   self->buffering = 100;
1575   self->is_eos = TRUE;
1576 }
1577 
1578 typedef struct
1579 {
1580   GstPlayer *player;
1581   gint percent;
1582 } BufferingSignalData;
1583 
1584 static void
buffering_dispatch(gpointer user_data)1585 buffering_dispatch (gpointer user_data)
1586 {
1587   BufferingSignalData *data = user_data;
1588 
1589   if (data->player->inhibit_sigs)
1590     return;
1591 
1592   if (data->player->target_state >= GST_STATE_PAUSED) {
1593     g_signal_emit (data->player, signals[SIGNAL_BUFFERING], 0, data->percent);
1594   }
1595 }
1596 
1597 static void
buffering_signal_data_free(BufferingSignalData * data)1598 buffering_signal_data_free (BufferingSignalData * data)
1599 {
1600   g_object_unref (data->player);
1601   g_free (data);
1602 }
1603 
1604 static void
buffering_cb(G_GNUC_UNUSED GstBus * bus,GstMessage * msg,gpointer user_data)1605 buffering_cb (G_GNUC_UNUSED GstBus * bus, GstMessage * msg, gpointer user_data)
1606 {
1607   GstPlayer *self = GST_PLAYER (user_data);
1608   gint percent;
1609 
1610   if (self->target_state < GST_STATE_PAUSED)
1611     return;
1612   if (self->is_live)
1613     return;
1614 
1615 #ifdef OHOS_EXT_FUNC
1616   /* ohos.ext.func.0007:
1617    * Queue2 will post buffering message(buffering-level is 100%) when playbin is GST_STATE_READY,
1618    * and player will be set to GST_PLAYER_STATE_PAUSED which will cause the prepared-message to be reported in advance
1619    * To avoid this, we do not handle buffering message which comes from queue2.
1620    */
1621   if ((GST_MESSAGE_SRC (msg) != NULL) &&
1622     (strncmp (gst_element_get_name (GST_MESSAGE_SRC(msg)), "queue2", strlen ("queue2")) == 0)) {
1623     GST_DEBUG_OBJECT (self, "buffering msg comes from queue2, do not handle it");
1624     return;
1625   }
1626 #endif
1627 
1628   gst_message_parse_buffering (msg, &percent);
1629   GST_LOG_OBJECT (self, "Buffering %d%%", percent);
1630 
1631   if (percent < 100 && self->target_state >= GST_STATE_PAUSED) {
1632     GstStateChangeReturn state_ret;
1633 
1634     GST_DEBUG_OBJECT (self, "Waiting for buffering to finish");
1635     state_ret = gst_element_set_state (self->playbin, GST_STATE_PAUSED);
1636 
1637     if (state_ret == GST_STATE_CHANGE_FAILURE) {
1638       emit_error (self, g_error_new (GST_PLAYER_ERROR, GST_PLAYER_ERROR_FAILED,
1639               "Failed to handle buffering"));
1640       return;
1641     }
1642 
1643     change_state (self, GST_PLAYER_STATE_BUFFERING);
1644   }
1645 
1646   if (self->buffering != percent) {
1647     if (g_signal_handler_find (self, G_SIGNAL_MATCH_ID,
1648             signals[SIGNAL_BUFFERING], 0, NULL, NULL, NULL) != 0) {
1649       BufferingSignalData *data = g_new (BufferingSignalData, 1);
1650 
1651       data->player = g_object_ref (self);
1652       data->percent = percent;
1653       gst_player_signal_dispatcher_dispatch (self->signal_dispatcher, self,
1654           buffering_dispatch, data,
1655           (GDestroyNotify) buffering_signal_data_free);
1656     }
1657 
1658     self->buffering = percent;
1659   }
1660 
1661 
1662   g_mutex_lock (&self->lock);
1663   if (percent == 100 && (self->seek_position != GST_CLOCK_TIME_NONE ||
1664           self->seek_pending)) {
1665     g_mutex_unlock (&self->lock);
1666 
1667     GST_DEBUG_OBJECT (self, "Buffering finished - seek pending");
1668   } else if (percent == 100 && self->target_state >= GST_STATE_PLAYING
1669       && self->current_state >= GST_STATE_PAUSED) {
1670     GstStateChangeReturn state_ret;
1671 
1672     g_mutex_unlock (&self->lock);
1673 
1674     GST_DEBUG_OBJECT (self, "Buffering finished - going to PLAYING");
1675     state_ret = gst_element_set_state (self->playbin, GST_STATE_PLAYING);
1676     /* Application state change is happening when the state change happened */
1677     if (state_ret == GST_STATE_CHANGE_FAILURE)
1678       emit_error (self, g_error_new (GST_PLAYER_ERROR, GST_PLAYER_ERROR_FAILED,
1679               "Failed to handle buffering"));
1680   } else if (percent == 100 && self->target_state >= GST_STATE_PAUSED) {
1681     g_mutex_unlock (&self->lock);
1682 
1683     GST_DEBUG_OBJECT (self, "Buffering finished - staying PAUSED");
1684 
1685 #ifdef OHOS_OPT_COMPAT
1686 /* ohos.opt.compat.0010
1687  * When the above state is switched in advance,
1688  * it is necessary to take a state switch here to solve the problem of the last business trip in the state */
1689   if (self->isStateChange) {
1690     change_state (self, GST_PLAYER_STATE_PAUSED);
1691     self->isStateChange = FALSE;
1692   }
1693 #else
1694   change_state (self, GST_PLAYER_STATE_PAUSED);
1695 #endif
1696   } else {
1697     g_mutex_unlock (&self->lock);
1698   }
1699 }
1700 
1701 #ifdef OHOS_EXT_FUNC
1702 // ohos.ext.func.0012
1703 typedef struct
1704 {
1705   GstPlayer *player;
1706   GstClockTime buffering_time;
1707   guint mq_num_id;
1708 } BufferingTimeSignalData;
1709 
1710 static void
buffering_time_dispatch(gpointer user_data)1711 buffering_time_dispatch (gpointer user_data)
1712 {
1713   BufferingTimeSignalData *data = user_data;
1714 
1715   if (data->player->inhibit_sigs)
1716     return;
1717 
1718   if (data->player->target_state >= GST_STATE_PAUSED) {
1719     g_signal_emit (data->player, signals[SIGNAL_BUFFERING_TIME], 0, data->buffering_time, data->mq_num_id);
1720   }
1721 }
1722 
1723 static void
buffering_time_signal_data_free(BufferingTimeSignalData * data)1724 buffering_time_signal_data_free (BufferingTimeSignalData * data)
1725 {
1726   g_object_unref (data->player);
1727   g_free (data);
1728 }
1729 
1730 static void
buffering_time_cb(GstMessage * msg,gpointer user_data)1731 buffering_time_cb (GstMessage * msg, gpointer user_data)
1732 {
1733   GstPlayer *self = GST_PLAYER (user_data);
1734   gint64 buffering_time;
1735   guint mq_num_id;
1736 
1737   gst_message_parse_buffering_time (msg, &buffering_time, &mq_num_id);
1738   GST_DEBUG_OBJECT (self, "mq_num_id = %u, Buffering time %" G_GUINT64_FORMAT, mq_num_id, buffering_time);
1739 
1740   if (g_signal_handler_find (self, G_SIGNAL_MATCH_ID, signals[SIGNAL_BUFFERING_TIME], 0, NULL, NULL, NULL) != 0) {
1741     BufferingTimeSignalData *data = g_new (BufferingTimeSignalData, 1);
1742 
1743     data->player = g_object_ref (self);
1744     data->buffering_time = buffering_time;
1745     data->mq_num_id = mq_num_id;
1746     gst_player_signal_dispatcher_dispatch (self->signal_dispatcher, self,
1747         buffering_time_dispatch, data,
1748         (GDestroyNotify) buffering_time_signal_data_free);
1749   }
1750 }
1751 
1752 // ohos.ext.func.0013
1753 typedef struct
1754 {
1755   GstPlayer *player;
1756   guint mq_num_use_buffering;
1757 } MqNumUseBufferingSignalData;
1758 
1759 static void
mq_num_use_buffering_dispatch(gpointer user_data)1760 mq_num_use_buffering_dispatch (gpointer user_data)
1761 {
1762   MqNumUseBufferingSignalData *data = user_data;
1763 
1764   if (data->player->inhibit_sigs)
1765     return;
1766   g_signal_emit (data->player, signals[SIGNAL_MQ_NUM_USE_BUFFERING], 0, data->mq_num_use_buffering);
1767 }
1768 
1769 static void
mq_num_use_buffering_signal_data_free(MqNumUseBufferingSignalData * data)1770 mq_num_use_buffering_signal_data_free (MqNumUseBufferingSignalData * data)
1771 {
1772   g_object_unref (data->player);
1773   g_free (data);
1774 }
1775 
1776 static void
mq_num_use_buffering_cb(GstMessage * msg,gpointer user_data)1777 mq_num_use_buffering_cb (GstMessage * msg, gpointer user_data)
1778 {
1779   GstPlayer *self = GST_PLAYER (user_data);
1780   guint mq_num_use_buffering;
1781 
1782   gst_message_parse_mq_num_use_buffering (msg, &mq_num_use_buffering);
1783   GST_DEBUG_OBJECT (self, "mq_num_use_buffering = %u", mq_num_use_buffering);
1784 
1785   if (g_signal_handler_find (self, G_SIGNAL_MATCH_ID, signals[SIGNAL_MQ_NUM_USE_BUFFERING], 0, NULL, NULL, NULL) != 0) {
1786     MqNumUseBufferingSignalData *data = g_new (MqNumUseBufferingSignalData, 1);
1787 
1788     data->player = g_object_ref (self);
1789     data->mq_num_use_buffering = mq_num_use_buffering;
1790     gst_player_signal_dispatcher_dispatch (self->signal_dispatcher, self,
1791         mq_num_use_buffering_dispatch, data,
1792         (GDestroyNotify) mq_num_use_buffering_signal_data_free);
1793   }
1794 }
1795 
1796 // ohos.ext.func.0014
1797 typedef struct {
1798   GstPlayer *player;
1799   gint width;
1800   gint height;
1801 } ResolutionChangedSignalData;
1802 
1803 static void
resolution_changed_dispatch(gpointer user_data)1804 resolution_changed_dispatch (gpointer user_data)
1805 {
1806   ResolutionChangedSignalData *data = user_data;
1807 
1808   if (data->player->inhibit_sigs)
1809     return;
1810 
1811   g_signal_emit (data->player, signals[SIGNAL_RESOLUTION_CHANGED], 0, data->width, data->height);
1812 }
1813 
1814 static void
resolution_changed_signal_data_free(ResolutionChangedSignalData * data)1815 resolution_changed_signal_data_free (ResolutionChangedSignalData * data)
1816 {
1817   g_object_unref (data->player);
1818   g_free (data);
1819 }
1820 
1821 static void
resulution_changed_cb(GstMessage * msg,gpointer user_data)1822 resulution_changed_cb (GstMessage * msg, gpointer user_data)
1823 {
1824   GstPlayer *self = GST_PLAYER (user_data);
1825   gint width;
1826   gint height;
1827 
1828   gst_message_parse_resulution_changed (msg, &width, &height);
1829   GST_DEBUG_OBJECT (self, "resulution width %d, height %d", width, height);
1830 
1831   if (g_signal_handler_find (self, G_SIGNAL_MATCH_ID, signals[SIGNAL_RESOLUTION_CHANGED], 0, NULL, NULL, NULL) != 0) {
1832     ResolutionChangedSignalData *data = g_new (ResolutionChangedSignalData, 1);
1833 
1834     data->player = g_object_ref (self);
1835     data->width = width;
1836     data->height = height;
1837     gst_player_signal_dispatcher_dispatch (self->signal_dispatcher, self,
1838         resolution_changed_dispatch, data,
1839         (GDestroyNotify) resolution_changed_signal_data_free);
1840     }
1841 }
1842 #endif
1843 
1844 static void
clock_lost_cb(G_GNUC_UNUSED GstBus * bus,G_GNUC_UNUSED GstMessage * msg,gpointer user_data)1845 clock_lost_cb (G_GNUC_UNUSED GstBus * bus, G_GNUC_UNUSED GstMessage * msg,
1846     gpointer user_data)
1847 {
1848   GstPlayer *self = GST_PLAYER (user_data);
1849   GstStateChangeReturn state_ret;
1850 
1851   GST_DEBUG_OBJECT (self, "Clock lost");
1852   if (self->target_state >= GST_STATE_PLAYING) {
1853     state_ret = gst_element_set_state (self->playbin, GST_STATE_PAUSED);
1854     if (state_ret != GST_STATE_CHANGE_FAILURE)
1855       state_ret = gst_element_set_state (self->playbin, GST_STATE_PLAYING);
1856 
1857     if (state_ret == GST_STATE_CHANGE_FAILURE)
1858       emit_error (self, g_error_new (GST_PLAYER_ERROR, GST_PLAYER_ERROR_FAILED,
1859               "Failed to handle clock loss"));
1860   }
1861 }
1862 
1863 typedef struct
1864 {
1865   GstPlayer *player;
1866   gint width, height;
1867 } VideoDimensionsChangedSignalData;
1868 
1869 static void
video_dimensions_changed_dispatch(gpointer user_data)1870 video_dimensions_changed_dispatch (gpointer user_data)
1871 {
1872   VideoDimensionsChangedSignalData *data = user_data;
1873 
1874   if (data->player->inhibit_sigs)
1875     return;
1876 
1877   if (data->player->target_state >= GST_STATE_PAUSED) {
1878     g_signal_emit (data->player, signals[SIGNAL_VIDEO_DIMENSIONS_CHANGED], 0,
1879         data->width, data->height);
1880   }
1881 }
1882 
1883 static void
video_dimensions_changed_signal_data_free(VideoDimensionsChangedSignalData * data)1884 video_dimensions_changed_signal_data_free (VideoDimensionsChangedSignalData *
1885     data)
1886 {
1887   g_object_unref (data->player);
1888   g_free (data);
1889 }
1890 
1891 static void
check_video_dimensions_changed(GstPlayer * self)1892 check_video_dimensions_changed (GstPlayer * self)
1893 {
1894   GstElement *video_sink;
1895   GstPad *video_sink_pad;
1896   GstCaps *caps;
1897   GstVideoInfo info;
1898   gint width = 0, height = 0;
1899 
1900   g_object_get (self->playbin, "video-sink", &video_sink, NULL);
1901   if (!video_sink)
1902     goto out;
1903 
1904   video_sink_pad = gst_element_get_static_pad (video_sink, "sink");
1905   if (!video_sink_pad) {
1906     gst_object_unref (video_sink);
1907     goto out;
1908   }
1909 
1910   caps = gst_pad_get_current_caps (video_sink_pad);
1911 
1912   if (caps) {
1913     if (gst_video_info_from_caps (&info, caps)) {
1914       info.width = info.width * info.par_n / info.par_d;
1915 
1916       GST_DEBUG_OBJECT (self, "Video dimensions changed: %dx%d", info.width,
1917           info.height);
1918       width = info.width;
1919       height = info.height;
1920     }
1921 
1922     gst_caps_unref (caps);
1923   }
1924   gst_object_unref (video_sink_pad);
1925   gst_object_unref (video_sink);
1926 
1927 out:
1928   if (g_signal_handler_find (self, G_SIGNAL_MATCH_ID,
1929           signals[SIGNAL_VIDEO_DIMENSIONS_CHANGED], 0, NULL, NULL, NULL) != 0) {
1930     VideoDimensionsChangedSignalData *data =
1931         g_new (VideoDimensionsChangedSignalData, 1);
1932 
1933     data->player = g_object_ref (self);
1934     data->width = width;
1935     data->height = height;
1936     gst_player_signal_dispatcher_dispatch (self->signal_dispatcher, self,
1937         video_dimensions_changed_dispatch, data,
1938         (GDestroyNotify) video_dimensions_changed_signal_data_free);
1939   }
1940 }
1941 
1942 static void
notify_caps_cb(G_GNUC_UNUSED GObject * object,G_GNUC_UNUSED GParamSpec * pspec,gpointer user_data)1943 notify_caps_cb (G_GNUC_UNUSED GObject * object,
1944     G_GNUC_UNUSED GParamSpec * pspec, gpointer user_data)
1945 {
1946   GstPlayer *self = GST_PLAYER (user_data);
1947 
1948   check_video_dimensions_changed (self);
1949 }
1950 
1951 typedef struct
1952 {
1953   GstPlayer *player;
1954   GstClockTime duration;
1955 } DurationChangedSignalData;
1956 
1957 static void
duration_changed_dispatch(gpointer user_data)1958 duration_changed_dispatch (gpointer user_data)
1959 {
1960   DurationChangedSignalData *data = user_data;
1961 
1962   if (data->player->inhibit_sigs)
1963     return;
1964 
1965   if (data->player->target_state >= GST_STATE_PAUSED) {
1966     g_signal_emit (data->player, signals[SIGNAL_DURATION_CHANGED], 0,
1967         data->duration);
1968     g_object_notify_by_pspec (G_OBJECT (data->player),
1969         param_specs[PROP_DURATION]);
1970   }
1971 }
1972 
1973 static void
duration_changed_signal_data_free(DurationChangedSignalData * data)1974 duration_changed_signal_data_free (DurationChangedSignalData * data)
1975 {
1976   g_object_unref (data->player);
1977   g_free (data);
1978 }
1979 
1980 static void
emit_duration_changed(GstPlayer * self,GstClockTime duration)1981 emit_duration_changed (GstPlayer * self, GstClockTime duration)
1982 {
1983   gboolean updated = FALSE;
1984 
1985   if (self->cached_duration == duration)
1986     return;
1987 
1988   GST_DEBUG_OBJECT (self, "Duration changed %" GST_TIME_FORMAT,
1989       GST_TIME_ARGS (duration));
1990 
1991   self->cached_duration = duration;
1992   g_mutex_lock (&self->lock);
1993   if (self->media_info) {
1994     self->media_info->duration = duration;
1995     updated = TRUE;
1996   }
1997   g_mutex_unlock (&self->lock);
1998   if (updated) {
1999     emit_media_info_updated_signal (self);
2000   }
2001 
2002   if (g_signal_handler_find (self, G_SIGNAL_MATCH_ID,
2003           signals[SIGNAL_DURATION_CHANGED], 0, NULL, NULL, NULL) != 0) {
2004     DurationChangedSignalData *data = g_new (DurationChangedSignalData, 1);
2005 
2006     data->player = g_object_ref (self);
2007     data->duration = duration;
2008     gst_player_signal_dispatcher_dispatch (self->signal_dispatcher, self,
2009         duration_changed_dispatch, data,
2010         (GDestroyNotify) duration_changed_signal_data_free);
2011   }
2012 }
2013 
2014 typedef struct
2015 {
2016   GstPlayer *player;
2017   GstClockTime position;
2018 } SeekDoneSignalData;
2019 
2020 static void
seek_done_dispatch(gpointer user_data)2021 seek_done_dispatch (gpointer user_data)
2022 {
2023   SeekDoneSignalData *data = user_data;
2024 
2025   if (data->player->inhibit_sigs)
2026     return;
2027 
2028   g_signal_emit (data->player, signals[SIGNAL_SEEK_DONE], 0, data->position);
2029 }
2030 
2031 static void
seek_done_signal_data_free(SeekDoneSignalData * data)2032 seek_done_signal_data_free (SeekDoneSignalData * data)
2033 {
2034   g_object_unref (data->player);
2035   g_free (data);
2036 }
2037 
2038 static void
emit_seek_done(GstPlayer * self)2039 emit_seek_done (GstPlayer * self)
2040 {
2041   if (g_signal_handler_find (self, G_SIGNAL_MATCH_ID,
2042           signals[SIGNAL_SEEK_DONE], 0, NULL, NULL, NULL) != 0) {
2043     SeekDoneSignalData *data = g_new (SeekDoneSignalData, 1);
2044 
2045     data->player = g_object_ref (self);
2046     data->position = gst_player_get_position (self);
2047     gst_player_signal_dispatcher_dispatch (self->signal_dispatcher, self,
2048         seek_done_dispatch, data, (GDestroyNotify) seek_done_signal_data_free);
2049   }
2050 }
2051 
2052 static void
state_changed_cb(G_GNUC_UNUSED GstBus * bus,GstMessage * msg,gpointer user_data)2053 state_changed_cb (G_GNUC_UNUSED GstBus * bus, GstMessage * msg,
2054     gpointer user_data)
2055 {
2056   GstPlayer *self = GST_PLAYER (user_data);
2057   GstState old_state, new_state, pending_state;
2058 
2059   gst_message_parse_state_changed (msg, &old_state, &new_state, &pending_state);
2060 
2061   if (GST_MESSAGE_SRC (msg) == GST_OBJECT (self->playbin)) {
2062     gchar *transition_name;
2063 
2064     GST_DEBUG_OBJECT (self, "Changed state old: %s new: %s pending: %s",
2065         gst_element_state_get_name (old_state),
2066         gst_element_state_get_name (new_state),
2067         gst_element_state_get_name (pending_state));
2068 
2069     transition_name = g_strdup_printf ("%s_%s",
2070         gst_element_state_get_name (old_state),
2071         gst_element_state_get_name (new_state));
2072     dump_dot_file (self, transition_name);
2073     g_free (transition_name);
2074 
2075     self->current_state = new_state;
2076 
2077     if (old_state == GST_STATE_READY && new_state == GST_STATE_PAUSED
2078         && pending_state == GST_STATE_VOID_PENDING) {
2079       GstElement *video_sink;
2080       GstPad *video_sink_pad;
2081       gint64 duration = -1;
2082 
2083       GST_DEBUG_OBJECT (self, "Initial PAUSED - pre-rolled");
2084 
2085       g_mutex_lock (&self->lock);
2086       if (self->media_info)
2087         g_object_unref (self->media_info);
2088       self->media_info = gst_player_media_info_create (self);
2089       g_mutex_unlock (&self->lock);
2090       emit_media_info_updated_signal (self);
2091 
2092       g_object_get (self->playbin, "video-sink", &video_sink, NULL);
2093 
2094       if (video_sink) {
2095         video_sink_pad = gst_element_get_static_pad (video_sink, "sink");
2096 
2097         if (video_sink_pad) {
2098           g_signal_connect (video_sink_pad, "notify::caps",
2099               (GCallback) notify_caps_cb, self);
2100           gst_object_unref (video_sink_pad);
2101         }
2102         gst_object_unref (video_sink);
2103       }
2104 
2105       check_video_dimensions_changed (self);
2106       if (gst_element_query_duration (self->playbin, GST_FORMAT_TIME,
2107               &duration)) {
2108         emit_duration_changed (self, duration);
2109       } else {
2110         self->cached_duration = GST_CLOCK_TIME_NONE;
2111       }
2112     }
2113 
2114     if (new_state == GST_STATE_PAUSED
2115         && pending_state == GST_STATE_VOID_PENDING) {
2116       remove_tick_source (self);
2117 
2118       g_mutex_lock (&self->lock);
2119       if (self->seek_pending) {
2120         self->seek_pending = FALSE;
2121 
2122         if (!self->media_info->seekable) {
2123           GST_DEBUG_OBJECT (self, "Media is not seekable");
2124           remove_seek_source (self);
2125           self->seek_position = GST_CLOCK_TIME_NONE;
2126           self->last_seek_time = GST_CLOCK_TIME_NONE;
2127         } else if (self->seek_source) {
2128           GST_DEBUG_OBJECT (self, "Seek finished but new seek is pending");
2129           gst_player_seek_internal_locked (self);
2130         } else {
2131           GST_DEBUG_OBJECT (self, "Seek finished");
2132           emit_seek_done (self);
2133         }
2134       }
2135 
2136       if (self->seek_position != GST_CLOCK_TIME_NONE) {
2137         GST_DEBUG_OBJECT (self, "Seeking now that we reached PAUSED state");
2138         gst_player_seek_internal_locked (self);
2139         g_mutex_unlock (&self->lock);
2140       } else if (!self->seek_pending) {
2141         g_mutex_unlock (&self->lock);
2142 
2143         tick_cb (self);
2144 
2145         if (self->target_state >= GST_STATE_PLAYING && self->buffering == 100) {
2146           GstStateChangeReturn state_ret;
2147 
2148           state_ret = gst_element_set_state (self->playbin, GST_STATE_PLAYING);
2149           if (state_ret == GST_STATE_CHANGE_FAILURE)
2150             emit_error (self, g_error_new (GST_PLAYER_ERROR,
2151                     GST_PLAYER_ERROR_FAILED, "Failed to play"));
2152         } else if (self->buffering == 100) {
2153           change_state (self, GST_PLAYER_STATE_PAUSED);
2154         }
2155         #ifdef OHOS_OPT_COMPAT
2156         // ohos.opt.compat.0010
2157         else {
2158           self->isStateChange = TRUE;
2159         }
2160         #endif
2161       } else {
2162         g_mutex_unlock (&self->lock);
2163       }
2164     } else if (new_state == GST_STATE_PLAYING
2165         && pending_state == GST_STATE_VOID_PENDING) {
2166 
2167       /* If no seek is currently pending, add the tick source. This can happen
2168        * if we seeked already but the state-change message was still queued up */
2169       if (!self->seek_pending) {
2170         add_tick_source (self);
2171         change_state (self, GST_PLAYER_STATE_PLAYING);
2172       }
2173     } else if (new_state == GST_STATE_READY && old_state > GST_STATE_READY) {
2174       change_state (self, GST_PLAYER_STATE_STOPPED);
2175     } else {
2176       /* Otherwise we neither reached PLAYING nor PAUSED, so must
2177        * wait for something to happen... i.e. are BUFFERING now */
2178       change_state (self, GST_PLAYER_STATE_BUFFERING);
2179     }
2180   }
2181 }
2182 
2183 static void
duration_changed_cb(G_GNUC_UNUSED GstBus * bus,G_GNUC_UNUSED GstMessage * msg,gpointer user_data)2184 duration_changed_cb (G_GNUC_UNUSED GstBus * bus, G_GNUC_UNUSED GstMessage * msg,
2185     gpointer user_data)
2186 {
2187   GstPlayer *self = GST_PLAYER (user_data);
2188   gint64 duration = GST_CLOCK_TIME_NONE;
2189 
2190   if (gst_element_query_duration (self->playbin, GST_FORMAT_TIME, &duration)) {
2191     emit_duration_changed (self, duration);
2192   }
2193 }
2194 
2195 static void
latency_cb(G_GNUC_UNUSED GstBus * bus,G_GNUC_UNUSED GstMessage * msg,gpointer user_data)2196 latency_cb (G_GNUC_UNUSED GstBus * bus, G_GNUC_UNUSED GstMessage * msg,
2197     gpointer user_data)
2198 {
2199   GstPlayer *self = GST_PLAYER (user_data);
2200 
2201   GST_DEBUG_OBJECT (self, "Latency changed");
2202 
2203   gst_bin_recalculate_latency (GST_BIN (self->playbin));
2204 }
2205 
2206 static void
request_state_cb(G_GNUC_UNUSED GstBus * bus,GstMessage * msg,gpointer user_data)2207 request_state_cb (G_GNUC_UNUSED GstBus * bus, GstMessage * msg,
2208     gpointer user_data)
2209 {
2210   GstPlayer *self = GST_PLAYER (user_data);
2211   GstState state;
2212   GstStateChangeReturn state_ret;
2213 
2214   gst_message_parse_request_state (msg, &state);
2215 
2216   GST_DEBUG_OBJECT (self, "State %s requested",
2217       gst_element_state_get_name (state));
2218 
2219   self->target_state = state;
2220   state_ret = gst_element_set_state (self->playbin, state);
2221   if (state_ret == GST_STATE_CHANGE_FAILURE)
2222     emit_error (self, g_error_new (GST_PLAYER_ERROR, GST_PLAYER_ERROR_FAILED,
2223             "Failed to change to requested state %s",
2224             gst_element_state_get_name (state)));
2225 }
2226 
2227 static void
media_info_update(GstPlayer * self,GstPlayerMediaInfo * info)2228 media_info_update (GstPlayer * self, GstPlayerMediaInfo * info)
2229 {
2230   g_free (info->title);
2231   info->title = get_from_tags (self, info, get_title);
2232 
2233   g_free (info->container);
2234   info->container = get_from_tags (self, info, get_container_format);
2235 
2236   if (info->image_sample)
2237     gst_sample_unref (info->image_sample);
2238   info->image_sample = get_from_tags (self, info, get_cover_sample);
2239 
2240   GST_DEBUG_OBJECT (self, "title: %s, container: %s "
2241       "image_sample: %p", info->title, info->container, info->image_sample);
2242 }
2243 
2244 static void
tags_cb(G_GNUC_UNUSED GstBus * bus,GstMessage * msg,gpointer user_data)2245 tags_cb (G_GNUC_UNUSED GstBus * bus, GstMessage * msg, gpointer user_data)
2246 {
2247   GstPlayer *self = GST_PLAYER (user_data);
2248   GstTagList *tags = NULL;
2249 
2250   gst_message_parse_tag (msg, &tags);
2251 
2252   GST_DEBUG_OBJECT (self, "received %s tags",
2253       gst_tag_list_get_scope (tags) ==
2254       GST_TAG_SCOPE_GLOBAL ? "global" : "stream");
2255 
2256   if (gst_tag_list_get_scope (tags) == GST_TAG_SCOPE_GLOBAL) {
2257     g_mutex_lock (&self->lock);
2258     if (self->media_info) {
2259       if (self->media_info->tags)
2260         gst_tag_list_unref (self->media_info->tags);
2261       self->media_info->tags = gst_tag_list_ref (tags);
2262       media_info_update (self, self->media_info);
2263       g_mutex_unlock (&self->lock);
2264       emit_media_info_updated_signal (self);
2265     } else {
2266       if (self->global_tags)
2267         gst_tag_list_unref (self->global_tags);
2268       self->global_tags = gst_tag_list_ref (tags);
2269       g_mutex_unlock (&self->lock);
2270     }
2271   }
2272 
2273   gst_tag_list_unref (tags);
2274 }
2275 
2276 static void
element_cb(G_GNUC_UNUSED GstBus * bus,GstMessage * msg,gpointer user_data)2277 element_cb (G_GNUC_UNUSED GstBus * bus, GstMessage * msg, gpointer user_data)
2278 {
2279   GstPlayer *self = GST_PLAYER (user_data);
2280   const GstStructure *s;
2281 
2282   s = gst_message_get_structure (msg);
2283   if (gst_structure_has_name (s, "redirect")) {
2284     const gchar *new_location;
2285 
2286     new_location = gst_structure_get_string (s, "new-location");
2287     if (!new_location) {
2288       const GValue *locations_list, *location_val;
2289       guint i, size;
2290 
2291       locations_list = gst_structure_get_value (s, "locations");
2292       size = gst_value_list_get_size (locations_list);
2293       for (i = 0; i < size; ++i) {
2294         const GstStructure *location_s;
2295 
2296         location_val = gst_value_list_get_value (locations_list, i);
2297         if (!GST_VALUE_HOLDS_STRUCTURE (location_val))
2298           continue;
2299 
2300         location_s = (const GstStructure *) g_value_get_boxed (location_val);
2301         if (!gst_structure_has_name (location_s, "redirect"))
2302           continue;
2303 
2304         new_location = gst_structure_get_string (location_s, "new-location");
2305         if (new_location)
2306           break;
2307       }
2308     }
2309 
2310     if (new_location) {
2311       GstState target_state;
2312 
2313       GST_DEBUG_OBJECT (self, "Redirect to '%s'", new_location);
2314 
2315       /* Remember target state and restore after setting the URI */
2316       target_state = self->target_state;
2317 
2318       gst_player_stop_internal (self, TRUE);
2319 
2320       g_mutex_lock (&self->lock);
2321       g_free (self->redirect_uri);
2322       self->redirect_uri = g_strdup (new_location);
2323       g_object_set (self->playbin, "uri", self->redirect_uri, NULL);
2324       g_mutex_unlock (&self->lock);
2325 
2326       if (target_state == GST_STATE_PAUSED)
2327         gst_player_pause_internal (self);
2328       else if (target_state == GST_STATE_PLAYING)
2329         gst_player_play_internal (self);
2330     }
2331   }
2332 #ifdef OHOS_EXT_FUNC
2333   // ohos.ext.func.0012
2334   else if (gst_structure_has_name (s, "message-buffering-time")) {
2335     buffering_time_cb(msg, user_data);
2336   } else if (gst_structure_has_name (s, "message-mq-num-use-buffering")) {
2337     mq_num_use_buffering_cb(msg, user_data);
2338   } else if (gst_structure_has_name (s, "resolution-changed")) {
2339     resulution_changed_cb(msg, user_data);
2340   }
2341 #endif
2342 }
2343 
2344 /* Must be called with lock */
2345 static gboolean
update_stream_collection(GstPlayer * self,GstStreamCollection * collection)2346 update_stream_collection (GstPlayer * self, GstStreamCollection * collection)
2347 {
2348   if (self->collection && self->collection == collection)
2349     return FALSE;
2350 
2351   if (self->collection && self->stream_notify_id)
2352     g_signal_handler_disconnect (self->collection, self->stream_notify_id);
2353 
2354   gst_object_replace ((GstObject **) & self->collection,
2355       (GstObject *) collection);
2356   if (self->media_info) {
2357     gst_object_unref (self->media_info);
2358     self->media_info = gst_player_media_info_create (self);
2359   }
2360 
2361   self->stream_notify_id =
2362       g_signal_connect (self->collection, "stream-notify",
2363       G_CALLBACK (stream_notify_cb), self);
2364 
2365   return TRUE;
2366 }
2367 
2368 static void
stream_collection_cb(G_GNUC_UNUSED GstBus * bus,GstMessage * msg,gpointer user_data)2369 stream_collection_cb (G_GNUC_UNUSED GstBus * bus, GstMessage * msg,
2370     gpointer user_data)
2371 {
2372   GstPlayer *self = GST_PLAYER (user_data);
2373   GstStreamCollection *collection = NULL;
2374   gboolean updated = FALSE;
2375 
2376   gst_message_parse_stream_collection (msg, &collection);
2377 
2378   if (!collection)
2379     return;
2380 
2381   g_mutex_lock (&self->lock);
2382   updated = update_stream_collection (self, collection);
2383   gst_object_unref (collection);
2384   g_mutex_unlock (&self->lock);
2385 
2386   if (self->media_info && updated)
2387     emit_media_info_updated_signal (self);
2388 }
2389 
2390 static void
streams_selected_cb(G_GNUC_UNUSED GstBus * bus,GstMessage * msg,gpointer user_data)2391 streams_selected_cb (G_GNUC_UNUSED GstBus * bus, GstMessage * msg,
2392     gpointer user_data)
2393 {
2394   GstPlayer *self = GST_PLAYER (user_data);
2395   GstStreamCollection *collection = NULL;
2396   gboolean updated = FALSE;
2397   guint i, len;
2398 
2399   gst_message_parse_streams_selected (msg, &collection);
2400 
2401   if (!collection)
2402     return;
2403 
2404   g_mutex_lock (&self->lock);
2405   updated = update_stream_collection (self, collection);
2406   gst_object_unref (collection);
2407 
2408   g_free (self->video_sid);
2409   g_free (self->audio_sid);
2410   g_free (self->subtitle_sid);
2411   self->video_sid = NULL;
2412   self->audio_sid = NULL;
2413   self->subtitle_sid = NULL;
2414 
2415   len = gst_message_streams_selected_get_size (msg);
2416   for (i = 0; i < len; i++) {
2417     GstStream *stream;
2418     GstStreamType stream_type;
2419     const gchar *stream_id;
2420     gchar **current_sid;
2421     stream = gst_message_streams_selected_get_stream (msg, i);
2422     stream_type = gst_stream_get_stream_type (stream);
2423     stream_id = gst_stream_get_stream_id (stream);
2424     if (stream_type & GST_STREAM_TYPE_AUDIO)
2425       current_sid = &self->audio_sid;
2426     else if (stream_type & GST_STREAM_TYPE_VIDEO)
2427       current_sid = &self->video_sid;
2428     else if (stream_type & GST_STREAM_TYPE_TEXT)
2429       current_sid = &self->subtitle_sid;
2430     else {
2431       GST_WARNING_OBJECT (self,
2432           "Unknown stream-id %s with type 0x%x", stream_id, stream_type);
2433       continue;
2434     }
2435 
2436     if (G_UNLIKELY (*current_sid)) {
2437       GST_FIXME_OBJECT (self,
2438           "Multiple streams are selected for type %s, choose the first one",
2439           gst_stream_type_get_name (stream_type));
2440       continue;
2441     }
2442 
2443     *current_sid = g_strdup (stream_id);
2444   }
2445   g_mutex_unlock (&self->lock);
2446 
2447   if (self->media_info && updated)
2448     emit_media_info_updated_signal (self);
2449 }
2450 
2451 static void
player_set_flag(GstPlayer * self,gint pos)2452 player_set_flag (GstPlayer * self, gint pos)
2453 {
2454   gint flags;
2455 
2456   g_object_get (self->playbin, "flags", &flags, NULL);
2457   flags |= pos;
2458   g_object_set (self->playbin, "flags", flags, NULL);
2459 
2460   GST_DEBUG_OBJECT (self, "setting flags=%#x", flags);
2461 }
2462 
2463 static void
player_clear_flag(GstPlayer * self,gint pos)2464 player_clear_flag (GstPlayer * self, gint pos)
2465 {
2466   gint flags;
2467 
2468   g_object_get (self->playbin, "flags", &flags, NULL);
2469   flags &= ~pos;
2470   g_object_set (self->playbin, "flags", flags, NULL);
2471 
2472   GST_DEBUG_OBJECT (self, "setting flags=%#x", flags);
2473 }
2474 
2475 typedef struct
2476 {
2477   GstPlayer *player;
2478   GstPlayerMediaInfo *info;
2479 } MediaInfoUpdatedSignalData;
2480 
2481 static void
media_info_updated_dispatch(gpointer user_data)2482 media_info_updated_dispatch (gpointer user_data)
2483 {
2484   MediaInfoUpdatedSignalData *data = user_data;
2485 
2486   if (data->player->inhibit_sigs)
2487     return;
2488 
2489   if (data->player->target_state >= GST_STATE_PAUSED) {
2490     g_signal_emit (data->player, signals[SIGNAL_MEDIA_INFO_UPDATED], 0,
2491         data->info);
2492   }
2493 }
2494 
2495 static void
free_media_info_updated_signal_data(MediaInfoUpdatedSignalData * data)2496 free_media_info_updated_signal_data (MediaInfoUpdatedSignalData * data)
2497 {
2498   g_object_unref (data->player);
2499   g_object_unref (data->info);
2500   g_free (data);
2501 }
2502 
2503 /*
2504  * emit_media_info_updated_signal:
2505  *
2506  * create a new copy of self->media_info object and emits the newly created
2507  * copy to user application. The newly created media_info will be unref'ed
2508  * as part of signal finalize method.
2509  */
2510 static void
emit_media_info_updated_signal(GstPlayer * self)2511 emit_media_info_updated_signal (GstPlayer * self)
2512 {
2513   MediaInfoUpdatedSignalData *data = g_new (MediaInfoUpdatedSignalData, 1);
2514   data->player = g_object_ref (self);
2515   g_mutex_lock (&self->lock);
2516   data->info = gst_player_media_info_copy (self->media_info);
2517   g_mutex_unlock (&self->lock);
2518 
2519   gst_player_signal_dispatcher_dispatch (self->signal_dispatcher, self,
2520       media_info_updated_dispatch, data,
2521       (GDestroyNotify) free_media_info_updated_signal_data);
2522 }
2523 
2524 static GstCaps *
get_caps(GstPlayer * self,gint stream_index,GType type)2525 get_caps (GstPlayer * self, gint stream_index, GType type)
2526 {
2527   GstPad *pad = NULL;
2528   GstCaps *caps = NULL;
2529 
2530   if (type == GST_TYPE_PLAYER_VIDEO_INFO)
2531     g_signal_emit_by_name (G_OBJECT (self->playbin),
2532         "get-video-pad", stream_index, &pad);
2533   else if (type == GST_TYPE_PLAYER_AUDIO_INFO)
2534     g_signal_emit_by_name (G_OBJECT (self->playbin),
2535         "get-audio-pad", stream_index, &pad);
2536   else
2537     g_signal_emit_by_name (G_OBJECT (self->playbin),
2538         "get-text-pad", stream_index, &pad);
2539 
2540   if (pad) {
2541     caps = gst_pad_get_current_caps (pad);
2542     gst_object_unref (pad);
2543   }
2544 
2545   return caps;
2546 }
2547 
2548 static void
gst_player_subtitle_info_update(GstPlayer * self,GstPlayerStreamInfo * stream_info)2549 gst_player_subtitle_info_update (GstPlayer * self,
2550     GstPlayerStreamInfo * stream_info)
2551 {
2552   GstPlayerSubtitleInfo *info = (GstPlayerSubtitleInfo *) stream_info;
2553 
2554   if (stream_info->tags) {
2555 
2556     /* free the old language info */
2557     g_free (info->language);
2558     info->language = NULL;
2559 
2560     /* First try to get the language full name from tag, if name is not
2561      * available then try language code. If we find the language code
2562      * then use gstreamer api to translate code to full name.
2563      */
2564     gst_tag_list_get_string (stream_info->tags, GST_TAG_LANGUAGE_NAME,
2565         &info->language);
2566     if (!info->language) {
2567       gchar *lang_code = NULL;
2568 
2569       gst_tag_list_get_string (stream_info->tags, GST_TAG_LANGUAGE_CODE,
2570           &lang_code);
2571       if (lang_code) {
2572         info->language = g_strdup (gst_tag_get_language_name (lang_code));
2573         g_free (lang_code);
2574       }
2575     }
2576 
2577     /* If we are still failed to find language name then check if external
2578      * subtitle is loaded and compare the stream index between current sub
2579      * stream index with our stream index and if matches then declare it as
2580      * external subtitle and use the filename.
2581      */
2582     if (!info->language) {
2583       gint text_index = -1;
2584       gchar *suburi = NULL;
2585 
2586       g_object_get (G_OBJECT (self->playbin), "current-suburi", &suburi, NULL);
2587       if (suburi) {
2588         if (self->use_playbin3) {
2589           if (g_str_equal (self->subtitle_sid, stream_info->stream_id))
2590             info->language = g_path_get_basename (suburi);
2591         } else {
2592           g_object_get (G_OBJECT (self->playbin), "current-text", &text_index,
2593               NULL);
2594           if (text_index == gst_player_stream_info_get_index (stream_info))
2595             info->language = g_path_get_basename (suburi);
2596         }
2597         g_free (suburi);
2598       }
2599     }
2600 
2601   } else {
2602     g_free (info->language);
2603     info->language = NULL;
2604   }
2605 
2606   GST_DEBUG_OBJECT (self, "language=%s", info->language);
2607 }
2608 
2609 static void
gst_player_video_info_update(GstPlayer * self,GstPlayerStreamInfo * stream_info)2610 gst_player_video_info_update (GstPlayer * self,
2611     GstPlayerStreamInfo * stream_info)
2612 {
2613   GstPlayerVideoInfo *info = (GstPlayerVideoInfo *) stream_info;
2614 
2615   if (stream_info->caps) {
2616     GstStructure *s;
2617 
2618     s = gst_caps_get_structure (stream_info->caps, 0);
2619     if (s) {
2620       gint width, height;
2621       gint fps_n, fps_d;
2622       gint par_n, par_d;
2623 
2624       if (gst_structure_get_int (s, "width", &width))
2625         info->width = width;
2626       else
2627         info->width = -1;
2628 
2629       if (gst_structure_get_int (s, "height", &height))
2630         info->height = height;
2631       else
2632         info->height = -1;
2633 
2634       if (gst_structure_get_fraction (s, "framerate", &fps_n, &fps_d)) {
2635         info->framerate_num = fps_n;
2636         info->framerate_denom = fps_d;
2637       } else {
2638         info->framerate_num = 0;
2639         info->framerate_denom = 1;
2640       }
2641 
2642 
2643       if (gst_structure_get_fraction (s, "pixel-aspect-ratio", &par_n, &par_d)) {
2644         info->par_num = par_n;
2645         info->par_denom = par_d;
2646       } else {
2647         info->par_num = 1;
2648         info->par_denom = 1;
2649       }
2650     }
2651   } else {
2652     info->width = info->height = -1;
2653     info->par_num = info->par_denom = 1;
2654     info->framerate_num = 0;
2655     info->framerate_denom = 1;
2656   }
2657 
2658   if (stream_info->tags) {
2659     guint bitrate, max_bitrate;
2660 
2661     if (gst_tag_list_get_uint (stream_info->tags, GST_TAG_BITRATE, &bitrate))
2662       info->bitrate = bitrate;
2663     else
2664       info->bitrate = -1;
2665 
2666     if (gst_tag_list_get_uint (stream_info->tags, GST_TAG_MAXIMUM_BITRATE,
2667             &max_bitrate) || gst_tag_list_get_uint (stream_info->tags,
2668             GST_TAG_NOMINAL_BITRATE, &max_bitrate))
2669       info->max_bitrate = max_bitrate;
2670     else
2671       info->max_bitrate = -1;
2672   } else {
2673     info->bitrate = info->max_bitrate = -1;
2674   }
2675 
2676   GST_DEBUG_OBJECT (self, "width=%d height=%d fps=%.2f par=%d:%d "
2677       "bitrate=%d max_bitrate=%d", info->width, info->height,
2678       (gdouble) info->framerate_num / info->framerate_denom,
2679       info->par_num, info->par_denom, info->bitrate, info->max_bitrate);
2680 }
2681 
2682 static void
gst_player_audio_info_update(GstPlayer * self,GstPlayerStreamInfo * stream_info)2683 gst_player_audio_info_update (GstPlayer * self,
2684     GstPlayerStreamInfo * stream_info)
2685 {
2686   GstPlayerAudioInfo *info = (GstPlayerAudioInfo *) stream_info;
2687 
2688   if (stream_info->caps) {
2689     GstStructure *s;
2690 
2691     s = gst_caps_get_structure (stream_info->caps, 0);
2692     if (s) {
2693       gint rate, channels;
2694 
2695       if (gst_structure_get_int (s, "rate", &rate))
2696         info->sample_rate = rate;
2697       else
2698         info->sample_rate = -1;
2699 
2700       if (gst_structure_get_int (s, "channels", &channels))
2701         info->channels = channels;
2702       else
2703         info->channels = 0;
2704     }
2705   } else {
2706     info->sample_rate = -1;
2707     info->channels = 0;
2708   }
2709 
2710   if (stream_info->tags) {
2711     guint bitrate, max_bitrate;
2712 
2713     if (gst_tag_list_get_uint (stream_info->tags, GST_TAG_BITRATE, &bitrate))
2714       info->bitrate = bitrate;
2715     else
2716       info->bitrate = -1;
2717 
2718     if (gst_tag_list_get_uint (stream_info->tags, GST_TAG_MAXIMUM_BITRATE,
2719             &max_bitrate) || gst_tag_list_get_uint (stream_info->tags,
2720             GST_TAG_NOMINAL_BITRATE, &max_bitrate))
2721       info->max_bitrate = max_bitrate;
2722     else
2723       info->max_bitrate = -1;
2724 
2725     /* if we have old language the free it */
2726     g_free (info->language);
2727     info->language = NULL;
2728 
2729     /* First try to get the language full name from tag, if name is not
2730      * available then try language code. If we find the language code
2731      * then use gstreamer api to translate code to full name.
2732      */
2733     gst_tag_list_get_string (stream_info->tags, GST_TAG_LANGUAGE_NAME,
2734         &info->language);
2735     if (!info->language) {
2736       gchar *lang_code = NULL;
2737 
2738       gst_tag_list_get_string (stream_info->tags, GST_TAG_LANGUAGE_CODE,
2739           &lang_code);
2740       if (lang_code) {
2741         info->language = g_strdup (gst_tag_get_language_name (lang_code));
2742         g_free (lang_code);
2743       }
2744     }
2745   } else {
2746     g_free (info->language);
2747     info->language = NULL;
2748     info->max_bitrate = info->bitrate = -1;
2749   }
2750 
2751   GST_DEBUG_OBJECT (self, "language=%s rate=%d channels=%d bitrate=%d "
2752       "max_bitrate=%d", info->language, info->sample_rate, info->channels,
2753       info->bitrate, info->max_bitrate);
2754 }
2755 
2756 static GstPlayerStreamInfo *
gst_player_stream_info_find(GstPlayerMediaInfo * media_info,GType type,gint stream_index)2757 gst_player_stream_info_find (GstPlayerMediaInfo * media_info,
2758     GType type, gint stream_index)
2759 {
2760   GList *list, *l;
2761   GstPlayerStreamInfo *info = NULL;
2762 
2763   if (!media_info)
2764     return NULL;
2765 
2766   list = gst_player_media_info_get_stream_list (media_info);
2767   for (l = list; l != NULL; l = l->next) {
2768     info = (GstPlayerStreamInfo *) l->data;
2769     if ((G_OBJECT_TYPE (info) == type) && (info->stream_index == stream_index)) {
2770       return info;
2771     }
2772   }
2773 
2774   return NULL;
2775 }
2776 
2777 static GstPlayerStreamInfo *
gst_player_stream_info_find_from_stream_id(GstPlayerMediaInfo * media_info,const gchar * stream_id)2778 gst_player_stream_info_find_from_stream_id (GstPlayerMediaInfo * media_info,
2779     const gchar * stream_id)
2780 {
2781   GList *list, *l;
2782   GstPlayerStreamInfo *info = NULL;
2783 
2784   if (!media_info)
2785     return NULL;
2786 
2787   list = gst_player_media_info_get_stream_list (media_info);
2788   for (l = list; l != NULL; l = l->next) {
2789     info = (GstPlayerStreamInfo *) l->data;
2790     if (g_str_equal (info->stream_id, stream_id)) {
2791       return info;
2792     }
2793   }
2794 
2795   return NULL;
2796 }
2797 
2798 static gboolean
is_track_enabled(GstPlayer * self,gint pos)2799 is_track_enabled (GstPlayer * self, gint pos)
2800 {
2801   gint flags;
2802 
2803   g_object_get (G_OBJECT (self->playbin), "flags", &flags, NULL);
2804 
2805   if ((flags & pos))
2806     return TRUE;
2807 
2808   return FALSE;
2809 }
2810 
2811 static GstPlayerStreamInfo *
gst_player_stream_info_get_current(GstPlayer * self,const gchar * prop,GType type)2812 gst_player_stream_info_get_current (GstPlayer * self, const gchar * prop,
2813     GType type)
2814 {
2815   gint current;
2816   GstPlayerStreamInfo *info;
2817 
2818   if (!self->media_info)
2819     return NULL;
2820 
2821   g_object_get (G_OBJECT (self->playbin), prop, &current, NULL);
2822   g_mutex_lock (&self->lock);
2823   info = gst_player_stream_info_find (self->media_info, type, current);
2824   if (info)
2825     info = gst_player_stream_info_copy (info);
2826   g_mutex_unlock (&self->lock);
2827 
2828   return info;
2829 }
2830 
2831 static GstPlayerStreamInfo *
gst_player_stream_info_get_current_from_stream_id(GstPlayer * self,const gchar * stream_id,GType type)2832 gst_player_stream_info_get_current_from_stream_id (GstPlayer * self,
2833     const gchar * stream_id, GType type)
2834 {
2835   GstPlayerStreamInfo *info;
2836 
2837   if (!self->media_info || !stream_id)
2838     return NULL;
2839 
2840   g_mutex_lock (&self->lock);
2841   info =
2842       gst_player_stream_info_find_from_stream_id (self->media_info, stream_id);
2843   if (info && G_OBJECT_TYPE (info) == type)
2844     info = gst_player_stream_info_copy (info);
2845   else
2846     info = NULL;
2847   g_mutex_unlock (&self->lock);
2848 
2849   return info;
2850 }
2851 
2852 static void
stream_notify_cb(GstStreamCollection * collection,GstStream * stream,GParamSpec * pspec,GstPlayer * self)2853 stream_notify_cb (GstStreamCollection * collection, GstStream * stream,
2854     GParamSpec * pspec, GstPlayer * self)
2855 {
2856   GstPlayerStreamInfo *info;
2857   const gchar *stream_id;
2858   gboolean emit_signal = FALSE;
2859 
2860   if (!self->media_info)
2861     return;
2862 
2863   if (G_PARAM_SPEC_VALUE_TYPE (pspec) != GST_TYPE_CAPS &&
2864       G_PARAM_SPEC_VALUE_TYPE (pspec) != GST_TYPE_TAG_LIST)
2865     return;
2866 
2867   stream_id = gst_stream_get_stream_id (stream);
2868   g_mutex_lock (&self->lock);
2869   info =
2870       gst_player_stream_info_find_from_stream_id (self->media_info, stream_id);
2871   if (info) {
2872     gst_player_stream_info_update_from_stream (self, info, stream);
2873     emit_signal = TRUE;
2874   }
2875   g_mutex_unlock (&self->lock);
2876 
2877   if (emit_signal)
2878     emit_media_info_updated_signal (self);
2879 }
2880 
2881 static void
gst_player_stream_info_update(GstPlayer * self,GstPlayerStreamInfo * s)2882 gst_player_stream_info_update (GstPlayer * self, GstPlayerStreamInfo * s)
2883 {
2884   if (GST_IS_PLAYER_VIDEO_INFO (s))
2885     gst_player_video_info_update (self, s);
2886   else if (GST_IS_PLAYER_AUDIO_INFO (s))
2887     gst_player_audio_info_update (self, s);
2888   else
2889     gst_player_subtitle_info_update (self, s);
2890 }
2891 
2892 static gchar *
stream_info_get_codec(GstPlayerStreamInfo * s)2893 stream_info_get_codec (GstPlayerStreamInfo * s)
2894 {
2895   const gchar *type;
2896   GstTagList *tags;
2897   gchar *codec = NULL;
2898 
2899   if (GST_IS_PLAYER_VIDEO_INFO (s))
2900     type = GST_TAG_VIDEO_CODEC;
2901   else if (GST_IS_PLAYER_AUDIO_INFO (s))
2902     type = GST_TAG_AUDIO_CODEC;
2903   else
2904     type = GST_TAG_SUBTITLE_CODEC;
2905 
2906   tags = gst_player_stream_info_get_tags (s);
2907   if (tags) {
2908     gst_tag_list_get_string (tags, type, &codec);
2909     if (!codec)
2910       gst_tag_list_get_string (tags, GST_TAG_CODEC, &codec);
2911   }
2912 
2913   if (!codec) {
2914     GstCaps *caps;
2915     caps = gst_player_stream_info_get_caps (s);
2916     if (caps) {
2917       codec = gst_pb_utils_get_codec_description (caps);
2918     }
2919   }
2920 
2921   return codec;
2922 }
2923 
2924 static void
gst_player_stream_info_update_tags_and_caps(GstPlayer * self,GstPlayerStreamInfo * s)2925 gst_player_stream_info_update_tags_and_caps (GstPlayer * self,
2926     GstPlayerStreamInfo * s)
2927 {
2928   GstTagList *tags;
2929   gint stream_index;
2930 
2931   stream_index = gst_player_stream_info_get_index (s);
2932 
2933   if (GST_IS_PLAYER_VIDEO_INFO (s))
2934     g_signal_emit_by_name (self->playbin, "get-video-tags",
2935         stream_index, &tags);
2936   else if (GST_IS_PLAYER_AUDIO_INFO (s))
2937     g_signal_emit_by_name (self->playbin, "get-audio-tags",
2938         stream_index, &tags);
2939   else
2940     g_signal_emit_by_name (self->playbin, "get-text-tags", stream_index, &tags);
2941 
2942   if (s->tags)
2943     gst_tag_list_unref (s->tags);
2944   s->tags = tags;
2945 
2946   if (s->caps)
2947     gst_caps_unref (s->caps);
2948   s->caps = get_caps (self, stream_index, G_OBJECT_TYPE (s));
2949 
2950   g_free (s->codec);
2951   s->codec = stream_info_get_codec (s);
2952 
2953   GST_DEBUG_OBJECT (self, "%s index: %d tags: %p caps: %p",
2954       gst_player_stream_info_get_stream_type (s), stream_index,
2955       s->tags, s->caps);
2956 
2957   gst_player_stream_info_update (self, s);
2958 }
2959 
2960 static void
gst_player_streams_info_create(GstPlayer * self,GstPlayerMediaInfo * media_info,const gchar * prop,GType type)2961 gst_player_streams_info_create (GstPlayer * self,
2962     GstPlayerMediaInfo * media_info, const gchar * prop, GType type)
2963 {
2964   gint i;
2965   gint total = -1;
2966   GstPlayerStreamInfo *s;
2967 
2968   if (!media_info)
2969     return;
2970 
2971   g_object_get (G_OBJECT (self->playbin), prop, &total, NULL);
2972 
2973   GST_DEBUG_OBJECT (self, "%s: %d", prop, total);
2974 
2975   for (i = 0; i < total; i++) {
2976     /* check if stream already exist in the list */
2977     s = gst_player_stream_info_find (media_info, type, i);
2978 
2979     if (!s) {
2980       /* create a new stream info instance */
2981       s = gst_player_stream_info_new (i, type);
2982 
2983       /* add the object in stream list */
2984       media_info->stream_list = g_list_append (media_info->stream_list, s);
2985 
2986       /* based on type, add the object in its corresponding stream_ list */
2987       if (GST_IS_PLAYER_AUDIO_INFO (s))
2988         media_info->audio_stream_list = g_list_append
2989             (media_info->audio_stream_list, s);
2990       else if (GST_IS_PLAYER_VIDEO_INFO (s))
2991         media_info->video_stream_list = g_list_append
2992             (media_info->video_stream_list, s);
2993       else
2994         media_info->subtitle_stream_list = g_list_append
2995             (media_info->subtitle_stream_list, s);
2996 
2997       GST_DEBUG_OBJECT (self, "create %s stream stream_index: %d",
2998           gst_player_stream_info_get_stream_type (s), i);
2999     }
3000 
3001     gst_player_stream_info_update_tags_and_caps (self, s);
3002   }
3003 }
3004 
3005 static void
gst_player_stream_info_update_from_stream(GstPlayer * self,GstPlayerStreamInfo * s,GstStream * stream)3006 gst_player_stream_info_update_from_stream (GstPlayer * self,
3007     GstPlayerStreamInfo * s, GstStream * stream)
3008 {
3009   if (s->tags)
3010     gst_tag_list_unref (s->tags);
3011   s->tags = gst_stream_get_tags (stream);
3012 
3013   if (s->caps)
3014     gst_caps_unref (s->caps);
3015   s->caps = gst_stream_get_caps (stream);
3016 
3017   g_free (s->codec);
3018   s->codec = stream_info_get_codec (s);
3019 
3020   GST_DEBUG_OBJECT (self, "%s index: %d tags: %p caps: %p",
3021       gst_player_stream_info_get_stream_type (s), s->stream_index,
3022       s->tags, s->caps);
3023 
3024   gst_player_stream_info_update (self, s);
3025 }
3026 
3027 static void
gst_player_streams_info_create_from_collection(GstPlayer * self,GstPlayerMediaInfo * media_info,GstStreamCollection * collection)3028 gst_player_streams_info_create_from_collection (GstPlayer * self,
3029     GstPlayerMediaInfo * media_info, GstStreamCollection * collection)
3030 {
3031   guint i;
3032   guint total;
3033   GstPlayerStreamInfo *s;
3034   guint n_audio = 0;
3035   guint n_video = 0;
3036   guint n_text = 0;
3037 
3038   if (!media_info || !collection)
3039     return;
3040 
3041   total = gst_stream_collection_get_size (collection);
3042 
3043   for (i = 0; i < total; i++) {
3044     GstStream *stream = gst_stream_collection_get_stream (collection, i);
3045     GstStreamType stream_type = gst_stream_get_stream_type (stream);
3046     const gchar *stream_id = gst_stream_get_stream_id (stream);
3047 
3048     if (stream_type & GST_STREAM_TYPE_AUDIO) {
3049       s = gst_player_stream_info_new (n_audio, GST_TYPE_PLAYER_AUDIO_INFO);
3050       n_audio++;
3051     } else if (stream_type & GST_STREAM_TYPE_VIDEO) {
3052       s = gst_player_stream_info_new (n_video, GST_TYPE_PLAYER_VIDEO_INFO);
3053       n_video++;
3054     } else if (stream_type & GST_STREAM_TYPE_TEXT) {
3055       s = gst_player_stream_info_new (n_text, GST_TYPE_PLAYER_SUBTITLE_INFO);
3056       n_text++;
3057     } else {
3058       GST_DEBUG_OBJECT (self, "Unknown type stream %d", i);
3059       continue;
3060     }
3061 
3062     s->stream_id = g_strdup (stream_id);
3063 
3064     /* add the object in stream list */
3065     media_info->stream_list = g_list_append (media_info->stream_list, s);
3066 
3067     /* based on type, add the object in its corresponding stream_ list */
3068     if (GST_IS_PLAYER_AUDIO_INFO (s))
3069       media_info->audio_stream_list = g_list_append
3070           (media_info->audio_stream_list, s);
3071     else if (GST_IS_PLAYER_VIDEO_INFO (s))
3072       media_info->video_stream_list = g_list_append
3073           (media_info->video_stream_list, s);
3074     else
3075       media_info->subtitle_stream_list = g_list_append
3076           (media_info->subtitle_stream_list, s);
3077 
3078     GST_DEBUG_OBJECT (self, "create %s stream stream_index: %d",
3079         gst_player_stream_info_get_stream_type (s), s->stream_index);
3080 
3081     gst_player_stream_info_update_from_stream (self, s, stream);
3082   }
3083 }
3084 
3085 static void
video_changed_cb(G_GNUC_UNUSED GObject * object,gpointer user_data)3086 video_changed_cb (G_GNUC_UNUSED GObject * object, gpointer user_data)
3087 {
3088   GstPlayer *self = GST_PLAYER (user_data);
3089 
3090   g_mutex_lock (&self->lock);
3091   gst_player_streams_info_create (self, self->media_info,
3092       "n-video", GST_TYPE_PLAYER_VIDEO_INFO);
3093   g_mutex_unlock (&self->lock);
3094 }
3095 
3096 static void
audio_changed_cb(G_GNUC_UNUSED GObject * object,gpointer user_data)3097 audio_changed_cb (G_GNUC_UNUSED GObject * object, gpointer user_data)
3098 {
3099   GstPlayer *self = GST_PLAYER (user_data);
3100 
3101   g_mutex_lock (&self->lock);
3102   gst_player_streams_info_create (self, self->media_info,
3103       "n-audio", GST_TYPE_PLAYER_AUDIO_INFO);
3104   g_mutex_unlock (&self->lock);
3105 }
3106 
3107 static void
subtitle_changed_cb(G_GNUC_UNUSED GObject * object,gpointer user_data)3108 subtitle_changed_cb (G_GNUC_UNUSED GObject * object, gpointer user_data)
3109 {
3110   GstPlayer *self = GST_PLAYER (user_data);
3111 
3112   g_mutex_lock (&self->lock);
3113   gst_player_streams_info_create (self, self->media_info,
3114       "n-text", GST_TYPE_PLAYER_SUBTITLE_INFO);
3115   g_mutex_unlock (&self->lock);
3116 }
3117 
3118 static void *
get_title(GstTagList * tags)3119 get_title (GstTagList * tags)
3120 {
3121   gchar *title = NULL;
3122 
3123   gst_tag_list_get_string (tags, GST_TAG_TITLE, &title);
3124   if (!title)
3125     gst_tag_list_get_string (tags, GST_TAG_TITLE_SORTNAME, &title);
3126 
3127   return title;
3128 }
3129 
3130 static void *
get_container_format(GstTagList * tags)3131 get_container_format (GstTagList * tags)
3132 {
3133   gchar *container = NULL;
3134 
3135   gst_tag_list_get_string (tags, GST_TAG_CONTAINER_FORMAT, &container);
3136 
3137   /* TODO: If container is not available then maybe consider
3138    * parsing caps or file extension to guess the container format.
3139    */
3140 
3141   return container;
3142 }
3143 
3144 static void *
get_from_tags(GstPlayer * self,GstPlayerMediaInfo * media_info,void * (* func)(GstTagList *))3145 get_from_tags (GstPlayer * self, GstPlayerMediaInfo * media_info,
3146     void *(*func) (GstTagList *))
3147 {
3148   GList *l;
3149   void *ret = NULL;
3150 
3151   if (media_info->tags) {
3152     ret = func (media_info->tags);
3153     if (ret)
3154       return ret;
3155   }
3156 
3157   /* if global tag does not exit then try video and audio streams */
3158   GST_DEBUG_OBJECT (self, "trying video tags");
3159   for (l = gst_player_media_info_get_video_streams (media_info); l != NULL;
3160       l = l->next) {
3161     GstTagList *tags;
3162 
3163     tags = gst_player_stream_info_get_tags ((GstPlayerStreamInfo *) l->data);
3164     if (tags)
3165       ret = func (tags);
3166 
3167     if (ret)
3168       return ret;
3169   }
3170 
3171   GST_DEBUG_OBJECT (self, "trying audio tags");
3172   for (l = gst_player_media_info_get_audio_streams (media_info); l != NULL;
3173       l = l->next) {
3174     GstTagList *tags;
3175 
3176     tags = gst_player_stream_info_get_tags ((GstPlayerStreamInfo *) l->data);
3177     if (tags)
3178       ret = func (tags);
3179 
3180     if (ret)
3181       return ret;
3182   }
3183 
3184   GST_DEBUG_OBJECT (self, "failed to get the information from tags");
3185   return NULL;
3186 }
3187 
3188 static void *
get_cover_sample(GstTagList * tags)3189 get_cover_sample (GstTagList * tags)
3190 {
3191   GstSample *cover_sample = NULL;
3192 
3193   gst_tag_list_get_sample (tags, GST_TAG_IMAGE, &cover_sample);
3194   if (!cover_sample)
3195     gst_tag_list_get_sample (tags, GST_TAG_PREVIEW_IMAGE, &cover_sample);
3196 
3197   return cover_sample;
3198 }
3199 
3200 static GstPlayerMediaInfo *
gst_player_media_info_create(GstPlayer * self)3201 gst_player_media_info_create (GstPlayer * self)
3202 {
3203   GstPlayerMediaInfo *media_info;
3204   GstQuery *query;
3205 
3206   GST_DEBUG_OBJECT (self, "begin");
3207   media_info = gst_player_media_info_new (self->uri);
3208   media_info->duration = gst_player_get_duration (self);
3209   media_info->tags = self->global_tags;
3210   media_info->is_live = self->is_live;
3211   self->global_tags = NULL;
3212 
3213   query = gst_query_new_seeking (GST_FORMAT_TIME);
3214   if (gst_element_query (self->playbin, query))
3215     gst_query_parse_seeking (query, NULL, &media_info->seekable, NULL, NULL);
3216   gst_query_unref (query);
3217 
3218   if (self->use_playbin3 && self->collection) {
3219     gst_player_streams_info_create_from_collection (self, media_info,
3220         self->collection);
3221   } else {
3222     /* create audio/video/sub streams */
3223     gst_player_streams_info_create (self, media_info, "n-video",
3224         GST_TYPE_PLAYER_VIDEO_INFO);
3225     gst_player_streams_info_create (self, media_info, "n-audio",
3226         GST_TYPE_PLAYER_AUDIO_INFO);
3227     gst_player_streams_info_create (self, media_info, "n-text",
3228         GST_TYPE_PLAYER_SUBTITLE_INFO);
3229   }
3230 
3231   media_info->title = get_from_tags (self, media_info, get_title);
3232   media_info->container =
3233       get_from_tags (self, media_info, get_container_format);
3234   media_info->image_sample = get_from_tags (self, media_info, get_cover_sample);
3235 
3236   GST_DEBUG_OBJECT (self, "uri: %s title: %s duration: %" GST_TIME_FORMAT
3237       " seekable: %s live: %s container: %s image_sample %p",
3238       media_info->uri, media_info->title, GST_TIME_ARGS (media_info->duration),
3239       media_info->seekable ? "yes" : "no", media_info->is_live ? "yes" : "no",
3240       media_info->container, media_info->image_sample);
3241 
3242   GST_DEBUG_OBJECT (self, "end");
3243   return media_info;
3244 }
3245 
3246 static void
tags_changed_cb(GstPlayer * self,gint stream_index,GType type)3247 tags_changed_cb (GstPlayer * self, gint stream_index, GType type)
3248 {
3249   GstPlayerStreamInfo *s;
3250 
3251 /* ohos.opt.stable.0001 */
3252 #ifdef OHOS_OPT_STABLE
3253   g_mutex_lock (&self->lock);
3254 
3255   if (!self->media_info) {
3256     g_mutex_unlock (&self->lock);
3257     return;
3258   }
3259 #else
3260   if (!self->media_info) {
3261     return;
3262   }
3263 
3264   /* update the stream information */
3265   g_mutex_lock (&self->lock);
3266 #endif
3267   s = gst_player_stream_info_find (self->media_info, type, stream_index);
3268   gst_player_stream_info_update_tags_and_caps (self, s);
3269   g_mutex_unlock (&self->lock);
3270 
3271   emit_media_info_updated_signal (self);
3272 }
3273 
3274 static void
video_tags_changed_cb(G_GNUC_UNUSED GstElement * playbin,gint stream_index,gpointer user_data)3275 video_tags_changed_cb (G_GNUC_UNUSED GstElement * playbin, gint stream_index,
3276     gpointer user_data)
3277 {
3278   tags_changed_cb (GST_PLAYER (user_data), stream_index,
3279       GST_TYPE_PLAYER_VIDEO_INFO);
3280 }
3281 
3282 static void
audio_tags_changed_cb(G_GNUC_UNUSED GstElement * playbin,gint stream_index,gpointer user_data)3283 audio_tags_changed_cb (G_GNUC_UNUSED GstElement * playbin, gint stream_index,
3284     gpointer user_data)
3285 {
3286   tags_changed_cb (GST_PLAYER (user_data), stream_index,
3287       GST_TYPE_PLAYER_AUDIO_INFO);
3288 }
3289 
3290 static void
subtitle_tags_changed_cb(G_GNUC_UNUSED GstElement * playbin,gint stream_index,gpointer user_data)3291 subtitle_tags_changed_cb (G_GNUC_UNUSED GstElement * playbin, gint stream_index,
3292     gpointer user_data)
3293 {
3294   tags_changed_cb (GST_PLAYER (user_data), stream_index,
3295       GST_TYPE_PLAYER_SUBTITLE_INFO);
3296 }
3297 
3298 static void
volume_changed_dispatch(gpointer user_data)3299 volume_changed_dispatch (gpointer user_data)
3300 {
3301   GstPlayer *player = user_data;
3302 
3303   if (player->inhibit_sigs)
3304     return;
3305 
3306   g_signal_emit (player, signals[SIGNAL_VOLUME_CHANGED], 0);
3307   g_object_notify_by_pspec (G_OBJECT (player), param_specs[PROP_VOLUME]);
3308 }
3309 
3310 static void
volume_notify_cb(G_GNUC_UNUSED GObject * obj,G_GNUC_UNUSED GParamSpec * pspec,GstPlayer * self)3311 volume_notify_cb (G_GNUC_UNUSED GObject * obj, G_GNUC_UNUSED GParamSpec * pspec,
3312     GstPlayer * self)
3313 {
3314   if (g_signal_handler_find (self, G_SIGNAL_MATCH_ID,
3315           signals[SIGNAL_VOLUME_CHANGED], 0, NULL, NULL, NULL) != 0) {
3316     gst_player_signal_dispatcher_dispatch (self->signal_dispatcher, self,
3317         volume_changed_dispatch, g_object_ref (self),
3318         (GDestroyNotify) g_object_unref);
3319   }
3320 }
3321 
3322 static void
mute_changed_dispatch(gpointer user_data)3323 mute_changed_dispatch (gpointer user_data)
3324 {
3325   GstPlayer *player = user_data;
3326 
3327   if (player->inhibit_sigs)
3328     return;
3329 
3330   g_signal_emit (player, signals[SIGNAL_MUTE_CHANGED], 0);
3331   g_object_notify_by_pspec (G_OBJECT (player), param_specs[PROP_MUTE]);
3332 }
3333 
3334 static void
mute_notify_cb(G_GNUC_UNUSED GObject * obj,G_GNUC_UNUSED GParamSpec * pspec,GstPlayer * self)3335 mute_notify_cb (G_GNUC_UNUSED GObject * obj, G_GNUC_UNUSED GParamSpec * pspec,
3336     GstPlayer * self)
3337 {
3338   if (g_signal_handler_find (self, G_SIGNAL_MATCH_ID,
3339           signals[SIGNAL_MUTE_CHANGED], 0, NULL, NULL, NULL) != 0) {
3340     gst_player_signal_dispatcher_dispatch (self->signal_dispatcher, self,
3341         mute_changed_dispatch, g_object_ref (self),
3342         (GDestroyNotify) g_object_unref);
3343   }
3344 }
3345 
3346 static void
source_setup_cb(GstElement * playbin,GstElement * source,GstPlayer * self)3347 source_setup_cb (GstElement * playbin, GstElement * source, GstPlayer * self)
3348 {
3349   gchar *user_agent;
3350 #ifdef OHOS_EXT_FUNC
3351   // ohos.ext.func.0006
3352   g_signal_emit (self, signals[SIGNAL_SOURCE_SETUP], 0, source);
3353 #endif
3354   user_agent = gst_player_config_get_user_agent (self->config);
3355   if (user_agent) {
3356     GParamSpec *prop;
3357 
3358     prop = g_object_class_find_property (G_OBJECT_GET_CLASS (source),
3359         "user-agent");
3360     if (prop && prop->value_type == G_TYPE_STRING) {
3361       GST_INFO_OBJECT (self, "Setting source user-agent: %s", user_agent);
3362       g_object_set (source, "user-agent", user_agent, NULL);
3363     }
3364 
3365     g_free (user_agent);
3366   }
3367 }
3368 
3369 #ifdef OHOS_EXT_FUNC
3370 /* ohos.ext.func.0017
3371  * report the signal when add new pads
3372  */
3373 static void
element_setup_cb(GstElement * playbin,GstElement * element,GstPlayer * self)3374 element_setup_cb (GstElement * playbin, GstElement * element, GstPlayer * self)
3375 {
3376   g_signal_emit (self, signals[SIGNAL_ELEMENT_SETUP], 0, element);
3377 }
3378 #endif
3379 
3380 static gpointer
gst_player_main(gpointer data)3381 gst_player_main (gpointer data)
3382 {
3383   GstPlayer *self = GST_PLAYER (data);
3384   GstBus *bus;
3385   GSource *source;
3386   GSource *bus_source;
3387   GstElement *scaletempo;
3388   const gchar *env;
3389 
3390   GST_TRACE_OBJECT (self, "Starting main thread");
3391 
3392   g_main_context_push_thread_default (self->context);
3393 
3394   source = g_idle_source_new ();
3395   g_source_set_callback (source, (GSourceFunc) main_loop_running_cb, self,
3396       NULL);
3397   g_source_attach (source, self->context);
3398   g_source_unref (source);
3399 
3400   env = g_getenv ("GST_PLAYER_USE_PLAYBIN3");
3401   if (env && g_str_has_prefix (env, "1"))
3402     self->use_playbin3 = TRUE;
3403 
3404   if (self->use_playbin3) {
3405     GST_DEBUG_OBJECT (self, "playbin3 enabled");
3406     self->playbin = gst_element_factory_make ("playbin3", "playbin3");
3407   } else {
3408     self->playbin = gst_element_factory_make ("playbin", "playbin");
3409   }
3410 
3411   if (!self->playbin) {
3412     g_error ("GstPlayer: 'playbin' element not found, please check your setup");
3413     g_assert_not_reached ();
3414   }
3415 
3416   if (self->video_renderer) {
3417     GstElement *video_sink =
3418         gst_player_video_renderer_create_video_sink (self->video_renderer,
3419         self);
3420 
3421     if (video_sink)
3422       g_object_set (self->playbin, "video-sink", video_sink, NULL);
3423   }
3424 
3425   scaletempo = gst_element_factory_make ("scaletempo", NULL);
3426   if (scaletempo) {
3427     g_object_set (self->playbin, "audio-filter", scaletempo, NULL);
3428   } else {
3429     g_warning ("GstPlayer: scaletempo element not available. Audio pitch "
3430         "will not be preserved during trick modes");
3431   }
3432 
3433   self->bus = bus = gst_element_get_bus (self->playbin);
3434   bus_source = gst_bus_create_watch (bus);
3435   g_source_set_callback (bus_source, (GSourceFunc) gst_bus_async_signal_func,
3436       NULL, NULL);
3437   g_source_attach (bus_source, self->context);
3438 
3439 #ifdef OHOS_EXT_FUNC
3440   // ohos.ext.func.0001
3441   g_signal_connect (G_OBJECT (bus), "message::error", G_CALLBACK (error_cb_msg),
3442       self);
3443 #else
3444   g_signal_connect (G_OBJECT (bus), "message::error", G_CALLBACK (error_cb),
3445       self);
3446 #endif
3447   g_signal_connect (G_OBJECT (bus), "message::warning", G_CALLBACK (warning_cb),
3448       self);
3449   g_signal_connect (G_OBJECT (bus), "message::eos", G_CALLBACK (eos_cb), self);
3450   g_signal_connect (G_OBJECT (bus), "message::state-changed",
3451       G_CALLBACK (state_changed_cb), self);
3452   g_signal_connect (G_OBJECT (bus), "message::buffering",
3453       G_CALLBACK (buffering_cb), self);
3454   g_signal_connect (G_OBJECT (bus), "message::clock-lost",
3455       G_CALLBACK (clock_lost_cb), self);
3456   g_signal_connect (G_OBJECT (bus), "message::duration-changed",
3457       G_CALLBACK (duration_changed_cb), self);
3458   g_signal_connect (G_OBJECT (bus), "message::latency",
3459       G_CALLBACK (latency_cb), self);
3460   g_signal_connect (G_OBJECT (bus), "message::request-state",
3461       G_CALLBACK (request_state_cb), self);
3462   g_signal_connect (G_OBJECT (bus), "message::element",
3463       G_CALLBACK (element_cb), self);
3464   g_signal_connect (G_OBJECT (bus), "message::tag", G_CALLBACK (tags_cb), self);
3465 
3466   if (self->use_playbin3) {
3467     g_signal_connect (G_OBJECT (bus), "message::stream-collection",
3468         G_CALLBACK (stream_collection_cb), self);
3469     g_signal_connect (G_OBJECT (bus), "message::streams-selected",
3470         G_CALLBACK (streams_selected_cb), self);
3471   } else {
3472     g_signal_connect (self->playbin, "video-changed",
3473         G_CALLBACK (video_changed_cb), self);
3474     g_signal_connect (self->playbin, "audio-changed",
3475         G_CALLBACK (audio_changed_cb), self);
3476     g_signal_connect (self->playbin, "text-changed",
3477         G_CALLBACK (subtitle_changed_cb), self);
3478 
3479     g_signal_connect (self->playbin, "video-tags-changed",
3480         G_CALLBACK (video_tags_changed_cb), self);
3481     g_signal_connect (self->playbin, "audio-tags-changed",
3482         G_CALLBACK (audio_tags_changed_cb), self);
3483     g_signal_connect (self->playbin, "text-tags-changed",
3484         G_CALLBACK (subtitle_tags_changed_cb), self);
3485   }
3486 
3487   g_signal_connect (self->playbin, "notify::volume",
3488       G_CALLBACK (volume_notify_cb), self);
3489   g_signal_connect (self->playbin, "notify::mute",
3490       G_CALLBACK (mute_notify_cb), self);
3491   g_signal_connect (self->playbin, "source-setup",
3492       G_CALLBACK (source_setup_cb), self);
3493 
3494 #ifdef OHOS_EXT_FUNC
3495   /* ohos.ext.func.0017
3496    * report the signal when add new pads
3497    */
3498   g_signal_connect (self->playbin, "element-setup",
3499       G_CALLBACK (element_setup_cb), self);
3500 #endif
3501   self->target_state = GST_STATE_NULL;
3502   self->current_state = GST_STATE_NULL;
3503   change_state (self, GST_PLAYER_STATE_STOPPED);
3504   self->buffering = 100;
3505   self->is_eos = FALSE;
3506   self->is_live = FALSE;
3507   self->rate = 1.0;
3508 #ifdef OHOS_EXT_FUNC
3509   // ohos.ext.func.0012
3510   self->timeout = DEFAULT_TIMEOUT;
3511 #endif
3512   GST_TRACE_OBJECT (self, "Starting main loop");
3513   g_main_loop_run (self->loop);
3514   GST_TRACE_OBJECT (self, "Stopped main loop");
3515 
3516   g_source_destroy (bus_source);
3517   g_source_unref (bus_source);
3518   gst_object_unref (bus);
3519 
3520   remove_tick_source (self);
3521   remove_ready_timeout_source (self);
3522 
3523   g_mutex_lock (&self->lock);
3524   if (self->media_info) {
3525     g_object_unref (self->media_info);
3526     self->media_info = NULL;
3527   }
3528 
3529   remove_seek_source (self);
3530   g_mutex_unlock (&self->lock);
3531 
3532   g_main_context_pop_thread_default (self->context);
3533 
3534   self->target_state = GST_STATE_NULL;
3535   self->current_state = GST_STATE_NULL;
3536   if (self->playbin) {
3537     gst_element_set_state (self->playbin, GST_STATE_NULL);
3538     gst_object_unref (self->playbin);
3539     self->playbin = NULL;
3540   }
3541 
3542   GST_TRACE_OBJECT (self, "Stopped main thread");
3543 
3544   return NULL;
3545 }
3546 
3547 static gpointer
gst_player_init_once(G_GNUC_UNUSED gpointer user_data)3548 gst_player_init_once (G_GNUC_UNUSED gpointer user_data)
3549 {
3550   gst_init (NULL, NULL);
3551 
3552   GST_DEBUG_CATEGORY_INIT (gst_player_debug, "gst-player", 0, "GstPlayer");
3553   gst_player_error_quark ();
3554 
3555   return NULL;
3556 }
3557 
3558 /**
3559  * gst_player_new:
3560  * @video_renderer: (transfer full) (allow-none): GstPlayerVideoRenderer to use
3561  * @signal_dispatcher: (transfer full) (allow-none): GstPlayerSignalDispatcher to use
3562  *
3563  * Creates a new #GstPlayer instance that uses @signal_dispatcher to dispatch
3564  * signals to some event loop system, or emits signals directly if NULL is
3565  * passed. See gst_player_g_main_context_signal_dispatcher_new().
3566  *
3567  * Video is going to be rendered by @video_renderer, or if %NULL is provided
3568  * no special video set up will be done and some default handling will be
3569  * performed.
3570  *
3571  * Returns: (transfer full): a new #GstPlayer instance
3572  */
3573 GstPlayer *
gst_player_new(GstPlayerVideoRenderer * video_renderer,GstPlayerSignalDispatcher * signal_dispatcher)3574 gst_player_new (GstPlayerVideoRenderer * video_renderer,
3575     GstPlayerSignalDispatcher * signal_dispatcher)
3576 {
3577   static GOnce once = G_ONCE_INIT;
3578   GstPlayer *self;
3579 
3580   g_once (&once, gst_player_init_once, NULL);
3581 
3582   self =
3583       g_object_new (GST_TYPE_PLAYER, "video-renderer", video_renderer,
3584       "signal-dispatcher", signal_dispatcher, NULL);
3585   gst_object_ref_sink (self);
3586 
3587   if (video_renderer)
3588     g_object_unref (video_renderer);
3589   if (signal_dispatcher)
3590     g_object_unref (signal_dispatcher);
3591 
3592   return self;
3593 }
3594 
3595 static gboolean
gst_player_play_internal(gpointer user_data)3596 gst_player_play_internal (gpointer user_data)
3597 {
3598   GstPlayer *self = GST_PLAYER (user_data);
3599   GstStateChangeReturn state_ret;
3600 
3601   GST_DEBUG_OBJECT (self, "Play");
3602 
3603   g_mutex_lock (&self->lock);
3604   if (!self->uri) {
3605     g_mutex_unlock (&self->lock);
3606     return G_SOURCE_REMOVE;
3607   }
3608   g_mutex_unlock (&self->lock);
3609 
3610   remove_ready_timeout_source (self);
3611   self->target_state = GST_STATE_PLAYING;
3612 
3613   if (self->current_state < GST_STATE_PAUSED)
3614     change_state (self, GST_PLAYER_STATE_BUFFERING);
3615 
3616   if (self->current_state >= GST_STATE_PAUSED && !self->is_eos
3617       && self->buffering >= 100 && !(self->seek_position != GST_CLOCK_TIME_NONE
3618           || self->seek_pending)) {
3619     state_ret = gst_element_set_state (self->playbin, GST_STATE_PLAYING);
3620   } else {
3621     state_ret = gst_element_set_state (self->playbin, GST_STATE_PAUSED);
3622   }
3623 
3624   if (state_ret == GST_STATE_CHANGE_FAILURE) {
3625     emit_error (self, g_error_new (GST_PLAYER_ERROR, GST_PLAYER_ERROR_FAILED,
3626             "Failed to play"));
3627     return G_SOURCE_REMOVE;
3628   } else if (state_ret == GST_STATE_CHANGE_NO_PREROLL) {
3629     self->is_live = TRUE;
3630     GST_DEBUG_OBJECT (self, "Pipeline is live");
3631   }
3632 
3633   if (self->is_eos) {
3634     gboolean ret;
3635 
3636     GST_DEBUG_OBJECT (self, "Was EOS, seeking to beginning");
3637     self->is_eos = FALSE;
3638     ret =
3639         gst_element_seek_simple (self->playbin, GST_FORMAT_TIME,
3640         GST_SEEK_FLAG_FLUSH, 0);
3641     if (!ret) {
3642       GST_ERROR_OBJECT (self, "Seek to beginning failed");
3643       gst_player_stop_internal (self, TRUE);
3644       gst_player_play_internal (self);
3645     }
3646   }
3647 
3648   return G_SOURCE_REMOVE;
3649 }
3650 
3651 /**
3652  * gst_player_play:
3653  * @player: #GstPlayer instance
3654  *
3655  * Request to play the loaded stream.
3656  */
3657 void
gst_player_play(GstPlayer * self)3658 gst_player_play (GstPlayer * self)
3659 {
3660   g_return_if_fail (GST_IS_PLAYER (self));
3661 
3662   g_mutex_lock (&self->lock);
3663   self->inhibit_sigs = FALSE;
3664   g_mutex_unlock (&self->lock);
3665 
3666   g_main_context_invoke_full (self->context, G_PRIORITY_DEFAULT,
3667       gst_player_play_internal, self, NULL);
3668 }
3669 
3670 static gboolean
gst_player_pause_internal(gpointer user_data)3671 gst_player_pause_internal (gpointer user_data)
3672 {
3673   GstPlayer *self = GST_PLAYER (user_data);
3674   GstStateChangeReturn state_ret;
3675 
3676   GST_DEBUG_OBJECT (self, "Pause");
3677 
3678   g_mutex_lock (&self->lock);
3679   if (!self->uri) {
3680     g_mutex_unlock (&self->lock);
3681     return G_SOURCE_REMOVE;
3682   }
3683   g_mutex_unlock (&self->lock);
3684 
3685   tick_cb (self);
3686   remove_tick_source (self);
3687   remove_ready_timeout_source (self);
3688 
3689   self->target_state = GST_STATE_PAUSED;
3690 
3691   if (self->current_state < GST_STATE_PAUSED)
3692     change_state (self, GST_PLAYER_STATE_BUFFERING);
3693 
3694   state_ret = gst_element_set_state (self->playbin, GST_STATE_PAUSED);
3695   if (state_ret == GST_STATE_CHANGE_FAILURE) {
3696     emit_error (self, g_error_new (GST_PLAYER_ERROR, GST_PLAYER_ERROR_FAILED,
3697             "Failed to pause"));
3698     return G_SOURCE_REMOVE;
3699   } else if (state_ret == GST_STATE_CHANGE_NO_PREROLL) {
3700     self->is_live = TRUE;
3701     GST_DEBUG_OBJECT (self, "Pipeline is live");
3702   }
3703 
3704   if (self->is_eos) {
3705     gboolean ret;
3706 
3707     GST_DEBUG_OBJECT (self, "Was EOS, seeking to beginning");
3708     self->is_eos = FALSE;
3709     ret =
3710         gst_element_seek_simple (self->playbin, GST_FORMAT_TIME,
3711         GST_SEEK_FLAG_FLUSH, 0);
3712     if (!ret) {
3713       GST_ERROR_OBJECT (self, "Seek to beginning failed");
3714       gst_player_stop_internal (self, TRUE);
3715       gst_player_pause_internal (self);
3716     }
3717   }
3718 
3719   return G_SOURCE_REMOVE;
3720 }
3721 
3722 /**
3723  * gst_player_pause:
3724  * @player: #GstPlayer instance
3725  *
3726  * Pauses the current stream.
3727  */
3728 void
gst_player_pause(GstPlayer * self)3729 gst_player_pause (GstPlayer * self)
3730 {
3731   g_return_if_fail (GST_IS_PLAYER (self));
3732 
3733   g_mutex_lock (&self->lock);
3734   self->inhibit_sigs = FALSE;
3735   g_mutex_unlock (&self->lock);
3736 
3737   g_main_context_invoke_full (self->context, G_PRIORITY_DEFAULT,
3738       gst_player_pause_internal, self, NULL);
3739 }
3740 
3741 static void
gst_player_stop_internal(GstPlayer * self,gboolean transient)3742 gst_player_stop_internal (GstPlayer * self, gboolean transient)
3743 {
3744   /* directly return if we're already stopped */
3745   if (self->current_state <= GST_STATE_READY &&
3746       self->target_state <= GST_STATE_READY)
3747     return;
3748 
3749   GST_DEBUG_OBJECT (self, "Stop (transient %d)", transient);
3750 
3751   tick_cb (self);
3752   remove_tick_source (self);
3753 
3754   add_ready_timeout_source (self);
3755 
3756   self->target_state = GST_STATE_NULL;
3757   self->current_state = GST_STATE_READY;
3758   self->is_live = FALSE;
3759   self->is_eos = FALSE;
3760   gst_bus_set_flushing (self->bus, TRUE);
3761   gst_element_set_state (self->playbin, GST_STATE_READY);
3762   gst_bus_set_flushing (self->bus, FALSE);
3763   change_state (self, transient
3764       && self->app_state !=
3765       GST_PLAYER_STATE_STOPPED ? GST_PLAYER_STATE_BUFFERING :
3766       GST_PLAYER_STATE_STOPPED);
3767   self->buffering = 100;
3768   self->cached_duration = GST_CLOCK_TIME_NONE;
3769   g_mutex_lock (&self->lock);
3770   if (self->media_info) {
3771     g_object_unref (self->media_info);
3772     self->media_info = NULL;
3773   }
3774   if (self->global_tags) {
3775     gst_tag_list_unref (self->global_tags);
3776     self->global_tags = NULL;
3777   }
3778   self->seek_pending = FALSE;
3779   remove_seek_source (self);
3780   self->seek_position = GST_CLOCK_TIME_NONE;
3781   self->last_seek_time = GST_CLOCK_TIME_NONE;
3782   self->rate = 1.0;
3783   if (self->collection) {
3784     if (self->stream_notify_id)
3785       g_signal_handler_disconnect (self->collection, self->stream_notify_id);
3786     self->stream_notify_id = 0;
3787     gst_object_unref (self->collection);
3788     self->collection = NULL;
3789   }
3790   g_free (self->video_sid);
3791   g_free (self->audio_sid);
3792   g_free (self->subtitle_sid);
3793   self->video_sid = NULL;
3794   self->audio_sid = NULL;
3795   self->subtitle_sid = NULL;
3796   g_mutex_unlock (&self->lock);
3797 }
3798 
3799 static gboolean
gst_player_stop_internal_dispatch(gpointer user_data)3800 gst_player_stop_internal_dispatch (gpointer user_data)
3801 {
3802   GstPlayer *self = GST_PLAYER (user_data);
3803 
3804   gst_player_stop_internal (self, FALSE);
3805 
3806   return G_SOURCE_REMOVE;
3807 }
3808 
3809 
3810 /**
3811  * gst_player_stop:
3812  * @player: #GstPlayer instance
3813  *
3814  * Stops playing the current stream and resets to the first position
3815  * in the stream.
3816  */
3817 void
gst_player_stop(GstPlayer * self)3818 gst_player_stop (GstPlayer * self)
3819 {
3820   g_return_if_fail (GST_IS_PLAYER (self));
3821 
3822   g_mutex_lock (&self->lock);
3823   self->inhibit_sigs = TRUE;
3824   g_mutex_unlock (&self->lock);
3825 
3826   g_main_context_invoke_full (self->context, G_PRIORITY_DEFAULT,
3827       gst_player_stop_internal_dispatch, self, NULL);
3828 }
3829 
3830 /* Must be called with lock from main context, releases lock! */
3831 static void
gst_player_seek_internal_locked(GstPlayer * self)3832 gst_player_seek_internal_locked (GstPlayer * self)
3833 {
3834   gboolean ret;
3835   GstClockTime position;
3836   gdouble rate;
3837   GstStateChangeReturn state_ret;
3838   GstEvent *s_event;
3839   GstSeekFlags flags = 0;
3840   gboolean accurate = FALSE;
3841 
3842   remove_seek_source (self);
3843 
3844   /* Only seek in PAUSED */
3845   if (self->current_state < GST_STATE_PAUSED) {
3846     return;
3847   } else if (self->current_state != GST_STATE_PAUSED) {
3848     g_mutex_unlock (&self->lock);
3849     state_ret = gst_element_set_state (self->playbin, GST_STATE_PAUSED);
3850     if (state_ret == GST_STATE_CHANGE_FAILURE) {
3851       emit_error (self, g_error_new (GST_PLAYER_ERROR, GST_PLAYER_ERROR_FAILED,
3852               "Failed to seek"));
3853       g_mutex_lock (&self->lock);
3854       return;
3855     }
3856     g_mutex_lock (&self->lock);
3857     return;
3858   }
3859 
3860   self->last_seek_time = gst_util_get_timestamp ();
3861   position = self->seek_position;
3862   self->seek_position = GST_CLOCK_TIME_NONE;
3863   self->seek_pending = TRUE;
3864   rate = self->rate;
3865   g_mutex_unlock (&self->lock);
3866 
3867   remove_tick_source (self);
3868   self->is_eos = FALSE;
3869 
3870   flags |= GST_SEEK_FLAG_FLUSH;
3871 
3872   accurate = gst_player_config_get_seek_accurate (self->config);
3873 
3874   if (accurate) {
3875     flags |= GST_SEEK_FLAG_ACCURATE;
3876   } else {
3877     flags &= ~GST_SEEK_FLAG_ACCURATE;
3878   }
3879 
3880   if (rate != 1.0) {
3881     flags |= GST_SEEK_FLAG_TRICKMODE;
3882   }
3883 #ifdef OHOS_EXT_FUNC
3884   // ohos.ext.func.0004
3885     flags |= self->seek_mode;
3886 #endif
3887   if (rate >= 0.0) {
3888     s_event = gst_event_new_seek (rate, GST_FORMAT_TIME, flags,
3889         GST_SEEK_TYPE_SET, position, GST_SEEK_TYPE_SET, GST_CLOCK_TIME_NONE);
3890   } else {
3891     s_event = gst_event_new_seek (rate, GST_FORMAT_TIME, flags,
3892         GST_SEEK_TYPE_SET, G_GINT64_CONSTANT (0), GST_SEEK_TYPE_SET, position);
3893   }
3894 
3895   GST_DEBUG_OBJECT (self, "Seek with rate %.2lf to %" GST_TIME_FORMAT,
3896       rate, GST_TIME_ARGS (position));
3897 
3898   ret = gst_element_send_event (self->playbin, s_event);
3899   if (!ret)
3900     emit_error (self, g_error_new (GST_PLAYER_ERROR, GST_PLAYER_ERROR_FAILED,
3901             "Failed to seek to %" GST_TIME_FORMAT, GST_TIME_ARGS (position)));
3902 
3903   g_mutex_lock (&self->lock);
3904 }
3905 
3906 static gboolean
gst_player_seek_internal(gpointer user_data)3907 gst_player_seek_internal (gpointer user_data)
3908 {
3909   GstPlayer *self = GST_PLAYER (user_data);
3910 
3911   g_mutex_lock (&self->lock);
3912   gst_player_seek_internal_locked (self);
3913   g_mutex_unlock (&self->lock);
3914 
3915   return G_SOURCE_REMOVE;
3916 }
3917 
3918 /**
3919  * gst_player_set_rate:
3920  * @player: #GstPlayer instance
3921  * @rate: playback rate
3922  *
3923  * Playback at specified rate
3924  */
3925 void
gst_player_set_rate(GstPlayer * self,gdouble rate)3926 gst_player_set_rate (GstPlayer * self, gdouble rate)
3927 {
3928   g_return_if_fail (GST_IS_PLAYER (self));
3929   g_return_if_fail (rate != 0.0);
3930 #ifdef OHOS_EXT_FUNC
3931   // ohos.ext.func.0004
3932   g_object_set (self, "seek-mode", 0, NULL);
3933 #endif
3934   g_object_set (self, "rate", rate, NULL);
3935 }
3936 
3937 /**
3938  * gst_player_get_rate:
3939  * @player: #GstPlayer instance
3940  *
3941  * Returns: current playback rate
3942  */
3943 gdouble
gst_player_get_rate(GstPlayer * self)3944 gst_player_get_rate (GstPlayer * self)
3945 {
3946   gdouble val;
3947 
3948   g_return_val_if_fail (GST_IS_PLAYER (self), DEFAULT_RATE);
3949 
3950   g_object_get (self, "rate", &val, NULL);
3951 
3952   return val;
3953 }
3954 
3955 /**
3956  * gst_player_seek:
3957  * @player: #GstPlayer instance
3958  * @position: position to seek in nanoseconds
3959  *
3960  * Seeks the currently-playing stream to the absolute @position time
3961  * in nanoseconds.
3962  */
3963 void
gst_player_seek(GstPlayer * self,GstClockTime position)3964 gst_player_seek (GstPlayer * self, GstClockTime position)
3965 {
3966   g_return_if_fail (GST_IS_PLAYER (self));
3967   g_return_if_fail (GST_CLOCK_TIME_IS_VALID (position));
3968 
3969   g_mutex_lock (&self->lock);
3970   if (self->media_info && !self->media_info->seekable) {
3971     GST_DEBUG_OBJECT (self, "Media is not seekable");
3972     g_mutex_unlock (&self->lock);
3973     return;
3974   }
3975 
3976   self->seek_position = position;
3977 
3978   /* If there is no seek being dispatch to the main context currently do that,
3979    * otherwise we just updated the seek position so that it will be taken by
3980    * the seek handler from the main context instead of the old one.
3981    */
3982   if (!self->seek_source) {
3983     GstClockTime now = gst_util_get_timestamp ();
3984 
3985     /* If no seek is pending or it was started more than 250 mseconds ago seek
3986      * immediately, otherwise wait until the 250 mseconds have passed */
3987     if (!self->seek_pending || (now - self->last_seek_time > 250 * GST_MSECOND)) {
3988       self->seek_source = g_idle_source_new ();
3989       g_source_set_callback (self->seek_source,
3990           (GSourceFunc) gst_player_seek_internal, self, NULL);
3991       GST_TRACE_OBJECT (self, "Dispatching seek to position %" GST_TIME_FORMAT,
3992           GST_TIME_ARGS (position));
3993       g_source_attach (self->seek_source, self->context);
3994     } else {
3995       guint delay = 250000 - (now - self->last_seek_time) / 1000;
3996 
3997       /* Note that last_seek_time must be set to something at this point and
3998        * it must be smaller than 250 mseconds */
3999       self->seek_source = g_timeout_source_new (delay);
4000       g_source_set_callback (self->seek_source,
4001           (GSourceFunc) gst_player_seek_internal, self, NULL);
4002 
4003       GST_TRACE_OBJECT (self,
4004           "Delaying seek to position %" GST_TIME_FORMAT " by %u us",
4005           GST_TIME_ARGS (position), delay);
4006       g_source_attach (self->seek_source, self->context);
4007     }
4008   }
4009   g_mutex_unlock (&self->lock);
4010 }
4011 
4012 static void
remove_seek_source(GstPlayer * self)4013 remove_seek_source (GstPlayer * self)
4014 {
4015   if (!self->seek_source)
4016     return;
4017 
4018   g_source_destroy (self->seek_source);
4019   g_source_unref (self->seek_source);
4020   self->seek_source = NULL;
4021 }
4022 
4023 /**
4024  * gst_player_get_uri:
4025  * @player: #GstPlayer instance
4026  *
4027  * Gets the URI of the currently-playing stream.
4028  *
4029  * Returns: (transfer full): a string containing the URI of the
4030  * currently-playing stream. g_free() after usage.
4031  */
4032 gchar *
gst_player_get_uri(GstPlayer * self)4033 gst_player_get_uri (GstPlayer * self)
4034 {
4035   gchar *val;
4036 
4037   g_return_val_if_fail (GST_IS_PLAYER (self), DEFAULT_URI);
4038 
4039   g_object_get (self, "uri", &val, NULL);
4040 
4041   return val;
4042 }
4043 
4044 /**
4045  * gst_player_set_uri:
4046  * @player: #GstPlayer instance
4047  * @uri: next URI to play.
4048  *
4049  * Sets the next URI to play.
4050  */
4051 void
gst_player_set_uri(GstPlayer * self,const gchar * val)4052 gst_player_set_uri (GstPlayer * self, const gchar * val)
4053 {
4054   g_return_if_fail (GST_IS_PLAYER (self));
4055 
4056   g_object_set (self, "uri", val, NULL);
4057 }
4058 
4059 /**
4060  * gst_player_set_subtitle_uri:
4061  * @player: #GstPlayer instance
4062  * @uri: subtitle URI
4063  *
4064  * Sets the external subtitle URI. This should be combined with a call to
4065  * gst_player_set_subtitle_track_enabled(@player, TRUE) so the subtitles are actually
4066  * rendered.
4067  */
4068 void
gst_player_set_subtitle_uri(GstPlayer * self,const gchar * suburi)4069 gst_player_set_subtitle_uri (GstPlayer * self, const gchar * suburi)
4070 {
4071   g_return_if_fail (GST_IS_PLAYER (self));
4072 
4073   g_object_set (self, "suburi", suburi, NULL);
4074 }
4075 
4076 /**
4077  * gst_player_get_subtitle_uri:
4078  * @player: #GstPlayer instance
4079  *
4080  * current subtitle URI
4081  *
4082  * Returns: (transfer full): URI of the current external subtitle.
4083  *   g_free() after usage.
4084  */
4085 gchar *
gst_player_get_subtitle_uri(GstPlayer * self)4086 gst_player_get_subtitle_uri (GstPlayer * self)
4087 {
4088   gchar *val = NULL;
4089 
4090   g_return_val_if_fail (GST_IS_PLAYER (self), NULL);
4091 
4092   g_object_get (self, "suburi", &val, NULL);
4093 
4094   return val;
4095 }
4096 
4097 /**
4098  * gst_player_get_position:
4099  * @player: #GstPlayer instance
4100  *
4101  * Returns: the absolute position time, in nanoseconds, of the
4102  * currently-playing stream.
4103  */
4104 GstClockTime
gst_player_get_position(GstPlayer * self)4105 gst_player_get_position (GstPlayer * self)
4106 {
4107   GstClockTime val;
4108 
4109   g_return_val_if_fail (GST_IS_PLAYER (self), DEFAULT_POSITION);
4110 
4111   g_object_get (self, "position", &val, NULL);
4112 
4113   return val;
4114 }
4115 
4116 /**
4117  * gst_player_get_duration:
4118  * @player: #GstPlayer instance
4119  *
4120  * Retrieves the duration of the media stream that self represents.
4121  *
4122  * Returns: the duration of the currently-playing media stream, in
4123  * nanoseconds.
4124  */
4125 GstClockTime
gst_player_get_duration(GstPlayer * self)4126 gst_player_get_duration (GstPlayer * self)
4127 {
4128   GstClockTime val;
4129 
4130   g_return_val_if_fail (GST_IS_PLAYER (self), DEFAULT_DURATION);
4131 
4132   g_object_get (self, "duration", &val, NULL);
4133 
4134   return val;
4135 }
4136 
4137 /**
4138  * gst_player_get_volume:
4139  * @player: #GstPlayer instance
4140  *
4141  * Returns the current volume level, as a percentage between 0 and 1.
4142  *
4143  * Returns: the volume as percentage between 0 and 1.
4144  */
4145 gdouble
gst_player_get_volume(GstPlayer * self)4146 gst_player_get_volume (GstPlayer * self)
4147 {
4148   gdouble val;
4149 
4150   g_return_val_if_fail (GST_IS_PLAYER (self), DEFAULT_VOLUME);
4151 
4152   g_object_get (self, "volume", &val, NULL);
4153 
4154   return val;
4155 }
4156 
4157 /**
4158  * gst_player_set_volume:
4159  * @player: #GstPlayer instance
4160  * @val: the new volume level, as a percentage between 0 and 1
4161  *
4162  * Sets the volume level of the stream as a percentage between 0 and 1.
4163  */
4164 void
gst_player_set_volume(GstPlayer * self,gdouble val)4165 gst_player_set_volume (GstPlayer * self, gdouble val)
4166 {
4167   g_return_if_fail (GST_IS_PLAYER (self));
4168 
4169   g_object_set (self, "volume", val, NULL);
4170 }
4171 
4172 /**
4173  * gst_player_get_mute:
4174  * @player: #GstPlayer instance
4175  *
4176  * Returns: %TRUE if the currently-playing stream is muted.
4177  */
4178 gboolean
gst_player_get_mute(GstPlayer * self)4179 gst_player_get_mute (GstPlayer * self)
4180 {
4181   gboolean val;
4182 
4183   g_return_val_if_fail (GST_IS_PLAYER (self), DEFAULT_MUTE);
4184 
4185   g_object_get (self, "mute", &val, NULL);
4186 
4187   return val;
4188 }
4189 
4190 /**
4191  * gst_player_set_mute:
4192  * @player: #GstPlayer instance
4193  * @val: Mute state the should be set
4194  *
4195  * %TRUE if the currently-playing stream should be muted.
4196  */
4197 void
gst_player_set_mute(GstPlayer * self,gboolean val)4198 gst_player_set_mute (GstPlayer * self, gboolean val)
4199 {
4200   g_return_if_fail (GST_IS_PLAYER (self));
4201 
4202   g_object_set (self, "mute", val, NULL);
4203 }
4204 
4205 /**
4206  * gst_player_get_pipeline:
4207  * @player: #GstPlayer instance
4208  *
4209  * Returns: (transfer full): The internal playbin instance
4210  */
4211 GstElement *
gst_player_get_pipeline(GstPlayer * self)4212 gst_player_get_pipeline (GstPlayer * self)
4213 {
4214   GstElement *val;
4215 
4216   g_return_val_if_fail (GST_IS_PLAYER (self), NULL);
4217 
4218   g_object_get (self, "pipeline", &val, NULL);
4219 
4220   return val;
4221 }
4222 
4223 /**
4224  * gst_player_get_media_info:
4225  * @player: #GstPlayer instance
4226  *
4227  * A Function to get the current media info #GstPlayerMediaInfo instance.
4228  *
4229  * Returns: (transfer full): media info instance.
4230  *
4231  * The caller should free it with g_object_unref()
4232  */
4233 GstPlayerMediaInfo *
gst_player_get_media_info(GstPlayer * self)4234 gst_player_get_media_info (GstPlayer * self)
4235 {
4236   GstPlayerMediaInfo *info;
4237 
4238   g_return_val_if_fail (GST_IS_PLAYER (self), NULL);
4239 
4240   if (!self->media_info)
4241     return NULL;
4242 
4243   g_mutex_lock (&self->lock);
4244   info = gst_player_media_info_copy (self->media_info);
4245   g_mutex_unlock (&self->lock);
4246 
4247   return info;
4248 }
4249 
4250 /**
4251  * gst_player_get_current_audio_track:
4252  * @player: #GstPlayer instance
4253  *
4254  * A Function to get current audio #GstPlayerAudioInfo instance.
4255  *
4256  * Returns: (transfer full): current audio track.
4257  *
4258  * The caller should free it with g_object_unref()
4259  */
4260 GstPlayerAudioInfo *
gst_player_get_current_audio_track(GstPlayer * self)4261 gst_player_get_current_audio_track (GstPlayer * self)
4262 {
4263   GstPlayerAudioInfo *info;
4264 
4265   g_return_val_if_fail (GST_IS_PLAYER (self), NULL);
4266 
4267   if (!is_track_enabled (self, GST_PLAY_FLAG_AUDIO))
4268     return NULL;
4269 
4270   if (self->use_playbin3) {
4271     info = (GstPlayerAudioInfo *)
4272         gst_player_stream_info_get_current_from_stream_id (self,
4273         self->audio_sid, GST_TYPE_PLAYER_AUDIO_INFO);
4274   } else {
4275     info = (GstPlayerAudioInfo *) gst_player_stream_info_get_current (self,
4276         "current-audio", GST_TYPE_PLAYER_AUDIO_INFO);
4277   }
4278 
4279   return info;
4280 }
4281 
4282 /**
4283  * gst_player_get_current_video_track:
4284  * @player: #GstPlayer instance
4285  *
4286  * A Function to get current video #GstPlayerVideoInfo instance.
4287  *
4288  * Returns: (transfer full): current video track.
4289  *
4290  * The caller should free it with g_object_unref()
4291  */
4292 GstPlayerVideoInfo *
gst_player_get_current_video_track(GstPlayer * self)4293 gst_player_get_current_video_track (GstPlayer * self)
4294 {
4295   GstPlayerVideoInfo *info;
4296 
4297   g_return_val_if_fail (GST_IS_PLAYER (self), NULL);
4298 
4299   if (!is_track_enabled (self, GST_PLAY_FLAG_VIDEO))
4300     return NULL;
4301 
4302   if (self->use_playbin3) {
4303     info = (GstPlayerVideoInfo *)
4304         gst_player_stream_info_get_current_from_stream_id (self,
4305         self->video_sid, GST_TYPE_PLAYER_VIDEO_INFO);
4306   } else {
4307     info = (GstPlayerVideoInfo *) gst_player_stream_info_get_current (self,
4308         "current-video", GST_TYPE_PLAYER_VIDEO_INFO);
4309   }
4310 
4311   return info;
4312 }
4313 
4314 /**
4315  * gst_player_get_current_subtitle_track:
4316  * @player: #GstPlayer instance
4317  *
4318  * A Function to get current subtitle #GstPlayerSubtitleInfo instance.
4319  *
4320  * Returns: (transfer none): current subtitle track.
4321  *
4322  * The caller should free it with g_object_unref()
4323  */
4324 GstPlayerSubtitleInfo *
gst_player_get_current_subtitle_track(GstPlayer * self)4325 gst_player_get_current_subtitle_track (GstPlayer * self)
4326 {
4327   GstPlayerSubtitleInfo *info;
4328 
4329   g_return_val_if_fail (GST_IS_PLAYER (self), NULL);
4330 
4331   if (!is_track_enabled (self, GST_PLAY_FLAG_SUBTITLE))
4332     return NULL;
4333 
4334   if (self->use_playbin3) {
4335     info = (GstPlayerSubtitleInfo *)
4336         gst_player_stream_info_get_current_from_stream_id (self,
4337         self->subtitle_sid, GST_TYPE_PLAYER_SUBTITLE_INFO);
4338   } else {
4339     info = (GstPlayerSubtitleInfo *) gst_player_stream_info_get_current (self,
4340         "current-text", GST_TYPE_PLAYER_SUBTITLE_INFO);
4341   }
4342 
4343   return info;
4344 }
4345 
4346 /* Must be called with lock */
4347 static gboolean
gst_player_select_streams(GstPlayer * self)4348 gst_player_select_streams (GstPlayer * self)
4349 {
4350   GList *stream_list = NULL;
4351   gboolean ret = FALSE;
4352 
4353   if (self->audio_sid)
4354     stream_list = g_list_append (stream_list, g_strdup (self->audio_sid));
4355   if (self->video_sid)
4356     stream_list = g_list_append (stream_list, g_strdup (self->video_sid));
4357   if (self->subtitle_sid)
4358     stream_list = g_list_append (stream_list, g_strdup (self->subtitle_sid));
4359 
4360   g_mutex_unlock (&self->lock);
4361   if (stream_list) {
4362     ret = gst_element_send_event (self->playbin,
4363         gst_event_new_select_streams (stream_list));
4364     g_list_free_full (stream_list, g_free);
4365   } else {
4366     GST_ERROR_OBJECT (self, "No available streams for select-streams");
4367   }
4368   g_mutex_lock (&self->lock);
4369 
4370   return ret;
4371 }
4372 
4373 /**
4374  * gst_player_set_audio_track:
4375  * @player: #GstPlayer instance
4376  * @stream_index: stream index
4377  *
4378  * Returns: %TRUE or %FALSE
4379  *
4380  * Sets the audio track @stream_idex.
4381  */
4382 gboolean
gst_player_set_audio_track(GstPlayer * self,gint stream_index)4383 gst_player_set_audio_track (GstPlayer * self, gint stream_index)
4384 {
4385   GstPlayerStreamInfo *info;
4386   gboolean ret = TRUE;
4387 
4388   g_return_val_if_fail (GST_IS_PLAYER (self), 0);
4389 
4390   g_mutex_lock (&self->lock);
4391   info = gst_player_stream_info_find (self->media_info,
4392       GST_TYPE_PLAYER_AUDIO_INFO, stream_index);
4393   g_mutex_unlock (&self->lock);
4394   if (!info) {
4395     GST_ERROR_OBJECT (self, "invalid audio stream index %d", stream_index);
4396     return FALSE;
4397   }
4398 
4399   if (self->use_playbin3) {
4400     g_mutex_lock (&self->lock);
4401     g_free (self->audio_sid);
4402     self->audio_sid = g_strdup (info->stream_id);
4403     ret = gst_player_select_streams (self);
4404     g_mutex_unlock (&self->lock);
4405   } else {
4406     g_object_set (G_OBJECT (self->playbin), "current-audio", stream_index,
4407         NULL);
4408   }
4409 
4410   GST_DEBUG_OBJECT (self, "set stream index '%d'", stream_index);
4411   return ret;
4412 }
4413 
4414 /**
4415  * gst_player_set_video_track:
4416  * @player: #GstPlayer instance
4417  * @stream_index: stream index
4418  *
4419  * Returns: %TRUE or %FALSE
4420  *
4421  * Sets the video track @stream_index.
4422  */
4423 gboolean
gst_player_set_video_track(GstPlayer * self,gint stream_index)4424 gst_player_set_video_track (GstPlayer * self, gint stream_index)
4425 {
4426   GstPlayerStreamInfo *info;
4427   gboolean ret = TRUE;
4428 
4429   g_return_val_if_fail (GST_IS_PLAYER (self), 0);
4430 
4431   /* check if stream_index exist in our internal media_info list */
4432   g_mutex_lock (&self->lock);
4433   info = gst_player_stream_info_find (self->media_info,
4434       GST_TYPE_PLAYER_VIDEO_INFO, stream_index);
4435   g_mutex_unlock (&self->lock);
4436   if (!info) {
4437     GST_ERROR_OBJECT (self, "invalid video stream index %d", stream_index);
4438     return FALSE;
4439   }
4440 
4441   if (self->use_playbin3) {
4442     g_mutex_lock (&self->lock);
4443     g_free (self->video_sid);
4444     self->video_sid = g_strdup (info->stream_id);
4445     ret = gst_player_select_streams (self);
4446     g_mutex_unlock (&self->lock);
4447   } else {
4448     g_object_set (G_OBJECT (self->playbin), "current-video", stream_index,
4449         NULL);
4450   }
4451 
4452   GST_DEBUG_OBJECT (self, "set stream index '%d'", stream_index);
4453   return ret;
4454 }
4455 
4456 /**
4457  * gst_player_set_subtitle_track:
4458  * @player: #GstPlayer instance
4459  * @stream_index: stream index
4460  *
4461  * Returns: %TRUE or %FALSE
4462  *
4463  * Sets the subtitle strack @stream_index.
4464  */
4465 gboolean
gst_player_set_subtitle_track(GstPlayer * self,gint stream_index)4466 gst_player_set_subtitle_track (GstPlayer * self, gint stream_index)
4467 {
4468   GstPlayerStreamInfo *info;
4469   gboolean ret = TRUE;
4470 
4471   g_return_val_if_fail (GST_IS_PLAYER (self), 0);
4472 
4473   g_mutex_lock (&self->lock);
4474   info = gst_player_stream_info_find (self->media_info,
4475       GST_TYPE_PLAYER_SUBTITLE_INFO, stream_index);
4476   g_mutex_unlock (&self->lock);
4477   if (!info) {
4478     GST_ERROR_OBJECT (self, "invalid subtitle stream index %d", stream_index);
4479     return FALSE;
4480   }
4481 
4482   if (self->use_playbin3) {
4483     g_mutex_lock (&self->lock);
4484     g_free (self->subtitle_sid);
4485     self->subtitle_sid = g_strdup (info->stream_id);
4486     ret = gst_player_select_streams (self);
4487     g_mutex_unlock (&self->lock);
4488   } else {
4489     g_object_set (G_OBJECT (self->playbin), "current-text", stream_index, NULL);
4490   }
4491 
4492   GST_DEBUG_OBJECT (self, "set stream index '%d'", stream_index);
4493   return ret;
4494 }
4495 
4496 /**
4497  * gst_player_set_audio_track_enabled:
4498  * @player: #GstPlayer instance
4499  * @enabled: TRUE or FALSE
4500  *
4501  * Enable or disable the current audio track.
4502  */
4503 void
gst_player_set_audio_track_enabled(GstPlayer * self,gboolean enabled)4504 gst_player_set_audio_track_enabled (GstPlayer * self, gboolean enabled)
4505 {
4506   g_return_if_fail (GST_IS_PLAYER (self));
4507 
4508   if (enabled)
4509     player_set_flag (self, GST_PLAY_FLAG_AUDIO);
4510   else
4511     player_clear_flag (self, GST_PLAY_FLAG_AUDIO);
4512 
4513   GST_DEBUG_OBJECT (self, "track is '%s'", enabled ? "Enabled" : "Disabled");
4514 }
4515 
4516 /**
4517  * gst_player_set_video_track_enabled:
4518  * @player: #GstPlayer instance
4519  * @enabled: TRUE or FALSE
4520  *
4521  * Enable or disable the current video track.
4522  */
4523 void
gst_player_set_video_track_enabled(GstPlayer * self,gboolean enabled)4524 gst_player_set_video_track_enabled (GstPlayer * self, gboolean enabled)
4525 {
4526   g_return_if_fail (GST_IS_PLAYER (self));
4527 
4528   if (enabled)
4529     player_set_flag (self, GST_PLAY_FLAG_VIDEO);
4530   else
4531     player_clear_flag (self, GST_PLAY_FLAG_VIDEO);
4532 
4533   GST_DEBUG_OBJECT (self, "track is '%s'", enabled ? "Enabled" : "Disabled");
4534 }
4535 
4536 /**
4537  * gst_player_set_subtitle_track_enabled:
4538  * @player: #GstPlayer instance
4539  * @enabled: TRUE or FALSE
4540  *
4541  * Enable or disable the current subtitle track.
4542  */
4543 void
gst_player_set_subtitle_track_enabled(GstPlayer * self,gboolean enabled)4544 gst_player_set_subtitle_track_enabled (GstPlayer * self, gboolean enabled)
4545 {
4546   g_return_if_fail (GST_IS_PLAYER (self));
4547 
4548   if (enabled)
4549     player_set_flag (self, GST_PLAY_FLAG_SUBTITLE);
4550   else
4551     player_clear_flag (self, GST_PLAY_FLAG_SUBTITLE);
4552 
4553   GST_DEBUG_OBJECT (self, "track is '%s'", enabled ? "Enabled" : "Disabled");
4554 }
4555 
4556 /**
4557  * gst_player_set_visualization:
4558  * @player: #GstPlayer instance
4559  * @name: visualization element obtained from
4560  * #gst_player_visualizations_get()
4561  *
4562  * Returns: %TRUE if the visualizations was set correctly. Otherwise,
4563  * %FALSE.
4564  */
4565 gboolean
gst_player_set_visualization(GstPlayer * self,const gchar * name)4566 gst_player_set_visualization (GstPlayer * self, const gchar * name)
4567 {
4568   g_return_val_if_fail (GST_IS_PLAYER (self), FALSE);
4569 
4570   g_mutex_lock (&self->lock);
4571   if (self->current_vis_element) {
4572     gst_object_unref (self->current_vis_element);
4573     self->current_vis_element = NULL;
4574   }
4575 
4576   if (name) {
4577     self->current_vis_element = gst_element_factory_make (name, NULL);
4578     if (!self->current_vis_element)
4579       goto error_no_element;
4580     gst_object_ref_sink (self->current_vis_element);
4581   }
4582   g_object_set (self->playbin, "vis-plugin", self->current_vis_element, NULL);
4583 
4584   g_mutex_unlock (&self->lock);
4585   GST_DEBUG_OBJECT (self, "set vis-plugin to '%s'", name);
4586 
4587   return TRUE;
4588 
4589 error_no_element:
4590   g_mutex_unlock (&self->lock);
4591   GST_WARNING_OBJECT (self, "could not find visualization '%s'", name);
4592   return FALSE;
4593 }
4594 
4595 /**
4596  * gst_player_get_current_visualization:
4597  * @player: #GstPlayer instance
4598  *
4599  * Returns: (transfer full): Name of the currently enabled visualization.
4600  *   g_free() after usage.
4601  */
4602 gchar *
gst_player_get_current_visualization(GstPlayer * self)4603 gst_player_get_current_visualization (GstPlayer * self)
4604 {
4605   gchar *name = NULL;
4606   GstElement *vis_plugin = NULL;
4607 
4608   g_return_val_if_fail (GST_IS_PLAYER (self), NULL);
4609 
4610   if (!is_track_enabled (self, GST_PLAY_FLAG_VIS))
4611     return NULL;
4612 
4613   g_object_get (self->playbin, "vis-plugin", &vis_plugin, NULL);
4614 
4615   if (vis_plugin) {
4616     GstElementFactory *factory = gst_element_get_factory (vis_plugin);
4617     if (factory)
4618       name = g_strdup (gst_plugin_feature_get_name (factory));
4619     gst_object_unref (vis_plugin);
4620   }
4621 
4622   GST_DEBUG_OBJECT (self, "vis-plugin '%s' %p", name, vis_plugin);
4623 
4624   return name;
4625 }
4626 
4627 /**
4628  * gst_player_set_visualization_enabled:
4629  * @player: #GstPlayer instance
4630  * @enabled: TRUE or FALSE
4631  *
4632  * Enable or disable the visualization.
4633  */
4634 void
gst_player_set_visualization_enabled(GstPlayer * self,gboolean enabled)4635 gst_player_set_visualization_enabled (GstPlayer * self, gboolean enabled)
4636 {
4637   g_return_if_fail (GST_IS_PLAYER (self));
4638 
4639   if (enabled)
4640     player_set_flag (self, GST_PLAY_FLAG_VIS);
4641   else
4642     player_clear_flag (self, GST_PLAY_FLAG_VIS);
4643 
4644   GST_DEBUG_OBJECT (self, "visualization is '%s'",
4645       enabled ? "Enabled" : "Disabled");
4646 }
4647 
4648 struct CBChannelMap
4649 {
4650   const gchar *label;           /* channel label name */
4651   const gchar *name;            /* get_name () */
4652 };
4653 
4654 static const struct CBChannelMap cb_channel_map[] = {
4655   /* GST_PLAYER_COLOR_BALANCE_BRIGHTNESS */ {"BRIGHTNESS", "brightness"},
4656   /* GST_PLAYER_COLOR_BALANCE_CONTRAST   */ {"CONTRAST", "contrast"},
4657   /* GST_PLAYER_COLOR_BALANCE_SATURATION */ {"SATURATION", "saturation"},
4658   /* GST_PLAYER_COLOR_BALANCE_HUE        */ {"HUE", "hue"},
4659 };
4660 
4661 static GstColorBalanceChannel *
gst_player_color_balance_find_channel(GstPlayer * self,GstPlayerColorBalanceType type)4662 gst_player_color_balance_find_channel (GstPlayer * self,
4663     GstPlayerColorBalanceType type)
4664 {
4665   GstColorBalanceChannel *channel;
4666   const GList *l, *channels;
4667 
4668   if (type < GST_PLAYER_COLOR_BALANCE_BRIGHTNESS ||
4669       type > GST_PLAYER_COLOR_BALANCE_HUE)
4670     return NULL;
4671 
4672   channels =
4673       gst_color_balance_list_channels (GST_COLOR_BALANCE (self->playbin));
4674   for (l = channels; l; l = l->next) {
4675     channel = l->data;
4676     if (g_strrstr (channel->label, cb_channel_map[type].label))
4677       return channel;
4678   }
4679 
4680   return NULL;
4681 }
4682 
4683 /**
4684  * gst_player_has_color_balance:
4685  * @player:#GstPlayer instance
4686  *
4687  * Checks whether the @player has color balance support available.
4688  *
4689  * Returns: %TRUE if @player has color balance support. Otherwise,
4690  *   %FALSE.
4691  */
4692 gboolean
gst_player_has_color_balance(GstPlayer * self)4693 gst_player_has_color_balance (GstPlayer * self)
4694 {
4695   const GList *channels;
4696 
4697   g_return_val_if_fail (GST_IS_PLAYER (self), FALSE);
4698 
4699   if (!GST_IS_COLOR_BALANCE (self->playbin))
4700     return FALSE;
4701 
4702   channels =
4703       gst_color_balance_list_channels (GST_COLOR_BALANCE (self->playbin));
4704   return (channels != NULL);
4705 }
4706 
4707 /**
4708  * gst_player_set_color_balance:
4709  * @player: #GstPlayer instance
4710  * @type: #GstPlayerColorBalanceType
4711  * @value: The new value for the @type, ranged [0,1]
4712  *
4713  * Sets the current value of the indicated channel @type to the passed
4714  * value.
4715  */
4716 void
gst_player_set_color_balance(GstPlayer * self,GstPlayerColorBalanceType type,gdouble value)4717 gst_player_set_color_balance (GstPlayer * self, GstPlayerColorBalanceType type,
4718     gdouble value)
4719 {
4720   GstColorBalanceChannel *channel;
4721   gdouble new_val;
4722 
4723   g_return_if_fail (GST_IS_PLAYER (self));
4724   g_return_if_fail (value >= 0.0 && value <= 1.0);
4725 
4726   if (!GST_IS_COLOR_BALANCE (self->playbin))
4727     return;
4728 
4729   channel = gst_player_color_balance_find_channel (self, type);
4730   if (!channel)
4731     return;
4732 
4733   value = CLAMP (value, 0.0, 1.0);
4734 
4735   /* Convert to channel range */
4736   new_val = channel->min_value + value * ((gdouble) channel->max_value -
4737       (gdouble) channel->min_value);
4738 
4739   gst_color_balance_set_value (GST_COLOR_BALANCE (self->playbin), channel,
4740       new_val);
4741 }
4742 
4743 /**
4744  * gst_player_get_color_balance:
4745  * @player: #GstPlayer instance
4746  * @type: #GstPlayerColorBalanceType
4747  *
4748  * Retrieve the current value of the indicated @type.
4749  *
4750  * Returns: The current value of @type, between [0,1]. In case of
4751  *   error -1 is returned.
4752  */
4753 gdouble
gst_player_get_color_balance(GstPlayer * self,GstPlayerColorBalanceType type)4754 gst_player_get_color_balance (GstPlayer * self, GstPlayerColorBalanceType type)
4755 {
4756   GstColorBalanceChannel *channel;
4757   gint value;
4758 
4759   g_return_val_if_fail (GST_IS_PLAYER (self), -1);
4760 
4761   if (!GST_IS_COLOR_BALANCE (self->playbin))
4762     return -1;
4763 
4764   channel = gst_player_color_balance_find_channel (self, type);
4765   if (!channel)
4766     return -1;
4767 
4768   value = gst_color_balance_get_value (GST_COLOR_BALANCE (self->playbin),
4769       channel);
4770 
4771   return ((gdouble) value -
4772       (gdouble) channel->min_value) / ((gdouble) channel->max_value -
4773       (gdouble) channel->min_value);
4774 }
4775 
4776 /**
4777  * gst_player_get_multiview_mode:
4778  * @player: #GstPlayer instance
4779  *
4780  * Retrieve the current value of the indicated @type.
4781  *
4782  * Returns: The current value of @type, Default: -1 "none"
4783  *
4784  * Since: 1.10
4785  */
4786 GstVideoMultiviewFramePacking
gst_player_get_multiview_mode(GstPlayer * self)4787 gst_player_get_multiview_mode (GstPlayer * self)
4788 {
4789   GstVideoMultiviewFramePacking val = GST_VIDEO_MULTIVIEW_FRAME_PACKING_NONE;
4790 
4791   g_return_val_if_fail (GST_IS_PLAYER (self),
4792       GST_VIDEO_MULTIVIEW_FRAME_PACKING_NONE);
4793 
4794   g_object_get (self, "video-multiview-mode", &val, NULL);
4795 
4796   return val;
4797 }
4798 
4799 /**
4800  * gst_player_set_multiview_mode:
4801  * @player: #GstPlayer instance
4802  * @mode: The new value for the @type
4803  *
4804  * Sets the current value of the indicated mode @type to the passed
4805  * value.
4806  *
4807  * Since: 1.10
4808  */
4809 void
gst_player_set_multiview_mode(GstPlayer * self,GstVideoMultiviewFramePacking mode)4810 gst_player_set_multiview_mode (GstPlayer * self,
4811     GstVideoMultiviewFramePacking mode)
4812 {
4813   g_return_if_fail (GST_IS_PLAYER (self));
4814 
4815   g_object_set (self, "video-multiview-mode", mode, NULL);
4816 }
4817 
4818 /**
4819  * gst_player_get_multiview_flags:
4820  * @player: #GstPlayer instance
4821  *
4822  * Retrieve the current value of the indicated @type.
4823  *
4824  * Returns: The current value of @type, Default: 0x00000000 "none
4825  *
4826  * Since: 1.10
4827  */
4828 GstVideoMultiviewFlags
gst_player_get_multiview_flags(GstPlayer * self)4829 gst_player_get_multiview_flags (GstPlayer * self)
4830 {
4831   GstVideoMultiviewFlags val = GST_VIDEO_MULTIVIEW_FLAGS_NONE;
4832 
4833   g_return_val_if_fail (GST_IS_PLAYER (self), val);
4834 
4835   g_object_get (self, "video-multiview-flags", &val, NULL);
4836 
4837   return val;
4838 }
4839 
4840 /**
4841  * gst_player_set_multiview_flags:
4842  * @player: #GstPlayer instance
4843  * @flags: The new value for the @type
4844  *
4845  * Sets the current value of the indicated mode @type to the passed
4846  * value.
4847  *
4848  * Since: 1.10
4849  */
4850 void
gst_player_set_multiview_flags(GstPlayer * self,GstVideoMultiviewFlags flags)4851 gst_player_set_multiview_flags (GstPlayer * self, GstVideoMultiviewFlags flags)
4852 {
4853   g_return_if_fail (GST_IS_PLAYER (self));
4854 
4855   g_object_set (self, "video-multiview-flags", flags, NULL);
4856 }
4857 
4858 /**
4859  * gst_player_get_audio_video_offset:
4860  * @player: #GstPlayer instance
4861  *
4862  * Retrieve the current value of audio-video-offset property
4863  *
4864  * Returns: The current value of audio-video-offset in nanoseconds
4865  *
4866  * Since: 1.10
4867  */
4868 gint64
gst_player_get_audio_video_offset(GstPlayer * self)4869 gst_player_get_audio_video_offset (GstPlayer * self)
4870 {
4871   gint64 val = 0;
4872 
4873   g_return_val_if_fail (GST_IS_PLAYER (self), DEFAULT_AUDIO_VIDEO_OFFSET);
4874 
4875   g_object_get (self, "audio-video-offset", &val, NULL);
4876 
4877   return val;
4878 }
4879 
4880 /**
4881  * gst_player_set_audio_video_offset:
4882  * @player: #GstPlayer instance
4883  * @offset: #gint64 in nanoseconds
4884  *
4885  * Sets audio-video-offset property by value of @offset
4886  *
4887  * Since: 1.10
4888  */
4889 void
gst_player_set_audio_video_offset(GstPlayer * self,gint64 offset)4890 gst_player_set_audio_video_offset (GstPlayer * self, gint64 offset)
4891 {
4892   g_return_if_fail (GST_IS_PLAYER (self));
4893 
4894   g_object_set (self, "audio-video-offset", offset, NULL);
4895 }
4896 
4897 /**
4898  * gst_player_get_subtitle_video_offset:
4899  * @player: #GstPlayer instance
4900  *
4901  * Retrieve the current value of subtitle-video-offset property
4902  *
4903  * Returns: The current value of subtitle-video-offset in nanoseconds
4904  *
4905  * Since: 1.16
4906  */
4907 gint64
gst_player_get_subtitle_video_offset(GstPlayer * self)4908 gst_player_get_subtitle_video_offset (GstPlayer * self)
4909 {
4910   gint64 val = 0;
4911 
4912   g_return_val_if_fail (GST_IS_PLAYER (self), DEFAULT_SUBTITLE_VIDEO_OFFSET);
4913 
4914   g_object_get (self, "subtitle-video-offset", &val, NULL);
4915 
4916   return val;
4917 }
4918 
4919 /**
4920  * gst_player_set_subtitle_video_offset:
4921  * @player: #GstPlayer instance
4922  * @offset: #gint64 in nanoseconds
4923  *
4924  * Sets subtitle-video-offset property by value of @offset
4925  *
4926  * Since: 1.16
4927  */
4928 void
gst_player_set_subtitle_video_offset(GstPlayer * self,gint64 offset)4929 gst_player_set_subtitle_video_offset (GstPlayer * self, gint64 offset)
4930 {
4931   g_return_if_fail (GST_IS_PLAYER (self));
4932 
4933   g_object_set (self, "subtitle-video-offset", offset, NULL);
4934 }
4935 
4936 
4937 #define C_ENUM(v) ((gint) v)
4938 #define C_FLAGS(v) ((guint) v)
4939 
4940 GType
gst_player_color_balance_type_get_type(void)4941 gst_player_color_balance_type_get_type (void)
4942 {
4943   static gsize id = 0;
4944   static const GEnumValue values[] = {
4945     {C_ENUM (GST_PLAYER_COLOR_BALANCE_HUE), "GST_PLAYER_COLOR_BALANCE_HUE",
4946         "hue"},
4947     {C_ENUM (GST_PLAYER_COLOR_BALANCE_BRIGHTNESS),
4948         "GST_PLAYER_COLOR_BALANCE_BRIGHTNESS", "brightness"},
4949     {C_ENUM (GST_PLAYER_COLOR_BALANCE_SATURATION),
4950         "GST_PLAYER_COLOR_BALANCE_SATURATION", "saturation"},
4951     {C_ENUM (GST_PLAYER_COLOR_BALANCE_CONTRAST),
4952         "GST_PLAYER_COLOR_BALANCE_CONTRAST", "contrast"},
4953     {0, NULL, NULL}
4954   };
4955 
4956   if (g_once_init_enter (&id)) {
4957     GType tmp = g_enum_register_static ("GstPlayerColorBalanceType", values);
4958     g_once_init_leave (&id, tmp);
4959   }
4960 
4961   return (GType) id;
4962 }
4963 
4964 /**
4965  * gst_player_color_balance_type_get_name:
4966  * @type: a #GstPlayerColorBalanceType
4967  *
4968  * Gets a string representing the given color balance type.
4969  *
4970  * Returns: (transfer none): a string with the name of the color
4971  *   balance type.
4972  */
4973 const gchar *
gst_player_color_balance_type_get_name(GstPlayerColorBalanceType type)4974 gst_player_color_balance_type_get_name (GstPlayerColorBalanceType type)
4975 {
4976   g_return_val_if_fail (type >= GST_PLAYER_COLOR_BALANCE_BRIGHTNESS &&
4977       type <= GST_PLAYER_COLOR_BALANCE_HUE, NULL);
4978 
4979   return cb_channel_map[type].name;
4980 }
4981 
4982 GType
gst_player_state_get_type(void)4983 gst_player_state_get_type (void)
4984 {
4985   static gsize id = 0;
4986   static const GEnumValue values[] = {
4987     {C_ENUM (GST_PLAYER_STATE_STOPPED), "GST_PLAYER_STATE_STOPPED", "stopped"},
4988     {C_ENUM (GST_PLAYER_STATE_BUFFERING), "GST_PLAYER_STATE_BUFFERING",
4989         "buffering"},
4990     {C_ENUM (GST_PLAYER_STATE_PAUSED), "GST_PLAYER_STATE_PAUSED", "paused"},
4991     {C_ENUM (GST_PLAYER_STATE_PLAYING), "GST_PLAYER_STATE_PLAYING", "playing"},
4992     {0, NULL, NULL}
4993   };
4994 
4995   if (g_once_init_enter (&id)) {
4996     GType tmp = g_enum_register_static ("GstPlayerState", values);
4997     g_once_init_leave (&id, tmp);
4998   }
4999 
5000   return (GType) id;
5001 }
5002 
5003 /**
5004  * gst_player_state_get_name:
5005  * @state: a #GstPlayerState
5006  *
5007  * Gets a string representing the given state.
5008  *
5009  * Returns: (transfer none): a string with the name of the state.
5010  */
5011 const gchar *
gst_player_state_get_name(GstPlayerState state)5012 gst_player_state_get_name (GstPlayerState state)
5013 {
5014   switch (state) {
5015     case GST_PLAYER_STATE_STOPPED:
5016       return "stopped";
5017     case GST_PLAYER_STATE_BUFFERING:
5018       return "buffering";
5019     case GST_PLAYER_STATE_PAUSED:
5020       return "paused";
5021     case GST_PLAYER_STATE_PLAYING:
5022       return "playing";
5023   }
5024 
5025   g_assert_not_reached ();
5026   return NULL;
5027 }
5028 
5029 GType
gst_player_error_get_type(void)5030 gst_player_error_get_type (void)
5031 {
5032   static gsize id = 0;
5033   static const GEnumValue values[] = {
5034     {C_ENUM (GST_PLAYER_ERROR_FAILED), "GST_PLAYER_ERROR_FAILED", "failed"},
5035     {0, NULL, NULL}
5036   };
5037 
5038   if (g_once_init_enter (&id)) {
5039     GType tmp = g_enum_register_static ("GstPlayerError", values);
5040     g_once_init_leave (&id, tmp);
5041   }
5042 
5043   return (GType) id;
5044 }
5045 
5046 /**
5047  * gst_player_error_get_name:
5048  * @error: a #GstPlayerError
5049  *
5050  * Gets a string representing the given error.
5051  *
5052  * Returns: (transfer none): a string with the given error.
5053  */
5054 const gchar *
gst_player_error_get_name(GstPlayerError error)5055 gst_player_error_get_name (GstPlayerError error)
5056 {
5057   switch (error) {
5058     case GST_PLAYER_ERROR_FAILED:
5059       return "failed";
5060   }
5061 
5062   g_assert_not_reached ();
5063   return NULL;
5064 }
5065 
5066 /**
5067  * gst_player_set_config:
5068  * @player: #GstPlayer instance
5069  * @config: (transfer full): a #GstStructure
5070  *
5071  * Set the configuration of the player. If the player is already configured, and
5072  * the configuration haven't change, this function will return %TRUE. If the
5073  * player is not in the GST_PLAYER_STATE_STOPPED, this method will return %FALSE
5074  * and active configuration will remain.
5075  *
5076  * @config is a #GstStructure that contains the configuration parameters for
5077  * the player.
5078  *
5079  * This function takes ownership of @config.
5080  *
5081  * Returns: %TRUE when the configuration could be set.
5082  * Since: 1.10
5083  */
5084 gboolean
gst_player_set_config(GstPlayer * self,GstStructure * config)5085 gst_player_set_config (GstPlayer * self, GstStructure * config)
5086 {
5087   g_return_val_if_fail (GST_IS_PLAYER (self), FALSE);
5088   g_return_val_if_fail (config != NULL, FALSE);
5089 
5090   g_mutex_lock (&self->lock);
5091 
5092   if (self->app_state != GST_PLAYER_STATE_STOPPED) {
5093     GST_INFO_OBJECT (self, "can't change config while player is %s",
5094         gst_player_state_get_name (self->app_state));
5095     g_mutex_unlock (&self->lock);
5096     return FALSE;
5097   }
5098 
5099   if (self->config)
5100     gst_structure_free (self->config);
5101   self->config = config;
5102   g_mutex_unlock (&self->lock);
5103 
5104   return TRUE;
5105 }
5106 
5107 /**
5108  * gst_player_get_config:
5109  * @player: #GstPlayer instance
5110  *
5111  * Get a copy of the current configuration of the player. This configuration
5112  * can either be modified and used for the gst_player_set_config() call
5113  * or it must be freed after usage.
5114  *
5115  * Returns: (transfer full): a copy of the current configuration of @player. Use
5116  * gst_structure_free() after usage or gst_player_set_config().
5117  *
5118  * Since: 1.10
5119  */
5120 GstStructure *
gst_player_get_config(GstPlayer * self)5121 gst_player_get_config (GstPlayer * self)
5122 {
5123   GstStructure *ret;
5124 
5125   g_return_val_if_fail (GST_IS_PLAYER (self), NULL);
5126 
5127   g_mutex_lock (&self->lock);
5128   ret = gst_structure_copy (self->config);
5129   g_mutex_unlock (&self->lock);
5130 
5131   return ret;
5132 }
5133 
5134 /**
5135  * gst_player_config_set_user_agent:
5136  * @config: a #GstPlayer configuration
5137  * @agent: the string to use as user agent
5138  *
5139  * Set the user agent to pass to the server if @player needs to connect
5140  * to a server during playback. This is typically used when playing HTTP
5141  * or RTSP streams.
5142  *
5143  * Since: 1.10
5144  */
5145 void
gst_player_config_set_user_agent(GstStructure * config,const gchar * agent)5146 gst_player_config_set_user_agent (GstStructure * config, const gchar * agent)
5147 {
5148   g_return_if_fail (config != NULL);
5149   g_return_if_fail (agent != NULL);
5150 
5151   gst_structure_id_set (config,
5152       CONFIG_QUARK (USER_AGENT), G_TYPE_STRING, agent, NULL);
5153 }
5154 
5155 /**
5156  * gst_player_config_get_user_agent:
5157  * @config: a #GstPlayer configuration
5158  *
5159  * Return the user agent which has been configured using
5160  * gst_player_config_set_user_agent() if any.
5161  *
5162  * Returns: (transfer full): the configured agent, or %NULL
5163  * Since: 1.10
5164  */
5165 gchar *
gst_player_config_get_user_agent(const GstStructure * config)5166 gst_player_config_get_user_agent (const GstStructure * config)
5167 {
5168   gchar *agent = NULL;
5169 
5170   g_return_val_if_fail (config != NULL, NULL);
5171 
5172   gst_structure_id_get (config,
5173       CONFIG_QUARK (USER_AGENT), G_TYPE_STRING, &agent, NULL);
5174 
5175   return agent;
5176 }
5177 
5178 /**
5179  * gst_player_config_set_position_update_interval:
5180  * @config: a #GstPlayer configuration
5181  * @interval: interval in ms
5182  *
5183  * set interval in milliseconds between two position-updated signals.
5184  * pass 0 to stop updating the position.
5185  * Since: 1.10
5186  */
5187 void
gst_player_config_set_position_update_interval(GstStructure * config,guint interval)5188 gst_player_config_set_position_update_interval (GstStructure * config,
5189     guint interval)
5190 {
5191   g_return_if_fail (config != NULL);
5192   g_return_if_fail (interval <= 10000);
5193 
5194   gst_structure_id_set (config,
5195       CONFIG_QUARK (POSITION_INTERVAL_UPDATE), G_TYPE_UINT, interval, NULL);
5196 }
5197 
5198 /**
5199  * gst_player_config_get_position_update_interval:
5200  * @config: a #GstPlayer configuration
5201  *
5202  * Returns: current position update interval in milliseconds
5203  *
5204  * Since: 1.10
5205  */
5206 guint
gst_player_config_get_position_update_interval(const GstStructure * config)5207 gst_player_config_get_position_update_interval (const GstStructure * config)
5208 {
5209   guint interval = DEFAULT_POSITION_UPDATE_INTERVAL_MS;
5210 
5211   g_return_val_if_fail (config != NULL, DEFAULT_POSITION_UPDATE_INTERVAL_MS);
5212 
5213   gst_structure_id_get (config,
5214       CONFIG_QUARK (POSITION_INTERVAL_UPDATE), G_TYPE_UINT, &interval, NULL);
5215 
5216   return interval;
5217 }
5218 
5219 /**
5220  * gst_player_config_set_seek_accurate:
5221  * @config: a #GstPlayer configuration
5222  * @accurate: accurate seek or not
5223  *
5224  * Enable or disable accurate seeking. When enabled, elements will try harder
5225  * to seek as accurately as possible to the requested seek position. Generally
5226  * it will be slower especially for formats that don't have any indexes or
5227  * timestamp markers in the stream.
5228  *
5229  * If accurate seeking is disabled, elements will seek as close as the request
5230  * position without slowing down seeking too much.
5231  *
5232  * Accurate seeking is disabled by default.
5233  *
5234  * Since: 1.12
5235  */
5236 void
gst_player_config_set_seek_accurate(GstStructure * config,gboolean accurate)5237 gst_player_config_set_seek_accurate (GstStructure * config, gboolean accurate)
5238 {
5239   g_return_if_fail (config != NULL);
5240 
5241   gst_structure_id_set (config,
5242       CONFIG_QUARK (ACCURATE_SEEK), G_TYPE_BOOLEAN, accurate, NULL);
5243 }
5244 
5245 /**
5246  * gst_player_config_get_seek_accurate:
5247  * @config: a #GstPlayer configuration
5248  *
5249  * Returns: %TRUE if accurate seeking is enabled
5250  *
5251  * Since: 1.12
5252  */
5253 gboolean
gst_player_config_get_seek_accurate(const GstStructure * config)5254 gst_player_config_get_seek_accurate (const GstStructure * config)
5255 {
5256   gboolean accurate = FALSE;
5257 
5258   g_return_val_if_fail (config != NULL, FALSE);
5259 
5260   gst_structure_id_get (config,
5261       CONFIG_QUARK (ACCURATE_SEEK), G_TYPE_BOOLEAN, &accurate, NULL);
5262 
5263   return accurate;
5264 }
5265 
5266 /**
5267  * gst_player_get_video_snapshot:
5268  * @player: #GstPlayer instance
5269  * @format: output format of the video snapshot
5270  * @config: (allow-none): Additional configuration
5271  *
5272  * Get a snapshot of the currently selected video stream, if any. The format can be
5273  * selected with @format and optional configuration is possible with @config
5274  * Currently supported settings are:
5275  * - width, height of type G_TYPE_INT
5276  * - pixel-aspect-ratio of type GST_TYPE_FRACTION
5277  *  Except for GST_PLAYER_THUMBNAIL_RAW_NATIVE format, if no config is set, pixel-aspect-ratio would be 1/1
5278  *
5279  * Returns: (transfer full):  Current video snapshot sample or %NULL on failure
5280  *
5281  * Since: 1.12
5282  */
5283 GstSample *
gst_player_get_video_snapshot(GstPlayer * self,GstPlayerSnapshotFormat format,const GstStructure * config)5284 gst_player_get_video_snapshot (GstPlayer * self,
5285     GstPlayerSnapshotFormat format, const GstStructure * config)
5286 {
5287   gint video_tracks = 0;
5288   GstSample *sample = NULL;
5289   GstCaps *caps = NULL;
5290   gint width = -1;
5291   gint height = -1;
5292   gint par_n = 1;
5293   gint par_d = 1;
5294   g_return_val_if_fail (GST_IS_PLAYER (self), NULL);
5295 
5296   g_object_get (self->playbin, "n-video", &video_tracks, NULL);
5297   if (video_tracks == 0) {
5298     GST_DEBUG_OBJECT (self, "total video track num is 0");
5299     return NULL;
5300   }
5301 
5302   switch (format) {
5303     case GST_PLAYER_THUMBNAIL_RAW_xRGB:
5304       caps = gst_caps_new_simple ("video/x-raw",
5305           "format", G_TYPE_STRING, "xRGB", NULL);
5306       break;
5307     case GST_PLAYER_THUMBNAIL_RAW_BGRx:
5308       caps = gst_caps_new_simple ("video/x-raw",
5309           "format", G_TYPE_STRING, "BGRx", NULL);
5310       break;
5311     case GST_PLAYER_THUMBNAIL_JPG:
5312       caps = gst_caps_new_empty_simple ("image/jpeg");
5313       break;
5314     case GST_PLAYER_THUMBNAIL_PNG:
5315       caps = gst_caps_new_empty_simple ("image/png");
5316       break;
5317     case GST_PLAYER_THUMBNAIL_RAW_NATIVE:
5318     default:
5319       caps = gst_caps_new_empty_simple ("video/x-raw");
5320       break;
5321   }
5322 
5323   if (NULL != config) {
5324     if (!gst_structure_get_int (config, "width", &width))
5325       width = -1;
5326     if (!gst_structure_get_int (config, "height", &height))
5327       height = -1;
5328     if (!gst_structure_get_fraction (config, "pixel-aspect-ratio", &par_n,
5329             &par_d)) {
5330       if (format != GST_PLAYER_THUMBNAIL_RAW_NATIVE) {
5331         par_n = 1;
5332         par_d = 1;
5333       } else {
5334         par_n = 0;
5335         par_d = 0;
5336       }
5337     }
5338   }
5339 
5340   if (width > 0 && height > 0) {
5341     gst_caps_set_simple (caps, "width", G_TYPE_INT, width,
5342         "height", G_TYPE_INT, height, NULL);
5343   }
5344 
5345   if (format != GST_PLAYER_THUMBNAIL_RAW_NATIVE) {
5346     gst_caps_set_simple (caps, "pixel-aspect-ratio", GST_TYPE_FRACTION,
5347         par_n, par_d, NULL);
5348   } else if (NULL != config && par_n != 0 && par_d != 0) {
5349     gst_caps_set_simple (caps, "pixel-aspect-ratio", GST_TYPE_FRACTION,
5350         par_n, par_d, NULL);
5351   }
5352 
5353   g_signal_emit_by_name (self->playbin, "convert-sample", caps, &sample);
5354   gst_caps_unref (caps);
5355   if (!sample) {
5356     GST_WARNING_OBJECT (self, "Failed to retrieve or convert video frame");
5357     return NULL;
5358   }
5359 
5360   return sample;
5361 }
5362