| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202 |
- /*
- * lws-api-test-lws_smd
- *
- * Written in 2020 by Andy Green <[email protected]>
- *
- * This file is made available under the Creative Commons CC0 1.0
- * Universal Public Domain Dedication.
- *
- * This api test confirms lws_smd System Message Distribution
- */
- #include <libwebsockets.h>
- #define HAVE_STRUCT_TIMESPEC
- #include <pthread.h>
- #include <signal.h>
- static int interrupted, ok, fail, _exp = 111;
- static lws_sorted_usec_list_t sul;
- struct lws_context *context;
- static pthread_t thread_spam;
- static void
- timeout_cb(lws_sorted_usec_list_t *sul)
- {
- /* We should have completed the test before this fires */
- interrupted = 1;
- lws_cancel_service(context);
- }
- static int
- smd_cb1int(void *opaque, lws_smd_class_t _class, lws_usec_t timestamp,
- void *buf, size_t len)
- {
- #if 0
- lwsl_notice("%s: ts %llu, len %d\n", __func__,
- (unsigned long long)timestamp, (int)len);
- lwsl_hexdump_notice(buf, len);
- #endif
- ok++;
- return 0;
- }
- static int
- smd_cb2int(void *opaque, lws_smd_class_t _class, lws_usec_t timestamp,
- void *buf, size_t len)
- {
- #if 0
- lwsl_notice("%s: ts %llu, len %d\n", __func__,
- (unsigned long long)timestamp, (int)len);
- lwsl_hexdump_notice(buf, len);
- #endif
- ok++;
- return 0;
- }
- static void *
- _thread_spam(void *d)
- {
- int n;
- n = 0;
- while (n++ < 100) {
- if (lws_smd_msg_printf(context, LWSSMDCL_SYSTEM_STATE,
- "{\"s\":\"state\",\"msg\":%d}",
- (unsigned int)n)) {
- lwsl_info("%s: send failed\n", __func__);
- n--;
- }
- #if defined(WIN32)
- Sleep(3);
- #else
- usleep(3000);
- #endif
- }
- #if !defined(WIN32)
- pthread_exit(NULL);
- #endif
- return NULL;
- }
- void sigint_handler(int sig)
- {
- interrupted = 1;
- }
- static int
- system_notify_cb(lws_state_manager_t *mgr, lws_state_notify_link_t *link,
- int current, int target)
- {
- // struct lws_context *context = mgr->parent;
- if (current != LWS_SYSTATE_OPERATIONAL || target != LWS_SYSTATE_OPERATIONAL)
- return 0;
- lwsl_info("%s: operational\n", __func__);
- /*
- * spawn the test thread, it's going to spam 100 messages at 20ms
- * intervals... check we got everything
- */
- if (pthread_create(&thread_spam, NULL, _thread_spam, NULL))
- lwsl_err("%s: failed to create the spamming thread\n", __func__);
- return 0;
- }
- int
- main(int argc, const char **argv)
- {
- lws_state_notify_link_t notifier = { {0}, system_notify_cb, "app" };
- lws_state_notify_link_t *na[] = { ¬ifier, NULL };
- int logs = LLL_USER | LLL_ERR | LLL_WARN | LLL_NOTICE;
- struct lws_context_creation_info info;
- const char *p;
- void *retval;
- /* the normal lws init */
- signal(SIGINT, sigint_handler);
- if ((p = lws_cmdline_option(argc, argv, "-d")))
- logs = atoi(p);
- lws_set_log_level(logs, NULL);
- lwsl_user("LWS API selftest: lws_smd\n");
- memset(&info, 0, sizeof info); /* otherwise uninitialized garbage */
- info.port = CONTEXT_PORT_NO_LISTEN;
- info.options = LWS_SERVER_OPTION_DO_SSL_GLOBAL_INIT;
- info.register_notifier_list = na;
- context = lws_create_context(&info);
- if (!context) {
- lwsl_err("lws init failed\n");
- return 1;
- }
- lws_sul_schedule(context, 0, &sul, timeout_cb, 5 * LWS_US_PER_SEC);
- /* register a messaging participant to hear INTERACTION class */
- if (!lws_smd_register(context, NULL, 0, LWSSMDCL_INTERACTION,
- smd_cb1int)) {
- lwsl_err("%s: smd register 1 failed\n", __func__);
- goto bail;
- }
- /* register a messaging participant to hear SYSTEM_STATE class */
- if (!lws_smd_register(context, NULL, 0, LWSSMDCL_SYSTEM_STATE,
- smd_cb2int)) {
- lwsl_err("%s: smd register 2 failed\n", __func__);
- goto bail;
- }
- /* generate an INTERACTION class message */
- if (lws_smd_msg_printf(context, LWSSMDCL_INTERACTION,
- "{\"s\":\"interaction\"}")) {
- lwsl_err("%s: problem sending smd\n", __func__);
- goto bail;
- }
- /* generate a SYSTEM_STATE class message */
- if (lws_smd_msg_printf(context, LWSSMDCL_SYSTEM_STATE,
- "{\"s\":\"state\"}")) {
- lwsl_err("%s: problem sending smd\n", __func__);
- goto bail;
- }
- /* no participant listens for this class, so it should be skipped */
- if (lws_smd_msg_printf(context, LWSSMDCL_NETWORK, "{\"s\":\"network\"}")) {
- lwsl_err("%s: problem sending smd\n", __func__);
- goto bail;
- }
- /* the usual lws event loop */
- while (!interrupted && lws_service(context, 0) >= 0)
- ;
- pthread_join(thread_spam, &retval);
- bail:
- lws_context_destroy(context);
- if (fail || ok >= _exp)
- lwsl_user("Completed: PASS: %d / %d, FAIL: %d\n", ok, _exp,
- fail);
- else
- lwsl_user("Completed: ALL PASS: %d / %d\n", ok, _exp);
- return !(ok >= _exp && !fail);
- }
|