mmap_cache: do not activate unless requested
Do not call mmap_cache functions until mmap_cache_enable is invoked. Change struct mmap_cache_t into a proxy structure, move all mmap_cache data from struct tcb inside this new structure. * Makefile.am (strace_SOURCES): Move mmap_cache.c and mmap_cache.h to libstrace_a_SOURCES. * defs.h (struct tcb): Remove mmap_cache_size and mmap_cache_generation * fields. * mmap_cache.h (struct mmap_cache_t): Rename to struct mmap_cache_entry_t, create a new struct mmap_cache_t, all users updated. (mmap_cache_delete): Remove. * mmap_cache.c (mmap_cache_delete): Rename to delete_mmap_cache, add static qualifier. (build_mmap_cache): Merge into mmap_cache_rebuild_if_invalid. * strace.c (droptcb): Replace mmap_cache_delete invocation with tcp->mmap_cache->free_fn.
This commit is contained in:
parent
a1ecb2a51c
commit
1d98b287a4
@ -75,6 +75,8 @@ libstrace_a_SOURCES = \
|
|||||||
fstatfs64.c \
|
fstatfs64.c \
|
||||||
getpagesize.c \
|
getpagesize.c \
|
||||||
ipc.c \
|
ipc.c \
|
||||||
|
mmap_cache.c \
|
||||||
|
mmap_cache.h \
|
||||||
sigreturn.c \
|
sigreturn.c \
|
||||||
socketcall.c \
|
socketcall.c \
|
||||||
statfs.c \
|
statfs.c \
|
||||||
@ -190,8 +192,6 @@ strace_SOURCES = \
|
|||||||
membarrier.c \
|
membarrier.c \
|
||||||
memfd_create.c \
|
memfd_create.c \
|
||||||
mknod.c \
|
mknod.c \
|
||||||
mmap_cache.c \
|
|
||||||
mmap_cache.h \
|
|
||||||
mmap_notify.c \
|
mmap_notify.c \
|
||||||
mmap_notify.h \
|
mmap_notify.h \
|
||||||
mmsghdr.c \
|
mmsghdr.c \
|
||||||
|
2
defs.h
2
defs.h
@ -219,8 +219,6 @@ struct tcb {
|
|||||||
struct timespec delay_expiration_time; /* When does the delay end */
|
struct timespec delay_expiration_time; /* When does the delay end */
|
||||||
|
|
||||||
struct mmap_cache_t *mmap_cache;
|
struct mmap_cache_t *mmap_cache;
|
||||||
unsigned int mmap_cache_size;
|
|
||||||
unsigned int mmap_cache_generation;
|
|
||||||
|
|
||||||
#ifdef ENABLE_STACKTRACE
|
#ifdef ENABLE_STACKTRACE
|
||||||
void *unwind_ctx;
|
void *unwind_ctx;
|
||||||
|
145
mmap_cache.c
145
mmap_cache.c
@ -46,9 +46,9 @@ mmap_cache_invalidate(struct tcb *tcp, void *unused)
|
|||||||
#endif
|
#endif
|
||||||
mmap_cache_generation++;
|
mmap_cache_generation++;
|
||||||
debug_func_msg("tgen=%u, ggen=%u, tcp=%p, cache=%p",
|
debug_func_msg("tgen=%u, ggen=%u, tcp=%p, cache=%p",
|
||||||
tcp->mmap_cache_generation,
|
tcp->mmap_cache ? tcp->mmap_cache->generation : 0,
|
||||||
mmap_cache_generation,
|
mmap_cache_generation, tcp,
|
||||||
tcp, tcp->mmap_cache);
|
tcp->mmap_cache ? tcp->mmap_cache->entry : 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -62,33 +62,66 @@ mmap_cache_enable(void)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* deleting the cache */
|
||||||
|
static void
|
||||||
|
delete_mmap_cache(struct tcb *tcp, const char *caller)
|
||||||
|
{
|
||||||
|
debug_func_msg("tgen=%u, ggen=%u, tcp=%p, cache=%p, caller=%s",
|
||||||
|
tcp->mmap_cache ? tcp->mmap_cache->generation : 0,
|
||||||
|
mmap_cache_generation, tcp,
|
||||||
|
tcp->mmap_cache ? tcp->mmap_cache->entry : 0, caller);
|
||||||
|
|
||||||
|
if (!tcp->mmap_cache)
|
||||||
|
return;
|
||||||
|
|
||||||
|
while (tcp->mmap_cache->size) {
|
||||||
|
unsigned int i = --tcp->mmap_cache->size;
|
||||||
|
free(tcp->mmap_cache->entry[i].binary_filename);
|
||||||
|
tcp->mmap_cache->entry[i].binary_filename = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
free(tcp->mmap_cache->entry);
|
||||||
|
tcp->mmap_cache->entry = NULL;
|
||||||
|
|
||||||
|
free(tcp->mmap_cache);
|
||||||
|
tcp->mmap_cache = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* caching of /proc/ID/maps for each process to speed up stack tracing
|
* caching of /proc/ID/maps for each process to speed up stack tracing
|
||||||
*
|
*
|
||||||
* The cache must be refreshed after syscalls that affect memory mappings,
|
* The cache must be refreshed after syscalls that affect memory mappings,
|
||||||
* e.g. mmap, mprotect, munmap, execve.
|
* e.g. mmap, mprotect, munmap, execve.
|
||||||
*/
|
*/
|
||||||
static void
|
extern enum mmap_cache_rebuild_result
|
||||||
build_mmap_cache(struct tcb *tcp)
|
mmap_cache_rebuild_if_invalid(struct tcb *tcp, const char *caller)
|
||||||
{
|
{
|
||||||
FILE *fp;
|
if (tcp->mmap_cache
|
||||||
struct mmap_cache_t *cache_head = tcp->mmap_cache;
|
&& tcp->mmap_cache->generation != mmap_cache_generation)
|
||||||
/* start with a small dynamically-allocated array and then expand it */
|
delete_mmap_cache(tcp, caller);
|
||||||
size_t cur_array_size = 0;
|
|
||||||
char filename[sizeof("/proc/4294967296/maps")];
|
|
||||||
char buffer[PATH_MAX + 80];
|
|
||||||
|
|
||||||
|
if (tcp->mmap_cache)
|
||||||
|
return MMAP_CACHE_REBUILD_READY;
|
||||||
|
|
||||||
|
char filename[sizeof("/proc/4294967296/maps")];
|
||||||
xsprintf(filename, "/proc/%u/maps", tcp->pid);
|
xsprintf(filename, "/proc/%u/maps", tcp->pid);
|
||||||
fp = fopen_stream(filename, "r");
|
|
||||||
|
FILE *fp = fopen_stream(filename, "r");
|
||||||
if (!fp) {
|
if (!fp) {
|
||||||
perror_msg("fopen: %s", filename);
|
perror_msg("fopen: %s", filename);
|
||||||
return;
|
return MMAP_CACHE_REBUILD_NOCACHE;
|
||||||
}
|
}
|
||||||
|
|
||||||
tcp->mmap_cache_size = 0;
|
struct mmap_cache_t cache = {
|
||||||
|
.free_fn = delete_mmap_cache,
|
||||||
|
.generation = mmap_cache_generation
|
||||||
|
};
|
||||||
|
|
||||||
|
/* start with a small dynamically-allocated array and then expand it */
|
||||||
|
size_t allocated = 0;
|
||||||
|
char buffer[PATH_MAX + 80];
|
||||||
|
|
||||||
while (fgets(buffer, sizeof(buffer), fp) != NULL) {
|
while (fgets(buffer, sizeof(buffer), fp) != NULL) {
|
||||||
struct mmap_cache_t *entry;
|
|
||||||
unsigned long start_addr, end_addr, mmap_offset;
|
unsigned long start_addr, end_addr, mmap_offset;
|
||||||
char read_bit;
|
char read_bit;
|
||||||
char write_bit;
|
char write_bit;
|
||||||
@ -120,12 +153,13 @@ build_mmap_cache(struct tcb *tcp)
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct mmap_cache_entry_t *entry;
|
||||||
/*
|
/*
|
||||||
* sanity check to make sure that we're storing
|
* sanity check to make sure that we're storing
|
||||||
* non-overlapping regions in ascending order
|
* non-overlapping regions in ascending order
|
||||||
*/
|
*/
|
||||||
if (tcp->mmap_cache_size > 0) {
|
if (cache.size > 0) {
|
||||||
entry = &cache_head[tcp->mmap_cache_size - 1];
|
entry = &cache.entry[cache.size - 1];
|
||||||
if (entry->start_addr == start_addr &&
|
if (entry->start_addr == start_addr &&
|
||||||
entry->end_addr == end_addr) {
|
entry->end_addr == end_addr) {
|
||||||
/* duplicate entry, e.g. [vsyscall] */
|
/* duplicate entry, e.g. [vsyscall] */
|
||||||
@ -143,11 +177,11 @@ build_mmap_cache(struct tcb *tcp)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (tcp->mmap_cache_size >= cur_array_size)
|
if (cache.size >= allocated)
|
||||||
cache_head = xgrowarray(cache_head, &cur_array_size,
|
cache.entry = xgrowarray(cache.entry, &allocated,
|
||||||
sizeof(*cache_head));
|
sizeof(*cache.entry));
|
||||||
|
|
||||||
entry = &cache_head[tcp->mmap_cache_size];
|
entry = &cache.entry[cache.size];
|
||||||
entry->start_addr = start_addr;
|
entry->start_addr = start_addr;
|
||||||
entry->end_addr = end_addr;
|
entry->end_addr = end_addr;
|
||||||
entry->mmap_offset = mmap_offset;
|
entry->mmap_offset = mmap_offset;
|
||||||
@ -161,73 +195,40 @@ build_mmap_cache(struct tcb *tcp)
|
|||||||
entry->major = major;
|
entry->major = major;
|
||||||
entry->minor = minor;
|
entry->minor = minor;
|
||||||
entry->binary_filename = xstrdup(binary_path);
|
entry->binary_filename = xstrdup(binary_path);
|
||||||
tcp->mmap_cache_size++;
|
cache.size++;
|
||||||
}
|
}
|
||||||
fclose(fp);
|
fclose(fp);
|
||||||
tcp->mmap_cache = cache_head;
|
|
||||||
tcp->mmap_cache_generation = mmap_cache_generation;
|
|
||||||
|
|
||||||
debug_func_msg("tgen=%u, ggen=%u, tcp=%p, cache=%p",
|
if (!cache.size)
|
||||||
tcp->mmap_cache_generation,
|
return MMAP_CACHE_REBUILD_NOCACHE;
|
||||||
mmap_cache_generation,
|
|
||||||
tcp, tcp->mmap_cache);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* deleting the cache */
|
tcp->mmap_cache = xmalloc(sizeof(*tcp->mmap_cache));
|
||||||
extern void
|
memcpy(tcp->mmap_cache, &cache, sizeof(cache));
|
||||||
mmap_cache_delete(struct tcb *tcp, const char *caller)
|
|
||||||
{
|
|
||||||
unsigned int i;
|
|
||||||
|
|
||||||
debug_func_msg("tgen=%u, ggen=%u, tcp=%p, cache=%p, caller=%s",
|
debug_func_msg("tgen=%u, ggen=%u, tcp=%p, cache=%p, caller=%s",
|
||||||
tcp->mmap_cache_generation,
|
tcp->mmap_cache->generation, mmap_cache_generation,
|
||||||
mmap_cache_generation,
|
tcp, tcp->mmap_cache->entry, caller);
|
||||||
tcp, tcp->mmap_cache, caller);
|
|
||||||
|
|
||||||
for (i = 0; i < tcp->mmap_cache_size; i++) {
|
return MMAP_CACHE_REBUILD_RENEWED;
|
||||||
free(tcp->mmap_cache[i].binary_filename);
|
|
||||||
tcp->mmap_cache[i].binary_filename = NULL;
|
|
||||||
}
|
|
||||||
free(tcp->mmap_cache);
|
|
||||||
tcp->mmap_cache = NULL;
|
|
||||||
tcp->mmap_cache_size = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
extern enum mmap_cache_rebuild_result
|
struct mmap_cache_entry_t *
|
||||||
mmap_cache_rebuild_if_invalid(struct tcb *tcp, const char *caller)
|
|
||||||
{
|
|
||||||
enum mmap_cache_rebuild_result r = MMAP_CACHE_REBUILD_READY;
|
|
||||||
if ((tcp->mmap_cache_generation != mmap_cache_generation)
|
|
||||||
&& tcp->mmap_cache)
|
|
||||||
mmap_cache_delete(tcp, caller);
|
|
||||||
|
|
||||||
if (!tcp->mmap_cache) {
|
|
||||||
r = MMAP_CACHE_REBUILD_RENEWED;
|
|
||||||
build_mmap_cache(tcp);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!(tcp->mmap_cache && tcp->mmap_cache_size))
|
|
||||||
r = MMAP_CACHE_REBUILD_NOCACHE;
|
|
||||||
|
|
||||||
return r;
|
|
||||||
}
|
|
||||||
|
|
||||||
struct mmap_cache_t *
|
|
||||||
mmap_cache_search(struct tcb *tcp, unsigned long ip)
|
mmap_cache_search(struct tcb *tcp, unsigned long ip)
|
||||||
{
|
{
|
||||||
|
if (!tcp->mmap_cache)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
int lower = 0;
|
int lower = 0;
|
||||||
int upper = (int) tcp->mmap_cache_size - 1;
|
int upper = (int) tcp->mmap_cache->size - 1;
|
||||||
|
|
||||||
while (lower <= upper) {
|
while (lower <= upper) {
|
||||||
struct mmap_cache_t *cur_mmap_cache;
|
|
||||||
int mid = (upper + lower) / 2;
|
int mid = (upper + lower) / 2;
|
||||||
|
struct mmap_cache_entry_t *entry = &tcp->mmap_cache->entry[mid];
|
||||||
|
|
||||||
cur_mmap_cache = &tcp->mmap_cache[mid];
|
if (ip >= entry->start_addr &&
|
||||||
|
ip < entry->end_addr)
|
||||||
if (ip >= cur_mmap_cache->start_addr &&
|
return entry;
|
||||||
ip < cur_mmap_cache->end_addr)
|
else if (ip < entry->start_addr)
|
||||||
return cur_mmap_cache;
|
|
||||||
else if (ip < cur_mmap_cache->start_addr)
|
|
||||||
upper = mid - 1;
|
upper = mid - 1;
|
||||||
else
|
else
|
||||||
lower = mid + 1;
|
lower = mid + 1;
|
||||||
|
13
mmap_cache.h
13
mmap_cache.h
@ -31,7 +31,15 @@
|
|||||||
* Keep a sorted array of cache entries,
|
* Keep a sorted array of cache entries,
|
||||||
* so that we can binary search through it.
|
* so that we can binary search through it.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
struct mmap_cache_t {
|
struct mmap_cache_t {
|
||||||
|
struct mmap_cache_entry_t *entry;
|
||||||
|
void (*free_fn)(struct tcb *, const char *caller);
|
||||||
|
unsigned int size;
|
||||||
|
unsigned int generation;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct mmap_cache_entry_t {
|
||||||
/**
|
/**
|
||||||
* example entry:
|
* example entry:
|
||||||
* 7fabbb09b000-7fabbb09f000 r-xp 00179000 fc:00 1180246 /lib/libc-2.11.1.so
|
* 7fabbb09b000-7fabbb09f000 r-xp 00179000 fc:00 1180246 /lib/libc-2.11.1.so
|
||||||
@ -68,13 +76,10 @@ enum mmap_cache_rebuild_result {
|
|||||||
extern void
|
extern void
|
||||||
mmap_cache_enable(void);
|
mmap_cache_enable(void);
|
||||||
|
|
||||||
extern void
|
|
||||||
mmap_cache_delete(struct tcb *, const char *caller);
|
|
||||||
|
|
||||||
extern enum mmap_cache_rebuild_result
|
extern enum mmap_cache_rebuild_result
|
||||||
mmap_cache_rebuild_if_invalid(struct tcb *, const char *caller);
|
mmap_cache_rebuild_if_invalid(struct tcb *, const char *caller);
|
||||||
|
|
||||||
extern struct mmap_cache_t *
|
extern struct mmap_cache_entry_t *
|
||||||
mmap_cache_search(struct tcb *, unsigned long ip);
|
mmap_cache_search(struct tcb *, unsigned long ip);
|
||||||
|
|
||||||
#endif /* !STRACE_MMAP_CACHE_H */
|
#endif /* !STRACE_MMAP_CACHE_H */
|
||||||
|
3
strace.c
3
strace.c
@ -819,7 +819,8 @@ droptcb(struct tcb *tcp)
|
|||||||
unwind_tcb_fin(tcp);
|
unwind_tcb_fin(tcp);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
mmap_cache_delete(tcp, __func__);
|
if (tcp->mmap_cache)
|
||||||
|
tcp->mmap_cache->free_fn(tcp, __func__);
|
||||||
|
|
||||||
nprocs--;
|
nprocs--;
|
||||||
debug_msg("dropped tcb for pid %d, %d remain", tcp->pid, nprocs);
|
debug_msg("dropped tcb for pid %d, %d remain", tcp->pid, nprocs);
|
||||||
|
@ -89,26 +89,25 @@ print_stack_frame(struct tcb *tcp,
|
|||||||
size_t *symbol_name_size)
|
size_t *symbol_name_size)
|
||||||
{
|
{
|
||||||
unw_word_t ip;
|
unw_word_t ip;
|
||||||
struct mmap_cache_t *cur_mmap_cache;
|
|
||||||
|
|
||||||
if (unw_get_reg(cursor, UNW_REG_IP, &ip) < 0) {
|
if (unw_get_reg(cursor, UNW_REG_IP, &ip) < 0) {
|
||||||
perror_msg("cannot walk the stack of process %d", tcp->pid);
|
perror_msg("cannot walk the stack of process %d", tcp->pid);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
cur_mmap_cache = mmap_cache_search(tcp, ip);
|
struct mmap_cache_entry_t *entry = mmap_cache_search(tcp, ip);
|
||||||
if (cur_mmap_cache
|
|
||||||
|
if (entry
|
||||||
/* ignore mappings that have no PROT_EXEC bit set */
|
/* ignore mappings that have no PROT_EXEC bit set */
|
||||||
&& (cur_mmap_cache->protections & MMAP_CACHE_PROT_EXECUTABLE)) {
|
&& (entry->protections & MMAP_CACHE_PROT_EXECUTABLE)) {
|
||||||
unsigned long true_offset;
|
|
||||||
unw_word_t function_offset;
|
unw_word_t function_offset;
|
||||||
|
|
||||||
get_symbol_name(cursor, symbol_name, symbol_name_size,
|
get_symbol_name(cursor, symbol_name, symbol_name_size,
|
||||||
&function_offset);
|
&function_offset);
|
||||||
true_offset = ip - cur_mmap_cache->start_addr +
|
unsigned long true_offset =
|
||||||
cur_mmap_cache->mmap_offset;
|
ip - entry->start_addr + entry->mmap_offset;
|
||||||
call_action(data,
|
call_action(data,
|
||||||
cur_mmap_cache->binary_filename,
|
entry->binary_filename,
|
||||||
*symbol_name,
|
*symbol_name,
|
||||||
function_offset,
|
function_offset,
|
||||||
true_offset);
|
true_offset);
|
||||||
@ -139,8 +138,6 @@ walk(struct tcb *tcp,
|
|||||||
|
|
||||||
if (!tcp->mmap_cache)
|
if (!tcp->mmap_cache)
|
||||||
error_func_msg_and_die("mmap_cache is NULL");
|
error_func_msg_and_die("mmap_cache is NULL");
|
||||||
if (tcp->mmap_cache_size == 0)
|
|
||||||
error_func_msg_and_die("mmap_cache is empty");
|
|
||||||
|
|
||||||
symbol_name = xmalloc(symbol_name_size);
|
symbol_name = xmalloc(symbol_name_size);
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user