clang 23.0.0git
SPIRV.cpp
Go to the documentation of this file.
1//===--- SPIRV.cpp - SPIR-V Tool Implementations ----------------*- C++ -*-===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8#include "SPIRV.h"
11#include "clang/Driver/Driver.h"
14
15using namespace clang::driver;
16using namespace clang::driver::toolchains;
17using namespace clang::driver::tools;
18using namespace llvm::opt;
19
21 const JobAction &JA,
22 const InputInfo &Output,
23 const InputInfo &Input,
24 const llvm::opt::ArgStringList &Args) {
25 llvm::opt::ArgStringList CmdArgs(Args);
26 CmdArgs.push_back(Input.getFilename());
27
28 assert(Input.getType() != types::TY_PP_Asm && "Unexpected input type");
29
30 if (Output.getType() == types::TY_PP_Asm)
31 CmdArgs.push_back("--spirv-tools-dis");
32
33 CmdArgs.append({"-o", Output.getFilename()});
34
35 // Try to find "llvm-spirv-<LLVM_VERSION_MAJOR>". Otherwise, fall back to
36 // plain "llvm-spirv".
37 using namespace std::string_literals;
38 auto VersionedTool = "llvm-spirv-"s + std::to_string(LLVM_VERSION_MAJOR);
39 std::string ExeCand = T.getToolChain().GetProgramPath(VersionedTool.c_str());
40 if (!llvm::sys::fs::can_execute(ExeCand))
41 ExeCand = T.getToolChain().GetProgramPath("llvm-spirv");
42
43 const char *Exec = C.getArgs().MakeArgString(ExeCand);
44 C.addCommand(std::make_unique<Command>(JA, T, ResponseFileSupport::None(),
45 Exec, CmdArgs, Input, Output));
46}
47
49 const JobAction &JA,
50 const InputInfo &Output,
51 const InputInfo &Input,
52 const llvm::opt::ArgStringList &Args) {
53 llvm::opt::ArgStringList CmdArgs(Args);
54 CmdArgs.push_back(Input.getFilename());
55
56 assert(Input.getType() == types::TY_PP_Asm && "Unexpected input type");
57
58 CmdArgs.append({"-o", Output.getFilename()});
59
60 // Try to find "spirv-as-<LLVM_VERSION_MAJOR>". Otherwise, fall back to
61 // plain "spirv-as".
62 using namespace std::string_literals;
63 auto VersionedTool = "spirv-as-"s + std::to_string(LLVM_VERSION_MAJOR);
64 std::string ExeCand = T.getToolChain().GetProgramPath(VersionedTool.c_str());
65 if (!llvm::sys::fs::can_execute(ExeCand))
66 ExeCand = T.getToolChain().GetProgramPath("spirv-as");
67
68 if (!llvm::sys::fs::can_execute(ExeCand) &&
69 !C.getArgs().hasArg(clang::options::OPT__HASH_HASH_HASH)) {
70 C.getDriver().Diag(clang::diag::err_drv_no_spv_tools) << "spirv-as";
71 return;
72 }
73 const char *Exec = C.getArgs().MakeArgString(ExeCand);
74 C.addCommand(std::make_unique<Command>(JA, T, ResponseFileSupport::None(),
75 Exec, CmdArgs, Input, Output));
76}
77
79 const JobAction &JA,
80 const InputInfo &Output,
81 const InputInfoList &Inputs,
82 const llvm::opt::ArgList &Args) {
83
84 ArgStringList LlvmLinkArgs;
85
86 for (auto Input : Inputs) {
87 if (Input.isFilename()) {
88 LlvmLinkArgs.push_back(Input.getFilename());
89 } else {
90 // Warn that any linker arguments will be dropped.
91 assert(Input.isInputArg() && "Unexpected linker input");
92 const llvm::opt::Arg &LinkerOpt = Input.getInputArg();
93 std::string LinkerOptStr = LinkerOpt.getAsString(Args);
94 const llvm::opt::Arg *EmitLLVM = Args.getLastArg(options::OPT_emit_llvm);
95 assert(EmitLLVM && "Unexpected linker input");
96 std::string EmitLLVMStr = EmitLLVM ? EmitLLVM->getAsString(Args) : "";
97 llvm::Triple Triple(T.getToolChain().getTriple());
98 C.getDriver().Diag(clang::diag::warn_drv_input_file_unused)
99 << Triple.getTriple() << LinkerOptStr << false << EmitLLVMStr;
100 }
101 }
102
103 tools::constructLLVMLinkCommand(C, T, JA, Inputs, LlvmLinkArgs, Output, Args);
104}
105
107 const InputInfo &Output,
108 const InputInfoList &Inputs,
109 const ArgList &Args,
110 const char *LinkingOutput) const {
111 claimNoWarnArgs(Args);
112 if (Inputs.size() != 1)
113 llvm_unreachable("Invalid number of input files.");
114 constructTranslateCommand(C, *this, JA, Output, Inputs[0], {});
115}
116
118 const InputInfo &Output,
119 const InputInfoList &Inputs,
120 const ArgList &Args,
121 const char *AssembleOutput) const {
122 claimNoWarnArgs(Args);
123 if (Inputs.size() != 1)
124 llvm_unreachable("Invalid number of input files.");
125 constructAssembleCommand(C, *this, JA, Output, Inputs[0], {});
126}
127
128clang::driver::Tool *SPIRVToolChain::getAssembler() const {
129 if (!Assembler)
130 Assembler = std::make_unique<SPIRV::Assembler>(*this);
131 return Assembler.get();
132}
133
138
140 switch (AC) {
141 default:
142 break;
144 return SPIRVToolChain::getAssembler();
145 }
146 return ToolChain::getTool(AC);
147}
151
153 const InputInfo &Output,
154 const InputInfoList &Inputs,
155 const ArgList &Args,
156 const char *LinkingOutput) const {
157 if (JA.getType() == types::TY_LLVM_BC) {
158 constructLLVMLinkCommand(C, *this, JA, Output, Inputs, Args);
159 return;
160 }
163 ArgStringList CmdArgs;
164 AddLinkerInputs(ToolChain, Inputs, Args, CmdArgs, JA);
165
166 CmdArgs.push_back("-o");
167 CmdArgs.push_back(Output.getFilename());
168
169 // TODO: Consider moving SPIR-V linking to a separate tool.
170 if (C.getDriver().isUsingLTO()) {
171 // Implement limited LTO support through llvm-lto.
172 if (Args.hasArg(options::OPT_sycl_link)) {
173 // For unsupported cases, throw the same error as when LTO isn't supported
174 // at all.
175 C.getDriver().Diag(clang::diag::err_drv_no_linker_llvm_support)
176 << ToolChain.getTriple().getTriple();
177 return;
178 }
179 Linker = ToolChain.GetProgramPath("llvm-lto");
180 // Disable internalization, otherwise GlobalDCE will optimize everything
181 // out.
182 CmdArgs.push_back("-enable-lto-internalization=false");
183 } else if (Args.hasArg(options::OPT_sycl_link)) {
184 // Use of --sycl-link will call the clang-sycl-linker instead of
185 // the default linker (spirv-link).
186 Linker = ToolChain.GetProgramPath("clang-sycl-linker");
187 } else if (!llvm::sys::fs::can_execute(Linker) &&
188 !C.getArgs().hasArg(clang::options::OPT__HASH_HASH_HASH)) {
189 C.getDriver().Diag(clang::diag::err_drv_no_spv_tools) << getShortName();
190 return;
191 }
192 C.addCommand(std::make_unique<Command>(JA, *this, ResponseFileSupport::None(),
193 Args.MakeArgString(Linker), CmdArgs,
194 Inputs, Output));
195}
196
197SPIRVToolChain::SPIRVToolChain(const Driver &D, const llvm::Triple &Triple,
198 const ArgList &Args)
199 : ToolChain(D, Triple, Args) {
200 // TODO: Revisit need/use of --sycl-link option once SYCL toolchain is
201 // available and SYCL linking support is moved there.
202 NativeLLVMSupport = Args.hasArg(options::OPT_sycl_link) || D.isUsingLTO();
203
204 // Lookup binaries into the driver directory.
205 getProgramPaths().push_back(getDriver().Dir);
206}
207
208bool SPIRVToolChain::HasNativeLLVMSupport() const { return NativeLLVMSupport; }
__device__ __2f16 float __ockl_bool s
types::ID getType() const
Definition Action.h:150
ActionClass getKind() const
Definition Action.h:149
Compilation - A set of tasks to perform for a single driver invocation.
Definition Compilation.h:45
Driver - Encapsulate logic for constructing compilation processes from a set of gcc-driver-like comma...
Definition Driver.h:99
InputInfo - Wrapper for information about an input source.
Definition InputInfo.h:22
const char * getFilename() const
Definition InputInfo.h:83
types::ID getType() const
Definition InputInfo.h:77
ToolChain - Access to tools for a single platform.
Definition ToolChain.h:92
const Driver & getDriver() const
Definition ToolChain.h:261
ToolChain(const Driver &D, const llvm::Triple &T, const llvm::opt::ArgList &Args)
Definition ToolChain.cpp:89
path_list & getProgramPaths()
Definition ToolChain.h:306
const llvm::Triple & getTriple() const
Definition ToolChain.h:263
std::string GetProgramPath(const char *Name) const
virtual Tool * getTool(Action::ActionClass AC) const
Tool - Information on a specific compilation tool.
Definition Tool.h:32
const ToolChain & getToolChain() const
Definition Tool.h:52
const char * getShortName() const
Definition Tool.h:50
Tool * buildLinker() const override
Definition SPIRV.cpp:148
clang::driver::Tool * getTool(Action::ActionClass AC) const override
Definition SPIRV.cpp:139
clang::driver::Tool * SelectTool(const JobAction &JA) const override
Choose a tool to use to handle the action JA.
Definition SPIRV.cpp:134
bool HasNativeLLVMSupport() const override
HasNativeLTOLinker - Check whether the linker and related tools have native LLVM support.
Definition SPIRV.cpp:208
SPIRVToolChain(const Driver &D, const llvm::Triple &Triple, const llvm::opt::ArgList &Args)
Definition SPIRV.cpp:197
void ConstructJob(Compilation &C, const JobAction &JA, const InputInfo &Output, const InputInfoList &Inputs, const llvm::opt::ArgList &TCArgs, const char *AssembleOutput) const override
ConstructJob - Construct jobs to perform the action JA, writing to Output and with Inputs,...
Definition SPIRV.cpp:117
Linker(const ToolChain &TC)
Definition SPIRV.h:51
void ConstructJob(Compilation &C, const JobAction &JA, const InputInfo &Output, const InputInfoList &Inputs, const llvm::opt::ArgList &TCArgs, const char *LinkingOutput) const override
ConstructJob - Construct jobs to perform the action JA, writing to Output and with Inputs,...
Definition SPIRV.cpp:152
void ConstructJob(Compilation &C, const JobAction &JA, const InputInfo &Output, const InputInfoList &Inputs, const llvm::opt::ArgList &TCArgs, const char *LinkingOutput) const override
ConstructJob - Construct jobs to perform the action JA, writing to Output and with Inputs,...
Definition SPIRV.cpp:106
void constructLLVMLinkCommand(Compilation &C, const Tool &T, const JobAction &JA, const InputInfo &Output, const InputInfoList &Inputs, const llvm::opt::ArgList &Args)
Definition SPIRV.cpp:78
void constructAssembleCommand(Compilation &C, const Tool &T, const JobAction &JA, const InputInfo &Output, const InputInfo &Input, const llvm::opt::ArgStringList &Args)
Definition SPIRV.cpp:48
void constructTranslateCommand(Compilation &C, const Tool &T, const JobAction &JA, const InputInfo &Output, const InputInfo &Input, const llvm::opt::ArgStringList &Args)
Definition SPIRV.cpp:20
void constructLLVMLinkCommand(Compilation &C, const Tool &T, const JobAction &JA, const InputInfoList &JobInputs, const llvm::opt::ArgStringList &LinkerInputs, const InputInfo &Output, const llvm::opt::ArgList &Args, const char *OutputFilename=nullptr)
void claimNoWarnArgs(const llvm::opt::ArgList &Args)
void AddLinkerInputs(const ToolChain &TC, const InputInfoList &Inputs, const llvm::opt::ArgList &Args, llvm::opt::ArgStringList &CmdArgs, const JobAction &JA)
SmallVector< InputInfo, 4 > InputInfoList
Definition Driver.h:50
static constexpr ResponseFileSupport None()
Returns a ResponseFileSupport indicating that response files are not supported.
Definition Job.h:78