1*ba2539a9Schs /* $NetBSD: dtrace_unload.c,v 1.7 2018/05/28 21:05:03 chs Exp $ */
201c9547eSdarran
3bb8023b5Sdarran /*
4bb8023b5Sdarran * CDDL HEADER START
5bb8023b5Sdarran *
6bb8023b5Sdarran * The contents of this file are subject to the terms of the
7bb8023b5Sdarran * Common Development and Distribution License (the "License").
8bb8023b5Sdarran * You may not use this file except in compliance with the License.
9bb8023b5Sdarran *
10bb8023b5Sdarran * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
11bb8023b5Sdarran * or http://www.opensolaris.org/os/licensing.
12bb8023b5Sdarran * See the License for the specific language governing permissions
13bb8023b5Sdarran * and limitations under the License.
14bb8023b5Sdarran *
15bb8023b5Sdarran * When distributing Covered Code, include this CDDL HEADER in each
16bb8023b5Sdarran * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
17bb8023b5Sdarran * If applicable, add the following below this CDDL HEADER, with the
18bb8023b5Sdarran * fields enclosed by brackets "[]" replaced with your own identifying
19bb8023b5Sdarran * information: Portions Copyright [yyyy] [name of copyright owner]
20bb8023b5Sdarran *
21bb8023b5Sdarran * CDDL HEADER END
22bb8023b5Sdarran *
23*ba2539a9Schs * $FreeBSD: head/sys/cddl/dev/dtrace/dtrace_unload.c 278166 2015-02-03 19:39:53Z pfg $
24bb8023b5Sdarran *
25bb8023b5Sdarran */
26bb8023b5Sdarran
27bb8023b5Sdarran static int
dtrace_unload()28bb8023b5Sdarran dtrace_unload()
29bb8023b5Sdarran {
30bb8023b5Sdarran dtrace_state_t *state;
31bb8023b5Sdarran int error = 0;
32bb8023b5Sdarran
33*ba2539a9Schs #ifdef __FreeBSD__
34*ba2539a9Schs destroy_dev(dtrace_dev);
35*ba2539a9Schs destroy_dev(helper_dev);
36*ba2539a9Schs #endif
37*ba2539a9Schs
38*ba2539a9Schs #ifdef __NetBSD__
39*ba2539a9Schs module_unregister_callbacks(dtrace_modcb);
40*ba2539a9Schs #endif
41*ba2539a9Schs
42bb8023b5Sdarran mutex_enter(&dtrace_provider_lock);
43bb8023b5Sdarran mutex_enter(&dtrace_lock);
44bb8023b5Sdarran mutex_enter(&cpu_lock);
45bb8023b5Sdarran
469ce3434fSozaki-r if (dtrace_opens > 0 || dtrace_helpers > 0) {
47bb8023b5Sdarran mutex_exit(&cpu_lock);
48bb8023b5Sdarran mutex_exit(&dtrace_lock);
49bb8023b5Sdarran mutex_exit(&dtrace_provider_lock);
50bb8023b5Sdarran return (EBUSY);
51bb8023b5Sdarran }
52bb8023b5Sdarran
53bb8023b5Sdarran if (dtrace_unregister((dtrace_provider_id_t)dtrace_provider) != 0) {
54bb8023b5Sdarran mutex_exit(&cpu_lock);
55bb8023b5Sdarran mutex_exit(&dtrace_lock);
56bb8023b5Sdarran mutex_exit(&dtrace_provider_lock);
57bb8023b5Sdarran return (EBUSY);
58bb8023b5Sdarran }
59bb8023b5Sdarran
60bb8023b5Sdarran dtrace_provider = NULL;
61*ba2539a9Schs #ifdef __FreeBSD__
62*ba2539a9Schs EVENTHANDLER_DEREGISTER(kld_load, dtrace_kld_load_tag);
63*ba2539a9Schs EVENTHANDLER_DEREGISTER(kld_unload_try, dtrace_kld_unload_try_tag);
64*ba2539a9Schs #endif
65bb8023b5Sdarran
66bb8023b5Sdarran if ((state = dtrace_anon_grab()) != NULL) {
67bb8023b5Sdarran /*
68bb8023b5Sdarran * If there were ECBs on this state, the provider should
69bb8023b5Sdarran * have not been allowed to detach; assert that there is
70bb8023b5Sdarran * none.
71bb8023b5Sdarran */
72bb8023b5Sdarran ASSERT(state->dts_necbs == 0);
73bb8023b5Sdarran dtrace_state_destroy(state);
74bb8023b5Sdarran }
75bb8023b5Sdarran
76bb8023b5Sdarran bzero(&dtrace_anon, sizeof (dtrace_anon_t));
77bb8023b5Sdarran
78bb8023b5Sdarran mutex_exit(&cpu_lock);
79bb8023b5Sdarran
80bb8023b5Sdarran if (dtrace_probes != NULL) {
81*ba2539a9Schs kmem_free(dtrace_probes, dtrace_nprobes * sizeof (dtrace_probe_t *));
82bb8023b5Sdarran dtrace_probes = NULL;
83bb8023b5Sdarran dtrace_nprobes = 0;
84bb8023b5Sdarran }
85bb8023b5Sdarran
86bb8023b5Sdarran dtrace_hash_destroy(dtrace_bymod);
87bb8023b5Sdarran dtrace_hash_destroy(dtrace_byfunc);
88bb8023b5Sdarran dtrace_hash_destroy(dtrace_byname);
89bb8023b5Sdarran dtrace_bymod = NULL;
90bb8023b5Sdarran dtrace_byfunc = NULL;
91bb8023b5Sdarran dtrace_byname = NULL;
92bb8023b5Sdarran
93bb8023b5Sdarran kmem_cache_destroy(dtrace_state_cache);
94bb8023b5Sdarran
95*ba2539a9Schs #ifdef __FreeBSD__
96*ba2539a9Schs delete_unrhdr(dtrace_arena);
97*ba2539a9Schs #endif
98*ba2539a9Schs #ifdef __NetBSD__
9901c9547eSdarran vmem_destroy(dtrace_arena);
100*ba2539a9Schs #endif
101bb8023b5Sdarran
102bb8023b5Sdarran if (dtrace_toxrange != NULL) {
103223b285dSyamt kmem_free(dtrace_toxrange,
104223b285dSyamt dtrace_toxranges_max * sizeof (dtrace_toxrange_t));
105bb8023b5Sdarran dtrace_toxrange = NULL;
106bb8023b5Sdarran dtrace_toxranges = 0;
107bb8023b5Sdarran dtrace_toxranges_max = 0;
108bb8023b5Sdarran }
109bb8023b5Sdarran
110bb8023b5Sdarran ASSERT(dtrace_vtime_references == 0);
111bb8023b5Sdarran ASSERT(dtrace_opens == 0);
112bb8023b5Sdarran ASSERT(dtrace_retained == NULL);
113bb8023b5Sdarran
114bb8023b5Sdarran mutex_exit(&dtrace_lock);
115bb8023b5Sdarran mutex_exit(&dtrace_provider_lock);
116bb8023b5Sdarran
117bb8023b5Sdarran mutex_destroy(&dtrace_meta_lock);
118bb8023b5Sdarran mutex_destroy(&dtrace_provider_lock);
119bb8023b5Sdarran mutex_destroy(&dtrace_lock);
120*ba2539a9Schs #ifdef DEBUG
121bb8023b5Sdarran mutex_destroy(&dtrace_errlock);
122*ba2539a9Schs #endif
123bb8023b5Sdarran
124*ba2539a9Schs #ifdef __FreeBSD__
125*ba2539a9Schs taskq_destroy(dtrace_taskq);
126*ba2539a9Schs #endif
127bb8023b5Sdarran
128bb8023b5Sdarran /* Reset our hook for exceptions. */
129bb8023b5Sdarran dtrace_invop_uninit();
130bb8023b5Sdarran
131bb8023b5Sdarran /*
132bb8023b5Sdarran * Reset our hook for thread switches, but ensure that vtime isn't
133bb8023b5Sdarran * active first.
134bb8023b5Sdarran */
135bb8023b5Sdarran dtrace_vtime_active = 0;
136bb8023b5Sdarran dtrace_vtime_switch_func = NULL;
137bb8023b5Sdarran
138bb8023b5Sdarran /* Unhook from the trap handler. */
139bb8023b5Sdarran dtrace_trap_func = NULL;
140bb8023b5Sdarran
141bb8023b5Sdarran return (error);
142bb8023b5Sdarran }
143