Search code examples
cstructoffsetobjdumpdwarf

How to get struct member offset from dwarf info?


I am trying to print the DW_AT_data_member_location attribute value from the dwarf info to get the member variables offset from a struct name but no helper function seems to work out.

Here is the dwarf_info from objdump:

<1><1bf>: Abbrev Number: 5 (DW_TAG_structure_type)
<1c0>   DW_AT_name        : (indirect string, offset: 0xf0): class  
<1c4>   DW_AT_byte_size   : 208 
<1c5>   DW_AT_decl_file   : 1   
<1c6>   DW_AT_decl_line   : 10  
<1c7>   DW_AT_sibling     : <0x1f8> 
<2><1cb>: Abbrev Number: 6 (DW_TAG_member)
<1cc>   DW_AT_name        : (indirect string, offset: 0xc7): schools    
<1d0>   DW_AT_decl_file   : 1   
<1d1>   DW_AT_decl_line   : 11  
<1d2>   DW_AT_type        : <0x1f8> 
<1d6>   DW_AT_data_member_location: 2 byte block: 23 0  (DW_OP_plus_uconst: 0)
<2><1d9>: Abbrev Number: 6 (DW_TAG_member)
<1da>   DW_AT_name        : (indirect string, offset: 0xd8): size   
<1de>   DW_AT_decl_file   : 1   
<1df>   DW_AT_decl_line   : 12  
<1e0>   DW_AT_type        : <0x159> 
<1e4>   DW_AT_data_member_location: 3 byte block: 23 c8 1   (DW_OP_plus_uconst: 200)
<2><1e8>: Abbrev Number: 6 (DW_TAG_member)
<1e9>   DW_AT_name        : (indirect string, offset: 0xf6): record 
<1ed>   DW_AT_decl_file   : 1   
<1ee>   DW_AT_decl_line   : 13  
<1ef>   DW_AT_type        : <0x1b8> 
<1f3>   DW_AT_data_member_location: 3 byte block: 23 cc 1   (DW_OP_plus_uconst: 204)

I am able to get the attribute code but not attribute value:

 if(dwarf_whatattr(attrs[i],&attrcode,&error) != DW_DLV_OK)
                      printf("Error");

 printf("Attrcode: %d\n",attrcode); // This one works

 if(attrcode==DW_AT_data_member_location) 
           dwarf_formudata(attrs[i],&offset,0) // This one does not work              

What is the type of this variable ? How to get its Value? Which helper function would work here ?


Solution

  • The DWARF debugging format allows the offset of a member to be expressed as either a simple constant or as an expression that needs to be evaluated in order calculate the offset. For whatever reason the debugging information you're trying to process has expressed a simple constant offset in the form of an expression. You'll need to "evaluate" the expression to determine what the offset is. Something like this:

    if (attrcode == DW_AT_data_member_location) {
        Dwarf_Half form;
        dwarf_whatform(attrs[i], &form, &error);
        if (form == DW_FORM_data1 || form == DW_FORM_data2
            form == DW_FORM_data2 || form == DW_FORM_data4
            form == DW_FORM_data8 || form == DW_FORM_udata) {
            dwarf_formudata(attrs[i], &offset, 0);
        } else if (form == DW_FORM_sdata) {
            Dwarf_Signed soffset;
            dwarf_formsdata(attrs[i], &soffset, 0);
            if (soffset < 0) {
                 printf("unsupported negative offset\n");
                 /* FAIL */
            }
            offset = (Dwarf_Unsigned) soffset;
        } else {
            Dwarf_Locdesc **locdescs;
            Dwarf_Signed len;
            if (dwarf_loclist_n(attrs[i], &locdescs, &len,  &error) == DW_DLV_ERROR) {
                 printf("unsupported member offset\n");
                 /* FAIL */
            }
            if (len != 1
                || locdescs[0]->ld_cents != 1
                || (locdescs[0]->ld_s[0]).lr_atom != DW_OP_plus_uconst) {
                 printf("unsupported location expression\n");
                 /* FAIL */
            }
            offset = (locdescs[0]->ld_s[0]).lr_number;
        }
    }