static struct traceframe_info * ctf_traceframe_info (struct target_ops *self) { struct traceframe_info *info = XCNEW (struct traceframe_info); const char *name; struct bt_iter_pos *pos; gdb_assert (ctf_iter != NULL); /* Save the current position. */ pos = bt_iter_get_pos (bt_ctf_get_iter (ctf_iter)); gdb_assert (pos->type == BT_SEEK_RESTORE); do { struct bt_ctf_event *event = bt_ctf_iter_read_event (ctf_iter); name = bt_ctf_event_name (event); if (name == NULL || strcmp (name, "register") == 0 || strcmp (name, "frame") == 0) ; else if (strcmp (name, "memory") == 0) { const struct bt_definition *scope = bt_ctf_get_top_level_scope (event, BT_EVENT_FIELDS); const struct bt_definition *def; struct mem_range *r; r = VEC_safe_push (mem_range_s, info->memory, NULL); def = bt_ctf_get_field (event, scope, "address"); r->start = bt_ctf_get_uint64 (def); def = bt_ctf_get_field (event, scope, "length"); r->length = (uint16_t) bt_ctf_get_uint64 (def); } else if (strcmp (name, "tsv") == 0) { int vnum; const struct bt_definition *scope = bt_ctf_get_top_level_scope (event, BT_EVENT_FIELDS); const struct bt_definition *def; def = bt_ctf_get_field (event, scope, "num"); vnum = (int) bt_ctf_get_int64 (def); VEC_safe_push (int, info->tvars, vnum); } else { warning (_("Unhandled trace block type (%s) " "while building trace frame info."), name); } if (bt_iter_next (bt_ctf_get_iter (ctf_iter)) < 0) break; }
static int ctf_get_trace_state_variable_value (struct target_ops *self, int tsvnum, LONGEST *val) { struct bt_iter_pos *pos; int found = 0; gdb_assert (ctf_iter != NULL); /* Save the current position. */ pos = bt_iter_get_pos (bt_ctf_get_iter (ctf_iter)); gdb_assert (pos->type == BT_SEEK_RESTORE); /* Iterate through the traceframe's blocks, looking for 'V' block. */ while (1) { struct bt_ctf_event *event = bt_ctf_iter_read_event (ctf_iter); const char *name = bt_ctf_event_name (event); if (name == NULL || strcmp (name, "frame") == 0) break; else if (strcmp (name, "tsv") == 0) { const struct bt_definition *scope; const struct bt_definition *def; scope = bt_ctf_get_top_level_scope (event, BT_EVENT_FIELDS); def = bt_ctf_get_field (event, scope, "num"); if (tsvnum == (int32_t) bt_ctf_get_uint64 (def)) { def = bt_ctf_get_field (event, scope, "val"); *val = bt_ctf_get_uint64 (def); found = 1; } } if (bt_iter_next (bt_ctf_get_iter (ctf_iter)) < 0) break; } /* Restore the position. */ bt_iter_set_pos (bt_ctf_get_iter (ctf_iter), pos); return found; }
int set_values_position(const LttvTracesetPosition *pos) { LttvTracesetPosition previous_pos; previous_pos.iter = pos->iter; previous_pos.bt_pos = bt_iter_get_pos(bt_ctf_get_iter(pos->iter)); /* Seek to the new desired position */ lttv_traceset_seek_to_position(pos); /*Read the event*/ struct bt_ctf_event *event = bt_ctf_iter_read_event(pos->iter); if(event != NULL){ ((LttvTracesetPosition *)pos)->timestamp = bt_ctf_get_timestamp(event); LttvEvent lttv_event; lttv_event.bt_event = event; ((LttvTracesetPosition *)pos)->cpu_id = lttv_traceset_get_cpuid_from_event(<tv_event); } else { /* The event is null */ return 0; } /* Reassign the previously saved position */ lttv_traceset_seek_to_position(&previous_pos); /*We must desallocate because the function bt_iter_get_pos() does a g_new */ bt_iter_free_pos(previous_pos.bt_pos); if (pos->timestamp == G_MAXUINT64) { return 0; } return 1; }
static CORE_ADDR ctf_get_traceframe_address (void) { struct bt_ctf_event *event = NULL; struct bt_iter_pos *pos; CORE_ADDR addr = 0; gdb_assert (ctf_iter != NULL); pos = bt_iter_get_pos (bt_ctf_get_iter (ctf_iter)); gdb_assert (pos->type == BT_SEEK_RESTORE); while (1) { const char *name; struct bt_ctf_event *event1; event1 = bt_ctf_iter_read_event (ctf_iter); name = bt_ctf_event_name (event1); if (name == NULL) break; else if (strcmp (name, "frame") == 0) { event = event1; break; } if (bt_iter_next (bt_ctf_get_iter (ctf_iter)) < 0) break; } if (event != NULL) { int tpnum = ctf_get_tpnum_from_frame_event (event); struct tracepoint *tp = get_tracepoint_by_number_on_target (tpnum); if (tp && tp->base.loc) addr = tp->base.loc->address; } /* Restore the position. */ bt_iter_set_pos (bt_ctf_get_iter (ctf_iter), pos); return addr; }
static void ctf_read_tp (struct uploaded_tp **uploaded_tps) { gdb_assert (ctf_iter != NULL); while (1) { struct bt_ctf_event *event; const struct bt_definition *scope; uint32_t u32; int32_t int32; uint64_t u64; struct uploaded_tp *utp = NULL; event = bt_ctf_iter_read_event (ctf_iter); scope = bt_ctf_get_top_level_scope (event, BT_STREAM_EVENT_HEADER); u32 = bt_ctf_get_uint64 (bt_ctf_get_field (event, scope, "id")); if (u32 != CTF_EVENT_ID_TP_DEF) break; scope = bt_ctf_get_top_level_scope (event, BT_EVENT_FIELDS); int32 = (int32_t) bt_ctf_get_int64 (bt_ctf_get_field (event, scope, "number")); u64 = bt_ctf_get_uint64 (bt_ctf_get_field (event, scope, "addr")); utp = get_uploaded_tp (int32, u64, uploaded_tps); SET_INT32_FIELD (event, scope, utp, enabled); SET_INT32_FIELD (event, scope, utp, step); SET_INT32_FIELD (event, scope, utp, pass); SET_INT32_FIELD (event, scope, utp, hit_count); SET_INT32_FIELD (event, scope, utp, type); /* Read 'cmd_strings'. */ SET_ARRAY_FIELD (event, scope, utp, cmd_num, cmd_strings); /* Read 'actions'. */ SET_ARRAY_FIELD (event, scope, utp, action_num, actions); /* Read 'step_actions'. */ SET_ARRAY_FIELD (event, scope, utp, step_action_num, step_actions); SET_STRING_FIELD(event, scope, utp, at_string); SET_STRING_FIELD(event, scope, utp, cond_string); if (bt_iter_next (bt_ctf_get_iter (ctf_iter)) < 0) break; } }
static int convert_trace(struct bt_trace_descriptor *td_write, struct bt_context *ctx) { struct bt_ctf_iter *iter; struct ctf_text_stream_pos *sout; struct bt_iter_pos *begin_pos = NULL, *end_pos = NULL; struct bt_ctf_event *ctf_event; int ret; sout = container_of(td_write, struct ctf_text_stream_pos, trace_descriptor); if (!sout->parent.event_cb) { return 0; } if (opt_stream_intersection) { iter = bt_ctf_iter_create_intersect(ctx, &begin_pos, &end_pos); } else { begin_pos = bt_iter_create_time_pos(NULL, 0); begin_pos->type = BT_SEEK_BEGIN; iter = bt_ctf_iter_create(ctx, begin_pos, NULL); } if (!iter) { ret = -1; goto error_iter; } while ((ctf_event = bt_ctf_iter_read_event(iter))) { ret = sout->parent.event_cb(&sout->parent, ctf_event->parent->stream); if (ret) { fprintf(stderr, "[error] Writing event failed.\n"); goto end; } ret = bt_iter_next(bt_ctf_get_iter(iter)); if (ret < 0) { goto end; } } ret = 0; end: bt_ctf_iter_destroy(iter); error_iter: bt_iter_free_pos(begin_pos); bt_iter_free_pos(end_pos); return ret; }
static void ctf_read_status (struct bt_ctf_event *event, struct trace_status *ts) { const struct bt_definition *scope = bt_ctf_get_top_level_scope (event, BT_EVENT_FIELDS); SET_INT32_FIELD (event, scope, ts, stop_reason); SET_INT32_FIELD (event, scope, ts, stopping_tracepoint); SET_INT32_FIELD (event, scope, ts, traceframe_count); SET_INT32_FIELD (event, scope, ts, traceframes_created); SET_INT32_FIELD (event, scope, ts, buffer_free); SET_INT32_FIELD (event, scope, ts, buffer_size); SET_INT32_FIELD (event, scope, ts, disconnected_tracing); SET_INT32_FIELD (event, scope, ts, circular_buffer); bt_iter_next (bt_ctf_get_iter (ctf_iter)); }
LttvTracesetPosition *lttv_traceset_create_current_position(const LttvTraceset *traceset) { LttvTracesetPosition *traceset_pos; traceset_pos = g_new(LttvTracesetPosition, 1); /* Check if the new passed */ if(traceset_pos == NULL) { return NULL; } traceset_pos->iter = traceset->iter; traceset_pos->bt_pos = bt_iter_get_pos(bt_ctf_get_iter(traceset->iter)); traceset_pos->timestamp = G_MAXUINT64; traceset_pos->cpu_id = INT_MAX; return traceset_pos; }
LttvTracesetPosition *lttv_traceset_create_time_position(LttvTraceset *traceset, LttTime timestamp) { LttvTracesetPosition *traceset_pos; traceset_pos = g_new(LttvTracesetPosition, 1); /* Check if the new passed */ if(traceset_pos == NULL) { return NULL; } traceset_pos->iter = traceset->iter; traceset_pos->bt_pos = bt_iter_create_time_pos( bt_ctf_get_iter(traceset_pos->iter), ltt_time_to_uint64(timestamp)); traceset_pos->timestamp = G_MAXUINT64; traceset_pos->cpu_id = INT_MAX; return traceset_pos; }
static void ctf_read_tsv (struct uploaded_tsv **uploaded_tsvs) { gdb_assert (ctf_iter != NULL); while (1) { struct bt_ctf_event *event; const struct bt_definition *scope; const struct bt_definition *def; uint32_t event_id; struct uploaded_tsv *utsv = NULL; event = bt_ctf_iter_read_event (ctf_iter); scope = bt_ctf_get_top_level_scope (event, BT_STREAM_EVENT_HEADER); event_id = bt_ctf_get_uint64 (bt_ctf_get_field (event, scope, "id")); if (event_id != CTF_EVENT_ID_TSV_DEF) break; scope = bt_ctf_get_top_level_scope (event, BT_EVENT_FIELDS); def = bt_ctf_get_field (event, scope, "number"); utsv = get_uploaded_tsv ((int32_t) bt_ctf_get_int64 (def), uploaded_tsvs); def = bt_ctf_get_field (event, scope, "builtin"); utsv->builtin = (int32_t) bt_ctf_get_int64 (def); def = bt_ctf_get_field (event, scope, "initial_value"); utsv->initial_value = bt_ctf_get_int64 (def); def = bt_ctf_get_field (event, scope, "name"); utsv->name = xstrdup (bt_ctf_get_string (def)); if (bt_iter_next (bt_ctf_get_iter (ctf_iter)) < 0) break; } }
static int ctf_trace_find (struct target_ops *self, enum trace_find_type type, int num, CORE_ADDR addr1, CORE_ADDR addr2, int *tpp) { int ret = -1; int tfnum = 0; int found = 0; struct bt_iter_pos pos; if (num == -1) { if (tpp != NULL) *tpp = -1; return -1; } gdb_assert (ctf_iter != NULL); /* Set iterator back to the start. */ bt_iter_set_pos (bt_ctf_get_iter (ctf_iter), start_pos); while (1) { int id; struct bt_ctf_event *event; const char *name; event = bt_ctf_iter_read_event (ctf_iter); name = bt_ctf_event_name (event); if (event == NULL || name == NULL) break; if (strcmp (name, "frame") == 0) { CORE_ADDR tfaddr; if (type == tfind_number) { /* Looking for a specific trace frame. */ if (tfnum == num) found = 1; } else { /* Start from the _next_ trace frame. */ if (tfnum > get_traceframe_number ()) { switch (type) { case tfind_tp: { struct tracepoint *tp = get_tracepoint (num); if (tp != NULL && (tp->number_on_target == ctf_get_tpnum_from_frame_event (event))) found = 1; break; } case tfind_pc: tfaddr = ctf_get_traceframe_address (); if (tfaddr == addr1) found = 1; break; case tfind_range: tfaddr = ctf_get_traceframe_address (); if (addr1 <= tfaddr && tfaddr <= addr2) found = 1; break; case tfind_outside: tfaddr = ctf_get_traceframe_address (); if (!(addr1 <= tfaddr && tfaddr <= addr2)) found = 1; break; default: internal_error (__FILE__, __LINE__, _("unknown tfind type")); } } } if (found) { if (tpp != NULL) *tpp = ctf_get_tpnum_from_frame_event (event); /* Skip the event "frame". */ bt_iter_next (bt_ctf_get_iter (ctf_iter)); return tfnum; } tfnum++; } if (bt_iter_next (bt_ctf_get_iter (ctf_iter)) < 0) break; } return -1; }
static enum target_xfer_status ctf_xfer_partial (struct target_ops *ops, enum target_object object, const char *annex, gdb_byte *readbuf, const gdb_byte *writebuf, ULONGEST offset, ULONGEST len, ULONGEST *xfered_len) { /* We're only doing regular memory for now. */ if (object != TARGET_OBJECT_MEMORY) return -1; if (readbuf == NULL) error (_("ctf_xfer_partial: trace file is read-only")); if (get_traceframe_number () != -1) { struct bt_iter_pos *pos; int i = 0; enum target_xfer_status res; /* Records the lowest available address of all blocks that intersects the requested range. */ ULONGEST low_addr_available = 0; gdb_assert (ctf_iter != NULL); /* Save the current position. */ pos = bt_iter_get_pos (bt_ctf_get_iter (ctf_iter)); gdb_assert (pos->type == BT_SEEK_RESTORE); /* Iterate through the traceframe's blocks, looking for memory. */ while (1) { ULONGEST amt; uint64_t maddr; uint16_t mlen; enum bfd_endian byte_order = gdbarch_byte_order (target_gdbarch ()); const struct bt_definition *scope; const struct bt_definition *def; struct bt_ctf_event *event = bt_ctf_iter_read_event (ctf_iter); const char *name = bt_ctf_event_name (event); if (name == NULL || strcmp (name, "frame") == 0) break; else if (strcmp (name, "memory") != 0) { if (bt_iter_next (bt_ctf_get_iter (ctf_iter)) < 0) break; continue; } scope = bt_ctf_get_top_level_scope (event, BT_EVENT_FIELDS); def = bt_ctf_get_field (event, scope, "address"); maddr = bt_ctf_get_uint64 (def); def = bt_ctf_get_field (event, scope, "length"); mlen = (uint16_t) bt_ctf_get_uint64 (def); /* If the block includes the first part of the desired range, return as much it has; GDB will re-request the remainder, which might be in a different block of this trace frame. */ if (maddr <= offset && offset < (maddr + mlen)) { const struct bt_definition *array = bt_ctf_get_field (event, scope, "contents"); const struct bt_declaration *decl = bt_ctf_get_decl_from_def (array); gdb_byte *contents; int k; contents = xmalloc (mlen); for (k = 0; k < mlen; k++) { const struct bt_definition *element = bt_ctf_get_index (event, array, k); contents[k] = (gdb_byte) bt_ctf_get_uint64 (element); } amt = (maddr + mlen) - offset; if (amt > len) amt = len; memcpy (readbuf, &contents[offset - maddr], amt); xfree (contents); /* Restore the position. */ bt_iter_set_pos (bt_ctf_get_iter (ctf_iter), pos); if (amt == 0) return TARGET_XFER_EOF; else { *xfered_len = amt; return TARGET_XFER_OK; } } if (offset < maddr && maddr < (offset + len)) if (low_addr_available == 0 || low_addr_available > maddr) low_addr_available = maddr; if (bt_iter_next (bt_ctf_get_iter (ctf_iter)) < 0) break; } /* Restore the position. */ bt_iter_set_pos (bt_ctf_get_iter (ctf_iter), pos); /* Requested memory is unavailable in the context of traceframes, and this address falls within a read-only section, fallback to reading from executable, up to LOW_ADDR_AVAILABLE */ if (offset < low_addr_available) len = min (len, low_addr_available - offset); res = exec_read_partial_read_only (readbuf, offset, len, xfered_len); if (res == TARGET_XFER_OK) return TARGET_XFER_OK; else { /* No use trying further, we know some memory starting at MEMADDR isn't available. */ *xfered_len = len; return TARGET_XFER_UNAVAILABLE; } } else { /* Fallback to reading from read-only sections. */ return section_table_read_available_memory (readbuf, offset, len, xfered_len); } }
static void ctf_fetch_registers (struct target_ops *ops, struct regcache *regcache, int regno) { struct gdbarch *gdbarch = get_regcache_arch (regcache); struct bt_ctf_event *event = NULL; struct bt_iter_pos *pos; /* An uninitialized reg size says we're not going to be successful at getting register blocks. */ if (trace_regblock_size == 0) return; gdb_assert (ctf_iter != NULL); /* Save the current position. */ pos = bt_iter_get_pos (bt_ctf_get_iter (ctf_iter)); gdb_assert (pos->type == BT_SEEK_RESTORE); while (1) { const char *name; struct bt_ctf_event *event1; event1 = bt_ctf_iter_read_event (ctf_iter); name = bt_ctf_event_name (event1); if (name == NULL || strcmp (name, "frame") == 0) break; else if (strcmp (name, "register") == 0) { event = event1; break; } if (bt_iter_next (bt_ctf_get_iter (ctf_iter)) < 0) break; } /* Restore the position. */ bt_iter_set_pos (bt_ctf_get_iter (ctf_iter), pos); if (event != NULL) { int offset, regsize, regn; const struct bt_definition *scope = bt_ctf_get_top_level_scope (event, BT_EVENT_FIELDS); const struct bt_definition *array = bt_ctf_get_field (event, scope, "contents"); gdb_byte *regs = (gdb_byte *) bt_ctf_get_char_array (array); /* Assume the block is laid out in GDB register number order, each register with the size that it has in GDB. */ offset = 0; for (regn = 0; regn < gdbarch_num_regs (gdbarch); regn++) { regsize = register_size (gdbarch, regn); /* Make sure we stay within block bounds. */ if (offset + regsize >= trace_regblock_size) break; if (regcache_register_status (regcache, regn) == REG_UNKNOWN) { if (regno == regn) { regcache_raw_supply (regcache, regno, regs + offset); break; } else if (regno == -1) { regcache_raw_supply (regcache, regn, regs + offset); } } offset += regsize; } } else tracefile_fetch_registers (regcache, regno); }
static void ctf_open (const char *dirname, int from_tty) { struct bt_ctf_event *event; uint32_t event_id; const struct bt_definition *scope; struct uploaded_tsv *uploaded_tsvs = NULL; struct uploaded_tp *uploaded_tps = NULL; if (!dirname) error (_("No CTF directory specified.")); ctf_open_dir (dirname); target_preopen (from_tty); /* Skip the first packet which about the trace status. The first event is "frame". */ event = bt_ctf_iter_read_event (ctf_iter); scope = bt_ctf_get_top_level_scope (event, BT_STREAM_EVENT_HEADER); event_id = bt_ctf_get_uint64 (bt_ctf_get_field (event, scope, "id")); if (event_id != CTF_EVENT_ID_FRAME) error (_("Wrong event id of the first event")); /* The second event is "status". */ bt_iter_next (bt_ctf_get_iter (ctf_iter)); event = bt_ctf_iter_read_event (ctf_iter); scope = bt_ctf_get_top_level_scope (event, BT_STREAM_EVENT_HEADER); event_id = bt_ctf_get_uint64 (bt_ctf_get_field (event, scope, "id")); if (event_id != CTF_EVENT_ID_STATUS) error (_("Wrong event id of the second event")); ctf_read_status (event, current_trace_status ()); ctf_read_tsv (&uploaded_tsvs); ctf_read_tp (&uploaded_tps); event = bt_ctf_iter_read_event (ctf_iter); /* EVENT can be NULL if we've already gone to the end of stream of events. */ if (event != NULL) { scope = bt_ctf_get_top_level_scope (event, BT_STREAM_EVENT_HEADER); event_id = bt_ctf_get_uint64 (bt_ctf_get_field (event, scope, "id")); if (event_id != CTF_EVENT_ID_FRAME) error (_("Wrong event id of the first event of the second packet")); } start_pos = bt_iter_get_pos (bt_ctf_get_iter (ctf_iter)); gdb_assert (start_pos->type == BT_SEEK_RESTORE); trace_dirname = xstrdup (dirname); push_target (&ctf_ops); inferior_appeared (current_inferior (), CTF_PID); inferior_ptid = pid_to_ptid (CTF_PID); add_thread_silent (inferior_ptid); merge_uploaded_trace_state_variables (&uploaded_tsvs); merge_uploaded_tracepoints (&uploaded_tps); post_create_inferior (&ctf_ops, from_tty); }
void lttv_traceset_seek_to_position(const LttvTracesetPosition *traceset_pos) { bt_iter_set_pos(bt_ctf_get_iter(traceset_pos->iter), traceset_pos->bt_pos); }