#define TSDL_MAGIC 0x75d11d57
+#define NSEC_PER_SEC 1000000000LL
+
struct packet_header {
uint32_t magic;
uint8_t uuid[16];
uint8_t minor;
} __attribute__((__packed__));
+BT_HIDDEN
+FILE *ctf_fs_metadata_open_file(const char *trace_path)
+{
+ GString *metadata_path = g_string_new(trace_path);
+ FILE *fp = NULL;
+
+ if (!metadata_path) {
+ goto error;
+ }
+
+ g_string_append(metadata_path, "/" CTF_FS_METADATA_FILENAME);
+ fp = fopen(metadata_path->str, "rb");
+ if (!fp) {
+ goto error;
+ }
+
+ goto end;
+
+error:
+ if (fp) {
+ fclose(fp);
+ fp = NULL;
+ }
+
+end:
+ g_string_free(metadata_path, TRUE);
+ return fp;
+}
+
static struct ctf_fs_file *get_file(struct ctf_fs_component *ctf_fs,
const char *trace_path)
{
return file;
}
-static
-bool is_packetized(struct ctf_fs_component *ctf_fs, struct ctf_fs_file *file)
+BT_HIDDEN
+bool ctf_metadata_is_packetized(FILE *fp, int *byte_order)
{
uint32_t magic;
size_t len;
int ret = 0;
- len = fread(&magic, sizeof(magic), 1, file->fp);
+ len = fread(&magic, sizeof(magic), 1, fp);
if (len != 1) {
goto end;
}
- if (magic == TSDL_MAGIC) {
- ret = 1;
- ctf_fs->metadata.bo = BYTE_ORDER;
- } else if (magic == GUINT32_SWAP_LE_BE(TSDL_MAGIC)) {
- ret = 1;
- ctf_fs->metadata.bo = BYTE_ORDER == BIG_ENDIAN ?
- LITTLE_ENDIAN : BIG_ENDIAN;
+ if (byte_order) {
+ if (magic == TSDL_MAGIC) {
+ ret = 1;
+ *byte_order = BYTE_ORDER;
+ } else if (magic == GUINT32_SWAP_LE_BE(TSDL_MAGIC)) {
+ ret = 1;
+ *byte_order = BYTE_ORDER == BIG_ENDIAN ?
+ LITTLE_ENDIAN : BIG_ENDIAN;
+ }
}
end:
- rewind(file->fp);
+ rewind(fp);
return ret;
}
}
static
-int decode_packet(struct ctf_fs_component *ctf_fs, FILE *in_fp, FILE *out_fp)
+int decode_packet(struct ctf_fs_component *ctf_fs, FILE *in_fp, FILE *out_fp,
+ int byte_order)
{
struct packet_header header;
size_t readlen, writelen, toread;
goto error;
}
- if (ctf_fs->metadata.bo != BYTE_ORDER) {
+ if (byte_order != BYTE_ORDER) {
header.magic = GUINT32_SWAP_LE_BE(header.magic);
header.checksum = GUINT32_SWAP_LE_BE(header.checksum);
header.content_size = GUINT32_SWAP_LE_BE(header.content_size);
}
/* Set expected trace UUID if not set; otherwise validate it */
- if (!ctf_fs->metadata.is_uuid_set) {
- memcpy(ctf_fs->metadata.uuid, header.uuid, sizeof(header.uuid));
- ctf_fs->metadata.is_uuid_set = true;
- } else if (bt_uuid_compare(header.uuid, ctf_fs->metadata.uuid)) {
- PERR("Metadata UUID mismatch between packets of the same file\n");
- goto error;
+ if (ctf_fs) {
+ if (!ctf_fs->metadata->is_uuid_set) {
+ memcpy(ctf_fs->metadata->uuid, header.uuid, sizeof(header.uuid));
+ ctf_fs->metadata->is_uuid_set = true;
+ } else if (bt_uuid_compare(header.uuid, ctf_fs->metadata->uuid)) {
+ PERR("Metadata UUID mismatch between packets of the same file\n");
+ goto error;
+ }
}
if ((header.content_size / CHAR_BIT) < sizeof(header)) {
return ret;
}
-static
-int packetized_file_to_buf(struct ctf_fs_component *ctf_fs, struct ctf_fs_file *file,
- uint8_t **buf)
+BT_HIDDEN
+int ctf_metadata_packetized_file_to_buf(struct ctf_fs_component *ctf_fs,
+ FILE *fp, uint8_t **buf, int byte_order)
{
FILE *out_fp;
size_t size;
}
for (;;) {
- if (feof(file->fp) != 0) {
+ if (feof(fp) != 0) {
break;
}
- tret = decode_packet(ctf_fs, file->fp, out_fp);
+ tret = decode_packet(ctf_fs, fp, out_fp, byte_order);
if (tret) {
PERR("Cannot decode packet #%zu\n", packet_index);
goto error;
return ret;
}
-void ctf_fs_metadata_set_trace(struct ctf_fs_component *ctf_fs)
+int ctf_fs_metadata_set_trace(struct ctf_fs_component *ctf_fs)
{
int ret = 0;
struct ctf_fs_file *file = get_file(ctf_fs, ctf_fs->trace_path->str);
// yydebug = 1;
}
- if (is_packetized(ctf_fs, file)) {
+ if (ctf_metadata_is_packetized(file->fp, &ctf_fs->metadata->bo)) {
PDBG("Metadata file \"%s\" is packetized\n", file->path->str);
- ret = packetized_file_to_buf(ctf_fs, file, &buf);
+ ret = ctf_metadata_packetized_file_to_buf(ctf_fs, file->fp,
+ &buf, ctf_fs->metadata->bo);
if (ret) {
// log: details
goto error;
}
cbuf = (char *) buf;
- ctf_fs->metadata.text = (char *) cbuf;
+ ctf_fs->metadata->text = (char *) cbuf;
/* Convert the real file pointer to a memory file pointer */
ret = fclose(file->fp);
}
ret = ctf_visitor_generate_ir(ctf_fs->error_fp, &scanner->ast->root,
- &ctf_fs->metadata.trace);
+ &ctf_fs->metadata->trace,
+ ctf_fs->options.clock_offset * NSEC_PER_SEC +
+ ctf_fs->options.clock_offset_ns);
if (ret) {
PERR("Cannot create trace object from metadata AST\n");
goto error;
goto end;
error:
- if (ctf_fs->metadata.text) {
- free(ctf_fs->metadata.text);
- ctf_fs->metadata.text = NULL;
+ if (ctf_fs->metadata->text) {
+ free(ctf_fs->metadata->text);
+ ctf_fs->metadata->text = NULL;
}
+ ret = -1;
+
end:
if (file) {
ctf_fs_file_destroy(file);
if (scanner) {
ctf_scanner_free(scanner);
}
+
+ return ret;
}
int ctf_fs_metadata_init(struct ctf_fs_metadata *metadata)
{
- /* Nothing to initialize for the moment */
+ /* Nothing to initialize for the moment. */
return 0;
}
-void ctf_fs_metadata_deinit(struct ctf_fs_metadata *metadata)
+void ctf_fs_metadata_fini(struct ctf_fs_metadata *metadata)
{
if (metadata->text) {
free(metadata->text);