lpac/driver/http/curl.c
Coelacanthus 8f43334888
refactor: always use \n for newline and make fuart unbuffered (#201)
* refactor: revert println refactor

This reverts commit ae3a87803c.
This reverts commit 37ec17eeab.

* fix(apdu/at): make fuart unbuffered

related #167

Signed-off-by: Coelacanthus <uwu@coelacanthus.name>

* refactor: always use \n for newline

> C89 Standard section 5.2.2 Character display semantics
> \n (new line) Moves the active position to the initial position of the next line.

'\n' in C have including meaning of both '\r' and '\n' on Windows, and
libc SHALL translate them when do input and output. If not, it SHALL be
a bug of libc.

FYI, all mingw64 toolchain code use '\n' directly, so it's impossible to
be caused by '\n', otherwise mingw64 itself will broken. Mingw64 also
set _fmode to _O_TEXT by default.[1] So as MSVC documents said[2], it
should do translate between LF and CRLF.

[1]: https://github.com/mingw-w64/mingw-w64/blob/master/mingw-w64-crt/crt/txtmode.c
[2]: https://learn.microsoft.com/en-us/cpp/c-runtime-library/reference/fopen-wfopen

Signed-off-by: Coelacanthus <uwu@coelacanthus.name>

---------

Signed-off-by: Coelacanthus <uwu@coelacanthus.name>
2025-03-03 06:19:32 +08:00

205 lines
5.9 KiB
C

#include "curl.h"
#include <inttypes.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <euicc/interface.h>
#ifndef _WIN32
#include <curl/curl.h>
#else
#include <dlfcn-win32/dlfcn.h>
#define CURL_GLOBAL_DEFAULT ((1 << 0) | (1 << 1))
#define CURLE_OK 0
#define CURLOPT_URL 10002
#define CURLOPT_WRITEFUNCTION 20011
#define CURLOPT_WRITEDATA 10001
#define CURLOPT_SSL_VERIFYPEER 64
#define CURLOPT_SSL_VERIFYHOST 81
#define CURLOPT_HTTPHEADER 10023
#define CURLOPT_POSTFIELDS 10015
#define CURLOPT_POSTFIELDSIZE 60
#define CURLINFO_RESPONSE_CODE 2097154
typedef void CURL;
typedef int CURLcode;
typedef int CURLoption;
typedef int CURLINFO;
static void *libcurl_interface_dlhandle = NULL;
#endif
struct http_trans_response_data
{
uint8_t *data;
size_t size;
};
static struct libcurl_interface
{
CURLcode (*_curl_global_init)(long flags);
CURL *(*_curl_easy_init)(void);
CURLcode (*_curl_easy_setopt)(CURL *curl, CURLoption option, ...);
CURLcode (*_curl_easy_perform)(CURL *curl);
CURLcode (*_curl_easy_getinfo)(CURL *curl, CURLINFO info, ...);
const char *(*_curl_easy_strerror)(CURLcode);
void (*_curl_easy_cleanup)(CURL *curl);
struct curl_slist *(*_curl_slist_append)(struct curl_slist *list, const char *data);
void (*_curl_slist_free_all)(struct curl_slist *list);
} libcurl;
static size_t http_trans_write_callback(void *contents, size_t size, size_t nmemb, void *userp)
{
size_t realsize = size * nmemb;
struct http_trans_response_data *mem = (struct http_trans_response_data *)userp;
mem->data = realloc(mem->data, mem->size + realsize + 1);
if (mem->data == NULL)
{
/* out of memory! */
printf("not enough memory (realloc returned NULL)\n");
return 0;
}
memcpy(&(mem->data[mem->size]), contents, realsize);
mem->size += realsize;
mem->data[mem->size] = 0;
return realsize;
}
static int http_interface_transmit(struct euicc_ctx *ctx, const char *url, uint32_t *rcode, uint8_t **rx, uint32_t *rx_len, const uint8_t *tx, uint32_t tx_len, const char **h)
{
int fret = 0;
CURL *curl;
CURLcode res;
struct http_trans_response_data responseData = {0};
struct curl_slist *headers = NULL, *nheaders = NULL;
long response_code;
(*rx) = NULL;
(*rcode) = 0;
curl = libcurl._curl_easy_init();
if (!curl)
{
goto err;
}
libcurl._curl_easy_setopt(curl, CURLOPT_URL, url);
libcurl._curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, http_trans_write_callback);
libcurl._curl_easy_setopt(curl, CURLOPT_WRITEDATA, (void *)&responseData);
libcurl._curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, 0L);
libcurl._curl_easy_setopt(curl, CURLOPT_SSL_VERIFYHOST, 0L);
for (int i = 0; h[i] != NULL; i++)
{
nheaders = libcurl._curl_slist_append(headers, h[i]);
if (nheaders == NULL)
{
goto err;
}
headers = nheaders;
}
libcurl._curl_easy_setopt(curl, CURLOPT_HTTPHEADER, headers);
if (tx != NULL)
{
libcurl._curl_easy_setopt(curl, CURLOPT_POSTFIELDS, tx);
libcurl._curl_easy_setopt(curl, CURLOPT_POSTFIELDSIZE, tx_len);
}
res = libcurl._curl_easy_perform(curl);
if (res != CURLE_OK)
{
fprintf(stderr, "curl_easy_perform() failed: %s\n", libcurl._curl_easy_strerror(res));
goto err;
}
libcurl._curl_easy_getinfo(curl, CURLINFO_RESPONSE_CODE, &response_code);
*rcode = response_code;
*rx = responseData.data;
*rx_len = responseData.size;
fret = 0;
goto exit;
err:
fret = -1;
free(responseData.data);
exit:
libcurl._curl_easy_cleanup(curl);
libcurl._curl_slist_free_all(headers);
return fret;
}
static int _init_libcurl(void)
{
#ifdef _WIN32
if (!(libcurl_interface_dlhandle = dlopen("libcurl.dll", RTLD_LAZY)))
{
fprintf(stderr, "libcurl init err: %s\n", dlerror());
return -1;
}
libcurl._curl_global_init = dlsym(libcurl_interface_dlhandle, "curl_global_init");
libcurl._curl_easy_init = dlsym(libcurl_interface_dlhandle, "curl_easy_init");
libcurl._curl_easy_setopt = dlsym(libcurl_interface_dlhandle, "curl_easy_setopt");
libcurl._curl_easy_perform = dlsym(libcurl_interface_dlhandle, "curl_easy_perform");
libcurl._curl_easy_getinfo = dlsym(libcurl_interface_dlhandle, "curl_easy_getinfo");
libcurl._curl_easy_strerror = dlsym(libcurl_interface_dlhandle, "curl_easy_strerror");
libcurl._curl_easy_cleanup = dlsym(libcurl_interface_dlhandle, "curl_easy_cleanup");
libcurl._curl_slist_append = dlsym(libcurl_interface_dlhandle, "curl_slist_append");
libcurl._curl_slist_free_all = dlsym(libcurl_interface_dlhandle, "curl_slist_free_all");
#else
libcurl._curl_global_init = curl_global_init;
libcurl._curl_easy_init = curl_easy_init;
libcurl._curl_easy_setopt = curl_easy_setopt;
libcurl._curl_easy_perform = curl_easy_perform;
libcurl._curl_easy_getinfo = curl_easy_getinfo;
libcurl._curl_easy_strerror = curl_easy_strerror;
libcurl._curl_easy_cleanup = curl_easy_cleanup;
libcurl._curl_slist_append = curl_slist_append;
libcurl._curl_slist_free_all = curl_slist_free_all;
#endif
return 0;
}
static int libhttpinterface_init(struct euicc_http_interface *ifstruct)
{
memset(ifstruct, 0, sizeof(struct euicc_http_interface));
if (_init_libcurl() != 0)
{
return -1;
}
if (libcurl._curl_global_init(CURL_GLOBAL_DEFAULT) != CURLE_OK)
{
return -1;
}
ifstruct->transmit = http_interface_transmit;
return 0;
}
static int libhttpinterface_main(int argc, char **argv)
{
return 0;
}
static void libhttpinterface_fini(struct euicc_http_interface *ifstruct)
{
}
const struct euicc_driver driver_http_curl = {
.type = DRIVER_HTTP,
.name = "curl",
.init = (int (*)(void *))libhttpinterface_init,
.main = libhttpinterface_main,
.fini = (void (*)(void *))libhttpinterface_fini,
};