Mercurial > audlegacy-plugins
view src/curl/curl.c @ 468:2e4d10f3f9cb trunk
[svn] - always make sure the returned data is strdup'd
author | nenolod |
---|---|
date | Sun, 21 Jan 2007 01:55:08 -0800 |
parents | 20657d994de8 |
children | 1256144d2a6e |
line wrap: on
line source
/* Audacious * Copyright (c) 2007 Daniel Barkalow * * 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 <audacious/vfs.h> #include <audacious/plugin.h> #include <curl/curl.h> #include <string.h> #define BUFFER_SIZE 256 * 1024 #define REVERSE_SEEK_SIZE 2048 #define DEBUG_CONNECTION 0 #define DEBUG_OPEN_CLOSE 0 #define DEBUG_SEEK 0 #define DEBUG_READ 0 #define DEBUG_HEADERS 0 #define DEBUG_ICY 0 #define DEBUG_ICY_VERBOSE 0 #define XXXX_FIXED_METADATA 0 #define DEBUG_METADATA_REPORT 0 typedef struct _CurlHandle CurlHandle; struct _CurlHandle { CURL *curl; gssize length; // the length of the file gsize rd_abs; // the absolute position for reading from the stream gsize wr_abs; // the absolute position where the input connection is gsize icy_left; gsize icy_interval; gint in_icy_meta; // 0=no, 1=before size, 2=in data gsize meta_abs; // the absolute position where the metadata changes gsize buffer_length; gchar *buffer; gsize rd_index; gsize wr_index; gsize hdr_index; gboolean header; // true if we haven't finished the header yet gboolean no_data; // true if we're only looking for length currently gboolean cancel; // true if the thread should be cancelled gboolean failed; // true if we've tried and failed already GThread *thread; // the thread that's reading from the connection gchar *name; gchar *title; }; /* TODO: * - Fix hang if the server closes the connection in the middle * - Clever buffer stuff when you read a bit of the beginning and a bit of the * end of a file */ /* The goal here is to have a buffering system which handles the following: * 1) open, seek, read (without fetching the beginning of the file) * 2) open, seek END, tell (using HEAD only) * 3) open, read, seek 0, read (without restarting fetch) */ static size_t buf_space(CurlHandle *handle) { size_t rd_edge = handle->rd_abs - REVERSE_SEEK_SIZE; size_t buffer_limit; size_t cont_limit; if (rd_edge < 0) rd_edge = 0; buffer_limit = handle->buffer_length - (handle->wr_abs - rd_edge); cont_limit = handle->buffer_length - handle->wr_index; if (cont_limit < buffer_limit) buffer_limit = cont_limit; if (handle->icy_interval) { if (handle->icy_left < buffer_limit) buffer_limit = handle->icy_left; } return buffer_limit; } static size_t buf_available(CurlHandle *handle) { size_t buffer_limit; size_t cont_limit; if (handle->header) return 0; buffer_limit = handle->wr_abs - handle->rd_abs; cont_limit = handle->buffer_length - handle->rd_index; if (buffer_limit <= 0) return 0; return buffer_limit < cont_limit ? buffer_limit : cont_limit; } static void check(CurlHandle *handle) { if (!((handle->wr_abs - handle->wr_index) % handle->buffer_length == (handle->rd_abs - handle->rd_index) % handle->buffer_length)) printf("%p Not aligned! wr %d rd %d\n", handle, (handle->wr_abs - handle->wr_index) % handle->buffer_length, (handle->rd_abs - handle->rd_index) % handle->buffer_length); } static void update_length(CurlHandle *handle) { if (handle->length == -1) { double value; int retcode = curl_easy_getinfo(handle->curl, CURLINFO_CONTENT_LENGTH_DOWNLOAD, &value); if (retcode == CURLE_OK) { handle->length = value; if (handle->length == 0) handle->length = -2; //g_print("Length: %d\n", handle->length); } else { handle->length = -2; g_print("getinfo gave error\n"); } } } #define PROBE 262140 #define ICY_NAME "icy-name:" #define ICY_METAINT "icy-metaint:" static gboolean match_header(CurlHandle *handle, size_t size, const char *header) { if (strlen(header) > size) return FALSE; // XXXX wrapped headers return !(strncmp(handle->buffer + handle->hdr_index, header, strlen(header))); } static gchar *get_value(CurlHandle *handle, size_t size, const char *header) { // XXXX wrapped headers return strdup(handle->buffer + (handle->hdr_index + strlen(header)) % handle->buffer_length); } static void got_header(CurlHandle *handle, size_t size) { if (DEBUG_HEADERS) g_print("Got header %d bytes\n", size); if (match_header(handle, size, ICY_NAME)) { handle->name = get_value(handle, size, ICY_NAME); if (DEBUG_HEADERS) { g_print("Stream name: %s\n", handle->name); } } if (match_header(handle, size, ICY_METAINT)) { gchar *value = get_value(handle, size, ICY_METAINT); handle->icy_interval = atoi(value); free(value); if (DEBUG_HEADERS) g_print("Metadata interval: %d\n", handle->icy_interval); } } #define TITLE_INLINE "StreamTitle=" static gboolean match_inline(CurlHandle *handle, size_t posn, const char *name) { // XXXX Wrapped return (!strncmp(handle->buffer + posn, name, strlen(name))); } static gchar *get_inline_value(CurlHandle *handle, size_t posn, const char *name) { // XXXX Wrapped size_t end; size_t sz; gchar *ret; posn += strlen(name); end = posn + 1; while (handle->buffer[end] != ';') end++; sz = end - posn - 1; ret = g_malloc(sz); memcpy(ret, handle->buffer + posn + 1, sz); ret[sz - 1] = '\0'; return ret; } static void got_inline_metadata(CurlHandle *handle) { size_t i = (handle->hdr_index + 1) % handle->buffer_length; if (match_inline(handle, i, TITLE_INLINE)) { handle->title = get_inline_value(handle, i, TITLE_INLINE); if (DEBUG_ICY) g_print("Title: '%s'\n", handle->title); } handle->meta_abs = handle->wr_abs; } static size_t curl_writecb(void *ptr, size_t size, size_t nmemb, void *stream) { CurlHandle *handle = stream; gint sz = size * nmemb; gint ret = 0; gint trans; if (!handle->header) update_length(handle); while (ret < sz) { while (!(trans = buf_space(handle)) && !handle->cancel) { g_usleep(10000); //g_print("Wait for free space\n"); } if (handle->cancel) break; if (trans > sz - ret) trans = sz - ret; memcpy(handle->buffer + handle->wr_index, ptr + ret, trans); if (!handle->header) { if (handle->icy_interval) handle->icy_left -= trans; if (!handle->in_icy_meta) { handle->wr_abs += trans; // write download here if (handle->icy_interval && !handle->icy_left) { if (DEBUG_ICY) g_print("Metadata inline after %d\n", handle->wr_abs); handle->in_icy_meta = 1; handle->icy_left = 1; } } else if (handle->in_icy_meta == 1) { // Track where the header insert starts handle->hdr_index = handle->wr_index; handle->icy_left = ((unsigned char)(handle->buffer[handle->wr_index])) * 16; if (DEBUG_ICY) g_print("Metadata of size %d\n", handle->icy_left); if (handle->icy_left) { handle->in_icy_meta = 2; } else { handle->in_icy_meta = 0; handle->icy_left = handle->icy_interval; handle->wr_index--; } } else { if (!handle->icy_left) { handle->wr_index = (handle->wr_index + trans) % handle->buffer_length; if (DEBUG_ICY_VERBOSE) { if (handle->wr_index < handle->hdr_index) { // wrapped fwrite(handle->buffer + handle->hdr_index + 1, handle->buffer_length - handle->hdr_index - 1, 1, stdout); fwrite(handle->buffer, handle->wr_index, 1, stdout); } else { fwrite(handle->buffer + handle->hdr_index, handle->wr_index - handle->hdr_index, 1, stdout); } g_print("\n"); } if (XXXX_FIXED_METADATA) got_inline_metadata(handle); // Rewind the buffer usage to write over the // metadata with content. -trans because we're about // to add it. handle->wr_index = handle->hdr_index - trans; handle->in_icy_meta = 0; handle->icy_left = handle->icy_interval; } } } handle->wr_index = (handle->wr_index + trans) % handle->buffer_length; ret += trans; if (handle->header) { gsize i = handle->hdr_index; while (1) { if ((i + 1) % handle->buffer_length == handle->wr_index) break; if (handle->buffer[i] == '\r' && handle->buffer[(i + 1) % handle->buffer_length] == '\n') { gsize size = (handle->buffer_length + i - handle->hdr_index) % handle->buffer_length; handle->buffer[i] = '\0'; got_header(handle, size); if (i == handle->hdr_index) { // Empty header means the end of the headers handle->header = 0; handle->hdr_index = (i + 2) % handle->buffer_length; // We read from the start of the data in the request handle->rd_index = handle->hdr_index; // We've already written the amount that's after // the header. handle->wr_abs += (handle->wr_index - handle->hdr_index + handle->buffer_length) % handle->buffer_length; // write download here... //handle->icy_interval = 0; handle->icy_left = handle->icy_interval; if (handle->icy_interval) { handle->icy_left -= (handle->wr_index - handle->hdr_index + handle->buffer_length) % handle->buffer_length; } if (DEBUG_ICY) g_print("Left %d\n", handle->icy_left); handle->in_icy_meta = 0; break; } handle->hdr_index = (i + 2) % handle->buffer_length; } i = (i + 1) % handle->buffer_length; } } } return ret; } static gpointer curl_manage_request(gpointer arg) { CurlHandle *handle = arg; CURLcode result; if (DEBUG_CONNECTION) g_print("Connect %p\n", handle); if (handle->no_data) curl_easy_setopt(handle->curl, CURLOPT_NOBODY, 1); else { if (DEBUG_CONNECTION) g_print("Start from %d\n", handle->wr_abs); curl_easy_setopt(handle->curl, CURLOPT_RESUME_FROM, handle->wr_abs); curl_easy_setopt(handle->curl, CURLOPT_NOBODY, 0); curl_easy_setopt(handle->curl, CURLOPT_HTTPGET, 1); } handle->header = 1; handle->hdr_index = 0; handle->icy_interval = 0; result = curl_easy_perform(handle->curl); if (result == CURLE_OK) update_length(handle); // We expect to get CURLE_WRITE_ERROR if we cancel. // We get CURLE_GOT_NOTHING if we send a HEAD request to a shoutcast server. // We get CURLE_HTTP_RANGE_ERROR if we try to use range with shoutcast. if (result != CURLE_OK && result != CURLE_WRITE_ERROR && result != CURLE_GOT_NOTHING && CURLE_HTTP_RANGE_ERROR) { g_print("Got curl error %d\n", result); handle->failed = 1; } if (DEBUG_CONNECTION) g_print("Done %p%s", handle, handle->cancel ? " (aborted)\n" : "\n"); handle->cancel = 1; return NULL; } static void curl_req_xfer(CurlHandle *handle) { if (handle->failed) { handle->cancel = 1; return; } if (!handle->thread) { handle->cancel = 0; handle->wr_index = 0; handle->rd_index = 0; handle->wr_abs = handle->rd_abs; if (DEBUG_CONNECTION) g_print("Starting connection %p at %d\n", handle, handle->wr_abs); handle->thread = g_thread_create(curl_manage_request, handle, TRUE, NULL); } } static void curl_req_sync_xfer(CurlHandle *handle, size_t old_rd_abs) { handle->rd_index = (handle->rd_index + handle->rd_abs - old_rd_abs + handle->buffer_length) % handle->buffer_length; } static void curl_req_no_xfer(CurlHandle *handle) { if (handle->thread) { handle->cancel = 1; g_thread_join(handle->thread); handle->thread = NULL; handle->cancel = 0; } } VFSFile * curl_vfs_fopen_impl(const gchar * path, const gchar * mode) { gchar *url = g_malloc(strlen(path) + strlen("http://") + 1); CurlHandle *handle; VFSFile *file; if (!path || !mode) return NULL; sprintf(url, "http://%s", path); file = g_new(VFSFile, 1); handle = g_new(CurlHandle, 1); handle->curl = curl_easy_init(); handle->rd_index = 0; handle->wr_index = 0; handle->meta_abs = 0; handle->rd_abs = 0; handle->wr_abs = 0; handle->buffer_length = BUFFER_SIZE; handle->buffer = g_malloc(handle->buffer_length); handle->thread = NULL; handle->length = -1; handle->cancel = 0; handle->failed = 0; handle->no_data = 0; curl_easy_setopt(handle->curl, CURLOPT_URL, url); curl_easy_setopt(handle->curl, CURLOPT_WRITEFUNCTION, curl_writecb); curl_easy_setopt(handle->curl, CURLOPT_WRITEDATA, handle); curl_easy_setopt(handle->curl, CURLOPT_HEADERDATA, handle); curl_easy_setopt(handle->curl, CURLOPT_CONNECTTIMEOUT, 10); //add header icy-metadata:1 (when we're ready for it) { struct curl_slist *hdr = NULL; hdr = curl_slist_append(hdr, "icy-metadata:1"); curl_easy_setopt(handle->curl, CURLOPT_HTTPHEADER, hdr); } file->handle = handle; if (DEBUG_OPEN_CLOSE) g_print("Open %s with curl => %p\n", url, handle); return file; } gint curl_vfs_fclose_impl(VFSFile * file) { gint ret = 0; if (file == NULL) return -1; if (DEBUG_OPEN_CLOSE) g_print("Close %p\n", file->handle); if (file->handle) { CurlHandle *handle = file->handle; if (DEBUG_CONNECTION) g_print("Cancel transfer\n"); curl_req_no_xfer(handle); if (DEBUG_CONNECTION) g_print("Okay\n"); g_free(handle->buffer); if (handle->name) g_free(handle->name); curl_easy_cleanup(handle->curl); g_free(handle); } return ret; } size_t curl_vfs_fread_impl(gpointer ptr, size_t size, size_t nmemb, VFSFile * file) { CurlHandle *handle = file->handle; ssize_t sz = size * nmemb; size_t ret = 0; if (sz < 0) return 0; //g_print("Reading %d*%d=%d from %p\n", size, nmemb, sz, handle); curl_req_xfer(handle); if (DEBUG_SEEK) check(handle); while (ret < sz) { size_t available; while (!(available = buf_available(handle)) && !handle->cancel) g_usleep(10000); if (available > sz - ret) available = sz - ret; memcpy(ptr + ret, handle->buffer + handle->rd_index, available); handle->rd_index = (handle->rd_index + available) % handle->buffer_length; if (handle->rd_abs < handle->meta_abs && handle->rd_abs + available >= handle->meta_abs) { if (DEBUG_METADATA_REPORT) g_print("New song: '%s'\n", handle->title); } handle->rd_abs += available; ret += available; if (!available) { //g_print("EOF reading from %p\n", handle); break; } } //g_print("Read %d from %p\n", ret, handle); return ret; } size_t curl_vfs_fwrite_impl(gconstpointer ptr, size_t size, size_t nmemb, VFSFile * file) { return 0; } gint curl_vfs_getc_impl(VFSFile *stream) { gchar c; if (curl_vfs_fread_impl(&c, 1, 1, stream) != 1) return -1; return c; } gint curl_vfs_ungetc_impl(gint c, VFSFile *stream) { g_print("Tried ungetc\n"); return -1; } gint curl_vfs_fseek_impl(VFSFile * file, glong offset, gint whence) { size_t posn; CurlHandle *handle = file->handle; if (DEBUG_SEEK) g_print("Seek %p to %d %d\n", handle, offset, whence); if (whence == SEEK_END && handle->length == -1) { if (!handle->thread) { // We need a HEAD to find out the length handle->no_data = 1; if (DEBUG_CONNECTION) g_print("Request for head info\n"); curl_manage_request(handle); if (DEBUG_CONNECTION) g_print("Completed\n"); handle->no_data = 0; } else { // Wait a bit? } } if (whence == SEEK_END && handle->length < 0) { if (DEBUG_SEEK) g_print("Tried to seek to the end of a file with unknown length\n"); // don't know how long it is... return -1; } posn = handle->rd_abs; if (whence == SEEK_SET) handle->rd_abs = offset; else if (whence == SEEK_END) handle->rd_abs = handle->length + offset; else handle->rd_abs = handle->rd_abs + offset; // XXXX // There's a race here between finding available space and // allocating it and the check below. if (handle->thread) { if (handle->rd_abs + handle->buffer_length < handle->wr_abs || handle->rd_abs > handle->wr_abs) { if (DEBUG_CONNECTION) g_print("Stop transfer\n"); curl_req_no_xfer(handle); if (DEBUG_CONNECTION) g_print("Okay\n"); } else { if (DEBUG_CONNECTION) g_print("Continue transfer\n"); curl_req_sync_xfer(handle, posn); } } if (DEBUG_SEEK) g_print("Seeked %p from %d to %d\n", handle, posn, handle->rd_abs); return 0; } void curl_vfs_rewind_impl(VFSFile * file) { curl_vfs_fseek_impl(file, 0, SEEK_SET); } glong curl_vfs_ftell_impl(VFSFile * file) { CurlHandle *handle = file->handle; return handle->rd_abs; } gboolean curl_vfs_feof_impl(VFSFile * file) { CurlHandle *handle = file->handle; return handle->rd_abs == handle->length; } gint curl_vfs_truncate_impl(VFSFile * file, glong size) { return -1; } gchar * curl_vfs_metadata_impl(VFSFile * file, const gchar * field) { CurlHandle *handle = file->handle; if (!strcmp(field, "stream-name")) return strdup(handle->name); if (!strcmp(field, "track-name")) return strdup(handle->title); return NULL; } VFSConstructor curl_const = { "http://", curl_vfs_fopen_impl, curl_vfs_fclose_impl, curl_vfs_fread_impl, curl_vfs_fwrite_impl, curl_vfs_getc_impl, curl_vfs_ungetc_impl, curl_vfs_fseek_impl, curl_vfs_rewind_impl, curl_vfs_ftell_impl, curl_vfs_feof_impl, curl_vfs_truncate_impl, curl_vfs_metadata_impl }; static void init(void) { vfs_register_transport(&curl_const); } static void cleanup(void) { #if 0 vfs_unregister_transport(&curl_const); #endif } LowlevelPlugin llp_curl = { NULL, NULL, "http:// URI Transport", init, cleanup, }; LowlevelPlugin *get_lplugin_info(void) { return &llp_curl; }