123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172 |
- #
- #
- # The Nim Compiler
- # (c) Copyright 2015 Andreas Rumpf
- #
- # See the file "copying.txt", included in this
- # distribution, for details about the copyright.
- #
- import std/[os, strutils, parseopt]
- when defined(nimPreviewSlimSystem):
- import std/assertions
- when defined(windows):
- when defined(gcc):
- when defined(x86):
- {.link: "../icons/nim.res".}
- else:
- {.link: "../icons/nim_icon.o".}
- when defined(amd64) and defined(vcc):
- {.link: "../icons/nim-amd64-windows-vcc.res".}
- when defined(i386) and defined(vcc):
- {.link: "../icons/nim-i386-windows-vcc.res".}
- import
- commands, options, msgs, extccomp, main, idents, lineinfos, cmdlinehelper,
- pathutils, modulegraphs
- from std/browsers import openDefaultBrowser
- from nodejs import findNodeJs
- when hasTinyCBackend:
- import tccgen
- when defined(profiler) or defined(memProfiler):
- {.hint: "Profiling support is turned on!".}
- import nimprof
- proc nimbleLockExists(config: ConfigRef): bool =
- const nimbleLock = "nimble.lock"
- let pd = if not config.projectPath.isEmpty: config.projectPath else: AbsoluteDir(getCurrentDir())
- if optSkipParentConfigFiles notin config.globalOptions:
- for dir in parentDirs(pd.string, fromRoot=true, inclusive=false):
- if fileExists(dir / nimbleLock):
- return true
- return fileExists(pd.string / nimbleLock)
- proc processCmdLine(pass: TCmdLinePass, cmd: string; config: ConfigRef) =
- var p = parseopt.initOptParser(cmd)
- var argsCount = 0
- config.commandLine.setLen 0
- # bugfix: otherwise, config.commandLine ends up duplicated
- while true:
- parseopt.next(p)
- case p.kind
- of cmdEnd: break
- of cmdLongOption, cmdShortOption:
- config.commandLine.add " "
- config.commandLine.addCmdPrefix p.kind
- config.commandLine.add p.key.quoteShell # quoteShell to be future proof
- if p.val.len > 0:
- config.commandLine.add ':'
- config.commandLine.add p.val.quoteShell
- if p.key == "": # `-` was passed to indicate main project is stdin
- p.key = "-"
- if processArgument(pass, p, argsCount, config): break
- else:
- processSwitch(pass, p, config)
- of cmdArgument:
- config.commandLine.add " "
- config.commandLine.add p.key.quoteShell
- if processArgument(pass, p, argsCount, config): break
- if pass == passCmd2:
- if {optRun, optWasNimscript} * config.globalOptions == {} and
- config.arguments.len > 0 and config.cmd notin {cmdTcc, cmdNimscript, cmdCrun}:
- rawMessage(config, errGenerated, errArgsNeedRunOption)
- if config.nimbleLockExists:
- # disable nimble path if nimble.lock is present.
- # see https://github.com/nim-lang/nimble/issues/1004
- disableNimblePath(config)
- proc getNimRunExe(conf: ConfigRef): string =
- # xxx consider defining `conf.getConfigVar("nimrun.exe")` to allow users to
- # customize the binary to run the command with, e.g. for custom `nodejs` or `wine`.
- if conf.isDefined("mingw"):
- if conf.isDefined("i386"): result = "wine"
- elif conf.isDefined("amd64"): result = "wine64"
- else: result = ""
- else:
- result = ""
- proc handleCmdLine(cache: IdentCache; conf: ConfigRef) =
- let self = NimProg(
- supportsStdinFile: true,
- processCmdLine: processCmdLine
- )
- self.initDefinesProg(conf, "nim_compiler")
- if paramCount() == 0:
- writeCommandLineUsage(conf)
- return
- self.processCmdLineAndProjectPath(conf)
- var graph = newModuleGraph(cache, conf)
- if not self.loadConfigsAndProcessCmdLine(cache, conf, graph):
- return
- if conf.cmd == cmdCheck and optWasNimscript notin conf.globalOptions and
- conf.backend == backendInvalid:
- conf.backend = backendC
- if conf.selectedGC == gcUnselected:
- if conf.backend in {backendC, backendCpp, backendObjc} or
- (conf.cmd in cmdDocLike and conf.backend != backendJs):
- initOrcDefines(conf)
- mainCommand(graph)
- if conf.hasHint(hintGCStats): echo(GC_getStatistics())
- #echo(GC_getStatistics())
- if conf.errorCounter != 0: return
- when hasTinyCBackend:
- if conf.cmd == cmdTcc:
- tccgen.run(conf, conf.arguments)
- if optRun in conf.globalOptions:
- let output = conf.absOutFile
- case conf.cmd
- of cmdBackends, cmdTcc:
- let nimRunExe = getNimRunExe(conf)
- var cmdPrefix = ""
- if nimRunExe.len > 0: cmdPrefix.add nimRunExe.quoteShell
- case conf.backend
- of backendC, backendCpp, backendObjc: discard
- of backendJs:
- # D20210217T215950:here this flag is needed for node < v15.0.0, otherwise
- # tasyncjs_fail` would fail, refs https://nodejs.org/api/cli.html#cli_unhandled_rejections_mode
- if cmdPrefix.len == 0: cmdPrefix = findNodeJs().quoteShell
- cmdPrefix.add " --unhandled-rejections=strict"
- else: raiseAssert $conf.backend
- if cmdPrefix.len > 0: cmdPrefix.add " "
- # without the `cmdPrefix.len > 0` check, on windows you'd get a cryptic:
- # `The parameter is incorrect`
- let cmd = cmdPrefix & output.quoteShell & ' ' & conf.arguments
- execExternalProgram(conf, cmd.strip(leading=false,trailing=true))
- of cmdDocLike, cmdRst2html, cmdRst2tex, cmdMd2html, cmdMd2tex: # bugfix(cmdRst2tex was missing)
- if conf.arguments.len > 0:
- # reserved for future use
- rawMessage(conf, errGenerated, "'$1 cannot handle arguments" % [$conf.cmd])
- openDefaultBrowser($output)
- else:
- # support as needed
- rawMessage(conf, errGenerated, "'$1 cannot handle --run" % [$conf.cmd])
- when declared(GC_setMaxPause):
- GC_setMaxPause 2_000
- when compileOption("gc", "refc"):
- # the new correct mark&sweet collector is too slow :-/
- GC_disableMarkAndSweep()
- when not defined(selftest):
- let conf = newConfigRef()
- handleCmdLine(newIdentCache(), conf)
- when declared(GC_setMaxPause):
- echo GC_getStatistics()
- msgQuit(int8(conf.errorCounter > 0))
|