// Copyright (c) 2018, Marc Tiehuis // 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 path from 'path'; import Semver from 'semver'; import _ from 'underscore'; import type {PreliminaryCompilerInfo} from '../../types/compiler.interfaces.js'; import type {ParseFiltersAndOutputOptions} from '../../types/features/filters.interfaces.js'; import type {SelectedLibraryVersion} from '../../types/libraries/libraries.interfaces.js'; import {BaseCompiler} from '../base-compiler.js'; import {asSafeVer} from '../utils.js'; export class ZigCompiler extends BaseCompiler { private readonly self_hosted_cli: boolean; static get key() { return 'zig'; } constructor(info: PreliminaryCompilerInfo, env) { super(info, env); this.compiler.supportsIntel = true; this.compiler.supportsIrView = true; this.self_hosted_cli = this.compiler.semver === 'trunk' || !!(this.compiler.semver && Semver.gt(asSafeVer(this.compiler.semver), '0.6.0', true)); if (this.self_hosted_cli) { this.compiler.irArg = ['-femit-llvm-ir']; } else { this.compiler.irArg = ['--emit', 'llvm-ir']; } this.compiler.minIrArgs = ['--emit llvm-ir', '-femit-llvm-ir']; } override getSharedLibraryPathsAsArguments(): string[] { return []; } override preProcess(source: string): string { if (Semver.eq(asSafeVer(this.compiler.semver), '0.2.0', true)) { source += '\n'; source += 'extern fn zig_panic() noreturn;\n'; source += 'pub fn panic(msg: []const u8, error_return_trace: ?&@import("builtin").StackTrace) noreturn {\n'; source += ' zig_panic();\n'; source += '}\n'; } else if ( Semver.gte(asSafeVer(this.compiler.semver), '0.3.0', true) && Semver.lte(asSafeVer(this.compiler.semver), '0.7.1', true) ) { source += '\n'; source += 'extern fn zig_panic() noreturn;\n'; source += 'pub fn panic(msg: []const u8, error_return_trace: ?*@import("builtin").StackTrace) noreturn {\n'; source += ' zig_panic();\n'; source += '}\n'; } else if ( Semver.gte(asSafeVer(this.compiler.semver), '0.8.0', true) && Semver.lte(asSafeVer(this.compiler.semver), '0.9.0', true) ) { source += '\n'; source += 'extern fn zig_panic() noreturn;\n'; source += 'pub fn panic(msg: []const u8, error_return_trace: ' + '?*@import("std").builtin.StackTrace) noreturn {\n'; source += ' _ = msg;\n'; source += ' _ = error_return_trace;\n'; source += ' zig_panic();\n'; source += '}\n'; } else { source += '\n'; source += 'extern fn zig_panic() noreturn;\n'; source += 'pub fn panic(msg: []const u8, error_return_trace: ' + '?*@import("std").builtin.StackTrace, ret_addr: ?usize) noreturn {\n'; source += ' _ = msg;\n'; source += ' _ = error_return_trace;\n'; source += ' _ = ret_addr;\n'; source += ' zig_panic();\n'; source += '}\n'; } return source; } override optionsForFilter( filters: ParseFiltersAndOutputOptions, outputFilename: string, userOptions: string[], ): string[] { let options = [filters.execute ? 'build-exe' : 'build-obj']; const desiredName = path.basename(outputFilename); // strip '.s' if we aren't executing const name = filters.execute ? desiredName : desiredName.slice(0, -2); if (this.self_hosted_cli) { // Versions after 0.6.0 use a different command line interface. const outputDir = path.dirname(outputFilename); options.push('--cache-dir', outputDir, '--name', name); if (filters.binary) { options.push('-femit-bin=' + desiredName); } else { options.push('-fno-emit-bin', '-femit-asm=' + desiredName); } return options; } if (this.compiler.semver && Semver.gt(asSafeVer(this.compiler.semver), '0.3.0', true)) { const outputDir = path.dirname(outputFilename); options.push('--cache-dir', outputDir, '--output-dir', outputDir, '--name', name); } else { // Older versions use a different command line interface (#1304) options.push( '--cache-dir', path.dirname(outputFilename), '--output', this.filename(outputFilename), '--output-h', '/dev/null', ); } if (!filters.binary) { const userRequestedEmit = _.any(userOptions, opt => opt.includes('--emit')); if (!userRequestedEmit) { options = options.concat('--emit', 'asm'); } if (filters.intel) options = options.concat('-mllvm', '--x86-asm-syntax=intel'); } return options; } override getIncludeArguments(libraries: SelectedLibraryVersion[]) { return libraries.flatMap(selectedLib => { const foundVersion = this.findLibVersion(selectedLib); if (!foundVersion) return []; // Zig should not have more than 1 path, but it's still an array so spread it return [ '--pkg-begin', ...(foundVersion.name ? [foundVersion.name] : []), ...foundVersion.path, '--pkg-end', ]; }); } override getIrOutputFilename(inputFilename: string): string { return this.getOutputFilename(path.dirname(inputFilename), this.outputFilebase).replace('.s', '.ll'); } override filterUserOptions(userOptions: string[]): string[] { const forbiddenOptions = /^(((--(cache-dir|name|output|verbose))|(-(mllvm|f(no-)?emit-))).*)$/; return _.filter(userOptions, option => !forbiddenOptions.test(option)); } override isCfgCompiler(/*compilerVersion*/): boolean { return true; } }