224
235
perror_plus("strdup");
238
ret = clock_gettime(CLOCK_MONOTONIC, &(new_server->last_seen));
240
perror_plus("clock_gettime");
227
243
/* Special case of first server */
228
244
if(*current_server == NULL){
229
245
new_server->next = new_server;
230
246
new_server->prev = new_server;
231
247
*current_server = new_server;
232
/* Place the new server last in the list */
249
/* Place the new server last in the list */
234
250
new_server->next = *current_server;
235
251
new_server->prev = (*current_server)->prev;
236
252
new_server->prev->next = new_server;
237
253
(*current_server)->prev = new_server;
239
ret = clock_gettime(CLOCK_MONOTONIC, &(*current_server)->last_seen);
241
perror_plus("clock_gettime");
248
259
* Initialize GPGME.
250
static bool init_gpgme(const char *seckey, const char *pubkey,
251
const char *tempdir, mandos_context *mc){
261
__attribute__((nonnull, warn_unused_result))
262
static bool init_gpgme(const char * const seckey,
263
const char * const pubkey,
264
const char * const tempdir,
252
266
gpgme_error_t rc;
253
267
gpgme_engine_info_t engine_info;
256
270
* Helper function to insert pub and seckey to the engine keyring.
258
bool import_key(const char *filename){
272
bool import_key(const char * const filename){
261
275
gpgme_data_t pgp_data;
779
789
char addrstr[(INET_ADDRSTRLEN > INET6_ADDRSTRLEN) ?
780
790
INET_ADDRSTRLEN : INET6_ADDRSTRLEN] = "";
782
791
if(af == AF_INET6){
783
pcret = inet_ntop(af, &(to.in6.sin6_addr), addrstr,
792
ret = getnameinfo((struct sockaddr *)&to,
793
sizeof(struct sockaddr_in6),
794
addrstr, sizeof(addrstr), NULL, 0,
786
pcret = inet_ntop(af, &(to.in.sin_addr), addrstr,
797
ret = getnameinfo((struct sockaddr *)&to,
798
sizeof(struct sockaddr_in),
799
addrstr, sizeof(addrstr), NULL, 0,
790
perror_plus("inet_ntop");
792
if(strcmp(addrstr, ip) != 0){
793
fprintf_plus(stderr, "Canonical address form: %s\n", addrstr);
802
if(ret == EAI_SYSTEM){
803
perror_plus("getnameinfo");
804
} else if(ret != 0) {
805
fprintf_plus(stderr, "getnameinfo: %s", gai_strerror(ret));
806
} else if(strcmp(addrstr, ip) != 0){
807
fprintf_plus(stderr, "Canonical address form: %s\n", addrstr);
1485
1514
return ret_errno;
1488
bool run_network_hooks(const char *mode, const char *interface,
1519
* Based on the example in the GNU LibC manual chapter 13.13 "File
1520
* Descriptor Flags".
1521
| [[info:libc:Descriptor%20Flags][File Descriptor Flags]] |
1523
__attribute__((warn_unused_result))
1524
static int set_cloexec_flag(int fd){
1525
int ret = (int)TEMP_FAILURE_RETRY(fcntl(fd, F_GETFD, 0));
1526
/* If reading the flags failed, return error indication now. */
1530
/* Store modified flag word in the descriptor. */
1531
return (int)TEMP_FAILURE_RETRY(fcntl(fd, F_SETFD,
1534
#endif /* not O_CLOEXEC */
1536
__attribute__((nonnull))
1537
void run_network_hooks(const char *mode, const char *interface,
1489
1538
const float delay){
1490
1539
struct dirent **direntries;
1540
if(hookdir_fd == -1){
1541
hookdir_fd = open(hookdir, O_RDONLY |
1544
#else /* not O_CLOEXEC */
1546
#endif /* not O_CLOEXEC */
1548
if(hookdir_fd == -1){
1549
if(errno == ENOENT){
1551
fprintf_plus(stderr, "Network hook directory \"%s\" not"
1552
" found\n", hookdir);
1555
perror_plus("open");
1560
if(set_cloexec_flag(hookdir_fd) < 0){
1561
perror_plus("set_cloexec_flag");
1562
if((int)TEMP_FAILURE_RETRY(close(hookdir_fd)) == -1){
1563
perror_plus("close");
1569
#endif /* not O_CLOEXEC */
1572
#if __GLIBC_PREREQ(2, 15)
1573
int numhooks = scandirat(hookdir_fd, ".", &direntries,
1574
runnable_hook, alphasort);
1575
#else /* not __GLIBC_PREREQ(2, 15) */
1576
int numhooks = scandir(hookdir, &direntries, runnable_hook,
1578
#endif /* not __GLIBC_PREREQ(2, 15) */
1579
#else /* not __GLIBC__ */
1580
int numhooks = scandir(hookdir, &direntries, runnable_hook,
1582
#endif /* not __GLIBC__ */
1584
perror_plus("scandir");
1491
1587
struct dirent *direntry;
1493
int numhooks = scandir(hookdir, &direntries, runnable_hook,
1496
if(errno == ENOENT){
1498
fprintf_plus(stderr, "Network hook directory \"%s\" not"
1499
" found\n", hookdir);
1502
perror_plus("scandir");
1589
int devnull = open("/dev/null", O_RDONLY);
1590
for(int i = 0; i < numhooks; i++){
1591
direntry = direntries[i];
1593
fprintf_plus(stderr, "Running network hook \"%s\"\n",
1505
int devnull = open("/dev/null", O_RDONLY);
1506
for(int i = 0; i < numhooks; i++){
1507
direntry = direntries[i];
1508
char *fullname = NULL;
1509
ret = asprintf(&fullname, "%s/%s", hookdir, direntry->d_name);
1596
pid_t hook_pid = fork();
1599
/* Raise privileges */
1600
if(raise_privileges_permanently() != 0){
1601
perror_plus("Failed to raise privileges");
1608
perror_plus("setgid");
1611
/* Reset supplementary groups */
1613
ret = setgroups(0, NULL);
1615
perror_plus("setgroups");
1618
ret = dup2(devnull, STDIN_FILENO);
1620
perror_plus("dup2(devnull, STDIN_FILENO)");
1623
ret = close(devnull);
1625
perror_plus("close");
1628
ret = dup2(STDERR_FILENO, STDOUT_FILENO);
1630
perror_plus("dup2(STDERR_FILENO, STDOUT_FILENO)");
1633
ret = setenv("MANDOSNETHOOKDIR", hookdir, 1);
1635
perror_plus("setenv");
1638
ret = setenv("DEVICE", interface, 1);
1640
perror_plus("setenv");
1643
ret = setenv("VERBOSITY", debug ? "1" : "0", 1);
1645
perror_plus("setenv");
1648
ret = setenv("MODE", mode, 1);
1650
perror_plus("setenv");
1654
ret = asprintf(&delaystring, "%f", (double)delay);
1511
1656
perror_plus("asprintf");
1515
fprintf_plus(stderr, "Running network hook \"%s\"\n",
1518
pid_t hook_pid = fork();
1521
/* Raise privileges */
1522
raise_privileges_permanently();
1527
perror_plus("setgid");
1529
/* Reset supplementary groups */
1531
ret = setgroups(0, NULL);
1533
perror_plus("setgroups");
1535
dup2(devnull, STDIN_FILENO);
1537
dup2(STDERR_FILENO, STDOUT_FILENO);
1538
ret = setenv("MANDOSNETHOOKDIR", hookdir, 1);
1540
perror_plus("setenv");
1543
ret = setenv("DEVICE", interface, 1);
1545
perror_plus("setenv");
1548
ret = setenv("VERBOSITY", debug ? "1" : "0", 1);
1550
perror_plus("setenv");
1553
ret = setenv("MODE", mode, 1);
1555
perror_plus("setenv");
1559
ret = asprintf(&delaystring, "%f", delay);
1561
perror_plus("asprintf");
1564
ret = setenv("DELAY", delaystring, 1);
1567
perror_plus("setenv");
1659
ret = setenv("DELAY", delaystring, 1);
1570
1661
free(delaystring);
1571
if(connect_to != NULL){
1572
ret = setenv("CONNECT", connect_to, 1);
1574
perror_plus("setenv");
1578
if(execl(fullname, direntry->d_name, mode, NULL) == -1){
1579
perror_plus("execl");
1580
_exit(EXIT_FAILURE);
1662
perror_plus("setenv");
1666
if(connect_to != NULL){
1667
ret = setenv("CONNECT", connect_to, 1);
1669
perror_plus("setenv");
1673
if(fexecve(hookdir_fd, (char *const [])
1674
{ direntry->d_name, NULL }, environ) == -1){
1675
perror_plus("fexecve");
1676
_exit(EXIT_FAILURE);
1680
if(TEMP_FAILURE_RETRY(waitpid(hook_pid, &status, 0)) == -1){
1681
perror_plus("waitpid");
1684
if(WIFEXITED(status)){
1685
if(WEXITSTATUS(status) != 0){
1686
fprintf_plus(stderr, "Warning: network hook \"%s\" exited"
1687
" with status %d\n", direntry->d_name,
1688
WEXITSTATUS(status));
1691
} else if(WIFSIGNALED(status)){
1692
fprintf_plus(stderr, "Warning: network hook \"%s\" died by"
1693
" signal %d\n", direntry->d_name,
1584
if(TEMP_FAILURE_RETRY(waitpid(hook_pid, &status, 0)) == -1){
1585
perror_plus("waitpid");
1589
if(WIFEXITED(status)){
1590
if(WEXITSTATUS(status) != 0){
1591
fprintf_plus(stderr, "Warning: network hook \"%s\" exited"
1592
" with status %d\n", direntry->d_name,
1593
WEXITSTATUS(status));
1597
} else if(WIFSIGNALED(status)){
1598
fprintf_plus(stderr, "Warning: network hook \"%s\" died by"
1599
" signal %d\n", direntry->d_name,
1604
fprintf_plus(stderr, "Warning: network hook \"%s\""
1605
" crashed\n", direntry->d_name);
1612
fprintf_plus(stderr, "Network hook \"%s\" ran successfully\n",
1697
fprintf_plus(stderr, "Warning: network hook \"%s\""
1698
" crashed\n", direntry->d_name);
1703
fprintf_plus(stderr, "Network hook \"%s\" ran successfully\n",
1707
if((int)TEMP_FAILURE_RETRY(close(hookdir_fd)) == -1){
1708
perror_plus("close");
1715
__attribute__((nonnull, warn_unused_result))
1621
1716
error_t bring_up_interface(const char *const interface,
1622
1717
const float delay){
1624
1718
error_t old_errno = errno;
1625
error_t ret_errno = 0;
1626
int ret, ret_setflags;
1627
1720
struct ifreq network;
1628
1721
unsigned int if_index = if_nametoindex(interface);
1629
1722
if(if_index == 0){
2449
/* Re-raise priviliges */
2586
/* Re-raise privileges */
2453
/* Run network hooks */
2454
run_network_hooks("stop", interfaces_hooks != NULL ?
2455
interfaces_hooks : "", delay);
2457
/* Take down the network interfaces which were brought up */
2459
char *interface = NULL;
2460
while((interface=argz_next(interfaces_to_take_down,
2461
interfaces_to_take_down_size,
2463
ret_errno = take_down_interface(interface);
2466
perror_plus("Failed to take down interface");
2469
if(debug and (interfaces_to_take_down == NULL)){
2470
fprintf_plus(stderr, "No interfaces needed to be taken"
2475
lower_privileges_permanently();
2588
ret_errno = raise_privileges();
2590
perror_plus("Failed to raise privileges");
2593
/* Run network hooks */
2594
run_network_hooks("stop", interfaces_hooks != NULL ?
2595
interfaces_hooks : "", delay);
2597
/* Take down the network interfaces which were brought up */
2599
char *interface = NULL;
2600
while((interface=argz_next(interfaces_to_take_down,
2601
interfaces_to_take_down_size,
2603
ret_errno = take_down_interface(interface);
2606
perror_plus("Failed to take down interface");
2609
if(debug and (interfaces_to_take_down == NULL)){
2610
fprintf_plus(stderr, "No interfaces needed to be taken"
2616
ret_errno = lower_privileges_permanently();
2618
perror_plus("Failed to lower privileges permanently");
2478
2622
free(interfaces_to_take_down);
2479
2623
free(interfaces_hooks);
2481
2625
/* Removes the GPGME temp directory and all files inside */
2482
if(tempdir_created){
2626
if(tempdir != NULL){
2483
2627
struct dirent **direntries = NULL;
2484
2628
struct dirent *direntry = NULL;
2485
2629
int numentries = scandir(tempdir, &direntries, notdotentries,
2487
if (numentries > 0){
2488
2632
for(int i = 0; i < numentries; i++){
2489
2633
direntry = direntries[i];
2490
2634
char *fullname = NULL;