mirror of
https://github.com/espressif/openthread.git
synced 2026-06-06 05:24:51 +00:00
bf41332061
This commit introduces native mDNS support within the OpenThread stack, implementing RFC 6762 compliant registration of hosts, services, and keys. It supports the following functionalities: - Sending probes to claim names. - Sending announcements on initial registration and changes. - Sending "goodbye" announcements when unregistered or upon record removal. - Negative responses (NSEC). - Support for service sub-types and their addition/removal. - Support for `_services._dns-sd._udp` queries (all service types). - Responding to queries (including "QU" questions). - Delay mechanism when responding to multi-question query messages, ensuring unique answers. - Providing extra records in the Additional Data section if not already in the Answer section (e.g., on a PTR query, include SRV and host AAAA addresses). - Implementing Known-Answer Suppression. - Supporting multi-packet queries with known answers in follow-up messages. - Rate-limiting record multicasts (once per second). - Rate-limiting probe responses (once per 250ms). - Detecting conflicts after probes. - Limiting the size of emitted responses or probes, breaking into multiple messages if necessary. - Detecting self originating messages (sent by mDNS module). - Support for service browser. - Support for service resolvers (SRV and TXT records) and IPv4/IPv6 address resolvers for hostnames. - Introduces smart cache management: - Passively caches service records for active browsers. - Passively caches address records for active service resolvers. - Enables multiple simultaneous browsers/resolvers for the same service/host. This commit introduces public `otMdns` OpenThread APIs and related CLI commands for the mDNS module. For platform abstraction, `otPlatMdns` APIs are defined in `mdns_socket.h` (e.g., to send or receive mDNS messages): - An implementation of the platform APIs is provided for posix. - Also under the simulation platform, a simplified implementation of the `otPlatMdns` APIs is provided (intended for testing). This commit also adds a detailed `test_mdns` unit test, validating various functionalities and covering potential edge cases.
346 lines
9.9 KiB
C
346 lines
9.9 KiB
C
/*
|
|
* Copyright (c) 2016, The OpenThread Authors.
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* 3. Neither the name of the copyright holder nor the
|
|
* names of its contributors may be used to endorse or promote products
|
|
* derived from this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
|
|
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
/**
|
|
* @file
|
|
* @brief
|
|
* This file includes the platform-specific initializers.
|
|
*/
|
|
|
|
#include "platform-simulation.h"
|
|
|
|
#if OPENTHREAD_SIMULATION_VIRTUAL_TIME == 0
|
|
|
|
#include <arpa/inet.h>
|
|
#include <assert.h>
|
|
#include <errno.h>
|
|
#include <getopt.h>
|
|
#include <ifaddrs.h>
|
|
#include <libgen.h>
|
|
#include <netinet/in.h>
|
|
#include <stddef.h>
|
|
#include <stdint.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <sys/types.h>
|
|
|
|
#include <openthread/tasklet.h>
|
|
#include <openthread/platform/alarm-milli.h>
|
|
#include <openthread/platform/radio.h>
|
|
|
|
#include "simul_utils.h"
|
|
#include "utils/code_utils.h"
|
|
|
|
uint32_t gNodeId = 1;
|
|
|
|
extern bool gPlatformPseudoResetWasRequested;
|
|
extern otRadioCaps gRadioCaps;
|
|
|
|
static volatile bool gTerminate = false;
|
|
|
|
static void handleSignal(int aSignal)
|
|
{
|
|
OT_UNUSED_VARIABLE(aSignal);
|
|
|
|
gTerminate = true;
|
|
}
|
|
|
|
/**
|
|
* Defines the argument return values.
|
|
*
|
|
*/
|
|
enum
|
|
{
|
|
OT_SIM_OPT_HELP = 'h',
|
|
OT_SIM_OPT_ENABLE_ENERGY_SCAN = 'E',
|
|
OT_SIM_OPT_LOCAL_HOST = 'L',
|
|
OT_SIM_OPT_SLEEP_TO_TX = 't',
|
|
OT_SIM_OPT_TIME_SPEED = 's',
|
|
OT_SIM_OPT_LOG_FILE = 'l',
|
|
OT_SIM_OPT_UNKNOWN = '?',
|
|
};
|
|
|
|
static void PrintUsage(const char *aProgramName, int aExitCode)
|
|
{
|
|
fprintf(stderr,
|
|
"Syntax:\n"
|
|
" %s [Options] NodeId\n"
|
|
"Options:\n"
|
|
" -h --help Display this usage information.\n"
|
|
" -E --enable-energy-scan Enable energy scan capability.\n"
|
|
" -t --sleep-to-tx Let radio support direct transition from sleep to TX with CSMA.\n"
|
|
" -s --time-speed=val Speed up the time in simulation.\n"
|
|
#if (OPENTHREAD_CONFIG_LOG_OUTPUT == OPENTHREAD_CONFIG_LOG_OUTPUT_PLATFORM_DEFINED)
|
|
" -l --log-file=name File name to write logs.\n"
|
|
#endif
|
|
,
|
|
aProgramName);
|
|
|
|
exit(aExitCode);
|
|
}
|
|
|
|
static const char *GetLocalHostAddress(const char *aLocalHost)
|
|
{
|
|
struct ifaddrs *ifaddr;
|
|
static char ipstr[INET_ADDRSTRLEN] = {0};
|
|
const char *rval = NULL;
|
|
|
|
{
|
|
struct in_addr addr;
|
|
|
|
otEXPECT_ACTION(inet_aton(aLocalHost, &addr) == 0, rval = aLocalHost);
|
|
}
|
|
|
|
if (getifaddrs(&ifaddr) == -1)
|
|
{
|
|
perror("getifaddrs");
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
|
|
for (struct ifaddrs *ifa = ifaddr; ifa != NULL; ifa = ifa->ifa_next)
|
|
{
|
|
if (ifa->ifa_addr == NULL || ifa->ifa_addr->sa_family != AF_INET)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (strcmp(ifa->ifa_name, aLocalHost) == 0)
|
|
{
|
|
struct sockaddr_in *addr = (struct sockaddr_in *)ifa->ifa_addr;
|
|
|
|
if (inet_ntop(AF_INET, &addr->sin_addr, ipstr, sizeof(ipstr)))
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
freeifaddrs(ifaddr);
|
|
|
|
if (ipstr[0] == '\0')
|
|
{
|
|
fprintf(stderr, "Local host address not found!\n");
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
|
|
rval = ipstr;
|
|
|
|
exit:
|
|
return rval;
|
|
}
|
|
|
|
void otSysInit(int aArgCount, char *aArgVector[])
|
|
{
|
|
char *endptr;
|
|
uint32_t speedUpFactor = 1;
|
|
|
|
static const struct option long_options[] = {
|
|
{"help", no_argument, 0, OT_SIM_OPT_HELP},
|
|
{"enable-energy-scan", no_argument, 0, OT_SIM_OPT_ENABLE_ENERGY_SCAN},
|
|
{"sleep-to-tx", no_argument, 0, OT_SIM_OPT_SLEEP_TO_TX},
|
|
{"time-speed", required_argument, 0, OT_SIM_OPT_TIME_SPEED},
|
|
{"local-host", required_argument, 0, OT_SIM_OPT_LOCAL_HOST},
|
|
#if (OPENTHREAD_CONFIG_LOG_OUTPUT == OPENTHREAD_CONFIG_LOG_OUTPUT_PLATFORM_DEFINED)
|
|
{"log-file", required_argument, 0, OT_SIM_OPT_LOG_FILE},
|
|
#endif
|
|
{0, 0, 0, 0},
|
|
};
|
|
|
|
#if (OPENTHREAD_CONFIG_LOG_OUTPUT == OPENTHREAD_CONFIG_LOG_OUTPUT_PLATFORM_DEFINED)
|
|
static const char options[] = "Ehts:L:l:";
|
|
#else
|
|
static const char options[] = "Ehts:L:";
|
|
#endif
|
|
|
|
if (gPlatformPseudoResetWasRequested)
|
|
{
|
|
gPlatformPseudoResetWasRequested = false;
|
|
return;
|
|
}
|
|
|
|
optind = 1;
|
|
|
|
while (true)
|
|
{
|
|
int c = getopt_long(aArgCount, aArgVector, options, long_options, NULL);
|
|
|
|
if (c == -1)
|
|
{
|
|
break;
|
|
}
|
|
|
|
switch (c)
|
|
{
|
|
case OT_SIM_OPT_UNKNOWN:
|
|
PrintUsage(aArgVector[0], EXIT_FAILURE);
|
|
break;
|
|
case OT_SIM_OPT_HELP:
|
|
PrintUsage(aArgVector[0], EXIT_SUCCESS);
|
|
break;
|
|
case OT_SIM_OPT_ENABLE_ENERGY_SCAN:
|
|
gRadioCaps |= OT_RADIO_CAPS_ENERGY_SCAN;
|
|
break;
|
|
case OT_SIM_OPT_SLEEP_TO_TX:
|
|
gRadioCaps |= OT_RADIO_CAPS_SLEEP_TO_TX;
|
|
break;
|
|
case OT_SIM_OPT_LOCAL_HOST:
|
|
gLocalHost = GetLocalHostAddress(optarg);
|
|
fprintf(stderr, "Simulate on %s\n", gLocalHost);
|
|
break;
|
|
case OT_SIM_OPT_TIME_SPEED:
|
|
speedUpFactor = (uint32_t)strtol(optarg, &endptr, 10);
|
|
if (*endptr != '\0' || speedUpFactor == 0)
|
|
{
|
|
fprintf(stderr, "Invalid value for TimerSpeedUpFactor: %s\n", optarg);
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
break;
|
|
#if OPENTHREAD_CONFIG_LOG_OUTPUT == OPENTHREAD_CONFIG_LOG_OUTPUT_PLATFORM_DEFINED
|
|
case OT_SIM_OPT_LOG_FILE:
|
|
platformLoggingSetFileName(optarg);
|
|
break;
|
|
#endif
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (optind != aArgCount - 1)
|
|
{
|
|
PrintUsage(aArgVector[0], EXIT_FAILURE);
|
|
}
|
|
|
|
gNodeId = (uint32_t)strtol(aArgVector[optind], &endptr, 0);
|
|
|
|
if (*endptr != '\0' || gNodeId < 1 || gNodeId > MAX_NETWORK_SIZE)
|
|
{
|
|
fprintf(stderr, "Invalid NodeId: %s\n", aArgVector[optind]);
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
|
|
signal(SIGTERM, &handleSignal);
|
|
signal(SIGHUP, &handleSignal);
|
|
|
|
platformLoggingInit(basename(aArgVector[0]));
|
|
platformAlarmInit(speedUpFactor);
|
|
platformRadioInit();
|
|
#if OPENTHREAD_CONFIG_RADIO_LINK_TREL_ENABLE
|
|
platformTrelInit(speedUpFactor);
|
|
#endif
|
|
#if OPENTHREAD_CONFIG_BORDER_ROUTING_ENABLE
|
|
platformInfraIfInit();
|
|
#endif
|
|
platformRandomInit();
|
|
}
|
|
|
|
bool otSysPseudoResetWasRequested(void) { return gPlatformPseudoResetWasRequested; }
|
|
|
|
void otSysDeinit(void)
|
|
{
|
|
platformRadioDeinit();
|
|
#if OPENTHREAD_CONFIG_RADIO_LINK_TREL_ENABLE
|
|
platformTrelDeinit();
|
|
#endif
|
|
#if OPENTHREAD_CONFIG_BORDER_ROUTING_ENABLE
|
|
// platformInfrIfDeinit();
|
|
#endif
|
|
platformLoggingDeinit();
|
|
}
|
|
|
|
void otSysProcessDrivers(otInstance *aInstance)
|
|
{
|
|
fd_set read_fds;
|
|
fd_set write_fds;
|
|
fd_set error_fds;
|
|
int max_fd = -1;
|
|
struct timeval timeout;
|
|
int rval;
|
|
|
|
FD_ZERO(&read_fds);
|
|
FD_ZERO(&write_fds);
|
|
FD_ZERO(&error_fds);
|
|
|
|
platformUartUpdateFdSet(&read_fds, &write_fds, &error_fds, &max_fd);
|
|
platformAlarmUpdateTimeout(&timeout);
|
|
platformRadioUpdateFdSet(&read_fds, &write_fds, &timeout, &max_fd);
|
|
#if OPENTHREAD_CONFIG_RADIO_LINK_TREL_ENABLE
|
|
platformTrelUpdateFdSet(&read_fds, &write_fds, &timeout, &max_fd);
|
|
#endif
|
|
#if OPENTHREAD_CONFIG_BORDER_ROUTING_ENABLE
|
|
platformInfraIfUpdateFdSet(&read_fds, &write_fds, &max_fd);
|
|
#endif
|
|
#if OPENTHREAD_CONFIG_MULTICAST_DNS_ENABLE && OPENTHREAD_SIMULATION_MDNS_SOCKET_IMPLEMENT_POSIX
|
|
platformMdnsSocketUpdateFdSet(&read_fds, &max_fd);
|
|
#endif
|
|
|
|
#if OPENTHREAD_CONFIG_BLE_TCAT_ENABLE
|
|
platformBleUpdateFdSet(&read_fds, &write_fds, &timeout, &max_fd);
|
|
#endif
|
|
|
|
if (otTaskletsArePending(aInstance))
|
|
{
|
|
timeout.tv_sec = 0;
|
|
timeout.tv_usec = 0;
|
|
}
|
|
|
|
rval = select(max_fd + 1, &read_fds, &write_fds, &error_fds, &timeout);
|
|
|
|
if (rval >= 0)
|
|
{
|
|
platformUartProcess();
|
|
platformRadioProcess(aInstance, &read_fds, &write_fds);
|
|
#if OPENTHREAD_CONFIG_BLE_TCAT_ENABLE
|
|
platformBleProcess(aInstance, &read_fds, &write_fds);
|
|
#endif
|
|
}
|
|
else if (errno != EINTR)
|
|
{
|
|
perror("select");
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
|
|
platformAlarmProcess(aInstance);
|
|
#if OPENTHREAD_CONFIG_RADIO_LINK_TREL_ENABLE
|
|
platformTrelProcess(aInstance, &read_fds, &write_fds);
|
|
#endif
|
|
#if OPENTHREAD_CONFIG_BORDER_ROUTING_ENABLE
|
|
platformInfraIfProcess(aInstance, &read_fds, &write_fds);
|
|
#endif
|
|
#if OPENTHREAD_CONFIG_MULTICAST_DNS_ENABLE && OPENTHREAD_SIMULATION_MDNS_SOCKET_IMPLEMENT_POSIX
|
|
platformMdnsSocketProcess(aInstance, &read_fds);
|
|
#endif
|
|
|
|
if (gTerminate)
|
|
{
|
|
exit(0);
|
|
}
|
|
}
|
|
|
|
#endif // OPENTHREAD_SIMULATION_VIRTUAL_TIME == 0
|