diff --git a/.gitignore b/.gitignore index 6b1edd0f..bc64f977 100644 --- a/.gitignore +++ b/.gitignore @@ -48,6 +48,7 @@ build/ *.recipe *.idb *.iobj +*.pdb # Visual Studio for Mac .idea/ diff --git a/src/webui.c b/src/webui.c index 0917e594..9946533b 100644 --- a/src/webui.c +++ b/src/webui.c @@ -67,6 +67,7 @@ #define WEBUI_CMD_SEND_RAW 0xF8 // Command: Send raw binary data to the UI #define WEBUI_CMD_ADD_ID 0xF7 // Command: Add new bind ID #define WEBUI_CMD_MULTI 0xF6 // Command: Multi packet data +#define WEBUI_CMD_CHECK_TK 0xF5 // Command: Check validity of a client's token #define WEBUI_PROTOCOL_SIZE (8) // Protocol header size in bytes #define WEBUI_PROTOCOL_SIGN (0) // Protocol byte position: Signature (1 Byte) #define WEBUI_PROTOCOL_TOKEN (1) // Protocol byte position: Token (4 Bytes) @@ -97,6 +98,8 @@ #define WEBUI_MAX_X (3000) // Maximal window X (4K Monitor) #define WEBUI_MAX_Y (1800) // Maximal window Y (4K Monitor) #define WEBUI_PROFILE_NAME "WebUI" // Default browser profile name (Used only for Firefox) +#define WEBUI_AUTH_COOKIE (32) // Authentification Cookie Len +#define WEBUI_BROWSERS_COUNT (16) // Total supported browser (enum webui_browser) #ifdef WEBUI_TLS #define WEBUI_SECURE "TLS-Encryption" @@ -151,6 +154,11 @@ typedef pthread_cond_t webui_condition_t; #define WEBUI_COMPILER "Unknown" #endif +// Verbose Log +#ifdef WEBUI_LOG + // #define WEBUI_LOG_VERBOSE +#endif + // Timer typedef struct _webui_timer_t { struct timespec start; @@ -159,10 +167,12 @@ typedef struct _webui_timer_t { // Event data typedef struct webui_event_inf_t { - char* event_data[WEBUI_MAX_ARG + 1]; // Event data (string | num | bool | raw) + char* event_data[WEBUI_MAX_ARG + 1]; // Event data (string | num | bool | raw) size_t event_size[WEBUI_MAX_ARG + 1]; // Event data size (in bytes) - char* response; // Event response (string) + char* response; // Event response (string) size_t count; + struct mg_connection* client; + size_t client_id; } webui_event_inf_t; // WebView @@ -181,14 +191,14 @@ typedef struct webui_event_inf_t { void* createWebViewControllerHandler_lpVtbl; // WebUI Window wchar_t* url; - volatile bool navigate; - volatile bool size; - volatile bool position; - volatile int width; - volatile int height; - volatile int x; - volatile int y; - volatile bool stop; + bool navigate; + bool size; + bool position; + int width; + int height; + int x; + int y; + bool stop; } _webui_wv_win32_t; #elif __linux__ void* libgtk; @@ -220,11 +230,11 @@ typedef struct webui_event_inf_t { gtk_window_move_func gtk_window_move = NULL; gtk_window_close_func gtk_window_close = NULL; gtk_window_resize_func gtk_window_resize = NULL; - gtk_window_set_position_func gtk_window_set_position = NULL; + gtk_window_set_position_func gtk_window_set_position = NULL; g_signal_connect_data_func g_signal_connect_data = NULL; g_idle_add_func g_idle_add = NULL; // WebKit Symbol Addresses - typedef void *(*webkit_web_view_new_func)(void); + typedef void *(*webkit_web_view_new_func)(void); typedef void (*webkit_web_view_load_uri_func)(void *, const char *); typedef const char *(*webkit_web_view_get_title_func)(void *); webkit_web_view_new_func webkit_web_view_new = NULL; @@ -235,17 +245,17 @@ typedef struct webui_event_inf_t { // Linux WebView void* gtk_win; void* gtk_wv; - volatile bool open; + bool open; // WebUI Window char* url; - volatile bool navigate; - volatile bool size; - volatile bool position; - volatile unsigned int width; - volatile unsigned int height; - volatile unsigned int x; - volatile unsigned int y; - volatile bool stop; + bool navigate; + bool size; + bool position; + unsigned int width; + unsigned int height; + unsigned int x; + unsigned int y; + bool stop; } _webui_wv_linux_t; #else extern bool _webui_macos_wv_new(int index); @@ -264,33 +274,47 @@ typedef struct webui_event_inf_t { int index; // WebUI Window char* url; - volatile bool navigate; - volatile bool size; - volatile bool position; - volatile unsigned int width; - volatile unsigned int height; - volatile unsigned int x; - volatile unsigned int y; - volatile bool stop; + bool navigate; + bool size; + bool position; + unsigned int width; + unsigned int height; + unsigned int x; + unsigned int y; + bool stop; } _webui_wv_macos_t; #endif // Window typedef struct _webui_window_t { - size_t window_number; - volatile bool server_running; - volatile bool connected; - volatile bool file_handled; - bool ws_block; - bool html_handled; - bool bridge_handled; - bool server_handled; - bool is_embedded_html; - bool is_closed; - size_t custom_server_port; + // Client + struct mg_connection* clients[WEBUI_MAX_IDS]; + size_t clients_count; + bool clients_token_check[WEBUI_MAX_IDS]; + // Server + bool wait; // Let server thread wait more time for websocket + bool server_running; // Slow check + bool connected; // Fast check size_t server_port; char* url; const char* html; + char* server_root_path; + #ifdef _WIN32 + HANDLE server_thread; + #else + pthread_t server_thread; + #endif + // Window + uint32_t token; + size_t window_number; + char* html_elements[WEBUI_MAX_IDS]; + bool has_all_events; + void(*cb[WEBUI_MAX_IDS])(webui_event_t* e); + void(*cb_interface[WEBUI_MAX_IDS])(size_t, size_t, char* , size_t, size_t); + bool ws_block; + bool is_embedded_html; + bool is_closed; + size_t custom_server_port; const char* icon; const char* icon_type; size_t current_browser; @@ -300,8 +324,6 @@ typedef struct _webui_window_t { char* profile_path; char* profile_name; size_t runtime; - bool has_events; - char* server_root_path; bool kiosk_mode; bool disable_browser_high_contrast; bool hide; @@ -312,14 +334,7 @@ typedef struct _webui_window_t { int y; bool position_set; size_t process_id; - #ifdef _WIN32 - HANDLE server_thread; - #else - pthread_t server_thread; - #endif - const void * ( * files_handler)(const char* filename, int * length); - uint32_t token; - struct mg_connection * mg_connection; + const void*(*files_handler)(const char* filename, int* length); webui_event_inf_t* events[WEBUI_MAX_IDS]; size_t events_count; bool is_public; @@ -327,7 +342,7 @@ typedef struct _webui_window_t { char *proxy_server; // WebView bool allow_webview; - volatile bool update_webview; + bool update_webview; #ifdef _WIN32 _webui_wv_win32_t* webView; #elif __linux__ @@ -345,29 +360,29 @@ typedef struct _webui_core_t { bool show_auto_js_inject; bool ws_block; bool folder_monitor; + bool multi_client; } config; - volatile size_t servers; - char* html_elements[WEBUI_MAX_IDS]; + bool auth_cookie_set[WEBUI_BROWSERS_COUNT]; + char auth_cookie[WEBUI_BROWSERS_COUNT][WEBUI_AUTH_COOKIE * 2]; + size_t servers; size_t used_ports[WEBUI_MAX_IDS]; size_t startup_timeout; - volatile bool exit_now; - volatile bool run_done[WEBUI_MAX_IDS]; // 2 Bytes ID + size_t cb_count; + bool exit_now; + bool run_done[WEBUI_MAX_IDS]; // 2 Bytes ID char* run_userBuffer[WEBUI_MAX_IDS]; size_t run_userBufferLen[WEBUI_MAX_IDS]; bool run_error[WEBUI_MAX_IDS]; uint16_t run_last_id; bool initialized; - void( * cb[WEBUI_MAX_IDS])(webui_event_t* e); - void( * cb_interface[WEBUI_MAX_IDS])(size_t, size_t, char* , size_t, size_t); char* executable_path; void * ptr_list[WEBUI_MAX_IDS * 2]; size_t ptr_position; size_t ptr_size[WEBUI_MAX_IDS * 2]; size_t current_browser; - _webui_window_t * wins[WEBUI_MAX_IDS]; + _webui_window_t* wins[WEBUI_MAX_IDS]; bool wins_reserved[WEBUI_MAX_IDS]; size_t last_win_number; - bool server_handled; webui_mutex_t mutex_server_start; webui_mutex_t mutex_send; webui_mutex_t mutex_receive; @@ -377,15 +392,16 @@ typedef struct _webui_core_t { webui_mutex_t mutex_win_connect; webui_mutex_t mutex_exit_now; webui_mutex_t mutex_webview_stop; + webui_mutex_t mutex_http_handler; + webui_mutex_t mutex_client; webui_condition_t condition_wait; char* default_server_root_path; bool ui; - // bool little_endian; #ifdef WEBUI_TLS - uint8_t * root_cert; - uint8_t * root_key; - uint8_t * ssl_cert; - uint8_t * ssl_key; + uint8_t* root_cert; + uint8_t* root_key; + uint8_t* ssl_cert; + uint8_t* ssl_key; #endif // WebView bool is_browser_main_run; @@ -403,22 +419,22 @@ _webui_core_t; typedef struct _webui_cb_arg_t { // Event - _webui_window_t * window; + _webui_window_t* window; size_t event_type; char* element; char* data; size_t event_number; - // Extras - char* webui_internal_id; } _webui_cb_arg_t; typedef struct _webui_recv_arg_t { - _webui_window_t * win; + _webui_window_t* win; void * ptr; size_t len; size_t recvNum; int event_type; + struct mg_connection* client; + size_t client_id; } _webui_recv_arg_t; @@ -429,7 +445,7 @@ typedef struct _webui_monitor_arg_t { _webui_monitor_arg_t; typedef struct _webui_cmd_async_t { - _webui_window_t * win; + _webui_window_t* win; char* cmd; } _webui_cmd_async_t; @@ -438,13 +454,13 @@ _webui_cmd_async_t; #ifdef _WIN32 static const char* webui_sep = "\\"; static DWORD WINAPI _webui_run_browser_task(LPVOID _arg); -static int _webui_system_win32(_webui_window_t * win, char* cmd, bool show); +static int _webui_system_win32(_webui_window_t* win, char* cmd, bool show); static int _webui_system_win32_out(const char* cmd, char ** output, bool show); static bool _webui_socket_test_listen_win32(size_t port_num); static bool _webui_get_windows_reg_value(HKEY key, LPCWSTR reg, LPCWSTR value_name, char value[WEBUI_MAX_PATH]); static bool _webui_str_to_wide(const char *s, wchar_t **w); #define WEBUI_THREAD_SERVER_START DWORD WINAPI _webui_server_thread(LPVOID arg) -#define WEBUI_THREAD_RECEIVE DWORD WINAPI _webui_process_thread(LPVOID _arg) +#define WEBUI_THREAD_RECEIVE DWORD WINAPI _webui_ws_process_thread(LPVOID _arg) #define WEBUI_THREAD_WEBVIEW DWORD WINAPI _webui_webview_thread(LPVOID arg) #define WEBUI_THREAD_MONITOR DWORD WINAPI _webui_folder_monitor_thread(LPVOID arg) #define WEBUI_THREAD_RETURN return 0; @@ -452,87 +468,98 @@ static bool _webui_str_to_wide(const char *s, wchar_t **w); static const char* webui_sep = "/"; static void * _webui_run_browser_task(void * _arg); #define WEBUI_THREAD_SERVER_START void * _webui_server_thread(void * arg) -#define WEBUI_THREAD_RECEIVE void * _webui_process_thread(void * _arg) +#define WEBUI_THREAD_RECEIVE void * _webui_ws_process_thread(void * _arg) #define WEBUI_THREAD_WEBVIEW void * _webui_webview_thread(void * arg) #define WEBUI_THREAD_MONITOR void * _webui_folder_monitor_thread(void * arg) #define WEBUI_THREAD_RETURN pthread_exit(NULL); #endif static void _webui_init(void); -static bool _webui_show(_webui_window_t * win, const char* content, size_t browser); -static size_t _webui_get_cb_index(char* webui_internal_id); -static size_t _webui_set_cb_index(char* webui_internal_id); +static bool _webui_show(_webui_window_t* win, struct mg_connection* client, const char* content, size_t browser); +static bool _webui_get_cb_index(_webui_window_t* win, char* element, size_t* id); static size_t _webui_get_free_port(void); static void _webui_free_port(size_t port); static char* _webui_get_current_path(void); -static void _webui_ws_send(_webui_window_t * win, char* packet, size_t packets_size); +static void _webui_send_client_ws(_webui_window_t* win, struct mg_connection* client, + size_t client_id, char* packet, size_t packets_size); static void _webui_window_event( - _webui_window_t * win, int event_type, char* element, size_t event_number, char* webui_internal_id + _webui_window_t* win, size_t client_id, int event_type, char* element, size_t event_number ); -static int _webui_cmd_sync(_webui_window_t * win, char* cmd, bool show); -static int _webui_cmd_async(_webui_window_t * win, char* cmd, bool show); -static int _webui_run_browser(_webui_window_t * win, char* cmd); +static int _webui_cmd_sync(_webui_window_t* win, char* cmd, bool show); +static int _webui_cmd_async(_webui_window_t* win, char* cmd, bool show); +static int _webui_run_browser(_webui_window_t* win, char* cmd); static void _webui_clean(void); -static bool _webui_browser_exist(_webui_window_t * win, size_t browser); +static bool _webui_browser_exist(_webui_window_t* win, size_t browser); static const char* _webui_get_temp_path(); static bool _webui_folder_exist(char* folder); static void _webui_delete_folder(char* folder); -static bool _webui_browser_create_new_profile(_webui_window_t * win, size_t browser); -static bool _webui_browser_start_chrome(_webui_window_t * win, const char* address); -static bool _webui_browser_start_edge(_webui_window_t * win, const char* address); -static bool _webui_browser_start_epic(_webui_window_t * win, const char* address); -static bool _webui_browser_start_vivaldi(_webui_window_t * win, const char* address); -static bool _webui_browser_start_brave(_webui_window_t * win, const char* address); -static bool _webui_browser_start_firefox(_webui_window_t * win, const char* address); -static bool _webui_browser_start_yandex(_webui_window_t * win, const char* address); -static bool _webui_browser_start_chromium(_webui_window_t * win, const char* address); -static bool _webui_browser_start(_webui_window_t * win, const char* address, size_t _browser); +static bool _webui_browser_create_new_profile(_webui_window_t* win, size_t browser); +static bool _webui_browser_start_chrome(_webui_window_t* win, const char* address); +static bool _webui_browser_start_edge(_webui_window_t* win, const char* address); +static bool _webui_browser_start_epic(_webui_window_t* win, const char* address); +static bool _webui_browser_start_vivaldi(_webui_window_t* win, const char* address); +static bool _webui_browser_start_brave(_webui_window_t* win, const char* address); +static bool _webui_browser_start_firefox(_webui_window_t* win, const char* address); +static bool _webui_browser_start_yandex(_webui_window_t* win, const char* address); +static bool _webui_browser_start_chromium(_webui_window_t* win, const char* address); +static bool _webui_browser_start(_webui_window_t* win, const char* address, size_t _browser); static long _webui_timer_diff(struct timespec * start, struct timespec* end); -static void _webui_timer_start(_webui_timer_t * t); -static bool _webui_timer_is_end(_webui_timer_t * t, size_t ms); +static void _webui_timer_start(_webui_timer_t* t); +static bool _webui_timer_is_end(_webui_timer_t* t, size_t ms); static void _webui_timer_clock_gettime(struct timespec * spec); -static bool _webui_set_root_folder(_webui_window_t * win, const char* path); -static const char* _webui_generate_js_bridge(_webui_window_t * win); +static bool _webui_set_root_folder(_webui_window_t* win, const char* path); +static const char* _webui_generate_js_bridge(_webui_window_t* win, struct mg_connection* client); static void _webui_free_mem(void * ptr); static size_t _webui_mb(size_t size); -static bool _webui_file_exist_mg(_webui_window_t * win, struct mg_connection * conn); +static bool _webui_file_exist_mg(_webui_window_t* win, struct mg_connection* client); static bool _webui_file_exist(char* path); static void _webui_free_all_mem(void); -static bool _webui_show_window(_webui_window_t * win, const char* content, int type, size_t browser); -static char* _webui_generate_internal_id(_webui_window_t * win, const char* element); +static bool _webui_show_window(_webui_window_t* win, struct mg_connection* client, + const char* content, int type, size_t browser); static bool _webui_is_empty(const char* s); static size_t _webui_strlen(const char* s); static uint16_t _webui_get_run_id(void); static void * _webui_malloc(size_t size); static void _webui_sleep(long unsigned int ms); -static size_t _webui_find_the_best_browser(_webui_window_t * win); +static size_t _webui_find_the_best_browser(_webui_window_t* win); static bool _webui_is_process_running(const char* process_name); static void _webui_panic(char* msg); static void _webui_kill_pid(size_t pid); -static _webui_window_t * _webui_dereference_win_ptr(void * ptr); -static int _webui_get_browser_args(_webui_window_t * win, size_t browser, char* buffer, size_t len); -static void _webui_mutex_init(webui_mutex_t * mutex); -static void _webui_mutex_lock(webui_mutex_t * mutex); -static void _webui_mutex_unlock(webui_mutex_t * mutex); -static void _webui_mutex_destroy(webui_mutex_t * mutex); -static bool _webui_mutex_is_connected(_webui_window_t * win, int update); +static _webui_window_t* _webui_dereference_win_ptr(void * ptr); +static int _webui_get_browser_args(_webui_window_t* win, size_t browser, char* buffer, size_t len); +static void _webui_mutex_init(webui_mutex_t* mutex); +static void _webui_mutex_lock(webui_mutex_t* mutex); +static void _webui_mutex_unlock(webui_mutex_t* mutex); +static void _webui_mutex_destroy(webui_mutex_t* mutex); +static bool _webui_mutex_is_connected(_webui_window_t* win, int update); static bool _webui_mutex_is_exit_now(int update); static bool _webui_mutex_is_webview_update(_webui_window_t* win, int update); -static void _webui_condition_init(webui_condition_t * cond); -static void _webui_condition_wait(webui_condition_t * cond, webui_mutex_t * mutex); -static void _webui_condition_signal(webui_condition_t * cond); -static void _webui_condition_destroy(webui_condition_t * cond); -static void _webui_http_send(struct mg_connection * conn, const char* mime_type, const char* body); -static void _webui_http_send_error_page(struct mg_connection * conn, const char* body, int status); -static int _webui_http_handler(struct mg_connection * conn, void * _win); -static int _webui_ws_connect_handler(const struct mg_connection * conn, void * _win); -static void _webui_ws_ready_handler(struct mg_connection * conn, void * _win); -static int _webui_ws_data_handler(struct mg_connection * conn, int opcode, char* data, size_t datasize, void * _win); -static void _webui_ws_close_handler(const struct mg_connection * conn, void * _win); -static void _webui_receive(_webui_window_t * win, int event_type, void * data, size_t len); -static void _webui_process(_webui_window_t * win, void * ptr, size_t len, size_t recvNum, int event_type); +static void _webui_condition_init(webui_condition_t* cond); +static void _webui_condition_wait(webui_condition_t* cond, webui_mutex_t* mutex); +static void _webui_condition_signal(webui_condition_t* cond); +static void _webui_condition_destroy(webui_condition_t* cond); +static void _webui_http_send(_webui_window_t* win, struct mg_connection* client, + const char* mime_type, const char* body, size_t body_len, bool cache); +static void _webui_http_send_file(_webui_window_t* win, struct mg_connection* client, + const char* mime_type, const char* path, bool cache); +static void _webui_http_send_header(_webui_window_t* win, struct mg_connection* client, + const char* mime_type, size_t body_len, bool cache); +static void _webui_http_send_error(struct mg_connection* client, const char* body, int status); +static int _webui_http_handler(struct mg_connection* client, void * _win); +static int _webui_ws_connect_handler(const struct mg_connection* client, void * _win); +static void _webui_ws_ready_handler(struct mg_connection* client, void * _win); +static int _webui_ws_data_handler(struct mg_connection* client, int opcode, char* data, size_t datasize, void * _win); +static void _webui_ws_close_handler(const struct mg_connection* client, void * _win); +static void _webui_receive(_webui_window_t* win, struct mg_connection* client, int event_type, void * data, size_t len); +static void _webui_ws_process(_webui_window_t* win, struct mg_connection* client, size_t client_id, + void* ptr, size_t len, size_t recvNum, int event_type); +static bool _webui_client_save(_webui_window_t* win, struct mg_connection* client, size_t* client_id); +static bool _webui_client_get_id(_webui_window_t* win, struct mg_connection* client, size_t* client_id); +static void _webui_client_remove(_webui_window_t* win, struct mg_connection* client); static void _webui_remove_firefox_profile_ini(const char* path, const char* profile_name); static bool _webui_is_firefox_ini_profile_exist(const char* path, const char* profile_name); -static void _webui_send(_webui_window_t * win, uint32_t token, uint16_t id, uint8_t cmd, const char* data, size_t len); +static void _webui_send_client(_webui_window_t* win, struct mg_connection *client, + uint16_t id, uint8_t cmd, const char* data, size_t len); +static void _webui_send_all(_webui_window_t* win, uint16_t id, uint8_t cmd, const char* data, size_t len); static uint16_t _webui_get_id(const char* data); static uint32_t _webui_get_token(const char* data); static uint32_t _webui_generate_random_uint32(); @@ -542,6 +569,8 @@ static bool _webui_is_valid_url(const char* url); static bool _webui_port_is_used(size_t port_num); static char* _webui_str_dup(const char* src); static void _webui_bridge_api_handler(webui_event_t* e); +static void _webui_generate_cookie(char* cookie, size_t length); +static bool _webui_check_auth_cookie(_webui_window_t *win, char* full_cookies); // WebView #ifdef _WIN32 // Microsoft Windows @@ -580,7 +609,7 @@ static bool _webui_check_certificate(const char* certificate_pem, const char* pr #ifdef WEBUI_LOG static void _webui_print_hex(const char* data, size_t len); static void _webui_print_ascii(const char* data, size_t len); -static int _webui_http_log(const struct mg_connection * conn, const char* message); +static int _webui_http_log(const struct mg_connection* client, const char* message); #endif static WEBUI_THREAD_SERVER_START; static WEBUI_THREAD_RECEIVE; @@ -589,52 +618,41 @@ static WEBUI_THREAD_MONITOR; // Safe C STD #ifdef _WIN32 -#define WEBUI_TOK(str, delim, context) strtok_s(str, delim, context) -#define WEBUI_FOPEN(file, filename, mode) fopen_s(&file, filename, mode) -#define WEBUI_SPF_DYN(buffer, buffer_size, format, ...) snprintf(buffer, _webui_mb(buffer_size), format, ##__VA_ARGS__) -#define WEBUI_SPF_STATIC(buffer, buffer_size, format, ...) snprintf(buffer, buffer_size, format, ##__VA_ARGS__) -#define WEBUI_SCOPY_DYN(dest, dest_size, src) strcpy_s(dest, _webui_mb(dest_size), src) -#define WEBUI_SCOPY_STATIC(dest, dest_size, src) strcpy_s(dest, dest_size, src) -#define WEBUI_SCAT_DYN(dest, dest_size, src) strcat_s(dest, _webui_mb(dest_size), src) -#define WEBUI_SCAT_STATIC(dest, dest_size, src) strcat_s(dest, dest_size, src) +#define WEBUI_STR_TOK(str, delim, context) strtok_s(str, delim, context) +#define WEBUI_FILE_OPEN(file, filename, mode) fopen_s(&file, filename, mode) +#define WEBUI_SN_PRINTF_DYN(buffer, buffer_size, format, ...) snprintf(buffer, _webui_mb(buffer_size), format, ##__VA_ARGS__) +#define WEBUI_SN_PRINTF_STATIC(buffer, buffer_size, format, ...) snprintf(buffer, buffer_size, format, ##__VA_ARGS__) +#define WEBUI_STR_COPY_DYN(dest, dest_size, src) strcpy_s(dest, _webui_mb(dest_size), src) +#define WEBUI_STR_COPY_STATIC(dest, dest_size, src) strcpy_s(dest, dest_size, src) +#define WEBUI_STR_CAT_DYN(dest, dest_size, src) strcat_s(dest, _webui_mb(dest_size), src) +#define WEBUI_STR_CAT_STATIC(dest, dest_size, src) strcat_s(dest, dest_size, src) #else -#define WEBUI_TOK(str, delim, context) strtok_r(str, delim, context) -#define WEBUI_FOPEN(file, filename, mode) ((file) = fopen(filename, mode)) -#define WEBUI_SPF_DYN(buffer, buffer_size, format, ...) snprintf(buffer, _webui_mb(buffer_size), format, ##__VA_ARGS__) -#define WEBUI_SPF_STATIC(buffer, buffer_size, format, ...) snprintf(buffer, buffer_size, format, ##__VA_ARGS__) -#define WEBUI_SCOPY_DYN(dest, dest_size, src) strncpy(dest, src, _webui_mb(dest_size)) -#define WEBUI_SCOPY_STATIC(dest, dest_size, src) strncpy(dest, src, dest_size) -#define WEBUI_SCAT_DYN(dest, dest_size, src) strncat(dest, src, _webui_mb(dest_size)) -#define WEBUI_SCAT_STATIC(dest, dest_size, src) strncat(dest, src, dest_size) +#define WEBUI_STR_TOK(str, delim, context) strtok_r(str, delim, context) +#define WEBUI_FILE_OPEN(file, filename, mode) ((file) = fopen(filename, mode)) +#define WEBUI_SN_PRINTF_DYN(buffer, buffer_size, format, ...) snprintf(buffer, _webui_mb(buffer_size), format, ##__VA_ARGS__) +#define WEBUI_SN_PRINTF_STATIC(buffer, buffer_size, format, ...) snprintf(buffer, buffer_size, format, ##__VA_ARGS__) +#define WEBUI_STR_COPY_DYN(dest, dest_size, src) strncpy(dest, src, _webui_mb(dest_size)) +#define WEBUI_STR_COPY_STATIC(dest, dest_size, src) strncpy(dest, src, dest_size) +#define WEBUI_STR_CAT_DYN(dest, dest_size, src) strncat(dest, src, _webui_mb(dest_size)) +#define WEBUI_STR_CAT_STATIC(dest, dest_size, src) strncat(dest, src, dest_size) #endif // Assert -#define WEBUI_ASSERT(s) _webui_panic(s); assert(0 && s); +#define WEBUI_ASSERT(s) _webui_panic(s);assert(0 && s); // -- Heap ---------------------------- -static _webui_core_t _webui_core; +static _webui_core_t _webui; static const char* webui_html_served = "
You can't access this content
because it's already in use in
another "
-"window.