<inttypes.h>: Small sync with FreeBSD.
[dragonfly.git] / sys / ddb / db_examine.c
blob01fd24cbf85e474704bc191fcf2d286c49ba7b1c
1 /*
2 * Mach Operating System
3 * Copyright (c) 1991,1990 Carnegie Mellon University
4 * All Rights Reserved.
6 * Permission to use, copy, modify and distribute this software and its
7 * documentation is hereby granted, provided that both the copyright
8 * notice and this permission notice appear in all copies of the
9 * software, derivative works or modified versions, and any portions
10 * thereof, and that both notices appear in supporting documentation.
12 * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS
13 * CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
14 * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
16 * Carnegie Mellon requests users of this software to return to
18 * Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU
19 * School of Computer Science
20 * Carnegie Mellon University
21 * Pittsburgh PA 15213-3890
23 * any improvements or extensions that they make and grant Carnegie the
24 * rights to redistribute these changes.
26 * $FreeBSD: src/sys/ddb/db_examine.c,v 1.27 1999/08/28 00:41:07 peter Exp $
30 * Author: David B. Golub, Carnegie Mellon University
31 * Date: 7/90
34 #include <sys/param.h>
35 #include <sys/systm.h>
37 #include <ddb/ddb.h>
39 #include <ddb/db_lex.h>
40 #include <ddb/db_output.h>
41 #include <ddb/db_command.h>
42 #include <ddb/db_sym.h>
43 #include <ddb/db_access.h>
45 static char db_examine_format[TOK_STRING_SIZE] = "x";
47 static void db_examine (db_addr_t, char *, int);
48 static void db_search (db_addr_t, int, db_expr_t, db_expr_t, u_int);
51 * Examine (print) data.
53 /*ARGSUSED*/
54 void
55 db_examine_cmd(db_expr_t addr, boolean_t have_addr, db_expr_t count,
56 char *modif)
58 if (modif[0] != '\0')
59 db_strcpy(db_examine_format, modif);
61 if (count == -1)
62 count = 1;
64 db_examine((db_addr_t) addr, db_examine_format, count);
68 * Parameters:
69 * fmt: format string
70 * count: repeat count
72 static void
73 db_examine(db_addr_t addr, char *fmt, int count)
75 int c;
76 db_expr_t value;
77 int size;
78 int width;
79 char * fp;
80 char tbuf[24];
82 while (--count >= 0) {
83 fp = fmt;
84 size = 4;
85 width = 16;
86 while ((c = *fp++) != 0) {
87 switch (c) {
88 case 'b':
89 size = 1;
90 width = 4;
91 break;
92 case 'h':
93 size = 2;
94 width = 8;
95 break;
96 case 'l':
97 size = 4;
98 width = 16;
99 break;
100 case 'g':
101 size = 8;
102 width = 32;
103 break;
104 default:
105 if (db_print_position() == 0) {
106 /* Print the address. */
107 db_printsym(addr, DB_STGY_ANY);
108 db_printf(":\t");
109 db_prev = addr;
112 switch (c) {
113 case 'a':
114 size = sizeof(void *);
115 value = db_get_value(addr, size, TRUE);
116 addr += size;
117 db_printsym(value, DB_STGY_ANY);
118 break;
119 case 'p':
120 size = sizeof(void *);
121 value = db_get_value(addr, size, TRUE);
122 addr += size;
123 db_printf("%p", (void *)value);
124 break;
125 case 'r': /* signed, current radix */
126 value = db_get_value(addr, size, TRUE);
127 addr += size;
128 db_format_radix(tbuf, 24, value, FALSE);
129 db_printf("%-*s", width, tbuf);
130 break;
131 case 'x': /* unsigned hex */
132 value = db_get_value(addr, size, FALSE);
133 addr += size;
134 db_printf("%-*lx", width, (long)value);
135 break;
136 case 'z': /* signed hex */
137 value = db_get_value(addr, size, TRUE);
138 addr += size;
139 db_format_hex(tbuf, 24, value, FALSE);
140 db_printf("%-*s", width, tbuf);
141 break;
142 case 'd': /* signed decimal */
143 value = db_get_value(addr, size, TRUE);
144 addr += size;
145 db_printf("%-*ld", width, (long)value);
146 break;
147 case 'u': /* unsigned decimal */
148 value = db_get_value(addr, size, FALSE);
149 addr += size;
150 db_printf("%-*lu", width, (long)value);
151 break;
152 case 'o': /* unsigned octal */
153 value = db_get_value(addr, size, FALSE);
154 addr += size;
155 db_printf("%-*lo", width, (long)value);
156 break;
157 case 'c': /* character */
158 value = db_get_value(addr, 1, FALSE);
159 addr += 1;
160 if (value >= ' ' && value <= '~')
161 db_printf("%c", (int)value);
162 else
163 db_printf("\\%03o", (int)value);
164 break;
165 case 's': /* null-terminated string */
166 for (;;) {
167 value = db_get_value(addr, 1, FALSE);
168 addr += 1;
169 if (value == 0)
170 break;
171 if (value >= ' ' && value <= '~')
172 db_printf("%c", (int)value);
173 else
174 db_printf("\\%03o", (int)value);
176 break;
177 case 'i': /* instruction */
178 addr = db_disasm(addr, FALSE, NULL);
179 break;
180 case 'I': /* instruction, alternate form */
181 addr = db_disasm(addr, TRUE, NULL);
182 break;
183 default:
184 break;
186 if (db_print_position() != 0 ||
187 c == 'a' || c == 'p') {
188 db_end_line(1);
190 break;
194 db_next = addr;
198 * Print value.
200 static char db_print_format = 'x';
202 /*ARGSUSED*/
203 void
204 db_print_cmd(db_expr_t addr, boolean_t have_addr, db_expr_t count, char *modif)
206 db_expr_t value;
208 if (modif[0] != '\0')
209 db_print_format = modif[0];
211 switch (db_print_format) {
212 case 'a':
213 db_printsym((db_addr_t)addr, DB_STGY_ANY);
214 break;
215 case 'r':
217 char tbuf[24];
219 db_format_radix(tbuf, 24, addr, FALSE);
220 db_printf("%11s", tbuf);
221 break;
223 case 'x':
224 db_printf("%8lx", (unsigned long)addr);
225 break;
226 case 'z':
228 char tbuf[24];
230 db_format_hex(tbuf, 24, addr, FALSE);
231 db_printf("%8s", tbuf);
232 break;
234 case 'd':
235 db_printf("%11ld", (long)addr);
236 break;
237 case 'u':
238 db_printf("%11lu", (unsigned long)addr);
239 break;
240 case 'o':
241 db_printf("%16lo", (unsigned long)addr);
242 break;
243 case 'c':
244 value = addr & 0xFF;
245 if (value >= ' ' && value <= '~')
246 db_printf("%c", (int)value);
247 else
248 db_printf("\\%03o", (int)value);
249 break;
251 db_printf("\n");
254 void
255 db_print_loc_and_inst(db_addr_t loc, db_regs_t *regs)
257 db_printsym(loc, DB_STGY_PROC);
258 db_printf(":\t");
259 db_disasm(loc, TRUE, regs);
263 * Search for a value in memory.
264 * Syntax: search [/bhl] addr value [mask] [,count]
266 void
267 db_search_cmd(db_expr_t dummy1, boolean_t dummy2, db_expr_t dummy3,
268 char *dummy4)
270 int t;
271 db_addr_t addr;
272 int size;
273 db_expr_t value;
274 db_expr_t mask;
275 db_expr_t count;
277 t = db_read_token();
278 if (t == tSLASH) {
279 t = db_read_token();
280 if (t != tIDENT) {
281 bad_modifier:
282 db_printf("Bad modifier\n");
283 db_flush_lex();
284 return;
287 if (!strcmp(db_tok_string, "b"))
288 size = 1;
289 else if (!strcmp(db_tok_string, "h"))
290 size = 2;
291 else if (!strcmp(db_tok_string, "l"))
292 size = 4;
293 else
294 goto bad_modifier;
295 } else {
296 db_unread_token(t);
297 size = 4;
300 if (!db_expression((db_expr_t *)&addr)) {
301 db_printf("Address missing\n");
302 db_flush_lex();
303 return;
306 if (!db_expression(&value)) {
307 db_printf("Value missing\n");
308 db_flush_lex();
309 return;
312 if (!db_expression(&mask))
313 mask = 0xffffffffUL;
315 t = db_read_token();
316 if (t == tCOMMA) {
317 if (!db_expression(&count)) {
318 db_printf("Count missing\n");
319 db_flush_lex();
320 return;
322 } else {
323 db_unread_token(t);
324 count = -1; /* effectively forever */
326 db_skip_to_eol();
328 db_search(addr, size, value, mask, count);
331 static void
332 db_search(db_addr_t addr, int size, db_expr_t value, db_expr_t mask,
333 unsigned int count)
335 while (count-- != 0) {
336 db_prev = addr;
337 if ((db_get_value(addr, size, FALSE) & mask) == value)
338 break;
339 addr += size;
341 db_next = addr;