xref: /openbsd-src/gnu/llvm/llvm/tools/obj2yaml/dxcontainer2yaml.cpp (revision d415bd752c734aee168c4ee86ff32e8cc249eb16)
1*d415bd75Srobert //===------ dxcontainer2yaml.cpp - obj2yaml conversion tool -----*- C++ -*-===//
2*d415bd75Srobert //
3*d415bd75Srobert // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*d415bd75Srobert // See https://llvm.org/LICENSE.txt for license information.
5*d415bd75Srobert // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*d415bd75Srobert //
7*d415bd75Srobert //===----------------------------------------------------------------------===//
8*d415bd75Srobert 
9*d415bd75Srobert #include "obj2yaml.h"
10*d415bd75Srobert #include "llvm/Object/DXContainer.h"
11*d415bd75Srobert #include "llvm/ObjectYAML/DXContainerYAML.h"
12*d415bd75Srobert #include "llvm/Support/Error.h"
13*d415bd75Srobert 
14*d415bd75Srobert #include <algorithm>
15*d415bd75Srobert 
16*d415bd75Srobert using namespace llvm;
17*d415bd75Srobert using namespace llvm::object;
18*d415bd75Srobert 
19*d415bd75Srobert static Expected<DXContainerYAML::Object *>
dumpDXContainer(MemoryBufferRef Source)20*d415bd75Srobert dumpDXContainer(MemoryBufferRef Source) {
21*d415bd75Srobert   assert(file_magic::dxcontainer_object == identify_magic(Source.getBuffer()));
22*d415bd75Srobert 
23*d415bd75Srobert   Expected<DXContainer> ExDXC = DXContainer::create(Source);
24*d415bd75Srobert   if (!ExDXC)
25*d415bd75Srobert     return ExDXC.takeError();
26*d415bd75Srobert   DXContainer Container = *ExDXC;
27*d415bd75Srobert 
28*d415bd75Srobert   std::unique_ptr<DXContainerYAML::Object> Obj =
29*d415bd75Srobert       std::make_unique<DXContainerYAML::Object>();
30*d415bd75Srobert 
31*d415bd75Srobert   for (uint8_t Byte : Container.getHeader().FileHash.Digest)
32*d415bd75Srobert     Obj->Header.Hash.push_back(Byte);
33*d415bd75Srobert   Obj->Header.Version.Major = Container.getHeader().Version.Major;
34*d415bd75Srobert   Obj->Header.Version.Minor = Container.getHeader().Version.Minor;
35*d415bd75Srobert   Obj->Header.FileSize = Container.getHeader().FileSize;
36*d415bd75Srobert   Obj->Header.PartCount = Container.getHeader().PartCount;
37*d415bd75Srobert 
38*d415bd75Srobert   Obj->Header.PartOffsets = std::vector<uint32_t>();
39*d415bd75Srobert   for (const auto P : Container) {
40*d415bd75Srobert     Obj->Header.PartOffsets->push_back(P.Offset);
41*d415bd75Srobert     Obj->Parts.push_back(
42*d415bd75Srobert         DXContainerYAML::Part(P.Part.getName().str(), P.Part.Size));
43*d415bd75Srobert     DXContainerYAML::Part &NewPart = Obj->Parts.back();
44*d415bd75Srobert     dxbc::PartType PT = dxbc::parsePartType(P.Part.getName());
45*d415bd75Srobert     switch (PT) {
46*d415bd75Srobert     case dxbc::PartType::DXIL: {
47*d415bd75Srobert       std::optional<DXContainer::DXILData> DXIL = Container.getDXIL();
48*d415bd75Srobert       assert(DXIL && "Since we are iterating and found a DXIL part, "
49*d415bd75Srobert                      "this should never not have a value");
50*d415bd75Srobert       NewPart.Program = DXContainerYAML::DXILProgram{
51*d415bd75Srobert           DXIL->first.MajorVersion,
52*d415bd75Srobert           DXIL->first.MinorVersion,
53*d415bd75Srobert           DXIL->first.ShaderKind,
54*d415bd75Srobert           DXIL->first.Size,
55*d415bd75Srobert           DXIL->first.Bitcode.MajorVersion,
56*d415bd75Srobert           DXIL->first.Bitcode.MinorVersion,
57*d415bd75Srobert           DXIL->first.Bitcode.Offset,
58*d415bd75Srobert           DXIL->first.Bitcode.Size,
59*d415bd75Srobert           std::vector<llvm::yaml::Hex8>(
60*d415bd75Srobert               DXIL->second, DXIL->second + DXIL->first.Bitcode.Size)};
61*d415bd75Srobert       break;
62*d415bd75Srobert     }
63*d415bd75Srobert     case dxbc::PartType::SFI0: {
64*d415bd75Srobert       std::optional<uint64_t> Flags = Container.getShaderFlags();
65*d415bd75Srobert       // Omit the flags in the YAML if they are missing or zero.
66*d415bd75Srobert       if (Flags && *Flags > 0)
67*d415bd75Srobert         NewPart.Flags = DXContainerYAML::ShaderFlags(*Flags);
68*d415bd75Srobert       break;
69*d415bd75Srobert     }
70*d415bd75Srobert     case dxbc::PartType::HASH: {
71*d415bd75Srobert       std::optional<dxbc::ShaderHash> Hash = Container.getShaderHash();
72*d415bd75Srobert       if (Hash && Hash->isPopulated())
73*d415bd75Srobert         NewPart.Hash = DXContainerYAML::ShaderHash(*Hash);
74*d415bd75Srobert       break;
75*d415bd75Srobert     }
76*d415bd75Srobert     case dxbc::PartType::Unknown:
77*d415bd75Srobert       break;
78*d415bd75Srobert     }
79*d415bd75Srobert   }
80*d415bd75Srobert 
81*d415bd75Srobert   return Obj.release();
82*d415bd75Srobert }
83*d415bd75Srobert 
dxcontainer2yaml(llvm::raw_ostream & Out,llvm::MemoryBufferRef Source)84*d415bd75Srobert llvm::Error dxcontainer2yaml(llvm::raw_ostream &Out,
85*d415bd75Srobert                              llvm::MemoryBufferRef Source) {
86*d415bd75Srobert   Expected<DXContainerYAML::Object *> YAMLOrErr = dumpDXContainer(Source);
87*d415bd75Srobert   if (!YAMLOrErr)
88*d415bd75Srobert     return YAMLOrErr.takeError();
89*d415bd75Srobert 
90*d415bd75Srobert   std::unique_ptr<DXContainerYAML::Object> YAML(YAMLOrErr.get());
91*d415bd75Srobert   yaml::Output Yout(Out);
92*d415bd75Srobert   Yout << *YAML;
93*d415bd75Srobert 
94*d415bd75Srobert   return Error::success();
95*d415bd75Srobert }
96