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.
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.
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]
23 * Copyright 2008 Sun Microsystems, Inc. All rights reserved.
24 * Use is subject to license terms.
36 #include <sys/types.h>
37 #include <sys/stack.h>
40 #include <sys/isa_defs.h>
41 #include <proc_service.h>
52 * Functions supporting library function call tracing.
61 * static functions in this file.
63 void function_entry(private_t
*, struct bkpt
*, struct callstack
*);
64 void function_return(private_t
*, struct callstack
*);
65 int object_iter(void *, const prmap_t
*, const char *);
66 int object_present(void *, const prmap_t
*, const char *);
67 int symbol_iter(void *, const GElf_Sym
*, const char *);
68 uintptr_t get_return_address(uintptr_t *);
69 int get_arguments(long *argp
);
70 uintptr_t previous_fp(uintptr_t, uintptr_t *);
71 int lwp_stack_traps(void *cd
, const lwpstatus_t
*Lsp
);
72 int thr_stack_traps(const td_thrhandle_t
*Thp
, void *cd
);
73 struct bkpt
*create_bkpt(uintptr_t, int, int);
74 void set_deferred_breakpoints(void);
76 #define DEF_MAXCALL 16 /* initial value of Stk->maxcall */
78 #define FAULT_ADDR ((uintptr_t)(0-8))
81 #define bpt_hash(addr) ((((addr) >> 13) ^ ((addr) >> 2)) & 0x7ff)
84 setup_thread_agent(void)
88 td_thr_events_t events
;
90 if (Thr_agent
!= NULL
) /* only once */
92 if (td_init() != TD_OK
|| td_ta_new(Proc
, &Thr_agent
) != TD_OK
)
95 td_event_emptyset(&events
);
96 td_event_addset(&events
, TD_CREATE
);
97 if (td_ta_event_addr(Thr_agent
, TD_CREATE
, ¬ify
) == TD_OK
&&
98 notify
.type
== NOTIFY_BPT
&&
99 td_ta_set_event(Thr_agent
, &events
) == TD_OK
&&
100 (Bp
= create_bkpt(notify
.u
.bptaddr
, 0, 1)) != NULL
)
101 Bp
->flags
|= BPT_TD_CREATE
;
106 * Delete all breakpoints in the range [base .. base+size)
107 * from the breakpoint hash table.
110 delete_breakpoints(uintptr_t base
, size_t size
)
116 if (bpt_hashtable
== NULL
)
118 for (i
= 0; i
< HASHSZ
; i
++) {
119 Bpp
= &bpt_hashtable
[i
];
120 while ((Bp
= *Bpp
) != NULL
) {
121 if (Bp
->addr
< base
|| Bp
->addr
>= base
+ size
) {
133 * Establishment of breakpoints on traced library functions.
136 establish_breakpoints(void)
141 /* allocate the breakpoint hash table */
142 if (bpt_hashtable
== NULL
) {
143 bpt_hashtable
= my_malloc(HASHSZ
* sizeof (struct bkpt
*),
145 (void) memset(bpt_hashtable
, 0,
146 HASHSZ
* sizeof (struct bkpt
*));
150 * Set special rtld_db event breakpoints, first time only.
152 if (Rdb_agent
== NULL
&&
153 (Rdb_agent
= Prd_agent(Proc
)) != NULL
) {
157 (void) rd_event_enable(Rdb_agent
, 1);
158 if (rd_event_addr(Rdb_agent
, RD_PREINIT
, ¬ify
) == RD_OK
&&
159 (Bp
= create_bkpt(notify
.u
.bptaddr
, 0, 1)) != NULL
)
160 Bp
->flags
|= BPT_PREINIT
;
161 if (rd_event_addr(Rdb_agent
, RD_POSTINIT
, ¬ify
) == RD_OK
&&
162 (Bp
= create_bkpt(notify
.u
.bptaddr
, 0, 1)) != NULL
)
163 Bp
->flags
|= BPT_POSTINIT
;
164 if (rd_event_addr(Rdb_agent
, RD_DLACTIVITY
, ¬ify
) == RD_OK
&&
165 (Bp
= create_bkpt(notify
.u
.bptaddr
, 0, 1)) != NULL
)
166 Bp
->flags
|= BPT_DLACTIVITY
;
170 * Set special thread event breakpoint, first time libc is seen.
172 if (Thr_agent
== NULL
)
173 setup_thread_agent();
176 * Tell libproc to update its mappings.
181 * If rtld_db told us a library was being deleted,
182 * first mark all of the dynlibs as not present, then
183 * iterate over the shared objects, marking only those
184 * present that really are present, and finally delete
185 * all of the not-present dynlibs.
187 if (delete_library
) {
191 for (Dp
= Dynlib
; Dp
!= NULL
; Dp
= Dp
->next
)
193 (void) Pobject_iter(Proc
, object_present
, NULL
);
195 while ((Dp
= *Dpp
) != NULL
) {
200 delete_breakpoints(Dp
->base
, Dp
->size
);
203 free(Dp
->match_name
);
207 delete_library
= FALSE
;
211 * Iterate over the shared objects, creating breakpoints.
213 (void) Pobject_iter(Proc
, object_iter
, NULL
);
216 * Now actually set all the breakpoints we just created.
218 set_deferred_breakpoints();
222 * Initial establishment of stacks in a newly-grabbed process.
223 * establish_breakpoints() has already been called.
226 establish_stacks(void)
228 const pstatus_t
*Psp
= Pstatus(Proc
);
232 prmap_t
*Pmap
= NULL
;
236 (void) sprintf(mapfile
, "/proc/%d/rmap", (int)Psp
->pr_pid
);
237 if ((mapfd
= open(mapfile
, O_RDONLY
)) < 0 ||
238 fstat(mapfd
, &statb
) != 0 ||
239 statb
.st_size
< sizeof (prmap_t
) ||
240 (Pmap
= my_malloc(statb
.st_size
, NULL
)) == NULL
||
241 (nmap
= pread(mapfd
, Pmap
, statb
.st_size
, 0L)) <= 0 ||
242 (nmap
/= sizeof (prmap_t
)) == 0) {
251 * Iterate over lwps, establishing stacks.
255 (void) Plwp_iter(Proc
, lwp_stack_traps
, &ph_map
);
258 if (Thr_agent
== NULL
)
262 * Iterate over unbound threads, establishing stacks.
264 (void) td_ta_thr_iter(Thr_agent
, thr_stack_traps
, NULL
,
265 TD_THR_ANY_STATE
, TD_THR_LOWEST_PRIORITY
,
266 TD_SIGNO_MASK
, TD_THR_ANY_USER_FLAGS
);
270 do_symbol_iter(const char *object_name
, struct dynpat
*Dyp
)
272 if (*Dyp
->Dp
->prt_name
== '\0')
273 object_name
= PR_OBJ_EXEC
;
276 * Always search the dynamic symbol table.
278 (void) Psymbol_iter(Proc
, object_name
,
279 PR_DYNSYM
, BIND_WEAK
|BIND_GLOBAL
|TYPE_FUNC
,
283 * Search the static symbol table if this is the
284 * executable file or if we are being asked to
285 * report internal calls within the library.
287 if (object_name
== PR_OBJ_EXEC
|| Dyp
->internal
)
288 (void) Psymbol_iter(Proc
, object_name
,
289 PR_SYMTAB
, BIND_ANY
|TYPE_FUNC
,
295 object_iter(void *cd
, const prmap_t
*pmp
, const char *object_name
)
304 if ((pmp
->pr_mflags
& MA_WRITE
) || !(pmp
->pr_mflags
& MA_EXEC
))
308 * Set special thread event breakpoint, first time libc is seen.
310 if (Thr_agent
== NULL
&& strstr(object_name
, "/libc.so.") != NULL
)
311 setup_thread_agent();
313 for (Dp
= Dynlib
; Dp
!= NULL
; Dp
= Dp
->next
)
314 if (strcmp(object_name
, Dp
->lib_name
) == 0 ||
315 (strcmp(Dp
->lib_name
, "a.out") == 0 &&
316 strcmp(pmp
->pr_mapname
, "a.out") == 0))
320 Dp
= my_malloc(sizeof (struct dynlib
), NULL
);
321 (void) memset(Dp
, 0, sizeof (struct dynlib
));
322 if (strcmp(pmp
->pr_mapname
, "a.out") == 0) {
323 Dp
->lib_name
= strdup(pmp
->pr_mapname
);
324 Dp
->match_name
= strdup(pmp
->pr_mapname
);
325 Dp
->prt_name
= strdup("");
327 Dp
->lib_name
= strdup(object_name
);
328 if ((str
= strrchr(object_name
, '/')) != NULL
)
332 (void) strncpy(name
, str
, sizeof (name
) - 2);
333 name
[sizeof (name
) - 2] = '\0';
334 if ((s
= strstr(name
, ".so")) != NULL
)
336 Dp
->match_name
= strdup(name
);
337 (void) strcat(name
, ":");
338 Dp
->prt_name
= strdup(name
);
345 (not_consist
&& strcmp(Dp
->prt_name
, "ld:") != 0)) /* kludge */
348 if (hflag
&& not_consist
)
349 (void) fprintf(stderr
, "not_consist is TRUE, building %s\n",
352 Dp
->base
= pmp
->pr_vaddr
;
353 Dp
->size
= pmp
->pr_size
;
356 * For every dynlib pattern that matches this library's name,
357 * iterate through all of the library's symbols looking for
358 * matching symbol name patterns.
360 for (Dyp
= Dynpat
; Dyp
!= NULL
; Dyp
= Dyp
->next
) {
361 if (interrupt
|sigusr1
)
363 for (i
= 0; i
< Dyp
->nlibpat
; i
++) {
364 if (interrupt
|sigusr1
)
366 if (fnmatch(Dyp
->libpat
[i
], Dp
->match_name
, 0) != 0)
367 continue; /* no match */
370 * Require an exact match for the executable (a.out)
371 * and for the dynamic linker (ld.so.1).
373 if ((strcmp(Dp
->match_name
, "a.out") == 0 ||
374 strcmp(Dp
->match_name
, "ld") == 0) &&
375 strcmp(Dyp
->libpat
[i
], Dp
->match_name
) != 0)
379 * Set Dyp->Dp to Dp so symbol_iter() can use it.
382 do_symbol_iter(object_name
, Dyp
);
388 return (interrupt
| sigusr1
);
393 object_present(void *cd
, const prmap_t
*pmp
, const char *object_name
)
397 for (Dp
= Dynlib
; Dp
!= NULL
; Dp
= Dp
->next
) {
398 if (Dp
->base
== pmp
->pr_vaddr
)
406 * Search for an existing breakpoint at the 'pc' location.
409 get_bkpt(uintptr_t pc
)
413 for (Bp
= bpt_hashtable
[bpt_hash(pc
)]; Bp
!= NULL
; Bp
= Bp
->next
)
421 * Create a breakpoint at 'pc', if one is not there already.
422 * 'ret' is true when creating a function return breakpoint, in which case
423 * fail and return NULL if the breakpoint would be created in writeable data.
424 * If 'set' it true, set the breakpoint in the process now.
427 create_bkpt(uintptr_t pc
, int ret
, int set
)
429 uint_t hix
= bpt_hash(pc
);
433 for (Bp
= bpt_hashtable
[hix
]; Bp
!= NULL
; Bp
= Bp
->next
)
438 * Don't set return breakpoints on writeable data
439 * or on any space other than executable text.
440 * Don't set breakpoints in the child of a vfork()
441 * because that would modify the parent's address space.
443 if (is_vfork_child
||
445 ((pmp
= Paddr_to_text_map(Proc
, pc
)) == NULL
||
446 !(pmp
->pr_mflags
& MA_EXEC
) ||
447 (pmp
->pr_mflags
& MA_WRITE
))))
450 /* create a new unnamed breakpoint */
451 Bp
= my_malloc(sizeof (struct bkpt
), NULL
);
457 if (set
&& Psetbkpt(Proc
, Bp
->addr
, &Bp
->instr
) == 0)
458 Bp
->flags
|= BPT_ACTIVE
;
459 Bp
->next
= bpt_hashtable
[hix
];
460 bpt_hashtable
[hix
] = Bp
;
466 * Set all breakpoints that haven't been set yet.
467 * Deactivate all breakpoints from modules that are not present any more.
470 set_deferred_breakpoints(void)
478 for (i
= 0; i
< HASHSZ
; i
++) {
479 for (Bp
= bpt_hashtable
[i
]; Bp
!= NULL
; Bp
= Bp
->next
) {
480 if (!(Bp
->flags
& BPT_ACTIVE
)) {
481 if (!(Bp
->flags
& BPT_EXCLUDE
) &&
482 Psetbkpt(Proc
, Bp
->addr
, &Bp
->instr
) == 0)
483 Bp
->flags
|= BPT_ACTIVE
;
484 } else if (Paddr_to_text_map(Proc
, Bp
->addr
) == NULL
) {
485 Bp
->flags
&= ~BPT_ACTIVE
;
492 symbol_iter(void *cd
, const GElf_Sym
*sym
, const char *sym_name
)
494 struct dynpat
*Dyp
= cd
;
495 struct dynlib
*Dp
= Dyp
->Dp
;
496 uintptr_t pc
= sym
->st_value
;
500 /* ignore any undefined symbols */
501 if (sym
->st_shndx
== SHN_UNDEF
)
505 * Arbitrarily omit "_start" from the executable.
506 * (Avoid indentation before main().)
508 if (*Dp
->prt_name
== '\0' && strcmp(sym_name
, "_start") == 0)
512 * Arbitrarily omit "_rt_boot" from the dynamic linker.
513 * (Avoid indentation before main().)
515 if (strcmp(Dp
->match_name
, "ld") == 0 &&
516 strcmp(sym_name
, "_rt_boot") == 0)
520 * Arbitrarily omit any symbols whose name starts with '.'.
521 * Apparantly putting a breakpoint on .umul causes a
522 * fatal error in libthread (%y is not restored correctly
523 * when a single step is taken). Looks like a /proc bug.
525 if (*sym_name
== '.')
529 * For each pattern in the array of symbol patterns,
530 * if the pattern matches the symbol name, then
531 * create a breakpoint at the function in question.
533 for (i
= 0; i
< Dyp
->nsympat
; i
++) {
534 if (interrupt
|sigusr1
)
536 if (fnmatch(Dyp
->sympat
[i
], sym_name
, 0) != 0)
539 if ((Bp
= create_bkpt(pc
, 0, 0)) == NULL
) /* can't fail */
543 * New breakpoints receive a name now.
544 * For existing breakpoints, prefer the subset name if possible,
545 * else prefer the shorter name.
547 if (Bp
->sym_name
== NULL
) {
548 Bp
->sym_name
= strdup(sym_name
);
549 } else if (strstr(Bp
->sym_name
, sym_name
) != NULL
||
550 strlen(Bp
->sym_name
) > strlen(sym_name
)) {
552 Bp
->sym_name
= strdup(sym_name
);
555 Bp
->flags
|= Dyp
->flag
;
557 Bp
->flags
|= BPT_EXCLUDE
;
558 else if (Dyp
->internal
|| *Dp
->prt_name
== '\0')
559 Bp
->flags
|= BPT_INTERNAL
;
563 return (interrupt
| sigusr1
);
566 /* For debugging only ---- */
568 report_htable_stats(void)
570 const pstatus_t
*Psp
= Pstatus(Proc
);
571 struct callstack
*Stk
;
573 uint_t Min
= 1000000;
578 uint_t bucket
[HASHSZ
];
580 if (Dynpat
== NULL
|| !hflag
)
584 (void) memset(bucket
, 0, sizeof (bucket
));
586 for (i
= 0; i
< HASHSZ
; i
++) {
588 for (Bp
= bpt_hashtable
[i
]; Bp
!= NULL
; Bp
= Bp
->next
)
598 Avg
= (Total
+ HASHSZ
/ 2) / HASHSZ
;
599 (void) fprintf(stderr
, "truss hash table statistics --------\n");
600 (void) fprintf(stderr
, " Total = %u\n", Total
);
601 (void) fprintf(stderr
, " Min = %u\n", Min
);
602 (void) fprintf(stderr
, " Max = %u\n", Max
);
603 (void) fprintf(stderr
, " Avg = %u\n", Avg
);
604 for (i
= 0; i
< HASHSZ
; i
++)
606 (void) fprintf(stderr
, " %3u buckets of size %d\n",
609 (void) fprintf(stderr
, "truss-detected stacks --------\n");
610 for (Stk
= callstack
; Stk
!= NULL
; Stk
= Stk
->next
) {
611 (void) fprintf(stderr
,
612 " base = 0x%.8lx end = 0x%.8lx size = %ld\n",
613 (ulong_t
)Stk
->stkbase
,
614 (ulong_t
)Stk
->stkend
,
615 (ulong_t
)(Stk
->stkend
- Stk
->stkbase
));
617 (void) fprintf(stderr
, "primary unix stack --------\n");
618 (void) fprintf(stderr
,
619 " base = 0x%.8lx end = 0x%.8lx size = %ld\n",
620 (ulong_t
)Psp
->pr_stkbase
,
621 (ulong_t
)(Psp
->pr_stkbase
+ Psp
->pr_stksize
),
622 (ulong_t
)Psp
->pr_stksize
);
623 (void) fprintf(stderr
, "nthr_create = %u\n", nthr_create
);
627 make_lwp_stack(const lwpstatus_t
*Lsp
, prmap_t
*Pmap
, int nmap
)
629 const pstatus_t
*Psp
= Pstatus(Proc
);
630 uintptr_t sp
= Lsp
->pr_reg
[R_SP
];
631 id_t lwpid
= Lsp
->pr_lwpid
;
632 struct callstack
*Stk
;
634 td_thrinfo_t thrinfo
;
636 if (data_model
!= PR_MODEL_LP64
)
639 /* check to see if we already have this stack */
642 for (Stk
= callstack
; Stk
!= NULL
; Stk
= Stk
->next
)
643 if (sp
>= Stk
->stkbase
&& sp
< Stk
->stkend
)
646 Stk
= my_malloc(sizeof (struct callstack
), NULL
);
647 Stk
->next
= callstack
;
652 Stk
->nthr_create
= 0;
654 Stk
->maxcall
= DEF_MAXCALL
;
655 Stk
->stack
= my_malloc(DEF_MAXCALL
* sizeof (*Stk
->stack
), NULL
);
658 if (sp
>= Psp
->pr_stkbase
&& sp
< Psp
->pr_stkbase
+ Psp
->pr_stksize
) {
659 Stk
->stkbase
= Psp
->pr_stkbase
;
660 Stk
->stkend
= Stk
->stkbase
+ Psp
->pr_stksize
;
664 /* alternate stack */
665 if ((Lsp
->pr_altstack
.ss_flags
& SS_ONSTACK
) &&
666 sp
>= (uintptr_t)Lsp
->pr_altstack
.ss_sp
&&
667 sp
< (uintptr_t)Lsp
->pr_altstack
.ss_sp
668 + Lsp
->pr_altstack
.ss_size
) {
669 Stk
->stkbase
= (uintptr_t)Lsp
->pr_altstack
.ss_sp
;
670 Stk
->stkend
= Stk
->stkbase
+ Lsp
->pr_altstack
.ss_size
;
675 if (Thr_agent
!= NULL
&&
676 td_ta_map_lwp2thr(Thr_agent
, lwpid
, &th
) == TD_OK
&&
677 td_thr_get_info(&th
, &thrinfo
) == TD_OK
&&
678 sp
>= (uintptr_t)thrinfo
.ti_stkbase
- thrinfo
.ti_stksize
&&
679 sp
< (uintptr_t)thrinfo
.ti_stkbase
) {
680 /* The bloody fools got this backwards! */
681 Stk
->stkend
= (uintptr_t)thrinfo
.ti_stkbase
;
682 Stk
->stkbase
= Stk
->stkend
- thrinfo
.ti_stksize
;
686 /* last chance -- try the raw memory map */
687 for (; nmap
; nmap
--, Pmap
++) {
688 if (sp
>= Pmap
->pr_vaddr
&&
689 sp
< Pmap
->pr_vaddr
+ Pmap
->pr_size
) {
690 Stk
->stkbase
= Pmap
->pr_vaddr
;
691 Stk
->stkend
= Pmap
->pr_vaddr
+ Pmap
->pr_size
;
696 callstack
= Stk
->next
;
703 make_thr_stack(const td_thrhandle_t
*Thp
, prgregset_t reg
)
705 const pstatus_t
*Psp
= Pstatus(Proc
);
706 td_thrinfo_t thrinfo
;
707 uintptr_t sp
= reg
[R_SP
];
708 struct callstack
*Stk
;
710 if (data_model
!= PR_MODEL_LP64
)
713 /* check to see if we already have this stack */
716 for (Stk
= callstack
; Stk
!= NULL
; Stk
= Stk
->next
)
717 if (sp
>= Stk
->stkbase
&& sp
< Stk
->stkend
)
720 Stk
= my_malloc(sizeof (struct callstack
), NULL
);
721 Stk
->next
= callstack
;
726 Stk
->nthr_create
= 0;
728 Stk
->maxcall
= DEF_MAXCALL
;
729 Stk
->stack
= my_malloc(DEF_MAXCALL
* sizeof (*Stk
->stack
), NULL
);
732 if (sp
>= Psp
->pr_stkbase
&& sp
< Psp
->pr_stkbase
+ Psp
->pr_stksize
) {
733 Stk
->stkbase
= Psp
->pr_stkbase
;
734 Stk
->stkend
= Stk
->stkbase
+ Psp
->pr_stksize
;
738 if (td_thr_get_info(Thp
, &thrinfo
) == TD_OK
&&
739 sp
>= (uintptr_t)thrinfo
.ti_stkbase
- thrinfo
.ti_stksize
&&
740 sp
< (uintptr_t)thrinfo
.ti_stkbase
) {
741 /* The bloody fools got this backwards! */
742 Stk
->stkend
= (uintptr_t)thrinfo
.ti_stkbase
;
743 Stk
->stkbase
= Stk
->stkend
- thrinfo
.ti_stksize
;
747 callstack
= Stk
->next
;
754 find_lwp_stack(uintptr_t sp
)
756 const pstatus_t
*Psp
= Pstatus(Proc
);
760 prmap_t
*Pmap
= NULL
;
761 prmap_t
*pmap
= NULL
;
763 struct callstack
*Stk
= NULL
;
766 * Get the address space map.
768 (void) sprintf(mapfile
, "/proc/%d/rmap", (int)Psp
->pr_pid
);
769 if ((mapfd
= open(mapfile
, O_RDONLY
)) < 0 ||
770 fstat(mapfd
, &statb
) != 0 ||
771 statb
.st_size
< sizeof (prmap_t
) ||
772 (Pmap
= my_malloc(statb
.st_size
, NULL
)) == NULL
||
773 (nmap
= pread(mapfd
, Pmap
, statb
.st_size
, 0L)) <= 0 ||
774 (nmap
/= sizeof (prmap_t
)) == 0) {
782 for (pmap
= Pmap
; nmap
--; pmap
++) {
783 if (sp
>= pmap
->pr_vaddr
&&
784 sp
< pmap
->pr_vaddr
+ pmap
->pr_size
) {
785 Stk
= my_malloc(sizeof (struct callstack
), NULL
);
786 Stk
->next
= callstack
;
789 Stk
->stkbase
= pmap
->pr_vaddr
;
790 Stk
->stkend
= pmap
->pr_vaddr
+ pmap
->pr_size
;
793 Stk
->nthr_create
= 0;
795 Stk
->maxcall
= DEF_MAXCALL
;
796 Stk
->stack
= my_malloc(
797 DEF_MAXCALL
* sizeof (*Stk
->stack
), NULL
);
807 find_stack(uintptr_t sp
)
809 const pstatus_t
*Psp
= Pstatus(Proc
);
810 private_t
*pri
= get_private();
811 const lwpstatus_t
*Lsp
= pri
->lwpstat
;
812 id_t lwpid
= Lsp
->pr_lwpid
;
814 prgreg_t tref
= Lsp
->pr_reg
[R_G7
];
815 #elif defined(__amd64)
816 prgreg_t tref
= Lsp
->pr_reg
[REG_FS
];
817 #elif defined(__i386)
818 prgreg_t tref
= Lsp
->pr_reg
[GS
];
820 struct callstack
*Stk
= NULL
;
822 td_thrinfo_t thrinfo
;
826 if (sp
>= Psp
->pr_stkbase
&& sp
< Psp
->pr_stkbase
+ Psp
->pr_stksize
) {
827 Stk
= my_malloc(sizeof (struct callstack
), NULL
);
828 Stk
->next
= callstack
;
831 Stk
->stkbase
= Psp
->pr_stkbase
;
832 Stk
->stkend
= Stk
->stkbase
+ Psp
->pr_stksize
;
835 Stk
->nthr_create
= 0;
837 Stk
->maxcall
= DEF_MAXCALL
;
838 Stk
->stack
= my_malloc(DEF_MAXCALL
* sizeof (*Stk
->stack
),
843 /* alternate stack */
844 if ((Lsp
->pr_altstack
.ss_flags
& SS_ONSTACK
) &&
845 sp
>= (uintptr_t)Lsp
->pr_altstack
.ss_sp
&&
846 sp
< (uintptr_t)Lsp
->pr_altstack
.ss_sp
847 + Lsp
->pr_altstack
.ss_size
) {
848 Stk
= my_malloc(sizeof (struct callstack
), NULL
);
849 Stk
->next
= callstack
;
852 Stk
->stkbase
= (uintptr_t)Lsp
->pr_altstack
.ss_sp
;
853 Stk
->stkend
= Stk
->stkbase
+ Lsp
->pr_altstack
.ss_size
;
856 Stk
->nthr_create
= 0;
858 Stk
->maxcall
= DEF_MAXCALL
;
859 Stk
->stack
= my_malloc(DEF_MAXCALL
* sizeof (*Stk
->stack
),
864 if (Thr_agent
== NULL
)
865 return (find_lwp_stack(sp
));
868 if ((error
= td_ta_map_lwp2thr(Thr_agent
, lwpid
, &th
)) != TD_OK
) {
870 (void) fprintf(stderr
,
871 "cannot get thread handle for "
872 "lwp#%d, error=%d, tref=0x%.8lx\n",
873 (int)lwpid
, error
, (long)tref
);
877 if ((error
= td_thr_get_info(&th
, &thrinfo
)) != TD_OK
) {
879 (void) fprintf(stderr
,
880 "cannot get thread info for "
881 "lwp#%d, error=%d, tref=0x%.8lx\n",
882 (int)lwpid
, error
, (long)tref
);
886 if (sp
>= (uintptr_t)thrinfo
.ti_stkbase
- thrinfo
.ti_stksize
&&
887 sp
< (uintptr_t)thrinfo
.ti_stkbase
) {
888 Stk
= my_malloc(sizeof (struct callstack
), NULL
);
889 Stk
->next
= callstack
;
892 /* The bloody fools got this backwards! */
893 Stk
->stkend
= (uintptr_t)thrinfo
.ti_stkbase
;
894 Stk
->stkbase
= Stk
->stkend
- thrinfo
.ti_stksize
;
896 Stk
->tid
= thrinfo
.ti_tid
;
897 Stk
->nthr_create
= nthr_create
;
899 Stk
->maxcall
= DEF_MAXCALL
;
900 Stk
->stack
= my_malloc(DEF_MAXCALL
* sizeof (*Stk
->stack
),
905 /* stack bounds failure -- complain bitterly */
907 (void) fprintf(stderr
,
908 "sp not within thread stack: "
909 "sp=0x%.8lx stkbase=0x%.8lx stkend=0x%.8lx\n",
911 /* The bloody fools got this backwards! */
912 (ulong_t
)thrinfo
.ti_stkbase
- thrinfo
.ti_stksize
,
913 (ulong_t
)thrinfo
.ti_stkbase
);
920 get_tid(struct callstack
*Stk
)
922 private_t
*pri
= get_private();
923 const lwpstatus_t
*Lsp
= pri
->lwpstat
;
924 id_t lwpid
= Lsp
->pr_lwpid
;
926 prgreg_t tref
= Lsp
->pr_reg
[R_G7
];
927 #elif defined(__amd64)
928 prgreg_t tref
= (data_model
== PR_MODEL_LP64
) ?
929 Lsp
->pr_reg
[REG_FS
] : Lsp
->pr_reg
[REG_GS
];
930 #elif defined(__i386)
931 prgreg_t tref
= Lsp
->pr_reg
[GS
];
934 td_thrinfo_t thrinfo
;
937 if (Thr_agent
== NULL
) {
940 Stk
->nthr_create
= 0;
946 * If we have a matching tref and no new threads have
947 * been created since the last time we encountered this
948 * stack, then we don't have to go through the overhead
949 * of calling td_ta_map_lwp2thr() to get the thread-id.
951 if (tref
== Stk
->tref
&& Stk
->nthr_create
== nthr_create
)
954 if ((error
= td_ta_map_lwp2thr(Thr_agent
, lwpid
, &th
)) != TD_OK
) {
956 (void) fprintf(stderr
,
957 "cannot get thread handle for "
958 "lwp#%d, error=%d, tref=0x%.8lx\n",
959 (int)lwpid
, error
, (long)tref
);
962 Stk
->nthr_create
= 0;
963 } else if ((error
= td_thr_get_info(&th
, &thrinfo
)) != TD_OK
) {
965 (void) fprintf(stderr
,
966 "cannot get thread info for "
967 "lwp#%d, error=%d, tref=0x%.8lx\n",
968 (int)lwpid
, error
, (long)tref
);
971 Stk
->nthr_create
= 0;
974 Stk
->tid
= thrinfo
.ti_tid
;
975 Stk
->nthr_create
= nthr_create
;
980 callstack_info(uintptr_t sp
, uintptr_t fp
, int makeid
)
982 struct callstack
*Stk
;
986 Pread(Proc
, &trash
, sizeof (trash
), sp
) != sizeof (trash
))
989 for (Stk
= callstack
; Stk
!= NULL
; Stk
= Stk
->next
)
990 if (sp
>= Stk
->stkbase
&& sp
< Stk
->stkend
)
994 * If we didn't find the stack, do it the hard way.
997 uintptr_t stkbase
= sp
;
1001 #if defined(i386) || defined(__amd64)
1003 if (data_model
== PR_MODEL_LP64
)
1004 minsize
= 2 * sizeof (uintptr_t); /* fp + pc */
1007 minsize
= 2 * sizeof (uint32_t);
1010 if (data_model
!= PR_MODEL_LP64
)
1011 minsize
= SA32(MINFRAME32
);
1013 minsize
= SA64(MINFRAME64
);
1015 minsize
= SA(MINFRAME
);
1018 stkend
= sp
+ minsize
;
1020 while (Stk
== NULL
&& fp
!= 0 && fp
>= sp
) {
1021 stkend
= fp
+ minsize
;
1022 for (Stk
= callstack
; Stk
!= NULL
; Stk
= Stk
->next
)
1023 if ((fp
>= Stk
->stkbase
&& fp
< Stk
->stkend
) ||
1024 (stkend
> Stk
->stkbase
&&
1025 stkend
<= Stk
->stkend
))
1028 fp
= previous_fp(fp
, NULL
);
1031 if (Stk
!= NULL
) /* the stack grew */
1032 Stk
->stkbase
= stkbase
;
1035 if (Stk
== NULL
&& makeid
) /* new stack */
1036 Stk
= find_stack(sp
);
1042 * Ensure that there is room for at least one more entry.
1044 if (Stk
->ncall
== Stk
->maxcall
) {
1046 Stk
->stack
= my_realloc(Stk
->stack
,
1047 Stk
->maxcall
* sizeof (*Stk
->stack
), NULL
);
1057 * Reset the breakpoint information (called on successful exec()).
1060 reset_breakpoints(void)
1064 struct callstack
*Stk
;
1070 /* destroy all previous dynamic library information */
1071 while ((Dp
= Dynlib
) != NULL
) {
1074 free(Dp
->match_name
);
1079 /* destroy all previous breakpoint trap information */
1080 if (bpt_hashtable
!= NULL
) {
1081 for (i
= 0; i
< HASHSZ
; i
++) {
1082 while ((Bp
= bpt_hashtable
[i
]) != NULL
) {
1083 bpt_hashtable
[i
] = Bp
->next
;
1090 /* destroy all the callstack information */
1091 while ((Stk
= callstack
) != NULL
) {
1092 callstack
= Stk
->next
;
1097 /* we are not a multi-threaded process anymore */
1098 if (Thr_agent
!= NULL
)
1099 (void) td_ta_delete(Thr_agent
);
1102 /* tell libproc to clear out its mapping information */
1106 /* Reestablish the symbols from the executable */
1107 (void) establish_breakpoints();
1111 * Clear breakpoints from the process (called before Prelease()).
1112 * Don't actually destroy the breakpoint table;
1113 * threads currently fielding breakpoints will need it.
1116 clear_breakpoints(void)
1125 * Change all breakpoint traps back to normal instructions.
1126 * We attempt to remove a breakpoint from every address which
1127 * may have ever contained a breakpoint to protect our victims.
1129 report_htable_stats(); /* report stats first */
1130 for (i
= 0; i
< HASHSZ
; i
++) {
1131 for (Bp
= bpt_hashtable
[i
]; Bp
!= NULL
; Bp
= Bp
->next
) {
1132 if (Bp
->flags
& BPT_ACTIVE
)
1133 (void) Pdelbkpt(Proc
, Bp
->addr
, Bp
->instr
);
1134 Bp
->flags
&= ~BPT_ACTIVE
;
1138 if (Thr_agent
!= NULL
) {
1139 td_thr_events_t events
;
1141 td_event_fillset(&events
);
1142 (void) td_ta_clear_event(Thr_agent
, &events
);
1143 (void) td_ta_delete(Thr_agent
);
1149 * Reestablish the breakpoint traps in the process.
1150 * Called after resuming from a vfork() in the parent.
1153 reestablish_traps(void)
1159 if (Dynpat
== NULL
|| is_vfork_child
)
1162 for (i
= 0; i
< HASHSZ
; i
++) {
1163 for (Bp
= bpt_hashtable
[i
]; Bp
!= NULL
; Bp
= Bp
->next
) {
1164 if ((Bp
->flags
& BPT_ACTIVE
) &&
1165 Psetbkpt(Proc
, Bp
->addr
, &instr
) != 0)
1166 Bp
->flags
&= ~BPT_ACTIVE
;
1172 show_function_call(private_t
*pri
,
1173 struct callstack
*Stk
, struct dynlib
*Dp
, struct bkpt
*Bp
)
1179 narg
= get_arguments(arg
);
1180 make_pname(pri
, (Stk
!= NULL
)? Stk
->tid
: 0);
1184 for (i
= 1; i
< Stk
->ncall
; i
++) {
1185 (void) fputc(' ', stdout
);
1186 (void) fputc(' ', stdout
);
1189 (void) printf("-> %s%s(", Dp
->prt_name
, Bp
->sym_name
);
1190 for (i
= 0; i
< narg
; i
++) {
1191 (void) printf("0x%lx", arg
[i
]);
1193 (void) fputc(',', stdout
);
1194 (void) fputc(' ', stdout
);
1197 (void) printf(")\n");
1203 show_function_return(private_t
*pri
, long rval
, int stret
,
1204 struct callstack
*Stk
, struct dynlib
*Dp
, struct bkpt
*Bp
)
1208 make_pname(pri
, Stk
->tid
);
1211 for (i
= 0; i
< Stk
->ncall
; i
++) {
1212 (void) fputc(' ', stdout
);
1213 (void) fputc(' ', stdout
);
1215 (void) printf("<- %s%s() = ", Dp
->prt_name
, Bp
->sym_name
);
1217 (void) printf("struct return\n");
1218 } else if (data_model
== PR_MODEL_LP64
) {
1219 if (rval
>= (64 * 1024) || -rval
>= (64 * 1024))
1220 (void) printf("0x%lx\n", rval
);
1222 (void) printf("%ld\n", rval
);
1224 int rval32
= (int)rval
;
1225 if (rval32
>= (64 * 1024) || -rval32
>= (64 * 1024))
1226 (void) printf("0x%x\n", rval32
);
1228 (void) printf("%d\n", rval32
);
1234 * Called to deal with function-call tracing.
1235 * Return 0 on normal success, 1 to indicate a BPT_HANG success,
1236 * and -1 on failure (not tracing functions or unknown breakpoint).
1239 function_trace(private_t
*pri
, int first
, int clear
, int dotrace
)
1241 struct ps_lwphandle
*Lwp
= pri
->Lwp
;
1242 const lwpstatus_t
*Lsp
= pri
->lwpstat
;
1243 uintptr_t pc
= Lsp
->pr_reg
[R_PC
];
1244 uintptr_t sp
= Lsp
->pr_reg
[R_SP
];
1245 uintptr_t fp
= Lsp
->pr_reg
[R_FP
];
1248 struct callstack
*Stk
;
1256 if (data_model
!= PR_MODEL_LP64
) {
1262 if ((Bp
= get_bkpt(pc
)) == NULL
) {
1264 (void) fprintf(stderr
,
1265 "function_trace(): "
1266 "cannot find breakpoint for pc: 0x%.8lx\n",
1271 if ((Bp
->flags
& (BPT_PREINIT
|BPT_POSTINIT
|BPT_DLACTIVITY
)) && !clear
) {
1272 rd_event_msg_t event_msg
;
1275 if (Bp
->flags
& BPT_PREINIT
)
1276 (void) fprintf(stderr
, "function_trace(): "
1277 "RD_PREINIT breakpoint\n");
1278 if (Bp
->flags
& BPT_POSTINIT
)
1279 (void) fprintf(stderr
, "function_trace(): "
1280 "RD_POSTINIT breakpoint\n");
1281 if (Bp
->flags
& BPT_DLACTIVITY
)
1282 (void) fprintf(stderr
, "function_trace(): "
1283 "RD_DLACTIVITY breakpoint\n");
1285 if (rd_event_getmsg(Rdb_agent
, &event_msg
) == RD_OK
) {
1286 if (event_msg
.type
== RD_DLACTIVITY
) {
1287 switch (event_msg
.u
.state
) {
1289 establish_breakpoints();
1292 not_consist
= TRUE
; /* kludge */
1293 establish_breakpoints();
1294 not_consist
= FALSE
;
1297 delete_library
= TRUE
;
1307 switch (event_msg
.type
) {
1318 et
= "RD_DLACTIVITY";
1321 (void) sprintf(buf
, "0x%x",
1326 (void) fprintf(stderr
,
1327 "event_msg.type = %s ", et
);
1328 switch (event_msg
.u
.state
) {
1333 et
= "RD_CONSISTENT";
1342 (void) sprintf(buf
, "0x%x",
1347 (void) fprintf(stderr
,
1348 "event_msg.u.state = %s\n", et
);
1353 if ((Bp
->flags
& BPT_TD_CREATE
) && !clear
) {
1356 (void) fprintf(stderr
, "function_trace(): "
1357 "BPT_TD_CREATE breakpoint\n");
1358 /* we don't care about the event message */
1364 if ((Stk
= callstack_info(sp
, fp
, 1)) == NULL
) {
1365 if (Dp
!= NULL
&& !clear
) {
1367 add_fcall(fcall_tbl
, Dp
->prt_name
,
1368 Bp
->sym_name
, (unsigned long)1);
1371 show_function_call(pri
, NULL
, Dp
, Bp
);
1372 if ((Bp
->flags
& BPT_HANG
) && !first
)
1375 } else if (!clear
) {
1377 function_entry(pri
, Bp
, Stk
);
1378 if ((Bp
->flags
& BPT_HANG
) && !first
)
1381 function_return(pri
, Stk
);
1387 * Single-step the traced instruction. Since it's possible that
1388 * another thread has deactivated this breakpoint, we indicate
1389 * that we have reactivated it by virtue of executing it.
1391 * To avoid a deadlock with some other thread in the process
1392 * performing a fork() or a thr_suspend() operation, we must
1393 * drop and later reacquire truss_lock. Some fancy dancing here.
1395 active
= (Bp
->flags
& BPT_ACTIVE
);
1396 Bp
->flags
|= BPT_ACTIVE
;
1398 (void) mutex_unlock(&truss_lock
);
1399 (void) Lxecbkpt(Lwp
, instr
);
1400 (void) mutex_lock(&truss_lock
);
1402 if (rval
|| clear
) { /* leave process stopped and abandoned */
1405 * Leave it stopped in a state that a stack trace is reasonable.
1407 /* XX64 needs to be updated for amd64 & gcc */
1408 if (rval
&& instr
== 0x55) { /* pushl %ebp */
1409 /* step it over the movl %esp,%ebp */
1410 (void) mutex_unlock(&truss_lock
);
1411 (void) Lsetrun(Lwp
, 0, PRCFAULT
|PRSTEP
);
1412 /* we're wrapping up; wait one second at most */
1413 (void) Lwait(Lwp
, MILLISEC
);
1414 (void) mutex_lock(&truss_lock
);
1417 if (get_bkpt(pc
) != Bp
)
1418 abend("function_trace: lost breakpoint", NULL
);
1419 (void) Pdelbkpt(Proc
, Bp
->addr
, Bp
->instr
);
1420 Bp
->flags
&= ~BPT_ACTIVE
;
1421 (void) mutex_unlock(&truss_lock
);
1422 (void) Lsetrun(Lwp
, 0, PRCFAULT
|PRSTOP
);
1423 /* we're wrapping up; wait one second at most */
1424 (void) Lwait(Lwp
, MILLISEC
);
1425 (void) mutex_lock(&truss_lock
);
1427 if (get_bkpt(pc
) != Bp
)
1428 abend("function_trace: lost breakpoint", NULL
);
1429 if (!active
|| !(Bp
->flags
& BPT_ACTIVE
)) {
1430 (void) Pdelbkpt(Proc
, Bp
->addr
, Bp
->instr
);
1431 Bp
->flags
&= ~BPT_ACTIVE
;
1438 function_entry(private_t
*pri
, struct bkpt
*Bp
, struct callstack
*Stk
)
1440 const lwpstatus_t
*Lsp
= pri
->lwpstat
;
1441 uintptr_t sp
= Lsp
->pr_reg
[R_SP
];
1442 uintptr_t rpc
= get_return_address(&sp
);
1443 struct dynlib
*Dp
= Bp
->dyn
;
1444 int oldframe
= FALSE
;
1448 if (data_model
!= PR_MODEL_LP64
) {
1450 rpc
= (uint32_t)rpc
;
1455 * If the sp is not within the stack bounds, forget it.
1456 * If the symbol's 'internal' flag is false,
1457 * don't report internal calls within the library.
1459 if (!(sp
>= Stk
->stkbase
&& sp
< Stk
->stkend
) ||
1460 (!(Bp
->flags
& BPT_INTERNAL
) &&
1461 rpc
>= Dp
->base
&& rpc
< Dp
->base
+ Dp
->size
))
1464 for (i
= 0; i
< Stk
->ncall
; i
++) {
1465 if (sp
>= Stk
->stack
[i
].sp
) {
1467 if (sp
== Stk
->stack
[i
].sp
)
1474 * Breakpoints for function returns are set here
1475 * If we're counting function calls, there is no need to set
1476 * a breakpoint upon return
1479 if (!oldframe
&& !cflag
) {
1480 (void) create_bkpt(rpc
, 1, 1); /* may or may not be set */
1481 Stk
->stack
[Stk
->ncall
].sp
= sp
; /* record it anyeay */
1482 Stk
->stack
[Stk
->ncall
].pc
= rpc
;
1483 Stk
->stack
[Stk
->ncall
].fcn
= Bp
;
1487 add_fcall(fcall_tbl
, Dp
->prt_name
, Bp
->sym_name
,
1490 show_function_call(pri
, Stk
, Dp
, Bp
);
1495 * We are here because we hit an unnamed breakpoint.
1496 * Attempt to match this up with a return pc on the stack
1497 * and report the function return.
1500 function_return(private_t
*pri
, struct callstack
*Stk
)
1502 const lwpstatus_t
*Lsp
= pri
->lwpstat
;
1503 uintptr_t sp
= Lsp
->pr_reg
[R_SP
];
1504 uintptr_t fp
= Lsp
->pr_reg
[R_FP
];
1508 if (data_model
!= PR_MODEL_LP64
) {
1517 for (i
= Stk
->ncall
- 1; i
>= 0; i
--) {
1518 if (sp
<= Stk
->stack
[i
].sp
&& fp
> Stk
->stack
[i
].sp
) {
1524 #if defined(i386) || defined(__amd64)
1526 /* probably __mul64() or friends -- try harder */
1528 for (j
= 0; i
< 0 && j
< 8; j
++) { /* up to 8 args */
1530 for (i
= Stk
->ncall
- 1; i
>= 0; i
--) {
1531 if (sp
<= Stk
->stack
[i
].sp
&&
1532 fp
> Stk
->stack
[i
].sp
) {
1541 if ((i
>= 0) && (!cflag
)) {
1542 show_function_return(pri
, Lsp
->pr_reg
[R_R0
], 0,
1543 Stk
, Stk
->stack
[i
].fcn
->dyn
, Stk
->stack
[i
].fcn
);
1547 #if defined(__sparc)
1549 #elif defined(__amd64)
1551 #elif defined(__i386)
1556 trap_one_stack(prgregset_t reg
)
1560 struct callstack
*Stk
;
1563 uintptr_t sp
= reg
[R_SP
];
1564 uintptr_t pc
= reg
[R_PC
];
1568 uint_t maxframe
= 8;
1570 uintptr_t sp
; /* %sp within called function */
1571 uintptr_t pc
; /* %pc within called function */
1572 uintptr_t rsp
; /* the return sp */
1573 uintptr_t rpc
; /* the return pc */
1574 } *frame
= my_malloc(maxframe
* sizeof (*frame
), NULL
);
1577 * Gather stack frames bottom to top.
1580 fp
= sp
; /* remember higest non-null sp */
1581 frame
[nframe
].sp
= sp
;
1582 frame
[nframe
].pc
= pc
;
1583 sp
= previous_fp(sp
, &pc
);
1584 frame
[nframe
].rsp
= sp
;
1585 frame
[nframe
].rpc
= pc
;
1586 if (++nframe
== maxframe
) {
1588 frame
= my_realloc(frame
, maxframe
* sizeof (*frame
),
1594 * Scan for function return breakpoints top to bottom.
1597 /* lookup the called function in the symbol tables */
1598 if (Plookup_by_addr(Proc
, frame
[nframe
].pc
, sym_name
,
1599 sizeof (sym_name
), &sym
) != 0)
1602 pc
= sym
.st_value
; /* entry point of the function */
1603 rpc
= frame
[nframe
].rpc
; /* caller's return pc */
1605 /* lookup the function in the breakpoint table */
1606 if ((Bp
= get_bkpt(pc
)) == NULL
|| (Dp
= Bp
->dyn
) == NULL
)
1609 if (!(Bp
->flags
& BPT_INTERNAL
) &&
1610 rpc
>= Dp
->base
&& rpc
< Dp
->base
+ Dp
->size
)
1613 sp
= frame
[nframe
].rsp
+ FPADJUST
; /* %sp at time of call */
1614 if ((Stk
= callstack_info(sp
, fp
, 0)) == NULL
)
1615 continue; /* can't happen? */
1617 if (create_bkpt(rpc
, 1, 1) != NULL
) {
1618 Stk
->stack
[Stk
->ncall
].sp
= sp
;
1619 Stk
->stack
[Stk
->ncall
].pc
= rpc
;
1620 Stk
->stack
[Stk
->ncall
].fcn
= Bp
;
1629 lwp_stack_traps(void *cd
, const lwpstatus_t
*Lsp
)
1631 ph_map_t
*ph_map
= (ph_map_t
*)cd
;
1634 (void) memcpy(reg
, Lsp
->pr_reg
, sizeof (prgregset_t
));
1635 make_lwp_stack(Lsp
, ph_map
->pmap
, ph_map
->nmap
);
1636 trap_one_stack(reg
);
1638 return (interrupt
| sigusr1
);
1643 thr_stack_traps(const td_thrhandle_t
*Thp
, void *cd
)
1648 * We have already dealt with all the lwps.
1649 * We only care about unbound threads here (TD_PARTIALREG).
1651 if (td_thr_getgregs(Thp
, reg
) != TD_PARTIALREG
)
1654 make_thr_stack(Thp
, reg
);
1655 trap_one_stack(reg
);
1657 return (interrupt
| sigusr1
);
1661 #if defined(__i386) || defined(__amd64)
1664 previous_fp(uintptr_t fp
, uintptr_t *rpc
)
1669 if (Pread(Proc
, frame
, sizeof (frame
), fp
) != sizeof (frame
) ||
1671 Pread(Proc
, trash
, sizeof (trash
), frame
[0]) != sizeof (trash
)))
1672 frame
[0] = frame
[1] = 0;
1681 #if defined(__amd64) || defined(__i386)
1684 * Examine the instruction at the return location of a function call
1685 * and return the byte count by which the stack is adjusted on return.
1686 * It the instruction at the return location is an addl, as expected,
1687 * then adjust the return pc by the size of that instruction so that
1688 * we will place the return breakpoint on the following instruction.
1689 * This allows programs that interrogate their own stacks and record
1690 * function calls and arguments to work correctly even while we interfere.
1691 * Return the count on success, -1 on failure.
1694 return_count32(uint32_t *ppc
)
1696 uintptr_t pc
= *ppc
;
1699 uchar_t instr
[6]; /* instruction at pc */
1701 if ((count
= Pread(Proc
, instr
, sizeof (instr
), pc
)) < 0)
1704 /* find the replaced instruction at pc (if any) */
1705 if ((Bp
= get_bkpt(pc
)) != NULL
&& (Bp
->flags
& BPT_ACTIVE
))
1706 instr
[0] = (uchar_t
)Bp
->instr
;
1708 if (count
!= sizeof (instr
) &&
1709 (count
< 3 || instr
[0] != 0x83))
1713 * A bit of disassembly of the instruction is required here.
1715 if (instr
[1] != 0xc4) { /* not an addl mumble,%esp inctruction */
1717 } else if (instr
[0] == 0x81) { /* count is a longword */
1718 count
= instr
[2]+(instr
[3]<<8)+(instr
[4]<<16)+(instr
[5]<<24);
1720 } else if (instr
[0] == 0x83) { /* count is a byte */
1723 } else { /* not an addl inctruction */
1731 get_return_address32(uintptr_t *psp
)
1737 *psp
+= 4; /* account for popping the stack on return */
1738 if (Pread(Proc
, &rpc
, sizeof (rpc
), sp
) != sizeof (rpc
))
1740 if ((count
= return_count32(&rpc
)) < 0)
1742 *psp
+= count
; /* expected sp on return */
1747 get_return_address(uintptr_t *psp
)
1751 uintptr_t sp
= *psp
;
1753 if (data_model
== PR_MODEL_LP64
) {
1754 if (Pread(Proc
, &rpc
, sizeof (rpc
), sp
) != sizeof (rpc
))
1757 * Ignore arguments pushed on the stack. See comments in
1763 return (get_return_address32(psp
));
1768 get_arguments32(long *argp
)
1770 private_t
*pri
= get_private();
1771 const lwpstatus_t
*Lsp
= pri
->lwpstat
;
1772 uint32_t frame
[5]; /* return pc + 4 args */
1777 narg
= Pread(Proc
, frame
, sizeof (frame
),
1778 (uintptr_t)Lsp
->pr_reg
[R_SP
]);
1779 narg
-= sizeof (greg32_t
);
1782 narg
/= sizeof (greg32_t
); /* no more than 4 */
1785 * Given the return PC, determine the number of arguments.
1787 if ((count
= return_count32(&frame
[0])) < 0)
1790 count
/= sizeof (greg32_t
);
1795 for (i
= 0; i
< narg
; i
++)
1796 argp
[i
] = (long)frame
[i
+1];
1802 get_arguments(long *argp
)
1805 private_t
*pri
= get_private();
1806 const lwpstatus_t
*Lsp
= pri
->lwpstat
;
1808 if (data_model
== PR_MODEL_LP64
) {
1810 * On amd64, we do not know how many arguments are passed to
1811 * each function. While it may be possible to detect if we
1812 * have more than 6 arguments, it is of marginal value.
1813 * Instead, assume that we always have 6 arguments, which are
1814 * passed via registers.
1816 argp
[0] = Lsp
->pr_reg
[REG_RDI
];
1817 argp
[1] = Lsp
->pr_reg
[REG_RSI
];
1818 argp
[2] = Lsp
->pr_reg
[REG_RDX
];
1819 argp
[3] = Lsp
->pr_reg
[REG_RCX
];
1820 argp
[4] = Lsp
->pr_reg
[REG_R8
];
1821 argp
[5] = Lsp
->pr_reg
[REG_R9
];
1825 return (get_arguments32(argp
));
1828 #endif /* __amd64 || __i386 */