]> git.ipfire.org Git - thirdparty/bind9.git/commitdiff
use netmgr for xfrin
authorEvan Hunt <each@isc.org>
Sun, 6 Sep 2020 07:38:50 +0000 (00:38 -0700)
committerOndřej Surý <ondrej@sury.org>
Mon, 9 Nov 2020 12:45:43 +0000 (13:45 +0100)
Use isc_nm_tcpdnsconnect() in xfrin.c for zone transfers.

bin/named/server.c
bin/tests/system/xfer/tests.sh
lib/dns/include/dns/xfrin.h
lib/dns/include/dns/zone.h
lib/dns/tests/dnstest.c
lib/dns/tests/zonemgr_test.c
lib/dns/xfrin.c
lib/dns/zone.c
lib/ns/tests/nstest.c

index db84a2c164ca9f14278e04c311fc9e7cb0d22eb2..be731815bac4049bac76adef2d8a94557c86b586 100644 (file)
@@ -10050,7 +10050,7 @@ named_server_create(isc_mem_t *mctx, named_server_t **serverp) {
 
        CHECKFATAL(dns_zonemgr_create(named_g_mctx, named_g_taskmgr,
                                      named_g_timermgr, named_g_socketmgr,
-                                     &server->zonemgr),
+                                     named_g_nm, &server->zonemgr),
                   "dns_zonemgr_create");
        CHECKFATAL(dns_zonemgr_setsize(server->zonemgr, 1000), "dns_zonemgr_"
                                                               "setsize");
index e9e9125f3fb0856e324a8b9745afb95ffe606978..ec98194451f892187e55c77b7838ab0e7ac8bc76 100755 (executable)
@@ -173,7 +173,7 @@ grep "^;" dig.out.ns6.test$n | cat_i
 
 $DIG $DIGOPTS primary. \
        @10.53.0.3 axfr > dig.out.ns3.test$n || tmp=1
-grep "^;" dig.out.ns3.test$n > /dev/null && cat_i dig.out.ns3.test$n
+grep "^;" dig.out.ns3.test$n > /dev/null && cat_i dig.out.ns3.test$n
 
 digcomp dig.out.ns6.test$n dig.out.ns3.test$n || tmp=1
 
index da5d7a88c47d84f7414096b113adbb4456079ad5..bc7b7498cec04b57d1aef4409c12a024c91a5d33 100644 (file)
@@ -49,9 +49,8 @@ dns_xfrin_create(dns_zone_t *zone, dns_rdatatype_t xfrtype,
                 const isc_sockaddr_t *masteraddr,
                 const isc_sockaddr_t *sourceaddr, isc_dscp_t dscp,
                 dns_tsigkey_t *tsigkey, isc_mem_t *mctx,
-                isc_timermgr_t *timermgr, isc_socketmgr_t *socketmgr,
-                isc_task_t *task, dns_xfrindone_t done,
-                dns_xfrin_ctx_t **xfrp);
+                isc_timermgr_t *timermgr, isc_nm_t *netmgr, isc_task_t *task,
+                dns_xfrindone_t done, dns_xfrin_ctx_t **xfrp);
 /*%<
  * Attempt to start an incoming zone transfer of 'zone'
  * from 'masteraddr', creating a dns_xfrin_ctx_t object to
index 963aa94831042259eedd77c4c27c671a3b436d70..c8342ca7dbfe66c9415415a6a922dee2a335cf22 100644 (file)
@@ -1630,7 +1630,7 @@ dns_zone_getkeydirectory(dns_zone_t *zone);
 isc_result_t
 dns_zonemgr_create(isc_mem_t *mctx, isc_taskmgr_t *taskmgr,
                   isc_timermgr_t *timermgr, isc_socketmgr_t *socketmgr,
-                  dns_zonemgr_t **zmgrp);
+                  isc_nm_t *netmgr, dns_zonemgr_t **zmgrp);
 /*%<
  * Create a zone manager.  Note: the zone manager will not be able to
  * manage any zones until dns_zonemgr_setsize() has been run.
index 4f31c2ddbe2535ee8b5ce383f065818f010c3811..a913b56a240f442484a286416d8af75e9eefa22c 100644 (file)
@@ -297,7 +297,7 @@ dns_test_setupzonemgr(void) {
        isc_result_t result;
        REQUIRE(zonemgr == NULL);
 
-       result = dns_zonemgr_create(dt_mctx, taskmgr, timermgr, socketmgr,
+       result = dns_zonemgr_create(dt_mctx, taskmgr, timermgr, socketmgr, NULL,
                                    &zonemgr);
        return (result);
 }
index 733d607996209156ef60660951800b57e84f2a21..e508a2a1d1b57b044cdfeb9bfb2b938cce7673da 100644 (file)
@@ -62,7 +62,7 @@ zonemgr_create(void **state) {
 
        UNUSED(state);
 
-       result = dns_zonemgr_create(dt_mctx, taskmgr, timermgr, socketmgr,
+       result = dns_zonemgr_create(dt_mctx, taskmgr, timermgr, socketmgr, NULL,
                                    &myzonemgr);
        assert_int_equal(result, ISC_R_SUCCESS);
 
@@ -80,7 +80,7 @@ zonemgr_managezone(void **state) {
 
        UNUSED(state);
 
-       result = dns_zonemgr_create(dt_mctx, taskmgr, timermgr, socketmgr,
+       result = dns_zonemgr_create(dt_mctx, taskmgr, timermgr, socketmgr, NULL,
                                    &myzonemgr);
        assert_int_equal(result, ISC_R_SUCCESS);
 
@@ -121,7 +121,7 @@ zonemgr_createzone(void **state) {
 
        UNUSED(state);
 
-       result = dns_zonemgr_create(dt_mctx, taskmgr, timermgr, socketmgr,
+       result = dns_zonemgr_create(dt_mctx, taskmgr, timermgr, socketmgr, NULL,
                                    &myzonemgr);
        assert_int_equal(result, ISC_R_SUCCESS);
 
@@ -160,7 +160,7 @@ zonemgr_unreachable(void **state) {
 
        TIME_NOW(&now);
 
-       result = dns_zonemgr_create(dt_mctx, taskmgr, timermgr, socketmgr,
+       result = dns_zonemgr_create(dt_mctx, taskmgr, timermgr, socketmgr, NULL,
                                    &myzonemgr);
        assert_int_equal(result, ISC_R_SUCCESS);
 
index 952d16a481bf7db50d793a6af88d289f77109666..9548ad8b0fc2ddea5e7e297eec1d8d56d9ecc862 100644 (file)
@@ -96,16 +96,18 @@ struct dns_xfrin_ctx {
        isc_mem_t *mctx;
        dns_zone_t *zone;
 
-       int refcount;
+       isc_refcount_t refs;
 
        isc_task_t *task;
        isc_timer_t *timer;
-       isc_socketmgr_t *socketmgr;
+       isc_nm_t *netmgr;
+
+       isc_refcount_t connects; /*%< Connect in progress */
+       isc_refcount_t sends;    /*%< Send in progress */
+       isc_refcount_t recvs;    /*%< Receive in progress */
 
-       int connects; /*%< Connect in progress */
-       int sends;    /*%< Send in progress */
-       int recvs;    /*%< Receive in progress */
        bool shuttingdown;
+
        isc_result_t shutdown_result;
 
        dns_name_t name; /*%< Name of zone to transfer */
@@ -123,16 +125,15 @@ struct dns_xfrin_ctx {
 
        isc_sockaddr_t masteraddr;
        isc_sockaddr_t sourceaddr;
-       isc_socket_t *socket;
+
+       isc_nmhandle_t *handle;
+       isc_nmhandle_t *readhandle;
+       isc_nmhandle_t *sendhandle;
 
        /*% Buffer for IXFR/AXFR request message */
        isc_buffer_t qbuffer;
        unsigned char qbuffer_data[512];
 
-       /*% Incoming reply TCP message */
-       dns_tcpmsg_t tcpmsg;
-       bool tcpmsg_valid;
-
        /*%
         * Whether the zone originally had a database attached at the time this
         * transfer context was created.  Used by maybe_free() when making
@@ -190,11 +191,10 @@ struct dns_xfrin_ctx {
 
 static isc_result_t
 xfrin_create(isc_mem_t *mctx, dns_zone_t *zone, dns_db_t *db, isc_task_t *task,
-            isc_timermgr_t *timermgr, isc_socketmgr_t *socketmgr,
-            dns_name_t *zonename, dns_rdataclass_t rdclass,
-            dns_rdatatype_t reqtype, const isc_sockaddr_t *masteraddr,
-            const isc_sockaddr_t *sourceaddr, isc_dscp_t dscp,
-            dns_tsigkey_t *tsigkey, dns_xfrin_ctx_t **xfrp);
+            isc_timermgr_t *timermgr, isc_nm_t *netmgr, dns_name_t *zonename,
+            dns_rdataclass_t rdclass, dns_rdatatype_t reqtype,
+            const isc_sockaddr_t *masteraddr, const isc_sockaddr_t *sourceaddr,
+            isc_dscp_t dscp, dns_tsigkey_t *tsigkey, dns_xfrin_ctx_t **xfrp);
 
 static isc_result_t
 axfr_init(dns_xfrin_ctx_t *xfr);
@@ -228,13 +228,14 @@ static isc_result_t
 xfrin_start(dns_xfrin_ctx_t *xfr);
 
 static void
-xfrin_connect_done(isc_task_t *task, isc_event_t *event);
+xfrin_connect_done(isc_nmhandle_t *handle, isc_result_t result, void *cbarg);
 static isc_result_t
 xfrin_send_request(dns_xfrin_ctx_t *xfr);
 static void
-xfrin_send_done(isc_task_t *task, isc_event_t *event);
+xfrin_send_done(isc_nmhandle_t *handle, isc_result_t result, void *cbarg);
 static void
-xfrin_recv_done(isc_task_t *task, isc_event_t *event);
+xfrin_recv_done(isc_nmhandle_t *handle, isc_result_t result,
+               isc_region_t *region, void *cbarg);
 static void
 xfrin_timeout(isc_task_t *task, isc_event_t *event);
 
@@ -375,7 +376,7 @@ failure:
 static isc_result_t
 ixfr_init(dns_xfrin_ctx_t *xfr) {
        isc_result_t result;
-       char *journalfile;
+       char *journalfile = NULL;
 
        if (xfr->reqtype != dns_rdatatype_ixfr) {
                xfrin_log(xfr, ISC_LOG_ERROR,
@@ -658,15 +659,15 @@ dns_xfrin_create(dns_zone_t *zone, dns_rdatatype_t xfrtype,
                 const isc_sockaddr_t *masteraddr,
                 const isc_sockaddr_t *sourceaddr, isc_dscp_t dscp,
                 dns_tsigkey_t *tsigkey, isc_mem_t *mctx,
-                isc_timermgr_t *timermgr, isc_socketmgr_t *socketmgr,
-                isc_task_t *task, dns_xfrindone_t done,
-                dns_xfrin_ctx_t **xfrp) {
+                isc_timermgr_t *timermgr, isc_nm_t *netmgr, isc_task_t *task,
+                dns_xfrindone_t done, dns_xfrin_ctx_t **xfrp) {
        dns_name_t *zonename = dns_zone_getorigin(zone);
        dns_xfrin_ctx_t *xfr = NULL;
        isc_result_t result;
        dns_db_t *db = NULL;
 
        REQUIRE(xfrp != NULL && *xfrp == NULL);
+       REQUIRE(done != NULL);
 
        (void)dns_zone_getdb(zone, &db);
 
@@ -674,7 +675,7 @@ dns_xfrin_create(dns_zone_t *zone, dns_rdatatype_t xfrtype,
                REQUIRE(db != NULL);
        }
 
-       CHECK(xfrin_create(mctx, zone, db, task, timermgr, socketmgr, zonename,
+       CHECK(xfrin_create(mctx, zone, db, task, timermgr, netmgr, zonename,
                           dns_zone_getclass(zone), xfrtype, masteraddr,
                           sourceaddr, dscp, tsigkey, &xfr));
 
@@ -682,14 +683,32 @@ dns_xfrin_create(dns_zone_t *zone, dns_rdatatype_t xfrtype,
                xfr->zone_had_db = true;
        }
 
-       CHECK(xfrin_start(xfr));
-
        xfr->done = done;
-       if (xfr->done != NULL) {
-               xfr->refcount++;
-       }
+
+       /*
+        * Initialize the reference counter to 2 not 1, as we need to
+        * be detached separately in both xfrin_recv_done() and the
+        * 'done' callback.
+        */
+       isc_refcount_init(&xfr->refs, 2);
+
+       /*
+        * Set *xfrp now, before calling xfrin_start(). Asynchronous
+        * netmgr processing could cause the 'done' callback to run in
+        * another thread before we reached the end of the present
+        * function. In that case, if *xfrp hadn't already been
+        * attached, the 'done' function would be unable to detach it.
+        */
        *xfrp = xfr;
 
+       result = xfrin_start(xfr);
+       if (result != ISC_R_SUCCESS) {
+               xfr->shuttingdown = true;
+               xfr->shutdown_result = result;
+               isc_refcount_decrement(&xfr->refs);
+               dns_xfrin_detach(xfrp);
+       }
+
 failure:
        if (db != NULL) {
                dns_db_detach(&db);
@@ -700,11 +719,13 @@ failure:
                xfrin_log1(ISC_LOG_ERROR, zonetext, masteraddr,
                           "zone transfer setup failed");
        }
+
        return (result);
 }
 
 void
 dns_xfrin_shutdown(dns_xfrin_ctx_t *xfr) {
+       REQUIRE(VALID_XFRIN(xfr));
        if (!xfr->shuttingdown) {
                xfrin_fail(xfr, ISC_R_CANCELED, "shut down");
        }
@@ -712,30 +733,33 @@ dns_xfrin_shutdown(dns_xfrin_ctx_t *xfr) {
 
 void
 dns_xfrin_attach(dns_xfrin_ctx_t *source, dns_xfrin_ctx_t **target) {
+       REQUIRE(VALID_XFRIN(source));
        REQUIRE(target != NULL && *target == NULL);
-       source->refcount++;
+       isc_refcount_increment(&source->refs);
        *target = source;
 }
 
 void
 dns_xfrin_detach(dns_xfrin_ctx_t **xfrp) {
-       dns_xfrin_ctx_t *xfr = *xfrp;
+       dns_xfrin_ctx_t *xfr = NULL;
+
+       REQUIRE(xfrp != NULL && VALID_XFRIN(*xfrp));
+
+       xfr = *xfrp;
        *xfrp = NULL;
-       INSIST(xfr->refcount > 0);
-       xfr->refcount--;
-       maybe_free(xfr);
+
+       if (isc_refcount_decrement(&xfr->refs) == 1) {
+               maybe_free(xfr);
+       }
 }
 
 static void
 xfrin_cancelio(dns_xfrin_ctx_t *xfr) {
-       if (xfr->connects > 0) {
-               isc_socket_cancel(xfr->socket, xfr->task,
-                                 ISC_SOCKCANCEL_CONNECT);
-       } else if (xfr->recvs > 0) {
-               dns_tcpmsg_cancelread(&xfr->tcpmsg);
-       } else if (xfr->sends > 0) {
-               isc_socket_cancel(xfr->socket, xfr->task, ISC_SOCKCANCEL_SEND);
+       if (xfr->readhandle == NULL) {
+               return;
        }
+       isc_nm_cancelread(xfr->readhandle);
+       isc_nmhandle_detach(&xfr->readhandle);
 }
 
 static void
@@ -746,8 +770,11 @@ xfrin_reset(dns_xfrin_ctx_t *xfr) {
 
        xfrin_cancelio(xfr);
 
-       if (xfr->socket != NULL) {
-               isc_socket_detach(&xfr->socket);
+       if (xfr->readhandle != NULL) {
+               isc_nmhandle_detach(&xfr->readhandle);
+       }
+       if (xfr->sendhandle != NULL) {
+               isc_nmhandle_detach(&xfr->sendhandle);
        }
 
        if (xfr->lasttsig != NULL) {
@@ -765,11 +792,6 @@ xfrin_reset(dns_xfrin_ctx_t *xfr) {
                (void)dns_db_endload(xfr->db, &xfr->axfr);
        }
 
-       if (xfr->tcpmsg_valid) {
-               dns_tcpmsg_invalidate(&xfr->tcpmsg);
-               xfr->tcpmsg_valid = false;
-       }
-
        if (xfr->ver != NULL) {
                dns_db_closeversion(xfr->db, &xfr->ver, false);
        }
@@ -803,82 +825,46 @@ xfrin_fail(dns_xfrin_ctx_t *xfr, isc_result_t result, const char *msg) {
 
 static isc_result_t
 xfrin_create(isc_mem_t *mctx, dns_zone_t *zone, dns_db_t *db, isc_task_t *task,
-            isc_timermgr_t *timermgr, isc_socketmgr_t *socketmgr,
-            dns_name_t *zonename, dns_rdataclass_t rdclass,
-            dns_rdatatype_t reqtype, const isc_sockaddr_t *masteraddr,
-            const isc_sockaddr_t *sourceaddr, isc_dscp_t dscp,
-            dns_tsigkey_t *tsigkey, dns_xfrin_ctx_t **xfrp) {
+            isc_timermgr_t *timermgr, isc_nm_t *netmgr, dns_name_t *zonename,
+            dns_rdataclass_t rdclass, dns_rdatatype_t reqtype,
+            const isc_sockaddr_t *masteraddr, const isc_sockaddr_t *sourceaddr,
+            isc_dscp_t dscp, dns_tsigkey_t *tsigkey, dns_xfrin_ctx_t **xfrp) {
        dns_xfrin_ctx_t *xfr = NULL;
        isc_result_t result;
 
        xfr = isc_mem_get(mctx, sizeof(*xfr));
-       xfr->mctx = NULL;
+       *xfr = (dns_xfrin_ctx_t){ .netmgr = netmgr,
+                                 .shutdown_result = ISC_R_UNSET,
+                                 .rdclass = rdclass,
+                                 .reqtype = reqtype,
+                                 .dscp = dscp,
+                                 .id = (dns_messageid_t)isc_random16(),
+                                 .maxrecords = dns_zone_getmaxrecords(zone),
+                                 .masteraddr = *masteraddr,
+                                 .sourceaddr = *sourceaddr };
+
        isc_mem_attach(mctx, &xfr->mctx);
-       xfr->refcount = 0;
-       xfr->zone = NULL;
        dns_zone_iattach(zone, &xfr->zone);
-       xfr->task = NULL;
        isc_task_attach(task, &xfr->task);
-       xfr->timer = NULL;
-       xfr->socketmgr = socketmgr;
-       xfr->done = NULL;
-
-       xfr->connects = 0;
-       xfr->sends = 0;
-       xfr->recvs = 0;
-       xfr->shuttingdown = false;
-       xfr->shutdown_result = ISC_R_UNSET;
-
        dns_name_init(&xfr->name, NULL);
-       xfr->rdclass = rdclass;
-       xfr->id = (dns_messageid_t)isc_random16();
-       xfr->reqtype = reqtype;
-       xfr->dscp = dscp;
-
-       /* sockaddr */
-       xfr->socket = NULL;
-       /* qbuffer */
-       /* qbuffer_data */
-       /* tcpmsg */
-       xfr->tcpmsg_valid = false;
-
-       xfr->zone_had_db = false;
-       xfr->db = NULL;
+
        if (db != NULL) {
                dns_db_attach(db, &xfr->db);
        }
-       xfr->ver = NULL;
+
        dns_diff_init(xfr->mctx, &xfr->diff);
-       xfr->difflen = 0;
 
        if (reqtype == dns_rdatatype_soa) {
                xfr->state = XFRST_SOAQUERY;
        } else {
                xfr->state = XFRST_INITIALSOA;
        }
-       /* end_serial */
 
-       xfr->nmsg = 0;
-       xfr->nrecs = 0;
-       xfr->nbytes = 0;
-       xfr->maxrecords = dns_zone_getmaxrecords(zone);
        isc_time_now(&xfr->start);
 
-       xfr->tsigkey = NULL;
        if (tsigkey != NULL) {
                dns_tsigkey_attach(tsigkey, &xfr->tsigkey);
        }
-       xfr->lasttsig = NULL;
-       xfr->tsigctx = NULL;
-       xfr->sincetsig = 0;
-       xfr->is_ixfr = false;
-
-       /* ixfr.request_serial */
-       /* ixfr.current_serial */
-       xfr->ixfr.journal = NULL;
-
-       xfr->axfr.add = NULL;
-       xfr->axfr.add_private = NULL;
 
        dns_name_dup(zonename, mctx, &xfr->name);
 
@@ -887,10 +873,7 @@ xfrin_create(isc_mem_t *mctx, dns_zone_t *zone, dns_db_t *db, isc_task_t *task,
        CHECK(dns_timer_setidle(xfr->timer, dns_zone_getmaxxfrin(xfr->zone),
                                dns_zone_getidlein(xfr->zone), false));
 
-       xfr->masteraddr = *masteraddr;
-
        INSIST(isc_sockaddr_pf(masteraddr) == isc_sockaddr_pf(sourceaddr));
-       xfr->sourceaddr = *sourceaddr;
        isc_sockaddr_setport(&xfr->sourceaddr, 0);
 
        /*
@@ -900,6 +883,7 @@ xfrin_create(isc_mem_t *mctx, dns_zone_t *zone, dns_db_t *db, isc_task_t *task,
                        sizeof(xfr->qbuffer_data) - 2);
 
        xfr->magic = XFRIN_MAGIC;
+
        *xfrp = xfr;
        return (ISC_R_SUCCESS);
 
@@ -926,21 +910,20 @@ failure:
 static isc_result_t
 xfrin_start(dns_xfrin_ctx_t *xfr) {
        isc_result_t result;
-       CHECK(isc_socket_create(xfr->socketmgr,
-                               isc_sockaddr_pf(&xfr->sourceaddr),
-                               isc_sockettype_tcp, &xfr->socket));
-       isc_socket_setname(xfr->socket, "xfrin", NULL);
-#ifndef BROKEN_TCP_BIND_BEFORE_CONNECT
-       CHECK(isc_socket_bind(xfr->socket, &xfr->sourceaddr,
-                             ISC_SOCKET_REUSEADDRESS));
-#endif /* ifndef BROKEN_TCP_BIND_BEFORE_CONNECT */
-       isc_socket_dscp(xfr->socket, xfr->dscp);
-       CHECK(isc_socket_connect(xfr->socket, &xfr->masteraddr, xfr->task,
-                                xfrin_connect_done, xfr));
-       xfr->connects++;
+       /*
+        * XXX: timeout hard-coded to 30 seconds; this needs to be
+        * configurable.
+        */
+       (void)isc_refcount_increment0(&xfr->connects);
+       CHECK(isc_nm_tcpdnsconnect(xfr->netmgr,
+                                  (isc_nmiface_t *)&xfr->sourceaddr,
+                                  (isc_nmiface_t *)&xfr->masteraddr,
+                                  xfrin_connect_done, xfr, 30000, 0));
+       /* TODO isc_socket_dscp(xfr->socket, xfr->dscp); */
        return (ISC_R_SUCCESS);
+
 failure:
-       xfrin_fail(xfr, result, "failed setting up socket");
+       isc_refcount_decrement0(&xfr->connects);
        return (result);
 }
 
@@ -972,30 +955,24 @@ failure:
  * A connection has been established.
  */
 static void
-xfrin_connect_done(isc_task_t *task, isc_event_t *event) {
-       isc_socket_connev_t *cev = (isc_socket_connev_t *)event;
-       dns_xfrin_ctx_t *xfr = (dns_xfrin_ctx_t *)event->ev_arg;
-       isc_result_t result = cev->result;
+xfrin_connect_done(isc_nmhandle_t *handle, isc_result_t result, void *cbarg) {
+       dns_xfrin_ctx_t *xfr = (dns_xfrin_ctx_t *)cbarg;
        char sourcetext[ISC_SOCKADDR_FORMATSIZE];
        char signerbuf[DNS_NAME_FORMATSIZE];
        const char *signer = "", *sep = "";
        isc_sockaddr_t sockaddr;
-       dns_zonemgr_t *zmgr;
+       dns_zonemgr_t *zmgr = NULL;
        isc_time_t now;
 
        REQUIRE(VALID_XFRIN(xfr));
 
-       UNUSED(task);
-
-       INSIST(event->ev_type == ISC_SOCKEVENT_CONNECT);
-       isc_event_free(&event);
-
-       xfr->connects--;
        if (xfr->shuttingdown) {
                maybe_free(xfr);
                return;
        }
 
+       isc_refcount_decrement0(&xfr->connects);
+
        zmgr = dns_zone_getmgr(xfr->zone);
        if (zmgr != NULL) {
                if (result != ISC_R_SUCCESS) {
@@ -1009,12 +986,9 @@ xfrin_connect_done(isc_task_t *task, isc_event_t *event) {
                }
        }
 
-       result = isc_socket_getsockname(xfr->socket, &sockaddr);
-       if (result == ISC_R_SUCCESS) {
-               isc_sockaddr_format(&sockaddr, sourcetext, sizeof(sourcetext));
-       } else {
-               strlcpy(sourcetext, "<UNKNOWN>", sizeof(sourcetext));
-       }
+       xfr->handle = handle;
+       sockaddr = isc_nmhandle_peeraddr(handle);
+       isc_sockaddr_format(&sockaddr, sourcetext, sizeof(sourcetext));
 
        if (xfr->tsigkey != NULL && xfr->tsigkey->key != NULL) {
                dns_name_format(dst_key_name(xfr->tsigkey->key), signerbuf,
@@ -1026,13 +1000,14 @@ xfrin_connect_done(isc_task_t *task, isc_event_t *event) {
        xfrin_log(xfr, ISC_LOG_INFO, "connected using %s%s%s", sourcetext, sep,
                  signer);
 
-       dns_tcpmsg_init(xfr->mctx, xfr->socket, &xfr->tcpmsg);
-       xfr->tcpmsg_valid = true;
-
        CHECK(xfrin_send_request(xfr));
+       isc_nmhandle_detach(&handle);
+
 failure:
        if (result != ISC_R_SUCCESS) {
                xfrin_fail(xfr, result, "failed to connect");
+       } else {
+               maybe_free(xfr);
        }
 }
 
@@ -1167,17 +1142,9 @@ xfrin_send_request(dns_xfrin_ctx_t *xfr) {
        isc_buffer_usedregion(&xfr->qbuffer, &region);
        INSIST(region.length <= 65535);
 
-       /*
-        * Record message length and adjust region to include TCP
-        * length field.
-        */
-       xfr->qbuffer_data[0] = (region.length >> 8) & 0xff;
-       xfr->qbuffer_data[1] = region.length & 0xff;
-       region.base -= 2;
-       region.length += 2;
-       CHECK(isc_socket_send(xfr->socket, &region, xfr->task, xfrin_send_done,
-                             xfr));
-       xfr->sends++;
+       isc_nmhandle_attach(xfr->handle, &xfr->sendhandle);
+       isc_refcount_increment0(&xfr->sends);
+       isc_nm_send(xfr->handle, &region, xfrin_send_done, xfr);
 
 failure:
        if (qname != NULL) {
@@ -1195,62 +1162,58 @@ failure:
        if (ver != NULL) {
                dns_db_closeversion(xfr->db, &ver, false);
        }
+
        return (result);
 }
 
 static void
-xfrin_send_done(isc_task_t *task, isc_event_t *event) {
-       isc_socketevent_t *sev = (isc_socketevent_t *)event;
-       dns_xfrin_ctx_t *xfr = (dns_xfrin_ctx_t *)event->ev_arg;
-       isc_result_t result;
+xfrin_send_done(isc_nmhandle_t *handle, isc_result_t result, void *cbarg) {
+       dns_xfrin_ctx_t *xfr = (dns_xfrin_ctx_t *)cbarg;
 
        REQUIRE(VALID_XFRIN(xfr));
 
-       UNUSED(task);
+       isc_refcount_decrement0(&xfr->sends);
 
-       INSIST(event->ev_type == ISC_SOCKEVENT_SENDDONE);
+       if (result != ISC_R_SUCCESS) {
+               isc_nmhandle_detach(&xfr->sendhandle);
+               xfrin_fail(xfr, result, "failed sending request data");
+               return;
+       }
 
-       xfr->sends--;
        xfrin_log(xfr, ISC_LOG_DEBUG(3), "sent request data");
-       CHECK(sev->result);
 
-       CHECK(dns_tcpmsg_readmessage(&xfr->tcpmsg, xfr->task, xfrin_recv_done,
-                                    xfr));
-       xfr->recvs++;
-failure:
-       isc_event_free(&event);
-       if (result != ISC_R_SUCCESS) {
-               xfrin_fail(xfr, result, "failed sending request data");
+       if (xfr->readhandle == NULL) {
+               isc_nmhandle_attach(handle, &xfr->readhandle);
        }
+
+       isc_nm_read(xfr->handle, xfrin_recv_done, xfr);
+
+       isc_refcount_increment0(&xfr->recvs);
+       isc_nmhandle_detach(&xfr->sendhandle);
+       maybe_free(xfr);
 }
 
 static void
-xfrin_recv_done(isc_task_t *task, isc_event_t *ev) {
-       dns_xfrin_ctx_t *xfr = (dns_xfrin_ctx_t *)ev->ev_arg;
-       isc_result_t result;
+xfrin_recv_done(isc_nmhandle_t *handle, isc_result_t result,
+               isc_region_t *region, void *cbarg) {
+       dns_xfrin_ctx_t *xfr = (dns_xfrin_ctx_t *)cbarg;
        dns_message_t *msg = NULL;
-       dns_name_t *name;
-       dns_tcpmsg_t *tcpmsg;
+       dns_name_t *name = NULL;
        const dns_name_t *tsigowner = NULL;
+       isc_buffer_t buffer;
+       isc_sockaddr_t peer;
 
        REQUIRE(VALID_XFRIN(xfr));
 
-       UNUSED(task);
-
-       INSIST(ev->ev_type == DNS_EVENT_TCPMSG);
-       tcpmsg = ev->ev_sender;
-       isc_event_free(&ev);
-
-       xfr->recvs--;
+       isc_refcount_decrement0(&xfr->recvs);
        if (xfr->shuttingdown) {
                maybe_free(xfr);
                return;
        }
 
-       CHECK(tcpmsg->result);
+       CHECK(result);
 
-       xfrin_log(xfr, ISC_LOG_DEBUG(7), "received %u bytes",
-                 tcpmsg->buffer.used);
+       xfrin_log(xfr, ISC_LOG_DEBUG(7), "received %u bytes", region->length);
 
        CHECK(isc_timer_touch(xfr->timer));
 
@@ -1268,12 +1231,15 @@ xfrin_recv_done(isc_task_t *task, isc_event_t *ev) {
                msg->tcp_continuation = 1;
        }
 
-       result = dns_message_parse(msg, &tcpmsg->buffer,
-                                  DNS_MESSAGEPARSE_PRESERVEORDER);
+       isc_buffer_init(&buffer, region->base, region->length);
+       isc_buffer_add(&buffer, region->length);
+       peer = isc_nmhandle_peeraddr(handle);
 
+       result = dns_message_parse(msg, &buffer,
+                                  DNS_MESSAGEPARSE_PRESERVEORDER);
        if (result == ISC_R_SUCCESS) {
-               dns_message_logpacket(msg, "received message from",
-                                     &tcpmsg->address, DNS_LOGCATEGORY_XFER_IN,
+               dns_message_logpacket(msg, "received message from", &peer,
+                                     DNS_LOGCATEGORY_XFER_IN,
                                      DNS_LOGMODULE_XFER_IN, ISC_LOG_DEBUG(10),
                                      xfr->mctx);
        } else {
@@ -1297,10 +1263,12 @@ xfrin_recv_done(isc_task_t *task, isc_event_t *ev) {
                } else if (result == ISC_R_SUCCESS || result == DNS_R_NOERROR) {
                        result = DNS_R_UNEXPECTEDID;
                }
+
                if (xfr->reqtype == dns_rdatatype_axfr ||
                    xfr->reqtype == dns_rdatatype_soa) {
                        goto failure;
                }
+
                xfrin_log(xfr, ISC_LOG_DEBUG(3), "got %s, retrying with AXFR",
                          isc_result_totext(result));
        try_axfr:
@@ -1308,7 +1276,10 @@ xfrin_recv_done(isc_task_t *task, isc_event_t *ev) {
                xfrin_reset(xfr);
                xfr->reqtype = dns_rdatatype_soa;
                xfr->state = XFRST_SOAQUERY;
-               (void)xfrin_start(xfr);
+               result = xfrin_start(xfr);
+               if (result != ISC_R_SUCCESS) {
+                       xfrin_fail(xfr, result, "failed setting up socket");
+               }
                return;
        }
 
@@ -1338,7 +1309,7 @@ xfrin_recv_done(isc_task_t *task, isc_event_t *ev) {
             result == ISC_R_SUCCESS;
             result = dns_message_nextname(msg, DNS_SECTION_QUESTION))
        {
-               dns_rdataset_t *rds;
+               dns_rdataset_t *rds = NULL;
 
                name = NULL;
                dns_message_currentname(msg, DNS_SECTION_QUESTION, &name);
@@ -1398,7 +1369,7 @@ xfrin_recv_done(isc_task_t *task, isc_event_t *ev) {
             result == ISC_R_SUCCESS;
             result = dns_message_nextname(msg, DNS_SECTION_ANSWER))
        {
-               dns_rdataset_t *rds;
+               dns_rdataset_t *rds = NULL;
 
                name = NULL;
                dns_message_currentname(msg, DNS_SECTION_ANSWER, &name);
@@ -1455,7 +1426,7 @@ xfrin_recv_done(isc_task_t *task, isc_event_t *ev) {
        /*
         * Update the number of bytes received.
         */
-       xfr->nbytes += tcpmsg->buffer.used;
+       xfr->nbytes += buffer.used;
 
        /*
         * Take the context back.
@@ -1471,10 +1442,11 @@ xfrin_recv_done(isc_task_t *task, isc_event_t *ev) {
                xfr->reqtype = dns_rdatatype_axfr;
                xfr->state = XFRST_INITIALSOA;
                CHECK(xfrin_send_request(xfr));
+               isc_nmhandle_detach(&xfr->readhandle);
                break;
        case XFRST_AXFR_END:
                CHECK(axfr_finalize(xfr));
-       /* FALLTHROUGH */
+               /* FALLTHROUGH */
        case XFRST_IXFR_END:
                /*
                 * Close the journal.
@@ -1490,21 +1462,20 @@ xfrin_recv_done(isc_task_t *task, isc_event_t *ev) {
                        (xfr->done)(xfr->zone, ISC_R_SUCCESS);
                        xfr->done = NULL;
                }
-               /*
-                * We should have no outstanding events at this
-                * point, thus maybe_free() should succeed.
-                */
+
                xfr->shuttingdown = true;
                xfr->shutdown_result = ISC_R_SUCCESS;
-               maybe_free(xfr);
+               dns_xfrin_detach(&xfr);
                break;
        default:
                /*
                 * Read the next message.
                 */
-               CHECK(dns_tcpmsg_readmessage(&xfr->tcpmsg, xfr->task,
-                                            xfrin_recv_done, xfr));
-               xfr->recvs++;
+               if (xfr->readhandle == NULL) {
+                       isc_nmhandle_attach(handle, &xfr->readhandle);
+               }
+               isc_nm_read(xfr->handle, xfrin_recv_done, xfr);
+               isc_refcount_increment0(&xfr->recvs);
        }
        return;
 
@@ -1512,9 +1483,12 @@ failure:
        if (msg != NULL) {
                dns_message_detach(&msg);
        }
+
        if (result != ISC_R_SUCCESS) {
                xfrin_fail(xfr, result, "failed while receiving responses");
        }
+
+       dns_xfrin_detach(&xfr);
 }
 
 static void
@@ -1540,21 +1514,22 @@ maybe_free(dns_xfrin_ctx_t *xfr) {
 
        REQUIRE(VALID_XFRIN(xfr));
 
-       if (!xfr->shuttingdown || xfr->refcount != 0 || xfr->connects != 0 ||
-           xfr->sends != 0 || xfr->recvs != 0)
+       if (!xfr->shuttingdown || isc_refcount_current(&xfr->refs) != 0 ||
+           isc_refcount_current(&xfr->connects) != 0 ||
+           isc_refcount_current(&xfr->sends) != 0 ||
+           isc_refcount_current(&xfr->recvs) != 0)
        {
                return;
        }
 
-       INSIST(!xfr->shuttingdown || xfr->shutdown_result != ISC_R_UNSET);
+       INSIST(xfr->shutdown_result != ISC_R_UNSET);
 
-       /* If we're called through dns_xfrin_detach() and are not
+       /*
+        * If we're called through dns_xfrin_detach() and are not
         * shutting down, we can't know what the transfer status is as
         * we are only called when the last reference is lost.
         */
-       result_str = (xfr->shuttingdown
-                             ? isc_result_totext(xfr->shutdown_result)
-                             : "unknown");
+       result_str = isc_result_totext(xfr->shutdown_result);
        xfrin_log(xfr, ISC_LOG_INFO, "Transfer status: %s", result_str);
 
        /*
@@ -1575,8 +1550,11 @@ maybe_free(dns_xfrin_ctx_t *xfr) {
                  (unsigned int)(msecs / 1000), (unsigned int)(msecs % 1000),
                  (unsigned int)persec, xfr->end_serial);
 
-       if (xfr->socket != NULL) {
-               isc_socket_detach(&xfr->socket);
+       if (xfr->readhandle != NULL) {
+               isc_nmhandle_detach(&xfr->readhandle);
+       }
+       if (xfr->sendhandle != NULL) {
+               isc_nmhandle_detach(&xfr->sendhandle);
        }
 
        if (xfr->timer != NULL) {
@@ -1591,6 +1569,11 @@ maybe_free(dns_xfrin_ctx_t *xfr) {
                dns_tsigkey_detach(&xfr->tsigkey);
        }
 
+       isc_refcount_destroy(&xfr->refs);
+       isc_refcount_destroy(&xfr->connects);
+       isc_refcount_destroy(&xfr->recvs);
+       isc_refcount_destroy(&xfr->sends);
+
        if (xfr->lasttsig != NULL) {
                isc_buffer_free(&xfr->lasttsig);
        }
@@ -1605,10 +1588,6 @@ maybe_free(dns_xfrin_ctx_t *xfr) {
                (void)dns_db_endload(xfr->db, &xfr->axfr);
        }
 
-       if (xfr->tcpmsg_valid) {
-               dns_tcpmsg_invalidate(&xfr->tcpmsg);
-       }
-
        if (xfr->tsigctx != NULL) {
                dst_context_destroy(&xfr->tsigctx);
        }
@@ -1626,7 +1605,7 @@ maybe_free(dns_xfrin_ctx_t *xfr) {
        }
 
        if (xfr->zone != NULL) {
-               if (!xfr->zone_had_db && xfr->shuttingdown &&
+               if (!xfr->zone_had_db &&
                    xfr->shutdown_result == ISC_R_SUCCESS &&
                    dns_zone_gettype(xfr->zone) == dns_zone_mirror)
                {
index 03427968ad0a717adbba0299ddeaa2113806ef2c..2e3886bbc6c72510d8798549435f8f49787073d4 100644 (file)
@@ -545,6 +545,7 @@ struct dns_zonemgr {
        isc_taskmgr_t *taskmgr;
        isc_timermgr_t *timermgr;
        isc_socketmgr_t *socketmgr;
+       isc_nm_t *netmgr;
        isc_taskpool_t *zonetasks;
        isc_taskpool_t *loadtasks;
        isc_task_t *task;
@@ -774,6 +775,8 @@ zone_unload(dns_zone_t *zone);
 static void
 zone_expire(dns_zone_t *zone);
 static void
+zone_refresh(dns_zone_t *zone);
+static void
 zone_iattach(dns_zone_t *source, dns_zone_t **target);
 static void
 zone_idetach(dns_zone_t **zonep);
@@ -10966,11 +10969,13 @@ zone_maintenance(dns_zone_t *zone) {
        case dns_zone_slave:
        case dns_zone_mirror:
        case dns_zone_stub:
+               LOCK_ZONE(zone);
                if (!DNS_ZONE_FLAG(zone, DNS_ZONEFLG_DIALREFRESH) &&
                    isc_time_compare(&now, &zone->refreshtime) >= 0)
                {
-                       dns_zone_refresh(zone);
+                       zone_refresh(zone);
                }
+               UNLOCK_ZONE(zone);
                break;
        default:
                break;
@@ -11215,14 +11220,15 @@ failure:
        zone_unload(zone);
 }
 
-void
-dns_zone_refresh(dns_zone_t *zone) {
+static void
+zone_refresh(dns_zone_t *zone) {
        isc_interval_t i;
        uint32_t oldflags;
        unsigned int j;
        isc_result_t result;
 
        REQUIRE(DNS_ZONE_VALID(zone));
+       REQUIRE(LOCKED_ZONE(zone));
 
        if (DNS_ZONE_FLAG(zone, DNS_ZONEFLG_EXITING)) {
                return;
@@ -11233,7 +11239,6 @@ dns_zone_refresh(dns_zone_t *zone) {
         * in progress at a time.
         */
 
-       LOCK_ZONE(zone);
        oldflags = atomic_load(&zone->flags);
        if (zone->masterscnt == 0) {
                DNS_ZONE_SETFLAG(zone, DNS_ZONEFLG_NOMASTERS);
@@ -11241,13 +11246,13 @@ dns_zone_refresh(dns_zone_t *zone) {
                        dns_zone_log(zone, ISC_LOG_ERROR,
                                     "cannot refresh: no primaries");
                }
-               goto unlock;
+               return;
        }
        DNS_ZONE_SETFLAG(zone, DNS_ZONEFLG_REFRESH);
        DNS_ZONE_CLRFLAG(zone, DNS_ZONEFLG_NOEDNS);
        DNS_ZONE_CLRFLAG(zone, DNS_ZONEFLG_USEALTXFRSRC);
        if ((oldflags & (DNS_ZONEFLG_REFRESH | DNS_ZONEFLG_LOADING)) != 0) {
-               goto unlock;
+               return;
        }
 
        /*
@@ -11279,7 +11284,12 @@ dns_zone_refresh(dns_zone_t *zone) {
        }
        /* initiate soa query */
        queue_soa_query(zone);
-unlock:
+}
+
+void
+dns_zone_refresh(dns_zone_t *zone) {
+       LOCK_ZONE(zone);
+       zone_refresh(zone);
        UNLOCK_ZONE(zone);
 }
 
@@ -14392,6 +14402,13 @@ zone_shutdown(isc_task_t *task, isc_event_t *event) {
                dns_xfrin_shutdown(zone->xfr);
        }
 
+       /*
+        * In case the shutdown didn't detach the xfr object...
+        */
+       if (zone->xfr != NULL) {
+               dns_xfrin_detach(&zone->xfr);
+       }
+
        /* Safe to release the zone now */
        if (zone->zmgr != NULL) {
                dns_zonemgr_releasezone(zone->zmgr, zone);
@@ -16882,7 +16899,7 @@ zone_xfrdone(dns_zone_t *zone, isc_result_t result) {
        /*
         * Obtaining a lock on the zone->secure (see zone_send_secureserial)
         * could result in a deadlock due to a LOR so we will spin if we
-        * can't obtain the both locks.
+        * can't obtain both locks.
         */
 again:
        LOCK_ZONE(zone);
@@ -17351,8 +17368,7 @@ got_transfer_quota(isc_task_t *task, isc_event_t *event) {
        INSIST(task == zone->task);
 
        if (DNS_ZONE_FLAG(zone, DNS_ZONEFLG_EXITING)) {
-               result = ISC_R_CANCELED;
-               goto cleanup;
+               CHECK(ISC_R_CANCELED);
        }
 
        TIME_NOW(&now);
@@ -17366,8 +17382,7 @@ got_transfer_quota(isc_task_t *task, isc_event_t *event) {
                              "got_transfer_quota: skipping zone transfer as "
                              "master %s (source %s) is unreachable (cached)",
                              master, source);
-               result = ISC_R_CANCELED;
-               goto cleanup;
+               CHECK(ISC_R_CANCELED);
        }
 
        isc_netaddr_fromsockaddr(&masterip, &zone->masteraddr);
@@ -17482,28 +17497,32 @@ got_transfer_quota(isc_task_t *task, isc_event_t *event) {
        }
        UNLOCK_ZONE(zone);
        INSIST(isc_sockaddr_pf(&masteraddr) == isc_sockaddr_pf(&sourceaddr));
-       result = dns_xfrin_create(zone, xfrtype, &masteraddr, &sourceaddr, dscp,
-                                 zone->tsigkey, zone->mctx,
-                                 zone->zmgr->timermgr, zone->zmgr->socketmgr,
-                                 zone->task, zone_xfrdone, &zone->xfr);
-       if (result == ISC_R_SUCCESS) {
-               LOCK_ZONE(zone);
-               if (xfrtype == dns_rdatatype_axfr) {
-                       if (isc_sockaddr_pf(&masteraddr) == PF_INET) {
-                               inc_stats(zone, dns_zonestatscounter_axfrreqv4);
-                       } else {
-                               inc_stats(zone, dns_zonestatscounter_axfrreqv6);
-                       }
-               } else if (xfrtype == dns_rdatatype_ixfr) {
-                       if (isc_sockaddr_pf(&masteraddr) == PF_INET) {
-                               inc_stats(zone, dns_zonestatscounter_ixfrreqv4);
-                       } else {
-                               inc_stats(zone, dns_zonestatscounter_ixfrreqv6);
-                       }
+
+       if (zone->xfr != NULL) {
+               dns_xfrin_detach(&zone->xfr);
+       }
+
+       CHECK(dns_xfrin_create(zone, xfrtype, &masteraddr, &sourceaddr, dscp,
+                              zone->tsigkey, zone->mctx, zone->zmgr->timermgr,
+                              zone->zmgr->netmgr, zone->task, zone_xfrdone,
+                              &zone->xfr));
+       LOCK_ZONE(zone);
+       if (xfrtype == dns_rdatatype_axfr) {
+               if (isc_sockaddr_pf(&masteraddr) == PF_INET) {
+                       inc_stats(zone, dns_zonestatscounter_axfrreqv4);
+               } else {
+                       inc_stats(zone, dns_zonestatscounter_axfrreqv6);
+               }
+       } else if (xfrtype == dns_rdatatype_ixfr) {
+               if (isc_sockaddr_pf(&masteraddr) == PF_INET) {
+                       inc_stats(zone, dns_zonestatscounter_ixfrreqv4);
+               } else {
+                       inc_stats(zone, dns_zonestatscounter_ixfrreqv6);
                }
-               UNLOCK_ZONE(zone);
        }
-cleanup:
+       UNLOCK_ZONE(zone);
+
+failure:
        /*
         * Any failure in this function is handled like a failed
         * zone transfer.  This ensures that we get removed from
@@ -17789,7 +17808,7 @@ dns_zone_first(dns_zonemgr_t *zmgr, dns_zone_t **first) {
 isc_result_t
 dns_zonemgr_create(isc_mem_t *mctx, isc_taskmgr_t *taskmgr,
                   isc_timermgr_t *timermgr, isc_socketmgr_t *socketmgr,
-                  dns_zonemgr_t **zmgrp) {
+                  isc_nm_t *netmgr, dns_zonemgr_t **zmgrp) {
        dns_zonemgr_t *zmgr;
        isc_result_t result;
 
@@ -17800,6 +17819,7 @@ dns_zonemgr_create(isc_mem_t *mctx, isc_taskmgr_t *taskmgr,
        zmgr->taskmgr = taskmgr;
        zmgr->timermgr = timermgr;
        zmgr->socketmgr = socketmgr;
+       zmgr->netmgr = netmgr;
        zmgr->zonetasks = NULL;
        zmgr->loadtasks = NULL;
        zmgr->mctxpool = NULL;
index 039e158a785db34a48dca0075e53280f301baa5f..20b746023cbc850bc9f1010c4dd0470327fd47eb 100644 (file)
@@ -459,7 +459,7 @@ ns_test_setupzonemgr(void) {
        isc_result_t result;
        REQUIRE(zonemgr == NULL);
 
-       result = dns_zonemgr_create(mctx, taskmgr, timermgr, socketmgr,
+       result = dns_zonemgr_create(mctx, taskmgr, timermgr, socketmgr, NULL,
                                    &zonemgr);
        return (result);
 }