100
47
#include <avahi-common/malloc.h>
101
48
#include <avahi-common/error.h>
104
#include <gnutls/gnutls.h> /* All GnuTLS types, constants and
107
init_gnutls_session(),
109
#include <gnutls/openpgp.h>
110
/* gnutls_certificate_set_openpgp_key_file(),
111
GNUTLS_OPENPGP_FMT_BASE64 */
114
#include <gpgme.h> /* All GPGME types, constants and
117
GPGME_PROTOCOL_OpenPGP,
51
#include <sys/types.h> /* socket(), inet_pton() */
52
#include <sys/socket.h> /* socket(), struct sockaddr_in6,
53
struct in6_addr, inet_pton() */
54
#include <gnutls/gnutls.h> /* All GnuTLS stuff */
55
#include <gnutls/openpgp.h> /* GnuTLS with openpgp stuff */
57
#include <unistd.h> /* close() */
58
#include <netinet/in.h>
59
#include <stdbool.h> /* true */
60
#include <string.h> /* memset */
61
#include <arpa/inet.h> /* inet_pton() */
62
#include <iso646.h> /* not */
65
#include <errno.h> /* perror() */
120
71
#define BUFFER_SIZE 256
122
#define PATHDIR "/conf/conf.d/mandos"
123
#define SECKEY "seckey.txt"
124
#define PUBKEY "pubkey.txt"
74
static const char *certdir = "/conf/conf.d/mandos";
75
static const char *certfile = "openpgp-client.txt";
76
static const char *certkey = "openpgp-client-key.txt";
126
78
bool debug = false;
127
static const char mandos_protocol_version[] = "1";
128
const char *argp_program_version = "mandos-client " VERSION;
129
const char *argp_program_bug_address = "<mandos@fukt.bsnet.se>";
130
static const char sys_class_net[] = "/sys/class/net";
131
char *connect_to = NULL;
133
/* Used for passing in values through the Avahi callback functions */
135
AvahiSimplePoll *simple_poll;
81
gnutls_session_t session;
137
82
gnutls_certificate_credentials_t cred;
138
unsigned int dh_bits;
139
83
gnutls_dh_params_t dh_params;
140
const char *priority;
87
static ssize_t pgp_packet_decrypt (char *packet, size_t packet_size,
90
gpgme_data_t dh_crypto, dh_plain;
144
/* global context so signal handler can reach it*/
145
mandos_context mc = { .simple_poll = NULL, .server = NULL,
146
.dh_bits = 1024, .priority = "SECURE256"
147
":!CTYPE-X.509:+CTYPE-OPENPGP" };
149
sig_atomic_t quit_now = 0;
150
int signal_received = 0;
153
* Make additional room in "buffer" for at least BUFFER_SIZE more
154
* bytes. "buffer_capacity" is how much is currently allocated,
155
* "buffer_length" is how much is already used.
157
size_t incbuffer(char **buffer, size_t buffer_length,
158
size_t buffer_capacity){
159
if(buffer_length + BUFFER_SIZE > buffer_capacity){
160
*buffer = realloc(*buffer, buffer_capacity + BUFFER_SIZE);
164
buffer_capacity += BUFFER_SIZE;
166
return buffer_capacity;
172
static bool init_gpgme(const char *seckey,
173
const char *pubkey, const char *tempdir){
94
ssize_t new_packet_capacity = 0;
95
ssize_t new_packet_length = 0;
175
96
gpgme_engine_info_t engine_info;
179
* Helper function to insert pub and seckey to the engine keyring.
181
bool import_key(const char *filename){
184
gpgme_data_t pgp_data;
186
fd = (int)TEMP_FAILURE_RETRY(open(filename, O_RDONLY));
192
rc = gpgme_data_new_from_fd(&pgp_data, fd);
193
if(rc != GPG_ERR_NO_ERROR){
194
fprintf(stderr, "bad gpgme_data_new_from_fd: %s: %s\n",
195
gpgme_strsource(rc), gpgme_strerror(rc));
199
rc = gpgme_op_import(mc.ctx, pgp_data);
200
if(rc != GPG_ERR_NO_ERROR){
201
fprintf(stderr, "bad gpgme_op_import: %s: %s\n",
202
gpgme_strsource(rc), gpgme_strerror(rc));
206
ret = (int)TEMP_FAILURE_RETRY(close(fd));
210
gpgme_data_release(pgp_data);
215
fprintf(stderr, "Initializing GPGME\n");
99
fprintf(stderr, "Trying to decrypt OpenPGP packet\n");
219
103
gpgme_check_version(NULL);
220
104
rc = gpgme_engine_check_version(GPGME_PROTOCOL_OpenPGP);
221
if(rc != GPG_ERR_NO_ERROR){
105
if (rc != GPG_ERR_NO_ERROR){
222
106
fprintf(stderr, "bad gpgme_engine_check_version: %s: %s\n",
223
107
gpgme_strsource(rc), gpgme_strerror(rc));
227
/* Set GPGME home directory for the OpenPGP engine only */
228
rc = gpgme_get_engine_info(&engine_info);
229
if(rc != GPG_ERR_NO_ERROR){
111
/* Set GPGME home directory */
112
rc = gpgme_get_engine_info (&engine_info);
113
if (rc != GPG_ERR_NO_ERROR){
230
114
fprintf(stderr, "bad gpgme_get_engine_info: %s: %s\n",
231
115
gpgme_strsource(rc), gpgme_strerror(rc));
234
118
while(engine_info != NULL){
235
119
if(engine_info->protocol == GPGME_PROTOCOL_OpenPGP){
236
120
gpgme_set_engine_info(GPGME_PROTOCOL_OpenPGP,
237
engine_info->file_name, tempdir);
121
engine_info->file_name, homedir);
240
124
engine_info = engine_info->next;
242
126
if(engine_info == NULL){
243
fprintf(stderr, "Could not set GPGME home dir to %s\n", tempdir);
247
/* Create new GPGME "context" */
248
rc = gpgme_new(&(mc.ctx));
249
if(rc != GPG_ERR_NO_ERROR){
250
fprintf(stderr, "bad gpgme_new: %s: %s\n",
251
gpgme_strsource(rc), gpgme_strerror(rc));
255
if(not import_key(pubkey) or not import_key(seckey)){
263
* Decrypt OpenPGP data.
264
* Returns -1 on error
266
static ssize_t pgp_packet_decrypt(const char *cryptotext,
269
gpgme_data_t dh_crypto, dh_plain;
272
size_t plaintext_capacity = 0;
273
ssize_t plaintext_length = 0;
276
fprintf(stderr, "Trying to decrypt OpenPGP data\n");
279
/* Create new GPGME data buffer from memory cryptotext */
280
rc = gpgme_data_new_from_mem(&dh_crypto, cryptotext, crypto_size,
282
if(rc != GPG_ERR_NO_ERROR){
127
fprintf(stderr, "Could not set home dir to %s\n", homedir);
131
/* Create new GPGME data buffer from packet buffer */
132
rc = gpgme_data_new_from_mem(&dh_crypto, packet, packet_size, 0);
133
if (rc != GPG_ERR_NO_ERROR){
283
134
fprintf(stderr, "bad gpgme_data_new_from_mem: %s: %s\n",
284
135
gpgme_strsource(rc), gpgme_strerror(rc));
334
fprintf(stderr, "Decryption of OpenPGP data succeeded\n");
197
/* Delete the GPGME FILE pointer cryptotext data buffer */
198
gpgme_data_release(dh_crypto);
337
200
/* Seek back to the beginning of the GPGME plaintext data buffer */
338
if(gpgme_data_seek(dh_plain, (off_t)0, SEEK_SET) == -1){
339
perror("gpgme_data_seek");
340
plaintext_length = -1;
201
if (gpgme_data_seek(dh_plain, (off_t) 0, SEEK_SET) == -1){
202
perror("pgpme_data_seek");
346
plaintext_capacity = incbuffer(plaintext,
347
(size_t)plaintext_length,
349
if(plaintext_capacity == 0){
351
plaintext_length = -1;
207
if (new_packet_length + BUFFER_SIZE > new_packet_capacity){
208
*new_packet = realloc(*new_packet,
209
(unsigned int)new_packet_capacity
211
if (*new_packet == NULL){
215
new_packet_capacity += BUFFER_SIZE;
355
ret = gpgme_data_read(dh_plain, *plaintext + plaintext_length,
218
ret = gpgme_data_read(dh_plain, *new_packet + new_packet_length,
357
220
/* Print the data, if any */
363
225
perror("gpgme_data_read");
364
plaintext_length = -1;
367
plaintext_length += ret;
371
fprintf(stderr, "Decrypted password is: ");
372
for(ssize_t i = 0; i < plaintext_length; i++){
373
fprintf(stderr, "%02hhX ", (*plaintext)[i]);
375
fprintf(stderr, "\n");
380
/* Delete the GPGME cryptotext data buffer */
381
gpgme_data_release(dh_crypto);
228
new_packet_length += ret;
231
/* FIXME: check characters before printing to screen so to not print
232
terminal control characters */
234
/* fprintf(stderr, "decrypted password is: "); */
235
/* fwrite(*new_packet, 1, new_packet_length, stderr); */
236
/* fprintf(stderr, "\n"); */
383
239
/* Delete the GPGME plaintext data buffer */
384
240
gpgme_data_release(dh_plain);
385
return plaintext_length;
241
return new_packet_length;
388
static const char * safer_gnutls_strerror(int value){
389
const char *ret = gnutls_strerror(value); /* Spurious warning from
390
-Wunreachable-code */
244
static const char * safer_gnutls_strerror (int value) {
245
const char *ret = gnutls_strerror (value);
392
247
ret = "(unknown)";
396
/* GnuTLS log function callback */
397
251
static void debuggnutls(__attribute__((unused)) int level,
398
252
const char* string){
399
fprintf(stderr, "GnuTLS: %s", string);
253
fprintf(stderr, "%s", string);
402
static int init_gnutls_global(const char *pubkeyfilename,
403
const char *seckeyfilename){
256
static int initgnutls(encrypted_session *es){
407
261
fprintf(stderr, "Initializing GnuTLS\n");
410
ret = gnutls_global_init();
411
if(ret != GNUTLS_E_SUCCESS){
412
fprintf(stderr, "GnuTLS global_init: %s\n",
413
safer_gnutls_strerror(ret));
264
if ((ret = gnutls_global_init ())
265
!= GNUTLS_E_SUCCESS) {
266
fprintf (stderr, "global_init: %s\n", safer_gnutls_strerror(ret));
418
/* "Use a log level over 10 to enable all debugging options."
421
271
gnutls_global_set_log_level(11);
422
272
gnutls_global_set_log_function(debuggnutls);
425
/* OpenPGP credentials */
426
gnutls_certificate_allocate_credentials(&mc.cred);
427
if(ret != GNUTLS_E_SUCCESS){
428
fprintf(stderr, "GnuTLS memory error: %s\n", /* Spurious warning
432
safer_gnutls_strerror(ret));
433
gnutls_global_deinit();
275
/* openpgp credentials */
276
if ((ret = gnutls_certificate_allocate_credentials (&es->cred))
277
!= GNUTLS_E_SUCCESS) {
278
fprintf (stderr, "memory error: %s\n",
279
safer_gnutls_strerror(ret));
438
fprintf(stderr, "Attempting to use OpenPGP public key %s and"
439
" secret key %s as GnuTLS credentials\n", pubkeyfilename,
284
fprintf(stderr, "Attempting to use OpenPGP certificate %s"
285
" and keyfile %s as GnuTLS credentials\n", certfile,
443
289
ret = gnutls_certificate_set_openpgp_key_file
444
(mc.cred, pubkeyfilename, seckeyfilename,
445
GNUTLS_OPENPGP_FMT_BASE64);
446
if(ret != GNUTLS_E_SUCCESS){
448
"Error[%d] while reading the OpenPGP key pair ('%s',"
449
" '%s')\n", ret, pubkeyfilename, seckeyfilename);
450
fprintf(stderr, "The GnuTLS error is: %s\n",
451
safer_gnutls_strerror(ret));
455
/* GnuTLS server initialization */
456
ret = gnutls_dh_params_init(&mc.dh_params);
457
if(ret != GNUTLS_E_SUCCESS){
458
fprintf(stderr, "Error in GnuTLS DH parameter initialization:"
459
" %s\n", safer_gnutls_strerror(ret));
462
ret = gnutls_dh_params_generate2(mc.dh_params, mc.dh_bits);
463
if(ret != GNUTLS_E_SUCCESS){
464
fprintf(stderr, "Error in GnuTLS prime generation: %s\n",
465
safer_gnutls_strerror(ret));
469
gnutls_certificate_set_dh_params(mc.cred, mc.dh_params);
475
gnutls_certificate_free_credentials(mc.cred);
476
gnutls_global_deinit();
477
gnutls_dh_params_deinit(mc.dh_params);
481
static int init_gnutls_session(gnutls_session_t *session){
483
/* GnuTLS session creation */
485
ret = gnutls_init(session, GNUTLS_SERVER);
489
} while(ret == GNUTLS_E_INTERRUPTED or ret == GNUTLS_E_AGAIN);
490
if(ret != GNUTLS_E_SUCCESS){
290
(es->cred, certfile, certkey, GNUTLS_OPENPGP_FMT_BASE64);
291
if (ret != GNUTLS_E_SUCCESS) {
293
(stderr, "Error[%d] while reading the OpenPGP key pair ('%s',"
295
ret, certfile, certkey);
296
fprintf(stdout, "The Error is: %s\n",
297
safer_gnutls_strerror(ret));
301
//GnuTLS server initialization
302
if ((ret = gnutls_dh_params_init (&es->dh_params))
303
!= GNUTLS_E_SUCCESS) {
304
fprintf (stderr, "Error in dh parameter initialization: %s\n",
305
safer_gnutls_strerror(ret));
309
if ((ret = gnutls_dh_params_generate2 (es->dh_params, DH_BITS))
310
!= GNUTLS_E_SUCCESS) {
311
fprintf (stderr, "Error in prime generation: %s\n",
312
safer_gnutls_strerror(ret));
316
gnutls_certificate_set_dh_params (es->cred, es->dh_params);
318
// GnuTLS session creation
319
if ((ret = gnutls_init (&es->session, GNUTLS_SERVER))
320
!= GNUTLS_E_SUCCESS){
491
321
fprintf(stderr, "Error in GnuTLS session initialization: %s\n",
492
322
safer_gnutls_strerror(ret));
498
ret = gnutls_priority_set_direct(*session, mc.priority, &err);
500
gnutls_deinit(*session);
503
} while(ret == GNUTLS_E_INTERRUPTED or ret == GNUTLS_E_AGAIN);
504
if(ret != GNUTLS_E_SUCCESS){
505
fprintf(stderr, "Syntax error at: %s\n", err);
506
fprintf(stderr, "GnuTLS error: %s\n",
507
safer_gnutls_strerror(ret));
508
gnutls_deinit(*session);
325
if ((ret = gnutls_priority_set_direct (es->session, "NORMAL", &err))
326
!= GNUTLS_E_SUCCESS) {
327
fprintf(stderr, "Syntax error at: %s\n", err);
328
fprintf(stderr, "GnuTLS error: %s\n",
329
safer_gnutls_strerror(ret));
514
ret = gnutls_credentials_set(*session, GNUTLS_CRD_CERTIFICATE,
517
gnutls_deinit(*session);
520
} while(ret == GNUTLS_E_INTERRUPTED or ret == GNUTLS_E_AGAIN);
521
if(ret != GNUTLS_E_SUCCESS){
522
fprintf(stderr, "Error setting GnuTLS credentials: %s\n",
333
if ((ret = gnutls_credentials_set
334
(es->session, GNUTLS_CRD_CERTIFICATE, es->cred))
335
!= GNUTLS_E_SUCCESS) {
336
fprintf(stderr, "Error setting a credentials set: %s\n",
523
337
safer_gnutls_strerror(ret));
524
gnutls_deinit(*session);
528
341
/* ignore client certificate if any. */
529
gnutls_certificate_server_set_request(*session, GNUTLS_CERT_IGNORE);
342
gnutls_certificate_server_set_request (es->session,
531
gnutls_dh_set_prime_bits(*session, mc.dh_bits);
345
gnutls_dh_set_prime_bits (es->session, DH_BITS);
536
/* Avahi log function callback */
537
350
static void empty_log(__attribute__((unused)) AvahiLogLevel level,
538
351
__attribute__((unused)) const char *txt){}
540
/* Called when a Mandos server is found */
541
353
static int start_mandos_communication(const char *ip, uint16_t port,
542
AvahiIfIndex if_index,
544
int ret, tcp_sd = -1;
547
struct sockaddr_in in;
548
struct sockaddr_in6 in6;
354
AvahiIfIndex if_index){
356
struct sockaddr_in6 to;
357
encrypted_session es;
550
358
char *buffer = NULL;
551
char *decrypted_buffer = NULL;
359
char *decrypted_buffer;
552
360
size_t buffer_length = 0;
553
361
size_t buffer_capacity = 0;
556
gnutls_session_t session;
557
int pf; /* Protocol family */
574
fprintf(stderr, "Bad address family: %d\n", af);
579
ret = init_gnutls_session(&session);
362
ssize_t decrypted_buffer_size;
365
char interface[IF_NAMESIZE];
585
fprintf(stderr, "Setting up a TCP connection to %s, port %" PRIu16
368
fprintf(stderr, "Setting up a tcp connection to %s, port %d\n",
589
tcp_sd = socket(pf, SOCK_STREAM, 0);
372
tcp_sd = socket(PF_INET6, SOCK_STREAM, 0);
592
374
perror("socket");
602
memset(&to, 0, sizeof(to));
604
to.in6.sin6_family = (sa_family_t)af;
605
ret = inet_pton(af, ip, &to.in6.sin6_addr);
607
to.in.sin_family = (sa_family_t)af;
608
ret = inet_pton(af, ip, &to.in.sin_addr);
379
if(if_indextoname((unsigned int)if_index, interface) == NULL){
381
perror("if_indextoname");
386
fprintf(stderr, "Binding to interface %s\n", interface);
389
memset(&to,0,sizeof(to)); /* Spurious warning */
390
to.sin6_family = AF_INET6;
391
ret = inet_pton(AF_INET6, ip, &to.sin6_addr);
612
393
perror("inet_pton");
618
397
fprintf(stderr, "Bad address: %s\n", ip);
623
to.in6.sin6_port = htons(port); /* Spurious warnings from
625
-Wunreachable-code */
627
if(IN6_IS_ADDR_LINKLOCAL /* Spurious warnings from */
628
(&to.in6.sin6_addr)){ /* -Wstrict-aliasing=2 or lower and
630
if(if_index == AVAHI_IF_UNSPEC){
631
fprintf(stderr, "An IPv6 link-local address is incomplete"
632
" without a network interface\n");
636
/* Set the network interface number as scope */
637
to.in6.sin6_scope_id = (uint32_t)if_index;
640
to.in.sin_port = htons(port); /* Spurious warnings from
642
-Wunreachable-code */
400
to.sin6_port = htons(port); /* Spurious warning */
402
to.sin6_scope_id = (uint32_t)if_index;
651
if(af == AF_INET6 and if_index != AVAHI_IF_UNSPEC){
652
char interface[IF_NAMESIZE];
653
if(if_indextoname((unsigned int)if_index, interface) == NULL){
654
perror("if_indextoname");
656
fprintf(stderr, "Connection to: %s%%%s, port %" PRIu16 "\n",
657
ip, interface, port);
660
fprintf(stderr, "Connection to: %s, port %" PRIu16 "\n", ip,
663
char addrstr[(INET_ADDRSTRLEN > INET6_ADDRSTRLEN) ?
664
INET_ADDRSTRLEN : INET6_ADDRSTRLEN] = "";
667
pcret = inet_ntop(af, &(to.in6.sin6_addr), addrstr,
670
pcret = inet_ntop(af, &(to.in.sin_addr), addrstr,
676
if(strcmp(addrstr, ip) != 0){
677
fprintf(stderr, "Canonical address form: %s\n", addrstr);
688
ret = connect(tcp_sd, &to.in6, sizeof(to));
690
ret = connect(tcp_sd, &to.in, sizeof(to)); /* IPv4 */
693
if ((errno != ECONNREFUSED and errno != ENETUNREACH) or debug){
706
const char *out = mandos_protocol_version;
709
size_t out_size = strlen(out);
710
ret = (int)TEMP_FAILURE_RETRY(write(tcp_sd, out + written,
711
out_size - written));
718
written += (size_t)ret;
719
if(written < out_size){
722
if(out == mandos_protocol_version){
405
fprintf(stderr, "Connection to: %s, port %d\n", ip, port);
406
/* char addrstr[INET6_ADDRSTRLEN]; */
407
/* if(inet_ntop(to.sin6_family, &(to.sin6_addr), addrstr, */
408
/* sizeof(addrstr)) == NULL){ */
409
/* perror("inet_ntop"); */
411
/* fprintf(stderr, "Really connecting to: %s, port %d\n", */
412
/* addrstr, ntohs(to.sin6_port)); */
416
ret = connect(tcp_sd, (struct sockaddr *) &to, sizeof(to));
422
ret = initgnutls (&es);
428
gnutls_transport_set_ptr (es.session,
429
(gnutls_transport_ptr_t) tcp_sd);
737
432
fprintf(stderr, "Establishing TLS session with %s\n", ip);
745
gnutls_transport_set_ptr(session, (gnutls_transport_ptr_t) tcp_sd);
753
ret = gnutls_handshake(session);
758
} while(ret == GNUTLS_E_AGAIN or ret == GNUTLS_E_INTERRUPTED);
760
if(ret != GNUTLS_E_SUCCESS){
435
ret = gnutls_handshake (es.session);
437
if (ret != GNUTLS_E_SUCCESS){
762
fprintf(stderr, "*** GnuTLS Handshake failed ***\n");
439
fprintf(stderr, "\n*** Handshake failed ***\n");
769
/* Read OpenPGP packet that contains the wanted password */
446
//Retrieve OpenPGP packet that contains the wanted password
772
fprintf(stderr, "Retrieving OpenPGP encrypted password from %s\n",
449
fprintf(stderr, "Retrieving pgp encrypted password from %s\n",
783
buffer_capacity = incbuffer(&buffer, buffer_length,
785
if(buffer_capacity == 0){
797
sret = gnutls_record_recv(session, buffer+buffer_length,
454
if (buffer_length + BUFFER_SIZE > buffer_capacity){
455
buffer = realloc(buffer, buffer_capacity + BUFFER_SIZE);
460
buffer_capacity += BUFFER_SIZE;
463
ret = gnutls_record_recv
464
(es.session, buffer+buffer_length, BUFFER_SIZE);
804
470
case GNUTLS_E_INTERRUPTED:
805
471
case GNUTLS_E_AGAIN:
807
473
case GNUTLS_E_REHANDSHAKE:
809
ret = gnutls_handshake(session);
815
} while(ret == GNUTLS_E_AGAIN or ret == GNUTLS_E_INTERRUPTED);
817
fprintf(stderr, "*** GnuTLS Re-handshake failed ***\n");
474
ret = gnutls_handshake (es.session);
476
fprintf(stderr, "\n*** Handshake failed ***\n");
824
483
fprintf(stderr, "Unknown error while reading data from"
825
" encrypted session with Mandos server\n");
826
gnutls_bye(session, GNUTLS_SHUT_RDWR);
484
" encrypted session with mandos server\n");
486
gnutls_bye (es.session, GNUTLS_SHUT_RDWR);
831
buffer_length += (size_t) sret;
836
fprintf(stderr, "Closing TLS session\n");
845
ret = gnutls_bye(session, GNUTLS_SHUT_RDWR);
850
} while(ret == GNUTLS_E_AGAIN or ret == GNUTLS_E_INTERRUPTED);
852
if(buffer_length > 0){
853
ssize_t decrypted_buffer_size;
490
buffer_length += (size_t) ret;
494
if (buffer_length > 0){
854
495
decrypted_buffer_size = pgp_packet_decrypt(buffer,
857
if(decrypted_buffer_size >= 0){
499
if (decrypted_buffer_size >= 0){
860
500
while(written < (size_t) decrypted_buffer_size){
866
ret = (int)fwrite(decrypted_buffer + written, 1,
867
(size_t)decrypted_buffer_size - written,
501
ret = (int)fwrite (decrypted_buffer + written, 1,
502
(size_t)decrypted_buffer_size - written,
869
504
if(ret == 0 and ferror(stdout)){
872
506
fprintf(stderr, "Error writing encrypted data: %s\n",
873
507
strerror(errno));
878
512
written += (size_t)ret;
884
/* Shutdown procedure */
889
free(decrypted_buffer);
892
ret = (int)TEMP_FAILURE_RETRY(close(tcp_sd));
900
gnutls_deinit(session);
514
free(decrypted_buffer);
523
fprintf(stderr, "Closing TLS session\n");
527
gnutls_bye (es.session, GNUTLS_SHUT_RDWR);
530
gnutls_deinit (es.session);
531
gnutls_certificate_free_credentials (es.cred);
532
gnutls_global_deinit ();
910
static void resolve_callback(AvahiSServiceResolver *r,
911
AvahiIfIndex interface,
913
AvahiResolverEvent event,
917
const char *host_name,
918
const AvahiAddress *address,
920
AVAHI_GCC_UNUSED AvahiStringList *txt,
921
AVAHI_GCC_UNUSED AvahiLookupResultFlags
923
AVAHI_GCC_UNUSED void* userdata){
536
static AvahiSimplePoll *simple_poll = NULL;
537
static AvahiServer *server = NULL;
539
static void resolve_callback(
540
AvahiSServiceResolver *r,
541
AvahiIfIndex interface,
542
AVAHI_GCC_UNUSED AvahiProtocol protocol,
543
AvahiResolverEvent event,
547
const char *host_name,
548
const AvahiAddress *address,
550
AVAHI_GCC_UNUSED AvahiStringList *txt,
551
AVAHI_GCC_UNUSED AvahiLookupResultFlags flags,
552
AVAHI_GCC_UNUSED void* userdata) {
554
assert(r); /* Spurious warning */
926
556
/* Called whenever a service has been resolved successfully or
935
561
case AVAHI_RESOLVER_FAILURE:
936
fprintf(stderr, "(Avahi Resolver) Failed to resolve service '%s'"
937
" of type '%s' in domain '%s': %s\n", name, type, domain,
938
avahi_strerror(avahi_server_errno(mc.server)));
562
fprintf(stderr, "(Resolver) Failed to resolve service '%s' of"
563
" type '%s' in domain '%s': %s\n", name, type, domain,
564
avahi_strerror(avahi_server_errno(server)));
941
567
case AVAHI_RESOLVER_FOUND:
943
569
char ip[AVAHI_ADDRESS_STR_MAX];
944
570
avahi_address_snprint(ip, sizeof(ip), address);
946
fprintf(stderr, "Mandos server \"%s\" found on %s (%s, %"
947
PRIdMAX ") on port %" PRIu16 "\n", name, host_name,
948
ip, (intmax_t)interface, port);
572
fprintf(stderr, "Mandos server \"%s\" found on %s (%s) on"
573
" port %d\n", name, host_name, ip, port);
950
int ret = start_mandos_communication(ip, port, interface,
951
avahi_proto_to_af(proto));
953
avahi_simple_poll_quit(mc.simple_poll);
575
int ret = start_mandos_communication(ip, port, interface);
957
581
avahi_s_service_resolver_free(r);
960
static void browse_callback(AvahiSServiceBrowser *b,
961
AvahiIfIndex interface,
962
AvahiProtocol protocol,
963
AvahiBrowserEvent event,
967
AVAHI_GCC_UNUSED AvahiLookupResultFlags
969
AVAHI_GCC_UNUSED void* userdata){
972
/* Called whenever a new services becomes available on the LAN or
973
is removed from the LAN */
981
case AVAHI_BROWSER_FAILURE:
983
fprintf(stderr, "(Avahi browser) %s\n",
984
avahi_strerror(avahi_server_errno(mc.server)));
985
avahi_simple_poll_quit(mc.simple_poll);
988
case AVAHI_BROWSER_NEW:
989
/* We ignore the returned Avahi resolver object. In the callback
990
function we free it. If the Avahi server is terminated before
991
the callback function is called the Avahi server will free the
994
if(avahi_s_service_resolver_new(mc.server, interface, protocol,
995
name, type, domain, protocol, 0,
996
resolve_callback, NULL) == NULL)
997
fprintf(stderr, "Avahi: Failed to resolve service '%s': %s\n",
998
name, avahi_strerror(avahi_server_errno(mc.server)));
1001
case AVAHI_BROWSER_REMOVE:
1004
case AVAHI_BROWSER_ALL_FOR_NOW:
1005
case AVAHI_BROWSER_CACHE_EXHAUSTED:
1007
fprintf(stderr, "No Mandos server found, still searching...\n");
1013
/* stop main loop after sigterm has been called */
1014
static void handle_sigterm(int sig){
1019
signal_received = sig;
1020
int old_errno = errno;
1021
if(mc.simple_poll != NULL){
1022
avahi_simple_poll_quit(mc.simple_poll);
1028
* This function determines if a directory entry in /sys/class/net
1029
* corresponds to an acceptable network device.
1030
* (This function is passed to scandir(3) as a filter function.)
1032
int good_interface(const struct dirent *if_entry){
1034
char *flagname = NULL;
1035
if(if_entry->d_name[0] == '.'){
1038
int ret = asprintf(&flagname, "%s/%s/flags", sys_class_net,
1044
int flags_fd = (int)TEMP_FAILURE_RETRY(open(flagname, O_RDONLY));
1051
typedef short ifreq_flags; /* ifreq.ifr_flags in netdevice(7) */
1052
/* read line from flags_fd */
1053
ssize_t to_read = (sizeof(ifreq_flags)*2)+3; /* "0x1003\n" */
1054
char *flagstring = malloc((size_t)to_read+1); /* +1 for final \0 */
1055
flagstring[(size_t)to_read] = '\0';
1056
if(flagstring == NULL){
1062
ssret = (ssize_t)TEMP_FAILURE_RETRY(read(flags_fd, flagstring,
1079
tmpmax = strtoimax(flagstring, &tmp, 0);
1080
if(errno != 0 or tmp == flagstring or (*tmp != '\0'
1081
and not (isspace(*tmp)))
1082
or tmpmax != (ifreq_flags)tmpmax){
1084
fprintf(stderr, "Invalid flags \"%s\" for interface \"%s\"\n",
1085
flagstring, if_entry->d_name);
1091
ifreq_flags flags = (ifreq_flags)tmpmax;
1092
/* Reject the loopback device */
1093
if(flags & IFF_LOOPBACK){
1095
fprintf(stderr, "Rejecting loopback interface \"%s\"\n",
1100
/* Accept point-to-point devices only if connect_to is specified */
1101
if(connect_to != NULL and (flags & IFF_POINTOPOINT)){
1103
fprintf(stderr, "Accepting point-to-point interface \"%s\"\n",
1108
/* Otherwise, reject non-broadcast-capable devices */
1109
if(not (flags & IFF_BROADCAST)){
1111
fprintf(stderr, "Rejecting non-broadcast interface \"%s\"\n",
1116
/* Accept this device */
1118
fprintf(stderr, "Interface \"%s\" is acceptable\n",
1124
int main(int argc, char *argv[]){
1125
AvahiSServiceBrowser *sb = NULL;
1130
int exitcode = EXIT_SUCCESS;
1131
const char *interface = "";
1132
struct ifreq network;
1134
bool take_down_interface = false;
1137
char tempdir[] = "/tmp/mandosXXXXXX";
1138
bool tempdir_created = false;
1139
AvahiIfIndex if_index = AVAHI_IF_UNSPEC;
1140
const char *seckey = PATHDIR "/" SECKEY;
1141
const char *pubkey = PATHDIR "/" PUBKEY;
1143
bool gnutls_initialized = false;
1144
bool gpgme_initialized = false;
1147
struct sigaction old_sigterm_action = { .sa_handler = SIG_DFL };
1148
struct sigaction sigterm_action = { .sa_handler = handle_sigterm };
1153
/* Lower any group privileges we might have, just to be safe */
1160
/* Lower user privileges (temporarily) */
1172
struct argp_option options[] = {
1173
{ .name = "debug", .key = 128,
1174
.doc = "Debug mode", .group = 3 },
1175
{ .name = "connect", .key = 'c',
1176
.arg = "ADDRESS:PORT",
1177
.doc = "Connect directly to a specific Mandos server",
1179
{ .name = "interface", .key = 'i',
1181
.doc = "Network interface that will be used to search for"
1184
{ .name = "seckey", .key = 's',
1186
.doc = "OpenPGP secret key file base name",
1188
{ .name = "pubkey", .key = 'p',
1190
.doc = "OpenPGP public key file base name",
1192
{ .name = "dh-bits", .key = 129,
1194
.doc = "Bit length of the prime number used in the"
1195
" Diffie-Hellman key exchange",
1197
{ .name = "priority", .key = 130,
1199
.doc = "GnuTLS priority string for the TLS handshake",
1201
{ .name = "delay", .key = 131,
1203
.doc = "Maximum delay to wait for interface startup",
1206
* These reproduce what we would get without ARGP_NO_HELP
1208
{ .name = "help", .key = '?',
1209
.doc = "Give this help list", .group = -1 },
1210
{ .name = "usage", .key = -3,
1211
.doc = "Give a short usage message", .group = -1 },
1212
{ .name = "version", .key = 'V',
1213
.doc = "Print program version", .group = -1 },
1217
error_t parse_opt(int key, char *arg,
1218
struct argp_state *state){
1221
case 128: /* --debug */
1224
case 'c': /* --connect */
1227
case 'i': /* --interface */
1230
case 's': /* --seckey */
1233
case 'p': /* --pubkey */
1236
case 129: /* --dh-bits */
1238
tmpmax = strtoimax(arg, &tmp, 10);
1239
if(errno != 0 or tmp == arg or *tmp != '\0'
1240
or tmpmax != (typeof(mc.dh_bits))tmpmax){
1241
argp_error(state, "Bad number of DH bits");
1243
mc.dh_bits = (typeof(mc.dh_bits))tmpmax;
1245
case 130: /* --priority */
1248
case 131: /* --delay */
1250
delay = strtof(arg, &tmp);
1251
if(errno != 0 or tmp == arg or *tmp != '\0'){
1252
argp_error(state, "Bad delay");
1256
* These reproduce what we would get without ARGP_NO_HELP
1258
case '?': /* --help */
1259
argp_state_help(state, state->out_stream,
1260
(ARGP_HELP_STD_HELP | ARGP_HELP_EXIT_ERR)
1261
& ~(unsigned int)ARGP_HELP_EXIT_OK);
1262
case -3: /* --usage */
1263
argp_state_help(state, state->out_stream,
1264
ARGP_HELP_USAGE | ARGP_HELP_EXIT_ERR);
1265
case 'V': /* --version */
1266
fprintf(state->out_stream, "%s\n", argp_program_version);
1267
exit(argp_err_exit_status);
1270
return ARGP_ERR_UNKNOWN;
1275
struct argp argp = { .options = options, .parser = parse_opt,
1277
.doc = "Mandos client -- Get and decrypt"
1278
" passwords from a Mandos server" };
1279
ret = argp_parse(&argp, argc, argv,
1280
ARGP_IN_ORDER | ARGP_NO_HELP, 0, NULL);
584
static void browse_callback(
585
AvahiSServiceBrowser *b,
586
AvahiIfIndex interface,
587
AvahiProtocol protocol,
588
AvahiBrowserEvent event,
592
AVAHI_GCC_UNUSED AvahiLookupResultFlags flags,
595
AvahiServer *s = userdata;
596
assert(b); /* Spurious warning */
598
/* Called whenever a new services becomes available on the LAN or
599
is removed from the LAN */
1287
perror("argp_parse");
1288
exitcode = EX_OSERR;
1291
exitcode = EX_USAGE;
1297
avahi_set_log_function(empty_log);
1300
if(interface[0] == '\0'){
1301
struct dirent **direntries;
1302
ret = scandir(sys_class_net, &direntries, good_interface,
1305
/* Pick the first good interface */
1306
interface = strdup(direntries[0]->d_name);
1308
fprintf(stderr, "Using interface \"%s\"\n", interface);
1310
if(interface == NULL){
1313
exitcode = EXIT_FAILURE;
1319
fprintf(stderr, "Could not find a network interface\n");
1320
exitcode = EXIT_FAILURE;
1325
/* Initialize Avahi early so avahi_simple_poll_quit() can be called
1326
from the signal handler */
1327
/* Initialize the pseudo-RNG for Avahi */
1328
srand((unsigned int) time(NULL));
1329
mc.simple_poll = avahi_simple_poll_new();
1330
if(mc.simple_poll == NULL){
1331
fprintf(stderr, "Avahi: Failed to create simple poll object.\n");
1332
exitcode = EX_UNAVAILABLE;
1336
sigemptyset(&sigterm_action.sa_mask);
1337
ret = sigaddset(&sigterm_action.sa_mask, SIGINT);
1339
perror("sigaddset");
1340
exitcode = EX_OSERR;
1343
ret = sigaddset(&sigterm_action.sa_mask, SIGHUP);
1345
perror("sigaddset");
1346
exitcode = EX_OSERR;
1349
ret = sigaddset(&sigterm_action.sa_mask, SIGTERM);
1351
perror("sigaddset");
1352
exitcode = EX_OSERR;
1355
/* Need to check if the handler is SIG_IGN before handling:
1356
| [[info:libc:Initial Signal Actions]] |
1357
| [[info:libc:Basic Signal Handling]] |
1359
ret = sigaction(SIGINT, NULL, &old_sigterm_action);
1361
perror("sigaction");
1364
if(old_sigterm_action.sa_handler != SIG_IGN){
1365
ret = sigaction(SIGINT, &sigterm_action, NULL);
1367
perror("sigaction");
1368
exitcode = EX_OSERR;
1372
ret = sigaction(SIGHUP, NULL, &old_sigterm_action);
1374
perror("sigaction");
1377
if(old_sigterm_action.sa_handler != SIG_IGN){
1378
ret = sigaction(SIGHUP, &sigterm_action, NULL);
1380
perror("sigaction");
1381
exitcode = EX_OSERR;
1385
ret = sigaction(SIGTERM, NULL, &old_sigterm_action);
1387
perror("sigaction");
1390
if(old_sigterm_action.sa_handler != SIG_IGN){
1391
ret = sigaction(SIGTERM, &sigterm_action, NULL);
1393
perror("sigaction");
1394
exitcode = EX_OSERR;
1399
/* If the interface is down, bring it up */
1400
if(strcmp(interface, "none") != 0){
603
case AVAHI_BROWSER_FAILURE:
605
fprintf(stderr, "(Browser) %s\n",
606
avahi_strerror(avahi_server_errno(server)));
607
avahi_simple_poll_quit(simple_poll);
610
case AVAHI_BROWSER_NEW:
611
/* We ignore the returned resolver object. In the callback
612
function we free it. If the server is terminated before
613
the callback function is called the server will free
614
the resolver for us. */
616
if (!(avahi_s_service_resolver_new(s, interface, protocol, name,
618
AVAHI_PROTO_INET6, 0,
619
resolve_callback, s)))
620
fprintf(stderr, "Failed to resolve service '%s': %s\n", name,
621
avahi_strerror(avahi_server_errno(s)));
624
case AVAHI_BROWSER_REMOVE:
627
case AVAHI_BROWSER_ALL_FOR_NOW:
628
case AVAHI_BROWSER_CACHE_EXHAUSTED:
633
/* Combines file name and path and returns the malloced new
634
string. some sane checks could/should be added */
635
static const char *combinepath(const char *first, const char *second){
636
size_t f_len = strlen(first);
637
size_t s_len = strlen(second);
638
char *tmp = malloc(f_len + s_len + 2);
643
memcpy(tmp, first, f_len);
647
memcpy(tmp + f_len + 1, second, s_len);
649
tmp[f_len + 1 + s_len] = '\0';
654
int main(AVAHI_GCC_UNUSED int argc, AVAHI_GCC_UNUSED char*argv[]) {
655
AvahiServerConfig config;
656
AvahiSServiceBrowser *sb = NULL;
659
int returncode = EXIT_SUCCESS;
660
const char *interface = "eth0";
661
struct ifreq network;
663
char *connect_to = NULL;
664
AvahiIfIndex if_index = AVAHI_IF_UNSPEC;
667
static struct option long_options[] = {
668
{"debug", no_argument, (int *)&debug, 1},
669
{"connect", required_argument, 0, 'C'},
670
{"interface", required_argument, 0, 'i'},
671
{"certdir", required_argument, 0, 'd'},
672
{"certkey", required_argument, 0, 'c'},
673
{"certfile", required_argument, 0, 'k'},
676
int option_index = 0;
677
ret = getopt_long (argc, argv, "i:", long_options,
707
certfile = combinepath(certdir, certfile);
708
if (certfile == NULL){
709
perror("combinepath");
710
returncode = EXIT_FAILURE;
714
certkey = combinepath(certdir, certkey);
715
if (certkey == NULL){
716
perror("combinepath");
717
returncode = EXIT_FAILURE;
1401
721
if_index = (AvahiIfIndex) if_nametoindex(interface);
1402
722
if(if_index == 0){
1403
723
fprintf(stderr, "No such interface: \"%s\"\n", interface);
1404
exitcode = EX_UNAVAILABLE;
1412
/* Re-raise priviliges */
1420
/* Lower kernel loglevel to KERN_NOTICE to avoid KERN_INFO
1421
messages about the network interface to mess up the prompt */
1422
ret = klogctl(8, NULL, 5);
1423
bool restore_loglevel = true;
1425
restore_loglevel = false;
1428
#endif /* __linux__ */
727
if(connect_to != NULL){
728
/* Connect directly, do not use Zeroconf */
729
/* (Mainly meant for debugging) */
730
char *address = strrchr(connect_to, ':');
732
fprintf(stderr, "No colon in address\n");
736
uint16_t port = (uint16_t) strtol(address+1, NULL, 10);
738
perror("Bad port number");
742
address = connect_to;
743
ret = start_mandos_communication(address, port, if_index);
1430
751
sd = socket(PF_INET6, SOCK_DGRAM, IPPROTO_IP);
1432
753
perror("socket");
1433
exitcode = EX_OSERR;
1435
if(restore_loglevel){
1436
ret = klogctl(7, NULL, 0);
1441
#endif /* __linux__ */
1442
/* Lower privileges */
754
returncode = EXIT_FAILURE;
1450
strcpy(network.ifr_name, interface);
757
strcpy(network.ifr_name, interface);
1451
758
ret = ioctl(sd, SIOCGIFFLAGS, &network);
1453
761
perror("ioctl SIOCGIFFLAGS");
1455
if(restore_loglevel){
1456
ret = klogctl(7, NULL, 0);
1461
#endif /* __linux__ */
1462
exitcode = EX_OSERR;
1463
/* Lower privileges */
762
returncode = EXIT_FAILURE;
1471
765
if((network.ifr_flags & IFF_UP) == 0){
1472
766
network.ifr_flags |= IFF_UP;
1473
take_down_interface = true;
1474
767
ret = ioctl(sd, SIOCSIFFLAGS, &network);
1476
take_down_interface = false;
1477
perror("ioctl SIOCSIFFLAGS +IFF_UP");
1478
exitcode = EX_OSERR;
1480
if(restore_loglevel){
1481
ret = klogctl(7, NULL, 0);
1486
#endif /* __linux__ */
1487
/* Lower privileges */
1496
/* sleep checking until interface is running */
1497
for(int i=0; i < delay * 4; i++){
1498
ret = ioctl(sd, SIOCGIFFLAGS, &network);
1500
perror("ioctl SIOCGIFFLAGS");
1501
} else if(network.ifr_flags & IFF_RUNNING){
1504
struct timespec sleeptime = { .tv_nsec = 250000000 };
1505
ret = nanosleep(&sleeptime, NULL);
1506
if(ret == -1 and errno != EINTR){
1507
perror("nanosleep");
1510
if(not take_down_interface){
1511
/* We won't need the socket anymore */
1512
ret = (int)TEMP_FAILURE_RETRY(close(sd));
1518
if(restore_loglevel){
1519
/* Restores kernel loglevel to default */
1520
ret = klogctl(7, NULL, 0);
1525
#endif /* __linux__ */
1526
/* Lower privileges */
1528
if(take_down_interface){
1529
/* Lower privileges */
1535
/* Lower privileges permanently */
1547
ret = init_gnutls_global(pubkey, seckey);
1549
fprintf(stderr, "init_gnutls_global failed\n");
1550
exitcode = EX_UNAVAILABLE;
1553
gnutls_initialized = true;
1560
tempdir_created = true;
1561
if(mkdtemp(tempdir) == NULL){
1562
tempdir_created = false;
1571
if(not init_gpgme(pubkey, seckey, tempdir)){
1572
fprintf(stderr, "init_gpgme failed\n");
1573
exitcode = EX_UNAVAILABLE;
1576
gpgme_initialized = true;
1583
if(connect_to != NULL){
1584
/* Connect directly, do not use Zeroconf */
1585
/* (Mainly meant for debugging) */
1586
char *address = strrchr(connect_to, ':');
1587
if(address == NULL){
1588
fprintf(stderr, "No colon in address\n");
1589
exitcode = EX_USAGE;
1599
tmpmax = strtoimax(address+1, &tmp, 10);
1600
if(errno != 0 or tmp == address+1 or *tmp != '\0'
1601
or tmpmax != (uint16_t)tmpmax){
1602
fprintf(stderr, "Bad port number\n");
1603
exitcode = EX_USAGE;
1611
port = (uint16_t)tmpmax;
1613
address = connect_to;
1614
/* Colon in address indicates IPv6 */
1616
if(strchr(address, ':') != NULL){
1626
while(not quit_now){
1627
ret = start_mandos_communication(address, port, if_index, af);
1628
if(quit_now or ret == 0){
1635
exitcode = EXIT_SUCCESS;
1646
AvahiServerConfig config;
1647
/* Do not publish any local Zeroconf records */
769
perror("ioctl SIOCSIFFLAGS");
770
returncode = EXIT_FAILURE;
777
avahi_set_log_function(empty_log);
780
/* Initialize the psuedo-RNG */
781
srand((unsigned int) time(NULL));
783
/* Allocate main loop object */
784
if (!(simple_poll = avahi_simple_poll_new())) {
785
fprintf(stderr, "Failed to create simple poll object.\n");
786
returncode = EXIT_FAILURE;
790
/* Do not publish any local records */
1648
791
avahi_server_config_init(&config);
1649
792
config.publish_hinfo = 0;
1650
793
config.publish_addresses = 0;
1651
794
config.publish_workstation = 0;
1652
795
config.publish_domain = 0;
1654
797
/* Allocate a new server */
1655
mc.server = avahi_server_new(avahi_simple_poll_get
1656
(mc.simple_poll), &config, NULL,
1659
/* Free the Avahi configuration data */
798
server = avahi_server_new(avahi_simple_poll_get(simple_poll),
799
&config, NULL, NULL, &error);
801
/* Free the configuration data */
1660
802
avahi_server_config_free(&config);
1663
/* Check if creating the Avahi server object succeeded */
1664
if(mc.server == NULL){
1665
fprintf(stderr, "Failed to create Avahi server: %s\n",
1666
avahi_strerror(error));
1667
exitcode = EX_UNAVAILABLE;
1675
/* Create the Avahi service browser */
1676
sb = avahi_s_service_browser_new(mc.server, if_index,
1677
AVAHI_PROTO_UNSPEC, "_mandos._tcp",
1678
NULL, 0, browse_callback, NULL);
1680
fprintf(stderr, "Failed to create service browser: %s\n",
1681
avahi_strerror(avahi_server_errno(mc.server)));
1682
exitcode = EX_UNAVAILABLE;
1690
/* Run the main loop */
1693
fprintf(stderr, "Starting Avahi loop search\n");
1696
avahi_simple_poll_loop(mc.simple_poll);
1701
fprintf(stderr, "%s exiting\n", argv[0]);
1704
/* Cleanup things */
1706
avahi_s_service_browser_free(sb);
1708
if(mc.server != NULL)
1709
avahi_server_free(mc.server);
1711
if(mc.simple_poll != NULL)
1712
avahi_simple_poll_free(mc.simple_poll);
1714
if(gnutls_initialized){
1715
gnutls_certificate_free_credentials(mc.cred);
1716
gnutls_global_deinit();
1717
gnutls_dh_params_deinit(mc.dh_params);
1720
if(gpgme_initialized){
1721
gpgme_release(mc.ctx);
1724
/* Take down the network interface */
1725
if(take_down_interface){
1726
/* Re-raise priviliges */
1733
ret = ioctl(sd, SIOCGIFFLAGS, &network);
1735
perror("ioctl SIOCGIFFLAGS");
1736
} else if(network.ifr_flags & IFF_UP) {
1737
network.ifr_flags &= ~(short)IFF_UP; /* clear flag */
1738
ret = ioctl(sd, SIOCSIFFLAGS, &network);
1740
perror("ioctl SIOCSIFFLAGS -IFF_UP");
1743
ret = (int)TEMP_FAILURE_RETRY(close(sd));
1747
/* Lower privileges permanently */
1756
/* Removes the temp directory used by GPGME */
1757
if(tempdir_created){
1759
struct dirent *direntry;
1760
d = opendir(tempdir);
1762
if(errno != ENOENT){
1767
direntry = readdir(d);
1768
if(direntry == NULL){
1771
/* Skip "." and ".." */
1772
if(direntry->d_name[0] == '.'
1773
and (direntry->d_name[1] == '\0'
1774
or (direntry->d_name[1] == '.'
1775
and direntry->d_name[2] == '\0'))){
1778
char *fullname = NULL;
1779
ret = asprintf(&fullname, "%s/%s", tempdir,
1785
ret = remove(fullname);
1787
fprintf(stderr, "remove(\"%s\"): %s\n", fullname,
1794
ret = rmdir(tempdir);
1795
if(ret == -1 and errno != ENOENT){
1801
sigemptyset(&old_sigterm_action.sa_mask);
1802
old_sigterm_action.sa_handler = SIG_DFL;
1803
ret = (int)TEMP_FAILURE_RETRY(sigaction(signal_received,
1804
&old_sigterm_action,
1807
perror("sigaction");
1810
ret = raise(signal_received);
1811
} while(ret != 0 and errno == EINTR);
1816
TEMP_FAILURE_RETRY(pause());
804
/* Check if creating the server object succeeded */
806
fprintf(stderr, "Failed to create server: %s\n",
807
avahi_strerror(error));
808
returncode = EXIT_FAILURE;
812
/* Create the service browser */
813
sb = avahi_s_service_browser_new(server, if_index,
815
"_mandos._tcp", NULL, 0,
816
browse_callback, server);
818
fprintf(stderr, "Failed to create service browser: %s\n",
819
avahi_strerror(avahi_server_errno(server)));
820
returncode = EXIT_FAILURE;
824
/* Run the main loop */
827
fprintf(stderr, "Starting avahi loop search\n");
830
avahi_simple_poll_loop(simple_poll);
835
fprintf(stderr, "%s exiting\n", argv[0]);
840
avahi_s_service_browser_free(sb);
843
avahi_server_free(server);
846
avahi_simple_poll_free(simple_poll);