32#ifndef _CRT_SECURE_NO_WARNINGS
33#define _CRT_SECURE_NO_WARNINGS 1
45#define sleep(x) Sleep(x * 1000)
55#ifndef OCPN_ENABLE_MDNS_DEBUG
59static char addrbuffer[64];
60static char namebuffer[256];
61static char sendbuffer[1024];
63static struct sockaddr_in service_address_ipv4;
64static struct sockaddr_in6 service_address_ipv6;
66volatile sig_atomic_t running_server = 1;
70 mdns_string_t service;
71 mdns_string_t hostname;
72 mdns_string_t service_instance;
73 mdns_string_t hostname_qualified;
74 struct sockaddr_in address_ipv4;
75 struct sockaddr_in6 address_ipv6;
77 mdns_record_t record_ptr;
78 mdns_record_t record_srv;
79 mdns_record_t record_a;
80 mdns_record_t record_aaaa;
81 mdns_record_t txt_record[2];
85int ocpn_service_callback(
int sock,
const struct sockaddr* from,
size_t addrlen,
86 mdns_entry_type_t entry, uint16_t query_id,
87 uint16_t rtype, uint16_t rclass, uint32_t ttl,
88 const void* data,
size_t size,
size_t name_offset,
89 size_t name_length,
size_t record_offset,
90 size_t record_length,
void* user_data) {
92 if (entry != MDNS_ENTRYTYPE_QUESTION)
return 0;
94 const char dns_sd[] =
"_services._dns-sd._udp.local.";
97 mdns_string_t fromaddrstr =
98 ip_address_to_string(addrbuffer,
sizeof(addrbuffer), from, addrlen);
100 size_t offset = name_offset;
102 mdns_string_extract(data, size, &offset, namebuffer,
sizeof(namebuffer));
104 const char* record_name = 0;
105 if (rtype == MDNS_RECORDTYPE_PTR)
107 else if (rtype == MDNS_RECORDTYPE_SRV)
109 else if (rtype == MDNS_RECORDTYPE_A)
111 else if (rtype == MDNS_RECORDTYPE_AAAA)
112 record_name =
"AAAA";
113 else if (rtype == MDNS_RECORDTYPE_TXT)
115 else if (rtype == MDNS_RECORDTYPE_ANY)
119 printf(
"Query %s %.*s\n", record_name, MDNS_STRING_FORMAT(name));
121 if ((name.length == (
sizeof(dns_sd) - 1)) &&
122 (strncmp(name.str, dns_sd,
sizeof(dns_sd) - 1) == 0)) {
123 if ((rtype == MDNS_RECORDTYPE_PTR) || (rtype == MDNS_RECORDTYPE_ANY)) {
130 mdns_record_t answer;
132 answer.type = MDNS_RECORDTYPE_PTR;
133 answer.data.ptr.name = service->service;
136 uint16_t unicast = (rclass & MDNS_UNICAST_RESPONSE);
137 printf(
" --> answer %.*s (%s)\n",
138 MDNS_STRING_FORMAT(answer.data.ptr.name),
139 (unicast ?
"unicast" :
"multicast"));
142 mdns_query_answer_unicast(sock, from, addrlen, sendbuffer,
143 sizeof(sendbuffer), query_id,
144 (mdns_record_type_t)rtype, name.str,
145 name.length, answer, 0, 0, 0, 0);
147 mdns_query_answer_multicast(sock, sendbuffer,
sizeof(sendbuffer),
151 }
else if ((name.length == service->service.length) &&
152 (strncmp(name.str, service->service.str, name.length) == 0)) {
153 if ((rtype == MDNS_RECORDTYPE_PTR) || (rtype == MDNS_RECORDTYPE_ANY)) {
165 mdns_record_t answer = service->record_ptr;
167 mdns_record_t additional[5]{{}};
168 size_t additional_count = 0;
172 additional[additional_count++] = service->record_srv;
175 if (service->address_ipv4.sin_family == AF_INET)
176 additional[additional_count++] = service->record_a;
177 if (service->address_ipv6.sin6_family == AF_INET6)
178 additional[additional_count++] = service->record_aaaa;
187 uint16_t unicast = (rclass & MDNS_UNICAST_RESPONSE);
188 printf(
" --> answer %.*s (%s)\n",
189 MDNS_STRING_FORMAT(service->record_ptr.data.ptr.name),
190 (unicast ?
"unicast" :
"multicast"));
193 mdns_query_answer_unicast(
194 sock, from, addrlen, sendbuffer,
sizeof(sendbuffer), query_id,
195 (mdns_record_type)rtype, name.str, name.length, answer, 0, 0,
196 additional, additional_count);
198 mdns_query_answer_multicast(sock, sendbuffer,
sizeof(sendbuffer),
199 answer, 0, 0, additional, additional_count);
202 }
else if ((name.length == service->service_instance.length) &&
203 (strncmp(name.str, service->service_instance.str, name.length) ==
205 if ((rtype == MDNS_RECORDTYPE_SRV) || (rtype == MDNS_RECORDTYPE_ANY)) {
214 mdns_record_t answer = service->record_srv;
216 mdns_record_t additional[5]{{}};
217 size_t additional_count = 0;
220 if (service->address_ipv4.sin_family == AF_INET)
221 additional[additional_count++] = service->record_a;
222 if (service->address_ipv6.sin6_family == AF_INET6)
223 additional[additional_count++] = service->record_aaaa;
228 additional[additional_count++] = service->txt_record[0];
229 additional[additional_count++] = service->txt_record[1];
232 uint16_t unicast = (rclass & MDNS_UNICAST_RESPONSE);
233 printf(
" --> answer %.*s port %d (%s)\n",
234 MDNS_STRING_FORMAT(service->record_srv.data.srv.name),
235 service->port, (unicast ?
"unicast" :
"multicast"));
238 mdns_query_answer_unicast(
239 sock, from, addrlen, sendbuffer,
sizeof(sendbuffer), query_id,
240 (mdns_record_type)rtype, name.str, name.length, answer, 0, 0,
241 additional, additional_count);
243 mdns_query_answer_multicast(sock, sendbuffer,
sizeof(sendbuffer),
244 answer, 0, 0, additional, additional_count);
247 }
else if ((name.length == service->hostname_qualified.length) &&
248 (strncmp(name.str, service->hostname_qualified.str, name.length) ==
250 if (((rtype == MDNS_RECORDTYPE_A) || (rtype == MDNS_RECORDTYPE_ANY)) &&
251 (service->address_ipv4.sin_family == AF_INET)) {
258 mdns_record_t answer = service->record_a;
260 mdns_record_t additional[5]{{}};
261 size_t additional_count = 0;
264 if (service->address_ipv6.sin6_family == AF_INET6)
265 additional[additional_count++] = service->record_aaaa;
270 additional[additional_count++] = service->txt_record[0];
271 additional[additional_count++] = service->txt_record[1];
274 uint16_t unicast = (rclass & MDNS_UNICAST_RESPONSE);
275 mdns_string_t addrstr =
276 ip_address_to_string(addrbuffer,
sizeof(addrbuffer),
277 (
struct sockaddr*)&service->record_a.data.a.addr,
278 sizeof(service->record_a.data.a.addr));
279 printf(
" --> answer %.*s IPv4 %.*s (%s)\n",
280 MDNS_STRING_FORMAT(service->record_a.name),
281 MDNS_STRING_FORMAT(addrstr), (unicast ?
"unicast" :
"multicast"));
284 mdns_query_answer_unicast(
285 sock, from, addrlen, sendbuffer,
sizeof(sendbuffer), query_id,
286 (mdns_record_type)rtype, name.str, name.length, answer, 0, 0,
287 additional, additional_count);
289 mdns_query_answer_multicast(sock, sendbuffer,
sizeof(sendbuffer),
290 answer, 0, 0, additional, additional_count);
292 }
else if (((rtype == MDNS_RECORDTYPE_AAAA) ||
293 (rtype == MDNS_RECORDTYPE_ANY)) &&
294 (service->address_ipv6.sin6_family == AF_INET6)) {
301 mdns_record_t answer = service->record_aaaa;
303 mdns_record_t additional[5]{{}};
304 size_t additional_count = 0;
307 if (service->address_ipv4.sin_family == AF_INET)
308 additional[additional_count++] = service->record_a;
313 additional[additional_count++] = service->txt_record[0];
314 additional[additional_count++] = service->txt_record[1];
317 uint16_t unicast = (rclass & MDNS_UNICAST_RESPONSE);
318 mdns_string_t addrstr = ip_address_to_string(
319 addrbuffer,
sizeof(addrbuffer),
320 (
struct sockaddr*)&service->record_aaaa.data.aaaa.addr,
321 sizeof(service->record_aaaa.data.aaaa.addr));
322 printf(
" --> answer %.*s IPv6 %.*s (%s)\n",
323 MDNS_STRING_FORMAT(service->record_aaaa.name),
324 MDNS_STRING_FORMAT(addrstr), (unicast ?
"unicast" :
"multicast"));
327 mdns_query_answer_unicast(
328 sock, from, addrlen, sendbuffer,
sizeof(sendbuffer), query_id,
329 (mdns_record_type)rtype, name.str, name.length, answer, 0, 0,
330 additional, additional_count);
332 mdns_query_answer_multicast(sock, sendbuffer,
sizeof(sendbuffer),
333 answer, 0, 0, additional, additional_count);
341void service_mdns(
const char* hostname,
const char* service_name,
345 open_service_sockets(sockets,
sizeof(sockets) /
sizeof(sockets[0]));
346 if (num_sockets <= 0) {
347 printf(
"Failed to open any client sockets\n");
350 printf(
"Opened %d socket%s for mDNS service\n", num_sockets,
351 num_sockets ?
"s" :
"");
353 size_t service_name_length = strlen(service_name);
354 if (!service_name_length) {
355 printf(
"Invalid service name\n");
359 char* service_name_buffer = (
char*)malloc(service_name_length + 2);
360 memcpy(service_name_buffer, service_name, service_name_length);
361 if (service_name_buffer[service_name_length - 1] !=
'.')
362 service_name_buffer[service_name_length++] =
'.';
363 service_name_buffer[service_name_length] = 0;
364 service_name = service_name_buffer;
366 printf(
"Service mDNS: %s:%d\n", service_name, service_port);
367 printf(
"Hostname: %s\n", hostname);
369 size_t capacity = 2048;
370 void* buffer = malloc(capacity);
372 mdns_string_t service_string;
373 service_string.str = service_name;
374 service_string.length = strlen(service_name);
375 mdns_string_t hostname_string;
376 hostname_string.str = hostname;
377 hostname_string.length = strlen(hostname);
380 char service_instance_buffer[256] = {0};
381 snprintf(service_instance_buffer,
sizeof(service_instance_buffer) - 1,
382 "%.*s.%.*s", MDNS_STRING_FORMAT(hostname_string),
383 MDNS_STRING_FORMAT(service_string));
385 mdns_string_t service_instance_string;
386 service_instance_string.str = service_instance_buffer;
387 service_instance_string.length = strlen(service_instance_buffer);
390 char qualified_hostname_buffer[256] = {0};
391 snprintf(qualified_hostname_buffer,
sizeof(qualified_hostname_buffer) - 1,
392 "%.*s.local.", MDNS_STRING_FORMAT(hostname_string));
393 mdns_string_t hostname_qualified_string;
394 hostname_qualified_string.str = qualified_hostname_buffer;
395 hostname_qualified_string.length = strlen(qualified_hostname_buffer);
398 service.service = service_string;
399 service.hostname = hostname_string;
400 service.service_instance = service_instance_string;
401 service.hostname_qualified = hostname_qualified_string;
402 service.address_ipv4 = service_address_ipv4;
403 service.address_ipv6 = service_address_ipv6;
404 service.port = service_port;
410 service.record_ptr.name = service.service;
411 service.record_ptr.type = MDNS_RECORDTYPE_PTR;
412 service.record_ptr.rclass = 0;
413 service.record_ptr.ttl = 0;
414 service.record_ptr.data.ptr.name = service.service_instance;
418 service.record_srv.name = service.service_instance;
419 service.record_srv.type = MDNS_RECORDTYPE_SRV;
420 service.record_srv.data.srv.name = service.hostname_qualified;
421 service.record_srv.data.srv.port = service.port;
422 service.record_srv.data.srv.priority = 0;
423 service.record_srv.data.srv.weight = 0;
424 service.record_srv.rclass = 0;
425 service.record_srv.ttl = 0;
429 service.record_a.name = service.hostname_qualified;
430 service.record_a.type = MDNS_RECORDTYPE_A;
431 service.record_a.data.a.addr = service.address_ipv4;
432 service.record_a.rclass = 0;
433 service.record_a.ttl = 0;
435 service.record_aaaa.name = service.hostname_qualified;
436 service.record_aaaa.type = MDNS_RECORDTYPE_AAAA;
437 service.record_aaaa.data.aaaa.addr = service.address_ipv6;
438 service.record_aaaa.rclass = 0;
439 service.record_aaaa.ttl = 0;
444 service.txt_record[0].name = service.service_instance;
445 service.txt_record[0].type = MDNS_RECORDTYPE_TXT;
446 service.txt_record[0].data.txt.key = {MDNS_STRING_CONST(
"test")};
447 service.txt_record[0].data.txt.value = {MDNS_STRING_CONST(
"1")};
448 service.txt_record[0].rclass = 0;
449 service.txt_record[0].ttl = 0;
451 service.txt_record[1].name = service.service_instance;
452 service.txt_record[1].type = MDNS_RECORDTYPE_TXT;
453 service.txt_record[1].data.txt.key = {MDNS_STRING_CONST(
"other")};
454 service.txt_record[1].data.txt.value = {MDNS_STRING_CONST(
"value")};
455 service.txt_record[1].rclass = 0;
456 service.txt_record[1].ttl = 0;
461 printf(
"Sending announce\n");
462 mdns_record_t additional[5]{{}};
463 size_t additional_count = 0;
464 additional[additional_count++] = service.record_srv;
465 if (service.address_ipv4.sin_family == AF_INET)
466 additional[additional_count++] = service.record_a;
467 if (service.address_ipv6.sin6_family == AF_INET6)
468 additional[additional_count++] = service.record_aaaa;
472 for (
int isock = 0; isock < num_sockets; ++isock)
473 mdns_announce_multicast(sockets[isock], buffer, capacity,
474 service.record_ptr, 0, 0, additional,
479 while (running_server) {
483 for (
int isock = 0; isock < num_sockets; ++isock) {
484 if (sockets[isock] >= nfds) nfds = sockets[isock] + 1;
485 FD_SET(sockets[isock], &readfs);
488 struct timeval timeout;
490 timeout.tv_usec = 100000;
492 if (select(nfds, &readfs, 0, 0, &timeout) >= 0) {
493 for (
int isock = 0; isock < num_sockets; ++isock) {
494 if (FD_ISSET(sockets[isock], &readfs)) {
495 mdns_socket_listen(sockets[isock], buffer, capacity,
496 ocpn_service_callback, &service);
498 FD_SET(sockets[isock], &readfs);
507 printf(
"Sending goodbye\n");
508 mdns_record_t additional[5]{{}};
509 size_t additional_count = 0;
510 additional[additional_count++] = service.record_srv;
511 if (service.address_ipv4.sin_family == AF_INET)
512 additional[additional_count++] = service.record_a;
513 if (service.address_ipv6.sin6_family == AF_INET6)
514 additional[additional_count++] = service.record_aaaa;
515 additional[additional_count++] = service.txt_record[0];
516 additional[additional_count++] = service.txt_record[1];
518 for (
int isock = 0; isock < num_sockets; ++isock)
519 mdns_goodbye_multicast(sockets[isock], buffer, capacity,
520 service.record_ptr, 0, 0, additional,
525 free(service_name_buffer);
527 for (
int isock = 0; isock < num_sockets; ++isock)
528 mdns_socket_close(sockets[isock]);
529 printf(
"Closed socket%s\n", num_sockets ?
"s" :
"");
537int StartMDNSService(std::string hostname, std::string service_name,
540 service = service_name;
542 std::thread mdns_service_thread(service_mdns, host.c_str(), service.c_str(),
544 mdns_service_thread.detach();
549bool StopMDNSService() {
return true; }