Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/opencrypto Re-work the module init and destroy code to a...
details: https://anonhg.NetBSD.org/src/rev/711ec4687be3
branches: trunk
changeset: 812035:711ec4687be3
user: pgoyette <pgoyette%NetBSD.org@localhost>
date: Sat Nov 28 03:06:45 2015 +0000
description:
Re-work the module init and destroy code to allow it to be unloaded and
then reloaded.
Should fix PR kern/49842
diffstat:
sys/opencrypto/crypto.c | 87 +++++++++++++++++++++++++++++++++++++--------
sys/opencrypto/cryptodev.h | 4 +-
2 files changed, 72 insertions(+), 19 deletions(-)
diffs (192 lines):
diff -r a0fe2ddd00c1 -r 711ec4687be3 sys/opencrypto/crypto.c
--- a/sys/opencrypto/crypto.c Fri Nov 27 22:02:15 2015 +0000
+++ b/sys/opencrypto/crypto.c Sat Nov 28 03:06:45 2015 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: crypto.c,v 1.45 2014/02/25 18:30:12 pooka Exp $ */
+/* $NetBSD: crypto.c,v 1.46 2015/11/28 03:06:45 pgoyette Exp $ */
/* $FreeBSD: src/sys/opencrypto/crypto.c,v 1.4.2.5 2003/02/26 00:14:05 sam Exp $ */
/* $OpenBSD: crypto.c,v 1.41 2002/07/17 23:52:38 art Exp $ */
@@ -53,7 +53,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: crypto.c,v 1.45 2014/02/25 18:30:12 pooka Exp $");
+__KERNEL_RCSID(0, "$NetBSD: crypto.c,v 1.46 2015/11/28 03:06:45 pgoyette Exp $");
#include <sys/param.h>
#include <sys/reboot.h>
@@ -97,6 +97,7 @@
static struct cryptocap *crypto_drivers;
static int crypto_drivers_num;
static void *softintr_cookie;
+static int crypto_exit_flag;
/*
* There are two queues for crypto requests; one for symmetric (e.g.
@@ -236,7 +237,7 @@
static void cryptointr(void); /* swi thread to dispatch ops */
static void cryptoret(void); /* kernel thread for callbacks*/
static struct lwp *cryptothread;
-static void crypto_destroy(void);
+static int crypto_destroy(bool);
static int crypto_invoke(struct cryptop *crp, int hint);
static int crypto_kinvoke(struct cryptkop *krp, int hint);
@@ -269,7 +270,7 @@
sizeof(struct cryptocap), M_CRYPTO_DATA, M_NOWAIT | M_ZERO);
if (crypto_drivers == NULL) {
printf("crypto_init: cannot malloc driver table\n");
- return 0;
+ return ENOMEM;
}
crypto_drivers_num = CRYPTO_DRIVERS_INITIAL;
@@ -279,7 +280,7 @@
if (error) {
printf("crypto_init: cannot start cryptoret thread; error %d",
error);
- crypto_destroy();
+ return crypto_destroy(false);
}
#ifdef _MODULE
@@ -288,21 +289,60 @@
return 0;
}
-void
+int
crypto_init(void)
{
static ONCE_DECL(crypto_init_once);
- RUN_ONCE(&crypto_init_once, crypto_init0);
+ return RUN_ONCE(&crypto_init_once, crypto_init0);
}
-static void
-crypto_destroy(void)
+static int
+crypto_destroy(bool exit_kthread)
{
- /* XXX no wait to reclaim zones */
+ int i;
+
+ if (exit_kthread) {
+ mutex_spin_enter(&crypto_ret_q_mtx);
+
+ /* if we have any in-progress requests, don't unload */
+ if (!TAILQ_EMPTY(&crp_q) || !TAILQ_EMPTY(&crp_kq))
+ return EBUSY;
+
+ for (i = 0; i < crypto_drivers_num; i++)
+ if (crypto_drivers[i].cc_sessions != 0)
+ break;
+ if (i < crypto_drivers_num)
+ return EBUSY;
+
+ /* kick the cryptoret thread and wait for it to exit */
+ crypto_exit_flag = 1;
+ cv_signal(&cryptoret_cv);
+
+ while (crypto_exit_flag != 0)
+ cv_wait(&cryptoret_cv, &crypto_ret_q_mtx);
+ mutex_spin_exit(&crypto_ret_q_mtx);
+ }
+
+ if (sysctl_opencrypto_clog != NULL)
+ sysctl_teardown(&sysctl_opencrypto_clog);
+
+ unregister_swi(SWI_CRYPTO, cryptointr);
+
if (crypto_drivers != NULL)
free(crypto_drivers, M_CRYPTO_DATA);
- unregister_swi(SWI_CRYPTO, cryptointr);
+
+ pool_destroy(&cryptop_pool);
+ pool_destroy(&cryptodesc_pool);
+ pool_destroy(&cryptkop_pool);
+
+ cv_destroy(&cryptoret_cv);
+
+ mutex_destroy(&crypto_ret_q_mtx);
+ mutex_destroy(&crypto_q_mtx);
+ mutex_destroy(&crypto_mtx);
+
+ return 0;
}
/*
@@ -441,7 +481,7 @@
struct cryptocap *newdrv;
int i;
- crypto_init(); /* XXX oh, this is foul! */
+ (void)crypto_init(); /* XXX oh, this is foul! */
mutex_enter(&crypto_mtx);
for (i = 0; i < crypto_drivers_num; i++)
@@ -1306,6 +1346,17 @@
/* drop before calling any callbacks. */
if (crp == NULL && krp == NULL) {
+
+ /* Check for the exit condition. */
+ if (crypto_exit_flag != 0) {
+
+ /* Time to die. */
+ crypto_exit_flag = 0;
+ cv_broadcast(&cryptoret_cv);
+ mutex_spin_exit(&crypto_ret_q_mtx);
+ kthread_exit(0);
+ }
+
cryptostats.cs_rets++;
cv_wait(&cryptoret_cv, &crypto_ret_q_mtx);
continue;
@@ -1345,19 +1396,21 @@
static int
opencrypto_modcmd(modcmd_t cmd, void *opaque)
{
+ int error = 0;
switch (cmd) {
case MODULE_CMD_INIT:
#ifdef _MODULE
- crypto_init();
+ error = crypto_init();
#endif
- return 0;
+ break;
case MODULE_CMD_FINI:
#ifdef _MODULE
- sysctl_teardown(&sysctl_opencrypto_clog);
+ error = crypto_destroy(true);
#endif
- return 0;
+ break;
default:
- return ENOTTY;
+ error = ENOTTY;
}
+ return error;
}
diff -r a0fe2ddd00c1 -r 711ec4687be3 sys/opencrypto/cryptodev.h
--- a/sys/opencrypto/cryptodev.h Fri Nov 27 22:02:15 2015 +0000
+++ b/sys/opencrypto/cryptodev.h Sat Nov 28 03:06:45 2015 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: cryptodev.h,v 1.25 2011/06/09 14:41:24 drochner Exp $ */
+/* $NetBSD: cryptodev.h,v 1.26 2015/11/28 03:06:45 pgoyette Exp $ */
/* $FreeBSD: src/sys/opencrypto/cryptodev.h,v 1.2.2.6 2003/07/02 17:04:50 sam Exp $ */
/* $OpenBSD: cryptodev.h,v 1.33 2002/07/17 23:52:39 art Exp $ */
@@ -625,7 +625,7 @@
* (This declaration doesnt really belong here but there's no header
* for the raw framework.)
*/
-void crypto_init(void);
+int crypto_init(void);
/*
* Crypto-related utility routines used mainly by drivers.
Home |
Main Index |
Thread Index |
Old Index