mirror of
https://github.com/tobast/libunwind-eh_elf.git
synced 2025-01-25 17:50:29 +01:00
Pull cache-related fields out of reg_state, and put them into reg_cache struct.
This commit is contained in:
parent
59ecb24ca2
commit
50a457abb0
2 changed files with 60 additions and 57 deletions
|
@ -260,15 +260,20 @@ typedef struct dwarf_reg_state
|
||||||
{
|
{
|
||||||
struct dwarf_reg_state *next; /* for rs_stack */
|
struct dwarf_reg_state *next; /* for rs_stack */
|
||||||
dwarf_save_loc_t reg[DWARF_NUM_PRESERVED_REGS + 2];
|
dwarf_save_loc_t reg[DWARF_NUM_PRESERVED_REGS + 2];
|
||||||
unw_word_t ip; /* ip this rs is for */
|
|
||||||
unw_word_t ret_addr_column; /* indicates which column in the rule table represents return address */
|
unw_word_t ret_addr_column; /* indicates which column in the rule table represents return address */
|
||||||
|
unsigned short signal_frame : 1; /* optional machine-dependent signal info */
|
||||||
|
}
|
||||||
|
dwarf_reg_state_t;
|
||||||
|
|
||||||
|
typedef struct dwarf_reg_cache_entry
|
||||||
|
{
|
||||||
|
unw_word_t ip; /* ip this rs is for */
|
||||||
unsigned short lru_chain; /* used for least-recently-used chain */
|
unsigned short lru_chain; /* used for least-recently-used chain */
|
||||||
unsigned short coll_chain; /* used for hash collisions */
|
unsigned short coll_chain; /* used for hash collisions */
|
||||||
unsigned short hint; /* hint for next rs to try (or -1) */
|
unsigned short hint; /* hint for next rs to try (or -1) */
|
||||||
unsigned short valid : 1; /* optional machine-dependent signal info */
|
unsigned short valid : 1; /* optional machine-dependent signal info */
|
||||||
unsigned short signal_frame : 1; /* optional machine-dependent signal info */
|
|
||||||
}
|
}
|
||||||
dwarf_reg_state_t;
|
dwarf_reg_cache_entry_t;
|
||||||
|
|
||||||
typedef struct dwarf_cie_info
|
typedef struct dwarf_cie_info
|
||||||
{
|
{
|
||||||
|
@ -349,10 +354,12 @@ struct dwarf_rs_cache
|
||||||
|
|
||||||
/* rs cache: */
|
/* rs cache: */
|
||||||
dwarf_reg_state_t *buckets;
|
dwarf_reg_state_t *buckets;
|
||||||
|
dwarf_reg_cache_entry_t *links;
|
||||||
|
|
||||||
/* default memory, loaded in BSS segment */
|
/* default memory, loaded in BSS segment */
|
||||||
unsigned short default_hash[DWARF_DEFAULT_UNW_HASH_SIZE];
|
unsigned short default_hash[DWARF_DEFAULT_UNW_HASH_SIZE];
|
||||||
dwarf_reg_state_t default_buckets[DWARF_DEFAULT_UNW_CACHE_SIZE];
|
dwarf_reg_state_t default_buckets[DWARF_DEFAULT_UNW_CACHE_SIZE];
|
||||||
|
dwarf_reg_cache_entry_t default_links[DWARF_DEFAULT_UNW_CACHE_SIZE];
|
||||||
};
|
};
|
||||||
|
|
||||||
/* A list of descriptors for loaded .debug_frame sections. */
|
/* A list of descriptors for loaded .debug_frame sections. */
|
||||||
|
|
|
@ -534,6 +534,7 @@ dwarf_flush_rs_cache (struct dwarf_rs_cache *cache)
|
||||||
|| !cache->hash) {
|
|| !cache->hash) {
|
||||||
cache->hash = cache->default_hash;
|
cache->hash = cache->default_hash;
|
||||||
cache->buckets = cache->default_buckets;
|
cache->buckets = cache->default_buckets;
|
||||||
|
cache->links = cache->default_links;
|
||||||
cache->log_size = DWARF_DEFAULT_LOG_UNW_CACHE_SIZE;
|
cache->log_size = DWARF_DEFAULT_LOG_UNW_CACHE_SIZE;
|
||||||
} else {
|
} else {
|
||||||
if (cache->hash && cache->hash != cache->default_hash)
|
if (cache->hash && cache->hash != cache->default_hash)
|
||||||
|
@ -542,11 +543,16 @@ dwarf_flush_rs_cache (struct dwarf_rs_cache *cache)
|
||||||
if (cache->buckets && cache->buckets != cache->default_buckets)
|
if (cache->buckets && cache->buckets != cache->default_buckets)
|
||||||
munmap(cache->buckets, DWARF_UNW_CACHE_SIZE(cache->prev_log_size)
|
munmap(cache->buckets, DWARF_UNW_CACHE_SIZE(cache->prev_log_size)
|
||||||
* sizeof (cache->buckets[0]));
|
* sizeof (cache->buckets[0]));
|
||||||
|
if (cache->links && cache->links != cache->default_links)
|
||||||
|
munmap(cache->links, DWARF_UNW_CACHE_SIZE(cache->prev_log_size)
|
||||||
|
* sizeof (cache->links[0]));
|
||||||
GET_MEMORY(cache->hash, DWARF_UNW_HASH_SIZE(cache->log_size)
|
GET_MEMORY(cache->hash, DWARF_UNW_HASH_SIZE(cache->log_size)
|
||||||
* sizeof (cache->hash[0]));
|
* sizeof (cache->hash[0]));
|
||||||
GET_MEMORY(cache->buckets, DWARF_UNW_CACHE_SIZE(cache->log_size)
|
GET_MEMORY(cache->buckets, DWARF_UNW_CACHE_SIZE(cache->log_size)
|
||||||
* sizeof (cache->buckets[0]));
|
* sizeof (cache->buckets[0]));
|
||||||
if (!cache->hash || !cache->buckets)
|
GET_MEMORY(cache->links, DWARF_UNW_CACHE_SIZE(cache->log_size)
|
||||||
|
* sizeof (cache->links[0]));
|
||||||
|
if (!cache->hash || !cache->buckets || !cache->links)
|
||||||
{
|
{
|
||||||
Debug (1, "Unable to allocate cache memory");
|
Debug (1, "Unable to allocate cache memory");
|
||||||
return -UNW_ENOMEM;
|
return -UNW_ENOMEM;
|
||||||
|
@ -560,10 +566,10 @@ dwarf_flush_rs_cache (struct dwarf_rs_cache *cache)
|
||||||
for (i = 0; i < DWARF_UNW_CACHE_SIZE(cache->log_size); ++i)
|
for (i = 0; i < DWARF_UNW_CACHE_SIZE(cache->log_size); ++i)
|
||||||
{
|
{
|
||||||
if (i > 0)
|
if (i > 0)
|
||||||
cache->buckets[i].lru_chain = (i - 1);
|
cache->links[i].lru_chain = (i - 1);
|
||||||
cache->buckets[i].coll_chain = -1;
|
cache->links[i].coll_chain = -1;
|
||||||
cache->buckets[i].ip = 0;
|
cache->links[i].ip = 0;
|
||||||
cache->buckets[i].valid = 0;
|
cache->links[i].valid = 0;
|
||||||
}
|
}
|
||||||
for (i = 0; i< DWARF_UNW_HASH_SIZE(cache->log_size); ++i)
|
for (i = 0; i< DWARF_UNW_HASH_SIZE(cache->log_size); ++i)
|
||||||
cache->hash[i] = -1;
|
cache->hash[i] = -1;
|
||||||
|
@ -618,35 +624,30 @@ hash (unw_word_t ip, unsigned short log_size)
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline long
|
static inline long
|
||||||
cache_match (dwarf_reg_state_t *rs, unw_word_t ip)
|
cache_match (struct dwarf_rs_cache *cache, unsigned short index, unw_word_t ip)
|
||||||
{
|
{
|
||||||
if (rs->valid && (ip == rs->ip))
|
return (cache->links[index].valid && (ip == cache->links[index].ip));
|
||||||
return 1;
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static dwarf_reg_state_t *
|
static dwarf_reg_state_t *
|
||||||
rs_lookup (struct dwarf_rs_cache *cache, struct dwarf_cursor *c)
|
rs_lookup (struct dwarf_rs_cache *cache, struct dwarf_cursor *c)
|
||||||
{
|
{
|
||||||
dwarf_reg_state_t *rs;
|
|
||||||
unsigned short index;
|
unsigned short index;
|
||||||
unw_word_t ip = c->ip;
|
unw_word_t ip = c->ip;
|
||||||
|
|
||||||
if (c->hint > 0)
|
if (c->hint > 0)
|
||||||
{
|
{
|
||||||
index = c->hint - 1;
|
index = c->hint - 1;
|
||||||
rs = cache->buckets + index;
|
if (cache_match (cache, index, ip))
|
||||||
if (cache_match (rs, ip))
|
return &cache->buckets[index];
|
||||||
return rs;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for (index = cache->hash[hash (ip, cache->log_size)];
|
for (index = cache->hash[hash (ip, cache->log_size)];
|
||||||
index < DWARF_UNW_CACHE_SIZE(cache->log_size);
|
index < DWARF_UNW_CACHE_SIZE(cache->log_size);
|
||||||
index = rs->coll_chain)
|
index = cache->links[index].coll_chain)
|
||||||
{
|
{
|
||||||
rs = cache->buckets + index;
|
if (cache_match (cache, index, ip))
|
||||||
if (cache_match (rs, ip))
|
return &cache->buckets[index];
|
||||||
return rs;
|
|
||||||
}
|
}
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
@ -654,50 +655,41 @@ rs_lookup (struct dwarf_rs_cache *cache, struct dwarf_cursor *c)
|
||||||
static inline dwarf_reg_state_t *
|
static inline dwarf_reg_state_t *
|
||||||
rs_new (struct dwarf_rs_cache *cache, struct dwarf_cursor * c)
|
rs_new (struct dwarf_rs_cache *cache, struct dwarf_cursor * c)
|
||||||
{
|
{
|
||||||
dwarf_reg_state_t *rs, *prev, *tmp;
|
dwarf_reg_state_t *rs;
|
||||||
unw_hash_index_t index;
|
unw_hash_index_t index;
|
||||||
unsigned short head;
|
unsigned short head;
|
||||||
|
|
||||||
head = cache->lru_head;
|
head = cache->lru_head;
|
||||||
rs = cache->buckets + head;
|
cache->lru_head = cache->links[head].lru_chain;
|
||||||
cache->lru_head = rs->lru_chain;
|
|
||||||
|
|
||||||
/* re-insert rs at the tail of the LRU chain: */
|
/* re-insert rs at the tail of the LRU chain: */
|
||||||
cache->buckets[cache->lru_tail].lru_chain = head;
|
cache->links[cache->lru_tail].lru_chain = head;
|
||||||
cache->lru_tail = head;
|
cache->lru_tail = head;
|
||||||
|
|
||||||
/* remove the old rs from the hash table (if it's there): */
|
/* remove the old rs from the hash table (if it's there): */
|
||||||
if (rs->ip)
|
if (cache->links[head].ip)
|
||||||
{
|
{
|
||||||
index = hash (rs->ip, cache->log_size);
|
unsigned short *pindex;
|
||||||
tmp = cache->buckets + cache->hash[index];
|
for (pindex = &cache->hash[hash (cache->links[head].ip, cache->log_size)];
|
||||||
prev = NULL;
|
*pindex < DWARF_UNW_CACHE_SIZE(cache->log_size);
|
||||||
while (1)
|
pindex = &cache->links[*pindex].coll_chain)
|
||||||
{
|
{
|
||||||
if (tmp == rs)
|
if (*pindex == head)
|
||||||
{
|
{
|
||||||
if (prev)
|
*pindex = cache->links[*pindex].coll_chain;
|
||||||
prev->coll_chain = tmp->coll_chain;
|
|
||||||
else
|
|
||||||
cache->hash[index] = tmp->coll_chain;
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
else
|
|
||||||
prev = tmp;
|
|
||||||
if (tmp->coll_chain >= DWARF_UNW_CACHE_SIZE(cache->log_size))
|
|
||||||
/* old rs wasn't in the hash-table */
|
|
||||||
break;
|
|
||||||
tmp = cache->buckets + tmp->coll_chain;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* enter new rs in the hash table */
|
/* enter new rs in the hash table */
|
||||||
index = hash (c->ip, cache->log_size);
|
index = hash (c->ip, cache->log_size);
|
||||||
rs->coll_chain = cache->hash[index];
|
cache->links[head].coll_chain = cache->hash[index];
|
||||||
cache->hash[index] = rs - cache->buckets;
|
cache->hash[index] = head;
|
||||||
|
|
||||||
rs->ip = c->ip;
|
cache->links[head].ip = c->ip;
|
||||||
rs->valid = 1;
|
cache->links[head].valid = 1;
|
||||||
|
rs = cache->buckets + head;
|
||||||
rs->ret_addr_column = c->ret_addr_column;
|
rs->ret_addr_column = c->ret_addr_column;
|
||||||
rs->signal_frame = 0;
|
rs->signal_frame = 0;
|
||||||
tdep_cache_frame (c, rs);
|
tdep_cache_frame (c, rs);
|
||||||
|
@ -889,9 +881,6 @@ dwarf_find_save_locs (struct dwarf_cursor *c)
|
||||||
(rs = rs_lookup(cache, c)))
|
(rs = rs_lookup(cache, c)))
|
||||||
{
|
{
|
||||||
/* update hint; no locking needed: single-word writes are atomic */
|
/* update hint; no locking needed: single-word writes are atomic */
|
||||||
cache->buckets[c->prev_rs].hint = rs - cache->buckets + 1;
|
|
||||||
c->prev_rs = rs - cache->buckets;
|
|
||||||
c->hint = rs->hint;
|
|
||||||
c->ret_addr_column = rs->ret_addr_column;
|
c->ret_addr_column = rs->ret_addr_column;
|
||||||
c->use_prev_instr = ! rs->signal_frame;
|
c->use_prev_instr = ! rs->signal_frame;
|
||||||
memcpy (&sr.rs_current, rs, sizeof (*rs));
|
memcpy (&sr.rs_current, rs, sizeof (*rs));
|
||||||
|
@ -906,15 +895,22 @@ dwarf_find_save_locs (struct dwarf_cursor *c)
|
||||||
if (cache && ret >= 0)
|
if (cache && ret >= 0)
|
||||||
{
|
{
|
||||||
rs = rs_new (cache, c);
|
rs = rs_new (cache, c);
|
||||||
memcpy(rs, &sr.rs_current, offsetof(struct dwarf_reg_state, ip));
|
cache->links[rs - cache->buckets].hint = 0;
|
||||||
cache->buckets[c->prev_rs].hint = rs - cache->buckets;
|
memcpy(rs, &sr.rs_current, offsetof(struct dwarf_reg_state, ret_addr_column));
|
||||||
c->prev_rs = rs - cache->buckets;
|
|
||||||
c->hint = rs->hint = 0;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (cache)
|
if (cache)
|
||||||
|
{
|
||||||
put_rs_cache (c->as, cache, &saved_mask);
|
put_rs_cache (c->as, cache, &saved_mask);
|
||||||
|
if (rs)
|
||||||
|
{
|
||||||
|
unsigned short index = rs - cache->buckets;
|
||||||
|
c->hint = cache->links[index].hint;
|
||||||
|
cache->links[c->prev_rs].hint = index + 1;
|
||||||
|
c->prev_rs = index;
|
||||||
|
}
|
||||||
|
}
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
return ret;
|
||||||
if (cache)
|
if (cache)
|
||||||
|
|
Loading…
Reference in a new issue