xref: /freebsd/share/examples/kld/dyn_sysctl/dyn_sysctl.c (revision 74bf4e164ba5851606a27d4feff27717452583e5)
1bd3cdc31SAndrzej Bialecki /*-
2bd3cdc31SAndrzej Bialecki  * Copyright (c) 2000 Andrzej Bialecki <abial@freebsd.org>
3bd3cdc31SAndrzej Bialecki  * All rights reserved.
4bd3cdc31SAndrzej Bialecki  *
5bd3cdc31SAndrzej Bialecki  * Redistribution and use in source and binary forms, with or without
6bd3cdc31SAndrzej Bialecki  * modification, are permitted provided that the following conditions
7bd3cdc31SAndrzej Bialecki  * are met:
8bd3cdc31SAndrzej Bialecki  * 1. Redistributions of source code must retain the above copyright
9bd3cdc31SAndrzej Bialecki  *    notice, this list of conditions and the following disclaimer.
10bd3cdc31SAndrzej Bialecki  * 2. Redistributions in binary form must reproduce the above copyright
11bd3cdc31SAndrzej Bialecki  *    notice, this list of conditions and the following disclaimer in the
12bd3cdc31SAndrzej Bialecki  *    documentation and/or other materials provided with the distribution.
13bd3cdc31SAndrzej Bialecki  *
14bd3cdc31SAndrzej Bialecki  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15bd3cdc31SAndrzej Bialecki  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16bd3cdc31SAndrzej Bialecki  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17bd3cdc31SAndrzej Bialecki  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18bd3cdc31SAndrzej Bialecki  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19bd3cdc31SAndrzej Bialecki  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20bd3cdc31SAndrzej Bialecki  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21bd3cdc31SAndrzej Bialecki  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22bd3cdc31SAndrzej Bialecki  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23bd3cdc31SAndrzej Bialecki  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24bd3cdc31SAndrzej Bialecki  * SUCH DAMAGE.
25bd3cdc31SAndrzej Bialecki  *
26bd3cdc31SAndrzej Bialecki  *     $FreeBSD$
27bd3cdc31SAndrzej Bialecki  */
28bd3cdc31SAndrzej Bialecki 
29bd3cdc31SAndrzej Bialecki #include <sys/types.h>
30bd3cdc31SAndrzej Bialecki #include <sys/param.h>
31bd3cdc31SAndrzej Bialecki #include <sys/systm.h>
32bd3cdc31SAndrzej Bialecki #include <sys/module.h>
33bd3cdc31SAndrzej Bialecki #include <sys/sysctl.h>
34bd3cdc31SAndrzej Bialecki #include <sys/kernel.h>
35bd3cdc31SAndrzej Bialecki 
36bd3cdc31SAndrzej Bialecki 
37bd3cdc31SAndrzej Bialecki /* Some example data */
38bd3cdc31SAndrzej Bialecki static long a = 100;
39bd3cdc31SAndrzej Bialecki static int b = 200;
40bd3cdc31SAndrzej Bialecki static char *c = "hi there from dyn_sysctl";
41bd3cdc31SAndrzej Bialecki static struct sysctl_oid *a_root, *a_root1, *b_root;
42bd3cdc31SAndrzej Bialecki static struct sysctl_ctx_list clist, clist1, clist2;
43bd3cdc31SAndrzej Bialecki 
44bd3cdc31SAndrzej Bialecki static int
45bd3cdc31SAndrzej Bialecki sysctl_dyn_sysctl_test (SYSCTL_HANDLER_ARGS)
46bd3cdc31SAndrzej Bialecki {
47bd3cdc31SAndrzej Bialecki 	char *buf = "let's produce some text...";
48bd3cdc31SAndrzej Bialecki 
49bd3cdc31SAndrzej Bialecki 	return (sysctl_handle_string(oidp, buf, strlen(buf), req));
50bd3cdc31SAndrzej Bialecki }
51bd3cdc31SAndrzej Bialecki 
52bd3cdc31SAndrzej Bialecki /*
53bd3cdc31SAndrzej Bialecki  * The function called at load/unload.
54bd3cdc31SAndrzej Bialecki  */
55bd3cdc31SAndrzej Bialecki static int
56bd3cdc31SAndrzej Bialecki load (module_t mod, int cmd, void *arg)
57bd3cdc31SAndrzej Bialecki {
58bd3cdc31SAndrzej Bialecki 	int error;
59bd3cdc31SAndrzej Bialecki 
60bd3cdc31SAndrzej Bialecki 	error = 0;
61bd3cdc31SAndrzej Bialecki 	switch (cmd) {
62bd3cdc31SAndrzej Bialecki 	case MOD_LOAD :
63bd3cdc31SAndrzej Bialecki 		/* Initialize the contexts */
64bd3cdc31SAndrzej Bialecki 		printf("Initializing contexts and creating subtrees.\n\n");
65bd3cdc31SAndrzej Bialecki 		sysctl_ctx_init(&clist);
66bd3cdc31SAndrzej Bialecki 		sysctl_ctx_init(&clist1);
67bd3cdc31SAndrzej Bialecki 		sysctl_ctx_init(&clist2);
68bd3cdc31SAndrzej Bialecki 		/*
69bd3cdc31SAndrzej Bialecki 		 * Create two partially overlapping subtrees, belonging
70bd3cdc31SAndrzej Bialecki 		 * to different contexts.
71bd3cdc31SAndrzej Bialecki 		 */
72bd3cdc31SAndrzej Bialecki 		printf("TREE		ROOT		  NAME\n");
73bd3cdc31SAndrzej Bialecki 		a_root = SYSCTL_ADD_NODE(&clist,
74bd3cdc31SAndrzej Bialecki 			SYSCTL_STATIC_CHILDREN(/* top of sysctl tree */),
7504d94960SAndrzej Bialecki 			OID_AUTO, "dyn_sysctl", CTLFLAG_RW, 0,
76bd3cdc31SAndrzej Bialecki 			"dyn_sysctl root node");
77bd3cdc31SAndrzej Bialecki 		a_root = SYSCTL_ADD_NODE(&clist1,
78bd3cdc31SAndrzej Bialecki 			SYSCTL_STATIC_CHILDREN(/* top of sysctl tree */),
7904d94960SAndrzej Bialecki 			OID_AUTO, "dyn_sysctl", CTLFLAG_RW, 0,
80bd3cdc31SAndrzej Bialecki 			"dyn_sysctl root node");
81bd3cdc31SAndrzej Bialecki 		if(a_root == NULL) {
82bd3cdc31SAndrzej Bialecki 			printf("SYSCTL_ADD_NODE failed!\n");
83bd3cdc31SAndrzej Bialecki 			return (EINVAL);
84bd3cdc31SAndrzej Bialecki 		}
85bd3cdc31SAndrzej Bialecki 		SYSCTL_ADD_LONG(&clist, SYSCTL_CHILDREN(a_root),
8604d94960SAndrzej Bialecki 		 OID_AUTO, "long_a", CTLFLAG_RW, &a, "just to try");
87bd3cdc31SAndrzej Bialecki 		SYSCTL_ADD_INT(&clist, SYSCTL_CHILDREN(a_root),
8804d94960SAndrzej Bialecki 		 OID_AUTO, "int_b", CTLFLAG_RW, &b, 0, "just to try 1");
89bd3cdc31SAndrzej Bialecki 		a_root1=SYSCTL_ADD_NODE(&clist, SYSCTL_CHILDREN(a_root),
9004d94960SAndrzej Bialecki 		 OID_AUTO, "nextlevel", CTLFLAG_RD, 0, "one level down");
91bd3cdc31SAndrzej Bialecki 		SYSCTL_ADD_STRING(&clist, SYSCTL_CHILDREN(a_root1),
9204d94960SAndrzej Bialecki 		 OID_AUTO, "string_c", CTLFLAG_RD, c, 0, "just to try 2");
93bd3cdc31SAndrzej Bialecki 		printf("1. (%p)	/		  dyn_sysctl\n", &clist);
94bd3cdc31SAndrzej Bialecki 
95bd3cdc31SAndrzej Bialecki 		/* Add a subtree under already existing category */
96bd3cdc31SAndrzej Bialecki 		a_root1 = SYSCTL_ADD_NODE(&clist, SYSCTL_STATIC_CHILDREN(_kern),
9704d94960SAndrzej Bialecki 		 OID_AUTO, "dyn_sysctl", CTLFLAG_RW, 0, "dyn_sysctl root node");
98bd3cdc31SAndrzej Bialecki 		if(a_root1 == NULL) {
99bd3cdc31SAndrzej Bialecki 			printf("SYSCTL_ADD_NODE failed!\n");
100bd3cdc31SAndrzej Bialecki 			return (EINVAL);
101bd3cdc31SAndrzej Bialecki 		}
102bd3cdc31SAndrzej Bialecki 		SYSCTL_ADD_PROC(&clist, SYSCTL_CHILDREN(a_root1),
10304d94960SAndrzej Bialecki 		 OID_AUTO, "procedure", CTLFLAG_RD, 0, 0,
104bd3cdc31SAndrzej Bialecki 		 sysctl_dyn_sysctl_test, "A", "I can be here, too");
105bd3cdc31SAndrzej Bialecki 		printf("   (%p)	/kern		  dyn_sysctl\n", &clist);
106bd3cdc31SAndrzej Bialecki 
107bd3cdc31SAndrzej Bialecki 		/* Overlap second tree with the first. */
108bd3cdc31SAndrzej Bialecki 		b_root = SYSCTL_ADD_NODE(&clist1, SYSCTL_CHILDREN(a_root),
10904d94960SAndrzej Bialecki 		 OID_AUTO, "nextlevel", CTLFLAG_RD, 0, "one level down");
110bd3cdc31SAndrzej Bialecki 		SYSCTL_ADD_STRING(&clist1, SYSCTL_CHILDREN(b_root),
11104d94960SAndrzej Bialecki 		 OID_AUTO, "string_c1", CTLFLAG_RD, c, 0, "just to try 2");
112bd3cdc31SAndrzej Bialecki 		printf("2. (%p)	/		  dyn_sysctl	(overlapping #1)\n", &clist1);
113bd3cdc31SAndrzej Bialecki 
114bd3cdc31SAndrzej Bialecki 		/*
115bd3cdc31SAndrzej Bialecki 		 * And now do something stupid. Connect another subtree to
116bd3cdc31SAndrzej Bialecki 		 * dynamic oid.
117bd3cdc31SAndrzej Bialecki 		 * WARNING: this is an example of WRONG use of dynamic sysctls.
118bd3cdc31SAndrzej Bialecki 		 */
119bd3cdc31SAndrzej Bialecki 		b_root=SYSCTL_ADD_NODE(&clist2, SYSCTL_CHILDREN(a_root1),
12004d94960SAndrzej Bialecki 		 OID_AUTO, "bad", CTLFLAG_RW, 0, "dependent node");
121bd3cdc31SAndrzej Bialecki 		SYSCTL_ADD_STRING(&clist2, SYSCTL_CHILDREN(b_root),
12204d94960SAndrzej Bialecki 		 OID_AUTO, "string_c", CTLFLAG_RD, c, 0, "shouldn't panic");
123bd3cdc31SAndrzej Bialecki 		printf("3. (%p)	/kern/dyn_sysctl  bad		(WRONG!)\n", &clist2);
124bd3cdc31SAndrzej Bialecki 		break;
125bd3cdc31SAndrzej Bialecki 	case MOD_UNLOAD :
126bd3cdc31SAndrzej Bialecki 		printf("1. Try to free ctx1 (%p): ", &clist);
127bd3cdc31SAndrzej Bialecki 		if(sysctl_ctx_free(&clist))
128bd3cdc31SAndrzej Bialecki 			printf("failed: expected. Need to remove ctx3 first.\n");
129bd3cdc31SAndrzej Bialecki 		else
130bd3cdc31SAndrzej Bialecki 			printf("HELP! sysctl_ctx_free(%p) succeeded. EXPECT PANIC!!!\n", &clist);
131bd3cdc31SAndrzej Bialecki 		printf("2. Try to free ctx3 (%p): ", &clist2);
132bd3cdc31SAndrzej Bialecki 		if(sysctl_ctx_free(&clist2)) {
133bd3cdc31SAndrzej Bialecki 			printf("sysctl_ctx_free(%p) failed!\n", &clist2);
134bd3cdc31SAndrzej Bialecki 			/* Remove subtree forcefully... */
135bd3cdc31SAndrzej Bialecki 			sysctl_remove_oid(b_root, 1, 1);
136bd3cdc31SAndrzej Bialecki 			printf("sysctl_remove_oid(%p) succeeded\n", b_root);
137bd3cdc31SAndrzej Bialecki 		} else
138bd3cdc31SAndrzej Bialecki 			printf("Ok\n");
139bd3cdc31SAndrzej Bialecki 		printf("3. Try to free ctx1 (%p) again: ", &clist);
140bd3cdc31SAndrzej Bialecki 		if(sysctl_ctx_free(&clist)) {
141bd3cdc31SAndrzej Bialecki 			printf("sysctl_ctx_free(%p) failed!\n", &clist);
142bd3cdc31SAndrzej Bialecki 			/* Remove subtree forcefully... */
143bd3cdc31SAndrzej Bialecki 			sysctl_remove_oid(a_root1, 1, 1);
144bd3cdc31SAndrzej Bialecki 			printf("sysctl_remove_oid(%p) succeeded\n", a_root1);
145bd3cdc31SAndrzej Bialecki 		} else
146bd3cdc31SAndrzej Bialecki 			printf("Ok\n");
147bd3cdc31SAndrzej Bialecki 		printf("4. Try to free ctx2 (%p): ", &clist1);
148bd3cdc31SAndrzej Bialecki 		if(sysctl_ctx_free(&clist1)) {
149bd3cdc31SAndrzej Bialecki 			printf("sysctl_ctx_free(%p) failed!\n", &clist1);
150bd3cdc31SAndrzej Bialecki 			/* Remove subtree forcefully... */
151bd3cdc31SAndrzej Bialecki 			sysctl_remove_oid(a_root, 1, 1);
152bd3cdc31SAndrzej Bialecki 		} else
153bd3cdc31SAndrzej Bialecki 			printf("Ok\n");
154bd3cdc31SAndrzej Bialecki 		break;
155bd3cdc31SAndrzej Bialecki 	default :
15674bf4e16SPawel Jakub Dawidek 		error = EOPNOTSUPP;
157bd3cdc31SAndrzej Bialecki 		break;
158bd3cdc31SAndrzej Bialecki 	}
159bd3cdc31SAndrzej Bialecki 	return error;
160bd3cdc31SAndrzej Bialecki }
161bd3cdc31SAndrzej Bialecki 
162bd3cdc31SAndrzej Bialecki static moduledata_t mod_data= {
163bd3cdc31SAndrzej Bialecki 	"dyn_sysctl",
164bd3cdc31SAndrzej Bialecki 	load,
165bd3cdc31SAndrzej Bialecki 	0
166bd3cdc31SAndrzej Bialecki };
167bd3cdc31SAndrzej Bialecki 
168bd3cdc31SAndrzej Bialecki DECLARE_MODULE(dyn_sysctl, mod_data, SI_SUB_EXEC, SI_ORDER_ANY);
169