Blame gst-libs/gst/rtsp/gstrtspmessage.c

Packit 971217
/* GStreamer
Packit 971217
 * Copyright (C) <2005,2006> Wim Taymans <wim@fluendo.com>
Packit 971217
 *               <2006> Lutz Mueller <lutz at topfrose dot de>
Packit 971217
 *               <2015> Tim-Philipp Müller <tim@centricular.com>
Packit 971217
 *
Packit 971217
 * This library is free software; you can redistribute it and/or
Packit 971217
 * modify it under the terms of the GNU Library General Public
Packit 971217
 * License as published by the Free Software Foundation; either
Packit 971217
 * version 2 of the License, or (at your option) any later version.
Packit 971217
 *
Packit 971217
 * This library is distributed in the hope that it will be useful,
Packit 971217
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 971217
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 971217
 * Library General Public License for more details.
Packit 971217
 *
Packit 971217
 * You should have received a copy of the GNU Library General Public
Packit 971217
 * License along with this library; if not, write to the
Packit 971217
 * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
Packit 971217
 * Boston, MA 02110-1301, USA.
Packit 971217
 */
Packit 971217
/*
Packit 971217
 * Unless otherwise indicated, Source Code is licensed under MIT license.
Packit 971217
 * See further explanation attached in License Statement (distributed in the file
Packit 971217
 * LICENSE).
Packit 971217
 *
Packit 971217
 * Permission is hereby granted, free of charge, to any person obtaining a copy of
Packit 971217
 * this software and associated documentation files (the "Software"), to deal in
Packit 971217
 * the Software without restriction, including without limitation the rights to
Packit 971217
 * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
Packit 971217
 * of the Software, and to permit persons to whom the Software is furnished to do
Packit 971217
 * so, subject to the following conditions:
Packit 971217
 *
Packit 971217
 * The above copyright notice and this permission notice shall be included in all
Packit 971217
 * copies or substantial portions of the Software.
Packit 971217
 *
Packit 971217
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
Packit 971217
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
Packit 971217
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
Packit 971217
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
Packit 971217
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
Packit 971217
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
Packit 971217
 * SOFTWARE.
Packit 971217
 */
Packit 971217
Packit 971217
/**
Packit 971217
 * SECTION:gstrtspmessage
Packit 971217
 * @title: GstRTSPMessage
Packit 971217
 * @short_description: RTSP messages
Packit 971217
 * @see_also: gstrtspconnection
Packit 971217
 *
Packit 971217
 * Provides methods for creating and parsing request, response and data messages.
Packit 971217
 */
Packit 971217
Packit 971217
#include <string.h>
Packit 971217
Packit 971217
#include <gst/gstutils.h>
Packit 971217
#include "gstrtspmessage.h"
Packit 971217
Packit 971217
typedef struct _RTSPKeyValue
Packit 971217
{
Packit 971217
  GstRTSPHeaderField field;
Packit 971217
  gchar *value;
Packit 971217
  gchar *custom_key;            /* custom header string (field is INVALID then) */
Packit 971217
} RTSPKeyValue;
Packit 971217
Packit 971217
static void
Packit 971217
key_value_foreach (GArray * array, GFunc func, gpointer user_data)
Packit 971217
{
Packit 971217
  guint i;
Packit 971217
Packit 971217
  g_return_if_fail (array != NULL);
Packit 971217
Packit 971217
  for (i = 0; i < array->len; i++) {
Packit 971217
    (*func) (&g_array_index (array, RTSPKeyValue, i), user_data);
Packit 971217
  }
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
key_value_append (const RTSPKeyValue * kv, GArray * array)
Packit 971217
{
Packit 971217
  RTSPKeyValue kvcopy;
Packit 971217
  g_return_if_fail (kv != NULL);
Packit 971217
  g_return_if_fail (array != NULL);
Packit 971217
Packit 971217
  kvcopy.field = kv->field;
Packit 971217
  kvcopy.value = g_strdup (kv->value);
Packit 971217
  kvcopy.custom_key = g_strdup (kv->custom_key);
Packit 971217
Packit 971217
  g_array_append_val (array, kvcopy);
Packit 971217
}
Packit 971217
Packit 971217
static GstRTSPMessage *
Packit 971217
gst_rtsp_message_boxed_copy (GstRTSPMessage * orig)
Packit 971217
{
Packit 971217
  GstRTSPMessage *copy;
Packit 971217
Packit 971217
  if (gst_rtsp_message_copy (orig, &copy) == GST_RTSP_OK)
Packit 971217
    return copy;
Packit 971217
Packit 971217
  return NULL;
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_rtsp_message_boxed_free (GstRTSPMessage * msg)
Packit 971217
{
Packit 971217
  gst_rtsp_message_free (msg);
Packit 971217
}
Packit 971217
Packit 971217
G_DEFINE_BOXED_TYPE (GstRTSPMessage, gst_rtsp_msg, gst_rtsp_message_boxed_copy,
Packit 971217
    gst_rtsp_message_boxed_free);
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_new:
Packit 971217
 * @msg: (out) (transfer full): a location for the new #GstRTSPMessage
Packit 971217
 *
Packit 971217
 * Create a new initialized #GstRTSPMessage. Free with gst_rtsp_message_free().
Packit 971217
 *
Packit 971217
 * Returns: a #GstRTSPResult.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_new (GstRTSPMessage ** msg)
Packit 971217
{
Packit 971217
  GstRTSPMessage *newmsg;
Packit 971217
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  newmsg = g_new0 (GstRTSPMessage, 1);
Packit 971217
Packit 971217
  *msg = newmsg;
Packit 971217
Packit 971217
  return gst_rtsp_message_init (newmsg);
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_init:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 *
Packit 971217
 * Initialize @msg. This function is mostly used when @msg is allocated on the
Packit 971217
 * stack. The reverse operation of this is gst_rtsp_message_unset().
Packit 971217
 *
Packit 971217
 * Returns: a #GstRTSPResult.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_init (GstRTSPMessage * msg)
Packit 971217
{
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  gst_rtsp_message_unset (msg);
Packit 971217
Packit 971217
  msg->type = GST_RTSP_MESSAGE_INVALID;
Packit 971217
  msg->hdr_fields = g_array_new (FALSE, FALSE, sizeof (RTSPKeyValue));
Packit 971217
Packit 971217
  return GST_RTSP_OK;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_get_type:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 *
Packit 971217
 * Get the message type of @msg.
Packit 971217
 *
Packit 971217
 * Returns: the message type.
Packit 971217
 */
Packit 971217
GstRTSPMsgType
Packit 971217
gst_rtsp_message_get_type (GstRTSPMessage * msg)
Packit 971217
{
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_MESSAGE_INVALID);
Packit 971217
Packit 971217
  return msg->type;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_new_request:
Packit 971217
 * @msg: (out) (transfer full): a location for the new #GstRTSPMessage
Packit 971217
 * @method: the request method to use
Packit 971217
 * @uri: (transfer none): the uri of the request
Packit 971217
 *
Packit 971217
 * Create a new #GstRTSPMessage with @method and @uri and store the result
Packit 971217
 * request message in @msg. Free with gst_rtsp_message_free().
Packit 971217
 *
Packit 971217
 * Returns: a #GstRTSPResult.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_new_request (GstRTSPMessage ** msg, GstRTSPMethod method,
Packit 971217
    const gchar * uri)
Packit 971217
{
Packit 971217
  GstRTSPMessage *newmsg;
Packit 971217
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
  g_return_val_if_fail (uri != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  newmsg = g_new0 (GstRTSPMessage, 1);
Packit 971217
Packit 971217
  *msg = newmsg;
Packit 971217
Packit 971217
  return gst_rtsp_message_init_request (newmsg, method, uri);
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_init_request:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 * @method: the request method to use
Packit 971217
 * @uri: (transfer none): the uri of the request
Packit 971217
 *
Packit 971217
 * Initialize @msg as a request message with @method and @uri. To clear @msg
Packit 971217
 * again, use gst_rtsp_message_unset().
Packit 971217
 *
Packit 971217
 * Returns: a #GstRTSPResult.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_init_request (GstRTSPMessage * msg, GstRTSPMethod method,
Packit 971217
    const gchar * uri)
Packit 971217
{
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
  g_return_val_if_fail (uri != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  gst_rtsp_message_unset (msg);
Packit 971217
Packit 971217
  msg->type = GST_RTSP_MESSAGE_REQUEST;
Packit 971217
  msg->type_data.request.method = method;
Packit 971217
  msg->type_data.request.uri = g_strdup (uri);
Packit 971217
  msg->type_data.request.version = GST_RTSP_VERSION_1_0;
Packit 971217
  msg->hdr_fields = g_array_new (FALSE, FALSE, sizeof (RTSPKeyValue));
Packit 971217
Packit 971217
  return GST_RTSP_OK;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_parse_request:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 * @method: (out) (allow-none): location to hold the method
Packit 971217
 * @uri: (out) (allow-none) (transfer none): location to hold the uri
Packit 971217
 * @version: (out) (allow-none) (transfer none): location to hold the version
Packit 971217
 *
Packit 971217
 * Parse the request message @msg and store the values @method, @uri and
Packit 971217
 * @version. The result locations can be %NULL if one is not interested in its
Packit 971217
 * value.
Packit 971217
 *
Packit 971217
 * @uri remains valid for as long as @msg is valid and unchanged.
Packit 971217
 *
Packit 971217
 * Returns: a #GstRTSPResult.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_parse_request (GstRTSPMessage * msg,
Packit 971217
    GstRTSPMethod * method, const gchar ** uri, GstRTSPVersion * version)
Packit 971217
{
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
  g_return_val_if_fail (msg->type == GST_RTSP_MESSAGE_REQUEST ||
Packit 971217
      msg->type == GST_RTSP_MESSAGE_HTTP_REQUEST, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  if (method)
Packit 971217
    *method = msg->type_data.request.method;
Packit 971217
  if (uri)
Packit 971217
    *uri = msg->type_data.request.uri;
Packit 971217
  if (version)
Packit 971217
    *version = msg->type_data.request.version;
Packit 971217
Packit 971217
  return GST_RTSP_OK;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_new_response:
Packit 971217
 * @msg: (out) (transfer full): a location for the new #GstRTSPMessage
Packit 971217
 * @code: the status code
Packit 971217
 * @reason: (transfer none) (allow-none): the status reason or %NULL
Packit 971217
 * @request: (transfer none) (allow-none): the request that triggered the response or %NULL
Packit 971217
 *
Packit 971217
 * Create a new response #GstRTSPMessage with @code and @reason and store the
Packit 971217
 * result message in @msg. Free with gst_rtsp_message_free().
Packit 971217
 *
Packit 971217
 * When @reason is %NULL, the default reason for @code will be used.
Packit 971217
 *
Packit 971217
 * When @request is not %NULL, the relevant headers will be copied to the new
Packit 971217
 * response message.
Packit 971217
 *
Packit 971217
 * Returns: a #GstRTSPResult.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_new_response (GstRTSPMessage ** msg, GstRTSPStatusCode code,
Packit 971217
    const gchar * reason, const GstRTSPMessage * request)
Packit 971217
{
Packit 971217
  GstRTSPMessage *newmsg;
Packit 971217
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  newmsg = g_new0 (GstRTSPMessage, 1);
Packit 971217
Packit 971217
  *msg = newmsg;
Packit 971217
Packit 971217
  return gst_rtsp_message_init_response (newmsg, code, reason, request);
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_init_response:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 * @code: the status code
Packit 971217
 * @reason: (transfer none) (allow-none): the status reason or %NULL
Packit 971217
 * @request: (transfer none) (allow-none): the request that triggered the response or %NULL
Packit 971217
 *
Packit 971217
 * Initialize @msg with @code and @reason.
Packit 971217
 *
Packit 971217
 * When @reason is %NULL, the default reason for @code will be used.
Packit 971217
 *
Packit 971217
 * When @request is not %NULL, the relevant headers will be copied to the new
Packit 971217
 * response message.
Packit 971217
 *
Packit 971217
 * Returns: a #GstRTSPResult.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_init_response (GstRTSPMessage * msg, GstRTSPStatusCode code,
Packit 971217
    const gchar * reason, const GstRTSPMessage * request)
Packit 971217
{
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  gst_rtsp_message_unset (msg);
Packit 971217
Packit 971217
  if (reason == NULL)
Packit 971217
    reason = gst_rtsp_status_as_text (code);
Packit 971217
Packit 971217
  msg->type = GST_RTSP_MESSAGE_RESPONSE;
Packit 971217
  msg->type_data.response.code = code;
Packit 971217
  msg->type_data.response.reason = g_strdup (reason);
Packit 971217
  msg->type_data.response.version = GST_RTSP_VERSION_1_0;
Packit 971217
  msg->hdr_fields = g_array_new (FALSE, FALSE, sizeof (RTSPKeyValue));
Packit 971217
Packit 971217
  if (request) {
Packit 971217
    if (request->type == GST_RTSP_MESSAGE_HTTP_REQUEST) {
Packit 971217
      msg->type = GST_RTSP_MESSAGE_HTTP_RESPONSE;
Packit 971217
      if (request->type_data.request.version != GST_RTSP_VERSION_INVALID)
Packit 971217
        msg->type_data.response.version = request->type_data.request.version;
Packit 971217
      else
Packit 971217
        msg->type_data.response.version = GST_RTSP_VERSION_1_1;
Packit 971217
    } else {
Packit 971217
      gchar *header;
Packit 971217
Packit 971217
      /* copy CSEQ */
Packit 971217
      if (gst_rtsp_message_get_header (request, GST_RTSP_HDR_CSEQ, &header,
Packit 971217
              0) == GST_RTSP_OK) {
Packit 971217
        gst_rtsp_message_add_header (msg, GST_RTSP_HDR_CSEQ, header);
Packit 971217
      }
Packit 971217
Packit 971217
      /* copy session id */
Packit 971217
      if (gst_rtsp_message_get_header (request, GST_RTSP_HDR_SESSION, &header,
Packit 971217
              0) == GST_RTSP_OK) {
Packit 971217
        char *pos;
Packit 971217
Packit 971217
        header = g_strdup (header);
Packit 971217
        if ((pos = strchr (header, ';'))) {
Packit 971217
          *pos = '\0';
Packit 971217
        }
Packit 971217
        g_strchomp (header);
Packit 971217
        gst_rtsp_message_take_header (msg, GST_RTSP_HDR_SESSION, header);
Packit 971217
      }
Packit 971217
Packit 971217
      /* FIXME copy more headers? */
Packit 971217
    }
Packit 971217
  }
Packit 971217
Packit 971217
  return GST_RTSP_OK;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_parse_response:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 * @code: (out) (allow-none): location to hold the status code
Packit 971217
 * @reason: (out) (allow-none) (transfer none): location to hold the status reason
Packit 971217
 * @version: (out) (allow-none) (transfer none): location to hold the version
Packit 971217
 *
Packit 971217
 * Parse the response message @msg and store the values @code, @reason and
Packit 971217
 * @version. The result locations can be %NULL if one is not interested in its
Packit 971217
 * value.
Packit 971217
 *
Packit 971217
 * @reason remains valid for as long as @msg is valid and unchanged.
Packit 971217
 *
Packit 971217
 * Returns: a #GstRTSPResult.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_parse_response (GstRTSPMessage * msg,
Packit 971217
    GstRTSPStatusCode * code, const gchar ** reason, GstRTSPVersion * version)
Packit 971217
{
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
  g_return_val_if_fail (msg->type == GST_RTSP_MESSAGE_RESPONSE ||
Packit 971217
      msg->type == GST_RTSP_MESSAGE_HTTP_RESPONSE, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  if (code)
Packit 971217
    *code = msg->type_data.response.code;
Packit 971217
  if (reason)
Packit 971217
    *reason = msg->type_data.response.reason;
Packit 971217
  if (version)
Packit 971217
    *version = msg->type_data.response.version;
Packit 971217
Packit 971217
  return GST_RTSP_OK;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_new_data:
Packit 971217
 * @msg: (out) (transfer full): a location for the new #GstRTSPMessage
Packit 971217
 * @channel: the channel
Packit 971217
 *
Packit 971217
 * Create a new data #GstRTSPMessage with @channel and store the
Packit 971217
 * result message in @msg. Free with gst_rtsp_message_free().
Packit 971217
 *
Packit 971217
 * Returns: a #GstRTSPResult.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_new_data (GstRTSPMessage ** msg, guint8 channel)
Packit 971217
{
Packit 971217
  GstRTSPMessage *newmsg;
Packit 971217
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  newmsg = g_new0 (GstRTSPMessage, 1);
Packit 971217
Packit 971217
  *msg = newmsg;
Packit 971217
Packit 971217
  return gst_rtsp_message_init_data (newmsg, channel);
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_init_data:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 * @channel: a channel
Packit 971217
 *
Packit 971217
 * Initialize a new data #GstRTSPMessage for @channel.
Packit 971217
 *
Packit 971217
 * Returns: a #GstRTSPResult.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_init_data (GstRTSPMessage * msg, guint8 channel)
Packit 971217
{
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  gst_rtsp_message_unset (msg);
Packit 971217
Packit 971217
  msg->type = GST_RTSP_MESSAGE_DATA;
Packit 971217
  msg->type_data.data.channel = channel;
Packit 971217
Packit 971217
  return GST_RTSP_OK;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_parse_data:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 * @channel: (out): location to hold the channel
Packit 971217
 *
Packit 971217
 * Parse the data message @msg and store the channel in @channel.
Packit 971217
 *
Packit 971217
 * Returns: a #GstRTSPResult.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_parse_data (GstRTSPMessage * msg, guint8 * channel)
Packit 971217
{
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
  g_return_val_if_fail (msg->type == GST_RTSP_MESSAGE_DATA, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  if (channel)
Packit 971217
    *channel = msg->type_data.data.channel;
Packit 971217
Packit 971217
  return GST_RTSP_OK;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_unset:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 *
Packit 971217
 * Unset the contents of @msg so that it becomes an uninitialized
Packit 971217
 * #GstRTSPMessage again. This function is mostly used in combination with
Packit 971217
 * gst_rtsp_message_init_request(), gst_rtsp_message_init_response() and
Packit 971217
 * gst_rtsp_message_init_data() on stack allocated #GstRTSPMessage structures.
Packit 971217
 *
Packit 971217
 * Returns: #GST_RTSP_OK.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_unset (GstRTSPMessage * msg)
Packit 971217
{
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  switch (msg->type) {
Packit 971217
    case GST_RTSP_MESSAGE_INVALID:
Packit 971217
      break;
Packit 971217
    case GST_RTSP_MESSAGE_REQUEST:
Packit 971217
    case GST_RTSP_MESSAGE_HTTP_REQUEST:
Packit 971217
      g_free (msg->type_data.request.uri);
Packit 971217
      break;
Packit 971217
    case GST_RTSP_MESSAGE_RESPONSE:
Packit 971217
    case GST_RTSP_MESSAGE_HTTP_RESPONSE:
Packit 971217
      g_free (msg->type_data.response.reason);
Packit 971217
      break;
Packit 971217
    case GST_RTSP_MESSAGE_DATA:
Packit 971217
      break;
Packit 971217
    default:
Packit 971217
      g_return_val_if_reached (GST_RTSP_EINVAL);
Packit 971217
  }
Packit 971217
Packit 971217
  if (msg->hdr_fields != NULL) {
Packit 971217
    guint i;
Packit 971217
Packit 971217
    for (i = 0; i < msg->hdr_fields->len; i++) {
Packit 971217
      RTSPKeyValue *keyval = &g_array_index (msg->hdr_fields, RTSPKeyValue, i);
Packit 971217
Packit 971217
      g_free (keyval->value);
Packit 971217
      g_free (keyval->custom_key);
Packit 971217
    }
Packit 971217
    g_array_free (msg->hdr_fields, TRUE);
Packit 971217
  }
Packit 971217
  g_free (msg->body);
Packit 971217
Packit 971217
  memset (msg, 0, sizeof (GstRTSPMessage));
Packit 971217
Packit 971217
  return GST_RTSP_OK;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_free:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 *
Packit 971217
 * Free the memory used by @msg.
Packit 971217
 *
Packit 971217
 * Returns: a #GstRTSPResult.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_free (GstRTSPMessage * msg)
Packit 971217
{
Packit 971217
  GstRTSPResult res;
Packit 971217
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  res = gst_rtsp_message_unset (msg);
Packit 971217
  if (res == GST_RTSP_OK)
Packit 971217
    g_free (msg);
Packit 971217
Packit 971217
  return res;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_copy:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 * @copy: (out) (transfer full): pointer to new #GstRTSPMessage
Packit 971217
 *
Packit 971217
 * Allocate a new copy of @msg and store the result in @copy. The value in
Packit 971217
 * @copy should be release with gst_rtsp_message_free function.
Packit 971217
 *
Packit 971217
 * Returns: a #GstRTSPResult
Packit 971217
 *
Packit 971217
 * Since: 1.14
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_copy (const GstRTSPMessage * msg, GstRTSPMessage ** copy)
Packit 971217
{
Packit 971217
  GstRTSPResult ret;
Packit 971217
  GstRTSPMessage *cp;
Packit 971217
Packit 971217
  if (msg == NULL)
Packit 971217
    return GST_RTSP_EINVAL;
Packit 971217
Packit 971217
  ret = gst_rtsp_message_new (copy);
Packit 971217
  if (ret != GST_RTSP_OK)
Packit 971217
    return ret;
Packit 971217
Packit 971217
  cp = *copy;
Packit 971217
Packit 971217
  cp->type = msg->type;
Packit 971217
  switch (cp->type) {
Packit 971217
    case GST_RTSP_MESSAGE_INVALID:
Packit 971217
      break;
Packit 971217
    case GST_RTSP_MESSAGE_REQUEST:
Packit 971217
    case GST_RTSP_MESSAGE_HTTP_REQUEST:
Packit 971217
      cp->type_data.request.method = msg->type_data.request.method;
Packit 971217
      cp->type_data.request.uri = g_strdup (msg->type_data.request.uri);
Packit 971217
      cp->type_data.request.version = msg->type_data.request.version;
Packit 971217
      break;
Packit 971217
    case GST_RTSP_MESSAGE_RESPONSE:
Packit 971217
    case GST_RTSP_MESSAGE_HTTP_RESPONSE:
Packit 971217
      cp->type_data.response.code = msg->type_data.response.code;
Packit 971217
      cp->type_data.response.reason = g_strdup (msg->type_data.response.reason);
Packit 971217
      cp->type_data.response.version = msg->type_data.response.version;
Packit 971217
      break;
Packit 971217
    case GST_RTSP_MESSAGE_DATA:
Packit 971217
      cp->type_data.data.channel = msg->type_data.data.channel;
Packit 971217
      break;
Packit 971217
    default:
Packit 971217
      return GST_RTSP_EINVAL;
Packit 971217
  }
Packit 971217
Packit 971217
  key_value_foreach (msg->hdr_fields, (GFunc) key_value_append, cp->hdr_fields);
Packit 971217
  gst_rtsp_message_set_body (cp, msg->body, msg->body_size);
Packit 971217
Packit 971217
  return GST_RTSP_OK;
Packit 971217
}
Packit 971217
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_take_header:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 * @field: a #GstRTSPHeaderField
Packit 971217
 * @value: (transfer full): the value of the header
Packit 971217
 *
Packit 971217
 * Add a header with key @field and @value to @msg. This function takes
Packit 971217
 * ownership of @value.
Packit 971217
 *
Packit 971217
 * Returns: a #GstRTSPResult.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_take_header (GstRTSPMessage * msg, GstRTSPHeaderField field,
Packit 971217
    gchar * value)
Packit 971217
{
Packit 971217
  RTSPKeyValue key_value;
Packit 971217
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
  g_return_val_if_fail (value != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  key_value.field = field;
Packit 971217
  key_value.value = value;
Packit 971217
  key_value.custom_key = NULL;
Packit 971217
Packit 971217
  g_array_append_val (msg->hdr_fields, key_value);
Packit 971217
Packit 971217
  return GST_RTSP_OK;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_add_header:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 * @field: a #GstRTSPHeaderField
Packit 971217
 * @value: (transfer none): the value of the header
Packit 971217
 *
Packit 971217
 * Add a header with key @field and @value to @msg. This function takes a copy
Packit 971217
 * of @value.
Packit 971217
 *
Packit 971217
 * Returns: a #GstRTSPResult.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_add_header (GstRTSPMessage * msg, GstRTSPHeaderField field,
Packit 971217
    const gchar * value)
Packit 971217
{
Packit 971217
  return gst_rtsp_message_take_header (msg, field, g_strdup (value));
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_remove_header:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 * @field: a #GstRTSPHeaderField
Packit 971217
 * @indx: the index of the header
Packit 971217
 *
Packit 971217
 * Remove the @indx header with key @field from @msg. If @indx equals -1, all
Packit 971217
 * headers will be removed.
Packit 971217
 *
Packit 971217
 * Returns: a #GstRTSPResult.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_remove_header (GstRTSPMessage * msg, GstRTSPHeaderField field,
Packit 971217
    gint indx)
Packit 971217
{
Packit 971217
  GstRTSPResult res = GST_RTSP_ENOTIMPL;
Packit 971217
  guint i = 0;
Packit 971217
  gint cnt = 0;
Packit 971217
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  while (i < msg->hdr_fields->len) {
Packit 971217
    RTSPKeyValue *key_value = &g_array_index (msg->hdr_fields, RTSPKeyValue, i);
Packit 971217
Packit 971217
    if (key_value->field == field && (indx == -1 || cnt++ == indx)) {
Packit 971217
      g_free (key_value->value);
Packit 971217
      g_array_remove_index (msg->hdr_fields, i);
Packit 971217
      res = GST_RTSP_OK;
Packit 971217
      if (indx != -1)
Packit 971217
        break;
Packit 971217
    } else {
Packit 971217
      i++;
Packit 971217
    }
Packit 971217
  }
Packit 971217
  return res;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_get_header:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 * @field: a #GstRTSPHeaderField
Packit 971217
 * @value: (out) (transfer none): pointer to hold the result
Packit 971217
 * @indx: the index of the header
Packit 971217
 *
Packit 971217
 * Get the @indx header value with key @field from @msg. The result in @value
Packit 971217
 * stays valid as long as it remains present in @msg.
Packit 971217
 *
Packit 971217
 * Returns: #GST_RTSP_OK when @field was found, #GST_RTSP_ENOTIMPL if the key
Packit 971217
 * was not found.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_get_header (const GstRTSPMessage * msg,
Packit 971217
    GstRTSPHeaderField field, gchar ** value, gint indx)
Packit 971217
{
Packit 971217
  guint i;
Packit 971217
  gint cnt = 0;
Packit 971217
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  /* no header initialized, there are no headers */
Packit 971217
  if (msg->hdr_fields == NULL)
Packit 971217
    return GST_RTSP_ENOTIMPL;
Packit 971217
Packit 971217
  for (i = 0; i < msg->hdr_fields->len; i++) {
Packit 971217
    RTSPKeyValue *key_value = &g_array_index (msg->hdr_fields, RTSPKeyValue, i);
Packit 971217
Packit 971217
    if (key_value->field == field && cnt++ == indx) {
Packit 971217
      if (value)
Packit 971217
        *value = key_value->value;
Packit 971217
      return GST_RTSP_OK;
Packit 971217
    }
Packit 971217
  }
Packit 971217
Packit 971217
  return GST_RTSP_ENOTIMPL;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_add_header_by_name:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 * @header: (transfer none): header string
Packit 971217
 * @value: (transfer none): the value of the header
Packit 971217
 *
Packit 971217
 * Add a header with key @header and @value to @msg. This function takes a copy
Packit 971217
 * of @value.
Packit 971217
 *
Packit 971217
 * Returns: a #GstRTSPResult.
Packit 971217
 *
Packit 971217
 * Since: 1.6
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_add_header_by_name (GstRTSPMessage * msg,
Packit 971217
    const gchar * header, const gchar * value)
Packit 971217
{
Packit 971217
  GstRTSPHeaderField field;
Packit 971217
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
  g_return_val_if_fail (header != NULL, GST_RTSP_EINVAL);
Packit 971217
  g_return_val_if_fail (value != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  field = gst_rtsp_find_header_field (header);
Packit 971217
  if (field != GST_RTSP_HDR_INVALID)
Packit 971217
    return gst_rtsp_message_take_header (msg, field, g_strdup (value));
Packit 971217
Packit 971217
  return gst_rtsp_message_take_header_by_name (msg, header, g_strdup (value));
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_take_header_by_name:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 * @header: (transfer none): a header string
Packit 971217
 * @value: (transfer full): the value of the header
Packit 971217
 *
Packit 971217
 * Add a header with key @header and @value to @msg. This function takes
Packit 971217
 * ownership of @value, but not of @header.
Packit 971217
 *
Packit 971217
 * Returns: a #GstRTSPResult.
Packit 971217
 *
Packit 971217
 * Since: 1.6
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_take_header_by_name (GstRTSPMessage * msg,
Packit 971217
    const gchar * header, gchar * value)
Packit 971217
{
Packit 971217
  RTSPKeyValue key_value;
Packit 971217
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
  g_return_val_if_fail (header != NULL, GST_RTSP_EINVAL);
Packit 971217
  g_return_val_if_fail (value != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  key_value.field = GST_RTSP_HDR_INVALID;
Packit 971217
  key_value.value = value;
Packit 971217
  key_value.custom_key = g_strdup (header);
Packit 971217
Packit 971217
  g_array_append_val (msg->hdr_fields, key_value);
Packit 971217
Packit 971217
  return GST_RTSP_OK;
Packit 971217
}
Packit 971217
Packit 971217
/* returns -1 if not found, otherwise index position within msg->hdr_fields */
Packit 971217
static gint
Packit 971217
gst_rtsp_message_find_header_by_name (GstRTSPMessage * msg,
Packit 971217
    const gchar * header, gint index)
Packit 971217
{
Packit 971217
  GstRTSPHeaderField field;
Packit 971217
  gint cnt = 0;
Packit 971217
  guint i;
Packit 971217
Packit 971217
  /* no header initialized, there are no headers */
Packit 971217
  if (msg->hdr_fields == NULL)
Packit 971217
    return -1;
Packit 971217
Packit 971217
  field = gst_rtsp_find_header_field (header);
Packit 971217
  for (i = 0; i < msg->hdr_fields->len; i++) {
Packit 971217
    RTSPKeyValue *key_val;
Packit 971217
Packit 971217
    key_val = &g_array_index (msg->hdr_fields, RTSPKeyValue, i);
Packit 971217
Packit 971217
    if (key_val->field != field)
Packit 971217
      continue;
Packit 971217
Packit 971217
    if (key_val->custom_key != NULL &&
Packit 971217
        g_ascii_strcasecmp (key_val->custom_key, header) != 0)
Packit 971217
      continue;
Packit 971217
Packit 971217
    if (index < 0 || cnt++ == index)
Packit 971217
      return i;
Packit 971217
  }
Packit 971217
Packit 971217
  return -1;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_remove_header_by_name:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 * @header: the header string
Packit 971217
 * @index: the index of the header
Packit 971217
 *
Packit 971217
 * Remove the @index header with key @header from @msg. If @index equals -1,
Packit 971217
 * all matching headers will be removed.
Packit 971217
 *
Packit 971217
 * Returns: a #GstRTSPResult
Packit 971217
 *
Packit 971217
 * Since: 1.6
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_remove_header_by_name (GstRTSPMessage * msg,
Packit 971217
    const gchar * header, gint index)
Packit 971217
{
Packit 971217
  GstRTSPResult res = GST_RTSP_ENOTIMPL;
Packit 971217
  RTSPKeyValue *kv;
Packit 971217
  gint pos;
Packit 971217
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
  g_return_val_if_fail (header != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  do {
Packit 971217
    pos = gst_rtsp_message_find_header_by_name (msg, header, index);
Packit 971217
Packit 971217
    if (pos < 0)
Packit 971217
      break;
Packit 971217
Packit 971217
    kv = &g_array_index (msg->hdr_fields, RTSPKeyValue, pos);
Packit 971217
    g_free (kv->value);
Packit 971217
    g_free (kv->custom_key);
Packit 971217
    g_array_remove_index (msg->hdr_fields, pos);
Packit 971217
    res = GST_RTSP_OK;
Packit 971217
  } while (index < 0);
Packit 971217
Packit 971217
  return res;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_get_header_by_name:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 * @header: a #GstRTSPHeaderField
Packit 971217
 * @value: (out) (transfer none): pointer to hold the result
Packit 971217
 * @index: the index of the header
Packit 971217
 *
Packit 971217
 * Get the @index header value with key @header from @msg. The result in @value
Packit 971217
 * stays valid as long as it remains present in @msg.
Packit 971217
 *
Packit 971217
 * Returns: #GST_RTSP_OK when @field was found, #GST_RTSP_ENOTIMPL if the key
Packit 971217
 * was not found.
Packit 971217
 *
Packit 971217
 * Since: 1.6
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_get_header_by_name (GstRTSPMessage * msg,
Packit 971217
    const gchar * header, gchar ** value, gint index)
Packit 971217
{
Packit 971217
  RTSPKeyValue *key_val;
Packit 971217
  gint pos;
Packit 971217
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
  g_return_val_if_fail (header != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  pos = gst_rtsp_message_find_header_by_name (msg, header, index);
Packit 971217
Packit 971217
  if (pos < 0)
Packit 971217
    return GST_RTSP_ENOTIMPL;
Packit 971217
Packit 971217
  key_val = &g_array_index (msg->hdr_fields, RTSPKeyValue, pos);
Packit 971217
Packit 971217
  if (value)
Packit 971217
    *value = key_val->value;
Packit 971217
Packit 971217
  return GST_RTSP_OK;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_append_headers:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 * @str: (transfer none): a string
Packit 971217
 *
Packit 971217
 * Append the currently configured headers in @msg to the #GString @str suitable
Packit 971217
 * for transmission.
Packit 971217
 *
Packit 971217
 * Returns: #GST_RTSP_OK.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_append_headers (const GstRTSPMessage * msg, GString * str)
Packit 971217
{
Packit 971217
  guint i;
Packit 971217
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
  g_return_val_if_fail (str != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  for (i = 0; i < msg->hdr_fields->len; i++) {
Packit 971217
    RTSPKeyValue *key_value;
Packit 971217
    const gchar *keystr;
Packit 971217
Packit 971217
    key_value = &g_array_index (msg->hdr_fields, RTSPKeyValue, i);
Packit 971217
Packit 971217
    if (key_value->custom_key != NULL)
Packit 971217
      keystr = key_value->custom_key;
Packit 971217
    else
Packit 971217
      keystr = gst_rtsp_header_as_text (key_value->field);
Packit 971217
Packit 971217
    g_string_append_printf (str, "%s: %s\r\n", keystr, key_value->value);
Packit 971217
  }
Packit 971217
  return GST_RTSP_OK;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_set_body:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 * @data: (array length=size) (transfer none): the data
Packit 971217
 * @size: the size of @data
Packit 971217
 *
Packit 971217
 * Set the body of @msg to a copy of @data.
Packit 971217
 *
Packit 971217
 * Returns: #GST_RTSP_OK.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_set_body (GstRTSPMessage * msg, const guint8 * data,
Packit 971217
    guint size)
Packit 971217
{
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  return gst_rtsp_message_take_body (msg, g_memdup (data, size), size);
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_take_body:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 * @data: (array length=size) (transfer full): the data
Packit 971217
 * @size: the size of @data
Packit 971217
 *
Packit 971217
 * Set the body of @msg to @data and @size. This method takes ownership of
Packit 971217
 * @data.
Packit 971217
 *
Packit 971217
 * Returns: #GST_RTSP_OK.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_take_body (GstRTSPMessage * msg, guint8 * data, guint size)
Packit 971217
{
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
  g_return_val_if_fail (data != NULL || size == 0, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  g_free (msg->body);
Packit 971217
Packit 971217
  msg->body = data;
Packit 971217
  msg->body_size = size;
Packit 971217
Packit 971217
  return GST_RTSP_OK;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_get_body:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 * @data: (out) (transfer none) (array length=size): location for the data
Packit 971217
 * @size: (out): location for the size of @data
Packit 971217
 *
Packit 971217
 * Get the body of @msg. @data remains valid for as long as @msg is valid and
Packit 971217
 * unchanged.
Packit 971217
 *
Packit 971217
 * Returns: #GST_RTSP_OK.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_get_body (const GstRTSPMessage * msg, guint8 ** data,
Packit 971217
    guint * size)
Packit 971217
{
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
  g_return_val_if_fail (data != NULL, GST_RTSP_EINVAL);
Packit 971217
  g_return_val_if_fail (size != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  *data = msg->body;
Packit 971217
  *size = msg->body_size;
Packit 971217
Packit 971217
  return GST_RTSP_OK;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_steal_body:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 * @data: (out) (transfer full) (array length=size): location for the data
Packit 971217
 * @size: (out): location for the size of @data
Packit 971217
 *
Packit 971217
 * Take the body of @msg and store it in @data and @size. After this method,
Packit 971217
 * the body and size of @msg will be set to %NULL and 0 respectively.
Packit 971217
 *
Packit 971217
 * Returns: #GST_RTSP_OK.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_steal_body (GstRTSPMessage * msg, guint8 ** data, guint * size)
Packit 971217
{
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
  g_return_val_if_fail (data != NULL, GST_RTSP_EINVAL);
Packit 971217
  g_return_val_if_fail (size != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  *data = msg->body;
Packit 971217
  *size = msg->body_size;
Packit 971217
Packit 971217
  msg->body = NULL;
Packit 971217
  msg->body_size = 0;
Packit 971217
Packit 971217
  return GST_RTSP_OK;
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
dump_key_value (gpointer data, gpointer user_data G_GNUC_UNUSED)
Packit 971217
{
Packit 971217
  RTSPKeyValue *key_value = (RTSPKeyValue *) data;
Packit 971217
  const gchar *key_string;
Packit 971217
Packit 971217
  if (key_value->custom_key != NULL)
Packit 971217
    key_string = key_value->custom_key;
Packit 971217
  else
Packit 971217
    key_string = gst_rtsp_header_as_text (key_value->field);
Packit 971217
Packit 971217
  g_print ("   key: '%s', value: '%s'\n", key_string, key_value->value);
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_dump:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 *
Packit 971217
 * Dump the contents of @msg to stdout.
Packit 971217
 *
Packit 971217
 * Returns: #GST_RTSP_OK.
Packit 971217
 */
Packit 971217
GstRTSPResult
Packit 971217
gst_rtsp_message_dump (GstRTSPMessage * msg)
Packit 971217
{
Packit 971217
  guint8 *data;
Packit 971217
  guint size;
Packit 971217
Packit 971217
  g_return_val_if_fail (msg != NULL, GST_RTSP_EINVAL);
Packit 971217
Packit 971217
  switch (msg->type) {
Packit 971217
    case GST_RTSP_MESSAGE_REQUEST:
Packit 971217
      g_print ("RTSP request message %p\n", msg);
Packit 971217
      g_print (" request line:\n");
Packit 971217
      g_print ("   method: '%s'\n",
Packit 971217
          gst_rtsp_method_as_text (msg->type_data.request.method));
Packit 971217
      g_print ("   uri:    '%s'\n", msg->type_data.request.uri);
Packit 971217
      g_print ("   version: '%s'\n",
Packit 971217
          gst_rtsp_version_as_text (msg->type_data.request.version));
Packit 971217
      g_print (" headers:\n");
Packit 971217
      key_value_foreach (msg->hdr_fields, dump_key_value, NULL);
Packit 971217
      g_print (" body:\n");
Packit 971217
      gst_rtsp_message_get_body (msg, &data, &size);
Packit 971217
      gst_util_dump_mem (data, size);
Packit 971217
      break;
Packit 971217
    case GST_RTSP_MESSAGE_RESPONSE:
Packit 971217
      g_print ("RTSP response message %p\n", msg);
Packit 971217
      g_print (" status line:\n");
Packit 971217
      g_print ("   code:   '%d'\n", msg->type_data.response.code);
Packit 971217
      g_print ("   reason: '%s'\n", msg->type_data.response.reason);
Packit 971217
      g_print ("   version: '%s'\n",
Packit 971217
          gst_rtsp_version_as_text (msg->type_data.response.version));
Packit 971217
      g_print (" headers:\n");
Packit 971217
      key_value_foreach (msg->hdr_fields, dump_key_value, NULL);
Packit 971217
      gst_rtsp_message_get_body (msg, &data, &size);
Packit 971217
      g_print (" body: length %d\n", size);
Packit 971217
      gst_util_dump_mem (data, size);
Packit 971217
      break;
Packit 971217
    case GST_RTSP_MESSAGE_HTTP_REQUEST:
Packit 971217
      g_print ("HTTP request message %p\n", msg);
Packit 971217
      g_print (" request line:\n");
Packit 971217
      g_print ("   method:  '%s'\n",
Packit 971217
          gst_rtsp_method_as_text (msg->type_data.request.method));
Packit 971217
      g_print ("   uri:     '%s'\n", msg->type_data.request.uri);
Packit 971217
      g_print ("   version: '%s'\n",
Packit 971217
          gst_rtsp_version_as_text (msg->type_data.request.version));
Packit 971217
      g_print (" headers:\n");
Packit 971217
      key_value_foreach (msg->hdr_fields, dump_key_value, NULL);
Packit 971217
      g_print (" body:\n");
Packit 971217
      gst_rtsp_message_get_body (msg, &data, &size);
Packit 971217
      gst_util_dump_mem (data, size);
Packit 971217
      break;
Packit 971217
    case GST_RTSP_MESSAGE_HTTP_RESPONSE:
Packit 971217
      g_print ("HTTP response message %p\n", msg);
Packit 971217
      g_print (" status line:\n");
Packit 971217
      g_print ("   code:    '%d'\n", msg->type_data.response.code);
Packit 971217
      g_print ("   reason:  '%s'\n", msg->type_data.response.reason);
Packit 971217
      g_print ("   version: '%s'\n",
Packit 971217
          gst_rtsp_version_as_text (msg->type_data.response.version));
Packit 971217
      g_print (" headers:\n");
Packit 971217
      key_value_foreach (msg->hdr_fields, dump_key_value, NULL);
Packit 971217
      gst_rtsp_message_get_body (msg, &data, &size);
Packit 971217
      g_print (" body: length %d\n", size);
Packit 971217
      gst_util_dump_mem (data, size);
Packit 971217
      break;
Packit 971217
    case GST_RTSP_MESSAGE_DATA:
Packit 971217
      g_print ("RTSP data message %p\n", msg);
Packit 971217
      g_print (" channel: '%d'\n", msg->type_data.data.channel);
Packit 971217
      g_print (" size:    '%d'\n", msg->body_size);
Packit 971217
      gst_rtsp_message_get_body (msg, &data, &size);
Packit 971217
      gst_util_dump_mem (data, size);
Packit 971217
      break;
Packit 971217
    default:
Packit 971217
      g_print ("unsupported message type %d\n", msg->type);
Packit 971217
      return GST_RTSP_EINVAL;
Packit 971217
  }
Packit 971217
  return GST_RTSP_OK;
Packit 971217
}
Packit 971217
Packit 971217
Packit 971217
static const gchar *
Packit 971217
skip_lws (const gchar * s)
Packit 971217
{
Packit 971217
  while (g_ascii_isspace (*s))
Packit 971217
    s++;
Packit 971217
  return s;
Packit 971217
}
Packit 971217
Packit 971217
static const gchar *
Packit 971217
skip_commas (const gchar * s)
Packit 971217
{
Packit 971217
  /* The grammar allows for multiple commas */
Packit 971217
  while (g_ascii_isspace (*s) || *s == ',')
Packit 971217
    s++;
Packit 971217
  return s;
Packit 971217
}
Packit 971217
Packit 971217
static const gchar *
Packit 971217
skip_scheme (const gchar * s)
Packit 971217
{
Packit 971217
  while (*s && !g_ascii_isspace (*s))
Packit 971217
    s++;
Packit 971217
  return s;
Packit 971217
}
Packit 971217
Packit 971217
static const gchar *
Packit 971217
skip_item (const gchar * s)
Packit 971217
{
Packit 971217
  gboolean quoted = FALSE;
Packit 971217
Packit 971217
  /* A list item ends at the last non-whitespace character
Packit 971217
   * before a comma which is not inside a quoted-string. Or at
Packit 971217
   * the end of the string.
Packit 971217
   */
Packit 971217
  while (*s) {
Packit 971217
    if (*s == '"') {
Packit 971217
      quoted = !quoted;
Packit 971217
    } else if (quoted) {
Packit 971217
      if (*s == '\\' && *(s + 1))
Packit 971217
        s++;
Packit 971217
    } else {
Packit 971217
      if (*s == ',' || g_ascii_isspace (*s))
Packit 971217
        break;
Packit 971217
    }
Packit 971217
    s++;
Packit 971217
  }
Packit 971217
Packit 971217
  return s;
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
decode_quoted_string (gchar * quoted_string)
Packit 971217
{
Packit 971217
  gchar *src, *dst;
Packit 971217
Packit 971217
  src = quoted_string + 1;
Packit 971217
  dst = quoted_string;
Packit 971217
  while (*src && *src != '"') {
Packit 971217
    if (*src == '\\' && *(src + 1))
Packit 971217
      src++;
Packit 971217
    *dst++ = *src++;
Packit 971217
  }
Packit 971217
  *dst = '\0';
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
parse_auth_credentials (GPtrArray * auth_credentials, const gchar * header,
Packit 971217
    GstRTSPHeaderField field)
Packit 971217
{
Packit 971217
  while (header[0] != '\0') {
Packit 971217
    const gchar *end;
Packit 971217
    GstRTSPAuthCredential *auth_credential;
Packit 971217
Packit 971217
    /* Skip whitespace at the start of the string */
Packit 971217
    header = skip_lws (header);
Packit 971217
    if (header[0] == '\0')
Packit 971217
      break;
Packit 971217
Packit 971217
    /* Skip until end of string or whitespace: end of scheme */
Packit 971217
    end = skip_scheme (header);
Packit 971217
Packit 971217
    auth_credential = g_new0 (GstRTSPAuthCredential, 1);
Packit 971217
Packit 971217
    if (g_ascii_strncasecmp (header, "basic", 5) == 0) {
Packit 971217
      auth_credential->scheme = GST_RTSP_AUTH_BASIC;
Packit 971217
    } else if (g_ascii_strncasecmp (header, "digest", 6) == 0) {
Packit 971217
      auth_credential->scheme = GST_RTSP_AUTH_DIGEST;
Packit 971217
    } else {
Packit 971217
      /* Not supported, skip */
Packit 971217
      g_free (auth_credential);
Packit 971217
      header = end;
Packit 971217
      continue;
Packit 971217
    }
Packit 971217
Packit 971217
    /* Basic Authorization request has only an unformated blurb following, all
Packit 971217
     * other variants have comma-separated name=value pairs */
Packit 971217
    if (end[0] != '\0' && field == GST_RTSP_HDR_AUTHORIZATION
Packit 971217
        && auth_credential->scheme == GST_RTSP_AUTH_BASIC) {
Packit 971217
      auth_credential->authorization = g_strdup (end + 1);
Packit 971217
      header = end;
Packit 971217
    } else if (end[0] != '\0') {
Packit 971217
      GPtrArray *params;
Packit 971217
Packit 971217
      params = g_ptr_array_new ();
Packit 971217
Packit 971217
      /* Space or start of param */
Packit 971217
      header = end;
Packit 971217
Packit 971217
      /* Parse a header whose content is described by RFC2616 as
Packit 971217
       * "#something", where "something" does not itself contain commas,
Packit 971217
       * except as part of quoted-strings, into a list of allocated strings.
Packit 971217
       */
Packit 971217
      while (*header) {
Packit 971217
        const gchar *item_end;
Packit 971217
        const gchar *eq;
Packit 971217
Packit 971217
        header = skip_commas (header);
Packit 971217
        item_end = skip_item (header);
Packit 971217
Packit 971217
        for (eq = header; *eq != '\0' && *eq != '=' && eq < item_end; eq++);
Packit 971217
        if (eq[0] == '=') {
Packit 971217
          GstRTSPAuthParam *auth_param = g_new0 (GstRTSPAuthParam, 1);
Packit 971217
          const gchar *value;
Packit 971217
Packit 971217
          /* have an actual param */
Packit 971217
          auth_param->name = g_strndup (header, eq - header);
Packit 971217
Packit 971217
          value = eq + 1;
Packit 971217
          value = skip_lws (value);
Packit 971217
          auth_param->value = g_strndup (value, item_end - value);
Packit 971217
          if (value[0] == '"')
Packit 971217
            decode_quoted_string (auth_param->value);
Packit 971217
Packit 971217
          g_ptr_array_add (params, auth_param);
Packit 971217
          header = item_end;
Packit 971217
        } else {
Packit 971217
          /* at next scheme, header at start of it */
Packit 971217
          break;
Packit 971217
        }
Packit 971217
      }
Packit 971217
      if (params->len)
Packit 971217
        g_ptr_array_add (params, NULL);
Packit 971217
      auth_credential->params =
Packit 971217
          (GstRTSPAuthParam **) g_ptr_array_free (params, FALSE);
Packit 971217
    } else {
Packit 971217
      header = end;
Packit 971217
    }
Packit 971217
    g_ptr_array_add (auth_credentials, auth_credential);
Packit 971217
Packit 971217
    /* WWW-Authenticate allows multiple, Authorization allows one */
Packit 971217
    if (field == GST_RTSP_HDR_AUTHORIZATION)
Packit 971217
      break;
Packit 971217
  }
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_message_parse_auth_credentials:
Packit 971217
 * @msg: a #GstRTSPMessage
Packit 971217
 * @field: a #GstRTSPHeaderField
Packit 971217
 *
Packit 971217
 * Parses the credentials given in a WWW-Authenticate or Authorization header.
Packit 971217
 *
Packit 971217
 * Returns: (array zero-terminated=1):
Packit 971217
 *     %NULL-terminated array of GstRTSPAuthCredential or %NULL.
Packit 971217
 *
Packit 971217
 * Since: 1.12
Packit 971217
 */
Packit 971217
GstRTSPAuthCredential **
Packit 971217
gst_rtsp_message_parse_auth_credentials (GstRTSPMessage * msg,
Packit 971217
    GstRTSPHeaderField field)
Packit 971217
{
Packit 971217
  gchar *header;
Packit 971217
  GPtrArray *auth_credentials;
Packit 971217
  gint i;
Packit 971217
Packit 971217
  g_return_val_if_fail (msg != NULL, NULL);
Packit 971217
Packit 971217
  auth_credentials = g_ptr_array_new ();
Packit 971217
Packit 971217
  i = 0;
Packit 971217
  while (gst_rtsp_message_get_header (msg, field, &header, i) == GST_RTSP_OK) {
Packit 971217
    parse_auth_credentials (auth_credentials, header, field);
Packit 971217
    i++;
Packit 971217
  }
Packit 971217
Packit 971217
  if (auth_credentials->len)
Packit 971217
    g_ptr_array_add (auth_credentials, NULL);
Packit 971217
Packit 971217
  return (GstRTSPAuthCredential **) g_ptr_array_free (auth_credentials, FALSE);
Packit 971217
}
Packit 971217
Packit 971217
GstRTSPAuthParam *
Packit 971217
gst_rtsp_auth_param_copy (GstRTSPAuthParam * param)
Packit 971217
{
Packit 971217
  GstRTSPAuthParam *copy;
Packit 971217
Packit 971217
  if (param == NULL)
Packit 971217
    return NULL;
Packit 971217
Packit 971217
  copy = g_new0 (GstRTSPAuthParam, 1);
Packit 971217
  copy->name = g_strdup (param->name);
Packit 971217
  copy->value = g_strdup (param->value);
Packit 971217
Packit 971217
  return copy;
Packit 971217
}
Packit 971217
Packit 971217
void
Packit 971217
gst_rtsp_auth_param_free (GstRTSPAuthParam * param)
Packit 971217
{
Packit 971217
  if (param != NULL) {
Packit 971217
    g_free (param->name);
Packit 971217
    g_free (param->value);
Packit 971217
    g_free (param);
Packit 971217
  }
Packit 971217
}
Packit 971217
Packit 971217
G_DEFINE_BOXED_TYPE (GstRTSPAuthParam, gst_rtsp_auth_param,
Packit 971217
    (GBoxedCopyFunc) gst_rtsp_auth_param_copy,
Packit 971217
    (GBoxedFreeFunc) gst_rtsp_auth_param_free);
Packit 971217
Packit 971217
static void
Packit 971217
gst_rtsp_auth_credential_free (GstRTSPAuthCredential * credential)
Packit 971217
{
Packit 971217
  GstRTSPAuthParam **p;
Packit 971217
Packit 971217
  if (credential == NULL)
Packit 971217
    return;
Packit 971217
Packit 971217
  for (p = credential->params; p != NULL && *p != NULL; ++p)
Packit 971217
    gst_rtsp_auth_param_free (*p);
Packit 971217
Packit 971217
  g_free (credential->params);
Packit 971217
  g_free (credential->authorization);
Packit 971217
  g_free (credential);
Packit 971217
}
Packit 971217
Packit 971217
static GstRTSPAuthCredential *
Packit 971217
gst_rtsp_auth_credential_copy (GstRTSPAuthCredential * cred)
Packit 971217
{
Packit 971217
  GstRTSPAuthCredential *copy;
Packit 971217
Packit 971217
  if (cred == NULL)
Packit 971217
    return NULL;
Packit 971217
Packit 971217
  copy = g_new0 (GstRTSPAuthCredential, 1);
Packit 971217
  copy->scheme = cred->scheme;
Packit 971217
  if (cred->params) {
Packit 971217
    guint i, n_params = g_strv_length ((gchar **) cred->params);
Packit 971217
Packit 971217
    copy->params = g_new0 (GstRTSPAuthParam *, n_params + 1);
Packit 971217
    for (i = 0; i < n_params; ++i)
Packit 971217
      copy->params[i] = gst_rtsp_auth_param_copy (cred->params[i]);
Packit 971217
  }
Packit 971217
  copy->authorization = g_strdup (cred->authorization);
Packit 971217
  return copy;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_rtsp_auth_credentials_free:
Packit 971217
 * @credentials: a %NULL-terminated array of #GstRTSPAuthCredential
Packit 971217
 *
Packit 971217
 * Free a %NULL-terminated array of credentials returned from
Packit 971217
 * gst_rtsp_message_parse_auth_credentials().
Packit 971217
 *
Packit 971217
 * Since: 1.12
Packit 971217
 */
Packit 971217
void
Packit 971217
gst_rtsp_auth_credentials_free (GstRTSPAuthCredential ** credentials)
Packit 971217
{
Packit 971217
  GstRTSPAuthCredential **p;
Packit 971217
Packit 971217
  if (!credentials)
Packit 971217
    return;
Packit 971217
Packit 971217
  for (p = credentials; p != NULL && *p != NULL; ++p)
Packit 971217
    gst_rtsp_auth_credential_free (*p);
Packit 971217
Packit 971217
  g_free (credentials);
Packit 971217
}
Packit 971217
Packit 971217
G_DEFINE_BOXED_TYPE (GstRTSPAuthCredential, gst_rtsp_auth_credential,
Packit 971217
    (GBoxedCopyFunc) gst_rtsp_auth_credential_copy,
Packit 971217
    (GBoxedFreeFunc) gst_rtsp_auth_credential_free);