15ffd83dbSDimitry Andric //===- SimpleTypeSerializer.cpp -----------------------------------------===//
25ffd83dbSDimitry Andric //
35ffd83dbSDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
45ffd83dbSDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
55ffd83dbSDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
65ffd83dbSDimitry Andric //
75ffd83dbSDimitry Andric //===----------------------------------------------------------------------===//
85ffd83dbSDimitry Andric 
90b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/SimpleTypeSerializer.h"
1081ad6265SDimitry Andric #include "llvm/DebugInfo/CodeView/CVRecord.h"
1181ad6265SDimitry Andric #include "llvm/DebugInfo/CodeView/RecordSerialization.h"
125ffd83dbSDimitry Andric #include "llvm/DebugInfo/CodeView/TypeRecordMapping.h"
135ffd83dbSDimitry Andric #include "llvm/Support/BinaryStreamWriter.h"
140b57cec5SDimitry Andric 
150b57cec5SDimitry Andric using namespace llvm;
160b57cec5SDimitry Andric using namespace llvm::codeview;
170b57cec5SDimitry Andric 
addPadding(BinaryStreamWriter & Writer)180b57cec5SDimitry Andric static void addPadding(BinaryStreamWriter &Writer) {
190b57cec5SDimitry Andric   uint32_t Align = Writer.getOffset() % 4;
200b57cec5SDimitry Andric   if (Align == 0)
210b57cec5SDimitry Andric     return;
220b57cec5SDimitry Andric 
230b57cec5SDimitry Andric   int PaddingBytes = 4 - Align;
240b57cec5SDimitry Andric   while (PaddingBytes > 0) {
250b57cec5SDimitry Andric     uint8_t Pad = static_cast<uint8_t>(LF_PAD0 + PaddingBytes);
260b57cec5SDimitry Andric     cantFail(Writer.writeInteger(Pad));
270b57cec5SDimitry Andric     --PaddingBytes;
280b57cec5SDimitry Andric   }
290b57cec5SDimitry Andric }
300b57cec5SDimitry Andric 
SimpleTypeSerializer()310b57cec5SDimitry Andric SimpleTypeSerializer::SimpleTypeSerializer() : ScratchBuffer(MaxRecordLength) {}
320b57cec5SDimitry Andric 
3381ad6265SDimitry Andric SimpleTypeSerializer::~SimpleTypeSerializer() = default;
340b57cec5SDimitry Andric 
350b57cec5SDimitry Andric template <typename T>
serialize(T & Record)360b57cec5SDimitry Andric ArrayRef<uint8_t> SimpleTypeSerializer::serialize(T &Record) {
375f757f3fSDimitry Andric   BinaryStreamWriter Writer(ScratchBuffer, llvm::endianness::little);
380b57cec5SDimitry Andric   TypeRecordMapping Mapping(Writer);
390b57cec5SDimitry Andric 
400b57cec5SDimitry Andric   // Write the record prefix first with a dummy length but real kind.
410b57cec5SDimitry Andric   RecordPrefix DummyPrefix(uint16_t(Record.getKind()));
420b57cec5SDimitry Andric   cantFail(Writer.writeObject(DummyPrefix));
430b57cec5SDimitry Andric 
440b57cec5SDimitry Andric   RecordPrefix *Prefix = reinterpret_cast<RecordPrefix *>(ScratchBuffer.data());
450b57cec5SDimitry Andric   CVType CVT(Prefix, sizeof(RecordPrefix));
460b57cec5SDimitry Andric 
470b57cec5SDimitry Andric   cantFail(Mapping.visitTypeBegin(CVT));
480b57cec5SDimitry Andric   cantFail(Mapping.visitKnownRecord(CVT, Record));
490b57cec5SDimitry Andric   cantFail(Mapping.visitTypeEnd(CVT));
500b57cec5SDimitry Andric 
510b57cec5SDimitry Andric   addPadding(Writer);
520b57cec5SDimitry Andric 
530b57cec5SDimitry Andric   // Update the size and kind after serialization.
540b57cec5SDimitry Andric   Prefix->RecordKind = CVT.kind();
550b57cec5SDimitry Andric   Prefix->RecordLen = Writer.getOffset() - sizeof(uint16_t);
560b57cec5SDimitry Andric 
57349cc55cSDimitry Andric   return {ScratchBuffer.data(), static_cast<size_t>(Writer.getOffset())};
580b57cec5SDimitry Andric }
590b57cec5SDimitry Andric 
600b57cec5SDimitry Andric // Explicitly instantiate the member function for each known type so that we can
610b57cec5SDimitry Andric // implement this in the cpp file.
620b57cec5SDimitry Andric #define TYPE_RECORD(EnumName, EnumVal, Name)                                   \
630b57cec5SDimitry Andric   template ArrayRef<uint8_t> llvm::codeview::SimpleTypeSerializer::serialize(  \
640b57cec5SDimitry Andric       Name##Record &Record);
650b57cec5SDimitry Andric #define TYPE_RECORD_ALIAS(EnumName, EnumVal, Name, AliasName)
660b57cec5SDimitry Andric #define MEMBER_RECORD(EnumName, EnumVal, Name)
670b57cec5SDimitry Andric #define MEMBER_RECORD_ALIAS(EnumName, EnumVal, Name, AliasName)
680b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/CodeViewTypes.def"
69