| // Copyright (c) 2022, 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 path from 'path'; |
| |
| import fs from 'fs-extra'; |
| import _ from 'underscore'; |
| |
| import {CompilationResult} from '../../types/compilation/compilation.interfaces'; |
| import {CompilerInfo} from '../../types/compiler.interfaces'; |
| import {ParseFiltersAndOutputOptions} from '../../types/features/filters.interfaces'; |
| import {ArtifactType} from '../../types/tool.interfaces'; |
| import {BaseCompiler} from '../base-compiler'; |
| import {CC65AsmParser} from '../parsers/asm-parser-cc65'; |
| import * as utils from '../utils'; |
| |
| export class Cc65Compiler extends BaseCompiler { |
| static get key() { |
| return 'cc65'; |
| } |
| |
| constructor(compilerInfo: CompilerInfo, env) { |
| super(compilerInfo, env); |
| |
| this.asm = new CC65AsmParser(this.compilerProps); |
| this.toolchainPath = path.resolve(path.dirname(compilerInfo.exe), '..'); |
| } |
| |
| override getSharedLibraryPathsAsArguments(libraries, libDownloadPath?) { |
| const libPathFlag = this.compiler.libpathFlag || '-L'; |
| |
| if (!libDownloadPath) { |
| libDownloadPath = '.'; |
| } |
| |
| return _.union( |
| [libPathFlag + libDownloadPath], |
| this.compiler.libPath.map(path => libPathFlag + path), |
| this.getSharedLibraryPaths(libraries).map(path => libPathFlag + path), |
| ) as string[]; |
| } |
| |
| override optionsForFilter(filters: ParseFiltersAndOutputOptions, outputFilename: string) { |
| if (filters.binary) { |
| return ['-g', '-o', this.filename(outputFilename)]; |
| } else { |
| return ['-g', '-S', '-c', '-o', this.filename(outputFilename)]; |
| } |
| } |
| |
| override getCompilerEnvironmentVariables(compilerflags) { |
| const allOptions = (this.compiler.options + ' ' + compilerflags).trim(); |
| return {...this.cmakeBaseEnv, CFLAGS: allOptions}; |
| } |
| |
| override async getCmakeBaseEnv() { |
| if (!this.compiler.exe) return {}; |
| |
| const env: Record<string, string> = {}; |
| |
| env.CC = this.compiler.exe; |
| |
| if (this.toolchainPath) { |
| const ldPath = `${this.toolchainPath}/bin/ld65`; |
| const arPath = `${this.toolchainPath}/bin/ar65`; |
| const asPath = `${this.toolchainPath}/bin/as65`; |
| |
| if (await utils.fileExists(ldPath)) env.LD = ldPath; |
| if (await utils.fileExists(arPath)) env.AR = arPath; |
| if (await utils.fileExists(asPath)) env.AS = asPath; |
| } |
| |
| return env; |
| } |
| |
| override async objdump( |
| outputFilename, |
| result: CompilationResult, |
| maxSize: number, |
| intelAsm, |
| demangle, |
| staticReloc: boolean, |
| dynamicReloc: boolean, |
| filters: ParseFiltersAndOutputOptions, |
| ) { |
| const res = await super.objdump( |
| outputFilename, |
| result, |
| maxSize, |
| intelAsm, |
| demangle, |
| staticReloc, |
| dynamicReloc, |
| filters, |
| ); |
| |
| const dirPath = path.dirname(outputFilename); |
| const nesFile = path.join(dirPath, 'example.nes'); |
| if (await utils.fileExists(nesFile)) { |
| await this.addArtifactToResult(res, nesFile, ArtifactType.nesrom); |
| } |
| |
| return res; |
| } |
| |
| override async doBuildstepAndAddToResult(result: CompilationResult, name, command, args, execParams) { |
| const stepResult = await super.doBuildstepAndAddToResult(result, name, command, args, execParams); |
| if (name === 'build') { |
| const mapFile = path.join(execParams.customCwd, 'map.txt'); |
| if (await utils.fileExists(mapFile)) { |
| const file_buffer = await fs.readFile(mapFile); |
| stepResult.stderr = stepResult.stderr.concat(utils.parseOutput(file_buffer.toString())); |
| } |
| } |
| return stepResult; |
| } |
| } |