[BACK]Return to db_watch.c CVS log [TXT][DIR] Up to [local] / sys / ddb

Annotation of sys/ddb/db_watch.c, Revision 1.1.1.1

1.1       nbrk        1: /*     $OpenBSD: db_watch.c,v 1.9 2006/03/13 06:23:20 jsg Exp $ */
                      2: /*     $NetBSD: db_watch.c,v 1.9 1996/03/30 22:30:12 christos Exp $    */
                      3:
                      4: /*
                      5:  * Mach Operating System
                      6:  * Copyright (c) 1993,1992,1991,1990 Carnegie Mellon University
                      7:  * All Rights Reserved.
                      8:  *
                      9:  * Permission to use, copy, modify and distribute this software and its
                     10:  * documentation is hereby granted, provided that both the copyright
                     11:  * notice and this permission notice appear in all copies of the
                     12:  * software, derivative works or modified versions, and any portions
                     13:  * thereof, and that both notices appear in supporting documentation.
                     14:  *
                     15:  * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"
                     16:  * CONDITION.  CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
                     17:  * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
                     18:  *
                     19:  * Carnegie Mellon requests users of this software to return to
                     20:  *
                     21:  *  Software Distribution Coordinator  or  Software.Distribution@CS.CMU.EDU
                     22:  *  School of Computer Science
                     23:  *  Carnegie Mellon University
                     24:  *  Pittsburgh PA 15213-3890
                     25:  *
                     26:  * any improvements or extensions that they make and grant Carnegie Mellon
                     27:  * the rights to redistribute these changes.
                     28:  *
                     29:  *     Author: Richard P. Draves, Carnegie Mellon University
                     30:  *     Date:   10/90
                     31:  */
                     32:
                     33: #include <sys/param.h>
                     34: #include <sys/proc.h>
                     35:
                     36: #include <machine/db_machdep.h>
                     37:
                     38: #include <ddb/db_break.h>
                     39: #include <ddb/db_watch.h>
                     40: #include <ddb/db_lex.h>
                     41: #include <ddb/db_access.h>
                     42: #include <ddb/db_run.h>
                     43: #include <ddb/db_sym.h>
                     44: #include <ddb/db_output.h>
                     45: #include <ddb/db_command.h>
                     46: #include <ddb/db_extern.h>
                     47:
                     48: /*
                     49:  * Watchpoints.
                     50:  */
                     51:
                     52: boolean_t      db_watchpoints_inserted = TRUE;
                     53:
                     54: #define        NWATCHPOINTS    100
                     55: struct db_watchpoint   db_watch_table[NWATCHPOINTS];
                     56: db_watchpoint_t                db_next_free_watchpoint = &db_watch_table[0];
                     57: db_watchpoint_t                db_free_watchpoints = 0;
                     58: db_watchpoint_t                db_watchpoint_list = 0;
                     59:
                     60: db_watchpoint_t
                     61: db_watchpoint_alloc(void)
                     62: {
                     63:        db_watchpoint_t watch;
                     64:
                     65:        if ((watch = db_free_watchpoints) != 0) {
                     66:            db_free_watchpoints = watch->link;
                     67:            return (watch);
                     68:        }
                     69:        if (db_next_free_watchpoint == &db_watch_table[NWATCHPOINTS]) {
                     70:            db_printf("All watchpoints used.\n");
                     71:            return (0);
                     72:        }
                     73:        watch = db_next_free_watchpoint;
                     74:        db_next_free_watchpoint++;
                     75:
                     76:        return (watch);
                     77: }
                     78:
                     79: void
                     80: db_watchpoint_free(db_watchpoint_t watch)
                     81: {
                     82:        watch->link = db_free_watchpoints;
                     83:        db_free_watchpoints = watch;
                     84: }
                     85:
                     86: void
                     87: db_set_watchpoint(struct vm_map *map, db_addr_t addr, vsize_t size)
                     88: {
                     89:        db_watchpoint_t watch;
                     90:
                     91:        if (map == NULL) {
                     92:            db_printf("No map.\n");
                     93:            return;
                     94:        }
                     95:
                     96:        /*
                     97:         *      Should we do anything fancy with overlapping regions?
                     98:         */
                     99:
                    100:        for (watch = db_watchpoint_list;
                    101:             watch != 0;
                    102:             watch = watch->link)
                    103:            if (db_map_equal(watch->map, map) &&
                    104:                (watch->loaddr == addr) &&
                    105:                (watch->hiaddr == addr+size)) {
                    106:                db_printf("Already set.\n");
                    107:                return;
                    108:            }
                    109:
                    110:        watch = db_watchpoint_alloc();
                    111:        if (watch == 0) {
                    112:            db_printf("Too many watchpoints.\n");
                    113:            return;
                    114:        }
                    115:
                    116:        watch->map = map;
                    117:        watch->loaddr = addr;
                    118:        watch->hiaddr = addr+size;
                    119:
                    120:        watch->link = db_watchpoint_list;
                    121:        db_watchpoint_list = watch;
                    122:
                    123:        db_watchpoints_inserted = FALSE;
                    124: }
                    125:
                    126: void
                    127: db_delete_watchpoint(struct vm_map *map, db_addr_t addr)
                    128: {
                    129:        db_watchpoint_t watch;
                    130:        db_watchpoint_t *prev;
                    131:
                    132:        for (prev = &db_watchpoint_list;
                    133:             (watch = *prev) != 0;
                    134:             prev = &watch->link)
                    135:            if (db_map_equal(watch->map, map) &&
                    136:                (watch->loaddr <= addr) &&
                    137:                (addr < watch->hiaddr)) {
                    138:                *prev = watch->link;
                    139:                db_watchpoint_free(watch);
                    140:                return;
                    141:            }
                    142:
                    143:        db_printf("Not set.\n");
                    144: }
                    145:
                    146: void
                    147: db_list_watchpoints(void)
                    148: {
                    149:        db_watchpoint_t watch;
                    150:
                    151:        if (db_watchpoint_list == 0) {
                    152:            db_printf("No watchpoints set\n");
                    153:            return;
                    154:        }
                    155:
                    156:        db_printf(" Map        Address  Size\n");
                    157:        for (watch = db_watchpoint_list;
                    158:             watch != 0;
                    159:             watch = watch->link)
                    160:            db_printf("%s%p  %8lx  %lx\n",
                    161:                      db_map_current(watch->map) ? "*" : " ",
                    162:                      watch->map, watch->loaddr,
                    163:                      watch->hiaddr - watch->loaddr);
                    164: }
                    165:
                    166: /* Delete watchpoint */
                    167: /*ARGSUSED*/
                    168: void
                    169: db_deletewatch_cmd(db_expr_t addr, int have_addr, db_expr_t count, char *modif)
                    170: {
                    171:        db_delete_watchpoint(db_map_addr(addr), addr);
                    172: }
                    173:
                    174: /* Set watchpoint */
                    175: /*ARGSUSED*/
                    176: void
                    177: db_watchpoint_cmd(db_expr_t addr, int have_addr, db_expr_t count, char *modif)
                    178: {
                    179:        vsize_t         size;
                    180:        db_expr_t       value;
                    181:
                    182:        if (db_expression(&value))
                    183:            size = (vsize_t) value;
                    184:        else
                    185:            size = 4;
                    186:        db_skip_to_eol();
                    187:
                    188:        db_set_watchpoint(db_map_addr(addr), addr, size);
                    189: }
                    190:
                    191: /* list watchpoints */
                    192: /*ARGSUSED*/
                    193: void
                    194: db_listwatch_cmd(db_expr_t addr, int have_addr, db_expr_t count, char *modif)
                    195: {
                    196:        db_list_watchpoints();
                    197: }
                    198:
                    199: void
                    200: db_set_watchpoints(void)
                    201: {
                    202:        db_watchpoint_t watch;
                    203:
                    204:        if (!db_watchpoints_inserted && db_watchpoint_list != NULL) {
                    205:            for (watch = db_watchpoint_list;
                    206:                 watch != 0;
                    207:                 watch = watch->link)
                    208:                pmap_protect(watch->map->pmap,
                    209:                             trunc_page(watch->loaddr),
                    210:                             round_page(watch->hiaddr),
                    211:                             VM_PROT_READ);
                    212:            pmap_update(watch->map->pmap);
                    213:            db_watchpoints_inserted = TRUE;
                    214:        }
                    215: }
                    216:
                    217: void
                    218: db_clear_watchpoints(void)
                    219: {
                    220:        db_watchpoints_inserted = FALSE;
                    221: }
                    222:
                    223: boolean_t
                    224: db_find_watchpoint(struct vm_map *map, db_addr_t addr, db_regs_t *regs)
                    225: {
                    226:        db_watchpoint_t watch;
                    227:        db_watchpoint_t found = 0;
                    228:
                    229:        for (watch = db_watchpoint_list;
                    230:             watch != 0;
                    231:             watch = watch->link)
                    232:            if (db_map_equal(watch->map, map)) {
                    233:                if ((watch->loaddr <= addr) &&
                    234:                    (addr < watch->hiaddr))
                    235:                    return (TRUE);
                    236:                else if ((trunc_page(watch->loaddr) <= addr) &&
                    237:                         (addr < round_page(watch->hiaddr)))
                    238:                    found = watch;
                    239:            }
                    240:
                    241:        /*
                    242:         *      We didn't hit exactly on a watchpoint, but we are
                    243:         *      in a protected region.  We want to single-step
                    244:         *      and then re-protect.
                    245:         */
                    246:
                    247:        if (found) {
                    248:            db_watchpoints_inserted = FALSE;
                    249:            db_single_step(regs);
                    250:        }
                    251:
                    252:        return (FALSE);
                    253: }

CVSweb