chiark / gitweb /
tools: add gdb command to dump hashmap information
[elogind.git] / src / shared / async.c
1 /*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
2
3 /***
4   This file is part of systemd.
5
6   Copyright 2013 Lennart Poettering
7
8   systemd is free software; you can redistribute it and/or modify it
9   under the terms of the GNU Lesser General Public License as published by
10   the Free Software Foundation; either version 2.1 of the License, or
11   (at your option) any later version.
12
13   systemd is distributed in the hope that it will be useful, but
14   WITHOUT ANY WARRANTY; without even the implied warranty of
15   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16   Lesser General Public License for more details.
17
18   You should have received a copy of the GNU Lesser General Public License
19   along with systemd; If not, see <http://www.gnu.org/licenses/>.
20 ***/
21
22 #include <pthread.h>
23 #include <unistd.h>
24
25 #include "async.h"
26 #include "log.h"
27 #include "util.h"
28
29 int asynchronous_job(void* (*func)(void *p), void *arg) {
30         pthread_attr_t a;
31         pthread_t t;
32         int r;
33
34         /* It kinda sucks that we have to resort to threads to
35          * implement an asynchronous sync(), but well, such is
36          * life.
37          *
38          * Note that issuing this command right before exiting a
39          * process will cause the process to wait for the sync() to
40          * complete. This function hence is nicely asynchronous really
41          * only in long running processes. */
42
43         r = pthread_attr_init(&a);
44         if (r != 0)
45                 return -r;
46
47         r = pthread_attr_setdetachstate(&a, PTHREAD_CREATE_DETACHED);
48         if (r != 0) {
49                 r = -r;
50                 goto finish;
51         }
52
53         r = pthread_create(&t, &a, func, arg);
54         if (r != 0) {
55                 r = -r;
56                 goto finish;
57         }
58
59 finish:
60         pthread_attr_destroy(&a);
61         return r;
62 }
63
64 static void *sync_thread(void *p) {
65         sync();
66         return NULL;
67 }
68
69 int asynchronous_sync(void) {
70         log_debug("Spawning new thread for sync");
71
72         return asynchronous_job(sync_thread, NULL);
73 }
74
75 static void *close_thread(void *p) {
76         assert_se(close_nointr(PTR_TO_INT(p)) != -EBADF);
77         return NULL;
78 }
79
80 int asynchronous_close(int fd) {
81         int r;
82
83         /* This is supposed to behave similar to safe_close(), but
84          * actually invoke close() asynchronously, so that it will
85          * never block. Ideally the kernel would have an API for this,
86          * but it doesn't, so we work around it, and hide this as a
87          * far away as we can. */
88
89         if (fd >= 0) {
90                 PROTECT_ERRNO;
91
92                 r = asynchronous_job(close_thread, INT_TO_PTR(fd));
93                 if (r < 0)
94                          assert_se(close_nointr(fd) != -EBADF);
95         }
96
97         return -1;
98 }