mirror of
https://github.com/libretro/RetroArch
synced 2025-03-02 19:13:34 +00:00
Remove unused http server
This commit is contained in:
parent
24a859fa16
commit
7ee547db0f
@ -1507,11 +1507,6 @@ XML
|
||||
/*============================================================
|
||||
HTTP SERVER
|
||||
============================================================ */
|
||||
#if defined(HAVE_HTTPSERVER) && defined(HAVE_ZLIB)
|
||||
#include "../deps/civetweb/civetweb.c"
|
||||
#include "network/httpserver/httpserver.c"
|
||||
#endif
|
||||
|
||||
#if defined(HAVE_DISCORD)
|
||||
#include "../discord/discord.c"
|
||||
#endif
|
||||
|
@ -1,735 +0,0 @@
|
||||
/* Copyright (c) 2016 the Civetweb developers
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
* of this software and associated documentation files (the "Software"), to deal
|
||||
* in the Software without restriction, including without limitation the rights
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
* copies of the Software, and to permit persons to whom the Software is
|
||||
* furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in
|
||||
* all copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||
* THE SOFTWARE.
|
||||
*/
|
||||
|
||||
static int
|
||||
url_encoded_field_found(const struct mg_connection *conn,
|
||||
const char *key,
|
||||
size_t key_len,
|
||||
const char *filename,
|
||||
size_t filename_len,
|
||||
char *path,
|
||||
size_t path_len,
|
||||
struct mg_form_data_handler *fdh)
|
||||
{
|
||||
char key_dec[1024];
|
||||
char filename_dec[1024];
|
||||
int key_dec_len;
|
||||
int filename_dec_len;
|
||||
int ret;
|
||||
|
||||
key_dec_len =
|
||||
mg_url_decode(key, (int)key_len, key_dec, (int)sizeof(key_dec), 1);
|
||||
|
||||
if (((size_t)key_dec_len >= (size_t)sizeof(key_dec)) || (key_dec_len < 0)) {
|
||||
return FORM_FIELD_STORAGE_SKIP;
|
||||
}
|
||||
|
||||
if (filename) {
|
||||
filename_dec_len = mg_url_decode(filename,
|
||||
(int)filename_len,
|
||||
filename_dec,
|
||||
(int)sizeof(filename_dec),
|
||||
1);
|
||||
|
||||
if (((size_t)filename_dec_len >= (size_t)sizeof(filename_dec))
|
||||
|| (filename_dec_len < 0)) {
|
||||
/* Log error message and skip this field. */
|
||||
mg_cry(conn, "%s: Cannot decode filename", __func__);
|
||||
return FORM_FIELD_STORAGE_SKIP;
|
||||
}
|
||||
}
|
||||
else {
|
||||
filename_dec[0] = 0;
|
||||
}
|
||||
|
||||
ret =
|
||||
fdh->field_found(key_dec, filename_dec, path, path_len, fdh->user_data);
|
||||
|
||||
if ((ret & 0xF) == FORM_FIELD_STORAGE_GET) {
|
||||
if (fdh->field_get == NULL) {
|
||||
mg_cry(conn, "%s: Function \"Get\" not available", __func__);
|
||||
return FORM_FIELD_STORAGE_SKIP;
|
||||
}
|
||||
}
|
||||
if ((ret & 0xF) == FORM_FIELD_STORAGE_STORE) {
|
||||
if (fdh->field_store == NULL) {
|
||||
mg_cry(conn, "%s: Function \"Store\" not available", __func__);
|
||||
return FORM_FIELD_STORAGE_SKIP;
|
||||
}
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int
|
||||
url_encoded_field_get(const struct mg_connection *conn,
|
||||
const char *key,
|
||||
size_t key_len,
|
||||
const char *value,
|
||||
size_t value_len,
|
||||
struct mg_form_data_handler *fdh)
|
||||
{
|
||||
char key_dec[1024];
|
||||
|
||||
char *value_dec = (char*)mg_malloc(value_len + 1);
|
||||
int value_dec_len;
|
||||
|
||||
if (!value_dec) {
|
||||
/* Log error message and stop parsing the form data. */
|
||||
mg_cry(conn,
|
||||
"%s: Not enough memory (required: %lu)",
|
||||
__func__,
|
||||
(unsigned long)(value_len + 1));
|
||||
return FORM_FIELD_STORAGE_ABORT;
|
||||
}
|
||||
|
||||
mg_url_decode(key, (int)key_len, key_dec, (int)sizeof(key_dec), 1);
|
||||
|
||||
value_dec_len =
|
||||
mg_url_decode(value, (int)value_len, value_dec, (int)value_len + 1, 1);
|
||||
|
||||
return fdh->field_get(key_dec,
|
||||
value_dec,
|
||||
(size_t)value_dec_len,
|
||||
fdh->user_data);
|
||||
}
|
||||
|
||||
static int
|
||||
field_stored(const struct mg_connection *conn,
|
||||
const char *path,
|
||||
size_t file_size,
|
||||
struct mg_form_data_handler *fdh)
|
||||
{
|
||||
/* Equivalent to "upload" callback of "mg_upload". */
|
||||
|
||||
(void)conn; /* we do not need mg_cry here, so conn is currently unused */
|
||||
|
||||
return fdh->field_store(path, file_size, fdh->user_data);
|
||||
}
|
||||
|
||||
static const char *
|
||||
search_boundary(const char *buf,
|
||||
size_t buf_len,
|
||||
const char *boundary,
|
||||
size_t boundary_len)
|
||||
{
|
||||
/* We must do a binary search here, not a string search, since the buffer
|
||||
* may contain '\x00' bytes, if binary data is transfered. */
|
||||
int clen = (int)buf_len - (int)boundary_len - 4;
|
||||
int i;
|
||||
|
||||
for (i = 0; i <= clen; i++) {
|
||||
if (!memcmp(buf + i, "\r\n--", 4)) {
|
||||
if (!memcmp(buf + i + 4, boundary, boundary_len)) {
|
||||
return buf + i;
|
||||
}
|
||||
}
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
int
|
||||
mg_handle_form_request(struct mg_connection *conn,
|
||||
struct mg_form_data_handler *fdh)
|
||||
{
|
||||
const char *content_type;
|
||||
char path[512];
|
||||
char buf[1024];
|
||||
int field_storage;
|
||||
int buf_fill = 0;
|
||||
int r;
|
||||
int field_count = 0;
|
||||
struct file fstore = STRUCT_FILE_INITIALIZER;
|
||||
size_t file_size = 0; /* init here, to a avoid a false positive
|
||||
"uninitialized variable used" warning */
|
||||
|
||||
int has_body_data =
|
||||
(conn->request_info.content_length > 0) || (conn->is_chunked);
|
||||
|
||||
/* There are three ways to encode data from a HTML form:
|
||||
* 1) method: GET (default)
|
||||
* The form data is in the HTTP query string.
|
||||
* 2) method: POST, enctype: "application/x-www-form-urlencoded"
|
||||
* The form data is in the request body.
|
||||
* The body is url encoded (the default encoding for POST).
|
||||
* 3) method: POST, enctype: "multipart/form-data".
|
||||
* The form data is in the request body of a multipart message.
|
||||
* This is the typical way to handle file upload from a form.
|
||||
*/
|
||||
|
||||
if (!has_body_data) {
|
||||
const char *data;
|
||||
|
||||
if (strcmp(conn->request_info.request_method, "GET")) {
|
||||
/* No body data, but not a GET request.
|
||||
* This is not a valid form request. */
|
||||
return -1;
|
||||
}
|
||||
|
||||
/* GET request: form data is in the query string. */
|
||||
/* The entire data has already been loaded, so there is no nead to
|
||||
* call mg_read. We just need to split the query string into key-value
|
||||
* pairs. */
|
||||
data = conn->request_info.query_string;
|
||||
if (!data) {
|
||||
/* No query string. */
|
||||
return -1;
|
||||
}
|
||||
|
||||
/* Split data in a=1&b=xy&c=3&c=4 ... */
|
||||
while (*data) {
|
||||
const char *val = strchr(data, '=');
|
||||
const char *next;
|
||||
ptrdiff_t keylen, vallen;
|
||||
|
||||
if (!val) {
|
||||
break;
|
||||
}
|
||||
keylen = val - data;
|
||||
|
||||
/* In every "field_found" callback we ask what to do with the
|
||||
* data ("field_storage"). This could be:
|
||||
* FORM_FIELD_STORAGE_SKIP (0) ... ignore the value of this field
|
||||
* FORM_FIELD_STORAGE_GET (1) ... read the data and call the get
|
||||
* callback function
|
||||
* FORM_FIELD_STORAGE_STORE (2) ... store the data in a file
|
||||
* FORM_FIELD_STORAGE_READ (3) ... let the user read the data
|
||||
* (for parsing long data on the fly)
|
||||
* (currently not implemented)
|
||||
* FORM_FIELD_STORAGE_ABORT (flag) ... stop parsing
|
||||
*/
|
||||
memset(path, 0, sizeof(path));
|
||||
field_count++;
|
||||
field_storage = url_encoded_field_found(conn,
|
||||
data,
|
||||
(size_t)keylen,
|
||||
NULL,
|
||||
0,
|
||||
path,
|
||||
sizeof(path) - 1,
|
||||
fdh);
|
||||
|
||||
val++;
|
||||
next = strchr(val, '&');
|
||||
if (next) {
|
||||
vallen = next - val;
|
||||
next++;
|
||||
}
|
||||
else {
|
||||
vallen = (ptrdiff_t)strlen(val);
|
||||
next = val + vallen;
|
||||
}
|
||||
|
||||
if (field_storage == FORM_FIELD_STORAGE_GET) {
|
||||
/* Call callback */
|
||||
url_encoded_field_get(
|
||||
conn, data, (size_t)keylen, val, (size_t)vallen, fdh);
|
||||
}
|
||||
if (field_storage == FORM_FIELD_STORAGE_STORE) {
|
||||
/* Store the content to a file */
|
||||
if (mg_fopen(conn, path, "wb", &fstore) == 0) {
|
||||
fstore.fp = NULL;
|
||||
}
|
||||
file_size = 0;
|
||||
if (fstore.fp != NULL) {
|
||||
size_t n =
|
||||
(size_t)fwrite(val, 1, (size_t)vallen, fstore.fp);
|
||||
if ((n != (size_t)vallen) || (ferror(fstore.fp))) {
|
||||
mg_cry(conn,
|
||||
"%s: Cannot write file %s",
|
||||
__func__,
|
||||
path);
|
||||
fclose(fstore.fp);
|
||||
fstore.fp = NULL;
|
||||
remove_bad_file(conn, path);
|
||||
}
|
||||
file_size += (size_t)n;
|
||||
|
||||
if (fstore.fp) {
|
||||
r = fclose(fstore.fp);
|
||||
if (r == 0) {
|
||||
/* stored successfully */
|
||||
field_stored(conn, path, file_size, fdh);
|
||||
}
|
||||
else {
|
||||
mg_cry(conn,
|
||||
"%s: Error saving file %s",
|
||||
__func__,
|
||||
path);
|
||||
remove_bad_file(conn, path);
|
||||
}
|
||||
fstore.fp = NULL;
|
||||
}
|
||||
|
||||
}
|
||||
else {
|
||||
mg_cry(conn, "%s: Cannot create file %s", __func__, path);
|
||||
}
|
||||
}
|
||||
|
||||
/* if (field_storage == FORM_FIELD_STORAGE_READ) { */
|
||||
/* The idea of "field_storage=read" is to let the API user read
|
||||
* data chunk by chunk and to some data processing on the fly.
|
||||
* This should avoid the need to store data in the server:
|
||||
* It should neither be stored in memory, like
|
||||
* "field_storage=get" does, nor in a file like
|
||||
* "field_storage=store".
|
||||
* However, for a "GET" request this does not make any much
|
||||
* sense, since the data is already stored in memory, as it is
|
||||
* part of the query string.
|
||||
*/
|
||||
/* } */
|
||||
|
||||
if ((field_storage & FORM_FIELD_STORAGE_ABORT)
|
||||
== FORM_FIELD_STORAGE_ABORT) {
|
||||
/* Stop parsing the request */
|
||||
break;
|
||||
}
|
||||
|
||||
/* Proceed to next entry */
|
||||
data = next;
|
||||
}
|
||||
|
||||
return field_count;
|
||||
}
|
||||
|
||||
content_type = mg_get_header(conn, "Content-Type");
|
||||
|
||||
if (!content_type
|
||||
|| !mg_strcasecmp(content_type, "APPLICATION/X-WWW-FORM-URLENCODED")) {
|
||||
/* The form data is in the request body data, encoded in key/value
|
||||
* pairs. */
|
||||
int all_data_read = 0;
|
||||
|
||||
/* Read body data and split it in a=1&b&c=3&c=4 ... */
|
||||
/* The encoding is like in the "GET" case above, but here we read data
|
||||
* on the fly */
|
||||
for (;;) {
|
||||
/* TODO(high): Handle (text) fields with data size > sizeof(buf). */
|
||||
const char *val;
|
||||
const char *next;
|
||||
ptrdiff_t keylen, vallen;
|
||||
ptrdiff_t used;
|
||||
int end_of_key_value_pair_found = 0;
|
||||
|
||||
if ((size_t)buf_fill < (sizeof(buf) - 1)) {
|
||||
|
||||
size_t to_read = sizeof(buf) - 1 - (size_t)buf_fill;
|
||||
r = mg_read(conn, buf + (size_t)buf_fill, to_read);
|
||||
if (r < 0) {
|
||||
/* read error */
|
||||
return -1;
|
||||
}
|
||||
if (r != (int)to_read) {
|
||||
/* TODO: Create a function to get "all_data_read" from
|
||||
* the conn object. Add data is read if the Content-Length
|
||||
* has been reached, or if chunked encoding is used and
|
||||
* the end marker has been read, or if the connection has
|
||||
* been closed. */
|
||||
all_data_read = 1;
|
||||
}
|
||||
buf_fill += r;
|
||||
buf[buf_fill] = 0;
|
||||
if (buf_fill < 1) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
val = strchr(buf, '=');
|
||||
|
||||
if (!val) {
|
||||
break;
|
||||
}
|
||||
keylen = val - buf;
|
||||
val++;
|
||||
|
||||
/* Call callback */
|
||||
memset(path, 0, sizeof(path));
|
||||
field_count++;
|
||||
field_storage = url_encoded_field_found(conn,
|
||||
buf,
|
||||
(size_t)keylen,
|
||||
NULL,
|
||||
0,
|
||||
path,
|
||||
sizeof(path) - 1,
|
||||
fdh);
|
||||
|
||||
if ((field_storage & FORM_FIELD_STORAGE_ABORT)
|
||||
== FORM_FIELD_STORAGE_ABORT) {
|
||||
/* Stop parsing the request */
|
||||
break;
|
||||
}
|
||||
|
||||
if (field_storage == FORM_FIELD_STORAGE_STORE) {
|
||||
if (mg_fopen(conn, path, "wb", &fstore) == 0) {
|
||||
fstore.fp = NULL;
|
||||
}
|
||||
file_size = 0;
|
||||
if (!fstore.fp) {
|
||||
mg_cry(conn, "%s: Cannot create file %s", __func__, path);
|
||||
}
|
||||
}
|
||||
|
||||
/* Loop to read values larger than sizeof(buf)-keylen-2 */
|
||||
do {
|
||||
next = strchr(val, '&');
|
||||
if (next) {
|
||||
vallen = next - val;
|
||||
next++;
|
||||
end_of_key_value_pair_found = 1;
|
||||
}
|
||||
else {
|
||||
vallen = (ptrdiff_t)strlen(val);
|
||||
next = val + vallen;
|
||||
}
|
||||
|
||||
if (fstore.fp) {
|
||||
size_t n =
|
||||
(size_t)fwrite(val, 1, (size_t)vallen, fstore.fp);
|
||||
if ((n != (size_t)vallen) || (ferror(fstore.fp))) {
|
||||
mg_cry(conn,
|
||||
"%s: Cannot write file %s",
|
||||
__func__,
|
||||
path);
|
||||
fclose(fstore.fp);
|
||||
fstore.fp = NULL;
|
||||
remove_bad_file(conn, path);
|
||||
}
|
||||
file_size += (size_t)n;
|
||||
}
|
||||
if (field_storage == FORM_FIELD_STORAGE_GET) {
|
||||
if (!end_of_key_value_pair_found && !all_data_read) {
|
||||
/* TODO: check for an easy way to get longer data */
|
||||
mg_cry(conn,
|
||||
"%s: Data too long for callback",
|
||||
__func__);
|
||||
return -1;
|
||||
}
|
||||
/* Call callback */
|
||||
url_encoded_field_get(
|
||||
conn, buf, (size_t)keylen, val, (size_t)vallen, fdh);
|
||||
}
|
||||
|
||||
if (!end_of_key_value_pair_found) {
|
||||
/* TODO: read more data */
|
||||
break;
|
||||
}
|
||||
|
||||
} while (!end_of_key_value_pair_found);
|
||||
|
||||
if (fstore.fp) {
|
||||
r = fclose(fstore.fp);
|
||||
if (r == 0) {
|
||||
/* stored successfully */
|
||||
field_stored(conn, path, file_size, fdh);
|
||||
}
|
||||
else {
|
||||
mg_cry(conn, "%s: Error saving file %s", __func__, path);
|
||||
remove_bad_file(conn, path);
|
||||
}
|
||||
fstore.fp = NULL;
|
||||
}
|
||||
|
||||
/* Proceed to next entry */
|
||||
used = next - buf;
|
||||
memmove(buf, buf + (size_t)used, sizeof(buf) - (size_t)used);
|
||||
buf_fill -= (int)used;
|
||||
}
|
||||
|
||||
return field_count;
|
||||
}
|
||||
|
||||
if (!mg_strncasecmp(content_type, "MULTIPART/FORM-DATA;", 20)) {
|
||||
/* The form data is in the request body data, encoded as multipart
|
||||
* content (see https://www.ietf.org/rfc/rfc1867.txt,
|
||||
* https://www.ietf.org/rfc/rfc2388.txt). */
|
||||
const char *boundary;
|
||||
size_t bl;
|
||||
ptrdiff_t used;
|
||||
struct mg_request_info part_header;
|
||||
char *hbuf, *hend, *fbeg, *fend, *nbeg, *nend;
|
||||
const char *content_disp;
|
||||
const char *next;
|
||||
|
||||
memset(&part_header, 0, sizeof(part_header));
|
||||
|
||||
/* There has to be a BOUNDARY definition in the Content-Type header */
|
||||
if (mg_strncasecmp(content_type + 21, "BOUNDARY=", 9)) {
|
||||
/* Malformed request */
|
||||
return -1;
|
||||
}
|
||||
|
||||
boundary = content_type + 30;
|
||||
bl = strlen(boundary);
|
||||
|
||||
if (bl + 800 > sizeof(buf)) {
|
||||
/* Sanity check: The algorithm can not work if bl >= sizeof(buf),
|
||||
* and it will not work effectively, if the buf is only a few byte
|
||||
* larger than bl, or it buf can not hold the multipart header
|
||||
* plus the boundary.
|
||||
* Check some reasonable number here, that should be fulfilled by
|
||||
* any reasonable request from every browser. If it is not
|
||||
* fulfilled, it might be a hand-made request, intended to
|
||||
* interfere with the algorithm. */
|
||||
return -1;
|
||||
}
|
||||
|
||||
for (;;) {
|
||||
|
||||
r = mg_read(conn,
|
||||
buf + (size_t)buf_fill,
|
||||
sizeof(buf) - 1 - (size_t)buf_fill);
|
||||
if (r < 0) {
|
||||
/* read error */
|
||||
return -1;
|
||||
}
|
||||
buf_fill += r;
|
||||
buf[buf_fill] = 0;
|
||||
if (buf_fill < 1) {
|
||||
/* No data */
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (buf[0] != '-' || buf[1] != '-') {
|
||||
/* Malformed request */
|
||||
return -1;
|
||||
}
|
||||
if (strncmp(buf + 2, boundary, bl)) {
|
||||
/* Malformed request */
|
||||
return -1;
|
||||
}
|
||||
if (buf[bl + 2] != '\r' || buf[bl + 3] != '\n') {
|
||||
/* Every part must end with \r\n, if there is another part.
|
||||
* The end of the request has an extra -- */
|
||||
if (((size_t)buf_fill != (size_t)(bl + 6))
|
||||
|| (strncmp(buf + bl + 2, "--\r\n", 4))) {
|
||||
/* Malformed request */
|
||||
return -1;
|
||||
}
|
||||
/* End of the request */
|
||||
break;
|
||||
}
|
||||
|
||||
/* Next, we need to get the part header: Read until \r\n\r\n */
|
||||
hbuf = buf + bl + 4;
|
||||
hend = strstr(hbuf, "\r\n\r\n");
|
||||
if (!hend) {
|
||||
/* Malformed request */
|
||||
return -1;
|
||||
}
|
||||
|
||||
parse_http_headers(&hbuf, &part_header);
|
||||
if ((hend + 2) != hbuf) {
|
||||
/* Malformed request */
|
||||
return -1;
|
||||
}
|
||||
|
||||
/* Skip \r\n\r\n */
|
||||
hend += 4;
|
||||
|
||||
/* According to the RFC, every part has to have a header field like:
|
||||
* Content-Disposition: form-data; name="..." */
|
||||
content_disp = get_header(&part_header, "Content-Disposition");
|
||||
if (!content_disp) {
|
||||
/* Malformed request */
|
||||
return -1;
|
||||
}
|
||||
|
||||
/* Get the mandatory name="..." part of the Content-Disposition
|
||||
* header. */
|
||||
nbeg = (char*)strstr(content_disp, "name=\"");
|
||||
if (!nbeg) {
|
||||
/* Malformed request */
|
||||
return -1;
|
||||
}
|
||||
nbeg += 6;
|
||||
nend = strchr(nbeg, '\"');
|
||||
if (!nend) {
|
||||
/* Malformed request */
|
||||
return -1;
|
||||
}
|
||||
|
||||
/* Get the optional filename="..." part of the Content-Disposition
|
||||
* header. */
|
||||
fbeg = (char*)strstr(content_disp, "filename=\"");
|
||||
if (fbeg) {
|
||||
fbeg += 10;
|
||||
fend = strchr(fbeg, '\"');
|
||||
if (!fend) {
|
||||
/* Malformed request (the filename field is optional, but if
|
||||
* it exists, it needs to be terminated correctly). */
|
||||
return -1;
|
||||
}
|
||||
|
||||
/* TODO: check Content-Type */
|
||||
/* Content-Type: application/octet-stream */
|
||||
|
||||
}
|
||||
else {
|
||||
fend = fbeg;
|
||||
}
|
||||
|
||||
memset(path, 0, sizeof(path));
|
||||
field_count++;
|
||||
field_storage = url_encoded_field_found(conn,
|
||||
nbeg,
|
||||
(size_t)(nend - nbeg),
|
||||
fbeg,
|
||||
(size_t)(fend - fbeg),
|
||||
path,
|
||||
sizeof(path) - 1,
|
||||
fdh);
|
||||
|
||||
/* If the boundary is already in the buffer, get the address,
|
||||
* otherwise next will be NULL. */
|
||||
next = search_boundary(hbuf,
|
||||
(size_t)((buf - hbuf) + buf_fill),
|
||||
boundary,
|
||||
bl);
|
||||
|
||||
if (field_storage == FORM_FIELD_STORAGE_GET) {
|
||||
if (!next) {
|
||||
/* TODO: check for an easy way to get longer data */
|
||||
mg_cry(conn, "%s: Data too long for callback", __func__);
|
||||
return -1;
|
||||
}
|
||||
|
||||
/* Call callback */
|
||||
url_encoded_field_get(conn,
|
||||
nbeg,
|
||||
(size_t)(nend - nbeg),
|
||||
hend,
|
||||
(size_t)(next - hend),
|
||||
fdh);
|
||||
}
|
||||
|
||||
if (field_storage == FORM_FIELD_STORAGE_STORE) {
|
||||
/* Store the content to a file */
|
||||
size_t towrite, n;
|
||||
if (mg_fopen(conn, path, "wb", &fstore) == 0) {
|
||||
fstore.fp = NULL;
|
||||
}
|
||||
file_size = 0;
|
||||
|
||||
if (!fstore.fp) {
|
||||
mg_cry(conn, "%s: Cannot create file %s", __func__, path);
|
||||
}
|
||||
|
||||
while (!next) {
|
||||
/* Set "towrite" to the number of bytes available
|
||||
* in the buffer */
|
||||
towrite = (size_t)(buf - hend + buf_fill);
|
||||
/* Subtract the boundary length, to deal with
|
||||
* cases the boundary is only partially stored
|
||||
* in the buffer. */
|
||||
towrite -= bl + 4;
|
||||
|
||||
if (fstore.fp) {
|
||||
|
||||
/* Store the content of the buffer. */
|
||||
n = (size_t)fwrite(hend, 1, towrite, fstore.fp);
|
||||
if ((n != towrite) || (ferror(fstore.fp))) {
|
||||
mg_cry(conn,
|
||||
"%s: Cannot write file %s",
|
||||
__func__,
|
||||
path);
|
||||
fclose(fstore.fp);
|
||||
fstore.fp = NULL;
|
||||
remove_bad_file(conn, path);
|
||||
}
|
||||
file_size += (size_t)n;
|
||||
}
|
||||
|
||||
memmove(buf, hend + towrite, bl + 4);
|
||||
buf_fill = (int)(bl + 4);
|
||||
hend = buf;
|
||||
|
||||
/* Read new data */
|
||||
r = mg_read(conn,
|
||||
buf + (size_t)buf_fill,
|
||||
sizeof(buf) - 1 - (size_t)buf_fill);
|
||||
if (r < 0) {
|
||||
/* read error */
|
||||
return -1;
|
||||
}
|
||||
buf_fill += r;
|
||||
buf[buf_fill] = 0;
|
||||
if (buf_fill < 1) {
|
||||
/* No data */
|
||||
return -1;
|
||||
}
|
||||
|
||||
/* Find boundary */
|
||||
next = search_boundary(buf, (size_t)buf_fill, boundary, bl);
|
||||
}
|
||||
|
||||
if (fstore.fp) {
|
||||
towrite = (size_t)(next - hend);
|
||||
n = (size_t)fwrite(hend, 1, towrite, fstore.fp);
|
||||
if ((n != towrite) || (ferror(fstore.fp))) {
|
||||
mg_cry(conn,
|
||||
"%s: Cannot write file %s",
|
||||
__func__,
|
||||
path);
|
||||
fclose(fstore.fp);
|
||||
fstore.fp = NULL;
|
||||
remove_bad_file(conn, path);
|
||||
}
|
||||
file_size += (size_t)n;
|
||||
}
|
||||
|
||||
if (fstore.fp) {
|
||||
r = fclose(fstore.fp);
|
||||
if (r == 0) {
|
||||
/* stored successfully */
|
||||
field_stored(conn, path, file_size, fdh);
|
||||
}
|
||||
else {
|
||||
mg_cry(conn,
|
||||
"%s: Error saving file %s",
|
||||
__func__,
|
||||
path);
|
||||
remove_bad_file(conn, path);
|
||||
}
|
||||
fstore.fp = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
if ((field_storage & FORM_FIELD_STORAGE_ABORT)
|
||||
== FORM_FIELD_STORAGE_ABORT) {
|
||||
/* Stop parsing the request */
|
||||
return -1;
|
||||
}
|
||||
|
||||
/* Remove from the buffer */
|
||||
used = next - buf + 2;
|
||||
memmove(buf, buf + (size_t)used, sizeof(buf) - (size_t)used);
|
||||
buf_fill -= (int)used;
|
||||
}
|
||||
|
||||
/* All parts handled */
|
||||
return field_count;
|
||||
}
|
||||
|
||||
/* Unknown Content-Type */
|
||||
return -1;
|
||||
}
|
@ -1,586 +0,0 @@
|
||||
/* RetroArch - A frontend for libretro.
|
||||
* Copyright (C) 2015-2017 - Andre Leiradella
|
||||
*
|
||||
* 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/>.
|
||||
*/
|
||||
|
||||
#define __STDC_FORMAT_MACROS
|
||||
#include <inttypes.h>
|
||||
#include <string.h>
|
||||
#include <stdarg.h>
|
||||
|
||||
#include <libretro.h>
|
||||
|
||||
#include <civetweb/civetweb.h>
|
||||
#include <string/stdstring.h>
|
||||
#include <compat/zlib.h>
|
||||
|
||||
#include "../../core.h"
|
||||
#include "../../retroarch.h"
|
||||
#include "../../core.h"
|
||||
#include "../../managers/core_option_manager.h"
|
||||
#include "../../cheevos-new/cheevos.h"
|
||||
#include "../../content.h"
|
||||
|
||||
#define BASIC_INFO "info"
|
||||
#define MEMORY_MAP "memoryMap"
|
||||
|
||||
static struct mg_callbacks s_httpserver_callbacks;
|
||||
static struct mg_context *s_httpserver_ctx = NULL;
|
||||
|
||||
/* Based on https://github.com/zeromq/rfc/blob/master/src/spec_32.c */
|
||||
static void httpserver_z85_encode_inplace(Bytef* data, size_t size)
|
||||
{
|
||||
static char digits[85 + 1] =
|
||||
{
|
||||
"0123456789"
|
||||
"abcdefghij"
|
||||
"klmnopqrst"
|
||||
"uvwxyzABCD"
|
||||
"EFGHIJKLMN"
|
||||
"OPQRSTUVWX"
|
||||
"YZ.-:+=^!/"
|
||||
"*?&<>()[]{"
|
||||
"}@%$#"
|
||||
};
|
||||
|
||||
uLong value;
|
||||
Bytef* source = data + size - 4;
|
||||
Bytef* dest = data + size * 5 / 4 - 5;
|
||||
|
||||
dest[5] = 0;
|
||||
|
||||
if (source >= data)
|
||||
{
|
||||
do
|
||||
{
|
||||
value = source[0] * 256 * 256 * 256;
|
||||
value += source[1] * 256 * 256;
|
||||
value += source[2] * 256;
|
||||
value += source[3];
|
||||
source -= 4;
|
||||
|
||||
dest[4] = digits[value % 85];
|
||||
value /= 85;
|
||||
dest[3] = digits[value % 85];
|
||||
value /= 85;
|
||||
dest[2] = digits[value % 85];
|
||||
value /= 85;
|
||||
dest[1] = digits[value % 85];
|
||||
dest[0] = digits[value / 85];
|
||||
dest -= 5;
|
||||
|
||||
} while (source >= data);
|
||||
}
|
||||
}
|
||||
|
||||
static void json_string_encode(char* output, size_t size, const char* input)
|
||||
{
|
||||
/* Don't use with UTF-8 strings. */
|
||||
char k;
|
||||
|
||||
if (*input != 0 && size != 0)
|
||||
{
|
||||
do
|
||||
{
|
||||
switch (k = *input++)
|
||||
{
|
||||
case '"': /* fall through */
|
||||
case '\\': /* fall through */
|
||||
case '/': if (size >= 3) { *output++ = '\\'; *output++ = k; size -= 2; } break;
|
||||
case '\b': if (size >= 3) { *output++ = '\\'; *output++ = 'b'; size -= 2; } break;
|
||||
case '\f': if (size >= 3) { *output++ = '\\'; *output++ = 'f'; size -= 2; } break;
|
||||
case '\n': if (size >= 3) { *output++ = '\\'; *output++ = 'n'; size -= 2; } break;
|
||||
case '\r': if (size >= 3) { *output++ = '\\'; *output++ = 'r'; size -= 2; } break;
|
||||
case '\t': if (size >= 3) { *output++ = '\\'; *output++ = 't'; size -= 2; } break;
|
||||
default: if (size >= 2) { *output++ = k; } size--; break;
|
||||
}
|
||||
}
|
||||
while (*input != 0);
|
||||
}
|
||||
|
||||
*output = 0;
|
||||
}
|
||||
|
||||
/*============================================================
|
||||
HTTP ERRORS
|
||||
============================================================ */
|
||||
|
||||
static int httpserver_error(struct mg_connection* conn, unsigned code, const char* fmt, ...)
|
||||
{
|
||||
va_list args;
|
||||
char buffer[1024] = {0};
|
||||
const char* reason = NULL;
|
||||
|
||||
switch (code)
|
||||
{
|
||||
case 404:
|
||||
reason = "Not Found";
|
||||
break;
|
||||
|
||||
case 405:
|
||||
reason = "Method Not Allowed";
|
||||
break;
|
||||
|
||||
default:
|
||||
/* Send unknown codes as 500 */
|
||||
code = 500;
|
||||
reason = "Internal Server Error";
|
||||
break;
|
||||
}
|
||||
|
||||
va_start(args, fmt);
|
||||
vsnprintf(buffer, sizeof(buffer), fmt, args);
|
||||
va_end(args);
|
||||
buffer[sizeof(buffer) - 1] = 0;
|
||||
|
||||
mg_printf(conn, "HTTP/1.1 %u %s\r\nContent-Type: text/html\r\n\r\n", code, reason);
|
||||
mg_printf(conn, "<html><body><h1>%u %s</h1><p>%s</p></body></html>", code, reason, buffer);
|
||||
return 1;
|
||||
}
|
||||
|
||||
/*============================================================
|
||||
INFO
|
||||
============================================================ */
|
||||
|
||||
static int httpserver_handle_basic_info(struct mg_connection* conn, void* cbdata)
|
||||
{
|
||||
static const char *libretro_btn_desc[] = {
|
||||
"B (bottom)", "Y (left)", "Select", "Start",
|
||||
"D-Pad Up", "D-Pad Down", "D-Pad Left", "D-Pad Right",
|
||||
"A (right)", "X (up)",
|
||||
"L", "R", "L2", "R2", "L3", "R3",
|
||||
};
|
||||
|
||||
unsigned p, q, r;
|
||||
retro_ctx_api_info_t api;
|
||||
retro_ctx_region_info_t region;
|
||||
retro_ctx_memory_info_t sram;
|
||||
retro_ctx_memory_info_t rtc;
|
||||
retro_ctx_memory_info_t sysram;
|
||||
retro_ctx_memory_info_t vram;
|
||||
char core_path[PATH_MAX_LENGTH] = {0};
|
||||
const char* pixel_format = NULL;
|
||||
const struct retro_subsystem_info* subsys = NULL;
|
||||
const struct retro_subsystem_rom_info* rom = NULL;
|
||||
const struct retro_subsystem_memory_info* mem = NULL;
|
||||
const struct retro_controller_description* ctrl = NULL;
|
||||
const char* comma = NULL;
|
||||
const struct core_option* opts = NULL;
|
||||
const struct retro_system_av_info* av_info = NULL;
|
||||
const core_option_manager_t* core_opts = NULL;
|
||||
const struct mg_request_info * req = mg_get_request_info(conn);
|
||||
const settings_t * settings = config_get_ptr();
|
||||
rarch_system_info_t *system = runloop_get_system_info();
|
||||
|
||||
if (string_is_empty(system->info.library_name))
|
||||
return httpserver_error(conn, 500, "Core not initialized in %s", __FUNCTION__);
|
||||
|
||||
if (!core_is_game_loaded())
|
||||
return httpserver_error(conn, 500, "Game not loaded in %s", __FUNCTION__);
|
||||
|
||||
json_string_encode(core_path, sizeof(core_path), config_get_active_core_path());
|
||||
|
||||
core_api_version(&api);
|
||||
core_get_region(®ion);
|
||||
|
||||
switch (video_driver_get_pixel_format())
|
||||
{
|
||||
case RETRO_PIXEL_FORMAT_0RGB1555:
|
||||
pixel_format = "RETRO_PIXEL_FORMAT_0RGB1555";
|
||||
break;
|
||||
case RETRO_PIXEL_FORMAT_XRGB8888:
|
||||
pixel_format = "RETRO_PIXEL_FORMAT_XRGB8888";
|
||||
break;
|
||||
case RETRO_PIXEL_FORMAT_RGB565:
|
||||
pixel_format = "RETRO_PIXEL_FORMAT_RGB565";
|
||||
break;
|
||||
default:
|
||||
pixel_format = "?";
|
||||
break;
|
||||
}
|
||||
|
||||
sram.id = RETRO_MEMORY_SAVE_RAM;
|
||||
core_get_memory(&sram);
|
||||
|
||||
rtc.id = RETRO_MEMORY_RTC;
|
||||
core_get_memory(&rtc);
|
||||
|
||||
sysram.id = RETRO_MEMORY_SYSTEM_RAM;
|
||||
core_get_memory(&sysram);
|
||||
|
||||
vram.id = RETRO_MEMORY_VIDEO_RAM;
|
||||
core_get_memory(&vram);
|
||||
|
||||
mg_printf(conn,
|
||||
"HTTP/1.1 200 OK\r\nContent-Type: application/json\r\n\r\n"
|
||||
"{"
|
||||
"\"corePath\":\"%s\","
|
||||
"\"apiVersion\":%u,"
|
||||
"\"systemInfo\":"
|
||||
"{"
|
||||
"\"libraryName\":\"%s\","
|
||||
"\"libraryVersion\":\"%s\","
|
||||
"\"validExtensions\":\"%s\","
|
||||
"\"needsFullpath\":%s,"
|
||||
"\"blockExtract\":%s"
|
||||
"},"
|
||||
"\"region\":\"%s\","
|
||||
"\"pixelFormat\":\"%s\","
|
||||
"\"rotation\":%u,"
|
||||
"\"performaceLevel\":%u,"
|
||||
"\"supportsNoGame\":%s,"
|
||||
#ifdef HAVE_CHEEVOS
|
||||
"\"frontendSupportsAchievements\":true,"
|
||||
"\"coreSupportsAchievements\":%s,"
|
||||
#else
|
||||
"\"frontendSupportsAchievements\":false,"
|
||||
"\"coreSupportsAchievements\":null,"
|
||||
#endif
|
||||
"\"saveRam\":{\"pointer\":\"%" PRIXPTR "\",\"size\":%" PRIu64 "},"
|
||||
"\"rtcRam\":{\"pointer\":\"%" PRIXPTR "\",\"size\":%" PRIu64 "},"
|
||||
"\"systemRam\":{\"pointer\":\"%" PRIXPTR "\",\"size\":%" PRIu64 "},"
|
||||
"\"videoRam\":{\"pointer\":\"%" PRIXPTR "\",\"size\":%" PRIu64 "},",
|
||||
core_path,
|
||||
api.version,
|
||||
system->info.library_name,
|
||||
system->info.library_version,
|
||||
system->info.valid_extensions,
|
||||
system->info.need_fullpath ? "true" : "false",
|
||||
system->info.block_extract ? "true" : "false",
|
||||
region.region ? "RETRO_REGION_PAL" : "RETRO_REGION_NTSC",
|
||||
pixel_format,
|
||||
system->rotation,
|
||||
system->performance_level,
|
||||
content_does_not_need_content() ? "true" : "false",
|
||||
#ifdef HAVE_CHEEVOS
|
||||
cheevos_get_support_cheevos() ? "true" : "false",
|
||||
#endif
|
||||
(uintptr_t)sram.data, sram.size,
|
||||
(uintptr_t)rtc.data, rtc.size,
|
||||
(uintptr_t)sysram.data, sysram.size,
|
||||
(uintptr_t)vram.data, vram.size
|
||||
);
|
||||
|
||||
mg_printf(conn, "\"subSystems\":[");
|
||||
subsys = system->subsystem.data;
|
||||
|
||||
for (p = 0; p < system->subsystem.size; p++, subsys++)
|
||||
{
|
||||
mg_printf(conn, "%s{\"id\":%u,\"description\":\"%s\",\"identifier\":\"%s\",\"roms\":[", p == 0 ? "" : ",", subsys->id, subsys->desc, subsys->ident);
|
||||
rom = subsys->roms;
|
||||
|
||||
for (q = 0; q < subsys->num_roms; q++, rom++)
|
||||
{
|
||||
mg_printf(conn,
|
||||
"%s{"
|
||||
"\"description\":\"%s\","
|
||||
"\"extensions\":\"%s\","
|
||||
"\"needsFullpath\":%s,"
|
||||
"\"blockExtract\":%s,"
|
||||
"\"required\":%s,"
|
||||
"\"memory\":[",
|
||||
q == 0 ? "" : ",",
|
||||
rom->desc,
|
||||
rom->valid_extensions,
|
||||
rom->need_fullpath ? "true" : "false",
|
||||
rom->block_extract ? "true" : "false",
|
||||
rom->required ? "true" : "false"
|
||||
);
|
||||
|
||||
mem = rom->memory;
|
||||
comma = "";
|
||||
|
||||
for (r = 0; r < rom->num_memory; r++, mem++)
|
||||
{
|
||||
mg_printf(conn, "%s{\"extension\":\"%s\",\"type\":%u}", comma, mem->extension, mem->type);
|
||||
comma = ",";
|
||||
}
|
||||
|
||||
mg_printf(conn, "]}");
|
||||
}
|
||||
|
||||
mg_printf(conn, "]}");
|
||||
}
|
||||
|
||||
av_info = video_viewport_get_system_av_info();
|
||||
|
||||
mg_printf(conn,
|
||||
"],\"avInfo\":{"
|
||||
"\"geometry\":{"
|
||||
"\"baseWidth\":%u,"
|
||||
"\"baseHeight\":%u,"
|
||||
"\"maxWidth\":%u,"
|
||||
"\"maxHeight\":%u,"
|
||||
"\"aspectRatio\":%f"
|
||||
"},"
|
||||
"\"timing\":{"
|
||||
"\"fps\":%f,"
|
||||
"\"sampleRate\":%f"
|
||||
"}"
|
||||
"},",
|
||||
av_info->geometry.base_width,
|
||||
av_info->geometry.base_height,
|
||||
av_info->geometry.max_width,
|
||||
av_info->geometry.max_height,
|
||||
av_info->geometry.aspect_ratio,
|
||||
av_info->timing.fps,
|
||||
av_info->timing.sample_rate
|
||||
);
|
||||
|
||||
mg_printf(conn, "\"ports\":[");
|
||||
comma = "";
|
||||
|
||||
for (p = 0; p < system->ports.size; p++)
|
||||
{
|
||||
ctrl = system->ports.data[p].types;
|
||||
|
||||
for (q = 0; q < system->ports.data[p].num_types; q++, ctrl++)
|
||||
{
|
||||
mg_printf(conn, "%s{\"id\":%u,\"description\":\"%s\"}", comma, ctrl->id, ctrl->desc);
|
||||
comma = ",";
|
||||
}
|
||||
}
|
||||
|
||||
mg_printf(conn, "],\"inputDescriptors\":[");
|
||||
comma = "";
|
||||
|
||||
if (core_has_set_input_descriptor())
|
||||
{
|
||||
for (p = 0; p < settings->input.max_users; p++)
|
||||
{
|
||||
for (q = 0; q < RARCH_FIRST_CUSTOM_BIND; q++)
|
||||
{
|
||||
const char* description = system->input_desc_btn[p][q];
|
||||
|
||||
if (description)
|
||||
{
|
||||
mg_printf(conn,
|
||||
"%s{\"player\":%u,\"button\":\"%s\",\"description\":\"%s\"}",
|
||||
comma,
|
||||
p + 1,
|
||||
libretro_btn_desc[q],
|
||||
description
|
||||
);
|
||||
|
||||
comma = ",";
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
mg_printf(conn, "],\"coreOptions\":[");
|
||||
rarch_ctl(RARCH_CTL_CORE_OPTIONS_LIST_GET, (void*)&core_opts);
|
||||
opts = core_opts->opts;
|
||||
|
||||
for (p = 0; p < core_opts->size; p++, opts++)
|
||||
{
|
||||
mg_printf(conn, "%s{\"key\":\"%s\",\"description\":\"%s\",\"values\":[", p == 0 ? "" : ",", opts->key, opts->desc);
|
||||
comma = "";
|
||||
|
||||
for (q = 0; q < opts->vals->size; q++)
|
||||
{
|
||||
mg_printf(conn, "%s\"%s\"", comma, opts->vals->elems[q].data);
|
||||
comma = ",";
|
||||
}
|
||||
|
||||
mg_printf(conn, "]}");
|
||||
}
|
||||
|
||||
mg_printf(conn, "]}");
|
||||
return 1;
|
||||
}
|
||||
|
||||
/*============================================================
|
||||
MMAPS
|
||||
============================================================ */
|
||||
|
||||
static int httpserver_handle_get_mmaps(struct mg_connection* conn, void* cbdata)
|
||||
{
|
||||
unsigned id;
|
||||
const struct mg_request_info* req = mg_get_request_info(conn);
|
||||
const char* comma = "";
|
||||
const struct retro_memory_map* mmaps = NULL;
|
||||
const struct retro_memory_descriptor* mmap = NULL;
|
||||
rarch_system_info_t *system = runloop_get_system_info();
|
||||
|
||||
if (strcmp(req->request_method, "GET"))
|
||||
return httpserver_error(conn, 405, "Unimplemented method in %s: %s", __FUNCTION__, req->request_method);
|
||||
|
||||
mmaps = &system->mmaps;
|
||||
mmap = mmaps->descriptors;
|
||||
|
||||
mg_printf(conn, "HTTP/1.1 200 OK\r\nContent-Type: application/json\r\n\r\n");
|
||||
mg_printf(conn, "[");
|
||||
|
||||
for (id = 0; id < mmaps->num_descriptors; id++, mmap++)
|
||||
{
|
||||
mg_printf(conn,
|
||||
"%s{"
|
||||
"\"id\":%u,"
|
||||
"\"flags\":%" PRIu64 ","
|
||||
"\"ptr\":\"%" PRIXPTR "\","
|
||||
"\"offset\":%" PRIu64 ","
|
||||
"\"start\":%" PRIu64 ","
|
||||
"\"select\":%" PRIu64 ","
|
||||
"\"disconnect\":%" PRIu64 ","
|
||||
"\"len\":%" PRIu64 ","
|
||||
"\"addrspace\":\"%s\""
|
||||
"}",
|
||||
comma,
|
||||
id,
|
||||
mmap->flags,
|
||||
(uintptr_t)mmap->ptr,
|
||||
mmap->offset,
|
||||
mmap->start,
|
||||
mmap->select,
|
||||
mmap->disconnect,
|
||||
mmap->len,
|
||||
mmap->addrspace ? mmap->addrspace : ""
|
||||
);
|
||||
|
||||
comma = ",";
|
||||
}
|
||||
|
||||
mg_printf(conn, "]");
|
||||
return 1;
|
||||
}
|
||||
|
||||
static int httpserver_handle_get_mmap(struct mg_connection* conn, void* cbdata)
|
||||
{
|
||||
size_t start, length;
|
||||
unsigned id;
|
||||
uLong buflen;
|
||||
const struct mg_request_info * req = mg_get_request_info(conn);
|
||||
const char * comma = "";
|
||||
const struct retro_memory_map* mmaps = NULL;
|
||||
const struct retro_memory_descriptor* mmap = NULL;
|
||||
const char* param = NULL;
|
||||
Bytef* buffer = NULL;
|
||||
rarch_system_info_t *system = runloop_get_system_info();
|
||||
|
||||
if (strcmp(req->request_method, "GET"))
|
||||
return httpserver_error(conn, 405, "Unimplemented method in %s: %s", __FUNCTION__, req->request_method);
|
||||
|
||||
if (sscanf(req->request_uri, "/" MEMORY_MAP "/%u", &id) != 1)
|
||||
return httpserver_error(conn, 500, "Malformed request in %s: %s", __FUNCTION__, req->request_uri);
|
||||
|
||||
mmaps = &system->mmaps;
|
||||
|
||||
if (id >= mmaps->num_descriptors)
|
||||
return httpserver_error(conn, 404, "Invalid memory map id in %s: %u", __FUNCTION__, id);
|
||||
|
||||
mmap = mmaps->descriptors + id;
|
||||
start = 0;
|
||||
length = mmap->len;
|
||||
|
||||
if (req->query_string != NULL)
|
||||
{
|
||||
param = strstr(req->query_string, "start=");
|
||||
|
||||
if (param != NULL)
|
||||
start = atoll(param + 6);
|
||||
|
||||
param = strstr(req->query_string, "length=");
|
||||
|
||||
if (param != NULL)
|
||||
length = atoll(param + 7);
|
||||
}
|
||||
|
||||
if (start >= mmap->len)
|
||||
start = mmap->len - 1;
|
||||
|
||||
if (length > mmap->len - start)
|
||||
length = mmap->len - start;
|
||||
|
||||
buflen = compressBound(length);
|
||||
buffer = (Bytef*)malloc(((buflen + 3) / 4) * 5);
|
||||
|
||||
if (buffer == NULL)
|
||||
return httpserver_error(conn, 500, "Out of memory in %s", __FUNCTION__);
|
||||
|
||||
if (compress2(buffer, &buflen, (Bytef*)mmap->ptr + start, length, Z_BEST_COMPRESSION) != Z_OK)
|
||||
{
|
||||
free((void*)buffer);
|
||||
return httpserver_error(conn, 500, "Error during compression in %s", __FUNCTION__);
|
||||
}
|
||||
|
||||
buffer[buflen] = 0;
|
||||
buffer[buflen + 1] = 0;
|
||||
buffer[buflen + 2] = 0;
|
||||
httpserver_z85_encode_inplace(buffer, (buflen + 3) & ~3);
|
||||
|
||||
mg_printf(conn, "HTTP/1.1 200 OK\r\nContent-Type: application/json\r\n\r\n");
|
||||
mg_printf(conn,
|
||||
"{"
|
||||
"\"start\":" STRING_REP_USIZE ","
|
||||
"\"length\":" STRING_REP_USIZE ","
|
||||
"\"compression\":\"deflate\","
|
||||
"\"compressedLength\":" STRING_REP_USIZE ","
|
||||
"\"encoding\":\"Z85\","
|
||||
"\"data\":\"%s\""
|
||||
"}",
|
||||
start,
|
||||
length,
|
||||
(size_t)buflen,
|
||||
(char*)buffer
|
||||
);
|
||||
|
||||
free((void*)buffer);
|
||||
return 1;
|
||||
}
|
||||
|
||||
static int httpserver_handle_mmaps(struct mg_connection* conn, void* cbdata)
|
||||
{
|
||||
unsigned id;
|
||||
const struct mg_request_info* req = mg_get_request_info(conn);
|
||||
|
||||
if (sscanf(req->request_uri, "/" MEMORY_MAP "/%u", &id) == 1)
|
||||
return httpserver_handle_get_mmap(conn, cbdata);
|
||||
|
||||
return httpserver_handle_get_mmaps(conn, cbdata);
|
||||
}
|
||||
|
||||
/*============================================================
|
||||
HTTP SERVER
|
||||
============================================================ */
|
||||
|
||||
int httpserver_init(unsigned port)
|
||||
{
|
||||
char str[16];
|
||||
snprintf(str, sizeof(str), "%u", port);
|
||||
str[sizeof(str) - 1] = 0;
|
||||
|
||||
const char* options[] =
|
||||
{
|
||||
"listening_ports", str,
|
||||
NULL, NULL
|
||||
};
|
||||
|
||||
memset(&s_httpserver_callbacks, 0, sizeof(s_httpserver_callbacks));
|
||||
s_httpserver_ctx = mg_start(&s_httpserver_callbacks, NULL, options);
|
||||
|
||||
if (s_httpserver_ctx == NULL)
|
||||
return -1;
|
||||
|
||||
mg_set_request_handler(s_httpserver_ctx, "/" BASIC_INFO, httpserver_handle_basic_info, NULL);
|
||||
|
||||
mg_set_request_handler(s_httpserver_ctx, "/" MEMORY_MAP, httpserver_handle_mmaps, NULL);
|
||||
mg_set_request_handler(s_httpserver_ctx, "/" MEMORY_MAP "/", httpserver_handle_mmaps, NULL);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
void httpserver_destroy(void)
|
||||
{
|
||||
mg_stop(s_httpserver_ctx);
|
||||
}
|
@ -1,29 +0,0 @@
|
||||
/* RetroArch - A frontend for libretro.
|
||||
* Copyright (C) 2015-2017 - Andre Leiradella
|
||||
*
|
||||
* 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/>.
|
||||
*/
|
||||
|
||||
#ifndef __RARCH_HTTPSERVR_H
|
||||
#define __RARCH_HTTPSERVR_H
|
||||
|
||||
#include <retro_common_api.h>
|
||||
|
||||
RETRO_BEGIN_DECLS
|
||||
|
||||
int httpserver_init(unsigned port);
|
||||
|
||||
void httpserver_destroy(void);
|
||||
|
||||
RETRO_END_DECLS
|
||||
|
||||
#endif /* __RARCH_HTTPSERVR_H */
|
10
retroarch.c
10
retroarch.c
@ -123,10 +123,6 @@
|
||||
#include "network/netplay/netplay.h"
|
||||
#endif
|
||||
|
||||
#if defined(HAVE_HTTPSERVER) && defined(HAVE_ZLIB)
|
||||
#include "network/httpserver/httpserver.h"
|
||||
#endif
|
||||
|
||||
#ifdef HAVE_THREADS
|
||||
#include <rthreads/rthreads.h>
|
||||
#endif
|
||||
@ -15914,14 +15910,8 @@ bool rarch_ctl(enum rarch_ctl_state state, void *data)
|
||||
}
|
||||
break;
|
||||
case RARCH_CTL_HTTPSERVER_INIT:
|
||||
#if defined(HAVE_HTTPSERVER) && defined(HAVE_ZLIB)
|
||||
httpserver_init(8888);
|
||||
#endif
|
||||
break;
|
||||
case RARCH_CTL_HTTPSERVER_DESTROY:
|
||||
#if defined(HAVE_HTTPSERVER) && defined(HAVE_ZLIB)
|
||||
httpserver_destroy();
|
||||
#endif
|
||||
break;
|
||||
case RARCH_CTL_CAMERA_SET_ACTIVE:
|
||||
camera_driver_active = true;
|
||||
|
Loading…
x
Reference in New Issue
Block a user