1 /* 2 * CDDL HEADER START 3 * 4 * The contents of this file are subject to the terms of the 5 * Common Development and Distribution License (the "License"). 6 * You may not use this file except in compliance with the License. 7 * 8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 9 * or http://www.opensolaris.org/os/licensing. 10 * See the License for the specific language governing permissions 11 * and limitations under the License. 12 * 13 * When distributing Covered Code, include this CDDL HEADER in each 14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 15 * If applicable, add the following below this CDDL HEADER, with the 16 * fields enclosed by brackets "[]" replaced with your own identifying 17 * information: Portions Copyright [yyyy] [name of copyright owner] 18 * 19 * CDDL HEADER END 20 */ 21 22 /* 23 * Copyright (c) 1995, 2010, Oracle and/or its affiliates. All rights reserved. 24 */ 25 26 #include <stdio.h> 27 #include <stdlib.h> 28 #include <unistd.h> 29 #include <sys/uio.h> 30 #include <fcntl.h> 31 #include <string.h> 32 #include <errno.h> 33 #include <sys/types.h> 34 #include <sys/signal.h> 35 #include <sys/fault.h> 36 #include <sys/syscall.h> 37 #include <procfs.h> 38 #include <sys/auxv.h> 39 #include <libelf.h> 40 #include <sys/param.h> 41 #include <sys/machelf.h> 42 #include <stdarg.h> 43 44 #include "rdb.h" 45 46 static const char *fault_strings[] = { 47 "<null string>", 48 "illegal instruction", 49 "privileged instruction", 50 "breakpoint instruction", 51 "trace trap (single-step)", 52 "Memory access (e.g., alignment)", 53 "Memory bounds (invalid address)", 54 "Integer overflow", 55 "Integer zero divide" 56 "Floating-point exception", 57 "Irrecoverable stack faul", 58 "Recoverable page fault (no associated sig)" 59 }; 60 61 #define MAXFAULT FLTPAGE 62 63 retc_t 64 set_breakpoint(struct ps_prochandle *ph, ulong_t addr, unsigned flags) 65 { 66 bptlist_t *new, *cur, *prev; 67 68 for (cur = ph->pp_breakpoints, prev = NULL; 69 (cur && (cur->bl_addr < addr)); 70 prev = cur, cur = cur->bl_next) 71 ; 72 if (cur && (cur->bl_addr == addr)) { 73 /* 74 * already have break point set here. 75 */ 76 cur->bl_flags |= flags; 77 return (RET_OK); 78 } 79 80 new = malloc(sizeof (bptlist_t)); 81 new->bl_addr = addr; 82 new->bl_flags = flags; 83 if (prev == NULL) { 84 /* 85 * insert at head 86 */ 87 new->bl_next = ph->pp_breakpoints; 88 ph->pp_breakpoints = new; 89 return (RET_OK); 90 } 91 92 prev->bl_next = new; 93 new->bl_next = cur; 94 return (RET_OK); 95 } 96 97 static bptlist_t * 98 find_bp(struct ps_prochandle *ph, ulong_t addr) 99 { 100 bptlist_t *cur; 101 102 for (cur = ph->pp_breakpoints; 103 (cur && (cur->bl_addr != addr)); 104 cur = cur->bl_next) 105 ; 106 107 if ((cur == NULL) || (cur->bl_addr != addr)) 108 return ((bptlist_t *)-1); 109 return (cur); 110 } 111 112 static retc_t 113 delete_bp(struct ps_prochandle *ph, ulong_t addr) 114 { 115 bptlist_t *cur, *prev; 116 117 for (cur = ph->pp_breakpoints, prev = NULL; 118 (cur && (cur->bl_addr < addr)); 119 prev = cur, cur = cur->bl_next) 120 ; 121 if ((cur == NULL) || (cur->bl_addr != addr)) 122 return (RET_FAILED); 123 124 if (prev == NULL) 125 ph->pp_breakpoints = cur->bl_next; 126 else 127 prev->bl_next = cur->bl_next; 128 129 free(cur); 130 return (RET_OK); 131 } 132 133 void 134 list_breakpoints(struct ps_prochandle *ph) 135 { 136 bptlist_t *cur; 137 138 if (ph->pp_breakpoints == NULL) { 139 (void) printf("no active breakpoints.\n"); 140 return; 141 } 142 143 (void) printf("active breakpoints:\n"); 144 for (cur = ph->pp_breakpoints; cur; cur = cur->bl_next) { 145 (void) printf("\t0x%08lx:0x%04x - %s\n", cur->bl_addr, 146 cur->bl_flags, print_address_ps(ph, cur->bl_addr, 147 FLG_PAP_SONAME)); 148 } 149 } 150 151 static void 152 set_breaks(struct ps_prochandle *ph) 153 { 154 bptlist_t *cur; 155 bptinstr_t bpt_instr = BPINSTR; 156 157 for (cur = ph->pp_breakpoints; cur; cur = cur->bl_next) { 158 bptinstr_t old_inst = 0; 159 160 if (ps_pread(ph, cur->bl_addr, (char *)&old_inst, 161 sizeof (bptinstr_t)) != PS_OK) 162 perr("sb: error setting breakpoint"); 163 164 cur->bl_instr = old_inst; 165 166 if (ps_pwrite(ph, cur->bl_addr, (char *)&bpt_instr, 167 sizeof (bptinstr_t)) != PS_OK) 168 perr("sb1: error setting breakpoint\n"); 169 } 170 171 } 172 173 static void 174 clear_breaks(struct ps_prochandle *ph) 175 { 176 bptlist_t *cur; 177 178 /* 179 * Restore all the original instructions 180 */ 181 for (cur = ph->pp_breakpoints; cur; cur = cur->bl_next) 182 if (ps_pwrite(ph, cur->bl_addr, (char *)&(cur->bl_instr), 183 sizeof (bptinstr_t)) != PS_OK) 184 perr("cb: error clearing breakpoint"); 185 } 186 187 retc_t 188 delete_all_breakpoints(struct ps_prochandle *ph) 189 { 190 bptlist_t *cur, *prev; 191 192 if (ph->pp_breakpoints == NULL) 193 return (RET_OK); 194 195 for (prev = NULL, cur = ph->pp_breakpoints; 196 cur; prev = cur, cur = cur->bl_next) 197 if (prev) 198 free(prev); 199 if (prev) 200 free(prev); 201 202 ph->pp_breakpoints = NULL; 203 return (RET_OK); 204 } 205 206 retc_t 207 delete_breakpoint(struct ps_prochandle *ph, ulong_t addr, unsigned flags) 208 { 209 bptlist_t *bpt; 210 211 if (((bpt = find_bp(ph, addr)) == (bptlist_t *)-1) || 212 ((bpt->bl_flags & flags) == 0)) 213 return (RET_FAILED); 214 215 bpt->bl_flags &= ~flags; 216 if (bpt->bl_flags) 217 return (RET_OK); 218 219 return (delete_bp(ph, addr)); 220 } 221 222 static void 223 handle_sp_break(struct ps_prochandle *ph) 224 { 225 rd_event_msg_t emt; 226 227 if (rd_event_getmsg(ph->pp_rap, &emt) != RD_OK) { 228 (void) fprintf(stderr, "hsb: failed rd_event_getmsg()\n"); 229 return; 230 } 231 232 if (emt.type == RD_DLACTIVITY) { 233 if (emt.u.state == RD_CONSISTENT) 234 ph->pp_flags |= FLG_PP_LMAPS; 235 else 236 ph->pp_flags &= ~FLG_PP_LMAPS; 237 if ((rdb_flags & RDB_FL_EVENTS) == 0) 238 return; 239 240 (void) printf("dlactivity: state changed to: "); 241 switch (emt.u.state) { 242 case RD_CONSISTENT: 243 (void) printf("RD_CONSISTENT\n"); 244 break; 245 case RD_ADD: 246 (void) printf("RD_ADD\n"); 247 break; 248 case RD_DELETE: 249 (void) printf("RD_DELETE\n"); 250 break; 251 default: 252 (void) printf("unknown: 0x%x\n", emt.u.state); 253 } 254 return; 255 } 256 257 if ((rdb_flags & RDB_FL_EVENTS) == 0) 258 return; 259 260 if (emt.type == RD_PREINIT) { 261 (void) printf("preinit reached\n"); 262 return; 263 } 264 265 if (emt.type == RD_POSTINIT) 266 (void) printf("postinit reached\n"); 267 } 268 269 unsigned 270 continue_to_break(struct ps_prochandle *ph) 271 { 272 bptlist_t *bpt; 273 pstatus_t pstatus; 274 struct iovec piov[5]; 275 long oper1, oper2, oper3, pflags = 0; 276 fltset_t faults; 277 278 /* 279 * We step by the first instruction incase their was 280 * a break-point there. 281 */ 282 (void) step_n(ph, 1, FLG_SN_NONE); 283 284 premptyset(&faults); 285 praddset(&faults, FLTBPT); 286 praddset(&faults, FLTILL); 287 praddset(&faults, FLTPRIV); 288 praddset(&faults, FLTACCESS); 289 praddset(&faults, FLTBOUNDS); 290 praddset(&faults, FLTIZDIV); 291 praddset(&faults, FLTSTACK); 292 praddset(&faults, FLTTRACE); 293 294 295 /* LINTED CONSTANT */ 296 while (1) { 297 set_breaks(ph); 298 oper1 = PCSFAULT; 299 piov[0].iov_base = (caddr_t)(&oper1); 300 piov[0].iov_len = sizeof (oper1); 301 302 piov[1].iov_base = (caddr_t)(&faults); 303 piov[1].iov_len = sizeof (faults); 304 305 oper2 = PCRUN; 306 piov[2].iov_base = (caddr_t)(&oper2); 307 piov[2].iov_len = sizeof (oper2); 308 pflags = PRCFAULT; 309 piov[3].iov_base = (caddr_t)(&pflags); 310 piov[3].iov_len = sizeof (pflags); 311 312 oper3 = PCWSTOP; 313 piov[4].iov_base = (caddr_t)(&oper3); 314 piov[4].iov_len = sizeof (oper3); 315 316 if (writev(ph->pp_ctlfd, piov, 5) == -1) { 317 if (errno == ENOENT) { 318 ph->pp_flags &= ~FLG_PP_PACT; 319 320 (void) ps_close(ph); 321 (void) printf("process terminated.\n"); 322 return (0); 323 } 324 perr("ctb: PCWSTOP"); 325 } 326 327 if (pread(ph->pp_statusfd, &pstatus, sizeof (pstatus), 0) == -1) 328 perr("ctb: reading status"); 329 330 331 if ((pstatus.pr_lwp.pr_why != PR_FAULTED) || 332 (pstatus.pr_lwp.pr_what != FLTBPT)) { 333 const char *fltmsg; 334 335 if ((pstatus.pr_lwp.pr_what <= MAXFAULT) && 336 (pstatus.pr_lwp.pr_why == PR_FAULTED)) 337 fltmsg = fault_strings[pstatus.pr_lwp.pr_what]; 338 else 339 fltmsg = "<unknown error>"; 340 341 (void) fprintf(stderr, "ctb: bad stop - stopped " 342 "on why: 0x%x what: %s(0x%x)\n", 343 pstatus.pr_lwp.pr_why, fltmsg, 344 pstatus.pr_lwp.pr_what); 345 return (0); 346 } 347 348 oper1 = PCCFAULT; 349 if (writev(ph->pp_ctlfd, piov, 1) == -1) 350 perr("ctb: PCCFAULT"); 351 352 if ((bpt = find_bp(ph, pstatus.pr_lwp.pr_reg[R_PC])) == 353 (bptlist_t *)-1) { 354 (void) fprintf(stderr, 355 "stopped at unregistered breakpoint! " 356 "addr: 0x%x\n", 357 EC_WORD(pstatus.pr_lwp.pr_reg[R_PC])); 358 break; 359 } 360 clear_breaks(ph); 361 362 /* 363 * If this was a BP at which we should stop 364 */ 365 if (bpt->bl_flags & MASK_BP_STOP) 366 break; 367 368 (void) step_n(ph, 1, FLG_SN_NONE); 369 } 370 371 if (bpt->bl_flags & FLG_BP_USERDEF) 372 (void) printf("break point reached at addr: 0x%x\n", 373 EC_WORD(pstatus.pr_lwp.pr_reg[R_PC])); 374 375 if (bpt->bl_flags & MASK_BP_SPECIAL) 376 handle_sp_break(ph); 377 378 if (ph->pp_flags & FLG_PP_LMAPS) { 379 if (get_linkmaps(ph) != RET_OK) 380 (void) fprintf(stderr, "problem loading linkmaps\n"); 381 } 382 383 return (bpt->bl_flags); 384 } 385 386 ulong_t 387 is_plt(struct ps_prochandle *ph, ulong_t pc) 388 { 389 map_info_t *mip; 390 ulong_t pltbase; 391 392 if ((mip = addr_to_map(ph, pc)) == (map_info_t *)0) 393 return ((ulong_t)0); 394 395 pltbase = mip->mi_pltbase; 396 if ((mip->mi_flags & FLG_MI_EXEC) == 0) 397 pltbase += mip->mi_addr; 398 399 if ((pc >= pltbase) && (pc <= (pltbase + mip->mi_pltsize))) 400 return (pltbase); 401 402 return ((ulong_t)0); 403 } 404 405 retc_t 406 step_n(struct ps_prochandle *ph, size_t count, sn_flags_e flgs) 407 { 408 pstatus_t pstatus; 409 fltset_t faults; 410 int i; 411 long oper; 412 long flags; 413 struct iovec piov[2]; 414 415 if (pread(ph->pp_statusfd, &pstatus, sizeof (pstatus), 0) == -1) 416 perr("stn: reading status"); 417 418 piov[0].iov_base = (caddr_t)(&oper); 419 piov[0].iov_len = sizeof (oper); 420 421 premptyset(&faults); 422 praddset(&faults, FLTTRACE); 423 424 flags = PRSTEP | PRCFAULT; 425 426 for (i = 0; i < count; i++) { 427 bptlist_t *bpt; 428 uintptr_t pc, pltbase; 429 430 pc = pstatus.pr_lwp.pr_reg[R_PC]; 431 432 if ((bpt = find_bp(ph, pc)) != (bptlist_t *)-1) { 433 if (bpt->bl_flags & MASK_BP_SPECIAL) 434 handle_sp_break(ph); 435 } 436 437 if (flgs & FLG_SN_VERBOSE) 438 disasm(ph, 1); 439 440 oper = PCSFAULT; 441 piov[1].iov_base = (caddr_t)(&faults); 442 piov[1].iov_len = sizeof (faults); 443 444 if (writev(ph->pp_ctlfd, piov, 2) == -1) 445 perr("stn: PCSFAULT"); 446 447 oper = PCRUN; 448 piov[1].iov_base = (caddr_t)(&flags); 449 piov[1].iov_len = sizeof (flags); 450 if (writev(ph->pp_ctlfd, piov, 2) == -1) 451 perr("stn: PCRUN(PRSETP)"); 452 453 oper = PCWSTOP; 454 if (writev(ph->pp_ctlfd, piov, 1) == -1) 455 perr("stn: PCWSTOP stepping"); 456 457 if (pread(ph->pp_statusfd, &pstatus, sizeof (pstatus), 0) == -1) 458 perr("stn1: reading status"); 459 pc = pstatus.pr_lwp.pr_reg[R_PC]; 460 461 462 if ((pstatus.pr_lwp.pr_why != PR_FAULTED) || 463 (pstatus.pr_lwp.pr_what != FLTTRACE)) { 464 (void) fprintf(stderr, "sn: bad stop - stopped on " 465 "why: 0x%x what: 0x%x\n", pstatus.pr_lwp.pr_why, 466 pstatus.pr_lwp.pr_what); 467 return (RET_FAILED); 468 } 469 470 if ((flgs & FLG_SN_PLTSKIP) && 471 ((pltbase = is_plt(ph, pc)) != (ulong_t)0)) { 472 rd_plt_info_t rp; 473 if (rd_plt_resolution(ph->pp_rap, pc, 474 pstatus.pr_lwp.pr_lwpid, pltbase, &rp) != RD_OK) { 475 (void) fprintf(stderr, 476 "sn: rd_plt_resolution failed\n"); 477 return (RET_FAILED); 478 } 479 if (rp.pi_skip_method == RD_RESOLVE_TARGET_STEP) { 480 unsigned bpflags; 481 482 (void) set_breakpoint(ph, rp.pi_target, 483 FLG_BP_PLTRES); 484 bpflags = continue_to_break(ph); 485 486 (void) delete_breakpoint(ph, rp.pi_target, 487 FLG_BP_PLTRES); 488 489 if (bpflags & FLG_BP_PLTRES) 490 (void) step_n(ph, rp.pi_nstep, 491 FLG_SN_NONE); 492 } else if (rp.pi_skip_method == RD_RESOLVE_STEP) 493 (void) step_n(ph, rp.pi_nstep, FLG_SN_NONE); 494 } 495 } 496 497 oper = PRCFAULT; 498 if (writev(ph->pp_ctlfd, piov, 1) == -1) 499 perr("stn: PRCFAULT"); 500 501 if ((flgs & FLG_SN_VERBOSE) && (ph->pp_flags & FLG_PP_LMAPS)) { 502 if (get_linkmaps(ph) != RET_OK) 503 (void) fprintf(stderr, "problem loading linkmaps\n"); 504 } 505 506 return (RET_OK); 507 } 508 509 void 510 step_to_addr(struct ps_prochandle *ph, ulong_t addr) 511 { 512 pstatus_t pstat; 513 int count = 0; 514 ulong_t caddr; 515 516 if (read(ph->pp_statusfd, &pstat, sizeof (pstat)) == -1) 517 perr("sta: reading status"); 518 519 caddr = pstat.pr_lwp.pr_reg[R_PC]; 520 521 while ((caddr > addr) || ((caddr + 0xff) < addr)) { 522 (void) step_n(ph, 1, FLG_SN_NONE); 523 if (read(ph->pp_statusfd, &pstat, sizeof (pstat)) == -1) 524 perr("sta1: reading status"); 525 caddr = pstat.pr_lwp.pr_reg[R_PC]; 526 if ((count % 10000) == 0) { 527 (void) printf("%d: ", count); 528 disasm(ph, 1); 529 } 530 531 count++; 532 } 533 534 (void) printf("address found %d instructions in: pc: 0x%lx addr: " 535 "0x%lx\n", count, caddr, addr); 536 }