| // 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 {ParseFilters} from '../../types/features/filters.interfaces'; |
| import {SelectedLibraryVersion} from '../../types/libraries/libraries.interfaces'; |
| import {BaseCompiler} from '../base-compiler'; |
| import {asSafeVer} from '../utils'; |
| |
| export class ZigCompiler extends BaseCompiler { |
| private readonly self_hosted_cli: boolean; |
| |
| static get key() { |
| return 'zig'; |
| } |
| |
| constructor(info, 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']; |
| } |
| } |
| |
| 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: ParseFilters, 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('-fno-emit-bin', '-femit-asm=' + desiredName); |
| } else { |
| options.push('-femit-bin=' + 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; |
| } |
| } |