1
/* -*- coding: utf-8 -*- */
3
* Mandos-client - get and decrypt data from a Mandos server
5
* This program is partly derived from an example program for an Avahi
6
* service browser, downloaded from
7
* <http://avahi.org/browser/examples/core-browse-services.c>. This
8
* includes the following functions: "resolve_callback",
9
* "browse_callback", and parts of "main".
12
* Copyright © 2008-2012 Teddy Hogeborn
13
* Copyright © 2008-2012 Björn Påhlsson
15
* This program is free software: you can redistribute it and/or
16
* modify it under the terms of the GNU General Public License as
17
* published by the Free Software Foundation, either version 3 of the
18
* License, or (at your option) any later version.
20
* This program is distributed in the hope that it will be useful, but
21
* WITHOUT ANY WARRANTY; without even the implied warranty of
22
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
23
* General Public License for more details.
25
* You should have received a copy of the GNU General Public License
26
* along with this program. If not, see
27
* <http://www.gnu.org/licenses/>.
29
* Contact the authors at <mandos@recompile.se>.
32
/* Needed by GPGME, specifically gpgme_data_seek() */
33
#ifndef _LARGEFILE_SOURCE
34
#define _LARGEFILE_SOURCE
36
#ifndef _FILE_OFFSET_BITS
37
#define _FILE_OFFSET_BITS 64
40
#define _GNU_SOURCE /* TEMP_FAILURE_RETRY(), asprintf() */
42
#include <stdio.h> /* fprintf(), stderr, fwrite(),
43
stdout, ferror(), remove() */
44
#include <stdint.h> /* uint16_t, uint32_t, intptr_t */
45
#include <stddef.h> /* NULL, size_t, ssize_t */
46
#include <stdlib.h> /* free(), EXIT_SUCCESS, srand(),
48
#include <stdbool.h> /* bool, false, true */
49
#include <string.h> /* memset(), strcmp(), strlen(),
50
strerror(), asprintf(), strcpy() */
51
#include <sys/ioctl.h> /* ioctl */
52
#include <sys/types.h> /* socket(), inet_pton(), sockaddr,
53
sockaddr_in6, PF_INET6,
54
SOCK_STREAM, uid_t, gid_t, open(),
56
#include <sys/stat.h> /* open(), S_ISREG */
57
#include <sys/socket.h> /* socket(), struct sockaddr_in6,
58
inet_pton(), connect() */
59
#include <fcntl.h> /* open() */
60
#include <dirent.h> /* opendir(), struct dirent, readdir()
62
#include <inttypes.h> /* PRIu16, PRIdMAX, intmax_t,
64
#include <errno.h> /* perror(), errno,
65
program_invocation_short_name */
66
#include <time.h> /* nanosleep(), time(), sleep() */
67
#include <net/if.h> /* ioctl, ifreq, SIOCGIFFLAGS, IFF_UP,
68
SIOCSIFFLAGS, if_indextoname(),
69
if_nametoindex(), IF_NAMESIZE */
70
#include <netinet/in.h> /* IN6_IS_ADDR_LINKLOCAL,
71
INET_ADDRSTRLEN, INET6_ADDRSTRLEN
73
#include <unistd.h> /* close(), SEEK_SET, off_t, write(),
74
getuid(), getgid(), seteuid(),
75
setgid(), pause(), _exit() */
76
#include <arpa/inet.h> /* inet_pton(), htons, inet_ntop() */
77
#include <iso646.h> /* not, or, and */
78
#include <argp.h> /* struct argp_option, error_t, struct
79
argp_state, struct argp,
80
argp_parse(), ARGP_KEY_ARG,
81
ARGP_KEY_END, ARGP_ERR_UNKNOWN */
82
#include <signal.h> /* sigemptyset(), sigaddset(),
83
sigaction(), SIGTERM, sig_atomic_t,
85
#include <sysexits.h> /* EX_OSERR, EX_USAGE, EX_UNAVAILABLE,
86
EX_NOHOST, EX_IOERR, EX_PROTOCOL */
87
#include <sys/wait.h> /* waitpid(), WIFEXITED(),
88
WEXITSTATUS(), WTERMSIG() */
89
#include <grp.h> /* setgroups() */
90
#include <argz.h> /* argz_add_sep(), argz_next(),
91
argz_delete(), argz_append(),
92
argz_stringify(), argz_add(),
96
#include <sys/klog.h> /* klogctl() */
97
#endif /* __linux__ */
100
/* All Avahi types, constants and functions
103
#include <avahi-core/core.h>
104
#include <avahi-core/lookup.h>
105
#include <avahi-core/log.h>
106
#include <avahi-common/simple-watch.h>
107
#include <avahi-common/malloc.h>
108
#include <avahi-common/error.h>
111
#include <gnutls/gnutls.h> /* All GnuTLS types, constants and
114
init_gnutls_session(),
116
#include <gnutls/openpgp.h>
117
/* gnutls_certificate_set_openpgp_key_file(),
118
GNUTLS_OPENPGP_FMT_BASE64 */
121
#include <gpgme.h> /* All GPGME types, constants and
124
GPGME_PROTOCOL_OpenPGP,
127
#define BUFFER_SIZE 256
129
#define PATHDIR "/conf/conf.d/mandos"
130
#define SECKEY "seckey.txt"
131
#define PUBKEY "pubkey.txt"
132
#define HOOKDIR "/lib/mandos/network-hooks.d"
135
static const char mandos_protocol_version[] = "1";
136
const char *argp_program_version = "mandos-client " VERSION;
137
const char *argp_program_bug_address = "<mandos@recompile.se>";
138
static const char sys_class_net[] = "/sys/class/net";
139
char *connect_to = NULL;
140
const char *hookdir = HOOKDIR;
144
/* Doubly linked list that need to be circularly linked when used */
145
typedef struct server{
148
AvahiIfIndex if_index;
150
struct timespec last_seen;
155
/* Used for passing in values through the Avahi callback functions */
157
AvahiSimplePoll *simple_poll;
159
gnutls_certificate_credentials_t cred;
160
unsigned int dh_bits;
161
gnutls_dh_params_t dh_params;
162
const char *priority;
164
server *current_server;
167
/* global context so signal handler can reach it*/
168
mandos_context mc = { .simple_poll = NULL, .server = NULL,
169
.dh_bits = 1024, .priority = "SECURE256"
170
":!CTYPE-X.509:+CTYPE-OPENPGP",
171
.current_server = NULL };
173
sig_atomic_t quit_now = 0;
174
int signal_received = 0;
176
/* Function to use when printing errors */
177
void perror_plus(const char *print_text){
179
fprintf(stderr, "Mandos plugin %s: ",
180
program_invocation_short_name);
185
__attribute__((format (gnu_printf, 2, 3)))
186
int fprintf_plus(FILE *stream, const char *format, ...){
188
va_start (ap, format);
190
TEMP_FAILURE_RETRY(fprintf(stream, "Mandos plugin %s: ",
191
program_invocation_short_name));
192
return TEMP_FAILURE_RETRY(vfprintf(stream, format, ap));
196
* Make additional room in "buffer" for at least BUFFER_SIZE more
197
* bytes. "buffer_capacity" is how much is currently allocated,
198
* "buffer_length" is how much is already used.
200
size_t incbuffer(char **buffer, size_t buffer_length,
201
size_t buffer_capacity){
202
if(buffer_length + BUFFER_SIZE > buffer_capacity){
203
*buffer = realloc(*buffer, buffer_capacity + BUFFER_SIZE);
207
buffer_capacity += BUFFER_SIZE;
209
return buffer_capacity;
212
/* Add server to set of servers to retry periodically */
213
bool add_server(const char *ip, in_port_t port, AvahiIfIndex if_index,
216
server *new_server = malloc(sizeof(server));
217
if(new_server == NULL){
218
perror_plus("malloc");
221
*new_server = (server){ .ip = strdup(ip),
223
.if_index = if_index,
225
if(new_server->ip == NULL){
226
perror_plus("strdup");
229
/* Special case of first server */
230
if (mc.current_server == NULL){
231
new_server->next = new_server;
232
new_server->prev = new_server;
233
mc.current_server = new_server;
234
/* Place the new server last in the list */
236
new_server->next = mc.current_server;
237
new_server->prev = mc.current_server->prev;
238
new_server->prev->next = new_server;
239
mc.current_server->prev = new_server;
241
ret = clock_gettime(CLOCK_MONOTONIC, &mc.current_server->last_seen);
243
perror_plus("clock_gettime");
252
static bool init_gpgme(const char *seckey, const char *pubkey,
253
const char *tempdir){
255
gpgme_engine_info_t engine_info;
259
* Helper function to insert pub and seckey to the engine keyring.
261
bool import_key(const char *filename){
264
gpgme_data_t pgp_data;
266
fd = (int)TEMP_FAILURE_RETRY(open(filename, O_RDONLY));
272
rc = gpgme_data_new_from_fd(&pgp_data, fd);
273
if(rc != GPG_ERR_NO_ERROR){
274
fprintf_plus(stderr, "bad gpgme_data_new_from_fd: %s: %s\n",
275
gpgme_strsource(rc), gpgme_strerror(rc));
279
rc = gpgme_op_import(mc.ctx, pgp_data);
280
if(rc != GPG_ERR_NO_ERROR){
281
fprintf_plus(stderr, "bad gpgme_op_import: %s: %s\n",
282
gpgme_strsource(rc), gpgme_strerror(rc));
286
ret = (int)TEMP_FAILURE_RETRY(close(fd));
288
perror_plus("close");
290
gpgme_data_release(pgp_data);
295
fprintf_plus(stderr, "Initializing GPGME\n");
299
gpgme_check_version(NULL);
300
rc = gpgme_engine_check_version(GPGME_PROTOCOL_OpenPGP);
301
if(rc != GPG_ERR_NO_ERROR){
302
fprintf_plus(stderr, "bad gpgme_engine_check_version: %s: %s\n",
303
gpgme_strsource(rc), gpgme_strerror(rc));
307
/* Set GPGME home directory for the OpenPGP engine only */
308
rc = gpgme_get_engine_info(&engine_info);
309
if(rc != GPG_ERR_NO_ERROR){
310
fprintf_plus(stderr, "bad gpgme_get_engine_info: %s: %s\n",
311
gpgme_strsource(rc), gpgme_strerror(rc));
314
while(engine_info != NULL){
315
if(engine_info->protocol == GPGME_PROTOCOL_OpenPGP){
316
gpgme_set_engine_info(GPGME_PROTOCOL_OpenPGP,
317
engine_info->file_name, tempdir);
320
engine_info = engine_info->next;
322
if(engine_info == NULL){
323
fprintf_plus(stderr, "Could not set GPGME home dir to %s\n",
328
/* Create new GPGME "context" */
329
rc = gpgme_new(&(mc.ctx));
330
if(rc != GPG_ERR_NO_ERROR){
331
fprintf_plus(stderr, "Mandos plugin mandos-client: "
332
"bad gpgme_new: %s: %s\n", gpgme_strsource(rc),
337
if(not import_key(pubkey) or not import_key(seckey)){
345
* Decrypt OpenPGP data.
346
* Returns -1 on error
348
static ssize_t pgp_packet_decrypt(const char *cryptotext,
351
gpgme_data_t dh_crypto, dh_plain;
354
size_t plaintext_capacity = 0;
355
ssize_t plaintext_length = 0;
358
fprintf_plus(stderr, "Trying to decrypt OpenPGP data\n");
361
/* Create new GPGME data buffer from memory cryptotext */
362
rc = gpgme_data_new_from_mem(&dh_crypto, cryptotext, crypto_size,
364
if(rc != GPG_ERR_NO_ERROR){
365
fprintf_plus(stderr, "bad gpgme_data_new_from_mem: %s: %s\n",
366
gpgme_strsource(rc), gpgme_strerror(rc));
370
/* Create new empty GPGME data buffer for the plaintext */
371
rc = gpgme_data_new(&dh_plain);
372
if(rc != GPG_ERR_NO_ERROR){
373
fprintf_plus(stderr, "Mandos plugin mandos-client: "
374
"bad gpgme_data_new: %s: %s\n",
375
gpgme_strsource(rc), gpgme_strerror(rc));
376
gpgme_data_release(dh_crypto);
380
/* Decrypt data from the cryptotext data buffer to the plaintext
382
rc = gpgme_op_decrypt(mc.ctx, dh_crypto, dh_plain);
383
if(rc != GPG_ERR_NO_ERROR){
384
fprintf_plus(stderr, "bad gpgme_op_decrypt: %s: %s\n",
385
gpgme_strsource(rc), gpgme_strerror(rc));
386
plaintext_length = -1;
388
gpgme_decrypt_result_t result;
389
result = gpgme_op_decrypt_result(mc.ctx);
391
fprintf_plus(stderr, "gpgme_op_decrypt_result failed\n");
393
fprintf_plus(stderr, "Unsupported algorithm: %s\n",
394
result->unsupported_algorithm);
395
fprintf_plus(stderr, "Wrong key usage: %u\n",
396
result->wrong_key_usage);
397
if(result->file_name != NULL){
398
fprintf_plus(stderr, "File name: %s\n", result->file_name);
400
gpgme_recipient_t recipient;
401
recipient = result->recipients;
402
while(recipient != NULL){
403
fprintf_plus(stderr, "Public key algorithm: %s\n",
404
gpgme_pubkey_algo_name
405
(recipient->pubkey_algo));
406
fprintf_plus(stderr, "Key ID: %s\n", recipient->keyid);
407
fprintf_plus(stderr, "Secret key available: %s\n",
408
recipient->status == GPG_ERR_NO_SECKEY
410
recipient = recipient->next;
418
fprintf_plus(stderr, "Decryption of OpenPGP data succeeded\n");
421
/* Seek back to the beginning of the GPGME plaintext data buffer */
422
if(gpgme_data_seek(dh_plain, (off_t)0, SEEK_SET) == -1){
423
perror_plus("gpgme_data_seek");
424
plaintext_length = -1;
430
plaintext_capacity = incbuffer(plaintext,
431
(size_t)plaintext_length,
433
if(plaintext_capacity == 0){
434
perror_plus("incbuffer");
435
plaintext_length = -1;
439
ret = gpgme_data_read(dh_plain, *plaintext + plaintext_length,
441
/* Print the data, if any */
447
perror_plus("gpgme_data_read");
448
plaintext_length = -1;
451
plaintext_length += ret;
455
fprintf_plus(stderr, "Decrypted password is: ");
456
for(ssize_t i = 0; i < plaintext_length; i++){
457
fprintf(stderr, "%02hhX ", (*plaintext)[i]);
459
fprintf(stderr, "\n");
464
/* Delete the GPGME cryptotext data buffer */
465
gpgme_data_release(dh_crypto);
467
/* Delete the GPGME plaintext data buffer */
468
gpgme_data_release(dh_plain);
469
return plaintext_length;
472
static const char * safer_gnutls_strerror(int value){
473
const char *ret = gnutls_strerror(value);
479
/* GnuTLS log function callback */
480
static void debuggnutls(__attribute__((unused)) int level,
482
fprintf_plus(stderr, "GnuTLS: %s", string);
485
static int init_gnutls_global(const char *pubkeyfilename,
486
const char *seckeyfilename){
490
fprintf_plus(stderr, "Initializing GnuTLS\n");
493
ret = gnutls_global_init();
494
if(ret != GNUTLS_E_SUCCESS){
495
fprintf_plus(stderr, "GnuTLS global_init: %s\n",
496
safer_gnutls_strerror(ret));
501
/* "Use a log level over 10 to enable all debugging options."
504
gnutls_global_set_log_level(11);
505
gnutls_global_set_log_function(debuggnutls);
508
/* OpenPGP credentials */
509
ret = gnutls_certificate_allocate_credentials(&mc.cred);
510
if(ret != GNUTLS_E_SUCCESS){
511
fprintf_plus(stderr, "GnuTLS memory error: %s\n",
512
safer_gnutls_strerror(ret));
513
gnutls_global_deinit();
518
fprintf_plus(stderr, "Attempting to use OpenPGP public key %s and"
519
" secret key %s as GnuTLS credentials\n",
524
ret = gnutls_certificate_set_openpgp_key_file
525
(mc.cred, pubkeyfilename, seckeyfilename,
526
GNUTLS_OPENPGP_FMT_BASE64);
527
if(ret != GNUTLS_E_SUCCESS){
529
"Error[%d] while reading the OpenPGP key pair ('%s',"
530
" '%s')\n", ret, pubkeyfilename, seckeyfilename);
531
fprintf_plus(stderr, "The GnuTLS error is: %s\n",
532
safer_gnutls_strerror(ret));
536
/* GnuTLS server initialization */
537
ret = gnutls_dh_params_init(&mc.dh_params);
538
if(ret != GNUTLS_E_SUCCESS){
539
fprintf_plus(stderr, "Error in GnuTLS DH parameter"
540
" initialization: %s\n",
541
safer_gnutls_strerror(ret));
544
ret = gnutls_dh_params_generate2(mc.dh_params, mc.dh_bits);
545
if(ret != GNUTLS_E_SUCCESS){
546
fprintf_plus(stderr, "Error in GnuTLS prime generation: %s\n",
547
safer_gnutls_strerror(ret));
551
gnutls_certificate_set_dh_params(mc.cred, mc.dh_params);
557
gnutls_certificate_free_credentials(mc.cred);
558
gnutls_global_deinit();
559
gnutls_dh_params_deinit(mc.dh_params);
563
static int init_gnutls_session(gnutls_session_t *session){
565
/* GnuTLS session creation */
567
ret = gnutls_init(session, GNUTLS_SERVER);
571
} while(ret == GNUTLS_E_INTERRUPTED or ret == GNUTLS_E_AGAIN);
572
if(ret != GNUTLS_E_SUCCESS){
574
"Error in GnuTLS session initialization: %s\n",
575
safer_gnutls_strerror(ret));
581
ret = gnutls_priority_set_direct(*session, mc.priority, &err);
583
gnutls_deinit(*session);
586
} while(ret == GNUTLS_E_INTERRUPTED or ret == GNUTLS_E_AGAIN);
587
if(ret != GNUTLS_E_SUCCESS){
588
fprintf_plus(stderr, "Syntax error at: %s\n", err);
589
fprintf_plus(stderr, "GnuTLS error: %s\n",
590
safer_gnutls_strerror(ret));
591
gnutls_deinit(*session);
597
ret = gnutls_credentials_set(*session, GNUTLS_CRD_CERTIFICATE,
600
gnutls_deinit(*session);
603
} while(ret == GNUTLS_E_INTERRUPTED or ret == GNUTLS_E_AGAIN);
604
if(ret != GNUTLS_E_SUCCESS){
605
fprintf_plus(stderr, "Error setting GnuTLS credentials: %s\n",
606
safer_gnutls_strerror(ret));
607
gnutls_deinit(*session);
611
/* ignore client certificate if any. */
612
gnutls_certificate_server_set_request(*session, GNUTLS_CERT_IGNORE);
614
gnutls_dh_set_prime_bits(*session, mc.dh_bits);
619
/* Avahi log function callback */
620
static void empty_log(__attribute__((unused)) AvahiLogLevel level,
621
__attribute__((unused)) const char *txt){}
623
/* Called when a Mandos server is found */
624
static int start_mandos_communication(const char *ip, in_port_t port,
625
AvahiIfIndex if_index,
627
int ret, tcp_sd = -1;
630
struct sockaddr_in in;
631
struct sockaddr_in6 in6;
634
char *decrypted_buffer = NULL;
635
size_t buffer_length = 0;
636
size_t buffer_capacity = 0;
639
gnutls_session_t session;
640
int pf; /* Protocol family */
657
fprintf_plus(stderr, "Bad address family: %d\n", af);
662
ret = init_gnutls_session(&session);
668
fprintf_plus(stderr, "Setting up a TCP connection to %s, port %"
669
PRIuMAX "\n", ip, (uintmax_t)port);
672
tcp_sd = socket(pf, SOCK_STREAM, 0);
675
perror_plus("socket");
685
memset(&to, 0, sizeof(to));
687
to.in6.sin6_family = (sa_family_t)af;
688
ret = inet_pton(af, ip, &to.in6.sin6_addr);
690
to.in.sin_family = (sa_family_t)af;
691
ret = inet_pton(af, ip, &to.in.sin_addr);
695
perror_plus("inet_pton");
701
fprintf_plus(stderr, "Bad address: %s\n", ip);
706
to.in6.sin6_port = htons(port);
707
if(IN6_IS_ADDR_LINKLOCAL /* Spurious warnings from */
708
(&to.in6.sin6_addr)){ /* -Wstrict-aliasing=2 or lower and
710
if(if_index == AVAHI_IF_UNSPEC){
711
fprintf_plus(stderr, "An IPv6 link-local address is"
712
" incomplete without a network interface\n");
716
/* Set the network interface number as scope */
717
to.in6.sin6_scope_id = (uint32_t)if_index;
720
to.in.sin_port = htons(port); /* Spurious warnings from
722
-Wunreachable-code */
731
if(af == AF_INET6 and if_index != AVAHI_IF_UNSPEC){
732
char interface[IF_NAMESIZE];
733
if(if_indextoname((unsigned int)if_index, interface) == NULL){
734
perror_plus("if_indextoname");
736
fprintf_plus(stderr, "Connection to: %s%%%s, port %" PRIuMAX
737
"\n", ip, interface, (uintmax_t)port);
740
fprintf_plus(stderr, "Connection to: %s, port %" PRIuMAX "\n",
741
ip, (uintmax_t)port);
743
char addrstr[(INET_ADDRSTRLEN > INET6_ADDRSTRLEN) ?
744
INET_ADDRSTRLEN : INET6_ADDRSTRLEN] = "";
747
pcret = inet_ntop(af, &(to.in6.sin6_addr), addrstr,
750
pcret = inet_ntop(af, &(to.in.sin_addr), addrstr,
754
perror_plus("inet_ntop");
756
if(strcmp(addrstr, ip) != 0){
757
fprintf_plus(stderr, "Canonical address form: %s\n", addrstr);
768
ret = connect(tcp_sd, &to.in6, sizeof(to));
770
ret = connect(tcp_sd, &to.in, sizeof(to)); /* IPv4 */
773
if ((errno != ECONNREFUSED and errno != ENETUNREACH) or debug){
775
perror_plus("connect");
786
const char *out = mandos_protocol_version;
789
size_t out_size = strlen(out);
790
ret = (int)TEMP_FAILURE_RETRY(write(tcp_sd, out + written,
791
out_size - written));
794
perror_plus("write");
798
written += (size_t)ret;
799
if(written < out_size){
802
if(out == mandos_protocol_version){
817
fprintf_plus(stderr, "Establishing TLS session with %s\n", ip);
825
/* This casting via intptr_t is to eliminate warning about casting
826
an int to a pointer type. This is exactly how the GnuTLS Guile
827
function "set-session-transport-fd!" does it. */
828
gnutls_transport_set_ptr(session,
829
(gnutls_transport_ptr_t)(intptr_t)tcp_sd);
837
ret = gnutls_handshake(session);
842
} while(ret == GNUTLS_E_AGAIN or ret == GNUTLS_E_INTERRUPTED);
844
if(ret != GNUTLS_E_SUCCESS){
846
fprintf_plus(stderr, "*** GnuTLS Handshake failed ***\n");
853
/* Read OpenPGP packet that contains the wanted password */
856
fprintf_plus(stderr, "Retrieving OpenPGP encrypted password from"
867
buffer_capacity = incbuffer(&buffer, buffer_length,
869
if(buffer_capacity == 0){
871
perror_plus("incbuffer");
881
sret = gnutls_record_recv(session, buffer+buffer_length,
888
case GNUTLS_E_INTERRUPTED:
891
case GNUTLS_E_REHANDSHAKE:
893
ret = gnutls_handshake(session);
899
} while(ret == GNUTLS_E_AGAIN or ret == GNUTLS_E_INTERRUPTED);
901
fprintf_plus(stderr, "*** GnuTLS Re-handshake failed "
909
fprintf_plus(stderr, "Unknown error while reading data from"
910
" encrypted session with Mandos server\n");
911
gnutls_bye(session, GNUTLS_SHUT_RDWR);
916
buffer_length += (size_t) sret;
921
fprintf_plus(stderr, "Closing TLS session\n");
930
ret = gnutls_bye(session, GNUTLS_SHUT_RDWR);
935
} while(ret == GNUTLS_E_AGAIN or ret == GNUTLS_E_INTERRUPTED);
937
if(buffer_length > 0){
938
ssize_t decrypted_buffer_size;
939
decrypted_buffer_size = pgp_packet_decrypt(buffer, buffer_length,
941
if(decrypted_buffer_size >= 0){
944
while(written < (size_t) decrypted_buffer_size){
950
ret = (int)fwrite(decrypted_buffer + written, 1,
951
(size_t)decrypted_buffer_size - written,
953
if(ret == 0 and ferror(stdout)){
956
fprintf_plus(stderr, "Error writing encrypted data: %s\n",
962
written += (size_t)ret;
968
/* Shutdown procedure */
973
free(decrypted_buffer);
976
ret = (int)TEMP_FAILURE_RETRY(close(tcp_sd));
982
perror_plus("close");
984
gnutls_deinit(session);
994
static void resolve_callback(AvahiSServiceResolver *r,
995
AvahiIfIndex interface,
997
AvahiResolverEvent event,
1001
const char *host_name,
1002
const AvahiAddress *address,
1004
AVAHI_GCC_UNUSED AvahiStringList *txt,
1005
AVAHI_GCC_UNUSED AvahiLookupResultFlags
1007
AVAHI_GCC_UNUSED void* userdata){
1012
/* Called whenever a service has been resolved successfully or
1021
case AVAHI_RESOLVER_FAILURE:
1022
fprintf_plus(stderr, "(Avahi Resolver) Failed to resolve service "
1023
"'%s' of type '%s' in domain '%s': %s\n", name, type,
1025
avahi_strerror(avahi_server_errno(mc.server)));
1028
case AVAHI_RESOLVER_FOUND:
1030
char ip[AVAHI_ADDRESS_STR_MAX];
1031
avahi_address_snprint(ip, sizeof(ip), address);
1033
fprintf_plus(stderr, "Mandos server \"%s\" found on %s (%s, %"
1034
PRIdMAX ") on port %" PRIu16 "\n", name,
1035
host_name, ip, (intmax_t)interface, port);
1037
int ret = start_mandos_communication(ip, (in_port_t)port,
1039
avahi_proto_to_af(proto));
1041
avahi_simple_poll_quit(mc.simple_poll);
1043
if(not add_server(ip, (in_port_t)port, interface,
1044
avahi_proto_to_af(proto))){
1045
fprintf_plus(stderr, "Failed to add server \"%s\" to server"
1051
avahi_s_service_resolver_free(r);
1054
static void browse_callback(AvahiSServiceBrowser *b,
1055
AvahiIfIndex interface,
1056
AvahiProtocol protocol,
1057
AvahiBrowserEvent event,
1061
AVAHI_GCC_UNUSED AvahiLookupResultFlags
1063
AVAHI_GCC_UNUSED void* userdata){
1068
/* Called whenever a new services becomes available on the LAN or
1069
is removed from the LAN */
1077
case AVAHI_BROWSER_FAILURE:
1079
fprintf_plus(stderr, "(Avahi browser) %s\n",
1080
avahi_strerror(avahi_server_errno(mc.server)));
1081
avahi_simple_poll_quit(mc.simple_poll);
1084
case AVAHI_BROWSER_NEW:
1085
/* We ignore the returned Avahi resolver object. In the callback
1086
function we free it. If the Avahi server is terminated before
1087
the callback function is called the Avahi server will free the
1090
if(avahi_s_service_resolver_new(mc.server, interface, protocol,
1091
name, type, domain, protocol, 0,
1092
resolve_callback, NULL) == NULL)
1093
fprintf_plus(stderr, "Avahi: Failed to resolve service '%s':"
1095
avahi_strerror(avahi_server_errno(mc.server)));
1098
case AVAHI_BROWSER_REMOVE:
1101
case AVAHI_BROWSER_ALL_FOR_NOW:
1102
case AVAHI_BROWSER_CACHE_EXHAUSTED:
1104
fprintf_plus(stderr, "No Mandos server found, still"
1111
/* Signal handler that stops main loop after SIGTERM */
1112
static void handle_sigterm(int sig){
1117
signal_received = sig;
1118
int old_errno = errno;
1119
/* set main loop to exit */
1120
if(mc.simple_poll != NULL){
1121
avahi_simple_poll_quit(mc.simple_poll);
1126
bool get_flags(const char *ifname, struct ifreq *ifr){
1130
int s = socket(PF_INET6, SOCK_DGRAM, IPPROTO_IP);
1133
perror_plus("socket");
1137
strcpy(ifr->ifr_name, ifname);
1138
ret = ioctl(s, SIOCGIFFLAGS, ifr);
1142
perror_plus("ioctl SIOCGIFFLAGS");
1150
bool good_flags(const char *ifname, const struct ifreq *ifr){
1152
/* Reject the loopback device */
1153
if(ifr->ifr_flags & IFF_LOOPBACK){
1155
fprintf_plus(stderr, "Rejecting loopback interface \"%s\"\n",
1160
/* Accept point-to-point devices only if connect_to is specified */
1161
if(connect_to != NULL and (ifr->ifr_flags & IFF_POINTOPOINT)){
1163
fprintf_plus(stderr, "Accepting point-to-point interface"
1164
" \"%s\"\n", ifname);
1168
/* Otherwise, reject non-broadcast-capable devices */
1169
if(not (ifr->ifr_flags & IFF_BROADCAST)){
1171
fprintf_plus(stderr, "Rejecting non-broadcast interface"
1172
" \"%s\"\n", ifname);
1176
/* Reject non-ARP interfaces (including dummy interfaces) */
1177
if(ifr->ifr_flags & IFF_NOARP){
1179
fprintf_plus(stderr, "Rejecting non-ARP interface \"%s\"\n",
1185
/* Accept this device */
1187
fprintf_plus(stderr, "Interface \"%s\" is good\n", ifname);
1193
* This function determines if a directory entry in /sys/class/net
1194
* corresponds to an acceptable network device.
1195
* (This function is passed to scandir(3) as a filter function.)
1197
int good_interface(const struct dirent *if_entry){
1198
if(if_entry->d_name[0] == '.'){
1203
if(not get_flags(if_entry->d_name, &ifr)){
1205
fprintf_plus(stderr, "Failed to get flags for interface "
1206
"\"%s\"\n", if_entry->d_name);
1211
if(not good_flags(if_entry->d_name, &ifr)){
1218
* This function determines if a network interface is up.
1220
bool interface_is_up(const char *interface){
1222
if(not get_flags(interface, &ifr)){
1224
fprintf_plus(stderr, "Failed to get flags for interface "
1225
"\"%s\"\n", interface);
1230
return (bool)(ifr.ifr_flags & IFF_UP);
1234
* This function determines if a network interface is running
1236
bool interface_is_running(const char *interface){
1238
if(not get_flags(interface, &ifr)){
1240
fprintf_plus(stderr, "Failed to get flags for interface "
1241
"\"%s\"\n", interface);
1246
return (bool)(ifr.ifr_flags & IFF_RUNNING);
1249
int notdotentries(const struct dirent *direntry){
1250
/* Skip "." and ".." */
1251
if(direntry->d_name[0] == '.'
1252
and (direntry->d_name[1] == '\0'
1253
or (direntry->d_name[1] == '.'
1254
and direntry->d_name[2] == '\0'))){
1260
/* Is this directory entry a runnable program? */
1261
int runnable_hook(const struct dirent *direntry){
1266
if((direntry->d_name)[0] == '\0'){
1271
sret = strspn(direntry->d_name, "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
1272
"abcdefghijklmnopqrstuvwxyz"
1275
if((direntry->d_name)[sret] != '\0'){
1276
/* Contains non-allowed characters */
1278
fprintf_plus(stderr, "Ignoring hook \"%s\" with bad name\n",
1284
char *fullname = NULL;
1285
ret = asprintf(&fullname, "%s/%s", hookdir, direntry->d_name);
1287
perror_plus("asprintf");
1291
ret = stat(fullname, &st);
1294
perror_plus("Could not stat hook");
1298
if(not (S_ISREG(st.st_mode))){
1299
/* Not a regular file */
1301
fprintf_plus(stderr, "Ignoring hook \"%s\" - not a file\n",
1306
if(not (st.st_mode & (S_IXUSR | S_IXGRP | S_IXOTH))){
1307
/* Not executable */
1309
fprintf_plus(stderr, "Ignoring hook \"%s\" - not executable\n",
1315
fprintf_plus(stderr, "Hook \"%s\" is acceptable\n",
1321
int avahi_loop_with_timeout(AvahiSimplePoll *s, int retry_interval){
1323
struct timespec now;
1324
struct timespec waited_time;
1325
intmax_t block_time;
1328
if(mc.current_server == NULL){
1330
fprintf_plus(stderr, "Wait until first server is found."
1333
ret = avahi_simple_poll_iterate(s, -1);
1336
fprintf_plus(stderr, "Check current_server if we should run"
1339
/* the current time */
1340
ret = clock_gettime(CLOCK_MONOTONIC, &now);
1342
perror_plus("clock_gettime");
1345
/* Calculating in ms how long time between now and server
1346
who we visted longest time ago. Now - last seen. */
1347
waited_time.tv_sec = (now.tv_sec
1348
- mc.current_server->last_seen.tv_sec);
1349
waited_time.tv_nsec = (now.tv_nsec
1350
- mc.current_server->last_seen.tv_nsec);
1351
/* total time is 10s/10,000ms.
1352
Converting to s from ms by dividing by 1,000,
1353
and ns to ms by dividing by 1,000,000. */
1354
block_time = ((retry_interval
1355
- ((intmax_t)waited_time.tv_sec * 1000))
1356
- ((intmax_t)waited_time.tv_nsec / 1000000));
1359
fprintf_plus(stderr, "Blocking for %" PRIdMAX " ms\n",
1363
if(block_time <= 0){
1364
ret = start_mandos_communication(mc.current_server->ip,
1365
mc.current_server->port,
1366
mc.current_server->if_index,
1367
mc.current_server->af);
1369
avahi_simple_poll_quit(mc.simple_poll);
1372
ret = clock_gettime(CLOCK_MONOTONIC,
1373
&mc.current_server->last_seen);
1375
perror_plus("clock_gettime");
1378
mc.current_server = mc.current_server->next;
1379
block_time = 0; /* Call avahi to find new Mandos
1380
servers, but don't block */
1383
ret = avahi_simple_poll_iterate(s, (int)block_time);
1386
if (ret > 0 or errno != EINTR){
1387
return (ret != 1) ? ret : 0;
1393
/* Set effective uid to 0, return errno */
1394
error_t raise_privileges(void){
1395
error_t old_errno = errno;
1396
error_t ret_errno = 0;
1397
if(seteuid(0) == -1){
1399
perror_plus("seteuid");
1405
/* Set effective and real user ID to 0. Return errno. */
1406
error_t raise_privileges_permanently(void){
1407
error_t old_errno = errno;
1408
error_t ret_errno = raise_privileges();
1413
if(setuid(0) == -1){
1415
perror_plus("seteuid");
1421
/* Set effective user ID to unprivileged saved user ID */
1422
error_t lower_privileges(void){
1423
error_t old_errno = errno;
1424
error_t ret_errno = 0;
1425
if(seteuid(uid) == -1){
1427
perror_plus("seteuid");
1433
/* Lower privileges permanently */
1434
error_t lower_privileges_permanently(void){
1435
error_t old_errno = errno;
1436
error_t ret_errno = 0;
1437
if(setuid(uid) == -1){
1439
perror_plus("setuid");
1445
bool run_network_hooks(const char *mode, const char *interface,
1447
struct dirent **direntries;
1448
struct dirent *direntry;
1450
int numhooks = scandir(hookdir, &direntries, runnable_hook,
1453
if(errno == ENOENT){
1455
fprintf_plus(stderr, "Network hook directory \"%s\" not"
1456
" found\n", hookdir);
1459
perror_plus("scandir");
1462
int devnull = open("/dev/null", O_RDONLY);
1463
for(int i = 0; i < numhooks; i++){
1464
direntry = direntries[i];
1465
char *fullname = NULL;
1466
ret = asprintf(&fullname, "%s/%s", hookdir, direntry->d_name);
1468
perror_plus("asprintf");
1472
fprintf_plus(stderr, "Running network hook \"%s\"\n",
1475
pid_t hook_pid = fork();
1478
/* Raise privileges */
1479
raise_privileges_permanently();
1484
perror_plus("setgid");
1486
/* Reset supplementary groups */
1488
ret = setgroups(0, NULL);
1490
perror_plus("setgroups");
1492
dup2(devnull, STDIN_FILENO);
1494
dup2(STDERR_FILENO, STDOUT_FILENO);
1495
ret = setenv("MANDOSNETHOOKDIR", hookdir, 1);
1497
perror_plus("setenv");
1500
ret = setenv("DEVICE", interface, 1);
1502
perror_plus("setenv");
1505
ret = setenv("VERBOSITY", debug ? "1" : "0", 1);
1507
perror_plus("setenv");
1510
ret = setenv("MODE", mode, 1);
1512
perror_plus("setenv");
1516
ret = asprintf(&delaystring, "%f", delay);
1518
perror_plus("asprintf");
1521
ret = setenv("DELAY", delaystring, 1);
1524
perror_plus("setenv");
1528
if(connect_to != NULL){
1529
ret = setenv("CONNECT", connect_to, 1);
1531
perror_plus("setenv");
1535
if(execl(fullname, direntry->d_name, mode, NULL) == -1){
1536
perror_plus("execl");
1537
_exit(EXIT_FAILURE);
1541
if(TEMP_FAILURE_RETRY(waitpid(hook_pid, &status, 0)) == -1){
1542
perror_plus("waitpid");
1546
if(WIFEXITED(status)){
1547
if(WEXITSTATUS(status) != 0){
1548
fprintf_plus(stderr, "Warning: network hook \"%s\" exited"
1549
" with status %d\n", direntry->d_name,
1550
WEXITSTATUS(status));
1554
} else if(WIFSIGNALED(status)){
1555
fprintf_plus(stderr, "Warning: network hook \"%s\" died by"
1556
" signal %d\n", direntry->d_name,
1561
fprintf_plus(stderr, "Warning: network hook \"%s\""
1562
" crashed\n", direntry->d_name);
1569
fprintf_plus(stderr, "Network hook \"%s\" ran successfully\n",
1578
error_t bring_up_interface(const char *const interface,
1581
error_t old_errno = errno;
1582
error_t ret_errno = 0;
1583
int ret, ret_setflags;
1584
struct ifreq network;
1585
unsigned int if_index = if_nametoindex(interface);
1587
fprintf_plus(stderr, "No such interface: \"%s\"\n", interface);
1597
if(not interface_is_up(interface)){
1598
if(not get_flags(interface, &network) and debug){
1600
fprintf_plus(stderr, "Failed to get flags for interface "
1601
"\"%s\"\n", interface);
1604
network.ifr_flags |= IFF_UP;
1606
sd = socket(PF_INET6, SOCK_DGRAM, IPPROTO_IP);
1609
perror_plus("socket");
1621
fprintf_plus(stderr, "Bringing up interface \"%s\"\n",
1625
/* Raise priviliges */
1629
/* Lower kernel loglevel to KERN_NOTICE to avoid KERN_INFO
1630
messages about the network interface to mess up the prompt */
1631
int ret_linux = klogctl(8, NULL, 5);
1632
bool restore_loglevel = true;
1633
if(ret_linux == -1){
1634
restore_loglevel = false;
1635
perror_plus("klogctl");
1637
#endif /* __linux__ */
1638
ret_setflags = ioctl(sd, SIOCSIFFLAGS, &network);
1641
if(restore_loglevel){
1642
ret_linux = klogctl(7, NULL, 0);
1643
if(ret_linux == -1){
1644
perror_plus("klogctl");
1647
#endif /* __linux__ */
1649
/* Lower privileges */
1652
/* Close the socket */
1653
ret = (int)TEMP_FAILURE_RETRY(close(sd));
1655
perror_plus("close");
1658
if(ret_setflags == -1){
1660
perror_plus("ioctl SIOCSIFFLAGS +IFF_UP");
1665
fprintf_plus(stderr, "Interface \"%s\" is already up; good\n",
1669
/* Sleep checking until interface is running.
1670
Check every 0.25s, up to total time of delay */
1671
for(int i=0; i < delay * 4; i++){
1672
if(interface_is_running(interface)){
1675
struct timespec sleeptime = { .tv_nsec = 250000000 };
1676
ret = nanosleep(&sleeptime, NULL);
1677
if(ret == -1 and errno != EINTR){
1678
perror_plus("nanosleep");
1686
error_t take_down_interface(const char *const interface){
1688
error_t old_errno = errno;
1689
error_t ret_errno = 0;
1690
int ret, ret_setflags;
1691
struct ifreq network;
1692
unsigned int if_index = if_nametoindex(interface);
1694
fprintf_plus(stderr, "No such interface: \"%s\"\n", interface);
1698
if(interface_is_up(interface)){
1699
if(not get_flags(interface, &network) and debug){
1701
fprintf_plus(stderr, "Failed to get flags for interface "
1702
"\"%s\"\n", interface);
1705
network.ifr_flags &= ~(short)IFF_UP; /* clear flag */
1707
sd = socket(PF_INET6, SOCK_DGRAM, IPPROTO_IP);
1710
perror_plus("socket");
1716
fprintf_plus(stderr, "Taking down interface \"%s\"\n",
1720
/* Raise priviliges */
1723
ret_setflags = ioctl(sd, SIOCSIFFLAGS, &network);
1726
/* Lower privileges */
1729
/* Close the socket */
1730
ret = (int)TEMP_FAILURE_RETRY(close(sd));
1732
perror_plus("close");
1735
if(ret_setflags == -1){
1737
perror_plus("ioctl SIOCSIFFLAGS -IFF_UP");
1742
fprintf_plus(stderr, "Interface \"%s\" is already down; odd\n",
1750
int main(int argc, char *argv[]){
1751
AvahiSServiceBrowser *sb = NULL;
1756
int exitcode = EXIT_SUCCESS;
1757
char *interfaces = NULL;
1758
size_t interfaces_size = 0;
1759
char *interfaces_to_take_down = NULL;
1760
size_t interfaces_to_take_down_size = 0;
1761
char tempdir[] = "/tmp/mandosXXXXXX";
1762
bool tempdir_created = false;
1763
AvahiIfIndex if_index = AVAHI_IF_UNSPEC;
1764
const char *seckey = PATHDIR "/" SECKEY;
1765
const char *pubkey = PATHDIR "/" PUBKEY;
1766
char *interfaces_hooks = NULL;
1767
size_t interfaces_hooks_size = 0;
1769
bool gnutls_initialized = false;
1770
bool gpgme_initialized = false;
1772
double retry_interval = 10; /* 10s between trying a server and
1773
retrying the same server again */
1775
struct sigaction old_sigterm_action = { .sa_handler = SIG_DFL };
1776
struct sigaction sigterm_action = { .sa_handler = handle_sigterm };
1781
/* Lower any group privileges we might have, just to be safe */
1785
perror_plus("setgid");
1788
/* Lower user privileges (temporarily) */
1792
perror_plus("seteuid");
1800
struct argp_option options[] = {
1801
{ .name = "debug", .key = 128,
1802
.doc = "Debug mode", .group = 3 },
1803
{ .name = "connect", .key = 'c',
1804
.arg = "ADDRESS:PORT",
1805
.doc = "Connect directly to a specific Mandos server",
1807
{ .name = "interface", .key = 'i',
1809
.doc = "Network interface that will be used to search for"
1812
{ .name = "seckey", .key = 's',
1814
.doc = "OpenPGP secret key file base name",
1816
{ .name = "pubkey", .key = 'p',
1818
.doc = "OpenPGP public key file base name",
1820
{ .name = "dh-bits", .key = 129,
1822
.doc = "Bit length of the prime number used in the"
1823
" Diffie-Hellman key exchange",
1825
{ .name = "priority", .key = 130,
1827
.doc = "GnuTLS priority string for the TLS handshake",
1829
{ .name = "delay", .key = 131,
1831
.doc = "Maximum delay to wait for interface startup",
1833
{ .name = "retry", .key = 132,
1835
.doc = "Retry interval used when denied by the Mandos server",
1837
{ .name = "network-hook-dir", .key = 133,
1839
.doc = "Directory where network hooks are located",
1842
* These reproduce what we would get without ARGP_NO_HELP
1844
{ .name = "help", .key = '?',
1845
.doc = "Give this help list", .group = -1 },
1846
{ .name = "usage", .key = -3,
1847
.doc = "Give a short usage message", .group = -1 },
1848
{ .name = "version", .key = 'V',
1849
.doc = "Print program version", .group = -1 },
1853
error_t parse_opt(int key, char *arg,
1854
struct argp_state *state){
1857
case 128: /* --debug */
1860
case 'c': /* --connect */
1863
case 'i': /* --interface */
1864
ret_errno = argz_add_sep(&interfaces, &interfaces_size, arg,
1867
argp_error(state, "%s", strerror(ret_errno));
1870
case 's': /* --seckey */
1873
case 'p': /* --pubkey */
1876
case 129: /* --dh-bits */
1878
tmpmax = strtoimax(arg, &tmp, 10);
1879
if(errno != 0 or tmp == arg or *tmp != '\0'
1880
or tmpmax != (typeof(mc.dh_bits))tmpmax){
1881
argp_error(state, "Bad number of DH bits");
1883
mc.dh_bits = (typeof(mc.dh_bits))tmpmax;
1885
case 130: /* --priority */
1888
case 131: /* --delay */
1890
delay = strtof(arg, &tmp);
1891
if(errno != 0 or tmp == arg or *tmp != '\0'){
1892
argp_error(state, "Bad delay");
1894
case 132: /* --retry */
1896
retry_interval = strtod(arg, &tmp);
1897
if(errno != 0 or tmp == arg or *tmp != '\0'
1898
or (retry_interval * 1000) > INT_MAX
1899
or retry_interval < 0){
1900
argp_error(state, "Bad retry interval");
1903
case 133: /* --network-hook-dir */
1907
* These reproduce what we would get without ARGP_NO_HELP
1909
case '?': /* --help */
1910
argp_state_help(state, state->out_stream,
1911
(ARGP_HELP_STD_HELP | ARGP_HELP_EXIT_ERR)
1912
& ~(unsigned int)ARGP_HELP_EXIT_OK);
1913
case -3: /* --usage */
1914
argp_state_help(state, state->out_stream,
1915
ARGP_HELP_USAGE | ARGP_HELP_EXIT_ERR);
1916
case 'V': /* --version */
1917
fprintf_plus(state->out_stream, "%s\n", argp_program_version);
1918
exit(argp_err_exit_status);
1921
return ARGP_ERR_UNKNOWN;
1926
struct argp argp = { .options = options, .parser = parse_opt,
1928
.doc = "Mandos client -- Get and decrypt"
1929
" passwords from a Mandos server" };
1930
ret = argp_parse(&argp, argc, argv,
1931
ARGP_IN_ORDER | ARGP_NO_HELP, 0, NULL);
1938
perror_plus("argp_parse");
1939
exitcode = EX_OSERR;
1942
exitcode = EX_USAGE;
1948
/* Work around Debian bug #633582:
1949
<http://bugs.debian.org/633582> */
1951
/* Re-raise priviliges */
1952
if(raise_privileges() == 0){
1955
if(strcmp(seckey, PATHDIR "/" SECKEY) == 0){
1956
int seckey_fd = open(seckey, O_RDONLY);
1957
if(seckey_fd == -1){
1958
perror_plus("open");
1960
ret = (int)TEMP_FAILURE_RETRY(fstat(seckey_fd, &st));
1962
perror_plus("fstat");
1964
if(S_ISREG(st.st_mode)
1965
and st.st_uid == 0 and st.st_gid == 0){
1966
ret = fchown(seckey_fd, uid, gid);
1968
perror_plus("fchown");
1972
TEMP_FAILURE_RETRY(close(seckey_fd));
1976
if(strcmp(pubkey, PATHDIR "/" PUBKEY) == 0){
1977
int pubkey_fd = open(pubkey, O_RDONLY);
1978
if(pubkey_fd == -1){
1979
perror_plus("open");
1981
ret = (int)TEMP_FAILURE_RETRY(fstat(pubkey_fd, &st));
1983
perror_plus("fstat");
1985
if(S_ISREG(st.st_mode)
1986
and st.st_uid == 0 and st.st_gid == 0){
1987
ret = fchown(pubkey_fd, uid, gid);
1989
perror_plus("fchown");
1993
TEMP_FAILURE_RETRY(close(pubkey_fd));
1997
/* Lower privileges */
2001
perror_plus("seteuid");
2006
/* Remove empty interface names */
2008
char *interface = NULL;
2009
while((interface = argz_next(interfaces, interfaces_size,
2011
if(if_nametoindex(interface) == 0){
2012
if(interface[0] != '\0' and strcmp(interface, "none") != 0){
2013
fprintf_plus(stderr, "Not using nonexisting interface"
2014
" \"%s\"\n", interface);
2016
argz_delete(&interfaces, &interfaces_size, interface);
2022
/* Run network hooks */
2024
ret_errno = argz_append(&interfaces_hooks, &interfaces_hooks_size,
2025
interfaces, interfaces_size);
2028
perror_plus("argz_append");
2031
argz_stringify(interfaces_hooks, interfaces_hooks_size, (int)',');
2032
if(not run_network_hooks("start", interfaces_hooks, delay)){
2038
avahi_set_log_function(empty_log);
2041
/* Initialize Avahi early so avahi_simple_poll_quit() can be called
2042
from the signal handler */
2043
/* Initialize the pseudo-RNG for Avahi */
2044
srand((unsigned int) time(NULL));
2045
mc.simple_poll = avahi_simple_poll_new();
2046
if(mc.simple_poll == NULL){
2047
fprintf_plus(stderr,
2048
"Avahi: Failed to create simple poll object.\n");
2049
exitcode = EX_UNAVAILABLE;
2053
sigemptyset(&sigterm_action.sa_mask);
2054
ret = sigaddset(&sigterm_action.sa_mask, SIGINT);
2056
perror_plus("sigaddset");
2057
exitcode = EX_OSERR;
2060
ret = sigaddset(&sigterm_action.sa_mask, SIGHUP);
2062
perror_plus("sigaddset");
2063
exitcode = EX_OSERR;
2066
ret = sigaddset(&sigterm_action.sa_mask, SIGTERM);
2068
perror_plus("sigaddset");
2069
exitcode = EX_OSERR;
2072
/* Need to check if the handler is SIG_IGN before handling:
2073
| [[info:libc:Initial Signal Actions]] |
2074
| [[info:libc:Basic Signal Handling]] |
2076
ret = sigaction(SIGINT, NULL, &old_sigterm_action);
2078
perror_plus("sigaction");
2081
if(old_sigterm_action.sa_handler != SIG_IGN){
2082
ret = sigaction(SIGINT, &sigterm_action, NULL);
2084
perror_plus("sigaction");
2085
exitcode = EX_OSERR;
2089
ret = sigaction(SIGHUP, NULL, &old_sigterm_action);
2091
perror_plus("sigaction");
2094
if(old_sigterm_action.sa_handler != SIG_IGN){
2095
ret = sigaction(SIGHUP, &sigterm_action, NULL);
2097
perror_plus("sigaction");
2098
exitcode = EX_OSERR;
2102
ret = sigaction(SIGTERM, NULL, &old_sigterm_action);
2104
perror_plus("sigaction");
2107
if(old_sigterm_action.sa_handler != SIG_IGN){
2108
ret = sigaction(SIGTERM, &sigterm_action, NULL);
2110
perror_plus("sigaction");
2111
exitcode = EX_OSERR;
2116
/* If no interfaces were specified, make a list */
2117
if(interfaces == NULL){
2118
struct dirent **direntries;
2119
/* Look for any good interfaces */
2120
ret = scandir(sys_class_net, &direntries, good_interface,
2123
/* Add all found interfaces to interfaces list */
2124
for(int i = 0; i < ret; ++i){
2125
ret_errno = argz_add(&interfaces, &interfaces_size,
2126
direntries[i]->d_name);
2128
perror_plus("argz_add");
2132
fprintf_plus(stderr, "Will use interface \"%s\"\n",
2133
direntries[i]->d_name);
2139
fprintf_plus(stderr, "Could not find a network interface\n");
2140
exitcode = EXIT_FAILURE;
2145
/* If we only got one interface, explicitly use only that one */
2146
if(argz_count(interfaces, interfaces_size) == 1){
2148
fprintf_plus(stderr, "Using only interface \"%s\"\n",
2151
if_index = (AvahiIfIndex)if_nametoindex(interfaces);
2154
/* Bring up interfaces which are down */
2155
if(not (argz_count(interfaces, interfaces_size) == 1
2156
and strcmp(interfaces, "none") == 0)){
2157
char *interface = NULL;
2158
while((interface = argz_next(interfaces, interfaces_size,
2160
bool interface_was_up = interface_is_up(interface);
2161
ret = bring_up_interface(interface, delay);
2162
if(not interface_was_up){
2165
perror_plus("Failed to bring up interface");
2167
ret_errno = argz_add(&interfaces_to_take_down,
2168
&interfaces_to_take_down_size,
2175
interfaces_size = 0;
2176
if(debug and (interfaces_to_take_down == NULL)){
2177
fprintf_plus(stderr, "No interfaces were brought up\n");
2185
ret = init_gnutls_global(pubkey, seckey);
2187
fprintf_plus(stderr, "init_gnutls_global failed\n");
2188
exitcode = EX_UNAVAILABLE;
2191
gnutls_initialized = true;
2198
if(mkdtemp(tempdir) == NULL){
2199
perror_plus("mkdtemp");
2202
tempdir_created = true;
2208
if(not init_gpgme(pubkey, seckey, tempdir)){
2209
fprintf_plus(stderr, "init_gpgme failed\n");
2210
exitcode = EX_UNAVAILABLE;
2213
gpgme_initialized = true;
2220
if(connect_to != NULL){
2221
/* Connect directly, do not use Zeroconf */
2222
/* (Mainly meant for debugging) */
2223
char *address = strrchr(connect_to, ':');
2225
if(address == NULL){
2226
fprintf_plus(stderr, "No colon in address\n");
2227
exitcode = EX_USAGE;
2237
tmpmax = strtoimax(address+1, &tmp, 10);
2238
if(errno != 0 or tmp == address+1 or *tmp != '\0'
2239
or tmpmax != (in_port_t)tmpmax){
2240
fprintf_plus(stderr, "Bad port number\n");
2241
exitcode = EX_USAGE;
2249
port = (in_port_t)tmpmax;
2251
/* Colon in address indicates IPv6 */
2253
if(strchr(connect_to, ':') != NULL){
2255
/* Accept [] around IPv6 address - see RFC 5952 */
2256
if(connect_to[0] == '[' and address[-1] == ']')
2264
address = connect_to;
2270
while(not quit_now){
2271
ret = start_mandos_communication(address, port, if_index, af);
2272
if(quit_now or ret == 0){
2276
fprintf_plus(stderr, "Retrying in %d seconds\n",
2277
(int)retry_interval);
2279
sleep((int)retry_interval);
2283
exitcode = EXIT_SUCCESS;
2294
AvahiServerConfig config;
2295
/* Do not publish any local Zeroconf records */
2296
avahi_server_config_init(&config);
2297
config.publish_hinfo = 0;
2298
config.publish_addresses = 0;
2299
config.publish_workstation = 0;
2300
config.publish_domain = 0;
2302
/* Allocate a new server */
2303
mc.server = avahi_server_new(avahi_simple_poll_get
2304
(mc.simple_poll), &config, NULL,
2307
/* Free the Avahi configuration data */
2308
avahi_server_config_free(&config);
2311
/* Check if creating the Avahi server object succeeded */
2312
if(mc.server == NULL){
2313
fprintf_plus(stderr, "Failed to create Avahi server: %s\n",
2314
avahi_strerror(ret_errno));
2315
exitcode = EX_UNAVAILABLE;
2323
/* Create the Avahi service browser */
2324
sb = avahi_s_service_browser_new(mc.server, if_index,
2325
AVAHI_PROTO_UNSPEC, "_mandos._tcp",
2326
NULL, 0, browse_callback, NULL);
2328
fprintf_plus(stderr, "Failed to create service browser: %s\n",
2329
avahi_strerror(avahi_server_errno(mc.server)));
2330
exitcode = EX_UNAVAILABLE;
2338
/* Run the main loop */
2341
fprintf_plus(stderr, "Starting Avahi loop search\n");
2344
ret = avahi_loop_with_timeout(mc.simple_poll,
2345
(int)(retry_interval * 1000));
2347
fprintf_plus(stderr, "avahi_loop_with_timeout exited %s\n",
2348
(ret == 0) ? "successfully" : "with error");
2354
fprintf_plus(stderr, "%s exiting\n", argv[0]);
2357
/* Cleanup things */
2359
avahi_s_service_browser_free(sb);
2361
if(mc.server != NULL)
2362
avahi_server_free(mc.server);
2364
if(mc.simple_poll != NULL)
2365
avahi_simple_poll_free(mc.simple_poll);
2367
if(gnutls_initialized){
2368
gnutls_certificate_free_credentials(mc.cred);
2369
gnutls_global_deinit();
2370
gnutls_dh_params_deinit(mc.dh_params);
2373
if(gpgme_initialized){
2374
gpgme_release(mc.ctx);
2377
/* Cleans up the circular linked list of Mandos servers the client
2379
if(mc.current_server != NULL){
2380
mc.current_server->prev->next = NULL;
2381
while(mc.current_server != NULL){
2382
server *next = mc.current_server->next;
2383
free(mc.current_server);
2384
mc.current_server = next;
2388
/* Re-raise priviliges */
2392
/* Run network hooks */
2393
run_network_hooks("stop", interfaces_hooks, delay);
2395
/* Take down the network interfaces which were brought up */
2397
char *interface = NULL;
2398
while((interface=argz_next(interfaces_to_take_down,
2399
interfaces_to_take_down_size,
2401
ret_errno = take_down_interface(interface);
2404
perror_plus("Failed to take down interface");
2407
if(debug and (interfaces_to_take_down == NULL)){
2408
fprintf_plus(stderr, "No interfaces needed to be taken"
2413
lower_privileges_permanently();
2416
free(interfaces_to_take_down);
2417
free(interfaces_hooks);
2419
/* Removes the GPGME temp directory and all files inside */
2420
if(tempdir_created){
2421
struct dirent **direntries = NULL;
2422
struct dirent *direntry = NULL;
2423
int numentries = scandir(tempdir, &direntries, notdotentries,
2425
if (numentries > 0){
2426
for(int i = 0; i < numentries; i++){
2427
direntry = direntries[i];
2428
char *fullname = NULL;
2429
ret = asprintf(&fullname, "%s/%s", tempdir,
2432
perror_plus("asprintf");
2435
ret = remove(fullname);
2437
fprintf_plus(stderr, "remove(\"%s\"): %s\n", fullname,
2444
/* need to clean even if 0 because man page doesn't specify */
2446
if (numentries == -1){
2447
perror_plus("scandir");
2449
ret = rmdir(tempdir);
2450
if(ret == -1 and errno != ENOENT){
2451
perror_plus("rmdir");
2456
sigemptyset(&old_sigterm_action.sa_mask);
2457
old_sigterm_action.sa_handler = SIG_DFL;
2458
ret = (int)TEMP_FAILURE_RETRY(sigaction(signal_received,
2459
&old_sigterm_action,
2462
perror_plus("sigaction");
2465
ret = raise(signal_received);
2466
} while(ret != 0 and errno == EINTR);
2468
perror_plus("raise");
2471
TEMP_FAILURE_RETRY(pause());