mirror of
https://github.com/tobast/libunwind-eh_elf.git
synced 2025-01-10 11:13:43 +01:00
Add support for multiple threads in core files
_UCD_select_thread() lets you select the active thread from the core file and _UCD_get_num_threads() tells you how many there are.
This commit is contained in:
parent
d627481783
commit
0c0ea23cf2
3 changed files with 60 additions and 27 deletions
|
@ -40,6 +40,8 @@ struct UCD_info;
|
||||||
extern struct UCD_info *_UCD_create(const char *filename);
|
extern struct UCD_info *_UCD_create(const char *filename);
|
||||||
extern void _UCD_destroy(struct UCD_info *);
|
extern void _UCD_destroy(struct UCD_info *);
|
||||||
|
|
||||||
|
extern int _UCD_get_num_threads(struct UCD_info *);
|
||||||
|
extern void _UCD_select_thread(struct UCD_info *, int);
|
||||||
extern pid_t _UCD_get_pid(struct UCD_info *);
|
extern pid_t _UCD_get_pid(struct UCD_info *);
|
||||||
extern int _UCD_get_cursig(struct UCD_info *);
|
extern int _UCD_get_cursig(struct UCD_info *);
|
||||||
extern int _UCD_add_backing_file_at_segment(struct UCD_info *, int phdr_no, const char *filename);
|
extern int _UCD_add_backing_file_at_segment(struct UCD_info *, int phdr_no, const char *filename);
|
||||||
|
|
|
@ -206,6 +206,9 @@ _UCD_create(const char *filename)
|
||||||
Debug(2, "phdr[%03d]: type:%d", i, cur->p_type);
|
Debug(2, "phdr[%03d]: type:%d", i, cur->p_type);
|
||||||
if (cur->p_type == PT_NOTE)
|
if (cur->p_type == PT_NOTE)
|
||||||
{
|
{
|
||||||
|
unsigned char *p, *note_end;
|
||||||
|
unsigned n_threads;
|
||||||
|
|
||||||
ui->note_phdr = malloc(cur->p_filesz);
|
ui->note_phdr = malloc(cur->p_filesz);
|
||||||
if (lseek(fd, cur->p_offset, SEEK_SET) != (off_t)cur->p_offset
|
if (lseek(fd, cur->p_offset, SEEK_SET) != (off_t)cur->p_offset
|
||||||
|| (uoff_t)read(fd, ui->note_phdr, cur->p_filesz) != cur->p_filesz
|
|| (uoff_t)read(fd, ui->note_phdr, cur->p_filesz) != cur->p_filesz
|
||||||
|
@ -215,33 +218,46 @@ _UCD_create(const char *filename)
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Note is three 32-bit words: */
|
note_end = (unsigned char *)ui->note_phdr + cur->p_filesz;
|
||||||
/* Elf32_Word n_namesz; Length of the note's name */
|
|
||||||
/* Elf32_Word n_descsz; Length of the note's descriptor */
|
/* Count number of threads */
|
||||||
/* Elf32_Word n_type; Type */
|
n_threads = 0;
|
||||||
/* followed by name (padded to 32 bits(?)) and then descr */
|
p = ui->note_phdr;
|
||||||
Elf32_Nhdr *note_hdr = ui->note_phdr;
|
while (p + sizeof (Elf32_Nhdr) <= note_end)
|
||||||
if (cur->p_filesz >= 3*4
|
|
||||||
&& note_hdr->n_type == NT_PRSTATUS
|
|
||||||
&& cur->p_filesz >= (3*4 + note_hdr->n_namesz + note_hdr->n_descsz + sizeof(*ui->prstatus))
|
|
||||||
)
|
|
||||||
{
|
{
|
||||||
ui->prstatus = (void*) ((((long)note_hdr + sizeof(*note_hdr) + note_hdr->n_namesz) + 3) & ~3L);
|
Elf32_Nhdr *note_hdr = (Elf32_Nhdr *)p;
|
||||||
#if 0
|
unsigned char *p_next;
|
||||||
printf("pid:%d\n", ui->prstatus->pr_pid);
|
|
||||||
printf("ebx:%ld\n", (long)ui->prstatus->pr_reg[0]);
|
p_next = p + sizeof (Elf32_Nhdr) + ((note_hdr->n_namesz + 3) & ~3L) + note_hdr->n_descsz;
|
||||||
printf("ecx:%ld\n", (long)ui->prstatus->pr_reg[1]);
|
|
||||||
printf("edx:%ld\n", (long)ui->prstatus->pr_reg[2]);
|
if (p_next >= note_end)
|
||||||
printf("esi:%ld\n", (long)ui->prstatus->pr_reg[3]);
|
break;
|
||||||
printf("edi:%ld\n", (long)ui->prstatus->pr_reg[4]);
|
|
||||||
printf("ebp:%ld\n", (long)ui->prstatus->pr_reg[5]);
|
if (note_hdr->n_type == NT_PRSTATUS)
|
||||||
printf("eax:%ld\n", (long)ui->prstatus->pr_reg[6]);
|
n_threads++;
|
||||||
printf("xds:%ld\n", (long)ui->prstatus->pr_reg[7]);
|
|
||||||
printf("xes:%ld\n", (long)ui->prstatus->pr_reg[8]);
|
p = p_next;
|
||||||
printf("xfs:%ld\n", (long)ui->prstatus->pr_reg[9]);
|
}
|
||||||
printf("xgs:%ld\n", (long)ui->prstatus->pr_reg[10]);
|
|
||||||
printf("orig_eax:%ld\n", (long)ui->prstatus->pr_reg[11]);
|
ui->n_threads = n_threads;
|
||||||
#endif
|
ui->threads = malloc(sizeof (void *) * n_threads);
|
||||||
|
|
||||||
|
n_threads = 0;
|
||||||
|
p = ui->note_phdr;
|
||||||
|
while (p + sizeof (Elf32_Nhdr) <= note_end)
|
||||||
|
{
|
||||||
|
Elf32_Nhdr *note_hdr = (Elf32_Nhdr *)p;
|
||||||
|
unsigned char *p_next;
|
||||||
|
|
||||||
|
p_next = p + sizeof (Elf32_Nhdr) + ((note_hdr->n_namesz + 3) & ~3L) + note_hdr->n_descsz;
|
||||||
|
|
||||||
|
if (p_next >= note_end)
|
||||||
|
break;
|
||||||
|
|
||||||
|
if (note_hdr->n_type == NT_PRSTATUS)
|
||||||
|
ui->threads[n_threads++] = (void*) ((((long)note_hdr + sizeof(*note_hdr) + note_hdr->n_namesz) + 3) & ~3L);
|
||||||
|
|
||||||
|
p = p_next;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (cur->p_type == PT_LOAD)
|
if (cur->p_type == PT_LOAD)
|
||||||
|
@ -263,12 +279,14 @@ _UCD_create(const char *filename)
|
||||||
cur++;
|
cur++;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!ui->prstatus)
|
if (ui->n_threads == 0)
|
||||||
{
|
{
|
||||||
Debug(0, "No NT_PRSTATUS note found in '%s'\n", filename);
|
Debug(0, "No NT_PRSTATUS note found in '%s'\n", filename);
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ui->prstatus = ui->threads[0];
|
||||||
|
|
||||||
return ui;
|
return ui;
|
||||||
|
|
||||||
err:
|
err:
|
||||||
|
@ -276,6 +294,17 @@ _UCD_create(const char *filename)
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int _UCD_get_num_threads(struct UCD_info *ui)
|
||||||
|
{
|
||||||
|
return ui->n_threads;
|
||||||
|
}
|
||||||
|
|
||||||
|
void _UCD_select_thread(struct UCD_info *ui, int n)
|
||||||
|
{
|
||||||
|
if (n >= 0 && n < ui->n_threads)
|
||||||
|
ui->prstatus = ui->threads[n];
|
||||||
|
}
|
||||||
|
|
||||||
pid_t _UCD_get_pid(struct UCD_info *ui)
|
pid_t _UCD_get_pid(struct UCD_info *ui)
|
||||||
{
|
{
|
||||||
return ui->prstatus->pr_pid;
|
return ui->prstatus->pr_pid;
|
||||||
|
|
|
@ -91,6 +91,8 @@ struct UCD_info
|
||||||
unsigned phdrs_count;
|
unsigned phdrs_count;
|
||||||
void *note_phdr; /* allocated or NULL */
|
void *note_phdr; /* allocated or NULL */
|
||||||
struct PRSTATUS_STRUCT *prstatus; /* points inside note_phdr */
|
struct PRSTATUS_STRUCT *prstatus; /* points inside note_phdr */
|
||||||
|
int n_threads;
|
||||||
|
struct PRSTATUS_STRUCT **threads;
|
||||||
|
|
||||||
struct elf_dyn_info edi;
|
struct elf_dyn_info edi;
|
||||||
};
|
};
|
||||||
|
|
Loading…
Reference in a new issue