2
0
mirror of https://gitlab.isc.org/isc-projects/bind9 synced 2025-08-22 18:19:42 +00:00
bind/bin/named/server.c

462 lines
12 KiB
C
Raw Normal View History

/*
1999-03-06 03:55:54 +00:00
* Copyright (C) 1999 Internet Software Consortium.
*
* Permission to use, copy, modify, and distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
*
* THE SOFTWARE IS PROVIDED "AS IS" AND INTERNET SOFTWARE CONSORTIUM DISCLAIMS
* ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL INTERNET SOFTWARE
* CONSORTIUM BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
* DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
* PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS
* ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
* SOFTWARE.
*/
#include <config.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <isc/assertions.h>
#include <isc/error.h>
#include <isc/mem.h>
#include <isc/task.h>
#include <isc/thread.h>
#include <isc/result.h>
#include <isc/socket.h>
#include <isc/timer.h>
#include <dns/types.h>
#include <dns/result.h>
#include <dns/name.h>
#include <dns/rdata.h>
#include <dns/rdatalist.h>
#include <dns/rdataset.h>
#include <dns/compress.h>
1999-01-31 12:31:31 +00:00
#include <dns/db.h>
#include <dns/message.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include "confparser.h"
#include "udpclient.h"
1999-01-28 05:52:20 +00:00
#include "tcpclient.h"
isc_mem_t *mctx = NULL;
1999-02-01 21:11:10 +00:00
isc_boolean_t want_stats = ISC_FALSE;
1999-01-31 12:31:31 +00:00
dns_db_t *db;
1999-05-02 07:03:13 +00:00
static inline isc_boolean_t
CHECKRESULT(dns_result_t result, char *msg)
{
if ((result) != DNS_R_SUCCESS) {
printf("%s: %s\n", (msg), dns_result_totext(result));
return (ISC_TRUE);
}
return (ISC_FALSE);
}
1999-01-31 12:31:31 +00:00
static void
makename(isc_mem_t *mctx, char *text, dns_name_t *name, dns_name_t *origin) {
char b[255];
isc_buffer_t source, target;
size_t len;
isc_region_t r1, r2;
dns_result_t result;
if (origin == NULL)
origin = dns_rootname;
dns_name_init(name, NULL);
len = strlen(text);
isc_buffer_init(&source, text, len, ISC_BUFFERTYPE_TEXT);
isc_buffer_add(&source, len);
isc_buffer_init(&target, b, sizeof b, ISC_BUFFERTYPE_BINARY);
result = dns_name_fromtext(name, &source, origin, ISC_FALSE, &target);
RUNTIME_CHECK(result == DNS_R_SUCCESS);
dns_name_toregion(name, &r1);
r2.base = isc_mem_get(mctx, r1.length);
RUNTIME_CHECK(r2.base != NULL);
r2.length = r1.length;
memcpy(r2.base, r1.base, r1.length);
dns_name_fromregion(name, &r2);
}
/*
* This is in bin/tests/wire_test.c, but should be in a debugging library.
*/
extern dns_result_t
printmessage(dns_message_t *);
dns_result_t
resolve_packet(isc_mem_t *mctx, dns_db_t *, dns_message_t *, isc_buffer_t *);
dns_result_t
resolve_packet(isc_mem_t *mctx, dns_db_t *db, dns_message_t *query,
isc_buffer_t *target)
{
dns_message_t *message;
dns_result_t result;
1999-05-02 07:00:05 +00:00
dns_name_t *name, name2;
dns_rdataset_t *rds, rdataset;
dns_dbnode_t *node;
result = dns_message_create(mctx, &message, DNS_MESSAGE_INTENT_RENDER);
CHECKRESULT(result, "dns_message_create failed");
message->id = query->id;
1999-05-02 07:00:05 +00:00
message->rcode = dns_rcode_noerror;
message->flags = query->flags;
message->flags |= DNS_MESSAGEFLAG_QR;
/*
* XXX This is a total and disgusting hack. We need a way to add
* a copy of a rdataset and a name to the new message, but for now
* I'll just steal the one from the existing query message, and
* make certain the query is not destroyed before our message is.
*/
result = dns_message_firstname(query, DNS_SECTION_QUESTION);
if (result != DNS_R_SUCCESS)
return (result);
1999-05-02 07:00:05 +00:00
name = NULL;
dns_message_currentname(query, DNS_SECTION_QUESTION, &name);
1999-05-02 07:00:05 +00:00
rds = ISC_LIST_HEAD(name->list);
if (rds == NULL)
return (DNS_R_UNEXPECTED);
dns_message_addname(message, name, DNS_SECTION_QUESTION);
result = printmessage(message);
INSIST(result == DNS_R_SUCCESS); /* XXX not in a real server */
1999-05-02 07:00:05 +00:00
/*
* Pull the name out of the database.
*/
if (!dns_name_issubdomain(name, dns_db_origin(db))) {
message->rcode = dns_rcode_nxdomain;
goto render;
}
dns_name_init(&name2, NULL);
dns_name_clone(name, &name2);
node = NULL;
result = dns_db_findnode(db, &name2, ISC_FALSE, &node);
if (result == DNS_R_NOTFOUND) {
message->flags |= DNS_MESSAGEFLAG_AA;
message->rcode = dns_rcode_nxdomain;
goto render;
}
if (result != DNS_R_SUCCESS) {
message->rcode = dns_rcode_servfail;
printf("Foo\n");
goto render;
}
dns_rdataset_init(&rdataset);
result = dns_db_findrdataset(db, node, NULL, rds->type, 0, &rdataset);
dns_db_detachnode(db, &node);
if (result == DNS_R_NOTFOUND) {
message->flags |= DNS_MESSAGEFLAG_AA;
message->rcode = dns_rcode_nxdomain;
goto render;
}
if (result != DNS_R_SUCCESS) {
message->rcode = dns_rcode_servfail;
printf("Foo2\n");
goto render;
}
ISC_LIST_APPEND(name2.list, &rdataset, link);
dns_message_addname(message, &name2, DNS_SECTION_ANSWER);
message->flags |= DNS_MESSAGEFLAG_AA;
render:
result = dns_message_renderbegin(message, target);
if (result != DNS_R_SUCCESS)
return (result);
result = dns_message_rendersection(message, DNS_SECTION_QUESTION,
0, 0);
if (result != DNS_R_SUCCESS)
return (result);
result = dns_message_rendersection(message, DNS_SECTION_ANSWER,
0, 0);
if (result != DNS_R_SUCCESS)
return (result);
result = dns_message_rendersection(message, DNS_SECTION_AUTHORITY,
0, 0);
if (result != DNS_R_SUCCESS)
return (result);
result = dns_message_rendersection(message, DNS_SECTION_ADDITIONAL,
0, 0);
if (result != DNS_R_SUCCESS)
return (result);
result = dns_message_rendersection(message, DNS_SECTION_OPT,
0, 0);
if (result != DNS_R_SUCCESS)
return (result);
result = dns_message_rendersection(message, DNS_SECTION_TSIG,
0, 0);
if (result != DNS_R_SUCCESS)
return (result);
result = dns_message_renderend(message);
dns_message_destroy(&message);
return (DNS_R_SUCCESS);
}
/*
* Process the wire format message given in r, and return a new packet to
* transmit.
*
* Return of DNS_R_SUCCESS means r->base is a newly allocated region of
* memory, and r->length is its length. The actual for-transmit packet
* begins at (r->length + reslen) to reserve (reslen) bytes at the front
* of the packet for transmission specific details.
*/
static dns_result_t
dispatch(isc_mem_t *mctx, isc_region_t *rxr, unsigned int reslen)
{
char t[512];
1999-01-31 12:31:31 +00:00
isc_buffer_t source;
isc_buffer_t target;
dns_result_t result;
isc_region_t txr;
dns_message_t *message;
1999-01-31 12:31:31 +00:00
/*
* Set up the input buffer from the contents of the region passed
* to us.
*/
isc_buffer_init(&source, rxr->base, rxr->length,
ISC_BUFFERTYPE_BINARY);
isc_buffer_add(&source, rxr->length);
result = dns_message_create(mctx, &message, DNS_MESSAGE_INTENT_PARSE);
if (CHECKRESULT(result, "dns_message_create failed")) {
return (result);
}
result = dns_message_parse(message, &source);
if (CHECKRESULT(result, "dns_message_parsed failed")) {
dns_message_destroy(&message);
return (result);
}
CHECKRESULT(result, "dns_message_parse failed");
result = printmessage(message);
if (CHECKRESULT(result, "printmessage failed")) {
dns_message_destroy(&message);
return (result);
}
isc_buffer_init(&target, t, sizeof(t), ISC_BUFFERTYPE_BINARY);
result = resolve_packet(mctx, db, message, &target);
if (result != DNS_R_SUCCESS) {
dns_message_destroy(&message);
1999-01-31 12:31:31 +00:00
return (result);
}
/*
* Copy the reply out, adjusting for reslen
*/
1999-01-31 12:31:31 +00:00
isc_buffer_used(&target, &txr);
txr.base = isc_mem_get(mctx, txr.length + reslen);
if (txr.base == NULL) {
dns_message_destroy(&message);
return (DNS_R_NOMEMORY);
}
memcpy(txr.base + reslen, t + reslen, txr.length);
rxr->base = txr.base;
rxr->length = txr.length + reslen;
1999-01-31 12:31:31 +00:00
printf("Base == %p, length == %u\n", txr.base, txr.length);
fflush(stdout);
1999-02-01 21:11:10 +00:00
if (want_stats)
isc_mem_stats(mctx, stdout);
dns_message_destroy(&message);
return (DNS_R_SUCCESS);
}
int
main(int argc, char *argv[])
{
isc_taskmgr_t *manager = NULL;
unsigned int workers;
isc_socketmgr_t *socketmgr;
1999-01-28 05:52:20 +00:00
isc_socket_t *so0, *so1;
isc_sockaddr_t sockaddr;
unsigned int addrlen;
1999-01-28 05:52:20 +00:00
udp_listener_t *ludp;
tcp_listener_t *ltcp;
1999-01-31 12:31:31 +00:00
dns_name_t base, *origin;
int ch;
char basetext[1000];
dns_rdatatype_t type = 2;
dns_result_t result;
#if 0 /* brister */
isc_cfgctx_t *configctx = NULL;
const char *conffile = "/etc/named.conf"; /* XXX hardwired */
#endif
1999-01-31 12:31:31 +00:00
/*+ XXX */
strcpy(basetext, "");
1999-02-01 21:11:10 +00:00
while ((ch = getopt(argc, argv, "z:t:s")) != -1) {
1999-01-31 12:31:31 +00:00
switch (ch) {
case 'z':
strcpy(basetext, optarg);
break;
case 't':
type = atoi(optarg);
break;
1999-02-01 21:11:10 +00:00
case 's':
want_stats = ISC_TRUE;
break;
1999-01-31 12:31:31 +00:00
}
}
argc -= optind;
argv += optind;
if (argc < 1) {
fprintf(stderr, "usage: named filename\n");
exit(1);
}
/*- XXX */
memset(&sockaddr, 0, sizeof(sockaddr));
sockaddr.type.sin.sin_port = htons(5544);
addrlen = sizeof(struct sockaddr_in);
1999-01-31 12:31:31 +00:00
workers = 2;
printf("%d workers\n", workers);
RUNTIME_CHECK(isc_mem_create(0, 0, &mctx) == ISC_R_SUCCESS);
#if 0 /* brister */
parser_init();
RUNTIME_CHECK(parse_configuration(conffile, mctx, &configctx) ==
ISC_R_SUCCESS);
#endif
1999-01-31 12:31:31 +00:00
/*+ XXX */
if (strcmp(basetext, "") == 0)
strcpy(basetext, "vix.com.");
makename(mctx, basetext, &base, NULL);
db = NULL;
result = dns_db_create(mctx, "rbt", &base, ISC_FALSE, 1, 0, NULL,
&db);
RUNTIME_CHECK(result == DNS_R_SUCCESS);
1999-01-31 12:31:31 +00:00
origin = &base;
printf("loading %s\n", argv[0]);
result = dns_db_load(db, argv[0]);
if (result != DNS_R_SUCCESS) {
printf("couldn't load master file: %s\n",
dns_result_totext(result));
exit(1);
}
/*- XXX */
RUNTIME_CHECK(isc_taskmgr_create(mctx, workers, 0, &manager) ==
ISC_R_SUCCESS);
1999-01-31 12:31:31 +00:00
/*
* Open up a database.
*/
socketmgr = NULL;
RUNTIME_CHECK(isc_socketmgr_create(mctx, &socketmgr) == ISC_R_SUCCESS);
/*
* open up a UDP socket
*/
1999-01-28 05:52:20 +00:00
so0 = NULL;
memset(&sockaddr, 0, sizeof(sockaddr));
sockaddr.type.sin.sin_family = AF_INET;
sockaddr.type.sin.sin_port = htons(5544);
addrlen = sizeof(struct sockaddr_in);
RUNTIME_CHECK(isc_socket_create(socketmgr, isc_socket_udp, &so0) ==
ISC_R_SUCCESS);
RUNTIME_CHECK(isc_socket_bind(so0, &sockaddr,
(int)addrlen) == ISC_R_SUCCESS);
ludp = udp_listener_allocate(mctx, workers);
RUNTIME_CHECK(udp_listener_start(ludp, so0, manager,
workers, workers, 0,
dispatch) == ISC_R_SUCCESS);
1999-01-28 05:52:20 +00:00
1999-02-01 21:11:10 +00:00
if (want_stats)
isc_mem_stats(mctx, stdout);
1999-01-28 05:52:20 +00:00
/*
* open up a TCP socket
*/
so1 = NULL;
memset(&sockaddr, 0, sizeof(sockaddr));
sockaddr.type.sin.sin_family = AF_INET;
sockaddr.type.sin.sin_port = htons(5544);
addrlen = sizeof(struct sockaddr_in);
1999-01-28 05:52:20 +00:00
RUNTIME_CHECK(isc_socket_create(socketmgr, isc_socket_tcp, &so1) ==
ISC_R_SUCCESS);
RUNTIME_CHECK(isc_socket_bind(so1, &sockaddr,
(int)addrlen) == ISC_R_SUCCESS);
1999-01-28 05:52:20 +00:00
ltcp = tcp_listener_allocate(mctx, workers);
RUNTIME_CHECK(tcp_listener_start(ltcp, so1, manager,
workers, workers, 0,
dispatch) == ISC_R_SUCCESS);
1999-02-01 21:11:10 +00:00
if (want_stats)
isc_mem_stats(mctx, stdout);
1999-02-11 06:38:12 +00:00
/*
* XXX Need to set up a condition variable here, and wait on it.
* For now, just semi-busy loop.
*/
for (;;)
sleep(10);
printf("Destroying socket manager\n");
isc_socketmgr_destroy(&socketmgr);
printf("Destroying task manager\n");
isc_taskmgr_destroy(&manager);
1999-02-01 21:11:10 +00:00
if (want_stats)
isc_mem_stats(mctx, stdout);
isc_mem_destroy(&mctx);
return (0);
}