1d8e91e46SDimitry Andric //===- FileHeaderReader.cpp - XRay File Header Reader --------------------===//
2d8e91e46SDimitry Andric //
3e6d15924SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4e6d15924SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
5e6d15924SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6d8e91e46SDimitry Andric //
7d8e91e46SDimitry Andric //===----------------------------------------------------------------------===//
8d8e91e46SDimitry Andric #include "llvm/XRay/FileHeaderReader.h"
9d8e91e46SDimitry Andric
10d8e91e46SDimitry Andric namespace llvm {
11d8e91e46SDimitry Andric namespace xray {
12d8e91e46SDimitry Andric
13d8e91e46SDimitry Andric // Populates the FileHeader reference by reading the first 32 bytes of the file.
readBinaryFormatHeader(DataExtractor & HeaderExtractor,uint64_t & OffsetPtr)14d8e91e46SDimitry Andric Expected<XRayFileHeader> readBinaryFormatHeader(DataExtractor &HeaderExtractor,
151d5ae102SDimitry Andric uint64_t &OffsetPtr) {
16d8e91e46SDimitry Andric // FIXME: Maybe deduce whether the data is little or big-endian using some
17d8e91e46SDimitry Andric // magic bytes in the beginning of the file?
18d8e91e46SDimitry Andric
19d8e91e46SDimitry Andric // First 32 bytes of the file will always be the header. We assume a certain
20d8e91e46SDimitry Andric // format here:
21d8e91e46SDimitry Andric //
22d8e91e46SDimitry Andric // (2) uint16 : version
23d8e91e46SDimitry Andric // (2) uint16 : type
24d8e91e46SDimitry Andric // (4) uint32 : bitfield
25d8e91e46SDimitry Andric // (8) uint64 : cycle frequency
26d8e91e46SDimitry Andric // (16) - : padding
27d8e91e46SDimitry Andric XRayFileHeader FileHeader;
28d8e91e46SDimitry Andric auto PreReadOffset = OffsetPtr;
29d8e91e46SDimitry Andric FileHeader.Version = HeaderExtractor.getU16(&OffsetPtr);
30d8e91e46SDimitry Andric if (OffsetPtr == PreReadOffset)
31d8e91e46SDimitry Andric return createStringError(
32d8e91e46SDimitry Andric std::make_error_code(std::errc::invalid_argument),
331d5ae102SDimitry Andric "Failed reading version from file header at offset %" PRId64 ".",
341d5ae102SDimitry Andric OffsetPtr);
35d8e91e46SDimitry Andric
36d8e91e46SDimitry Andric PreReadOffset = OffsetPtr;
37d8e91e46SDimitry Andric FileHeader.Type = HeaderExtractor.getU16(&OffsetPtr);
38d8e91e46SDimitry Andric if (OffsetPtr == PreReadOffset)
39d8e91e46SDimitry Andric return createStringError(
40d8e91e46SDimitry Andric std::make_error_code(std::errc::invalid_argument),
411d5ae102SDimitry Andric "Failed reading file type from file header at offset %" PRId64 ".",
421d5ae102SDimitry Andric OffsetPtr);
43d8e91e46SDimitry Andric
44d8e91e46SDimitry Andric PreReadOffset = OffsetPtr;
45d8e91e46SDimitry Andric uint32_t Bitfield = HeaderExtractor.getU32(&OffsetPtr);
46d8e91e46SDimitry Andric if (OffsetPtr == PreReadOffset)
47d8e91e46SDimitry Andric return createStringError(
48d8e91e46SDimitry Andric std::make_error_code(std::errc::invalid_argument),
491d5ae102SDimitry Andric "Failed reading flag bits from file header at offset %" PRId64 ".",
501d5ae102SDimitry Andric OffsetPtr);
51d8e91e46SDimitry Andric
52d8e91e46SDimitry Andric FileHeader.ConstantTSC = Bitfield & 1uL;
53d8e91e46SDimitry Andric FileHeader.NonstopTSC = Bitfield & 1uL << 1;
54d8e91e46SDimitry Andric PreReadOffset = OffsetPtr;
55d8e91e46SDimitry Andric FileHeader.CycleFrequency = HeaderExtractor.getU64(&OffsetPtr);
56d8e91e46SDimitry Andric if (OffsetPtr == PreReadOffset)
57d8e91e46SDimitry Andric return createStringError(
58d8e91e46SDimitry Andric std::make_error_code(std::errc::invalid_argument),
591d5ae102SDimitry Andric "Failed reading cycle frequency from file header at offset %" PRId64
601d5ae102SDimitry Andric ".",
61d8e91e46SDimitry Andric OffsetPtr);
62d8e91e46SDimitry Andric
63d8e91e46SDimitry Andric std::memcpy(&FileHeader.FreeFormData,
64d8e91e46SDimitry Andric HeaderExtractor.getData().bytes_begin() + OffsetPtr, 16);
65d8e91e46SDimitry Andric
66d8e91e46SDimitry Andric // Manually advance the offset pointer 16 bytes, after getting a raw memcpy
67d8e91e46SDimitry Andric // from the underlying data.
68d8e91e46SDimitry Andric OffsetPtr += 16;
69d8e91e46SDimitry Andric return std::move(FileHeader);
70d8e91e46SDimitry Andric }
71d8e91e46SDimitry Andric
72d8e91e46SDimitry Andric } // namespace xray
73d8e91e46SDimitry Andric } // namespace llvm
74