/* Generate CodeView debugging info from the GCC DWARF.
Copyright (C) 2023 Free Software Foundation, Inc.
This file is part of GCC.
GCC 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 3, or (at your option) any later
version.
GCC 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 GCC; see the file COPYING3. If not see
. */
/* See gas/codeview.h in binutils for more about the constants and structs
listed below. References to Microsoft files refer to Microsoft's PDB
repository: https://github.com/microsoft/microsoft-pdb. */
#include "config.h"
#include "system.h"
#include "coretypes.h"
#include "target.h"
#include "output.h"
#include "errors.h"
#include "md5.h"
#include "function.h"
#include "version.h"
#include "tree.h"
#include "langhooks.h"
#include "dwarf2out.h"
#include "dwarf2codeview.h"
#include "rtl.h"
#ifdef CODEVIEW_DEBUGGING_INFO
#define CV_SIGNATURE_C13 4
#define DEBUG_S_SYMBOLS 0xf1
#define DEBUG_S_LINES 0xf2
#define DEBUG_S_STRINGTABLE 0xf3
#define DEBUG_S_FILECHKSMS 0xf4
#define CHKSUM_TYPE_MD5 1
#define CV_CFL_80386 0x03
#define CV_CFL_X64 0xD0
#define CV_CFL_C 0x00
#define CV_CFL_CXX 0x01
#define FIRST_TYPE 0x1000
#define LINE_LABEL "Lcvline"
#define END_FUNC_LABEL "Lcvendfunc"
#define SYMBOL_START_LABEL "Lcvsymstart"
#define SYMBOL_END_LABEL "Lcvsymend"
/* There's two bytes available for each type's size, but follow MSVC's lead in
capping the LF_FIELDLIST size at fb00 (minus 8 bytes for the LF_INDEX
pointing to the overflow entry). */
#define MAX_FIELDLIST_SIZE 0xfaf8
#define HASH_SIZE 16
/* This is enum SYM_ENUM_e in Microsoft's cvinfo.h. */
enum cv_sym_type {
S_LDATA32 = 0x110c,
S_GDATA32 = 0x110d,
S_COMPILE3 = 0x113c,
S_LPROC32_ID = 0x1146,
S_GPROC32_ID = 0x1147,
S_PROC_ID_END = 0x114f
};
/* This is enum LEAF_ENUM_e in Microsoft's cvinfo.h. */
enum cv_leaf_type {
LF_PAD1 = 0xf1,
LF_PAD2 = 0xf2,
LF_PAD3 = 0xf3,
LF_MODIFIER = 0x1001,
LF_POINTER = 0x1002,
LF_PROCEDURE = 0x1008,
LF_ARGLIST = 0x1201,
LF_FIELDLIST = 0x1203,
LF_BITFIELD = 0x1205,
LF_INDEX = 0x1404,
LF_ENUMERATE = 0x1502,
LF_ARRAY = 0x1503,
LF_CLASS = 0x1504,
LF_STRUCTURE = 0x1505,
LF_UNION = 0x1506,
LF_ENUM = 0x1507,
LF_MEMBER = 0x150d,
LF_FUNC_ID = 0x1601,
LF_CHAR = 0x8000,
LF_SHORT = 0x8001,
LF_USHORT = 0x8002,
LF_LONG = 0x8003,
LF_ULONG = 0x8004,
LF_QUADWORD = 0x8009,
LF_UQUADWORD = 0x800a
};
struct codeview_string
{
codeview_string *next;
uint32_t offset;
char *string;
};
struct string_hasher : free_ptr_hash
{
typedef const char *compare_type;
static hashval_t hash (const codeview_string *x)
{
return htab_hash_string (x->string);
}
static bool equal (const codeview_string *x, const char *y)
{
return !strcmp (x->string, y);
}
static void mark_empty (codeview_string *x)
{
if (x->string)
{
free (x->string);
x->string = NULL;
}
}
static void remove (codeview_string *&x)
{
free (x->string);
}
};
struct codeview_source_file
{
codeview_source_file *next;
unsigned int file_num;
uint32_t string_offset;
char *filename;
uint8_t hash[HASH_SIZE];
};
struct codeview_line
{
codeview_line *next;
unsigned int line_no;
unsigned int label_num;
};
struct codeview_line_block
{
codeview_line_block *next;
uint32_t file_id;
unsigned int num_lines;
codeview_line *lines, *last_line;
};
struct codeview_function
{
codeview_function *next;
function *func;
unsigned int end_label;
codeview_line_block *blocks, *last_block;
};
struct codeview_symbol
{
codeview_symbol *next;
enum cv_sym_type kind;
union
{
struct
{
uint32_t type;
char *name;
dw_die_ref die;
} data_symbol;
struct
{
uint32_t parent;
uint32_t end;
uint32_t next;
uint32_t type;
uint8_t flags;
char *name;
dw_die_ref die;
} function;
};
};
struct codeview_type
{
dw_die_ref die;
uint32_t num;
bool is_fwd_ref;
};
struct die_hasher : free_ptr_hash
{
typedef dw_die_ref compare_type;
static hashval_t hash (const codeview_type *x)
{
return htab_hash_pointer (x->die);
}
static bool equal (const codeview_type *x, const dw_die_ref y)
{
return x->die == y;
}
};
struct codeview_integer
{
bool neg;
uint64_t num;
};
struct codeview_subtype
{
struct codeview_subtype *next;
enum cv_leaf_type kind;
union
{
struct
{
char *name;
struct codeview_integer value;
} lf_enumerate;
struct
{
uint32_t type_num;
} lf_index;
struct
{
uint16_t attributes;
uint32_t type;
codeview_integer offset;
char *name;
} lf_member;
};
};
struct codeview_custom_type
{
struct codeview_custom_type *next;
uint32_t num;
enum cv_leaf_type kind;
union
{
struct
{
uint32_t base_type;
uint32_t attributes;
} lf_pointer;
struct
{
uint32_t base_type;
uint16_t modifier;
} lf_modifier;
struct
{
size_t length;
codeview_subtype *subtypes;
codeview_subtype *last_subtype;
} lf_fieldlist;
struct
{
uint16_t count;
uint16_t properties;
uint32_t underlying_type;
uint32_t fieldlist;
char *name;
} lf_enum;
struct
{
uint16_t num_members;
uint16_t properties;
uint32_t field_list;
uint32_t derived_from;
uint32_t vshape;
codeview_integer length;
char *name;
} lf_structure;
struct
{
uint32_t element_type;
uint32_t index_type;
codeview_integer length_in_bytes;
} lf_array;
struct
{
uint32_t base_type;
uint8_t length;
uint8_t position;
} lf_bitfield;
struct
{
uint32_t return_type;
uint8_t calling_convention;
uint8_t attributes;
uint16_t num_parameters;
uint32_t arglist;
} lf_procedure;
struct
{
uint32_t num_entries;
uint32_t *args;
} lf_arglist;
struct
{
uint32_t parent_scope;
uint32_t function_type;
char *name;
} lf_func_id;
};
};
struct codeview_deferred_type
{
struct codeview_deferred_type *next;
dw_die_ref type;
};
static unsigned int line_label_num;
static unsigned int func_label_num;
static unsigned int sym_label_num;
static codeview_source_file *files, *last_file;
static unsigned int num_files;
static uint32_t string_offset = 1;
static hash_table *strings_htab;
static codeview_string *strings, *last_string;
static codeview_function *funcs, *last_func;
static const char* last_filename;
static uint32_t last_file_id;
static codeview_symbol *sym, *last_sym;
static hash_table *types_htab;
static codeview_custom_type *custom_types, *last_custom_type;
static codeview_deferred_type *deferred_types, *last_deferred_type;
static uint32_t get_type_num (dw_die_ref type, bool in_struct, bool no_fwd_ref);
/* Record new line number against the current function. */
void
codeview_source_line (unsigned int line_no, const char *filename)
{
codeview_line *l;
uint32_t file_id = last_file_id;
unsigned int label_num = ++line_label_num;
targetm.asm_out.internal_label (asm_out_file, LINE_LABEL, label_num);
if (!last_func || last_func->func != cfun)
{
codeview_function *f = (codeview_function *)
xmalloc (sizeof (codeview_function));
f->next = NULL;
f->func = cfun;
f->end_label = 0;
f->blocks = f->last_block = NULL;
if (!funcs)
funcs = f;
else
last_func->next = f;
last_func = f;
}
if (filename != last_filename)
{
codeview_source_file *sf = files;
while (sf)
{
if (!strcmp (sf->filename, filename))
{
/* 0x18 is the size of the checksum entry for each file.
0x6 bytes for the header, plus 0x10 bytes for the hash,
then padded to a multiple of 4. */
file_id = sf->file_num * 0x18;
last_filename = filename;
last_file_id = file_id;
break;
}
sf = sf->next;
}
}
if (!last_func->last_block || last_func->last_block->file_id != file_id)
{
codeview_line_block *b;
b = (codeview_line_block *) xmalloc (sizeof (codeview_line_block));
b->next = NULL;
b->file_id = file_id;
b->num_lines = 0;
b->lines = b->last_line = NULL;
if (!last_func->blocks)
last_func->blocks = b;
else
last_func->last_block->next = b;
last_func->last_block = b;
}
if (last_func->last_block->last_line
&& last_func->last_block->last_line->line_no == line_no)
return;
l = (codeview_line *) xmalloc (sizeof (codeview_line));
l->next = NULL;
l->line_no = line_no;
l->label_num = label_num;
if (!last_func->last_block->lines)
last_func->last_block->lines = l;
else
last_func->last_block->last_line->next = l;
last_func->last_block->last_line = l;
last_func->last_block->num_lines++;
}
/* Adds string to the string table, returning its offset. If already present,
this returns the offset of the existing string. */
static uint32_t
add_string (const char *string)
{
codeview_string **slot;
codeview_string *s;
size_t len;
if (!strings_htab)
strings_htab = new hash_table (10);
slot = strings_htab->find_slot_with_hash (string, htab_hash_string (string),
INSERT);
if (*slot)
return (*slot)->offset;
s = (codeview_string *) xmalloc (sizeof (codeview_string));
len = strlen (string);
s->next = NULL;
s->offset = string_offset;
string_offset += len + 1;
s->string = xstrdup (string);
if (last_string)
last_string->next = s;
else
strings = s;
last_string = s;
*slot = s;
return s->offset;
}
/* A new source file has been encountered - record the details and calculate
its hash. */
void
codeview_start_source_file (const char *filename)
{
codeview_source_file *sf;
char *path;
uint32_t string_offset;
FILE *f;
path = lrealpath (filename);
string_offset = add_string (path);
free (path);
sf = files;
while (sf)
{
if (sf->string_offset == string_offset)
return;
sf = sf->next;
}
sf = (codeview_source_file *) xmalloc (sizeof (codeview_source_file));
sf->next = NULL;
sf->file_num = num_files;
sf->string_offset = string_offset;
sf->filename = xstrdup (filename);
f = fopen (filename, "r");
if (!f)
internal_error ("could not open %s for reading", filename);
if (md5_stream (f, sf->hash))
{
fclose (f);
internal_error ("md5_stream failed");
}
fclose (f);
if (last_file)
last_file->next = sf;
else
files = sf;
last_file = sf;
num_files++;
}
/* Write out the strings table into the .debug$S section. The linker will
parse this, and handle the deduplication and hashing for all the object
files. */
static void
write_strings_table (void)
{
codeview_string *string;
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, DEBUG_S_STRINGTABLE);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
asm_fprintf (asm_out_file, "%LLcv_strings_end - %LLcv_strings_start\n");
asm_fprintf (asm_out_file, "%LLcv_strings_start:\n");
/* The first entry is always an empty string. */
fputs (integer_asm_op (1, false), asm_out_file);
fprint_whex (asm_out_file, 0);
putc ('\n', asm_out_file);
string = strings;
while (string)
{
ASM_OUTPUT_ASCII (asm_out_file, string->string,
strlen (string->string) + 1);
string = string->next;
}
delete strings_htab;
asm_fprintf (asm_out_file, "%LLcv_strings_end:\n");
ASM_OUTPUT_ALIGN (asm_out_file, 2);
}
/* Write out the file checksums data into the .debug$S section. */
static void
write_source_files (void)
{
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, DEBUG_S_FILECHKSMS);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
asm_fprintf (asm_out_file,
"%LLcv_filechksms_end - %LLcv_filechksms_start\n");
asm_fprintf (asm_out_file, "%LLcv_filechksms_start:\n");
while (files)
{
codeview_source_file *next = files->next;
/* This is struct file_checksum in binutils, or filedata in Microsoft's
dumpsym7.cpp:
struct file_checksum
{
uint32_t file_id;
uint8_t checksum_length;
uint8_t checksum_type;
} ATTRIBUTE_PACKED;
followed then by the bytes of the hash, padded to the next 4 bytes.
file_id here is actually the offset in the strings table. */
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, files->string_offset);
putc ('\n', asm_out_file);
fputs (integer_asm_op (1, false), asm_out_file);
fprint_whex (asm_out_file, HASH_SIZE);
putc ('\n', asm_out_file);
fputs (integer_asm_op (1, false), asm_out_file);
fprint_whex (asm_out_file, CHKSUM_TYPE_MD5);
putc ('\n', asm_out_file);
for (unsigned int i = 0; i < HASH_SIZE; i++)
{
fputs (integer_asm_op (1, false), asm_out_file);
fprint_whex (asm_out_file, files->hash[i]);
putc ('\n', asm_out_file);
}
ASM_OUTPUT_ALIGN (asm_out_file, 2);
free (files->filename);
free (files);
files = next;
}
asm_fprintf (asm_out_file, "%LLcv_filechksms_end:\n");
}
/* Write out the line number information for each function into the
.debug$S section. */
static void
write_line_numbers (void)
{
unsigned int func_num = 0;
while (funcs)
{
codeview_function *next = funcs->next;
unsigned int first_label_num;
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, DEBUG_S_LINES);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
asm_fprintf (asm_out_file, "%LLcv_lines%u_end - %LLcv_lines%u_start\n",
func_num, func_num);
asm_fprintf (asm_out_file, "%LLcv_lines%u_start:\n", func_num);
/* Output the header (struct cv_lines_header in binutils or
CV_DebugSLinesHeader_t in Microsoft's cvinfo.h):
struct cv_lines_header
{
uint32_t offset;
uint16_t section;
uint16_t flags;
uint32_t length;
};
*/
asm_fprintf (asm_out_file, "\t.secrel32\t%L" LINE_LABEL "%u\n",
funcs->blocks->lines->label_num);
asm_fprintf (asm_out_file, "\t.secidx\t%L" LINE_LABEL "%u\n",
funcs->blocks->lines->label_num);
/* flags */
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, 0);
putc ('\n', asm_out_file);
first_label_num = funcs->blocks->lines->label_num;
/* length */
fputs (integer_asm_op (4, false), asm_out_file);
asm_fprintf (asm_out_file,
"%L" END_FUNC_LABEL "%u - %L" LINE_LABEL "%u\n",
funcs->end_label, first_label_num);
while (funcs->blocks)
{
codeview_line_block *next = funcs->blocks->next;
/* Next comes the blocks, each block being a part of a function
within the same source file (struct cv_lines_block in binutils or
CV_DebugSLinesFileBlockHeader_t in Microsoft's cvinfo.h):
struct cv_lines_block
{
uint32_t file_id;
uint32_t num_lines;
uint32_t length;
};
*/
/* file ID */
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, funcs->blocks->file_id);
putc ('\n', asm_out_file);
/* number of lines */
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, funcs->blocks->num_lines);
putc ('\n', asm_out_file);
/* length of code block: (num_lines * sizeof (struct cv_line)) +
sizeof (struct cv_lines_block) */
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, (funcs->blocks->num_lines * 0x8) + 0xc);
putc ('\n', asm_out_file);
while (funcs->blocks->lines)
{
codeview_line *next = funcs->blocks->lines->next;
/* Finally comes the line number information (struct cv_line in
binutils or CV_Line_t in Microsoft's cvinfo.h):
struct cv_line
{
uint32_t offset;
uint32_t line_no;
};
Strictly speaking line_no is a bitfield: the bottom 24 bits
are the line number, and the top bit means "is a statement".
*/
fputs (integer_asm_op (4, false), asm_out_file);
asm_fprintf (asm_out_file,
"%L" LINE_LABEL "%u - %L" LINE_LABEL "%u\n",
funcs->blocks->lines->label_num, first_label_num);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file,
0x80000000
| (funcs->blocks->lines->line_no & 0xffffff));
putc ('\n', asm_out_file);
free (funcs->blocks->lines);
funcs->blocks->lines = next;
}
free (funcs->blocks);
funcs->blocks = next;
}
free (funcs);
asm_fprintf (asm_out_file, "%LLcv_lines%u_end:\n", func_num);
func_num++;
funcs = next;
}
}
/* Treat cold sections as separate functions, for the purposes of line
numbers. */
void
codeview_switch_text_section (void)
{
codeview_function *f;
if (last_func && last_func->end_label == 0)
{
unsigned int label_num = ++func_label_num;
targetm.asm_out.internal_label (asm_out_file, END_FUNC_LABEL,
label_num);
last_func->end_label = label_num;
}
f = (codeview_function *) xmalloc (sizeof (codeview_function));
f->next = NULL;
f->func = cfun;
f->end_label = 0;
f->blocks = f->last_block = NULL;
if (!funcs)
funcs = f;
else
last_func->next = f;
last_func = f;
}
/* Mark the end of the current function. */
void
codeview_end_epilogue (void)
{
if (last_func && last_func->end_label == 0)
{
unsigned int label_num = ++func_label_num;
targetm.asm_out.internal_label (asm_out_file, END_FUNC_LABEL,
label_num);
last_func->end_label = label_num;
}
}
/* Return the CodeView constant for the selected architecture. */
static uint16_t
target_processor (void)
{
if (TARGET_64BIT)
return CV_CFL_X64;
else
return CV_CFL_80386;
}
/* Return the CodeView constant for the language being used. */
static uint32_t
language_constant (void)
{
const char *language_string = lang_hooks.name;
if (startswith (language_string, "GNU C++"))
return CV_CFL_CXX;
else if (startswith (language_string, "GNU C"))
return CV_CFL_C;
return 0;
}
/* Write a S_COMPILE3 symbol, which records the details of the compiler
being used. */
static void
write_compile3_symbol (void)
{
unsigned int label_num = ++sym_label_num;
static const char compiler_name[] = "GCC ";
/* This is struct COMPILESYM3 in binutils and Microsoft's cvinfo.h:
struct COMPILESYM3
{
uint16_t length;
uint16_t type;
uint32_t flags;
uint16_t machine;
uint16_t frontend_major;
uint16_t frontend_minor;
uint16_t frontend_build;
uint16_t frontend_qfe;
uint16_t backend_major;
uint16_t backend_minor;
uint16_t backend_build;
uint16_t backend_qfe;
} ATTRIBUTE_PACKED;
*/
fputs (integer_asm_op (2, false), asm_out_file);
asm_fprintf (asm_out_file,
"%L" SYMBOL_END_LABEL "%u - %L" SYMBOL_START_LABEL "%u\n",
label_num, label_num);
targetm.asm_out.internal_label (asm_out_file, SYMBOL_START_LABEL, label_num);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, S_COMPILE3);
putc ('\n', asm_out_file);
/* Microsoft has the flags as a bitfield, with the bottom 8 bits being the
language constant, and the reset being MSVC-specific stuff. */
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, language_constant ());
putc ('\n', asm_out_file);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, target_processor ());
putc ('\n', asm_out_file);
/* Write 8 uint16_ts for the frontend and backend versions. As with GAS, we
zero these, as it's easier to record the version in the compiler
string. */
for (unsigned int i = 0; i < 8; i++)
{
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, 0);
putc ('\n', asm_out_file);
}
ASM_OUTPUT_ASCII (asm_out_file, compiler_name, sizeof (compiler_name) - 1);
ASM_OUTPUT_ASCII (asm_out_file, version_string, strlen (version_string) + 1);
ASM_OUTPUT_ALIGN (asm_out_file, 2);
targetm.asm_out.internal_label (asm_out_file, SYMBOL_END_LABEL, label_num);
}
/* Write an S_GDATA32 symbol, representing a global variable, or an S_LDATA32
symbol, for a static global variable. */
static void
write_data_symbol (codeview_symbol *s)
{
unsigned int label_num = ++sym_label_num;
dw_attr_node *loc;
dw_loc_descr_ref loc_ref;
/* This is struct datasym in binutils:
struct datasym
{
uint16_t size;
uint16_t kind;
uint32_t type;
uint32_t offset;
uint16_t section;
char name[];
} ATTRIBUTE_PACKED;
*/
/* Extract the DW_AT_location attribute from the DIE, and make sure it's in
in a format we can parse. */
loc = get_AT (s->data_symbol.die, DW_AT_location);
if (!loc)
goto end;
if (loc->dw_attr_val.val_class != dw_val_class_loc)
goto end;
loc_ref = loc->dw_attr_val.v.val_loc;
if (!loc_ref || loc_ref->dw_loc_opc != DW_OP_addr)
goto end;
/* Output the S_GDATA32 / S_LDATA32 record. */
fputs (integer_asm_op (2, false), asm_out_file);
asm_fprintf (asm_out_file,
"%L" SYMBOL_END_LABEL "%u - %L" SYMBOL_START_LABEL "%u\n",
label_num, label_num);
targetm.asm_out.internal_label (asm_out_file, SYMBOL_START_LABEL, label_num);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, s->kind);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, s->data_symbol.type);
putc ('\n', asm_out_file);
asm_fprintf (asm_out_file, "\t.secrel32 ");
output_addr_const (asm_out_file, loc_ref->dw_loc_oprnd1.v.val_addr);
fputc ('\n', asm_out_file);
asm_fprintf (asm_out_file, "\t.secidx ");
output_addr_const (asm_out_file, loc_ref->dw_loc_oprnd1.v.val_addr);
fputc ('\n', asm_out_file);
ASM_OUTPUT_ASCII (asm_out_file, s->data_symbol.name,
strlen (s->data_symbol.name) + 1);
ASM_OUTPUT_ALIGN (asm_out_file, 2);
targetm.asm_out.internal_label (asm_out_file, SYMBOL_END_LABEL, label_num);
end:
free (s->data_symbol.name);
}
/* Write an S_GPROC32_ID symbol, representing a global function, or an
S_LPROC32_ID symbol, for a static function. */
static void
write_function (codeview_symbol *s)
{
unsigned int label_num = ++sym_label_num;
dw_attr_node *loc_low, *loc_high;
const char *label_low, *label_high;
rtx rtx_low, rtx_high;
/* This is struct procsym in binutils and PROCSYM32 in Microsoft's cvinfo.h:
struct procsym
{
uint16_t size;
uint16_t kind;
uint32_t parent;
uint32_t end;
uint32_t next;
uint32_t proc_len;
uint32_t debug_start;
uint32_t debug_end;
uint32_t type;
uint32_t offset;
uint16_t section;
uint8_t flags;
char name[];
} ATTRIBUTE_PACKED;
*/
loc_low = get_AT (s->function.die, DW_AT_low_pc);
if (!loc_low)
goto end;
if (loc_low->dw_attr_val.val_class != dw_val_class_lbl_id)
goto end;
label_low = loc_low->dw_attr_val.v.val_lbl_id;
if (!label_low)
goto end;
rtx_low = gen_rtx_SYMBOL_REF (Pmode, label_low);
loc_high = get_AT (s->function.die, DW_AT_high_pc);
if (!loc_high)
goto end;
if (loc_high->dw_attr_val.val_class != dw_val_class_high_pc)
goto end;
label_high = loc_high->dw_attr_val.v.val_lbl_id;
if (!label_high)
goto end;
rtx_high = gen_rtx_SYMBOL_REF (Pmode, label_high);
/* Output the S_GPROC32_ID / S_LPROC32_ID record. */
fputs (integer_asm_op (2, false), asm_out_file);
asm_fprintf (asm_out_file,
"%L" SYMBOL_END_LABEL "%u - %L" SYMBOL_START_LABEL "%u\n",
label_num, label_num);
targetm.asm_out.internal_label (asm_out_file, SYMBOL_START_LABEL, label_num);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, s->kind);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, s->function.parent);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, s->function.end);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, s->function.next);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
output_addr_const (asm_out_file, rtx_high);
fputs (" - ", asm_out_file);
output_addr_const (asm_out_file, rtx_low);
putc ('\n', asm_out_file);
/* FIXME - debug_start should be the end of the prologue, and debug_end
the beginning of the epilogue. Do the whole function for
now. */
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, 0);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
output_addr_const (asm_out_file, rtx_high);
fputs (" - ", asm_out_file);
output_addr_const (asm_out_file, rtx_low);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, s->function.type);
putc ('\n', asm_out_file);
asm_fprintf (asm_out_file, "\t.secrel32 ");
output_addr_const (asm_out_file, rtx_low);
fputc ('\n', asm_out_file);
asm_fprintf (asm_out_file, "\t.secidx ");
output_addr_const (asm_out_file, rtx_low);
fputc ('\n', asm_out_file);
fputs (integer_asm_op (1, false), asm_out_file);
fprint_whex (asm_out_file, s->function.flags);
putc ('\n', asm_out_file);
ASM_OUTPUT_ASCII (asm_out_file, s->function.name,
strlen (s->function.name) + 1);
ASM_OUTPUT_ALIGN (asm_out_file, 2);
targetm.asm_out.internal_label (asm_out_file, SYMBOL_END_LABEL, label_num);
/* Output the S_PROC_ID_END record. */
label_num = ++sym_label_num;
fputs (integer_asm_op (2, false), asm_out_file);
asm_fprintf (asm_out_file,
"%L" SYMBOL_END_LABEL "%u - %L" SYMBOL_START_LABEL "%u\n",
label_num, label_num);
targetm.asm_out.internal_label (asm_out_file, SYMBOL_START_LABEL, label_num);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, S_PROC_ID_END);
putc ('\n', asm_out_file);
targetm.asm_out.internal_label (asm_out_file, SYMBOL_END_LABEL, label_num);
end:
free (s->function.name);
}
/* Write the CodeView symbols into the .debug$S section. */
static void
write_codeview_symbols (void)
{
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, DEBUG_S_SYMBOLS);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
asm_fprintf (asm_out_file, "%LLcv_syms_end - %LLcv_syms_start\n");
asm_fprintf (asm_out_file, "%LLcv_syms_start:\n");
write_compile3_symbol ();
while (sym)
{
codeview_symbol *n = sym->next;
switch (sym->kind)
{
case S_LDATA32:
case S_GDATA32:
write_data_symbol (sym);
break;
case S_LPROC32_ID:
case S_GPROC32_ID:
write_function (sym);
break;
default:
break;
}
free (sym);
sym = n;
}
asm_fprintf (asm_out_file, "%LLcv_syms_end:\n");
}
/* Write an LF_POINTER type. */
static void
write_lf_pointer (codeview_custom_type *t)
{
/* This is lf_pointer in binutils and lfPointer in Microsoft's cvinfo.h:
struct lf_pointer
{
uint16_t size;
uint16_t kind;
uint32_t base_type;
uint32_t attributes;
} ATTRIBUTE_PACKED;
*/
fputs (integer_asm_op (2, false), asm_out_file);
asm_fprintf (asm_out_file, "%LLcv_type%x_end - %LLcv_type%x_start\n",
t->num, t->num);
asm_fprintf (asm_out_file, "%LLcv_type%x_start:\n", t->num);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, t->kind);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_pointer.base_type);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_pointer.attributes);
putc ('\n', asm_out_file);
asm_fprintf (asm_out_file, "%LLcv_type%x_end:\n", t->num);
}
/* All CodeView type definitions have to be aligned to a four-byte boundary,
so write some padding bytes if necessary. These have to be specific values:
LF_PAD3, LF_PAD2, LF_PAD1. */
static void
write_cv_padding (size_t padding)
{
if (padding == 4 || padding == 0)
return;
if (padding == 3)
{
fputs (integer_asm_op (1, false), asm_out_file);
fprint_whex (asm_out_file, LF_PAD3);
putc ('\n', asm_out_file);
}
if (padding >= 2)
{
fputs (integer_asm_op (1, false), asm_out_file);
fprint_whex (asm_out_file, LF_PAD2);
putc ('\n', asm_out_file);
}
fputs (integer_asm_op (1, false), asm_out_file);
fprint_whex (asm_out_file, LF_PAD1);
putc ('\n', asm_out_file);
}
/* Write an LF_MODIFIER type, representing a const and/or volatile modification
of another type. */
static void
write_lf_modifier (codeview_custom_type *t)
{
/* This is lf_modifier in binutils and lfModifier in Microsoft's cvinfo.h:
struct lf_modifier
{
uint16_t size;
uint16_t kind;
uint32_t base_type;
uint16_t modifier;
uint16_t padding;
} ATTRIBUTE_PACKED;
*/
fputs (integer_asm_op (2, false), asm_out_file);
asm_fprintf (asm_out_file, "%LLcv_type%x_end - %LLcv_type%x_start\n",
t->num, t->num);
asm_fprintf (asm_out_file, "%LLcv_type%x_start:\n", t->num);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, t->kind);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_modifier.base_type);
putc ('\n', asm_out_file);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_modifier.modifier);
putc ('\n', asm_out_file);
write_cv_padding (2);
asm_fprintf (asm_out_file, "%LLcv_type%x_end:\n", t->num);
}
/* Write a CodeView extensible integer. If the value is non-negative and
< 0x8000, the value gets written directly as an uint16_t. Otherwise, we
output two bytes for the integer type (LF_CHAR, LF_SHORT, ...), and the
actual value follows. Returns the total number of bytes written. */
static size_t
write_cv_integer (codeview_integer *i)
{
if (i->neg)
{
if (i->num <= 0x80)
{
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, LF_CHAR);
putc ('\n', asm_out_file);
fputs (integer_asm_op (1, false), asm_out_file);
fprint_whex (asm_out_file, -i->num);
putc ('\n', asm_out_file);
return 3;
}
else if (i->num <= 0x8000)
{
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, LF_SHORT);
putc ('\n', asm_out_file);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, -i->num);
putc ('\n', asm_out_file);
return 4;
}
else if (i->num <= 0x80000000)
{
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, LF_LONG);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, -i->num);
putc ('\n', asm_out_file);
return 6;
}
else
{
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, LF_QUADWORD);
putc ('\n', asm_out_file);
fputs (integer_asm_op (8, false), asm_out_file);
fprint_whex (asm_out_file, -i->num);
putc ('\n', asm_out_file);
return 10;
}
}
else
{
if (i->num <= 0x7fff)
{
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, i->num);
putc ('\n', asm_out_file);
return 2;
}
else if (i->num <= 0xffff)
{
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, LF_USHORT);
putc ('\n', asm_out_file);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, i->num);
putc ('\n', asm_out_file);
return 4;
}
else if (i->num <= 0xffffffff)
{
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, LF_ULONG);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, i->num);
putc ('\n', asm_out_file);
return 6;
}
else
{
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, LF_UQUADWORD);
putc ('\n', asm_out_file);
fputs (integer_asm_op (8, false), asm_out_file);
fprint_whex (asm_out_file, i->num);
putc ('\n', asm_out_file);
return 10;
}
}
}
/* Return the extra size needed for an extensible integer. */
static size_t
cv_integer_len (codeview_integer *i)
{
if (i->neg)
{
if (i->num <= 0x80)
return sizeof (int8_t);
else if (i->num <= 0x8000)
return sizeof (int16_t);
else if (i->num <= 0x80000000)
return sizeof (int32_t);
else
return sizeof (int64_t);
}
else
{
if (i->num <= 0x7fff)
return 0;
else if (i->num <= 0xffff)
return sizeof (uint16_t);
else if (i->num <= 0xffffffff)
return sizeof (uint32_t);
else
return sizeof (uint64_t);
}
}
/* Write an LF_FIELDLIST type, which is a container for various subtypes. This
has two uses: for the values in an enum, and for the member, operators etc.
for a struct, class, or union. */
static void
write_lf_fieldlist (codeview_custom_type *t)
{
fputs (integer_asm_op (2, false), asm_out_file);
asm_fprintf (asm_out_file, "%LLcv_type%x_end - %LLcv_type%x_start\n",
t->num, t->num);
asm_fprintf (asm_out_file, "%LLcv_type%x_start:\n", t->num);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, t->kind);
putc ('\n', asm_out_file);
while (t->lf_fieldlist.subtypes)
{
codeview_subtype *v = t->lf_fieldlist.subtypes;
codeview_subtype *next = v->next;
size_t name_len, leaf_len;
switch (v->kind)
{
case LF_ENUMERATE:
/* This is lf_enumerate in binutils and lfEnumerate in Microsoft's
cvinfo.h:
struct lf_enumerate
{
uint16_t kind;
uint16_t attributes;
uint16_t value;
(then actual value if value >= 0x8000)
char name[];
} ATTRIBUTE_PACKED;
*/
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, LF_ENUMERATE);
putc ('\n', asm_out_file);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, CV_ACCESS_PUBLIC);
putc ('\n', asm_out_file);
leaf_len = 4 + write_cv_integer (&v->lf_enumerate.value);
name_len = strlen (v->lf_enumerate.name) + 1;
ASM_OUTPUT_ASCII (asm_out_file, v->lf_enumerate.name, name_len);
leaf_len += name_len;
write_cv_padding (4 - (leaf_len % 4));
free (v->lf_enumerate.name);
break;
case LF_MEMBER:
/* This is lf_member in binutils and lfMember in Microsoft's
cvinfo.h:
struct lf_member
{
uint16_t kind;
uint16_t attributes;
uint32_t type;
uint16_t offset;
char name[];
} ATTRIBUTE_PACKED;
*/
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, LF_MEMBER);
putc ('\n', asm_out_file);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, v->lf_member.attributes);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, v->lf_member.type);
putc ('\n', asm_out_file);
leaf_len = 8 + write_cv_integer (&v->lf_member.offset);
if (v->lf_member.name)
{
name_len = strlen (v->lf_member.name) + 1;
ASM_OUTPUT_ASCII (asm_out_file, v->lf_member.name, name_len);
}
else
{
name_len = 1;
ASM_OUTPUT_ASCII (asm_out_file, "", name_len);
}
leaf_len += name_len;
write_cv_padding (4 - (leaf_len % 4));
free (v->lf_member.name);
break;
case LF_INDEX:
/* This is lf_index in binutils and lfIndex in Microsoft's cvinfo.h:
struct lf_index
{
uint16_t kind;
uint16_t padding;
uint32_t index;
} ATTRIBUTE_PACKED;
*/
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, LF_INDEX);
putc ('\n', asm_out_file);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, 0);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, v->lf_index.type_num);
putc ('\n', asm_out_file);
break;
default:
break;
}
t->lf_fieldlist.subtypes = next;
free (v);
}
asm_fprintf (asm_out_file, "%LLcv_type%x_end:\n", t->num);
}
/* Write an LF_ENUM type. */
static void
write_lf_enum (codeview_custom_type *t)
{
size_t name_len, leaf_len;
/* This is lf_enum in binutils and lfEnum in Microsoft's cvinfo.h:
struct lf_enum
{
uint16_t size;
uint16_t kind;
uint16_t num_elements;
uint16_t properties;
uint32_t underlying_type;
uint32_t field_list;
char name[];
} ATTRIBUTE_PACKED;
*/
fputs (integer_asm_op (2, false), asm_out_file);
asm_fprintf (asm_out_file, "%LLcv_type%x_end - %LLcv_type%x_start\n",
t->num, t->num);
asm_fprintf (asm_out_file, "%LLcv_type%x_start:\n", t->num);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, t->kind);
putc ('\n', asm_out_file);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_enum.count);
putc ('\n', asm_out_file);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_enum.properties);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_enum.underlying_type);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_enum.fieldlist);
putc ('\n', asm_out_file);
name_len = strlen (t->lf_enum.name) + 1;
ASM_OUTPUT_ASCII (asm_out_file, t->lf_enum.name, name_len);
leaf_len = 14 + name_len;
write_cv_padding (4 - (leaf_len % 4));
free (t->lf_enum.name);
asm_fprintf (asm_out_file, "%LLcv_type%x_end:\n", t->num);
}
/* Write an LF_STRUCTURE or LF_CLASS type (the two have the same structure). */
static void
write_lf_structure (codeview_custom_type *t)
{
size_t name_len, leaf_len;
/* This is lf_class in binutils and lfClass in Microsoft's cvinfo.h:
struct lf_class
{
uint16_t size;
uint16_t kind;
uint16_t num_members;
uint16_t properties;
uint32_t field_list;
uint32_t derived_from;
uint32_t vshape;
uint16_t length;
char name[];
} ATTRIBUTE_PACKED;
*/
fputs (integer_asm_op (2, false), asm_out_file);
asm_fprintf (asm_out_file, "%LLcv_type%x_end - %LLcv_type%x_start\n",
t->num, t->num);
asm_fprintf (asm_out_file, "%LLcv_type%x_start:\n", t->num);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, t->kind);
putc ('\n', asm_out_file);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_structure.num_members);
putc ('\n', asm_out_file);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_structure.properties);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_structure.field_list);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_structure.derived_from);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_structure.vshape);
putc ('\n', asm_out_file);
leaf_len = 20 + write_cv_integer (&t->lf_structure.length);
if (t->lf_structure.name)
{
name_len = strlen (t->lf_structure.name) + 1;
ASM_OUTPUT_ASCII (asm_out_file, t->lf_structure.name, name_len);
}
else
{
static const char unnamed_struct[] = "";
name_len = sizeof (unnamed_struct);
ASM_OUTPUT_ASCII (asm_out_file, unnamed_struct, name_len);
}
leaf_len += name_len;
write_cv_padding (4 - (leaf_len % 4));
free (t->lf_structure.name);
asm_fprintf (asm_out_file, "%LLcv_type%x_end:\n", t->num);
}
/* Write an LF_UNION type. */
static void
write_lf_union (codeview_custom_type *t)
{
size_t name_len, leaf_len;
/* This is lf_union in binutils and lfUnion in Microsoft's cvinfo.h:
struct lf_union
{
uint16_t size;
uint16_t kind;
uint16_t num_members;
uint16_t properties;
uint32_t field_list;
uint16_t length;
char name[];
} ATTRIBUTE_PACKED;
*/
fputs (integer_asm_op (2, false), asm_out_file);
asm_fprintf (asm_out_file, "%LLcv_type%x_end - %LLcv_type%x_start\n",
t->num, t->num);
asm_fprintf (asm_out_file, "%LLcv_type%x_start:\n", t->num);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, t->kind);
putc ('\n', asm_out_file);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_structure.num_members);
putc ('\n', asm_out_file);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_structure.properties);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_structure.field_list);
putc ('\n', asm_out_file);
leaf_len = 12 + write_cv_integer (&t->lf_structure.length);
if (t->lf_structure.name)
{
name_len = strlen (t->lf_structure.name) + 1;
ASM_OUTPUT_ASCII (asm_out_file, t->lf_structure.name, name_len);
}
else
{
static const char unnamed_struct[] = "";
name_len = sizeof (unnamed_struct);
ASM_OUTPUT_ASCII (asm_out_file, unnamed_struct, name_len);
}
leaf_len += name_len;
write_cv_padding (4 - (leaf_len % 4));
free (t->lf_structure.name);
asm_fprintf (asm_out_file, "%LLcv_type%x_end:\n", t->num);
}
/* Write an LF_ARRAY type. */
static void
write_lf_array (codeview_custom_type *t)
{
size_t leaf_len;
/* This is lf_array in binutils and lfArray in Microsoft's cvinfo.h:
struct lf_array
{
uint16_t size;
uint16_t kind;
uint32_t element_type;
uint32_t index_type;
uint16_t length_in_bytes;
char name[];
} ATTRIBUTE_PACKED;
*/
fputs (integer_asm_op (2, false), asm_out_file);
asm_fprintf (asm_out_file, "%LLcv_type%x_end - %LLcv_type%x_start\n",
t->num, t->num);
asm_fprintf (asm_out_file, "%LLcv_type%x_start:\n", t->num);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, t->kind);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_array.element_type);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_array.index_type);
putc ('\n', asm_out_file);
leaf_len = 13 + write_cv_integer (&t->lf_array.length_in_bytes);
ASM_OUTPUT_ASCII (asm_out_file, "", 1);
write_cv_padding (4 - (leaf_len % 4));
asm_fprintf (asm_out_file, "%LLcv_type%x_end:\n", t->num);
}
/* Write an LF_BITFIELD type. */
static void
write_lf_bitfield (codeview_custom_type *t)
{
/* This is lf_bitfield in binutils and lfBitfield in Microsoft's cvinfo.h:
struct lf_bitfield
{
uint16_t size;
uint16_t kind;
uint32_t base_type;
uint8_t length;
uint8_t position;
} ATTRIBUTE_PACKED;
*/
fputs (integer_asm_op (2, false), asm_out_file);
asm_fprintf (asm_out_file, "%LLcv_type%x_end - %LLcv_type%x_start\n",
t->num, t->num);
asm_fprintf (asm_out_file, "%LLcv_type%x_start:\n", t->num);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, t->kind);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_bitfield.base_type);
putc ('\n', asm_out_file);
fputs (integer_asm_op (1, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_bitfield.length);
putc ('\n', asm_out_file);
fputs (integer_asm_op (1, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_bitfield.position);
putc ('\n', asm_out_file);
write_cv_padding (2);
asm_fprintf (asm_out_file, "%LLcv_type%x_end:\n", t->num);
}
/* Write an LF_PROCEDURE type. Function pointers are implemented as pointers
to one of these. */
static void
write_lf_procedure (codeview_custom_type *t)
{
/* This is lf_procedure in binutils and lfProc in Microsoft's cvinfo.h:
struct lf_procedure
{
uint16_t size;
uint16_t kind;
uint32_t return_type;
uint8_t calling_convention;
uint8_t attributes;
uint16_t num_parameters;
uint32_t arglist;
} ATTRIBUTE_PACKED;
*/
fputs (integer_asm_op (2, false), asm_out_file);
asm_fprintf (asm_out_file, "%LLcv_type%x_end - %LLcv_type%x_start\n",
t->num, t->num);
asm_fprintf (asm_out_file, "%LLcv_type%x_start:\n", t->num);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, t->kind);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_procedure.return_type);
putc ('\n', asm_out_file);
fputs (integer_asm_op (1, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_procedure.calling_convention);
putc ('\n', asm_out_file);
fputs (integer_asm_op (1, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_procedure.attributes);
putc ('\n', asm_out_file);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_procedure.num_parameters);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_procedure.arglist);
putc ('\n', asm_out_file);
asm_fprintf (asm_out_file, "%LLcv_type%x_end:\n", t->num);
}
/* Write an LF_ARGLIST type. This is just a list of other types. LF_PROCEDURE
entries point to one of these. */
static void
write_lf_arglist (codeview_custom_type *t)
{
/* This is lf_arglist in binutils and lfArgList in Microsoft's cvinfo.h:
struct lf_arglist
{
uint16_t size;
uint16_t kind;
uint32_t num_entries;
uint32_t args[];
} ATTRIBUTE_PACKED;
*/
fputs (integer_asm_op (2, false), asm_out_file);
asm_fprintf (asm_out_file, "%LLcv_type%x_end - %LLcv_type%x_start\n",
t->num, t->num);
asm_fprintf (asm_out_file, "%LLcv_type%x_start:\n", t->num);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, t->kind);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_arglist.num_entries);
putc ('\n', asm_out_file);
for (uint32_t i = 0; i < t->lf_arglist.num_entries; i++)
{
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_arglist.args[i]);
putc ('\n', asm_out_file);
}
free (t->lf_arglist.args);
asm_fprintf (asm_out_file, "%LLcv_type%x_end:\n", t->num);
}
/* Write an LF_FUNC_ID type, which marries together a function type with its
name. This will end up in the alternative types stream in the final PDB,
but we can just stick it in the normal .debug$T section. */
static void
write_lf_func_id (codeview_custom_type *t)
{
size_t name_len;
/* This is lf_func_id in binutils and lfFuncId in Microsoft's cvinfo.h:
struct lf_func_id
{
uint16_t size;
uint16_t kind;
uint32_t parent_scope;
uint32_t function_type;
char name[];
} ATTRIBUTE_PACKED;
*/
fputs (integer_asm_op (2, false), asm_out_file);
asm_fprintf (asm_out_file, "%LLcv_type%x_end - %LLcv_type%x_start\n",
t->num, t->num);
asm_fprintf (asm_out_file, "%LLcv_type%x_start:\n", t->num);
fputs (integer_asm_op (2, false), asm_out_file);
fprint_whex (asm_out_file, t->kind);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_func_id.parent_scope);
putc ('\n', asm_out_file);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, t->lf_func_id.function_type);
putc ('\n', asm_out_file);
name_len = strlen (t->lf_func_id.name) + 1;
ASM_OUTPUT_ASCII (asm_out_file, t->lf_func_id.name, name_len);
write_cv_padding (4 - (name_len % 4));
free (t->lf_func_id.name);
asm_fprintf (asm_out_file, "%LLcv_type%x_end:\n", t->num);
}
/* Write the .debug$T section, which contains all of our custom type
definitions. */
static void
write_custom_types (void)
{
targetm.asm_out.named_section (".debug$T", SECTION_DEBUG, NULL);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, CV_SIGNATURE_C13);
putc ('\n', asm_out_file);
while (custom_types)
{
codeview_custom_type *n = custom_types->next;
switch (custom_types->kind)
{
case LF_POINTER:
write_lf_pointer (custom_types);
break;
case LF_MODIFIER:
write_lf_modifier (custom_types);
break;
case LF_FIELDLIST:
write_lf_fieldlist (custom_types);
break;
case LF_ENUM:
write_lf_enum (custom_types);
break;
case LF_STRUCTURE:
case LF_CLASS:
write_lf_structure (custom_types);
break;
case LF_UNION:
write_lf_union (custom_types);
break;
case LF_ARRAY:
write_lf_array (custom_types);
break;
case LF_BITFIELD:
write_lf_bitfield (custom_types);
break;
case LF_PROCEDURE:
write_lf_procedure (custom_types);
break;
case LF_ARGLIST:
write_lf_arglist (custom_types);
break;
case LF_FUNC_ID:
write_lf_func_id (custom_types);
break;
default:
break;
}
free (custom_types);
custom_types = n;
}
}
/* Finish CodeView debug info emission. */
void
codeview_debug_finish (void)
{
targetm.asm_out.named_section (".debug$S", SECTION_DEBUG, NULL);
fputs (integer_asm_op (4, false), asm_out_file);
fprint_whex (asm_out_file, CV_SIGNATURE_C13);
putc ('\n', asm_out_file);
write_strings_table ();
write_source_files ();
write_line_numbers ();
write_codeview_symbols ();
if (custom_types)
write_custom_types ();
if (types_htab)
delete types_htab;
}
/* Translate a DWARF base type (DW_TAG_base_type) into its CodeView
equivalent. */
static uint32_t
get_type_num_base_type (dw_die_ref type)
{
unsigned int size = get_AT_unsigned (type, DW_AT_byte_size);
switch (get_AT_unsigned (type, DW_AT_encoding))
{
case DW_ATE_signed_char:
{
const char *name = get_AT_string (type, DW_AT_name);
if (size != 1)
return 0;
if (name && !strcmp (name, "signed char"))
return T_CHAR;
else
return T_RCHAR;
}
case DW_ATE_unsigned_char:
if (size != 1)
return 0;
return T_UCHAR;
case DW_ATE_signed:
switch (size)
{
case 2:
return T_SHORT;
case 4:
{
const char *name = get_AT_string (type, DW_AT_name);
if (name && !strcmp (name, "int"))
return T_INT4;
else
return T_LONG;
}
case 8:
return T_QUAD;
default:
return 0;
}
case DW_ATE_unsigned:
switch (size)
{
case 2:
{
const char *name = get_AT_string (type, DW_AT_name);
if (name && !strcmp (name, "wchar_t"))
return T_WCHAR;
else
return T_USHORT;
}
case 4:
{
const char *name = get_AT_string (type, DW_AT_name);
if (name && !strcmp (name, "unsigned int"))
return T_UINT4;
else
return T_ULONG;
}
case 8:
return T_UQUAD;
default:
return 0;
}
case DW_ATE_UTF:
switch (size)
{
case 1:
return T_CHAR8;
case 2:
return T_CHAR16;
case 4:
return T_CHAR32;
default:
return 0;
}
case DW_ATE_float:
switch (size)
{
case 4:
return T_REAL32;
case 8:
return T_REAL64;
case 12:
return T_REAL80;
case 16:
return T_REAL128;
default:
return 0;
}
case DW_ATE_boolean:
if (size == 1)
return T_BOOL08;
else
return 0;
default:
return 0;
}
}
/* Add a new codeview_custom_type to our singly-linked custom_types list. */
static void
add_custom_type (codeview_custom_type *ct)
{
uint32_t num;
if (last_custom_type)
{
num = last_custom_type->num + 1;
last_custom_type->next = ct;
}
else
{
num = FIRST_TYPE;
custom_types = ct;
}
last_custom_type = ct;
ct->num = num;
}
/* Process a DW_TAG_pointer_type DIE. If this is a pointer to a builtin
type, return the predefined constant for this. Otherwise, add a new
LF_POINTER type and return its number. */
static uint32_t
get_type_num_pointer_type (dw_die_ref type, bool in_struct)
{
uint32_t base_type_num, byte_size;
dw_die_ref base_type;
codeview_custom_type *ct;
byte_size = get_AT_unsigned (type, DW_AT_byte_size);
if (byte_size != 4 && byte_size != 8)
return 0;
base_type = get_AT_ref (type, DW_AT_type);
/* If DW_AT_type is not set, this must be a void pointer. */
if (!base_type)
return byte_size == 4 ? T_32PVOID : T_64PVOID;
base_type_num = get_type_num (base_type, in_struct, false);
if (base_type_num == 0)
return 0;
/* Pointers to builtin types have predefined type numbers, with the top byte
determining the pointer size - 0x0400 for a 32-bit pointer and 0x0600
for 64-bit. */
if (base_type_num < FIRST_TYPE && !(base_type_num & 0xff00))
{
if (byte_size == 4)
return CV_POINTER_32 | base_type_num;
else
return CV_POINTER_64 | base_type_num;
}
ct = (codeview_custom_type *) xmalloc (sizeof (codeview_custom_type));
ct->next = NULL;
ct->kind = LF_POINTER;
ct->lf_pointer.base_type = base_type_num;
if (byte_size == 4)
ct->lf_pointer.attributes = CV_PTR_NEAR32;
else
ct->lf_pointer.attributes = CV_PTR_64;
ct->lf_pointer.attributes |= byte_size << 13;
add_custom_type (ct);
return ct->num;
}
/* Process a DW_TAG_reference_type or DW_TAG_rvalue_reference_type DIE, add a
new LF_POINTER type, and return its number. */
static uint32_t
get_type_num_reference_type (dw_die_ref type, bool in_struct, bool rvref)
{
uint32_t base_type_num, byte_size;
dw_die_ref base_type;
codeview_custom_type *ct;
byte_size = get_AT_unsigned (type, DW_AT_byte_size);
if (byte_size != 4 && byte_size != 8)
return 0;
base_type = get_AT_ref (type, DW_AT_type);
base_type_num = get_type_num (base_type, in_struct, false);
if (base_type_num == 0)
return 0;
ct = (codeview_custom_type *) xmalloc (sizeof (codeview_custom_type));
ct->next = NULL;
ct->kind = LF_POINTER;
ct->lf_pointer.base_type = base_type_num;
ct->lf_pointer.attributes = rvref ? CV_PTR_MODE_RVREF : CV_PTR_MODE_LVREF;
if (byte_size == 4)
ct->lf_pointer.attributes |= CV_PTR_NEAR32;
else
ct->lf_pointer.attributes |= CV_PTR_64;
ct->lf_pointer.attributes |= byte_size << 13;
add_custom_type (ct);
return ct->num;
}
/* Process a DW_TAG_const_type DIE, adding an LF_MODIFIER type and returning
its number. */
static uint32_t
get_type_num_const_type (dw_die_ref type, bool in_struct)
{
dw_die_ref base_type;
uint32_t base_type_num;
codeview_custom_type *ct;
bool is_volatile = false;
base_type = get_AT_ref (type, DW_AT_type);
/* Handle case when this is a const volatile type - we only need one
LF_MODIFIER for this. */
if (base_type && dw_get_die_tag (base_type) == DW_TAG_volatile_type)
{
is_volatile = true;
base_type = get_AT_ref (base_type, DW_AT_type);
}
if (!base_type)
{
base_type_num = T_VOID;
}
else
{
base_type_num = get_type_num (base_type, in_struct, false);
if (base_type_num == 0)
return 0;
}
ct = (codeview_custom_type *) xmalloc (sizeof (codeview_custom_type));
ct->next = NULL;
ct->kind = LF_MODIFIER;
ct->lf_modifier.base_type = base_type_num;
ct->lf_modifier.modifier = MOD_const;
if (is_volatile)
ct->lf_modifier.modifier |= MOD_volatile;
add_custom_type (ct);
return ct->num;
}
/* Process a DW_TAG_volatile_type DIE, adding an LF_MODIFIER type and
returning its number. */
static uint32_t
get_type_num_volatile_type (dw_die_ref type, bool in_struct)
{
dw_die_ref base_type;
uint32_t base_type_num;
codeview_custom_type *ct;
base_type = get_AT_ref (type, DW_AT_type);
if (base_type)
{
base_type_num = get_type_num (base_type, in_struct, false);
if (base_type_num == 0)
return 0;
}
else
{
base_type_num = T_VOID;
}
ct = (codeview_custom_type *) xmalloc (sizeof (codeview_custom_type));
ct->next = NULL;
ct->kind = LF_MODIFIER;
ct->lf_modifier.base_type = base_type_num;
ct->lf_modifier.modifier = MOD_volatile;
add_custom_type (ct);
return ct->num;
}
/* Add a forward declaration for an enum. This is legal from C++11 onwards. */
static uint32_t
add_enum_forward_def (dw_die_ref type)
{
codeview_custom_type *ct;
ct = (codeview_custom_type *) xmalloc (sizeof (codeview_custom_type));
ct->next = NULL;
ct->kind = LF_ENUM;
ct->lf_enum.count = 0;
ct->lf_enum.properties = CV_PROP_FWDREF;
ct->lf_enum.underlying_type = get_type_num (get_AT_ref (type, DW_AT_type),
false, false);
ct->lf_enum.fieldlist = 0;
ct->lf_enum.name = xstrdup (get_AT_string (type, DW_AT_name));
add_custom_type (ct);
return ct->num;
}
/* Process a DW_TAG_enumeration_type DIE, adding an LF_FIELDLIST and an LF_ENUM
type, returning the number of the latter. */
static uint32_t
get_type_num_enumeration_type (dw_die_ref type, bool in_struct)
{
dw_die_ref first_child;
codeview_custom_type *ct;
uint16_t count = 0;
uint32_t last_type;
if (get_AT_flag (type, DW_AT_declaration))
return add_enum_forward_def (type);
/* First, add an LF_FIELDLIST for the enum's values. We don't need to worry
about deduplication here, as ld will take care of that for us. If there's
a lot of entries, add more LF_FIELDLISTs with LF_INDEXes pointing to
the overflow lists. */
first_child = dw_get_die_child (type);
ct = (codeview_custom_type *) xmalloc (sizeof (codeview_custom_type));
ct->next = NULL;
ct->kind = LF_FIELDLIST;
ct->lf_fieldlist.length = 0;
ct->lf_fieldlist.subtypes = NULL;
ct->lf_fieldlist.last_subtype = NULL;
if (first_child)
{
dw_die_ref c;
c = first_child;
do
{
dw_attr_node *att;
codeview_subtype *el;
size_t el_len;
c = dw_get_die_sib (c);
if (dw_get_die_tag (c) != DW_TAG_enumerator)
continue;
att = get_AT (c, DW_AT_const_value);
if (!att)
continue;
el = (codeview_subtype *) xmalloc (sizeof (*el));
el->next = NULL;
el->kind = LF_ENUMERATE;
switch (AT_class (att))
{
case dw_val_class_unsigned_const:
case dw_val_class_unsigned_const_implicit:
el->lf_enumerate.value.neg = false;
el->lf_enumerate.value.num = att->dw_attr_val.v.val_unsigned;
break;
case dw_val_class_const:
case dw_val_class_const_implicit:
if (att->dw_attr_val.v.val_int < 0)
{
el->lf_enumerate.value.neg = true;
el->lf_enumerate.value.num = -att->dw_attr_val.v.val_int;
}
else
{
el->lf_enumerate.value.neg = false;
el->lf_enumerate.value.num = att->dw_attr_val.v.val_int;
}
break;
default:
free (el);
continue;
}
el->lf_enumerate.name = xstrdup (get_AT_string (c, DW_AT_name));
el_len = 7 + strlen (el->lf_enumerate.name);
el_len += cv_integer_len (&el->lf_enumerate.value);
if (el_len % 4)
el_len += 4 - (el_len % 4);
if (ct->lf_fieldlist.length + el_len > MAX_FIELDLIST_SIZE)
{
codeview_subtype *idx;
codeview_custom_type *ct2;
idx = (codeview_subtype *) xmalloc (sizeof (*idx));
idx->next = NULL;
idx->kind = LF_INDEX;
idx->lf_index.type_num = 0;
ct->lf_fieldlist.last_subtype->next = idx;
ct->lf_fieldlist.last_subtype = idx;
ct2 = (codeview_custom_type *)
xmalloc (sizeof (codeview_custom_type));
ct2->next = ct;
ct2->kind = LF_FIELDLIST;
ct2->lf_fieldlist.length = 0;
ct2->lf_fieldlist.subtypes = NULL;
ct2->lf_fieldlist.last_subtype = NULL;
ct = ct2;
}
ct->lf_fieldlist.length += el_len;
if (ct->lf_fieldlist.last_subtype)
ct->lf_fieldlist.last_subtype->next = el;
else
ct->lf_fieldlist.subtypes = el;
ct->lf_fieldlist.last_subtype = el;
count++;
}
while (c != first_child);
}
while (ct)
{
codeview_custom_type *ct2;
ct2 = ct->next;
ct->next = NULL;
if (ct->lf_fieldlist.last_subtype->kind == LF_INDEX)
ct->lf_fieldlist.last_subtype->lf_index.type_num = last_type;
add_custom_type (ct);
last_type = ct->num;
ct = ct2;
}
/* Now add an LF_ENUM, pointing to the LF_FIELDLIST we just added. */
ct = (codeview_custom_type *) xmalloc (sizeof (codeview_custom_type));
ct->next = NULL;
ct->kind = LF_ENUM;
ct->lf_enum.count = count;
ct->lf_enum.properties = 0;
ct->lf_enum.underlying_type = get_type_num (get_AT_ref (type, DW_AT_type),
in_struct, false);
ct->lf_enum.fieldlist = last_type;
ct->lf_enum.name = xstrdup (get_AT_string (type, DW_AT_name));
add_custom_type (ct);
return ct->num;
}
/* Add a DIE to our deferred_types list. This happens when we have a struct
with a pointer to a type that hasn't been defined yet, but which gets
defined later on. */
static void
add_deferred_type (dw_die_ref type)
{
codeview_deferred_type *def;
def = (codeview_deferred_type *) xmalloc (sizeof (codeview_deferred_type));
def->next = NULL;
def->type = type;
if (!deferred_types)
deferred_types = def;
else
last_deferred_type->next = def;
last_deferred_type = def;
}
/* Flush the contents of our deferred_types list. This happens after everything
else has been written. We call get_type_num to ensure that a type gets
added to custom_types, if it hasn't been already. */
static void
flush_deferred_types (void)
{
while (deferred_types)
{
codeview_deferred_type *next;
next = deferred_types->next;
get_type_num (deferred_types->type, false, true);
free (deferred_types);
deferred_types = next;
}
last_deferred_type = NULL;
}
/* Add a forward definition for a struct, class, or union. */
static uint32_t
add_struct_forward_def (dw_die_ref type)
{
codeview_custom_type *ct;
ct = (codeview_custom_type *) xmalloc (sizeof (codeview_custom_type));
ct->next = NULL;
switch (dw_get_die_tag (type))
{
case DW_TAG_class_type:
ct->kind = LF_CLASS;
break;
case DW_TAG_structure_type:
ct->kind = LF_STRUCTURE;
break;
case DW_TAG_union_type:
ct->kind = LF_UNION;
break;
default:
break;
}
ct->lf_structure.num_members = 0;
ct->lf_structure.properties = CV_PROP_FWDREF;
ct->lf_structure.field_list = 0;
ct->lf_structure.derived_from = 0;
ct->lf_structure.vshape = 0;
ct->lf_structure.length.neg = false;
ct->lf_structure.length.num = 0;
ct->lf_structure.name = xstrdup (get_AT_string (type, DW_AT_name));
add_custom_type (ct);
if (!get_AT_flag (type, DW_AT_declaration))
add_deferred_type (type);
return ct->num;
}
/* Add an LF_BITFIELD type, returning its number. DWARF represents bitfields
as members in a struct with a DW_AT_data_bit_offset attribute, whereas in
CodeView they're a distinct type. */
static uint32_t
create_bitfield (dw_die_ref c)
{
codeview_custom_type *ct;
uint32_t base_type;
base_type = get_type_num (get_AT_ref (c, DW_AT_type), true, false);
if (base_type == 0)
return 0;
ct = (codeview_custom_type *) xmalloc (sizeof (codeview_custom_type));
ct->next = NULL;
ct->kind = LF_BITFIELD;
ct->lf_bitfield.base_type = base_type;
ct->lf_bitfield.length = get_AT_unsigned (c, DW_AT_bit_size);
ct->lf_bitfield.position = get_AT_unsigned (c, DW_AT_data_bit_offset);
add_custom_type (ct);
return ct->num;
}
/* Process a DW_TAG_structure_type, DW_TAG_class_type, or DW_TAG_union_type
DIE, add an LF_FIELDLIST and an LF_STRUCTURE / LF_CLASS / LF_UNION type,
and return the number of the latter. */
static uint32_t
get_type_num_struct (dw_die_ref type, bool in_struct, bool *is_fwd_ref)
{
dw_die_ref first_child;
codeview_custom_type *ct;
uint16_t num_members = 0;
uint32_t last_type;
const char *name;
if ((in_struct && get_AT_string (type, DW_AT_name))
|| get_AT_flag (type, DW_AT_declaration))
{
*is_fwd_ref = true;
return add_struct_forward_def (type);
}
*is_fwd_ref = false;
/* First, add an LF_FIELDLIST for the structure's members. We don't need to
worry about deduplication here, as ld will take care of that for us.
If there's a lot of entries, add more LF_FIELDLISTs with LF_INDEXes
pointing to the overflow lists. */
first_child = dw_get_die_child (type);
ct = (codeview_custom_type *) xmalloc (sizeof (codeview_custom_type));
ct->next = NULL;
ct->kind = LF_FIELDLIST;
ct->lf_fieldlist.length = 0;
ct->lf_fieldlist.subtypes = NULL;
ct->lf_fieldlist.last_subtype = NULL;
if (first_child)
{
dw_die_ref c;
c = first_child;
do
{
codeview_subtype *el;
size_t el_len;
c = dw_get_die_sib (c);
if (dw_get_die_tag (c) != DW_TAG_member)
continue;
el = (codeview_subtype *) xmalloc (sizeof (*el));
el->next = NULL;
el->kind = LF_MEMBER;
switch (get_AT_unsigned (c, DW_AT_accessibility))
{
case DW_ACCESS_private:
el->lf_member.attributes = CV_ACCESS_PRIVATE;
break;
case DW_ACCESS_protected:
el->lf_member.attributes = CV_ACCESS_PROTECTED;
break;
case DW_ACCESS_public:
el->lf_member.attributes = CV_ACCESS_PUBLIC;
break;
/* Members in a C++ struct or union are public by default, members
in a class are private. */
default:
if (dw_get_die_tag (type) == DW_TAG_class_type)
el->lf_member.attributes = CV_ACCESS_PRIVATE;
else
el->lf_member.attributes = CV_ACCESS_PUBLIC;
break;
}
if (get_AT (c, DW_AT_data_bit_offset))
el->lf_member.type = create_bitfield (c);
else
el->lf_member.type = get_type_num (get_AT_ref (c, DW_AT_type),
true, false);
el->lf_member.offset.neg = false;
el->lf_member.offset.num = get_AT_unsigned (c,
DW_AT_data_member_location);
el_len = 11;
el_len += cv_integer_len (&el->lf_member.offset);
if (get_AT_string (c, DW_AT_name))
{
el->lf_member.name = xstrdup (get_AT_string (c, DW_AT_name));
el_len += strlen (el->lf_member.name);
}
else
{
el->lf_member.name = NULL;
}
if (el_len % 4)
el_len += 4 - (el_len % 4);
/* Add an LF_INDEX subtype if everything's too big for one
LF_FIELDLIST. */
if (ct->lf_fieldlist.length + el_len > MAX_FIELDLIST_SIZE)
{
codeview_subtype *idx;
codeview_custom_type *ct2;
idx = (codeview_subtype *) xmalloc (sizeof (*idx));
idx->next = NULL;
idx->kind = LF_INDEX;
idx->lf_index.type_num = 0;
ct->lf_fieldlist.last_subtype->next = idx;
ct->lf_fieldlist.last_subtype = idx;
ct2 = (codeview_custom_type *)
xmalloc (sizeof (codeview_custom_type));
ct2->next = ct;
ct2->kind = LF_FIELDLIST;
ct2->lf_fieldlist.length = 0;
ct2->lf_fieldlist.subtypes = NULL;
ct2->lf_fieldlist.last_subtype = NULL;
ct = ct2;
}
ct->lf_fieldlist.length += el_len;
if (ct->lf_fieldlist.last_subtype)
ct->lf_fieldlist.last_subtype->next = el;
else
ct->lf_fieldlist.subtypes = el;
ct->lf_fieldlist.last_subtype = el;
num_members++;
}
while (c != first_child);
}
while (ct)
{
codeview_custom_type *ct2;
ct2 = ct->next;
ct->next = NULL;
if (ct->lf_fieldlist.last_subtype
&& ct->lf_fieldlist.last_subtype->kind == LF_INDEX)
{
ct->lf_fieldlist.last_subtype->lf_index.type_num = last_type;
}
add_custom_type (ct);
last_type = ct->num;
ct = ct2;
}
/* Now add an LF_STRUCTURE / LF_CLASS / LF_UNION, pointing to the
LF_FIELDLIST we just added. */
ct = (codeview_custom_type *) xmalloc (sizeof (codeview_custom_type));
ct->next = NULL;
switch (dw_get_die_tag (type))
{
case DW_TAG_class_type:
ct->kind = LF_CLASS;
break;
case DW_TAG_structure_type:
ct->kind = LF_STRUCTURE;
break;
case DW_TAG_union_type:
ct->kind = LF_UNION;
break;
default:
break;
}
ct->lf_structure.num_members = num_members;
ct->lf_structure.properties = 0;
ct->lf_structure.field_list = last_type;
ct->lf_structure.derived_from = 0;
ct->lf_structure.vshape = 0;
ct->lf_structure.length.neg = false;
ct->lf_structure.length.num = get_AT_unsigned (type, DW_AT_byte_size);
name = get_AT_string (type, DW_AT_name);
if (name)
ct->lf_structure.name = xstrdup (name);
else
ct->lf_structure.name = NULL;
add_custom_type (ct);
return ct->num;
}
/* Process a DW_TAG_subroutine_type DIE, adding an LF_ARGLIST and an
LF_PROCEDURE type, and returning the number of the latter. */
static uint32_t
get_type_num_subroutine_type (dw_die_ref type, bool in_struct)
{
codeview_custom_type *ct;
uint32_t return_type, arglist_type;
uint16_t num_args;
dw_die_ref first_child;
/* Find the return type. */
if (get_AT_ref (type, DW_AT_type))
{
return_type = get_type_num (get_AT_ref (type, DW_AT_type), in_struct,
false);
if (return_type == 0)
return 0;
}
else
{
return_type = T_VOID;
}
/* Count the arguments. */
first_child = dw_get_die_child (type);
num_args = 0;
if (first_child)
{
dw_die_ref c;
c = first_child;
do
{
c = dw_get_die_sib (c);
if (dw_get_die_tag (c) != DW_TAG_formal_parameter
&& dw_get_die_tag (c) != DW_TAG_unspecified_parameters)
continue;
num_args++;
}
while (c != first_child);
}
/* Create an LF_ARGLIST for the arguments. If this is a duplicate, ld
will take care of this for us. */
first_child = dw_get_die_child (type);
ct = (codeview_custom_type *) xmalloc (sizeof (codeview_custom_type));
ct->next = NULL;
ct->kind = LF_ARGLIST;
ct->lf_arglist.num_entries = num_args;
if (num_args > 0)
{
dw_die_ref c;
uint32_t *argptr;
ct->lf_arglist.args = (uint32_t *) xmalloc (sizeof (uint32_t) * num_args);
argptr = ct->lf_arglist.args;
c = first_child;
do
{
c = dw_get_die_sib (c);
switch (dw_get_die_tag (c))
{
case DW_TAG_formal_parameter:
*argptr = get_type_num (get_AT_ref (c, DW_AT_type), in_struct,
false);
argptr++;
break;
case DW_TAG_unspecified_parameters:
*argptr = 0;
argptr++;
break;
default:
break;
}
}
while (c != first_child);
}
else
{
ct->lf_arglist.args = NULL;
}
add_custom_type (ct);
arglist_type = ct->num;
/* Finally, create an LF_PROCEDURE. */
ct = (codeview_custom_type *) xmalloc (sizeof (codeview_custom_type));
ct->next = NULL;
ct->kind = LF_PROCEDURE;
ct->lf_procedure.return_type = return_type;
ct->lf_procedure.calling_convention = 0;
ct->lf_procedure.attributes = 0;
ct->lf_procedure.num_parameters = num_args;
ct->lf_procedure.arglist = arglist_type;
add_custom_type (ct);
return ct->num;
}
/* Process a DW_TAG_array_type DIE, adding an LF_ARRAY type and returning its
number. */
static uint32_t
get_type_num_array_type (dw_die_ref type, bool in_struct)
{
dw_die_ref base_type, t, first_child, c, *dimension_arr;
uint64_t size = 0;
unsigned int dimensions, i;
uint32_t element_type;
base_type = get_AT_ref (type, DW_AT_type);
if (!base_type)
return 0;
/* We need to know the size of our base type. Loop through until we find
it. */
t = base_type;
while (t && size == 0)
{
switch (dw_get_die_tag (t))
{
case DW_TAG_const_type:
case DW_TAG_volatile_type:
case DW_TAG_typedef:
case DW_TAG_enumeration_type:
t = get_AT_ref (t, DW_AT_type);
break;
case DW_TAG_base_type:
case DW_TAG_structure_type:
case DW_TAG_class_type:
case DW_TAG_union_type:
case DW_TAG_pointer_type:
case DW_TAG_reference_type:
case DW_TAG_rvalue_reference_type:
size = get_AT_unsigned (t, DW_AT_byte_size);
break;
default:
return 0;
}
}
if (size == 0)
return 0;
first_child = dw_get_die_child (type);
if (!first_child)
return 0;
element_type = get_type_num (base_type, in_struct, false);
if (element_type == 0)
return 0;
/* Create an array of our DW_TAG_subrange_type children, in reverse order.
We have to do this because unlike DWARF CodeView doesn't have
multidimensional arrays, so instead we do arrays of arrays. */
dimensions = 0;
c = first_child;
do
{
c = dw_get_die_sib (c);
if (dw_get_die_tag (c) != DW_TAG_subrange_type)
continue;
dimensions++;
}
while (c != first_child);
if (dimensions == 0)
return 0;
dimension_arr = (dw_die_ref *) xmalloc (sizeof (dw_die_ref) * dimensions);
c = first_child;
i = 0;
do
{
c = dw_get_die_sib (c);
if (dw_get_die_tag (c) != DW_TAG_subrange_type)
continue;
dimension_arr[dimensions - i - 1] = c;
i++;
}
while (c != first_child);
/* Record an LF_ARRAY entry for each array dimension. If this leads to
duplicate types, ld will take care of it for us. */
for (i = 0; i < dimensions; i++)
{
codeview_custom_type *ct;
dw_die_ref index;
ct = (codeview_custom_type *) xmalloc (sizeof (codeview_custom_type));
size *= get_AT_unsigned (dimension_arr[i], DW_AT_upper_bound) + 1;
index = get_AT_ref (dimension_arr[i], DW_AT_type);
ct->next = NULL;
ct->kind = LF_ARRAY;
ct->lf_array.element_type = element_type;
ct->lf_array.index_type = get_type_num (index, in_struct, false);
ct->lf_array.length_in_bytes.neg = false;
ct->lf_array.length_in_bytes.num = size;
add_custom_type (ct);
element_type = ct->num;
}
free (dimension_arr);
return element_type;
}
/* Process a DIE representing a type definition, add a CodeView type if
necessary, and return its number. If it's something we can't handle, return
0. We keep a hash table so that we're not adding the same type multiple
times - though if we do it's not disastrous, as ld will deduplicate
everything for us. */
static uint32_t
get_type_num (dw_die_ref type, bool in_struct, bool no_fwd_ref)
{
codeview_type **slot, *t;
uint32_t num;
bool is_fwd_ref;
if (!type)
return 0;
if (!types_htab)
types_htab = new hash_table (10);
slot = types_htab->find_slot_with_hash (type, htab_hash_pointer (type),
NO_INSERT);
if (slot && *slot && (!no_fwd_ref || !(*slot)->is_fwd_ref))
return (*slot)->num;
is_fwd_ref = false;
switch (dw_get_die_tag (type))
{
case DW_TAG_base_type:
num = get_type_num_base_type (type);
break;
case DW_TAG_typedef:
/* FIXME - signed longs typedef'd as "HRESULT" should get their
own type (T_HRESULT) */
num = get_type_num (get_AT_ref (type, DW_AT_type), in_struct, false);
break;
case DW_TAG_pointer_type:
num = get_type_num_pointer_type (type, in_struct);
break;
case DW_TAG_reference_type:
num = get_type_num_reference_type (type, in_struct, false);
break;
case DW_TAG_rvalue_reference_type:
num = get_type_num_reference_type (type, in_struct, true);
break;
case DW_TAG_const_type:
num = get_type_num_const_type (type, in_struct);
break;
case DW_TAG_volatile_type:
num = get_type_num_volatile_type (type, in_struct);
break;
case DW_TAG_enumeration_type:
num = get_type_num_enumeration_type (type, in_struct);
break;
case DW_TAG_structure_type:
case DW_TAG_class_type:
case DW_TAG_union_type:
num = get_type_num_struct (type, in_struct, &is_fwd_ref);
break;
case DW_TAG_array_type:
num = get_type_num_array_type (type, in_struct);
break;
case DW_TAG_subroutine_type:
num = get_type_num_subroutine_type (type, in_struct);
break;
default:
num = 0;
break;
}
/* Check hash table again, and account for the fact that self-referential
structs will have created a forward reference to themselves. */
slot = types_htab->find_slot_with_hash (type, htab_hash_pointer (type),
INSERT);
if (*slot && (*slot)->is_fwd_ref && !is_fwd_ref)
{
(*slot)->num = num;
(*slot)->is_fwd_ref = false;
return num;
}
t = (codeview_type *) xmalloc (sizeof (codeview_type));
t->die = type;
t->num = num;
t->is_fwd_ref = is_fwd_ref;
*slot = t;
return t->num;
}
/* Process a DW_TAG_variable DIE, and add an S_GDATA32 or S_LDATA32 symbol for
this. */
static void
add_variable (dw_die_ref die)
{
codeview_symbol *s;
const char *name;
name = get_AT_string (die, DW_AT_name);
if (!name)
return;
s = (codeview_symbol *) xmalloc (sizeof (codeview_symbol));
s->next = NULL;
s->kind = get_AT (die, DW_AT_external) ? S_GDATA32 : S_LDATA32;
s->data_symbol.type = get_type_num (get_AT_ref (die, DW_AT_type), false,
false);
s->data_symbol.name = xstrdup (name);
s->data_symbol.die = die;
if (last_sym)
last_sym->next = s;
else
sym = s;
last_sym = s;
}
/* Process a DW_TAG_subprogram DIE, and add an S_GPROC32_ID or S_LPROC32_ID
symbol for this. */
static void
add_function (dw_die_ref die)
{
codeview_custom_type *ct;
const char *name = get_AT_string (die, DW_AT_name);
uint32_t function_type, func_id_type;
codeview_symbol *s;
if (!name)
return;
/* Add an LF_FUNC_ID type for this function. */
function_type = get_type_num_subroutine_type (die, false);
ct = (codeview_custom_type *) xmalloc (sizeof (codeview_custom_type));
ct->next = NULL;
ct->kind = LF_FUNC_ID;
ct->lf_func_id.parent_scope = 0;
ct->lf_func_id.function_type = function_type;
ct->lf_func_id.name = xstrdup (name);
add_custom_type (ct);
func_id_type = ct->num;
/* Add an S_GPROC32_ID / S_LPROC32_ID symbol. */
s = (codeview_symbol *) xmalloc (sizeof (codeview_symbol));
s->next = NULL;
s->kind = get_AT (die, DW_AT_external) ? S_GPROC32_ID : S_LPROC32_ID;
s->function.parent = 0;
s->function.end = 0;
s->function.next = 0;
s->function.type = func_id_type;
s->function.flags = 0;
s->function.name = xstrdup (name);
s->function.die = die;
if (last_sym)
last_sym->next = s;
else
sym = s;
last_sym = s;
}
/* Loop through the DIEs that have been output for our TU, and add CodeView
symbols for them. */
void
codeview_debug_early_finish (dw_die_ref die)
{
dw_die_ref first_child, c;
first_child = dw_get_die_child (die);
if (!first_child)
return;
c = first_child;
do
{
switch (dw_get_die_tag (c))
{
case DW_TAG_variable:
add_variable (c);
break;
case DW_TAG_subprogram:
add_function (c);
break;
default:
break;
}
c = dw_get_die_sib (c);
}
while (c != first_child);
flush_deferred_types ();
}
#endif