gst-rtsp-сервер gst_rtsp_client_close - PullRequest
0 голосов
/ 03 мая 2018

Я использую gst-rtsp-server и хочу иметь возможность закрыть текущий поток с сервера. Я пытаюсь использовать функцию gst_rtsp_client_close(). Однако, когда я называю это, приложение иногда перестает отвечать на запросы. В журнале я вижу: GLib-CRITICAL **: g_main_context_unref: assertion 'g_atomic_int_get (&context->ref_count) > 0' failed​.

Я изменил пример test-uri, чтобы вызвать эту проблему. Когда клиент подключается, запускается таймер. Когда таймер закончится, для этого клиента будет вызван gst_rtsp_client_close().

/* GStreamer
 * Copyright (C) 2008 Wim Taymans <wim.taymans at gmail.com>
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Library General Public
 * License as published by the Free Software Foundation; either
 * version 2 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Library General Public License for more details.
 *
 * You should have received a copy of the GNU Library General Public
 * License along with this library; if not, write to the
 * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
 * Boston, MA 02110-1301, USA.
 */

#include <gst/gst.h>

#include <gst/rtsp-server/rtsp-server.h>
#include <gst/rtsp-server/rtsp-media-factory-uri.h>

#define DEFAULT_RTSP_PORT "8554"

static char *port = (char *) DEFAULT_RTSP_PORT;

static GOptionEntry entries[] = {
  {"port", 'p', 0, G_OPTION_ARG_STRING, &port,
      "Port to listen on (default: " DEFAULT_RTSP_PORT ")", "PORT"},
  {NULL}
};

static void
client_closed(GstRTSPClient *client,
              gpointer user_data)
{
  g_print("client closed %p\n", client);
}

static gboolean
disconnect (GstRTSPClient * client)
{
  g_print("disconnect client %p\n", client);
  gst_rtsp_client_close(client);
  return FALSE;
}

static void
client_connected(GstRTSPServer *server,
                 GstRTSPClient *client,
                 gpointer user_data)
{
  g_print("client connected %p\n", client);
  g_signal_connect(client, "closed", (GCallback)client_closed,
                   NULL);
  g_timeout_add_seconds (8, (GSourceFunc) disconnect, client);
}

int
main (int argc, gchar * argv[])
{
  GMainLoop *loop;
  GstRTSPServer *server;
  GstRTSPMountPoints *mounts;
  GstRTSPMediaFactoryURI *factory;
  GOptionContext *optctx;
  GError *error = NULL;
  gchar *uri;


  guint major;
  guint minor;
  guint micro;
  guint nano;
  gst_version(&major, &minor, &micro, &nano);
  g_print("%s started. GST version %u.%u.%u.%u\n", argv[0], major, minor, micro,     nano);
  optctx = g_option_context_new ("<uri> - Test RTSP Server, URI");
  g_option_context_add_main_entries (optctx, entries, NULL);
  g_option_context_add_group (optctx, gst_init_get_option_group ());
  if (!g_option_context_parse (optctx, &argc, &argv, &error)) {
    g_printerr ("Error parsing options: %s\n", error->message);
    g_option_context_free (optctx);
    g_clear_error (&error);
    return -1;
  }
  g_option_context_free (optctx);

  if (argc < 2) {
    g_printerr ("Please pass an URI or file as argument!\n");
    return -1;
  }

  loop = g_main_loop_new (NULL, FALSE);

  /* create a server instance */
  server = gst_rtsp_server_new ();
  g_object_set (server, "service", port, NULL);
  g_signal_connect(server, "client-connected", (GCallback)client_connected,
                   NULL);
  /* get the mount points for this server, every server has a default object
   * that be used to map uri mount points to media factories */
  mounts = gst_rtsp_server_get_mount_points (server);

  /* make a URI media factory for a test stream. */
  factory = gst_rtsp_media_factory_uri_new ();

  /* check if URI is valid, otherwise convert filename to URI if it's a file */
  if (gst_uri_is_valid (argv[1])) {
    uri = g_strdup (argv[1]);
  } else if (g_file_test (argv[1], G_FILE_TEST_EXISTS)) {
    uri = gst_filename_to_uri (argv[1], NULL);
  } else {
    g_printerr ("Unrecognised command line argument '%s'.\n"
        "Please pass an URI or file as argument!\n", argv[1]);
    return -1;
  }

  gst_rtsp_media_factory_uri_set_uri (factory, uri);
  g_free (uri);

  /* attach the test factory to the /test url */
  gst_rtsp_mount_points_add_factory (mounts, "/test",
      GST_RTSP_MEDIA_FACTORY (factory));

  /* don't need the ref to the mapper anymore */
  g_object_unref (mounts);

  /* attach the server to the default maincontext */
  if (gst_rtsp_server_attach (server, NULL) == 0)
    goto failed;

  /* start serving */
  g_print ("stream ready at rtsp://127.0.0.1:%s/test\n", port);
  g_main_loop_run (loop);

  return 0;

  /* ERRORS */
failed:
  {
    g_print ("failed to attach the server\n");
    return -1;
  }
}

Я запускаю приложение с: ./close-test-uri.out ~/Videos/big_buck_bunny.mp4 и затем я подключаюсь к потоку RTSP с помощью клиента (VLC): vlc -vvv "rtsp://127.0.0.1:8554/test" Видео начнет воспроизводиться и через 8 секунд (как и ожидалось) будет отключено. Однако, как упоминалось ранее, я получаю критическую ошибку, и приложение перестает отвечать на запросы (иногда происходит сбой).

Я использую gst_rtsp_client_close() это неправильно?

Заранее спасибо!

...