summaryrefslogtreecommitdiff
path: root/src/lib/dev.c
diff options
context:
space:
mode:
authorDimitri Staessens <[email protected]>2024-02-17 10:19:46 +0100
committerSander Vrijders <[email protected]>2024-02-19 11:49:07 +0100
commit06ee3370998f965b469d1c2859e3e34159c71e20 (patch)
tree93881747a4f1e99f6932231b0cb2358941cb9741 /src/lib/dev.c
parent7bb8aed15c7f29de4d9719acf8db7fdf73731af5 (diff)
downloadouroboros-06ee3370998f965b469d1c2859e3e34159c71e20.tar.gz
ouroboros-06ee3370998f965b469d1c2859e3e34159c71e20.zip
irmd: Revise IRMd internals
This is a full revision of the IRMd internal implementation. The registry is now a proper subcomponent managing its own internal lock (a single mutex). Some tests are added for the registry and its data structures. Some macros for tests are added in <ouroboros/test.h>. Flow allocation is now more symmetric between the client side (alloc) and server size (accept). Each will create a flow in pending state (ALLOC_PENDING/ACCEPT_PENDING) that is potentially fulfilled by an IPCP using respond_alloc and respond_accept primitives. Deallocation is split in flow_dealloc (application side) and ipcp_flow_dealloc (IPCP side) to get the flow in DEALLOC_PENDING and DEALLOCATED state. Cleanup of failed flow allocation is now properly handled instead of relying on the sanitizer thread. The new sanitizer only needs to monitor crashed processes. On shutdown, the IRMd will now detect hanging processes and SIGKILL them and clean up their fuse mountpoints if needed. A lot of other things have been cleaned up and shuffled around a bit. Signed-off-by: Dimitri Staessens <[email protected]> Signed-off-by: Sander Vrijders <[email protected]>
Diffstat (limited to 'src/lib/dev.c')
-rw-r--r--src/lib/dev.c99
1 files changed, 85 insertions, 14 deletions
diff --git a/src/lib/dev.c b/src/lib/dev.c
index 216bf670..9e37978c 100644
--- a/src/lib/dev.c
+++ b/src/lib/dev.c
@@ -233,9 +233,8 @@ static int proc_announce(char * prog)
msg.prog = prog;
recv_msg = send_recv_irm_msg(&msg);
- if (recv_msg == NULL) {
+ if (recv_msg == NULL)
return -EIRMD;
- }
if (!recv_msg->has_result || (ret = recv_msg->result)) {
irm_msg__free_unpacked(recv_msg, NULL);
@@ -247,11 +246,35 @@ static int proc_announce(char * prog)
return ret;
}
+static void proc_exit(void)
+{
+ irm_msg_t msg = IRM_MSG__INIT;
+ irm_msg_t * recv_msg;
+ int ret = -1;
+
+ msg.code = IRM_MSG_CODE__IRM_PROC_EXIT;
+ msg.has_pid = true;
+ msg.pid = getpid();
+
+ recv_msg = send_recv_irm_msg(&msg);
+ if (recv_msg == NULL)
+ return;
+
+ if (!recv_msg->has_result || (ret = recv_msg->result)) {
+ irm_msg__free_unpacked(recv_msg, NULL);
+ return;
+ }
+
+ irm_msg__free_unpacked(recv_msg, NULL);
+
+ return;
+}
+
#include "frct.c"
void * flow_tx(void * o)
{
- struct timespec tic = {0, TICTIME};
+ struct timespec tic = TIMESPEC_INIT_NS(TICTIME);
(void) o;
@@ -347,7 +370,7 @@ static void __cleanup_fqueue_destroy(void * fq)
void * flow_rx(void * o)
{
- struct timespec tic = {0, TICTIME};
+ struct timespec tic = TIMESPEC_INIT_NS(TICTIME);
int ret;
struct fqueue * fq;
@@ -549,7 +572,7 @@ static void init(int argc,
prog = path_strip(prog);
if (prog == NULL) {
- fprintf(stderr, "FATAL: Could not find program name.\n");
+ fprintf(stderr, "FATAL: Could not determine program name.\n");
goto fail_prog;
}
@@ -588,7 +611,7 @@ static void init(int argc,
ai.flows = malloc(sizeof(*ai.flows) * PROG_MAX_FLOWS);
if (ai.flows == NULL) {
- fprintf(stderr, "FATAL: Could not allocate flows.\n");
+ fprintf(stderr, "FATAL: Could not malloc flows.\n");
goto fail_flows;
}
@@ -597,7 +620,7 @@ static void init(int argc,
ai.id_to_fd = malloc(sizeof(*ai.id_to_fd) * SYS_MAX_FLOWS);
if (ai.id_to_fd == NULL) {
- fprintf(stderr, "FATAL: Could not allocate id_to_fd.\n");
+ fprintf(stderr, "FATAL: Could not malloc id_to_fd.\n");
goto fail_id_to_fd;
}
@@ -716,7 +739,6 @@ static void fini(void)
#ifdef PROC_FLOW_STATS
rib_fini();
#endif
-
timerwheel_fini();
fset_destroy(ai.frct_set);
@@ -733,6 +755,8 @@ static void fini(void)
bmp_destroy(ai.fds);
bmp_destroy(ai.fqueues);
+ proc_exit();
+
memset(&ai, 0, sizeof(ai));
}
@@ -946,7 +970,7 @@ int flow_dealloc(int fd)
irm_msg_t msg = IRM_MSG__INIT;
irm_msg_t * recv_msg;
uint8_t buf[128];
- struct timespec tic = {0, TICTIME};
+ struct timespec tic = TIMESPEC_INIT_NS(TICTIME);
struct flow * f;
time_t timeo;
@@ -1027,6 +1051,53 @@ int flow_dealloc(int fd)
return 0;
}
+int ipcp_flow_dealloc(int fd)
+{
+ irm_msg_t msg = IRM_MSG__INIT;
+ irm_msg_t * recv_msg;
+ struct flow * f;
+
+ if (fd < 0 || fd >= SYS_MAX_FLOWS )
+ return -EINVAL;
+
+ msg.code = IRM_MSG_CODE__IPCP_FLOW_DEALLOC;
+ msg.has_pid = true;
+ msg.pid = getpid();
+ msg.has_flow_id = true;
+
+ f = &ai.flows[fd];
+
+ pthread_rwlock_rdlock(&ai.lock);
+
+ if (f->flow_id < 0) {
+ pthread_rwlock_unlock(&ai.lock);
+ return -ENOTALLOC;
+ }
+
+ msg.flow_id = f->flow_id;
+
+ pthread_rwlock_unlock(&ai.lock);
+
+ recv_msg = send_recv_irm_msg(&msg);
+ if (recv_msg == NULL)
+ return -EIRMD;
+
+ if (!recv_msg->has_result) {
+ irm_msg__free_unpacked(recv_msg, NULL);
+ return -EIRMD;
+ }
+
+ irm_msg__free_unpacked(recv_msg, NULL);
+
+ pthread_rwlock_wrlock(&ai.lock);
+
+ flow_fini(fd);
+
+ pthread_rwlock_unlock(&ai.lock);
+
+ return 0;
+}
+
int fccntl(int fd,
int cmd,
...)
@@ -1801,19 +1872,19 @@ int np1_flow_resp(int flow_id)
return fd;
}
-int ipcp_create_r(int result)
+int ipcp_create_r(const struct ipcp_info * info)
{
irm_msg_t msg = IRM_MSG__INIT;
irm_msg_t * recv_msg;
int ret;
msg.code = IRM_MSG_CODE__IPCP_CREATE_R;
- msg.has_pid = true;
- msg.pid = getpid();
- msg.has_result = true;
- msg.result = result;
+ msg.ipcp_info = ipcp_info_s_to_msg(info);
recv_msg = send_recv_irm_msg(&msg);
+
+ ipcp_info_msg__free_unpacked(msg.ipcp_info, NULL);
+
if (recv_msg == NULL)
return -EIRMD;