X-Git-Url: https://git.cworth.org/git?a=blobdiff_plain;f=common%2Ftrace_parser.cpp;h=a9cdd0fbc0586465be407522a0148146e05f283a;hb=refs%2Fheads%2Fglx-copy-sub-buffer;hp=095af670e2159f79299fdba9bc4e642cbb4ac5e4;hpb=677f59a10bf71b0bf1b73bfc9c1accbf719fafad;p=apitrace diff --git a/common/trace_parser.cpp b/common/trace_parser.cpp index 095af67..a9cdd0f 100644 --- a/common/trace_parser.cpp +++ b/common/trace_parser.cpp @@ -65,6 +65,8 @@ bool Parser::open(const char *filename) { version = read_uint(); if (version > TRACE_VERSION) { std::cerr << "error: unsupported trace format version " << version << "\n"; + delete file; + file = NULL; return false; } api = API_UNKNOWN; @@ -187,8 +189,11 @@ Call *Parser::parse_call(Mode mode) { std::cerr << "\tLEAVE\n"; #endif call = parse_leave(mode); - adjust_call_flags(call); - return call; + if (call) { + adjust_call_flags(call); + return call; + } + break; default: std::cerr << "error: unknown event " << c << "\n"; exit(1); @@ -238,7 +243,7 @@ Parser::parse_function_sig(void) { } sig->arg_names = arg_names; sig->flags = lookupCallFlags(sig->name); - sig->offset = file->currentOffset(); + sig->fileOffset = file->currentOffset(); functions[id] = sig; /** @@ -276,7 +281,7 @@ Parser::parse_function_sig(void) { glGetErrorSig = sig; } - } else if (file->currentOffset() < sig->offset) { + } else if (file->currentOffset() < sig->fileOffset) { /* skip over the signature */ skip_string(); /* name */ unsigned num_args = read_uint(); @@ -306,9 +311,9 @@ StructSig *Parser::parse_struct_sig() { member_names[i] = read_string(); } sig->member_names = member_names; - sig->offset = file->currentOffset(); + sig->fileOffset = file->currentOffset(); structs[id] = sig; - } else if (file->currentOffset() < sig->offset) { + } else if (file->currentOffset() < sig->fileOffset) { /* skip over the signature */ skip_string(); /* name */ unsigned num_members = read_uint(); @@ -342,9 +347,9 @@ EnumSig *Parser::parse_old_enum_sig() { values->name = read_string(); values->value = read_sint(); sig->values = values; - sig->offset = file->currentOffset(); + sig->fileOffset = file->currentOffset(); enums[id] = sig; - } else if (file->currentOffset() < sig->offset) { + } else if (file->currentOffset() < sig->fileOffset) { /* skip over the signature */ skip_string(); /*name*/ scan_value(); @@ -371,9 +376,9 @@ EnumSig *Parser::parse_enum_sig() { it->value = read_sint(); } sig->values = values; - sig->offset = file->currentOffset(); + sig->fileOffset = file->currentOffset(); enums[id] = sig; - } else if (file->currentOffset() < sig->offset) { + } else if (file->currentOffset() < sig->fileOffset) { /* skip over the signature */ int num_values = read_uint(); for (int i = 0; i < num_values; ++i) { @@ -406,9 +411,9 @@ BitmaskSig *Parser::parse_bitmask_sig() { } } sig->flags = flags; - sig->offset = file->currentOffset(); + sig->fileOffset = file->currentOffset(); bitmasks[id] = sig; - } else if (file->currentOffset() < sig->offset) { + } else if (file->currentOffset() < sig->fileOffset) { /* skip over the signature */ int num_flags = read_uint(); for (int i = 0; i < num_flags; ++i) { @@ -456,6 +461,14 @@ Call *Parser::parse_leave(Mode mode) { } } if (!call) { + /* This might happen on random access, when an asynchronous call is stranded + * between two frames. We won't return this call, but we still need to skip + * over its data. + */ + const FunctionSig sig = {0, NULL, 0, NULL}; + call = new Call(&sig, 0, 0); + parse_call_details(call, SCAN); + delete call; return NULL; } @@ -489,6 +502,12 @@ bool Parser::parse_call_details(Call *call, Mode mode) { #endif call->ret = parse_value(mode); break; + case trace::CALL_BACKTRACE: +#if TRACE_VERBOSE + std::cerr << "\tCALL_BACKTRACE\n"; +#endif + parse_call_backtrace(call, mode); + break; default: std::cerr << "error: ("<name()<< ") unknown call detail " << c << "\n"; @@ -499,6 +518,83 @@ bool Parser::parse_call_details(Call *call, Mode mode) { } while(true); } +bool Parser::parse_call_backtrace(Call *call, Mode mode) { + unsigned num_frames = read_uint(); + Backtrace* backtrace = new Backtrace(num_frames); + for (unsigned i = 0; i < num_frames; ++i) { + (*backtrace)[i] = parse_backtrace_frame(mode); + } + call->backtrace = backtrace; + return true; +} + +StackFrame * Parser::parse_backtrace_frame(Mode mode) { + size_t id = read_uint(); + + StackFrameState *frame = lookup(frames, id); + + if (!frame) { + frame = new StackFrameState; + int c = read_byte(); + while (c != trace::BACKTRACE_END && + c != -1) { + switch (c) { + case trace::BACKTRACE_MODULE: + frame->module = read_string(); + break; + case trace::BACKTRACE_FUNCTION: + frame->function = read_string(); + break; + case trace::BACKTRACE_FILENAME: + frame->filename = read_string(); + break; + case trace::BACKTRACE_LINENUMBER: + frame->linenumber = read_uint(); + break; + case trace::BACKTRACE_OFFSET: + frame->offset = read_uint(); + break; + default: + std::cerr << "error: unknown backtrace detail " + << c << "\n"; + exit(1); + } + c = read_byte(); + } + + frame->fileOffset = file->currentOffset(); + frames[id] = frame; + } else if (file->currentOffset() < frame->fileOffset) { + int c = read_byte(); + while (c != trace::BACKTRACE_END && + c != -1) { + switch (c) { + case trace::BACKTRACE_MODULE: + scan_string(); + break; + case trace::BACKTRACE_FUNCTION: + scan_string(); + break; + case trace::BACKTRACE_FILENAME: + scan_string(); + break; + case trace::BACKTRACE_LINENUMBER: + scan_uint(); + break; + case trace::BACKTRACE_OFFSET: + scan_uint(); + break; + default: + std::cerr << "error: unknown backtrace detail " + << c << "\n"; + exit(1); + } + c = read_byte(); + } + } + + return frame; +} /** * Make adjustments to this particular call flags.