mirror of
https://github.com/isc-projects/bind9.git
synced 2026-02-24 02:10:30 -05:00
as there is no further use of isc_task in BIND, this commit removes it, along with isc_taskmgr, isc_event, and all other related types. functions that accepted taskmgr as a parameter have been cleaned up. as a result of this change, some functions can no longer fail, so they've been changed to type void, and their callers have been updated accordingly. the tasks table has been removed from the statistics channel and the stats version has been updated. dns_dyndbctx has been changed to reference the loopmgr instead of taskmgr, and DNS_DYNDB_VERSION has been udpated as well.
305 lines
6.7 KiB
C
305 lines
6.7 KiB
C
/*
|
|
* Copyright (C) Internet Systems Consortium, Inc. ("ISC")
|
|
*
|
|
* SPDX-License-Identifier: MPL-2.0
|
|
*
|
|
* 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 <getopt.h>
|
|
#include <netinet/in.h>
|
|
#include <signal.h>
|
|
#include <stdbool.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <strings.h>
|
|
|
|
#include <isc/managers.h>
|
|
#include <isc/mem.h>
|
|
#include <isc/netaddr.h>
|
|
#include <isc/netmgr.h>
|
|
#include <isc/os.h>
|
|
#include <isc/sockaddr.h>
|
|
#include <isc/string.h>
|
|
#include <isc/util.h>
|
|
|
|
typedef enum { UDP, TCP, DOT, HTTPS, HTTP } protocol_t;
|
|
|
|
static const char *protocols[] = { "udp", "tcp", "dot", "https", "http-plain" };
|
|
|
|
static isc_mem_t *mctx = NULL;
|
|
static isc_loopmgr_t *loopmgr = NULL;
|
|
static isc_nm_t *netmgr = NULL;
|
|
|
|
static protocol_t protocol;
|
|
static in_port_t port;
|
|
static isc_netaddr_t netaddr;
|
|
static isc_sockaddr_t sockaddr __attribute__((unused));
|
|
static int workers;
|
|
|
|
static isc_tlsctx_t *tls_ctx = NULL;
|
|
|
|
static void
|
|
read_cb(isc_nmhandle_t *handle, isc_result_t eresult, isc_region_t *region,
|
|
void *cbarg);
|
|
static void
|
|
send_cb(isc_nmhandle_t *handle, isc_result_t eresult, void *cbarg);
|
|
|
|
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 = (in_port_t)val;
|
|
|
|
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) {
|
|
isc_netaddr_fromin6(&netaddr, &in6);
|
|
return (ISC_R_SUCCESS);
|
|
}
|
|
|
|
if (inet_pton(AF_INET, input, &in) == 1) {
|
|
isc_netaddr_fromin(&netaddr, &in);
|
|
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 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("::1") == 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' },
|
|
{ 0, 0, NULL, 0 }
|
|
};
|
|
|
|
c = getopt_long(argc, argv, "a:p:P:w:", 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;
|
|
|
|
default:
|
|
UNREACHABLE();
|
|
}
|
|
}
|
|
|
|
isc_sockaddr_fromnetaddr(&sockaddr, &netaddr, port);
|
|
|
|
isc_sockaddr_format(&sockaddr, buf, sizeof(buf));
|
|
|
|
printf("Will listen at %s://%s, %d workers\n", protocols[protocol], buf,
|
|
workers);
|
|
}
|
|
|
|
static void
|
|
setup(void) {
|
|
isc_managers_create(&mctx, workers, &loopmgr, &netmgr);
|
|
}
|
|
|
|
static void
|
|
teardown(void) {
|
|
if (tls_ctx) {
|
|
isc_tlsctx_free(&tls_ctx);
|
|
}
|
|
|
|
isc_managers_destroy(&mctx, &loopmgr, &netmgr);
|
|
}
|
|
|
|
static void
|
|
test_server_yield(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_region_t *reply = NULL;
|
|
|
|
REQUIRE(handle != NULL);
|
|
REQUIRE(eresult == ISC_R_SUCCESS);
|
|
UNUSED(cbarg);
|
|
|
|
fprintf(stderr, "RECEIVED %u bytes\n", region->length);
|
|
|
|
if (region->length >= 12) {
|
|
/* long enough to be a DNS header, set QR bit */
|
|
((uint8_t *)region->base)[2] ^= 0x80;
|
|
}
|
|
|
|
reply = isc_mem_get(mctx, sizeof(isc_region_t) + region->length);
|
|
reply->length = region->length;
|
|
reply->base = (uint8_t *)reply + sizeof(isc_region_t);
|
|
memmove(reply->base, region->base, region->length);
|
|
|
|
isc_nm_send(handle, reply, send_cb, reply);
|
|
return;
|
|
}
|
|
|
|
static void
|
|
send_cb(isc_nmhandle_t *handle, isc_result_t eresult, void *cbarg) {
|
|
isc_region_t *reply = cbarg;
|
|
|
|
REQUIRE(handle != NULL);
|
|
REQUIRE(eresult == ISC_R_SUCCESS);
|
|
|
|
isc_mem_put(mctx, cbarg, sizeof(isc_region_t) + reply->length);
|
|
}
|
|
|
|
static isc_result_t
|
|
accept_cb(isc_nmhandle_t *handle, isc_result_t eresult, void *cbarg) {
|
|
REQUIRE(handle != NULL);
|
|
REQUIRE(eresult == ISC_R_SUCCESS);
|
|
UNUSED(cbarg);
|
|
|
|
return (ISC_R_SUCCESS);
|
|
}
|
|
|
|
static void
|
|
run(void) {
|
|
isc_result_t result;
|
|
isc_nmsocket_t *sock = NULL;
|
|
|
|
switch (protocol) {
|
|
case UDP:
|
|
result = isc_nm_listenudp(netmgr, ISC_NM_LISTEN_ALL, &sockaddr,
|
|
read_cb, NULL, &sock);
|
|
break;
|
|
case TCP:
|
|
result = isc_nm_listenstreamdns(
|
|
netmgr, ISC_NM_LISTEN_ALL, &sockaddr, read_cb, NULL,
|
|
accept_cb, NULL, 0, NULL, NULL, &sock);
|
|
break;
|
|
case DOT: {
|
|
isc_tlsctx_createserver(NULL, NULL, &tls_ctx);
|
|
|
|
result = isc_nm_listenstreamdns(
|
|
netmgr, ISC_NM_LISTEN_ALL, &sockaddr, read_cb, NULL,
|
|
accept_cb, NULL, 0, NULL, tls_ctx, &sock);
|
|
break;
|
|
}
|
|
#if HAVE_LIBNGHTTP2
|
|
case HTTPS:
|
|
case HTTP: {
|
|
bool is_https = protocol == HTTPS;
|
|
isc_nm_http_endpoints_t *eps = NULL;
|
|
if (is_https) {
|
|
isc_tlsctx_createserver(NULL, NULL, &tls_ctx);
|
|
}
|
|
eps = isc_nm_http_endpoints_new(mctx);
|
|
result = isc_nm_http_endpoints_add(
|
|
eps, ISC_NM_HTTP_DEFAULT_PATH, read_cb, NULL);
|
|
|
|
if (result == ISC_R_SUCCESS) {
|
|
result = isc_nm_listenhttp(netmgr, ISC_NM_LISTEN_ALL,
|
|
&sockaddr, 0, NULL, tls_ctx,
|
|
eps, 0, &sock);
|
|
}
|
|
isc_nm_http_endpoints_detach(&eps);
|
|
} break;
|
|
#endif
|
|
default:
|
|
UNREACHABLE();
|
|
}
|
|
REQUIRE(result == ISC_R_SUCCESS);
|
|
|
|
test_server_yield();
|
|
|
|
isc_nm_stoplistening(sock);
|
|
isc_nmsocket_close(&sock);
|
|
}
|
|
|
|
int
|
|
main(int argc, char **argv) {
|
|
parse_options(argc, argv);
|
|
|
|
setup();
|
|
|
|
run();
|
|
|
|
teardown();
|
|
|
|
exit(EXIT_SUCCESS);
|
|
}
|