mirror of
https://github.com/isc-projects/bind9.git
synced 2026-02-27 12:02:10 -05:00
When shutting down the whole server, the reading could stop and detach from controlconnection before sending is done. If send callback then detaches from the last controlconnection handle, the ccmsg would be invalidated after the send callback and thus we must not access ccmsg after calling the send_cb().
204 lines
4.8 KiB
C
204 lines
4.8 KiB
C
/*
|
|
* Copyright (C) Internet Systems Consortium, Inc. ("ISC")
|
|
*
|
|
* SPDX-License-Identifier: MPL-2.0 AND 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.
|
|
*/
|
|
|
|
/*
|
|
* Copyright (C) 2001 Nominum, Inc.
|
|
*
|
|
* Permission to use, copy, modify, and/or 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 ISC AND NOMINUM DISCLAIMS ALL
|
|
* WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES
|
|
* OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL ISC 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.
|
|
*/
|
|
|
|
/*! \file */
|
|
|
|
#include <inttypes.h>
|
|
|
|
#include <isc/mem.h>
|
|
#include <isc/netmgr.h>
|
|
#include <isc/result.h>
|
|
#include <isc/string.h>
|
|
#include <isc/util.h>
|
|
|
|
#include <isccc/ccmsg.h>
|
|
|
|
#define CCMSG_MAGIC ISC_MAGIC('C', 'C', 'm', 's')
|
|
#define VALID_CCMSG(foo) ISC_MAGIC_VALID(foo, CCMSG_MAGIC)
|
|
|
|
static void
|
|
recv_data(isc_nmhandle_t *handle, isc_result_t eresult, isc_region_t *region,
|
|
void *arg) {
|
|
isccc_ccmsg_t *ccmsg = arg;
|
|
|
|
REQUIRE(VALID_CCMSG(ccmsg));
|
|
|
|
REQUIRE(handle == ccmsg->handle);
|
|
if (eresult != ISC_R_SUCCESS) {
|
|
goto done;
|
|
}
|
|
|
|
REQUIRE(region != NULL);
|
|
|
|
if (!ccmsg->length_received) {
|
|
if (region->length < sizeof(uint32_t)) {
|
|
eresult = ISC_R_UNEXPECTEDEND;
|
|
goto done;
|
|
}
|
|
|
|
ccmsg->size = ntohl(*(uint32_t *)region->base);
|
|
|
|
if (ccmsg->size == 0) {
|
|
eresult = ISC_R_UNEXPECTEDEND;
|
|
goto done;
|
|
}
|
|
if (ccmsg->size > ccmsg->maxsize) {
|
|
eresult = ISC_R_RANGE;
|
|
goto done;
|
|
}
|
|
|
|
isc_region_consume(region, sizeof(uint32_t));
|
|
isc_buffer_allocate(ccmsg->mctx, &ccmsg->buffer, ccmsg->size);
|
|
|
|
ccmsg->length_received = true;
|
|
}
|
|
|
|
/*
|
|
* If there's no more data, wait for more
|
|
*/
|
|
if (region->length == 0) {
|
|
return;
|
|
}
|
|
|
|
/* We have some data in the buffer, read it */
|
|
|
|
size_t size = ISC_MIN(isc_buffer_availablelength(ccmsg->buffer),
|
|
region->length);
|
|
isc_buffer_putmem(ccmsg->buffer, region->base, size);
|
|
isc_region_consume(region, size);
|
|
|
|
if (isc_buffer_usedlength(ccmsg->buffer) == ccmsg->size) {
|
|
goto done;
|
|
}
|
|
|
|
/* Wait for more data to come */
|
|
return;
|
|
|
|
done:
|
|
isc_nm_read_stop(handle);
|
|
ccmsg->recv_cb(handle, eresult, ccmsg->recv_cbarg);
|
|
|
|
return;
|
|
}
|
|
|
|
void
|
|
isccc_ccmsg_init(isc_mem_t *mctx, isc_nmhandle_t *handle,
|
|
isccc_ccmsg_t *ccmsg) {
|
|
REQUIRE(mctx != NULL);
|
|
REQUIRE(handle != NULL);
|
|
REQUIRE(ccmsg != NULL);
|
|
|
|
*ccmsg = (isccc_ccmsg_t){
|
|
.magic = CCMSG_MAGIC,
|
|
.maxsize = 0xffffffffU, /* Largest message possible. */
|
|
.mctx = mctx,
|
|
};
|
|
|
|
isc_nmhandle_attach(handle, &ccmsg->handle);
|
|
}
|
|
|
|
void
|
|
isccc_ccmsg_setmaxsize(isccc_ccmsg_t *ccmsg, unsigned int maxsize) {
|
|
REQUIRE(VALID_CCMSG(ccmsg));
|
|
|
|
ccmsg->maxsize = maxsize;
|
|
}
|
|
|
|
void
|
|
isccc_ccmsg_readmessage(isccc_ccmsg_t *ccmsg, isc_nm_cb_t cb, void *cbarg) {
|
|
REQUIRE(VALID_CCMSG(ccmsg));
|
|
|
|
if (ccmsg->buffer != NULL) {
|
|
isc_buffer_free(&ccmsg->buffer);
|
|
}
|
|
|
|
ccmsg->recv_cb = cb;
|
|
ccmsg->recv_cbarg = cbarg;
|
|
ccmsg->length_received = false;
|
|
|
|
isc_nm_read(ccmsg->handle, recv_data, ccmsg);
|
|
}
|
|
|
|
static void
|
|
ccmsg_senddone(isc_nmhandle_t *handle, isc_result_t eresult, void *arg) {
|
|
isccc_ccmsg_t *ccmsg = arg;
|
|
|
|
REQUIRE(VALID_CCMSG(ccmsg));
|
|
REQUIRE(ccmsg->send_cb != NULL);
|
|
|
|
isc_nm_cb_t send_cb = ccmsg->send_cb;
|
|
ccmsg->send_cb = NULL;
|
|
|
|
send_cb(handle, eresult, ccmsg->send_cbarg);
|
|
|
|
isc_nmhandle_detach(&handle);
|
|
}
|
|
|
|
void
|
|
isccc_ccmsg_sendmessage(isccc_ccmsg_t *ccmsg, isc_region_t *region,
|
|
isc_nm_cb_t cb, void *cbarg) {
|
|
REQUIRE(VALID_CCMSG(ccmsg));
|
|
REQUIRE(ccmsg->send_cb == NULL);
|
|
|
|
ccmsg->send_cb = cb;
|
|
ccmsg->send_cbarg = cbarg;
|
|
|
|
isc_nmhandle_ref(ccmsg->handle);
|
|
isc_nm_send(ccmsg->handle, region, ccmsg_senddone, ccmsg);
|
|
}
|
|
|
|
void
|
|
isccc_ccmsg_disconnect(isccc_ccmsg_t *ccmsg) {
|
|
REQUIRE(VALID_CCMSG(ccmsg));
|
|
|
|
if (ccmsg->handle != NULL) {
|
|
isc_nm_read_stop(ccmsg->handle);
|
|
isc_nmhandle_close(ccmsg->handle);
|
|
isc_nmhandle_detach(&ccmsg->handle);
|
|
}
|
|
}
|
|
|
|
void
|
|
isccc_ccmsg_invalidate(isccc_ccmsg_t *ccmsg) {
|
|
REQUIRE(VALID_CCMSG(ccmsg));
|
|
|
|
ccmsg->magic = 0;
|
|
|
|
if (ccmsg->buffer != NULL) {
|
|
isc_buffer_free(&ccmsg->buffer);
|
|
}
|
|
}
|
|
|
|
void
|
|
isccc_ccmsg_toregion(isccc_ccmsg_t *ccmsg, isccc_region_t *ccregion) {
|
|
REQUIRE(VALID_CCMSG(ccmsg));
|
|
|
|
ccregion->rstart = isc_buffer_base(ccmsg->buffer);
|
|
ccregion->rend = isc_buffer_used(ccmsg->buffer);
|
|
}
|