db_variables.c revision 1.33 1 1.33 drochner /* $NetBSD: db_variables.c,v 1.33 2005/06/01 12:25:27 drochner Exp $ */
2 1.6 cgd
3 1.24 simonb /*
4 1.1 cgd * Mach Operating System
5 1.1 cgd * Copyright (c) 1991,1990 Carnegie Mellon University
6 1.1 cgd * All Rights Reserved.
7 1.24 simonb *
8 1.1 cgd * Permission to use, copy, modify and distribute this software and its
9 1.1 cgd * documentation is hereby granted, provided that both the copyright
10 1.1 cgd * notice and this permission notice appear in all copies of the
11 1.1 cgd * software, derivative works or modified versions, and any portions
12 1.1 cgd * thereof, and that both notices appear in supporting documentation.
13 1.24 simonb *
14 1.16 pk * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"
15 1.1 cgd * CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
16 1.1 cgd * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
17 1.24 simonb *
18 1.1 cgd * Carnegie Mellon requests users of this software to return to
19 1.24 simonb *
20 1.1 cgd * Software Distribution Coordinator or Software.Distribution (at) CS.CMU.EDU
21 1.1 cgd * School of Computer Science
22 1.1 cgd * Carnegie Mellon University
23 1.1 cgd * Pittsburgh PA 15213-3890
24 1.24 simonb *
25 1.1 cgd * any improvements or extensions that they make and grant Carnegie the
26 1.1 cgd * rights to redistribute these changes.
27 1.1 cgd */
28 1.22 lukem
29 1.22 lukem #include <sys/cdefs.h>
30 1.33 drochner __KERNEL_RCSID(0, "$NetBSD: db_variables.c,v 1.33 2005/06/01 12:25:27 drochner Exp $");
31 1.1 cgd
32 1.25 itohy #include "opt_ddbparam.h"
33 1.13 jonathan
34 1.5 mycroft #include <sys/param.h>
35 1.5 mycroft #include <sys/proc.h>
36 1.18 mrg #include <uvm/uvm_extern.h>
37 1.9 thorpej #include <sys/sysctl.h>
38 1.5 mycroft
39 1.1 cgd #include <machine/db_machdep.h>
40 1.1 cgd
41 1.13 jonathan #include <ddb/ddbvar.h>
42 1.13 jonathan
43 1.1 cgd #include <ddb/db_lex.h>
44 1.1 cgd #include <ddb/db_variables.h>
45 1.8 christos #include <ddb/db_command.h>
46 1.8 christos #include <ddb/db_sym.h>
47 1.8 christos #include <ddb/db_extern.h>
48 1.17 lukem #include <ddb/db_output.h>
49 1.13 jonathan
50 1.1 cgd
51 1.9 thorpej /*
52 1.9 thorpej * If this is non-zero, the DDB will be entered when the system
53 1.9 thorpej * panics. Initialize it so that it's patchable.
54 1.9 thorpej */
55 1.9 thorpej #ifndef DDB_ONPANIC
56 1.9 thorpej #define DDB_ONPANIC 1
57 1.9 thorpej #endif
58 1.9 thorpej int db_onpanic = DDB_ONPANIC;
59 1.1 cgd
60 1.14 jonathan /*
61 1.14 jonathan * Can DDB can be entered from the console?
62 1.14 jonathan */
63 1.14 jonathan #ifndef DDB_FROMCONSOLE
64 1.14 jonathan #define DDB_FROMCONSOLE 1
65 1.14 jonathan #endif
66 1.14 jonathan int db_fromconsole = DDB_FROMCONSOLE;
67 1.14 jonathan
68 1.32 reinoud /*
69 1.32 reinoud * Output DDB output to the message buffer?
70 1.32 reinoud */
71 1.32 reinoud #ifndef DDB_TEE_MSGBUF
72 1.32 reinoud #define DDB_TEE_MSGBUF 0
73 1.32 reinoud #endif
74 1.32 reinoud int db_tee_msgbuf = DDB_TEE_MSGBUF;
75 1.32 reinoud
76 1.32 reinoud
77 1.24 simonb static int db_rw_internal_variable(const struct db_variable *, db_expr_t *,
78 1.24 simonb int);
79 1.24 simonb static int db_find_variable(const struct db_variable **);
80 1.11 cgd
81 1.11 cgd /* XXX must all be ints for sysctl. */
82 1.20 jdolecek const struct db_variable db_vars[] = {
83 1.26 thorpej { "radix", (void *)&db_radix, db_rw_internal_variable },
84 1.26 thorpej { "maxoff", (void *)&db_maxoff, db_rw_internal_variable },
85 1.26 thorpej { "maxwidth", (void *)&db_max_width, db_rw_internal_variable },
86 1.26 thorpej { "tabstops", (void *)&db_tab_stop_width, db_rw_internal_variable },
87 1.26 thorpej { "lines", (void *)&db_max_line, db_rw_internal_variable },
88 1.26 thorpej { "onpanic", (void *)&db_onpanic, db_rw_internal_variable },
89 1.26 thorpej { "fromconsole", (void *)&db_fromconsole, db_rw_internal_variable },
90 1.32 reinoud { "tee_msgbuf", (void *)&db_tee_msgbuf, db_rw_internal_variable },
91 1.1 cgd };
92 1.20 jdolecek const struct db_variable * const db_evars = db_vars + sizeof(db_vars)/sizeof(db_vars[0]);
93 1.11 cgd
94 1.11 cgd /*
95 1.11 cgd * ddb command line access to the DDB variables defined above.
96 1.11 cgd */
97 1.11 cgd static int
98 1.24 simonb db_rw_internal_variable(const struct db_variable *vp, db_expr_t *valp, int rw)
99 1.11 cgd {
100 1.11 cgd
101 1.24 simonb if (rw == DB_VAR_GET)
102 1.11 cgd *valp = *(int *)vp->valuep;
103 1.24 simonb else
104 1.11 cgd *(int *)vp->valuep = *valp;
105 1.11 cgd return (0);
106 1.11 cgd }
107 1.9 thorpej
108 1.9 thorpej /*
109 1.9 thorpej * sysctl(3) access to the DDB variables defined above.
110 1.9 thorpej */
111 1.29 atatat SYSCTL_SETUP(sysctl_ddb_setup, "sysctl ddb subtree setup")
112 1.9 thorpej {
113 1.9 thorpej
114 1.30 atatat sysctl_createv(clog, 0, NULL, NULL,
115 1.30 atatat CTLFLAG_PERMANENT,
116 1.29 atatat CTLTYPE_NODE, "ddb", NULL,
117 1.29 atatat NULL, 0, NULL, 0,
118 1.29 atatat CTL_DDB, CTL_EOL);
119 1.30 atatat
120 1.30 atatat sysctl_createv(clog, 0, NULL, NULL,
121 1.30 atatat CTLFLAG_PERMANENT|CTLFLAG_READWRITE,
122 1.31 atatat CTLTYPE_INT, "radix",
123 1.31 atatat SYSCTL_DESCR("Input and output radix"),
124 1.29 atatat NULL, 0, &db_radix, 0,
125 1.29 atatat CTL_DDB, DDBCTL_RADIX, CTL_EOL);
126 1.30 atatat sysctl_createv(clog, 0, NULL, NULL,
127 1.30 atatat CTLFLAG_PERMANENT|CTLFLAG_READWRITE,
128 1.31 atatat CTLTYPE_INT, "maxoff",
129 1.31 atatat SYSCTL_DESCR("Maximum symbol offset"),
130 1.29 atatat NULL, 0, &db_maxoff, 0,
131 1.29 atatat CTL_DDB, DDBCTL_MAXOFF, CTL_EOL);
132 1.30 atatat sysctl_createv(clog, 0, NULL, NULL,
133 1.30 atatat CTLFLAG_PERMANENT|CTLFLAG_READWRITE,
134 1.31 atatat CTLTYPE_INT, "maxwidth",
135 1.31 atatat SYSCTL_DESCR("Maximum output line width"),
136 1.29 atatat NULL, 0, &db_max_width, 0,
137 1.29 atatat CTL_DDB, DDBCTL_MAXWIDTH, CTL_EOL);
138 1.30 atatat sysctl_createv(clog, 0, NULL, NULL,
139 1.30 atatat CTLFLAG_PERMANENT|CTLFLAG_READWRITE,
140 1.31 atatat CTLTYPE_INT, "lines",
141 1.31 atatat SYSCTL_DESCR("Number of display lines"),
142 1.29 atatat NULL, 0, &db_max_line, 0,
143 1.29 atatat CTL_DDB, DDBCTL_LINES, CTL_EOL);
144 1.30 atatat sysctl_createv(clog, 0, NULL, NULL,
145 1.30 atatat CTLFLAG_PERMANENT|CTLFLAG_READWRITE,
146 1.31 atatat CTLTYPE_INT, "tabstops",
147 1.31 atatat SYSCTL_DESCR("Output tab width"),
148 1.29 atatat NULL, 0, &db_tab_stop_width, 0,
149 1.29 atatat CTL_DDB, DDBCTL_TABSTOPS, CTL_EOL);
150 1.30 atatat sysctl_createv(clog, 0, NULL, NULL,
151 1.30 atatat CTLFLAG_PERMANENT|CTLFLAG_READWRITE,
152 1.31 atatat CTLTYPE_INT, "onpanic",
153 1.31 atatat SYSCTL_DESCR("Whether to enter ddb on a kernel panic"),
154 1.29 atatat NULL, 0, &db_onpanic, 0,
155 1.29 atatat CTL_DDB, DDBCTL_ONPANIC, CTL_EOL);
156 1.30 atatat sysctl_createv(clog, 0, NULL, NULL,
157 1.30 atatat CTLFLAG_PERMANENT|CTLFLAG_READWRITE,
158 1.31 atatat CTLTYPE_INT, "fromconsole",
159 1.31 atatat SYSCTL_DESCR("Whether ddb can be entered from the "
160 1.31 atatat "console"),
161 1.29 atatat NULL, 0, &db_fromconsole, 0,
162 1.29 atatat CTL_DDB, DDBCTL_FROMCONSOLE, CTL_EOL);
163 1.32 reinoud sysctl_createv(clog, 0, NULL, NULL,
164 1.32 reinoud CTLFLAG_PERMANENT|CTLFLAG_READWRITE,
165 1.32 reinoud CTLTYPE_INT, "tee_msgbuf",
166 1.32 reinoud SYSCTL_DESCR("Whether to tee ddb output to the msgbuf"),
167 1.32 reinoud NULL, 0, &db_tee_msgbuf, 0,
168 1.32 reinoud CTL_DDB, CTL_CREATE, CTL_EOL);
169 1.9 thorpej }
170 1.1 cgd
171 1.1 cgd int
172 1.24 simonb db_find_variable(const struct db_variable **varp)
173 1.1 cgd {
174 1.1 cgd int t;
175 1.20 jdolecek const struct db_variable *vp;
176 1.1 cgd
177 1.1 cgd t = db_read_token();
178 1.1 cgd if (t == tIDENT) {
179 1.24 simonb for (vp = db_vars; vp < db_evars; vp++) {
180 1.24 simonb if (!strcmp(db_tok_string, vp->name)) {
181 1.24 simonb *varp = vp;
182 1.24 simonb return (1);
183 1.24 simonb }
184 1.1 cgd }
185 1.24 simonb for (vp = db_regs; vp < db_eregs; vp++) {
186 1.24 simonb if (!strcmp(db_tok_string, vp->name)) {
187 1.24 simonb *varp = vp;
188 1.24 simonb return (1);
189 1.24 simonb }
190 1.1 cgd }
191 1.1 cgd }
192 1.1 cgd db_error("Unknown variable\n");
193 1.7 mycroft /*NOTREACHED*/
194 1.8 christos return 0;
195 1.1 cgd }
196 1.1 cgd
197 1.1 cgd int
198 1.24 simonb db_get_variable(db_expr_t *valuep)
199 1.1 cgd {
200 1.20 jdolecek const struct db_variable *vp;
201 1.1 cgd
202 1.1 cgd if (!db_find_variable(&vp))
203 1.24 simonb return (0);
204 1.1 cgd
205 1.1 cgd db_read_variable(vp, valuep);
206 1.1 cgd
207 1.1 cgd return (1);
208 1.1 cgd }
209 1.1 cgd
210 1.1 cgd int
211 1.24 simonb db_set_variable(db_expr_t value)
212 1.1 cgd {
213 1.20 jdolecek const struct db_variable *vp;
214 1.1 cgd
215 1.1 cgd if (!db_find_variable(&vp))
216 1.24 simonb return (0);
217 1.1 cgd
218 1.1 cgd db_write_variable(vp, &value);
219 1.1 cgd
220 1.1 cgd return (1);
221 1.1 cgd }
222 1.1 cgd
223 1.1 cgd
224 1.8 christos void
225 1.24 simonb db_read_variable(const struct db_variable *vp, db_expr_t *valuep)
226 1.1 cgd {
227 1.24 simonb int (*func)(const struct db_variable *, db_expr_t *, int) = vp->fcn;
228 1.1 cgd
229 1.1 cgd if (func == FCN_NULL)
230 1.24 simonb *valuep = *(vp->valuep);
231 1.1 cgd else
232 1.24 simonb (*func)(vp, valuep, DB_VAR_GET);
233 1.1 cgd }
234 1.1 cgd
235 1.8 christos void
236 1.24 simonb db_write_variable(const struct db_variable *vp, db_expr_t *valuep)
237 1.1 cgd {
238 1.24 simonb int (*func)(const struct db_variable *, db_expr_t *, int) = vp->fcn;
239 1.1 cgd
240 1.1 cgd if (func == FCN_NULL)
241 1.24 simonb *(vp->valuep) = *valuep;
242 1.1 cgd else
243 1.24 simonb (*func)(vp, valuep, DB_VAR_SET);
244 1.1 cgd }
245 1.1 cgd
246 1.8 christos /*ARGSUSED*/
247 1.1 cgd void
248 1.33 drochner db_set_cmd(db_expr_t addr, int have_addr, db_expr_t count, const char *modif)
249 1.1 cgd {
250 1.1 cgd db_expr_t value;
251 1.19 jhawk db_expr_t old_value;
252 1.20 jdolecek const struct db_variable *vp;
253 1.1 cgd int t;
254 1.1 cgd
255 1.1 cgd t = db_read_token();
256 1.1 cgd if (t != tDOLLAR) {
257 1.24 simonb db_error("Unknown variable\n");
258 1.24 simonb /*NOTREACHED*/
259 1.1 cgd }
260 1.1 cgd if (!db_find_variable(&vp)) {
261 1.24 simonb db_error("Unknown variable\n");
262 1.24 simonb /*NOTREACHED*/
263 1.1 cgd }
264 1.1 cgd
265 1.1 cgd t = db_read_token();
266 1.1 cgd if (t != tEQ)
267 1.24 simonb db_unread_token(t);
268 1.1 cgd
269 1.1 cgd if (!db_expression(&value)) {
270 1.24 simonb db_error("No value\n");
271 1.24 simonb /*NOTREACHED*/
272 1.1 cgd }
273 1.1 cgd if (db_read_token() != tEOL) {
274 1.24 simonb db_error("?\n");
275 1.24 simonb /*NOTREACHED*/
276 1.1 cgd }
277 1.1 cgd
278 1.19 jhawk db_read_variable(vp, &old_value);
279 1.19 jhawk db_printf("$%s\t\t%s = ", vp->name, db_num_to_str(old_value));
280 1.19 jhawk db_printf("%s\n", db_num_to_str(value));
281 1.1 cgd db_write_variable(vp, &value);
282 1.1 cgd }
283