Blame lib/isc/taskpool.c

Packit Service ae04f2
/*
Packit Service ae04f2
 * Copyright (C) Internet Systems Consortium, Inc. ("ISC")
Packit Service ae04f2
 *
Packit Service ae04f2
 * This Source Code Form is subject to the terms of the Mozilla Public
Packit Service ae04f2
 * License, v. 2.0. If a copy of the MPL was not distributed with this
Packit Service ae04f2
 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
Packit Service ae04f2
 *
Packit Service ae04f2
 * See the COPYRIGHT file distributed with this work for additional
Packit Service ae04f2
 * information regarding copyright ownership.
Packit Service ae04f2
 */
Packit Service ae04f2
Packit Service ae04f2
Packit Service ae04f2
/*! \file */
Packit Service ae04f2
Packit Service ae04f2
#include <config.h>
Packit Service ae04f2
Packit Service ae04f2
#include <stdbool.h>
Packit Service ae04f2
Packit Service ae04f2
#include <isc/mem.h>
Packit Service ae04f2
#include <isc/random.h>
Packit Service ae04f2
#include <isc/taskpool.h>
Packit Service ae04f2
#include <isc/util.h>
Packit Service ae04f2
Packit Service ae04f2
/***
Packit Service ae04f2
 *** Types.
Packit Service ae04f2
 ***/
Packit Service ae04f2
Packit Service ae04f2
struct isc_taskpool {
Packit Service ae04f2
	isc_mem_t *			mctx;
Packit Service ae04f2
	isc_taskmgr_t *			tmgr;
Packit Service ae04f2
	unsigned int			ntasks;
Packit Service ae04f2
	unsigned int			quantum;
Packit Service ae04f2
	isc_task_t **			tasks;
Packit Service ae04f2
};
Packit Service ae04f2
Packit Service ae04f2
/***
Packit Service ae04f2
 *** Functions.
Packit Service ae04f2
 ***/
Packit Service ae04f2
Packit Service ae04f2
static isc_result_t
Packit Service ae04f2
alloc_pool(isc_taskmgr_t *tmgr, isc_mem_t *mctx, unsigned int ntasks,
Packit Service ae04f2
	   unsigned int quantum, isc_taskpool_t **poolp)
Packit Service ae04f2
{
Packit Service ae04f2
	isc_taskpool_t *pool;
Packit Service ae04f2
	unsigned int i;
Packit Service ae04f2
Packit Service ae04f2
	pool = isc_mem_get(mctx, sizeof(*pool));
Packit Service ae04f2
	if (pool == NULL)
Packit Service ae04f2
		return (ISC_R_NOMEMORY);
Packit Service ae04f2
Packit Service ae04f2
	pool->mctx = NULL;
Packit Service ae04f2
	isc_mem_attach(mctx, &pool->mctx);
Packit Service ae04f2
	pool->ntasks = ntasks;
Packit Service ae04f2
	pool->quantum = quantum;
Packit Service ae04f2
	pool->tmgr = tmgr;
Packit Service ae04f2
	pool->tasks = isc_mem_get(mctx, ntasks * sizeof(isc_task_t *));
Packit Service ae04f2
	if (pool->tasks == NULL) {
Packit Service ae04f2
		isc_mem_putanddetach(&pool->mctx, pool, sizeof(*pool));
Packit Service ae04f2
		return (ISC_R_NOMEMORY);
Packit Service ae04f2
	}
Packit Service ae04f2
	for (i = 0; i < ntasks; i++)
Packit Service ae04f2
		pool->tasks[i] = NULL;
Packit Service ae04f2
Packit Service ae04f2
	*poolp = pool;
Packit Service ae04f2
	return (ISC_R_SUCCESS);
Packit Service ae04f2
}
Packit Service ae04f2
Packit Service ae04f2
isc_result_t
Packit Service ae04f2
isc_taskpool_create(isc_taskmgr_t *tmgr, isc_mem_t *mctx,
Packit Service ae04f2
		    unsigned int ntasks, unsigned int quantum,
Packit Service ae04f2
		    isc_taskpool_t **poolp)
Packit Service ae04f2
{
Packit Service ae04f2
	unsigned int i;
Packit Service ae04f2
	isc_taskpool_t *pool = NULL;
Packit Service ae04f2
	isc_result_t result;
Packit Service ae04f2
Packit Service ae04f2
	INSIST(ntasks > 0);
Packit Service ae04f2
Packit Service ae04f2
	/* Allocate the pool structure */
Packit Service ae04f2
	result = alloc_pool(tmgr, mctx, ntasks, quantum, &pool);
Packit Service ae04f2
	if (result != ISC_R_SUCCESS)
Packit Service ae04f2
		return (result);
Packit Service ae04f2
Packit Service ae04f2
	/* Create the tasks */
Packit Service ae04f2
	for (i = 0; i < ntasks; i++) {
Packit Service ae04f2
		result = isc_task_create(tmgr, quantum, &pool->tasks[i]);
Packit Service ae04f2
		if (result != ISC_R_SUCCESS) {
Packit Service ae04f2
			isc_taskpool_destroy(&pool);
Packit Service ae04f2
			return (result);
Packit Service ae04f2
		}
Packit Service ae04f2
		isc_task_setname(pool->tasks[i], "taskpool", NULL);
Packit Service ae04f2
	}
Packit Service ae04f2
Packit Service ae04f2
	*poolp = pool;
Packit Service ae04f2
	return (ISC_R_SUCCESS);
Packit Service ae04f2
}
Packit Service ae04f2
Packit Service ae04f2
void
Packit Service ae04f2
isc_taskpool_gettask(isc_taskpool_t *pool, isc_task_t **targetp) {
Packit Service ae04f2
	uint32_t i;
Packit Service ae04f2
	isc_random_get(&i);
Packit Service ae04f2
	isc_task_attach(pool->tasks[i % pool->ntasks], targetp);
Packit Service ae04f2
}
Packit Service ae04f2
Packit Service ae04f2
int
Packit Service ae04f2
isc_taskpool_size(isc_taskpool_t *pool) {
Packit Service ae04f2
	REQUIRE(pool != NULL);
Packit Service ae04f2
	return (pool->ntasks);
Packit Service ae04f2
}
Packit Service ae04f2
Packit Service ae04f2
isc_result_t
Packit Service ae04f2
isc_taskpool_expand(isc_taskpool_t **sourcep, unsigned int size,
Packit Service ae04f2
		    isc_taskpool_t **targetp)
Packit Service ae04f2
{
Packit Service ae04f2
	isc_result_t result;
Packit Service ae04f2
	isc_taskpool_t *pool;
Packit Service ae04f2
Packit Service ae04f2
	REQUIRE(sourcep != NULL && *sourcep != NULL);
Packit Service ae04f2
	REQUIRE(targetp != NULL && *targetp == NULL);
Packit Service ae04f2
Packit Service ae04f2
	pool = *sourcep;
Packit Service ae04f2
	if (size > pool->ntasks) {
Packit Service ae04f2
		isc_taskpool_t *newpool = NULL;
Packit Service ae04f2
		unsigned int i;
Packit Service ae04f2
Packit Service ae04f2
		/* Allocate a new pool structure */
Packit Service ae04f2
		result = alloc_pool(pool->tmgr, pool->mctx, size,
Packit Service ae04f2
				    pool->quantum, &newpool);
Packit Service ae04f2
		if (result != ISC_R_SUCCESS)
Packit Service ae04f2
			return (result);
Packit Service ae04f2
Packit Service ae04f2
		/* Copy over the tasks from the old pool */
Packit Service ae04f2
		for (i = 0; i < pool->ntasks; i++) {
Packit Service ae04f2
			newpool->tasks[i] = pool->tasks[i];
Packit Service ae04f2
			pool->tasks[i] = NULL;
Packit Service ae04f2
		}
Packit Service ae04f2
Packit Service ae04f2
		/* Create new tasks */
Packit Service ae04f2
		for (i = pool->ntasks; i < size; i++) {
Packit Service ae04f2
			result = isc_task_create(pool->tmgr, pool->quantum,
Packit Service ae04f2
						 &newpool->tasks[i]);
Packit Service ae04f2
			if (result != ISC_R_SUCCESS) {
Packit Service ae04f2
				isc_taskpool_destroy(&newpool);
Packit Service ae04f2
				return (result);
Packit Service ae04f2
			}
Packit Service ae04f2
			isc_task_setname(newpool->tasks[i], "taskpool", NULL);
Packit Service ae04f2
		}
Packit Service ae04f2
Packit Service ae04f2
		isc_taskpool_destroy(&pool);
Packit Service ae04f2
		pool = newpool;
Packit Service ae04f2
	}
Packit Service ae04f2
Packit Service ae04f2
	*sourcep = NULL;
Packit Service ae04f2
	*targetp = pool;
Packit Service ae04f2
	return (ISC_R_SUCCESS);
Packit Service ae04f2
}
Packit Service ae04f2
Packit Service ae04f2
void
Packit Service ae04f2
isc_taskpool_destroy(isc_taskpool_t **poolp) {
Packit Service ae04f2
	unsigned int i;
Packit Service ae04f2
	isc_taskpool_t *pool = *poolp;
Packit Service ae04f2
	for (i = 0; i < pool->ntasks; i++) {
Packit Service ae04f2
		if (pool->tasks[i] != NULL)
Packit Service ae04f2
			isc_task_detach(&pool->tasks[i]);
Packit Service ae04f2
	}
Packit Service ae04f2
	isc_mem_put(pool->mctx, pool->tasks,
Packit Service ae04f2
		    pool->ntasks * sizeof(isc_task_t *));
Packit Service ae04f2
	isc_mem_putanddetach(&pool->mctx, pool, sizeof(*pool));
Packit Service ae04f2
	*poolp = NULL;
Packit Service ae04f2
}
Packit Service ae04f2
Packit Service ae04f2
void
Packit Service ae04f2
isc_taskpool_setprivilege(isc_taskpool_t *pool, bool priv) {
Packit Service ae04f2
	unsigned int i;
Packit Service ae04f2
Packit Service ae04f2
	REQUIRE(pool != NULL);
Packit Service ae04f2
Packit Service ae04f2
	for (i = 0; i < pool->ntasks; i++) {
Packit Service ae04f2
		if (pool->tasks[i] != NULL)
Packit Service ae04f2
			isc_task_setprivilege(pool->tasks[i], priv);
Packit Service ae04f2
	}
Packit Service ae04f2
}