/* RetroArch - A frontend for libretro. * Copyright (C) 2011-2017 - Daniel De Matteis * * RetroArch 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 Found- * ation, either version 3 of the License, or (at your option) any later version. * * RetroArch 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 RetroArch. * If not, see . */ #ifndef EMSCRIPTEN #error "task_http_emscripten only makes sense in emscripten builds" #endif #include #include "verbosity.h" #include #include #include #include #include #include #ifdef RARCH_INTERNAL #include "../gfx/video_display_server.h" #endif #include "task_file_transfer.h" #include "tasks_internal.h" struct http_handle { int handle; char connection_url[NAME_MAX_LENGTH]; http_transfer_data_t *response; }; typedef struct http_handle http_handle_t; static void task_http_transfer_handler(retro_task_t *task) { http_handle_t *http = (http_handle_t*)task->state; uint8_t flg = task_get_flags(task); if ((flg & RETRO_TASK_FLG_CANCELLED) > 0) goto task_finished; if (http->response || (http->handle == -1)) goto task_finished; return; task_finished: task_set_flags(task, RETRO_TASK_FLG_FINISHED, true); if (http->response) { if ((flg & RETRO_TASK_FLG_CANCELLED) > 0) { string_list_free(http->response->headers); free(http->response->data); free(http->response); http->response = NULL; task_set_error(task, strldup("Task cancelled.", sizeof("Task cancelled."))); } else { bool mute; task_set_data(task, http->response); mute = ((task->flags & RETRO_TASK_FLG_MUTE) > 0); if (!mute && http->response->status >= 400) task_set_error(task, strldup("Download failed.", sizeof("Download failed."))); } } free(http); } static void http_transfer_progress_cb(retro_task_t *task) { if (task) video_display_server_set_window_progress(task->progress, ((task->flags & RETRO_TASK_FLG_FINISHED) > 0)); } static bool task_http_finder(retro_task_t *task, void *user_data) { http_handle_t *http = NULL; if (task && (task->handler == task_http_transfer_handler) && user_data) if ((http = (http_handle_t*)task->state)) return string_is_equal(http->connection_url, (const char*)user_data); return false; } static void task_http_transfer_cleanup(retro_task_t *task) { http_transfer_data_t* data = (http_transfer_data_t*)task_get_data(task); if (data) { string_list_free(data->headers); if (data->data) free(data->data); free(data); } } void wget_onload_cb(unsigned handle, void *t_ptr, void *data, unsigned len) { retro_task_t *task = (retro_task_t *)t_ptr; http_handle_t *http = (http_handle_t*)task->state; http_transfer_data_t *resp; if (!(resp = (http_transfer_data_t*)malloc(sizeof(*resp)))) { http->handle = -1; return; } else { resp->data = data; resp->len = len; resp->status = 200; resp->headers = NULL; // sorry webdav http->response = resp; } } void wget_onerror_cb(unsigned handle, void *t_ptr, int status, const char *err) { retro_task_t *task = (retro_task_t *)t_ptr; http_handle_t *http = (http_handle_t*)task->state; bool mute = ((task->flags & RETRO_TASK_FLG_MUTE) > 0); if (!mute) task_set_error(task, strldup("Download failed.", sizeof("Download failed."))); http->handle = -1; } void wget_onprogress_cb(unsigned handle, void *t_ptr, int pos, int tot) { retro_task_t *task = (retro_task_t *)t_ptr; if (tot == 0) task_set_progress(task, -1); else if (pos < (((size_t)-1) / 100)) /* prefer multiply then divide for more accurate results */ task_set_progress(task, (signed)(pos * 100 / tot)); else /* but invert the logic if it would cause an overflow */ task_set_progress(task, MIN((signed)pos / (tot / 100), 100)); } static void *task_push_http_transfer_generic( const char *url, const char *method, const char *data, const char *user_agent, const char *headers, bool mute, retro_task_callback_t cb, void *user_data) { retro_task_t *t = NULL; http_handle_t *http = NULL; int wget_handle = -1; if (!url) return NULL; if (!string_is_equal(method, "GET")) { /* POST requests usually mutate the server, so assume multiple calls are * intended, even if they're duplicated. Additionally, they may differ * only by the POST data, and task_http_finder doesn't look at that, so * unique requests could be misclassified as duplicates. */ } else { task_finder_data_t find_data; find_data.func = task_http_finder; find_data.userdata = (void*)url; /* Concurrent download of the same file is not allowed */ if (task_queue_find(&find_data)) return NULL; } if (!(http = (http_handle_t*)malloc(sizeof(*http)))) goto error; http->handle = -1; http->response = NULL; http->connection_url[0] = '\0'; strlcpy(http->connection_url, url, sizeof(http->connection_url)); if (!(t = task_init())) goto error; t->handler = task_http_transfer_handler; t->state = http; t->callback = cb; t->progress_cb = http_transfer_progress_cb; t->cleanup = task_http_transfer_cleanup; t->user_data = user_data; t->progress = -1; if (mute) t->flags |= RETRO_TASK_FLG_MUTE; else t->flags &= ~RETRO_TASK_FLG_MUTE; wget_handle = emscripten_async_wget2_data(url, method, data, t, false, wget_onload_cb, wget_onerror_cb, wget_onprogress_cb); http->handle = wget_handle; task_queue_push(t); return t; error: if (http) free(http); if (t) free(t); return NULL; } void* task_push_http_transfer(const char *url, bool mute, const char *type, retro_task_callback_t cb, void *user_data) { return task_push_http_transfer_generic(url, type ? type : "GET", NULL, NULL, NULL, mute, cb, user_data); } void *task_push_webdav_stat(const char *url, bool mute, const char *headers, retro_task_callback_t cb, void *user_data) { RARCH_ERR("[http] response headers not supported, webdav won't work\n"); return task_push_http_transfer_generic(url, "OPTIONS", NULL, NULL, headers, mute, cb, user_data); } void* task_push_webdav_mkdir(const char *url, bool mute, const char *headers, retro_task_callback_t cb, void *user_data) { RARCH_ERR("[http] response headers not supported, webdav won't work\n"); return task_push_http_transfer_generic(url, "MKCOL", NULL, NULL, headers, mute, cb, user_data); } void* task_push_webdav_put(const char *url, const void *put_data, size_t len, bool mute, const char *headers, retro_task_callback_t cb, void *user_data) { char expect[1024]; /* TODO/FIXME - check size */ size_t _len; RARCH_ERR("[http] response headers not supported, webdav won't work\n"); _len = strlcpy(expect, "Expect: 100-continue\r\n", sizeof(expect)); if (headers) { strlcpy(expect + _len, headers, sizeof(expect) - _len); } return task_push_http_transfer_generic(url, "PUT", put_data, NULL, expect, mute, cb, user_data); } void* task_push_webdav_delete(const char *url, bool mute, const char *headers, retro_task_callback_t cb, void *user_data) { RARCH_ERR("[http] response headers not supported, webdav won't work\n"); return task_push_http_transfer_generic(url, "DELETE", NULL, NULL, headers, mute, cb, user_data); } void *task_push_webdav_move(const char *url, const char *dest, bool mute, const char *headers, retro_task_callback_t cb, void *user_data) { size_t _len; char dest_header[PATH_MAX_LENGTH + 512]; RARCH_ERR("[http] response headers not supported, webdav won't work\n"); _len = strlcpy(dest_header, "Destination: ", sizeof(dest_header)); _len += strlcpy(dest_header + _len, dest, sizeof(dest_header) - _len); _len += strlcpy(dest_header + _len, "\r\n", sizeof(dest_header) - _len); if (headers) strlcpy(dest_header + _len, headers, sizeof(dest_header) - _len); return task_push_http_transfer_generic(url, "MOVE", NULL, NULL, dest_header, mute, cb, user_data); } void* task_push_http_transfer_file(const char* url, bool mute, const char* type, retro_task_callback_t cb, file_transfer_t* transfer_data) { size_t _len; const char *s = NULL; char tmp[NAME_MAX_LENGTH] = ""; retro_task_t *t = NULL; if (string_is_empty(url)) return NULL; if (!(t = (retro_task_t*)task_push_http_transfer_generic( /* should be using type but some callers now rely on type being ignored */ url, "GET", NULL, NULL, NULL, mute, cb, transfer_data))) return NULL; if (transfer_data) s = transfer_data->path; else s = url; _len = strlcpy(tmp, msg_hash_to_str(MSG_DOWNLOADING), sizeof(tmp)); tmp[ _len] = ' '; tmp[++_len] = '\0'; if (string_ends_with_size(s, ".index", strlen(s), STRLEN_CONST(".index"))) s = msg_hash_to_str(MSG_INDEX_FILE); strlcpy(tmp + _len, s, sizeof(tmp) - _len); t->title = strdup(tmp); return t; } void* task_push_http_transfer_with_user_agent(const char *url, bool mute, const char *type, const char *user_agent, retro_task_callback_t cb, void *user_data) { return task_push_http_transfer_generic(url, type ? type : "GET", NULL, user_agent, NULL, mute, cb, user_data); } void* task_push_http_transfer_with_headers(const char *url, bool mute, const char *type, const char *headers, retro_task_callback_t cb, void *user_data) { return task_push_http_transfer_generic(url, type ? type : "GET", NULL, NULL, headers, mute, cb, user_data); } void* task_push_http_post_transfer(const char *url, const char *post_data, bool mute, const char *type, retro_task_callback_t cb, void *user_data) { return task_push_http_transfer_generic(url, type ? type : "POST", post_data, NULL, NULL, mute, cb, user_data); } void* task_push_http_post_transfer_with_user_agent(const char *url, const char *post_data, bool mute, const char *type, const char *user_agent, retro_task_callback_t cb, void *user_data) { return task_push_http_transfer_generic(url, type ? type : "POST", post_data, user_agent, NULL, mute, cb, user_data); } void* task_push_http_post_transfer_with_headers(const char *url, const char *post_data, bool mute, const char *type, const char *headers, retro_task_callback_t cb, void *user_data) { return task_push_http_transfer_generic(url, type ? type : "POST", post_data, NULL, headers, mute, cb, user_data); }