summaryrefslogtreecommitdiff
path: root/usr/src/uts/common/inet/cc/cc.c
diff options
context:
space:
mode:
Diffstat (limited to 'usr/src/uts/common/inet/cc/cc.c')
-rw-r--r--usr/src/uts/common/inet/cc/cc.c200
1 files changed, 200 insertions, 0 deletions
diff --git a/usr/src/uts/common/inet/cc/cc.c b/usr/src/uts/common/inet/cc/cc.c
new file mode 100644
index 0000000000..7bb213f74e
--- /dev/null
+++ b/usr/src/uts/common/inet/cc/cc.c
@@ -0,0 +1,200 @@
+/*
+ * Copyright (c) 2007-2008
+ * Swinburne University of Technology, Melbourne, Australia.
+ * Copyright (c) 2009-2010 Lawrence Stewart <lstewart@freebsd.org>
+ * Copyright (c) 2010 The FreeBSD Foundation
+ * All rights reserved.
+ * Copyright (c) 2017 by Delphix. All rights reserved.
+ *
+ * This software was developed at the Centre for Advanced Internet
+ * Architectures, Swinburne University of Technology, by Lawrence Stewart and
+ * James Healy, made possible in part by a grant from the Cisco University
+ * Research Program Fund at Community Foundation Silicon Valley.
+ *
+ * Portions of this software were developed at the Centre for Advanced
+ * Internet Architectures, Swinburne University of Technology, Melbourne,
+ * Australia by David Hayes under sponsorship from the FreeBSD Foundation.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ */
+
+/*
+ * This software was first released in 2007 by James Healy and Lawrence Stewart
+ * whilst working on the NewTCP research project at Swinburne University of
+ * Technology's Centre for Advanced Internet Architectures, Melbourne,
+ * Australia, which was made possible in part by a grant from the Cisco
+ * University Research Program Fund at Community Foundation Silicon Valley.
+ * More details are available at:
+ * http://caia.swin.edu.au/urp/newtcp/
+ */
+
+#include <sys/param.h>
+#include <sys/errno.h>
+#include <sys/systm.h>
+#include <sys/queue.h>
+#include <inet/cc.h>
+#include <inet/tcp.h>
+#include <sys/sdt.h>
+
+#define CC_KMODDIR "cc"
+
+/*
+ * List of available cc algorithms on the current system. Access is
+ * synchronized using cc_list_lock.
+ */
+static STAILQ_HEAD(cc_head, cc_algo) cc_list = STAILQ_HEAD_INITIALIZER(cc_list);
+static kmutex_t cc_list_lock;
+
+static struct modlmisc cc_modlmisc = {
+ &mod_miscops,
+ "Pluggable Congestion Control Framework"
+};
+
+static struct modlinkage cc_modlinkage = {
+ MODREV_1,
+ &cc_modlmisc,
+ NULL
+};
+
+/*
+ * Initialise CC subsystem on system boot.
+ */
+int
+_init(void)
+{
+ STAILQ_INIT(&cc_list);
+
+ return (mod_install(&cc_modlinkage));
+}
+
+int
+_fini(void)
+{
+ return (EBUSY);
+}
+
+int
+_info(struct modinfo *modinfop)
+{
+ return (mod_info(&cc_modlinkage, modinfop));
+}
+
+int
+cc_walk_algos(cc_walk_func_t *func, void *cd)
+{
+ struct cc_algo *algo;
+ int ret = 0;
+
+ mutex_enter(&cc_list_lock);
+ STAILQ_FOREACH(algo, &cc_list, entries) {
+ if ((ret = func(cd, algo)) != 0) {
+ break;
+ }
+ }
+ mutex_exit(&cc_list_lock);
+
+ return (ret);
+}
+
+/*
+ * Search for an algorithm of a given name, and return the corresponding set of
+ * operations. If there is no algorithm with the given name present, then this
+ * function returns NULL.
+ *
+ * Since this function is passed names from userland, it needs to be paranoid
+ * about the string, in case it's missing a terminating NUL character.
+ */
+struct cc_algo *
+cc_load_algo(const char *name)
+{
+ struct cc_algo *algo;
+ boolean_t found = B_FALSE;
+
+ if (strnlen(name, CC_ALGO_NAME_MAX) >= CC_ALGO_NAME_MAX) {
+ return (NULL);
+ }
+
+ mutex_enter(&cc_list_lock);
+ STAILQ_FOREACH(algo, &cc_list, entries) {
+ if (strncmp(algo->name, name, CC_ALGO_NAME_MAX) == 0) {
+ found = B_TRUE;
+ break;
+ }
+ }
+ mutex_exit(&cc_list_lock);
+
+ return (found ? algo : NULL);
+}
+
+/*
+ * Returns non-zero on success, 0 on failure.
+ */
+int
+cc_deregister_algo(struct cc_algo *remove_cc)
+{
+ struct cc_algo *funcs, *tmpfuncs;
+ int err = ENOENT;
+
+ mutex_enter(&cc_list_lock);
+ STAILQ_FOREACH_SAFE(funcs, &cc_list, entries, tmpfuncs) {
+ if (funcs == remove_cc) {
+ STAILQ_REMOVE(&cc_list, funcs, cc_algo, entries);
+ err = 0;
+ break;
+ }
+ }
+ mutex_exit(&cc_list_lock);
+ return (err);
+}
+
+/*
+ * Returns 0 on success, non-zero on failure.
+ */
+int
+cc_register_algo(struct cc_algo *add_cc)
+{
+ struct cc_algo *funcs;
+ size_t nlen;
+ int err = 0;
+
+ nlen = strnlen(add_cc->name, CC_ALGO_NAME_MAX);
+ if (nlen == 0 || nlen >= CC_ALGO_NAME_MAX) {
+ return (EINVAL);
+ }
+
+ /*
+ * Iterate over list of registered CC algorithms and make sure
+ * we're not trying to add a duplicate.
+ */
+ mutex_enter(&cc_list_lock);
+ STAILQ_FOREACH(funcs, &cc_list, entries) {
+ if (strncmp(funcs->name, add_cc->name, CC_ALGO_NAME_MAX) == 0)
+ err = EEXIST;
+ }
+
+ if (err == 0)
+ STAILQ_INSERT_TAIL(&cc_list, add_cc, entries);
+
+ mutex_exit(&cc_list_lock);
+
+ return (err);
+}