aboutsummaryrefslogtreecommitdiffstats
path: root/src/libunixonacid
diff options
context:
space:
mode:
authorLaurent Bercot <ska-skaware@skarnet.org>2026-02-20 20:55:45 +0000
committerLaurent Bercot <ska-skaware@skarnet.org>2026-02-20 20:55:45 +0000
commit48489c67e90a664466ec71c74bda80b1fc0da879 (patch)
tree02a0dd769afa0ac300bbb9fbf586e4dd48850ed0 /src/libunixonacid
parentee9c49369fe58e8159395c5624c654d8ed242a60 (diff)
downloadskalibs-48489c67e90a664466ec71c74bda80b1fc0da879.tar.gz
Add sassclient/sassserver, prepare for 2.14.6.0
Diffstat (limited to 'src/libunixonacid')
-rw-r--r--src/libunixonacid/sassclient-internal.h22
-rw-r--r--src/libunixonacid/sassclient_ack.c24
-rw-r--r--src/libunixonacid/sassclient_cancel.c39
-rw-r--r--src/libunixonacid/sassclient_cancel_internal.c21
-rw-r--r--src/libunixonacid/sassclient_end.c20
-rw-r--r--src/libunixonacid/sassclient_send.c11
-rw-r--r--src/libunixonacid/sassclient_sendv.c63
-rw-r--r--src/libunixonacid/sassclient_start.c36
-rw-r--r--src/libunixonacid/sassclient_update.c58
-rw-r--r--src/libunixonacid/sassserver.c234
10 files changed, 528 insertions, 0 deletions
diff --git a/src/libunixonacid/sassclient-internal.h b/src/libunixonacid/sassclient-internal.h
new file mode 100644
index 0000000..5ddf0f4
--- /dev/null
+++ b/src/libunixonacid/sassclient-internal.h
@@ -0,0 +1,22 @@
+/* ISC license. */
+
+#ifndef SKALIBS_SASSCLIENT_INTERNAL_H
+#define SKALIBS_SASSCLIENT_INTERNAL_H
+
+#include <stddef.h>
+#include <stdint.h>
+
+#include <skalibs/tai.h>
+#include <skalibs/sassclient.h>
+
+typedef struct sassclient_data_s sassclient_data, *sassclient_data_ref ;
+struct sassclient_data_s
+{
+ void *data ;
+ sassclient_cb_func_ref cb ;
+} ;
+
+extern int sassclient_cancel_internal (sassclient *, uint32_t, tain const *, tain *) ;
+#define asyncnss_cancel_internal_g(a, id, deadline) sassclient_cancel_internal(a, id, (deadline), &STAMP)
+
+#endif
diff --git a/src/libunixonacid/sassclient_ack.c b/src/libunixonacid/sassclient_ack.c
new file mode 100644
index 0000000..4fd765f
--- /dev/null
+++ b/src/libunixonacid/sassclient_ack.c
@@ -0,0 +1,24 @@
+/* ISC license. */
+
+#include <errno.h>
+#include <pthread.h>
+
+#include <skalibs/uint32.h>
+#include <skalibs/genqdyn.h>
+#include <skalibs/sassclient.h>
+
+int sassclient_ack (sassclient *a, uint32_t *id, int *status)
+{
+ uint32_t x ;
+ char const *s ;
+ int e = pthread_mutex_lock(&a->results_mutex) ;
+ if (e) return (errno = e, -1) ;
+ if (!genqdyn_n(&a->results)) { pthread_mutex_unlock(&a->results_mutex) ; return 0 ; }
+ s = GENQDYN_PEEK(char const, &a->results) ;
+ uint32_unpack_big(s, id) ;
+ uint32_unpack_big(s + 4, &x) ;
+ genqdyn_pop(&a->results) ;
+ pthread_mutex_unlock(&a->results_mutex) ;
+ *status = x ;
+ return 1 ;
+}
diff --git a/src/libunixonacid/sassclient_cancel.c b/src/libunixonacid/sassclient_cancel.c
new file mode 100644
index 0000000..bc4273e
--- /dev/null
+++ b/src/libunixonacid/sassclient_cancel.c
@@ -0,0 +1,39 @@
+/* ISC license. */
+
+#include <stdint.h>
+#include <errno.h>
+#include <time.h>
+#include <pthread.h>
+
+#include <skalibs/uint32.h>
+#include <skalibs/tai.h>
+#include <skalibs/textclient.h>
+#include <skalibs/gensetdyn.h>
+#include <skalibs/sassclient.h>
+#include "sassclient-internal.h"
+
+int sassclient_cancel (sassclient *a, uint32_t id, tain const *deadline, tain *stamp)
+{
+ int e ;
+
+ {
+ struct timespec ts ;
+ tain diff ;
+ tain_sub(&diff, deadline, stamp) ;
+ if (!timespec_from_tain_relative(&ts, &diff)) return 0 ;
+ e = pthread_mutex_timedlock(&a->connection_mutex, &ts) ;
+// e = pthread_mutex_clocklock(&a->connection_mutex, CLOCK_MONOTONIC, &ts) ;
+ if (e) return (errno = e, 0) ;
+ }
+
+ e = sassclient_cancel_internal(a, id, deadline, stamp) ;
+ if (e) goto err ;
+ gensetdyn_delete(&a->store, id) ;
+ pthread_mutex_unlock(&a->connection_mutex) ;
+ return 1 ;
+
+ err:
+ pthread_mutex_unlock(&a->connection_mutex) ;
+ errno = e ;
+ return 0 ;
+}
diff --git a/src/libunixonacid/sassclient_cancel_internal.c b/src/libunixonacid/sassclient_cancel_internal.c
new file mode 100644
index 0000000..d9fbd5d
--- /dev/null
+++ b/src/libunixonacid/sassclient_cancel_internal.c
@@ -0,0 +1,21 @@
+/* ISC license. */
+
+#include <sys/uio.h>
+#include <stdint.h>
+#include <errno.h>
+
+#include <skalibs/uint32.h>
+#include <skalibs/textclient.h>
+
+#include "sassclient-internal.h"
+
+int asyncnss_cancel_internal (sassclient *a, uint32_t id, tain const *deadline, tain *stamp)
+{
+ struct iovec answer ;
+ char pack[5] = "-" ;
+
+ uint32_pack_big(pack + 1, id) ;
+ if (!textclient_exchange(&a->connection, pack, 5, &answer, deadline, stamp)) return errno ;
+ if (answer.iov_len != 1) return EPROTO ;
+ return *(uint8_t *)answer.iov_base ;
+}
diff --git a/src/libunixonacid/sassclient_end.c b/src/libunixonacid/sassclient_end.c
new file mode 100644
index 0000000..3e8b106
--- /dev/null
+++ b/src/libunixonacid/sassclient_end.c
@@ -0,0 +1,20 @@
+/* ISC license. */
+
+#include <pthread.h>
+
+#include <skalibs/textclient.h>
+#include <skalibs/gensetdyn.h>
+#include <skalibs/genqdyn.h>
+#include <skalibs/sassclient.h>
+
+void sassclient_end (sassclient *a)
+{
+ static sassclient const zero = SASSCLIENT_ZERO ;
+ if (textclient_fd(&a->connection) == -1) return ;
+ textclient_end(&a->connection) ;
+ gensetdyn_free(&a->store) ;
+ genqdyn_free(&a->results) ;
+ pthread_mutex_destroy(&a->connection_mutex) ;
+ pthread_mutex_destroy(&a->results_mutex) ;
+ *a = zero ;
+}
diff --git a/src/libunixonacid/sassclient_send.c b/src/libunixonacid/sassclient_send.c
new file mode 100644
index 0000000..9ede61a
--- /dev/null
+++ b/src/libunixonacid/sassclient_send.c
@@ -0,0 +1,11 @@
+/* ISC license. */
+
+#include <sys/uio.h>
+
+#include <skalibs/sassclient.h>
+
+int sassclient_send (sassclient *a, uint32_t *id, uint32_t timeout, uint32_t opcode, char const *s, size_t len, sassclient_cb_func_ref cb, void *data, tain const *deadline, tain *stamp)
+{
+ struct iovec v = { .iov_base = (char *)s, .iov_len = len } ;
+ return sassclient_sendv(a, id, timeout, opcode, &v, 1, cb, data, deadline, stamp) ;
+}
diff --git a/src/libunixonacid/sassclient_sendv.c b/src/libunixonacid/sassclient_sendv.c
new file mode 100644
index 0000000..ef38679
--- /dev/null
+++ b/src/libunixonacid/sassclient_sendv.c
@@ -0,0 +1,63 @@
+/* ISC license. */
+
+#include <sys/uio.h>
+#include <stdint.h>
+#include <errno.h>
+#include <time.h>
+#include <limits.h>
+#include <pthread.h>
+
+#include <skalibs/uint32.h>
+#include <skalibs/tai.h>
+#include <skalibs/siovec.h>
+#include <skalibs/gensetdyn.h>
+#include <skalibs/textclient.h>
+#include <skalibs/sassclient.h>
+#include "sassclient-internal.h"
+
+int sassclient_sendv (sassclient *a, uint32_t *cid, uint32_t timeout, uint32_t opcode, struct iovec const *vv, unsigned int n, sassclient_cb_func_ref cb, void *data, tain const *deadline, tain *stamp)
+{
+ size_t len = siovec_len(vv, n) ;
+ uint32_t id ;
+ int e ;
+ struct iovec answer ;
+ sassclient_data *p ;
+ char pack[17] = "+" ;
+ struct iovec v[1 + n] ;
+ if (len + 11 > UINT32_MAX) return (errno = ENAMETOOLONG, 0) ;
+
+ {
+ struct timespec ts ;
+ tain diff ;
+ tain_sub(&diff, deadline, stamp) ;
+ if (!timespec_from_tain_relative(&ts, &diff)) return 0 ;
+ e = pthread_mutex_timedlock(&a->connection_mutex, &ts) ;
+// e = pthread_mutex_clocklock(&a->connection_mutex, CLOCK_MONOTONIC, &ts) ;
+ if (e) return (errno = e, 0) ;
+ }
+
+ if (!gensetdyn_new(&a->store, &id)) goto err ;
+ v[0].iov_base = pack ;
+ v[0].iov_len = 17 ;
+ for (unsigned int i = 0 ; i < n ; i++) v[i+1] = vv[i] ;
+ uint32_pack_big(pack + 1, id) ;
+ uint32_pack_big(pack + 5, timeout) ;
+ uint32_pack_big(pack + 9, opcode) ;
+ uint32_pack_big(pack + 13, len) ;
+ if (!textclient_exchangev(&a->connection, v, 1 + n, &answer, deadline, stamp)) { e = errno ; goto err0 ; }
+ if (answer.iov_len == 1) { e = *(unsigned char *)answer.iov_base ; goto err0 ; }
+ if (answer.iov_len != 5 || *(unsigned char *)answer.iov_base) { e = EPROTO ; goto err0 ; }
+ p = GENSETDYN_P(sassclient_data, &a->store, id) ;
+ p->cb = cb ;
+ p->data = data ;
+ pthread_mutex_unlock(&a->connection_mutex) ;
+ *cid = id ;
+ return 1 ;
+
+ err0:
+ gensetdyn_delete(&a->store, id) ;
+ errno = e ;
+ err:
+ pthread_mutex_unlock(&a->connection_mutex) ;
+ return 0 ;
+}
diff --git a/src/libunixonacid/sassclient_start.c b/src/libunixonacid/sassclient_start.c
new file mode 100644
index 0000000..73ad732
--- /dev/null
+++ b/src/libunixonacid/sassclient_start.c
@@ -0,0 +1,36 @@
+/* ISC license. */
+
+#include <string.h>
+#include <errno.h>
+#include <pthread.h>
+
+#include <skalibs/posixplz.h>
+#include <skalibs/textclient.h>
+#include <skalibs/gensetdyn.h>
+#include <skalibs/genqdyn.h>
+#include <skalibs/sassclient.h>
+#include "sassclient-internal.h"
+
+int sassclient_start (sassclient *a, char const *const *argv, char const *banner1, char const *banner2, tain const *deadline, tain *stamp)
+{
+ int e ;
+ if (sassclient_fd(a) >= 0) return (errno = EBUSY, 0) ;
+ e = pthread_mutex_init(&a->connection_mutex, 0) ;
+ if (e) goto err ;
+ e = pthread_mutex_init(&a->results_mutex, 0) ;
+ if (e) goto err0 ;
+ if (!textclient_startf(&a->connection, argv, (char const *const *)environ, TEXTCLIENT_OPTION_WAITPID, banner1, strlen(banner1), banner2, strlen(banner2), deadline, stamp))
+ goto err1 ;
+ gensetdyn_init(&a->store, sizeof(sassclient_data), 8, 3, 8) ;
+ genqdyn_init(&a->results, 8, 1, 10) ;
+ return 1 ;
+
+ err1:
+ e = errno ;
+ pthread_mutex_destroy(&a->results_mutex) ;
+ err0:
+ pthread_mutex_destroy(&a->connection_mutex) ;
+ err:
+ errno = e ;
+ return 0 ;
+}
diff --git a/src/libunixonacid/sassclient_update.c b/src/libunixonacid/sassclient_update.c
new file mode 100644
index 0000000..01a0427
--- /dev/null
+++ b/src/libunixonacid/sassclient_update.c
@@ -0,0 +1,58 @@
+/* ISC license. */
+
+#include <stdint.h>
+#include <string.h>
+#include <errno.h>
+#include <pthread.h>
+
+#include <skalibs/uint32.h>
+#include <skalibs/textclient.h>
+#include <skalibs/gensetdyn.h>
+#include <skalibs/genqdyn.h>
+#include <skalibs/sassclient.h>
+#include "sassclient-internal.h"
+
+#include <skalibs/posixishard.h>
+
+static int sassclient_msghandler (struct iovec const *v, void *aux)
+{
+ sassclient *a = aux ;
+ char const *s = v->iov_base ;
+ int e ;
+ sassclient_data *p ;
+ uint32_t id ;
+ char res[8] ;
+ if (v->iov_len < 8) return (errno = EPROTO, 0) ;
+ uint32_unpack_big(s, &id) ;
+ memcpy(res, s, 4) ; s += 4 ;
+ p = GENSETDYN_P(sassclient_data, &a->store, id) ;
+ if (memcmp(s, "\0\0\0", 4))
+ {
+ if (v->iov_len != 8) return (errno = EPROTO, 0) ;
+ memcpy(res + 4, s, 4) ;
+ }
+ else
+ {
+ e = (*p->cb)(s + 8, v->iov_len - 8, p->data) ;
+ uint32_pack_big(res + 4, e) ;
+ }
+ e = pthread_mutex_lock(&a->results_mutex) ;
+ if (e) return (errno = e, 0) ;
+ if (!genqdyn_push(&a->results, res))
+ {
+ pthread_mutex_unlock(&a->results_mutex) ;
+ return 0 ;
+ }
+ pthread_mutex_unlock(&a->results_mutex) ;
+ if (!gensetdyn_delete(&a->store, id)) return 0 ;
+ return 1 ;
+}
+
+int sassclient_update (sassclient *a)
+{
+ int e = pthread_mutex_lock(&a->connection_mutex) ;
+ if (e) return (errno = e, -1) ;
+ e = textclient_update(&a->connection, &sassclient_msghandler, a) ;
+ pthread_mutex_unlock(&a->connection_mutex) ;
+ return e < 0 ? -1 : !!e ;
+}
diff --git a/src/libunixonacid/sassserver.c b/src/libunixonacid/sassserver.c
new file mode 100644
index 0000000..7ca3ad7
--- /dev/null
+++ b/src/libunixonacid/sassserver.c
@@ -0,0 +1,234 @@
+/* ISC license. */
+
+#include <sys/uio.h>
+#include <string.h>
+#include <stdint.h>
+#include <errno.h>
+
+#include <skalibs/uint32.h>
+#include <skalibs/error.h>
+#include <skalibs/strerr.h>
+#include <skalibs/tai.h>
+#include <skalibs/iopause.h>
+#include <skalibs/gensetdyn.h>
+#include <skalibs/avltree.h>
+#include <skalibs/textmessage.h>
+#include <skalibs/textclient.h>
+#include <skalibs/sassserver.h>
+
+typedef struct sassserver_query_s sassserver_query, sassserver_query_ref ;
+struct sassserver_query_s
+{
+ uint32_t id ;
+ tain deadline ;
+ void *data ;
+} ;
+
+static sassserver_send_func_ref sassserver_sendf ;
+static sassserver_cancel_func_ref sassserver_cancelf ;
+
+static gensetdyn sassserver_queries = GENSETDYN_INIT(sassserver_query, 8, 3, 8) ;
+#define SASSSERVER_QUERY(i) GENSETDYN_P(sassserver_query, &sassserver_queries, (i))
+
+void *sassserver_data (uint32_t handle)
+{
+ return SASSSERVER_QUERY(handle)->data ;
+}
+
+static void *sassserver_deadline_dtok (uint32_t d, void *aux)
+{
+ return &GENSETDYN_P(sassserver_query, (gensetdyn *)aux, d)->deadline ;
+}
+
+static int sassserver_deadline_cmp (void const *a, void const *b, void *aux)
+{
+ tain const *aa = a ;
+ tain const *bb = b ;
+ (void)aux ;
+ return tain_less(aa, bb) ? -1 : tain_less(bb, aa) ;
+}
+
+static avltree sassserver_by_deadline = AVLTREE_INIT(8, 3, 8, &sassserver_deadline_dtok, &sassserver_deadline_cmp, &sassserver_queries) ;
+
+static void *sassserver_id_dtok (uint32_t d, void *aux)
+{
+ return &GENSETDYN_P(sassserver_query, (gensetdyn *)aux, d)->id ;
+}
+
+static int sassserver_id_cmp (void const *a, void const *b, void *aux)
+{
+ uint32_t const *aa = a ;
+ uint32_t const *bb = b ;
+ (void)aux ;
+ return *aa < *bb ? -1 : *aa > *bb ;
+}
+
+static avltree sassserver_by_id = AVLTREE_INIT(8, 3, 8, &sassserver_id_dtok, &sassserver_id_cmp, &sassserver_queries) ;
+
+static void sassserver_sync_answer (int e)
+{
+ char pack[4] ;
+ uint32_pack_big(pack, (uint32_t)e) ;
+ if (!textmessage_put(textmessage_sender_1, pack, 4))
+ strerr_diefu1sys(111, "textmessage_put") ;
+}
+
+static void sassserver_remove (uint32_t handle)
+{
+ sassserver_query *p = SASSSERVER_QUERY(handle) ;
+ avltree_delete(&sassserver_by_deadline, &p->deadline) ;
+ avltree_delete(&sassserver_by_id, &p->id) ;
+ gensetdyn_delete(&sassserver_queries, handle) ;
+}
+
+void sassserver_async_failure (uint32_t handle, int e)
+{
+ sassserver_query *p = SASSSERVER_QUERY(handle) ;
+ char pack[8] ;
+ uint32_pack_big(pack, p->id) ;
+ uint32_pack_big(pack + 4, (uint32_t)e) ;
+ if (!textmessage_put(textmessage_sender_x, pack, 8))
+ strerr_diefu1sys(111, "textmessage_put") ;
+ sassserver_remove(handle) ;
+}
+
+void sassserver_async_successv (uint32_t handle, struct iovec const *v, unsigned int n)
+{
+ sassserver_query *p = SASSSERVER_QUERY(handle) ;
+ char pack[8] = "\0\0\0\0\0\0\0" ;
+ struct iovec vv[n+1] ;
+ vv[0].iov_base = pack ; vv[0].iov_len = 8 ;
+ for (unsigned int i = 0 ; i < n ; i++) vv[i+1] = v[i] ;
+ uint32_pack_big(pack, p->id) ;
+ if (!textmessage_putv(textmessage_sender_x, vv, n+1))
+ strerr_diefu1sys(111, "textmessage_putv") ;
+ sassserver_remove(handle) ;
+}
+
+void sassserver_async_success (uint32_t handle, char const *s, size_t len)
+{
+ struct iovec v = { .iov_base = (char *)s, .iov_len = len } ;
+ sassserver_async_successv(handle, &v, 1) ;
+}
+
+static inline void sassserver_uniquify (tain *deadline)
+{
+ static tain const nanosec = { .sec = TAI_ZERO, .nano = 1 } ;
+ uint32_t dummy ;
+ while (avltree_search(&sassserver_by_deadline, deadline, &dummy))
+ tain_add(deadline, deadline, &nanosec) ;
+}
+
+static int sassserver_parse_protocol (struct iovec const *v, void *aux)
+{
+ char const *s = v->iov_base ;
+ size_t vlen = v->iov_len ;
+ (void)aux ;
+ if (vlen-- < 5) strerr_dief1x(100, "invalid client request") ;
+ switch (*s++)
+ {
+ case '-' : /* cancel */
+ {
+ uint32_t handle, id ;
+ if (vlen != 4) strerr_dief1x(100, "invalid client request") ;
+ uint32_unpack_big(s, &id) ;
+ if (!avltree_search(&sassserver_by_id, &id, &handle)) sassserver_sync_answer(EINVAL) ;
+ (*sassserver_cancelf)(SASSSERVER_QUERY(handle)->data) ;
+ sassserver_remove(handle) ;
+ sassserver_sync_answer(0) ;
+ break ;
+ }
+ case '+' : /* send */
+ {
+ sassserver_query *p ;
+ uint32_t handle ;
+ uint32_t timeout ;
+ uint32_t opcode ;
+ uint32_t len ;
+ if (vlen < 16) strerr_dief1x(100, "invalid client request") ;
+ if (!gensetdyn_new(&sassserver_queries, &handle)) strerr_diefu1sys(111, "gensetdyn_new") ;
+ p = SASSSERVER_QUERY(handle) ;
+ uint32_unpack_big(s, &p->id) ; s += 4 ; vlen -= 4 ;
+ uint32_unpack_big(s, &timeout) ; s += 4 ; vlen -= 4 ;
+ uint32_unpack_big(s, &opcode) ; s += 4 ; vlen -= 4 ;
+ uint32_unpack_big(s, &len) ; s += 4 ; vlen -= 4 ;
+ if (len != vlen) strerr_dief1x(100, "invalid client request") ;
+ if (timeout)
+ {
+ if (!tain_from_millisecs(&p->deadline, timeout)) strerr_dief1x(100, "invalid client request") ;
+ tain_add_g(&p->deadline, &p->deadline) ;
+ }
+ else tain_add_g(&p->deadline, &tain_infinite_relative) ;
+ sassserver_uniquify(&p->deadline) ;
+ if (!avltree_insert(&sassserver_by_deadline, handle)) strerr_diefu1sys(111, "avltree_insert") ;
+ if (!avltree_insert(&sassserver_by_id, handle)) strerr_diefu1sys(111, "avltree_insert") ;
+ p->data = (*sassserver_sendf)(handle, opcode, s, len) ;
+ if (!p->data)
+ {
+ int e = errno ;
+ sassserver_remove(handle) ;
+ sassserver_sync_answer(e) ;
+ }
+ else sassserver_sync_answer(0) ;
+ break ;
+ }
+ default : strerr_dief1x(100, "invalid client request") ;
+ }
+ return 1 ;
+}
+
+void sassserver_init (char const *banner1, char const *banner2, sassserver_send_func_ref sendf, sassserver_cancel_func_ref cancelf, tain const *deadline, tain *stamp)
+{
+ if (!textclient_server_01x_init(banner1, strlen(banner1), banner2, strlen(banner2), deadline, stamp))
+ strerr_diefu1sys(111, "sync with client") ;
+ sassserver_sendf = sendf ;
+ sassserver_cancelf = cancelf ;
+}
+
+unsigned int sassserver_prepare_iopause (iopause_fd *x, tain *deadline)
+{
+ uint32_t i ;
+ if (avltree_min(&sassserver_by_deadline, &i)) tain_earliest1(deadline, &SASSSERVER_QUERY(i)->deadline) ;
+ x[0].fd = 0 ;
+ x[0].events = IOPAUSE_READ ;
+ x[1].fd = 1 ;
+ x[1].events = textmessage_sender_isempty(textmessage_sender_1) ? 0 : IOPAUSE_WRITE ;
+ x[2].fd = textmessage_sender_fd(textmessage_sender_x) ;
+ x[2].events = textmessage_sender_isempty(textmessage_sender_x) ? 0 : IOPAUSE_WRITE ;
+ return 3 ;
+}
+
+void sassserver_timeout (void)
+{
+ uint32_t i ;
+ while (avltree_min(&sassserver_by_deadline, &i))
+ {
+ sassserver_query *p = SASSSERVER_QUERY(i) ;
+ if (tain_future(&p->deadline)) break ;
+ avltree_delete(&sassserver_by_deadline, &p->deadline) ;
+ avltree_delete(&sassserver_by_id, &p->id) ;
+ (*sassserver_cancelf)(p->data) ;
+ sassserver_async_failure(p->id, ETIMEDOUT) ;
+ gensetdyn_delete(&sassserver_queries, i) ;
+ }
+}
+
+int sassserver_event (iopause_fd const *x)
+{
+ if (x[1].revents & IOPAUSE_WRITE)
+ if (!textmessage_sender_flush(textmessage_sender_1) && !error_isagain(errno))
+ strerr_diefu1sys(111, "flush stdout") ;
+ if (x[2].revents & IOPAUSE_WRITE)
+ if (!textmessage_sender_flush(textmessage_sender_x) && !error_isagain(errno))
+ strerr_diefu1sys(111, "flush asyncout") ;
+
+ if (!textmessage_receiver_isempty(textmessage_receiver_0) || x[0].revents & IOPAUSE_READ)
+ {
+ if (textmessage_handle(textmessage_receiver_0, &sassserver_parse_protocol, 0) == -1)
+ {
+ if (errno != EPIPE) strerr_diefu1sys(111, "read messages from client") ;
+ return 1 ;
+ }
+ }
+ return 0 ;
+}