Mercurial > audlegacy-plugins
view src/neon/neon.c @ 2890:5e97b55f87cf
Manual merge to fix repository breakage.
author | Matti Hamalainen <ccr@tnsp.org> |
---|---|
date | Tue, 12 Aug 2008 20:40:31 +0300 |
parents | f1b6f1b2cdb3 |
children | 3134a0987162 |
line wrap: on
line source
/* * A neon HTTP input plugin for Audacious * Copyright (C) 2007 Ralf Ertzinger * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include "neon.h" #include <audacious/plugin.h> #include <ne_socket.h> #include <ne_utils.h> #include <ne_redirect.h> #include <ne_request.h> #include <ne_auth.h> #include "debug.h" #include "rb.h" #define NEON_BUFSIZE (128u*1024u) #define NEON_NETBLKSIZE (4096u) #define NEON_ICY_BUFSIZE (4096) VFSFile *neon_aud_vfs_fopen_impl(const gchar* path, const gchar* mode); gint neon_aud_vfs_fclose_impl(VFSFile* file); gsize neon_aud_vfs_fread_impl(gpointer ptr_, gsize size, gsize nmemb, VFSFile* file); gsize neon_aud_vfs_fwrite_impl(gconstpointer ptr, gsize size, gsize nmemb, VFSFile* file); gint neon_aud_vfs_getc_impl(VFSFile* file); gint neon_aud_vfs_ungetc_impl(gint c, VFSFile* file); void neon_aud_vfs_rewind_impl(VFSFile* file); glong neon_aud_vfs_ftell_impl(VFSFile* file); gboolean neon_aud_vfs_feof_impl(VFSFile* file); gint neon_aud_vfs_truncate_impl(VFSFile* file, glong size); gint neon_aud_vfs_fseek_impl(VFSFile* file, glong offset, gint whence); gchar *neon_aud_vfs_metadata_impl(VFSFile* file, const gchar * field); off_t neon_aud_vfs_fsize_impl(VFSFile* file); VFSConstructor neon_http_const = { "http://", neon_aud_vfs_fopen_impl, neon_aud_vfs_fclose_impl, neon_aud_vfs_fread_impl, neon_aud_vfs_fwrite_impl, neon_aud_vfs_getc_impl, neon_aud_vfs_ungetc_impl, neon_aud_vfs_fseek_impl, neon_aud_vfs_rewind_impl, neon_aud_vfs_ftell_impl, neon_aud_vfs_feof_impl, neon_aud_vfs_truncate_impl, neon_aud_vfs_fsize_impl, neon_aud_vfs_metadata_impl }; VFSConstructor neon_https_const = { "https://", neon_aud_vfs_fopen_impl, neon_aud_vfs_fclose_impl, neon_aud_vfs_fread_impl, neon_aud_vfs_fwrite_impl, neon_aud_vfs_getc_impl, neon_aud_vfs_ungetc_impl, neon_aud_vfs_fseek_impl, neon_aud_vfs_rewind_impl, neon_aud_vfs_ftell_impl, neon_aud_vfs_feof_impl, neon_aud_vfs_truncate_impl, neon_aud_vfs_fsize_impl, neon_aud_vfs_metadata_impl }; /* * ---- */ static void neon_plugin_init(void) { gint ret; _ENTER; if (0 != (ret = ne_sock_init())) { _ERROR("Could not initialize neon library: %d\n", ret); _LEAVE; } aud_vfs_register_transport(&neon_http_const); if (0 != ne_has_support(NE_FEATURE_SSL)) { _DEBUG("neon compiled with thread-safe SSL, enabling https:// transport"); aud_vfs_register_transport(&neon_https_const); } _LEAVE; } /* * ----- */ static void neon_plugin_fini(void) { _ENTER; ne_sock_exit(); _LEAVE; } DECLARE_PLUGIN(neon, neon_plugin_init, neon_plugin_fini) /* * ======== */ static struct neon_handle* handle_init(void) { struct neon_handle* h; _ENTER; if (NULL == (h = g_new0(struct neon_handle, 1))) { _ERROR("Could not allocate memory for handle"); _LEAVE NULL; } h->reader_status.mutex = g_mutex_new(); h->reader_status.cond = g_cond_new(); h->reader_status.reading = FALSE; h->reader_status.status = NEON_READER_INIT; if (0 != init_rb_with_lock(&(h->rb), NEON_BUFSIZE, h->reader_status.mutex)) { _ERROR("Could not initialize buffer"); g_free(h); _LEAVE NULL; } h->purl = g_new0(ne_uri, 1); h->content_length = -1; _LEAVE h; } /* * ----- */ static void handle_free(struct neon_handle* h) { _ENTER; _DEBUG("<%p> freeing handle", h); ne_uri_free(h->purl); g_free(h->purl); destroy_rb(&h->rb); g_free(h->icy_metadata.stream_name); g_free(h->icy_metadata.stream_title); g_free(h->icy_metadata.stream_url); g_free(h->icy_metadata.stream_contenttype); g_free(h); _LEAVE; } /* * ----- */ static gboolean neon_strcmp(const gchar *str, const gchar *cmp) { return (g_ascii_strncasecmp(str, cmp, strlen(cmp)) == 0); } static void add_icy(struct icy_metadata* m, gchar* name, gchar* value) { _ENTER; if (neon_strcmp(name, "StreamTitle")) { _DEBUG("Found StreamTitle: %s", value); g_free(m->stream_title); m->stream_title = g_strdup(value); } if (neon_strcmp(name, "StreamUrl")) { _DEBUG("Found StreamUrl: %s", value); g_free(m->stream_url); m->stream_url = g_strdup(value); } _LEAVE; } /* * ----- */ static void parse_icy(struct icy_metadata* m, gchar* metadata, gsize len) { gchar* p; gchar* tstart; gchar* tend; gchar name[NEON_ICY_BUFSIZE]; gchar value[NEON_ICY_BUFSIZE]; gint state; gsize pos; _ENTER; p = metadata; state = 1; pos = 0; name[0] = '\0'; value[0] = '\0'; tstart = metadata; tend = metadata; while ((pos < len) && (*p != '\0')) { switch (state) { case 1: /* * Reading tag name */ if ('=' == *p) { /* * End of tag name. */ *p = '\0'; g_strlcpy(name, tstart, NEON_ICY_BUFSIZE); _DEBUG("Found tag name: %s", name); state = 2; } else { tend = p; }; break; case 2: /* * Waiting for start of value */ if ('\'' == *p) { /* * Leading ' of value */ tend = tstart = p + 1; state = 3; value[0] = '\0'; } break; case 3: /* * Reading value */ if ('\'' == *p && ';' == *(p+1)) { /* * End of value */ *p = '\0'; g_strlcpy(value, tstart, NEON_ICY_BUFSIZE); _DEBUG("Found tag value: %s", value); add_icy(m, name, value); state = 4; } else { tend = p; } break; case 4: /* * Waiting for next tag start */ if (';' == *p) { /* * Next tag name starts after this char */ tend = tstart = p + 1; state = 1; name[0] = '\0'; value[0] = '\0'; } break; default: /* * This should not happen */ _ERROR("Invalid state while parsing metadata, metadata may be corrupted: %d", state); _LEAVE; break; } p++; pos++; } _LEAVE; } /* * ----- */ static void kill_reader(struct neon_handle* h) { _ENTER; if (NULL == h) { _LEAVE; } _DEBUG("Signaling reader thread to terminate"); g_mutex_lock(h->reader_status.mutex); h->reader_status.reading = FALSE; g_cond_signal(h->reader_status.cond); g_mutex_unlock(h->reader_status.mutex); _DEBUG("Waiting for reader thread to die..."); g_thread_join(h->reader); _DEBUG("Reader thread has died"); h->reader = NULL; _LEAVE; } /* * ----- */ static int server_auth_callback(void* userdata, const char* realm, int attempt, char* username, char* password) { struct neon_handle* h = (struct neon_handle*)userdata; gchar* authcpy; gchar** authtok; _ENTER; if ((NULL == h->purl->userinfo) || ('\0' == *(h->purl->userinfo))) { _ERROR("Authentication required, but no credentials set"); _LEAVE 1; } if (NULL == (authcpy = g_strdup(h->purl->userinfo))) { /* * No auth data */ _ERROR("Could not allocate memory for authentication data"); _LEAVE 1; } authtok = g_strsplit(authcpy, ":", 2); if ((strlen(authtok[1]) > (NE_ABUFSIZ-1)) || (strlen(authtok[0]) > (NE_ABUFSIZ-1))) { _ERROR("Username/Password too long"); g_strfreev(authtok); g_free(authcpy); _LEAVE 1; } g_strlcpy(username, authtok[0], NE_ABUFSIZ); g_strlcpy(password, authtok[1], NE_ABUFSIZ); _DEBUG("Authenticating: Username: %s, Password: %s", username, password); g_strfreev(authtok); g_free(authcpy); _LEAVE attempt; } /* * ----- */ static void handle_headers(struct neon_handle* h) { const gchar* name; const gchar* value; void* cursor = NULL; long len; gchar* endptr; _ENTER; _DEBUG("Header responses:"); while(NULL != (cursor = ne_response_header_iterate(h->request, cursor, &name, &value))) { _DEBUG("HEADER: %s: %s", name, value); if (neon_strcmp(name, "accept-ranges")) { /* * The server advertises range capability. we need "bytes" */ if (NULL != g_strrstr(value, "bytes")) { _DEBUG("server can_ranges"); h->can_ranges = TRUE; } continue; } if (neon_strcmp(name, "content-length")) { /* * The server sent us the content length. Parse and store. */ len = strtol(value, &endptr, 10); if ((*value != '\0') && (*endptr == '\0') && (len >= 0)) { /* * Valid data. */ _DEBUG("Content length as advertised by server: %ld", len); h->content_length = len; } else { _ERROR("Invalid content length header: %s", value); } continue; } if (neon_strcmp(name, "content-type")) { /* * The server sent us a content type. Save it for later */ _DEBUG("Content-Type: %s", value); g_free(h->icy_metadata.stream_contenttype); h->icy_metadata.stream_contenttype = g_strdup(value); continue; } if (neon_strcmp(name, "icy-metaint")) { /* * The server sent us a ICY metaint header. Parse and store. */ len = strtol(value, &endptr, 10); if ((*value != '\0') && (*endptr == '\0') && (len > 0)) { /* * Valid data */ _DEBUG("ICY MetaInt as advertised by server: %ld", len); h->icy_metaint = len; h->icy_metaleft = len; } else { _ERROR("Invalid ICY MetaInt header: %s", value); } continue; } if (neon_strcmp(name, "icy-name")) { /* * The server sent us a ICY name. Save it for later */ _DEBUG("ICY stream name: %s", value); g_free(h->icy_metadata.stream_name); h->icy_metadata.stream_name = g_strdup(value); } continue; } _LEAVE; } /* * ----- */ static int neon_proxy_auth_cb(void *userdata, const char *realm, int attempt, char *username, char *password) { mcs_handle_t *db; gchar *value = NULL; _ENTER; if ((db = aud_cfg_db_open()) == NULL) { _DEBUG("<%p> configdb failed to open!", userdata); _LEAVE -1; } aud_cfg_db_get_string(db, NULL, "proxy_user", &value); if (!value) { _DEBUG("<%p> proxy_auth requested but no proxy_user", userdata); _LEAVE -1; } g_strlcpy(username, value, NE_ABUFSIZ); value = NULL; aud_cfg_db_get_string(db, NULL, "proxy_pass", &value); if (!value) { _DEBUG("<%p> proxy_auth requested but no proxy_pass", userdata); _LEAVE -1; } g_strlcpy(password, value, NE_ABUFSIZ); value = NULL; _LEAVE attempt; } /* * ----- */ static int open_request(struct neon_handle* handle, gulong startbyte) { int ret; const ne_status* status; ne_uri* rediruri; _ENTER; if (handle->purl->query && *(handle->purl->query)) { gchar *tmp = g_strdup_printf("%s?%s", handle->purl->path, handle->purl->query); handle->request = ne_request_create(handle->session, "GET", tmp); g_free(tmp); } else { handle->request = ne_request_create(handle->session, "GET", handle->purl->path); } if (0 < startbyte) { ne_print_request_header(handle->request, "Range", "bytes=%ld-", startbyte); } ne_print_request_header(handle->request, "Icy-MetaData", "1"); /* * Try to connect to the server. */ _DEBUG("<%p> Connecting...", handle); ret = ne_begin_request(handle->request); status = ne_get_status(handle->request); _DEBUG("<%p> Return: %d, Status: %d", handle, ret, status->code); if ((NE_OK == ret) && (401 == status->code)) { /* * Authorization required. Reconnect to * authenticate */ _DEBUG("Reconnecting due to 401"); ne_end_request(handle->request); ret = ne_begin_request(handle->request); } if ((NE_OK == ret) && ((301 == status->code) || (302 == status->code) || (303 == status->code) || (307 == status->code))) { /* * Redirect encountered. Reconnect. */ ne_end_request(handle->request); ret = NE_REDIRECT; } if ((NE_OK == ret) && (407 == status->code)) { /* * Proxy auth required. Reconnect to authenticate */ _DEBUG("Reconnecting due to 407"); ne_end_request(handle->request); ret = ne_begin_request(handle->request); } switch (ret) { case NE_OK: if ((status->code > 199) && (status->code < 300)) { /* URL opened OK */ _DEBUG("<%p> URL opened OK", handle); handle->content_start = startbyte; handle->pos = startbyte; handle_headers(handle); _LEAVE 0; break; } case NE_REDIRECT: /* We hit a redirect. Handle it. */ _DEBUG("<%p> Redirect encountered", handle); handle->redircount += 1; rediruri = (ne_uri*)ne_redirect_location(handle->session); ne_request_destroy(handle->request); if (NULL == rediruri) { _ERROR("<%p> Could not parse redirect response", handle); _LEAVE -1; } ne_uri_free(handle->purl); ne_uri_copy(handle->purl, rediruri); _LEAVE 1; break; default: /* Something went wrong. */ _ERROR("<%p> Could not open URL: %d (%d)", handle, ret, status->code); if (1 == ret) { _ERROR("<%p> neon error string: %s", handle, ne_get_error(handle->session)); } ne_request_destroy(handle->request); _LEAVE -1; break; } } /* * ----- */ static gint open_handle(struct neon_handle* handle, gulong startbyte) { gint ret; mcs_handle_t* db; gchar* proxy_host = NULL; gchar* proxy_port_s = NULL; gchar* endptr; guint proxy_port = 0; gboolean use_proxy, use_proxy_auth; _ENTER; db = aud_cfg_db_open(); if (FALSE == aud_cfg_db_get_bool(db, NULL, "use_proxy", &use_proxy)) { use_proxy = FALSE; } if (FALSE == aud_cfg_db_get_bool(db, NULL, "use_proxy_auth", &use_proxy_auth)) { use_proxy_auth = FALSE; } if (use_proxy) { if (FALSE == aud_cfg_db_get_string(db, NULL, "proxy_host", &proxy_host)) { _ERROR("<%p> Could not read proxy host, disabling proxy use", handle); use_proxy = FALSE; } if (FALSE == aud_cfg_db_get_string(db, NULL, "proxy_port", &proxy_port_s)) { _ERROR("<%p> Could not read proxy port, disabling proxy use", handle); use_proxy = FALSE; } proxy_port = strtoul(proxy_port_s, &endptr, 10); if (!((*proxy_port_s != '\0') && (*endptr == '\0') && (proxy_port < 65536))) { /* * Invalid data */ _ERROR("<%p> Invalid proxy port, disabling proxy use", handle); use_proxy = FALSE; } } aud_cfg_db_close(db); handle->redircount = 0; _DEBUG("<%p> Parsing URL", handle); if (0 != ne_uri_parse(handle->url, handle->purl)) { _ERROR("<%p> Could not parse URL '%s'", handle, handle->url); _LEAVE -1; } while (handle->redircount < 10) { if (0 == handle->purl->port) { handle->purl->port = ne_uri_defaultport(handle->purl->scheme); } _DEBUG("<%p> Creating session to %s://%s:%d", handle, handle->purl->scheme, handle->purl->host, handle->purl->port); handle->session = ne_session_create(handle->purl->scheme, handle->purl->host, handle->purl->port); ne_redirect_register(handle->session); ne_add_server_auth(handle->session, NE_AUTH_BASIC, server_auth_callback, (void *)handle); ne_set_session_flag(handle->session, NE_SESSFLAG_ICYPROTO, 1); ne_set_session_flag(handle->session, NE_SESSFLAG_PERSIST, 0); #ifdef HAVE_NE_SET_CONNECT_TIMEOUT ne_set_connect_timeout(handle->session, 10); #endif ne_set_read_timeout(handle->session, 10); ne_set_useragent(handle->session, "Audacious/" PACKAGE_VERSION ); if (use_proxy) { _DEBUG("<%p> Using proxy: %s:%d", handle, proxy_host, proxy_port); ne_session_proxy(handle->session, proxy_host, proxy_port); if (use_proxy_auth) { _DEBUG("<%p> Using proxy authentication", handle); ne_add_proxy_auth(handle->session, NE_AUTH_BASIC, neon_proxy_auth_cb, (void *)handle); } } _DEBUG("<%p> Creating request", handle); ret = open_request(handle, startbyte); if (0 == ret) { _LEAVE 0; } else if (-1 == ret) { ne_session_destroy(handle->session); _LEAVE -1; } _DEBUG("<%p> Following redirect...", handle); ne_session_destroy(handle->session); } /* * If we get here, our redirect count exceeded */ _ERROR("<%p> Redirect count exceeded for URL %s", handle, handle->url); _LEAVE 1; } /* * ----- */ static gint fill_buffer(struct neon_handle* h) { gssize bsize; gchar buffer[NEON_NETBLKSIZE]; gssize to_read; _ENTER; bsize = free_rb(&h->rb); to_read = MIN(bsize, NEON_NETBLKSIZE); _DEBUG("<%p> %d bytes free in buffer, trying to read %d bytes max", h, bsize, to_read); _DEBUG("<%p> Reading from the network....", h); if (0 >= (bsize = ne_read_response_block(h->request, buffer, to_read))) { if (0 == bsize) { _DEBUG("<%p> End of file encountered", h); _LEAVE 1; } else { _ERROR("<%p> Error while reading from the network", h); _LEAVE -1; } } _DEBUG("<%p> Read %d bytes from the network", h, bsize); if (0 != write_rb(&(h->rb), buffer, bsize)) { _ERROR("<%p> Error putting data into buffer", h); _LEAVE -1; } _LEAVE 0; } /* * ----- */ static int fill_buffer_limit(struct neon_handle* h, unsigned int maxfree) { gssize bfree; gint ret; _ENTER; bfree = free_rb(&h->rb); _DEBUG("<%p> Filling buffer up to max %d bytes free, %d bytes free now", h, maxfree, bfree); while (bfree > maxfree) { ret = fill_buffer(h); if (-1 == ret) { _ERROR("<%p> Error while filling buffer", h); _LEAVE ret; } else if (1 == ret) { /* * EOF while filling the buffer. Return what we have. */ _LEAVE 1; } bfree = free_rb(&h->rb); } _LEAVE 0; } /* * ----- */ static gpointer reader_thread(void* data) { struct neon_handle* h = (struct neon_handle*)data; gint ret; _ENTER; g_mutex_lock(h->reader_status.mutex); while(h->reader_status.reading) { /* * Hit the network only if we have more than NEON_NETBLKSIZE of free buffer */ if (NEON_NETBLKSIZE < free_rb_locked(&h->rb)) { g_mutex_unlock(h->reader_status.mutex); _DEBUG("<%p> Filling buffer...", h); ret = fill_buffer(h); g_mutex_lock(h->reader_status.mutex); if (-1 == ret) { /* * Error encountered while reading from the network. * Set the error flag and terminate the * reader thread. */ _ERROR("<%p> Error while reading from the network. Terminating reader thread", h); h->reader_status.status = NEON_READER_ERROR; g_mutex_unlock(h->reader_status.mutex); _LEAVE NULL; } else if (1 == ret) { /* * EOF encountered while reading from the * network. Set the EOF status and exit. */ _DEBUG("<%p> EOF encountered while reading from the network. Terminating reader thread", h); h->reader_status.status = NEON_READER_EOF; g_mutex_unlock(h->reader_status.mutex); _LEAVE NULL; } /* * So we actually got some data out of the stream. */ _DEBUG("<%p> Network read succeeded", h); } else { /* * Not enough free space in the buffer. * Sleep until the main thread wakes us up. */ _DEBUG("<%p> Reader thread going to sleep", h); g_cond_wait(h->reader_status.cond, h->reader_status.mutex); _DEBUG("<%p> Reader thread woke up", h); } } _DEBUG("<%p> Reader thread terminating gracefully", h); h->reader_status.status = NEON_READER_TERM; g_mutex_unlock(h->reader_status.mutex); _LEAVE NULL; } /* * ----- */ VFSFile* neon_aud_vfs_fopen_impl(const gchar* path, const gchar* mode) { VFSFile* file; struct neon_handle* handle; _ENTER; _DEBUG("Trying to open '%s' with neon", path); if (NULL == (file = g_new0(VFSFile, 1))) { _ERROR("Could not allocate memory for filehandle"); _LEAVE NULL; } if (NULL == (handle = handle_init())) { _ERROR("Could not allocate memory for neon handle"); g_free(file); _LEAVE NULL; } _DEBUG("Allocated new handle: %p", handle); if (NULL == (handle->url = strdup(path))) { _ERROR("<%p> Could not copy URL string", handle); handle_free(handle); g_free(file); _LEAVE NULL; } if (0 != open_handle(handle, 0)) { _ERROR("<%p> Could not open URL", handle); handle_free(handle); g_free(file); _LEAVE NULL; } file->handle = handle; file->base = &neon_http_const; _LEAVE file; } /* * ---- */ gint neon_aud_vfs_fclose_impl(VFSFile* file) { struct neon_handle* h = (struct neon_handle *)file->handle; _ENTER; if (NULL != h->reader) { kill_reader(h); } _DEBUG("<%p> Destroying request", h); if (NULL != h->request) { ne_request_destroy(h->request); } _DEBUG("<%p> Destroying session", h); ne_session_destroy(h->session); handle_free(h); _LEAVE 0; } /* * ----- */ gsize neon_aud_vfs_fread_impl(gpointer ptr_, gsize size, gsize nmemb, VFSFile* file) { struct neon_handle* h = (struct neon_handle*)file->handle; gint belem; gint relem; gint ret; gchar icy_metadata[NEON_ICY_BUFSIZE]; guchar icy_metalen; _ENTER; if (NULL == h->request) { _ERROR("<%p> No request to read from, seek gone wrong?", h); _LEAVE 0; } _DEBUG("<%p> Requesting %d elements of %d bytes size each (%d bytes total), to be stored at %p", h, nmemb, size, (nmemb*size), ptr_); /* * Look how much data is in the buffer */ belem = used_rb(&h->rb) / size; if ((NULL != h->reader) && (0 == belem)) { /* * There is a reader thread, but the buffer is empty. * If we are running normally we will have to rebuffer. * Kill the reader thread and restart. */ g_mutex_lock(h->reader_status.mutex); if (NEON_READER_RUN == h->reader_status.status) { g_mutex_unlock(h->reader_status.mutex); _ERROR("<%p> Buffer underrun, trying rebuffering", h); kill_reader(h); /* * We have to check if the reader terminated gracefully * again */ if ((NEON_READER_TERM != h->reader_status.status) && (NEON_READER_EOF != h->reader_status.status)) { /* * Reader thread did not terminate gracefully. */ _ERROR("<%p> Reader thread did not terminate gracefully: %d", h, h->reader_status.status); _LEAVE 0; } } else { g_mutex_unlock(h->reader_status.mutex); } } if (NULL == h->reader) { if (NEON_READER_EOF != h->reader_status.status) { /* * There is no reader thread yet. Read the first bytes from * the network ourselves, and then fire up the reader thread * to keep the buffer filled up. */ _DEBUG("<%p> Doing initial buffer fill", h); ret = fill_buffer_limit(h, NEON_BUFSIZE/2); if (-1 == ret) { _ERROR("<%p> Error while reading from the network", h); _LEAVE 0; } else if (1 == ret) { _DEBUG("<%p> EOF during initial read", h); } /* * We have some data in the buffer now. * Start the reader thread if we did not reach EOF during * the initial fill */ g_mutex_lock(h->reader_status.mutex); if (0 == ret) { h->reader_status.reading = TRUE; _DEBUG("<%p> Starting reader thread", h); if (NULL == (h->reader = g_thread_create(reader_thread, h, TRUE, NULL))) { h->reader_status.reading = FALSE; g_mutex_unlock(h->reader_status.mutex); _ERROR("<%p> Error creating reader thread!", h); _LEAVE 0; } h->reader_status.status = NEON_READER_RUN; } else { _DEBUG("<%p> No reader thread needed (stream has reached EOF during fill)", h); h->reader_status.reading = FALSE; h->reader_status.status = NEON_READER_EOF; } g_mutex_unlock(h->reader_status.mutex); } } else { /* * There already is a reader thread. Look if it is in good * shape. */ g_mutex_lock(h->reader_status.mutex); _DEBUG("<%p> Reader thread status: %d", h, h->reader_status.status); switch (h->reader_status.status) { case NEON_READER_INIT: case NEON_READER_RUN: /* * All is well, nothing to be done. */ break; case NEON_READER_EOF: /* * If there still is data in the buffer, carry on. * If not, terminate the reader thread and return 0. */ if (0 == used_rb_locked(&h->rb)) { _DEBUG("<%p> Reached end of stream", h); g_mutex_unlock(h->reader_status.mutex); kill_reader(h); h->eof = TRUE; _LEAVE 0; } break; case NEON_READER_ERROR: /* Terminate the reader and return 0 */ g_mutex_unlock(h->reader_status.mutex); kill_reader(h); _LEAVE 0; break; case NEON_READER_TERM: /* * The reader thread terminated gracefully, most * likely on our own request. * We should not get here. */ _ERROR("<%p> Reader thread terminated and fread() called. How did we get here?", h); g_mutex_unlock(h->reader_status.mutex); kill_reader(h); _LEAVE 0; } g_mutex_unlock(h->reader_status.mutex); } /* * Deliver data from the buffer */ if (0 == used_rb(&h->rb)) { /* * The buffer is still empty, we can deliver no data! */ _ERROR("<%p> Buffer still underrun, fatal.", h); _LEAVE 0; } if (0 != h->icy_metaint) { _DEBUG("<%p> %ld bytes left before next ICY metadata announcement", h, h->icy_metaleft); if (0 == h->icy_metaleft) { /* * The next data in the buffer is a ICY metadata announcement. * Get the length byte */ read_rb(&h->rb, &icy_metalen, 1); /* * We need enough data in the buffer to * a) Read the complete ICY metadata block * b) deliver at least one byte to the reader */ _DEBUG("<%p> Expecting %d bytes of ICY metadata", h, (icy_metalen*16)); if ((free_rb(&h->rb)-(icy_metalen*16)) < size) { /* There is not enough data. We do not have much choice at this point, * so we'll deliver the metadata as normal data to the reader and * hope for the best. */ _ERROR("<%p> Buffer underrun when reading metadata. Expect audio degradation", h); h->icy_metaleft = h->icy_metaint + (icy_metalen*16); } else { /* * Grab the metadata from the buffer and send it to the parser */ read_rb(&h->rb, icy_metadata, (icy_metalen*16)); parse_icy(&h->icy_metadata, icy_metadata, (icy_metalen*16)); h->icy_metaleft = h->icy_metaint; } } /* * The maximum number of bytes we can deliver is determined * by the number of bytes left until the next metadata announcement */ belem = MIN(used_rb(&h->rb), h->icy_metaleft) / size; } else { belem = used_rb(&h->rb) / size; } relem = MIN(belem, nmemb); _DEBUG("<%p> %d elements of returnable data in the buffer", h, belem); read_rb(&h->rb, ptr_, relem*size); /* * Signal the network thread to continue reading */ g_mutex_lock(h->reader_status.mutex); if (NEON_READER_EOF == h->reader_status.status) { if (0 == free_rb_locked(&h->rb)) { _DEBUG("<%p> stream EOF reached and buffer empty", h); h->eof = TRUE; } } else { _DEBUG("<%p> Waking up reader thread", h); g_cond_signal(h->reader_status.cond); } g_mutex_unlock(h->reader_status.mutex); h->pos += (relem*size); h->icy_metaleft -= (relem*size); _DEBUG("<%p> Returning %d elements", h, relem); _LEAVE relem; } /* * ----- */ gsize neon_aud_vfs_fwrite_impl(gconstpointer ptr, gsize size, gsize nmemb, VFSFile* file) { _ENTER; _ERROR("<%p> NOT IMPLEMENTED", file->handle); _LEAVE 0; } /* * ----- */ gint neon_aud_vfs_getc_impl(VFSFile* file) { gchar c; _ENTER; if (1 != neon_aud_vfs_fread_impl(&c, 1, 1, file)) { _ERROR("<%p> Could not getc()!", file->handle); _LEAVE -1; } _LEAVE c; } /* * ----- */ gint neon_aud_vfs_ungetc_impl(gint c, VFSFile* stream) { _ENTER; _ERROR("<%p> NOT IMPLEMENTED", stream->handle); _LEAVE 0; } /* * ----- */ void neon_aud_vfs_rewind_impl(VFSFile* file) { _ENTER; (void)neon_aud_vfs_fseek_impl(file, 0L, SEEK_SET); _LEAVE; } /* * ----- */ glong neon_aud_vfs_ftell_impl(VFSFile* file) { struct neon_handle* h = (struct neon_handle *)file->handle; _ENTER; _DEBUG("<%p> Current file position: %ld", h, h->pos); _LEAVE h->pos; } /* * ----- */ gboolean neon_aud_vfs_feof_impl(VFSFile* file) { struct neon_handle* h = (struct neon_handle*)file->handle; _ENTER; _DEBUG("<%p> EOF status: %s", h, h->eof?"TRUE":"FALSE"); _LEAVE h->eof; } /* * ----- */ gint neon_aud_vfs_truncate_impl(VFSFile* file, glong size) { _ENTER; _ERROR("<%p> NOT IMPLEMENTED", file->handle); _LEAVE 0; } /* * ----- */ gint neon_aud_vfs_fseek_impl(VFSFile* file, glong offset, gint whence) { struct neon_handle* h = (struct neon_handle*)file->handle; long newpos; long content_length; _ENTER; _DEBUG("<%p> Seek requested: offset %ld, whence %d", h, offset, whence); /* * Two things must be satisfied for us to be able to seek: * - the server must advertise a content-length * - the server must advertise accept-ranges: bytes */ if ((-1 == h->content_length) || !h->can_ranges) { _DEBUG("<%p> Can not seek due to server restrictions", h); _LEAVE -1; } content_length = h->content_length + h->content_start; switch (whence) { case SEEK_SET: newpos = offset; break; case SEEK_CUR: newpos = h->pos + offset; break; case SEEK_END: newpos = content_length + offset; break; default: _ERROR("<%p> Invalid whence specified", h); _LEAVE -1; } _DEBUG("<%p> Position to seek to: %ld, current: %ld", h, newpos, h->pos); if (0 > newpos) { _ERROR("<%p> Can not seek before start of stream", h); _LEAVE -1; } if (newpos >= content_length) { _ERROR("<%p> Can not seek beyond end of stream", h); _LEAVE -1; } if (newpos == h->pos) { _LEAVE 0; } /* * To seek to the new position we have to * - stop the current reader thread, if there is one * - destroy the current request * - dump all data currently in the ringbuffer * - create a new request starting at newpos */ if (NULL != h->reader) { /* * There may be a thread still running. */ kill_reader(h); } ne_request_destroy(h->request); ne_session_destroy(h->session); reset_rb(&h->rb); if (0 != open_handle(h, newpos)) { /* * Something went wrong while creating the new request. * There is not much we can do now, we'll set the request * to NULL, so that fread() will error out on the next * read request */ _ERROR("<%p> Error while creating new request!", h); h->request = NULL; _LEAVE -1; } /* * Things seem to have worked. The next read request will start * the reader thread again. */ _LEAVE 0; } /* * ----- */ gchar *neon_aud_vfs_metadata_impl(VFSFile* file, const gchar* field) { struct neon_handle* h = (struct neon_handle*)file->handle; _ENTER; _DEBUG("<%p> Field name: %s", h, field); if (neon_strcmp(field, "track-name")) { _LEAVE g_strdup(h->icy_metadata.stream_title); } if (neon_strcmp(field, "stream-name")) { _LEAVE g_strdup(h->icy_metadata.stream_name); } if (neon_strcmp(field, "content-type")) { _LEAVE g_strdup(h->icy_metadata.stream_contenttype); } _LEAVE NULL; } /* * ----- */ off_t neon_aud_vfs_fsize_impl(VFSFile* file) { struct neon_handle* h = (struct neon_handle*)file->handle; _ENTER; if (-1 == h->content_length) { _DEBUG("<%p> Unknown content length", h); _LEAVE -1; } _LEAVE (h->content_start + h->content_length); }