2 common system utilities
4 Copyright (C) Amitay Isaacs 2014
5 Copyright (C) Martin Schwenke 2014
7 This program is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 3 of the License, or
10 (at your option) any later version.
12 This program is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
17 You should have received a copy of the GNU General Public License
18 along with this program; if not, see <http://www.gnu.org/licenses/>.
22 #include "system/filesys.h"
23 #include "system/shmem.h"
27 #include "ctdb_private.h"
38 if possible, make this task real time
40 bool set_scheduler(void)
43 #if HAVE_THREAD_SETSCHED
44 struct thrdentry64 te
;
48 if (getthrds64(getpid(), &te
, sizeof(te
), &ti
, 1) != 1) {
49 DEBUG(DEBUG_ERR
, ("Unable to get thread information\n"));
53 if (thread_setsched(te
.ti_tid
, 0, SCHED_RR
) == -1) {
54 DEBUG(DEBUG_ERR
, ("Unable to set scheduler to SCHED_RR (%s)\n",
62 #if HAVE_SCHED_SETSCHEDULER
64 int policy
= SCHED_FIFO
;
68 #ifdef SCHED_RESET_ON_FORK
69 policy
|= SCHED_RESET_ON_FORK
;
71 if (sched_setscheduler(0, policy
, &p
) == -1) {
72 DEBUG(DEBUG_CRIT
,("Unable to set scheduler to SCHED_FIFO (%s)\n",
80 DEBUG(DEBUG_CRIT
,("No way to set real-time priority.\n"));
85 reset scheduler from real-time to normal scheduling
87 void reset_scheduler(void)
90 #if HAVE_THREAD_SETSCHED
91 struct thrdentry64 te
;
95 if (getthrds64(getpid(), &te
, sizeof(te
), &ti
, 1) != 1) {
96 DEBUG(DEBUG_ERR
, ("Unable to get thread information\n"));
98 if (thread_setsched(te
.ti_tid
, 0, SCHED_OTHER
) == -1) {
99 DEBUG(DEBUG_ERR
, ("Unable to set scheduler to SCHED_OTHER\n"));
103 #if HAVE_SCHED_SETSCHEDULER
104 #ifndef SCHED_RESET_ON_FORK
105 struct sched_param p
;
107 p
.sched_priority
= 0;
108 if (sched_setscheduler(0, SCHED_OTHER
, &p
) == -1) {
109 DEBUG(DEBUG_ERR
, ("Unable to set scheduler to SCHED_OTHER\n"));
116 void set_nonblocking(int fd
)
120 v
= fcntl(fd
, F_GETFL
, 0);
122 DEBUG(DEBUG_WARNING
, ("Failed to get file status flags - %s\n",
126 if (fcntl(fd
, F_SETFL
, v
| O_NONBLOCK
) == -1) {
127 DEBUG(DEBUG_WARNING
, ("Failed to set non_blocking on fd - %s\n",
132 void set_close_on_exec(int fd
)
136 v
= fcntl(fd
, F_GETFD
, 0);
138 DEBUG(DEBUG_WARNING
, ("Failed to get file descriptor flags - %s\n",
142 if (fcntl(fd
, F_SETFD
, v
| FD_CLOEXEC
) != 0) {
143 DEBUG(DEBUG_WARNING
, ("Failed to set close_on_exec on fd - %s\n",
149 bool parse_ipv4(const char *s
, unsigned port
, struct sockaddr_in
*sin
)
151 sin
->sin_family
= AF_INET
;
152 sin
->sin_port
= htons(port
);
154 if (inet_pton(AF_INET
, s
, &sin
->sin_addr
) != 1) {
155 DEBUG(DEBUG_ERR
, (__location__
" Failed to translate %s into sin_addr\n", s
));
159 #ifdef HAVE_SOCK_SIN_LEN
160 sin
->ip
.sin_len
= sizeof(*sin
);
165 static bool parse_ipv6(const char *s
, const char *ifaces
, unsigned port
, ctdb_sock_addr
*saddr
)
167 saddr
->ip6
.sin6_family
= AF_INET6
;
168 saddr
->ip6
.sin6_port
= htons(port
);
169 saddr
->ip6
.sin6_flowinfo
= 0;
170 saddr
->ip6
.sin6_scope_id
= 0;
172 if (inet_pton(AF_INET6
, s
, &saddr
->ip6
.sin6_addr
) != 1) {
173 DEBUG(DEBUG_ERR
, (__location__
" Failed to translate %s into sin6_addr\n", s
));
177 if (ifaces
&& IN6_IS_ADDR_LINKLOCAL(&saddr
->ip6
.sin6_addr
)) {
178 if (strchr(ifaces
, ',')) {
179 DEBUG(DEBUG_ERR
, (__location__
" Link local address %s "
180 "is specified for multiple ifaces %s\n",
184 saddr
->ip6
.sin6_scope_id
= if_nametoindex(ifaces
);
187 #ifdef HAVE_SOCK_SIN_LEN
188 saddr
->ip6
.sin6_len
= sizeof(*saddr
);
196 bool parse_ip(const char *addr
, const char *ifaces
, unsigned port
, ctdb_sock_addr
*saddr
)
201 ZERO_STRUCTP(saddr
); /* valgrind :-) */
203 /* now is this a ipv4 or ipv6 address ?*/
204 p
= index(addr
, ':');
206 ret
= parse_ipv4(addr
, port
, &saddr
->ip
);
208 ret
= parse_ipv6(addr
, ifaces
, port
, saddr
);
217 bool parse_ip_mask(const char *str
, const char *ifaces
, ctdb_sock_addr
*addr
, unsigned *mask
)
220 char s
[64]; /* Much longer than INET6_ADDRSTRLEN */
228 if (len
>= sizeof(s
)) {
229 DEBUG(DEBUG_ERR
, ("Address %s is unreasonably long\n", str
));
233 strncpy(s
, str
, len
+1);
237 DEBUG(DEBUG_ERR
, (__location__
" This addr: %s does not contain a mask\n", s
));
241 *mask
= strtoul(p
+1, &endp
, 10);
242 if (endp
== NULL
|| *endp
!= 0) {
243 /* trailing garbage */
244 DEBUG(DEBUG_ERR
, (__location__
" Trailing garbage after the mask in %s\n", s
));
250 /* now is this a ipv4 or ipv6 address ?*/
251 ret
= parse_ip(s
, ifaces
, 0, addr
);
259 bool parse_ip_port(const char *addr
, ctdb_sock_addr
*saddr
)
262 char s
[64]; /* Much longer than INET6_ADDRSTRLEN */
269 if (len
>= sizeof(s
)) {
270 DEBUG(DEBUG_ERR
, ("Address %s is unreasonably long\n", addr
));
274 strncpy(s
, addr
, len
+1);
278 DEBUG(DEBUG_ERR
, (__location__
" This addr: %s does not contain a port number\n", s
));
282 port
= strtoul(p
+1, &endp
, 10);
283 if (endp
== NULL
|| *endp
!= 0) {
284 /* trailing garbage */
285 DEBUG(DEBUG_ERR
, (__location__
" Trailing garbage after the port in %s\n", s
));
290 /* now is this a ipv4 or ipv6 address ?*/
291 ret
= parse_ip(s
, NULL
, port
, saddr
);
296 /* we don't lock future pages here; it would increase the chance that
297 * we'd fail to mmap later on. */
298 void lockdown_memory(bool valgrinding
)
300 #if defined(HAVE_MLOCKALL) && !defined(_AIX_)
301 /* Extra stack, please! */
303 memset(dummy
, 0, sizeof(dummy
));
309 /* Ignore when running in local daemons mode */
314 /* Avoid compiler optimizing out dummy. */
315 mlock(dummy
, sizeof(dummy
));
316 if (mlockall(MCL_CURRENT
) != 0) {
317 DEBUG(DEBUG_WARNING
,("Failed to lockdown memory: %s'\n",
323 int mkdir_p(const char *dir
, int mode
)
329 if (strcmp(dir
, "/") == 0) {
333 if (strcmp(dir
, ".") == 0) {
337 /* Try to create directory */
338 ret
= mkdir(dir
, mode
);
339 /* Succeed if that worked or if it already existed */
340 if (ret
== 0 || errno
== EEXIST
) {
343 /* Fail on anything else except ENOENT */
344 if (errno
!= ENOENT
) {
348 /* Create ancestors */
350 if (len
>= PATH_MAX
) {
351 errno
= ENAMETOOLONG
;
354 strncpy(t
, dir
, len
+1);
356 ret
= mkdir_p(dirname(t
), mode
);
361 /* Create directory */
362 ret
= mkdir(dir
, mode
);
363 if ((ret
== -1) && (errno
== EEXIST
)) {
370 void mkdir_p_or_die(const char *dir
, int mode
)
374 ret
= mkdir_p(dir
, mode
);
377 ("ctdb exiting with error: "
378 "failed to create directory \"%s\" (%s)\n",
379 dir
, strerror(errno
)));
384 /* A read wrapper that will deal with EINTR. For now, copied from
385 * source3/lib/system.c
387 ssize_t
sys_read(int fd
, void *buf
, size_t count
)
392 ret
= read(fd
, buf
, count
);
393 #if defined(EWOULDBLOCK)
394 } while (ret
== -1 && (errno
== EINTR
|| errno
== EAGAIN
|| errno
== EWOULDBLOCK
));
396 } while (ret
== -1 && (errno
== EINTR
|| errno
== EAGAIN
));
401 /* A write wrapper that will deal with EINTR. For now, copied from
402 * source3/lib/system.c
404 ssize_t
sys_write(int fd
, const void *buf
, size_t count
)
409 ret
= write(fd
, buf
, count
);
410 #if defined(EWOULDBLOCK)
411 } while (ret
== -1 && (errno
== EINTR
|| errno
== EAGAIN
|| errno
== EWOULDBLOCK
));
413 } while (ret
== -1 && (errno
== EINTR
|| errno
== EAGAIN
));