mirror of
https://github.com/libretro/RetroArch.git
synced 2024-11-27 18:20:27 +00:00
474 lines
12 KiB
C
474 lines
12 KiB
C
/* 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 <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <net/net_http.h>
|
|
#include <string/stdstring.h>
|
|
#include <compat/strl.h>
|
|
#include <file/file_path.h>
|
|
#include <net/net_compat.h>
|
|
#include <retro_timers.h>
|
|
|
|
#ifdef RARCH_INTERNAL
|
|
#include "../gfx/video_display_server.h"
|
|
#endif
|
|
#include "task_file_transfer.h"
|
|
#include "tasks_internal.h"
|
|
|
|
enum http_status_enum
|
|
{
|
|
HTTP_STATUS_CONNECTION_TRANSFER = 0,
|
|
HTTP_STATUS_CONNECTION_TRANSFER_PARSE,
|
|
HTTP_STATUS_TRANSFER,
|
|
HTTP_STATUS_TRANSFER_PARSE,
|
|
HTTP_STATUS_TRANSFER_PARSE_FREE
|
|
};
|
|
|
|
struct http_transfer_info
|
|
{
|
|
int progress;
|
|
char url[255];
|
|
};
|
|
|
|
struct http_handle
|
|
{
|
|
struct http_t *handle;
|
|
transfer_cb_t cb;
|
|
struct
|
|
{
|
|
struct http_connection_t *handle;
|
|
transfer_cb_t cb;
|
|
} connection;
|
|
unsigned status;
|
|
bool error;
|
|
char connection_elem[255];
|
|
char connection_url[255];
|
|
};
|
|
|
|
typedef struct http_transfer_info http_transfer_info_t;
|
|
typedef struct http_handle http_handle_t;
|
|
|
|
static int task_http_con_iterate_transfer(http_handle_t *http)
|
|
{
|
|
if (!net_http_connection_iterate(http->connection.handle))
|
|
return -1;
|
|
return 0;
|
|
}
|
|
|
|
static int task_http_conn_iterate_transfer_parse(
|
|
http_handle_t *http)
|
|
{
|
|
if (net_http_connection_done(http->connection.handle))
|
|
{
|
|
if (http->connection.handle && http->connection.cb)
|
|
http->connection.cb(http, 0);
|
|
}
|
|
|
|
net_http_connection_free(http->connection.handle);
|
|
|
|
http->connection.handle = NULL;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int cb_http_conn_default(void *data_, size_t len)
|
|
{
|
|
http_handle_t *http = (http_handle_t*)data_;
|
|
|
|
if (!http)
|
|
return -1;
|
|
|
|
if (!network_init())
|
|
return -1;
|
|
|
|
http->handle = net_http_new(http->connection.handle);
|
|
|
|
if (!http->handle)
|
|
{
|
|
http->error = true;
|
|
return -1;
|
|
}
|
|
|
|
http->cb = NULL;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* task_http_iterate_transfer:
|
|
*
|
|
* Resumes HTTP transfer update.
|
|
*
|
|
* Returns: 0 when finished, -1 when we should continue
|
|
* with the transfer on the next frame.
|
|
**/
|
|
static int task_http_iterate_transfer(retro_task_t *task)
|
|
{
|
|
http_handle_t *http = (http_handle_t*)task->state;
|
|
size_t pos = 0, tot = 0;
|
|
|
|
/* FIXME: This wouldn't be needed if we could wait for a timeout */
|
|
if (task_queue_is_threaded())
|
|
retro_sleep(1);
|
|
|
|
if (!net_http_update(http->handle, &pos, &tot))
|
|
{
|
|
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));
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void task_http_transfer_handler(retro_task_t *task)
|
|
{
|
|
http_transfer_data_t *data = NULL;
|
|
http_handle_t *http = (http_handle_t*)task->state;
|
|
|
|
if (task_get_cancelled(task))
|
|
goto task_finished;
|
|
|
|
switch (http->status)
|
|
{
|
|
case HTTP_STATUS_CONNECTION_TRANSFER_PARSE:
|
|
task_http_conn_iterate_transfer_parse(http);
|
|
http->status = HTTP_STATUS_TRANSFER;
|
|
break;
|
|
case HTTP_STATUS_CONNECTION_TRANSFER:
|
|
if (!task_http_con_iterate_transfer(http))
|
|
http->status = HTTP_STATUS_CONNECTION_TRANSFER_PARSE;
|
|
break;
|
|
case HTTP_STATUS_TRANSFER:
|
|
if (!task_http_iterate_transfer(task))
|
|
goto task_finished;
|
|
break;
|
|
case HTTP_STATUS_TRANSFER_PARSE:
|
|
goto task_finished;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (http->error)
|
|
goto task_finished;
|
|
|
|
return;
|
|
task_finished:
|
|
task_set_finished(task, true);
|
|
|
|
if (http->handle)
|
|
{
|
|
size_t len = 0;
|
|
char *tmp = (char*)net_http_data(http->handle, &len, false);
|
|
|
|
if (tmp && http->cb)
|
|
http->cb(tmp, len);
|
|
|
|
if (net_http_error(http->handle) || task_get_cancelled(task))
|
|
{
|
|
tmp = (char*)net_http_data(http->handle, &len, true);
|
|
|
|
if (tmp)
|
|
free(tmp);
|
|
|
|
if (task_get_cancelled(task))
|
|
{
|
|
task_set_error(task, strdup("Task cancelled."));
|
|
}
|
|
else
|
|
{
|
|
data = (http_transfer_data_t*)malloc(sizeof(*data));
|
|
data->data = NULL;
|
|
data->len = 0;
|
|
data->status = net_http_status(http->handle);
|
|
|
|
task_set_data(task, data);
|
|
|
|
if (!task->mute)
|
|
task_set_error(task, strdup("Download failed."));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
data = (http_transfer_data_t*)malloc(sizeof(*data));
|
|
data->data = tmp;
|
|
data->len = len;
|
|
data->status = net_http_status(http->handle);
|
|
|
|
task_set_data(task, data);
|
|
}
|
|
|
|
net_http_delete(http->handle);
|
|
} else if (http->error)
|
|
task_set_error(task, strdup("Internal error."));
|
|
|
|
free(http);
|
|
}
|
|
|
|
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)
|
|
{
|
|
if (data->data)
|
|
free(data->data);
|
|
free(data);
|
|
}
|
|
}
|
|
|
|
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))
|
|
return false;
|
|
|
|
if (!user_data)
|
|
return false;
|
|
|
|
http = (http_handle_t*)task->state;
|
|
if (!http)
|
|
return false;
|
|
|
|
return string_is_equal(http->connection_url, (const char*)user_data);
|
|
}
|
|
|
|
static bool task_http_retriever(retro_task_t *task, void *data)
|
|
{
|
|
http_transfer_info_t *info = (http_transfer_info_t*)data;
|
|
|
|
/* Extract HTTP handle and return already if invalid */
|
|
http_handle_t *http = (http_handle_t *)task->state;
|
|
if (!http)
|
|
return false;
|
|
|
|
/* Fill HTTP info link */
|
|
strlcpy(info->url, http->connection_url, sizeof(info->url));
|
|
info->progress = task_get_progress(task);
|
|
return true;
|
|
}
|
|
|
|
static void http_transfer_progress_cb(retro_task_t *task)
|
|
{
|
|
#ifdef RARCH_INTERNAL
|
|
if (task)
|
|
video_display_server_set_window_progress(task->progress, task->finished);
|
|
#endif
|
|
}
|
|
|
|
static void* task_push_http_transfer_generic(
|
|
struct http_connection_t *conn,
|
|
const char *url, bool mute, const char *type,
|
|
retro_task_callback_t cb, void *user_data)
|
|
{
|
|
retro_task_t *t = NULL;
|
|
http_handle_t *http = NULL;
|
|
const char *method = NULL;
|
|
|
|
if (!conn)
|
|
return NULL;
|
|
|
|
method = net_http_connection_method(conn);
|
|
if (method && (method[0] == 'P' || method[0] == 'p'))
|
|
{
|
|
/* 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))
|
|
{
|
|
net_http_connection_free(conn);
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
http = (http_handle_t*)malloc(sizeof(*http));
|
|
|
|
if (!http)
|
|
goto error;
|
|
|
|
http->connection.handle = conn;
|
|
http->connection.cb = &cb_http_conn_default;
|
|
http->connection_elem[0] = '\0';
|
|
http->connection_url[0] = '\0';
|
|
http->handle = NULL;
|
|
http->cb = NULL;
|
|
http->status = 0;
|
|
http->error = false;
|
|
|
|
if (type)
|
|
strlcpy(http->connection_elem, type, sizeof(http->connection_elem));
|
|
|
|
strlcpy(http->connection_url, url, sizeof(http->connection_url));
|
|
|
|
http->status = HTTP_STATUS_CONNECTION_TRANSFER;
|
|
t = task_init();
|
|
|
|
if (!t)
|
|
goto error;
|
|
|
|
t->handler = task_http_transfer_handler;
|
|
t->state = http;
|
|
t->mute = mute;
|
|
t->callback = cb;
|
|
t->progress_cb = http_transfer_progress_cb;
|
|
t->cleanup = task_http_transfer_cleanup;
|
|
t->user_data = user_data;
|
|
t->progress = -1;
|
|
|
|
task_queue_push(t);
|
|
|
|
return t;
|
|
|
|
error:
|
|
if (conn)
|
|
net_http_connection_free(conn);
|
|
if (http)
|
|
free(http);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
void* task_push_http_transfer(const char *url, bool mute,
|
|
const char *type,
|
|
retro_task_callback_t cb, void *user_data)
|
|
{
|
|
if (string_is_empty(url))
|
|
return NULL;
|
|
|
|
return task_push_http_transfer_generic(
|
|
net_http_connection_new(url, "GET", NULL),
|
|
url, mute, type, 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)
|
|
{
|
|
const char *s = NULL;
|
|
char tmp[255] = "";
|
|
retro_task_t *t = NULL;
|
|
|
|
if (string_is_empty(url))
|
|
return NULL;
|
|
|
|
t = (retro_task_t*)task_push_http_transfer_generic(
|
|
net_http_connection_new(url, "GET", NULL),
|
|
url, mute, type, cb, transfer_data);
|
|
|
|
if (!t)
|
|
return NULL;
|
|
|
|
if (transfer_data)
|
|
s = transfer_data->path;
|
|
else
|
|
s = url;
|
|
|
|
strlcpy(tmp, msg_hash_to_str(MSG_DOWNLOADING), sizeof(tmp));
|
|
strlcat(tmp, " ", sizeof(tmp));
|
|
|
|
if (string_ends_with_size(s, ".index",
|
|
strlen(s), STRLEN_CONST(".index")))
|
|
strlcat(tmp, msg_hash_to_str(MSG_INDEX_FILE), sizeof(tmp));
|
|
else
|
|
strlcat(tmp, s, sizeof(tmp));
|
|
|
|
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)
|
|
{
|
|
struct http_connection_t* conn;
|
|
|
|
if (string_is_empty(url))
|
|
return NULL;
|
|
|
|
conn = net_http_connection_new(url, "GET", NULL);
|
|
if (!conn)
|
|
return NULL;
|
|
|
|
if (user_agent != NULL)
|
|
net_http_connection_set_user_agent(conn, user_agent);
|
|
|
|
/* assert: task_push_http_transfer_generic will free conn on failure */
|
|
return task_push_http_transfer_generic(conn, url, mute, type, 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)
|
|
{
|
|
if (string_is_empty(url))
|
|
return NULL;
|
|
return task_push_http_transfer_generic(
|
|
net_http_connection_new(url, "POST", post_data),
|
|
url, mute, type, 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)
|
|
{
|
|
struct http_connection_t* conn;
|
|
|
|
if (string_is_empty(url))
|
|
return NULL;
|
|
|
|
conn = net_http_connection_new(url, "POST", post_data);
|
|
if (!conn)
|
|
return NULL;
|
|
|
|
if (user_agent != NULL)
|
|
net_http_connection_set_user_agent(conn, user_agent);
|
|
|
|
/* assert: task_push_http_transfer_generic will free conn on failure */
|
|
return task_push_http_transfer_generic(conn, url, mute, type, cb, user_data);
|
|
}
|
|
|
|
task_retriever_info_t *http_task_get_transfer_list(void)
|
|
{
|
|
task_retriever_data_t retrieve_data;
|
|
|
|
/* Fill retrieve data */
|
|
retrieve_data.handler = task_http_transfer_handler;
|
|
retrieve_data.element_size = sizeof(http_transfer_info_t);
|
|
retrieve_data.func = task_http_retriever;
|
|
|
|
/* Build list of current HTTP transfers and return it */
|
|
task_queue_retrieve(&retrieve_data);
|
|
|
|
return retrieve_data.list;
|
|
}
|