| |
| /*--------------------------------------------------------------------*/ |
| /*--- Stuff relating to skin data structures. ---*/ |
| /*--- vg_needs.c ---*/ |
| /*--------------------------------------------------------------------*/ |
| |
| /* |
| This file is part of Valgrind, an extensible x86 protected-mode |
| emulator for monitoring program execution on x86-Unixes. |
| |
| Copyright (C) 2000-2003 Nicholas Nethercote |
| njn25@cam.ac.uk |
| |
| This program is free software; you can redistribute it and/or |
| modify it under the terms of the GNU General Public License as |
| published by the Free Software Foundation; either version 2 of the |
| License, or (at your option) any later version. |
| |
| This program is distributed in the hope that it will be useful, but |
| WITHOUT ANY WARRANTY; without even the implied warranty of |
| MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| General Public License for more details. |
| |
| You should have received a copy of the GNU General Public License |
| along with this program; if not, write to the Free Software |
| Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA |
| 02111-1307, USA. |
| |
| The GNU General Public License is contained in the file COPYING. |
| */ |
| |
| #include "vg_include.h" |
| |
| |
| /* --------------------------------------------------------------------- |
| Skin data structure initialisation |
| ------------------------------------------------------------------ */ |
| |
| /* Init with default values. */ |
| VgDetails VG_(details) = { |
| .name = NULL, |
| .version = NULL, |
| .description = NULL, |
| .copyright_author = NULL, |
| .bug_reports_to = NULL, |
| .avg_translation_sizeB = VG_DEFAULT_TRANS_SIZEB, |
| }; |
| |
| VgNeeds VG_(needs) = { |
| .core_errors = False, |
| .skin_errors = False, |
| .libc_freeres = False, |
| .basic_block_discards = False, |
| .shadow_regs = False, |
| .command_line_options = False, |
| .client_requests = False, |
| .extended_UCode = False, |
| .syscall_wrapper = False, |
| .sanity_checks = False, |
| .data_syms = False, |
| }; |
| |
| VgTrackEvents VG_(track_events) = { |
| /* Memory events */ |
| .new_mem_startup = NULL, |
| .new_mem_stack_signal = NULL, |
| .new_mem_brk = NULL, |
| .new_mem_mmap = NULL, |
| |
| .copy_mem_remap = NULL, |
| .change_mem_mprotect = NULL, |
| |
| .die_mem_stack_signal = NULL, |
| .die_mem_brk = NULL, |
| .die_mem_munmap = NULL, |
| |
| .new_mem_stack_4 = NULL, |
| .new_mem_stack_8 = NULL, |
| .new_mem_stack_12 = NULL, |
| .new_mem_stack_16 = NULL, |
| .new_mem_stack_32 = NULL, |
| .new_mem_stack = NULL, |
| |
| .die_mem_stack_4 = NULL, |
| .die_mem_stack_8 = NULL, |
| .die_mem_stack_12 = NULL, |
| .die_mem_stack_16 = NULL, |
| .die_mem_stack_32 = NULL, |
| .die_mem_stack = NULL, |
| |
| .ban_mem_stack = NULL, |
| |
| .pre_mem_read = NULL, |
| .pre_mem_read_asciiz = NULL, |
| .pre_mem_write = NULL, |
| .post_mem_write = NULL, |
| |
| /* Register events */ |
| .post_regs_write_init = NULL, |
| .post_reg_write_syscall_return = NULL, |
| .post_reg_write_deliver_signal = NULL, |
| .post_reg_write_pthread_return = NULL, |
| .post_reg_write_clientreq_return = NULL, |
| .post_reg_write_clientcall_return = NULL, |
| |
| /* Scheduler events */ |
| .thread_run = NULL, |
| |
| /* Mutex events */ |
| .post_mutex_lock = NULL, |
| .post_mutex_unlock = NULL, |
| |
| /* Signal events */ |
| .pre_deliver_signal = NULL, |
| .post_deliver_signal = NULL, |
| }; |
| |
| /* static */ |
| void VG_(sanity_check_needs) ( void) |
| { |
| #define CHECK_NOT(var, value) \ |
| if ((var)==(value)) { \ |
| VG_(printf)("\nSkin error: `%s' not initialised\n", \ |
| VG__STRING(var)); \ |
| VG_(skin_panic)("Uninitialised details field\n"); \ |
| } |
| |
| /* Ones that must be set */ |
| CHECK_NOT(VG_(details).name, NULL); |
| /* Nb: .version can be NULL */ |
| CHECK_NOT(VG_(details).description, NULL); |
| CHECK_NOT(VG_(details).copyright_author, NULL); |
| CHECK_NOT(VG_(details).bug_reports_to, NULL); |
| |
| if ( (VG_(track_events).new_mem_stack_4 || |
| VG_(track_events).new_mem_stack_8 || |
| VG_(track_events).new_mem_stack_12 || |
| VG_(track_events).new_mem_stack_16 || |
| VG_(track_events).new_mem_stack_32) && |
| ! VG_(track_events).new_mem_stack) |
| { |
| VG_(printf)("\nSkin error: one of the specialised `new_mem_stack_n'\n" |
| "events tracked, but not the generic `new_mem_stack' one.\n"); |
| VG_(skin_panic)("`new_mem_stack' should be defined\n"); |
| } |
| |
| if ( (VG_(track_events).die_mem_stack_4 || |
| VG_(track_events).die_mem_stack_8 || |
| VG_(track_events).die_mem_stack_12 || |
| VG_(track_events).die_mem_stack_16 || |
| VG_(track_events).die_mem_stack_32) && |
| ! VG_(track_events).die_mem_stack) |
| { |
| VG_(printf)("\nSkin error: one of the specialised `die_mem_stack_n'\n" |
| "events tracked, but not the generic `die_mem_stack' one.\n"); |
| VG_(skin_panic)("`die_mem_stack' should be defined\n"); |
| } |
| |
| if ( (VG_(track_events).post_reg_write_syscall_return || |
| VG_(track_events).post_reg_write_deliver_signal || |
| VG_(track_events).post_reg_write_pthread_return || |
| VG_(track_events).post_reg_write_clientreq_return || |
| VG_(track_events).post_reg_write_clientcall_return) && |
| ! VG_(needs).shadow_regs) |
| { |
| VG_(printf)("\nSkin error: one of the `post_reg_write'\n" |
| "events tracked, but `shadow_regs' need not set.\n"); |
| VG_(skin_panic)("`shadow_regs' should be set\n"); |
| } |
| |
| #undef CHECK_NOT |
| #undef INVALID_Bool |
| } |
| |
| /*--------------------------------------------------------------------*/ |
| /* Setting details */ |
| |
| /* Use macro because they're so repetitive */ |
| #define DETAILS(type, detail) \ |
| extern void VG_(details_##detail)(type detail) \ |
| { \ |
| VG_(details).detail = detail; \ |
| } |
| |
| DETAILS(Char*, name) |
| DETAILS(Char*, version) |
| DETAILS(Char*, description) |
| DETAILS(Char*, copyright_author) |
| DETAILS(Char*, bug_reports_to) |
| DETAILS(UInt, avg_translation_sizeB) |
| |
| /*--------------------------------------------------------------------*/ |
| /* Setting needs */ |
| |
| /* Use macro because they're so repetitive */ |
| #define NEEDS(need) \ |
| extern void VG_(needs_##need)(void) \ |
| { \ |
| VG_(needs).need = True; \ |
| } |
| |
| NEEDS(libc_freeres) |
| NEEDS(core_errors) |
| NEEDS(skin_errors) |
| NEEDS(basic_block_discards) |
| NEEDS(shadow_regs) |
| NEEDS(command_line_options) |
| NEEDS(client_requests) |
| NEEDS(extended_UCode) |
| NEEDS(syscall_wrapper) |
| NEEDS(sanity_checks) |
| NEEDS(data_syms) |
| |
| /*--------------------------------------------------------------------*/ |
| #define TRACK(event, args...) \ |
| void VG_(track_##event)(void (*f)(args)) \ |
| { \ |
| VG_(track_events).event = f; \ |
| } |
| |
| /* Memory events */ |
| TRACK(new_mem_startup, Addr a, UInt len, Bool rr, Bool ww, Bool xx) |
| TRACK(new_mem_stack_signal, Addr a, UInt len) |
| TRACK(new_mem_brk, Addr a, UInt len) |
| TRACK(new_mem_mmap, Addr a, UInt len, Bool rr, Bool ww, Bool xx) |
| |
| TRACK(copy_mem_remap, Addr from, Addr to, UInt len) |
| TRACK(change_mem_mprotect, Addr a, UInt len, Bool rr, Bool ww, Bool xx) |
| |
| TRACK(die_mem_stack_signal, Addr a, UInt len) |
| TRACK(die_mem_brk, Addr a, UInt len) |
| TRACK(die_mem_munmap, Addr a, UInt len) |
| |
| TRACK(new_mem_stack_4, Addr new_ESP) |
| TRACK(new_mem_stack_8, Addr new_ESP) |
| TRACK(new_mem_stack_12, Addr new_ESP) |
| TRACK(new_mem_stack_16, Addr new_ESP) |
| TRACK(new_mem_stack_32, Addr new_ESP) |
| TRACK(new_mem_stack, Addr a, UInt len) |
| |
| TRACK(die_mem_stack_4, Addr new_ESP) |
| TRACK(die_mem_stack_8, Addr new_ESP) |
| TRACK(die_mem_stack_12, Addr new_ESP) |
| TRACK(die_mem_stack_16, Addr new_ESP) |
| TRACK(die_mem_stack_32, Addr new_ESP) |
| TRACK(die_mem_stack, Addr a, UInt len) |
| |
| TRACK(ban_mem_stack, Addr a, UInt len) |
| |
| TRACK(pre_mem_read, CorePart part, ThreadId tid, Char* s, Addr a, |
| UInt size) |
| TRACK(pre_mem_read_asciiz, CorePart part, ThreadId tid, Char* s, Addr a) |
| TRACK(pre_mem_write, CorePart part, ThreadId tid, Char* s, Addr a, |
| UInt size) |
| TRACK(post_mem_write, Addr a, UInt size) |
| |
| TRACK(post_regs_write_init, void ); |
| TRACK(post_reg_write_syscall_return, ThreadId tid, UInt reg ); |
| TRACK(post_reg_write_deliver_signal, ThreadId tid, UInt reg ); |
| TRACK(post_reg_write_pthread_return, ThreadId tid, UInt reg ); |
| TRACK(post_reg_write_clientreq_return, ThreadId tid, UInt reg ); |
| TRACK(post_reg_write_clientcall_return, ThreadId tid, UInt reg, Addr f ); |
| |
| TRACK(thread_run, ThreadId tid) |
| |
| TRACK(post_thread_create, ThreadId tid, ThreadId child) |
| TRACK(post_thread_join, ThreadId joiner, ThreadId joinee) |
| |
| TRACK( pre_mutex_lock, ThreadId tid, void* /*pthread_mutex_t* */ mutex) |
| TRACK(post_mutex_lock, ThreadId tid, void* /*pthread_mutex_t* */ mutex) |
| TRACK(post_mutex_unlock, ThreadId tid, void* /*pthread_mutex_t* */ mutex) |
| |
| TRACK( pre_deliver_signal, ThreadId tid, Int sigNum, Bool alt_stack) |
| TRACK(post_deliver_signal, ThreadId tid, Int sigNum) |
| |
| /*--------------------------------------------------------------------*/ |
| /* UCodeBlocks */ |
| |
| Int VG_(get_num_instrs) ( UCodeBlock* cb ) |
| { |
| return cb->used; |
| } |
| |
| Int VG_(get_num_temps) ( UCodeBlock* cb ) |
| { |
| return cb->nextTemp; |
| } |
| |
| UInstr* VG_(get_instr) ( UCodeBlock* cb, Int i ) |
| { |
| return & cb->instrs[i]; |
| } |
| |
| UInstr* VG_(get_last_instr) ( UCodeBlock* cb ) |
| { |
| return & cb->instrs[cb->used-1]; |
| } |
| |
| /*--------------------------------------------------------------------*/ |
| /* Suppressions */ |
| |
| SuppKind VG_(get_supp_kind) ( Supp* su ) |
| { |
| return su->skind; |
| } |
| |
| Char* VG_(get_supp_string) ( Supp* su ) |
| { |
| return su->string; |
| } |
| |
| void* VG_(get_supp_extra) ( Supp* su ) |
| { |
| return su->extra; |
| } |
| |
| |
| void VG_(set_supp_kind) ( Supp* su, SuppKind skind ) |
| { |
| su->skind = skind; |
| } |
| |
| void VG_(set_supp_string) ( Supp* su, Char* string ) |
| { |
| su->string = string; |
| } |
| |
| void VG_(set_supp_extra) ( Supp* su, void* extra ) |
| { |
| su->extra = extra; |
| } |
| |
| /*--------------------------------------------------------------------*/ |
| /* Errors */ |
| |
| ExeContext* VG_(get_error_where) ( Error* err ) |
| { |
| return err->where; |
| } |
| |
| ErrorKind VG_(get_error_kind) ( Error* err ) |
| { |
| return err->ekind; |
| } |
| |
| Addr VG_(get_error_address) ( Error* err ) |
| { |
| return err->addr; |
| } |
| |
| Char* VG_(get_error_string) ( Error* err ) |
| { |
| return err->string; |
| } |
| |
| void* VG_(get_error_extra) ( Error* err ) |
| { |
| return err->extra; |
| } |
| |
| /*--------------------------------------------------------------------*/ |
| /*--- end vg_needs.c ---*/ |
| /*--------------------------------------------------------------------*/ |
| |
| |