mirror of
https://github.com/isc-projects/bind9.git
synced 2026-05-04 17:24:47 -04:00
With taskmgr running on top of netmgr, the ordering of how the tasks and
netmgr shutdown interacts was wrong as previously isc_taskmgr_destroy()
was waiting until all tasks were properly shutdown and detached. This
responsibility was moved to netmgr, so we now need to do the following:
1. shutdown all the tasks - this schedules all shutdown events onto
the netmgr queue
2. shutdown the netmgr - this also makes sure all the tasks and
events are properly executed
3. Shutdown the taskmgr - this now waits for all the tasks to finish
running before returning
4. Shutdown the netmgr - this call waits for all the netmgr netievents
to finish before returning
This solves the race when the taskmgr object would be destroyed before
all the tasks were finished running in the netmgr loops.
481 lines
11 KiB
C
481 lines
11 KiB
C
/*
|
|
* Copyright (C) Internet Systems Consortium, Inc. ("ISC")
|
|
*
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, you can obtain one at https://mozilla.org/MPL/2.0/.
|
|
*
|
|
* See the COPYRIGHT file distributed with this work for additional
|
|
* information regarding copyright ownership.
|
|
*/
|
|
|
|
#include <fcntl.h>
|
|
#include <getopt.h>
|
|
#include <netdb.h>
|
|
#include <netinet/in.h>
|
|
#include <signal.h>
|
|
#include <stdbool.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <strings.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/types.h>
|
|
#include <unistd.h>
|
|
|
|
#include <isc/managers.h>
|
|
#include <isc/mem.h>
|
|
#include <isc/netaddr.h>
|
|
#include <isc/netmgr.h>
|
|
#include <isc/os.h>
|
|
#include <isc/print.h>
|
|
#include <isc/sockaddr.h>
|
|
#include <isc/string.h>
|
|
#include <isc/util.h>
|
|
|
|
#define DEFAULT_DOH_PATH "/dns-query"
|
|
|
|
typedef enum {
|
|
UDP,
|
|
TCP,
|
|
DOT,
|
|
HTTPS_POST,
|
|
HTTPS_GET,
|
|
HTTP_POST,
|
|
HTTP_GET
|
|
} protocol_t;
|
|
|
|
static const char *protocols[] = { "udp", "tcp",
|
|
"dot", "https-post",
|
|
"https-get", "http-plain-post",
|
|
"http-plain-get" };
|
|
|
|
static isc_mem_t *mctx = NULL;
|
|
static isc_nm_t *netmgr = NULL;
|
|
|
|
static protocol_t protocol;
|
|
static const char *address;
|
|
static const char *port;
|
|
static int family = AF_UNSPEC;
|
|
static isc_sockaddr_t sockaddr_local;
|
|
static isc_sockaddr_t sockaddr_remote;
|
|
static int workers;
|
|
static int timeout;
|
|
static uint8_t messagebuf[2 * 65536];
|
|
static isc_region_t message = { .length = 0, .base = messagebuf };
|
|
static int out = -1;
|
|
|
|
static isc_tlsctx_t *tls_ctx = NULL;
|
|
|
|
static isc_result_t
|
|
parse_port(const char *input) {
|
|
char *endptr = NULL;
|
|
long val = strtol(input, &endptr, 10);
|
|
|
|
if ((*endptr != '\0') || (val <= 0) || (val >= 65536)) {
|
|
return (ISC_R_BADNUMBER);
|
|
}
|
|
|
|
port = input;
|
|
|
|
return (ISC_R_SUCCESS);
|
|
}
|
|
|
|
static isc_result_t
|
|
parse_protocol(const char *input) {
|
|
for (size_t i = 0; i < ARRAY_SIZE(protocols); i++) {
|
|
if (!strcasecmp(input, protocols[i])) {
|
|
protocol = i;
|
|
return (ISC_R_SUCCESS);
|
|
}
|
|
}
|
|
|
|
return (ISC_R_BADNUMBER);
|
|
}
|
|
|
|
static isc_result_t
|
|
parse_address(const char *input) {
|
|
struct in6_addr in6;
|
|
struct in_addr in;
|
|
|
|
if (inet_pton(AF_INET6, input, &in6) == 1) {
|
|
family = AF_INET6;
|
|
address = input;
|
|
return (ISC_R_SUCCESS);
|
|
}
|
|
|
|
if (inet_pton(AF_INET, input, &in) == 1) {
|
|
family = AF_INET;
|
|
address = input;
|
|
return (ISC_R_SUCCESS);
|
|
}
|
|
|
|
return (ISC_R_BADADDRESSFORM);
|
|
}
|
|
|
|
static int
|
|
parse_workers(const char *input) {
|
|
char *endptr = NULL;
|
|
long val = strtol(input, &endptr, 10);
|
|
|
|
if ((*endptr != '\0') || (val <= 0) || (val >= 128)) {
|
|
return (ISC_R_BADNUMBER);
|
|
}
|
|
|
|
workers = val;
|
|
|
|
return (ISC_R_SUCCESS);
|
|
}
|
|
|
|
static isc_result_t
|
|
parse_timeout(const char *input) {
|
|
char *endptr = NULL;
|
|
long val = strtol(input, &endptr, 10);
|
|
|
|
if ((*endptr != '\0') || (val <= 0) || (val >= 120)) {
|
|
return (ISC_R_BADNUMBER);
|
|
}
|
|
|
|
timeout = (in_port_t)val * 1000;
|
|
|
|
return (ISC_R_SUCCESS);
|
|
}
|
|
|
|
static isc_result_t
|
|
parse_input(const char *input) {
|
|
int in = -1;
|
|
|
|
if (!strcmp(input, "-")) {
|
|
in = 0;
|
|
} else {
|
|
in = open(input, O_RDONLY);
|
|
}
|
|
RUNTIME_CHECK(in >= 0);
|
|
|
|
message.length = read(in, message.base, sizeof(messagebuf));
|
|
|
|
close(in);
|
|
|
|
return (ISC_R_SUCCESS);
|
|
}
|
|
|
|
static isc_result_t
|
|
parse_output(const char *input) {
|
|
if (!strcmp(input, "-")) {
|
|
out = 1;
|
|
} else {
|
|
out = open(input, O_WRONLY | O_CREAT,
|
|
S_IRUSR | S_IRGRP | S_IROTH);
|
|
}
|
|
RUNTIME_CHECK(out >= 0);
|
|
|
|
return (ISC_R_SUCCESS);
|
|
}
|
|
|
|
static void
|
|
parse_options(int argc, char **argv) {
|
|
char buf[ISC_NETADDR_FORMATSIZE];
|
|
|
|
/* Set defaults */
|
|
RUNTIME_CHECK(parse_protocol("UDP") == ISC_R_SUCCESS);
|
|
RUNTIME_CHECK(parse_port("53000") == ISC_R_SUCCESS);
|
|
RUNTIME_CHECK(parse_address("::0") == ISC_R_SUCCESS);
|
|
workers = isc_os_ncpus();
|
|
|
|
while (true) {
|
|
int c;
|
|
int option_index = 0;
|
|
static struct option long_options[] = {
|
|
{ "port", required_argument, NULL, 'p' },
|
|
{ "address", required_argument, NULL, 'a' },
|
|
{ "protocol", required_argument, NULL, 'P' },
|
|
{ "workers", required_argument, NULL, 'w' },
|
|
{ "timeout", required_argument, NULL, 't' },
|
|
{ "input", required_argument, NULL, 'i' },
|
|
{ "output", required_argument, NULL, 'o' },
|
|
{ 0, 0, NULL, 0 }
|
|
};
|
|
|
|
c = getopt_long(argc, argv, "a:p:P:w:t:i:o:", long_options,
|
|
&option_index);
|
|
if (c == -1) {
|
|
break;
|
|
}
|
|
|
|
switch (c) {
|
|
case 'a':
|
|
RUNTIME_CHECK(parse_address(optarg) == ISC_R_SUCCESS);
|
|
break;
|
|
|
|
case 'p':
|
|
RUNTIME_CHECK(parse_port(optarg) == ISC_R_SUCCESS);
|
|
break;
|
|
|
|
case 'P':
|
|
RUNTIME_CHECK(parse_protocol(optarg) == ISC_R_SUCCESS);
|
|
break;
|
|
|
|
case 'w':
|
|
RUNTIME_CHECK(parse_workers(optarg) == ISC_R_SUCCESS);
|
|
break;
|
|
|
|
case 't':
|
|
RUNTIME_CHECK(parse_timeout(optarg) == ISC_R_SUCCESS);
|
|
break;
|
|
|
|
case 'i':
|
|
RUNTIME_CHECK(parse_input(optarg) == ISC_R_SUCCESS);
|
|
break;
|
|
|
|
case 'o':
|
|
RUNTIME_CHECK(parse_output(optarg) == ISC_R_SUCCESS);
|
|
break;
|
|
|
|
default:
|
|
INSIST(0);
|
|
}
|
|
}
|
|
|
|
{
|
|
struct addrinfo hints = {
|
|
.ai_family = family,
|
|
.ai_socktype = (protocol == UDP) ? SOCK_DGRAM
|
|
: SOCK_STREAM,
|
|
};
|
|
struct addrinfo *result = NULL;
|
|
int r = getaddrinfo(address, NULL, &hints, &result);
|
|
RUNTIME_CHECK(r == 0);
|
|
|
|
for (struct addrinfo *rp = result; rp != NULL; rp = rp->ai_next)
|
|
{
|
|
RUNTIME_CHECK(isc_sockaddr_fromsockaddr(&sockaddr_local,
|
|
rp->ai_addr) ==
|
|
ISC_R_SUCCESS);
|
|
}
|
|
freeaddrinfo(result);
|
|
}
|
|
|
|
{
|
|
struct addrinfo hints = {
|
|
.ai_family = family,
|
|
.ai_socktype = (protocol == UDP) ? SOCK_DGRAM
|
|
: SOCK_STREAM,
|
|
};
|
|
struct addrinfo *result = NULL;
|
|
int r = getaddrinfo(argv[optind], port, &hints, &result);
|
|
RUNTIME_CHECK(r == 0);
|
|
|
|
for (struct addrinfo *rp = result; rp != NULL; rp = rp->ai_next)
|
|
{
|
|
RUNTIME_CHECK(isc_sockaddr_fromsockaddr(
|
|
&sockaddr_remote, rp->ai_addr) ==
|
|
ISC_R_SUCCESS);
|
|
}
|
|
freeaddrinfo(result);
|
|
}
|
|
|
|
isc_sockaddr_format(&sockaddr_local, buf, sizeof(buf));
|
|
|
|
printf("Will connect from %s://%s", protocols[protocol], buf);
|
|
|
|
isc_sockaddr_format(&sockaddr_remote, buf, sizeof(buf));
|
|
|
|
printf(" to %s, %d workers\n", buf, workers);
|
|
}
|
|
|
|
static void
|
|
_signal(int sig, void (*handler)(int)) {
|
|
struct sigaction sa = { .sa_handler = handler };
|
|
|
|
RUNTIME_CHECK(sigfillset(&sa.sa_mask) == 0);
|
|
RUNTIME_CHECK(sigaction(sig, &sa, NULL) >= 0);
|
|
}
|
|
|
|
static void
|
|
setup(void) {
|
|
sigset_t sset;
|
|
|
|
_signal(SIGPIPE, SIG_IGN);
|
|
_signal(SIGHUP, SIG_DFL);
|
|
_signal(SIGTERM, SIG_DFL);
|
|
_signal(SIGINT, SIG_DFL);
|
|
|
|
RUNTIME_CHECK(sigemptyset(&sset) == 0);
|
|
RUNTIME_CHECK(sigaddset(&sset, SIGHUP) == 0);
|
|
RUNTIME_CHECK(sigaddset(&sset, SIGINT) == 0);
|
|
RUNTIME_CHECK(sigaddset(&sset, SIGTERM) == 0);
|
|
RUNTIME_CHECK(pthread_sigmask(SIG_BLOCK, &sset, NULL) == 0);
|
|
|
|
isc_mem_create(&mctx);
|
|
|
|
isc_managers_create(mctx, workers, 0, 0, &netmgr, NULL, NULL, NULL);
|
|
}
|
|
|
|
static void
|
|
teardown(void) {
|
|
if (out > 0) {
|
|
close(out);
|
|
}
|
|
|
|
isc_managers_destroy(&netmgr, NULL, NULL, NULL);
|
|
isc_mem_destroy(&mctx);
|
|
if (tls_ctx) {
|
|
isc_tlsctx_free(&tls_ctx);
|
|
}
|
|
}
|
|
|
|
static void
|
|
waitforsignal(void) {
|
|
sigset_t sset;
|
|
int sig;
|
|
|
|
RUNTIME_CHECK(sigemptyset(&sset) == 0);
|
|
RUNTIME_CHECK(sigaddset(&sset, SIGHUP) == 0);
|
|
RUNTIME_CHECK(sigaddset(&sset, SIGINT) == 0);
|
|
RUNTIME_CHECK(sigaddset(&sset, SIGTERM) == 0);
|
|
RUNTIME_CHECK(sigwait(&sset, &sig) == 0);
|
|
|
|
fprintf(stderr, "Shutting down...\n");
|
|
}
|
|
|
|
static void
|
|
read_cb(isc_nmhandle_t *handle, isc_result_t eresult, isc_region_t *region,
|
|
void *cbarg) {
|
|
isc_nmhandle_t *readhandle = cbarg;
|
|
|
|
REQUIRE(handle != NULL);
|
|
REQUIRE(eresult == ISC_R_SUCCESS || eresult == ISC_R_CANCELED ||
|
|
eresult == ISC_R_EOF);
|
|
REQUIRE(cbarg != NULL);
|
|
|
|
fprintf(stderr, "%s(..., %s, ...)\n", __func__,
|
|
isc_result_totext(eresult));
|
|
|
|
if (eresult == ISC_R_SUCCESS) {
|
|
printf("RECEIVED %u bytes\n", region->length);
|
|
if (out >= 0) {
|
|
ssize_t len = write(out, region->base, region->length);
|
|
close(out);
|
|
REQUIRE((size_t)len == region->length);
|
|
}
|
|
}
|
|
|
|
isc_nmhandle_detach(&readhandle);
|
|
kill(getpid(), SIGTERM);
|
|
}
|
|
|
|
static void
|
|
send_cb(isc_nmhandle_t *handle, isc_result_t eresult, void *cbarg) {
|
|
REQUIRE(handle != NULL);
|
|
REQUIRE(eresult == ISC_R_SUCCESS || eresult == ISC_R_CANCELED ||
|
|
eresult == ISC_R_EOF);
|
|
REQUIRE(cbarg == NULL);
|
|
}
|
|
|
|
static void
|
|
connect_cb(isc_nmhandle_t *handle, isc_result_t eresult, void *cbarg) {
|
|
isc_nmhandle_t *readhandle = NULL;
|
|
|
|
REQUIRE(handle != NULL);
|
|
UNUSED(cbarg);
|
|
|
|
fprintf(stderr, "ECHO_CLIENT:%s:%s\n", __func__,
|
|
isc_result_totext(eresult));
|
|
|
|
if (eresult != ISC_R_SUCCESS) {
|
|
kill(getpid(), SIGTERM);
|
|
return;
|
|
}
|
|
|
|
isc_nmhandle_attach(handle, &readhandle);
|
|
isc_nm_read(handle, read_cb, readhandle);
|
|
isc_nm_send(handle, &message, send_cb, NULL);
|
|
}
|
|
|
|
static void
|
|
sockaddr_to_url(isc_sockaddr_t *sa, const bool https, char *outbuf,
|
|
size_t outbuf_len, const char *append) {
|
|
uint16_t sa_port;
|
|
char saddr[INET6_ADDRSTRLEN] = { 0 };
|
|
int sa_family;
|
|
|
|
if (sa == NULL || outbuf == NULL || outbuf_len == 0) {
|
|
return;
|
|
}
|
|
|
|
sa_family = ((struct sockaddr *)&sa->type.sa)->sa_family;
|
|
|
|
sa_port = ntohs(sa_family == AF_INET ? sa->type.sin.sin_port
|
|
: sa->type.sin6.sin6_port);
|
|
inet_ntop(sa_family,
|
|
sa_family == AF_INET
|
|
? (struct sockaddr *)&sa->type.sin.sin_addr
|
|
: (struct sockaddr *)&sa->type.sin6.sin6_addr,
|
|
saddr, sizeof(saddr));
|
|
|
|
snprintf(outbuf, outbuf_len, "%s://%s%s%s:%u%s",
|
|
https ? "https" : "http", sa_family == AF_INET ? "" : "[",
|
|
saddr, sa_family == AF_INET ? "" : "]", sa_port,
|
|
append ? append : "");
|
|
}
|
|
|
|
static void
|
|
run(void) {
|
|
switch (protocol) {
|
|
case UDP:
|
|
isc_nm_udpconnect(netmgr, (isc_nmiface_t *)&sockaddr_local,
|
|
(isc_nmiface_t *)&sockaddr_remote, connect_cb,
|
|
NULL, timeout, 0);
|
|
break;
|
|
case TCP:
|
|
isc_nm_tcpdnsconnect(netmgr, (isc_nmiface_t *)&sockaddr_local,
|
|
(isc_nmiface_t *)&sockaddr_remote,
|
|
connect_cb, NULL, timeout, 0);
|
|
break;
|
|
case DOT: {
|
|
isc_tlsctx_createclient(&tls_ctx);
|
|
|
|
isc_nm_tlsdnsconnect(netmgr, (isc_nmiface_t *)&sockaddr_local,
|
|
(isc_nmiface_t *)&sockaddr_remote,
|
|
connect_cb, NULL, timeout, 0, tls_ctx);
|
|
break;
|
|
}
|
|
case HTTP_GET:
|
|
case HTTPS_GET:
|
|
case HTTPS_POST:
|
|
case HTTP_POST: {
|
|
bool is_https = (protocol == HTTPS_POST ||
|
|
protocol == HTTPS_GET);
|
|
bool is_post = (protocol == HTTPS_POST ||
|
|
protocol == HTTP_POST);
|
|
char req_url[256];
|
|
sockaddr_to_url(&sockaddr_remote, is_https, req_url,
|
|
sizeof(req_url), DEFAULT_DOH_PATH);
|
|
if (is_https) {
|
|
isc_tlsctx_createclient(&tls_ctx);
|
|
}
|
|
isc_nm_httpconnect(netmgr, (isc_nmiface_t *)&sockaddr_local,
|
|
(isc_nmiface_t *)&sockaddr_remote, req_url,
|
|
is_post, connect_cb, NULL, tls_ctx, timeout,
|
|
0);
|
|
} break;
|
|
default:
|
|
INSIST(0);
|
|
ISC_UNREACHABLE();
|
|
}
|
|
|
|
waitforsignal();
|
|
}
|
|
|
|
int
|
|
main(int argc, char **argv) {
|
|
parse_options(argc, argv);
|
|
|
|
setup();
|
|
|
|
run();
|
|
|
|
teardown();
|
|
|
|
exit(EXIT_SUCCESS);
|
|
}
|