Fast Events (Internal Core changes)

- Changing the way WS events process for more speed
- Changing the WS events process for less data copy as possible
This commit is contained in:
Hassan DRAGA 2023-09-30 23:15:55 -04:00
parent 4eda2b6670
commit 91c9445912
5 changed files with 1603 additions and 1480 deletions

File diff suppressed because it is too large Load Diff

View File

@ -44,8 +44,8 @@ class WebuiBridge {
#closeReason = 0
#closeValue: string
#hasEvents = false
#fnId = 1
#fnPromiseResolve: (((data: string) => unknown) | undefined)[] = []
#callPromiseID = new Uint8Array(1)
#callPromiseResolve: (((data: string) => unknown) | undefined)[] = []
#allowNavigation = false
// WebUI const
@ -224,7 +224,7 @@ class WebuiBridge {
this.#ws.onopen = () => {
this.#wsStatus = true
this.#wsStatusOnce = true
this.#fnId = 1
this.#callPromiseID[0] = 0
if (this.#log) console.log('WebUI -> Connected')
this.#clicksListener()
}
@ -278,14 +278,14 @@ class WebuiBridge {
if (this.#log) {
console.log(`WebUI -> CMD -> Call Response [${callResponse}]`)
}
if (this.#fnPromiseResolve[callId]) {
if (this.#callPromiseResolve[callId]) {
if (this.#log) {
console.log(
`WebUI -> CMD -> Resolving Response #${callId}...`
)
}
this.#fnPromiseResolve[callId]?.(callResponse)
this.#fnPromiseResolve[callId] = undefined
this.#callPromiseResolve[callId]?.(callResponse)
this.#callPromiseResolve[callId] = undefined
}
}
break
@ -490,9 +490,9 @@ class WebuiBridge {
globalThis.close()
}, 1000)
}
#fnPromise(fn: string, value: any) {
if (this.#log) console.log(`WebUI -> Call [${fn}](${value})`)
const callId = this.#fnId++
#callPromise(fn: string, value: any) {
const callId = --this.#callPromiseID[0]
if (this.#log) console.log(`WebUI -> Call [${fn}](${value}) (ID:${this.#callPromiseID[0]})`)
// Response Packet
// 0: [Signature]
@ -513,7 +513,7 @@ class WebuiBridge {
)
return new Promise((resolve) => {
this.#fnPromiseResolve[callId] = resolve
this.#callPromiseResolve[callId] = resolve
this.#ws.send(packet.buffer)
})
}
@ -588,7 +588,7 @@ class WebuiBridge {
)
// Get the binding response
const response = (await this.#fnPromise(
const response = (await this.#callPromise(
bindingName,
payload === undefined
? ''

View File

@ -61,7 +61,11 @@ int main() {
" <body>"
" <h1>WebUI - Call JavaScript from C</h1>"
" <br>"
" <button id=\"MyButton1\">Count <span id=\"count\">0</span></button>"
" <h1 id=\"count\">0</h1>"
" <br>"
" <button id=\"MyButton1\">Manual Count</button>"
" <br>"
" <button id=\"MyTest\" OnClick=\"AutoTest();\">Auto Count (Every 100ms)</button>"
" <br>"
" <button id=\"MyButton2\">Exit</button>"
" <script>"
@ -73,6 +77,9 @@ int main() {
" document.getElementById('count').innerHTML = number;"
" count = number;"
" }"
" function AutoTest(number) {"
" setInterval(function(){ webui.call('MyButton1'); }, 100);"
" }"
" </script>"
" </body>"
"</html>";

View File

@ -448,7 +448,7 @@ WEBUI_EXPORT void webui_navigate(size_t window, const char* url);
WEBUI_EXPORT void webui_clean();
/**
* @brief Delete all local web-browser profile's folder. Should be called at the end.
* @brief Delete all local web-browser profiles folder. It should called at the end.
*
* @example
* webui_wait();

View File

@ -112,13 +112,13 @@ typedef struct _webui_window_t {
// Core
typedef struct _webui_core_t {
volatile size_t servers;
volatile size_t connections;
char* html_elements[WEBUI_MAX_IDS];
size_t used_ports[WEBUI_MAX_IDS];
size_t startup_timeout;
volatile bool exit_now;
const char* run_responses[WEBUI_MAX_IDS];
volatile bool run_done[WEBUI_MAX_IDS];
char* run_userBuffer[WEBUI_MAX_IDS];
size_t run_userBufferLen[WEBUI_MAX_IDS];
bool run_error[WEBUI_MAX_IDS];
unsigned char run_last_id;
bool initialized;
@ -135,7 +135,7 @@ typedef struct _webui_core_t {
bool server_handled;
webui_mutex_t mutex_server_start;
webui_mutex_t mutex_send;
webui_mutex_t mutex_receive;
// webui_mutex_t mutex_receive;
webui_mutex_t mutex_wait;
webui_mutex_t mutex_bridge;
webui_condition_t condition_wait;
@ -154,15 +154,18 @@ typedef struct _webui_cb_arg_t {
char* webui_internal_id;
} _webui_cb_arg_t;
typedef struct _webui_recv_arg_t {
_webui_window_t* win;
const char* packet;
size_t len;
size_t recvNum;
} _webui_recv_arg_t;
typedef struct _webui_cmd_async_t {
_webui_window_t* win;
char* cmd;
} _webui_cmd_async_t;
#ifdef WEBUI_LOG
static size_t ReceiveNum = 0;
#endif
// -- Definitions ---------------------
#ifdef _WIN32
static const char* webui_sep = "\\";
@ -173,17 +176,19 @@ typedef struct _webui_cmd_async_t {
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]);
#define WEBUI_CB DWORD WINAPI _webui_cb(LPVOID _arg)
#define WEBUI_SERVER_START DWORD WINAPI _webui_server_start(LPVOID arg)
#define THREAD_RETURN return 0;
// #define WEBUI_THREAD_CB DWORD WINAPI _webui_cb(LPVOID _arg)
#define WEBUI_THREAD_SERVER_START DWORD WINAPI _webui_server_start(LPVOID arg)
#define WEBUI_THREAD_RECEIVE DWORD WINAPI _webui_receive_thread(LPVOID _arg)
#define WEBUI_THREAD_RETURN return 0;
#else
static const char* webui_sep = "/";
static void* _webui_cb(void* _arg);
static void* _webui_run_browser_task(void* _arg);
#define WEBUI_CB void* _webui_cb(void* _arg)
#define WEBUI_SERVER_START void* _webui_server_start(void* arg)
#define THREAD_RETURN pthread_exit(NULL);
// #define WEBUI_THREAD_CB void* _webui_cb(void* _arg)
#define WEBUI_THREAD_SERVER_START void* _webui_server_start(void* arg)
#define WEBUI_THREAD_RECEIVE void* _webui_receive_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);
@ -192,7 +197,7 @@ static size_t _webui_set_cb_index(char* webui_internal_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_window_receive(_webui_window_t* win, const char* packet, size_t len);
static void _webui_window_receive(_webui_window_t* win, const char* packet, size_t len, size_t recvNum);
static void _webui_window_send(_webui_window_t* win, char* packet, size_t packets_size);
static void _webui_window_event(_webui_window_t* win, int event_type, char* element, char* data, size_t event_number, char* webui_internal_id);
static int _webui_cmd_sync(_webui_window_t* win, char* cmd, bool show);
@ -256,8 +261,9 @@ static int _webui_ws_connect_handler(const struct mg_connection *conn, void *_wi
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 WEBUI_SERVER_START;
static WEBUI_CB;
static WEBUI_THREAD_SERVER_START;
// static WEBUI_THREAD_CB;
static WEBUI_THREAD_RECEIVE;
// -- Heap ----------------------------
static _webui_core_t _webui_core;
@ -292,9 +298,9 @@ void webui_run(size_t window, const char* script) {
unsigned char run_id = _webui_get_run_id();
_webui_core.run_done[run_id] = false;
_webui_core.run_error[run_id] = false;
if((void*)_webui_core.run_responses[run_id] != NULL)
_webui_free_mem((void*)_webui_core.run_responses[run_id]);
_webui_core.run_userBuffer[run_id] = NULL;
_webui_core.run_userBufferLen[run_id] = 0;
// Prepare the packet
size_t packet_len = 3 + js_len; // [header][js]
char* packet = (char*) _webui_malloc(packet_len);
@ -342,7 +348,7 @@ bool webui_script(size_t window, const char* script, size_t timeout_second, char
// Initializing response buffer
if(buffer_length > 0)
memset(buffer, 0, buffer_length);
memset(buffer, 0, buffer_length);
size_t js_len = _webui_strlen(script);
@ -353,8 +359,8 @@ bool webui_script(size_t window, const char* script, size_t timeout_second, char
unsigned char run_id = _webui_get_run_id();
_webui_core.run_done[run_id] = false;
_webui_core.run_error[run_id] = false;
if((void*)_webui_core.run_responses[run_id] != NULL)
_webui_free_mem((void*)_webui_core.run_responses[run_id]);
_webui_core.run_userBuffer[run_id] = buffer;
_webui_core.run_userBufferLen[run_id] = buffer_length;
// 0: [Signature]
// 1: [CMD]
@ -379,7 +385,7 @@ bool webui_script(size_t window, const char* script, size_t timeout_second, char
// Wait forever
for(;;) {
_webui_sleep(10);
_webui_sleep(1);
if(_webui_core.run_done[run_id])
break;
}
@ -390,7 +396,7 @@ bool webui_script(size_t window, const char* script, size_t timeout_second, char
_webui_timer_t timer;
_webui_timer_start(&timer);
for(;;) {
_webui_sleep(10);
_webui_sleep(1);
if(_webui_core.run_done[run_id])
break;
if(_webui_timer_is_end(&timer, (timeout_second * 1000)))
@ -398,25 +404,21 @@ bool webui_script(size_t window, const char* script, size_t timeout_second, char
}
}
if(_webui_core.run_responses[run_id] != NULL) {
if(_webui_core.run_done[run_id]) {
#ifdef WEBUI_LOG
printf("[User] webui_script -> Response found [%s] \n", _webui_core.run_responses[run_id]);
printf("[User] webui_script -> Response found. User buffer len: %zu bytes \n", _webui_core.run_userBufferLen[run_id]);
printf("[User] webui_script -> Response found. User buffer data: [%s] \n", _webui_core.run_userBuffer[run_id]);
#endif
// Response found
if(buffer != NULL && buffer_length > 1) {
// Copy response to the user's response buffer
size_t response_len = _webui_strlen(_webui_core.run_responses[run_id]) + 1;
size_t bytes_to_cpy = (response_len <= buffer_length ? response_len : buffer_length);
memcpy(buffer, _webui_core.run_responses[run_id], bytes_to_cpy);
}
_webui_free_mem((void*)_webui_core.run_responses[run_id]);
return _webui_core.run_error[run_id];
}
else {
#ifdef WEBUI_LOG
printf("[User] webui_script -> No Response is found. \n");
#endif
}
return false;
}
@ -1880,7 +1882,7 @@ static void _webui_ptr_add(void* ptr, size_t size) {
if(_webui_core.ptr_list[i] == NULL) {
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_ptr_add(0x%p) -> Allocate %zu bytes\n", ptr, size);
// printf("[Core]\t\t_webui_ptr_add(0x%p) -> Allocate %zu bytes\n", ptr, size);
#endif
_webui_core.ptr_list[i] = ptr;
@ -1890,7 +1892,7 @@ static void _webui_ptr_add(void* ptr, size_t size) {
}
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_ptr_add(0x%p) -> Allocate %zu bytes\n", ptr, size);
// printf("[Core]\t\t_webui_ptr_add(0x%p) -> Allocate %zu bytes\n", ptr, size);
#endif
_webui_core.ptr_list[_webui_core.ptr_position] = ptr;
@ -1994,7 +1996,7 @@ static size_t _webui_round_to_memory_block(size_t size) {
static void* _webui_malloc(size_t size) {
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_malloc([%zu])...\n", size);
// printf("[Core]\t\t_webui_malloc([%zu])...\n", size);
#endif
// Make sure we have the null
@ -3694,7 +3696,7 @@ static void _webui_clean(void) {
// Destroy all mutex
_webui_mutex_destroy(&_webui_core.mutex_server_start);
_webui_mutex_destroy(&_webui_core.mutex_send);
_webui_mutex_destroy(&_webui_core.mutex_receive);
// _webui_mutex_destroy(&_webui_core.mutex_receive);
_webui_mutex_destroy(&_webui_core.mutex_wait);
_webui_condition_destroy(&_webui_core.condition_wait);
@ -4134,9 +4136,13 @@ static bool _webui_browser_start(_webui_window_t* win, const char* address, size
#endif
// Non existing browser
if(browser > 11)
if(browser > 12)
return false;
// No browser mode
if(browser == NoBrowser)
return true;
// Current browser used in the last opened window
if(browser == AnyBrowser && _webui_core.current_browser != 0)
browser = _webui_core.current_browser;
@ -4405,7 +4411,6 @@ static bool _webui_show(_webui_window_t* win, const char* content, size_t browse
// Some wrappers do not guarantee pointers stay valid,
// so, let's make our copy.
size_t content_len = _webui_strlen(content);
const char* content_cpy = (const char*)_webui_malloc(content_len);
memcpy((char*)content_cpy, content, content_len);
@ -4546,32 +4551,83 @@ static void _webui_window_event(_webui_window_t* win, int event_type, char* elem
printf("[Core]\t\t_webui_window_event([%s], [%s])...\n", webui_internal_id, element);
#endif
// Create a thread, and call the used cb function
// no need to wait for the response. This is fire
// and forget.
// New Event
webui_event_t e;
e.window = win->window_number;
e.event_type = event_type;
e.element = element;
e.data = data;
e.event_number = event_number;
// Create a new CB args struct
_webui_cb_arg_t* arg = (_webui_cb_arg_t*) _webui_malloc(sizeof(_webui_cb_arg_t));
// Event
arg->window = win;
arg->event_type = event_type;
arg->element = element;
arg->data = data;
arg->event_number = event_number;
// Extras
arg->webui_internal_id = webui_internal_id;
// Check for all events-bind functions
if(!_webui_core.exit_now && win->has_events) {
// fire and forget.
#ifdef _WIN32
HANDLE user_fun_thread = CreateThread(NULL, 0, _webui_cb, (void*)arg, 0, NULL);
if(user_fun_thread != NULL)
CloseHandle(user_fun_thread);
#else
pthread_t thread;
pthread_create(&thread, NULL, &_webui_cb, (void*)arg);
pthread_detach(thread);
char* events_id = _webui_generate_internal_id(win, "");
size_t events_cb_index = _webui_get_cb_index(events_id);
_webui_free_mem((void*)events_id);
if(events_cb_index > 0 && _webui_core.cb[events_cb_index] != NULL) {
// Call user all events cb
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_window_event() -> Calling user callback...\n[Call]\n");
#endif
_webui_core.cb[events_cb_index](&e);
}
}
if(!_webui_core.exit_now) {
// Check for the regular bind functions
if(!_webui_is_empty(element)) {
size_t cb_index = _webui_get_cb_index(webui_internal_id);
if(cb_index > 0 && _webui_core.cb[cb_index] != NULL) {
// Call user cb
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_window_event() -> Calling user callback...\n[Call]\n");
#endif
_webui_core.cb[cb_index](&e);
}
}
}
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_window_event() -> Finished.\n");
#endif
// Free event
_webui_free_mem((void*)webui_internal_id);
/*
// Create a thread, and call the used cb function
// no need to wait for the response. This is fire
// and forget.
// Create a new CB args struct
_webui_cb_arg_t* arg = (_webui_cb_arg_t*) _webui_malloc(sizeof(_webui_cb_arg_t));
// Event
arg->window = win;
arg->event_type = event_type;
arg->element = element;
arg->data = data;
arg->event_number = event_number;
// Extras
arg->webui_internal_id = webui_internal_id;
// fire and forget.
#ifdef _WIN32
HANDLE thread = CreateThread(NULL, 0, _webui_cb, (void*)arg, 0, NULL);
if(thread != NULL)
CloseHandle(thread);
#else
pthread_t thread;
pthread_create(&thread, NULL, &_webui_cb, (void*)arg);
pthread_detach(thread);
#endif
*/
}
static void _webui_window_send(_webui_window_t* win, char* packet, size_t packets_size) {
@ -4614,6 +4670,7 @@ static void _webui_window_send(_webui_window_t* win, char* packet, size_t packet
#endif
}
/*
static bool _webui_get_data(const char* packet, size_t packet_len, size_t pos, size_t* data_len, char** data) {
#ifdef WEBUI_LOG
@ -4667,27 +4724,27 @@ static bool _webui_get_data(const char* packet, size_t packet_len, size_t pos, s
return true;
}
*/
static void _webui_window_receive(_webui_window_t* win, const char* packet, size_t len) {
static void _webui_window_receive(_webui_window_t* win, const char* packet, size_t len, size_t recvNum) {
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_window_receive()...\n");
printf("[Core]\t\t[Thread %zu] _webui_window_receive()...\n", recvNum);
#endif
if((unsigned char) packet[0] != WEBUI_HEADER_SIGNATURE || len < 4 || _webui_core.exit_now)
return;
// Mutex
// wait for previous event to finish
_webui_mutex_lock(&_webui_core.mutex_receive);
// _webui_mutex_lock(&_webui_core.mutex_receive);
// Check if the previous event calls exit()
if(_webui_core.exit_now)
return;
#ifdef WEBUI_LOG
ReceiveNum++;
printf("[Core]\t\t_webui_window_receive() -> #%zu -> Start\n", ReceiveNum);
printf("[Core]\t\t[Thread %zu] _webui_window_receive() -> Start\n", recvNum);
#endif
if((unsigned char) packet[1] == WEBUI_HEADER_CLICK) {
@ -4698,18 +4755,14 @@ static void _webui_window_receive(_webui_window_t* win, const char* packet, size
// 1: [CMD]
// 2: [Element]
// We should copy `element` using `_webui_get_data()` as
// the `packet[]` will be freed by WS after end of this call.
// Get html element id
char* element;
size_t element_len;
_webui_get_data(packet, len, 2, &element_len, &element);
char* element = (char*)&packet[2];
size_t element_len = _webui_strlen(element);
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_window_receive() -> #%zu -> WEBUI_HEADER_CLICK \n", ReceiveNum);
printf("[Core]\t\t_webui_window_receive() -> #%zu -> Element size: %zu bytes \n", ReceiveNum, element_len);
printf("[Core]\t\t_webui_window_receive() -> #%zu -> Element : [%s] \n", ReceiveNum, element);
printf("[Core]\t\t[Thread %zu] _webui_window_receive() -> WEBUI_HEADER_CLICK \n", recvNum);
printf("[Core]\t\t[Thread %zu] _webui_window_receive() -> Element size: %zu bytes \n", recvNum, element_len);
printf("[Core]\t\t[Thread %zu] _webui_window_receive() -> Element : [%s] \n", recvNum, element);
#endif
// Generate WebUI internal id
@ -4741,17 +4794,14 @@ static void _webui_window_receive(_webui_window_t* win, const char* packet, size
// Fatal.
// The pipe ID is not valid
// we can't send the ready signal to webui_script()
_webui_mutex_unlock(&_webui_core.mutex_receive);
// _webui_mutex_unlock(&_webui_core.mutex_receive);
return;
}
// We should copy `data` using `_webui_get_data()` as
// the `packet[]` will be freed by WS after end of this call.
_webui_core.run_done[run_id] = false;
// Get data part
char* data;
size_t data_len;
bool data_status = _webui_get_data(packet, len, 4, &data_len, &data);
char* data = (char*)&packet[4];
size_t data_len = _webui_strlen(data);
// Get js-error
bool error = true;
@ -4759,28 +4809,26 @@ static void _webui_window_receive(_webui_window_t* win, const char* packet, size
error = false;
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_window_receive() -> #%zu -> WEBUI_HEADER_JS \n", ReceiveNum);
printf("[Core]\t\t_webui_window_receive() -> #%zu -> run_id = 0x%02x \n", ReceiveNum, run_id);
printf("[Core]\t\t_webui_window_receive() -> #%zu -> error = 0x%02x \n", ReceiveNum, error);
printf("[Core]\t\t_webui_window_receive() -> #%zu -> %zu bytes of data\n", ReceiveNum, data_len);
printf("[Core]\t\t_webui_window_receive() -> #%zu -> data = [%s] @ 0x%p\n", ReceiveNum, data, data);
printf("[Core]\t\t[Thread %zu] _webui_window_receive() -> WEBUI_HEADER_JS \n", recvNum);
printf("[Core]\t\t[Thread %zu] _webui_window_receive() -> run_id = 0x%02x \n", recvNum, run_id);
printf("[Core]\t\t[Thread %zu] _webui_window_receive() -> error = 0x%02x \n", recvNum, error);
printf("[Core]\t\t[Thread %zu] _webui_window_receive() -> %zu bytes of data\n", recvNum, data_len);
printf("[Core]\t\t[Thread %zu] _webui_window_receive() -> data = [%s] @ 0x%p\n", recvNum, data, data);
#endif
// Initialize pipe
if((void*)_webui_core.run_responses[run_id] != NULL)
_webui_free_mem((void*)_webui_core.run_responses[run_id]);
// Set pipe
if(data_status && data_len > 0) {
_webui_core.run_error[run_id] = error;
if(data_len > 0) {
_webui_core.run_error[run_id] = error;
_webui_core.run_responses[run_id] = data;
// Copy response to the user's response buffer directly
size_t response_len = data_len + 1;
size_t bytes_to_cpy = (response_len <= _webui_core.run_userBufferLen[run_id] ? response_len : _webui_core.run_userBufferLen[run_id]);
memcpy(_webui_core.run_userBuffer[run_id], data, bytes_to_cpy);
}
else {
// Empty Result
_webui_core.run_error[run_id] = error;
_webui_core.run_responses[run_id] = "";
memcpy(_webui_core.run_userBuffer[run_id], 0x00, 1);
}
// Send ready signal to webui_script()
@ -4797,22 +4845,14 @@ static void _webui_window_receive(_webui_window_t* win, const char* packet, size
// Events
if(win->has_events) {
// We should copy `url` using `_webui_get_data()` as
// the `packet[]` will be freed by WS after end of this call.
// Get URL
char* url;
size_t url_len;
if(!_webui_get_data(packet, len, 2, &url_len, &url)) {
_webui_mutex_unlock(&_webui_core.mutex_receive);
return;
}
char* url = (char*)&packet[2];
size_t url_len = _webui_strlen(url);
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_window_receive() -> #%zu -> WEBUI_HEADER_NAVIGATION \n", ReceiveNum);
printf("[Core]\t\t_webui_window_receive() -> #%zu -> URL size: %zu bytes \n", ReceiveNum, url_len);
printf("[Core]\t\t_webui_window_receive() -> #%zu -> URL: [%s] \n", ReceiveNum, url);
printf("[Core]\t\t[Thread %zu] _webui_window_receive() -> WEBUI_HEADER_NAVIGATION \n", recvNum);
printf("[Core]\t\t[Thread %zu] _webui_window_receive() -> URL size: %zu bytes \n", recvNum, url_len);
printf("[Core]\t\t[Thread %zu] _webui_window_receive() -> URL: [%s] \n", recvNum, url);
#endif
// Generate WebUI internal id
@ -4836,9 +4876,6 @@ static void _webui_window_receive(_webui_window_t* win, const char* packet, size
// 1: [CMD]
// 2: [Call ID]
// 3: [Element ID, Null, Len, Null, Data, Null]
// No need to copy `element` and `data` using `_webui_get_data()` as
// we don't need the `packet[]` after the end of this call.
// Get html element id
char* element = (char*)&packet[3];
@ -4855,7 +4892,7 @@ static void _webui_window_receive(_webui_window_t* win, const char* packet, size
len = strtoll((const char*)len_s, &endptr, 10);
if(len < 0) {
_webui_mutex_unlock(&_webui_core.mutex_receive);
// _webui_mutex_unlock(&_webui_core.mutex_receive);
return;
}
}
@ -4864,14 +4901,14 @@ static void _webui_window_receive(_webui_window_t* win, const char* packet, size
char* data = (char*)&packet[3 + element_strlen + 1 + len_s_strlen + 1];
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_window_receive() -> #%zu -> WEBUI_HEADER_CALL_FUNC \n", ReceiveNum);
printf("[Core]\t\t_webui_window_receive() -> #%zu -> Call ID: [0x%02x] \n", ReceiveNum, packet[2]);
printf("[Core]\t\t_webui_window_receive() -> #%zu -> Element: [%s] \n", ReceiveNum, element);
printf("[Core]\t\t_webui_window_receive() -> #%zu -> Data size: %lld Bytes \n", ReceiveNum, len);
printf("[Core]\t\t_webui_window_receive() -> #%zu -> Data: Hex [ ", ReceiveNum);
printf("[Core]\t\t[Thread %zu] _webui_window_receive() -> WEBUI_HEADER_CALL_FUNC \n", recvNum);
printf("[Core]\t\t[Thread %zu] _webui_window_receive() -> Call ID: [0x%02x] \n", recvNum, packet[2]);
printf("[Core]\t\t[Thread %zu] _webui_window_receive() -> Element: [%s] \n", recvNum, element);
printf("[Core]\t\t[Thread %zu] _webui_window_receive() -> Data size: %lld Bytes \n", recvNum, len);
printf("[Core]\t\t[Thread %zu] _webui_window_receive() -> Data: Hex [ ", recvNum);
_webui_print_hex(data, len);
printf("]\n");
printf("[Core]\t\t_webui_window_receive() -> #%zu -> Data: ASCII [ ", ReceiveNum);
printf("[Core]\t\t[Thread %zu] _webui_window_receive() -> Data: ASCII [ ", recvNum);
_webui_print_ascii(data, len);
printf("]\n");
#endif
@ -4900,7 +4937,7 @@ static void _webui_window_receive(_webui_window_t* win, const char* packet, size
// Call user cb
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_window_receive() -> #%zu -> Calling user callback...\n[Call]\n", ReceiveNum);
printf("[Core]\t\t[Thread %zu] _webui_window_receive() -> Calling user callback...\n[Call]\n", recvNum);
#endif
_webui_core.cb[cb_index](&e);
}
@ -4910,7 +4947,7 @@ static void _webui_window_receive(_webui_window_t* win, const char* packet, size
*response = (char*)"";
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_window_receive() -> #%zu -> user-callback response [%s]\n", ReceiveNum, *response);
printf("[Core]\t\t[Thread %zu] _webui_window_receive() -> user-callback response [%s]\n", recvNum, *response);
#endif
// 0: [Signature]
@ -4939,10 +4976,10 @@ static void _webui_window_receive(_webui_window_t* win, const char* packet, size
}
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_window_receive() -> #%zu -> End\n", ReceiveNum);
printf("[Core]\t\t[Thread %zu] _webui_window_receive() -> End\n", recvNum);
#endif
_webui_mutex_unlock(&_webui_core.mutex_receive);
// _webui_mutex_unlock(&_webui_core.mutex_receive);
}
static char* _webui_get_current_path(void) {
@ -5044,7 +5081,7 @@ static void _webui_init(void) {
// Initializing mutex
_webui_mutex_init(&_webui_core.mutex_server_start);
_webui_mutex_init(&_webui_core.mutex_send);
_webui_mutex_init(&_webui_core.mutex_receive);
// _webui_mutex_init(&_webui_core.mutex_receive);
_webui_mutex_init(&_webui_core.mutex_wait);
_webui_mutex_init(&_webui_core.mutex_bridge);
_webui_condition_init(&_webui_core.condition_wait);
@ -5477,37 +5514,41 @@ static void _webui_ws_ready_handler(struct mg_connection *conn, void *_win) {
printf("[Core]\t\t_webui_ws_ready_handler()...\n");
#endif
// Dereference
_webui_window_t* win = _webui_dereference_win_ptr(_win);
if(_webui_core.exit_now || win == NULL) return;
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_ws_ready_handler() -> WebSocket Connected\n");
#endif
// Ini
win->connections++;
int event_type = WEBUI_EVENT_CONNECTED;
if(!win->connected) {
if(win->connections < 2) {
// First connection
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_ws_ready_handler() -> WebSocket %zu Connected\n", win->connections);
#endif
win->connected = true; // server thread
_webui_core.connections++; // main loop
_webui_core.mg_connections[win->window_number] = conn; // websocket send func
win->connected = true; // server thread
}
else {
// Multi connections
if(win->multi_access) {
// Multi connections
win->connections++;
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_ws_ready_handler() -> MULTI_CONNECTION -> WebSocket %zu Connected\n", win->connections);
#endif
event_type = WEBUI_EVENT_MULTI_CONNECTION;
}
else {
// UNWANTED Multi connections
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_ws_ready_handler() -> UNWANTED Multi Connections\n");
printf("[Core]\t\t_webui_ws_ready_handler() -> UNWANTED_CONNECTION -> WebSocket %zu Connected\n", win->connections);
#endif
mg_close_connection(conn);
@ -5538,6 +5579,7 @@ static int _webui_ws_data_handler(struct mg_connection *conn, int opcode, char*
printf("[Core]\t\t_webui_ws_data_handler()...\n");
#endif
static size_t recvNum = 0;
if(_webui_core.exit_now || datasize < 1)
return 1; // OK
@ -5554,10 +5596,28 @@ static int _webui_ws_data_handler(struct mg_connection *conn, int opcode, char*
switch (opcode & 0xf) {
case MG_WEBSOCKET_OPCODE_BINARY: {
// Process the packet
_webui_window_t* win = _webui_dereference_win_ptr(_win);
if(win != NULL)
_webui_window_receive(win, data, datasize);
if(win != NULL) {
// Copy packet
const char* data_cpy = (const char*)_webui_malloc(datasize);
memcpy((char*)data_cpy, data, datasize);
// Create args
_webui_recv_arg_t* arg = (_webui_recv_arg_t*) _webui_malloc(sizeof(_webui_recv_arg_t));
arg->win = win;
arg->packet = data_cpy;
arg->len = datasize;
arg->recvNum = ++recvNum;
// Process the packet in a new thread
#ifdef _WIN32
HANDLE thread = CreateThread(NULL, 0, _webui_receive_thread, (void*)arg, 0, NULL);
if(thread != NULL)
CloseHandle(thread);
#else
pthread_t thread;
pthread_create(&thread, NULL, &_webui_receive_thread, (void*)arg);
pthread_detach(thread);
#endif
}
break;
}
case MG_WEBSOCKET_OPCODE_TEXT: {
@ -5585,29 +5645,43 @@ static void _webui_ws_close_handler(const struct mg_connection *conn, void *_win
_webui_window_t* win = _webui_dereference_win_ptr(_win);
if(_webui_core.exit_now || win == NULL || !win->connected) return;
win->html_handled = false;
win->server_handled = false;
win->bridge_handled = false;
if(win->connections < 2) {
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_ws_close_handler() -> WebSocket Closed\n");
#endif
// Main connection close
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_ws_close_handler() -> WebSocket %zu Closed\n", win->connections);
#endif
if(win->connected) {
win->html_handled = false;
win->server_handled = false;
win->bridge_handled = false;
win->connected = false; // server thread
}
else {
if(win->multi_access && win->connections > 0) {
// Multi connections close
// Multi connections close
win->connections--;
if(win->multi_access) {
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_ws_close_handler() -> MULTI_CONNECTION -> WebSocket %zu Closed\n", win->connections);
#endif
win->html_handled = false;
win->server_handled = false;
win->bridge_handled = false;
}
else {
// Main connection close
_webui_core.connections--; // main loop
win->connected = false; // server thread
// UNWANTED Multi connections close
#ifdef WEBUI_LOG
printf("[Core]\t\t_webui_ws_close_handler() -> UNWANTED_CONNECTION -> WebSocket %zu Closed\n", win->connections);
#endif
}
}
win->connections--;
// Events
if(win->has_events) {
@ -5625,7 +5699,7 @@ static void _webui_ws_close_handler(const struct mg_connection *conn, void *_win
}
}
static WEBUI_SERVER_START
static WEBUI_THREAD_SERVER_START
{
#ifdef WEBUI_LOG
printf("[Core]\t\t[Thread] _webui_server_start()...\n");
@ -5636,7 +5710,7 @@ static WEBUI_SERVER_START
_webui_window_t* win = _webui_dereference_win_ptr(arg);
if(win == NULL) {
THREAD_RETURN
WEBUI_THREAD_RETURN
}
#ifdef WEBUI_LOG
@ -5911,10 +5985,11 @@ static WEBUI_SERVER_START
_webui_condition_signal(&_webui_core.condition_wait);
}
THREAD_RETURN
WEBUI_THREAD_RETURN
}
static WEBUI_CB
/*
static WEBUI_THREAD_CB
{
#ifdef WEBUI_LOG
printf("[Core]\t\t[Thread] _webui_cb()...\n");
@ -5975,7 +6050,39 @@ static WEBUI_CB
_webui_free_mem((void*)arg->webui_internal_id);
_webui_free_mem((void*)arg);
THREAD_RETURN
WEBUI_THREAD_RETURN
}
*/
static WEBUI_THREAD_RECEIVE
{
#ifdef WEBUI_LOG
printf("[Core]\t\t[Thread] _webui_receive_thread()...\n");
#endif
// Arguments
_webui_recv_arg_t* arg = (_webui_recv_arg_t*) _arg;
_webui_window_t* win = arg->win;
const char* packet = arg->packet;
size_t len = arg->len;
size_t recvNum = arg->recvNum;
#ifdef WEBUI_LOG
printf("[Core]\t\t[Thread %zu] _webui_receive_thread() -> Start...\n", recvNum);
#endif
// Process the packet in this current thread
_webui_window_receive(win, packet, len, recvNum);
#ifdef WEBUI_LOG
printf("[Core]\t\t[Thread %zu] _webui_receive_thread() -> Finished.\n", recvNum);
#endif
// Free
_webui_free_mem((void*)arg->packet);
_webui_free_mem((void*)arg);
WEBUI_THREAD_RETURN
}
#ifdef _WIN32