223
void debuggnutls(int level, const char* string){
224
fprintf(stderr, "%s", string);
287
/* GnuTLS log function callback */
288
static void debuggnutls(__attribute__((unused)) int level,
290
fprintf(stderr, "GnuTLS: %s", string);
227
int initgnutls(encrypted_session *es){
293
static int init_gnutls_global(mandos_context *mc,
294
const char *pubkeyfile,
295
const char *seckeyfile){
232
fprintf(stderr, "Initializing gnutls\n");
299
fprintf(stderr, "Initializing GnuTLS\n");
236
302
if ((ret = gnutls_global_init ())
237
303
!= GNUTLS_E_SUCCESS) {
238
fprintf (stderr, "global_init: %s\n", safer_gnutls_strerror(ret));
304
fprintf (stderr, "GnuTLS global_init: %s\n",
305
safer_gnutls_strerror(ret));
310
/* "Use a log level over 10 to enable all debugging options."
243
313
gnutls_global_set_log_level(11);
244
314
gnutls_global_set_log_function(debuggnutls);
248
/* openpgp credentials */
249
if ((ret = gnutls_certificate_allocate_credentials (&es->cred))
317
/* OpenPGP credentials */
318
if ((ret = gnutls_certificate_allocate_credentials (&mc->cred))
250
319
!= GNUTLS_E_SUCCESS) {
251
fprintf (stderr, "memory error: %s\n", safer_gnutls_strerror(ret));
320
fprintf (stderr, "GnuTLS memory error: %s\n",
321
safer_gnutls_strerror(ret));
256
fprintf(stderr, "Attempting to use openpgp certificate %s"
257
" and keyfile %s as gnutls credentials\n", CERTFILE, KEYFILE);
326
fprintf(stderr, "Attempting to use OpenPGP certificate %s"
327
" and keyfile %s as GnuTLS credentials\n", pubkeyfile,
260
331
ret = gnutls_certificate_set_openpgp_key_file
261
(es->cred, CERTFILE, KEYFILE, GNUTLS_OPENPGP_FMT_BASE64);
262
if (ret != GNUTLS_E_SUCCESS) {
264
(stderr, "Error[%d] while reading the OpenPGP key pair ('%s', '%s')\n",
265
ret, CERTFILE, KEYFILE);
266
fprintf(stdout, "The Error is: %s\n",
267
safer_gnutls_strerror(ret));
271
//Gnutls server initialization
272
if ((ret = gnutls_dh_params_init (&es->dh_params))
273
!= GNUTLS_E_SUCCESS) {
274
fprintf (stderr, "Error in dh parameter initialization: %s\n",
275
safer_gnutls_strerror(ret));
279
if ((ret = gnutls_dh_params_generate2 (es->dh_params, DH_BITS))
280
!= GNUTLS_E_SUCCESS) {
281
fprintf (stderr, "Error in prime generation: %s\n",
282
safer_gnutls_strerror(ret));
286
gnutls_certificate_set_dh_params (es->cred, es->dh_params);
288
// Gnutls session creation
289
if ((ret = gnutls_init (&es->session, GNUTLS_SERVER))
290
!= GNUTLS_E_SUCCESS){
291
fprintf(stderr, "Error in gnutls session initialization: %s\n",
292
safer_gnutls_strerror(ret));
295
if ((ret = gnutls_priority_set_direct (es->session, "NORMAL", &err))
296
!= GNUTLS_E_SUCCESS) {
297
fprintf(stderr, "Syntax error at: %s\n", err);
298
fprintf(stderr, "Gnutls error: %s\n",
299
safer_gnutls_strerror(ret));
303
if ((ret = gnutls_credentials_set
304
(es->session, GNUTLS_CRD_CERTIFICATE, es->cred))
305
!= GNUTLS_E_SUCCESS) {
306
fprintf(stderr, "Error setting a credentials set: %s\n",
307
safer_gnutls_strerror(ret));
332
(mc->cred, pubkeyfile, seckeyfile, GNUTLS_OPENPGP_FMT_BASE64);
333
if (ret != GNUTLS_E_SUCCESS) {
335
"Error[%d] while reading the OpenPGP key pair ('%s',"
336
" '%s')\n", ret, pubkeyfile, seckeyfile);
337
fprintf(stdout, "The GnuTLS error is: %s\n",
338
safer_gnutls_strerror(ret));
342
/* GnuTLS server initialization */
343
ret = gnutls_dh_params_init(&mc->dh_params);
344
if (ret != GNUTLS_E_SUCCESS) {
345
fprintf (stderr, "Error in GnuTLS DH parameter initialization:"
346
" %s\n", safer_gnutls_strerror(ret));
349
ret = gnutls_dh_params_generate2(mc->dh_params, mc->dh_bits);
350
if (ret != GNUTLS_E_SUCCESS) {
351
fprintf (stderr, "Error in GnuTLS prime generation: %s\n",
352
safer_gnutls_strerror(ret));
356
gnutls_certificate_set_dh_params(mc->cred, mc->dh_params);
361
static int init_gnutls_session(mandos_context *mc,
362
gnutls_session_t *session){
364
/* GnuTLS session creation */
365
ret = gnutls_init(session, GNUTLS_SERVER);
366
if (ret != GNUTLS_E_SUCCESS){
367
fprintf(stderr, "Error in GnuTLS session initialization: %s\n",
368
safer_gnutls_strerror(ret));
373
ret = gnutls_priority_set_direct(*session, mc->priority, &err);
374
if (ret != GNUTLS_E_SUCCESS) {
375
fprintf(stderr, "Syntax error at: %s\n", err);
376
fprintf(stderr, "GnuTLS error: %s\n",
377
safer_gnutls_strerror(ret));
382
ret = gnutls_credentials_set(*session, GNUTLS_CRD_CERTIFICATE,
384
if (ret != GNUTLS_E_SUCCESS) {
385
fprintf(stderr, "Error setting GnuTLS credentials: %s\n",
386
safer_gnutls_strerror(ret));
311
390
/* ignore client certificate if any. */
312
gnutls_certificate_server_set_request (es->session, GNUTLS_CERT_IGNORE);
391
gnutls_certificate_server_set_request (*session,
314
gnutls_dh_set_prime_bits (es->session, DH_BITS);
394
gnutls_dh_set_prime_bits (*session, mc->dh_bits);
319
void empty_log(AvahiLogLevel level, const char *txt){}
399
/* Avahi log function callback */
400
static void empty_log(__attribute__((unused)) AvahiLogLevel level,
401
__attribute__((unused)) const char *txt){}
321
int start_mandos_communcation(char *ip, uint16_t port){
403
/* Called when a Mandos server is found */
404
static int start_mandos_communication(const char *ip, uint16_t port,
405
AvahiIfIndex if_index,
323
struct sockaddr_in6 to;
324
struct in6_addr ip_addr;
325
encrypted_session es;
408
union { struct sockaddr in; struct sockaddr_in6 in6; } to;
326
409
char *buffer = NULL;
327
410
char *decrypted_buffer;
328
411
size_t buffer_length = 0;
329
412
size_t buffer_capacity = 0;
330
413
ssize_t decrypted_buffer_size;
416
char interface[IF_NAMESIZE];
417
gnutls_session_t session;
419
ret = init_gnutls_session (mc, &session);
334
fprintf(stderr, "Setting up a tcp connection to %s\n", ip);
425
fprintf(stderr, "Setting up a tcp connection to %s, port %d\n",
337
429
tcp_sd = socket(PF_INET6, SOCK_STREAM, 0);
436
if(if_indextoname((unsigned int)if_index, interface) == NULL){
437
perror("if_indextoname");
344
440
fprintf(stderr, "Binding to interface %s\n", interface);
347
ret = setsockopt(tcp_sd, SOL_SOCKET, SO_BINDTODEVICE, interface, 5);
349
perror("setsockopt bindtodevice");
353
memset(&to,0,sizeof(to));
354
to.sin6_family = AF_INET6;
355
ret = inet_pton(AF_INET6, ip, &ip_addr);
443
memset(&to,0,sizeof(to)); /* Spurious warning */
444
to.in6.sin6_family = AF_INET6;
445
/* It would be nice to have a way to detect if we were passed an
446
IPv4 address here. Now we assume an IPv6 address. */
447
ret = inet_pton(AF_INET6, ip, &to.in6.sin6_addr);
357
449
perror("inet_pton");
361
453
fprintf(stderr, "Bad address: %s\n", ip);
364
to.sin6_port = htons(port);
365
to.sin6_scope_id = if_nametoindex(interface);
456
to.in6.sin6_port = htons(port); /* Spurious warning */
458
to.in6.sin6_scope_id = (uint32_t)if_index;
368
fprintf(stderr, "Connection to: %s\n", ip);
461
fprintf(stderr, "Connection to: %s, port %d\n", ip, port);
462
char addrstr[INET6_ADDRSTRLEN] = "";
463
if(inet_ntop(to.in6.sin6_family, &(to.in6.sin6_addr), addrstr,
464
sizeof(addrstr)) == NULL){
467
if(strcmp(addrstr, ip) != 0){
468
fprintf(stderr, "Canonical address form: %s\n", addrstr);
371
ret = connect(tcp_sd, (struct sockaddr *) &to, sizeof(to));
473
ret = connect(tcp_sd, &to.in, sizeof(to));
373
475
perror("connect");
377
ret = initgnutls (&es);
384
gnutls_transport_set_ptr (es.session, (gnutls_transport_ptr_t) tcp_sd);
479
const char *out = mandos_protocol_version;
482
size_t out_size = strlen(out);
483
ret = TEMP_FAILURE_RETRY(write(tcp_sd, out + written,
484
out_size - written));
490
written += (size_t)ret;
491
if(written < out_size){
494
if (out == mandos_protocol_version){
387
fprintf(stderr, "Establishing tls session with %s\n", ip);
504
fprintf(stderr, "Establishing TLS session with %s\n", ip);
391
ret = gnutls_handshake (es.session);
507
gnutls_transport_set_ptr (session, (gnutls_transport_ptr_t) tcp_sd);
509
ret = gnutls_handshake (session);
393
511
if (ret != GNUTLS_E_SUCCESS){
394
fprintf(stderr, "\n*** Handshake failed ***\n");
513
fprintf(stderr, "*** GnuTLS Handshake failed ***\n");
400
//Retrieve gpg packet that contains the wanted password
520
/* Read OpenPGP packet that contains the wanted password */
403
fprintf(stderr, "Retrieving pgp encrypted password from %s\n", ip);
523
fprintf(stderr, "Retrieving pgp encrypted password from %s\n",
407
if (buffer_length + BUFFER_SIZE > buffer_capacity){
408
buffer = realloc(buffer, buffer_capacity + BUFFER_SIZE);
413
buffer_capacity += BUFFER_SIZE;
528
buffer_capacity = adjustbuffer(&buffer, buffer_length,
530
if (buffer_capacity == 0){
531
perror("adjustbuffer");
416
ret = gnutls_record_recv
417
(es.session, buffer+buffer_length, BUFFER_SIZE);
536
ret = gnutls_record_recv(session, buffer+buffer_length,
424
544
case GNUTLS_E_AGAIN:
426
546
case GNUTLS_E_REHANDSHAKE:
427
ret = gnutls_handshake (es.session);
547
ret = gnutls_handshake (session);
429
fprintf(stderr, "\n*** Handshake failed ***\n");
549
fprintf(stderr, "*** GnuTLS Re-handshake failed ***\n");
430
550
gnutls_perror (ret);
436
fprintf(stderr, "Unknown error while reading data from encrypted session with mandos server\n");
556
fprintf(stderr, "Unknown error while reading data from"
557
" encrypted session with Mandos server\n");
438
gnutls_bye (es.session, GNUTLS_SHUT_RDWR);
559
gnutls_bye (session, GNUTLS_SHUT_RDWR);
442
buffer_length += ret;
563
buffer_length += (size_t) ret;
568
fprintf(stderr, "Closing TLS session\n");
571
gnutls_bye (session, GNUTLS_SHUT_RDWR);
446
573
if (buffer_length > 0){
447
if ((decrypted_buffer_size = gpg_packet_decrypt(buffer, buffer_length, &decrypted_buffer, CERT_ROOT)) >= 0){
448
fwrite (decrypted_buffer, 1, decrypted_buffer_size, stdout);
574
decrypted_buffer_size = pgp_packet_decrypt(buffer,
578
if (decrypted_buffer_size >= 0){
580
while(written < (size_t) decrypted_buffer_size){
581
ret = (int)fwrite (decrypted_buffer + written, 1,
582
(size_t)decrypted_buffer_size - written,
584
if(ret == 0 and ferror(stdout)){
586
fprintf(stderr, "Error writing encrypted data: %s\n",
592
written += (size_t)ret;
449
594
free(decrypted_buffer);
458
fprintf(stderr, "Closing tls session\n");
600
/* Shutdown procedure */
462
gnutls_bye (es.session, GNUTLS_SHUT_RDWR);
465
gnutls_deinit (es.session);
466
gnutls_certificate_free_credentials (es.cred);
605
gnutls_deinit (session);
606
gnutls_certificate_free_credentials (mc->cred);
467
607
gnutls_global_deinit ();
471
static AvahiSimplePoll *simple_poll = NULL;
472
static AvahiServer *server = NULL;
474
static void resolve_callback(
475
AvahiSServiceResolver *r,
476
AVAHI_GCC_UNUSED AvahiIfIndex interface,
477
AVAHI_GCC_UNUSED AvahiProtocol protocol,
478
AvahiResolverEvent event,
482
const char *host_name,
483
const AvahiAddress *address,
485
AvahiStringList *txt,
486
AvahiLookupResultFlags flags,
487
AVAHI_GCC_UNUSED void* userdata) {
491
/* Called whenever a service has been resolved successfully or timed out */
494
case AVAHI_RESOLVER_FAILURE:
495
fprintf(stderr, "(Resolver) Failed to resolve service '%s' of type '%s' in domain '%s': %s\n", name, type, domain, avahi_strerror(avahi_server_errno(server)));
498
case AVAHI_RESOLVER_FOUND: {
499
char ip[AVAHI_ADDRESS_STR_MAX];
500
avahi_address_snprint(ip, sizeof(ip), address);
502
fprintf(stderr, "Mandos server found at %s on port %d\n", ip, port);
504
int ret = start_mandos_communcation(ip, port);
512
avahi_s_service_resolver_free(r);
515
static void browse_callback(
516
AvahiSServiceBrowser *b,
517
AvahiIfIndex interface,
518
AvahiProtocol protocol,
519
AvahiBrowserEvent event,
523
AVAHI_GCC_UNUSED AvahiLookupResultFlags flags,
526
AvahiServer *s = userdata;
529
/* Called whenever a new services becomes available on the LAN or is removed from the LAN */
533
case AVAHI_BROWSER_FAILURE:
535
fprintf(stderr, "(Browser) %s\n", avahi_strerror(avahi_server_errno(server)));
536
avahi_simple_poll_quit(simple_poll);
539
case AVAHI_BROWSER_NEW:
540
/* We ignore the returned resolver object. In the callback
541
function we free it. If the server is terminated before
542
the callback function is called the server will free
543
the resolver for us. */
545
if (!(avahi_s_service_resolver_new(s, interface, protocol, name, type, domain, AVAHI_PROTO_INET6, 0, resolve_callback, s)))
546
fprintf(stderr, "Failed to resolve service '%s': %s\n", name, avahi_strerror(avahi_server_errno(s)));
550
case AVAHI_BROWSER_REMOVE:
553
case AVAHI_BROWSER_ALL_FOR_NOW:
554
case AVAHI_BROWSER_CACHE_EXHAUSTED:
559
int main(AVAHI_GCC_UNUSED int argc, AVAHI_GCC_UNUSED char*argv[]) {
560
AvahiServerConfig config;
611
static void resolve_callback(AvahiSServiceResolver *r,
612
AvahiIfIndex interface,
613
AVAHI_GCC_UNUSED AvahiProtocol protocol,
614
AvahiResolverEvent event,
618
const char *host_name,
619
const AvahiAddress *address,
621
AVAHI_GCC_UNUSED AvahiStringList *txt,
622
AVAHI_GCC_UNUSED AvahiLookupResultFlags
625
mandos_context *mc = userdata;
626
assert(r); /* Spurious warning */
628
/* Called whenever a service has been resolved successfully or
633
case AVAHI_RESOLVER_FAILURE:
634
fprintf(stderr, "(Avahi Resolver) Failed to resolve service '%s'"
635
" of type '%s' in domain '%s': %s\n", name, type, domain,
636
avahi_strerror(avahi_server_errno(mc->server)));
639
case AVAHI_RESOLVER_FOUND:
641
char ip[AVAHI_ADDRESS_STR_MAX];
642
avahi_address_snprint(ip, sizeof(ip), address);
644
fprintf(stderr, "Mandos server \"%s\" found on %s (%s, %d) on"
645
" port %d\n", name, host_name, ip, interface, port);
647
int ret = start_mandos_communication(ip, port, interface, mc);
653
avahi_s_service_resolver_free(r);
656
static void browse_callback( AvahiSServiceBrowser *b,
657
AvahiIfIndex interface,
658
AvahiProtocol protocol,
659
AvahiBrowserEvent event,
663
AVAHI_GCC_UNUSED AvahiLookupResultFlags
666
mandos_context *mc = userdata;
667
assert(b); /* Spurious warning */
669
/* Called whenever a new services becomes available on the LAN or
670
is removed from the LAN */
674
case AVAHI_BROWSER_FAILURE:
676
fprintf(stderr, "(Avahi browser) %s\n",
677
avahi_strerror(avahi_server_errno(mc->server)));
678
avahi_simple_poll_quit(mc->simple_poll);
681
case AVAHI_BROWSER_NEW:
682
/* We ignore the returned Avahi resolver object. In the callback
683
function we free it. If the Avahi server is terminated before
684
the callback function is called the Avahi server will free the
687
if (!(avahi_s_service_resolver_new(mc->server, interface,
688
protocol, name, type, domain,
689
AVAHI_PROTO_INET6, 0,
690
resolve_callback, mc)))
691
fprintf(stderr, "Avahi: Failed to resolve service '%s': %s\n",
692
name, avahi_strerror(avahi_server_errno(mc->server)));
695
case AVAHI_BROWSER_REMOVE:
698
case AVAHI_BROWSER_ALL_FOR_NOW:
699
case AVAHI_BROWSER_CACHE_EXHAUSTED:
701
fprintf(stderr, "No Mandos server found, still searching...\n");
707
/* Combines file name and path and returns the malloced new
708
string. some sane checks could/should be added */
709
static const char *combinepath(const char *first, const char *second){
710
size_t f_len = strlen(first);
711
size_t s_len = strlen(second);
712
char *tmp = malloc(f_len + s_len + 2);
717
memcpy(tmp, first, f_len); /* Spurious warning */
721
memcpy(tmp + f_len + 1, second, s_len); /* Spurious warning */
723
tmp[f_len + 1 + s_len] = '\0';
728
int main(int argc, char *argv[]){
561
729
AvahiSServiceBrowser *sb = NULL;
564
int returncode = EXIT_SUCCESS;
567
static struct option long_options[] = {
568
{"debug", no_argument, (int *)&debug, 1},
569
{"interface", required_argument, 0, 'i'},
572
int option_index = 0;
573
ret = getopt_long (argc, argv, "i:", long_options, &option_index);
732
int exitcode = EXIT_SUCCESS;
733
const char *interface = "eth0";
734
struct ifreq network;
738
char *connect_to = NULL;
739
AvahiIfIndex if_index = AVAHI_IF_UNSPEC;
740
const char *pubkeyfile = "pubkey.txt";
741
const char *seckeyfile = "seckey.txt";
742
mandos_context mc = { .simple_poll = NULL, .server = NULL,
743
.dh_bits = 1024, .priority = "SECURE256"};
746
struct argp_option options[] = {
747
{ .name = "debug", .key = 128,
748
.doc = "Debug mode", .group = 3 },
749
{ .name = "connect", .key = 'c',
751
.doc = "Connect directly to a sepcified mandos server",
753
{ .name = "interface", .key = 'i',
755
.doc = "Interface that Avahi will conntect through",
757
{ .name = "keydir", .key = 'd',
759
.doc = "Directory where the openpgp keyring is",
761
{ .name = "seckey", .key = 's',
763
.doc = "Secret openpgp key for gnutls authentication",
765
{ .name = "pubkey", .key = 'p',
767
.doc = "Public openpgp key for gnutls authentication",
769
{ .name = "dh-bits", .key = 129,
771
.doc = "dh-bits to use in gnutls communication",
773
{ .name = "priority", .key = 130,
775
.doc = "GNUTLS priority", .group = 1 },
780
error_t parse_opt (int key, char *arg,
781
struct argp_state *state) {
782
/* Get the INPUT argument from `argp_parse', which we know is
783
a pointer to our plugin list pointer. */
805
mc.dh_bits = (unsigned int) strtol(arg, NULL, 10);
820
return ARGP_ERR_UNKNOWN;
825
struct argp argp = { .options = options, .parser = parse_opt,
827
.doc = "Mandos client -- Get and decrypt"
828
" passwords from mandos server" };
829
argp_parse (&argp, argc, argv, 0, 0, NULL);
832
pubkeyfile = combinepath(keydir, pubkeyfile);
833
if (pubkeyfile == NULL){
834
perror("combinepath");
835
exitcode = EXIT_FAILURE;
839
seckeyfile = combinepath(keydir, seckeyfile);
840
if (seckeyfile == NULL){
841
perror("combinepath");
845
ret = init_gnutls_global(&mc, pubkeyfile, seckeyfile);
847
fprintf(stderr, "init_gnutls_global\n");
864
if_index = (AvahiIfIndex) if_nametoindex(interface);
866
fprintf(stderr, "No such interface: \"%s\"\n", interface);
870
if(connect_to != NULL){
871
/* Connect directly, do not use Zeroconf */
872
/* (Mainly meant for debugging) */
873
char *address = strrchr(connect_to, ':');
875
fprintf(stderr, "No colon in address\n");
876
exitcode = EXIT_FAILURE;
880
uint16_t port = (uint16_t) strtol(address+1, NULL, 10);
882
perror("Bad port number");
883
exitcode = EXIT_FAILURE;
887
address = connect_to;
888
ret = start_mandos_communication(address, port, if_index, &mc);
890
exitcode = EXIT_FAILURE;
892
exitcode = EXIT_SUCCESS;
897
/* If the interface is down, bring it up */
899
sd = socket(PF_INET6, SOCK_DGRAM, IPPROTO_IP);
902
exitcode = EXIT_FAILURE;
905
strcpy(network.ifr_name, interface); /* Spurious warning */
906
ret = ioctl(sd, SIOCGIFFLAGS, &network);
908
perror("ioctl SIOCGIFFLAGS");
909
exitcode = EXIT_FAILURE;
912
if((network.ifr_flags & IFF_UP) == 0){
913
network.ifr_flags |= IFF_UP;
914
ret = ioctl(sd, SIOCSIFFLAGS, &network);
916
perror("ioctl SIOCSIFFLAGS");
917
exitcode = EXIT_FAILURE;
591
925
avahi_set_log_function(empty_log);
594
/* Initialize the psuedo-RNG */
597
/* Allocate main loop object */
598
if (!(simple_poll = avahi_simple_poll_new())) {
599
fprintf(stderr, "Failed to create simple poll object.\n");
604
/* Do not publish any local records */
605
avahi_server_config_init(&config);
606
config.publish_hinfo = 0;
607
config.publish_addresses = 0;
608
config.publish_workstation = 0;
609
config.publish_domain = 0;
611
/* Allocate a new server */
612
server = avahi_server_new(avahi_simple_poll_get(simple_poll), &config, NULL, NULL, &error);
614
/* Free the configuration data */
615
avahi_server_config_free(&config);
617
/* Check if creating the server object succeeded */
619
fprintf(stderr, "Failed to create server: %s\n", avahi_strerror(error));
620
returncode = EXIT_FAILURE;
624
/* Create the service browser */
625
if (!(sb = avahi_s_service_browser_new(server, if_nametoindex("eth0"), AVAHI_PROTO_INET6, "_mandos._tcp", NULL, 0, browse_callback, server))) {
626
fprintf(stderr, "Failed to create service browser: %s\n", avahi_strerror(avahi_server_errno(server)));
627
returncode = EXIT_FAILURE;
928
/* Initialize the pseudo-RNG for Avahi */
929
srand((unsigned int) time(NULL));
931
/* Allocate main Avahi loop object */
932
mc.simple_poll = avahi_simple_poll_new();
933
if (mc.simple_poll == NULL) {
934
fprintf(stderr, "Avahi: Failed to create simple poll"
936
exitcode = EXIT_FAILURE;
941
AvahiServerConfig config;
942
/* Do not publish any local Zeroconf records */
943
avahi_server_config_init(&config);
944
config.publish_hinfo = 0;
945
config.publish_addresses = 0;
946
config.publish_workstation = 0;
947
config.publish_domain = 0;
949
/* Allocate a new server */
950
mc.server = avahi_server_new(avahi_simple_poll_get
951
(mc.simple_poll), &config, NULL,
954
/* Free the Avahi configuration data */
955
avahi_server_config_free(&config);
958
/* Check if creating the Avahi server object succeeded */
959
if (mc.server == NULL) {
960
fprintf(stderr, "Failed to create Avahi server: %s\n",
961
avahi_strerror(error));
962
exitcode = EXIT_FAILURE;
966
/* Create the Avahi service browser */
967
sb = avahi_s_service_browser_new(mc.server, if_index,
969
"_mandos._tcp", NULL, 0,
970
browse_callback, &mc);
972
fprintf(stderr, "Failed to create service browser: %s\n",
973
avahi_strerror(avahi_server_errno(mc.server)));
974
exitcode = EXIT_FAILURE;
631
978
/* Run the main loop */
634
fprintf(stderr, "Starting avahi loop search\n");
981
fprintf(stderr, "Starting Avahi loop search\n");
637
avahi_simple_poll_loop(simple_poll);
984
avahi_simple_poll_loop(mc.simple_poll);
642
989
fprintf(stderr, "%s exiting\n", argv[0]);
645
992
/* Cleanup things */
647
994
avahi_s_service_browser_free(sb);
650
avahi_server_free(server);
653
avahi_simple_poll_free(simple_poll);
996
if (mc.server != NULL)
997
avahi_server_free(mc.server);
999
if (mc.simple_poll != NULL)
1000
avahi_simple_poll_free(mc.simple_poll);