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, ¤t, 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