The Design and Implementation of the FreeBSD Operating System, Second Edition
Now available: The Design and Implementation of the FreeBSD Operating System (Second Edition)


[ source navigation ] [ diff markup ] [ identifier search ] [ freetext search ] [ file search ] [ list types ] [ track identifier ]

FreeBSD/Linux Kernel Cross Reference
sys/ddb/db_examine.c

Version: -  FREEBSD  -  FREEBSD-13-STABLE  -  FREEBSD-13-0  -  FREEBSD-12-STABLE  -  FREEBSD-12-0  -  FREEBSD-11-STABLE  -  FREEBSD-11-0  -  FREEBSD-10-STABLE  -  FREEBSD-10-0  -  FREEBSD-9-STABLE  -  FREEBSD-9-0  -  FREEBSD-8-STABLE  -  FREEBSD-8-0  -  FREEBSD-7-STABLE  -  FREEBSD-7-0  -  FREEBSD-6-STABLE  -  FREEBSD-6-0  -  FREEBSD-5-STABLE  -  FREEBSD-5-0  -  FREEBSD-4-STABLE  -  FREEBSD-3-STABLE  -  FREEBSD22  -  l41  -  OPENBSD  -  linux-2.6  -  MK84  -  PLAN9  -  xnu-8792 
SearchContext: -  none  -  3  -  10 

    1 /*-
    2  * Mach Operating System
    3  * Copyright (c) 1991,1990 Carnegie Mellon University
    4  * All Rights Reserved.
    5  *
    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.
   11  *
   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.
   15  *
   16  * Carnegie Mellon requests users of this software to return to
   17  *
   18  *  Software Distribution Coordinator  or  Software.Distribution@CS.CMU.EDU
   19  *  School of Computer Science
   20  *  Carnegie Mellon University
   21  *  Pittsburgh PA 15213-3890
   22  *
   23  * any improvements or extensions that they make and grant Carnegie the
   24  * rights to redistribute these changes.
   25  */
   26 /*
   27  *      Author: David B. Golub, Carnegie Mellon University
   28  *      Date:   7/90
   29  */
   30 
   31 #include <sys/cdefs.h>
   32 __FBSDID("$FreeBSD: releng/11.2/sys/ddb/db_examine.c 308418 2016-11-07 12:10:17Z kib $");
   33 
   34 #include <sys/param.h>
   35 #include <sys/systm.h>
   36 
   37 #include <ddb/ddb.h>
   38 
   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>
   44 
   45 static char     db_examine_format[TOK_STRING_SIZE] = "x";
   46 
   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);
   49 
   50 /*
   51  * Examine (print) data.
   52  */
   53 /*ARGSUSED*/
   54 void
   55 db_examine_cmd(db_expr_t addr, bool have_addr, db_expr_t count, char *modif)
   56 {
   57         if (modif[0] != '\0')
   58             db_strcpy(db_examine_format, modif);
   59 
   60         if (count == -1)
   61             count = 1;
   62 
   63         db_examine((db_addr_t) addr, db_examine_format, count);
   64 }
   65 
   66 static void
   67 db_examine(db_addr_t addr, char *fmt, int count)
   68 {
   69         int             c;
   70         db_expr_t       value;
   71         int             size;
   72         int             width;
   73         char *          fp;
   74 
   75         while (--count >= 0 && !db_pager_quit) {
   76             fp = fmt;
   77             size = 4;
   78             while ((c = *fp++) != 0) {
   79                 switch (c) {
   80                     case 'b':
   81                         size = 1;
   82                         break;
   83                     case 'h':
   84                         size = 2;
   85                         break;
   86                     case 'l':
   87                         size = 4;
   88                         break;
   89                     case 'g':
   90                         size = 8;
   91                         break;
   92                     case 'a':   /* address */
   93                         size = sizeof(void *);
   94                         /* always forces a new line */
   95                         if (db_print_position() != 0)
   96                             db_printf("\n");
   97                         db_prev = addr;
   98                         db_printsym(addr, DB_STGY_ANY);
   99                         db_printf(":\t");
  100                         break;
  101                     default:
  102                         if (db_print_position() == 0) {
  103                             /* Print the address. */
  104                             db_printsym(addr, DB_STGY_ANY);
  105                             db_printf(":\t");
  106                             db_prev = addr;
  107                         }
  108 
  109                         width = size * 4;
  110                         switch (c) {
  111                             case 'r':   /* signed, current radix */
  112                                 value = db_get_value(addr, size, true);
  113                                 addr += size;
  114                                 db_printf("%+-*lr", width, (long)value);
  115                                 break;
  116                             case 'x':   /* unsigned hex */
  117                                 value = db_get_value(addr, size, false);
  118                                 addr += size;
  119                                 db_printf("%-*lx", width, (long)value);
  120                                 break;
  121                             case 'z':   /* signed hex */
  122                                 value = db_get_value(addr, size, true);
  123                                 addr += size;
  124                                 db_printf("%-*ly", width, (long)value);
  125                                 break;
  126                             case 'd':   /* signed decimal */
  127                                 value = db_get_value(addr, size, true);
  128                                 addr += size;
  129                                 db_printf("%-*ld", width, (long)value);
  130                                 break;
  131                             case 'u':   /* unsigned decimal */
  132                                 value = db_get_value(addr, size, false);
  133                                 addr += size;
  134                                 db_printf("%-*lu", width, (long)value);
  135                                 break;
  136                             case 'o':   /* unsigned octal */
  137                                 value = db_get_value(addr, size, false);
  138                                 addr += size;
  139                                 db_printf("%-*lo", width, (long)value);
  140                                 break;
  141                             case 'c':   /* character */
  142                                 value = db_get_value(addr, 1, false);
  143                                 addr += 1;
  144                                 if (value >= ' ' && value <= '~')
  145                                     db_printf("%c", (int)value);
  146                                 else
  147                                     db_printf("\\%03o", (int)value);
  148                                 break;
  149                             case 's':   /* null-terminated string */
  150                                 for (;;) {
  151                                     value = db_get_value(addr, 1, false);
  152                                     addr += 1;
  153                                     if (value == 0)
  154                                         break;
  155                                     if (value >= ' ' && value <= '~')
  156                                         db_printf("%c", (int)value);
  157                                     else
  158                                         db_printf("\\%03o", (int)value);
  159                                 }
  160                                 break;
  161                             case 'S':   /* symbol */
  162                                 value = db_get_value(addr, sizeof(void *),
  163                                     false);
  164                                 addr += sizeof(void *);
  165                                 db_printsym(value, DB_STGY_ANY);
  166                                 break;
  167                             case 'i':   /* instruction */
  168                                 addr = db_disasm(addr, false);
  169                                 break;
  170                             case 'I':   /* instruction, alternate form */
  171                                 addr = db_disasm(addr, true);
  172                                 break;
  173                             default:
  174                                 break;
  175                         }
  176                         if (db_print_position() != 0)
  177                             db_end_line(1);
  178                         break;
  179                 }
  180             }
  181         }
  182         db_next = addr;
  183 }
  184 
  185 /*
  186  * Print value.
  187  */
  188 static char     db_print_format = 'x';
  189 
  190 /*ARGSUSED*/
  191 void
  192 db_print_cmd(db_expr_t addr, bool have_addr, db_expr_t count, char *modif)
  193 {
  194         db_expr_t       value;
  195 
  196         if (modif[0] != '\0')
  197             db_print_format = modif[0];
  198 
  199         switch (db_print_format) {
  200             case 'a':
  201                 db_printsym((db_addr_t)addr, DB_STGY_ANY);
  202                 break;
  203             case 'r':
  204                 db_printf("%+11lr", (long)addr);
  205                 break;
  206             case 'x':
  207                 db_printf("%8lx", (unsigned long)addr);
  208                 break;
  209             case 'z':
  210                 db_printf("%8ly", (long)addr);
  211                 break;
  212             case 'd':
  213                 db_printf("%11ld", (long)addr);
  214                 break;
  215             case 'u':
  216                 db_printf("%11lu", (unsigned long)addr);
  217                 break;
  218             case 'o':
  219                 db_printf("%16lo", (unsigned long)addr);
  220                 break;
  221             case 'c':
  222                 value = addr & 0xFF;
  223                 if (value >= ' ' && value <= '~')
  224                     db_printf("%c", (int)value);
  225                 else
  226                     db_printf("\\%03o", (int)value);
  227                 break;
  228             default:
  229                 db_print_format = 'x';
  230                 db_error("Syntax error: unsupported print modifier\n");
  231                 /*NOTREACHED*/
  232         }
  233         db_printf("\n");
  234 }
  235 
  236 void
  237 db_print_loc_and_inst(db_addr_t loc)
  238 {
  239         db_expr_t off;
  240 
  241         db_printsym(loc, DB_STGY_PROC);
  242         if (db_search_symbol(loc, DB_STGY_PROC, &off) != C_DB_SYM_NULL) {
  243                 db_printf(":\t");
  244                 (void)db_disasm(loc, false);
  245         }
  246 }
  247 
  248 /*
  249  * Search for a value in memory.
  250  * Syntax: search [/bhl] addr value [mask] [,count]
  251  */
  252 void
  253 db_search_cmd(db_expr_t dummy1, bool dummy2, db_expr_t dummy3, char *dummy4)
  254 {
  255         int             t;
  256         db_addr_t       addr;
  257         int             size;
  258         db_expr_t       value;
  259         db_expr_t       mask;
  260         db_expr_t       count;
  261 
  262         t = db_read_token();
  263         if (t == tSLASH) {
  264             t = db_read_token();
  265             if (t != tIDENT) {
  266               bad_modifier:
  267                 db_printf("Bad modifier\n");
  268                 db_flush_lex();
  269                 return;
  270             }
  271 
  272             if (!strcmp(db_tok_string, "b"))
  273                 size = 1;
  274             else if (!strcmp(db_tok_string, "h"))
  275                 size = 2;
  276             else if (!strcmp(db_tok_string, "l"))
  277                 size = 4;
  278             else
  279                 goto bad_modifier;
  280         } else {
  281             db_unread_token(t);
  282             size = 4;
  283         }
  284 
  285         if (!db_expression((db_expr_t *)&addr)) {
  286             db_printf("Address missing\n");
  287             db_flush_lex();
  288             return;
  289         }
  290 
  291         if (!db_expression(&value)) {
  292             db_printf("Value missing\n");
  293             db_flush_lex();
  294             return;
  295         }
  296 
  297         if (!db_expression(&mask))
  298             mask = 0xffffffffUL;
  299 
  300         t = db_read_token();
  301         if (t == tCOMMA) {
  302             if (!db_expression(&count)) {
  303                 db_printf("Count missing\n");
  304                 db_flush_lex();
  305                 return;
  306             }
  307         } else {
  308             db_unread_token(t);
  309             count = -1;         /* effectively forever */
  310         }
  311         db_skip_to_eol();
  312 
  313         db_search(addr, size, value, mask, count);
  314 }
  315 
  316 static void
  317 db_search(db_addr_t addr, int size, db_expr_t value, db_expr_t mask,
  318     unsigned int count)
  319 {
  320         while (count-- != 0) {
  321                 db_prev = addr;
  322                 if ((db_get_value(addr, size, false) & mask) == value)
  323                         break;
  324                 addr += size;
  325         }
  326         db_next = addr;
  327 }

Cache object: 2f3f278ba2e4dc2aa5268783b02de67b


[ source navigation ] [ diff markup ] [ identifier search ] [ freetext search ] [ file search ] [ list types ] [ track identifier ]


This page is part of the FreeBSD/Linux Linux Kernel Cross-Reference, and was automatically generated using a modified version of the LXR engine.