| // Copyright (c) 2018, Compiler Explorer Authors |
| // All rights reserved. |
| // |
| // Redistribution and use in source and binary forms, with or without |
| // modification, are permitted provided that the following conditions are met: |
| // |
| // * Redistributions of source code must retain the above copyright notice, |
| // this list of conditions and the following disclaimer. |
| // * Redistributions in binary form must reproduce the above copyright |
| // notice, this list of conditions and the following disclaimer in the |
| // documentation and/or other materials provided with the distribution. |
| // |
| // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" |
| // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
| // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE |
| // ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE |
| // LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR |
| // CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF |
| // SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS |
| // INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN |
| // CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) |
| // ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE |
| // POSSIBILITY OF SUCH DAMAGE. |
| |
| import fs from 'fs'; |
| import path from 'path'; |
| |
| import _ from 'underscore'; |
| |
| import type {ExecutionOptions} from '../../types/compilation/compilation.interfaces.js'; |
| import type {PreliminaryCompilerInfo} from '../../types/compiler.interfaces.js'; |
| import type {ExecutableExecutionOptions, UnprocessedExecResult} from '../../types/execution/execution.interfaces.js'; |
| import type {ParseFiltersAndOutputOptions} from '../../types/features/filters.interfaces.js'; |
| import {BaseCompiler} from '../base-compiler.js'; |
| import {AmdgpuAsmParser} from '../parsers/asm-parser-amdgpu.js'; |
| import {SassAsmParser} from '../parsers/asm-parser-sass.js'; |
| |
| const offloadRegexp = /^#\s+__CLANG_OFFLOAD_BUNDLE__(__START__|__END__)\s+(.*)$/gm; |
| |
| export class ClangCompiler extends BaseCompiler { |
| protected asanSymbolizerPath?: string; |
| protected offloadBundlerPath?: string; |
| protected llvmDisassemblerPath?: string; |
| |
| static get key() { |
| return 'clang'; |
| } |
| |
| constructor(info: PreliminaryCompilerInfo, env) { |
| super(info, env); |
| this.compiler.supportsDeviceAsmView = true; |
| |
| const asanSymbolizerPath = path.join(path.dirname(this.compiler.exe), 'llvm-symbolizer'); |
| if (fs.existsSync(asanSymbolizerPath)) { |
| this.asanSymbolizerPath = asanSymbolizerPath; |
| } |
| |
| const offloadBundlerPath = path.join(path.dirname(this.compiler.exe), 'clang-offload-bundler'); |
| if (fs.existsSync(offloadBundlerPath)) { |
| this.offloadBundlerPath = offloadBundlerPath; |
| } |
| |
| const llvmDisassemblerPath = path.join(path.dirname(this.compiler.exe), 'llvm-dis'); |
| if (fs.existsSync(llvmDisassemblerPath)) { |
| this.llvmDisassemblerPath = llvmDisassemblerPath; |
| } else { |
| this.llvmDisassemblerPath = this.compilerProps<string | undefined>('llvmDisassembler'); |
| } |
| } |
| |
| override runExecutable(executable, executeParameters: ExecutableExecutionOptions, homeDir) { |
| if (this.asanSymbolizerPath) { |
| executeParameters.env = { |
| ASAN_SYMBOLIZER_PATH: this.asanSymbolizerPath, |
| ...executeParameters.env, |
| }; |
| } |
| return super.runExecutable(executable, executeParameters, homeDir); |
| } |
| |
| forceDwarf4UnlessOverridden(options: string[]) { |
| const hasOverride = _.any(options, option => { |
| return option.includes('-gdwarf-') || option.includes('-fdebug-default-version='); |
| }); |
| |
| if (!hasOverride) return ['-gdwarf-4'].concat(options); |
| |
| return options; |
| } |
| |
| override optionsForFilter(filters: ParseFiltersAndOutputOptions, outputFilename: string, userOptions?: string[]) { |
| const options = super.optionsForFilter(filters, outputFilename); |
| |
| return this.forceDwarf4UnlessOverridden(options); |
| } |
| |
| override runCompiler(compiler: string, options: string[], inputFilename: string, execOptions: ExecutionOptions) { |
| if (!execOptions) { |
| execOptions = this.getDefaultExecOptions(); |
| } |
| |
| execOptions.customCwd = path.dirname(inputFilename); |
| |
| return super.runCompiler(compiler, options, inputFilename, execOptions); |
| } |
| |
| async splitDeviceCode(assembly) { |
| // Check to see if there is any offload code in the assembly file. |
| if (!offloadRegexp.test(assembly)) return null; |
| |
| offloadRegexp.lastIndex = 0; |
| const matches = assembly.matchAll(offloadRegexp); |
| let prevStart = 0; |
| const devices = {}; |
| for (const match of matches) { |
| const [full, startOrEnd, triple] = match; |
| if (startOrEnd === '__START__') { |
| prevStart = match.index + full.length + 1; |
| } else { |
| devices[triple] = assembly.substr(prevStart, match.index - prevStart); |
| } |
| } |
| return devices; |
| } |
| |
| override async extractDeviceCode(result, filters, compilationInfo) { |
| const split = await this.splitDeviceCode(result.asm); |
| if (!split) return result; |
| |
| const devices = (result.devices = {}); |
| for (const key of Object.keys(split)) { |
| if (key.indexOf('host-') === 0) result.asm = split[key]; |
| else devices[key] = await this.processDeviceAssembly(key, split[key], filters, compilationInfo); |
| } |
| return result; |
| } |
| |
| async extractBitcodeFromBundle(bundlefile, devicename): Promise<string> { |
| const bcfile = path.join(path.dirname(bundlefile), devicename + '.bc'); |
| |
| const env = this.getDefaultExecOptions(); |
| env.customCwd = path.dirname(bundlefile); |
| |
| if (this.offloadBundlerPath) { |
| const unbundleResult: UnprocessedExecResult = await this.exec( |
| this.offloadBundlerPath, |
| ['-unbundle', '--type', 's', '--inputs', bundlefile, '--outputs', bcfile, '--targets', devicename], |
| env, |
| ); |
| if (unbundleResult.code !== 0) { |
| return unbundleResult.stderr; |
| } |
| } else { |
| return '<error: no offload bundler found to unbundle device code>'; |
| } |
| |
| if (this.llvmDisassemblerPath) { |
| const llvmirFile = path.join(path.dirname(bundlefile), devicename + '.ll'); |
| |
| const disassembleResult: UnprocessedExecResult = await this.exec(this.llvmDisassemblerPath, [bcfile], env); |
| if (disassembleResult.code !== 0) { |
| return disassembleResult.stderr; |
| } |
| |
| return fs.readFileSync(llvmirFile, 'utf8'); |
| } else { |
| return '<error: no llvm-dis found to disassemble bitcode>'; |
| } |
| } |
| |
| async processDeviceAssembly(deviceName, deviceAsm, filters, compilationInfo) { |
| if (deviceAsm.startsWith('BC')) { |
| deviceAsm = await this.extractBitcodeFromBundle(compilationInfo.outputFilename, deviceName); |
| } |
| |
| return this.llvmIr.isLlvmIr(deviceAsm) |
| ? this.llvmIr.process(deviceAsm, filters) |
| : this.asm.process(deviceAsm, filters); |
| } |
| } |
| |
| export class ClangCudaCompiler extends ClangCompiler { |
| static override get key() { |
| return 'clang-cuda'; |
| } |
| |
| constructor(info: PreliminaryCompilerInfo, env) { |
| super(info, env); |
| |
| this.asm = new SassAsmParser(); |
| } |
| |
| override getCompilerResultLanguageId() { |
| return 'ptx'; |
| } |
| |
| override optionsForFilter(filters: ParseFiltersAndOutputOptions, outputFilename: string) { |
| return ['-o', this.filename(outputFilename), '-g1', filters.binary ? '-c' : '-S']; |
| } |
| |
| override async objdump(outputFilename, result, maxSize) { |
| // For nvdisasm. |
| const args = [outputFilename, '-c', '-g', '-hex']; |
| const execOptions = {maxOutput: maxSize, customCwd: path.dirname(outputFilename)}; |
| |
| const objResult = await this.exec(this.compiler.objdumper, args, execOptions); |
| result.asm = objResult.stdout; |
| if (objResult.code === 0) { |
| result.objdumpTime = objResult.execTime; |
| } else { |
| result.asm = `<No output: nvdisasm returned ${objResult.code}>`; |
| } |
| return result; |
| } |
| } |
| |
| export class ClangHipCompiler extends ClangCompiler { |
| static override get key() { |
| return 'clang-hip'; |
| } |
| |
| constructor(info: PreliminaryCompilerInfo, env) { |
| super(info, env); |
| |
| this.asm = new AmdgpuAsmParser(); |
| } |
| |
| override optionsForFilter(filters: ParseFiltersAndOutputOptions, outputFilename: string) { |
| return ['-o', this.filename(outputFilename), '-g1', '--no-gpu-bundle-output', filters.binary ? '-c' : '-S']; |
| } |
| } |
| |
| export class ClangIntelCompiler extends ClangCompiler { |
| static override get key() { |
| return 'clang-intel'; |
| } |
| |
| constructor(info: PreliminaryCompilerInfo, env) { |
| super(info, env); |
| |
| if (!this.offloadBundlerPath) { |
| const offloadBundlerPath = path.join(path.dirname(this.compiler.exe), '../bin-llvm/clang-offload-bundler'); |
| if (fs.existsSync(offloadBundlerPath)) { |
| this.offloadBundlerPath = path.resolve(offloadBundlerPath); |
| } |
| } |
| } |
| |
| override getDefaultExecOptions(): ExecutionOptions { |
| const opts = super.getDefaultExecOptions(); |
| opts.env.PATH = process.env.PATH + path.delimiter + path.dirname(this.compiler.exe); |
| |
| return opts; |
| } |
| |
| override runExecutable(executable, executeParameters: ExecutableExecutionOptions, homeDir) { |
| executeParameters.env = { |
| OCL_ICD_FILENAMES: path.resolve(path.dirname(this.compiler.exe) + '/../lib/x64/libintelocl.so'), |
| ...executeParameters.env, |
| }; |
| return super.runExecutable(executable, executeParameters, homeDir); |
| } |
| } |