2023-07-05 20:09:28 +01:00
/**
* SPDX - FileCopyrightText : 2023 Albert Vaca Cintora < albertvaka @ gmail . com >
*
* SPDX - License - Identifier : GPL - 2.0 - only OR GPL - 3.0 - only OR LicenseRef - KDE - Accepted - GPL
*/
# include "mdns_wrapper.h"
# include "core_debug.h"
# include "mdns.h"
# include <errno.h>
2023-07-08 12:22:45 +01:00
# include <QHostInfo>
2023-07-05 20:09:28 +01:00
# include <QNetworkInterface>
# include <QSocketNotifier>
2023-07-10 11:11:38 +01:00
namespace MdnsWrapper
{
2023-07-05 20:09:28 +01:00
const char * recordTypeToStr ( int rtype )
{
switch ( rtype ) {
2023-07-16 13:43:37 +01:00
case MDNS_RECORDTYPE_PTR :
return " PTR " ;
case MDNS_RECORDTYPE_SRV :
return " SRV " ;
case MDNS_RECORDTYPE_TXT :
return " TXT " ;
case MDNS_RECORDTYPE_A :
return " A " ;
case MDNS_RECORDTYPE_AAAA :
return " AAAA " ;
case MDNS_RECORDTYPE_ANY :
return " ANY " ;
default :
return " UNKNOWN " ;
2023-07-05 20:09:28 +01:00
}
}
const char * entryTypeToStr ( int entry )
{
switch ( entry ) {
2023-07-16 13:43:37 +01:00
case MDNS_ENTRYTYPE_QUESTION :
return " QUESTION " ;
case MDNS_ENTRYTYPE_ANSWER :
return " ANSWER " ;
case MDNS_ENTRYTYPE_AUTHORITY :
return " AUTHORITY " ;
case MDNS_ENTRYTYPE_ADDITIONAL :
return " ADDITIONAL " ;
default :
return " UNKNOWN " ;
2023-07-05 20:09:28 +01:00
}
}
2023-07-10 13:50:39 +01:00
static sockaddr_in qHostAddresstoSockaddr ( QHostAddress hostAddress )
{
Q_ASSERT ( hostAddress . protocol ( ) = = QAbstractSocket : : IPv4Protocol ) ;
sockaddr_in socketAddress ;
memset ( & socketAddress , 0 , sizeof ( socketAddress ) ) ;
socketAddress . sin_family = AF_INET ;
socketAddress . sin_addr . s_addr = htonl ( hostAddress . toIPv4Address ( ) ) ;
return socketAddress ;
}
static sockaddr_in6 qHostAddresstoSockaddr6 ( QHostAddress hostAddress )
{
Q_ASSERT ( hostAddress . protocol ( ) = = QAbstractSocket : : IPv6Protocol ) ;
sockaddr_in6 socketAddress ;
memset ( & socketAddress , 0 , sizeof ( socketAddress ) ) ;
socketAddress . sin6_family = AF_INET6 ;
Q_IPV6ADDR ipv6Address = hostAddress . toIPv6Address ( ) ;
for ( int i = 0 ; i < 16 ; + + i ) {
socketAddress . sin6_addr . s6_addr [ i ] = ipv6Address [ i ] ;
}
return socketAddress ;
}
2023-07-05 20:09:28 +01:00
// Callback that handles responses to a query
static int query_callback ( int sock , const struct sockaddr * from , size_t addrlen , mdns_entry_type_t entry_type ,
uint16_t query_id , uint16_t record_type , uint16_t rclass , uint32_t ttl , const void * data ,
size_t size , size_t name_offset , size_t name_length , size_t record_offset ,
size_t record_length , void * user_data ) {
Q_UNUSED ( sock ) ;
Q_UNUSED ( addrlen ) ;
Q_UNUSED ( query_id ) ;
Q_UNUSED ( entry_type ) ;
Q_UNUSED ( rclass ) ;
Q_UNUSED ( ttl ) ;
Q_UNUSED ( name_offset ) ;
Q_UNUSED ( name_length ) ;
//qCDebug(KDECONNECT_CORE) << "Received DNS record of type" << recordTypeToStr(record_type) << "from socket" << sock << "with type" << entryTypeToStr(entry_type);
2023-07-10 11:11:38 +01:00
Discoverer : : MdnsService * discoveredService = ( Discoverer : : MdnsService * ) user_data ;
2023-07-05 20:09:28 +01:00
switch ( record_type ) {
case MDNS_RECORDTYPE_PTR : {
2023-07-10 09:45:10 +01:00
// We don't use mdns_record_parse_ptr() because we want to extract just the service name instead of the full "<service-name>.<_service-type>._tcp.local." string
2023-07-06 23:49:28 +01:00
mdns_string_pair_t serviceNamePos = mdns_get_next_substring ( data , size , record_offset ) ;
2023-07-07 17:03:01 +01:00
discoveredService - > name = QString : : fromLatin1 ( ( char * ) data + serviceNamePos . offset , serviceNamePos . length ) ;
2023-07-06 23:49:28 +01:00
//static char serviceNameBuffer[256];
//mdns_string_t serviceName = mdns_record_parse_ptr(data, size, record_offset, record_length, serviceNameBuffer, sizeof(serviceNameBuffer));
2023-07-07 17:03:01 +01:00
//discoveredService->name = QString::fromLatin1(serviceName.str, serviceName.length);
2023-07-05 20:09:28 +01:00
if ( discoveredService - > address = = QHostAddress : : Null ) {
discoveredService - > address = QHostAddress ( from ) ; // In case we don't receive a A record, use from as address
}
} break ;
case MDNS_RECORDTYPE_SRV : {
2023-07-06 23:49:28 +01:00
static char nameBuffer [ 256 ] ;
mdns_record_srv_t record = mdns_record_parse_srv ( data , size , record_offset , record_length , nameBuffer , sizeof ( nameBuffer ) ) ;
// We can use the IP to connect so we don't need to store the "<hostname>.local." address.
//discoveredService->qualifiedHostname = QString::fromLatin1(record.name.str, record.name.length);
2023-07-05 20:09:28 +01:00
discoveredService - > port = record . port ;
} break ;
case MDNS_RECORDTYPE_A : {
sockaddr_in addr ;
mdns_record_parse_a ( data , size , record_offset , record_length , & addr ) ;
discoveredService - > address = QHostAddress ( ntohl ( addr . sin_addr . s_addr ) ) ;
} break ;
case MDNS_RECORDTYPE_AAAA :
// Ignore IPv6 for now
2023-07-06 23:49:28 +01:00
//sockaddr_in6 addr6;
//mdns_record_parse_aaaa(data, size, record_offset, record_length, &addr6);
2023-07-05 20:09:28 +01:00
break ;
case MDNS_RECORDTYPE_TXT : {
mdns_record_txt_t records [ 24 ] ;
size_t parsed = mdns_record_parse_txt ( data , size , record_offset , record_length , records , sizeof ( records ) / sizeof ( mdns_record_txt_t ) ) ;
for ( size_t itxt = 0 ; itxt < parsed ; + + itxt ) {
QString key = QString : : fromLatin1 ( records [ itxt ] . key . str , records [ itxt ] . key . length ) ;
QString value = QString : : fromLatin1 ( records [ itxt ] . value . str , records [ itxt ] . value . length ) ;
discoveredService - > txtRecords [ key ] = value ;
}
} break ;
default :
// Ignore unknown record types
break ;
}
return 0 ;
}
2023-07-10 11:11:38 +01:00
void Discoverer : : startDiscovering ( const QString & serviceType )
2023-07-05 20:09:28 +01:00
{
int num_sockets = listenForQueryResponses ( ) ;
if ( num_sockets < = 0 ) {
qWarning ( ) < < " Failed to open any client sockets " ;
2023-07-08 12:22:45 +01:00
return ;
2023-07-05 20:09:28 +01:00
}
2023-07-08 12:22:45 +01:00
sendQuery ( serviceType ) ;
2023-07-05 20:09:28 +01:00
}
2023-07-10 11:11:38 +01:00
void Discoverer : : stopDiscovering ( )
2023-07-05 20:09:28 +01:00
{
stopListeningForQueryResponses ( ) ;
}
2023-07-10 11:11:38 +01:00
void Discoverer : : stopListeningForQueryResponses ( )
2023-07-05 20:09:28 +01:00
{
qCDebug ( KDECONNECT_CORE ) < < " Closing " < < responseSocketNotifiers . size ( ) < < " sockets " ;
2023-07-29 15:28:37 +01:00
for ( QSocketNotifier * socketNotifier : qAsConst ( responseSocketNotifiers ) ) {
2023-07-05 20:09:28 +01:00
mdns_socket_close ( socketNotifier - > socket ( ) ) ;
delete socketNotifier ;
}
responseSocketNotifiers . clear ( ) ;
}
2023-07-10 11:11:38 +01:00
int Discoverer : : listenForQueryResponses ( )
2023-07-05 20:09:28 +01:00
{
// Open a socket for each interface
QVector < int > sockets ;
for ( const QNetworkInterface & iface : QNetworkInterface : : allInterfaces ( ) ) {
int flags = iface . flags ( ) ;
if ( ! ( flags & QNetworkInterface : : IsUp ) | | ! ( flags & QNetworkInterface : : CanMulticast ) | | ( flags & QNetworkInterface : : IsLoopBack ) ) {
continue ;
}
for ( const QNetworkAddressEntry & ifaceAddress : iface . addressEntries ( ) ) {
QHostAddress sourceAddress = ifaceAddress . ip ( ) ;
if ( sourceAddress . protocol ( ) = = QAbstractSocket : : IPv4Protocol & & sourceAddress ! = QHostAddress : : LocalHost ) {
qCDebug ( KDECONNECT_CORE ) < < " Opening socket for address " < < sourceAddress ;
2023-07-10 13:50:39 +01:00
struct sockaddr_in saddr = qHostAddresstoSockaddr ( sourceAddress ) ;
2023-07-05 20:09:28 +01:00
int socket = mdns_socket_open_ipv4 ( & saddr ) ;
sockets . append ( socket ) ;
2023-07-10 13:50:39 +01:00
} else if ( sourceAddress . protocol ( ) = = QAbstractSocket : : IPv6Protocol & & sourceAddress ! = QHostAddress : : LocalHostIPv6 ) {
qCDebug ( KDECONNECT_CORE ) < < " Opening socket for address6 " < < sourceAddress ;
struct sockaddr_in6 saddr = qHostAddresstoSockaddr6 ( sourceAddress ) ;
int socket = mdns_socket_open_ipv6 ( & saddr ) ;
sockets . append ( socket ) ;
2023-07-05 20:09:28 +01:00
}
}
}
// Start listening on all sockets
for ( int socket : sockets ) {
QSocketNotifier * socketNotifier = new QSocketNotifier ( socket , QSocketNotifier : : Read ) ;
QObject : : connect ( socketNotifier , & QSocketNotifier : : activated , [ this ] ( QSocketDescriptor socket ) {
MdnsService discoveredService ;
2023-07-10 09:45:10 +01:00
static char buffer [ 2048 ] ;
size_t num_records = mdns_query_recv ( socket , buffer , sizeof ( buffer ) , query_callback , ( void * ) & discoveredService , 0 ) ;
2023-07-10 13:34:26 +01:00
Q_UNUSED ( num_records ) ;
2023-07-05 20:09:28 +01:00
2023-07-07 17:03:01 +01:00
// qCDebug(KDECONNECT_CORE) << "Discovered service" << discoveredService.name << "at" << discoveredService.address << "in" << num_records <<
2023-07-05 20:09:28 +01:00
// "records via socket" << socket;
Q_EMIT serviceFound ( discoveredService ) ;
} ) ;
responseSocketNotifiers . append ( socketNotifier ) ;
}
qCDebug ( KDECONNECT_CORE ) < < " Opened " < < sockets . size ( ) < < " sockets to listen for MDNS query responses " ;
return sockets . size ( ) ;
}
2023-07-10 11:11:38 +01:00
void Discoverer : : sendQuery ( const QString & serviceType )
2023-07-05 20:09:28 +01:00
{
2023-07-07 17:03:01 +01:00
qCDebug ( KDECONNECT_CORE ) < < " Sending MDNS query for service " < < serviceType ;
2023-07-05 20:09:28 +01:00
mdns_query_t query ;
2023-07-08 12:22:45 +01:00
QByteArray serviceTypeBytes = serviceType . toLatin1 ( ) ;
2023-07-10 09:45:10 +01:00
query . name = serviceTypeBytes . constData ( ) ;
2023-07-08 12:22:45 +01:00
query . length = serviceTypeBytes . length ( ) ;
2023-07-05 20:09:28 +01:00
query . type = MDNS_RECORDTYPE_PTR ;
2023-07-10 09:45:10 +01:00
static char buffer [ 2048 ] ;
2023-07-29 15:28:37 +01:00
for ( QSocketNotifier * socketNotifier : qAsConst ( responseSocketNotifiers ) ) {
2023-07-05 20:09:28 +01:00
int socket = socketNotifier - > socket ( ) ;
qCDebug ( KDECONNECT_CORE ) < < " Sending mDNS query via socket " < < socket ;
2023-07-10 09:45:10 +01:00
int ret = mdns_multiquery_send ( socket , & query , 1 , buffer , sizeof ( buffer ) , 0 ) ;
2023-07-05 20:09:28 +01:00
if ( ret < 0 ) {
qWarning ( ) < < " Failed to send mDNS query: " < < strerror ( errno ) ;
}
}
}
2023-07-10 09:45:10 +01:00
const QByteArray dnsSdName = QByteArray ( " _services._dns-sd._udp.local. " ) ;
2023-07-05 20:09:28 +01:00
2023-07-10 09:45:10 +01:00
static mdns_string_t createMdnsString ( const QByteArray & str )
{
return mdns_string_t { str . constData ( ) , ( size_t ) str . length ( ) } ;
2023-07-05 20:09:28 +01:00
}
2023-07-10 13:34:26 +01:00
static QHostAddress findBestAddressMatch ( QVector < QHostAddress > hostAddresses , const struct sockaddr * fromAddress )
{
if ( hostAddresses . size ( ) = = 1 | | fromAddress = = nullptr ) {
return hostAddresses [ 0 ] ;
}
// FIXME
return hostAddresses [ 0 ] ;
}
2023-07-10 11:11:38 +01:00
static mdns_record_t createMdnsRecord ( const Announcer : : AnnouncedInfo & self ,
2023-07-10 09:45:10 +01:00
mdns_record_type_t record_type ,
2023-07-10 13:34:26 +01:00
const struct sockaddr * fromAddress = nullptr , // used to determine IP to set in A and AAAA records
2023-07-10 09:45:10 +01:00
QHash < QByteArray , QByteArray > : : const_iterator txtIterator = { } )
{
mdns_record_t answer ;
answer . type = record_type ;
answer . rclass = 0 ;
answer . ttl = 0 ;
switch ( record_type ) {
case MDNS_RECORDTYPE_PTR : // maps "<_service-type>._tcp.local." to "<service-name>.<_service-type>._tcp.local."
answer . name = createMdnsString ( self . serviceType ) ;
answer . data . ptr . name = createMdnsString ( self . serviceInstance ) ;
break ;
case MDNS_RECORDTYPE_SRV : // maps "<service-name>.<_service-type>._tcp.local." to "<hostname>.local." and port
answer . name = createMdnsString ( self . serviceInstance ) ;
answer . data . srv . name = createMdnsString ( self . hostname ) ;
2023-07-16 13:43:37 +01:00
answer . data . srv . port = self . port ;
answer . data . srv . priority = 0 ;
2023-07-10 09:45:10 +01:00
answer . data . srv . weight = 0 ;
break ;
case MDNS_RECORDTYPE_A : // maps "<hostname>.local." to IPv4
answer . name = createMdnsString ( self . hostname ) ;
2023-07-10 13:34:26 +01:00
answer . data . a . addr = qHostAddresstoSockaddr ( findBestAddressMatch ( self . addressesV4 , fromAddress ) ) ;
2023-07-10 09:45:10 +01:00
break ;
case MDNS_RECORDTYPE_AAAA : // maps "<hostname>.local." to IPv6
answer . name = createMdnsString ( self . hostname ) ;
2023-07-10 13:34:26 +01:00
answer . data . aaaa . addr = qHostAddresstoSockaddr6 ( findBestAddressMatch ( self . addressesV6 , fromAddress ) ) ;
2023-07-10 09:45:10 +01:00
break ;
case MDNS_RECORDTYPE_TXT :
answer . name = createMdnsString ( self . serviceInstance ) ;
answer . type = MDNS_RECORDTYPE_TXT ;
answer . data . txt . key = createMdnsString ( txtIterator . key ( ) ) ;
answer . data . txt . value = createMdnsString ( txtIterator . value ( ) ) ;
break ;
default :
2023-07-10 13:34:26 +01:00
Q_ASSERT ( false ) ;
2023-07-05 20:09:28 +01:00
}
2023-07-10 09:45:10 +01:00
return answer ;
2023-07-05 20:09:28 +01:00
}
// Callback handling questions incoming on service sockets
2023-07-08 12:22:45 +01:00
static int service_callback ( int sock , const struct sockaddr * from , size_t addrlen , mdns_entry_type_t entry_type ,
uint16_t query_id , uint16_t record_type , uint16_t rclass , uint32_t ttl , const void * data ,
2023-07-05 20:09:28 +01:00
size_t size , size_t name_offset , size_t name_length , size_t record_offset ,
size_t record_length , void * user_data ) {
2023-07-08 12:22:45 +01:00
Q_UNUSED ( ttl ) ;
Q_UNUSED ( name_length ) ;
Q_UNUSED ( record_offset ) ;
Q_UNUSED ( record_length ) ;
2023-07-10 11:11:38 +01:00
static char sendbuffer [ 2024 ] ;
2023-07-05 20:09:28 +01:00
2023-07-10 11:11:38 +01:00
const Announcer : : AnnouncedInfo & self = * ( ( Announcer : : AnnouncedInfo * ) user_data ) ;
2023-07-05 20:09:28 +01:00
2023-07-08 12:22:45 +01:00
if ( entry_type ! = MDNS_ENTRYTYPE_QUESTION ) {
2023-07-05 20:09:28 +01:00
return 0 ;
2023-07-08 12:22:45 +01:00
}
2023-07-05 20:09:28 +01:00
2023-07-08 12:22:45 +01:00
static char nameBuffer [ 256 ] ;
mdns_string_t nameMdnsString = mdns_string_extract ( data , size , & name_offset , nameBuffer , sizeof ( nameBuffer ) ) ;
2023-07-10 09:45:10 +01:00
QByteArray name = QByteArray ( nameMdnsString . str , nameMdnsString . length ) ;
2023-07-05 20:09:28 +01:00
2023-07-08 12:22:45 +01:00
if ( name = = dnsSdName ) {
qWarning ( ) < < " Someone queried all services for " < < recordTypeToStr ( record_type ) ;
if ( ( record_type = = MDNS_RECORDTYPE_PTR ) | | ( record_type = = MDNS_RECORDTYPE_ANY ) ) {
2023-07-10 09:45:10 +01:00
// The PTR query was for the DNS-SD domain, send answer with a PTR record for the service name we advertise.
2023-07-05 20:09:28 +01:00
2023-07-10 09:45:10 +01:00
mdns_record_t answer = createMdnsRecord ( self , MDNS_RECORDTYPE_PTR ) ;
2023-07-05 20:09:28 +01:00
// Send the answer, unicast or multicast depending on flag in query
uint16_t unicast = ( rclass & MDNS_UNICAST_RESPONSE ) ;
2023-07-10 09:45:10 +01:00
printf ( " --> answer %.*s (%s) \n " , MDNS_STRING_FORMAT ( answer . data . ptr . name ) , ( unicast ? " unicast " : " multicast " ) ) ;
2023-07-05 20:09:28 +01:00
if ( unicast ) {
2023-07-10 09:45:10 +01:00
mdns_query_answer_unicast ( sock , from , addrlen , sendbuffer , sizeof ( sendbuffer ) , query_id ,
( mdns_record_type_t ) record_type , nameMdnsString . str , nameMdnsString . length ,
2023-07-29 15:28:37 +01:00
answer , nullptr , 0 , nullptr , 0 ) ;
2023-07-05 20:09:28 +01:00
} else {
2023-07-29 15:28:37 +01:00
mdns_query_answer_multicast ( sock , sendbuffer , sizeof ( sendbuffer ) , answer , nullptr , 0 , nullptr , 0 ) ;
2023-07-05 20:09:28 +01:00
}
}
2023-07-10 09:45:10 +01:00
} else if ( name = = self . serviceType ) {
2023-07-08 12:22:45 +01:00
qWarning ( ) < < " Someone queried my service type for " < < recordTypeToStr ( record_type ) ;
if ( ( record_type = = MDNS_RECORDTYPE_PTR ) | | ( record_type = = MDNS_RECORDTYPE_ANY ) ) {
2023-07-10 09:45:10 +01:00
// The PTR query was for our service, answer a PTR record reverse mapping the queried service name
// to our service instance name and add additional records containing the SRV record mapping the
// service instance name to our qualified hostname and port, as well as any IPv4/IPv6 and TXT records
mdns_record_t answer = createMdnsRecord ( self , MDNS_RECORDTYPE_PTR ) ;
QVector < mdns_record_t > additional ;
additional . append ( createMdnsRecord ( self , MDNS_RECORDTYPE_SRV ) ) ;
2023-07-10 13:34:26 +01:00
if ( ! self . addressesV4 . empty ( ) ) {
additional . append ( createMdnsRecord ( self , MDNS_RECORDTYPE_A , from ) ) ;
2023-07-10 09:45:10 +01:00
}
2023-07-10 13:34:26 +01:00
if ( ! self . addressesV6 . empty ( ) ) {
additional . append ( createMdnsRecord ( self , MDNS_RECORDTYPE_AAAA , from ) ) ;
2023-07-10 09:45:10 +01:00
}
for ( auto txtIterator = self . txtRecords . cbegin ( ) ; txtIterator ! = self . txtRecords . cend ( ) ; txtIterator + + ) {
2023-07-10 13:34:26 +01:00
additional . append ( createMdnsRecord ( self , MDNS_RECORDTYPE_TXT , nullptr , txtIterator ) ) ;
2023-07-10 09:45:10 +01:00
}
2023-07-05 20:09:28 +01:00
// Send the answer, unicast or multicast depending on flag in query
uint16_t unicast = ( rclass & MDNS_UNICAST_RESPONSE ) ;
2023-07-10 09:45:10 +01:00
printf ( " --> answer %.*s (%s) \n " , MDNS_STRING_FORMAT ( answer . data . ptr . name ) , ( unicast ? " unicast " : " multicast " ) ) ;
2023-07-05 20:09:28 +01:00
if ( unicast ) {
2023-07-10 09:45:10 +01:00
mdns_query_answer_unicast ( sock , from , addrlen , sendbuffer , sizeof ( sendbuffer ) , query_id ,
( mdns_record_type_t ) record_type , nameMdnsString . str , nameMdnsString . length ,
2023-07-29 15:28:37 +01:00
answer , nullptr , 0 , additional . constData ( ) , additional . length ( ) ) ;
2023-07-05 20:09:28 +01:00
} else {
2023-07-29 15:28:37 +01:00
mdns_query_answer_multicast ( sock , sendbuffer , sizeof ( sendbuffer ) , answer , nullptr , 0 ,
2023-07-10 09:45:10 +01:00
additional . constData ( ) , additional . length ( ) ) ;
2023-07-05 20:09:28 +01:00
}
}
2023-07-10 09:45:10 +01:00
} else if ( name = = self . serviceInstance ) {
2023-07-08 12:22:45 +01:00
qWarning ( ) < < " Someone queried my service instance " < < recordTypeToStr ( record_type ) ;
if ( ( record_type = = MDNS_RECORDTYPE_SRV ) | | ( record_type = = MDNS_RECORDTYPE_ANY ) ) {
2023-07-10 09:45:10 +01:00
// The SRV query was for our service instance, answer a SRV record mapping the service
2023-07-05 20:09:28 +01:00
// instance name to our qualified hostname (typically "<hostname>.local.") and port, as
2023-07-10 09:45:10 +01:00
// well as any IPv4/IPv6 address for the hostname as A/AAAA records and TXT records
2023-07-05 20:09:28 +01:00
2023-07-10 09:45:10 +01:00
mdns_record_t answer = createMdnsRecord ( self , MDNS_RECORDTYPE_SRV ) ;
2023-07-05 20:09:28 +01:00
2023-07-10 09:45:10 +01:00
QVector < mdns_record_t > additional ;
2023-07-10 13:34:26 +01:00
if ( ! self . addressesV4 . empty ( ) ) {
additional . append ( createMdnsRecord ( self , MDNS_RECORDTYPE_A , from ) ) ;
2023-07-10 09:45:10 +01:00
}
2023-07-10 13:34:26 +01:00
if ( ! self . addressesV6 . empty ( ) ) {
additional . append ( createMdnsRecord ( self , MDNS_RECORDTYPE_AAAA , from ) ) ;
2023-07-10 09:45:10 +01:00
}
2023-07-05 20:09:28 +01:00
2023-07-10 09:45:10 +01:00
for ( auto txtIterator = self . txtRecords . cbegin ( ) ; txtIterator ! = self . txtRecords . cend ( ) ; txtIterator + + ) {
2023-07-10 13:34:26 +01:00
additional . append ( createMdnsRecord ( self , MDNS_RECORDTYPE_TXT , nullptr , txtIterator ) ) ;
2023-07-10 09:45:10 +01:00
}
2023-07-05 20:09:28 +01:00
// Send the answer, unicast or multicast depending on flag in query
uint16_t unicast = ( rclass & MDNS_UNICAST_RESPONSE ) ;
2023-07-10 09:45:10 +01:00
printf ( " --> answer %.*s port %d (%s) \n " , MDNS_STRING_FORMAT ( answer . data . srv . name ) , answer . data . srv . port , ( unicast ? " unicast " : " multicast " ) ) ;
2023-07-05 20:09:28 +01:00
if ( unicast ) {
2023-07-10 09:45:10 +01:00
mdns_query_answer_unicast ( sock , from , addrlen , sendbuffer , sizeof ( sendbuffer ) , query_id ,
( mdns_record_type_t ) record_type , nameMdnsString . str , nameMdnsString . length ,
2023-07-29 15:28:37 +01:00
answer , nullptr , 0 , additional . constData ( ) , additional . length ( ) ) ;
2023-07-05 20:09:28 +01:00
} else {
2023-07-29 15:28:37 +01:00
mdns_query_answer_multicast ( sock , sendbuffer , sizeof ( sendbuffer ) , answer , nullptr , 0 ,
2023-07-10 09:45:10 +01:00
additional . constData ( ) , additional . length ( ) ) ;
2023-07-05 20:09:28 +01:00
}
}
2023-07-10 09:45:10 +01:00
} else if ( name = = self . hostname ) {
2023-07-08 12:22:45 +01:00
qWarning ( ) < < " Someone queried my host for " < < recordTypeToStr ( record_type ) ;
2023-07-10 13:34:26 +01:00
if ( ( ( record_type = = MDNS_RECORDTYPE_A ) | | ( record_type = = MDNS_RECORDTYPE_ANY ) ) & & ! self . addressesV4 . empty ( ) ) {
2023-07-10 09:45:10 +01:00
// The A query was for our qualified hostname and we have an IPv4 address, answer with an A
// record mapping the hostname to an IPv4 address, as well as an AAAA record and TXT records
2023-07-08 12:22:45 +01:00
2023-07-10 13:34:26 +01:00
mdns_record_t answer = createMdnsRecord ( self , MDNS_RECORDTYPE_A , from ) ;
2023-07-08 12:22:45 +01:00
2023-07-10 09:45:10 +01:00
QVector < mdns_record_t > additional ;
2023-07-10 13:34:26 +01:00
if ( ! self . addressesV6 . empty ( ) ) {
additional . append ( createMdnsRecord ( self , MDNS_RECORDTYPE_AAAA , from ) ) ;
2023-07-10 09:45:10 +01:00
}
2023-07-08 12:22:45 +01:00
2023-07-10 09:45:10 +01:00
for ( auto txtIterator = self . txtRecords . cbegin ( ) ; txtIterator ! = self . txtRecords . cend ( ) ; txtIterator + + ) {
2023-07-10 13:34:26 +01:00
additional . append ( createMdnsRecord ( self , MDNS_RECORDTYPE_TXT , nullptr , txtIterator ) ) ;
2023-07-10 09:45:10 +01:00
}
2023-07-08 12:22:45 +01:00
// Send the answer, unicast or multicast depending on flag in query
uint16_t unicast = ( rclass & MDNS_UNICAST_RESPONSE ) ;
2023-07-10 09:45:10 +01:00
printf ( " --> answer %.*s IPv4 (%s) \n " , MDNS_STRING_FORMAT ( answer . name ) , ( unicast ? " unicast " : " multicast " ) ) ;
2023-07-08 12:22:45 +01:00
if ( unicast ) {
2023-07-10 09:45:10 +01:00
mdns_query_answer_unicast ( sock , from , addrlen , sendbuffer , sizeof ( sendbuffer ) , query_id ,
( mdns_record_type_t ) record_type , nameMdnsString . str , nameMdnsString . length ,
2023-07-29 15:28:37 +01:00
answer , nullptr , 0 , additional . constData ( ) , additional . length ( ) ) ;
2023-07-08 12:22:45 +01:00
} else {
2023-07-29 15:28:37 +01:00
mdns_query_answer_multicast ( sock , sendbuffer , sizeof ( sendbuffer ) , answer , nullptr , 0 ,
2023-07-10 09:45:10 +01:00
additional . constData ( ) , additional . length ( ) ) ;
2023-07-08 12:22:45 +01:00
}
2023-07-10 13:34:26 +01:00
} else if ( ( ( record_type = = MDNS_RECORDTYPE_AAAA ) | | ( record_type = = MDNS_RECORDTYPE_ANY ) ) & & ! self . addressesV6 . empty ( ) ) {
2023-07-10 09:45:10 +01:00
// The AAAA query was for our qualified hostname and we have an IPv6 address, answer with an AAAA
// record mapping the hostname to an IPv4 address, as well as an A record and TXT records
2023-07-08 12:22:45 +01:00
2023-07-10 13:34:26 +01:00
mdns_record_t answer = createMdnsRecord ( self , MDNS_RECORDTYPE_AAAA , from ) ;
2023-07-08 12:22:45 +01:00
2023-07-10 09:45:10 +01:00
QVector < mdns_record_t > additional ;
2023-07-10 13:34:26 +01:00
if ( ! self . addressesV4 . empty ( ) ) {
additional . append ( createMdnsRecord ( self , MDNS_RECORDTYPE_A , from ) ) ;
2023-07-10 09:45:10 +01:00
}
2023-07-08 12:22:45 +01:00
2023-07-10 09:45:10 +01:00
for ( auto txtIterator = self . txtRecords . cbegin ( ) ; txtIterator ! = self . txtRecords . cend ( ) ; txtIterator + + ) {
2023-07-10 13:34:26 +01:00
additional . append ( createMdnsRecord ( self , MDNS_RECORDTYPE_TXT , nullptr , txtIterator ) ) ;
2023-07-10 09:45:10 +01:00
}
2023-07-08 12:22:45 +01:00
// Send the answer, unicast or multicast depending on flag in query
uint16_t unicast = ( rclass & MDNS_UNICAST_RESPONSE ) ;
2023-07-10 09:45:10 +01:00
printf ( " --> answer %.*s IPv6 (%s) \n " , MDNS_STRING_FORMAT ( answer . name ) , ( unicast ? " unicast " : " multicast " ) ) ;
2023-07-08 12:22:45 +01:00
if ( unicast ) {
2023-07-10 09:45:10 +01:00
mdns_query_answer_unicast ( sock , from , addrlen , sendbuffer , sizeof ( sendbuffer ) , query_id ,
( mdns_record_type_t ) record_type , nameMdnsString . str , nameMdnsString . length ,
2023-07-29 15:28:37 +01:00
answer , nullptr , 0 , additional . constData ( ) , additional . length ( ) ) ;
2023-07-08 12:22:45 +01:00
} else {
2023-07-29 15:28:37 +01:00
mdns_query_answer_multicast ( sock , sendbuffer , sizeof ( sendbuffer ) , answer , nullptr , 0 ,
2023-07-10 09:45:10 +01:00
additional . constData ( ) , additional . length ( ) ) ;
2023-07-05 20:09:28 +01:00
}
}
2023-07-08 12:22:45 +01:00
} // else request is not for me
2023-07-05 20:09:28 +01:00
return 0 ;
}
2023-07-08 12:22:45 +01:00
// Open sockets to listen to incoming mDNS queries on port 5353
// When recieving, each socket can recieve data from all network interfaces
// Thus we only need to open one socket for each address family
2023-07-10 11:11:38 +01:00
int Announcer : : listenForQueries ( )
2023-07-05 20:09:28 +01:00
{
2023-07-08 12:22:45 +01:00
auto callback = [ this ] ( QSocketDescriptor socket ) {
2023-07-10 09:45:10 +01:00
static char buffer [ 2048 ] ;
mdns_socket_listen ( socket , buffer , sizeof ( buffer ) , service_callback , & self ) ;
2023-07-08 12:22:45 +01:00
} ;
2023-07-05 20:09:28 +01:00
2023-07-08 12:22:45 +01:00
int numSockets = 0 ;
2023-07-05 20:09:28 +01:00
2023-07-08 12:22:45 +01:00
{
struct sockaddr_in sock_addr ;
memset ( & sock_addr , 0 , sizeof ( struct sockaddr_in ) ) ;
sock_addr . sin_family = AF_INET ;
# ifdef _WIN32
sock_addr . sin_addr = in4addr_any ;
# else
sock_addr . sin_addr . s_addr = INADDR_ANY ;
# endif
sock_addr . sin_port = htons ( MDNS_PORT ) ;
# ifdef __APPLE__
sock_addr . sin_len = sizeof ( struct sockaddr_in ) ;
# endif
int socket = mdns_socket_open_ipv4 ( & sock_addr ) ;
if ( socket > = 0 ) {
socketNotifier = new QSocketNotifier ( socket , QSocketNotifier : : Read ) ;
QObject : : connect ( socketNotifier , & QSocketNotifier : : activated , callback ) ;
numSockets + + ;
}
}
2023-07-05 20:09:28 +01:00
2023-07-08 12:22:45 +01:00
{
struct sockaddr_in6 sock_addr ;
memset ( & sock_addr , 0 , sizeof ( struct sockaddr_in6 ) ) ;
sock_addr . sin6_family = AF_INET6 ;
sock_addr . sin6_addr = in6addr_any ;
sock_addr . sin6_port = htons ( MDNS_PORT ) ;
# ifdef __APPLE__
sock_addr . sin6_len = sizeof ( struct sockaddr_in6 ) ;
# endif
int socket = mdns_socket_open_ipv6 ( & sock_addr ) ;
if ( socket > = 0 ) {
socketNotifierV6 = new QSocketNotifier ( socket , QSocketNotifier : : Read ) ;
QObject : : connect ( socketNotifierV6 , & QSocketNotifier : : activated , callback ) ;
numSockets + + ;
}
}
2023-07-05 20:09:28 +01:00
2023-07-08 12:22:45 +01:00
return numSockets ;
}
2023-07-05 20:09:28 +01:00
2023-07-10 11:11:38 +01:00
Announcer : : Announcer ( const QString & serviceName , const QString & serviceType , uint16_t port )
2023-07-08 12:22:45 +01:00
{
2023-07-10 09:45:10 +01:00
self . serviceType = serviceType . toLatin1 ( ) ;
if ( ! self . serviceType . endsWith ( ' . ' ) ) {
2023-07-08 12:22:45 +01:00
// mdns.h needs all the qualified names to end with dot for some reason
2023-07-10 09:45:10 +01:00
self . serviceType . append ( ' . ' ) ;
2023-07-08 12:22:45 +01:00
}
2023-07-10 09:45:10 +01:00
self . serviceInstance = serviceName . toLatin1 ( ) + ' . ' + self . serviceType ;
2023-07-10 13:34:26 +01:00
self . hostname = QHostInfo : : localHostName ( ) . toLatin1 ( ) + " .local. " ;
self . port = port ;
detectHostAddresses ( ) ;
}
void Announcer : : detectHostAddresses ( )
{
qWarning ( ) < < " detectHostAddresses " ;
self . addressesV4 . clear ( ) ;
self . addressesV6 . clear ( ) ;
for ( const QNetworkInterface & iface : QNetworkInterface : : allInterfaces ( ) ) {
int flags = iface . flags ( ) ;
if ( ! ( flags & QNetworkInterface : : IsUp ) | | ! ( flags & QNetworkInterface : : CanMulticast ) | | ( flags & QNetworkInterface : : IsLoopBack ) ) {
continue ;
}
for ( const QNetworkAddressEntry & ifaceAddress : iface . addressEntries ( ) ) {
QHostAddress sourceAddress = ifaceAddress . ip ( ) ;
if ( sourceAddress . protocol ( ) = = QAbstractSocket : : IPv4Protocol & & sourceAddress ! = QHostAddress : : LocalHost ) {
qWarning ( ) < < " Found ipv4 " < < sourceAddress ;
self . addressesV4 . append ( sourceAddress ) ;
} else if ( sourceAddress . protocol ( ) = = QAbstractSocket : : IPv6Protocol & & sourceAddress ! = QHostAddress : : LocalHostIPv6 ) {
qWarning ( ) < < " Found ipv6 " < < sourceAddress ;
self . addressesV6 . append ( sourceAddress ) ;
}
}
}
2023-07-08 12:22:45 +01:00
}
2023-07-05 20:09:28 +01:00
2023-07-10 11:11:38 +01:00
void Announcer : : startAnnouncing ( )
2023-07-08 12:22:45 +01:00
{
int num_sockets = listenForQueries ( ) ;
if ( num_sockets < = 0 ) {
qWarning ( ) < < " Failed to open any client sockets " ;
return ;
2023-07-05 20:09:28 +01:00
}
2023-07-08 12:22:45 +01:00
sendMulticastAnnounce ( false ) ;
2023-07-05 20:09:28 +01:00
}
2023-07-10 11:11:38 +01:00
void Announcer : : stopAnnouncing ( )
2023-07-08 12:22:45 +01:00
{
sendMulticastAnnounce ( true ) ;
stopListeningForQueries ( ) ;
}
2023-07-05 20:09:28 +01:00
2023-07-10 11:11:38 +01:00
void Announcer : : stopListeningForQueries ( )
2023-07-08 12:22:45 +01:00
{
if ( socketNotifier ! = nullptr ) {
delete socketNotifier ;
socketNotifier = nullptr ;
}
if ( socketNotifierV6 ! = nullptr ) {
delete socketNotifierV6 ;
socketNotifierV6 = nullptr ;
}
}
2023-07-05 20:09:28 +01:00
2023-07-10 11:11:38 +01:00
void Announcer : : sendMulticastAnnounce ( bool isGoodbye )
2023-07-05 20:09:28 +01:00
{
2023-07-10 09:45:10 +01:00
mdns_record_t ptr_record = createMdnsRecord ( self , MDNS_RECORDTYPE_PTR ) ;
QVector < mdns_record_t > additional ;
additional . append ( createMdnsRecord ( self , MDNS_RECORDTYPE_SRV ) ) ;
2023-07-10 13:34:26 +01:00
if ( ! self . addressesV4 . empty ( ) ) {
2023-07-10 09:45:10 +01:00
additional . append ( createMdnsRecord ( self , MDNS_RECORDTYPE_A ) ) ;
}
2023-07-10 13:34:26 +01:00
if ( ! self . addressesV6 . empty ( ) ) {
2023-07-10 09:45:10 +01:00
additional . append ( createMdnsRecord ( self , MDNS_RECORDTYPE_AAAA ) ) ;
}
for ( auto txtIterator = self . txtRecords . cbegin ( ) ; txtIterator ! = self . txtRecords . cend ( ) ; txtIterator + + ) {
2023-07-10 13:34:26 +01:00
additional . append ( createMdnsRecord ( self , MDNS_RECORDTYPE_TXT , nullptr , txtIterator ) ) ;
2023-07-10 09:45:10 +01:00
}
2023-07-05 20:09:28 +01:00
2023-07-10 09:45:10 +01:00
static char buffer [ 2048 ] ;
2023-07-08 12:22:45 +01:00
if ( isGoodbye ) {
qCDebug ( KDECONNECT_CORE ) < < " Sending goodbye " ;
2023-07-16 13:43:37 +01:00
if ( socketNotifier )
2023-07-29 15:28:37 +01:00
mdns_goodbye_multicast ( socketNotifier - > socket ( ) , buffer , sizeof ( buffer ) , ptr_record , nullptr , 0 , additional . constData ( ) , additional . length ( ) ) ;
2023-07-16 13:43:37 +01:00
if ( socketNotifierV6 )
2023-07-29 15:28:37 +01:00
mdns_goodbye_multicast ( socketNotifierV6 - > socket ( ) , buffer , sizeof ( buffer ) , ptr_record , nullptr , 0 , additional . constData ( ) , additional . length ( ) ) ;
2023-07-08 12:22:45 +01:00
} else {
qCDebug ( KDECONNECT_CORE ) < < " Sending announce " ;
2023-07-16 13:43:37 +01:00
if ( socketNotifier )
2023-07-29 15:28:37 +01:00
mdns_announce_multicast ( socketNotifier - > socket ( ) , buffer , sizeof ( buffer ) , ptr_record , nullptr , 0 , additional . constData ( ) , additional . length ( ) ) ;
2023-07-16 13:43:37 +01:00
if ( socketNotifierV6 )
2023-07-29 15:28:37 +01:00
mdns_announce_multicast ( socketNotifierV6 - > socket ( ) , buffer , sizeof ( buffer ) , ptr_record , nullptr , 0 , additional . constData ( ) , additional . length ( ) ) ;
2023-07-05 20:09:28 +01:00
}
}
2023-07-10 11:11:38 +01:00
} // namespace MdnsWrapper