diff options
Diffstat (limited to 'sys/wininet/gstwininetsrc.c')
-rw-r--r-- | sys/wininet/gstwininetsrc.c | 453 |
1 files changed, 453 insertions, 0 deletions
diff --git a/sys/wininet/gstwininetsrc.c b/sys/wininet/gstwininetsrc.c new file mode 100644 index 00000000..ea194d30 --- /dev/null +++ b/sys/wininet/gstwininetsrc.c @@ -0,0 +1,453 @@ +/* GStreamer Windows network source + * Copyright (C) 2008 Ole André Vadla Ravnås <ole.andre.ravnas@tandberg.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., 59 Temple Place - Suite 330, + * Boston, MA 02111-1307, USA. + */ + +/** + * SECTION:element-wininetsrc + * + * <refsect2> + * <title>Example launch line</title> + * <para> + * <programlisting> + * gst-launch-0.10 -v wininetsrc location="http://71.83.57.210:9000" ! application/x-icy,metadata-interval=0 ! icydemux ! mad ! audioconvert ! directsoundsink + * </programlisting> + * </para> + * </refsect2> + */ + +#ifdef HAVE_CONFIG_H +#include <config.h> +#endif + +#include "gstwininetsrc.h" + +#include <string.h> + +#define DEFAULT_LOCATION "http://localhost/" +#define DEFAULT_POLL_MODE FALSE +#define DEFAULT_IRADIO_MODE FALSE + +enum +{ + PROP_0, + PROP_LOCATION, + PROP_POLL_MODE, + PROP_IRADIO_MODE +}; + +GST_DEBUG_CATEGORY_STATIC (gst_win_inet_src_debug); +#define GST_CAT_DEFAULT gst_win_inet_src_debug + +static GstStaticPadTemplate src_template = GST_STATIC_PAD_TEMPLATE ("src", + GST_PAD_SRC, + GST_PAD_ALWAYS, + GST_STATIC_CAPS_ANY); + +static void gst_win_inet_src_init_interfaces (GType type); +static void gst_win_inet_src_uri_handler_init (gpointer g_iface, + gpointer iface_data); + +static void gst_win_inet_src_dispose (GObject * object); +static void gst_win_inet_src_finalize (GObject * object); +static void gst_win_inet_src_get_property (GObject * object, guint prop_id, + GValue * value, GParamSpec * pspec); +static void gst_win_inet_src_set_property (GObject * object, guint prop_id, + const GValue * value, GParamSpec * pspec); + +static gboolean gst_win_inet_src_start (GstBaseSrc * basesrc); +static gboolean gst_win_inet_src_stop (GstBaseSrc * basesrc); + +static GstFlowReturn gst_win_inet_src_create (GstPushSrc * pushsrc, + GstBuffer ** buffer); + +static void gst_win_inet_src_reset (GstWinInetSrc * self); + +GST_BOILERPLATE_FULL (GstWinInetSrc, gst_win_inet_src, GstPushSrc, + GST_TYPE_PUSH_SRC, gst_win_inet_src_init_interfaces); + +static void +gst_win_inet_src_base_init (gpointer gclass) +{ + GstElementClass *element_class = GST_ELEMENT_CLASS (gclass); + + gst_element_class_add_pad_template (element_class, + gst_static_pad_template_get (&src_template)); + + gst_element_class_set_details_simple (element_class, + "Windows Network Source", "Source/Network", + "Receive data as a client over the network via HTTP or FTP", + "Ole André Vadla Ravnås <ole.andre.ravnas@tandberg.com>"); +} + +static void +gst_win_inet_src_class_init (GstWinInetSrcClass * klass) +{ + GObjectClass *gobject_class = G_OBJECT_CLASS (klass); + GstBaseSrcClass *gstbasesrc_class = GST_BASE_SRC_CLASS (klass); + GstPushSrcClass *gstpushsrc_class = GST_PUSH_SRC_CLASS (klass); + + gobject_class->dispose = gst_win_inet_src_dispose; + gobject_class->finalize = gst_win_inet_src_finalize; + gobject_class->get_property = gst_win_inet_src_get_property; + gobject_class->set_property = gst_win_inet_src_set_property; + + gstbasesrc_class->start = GST_DEBUG_FUNCPTR (gst_win_inet_src_start); + gstbasesrc_class->stop = GST_DEBUG_FUNCPTR (gst_win_inet_src_stop); + gstpushsrc_class->create = GST_DEBUG_FUNCPTR (gst_win_inet_src_create); + + g_object_class_install_property (gobject_class, + PROP_LOCATION, g_param_spec_string ("location", "Location", + "Location to read from", DEFAULT_LOCATION, G_PARAM_READWRITE)); + + g_object_class_install_property (gobject_class, + PROP_POLL_MODE, g_param_spec_boolean ("poll-mode", "poll-mode", + "Enable poll mode (keep re-issuing request)", + DEFAULT_POLL_MODE, G_PARAM_READWRITE)); + + g_object_class_install_property (gobject_class, + PROP_IRADIO_MODE, g_param_spec_boolean ("iradio-mode", "iradio-mode", + "Enable Internet radio mode " + "(extraction of shoutcast/icecast metadata)", + DEFAULT_IRADIO_MODE, G_PARAM_READWRITE)); +} + +static void +gst_win_inet_src_init_interfaces (GType type) +{ + static const GInterfaceInfo uri_handler_info = { + gst_win_inet_src_uri_handler_init, + NULL, + NULL + }; + + g_type_add_interface_static (type, GST_TYPE_URI_HANDLER, &uri_handler_info); + + GST_DEBUG_CATEGORY_INIT (gst_win_inet_src_debug, "wininetsrc", + 0, "Wininet source"); +} + +static void +gst_win_inet_src_init (GstWinInetSrc * self, GstWinInetSrcClass * gclass) +{ + self->location = g_strdup (DEFAULT_LOCATION); + self->poll_mode = DEFAULT_POLL_MODE; + self->iradio_mode = DEFAULT_IRADIO_MODE; + + self->inet = NULL; + self->url = NULL; + self->cur_offset = 0; + self->icy_caps = NULL; +} + +static void +gst_win_inet_src_dispose (GObject * object) +{ + GstWinInetSrc *self = GST_WIN_INET_SRC (object); + + gst_win_inet_src_reset (self); + + G_OBJECT_CLASS (parent_class)->dispose (object); +} + +static void +gst_win_inet_src_finalize (GObject * object) +{ + GstWinInetSrc *self = GST_WIN_INET_SRC (object); + + g_free (self->location); + + G_OBJECT_CLASS (parent_class)->finalize (object); +} + +static void +gst_win_inet_src_get_property (GObject * object, guint prop_id, + GValue * value, GParamSpec * pspec) +{ + GstWinInetSrc *self = GST_WIN_INET_SRC (object); + + switch (prop_id) { + case PROP_LOCATION: + g_value_set_string (value, self->location); + break; + + case PROP_POLL_MODE: + g_value_set_boolean (value, self->poll_mode); + break; + + case PROP_IRADIO_MODE: + g_value_set_boolean (value, self->iradio_mode); + break; + + default: + G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); + break; + } +} + +static void +gst_win_inet_src_set_property (GObject * object, guint prop_id, + const GValue * value, GParamSpec * pspec) +{ + GstWinInetSrc *self = GST_WIN_INET_SRC (object); + + switch (prop_id) { + case PROP_LOCATION: + if (GST_STATE (self) == GST_STATE_PLAYING || + GST_STATE (self) == GST_STATE_PAUSED) { + GST_WARNING_OBJECT (self, "element must be in stopped or paused state " + "in order to change location"); + break; + } + + g_free (self->location); + self->location = g_value_dup_string (value); + break; + + case PROP_POLL_MODE: + self->poll_mode = g_value_get_boolean (value); + break; + + case PROP_IRADIO_MODE: + self->iradio_mode = g_value_get_boolean (value); + break; + + default: + G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); + break; + } +} + +static void +gst_win_inet_src_reset (GstWinInetSrc * self) +{ + if (self->url != NULL) { + InternetCloseHandle (self->url); + self->url = NULL; + } + + if (self->inet != NULL) { + InternetCloseHandle (self->inet); + self->inet = NULL; + } + + if (self->icy_caps != NULL) { + gst_caps_unref (self->icy_caps); + self->icy_caps = NULL; + } + + self->cur_offset = 0; +} + +static gboolean +gst_win_inet_src_get_header_value_as_int (GstWinInetSrc * self, + const gchar * header_name, gint * header_value, gboolean log_failure) +{ + gchar buf[16] = { 0, }; + DWORD buf_size = sizeof (buf); + gint *value = (gint *) buf; + + strcpy (buf, header_name); + + if (!HttpQueryInfo (self->url, HTTP_QUERY_CUSTOM | HTTP_QUERY_FLAG_NUMBER, + buf, &buf_size, NULL)) { + if (log_failure) { + DWORD error_code = GetLastError (); + const gchar *error_str = "unknown error"; + + if (error_code == ERROR_HTTP_HEADER_NOT_FOUND) + error_str = "ERROR_HTTP_HEADER_NOT_FOUND"; + + GST_WARNING_OBJECT (self, "HttpQueryInfo for header '%s' failed: %s " + "(0x%08x)", header_name, error_str, error_code); + } + + return FALSE; + } + + *header_value = *value; + return TRUE; +} + +static gboolean +gst_win_inet_src_open (GstWinInetSrc * self) +{ + const gchar *extra_headers = NULL; + + gst_win_inet_src_reset (self); + + self->inet = InternetOpen (NULL, INTERNET_OPEN_TYPE_PRECONFIG, NULL, NULL, 0); + if (self->inet == NULL) + goto error; + + if (self->iradio_mode) + extra_headers = "Icy-MetaData:1"; /* exactly as sent by WinAmp, no space */ + + self->url = InternetOpenUrl (self->inet, self->location, extra_headers, + (extra_headers != NULL) ? -1 : 0, INTERNET_FLAG_NO_UI, (DWORD_PTR) self); + if (self->url == NULL) + goto error; + + if (self->iradio_mode) { + gint value; + + if (gst_win_inet_src_get_header_value_as_int (self, "icy-metaint", &value, + TRUE)) { + self->icy_caps = gst_caps_new_simple ("application/x-icy", + "metadata-interval", G_TYPE_INT, value, NULL); + } + } + + return TRUE; + +error: + GST_ELEMENT_ERROR (self, RESOURCE, NOT_FOUND, (NULL), + ("Could not open location \"%s\" for reading: 0x%08x", + self->location, GetLastError ())); + gst_win_inet_src_reset (self); + + return FALSE; +} + +static gboolean +gst_win_inet_src_start (GstBaseSrc * basesrc) +{ + GstWinInetSrc *self = GST_WIN_INET_SRC (basesrc); + + return gst_win_inet_src_open (self); +} + +static gboolean +gst_win_inet_src_stop (GstBaseSrc * basesrc) +{ + gst_win_inet_src_reset (GST_WIN_INET_SRC (basesrc)); + + return TRUE; +} + +static GstFlowReturn +gst_win_inet_src_create (GstPushSrc * pushsrc, GstBuffer ** buffer) +{ + GstWinInetSrc *self = GST_WIN_INET_SRC (pushsrc); + GstBaseSrc *basesrc = GST_BASE_SRC (pushsrc); + GstBuffer *buf = NULL; + GstFlowReturn ret = GST_FLOW_OK; + DWORD bytes_read = 0; + + do { + GstCaps *caps = GST_PAD_CAPS (GST_BASE_SRC_PAD (self)); + + if (self->icy_caps != NULL) + caps = self->icy_caps; + + ret = gst_pad_alloc_buffer (GST_BASE_SRC_PAD (basesrc), + self->cur_offset, basesrc->blocksize, caps, &buf); + + if (G_LIKELY (ret == GST_FLOW_OK)) { + if (InternetReadFile (self->url, GST_BUFFER_DATA (buf), + basesrc->blocksize, &bytes_read)) { + if (bytes_read == 0) { + if (self->poll_mode) { + if (gst_win_inet_src_open (self)) { + gst_buffer_unref (buf); + buf = NULL; + } else { + ret = GST_FLOW_ERROR; + } + } else { + GST_ERROR_OBJECT (self, "short read (eof?)"); + ret = GST_FLOW_UNEXPECTED; + } + } + } else { + GST_ERROR_OBJECT (self, "InternetReadFile failed: 0x%08x", + GetLastError ()); + + ret = GST_FLOW_ERROR; + } + } + } + while (bytes_read == 0 && ret == GST_FLOW_OK); + + if (ret == GST_FLOW_OK) { + GST_BUFFER_SIZE (buf) = bytes_read; + self->cur_offset += bytes_read; + + *buffer = buf; + } else { + if (buf != NULL) + gst_buffer_unref (buf); + } + + return ret; +} + +static GstURIType +gst_win_inet_src_uri_get_type (void) +{ + return GST_URI_SRC; +} + +static gchar ** +gst_win_inet_src_uri_get_protocols (void) +{ + static gchar *protocols[] = { "http", "https", "ftp", NULL }; + + return protocols; +} + +static G_CONST_RETURN gchar * +gst_win_inet_src_uri_get_uri (GstURIHandler * handler) +{ + GstWinInetSrc *src = GST_WIN_INET_SRC (handler); + + return src->location; +} + +static gboolean +gst_win_inet_src_uri_set_uri (GstURIHandler * handler, const gchar * uri) +{ + GstWinInetSrc *src = GST_WIN_INET_SRC (handler); + + g_free (src->location); + src->location = g_strdup (uri); + return TRUE; +} + +static void +gst_win_inet_src_uri_handler_init (gpointer g_iface, gpointer iface_data) +{ + GstURIHandlerInterface *iface = (GstURIHandlerInterface *) g_iface; + + iface->get_type = gst_win_inet_src_uri_get_type; + iface->get_protocols = gst_win_inet_src_uri_get_protocols; + iface->get_uri = gst_win_inet_src_uri_get_uri; + iface->set_uri = gst_win_inet_src_uri_set_uri; +} + +static gboolean +plugin_init (GstPlugin * plugin) +{ + return gst_element_register (plugin, "wininetsrc", + GST_RANK_NONE, GST_TYPE_WIN_INET_SRC); +} + +GST_PLUGIN_DEFINE (GST_VERSION_MAJOR, + GST_VERSION_MINOR, + "wininet", + "Windows network plugins", + plugin_init, VERSION, GST_LICENSE, GST_PACKAGE_NAME, GST_PACKAGE_ORIGIN) |