mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-10-21 09:53:47 +00:00
Allow internal codes from last.fm
Original commit message from CVS: Allow internal codes from last.fm
This commit is contained in:
parent
90c0981260
commit
1b8664b4c5
4 changed files with 156 additions and 60 deletions
11
ChangeLog
11
ChangeLog
|
@ -1,3 +1,14 @@
|
||||||
|
2006-09019 Edgard Lima <edgard.lima@indt.org.br>
|
||||||
|
|
||||||
|
Patch by: deadchip <internalerror@gmail.com> and
|
||||||
|
Rosfran Borges <rosfran.borges@indt.org.br>
|
||||||
|
|
||||||
|
* ext/neon/Makefile.am:
|
||||||
|
* ext/neon/gstneonhttpsrc.c:
|
||||||
|
* ext/neon/gstneonhttpsrc.h:
|
||||||
|
Allow internal codes from last.fm
|
||||||
|
|
||||||
|
|
||||||
2006-09-19 Edgard Lima <edgard.lima@indt.org.br>
|
2006-09-19 Edgard Lima <edgard.lima@indt.org.br>
|
||||||
|
|
||||||
* tests/icles/v4l2src-test.c:
|
* tests/icles/v4l2src-test.c:
|
||||||
|
|
|
@ -4,8 +4,9 @@ plugin_LTLIBRARIES = libgstneonhttpsrc.la
|
||||||
|
|
||||||
libgstneonhttpsrc_la_SOURCES = gstneonhttpsrc.c
|
libgstneonhttpsrc_la_SOURCES = gstneonhttpsrc.c
|
||||||
|
|
||||||
|
libgstneonhttpsrc_la_CFLAGS = $(GST_CFLAGS) \
|
||||||
libgstneonhttpsrc_la_CFLAGS = $(GST_CFLAGS) $(NEON_CFLAGS)
|
$(NEON_CFLAGS) \
|
||||||
|
-D_GNU_SOURCE
|
||||||
libgstneonhttpsrc_la_LDFLAGS = $(GST_PLUGIN_LDFLAGS)
|
libgstneonhttpsrc_la_LDFLAGS = $(GST_PLUGIN_LDFLAGS)
|
||||||
libgstneonhttpsrc_la_LIBADD = $(GST_BASE_LIBS) \
|
libgstneonhttpsrc_la_LIBADD = $(GST_BASE_LIBS) \
|
||||||
$(NEON_LIBS)
|
$(NEON_LIBS)
|
||||||
|
|
|
@ -14,13 +14,17 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifdef HAVE_CONFIG_H
|
#ifdef HAVE_CONFIG_H
|
||||||
#include "config.h"
|
#include <config.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef HAVE_MEMMEM
|
||||||
|
# include "memmem.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include "gstneonhttpsrc.h"
|
#include "gstneonhttpsrc.h"
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <strings.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
|
||||||
#include <ne_redirect.h>
|
#include <ne_redirect.h>
|
||||||
|
|
||||||
#ifndef NE_FREE
|
#ifndef NE_FREE
|
||||||
|
@ -57,13 +61,14 @@ enum
|
||||||
PROP_URI,
|
PROP_URI,
|
||||||
PROP_PROXY,
|
PROP_PROXY,
|
||||||
PROP_USER_AGENT,
|
PROP_USER_AGENT,
|
||||||
|
PROP_LASTFM_MODE,
|
||||||
PROP_IRADIO_MODE,
|
PROP_IRADIO_MODE,
|
||||||
PROP_IRADIO_NAME,
|
PROP_IRADIO_NAME,
|
||||||
PROP_IRADIO_GENRE,
|
PROP_IRADIO_GENRE,
|
||||||
PROP_IRADIO_URL,
|
PROP_IRADIO_URL,
|
||||||
PROP_NEON_HTTP_REDIRECT
|
PROP_NEON_HTTP_REDIRECT,
|
||||||
#ifndef GST_DISABLE_GST_DEBUG
|
#ifndef GST_DISABLE_GST_DEBUG
|
||||||
, PROP_NEON_HTTP_DBG
|
PROP_NEON_HTTP_DBG
|
||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -106,10 +111,9 @@ _urihandler_init (GType type)
|
||||||
}
|
}
|
||||||
|
|
||||||
GST_BOILERPLATE_FULL (GstNeonhttpSrc, gst_neonhttp_src, GstPushSrc,
|
GST_BOILERPLATE_FULL (GstNeonhttpSrc, gst_neonhttp_src, GstPushSrc,
|
||||||
GST_TYPE_PUSH_SRC, _urihandler_init);
|
GST_TYPE_PUSH_SRC, _urihandler_init)
|
||||||
|
|
||||||
static void
|
static void gst_neonhttp_src_base_init (gpointer g_class)
|
||||||
gst_neonhttp_src_base_init (gpointer g_class)
|
|
||||||
{
|
{
|
||||||
GstElementClass *element_class = GST_ELEMENT_CLASS (g_class);
|
GstElementClass *element_class = GST_ELEMENT_CLASS (g_class);
|
||||||
|
|
||||||
|
@ -171,6 +175,13 @@ gst_neonhttp_src_class_init (GstNeonhttpSrcClass * klass)
|
||||||
"Enable internet radio mode (extraction of shoutcast/icecast metadata)",
|
"Enable internet radio mode (extraction of shoutcast/icecast metadata)",
|
||||||
FALSE, G_PARAM_READWRITE));
|
FALSE, G_PARAM_READWRITE));
|
||||||
|
|
||||||
|
/* lastFM support - patch sent by deadcip <internalerror@gmail.com> */
|
||||||
|
g_object_class_install_property
|
||||||
|
(gobject_class, PROP_LASTFM_MODE,
|
||||||
|
g_param_spec_boolean ("lastfm-mode", "lastfm-mode",
|
||||||
|
"Enable LastFM mode (check for lastfm-stats and sync messages)",
|
||||||
|
FALSE, G_PARAM_READWRITE));
|
||||||
|
|
||||||
g_object_class_install_property (gobject_class,
|
g_object_class_install_property (gobject_class,
|
||||||
PROP_IRADIO_NAME,
|
PROP_IRADIO_NAME,
|
||||||
g_param_spec_string ("iradio-name",
|
g_param_spec_string ("iradio-name",
|
||||||
|
@ -211,61 +222,64 @@ gst_neonhttp_src_class_init (GstNeonhttpSrcClass * klass)
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_neonhttp_src_init (GstNeonhttpSrc * this, GstNeonhttpSrcClass * g_class)
|
gst_neonhttp_src_init (GstNeonhttpSrc * neonhttpsrc,
|
||||||
|
GstNeonhttpSrcClass * g_class)
|
||||||
{
|
{
|
||||||
this->session = NULL;
|
neonhttpsrc->session = NULL;
|
||||||
this->request = NULL;
|
neonhttpsrc->request = NULL;
|
||||||
|
|
||||||
memset (&this->uri, 0, sizeof (this->uri));
|
memset (&neonhttpsrc->uri, 0, sizeof (neonhttpsrc->uri));
|
||||||
this->uristr = NULL;
|
neonhttpsrc->uristr = NULL;
|
||||||
memset (&this->proxy, 0, sizeof (this->proxy));
|
memset (&neonhttpsrc->proxy, 0, sizeof (neonhttpsrc->proxy));
|
||||||
this->ishttps = FALSE;
|
neonhttpsrc->ishttps = FALSE;
|
||||||
this->content_size = -1;
|
neonhttpsrc->content_size = -1;
|
||||||
|
|
||||||
set_uri (this, NULL, &this->uri, &this->ishttps, &this->uristr, TRUE);
|
set_uri (neonhttpsrc, NULL, &neonhttpsrc->uri, &neonhttpsrc->ishttps,
|
||||||
set_proxy (this, NULL, &this->proxy, TRUE);
|
&neonhttpsrc->uristr, TRUE);
|
||||||
|
set_proxy (neonhttpsrc, NULL, &neonhttpsrc->proxy, TRUE);
|
||||||
|
|
||||||
this->user_agent = g_strdup ("neonhttpsrc");
|
neonhttpsrc->user_agent = g_strdup ("Neon HTTP 26");
|
||||||
|
|
||||||
this->iradio_mode = FALSE;
|
neonhttpsrc->lastfm_mode = FALSE;
|
||||||
this->iradio_name = NULL;
|
neonhttpsrc->iradio_mode = FALSE;
|
||||||
this->iradio_genre = NULL;
|
neonhttpsrc->iradio_name = NULL;
|
||||||
this->iradio_url = NULL;
|
neonhttpsrc->iradio_genre = NULL;
|
||||||
this->icy_caps = NULL;
|
neonhttpsrc->iradio_url = NULL;
|
||||||
this->icy_metaint = 0;
|
neonhttpsrc->icy_caps = NULL;
|
||||||
|
neonhttpsrc->icy_metaint = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_neonhttp_src_finalize (GObject * gobject)
|
gst_neonhttp_src_finalize (GObject * gobject)
|
||||||
{
|
{
|
||||||
GstNeonhttpSrc *this = GST_NEONHTTP_SRC (gobject);
|
GstNeonhttpSrc *neonhttpsrc = GST_NEONHTTP_SRC (gobject);
|
||||||
|
|
||||||
ne_uri_free (&this->uri);
|
ne_uri_free (&neonhttpsrc->uri);
|
||||||
ne_uri_free (&this->proxy);
|
ne_uri_free (&neonhttpsrc->proxy);
|
||||||
|
|
||||||
g_free (this->user_agent);
|
g_free (neonhttpsrc->user_agent);
|
||||||
g_free (this->iradio_name);
|
g_free (neonhttpsrc->iradio_name);
|
||||||
g_free (this->iradio_genre);
|
g_free (neonhttpsrc->iradio_genre);
|
||||||
g_free (this->iradio_url);
|
g_free (neonhttpsrc->iradio_url);
|
||||||
|
|
||||||
if (this->icy_caps) {
|
if (neonhttpsrc->icy_caps) {
|
||||||
gst_caps_unref (this->icy_caps);
|
gst_caps_unref (neonhttpsrc->icy_caps);
|
||||||
this->icy_caps = NULL;
|
neonhttpsrc->icy_caps = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this->request) {
|
if (neonhttpsrc->request) {
|
||||||
ne_request_destroy (this->request);
|
ne_request_destroy (neonhttpsrc->request);
|
||||||
this->request = NULL;
|
neonhttpsrc->request = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this->session) {
|
if (neonhttpsrc->session) {
|
||||||
ne_close_connection (this->session);
|
ne_close_connection (neonhttpsrc->session);
|
||||||
ne_session_destroy (this->session);
|
ne_session_destroy (neonhttpsrc->session);
|
||||||
this->session = NULL;
|
neonhttpsrc->session = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this->uristr) {
|
if (neonhttpsrc->uristr) {
|
||||||
ne_free (this->uristr);
|
ne_free (neonhttpsrc->uristr);
|
||||||
}
|
}
|
||||||
|
|
||||||
G_OBJECT_CLASS (parent_class)->finalize (gobject);
|
G_OBJECT_CLASS (parent_class)->finalize (gobject);
|
||||||
|
@ -284,8 +298,10 @@ request_dispatch (GstNeonhttpSrc * src, GstBuffer * outbuf)
|
||||||
ssize_t len = 0;
|
ssize_t len = 0;
|
||||||
|
|
||||||
while (sizetoread > 0) {
|
while (sizetoread > 0) {
|
||||||
len = ne_read_response_block (src->request,
|
len =
|
||||||
|
ne_read_response_block (src->request,
|
||||||
(char *) GST_BUFFER_DATA (outbuf) + read, sizetoread);
|
(char *) GST_BUFFER_DATA (outbuf) + read, sizetoread);
|
||||||
|
|
||||||
if (len > 0) {
|
if (len > 0) {
|
||||||
read += len;
|
read += len;
|
||||||
sizetoread -= len;
|
sizetoread -= len;
|
||||||
|
@ -313,10 +329,67 @@ request_dispatch (GstNeonhttpSrc * src, GstBuffer * outbuf)
|
||||||
}
|
}
|
||||||
|
|
||||||
done:
|
done:
|
||||||
|
/* lastFM support - patch sent by deadchip <internalerror@gmail.com> */
|
||||||
|
if (src->lastfm_mode) {
|
||||||
|
const void *buf = (const void *) (GST_BUFFER_DATA (outbuf));
|
||||||
|
static char SYNC[4] = { 'S', 'Y', 'N', 'C' };
|
||||||
|
|
||||||
|
if (memmem (buf, GST_BUFFER_SIZE (outbuf), &SYNC, 4) != NULL) {
|
||||||
|
GstStructure *structure =
|
||||||
|
gst_structure_new ("lastfm-sync", "sync", G_TYPE_BOOLEAN, TRUE, NULL);
|
||||||
|
GstMessage *message =
|
||||||
|
gst_message_new_application (GST_OBJECT (src), structure);
|
||||||
|
gst_element_post_message (GST_ELEMENT (src), message);
|
||||||
|
}
|
||||||
|
|
||||||
|
static const struct
|
||||||
|
{
|
||||||
|
gchar *status_msg;
|
||||||
|
gint msg_code;
|
||||||
|
} http_status_msgs_tb[4] = {
|
||||||
|
{
|
||||||
|
"HTTP/1.0 401", 401}, {
|
||||||
|
"HTTP/1.0 503", 503}, {
|
||||||
|
"HTTP/1.0 666", 666}, {
|
||||||
|
"HTTP/1.0 667", 667},};
|
||||||
|
|
||||||
|
guint iter;
|
||||||
|
|
||||||
|
gchar *http_chunk = g_new0 (gchar, sizeof (gchar));
|
||||||
|
|
||||||
|
while ((http_chunk =
|
||||||
|
memmem (buf, GST_BUFFER_SIZE (outbuf), "HTTP/1.0", 8)) != NULL) {
|
||||||
|
|
||||||
|
for (iter = 0; iter < 4; iter++) {
|
||||||
|
|
||||||
|
if (strncmp (http_chunk, http_status_msgs_tb[iter].status_msg, 12) == 0) {
|
||||||
|
#ifndef GST_DISABLE_GST_DEBUG
|
||||||
|
g_print ("[%s] ok, we found a HTTP/1.0 tag = %s\n", __FUNCTION__,
|
||||||
|
http_status_msgs_tb[iter].status_msg);
|
||||||
|
#endif
|
||||||
|
GstStructure *structure =
|
||||||
|
gst_structure_new ("lastfm-status", "status", G_TYPE_INT,
|
||||||
|
http_status_msgs_tb[iter].msg_code, NULL);
|
||||||
|
GstMessage *message =
|
||||||
|
gst_message_new_application (GST_OBJECT (src), structure);
|
||||||
|
gst_element_post_message (GST_ELEMENT (src), message);
|
||||||
|
gst_element_post_message (GST_ELEMENT (src),
|
||||||
|
gst_message_new_eos (GST_OBJECT (src)));
|
||||||
|
src->eos = TRUE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} /* for */
|
||||||
|
|
||||||
|
/* do move by 8-bytes offset */
|
||||||
|
buf = http_chunk + 8;
|
||||||
|
|
||||||
|
} /* while - ( all the HTTP/1.0 occurrences ) */
|
||||||
|
|
||||||
|
}
|
||||||
|
/* lastFM packet discarding code */
|
||||||
return read;
|
return read;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static GstFlowReturn
|
static GstFlowReturn
|
||||||
gst_neonhttp_src_create (GstPushSrc * psrc, GstBuffer ** outbuf)
|
gst_neonhttp_src_create (GstPushSrc * psrc, GstBuffer ** outbuf)
|
||||||
{
|
{
|
||||||
|
@ -452,6 +525,7 @@ send_request_and_redirect (GstNeonhttpSrc * src, gboolean do_redir)
|
||||||
/* When the HTTP status code is 302, it is not the SHOUTcast streaming content yet;
|
/* When the HTTP status code is 302, it is not the SHOUTcast streaming content yet;
|
||||||
* Reload the HTTP request with a new URI value */
|
* Reload the HTTP request with a new URI value */
|
||||||
http_status = ne_get_status (src->request)->code;
|
http_status = ne_get_status (src->request)->code;
|
||||||
|
|
||||||
if (http_status == 302) {
|
if (http_status == 302) {
|
||||||
/* the new URI value to go when redirecting can be found on the 'Location' HTTP header */
|
/* the new URI value to go when redirecting can be found on the 'Location' HTTP header */
|
||||||
redir = ne_get_response_header (src->request, "Location");
|
redir = ne_get_response_header (src->request, "Location");
|
||||||
|
@ -777,7 +851,7 @@ static void
|
||||||
gst_neonhttp_src_set_property (GObject * object, guint prop_id,
|
gst_neonhttp_src_set_property (GObject * object, guint prop_id,
|
||||||
const GValue * value, GParamSpec * pspec)
|
const GValue * value, GParamSpec * pspec)
|
||||||
{
|
{
|
||||||
GstNeonhttpSrc *this = GST_NEONHTTP_SRC (object);
|
GstNeonhttpSrc *neonhttpsrc = GST_NEONHTTP_SRC (object);
|
||||||
|
|
||||||
switch (prop_id) {
|
switch (prop_id) {
|
||||||
case PROP_PROXY:
|
case PROP_PROXY:
|
||||||
|
@ -786,7 +860,8 @@ gst_neonhttp_src_set_property (GObject * object, guint prop_id,
|
||||||
GST_WARNING ("proxy property cannot be NULL");
|
GST_WARNING ("proxy property cannot be NULL");
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
if (!set_proxy (this, g_value_get_string (value), &this->proxy, FALSE)) {
|
if (!set_proxy (neonhttpsrc, g_value_get_string (value),
|
||||||
|
&neonhttpsrc->proxy, FALSE)) {
|
||||||
GST_WARNING ("badly formated proxy");
|
GST_WARNING ("badly formated proxy");
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
@ -799,8 +874,8 @@ gst_neonhttp_src_set_property (GObject * object, guint prop_id,
|
||||||
GST_WARNING ("location property cannot be NULL");
|
GST_WARNING ("location property cannot be NULL");
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
if (!set_uri (this, g_value_get_string (value), &this->uri,
|
if (!set_uri (neonhttpsrc, g_value_get_string (value), &neonhttpsrc->uri,
|
||||||
&this->ishttps, &this->uristr, FALSE)) {
|
&neonhttpsrc->ishttps, &neonhttpsrc->uristr, FALSE)) {
|
||||||
GST_WARNING ("badly formated location");
|
GST_WARNING ("badly formated location");
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
@ -808,29 +883,34 @@ gst_neonhttp_src_set_property (GObject * object, guint prop_id,
|
||||||
}
|
}
|
||||||
case PROP_USER_AGENT:
|
case PROP_USER_AGENT:
|
||||||
{
|
{
|
||||||
if (this->user_agent) {
|
if (neonhttpsrc->user_agent) {
|
||||||
g_free (this->user_agent);
|
g_free (neonhttpsrc->user_agent);
|
||||||
this->user_agent = NULL;
|
neonhttpsrc->user_agent = NULL;
|
||||||
}
|
}
|
||||||
if (g_value_get_string (value)) {
|
if (g_value_get_string (value)) {
|
||||||
this->user_agent = g_strdup (g_value_get_string (value));
|
neonhttpsrc->user_agent = g_strdup (g_value_get_string (value));
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
case PROP_LASTFM_MODE:
|
||||||
|
{
|
||||||
|
neonhttpsrc->lastfm_mode = g_value_get_boolean (value);
|
||||||
|
break;
|
||||||
|
}
|
||||||
case PROP_IRADIO_MODE:
|
case PROP_IRADIO_MODE:
|
||||||
{
|
{
|
||||||
this->iradio_mode = g_value_get_boolean (value);
|
neonhttpsrc->iradio_mode = g_value_get_boolean (value);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case PROP_NEON_HTTP_REDIRECT:
|
case PROP_NEON_HTTP_REDIRECT:
|
||||||
{
|
{
|
||||||
this->neon_http_redirect = g_value_get_boolean (value);
|
neonhttpsrc->neon_http_redirect = g_value_get_boolean (value);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
#ifndef GST_DISABLE_GST_DEBUG
|
#ifndef GST_DISABLE_GST_DEBUG
|
||||||
case PROP_NEON_HTTP_DBG:
|
case PROP_NEON_HTTP_DBG:
|
||||||
{
|
{
|
||||||
this->neon_http_msgs_dbg = g_value_get_boolean (value);
|
neonhttpsrc->neon_http_msgs_dbg = g_value_get_boolean (value);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
@ -885,6 +965,9 @@ gst_neonhttp_src_get_property (GObject * object, guint prop_id,
|
||||||
g_value_set_string (value, neonhttpsrc->user_agent);
|
g_value_set_string (value, neonhttpsrc->user_agent);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
case PROP_LASTFM_MODE:
|
||||||
|
g_value_set_boolean (value, neonhttpsrc->lastfm_mode);
|
||||||
|
break;
|
||||||
case PROP_IRADIO_MODE:
|
case PROP_IRADIO_MODE:
|
||||||
g_value_set_boolean (value, neonhttpsrc->iradio_mode);
|
g_value_set_boolean (value, neonhttpsrc->iradio_mode);
|
||||||
break;
|
break;
|
||||||
|
@ -923,7 +1006,7 @@ plugin_init (GstPlugin * plugin)
|
||||||
GST_TYPE_NEONHTTP_SRC);
|
GST_TYPE_NEONHTTP_SRC);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* this is the structure that gst-register looks for
|
/* neonhttpsrc is the structure that gst-register looks for
|
||||||
* so keep the name plugin_desc, or you cannot get your plug-in registered */
|
* so keep the name plugin_desc, or you cannot get your plug-in registered */
|
||||||
GST_PLUGIN_DEFINE (GST_VERSION_MAJOR,
|
GST_PLUGIN_DEFINE (GST_VERSION_MAJOR,
|
||||||
GST_VERSION_MINOR,
|
GST_VERSION_MINOR,
|
||||||
|
@ -933,7 +1016,7 @@ GST_PLUGIN_DEFINE (GST_VERSION_MAJOR,
|
||||||
|
|
||||||
|
|
||||||
/*** GSTURIHANDLER INTERFACE *************************************************/
|
/*** GSTURIHANDLER INTERFACE *************************************************/
|
||||||
static guint gst_neonhttp_src_uri_get_type (void)
|
static GstURIType gst_neonhttp_src_uri_get_type (void)
|
||||||
{
|
{
|
||||||
return GST_URI_SRC;
|
return GST_URI_SRC;
|
||||||
}
|
}
|
||||||
|
|
|
@ -59,6 +59,7 @@ struct _GstNeonhttpSrc {
|
||||||
|
|
||||||
/* icecast/audiocast metadata extraction handling */
|
/* icecast/audiocast metadata extraction handling */
|
||||||
gboolean iradio_mode;
|
gboolean iradio_mode;
|
||||||
|
gboolean lastfm_mode;
|
||||||
gchar *iradio_name;
|
gchar *iradio_name;
|
||||||
gchar *iradio_genre;
|
gchar *iradio_genre;
|
||||||
gchar *iradio_url;
|
gchar *iradio_url;
|
||||||
|
|
Loading…
Reference in a new issue