mirror of
https://github.com/nim-lang/Nim.git
synced 2025-12-29 09:24:36 +00:00
626 lines
18 KiB
Nim
626 lines
18 KiB
Nim
#
|
|
#
|
|
# The Nim Compiler
|
|
# (c) Copyright 2017 Andreas Rumpf
|
|
#
|
|
# See the file "copying.txt", included in this
|
|
# distribution, for details about the copyright.
|
|
#
|
|
|
|
## Nimsuggest is a tool that helps to give editors IDE like capabilities.
|
|
|
|
import strutils, os, parseopt, parseutils, sequtils, net, rdstdin, sexp
|
|
# Do NOT import suggest. It will lead to wierd bugs with
|
|
# suggestionResultHook, because suggest.nim is included by sigmatch.
|
|
# So we import that one instead.
|
|
import compiler / [options, commands, modules, sem,
|
|
passes, passaux, msgs, nimconf,
|
|
extccomp, condsyms,
|
|
sigmatch, ast, scriptconfig,
|
|
idents, modulegraphs, vm, prefixmatches]
|
|
|
|
when defined(windows):
|
|
import winlean
|
|
else:
|
|
import posix
|
|
|
|
const DummyEof = "!EOF!"
|
|
const Usage = """
|
|
Nimsuggest - Tool to give every editor IDE like capabilities for Nim
|
|
Usage:
|
|
nimsuggest [options] projectfile.nim
|
|
|
|
Options:
|
|
--port:PORT port, by default 6000
|
|
--address:HOST binds to that address, by default ""
|
|
--stdin read commands from stdin and write results to
|
|
stdout instead of using sockets
|
|
--epc use emacs epc mode
|
|
--debug enable debug output
|
|
--log enable verbose logging to nimsuggest.log file
|
|
--v1 use version 1 of the protocol; for backwards compatibility
|
|
--refresh perform automatic refreshes to keep the analysis precise
|
|
--maxresults:N limit the number of suggestions to N
|
|
--tester implies --stdin and outputs a line
|
|
'""" & DummyEof & """' for the tester
|
|
|
|
The server then listens to the connection and takes line-based commands.
|
|
|
|
In addition, all command line options of Nim that do not affect code generation
|
|
are supported.
|
|
"""
|
|
type
|
|
Mode = enum mstdin, mtcp, mepc, mcmdsug, mcmdcon
|
|
CachedMsg = object
|
|
info: TLineInfo
|
|
msg: string
|
|
sev: Severity
|
|
CachedMsgs = seq[CachedMsg]
|
|
|
|
var
|
|
gPort = 6000.Port
|
|
gAddress = ""
|
|
gMode: Mode
|
|
gEmitEof: bool # whether we write '!EOF!' dummy lines
|
|
gLogging = defined(logging)
|
|
gRefresh: bool
|
|
|
|
requests: Channel[string]
|
|
results: Channel[Suggest]
|
|
|
|
proc writelnToChannel(line: string) =
|
|
results.send(Suggest(section: ideMsg, doc: line))
|
|
|
|
proc sugResultHook(s: Suggest) =
|
|
results.send(s)
|
|
|
|
proc errorHook(info: TLineInfo; msg: string; sev: Severity) =
|
|
results.send(Suggest(section: ideChk, filePath: toFullPath(info),
|
|
line: toLinenumber(info), column: toColumn(info), doc: msg,
|
|
forth: $sev))
|
|
|
|
proc myLog(s: string) =
|
|
if gLogging: log(s)
|
|
|
|
const
|
|
seps = {':', ';', ' ', '\t'}
|
|
Help = "usage: sug|con|def|use|dus|chk|mod|highlight|outline|known file.nim[;dirtyfile.nim]:line:col\n" &
|
|
"type 'quit' to quit\n" &
|
|
"type 'debug' to toggle debug mode on/off\n" &
|
|
"type 'terse' to toggle terse mode on/off"
|
|
|
|
type
|
|
EUnexpectedCommand = object of Exception
|
|
|
|
proc parseQuoted(cmd: string; outp: var string; start: int): int =
|
|
var i = start
|
|
i += skipWhitespace(cmd, i)
|
|
if cmd[i] == '"':
|
|
i += parseUntil(cmd, outp, '"', i+1)+2
|
|
else:
|
|
i += parseUntil(cmd, outp, seps, i)
|
|
result = i
|
|
|
|
proc sexp(s: IdeCmd|TSymKind|PrefixMatch): SexpNode = sexp($s)
|
|
|
|
proc sexp(s: Suggest): SexpNode =
|
|
# If you change the order here, make sure to change it over in
|
|
# nim-mode.el too.
|
|
let qp = if s.qualifiedPath.isNil: @[] else: s.qualifiedPath
|
|
result = convertSexp([
|
|
s.section,
|
|
s.symkind,
|
|
qp.map(newSString),
|
|
s.filePath,
|
|
s.forth,
|
|
s.line,
|
|
s.column,
|
|
s.doc,
|
|
s.quality
|
|
])
|
|
if s.section == ideSug:
|
|
result.add convertSexp(s.prefix)
|
|
|
|
proc sexp(s: seq[Suggest]): SexpNode =
|
|
result = newSList()
|
|
for sug in s:
|
|
result.add(sexp(sug))
|
|
|
|
proc listEpc(): SexpNode =
|
|
# This function is called from Emacs to show available options.
|
|
let
|
|
argspecs = sexp("file line column dirtyfile".split(" ").map(newSSymbol))
|
|
docstring = sexp("line starts at 1, column at 0, dirtyfile is optional")
|
|
result = newSList()
|
|
for command in ["sug", "con", "def", "use", "dus", "chk", "mod"]:
|
|
let
|
|
cmd = sexp(command)
|
|
methodDesc = newSList()
|
|
methodDesc.add(cmd)
|
|
methodDesc.add(argspecs)
|
|
methodDesc.add(docstring)
|
|
result.add(methodDesc)
|
|
|
|
proc findNode(n: PNode): PSym =
|
|
#echo "checking node ", n.info
|
|
if n.kind == nkSym:
|
|
if isTracked(n.info, n.sym.name.s.len): return n.sym
|
|
else:
|
|
for i in 0 ..< safeLen(n):
|
|
let res = n.sons[i].findNode
|
|
if res != nil: return res
|
|
|
|
proc symFromInfo(graph: ModuleGraph; gTrackPos: TLineInfo): PSym =
|
|
let m = graph.getModule(gTrackPos.fileIndex)
|
|
#echo m.isNil, " I knew it ", gTrackPos.fileIndex
|
|
if m != nil and m.ast != nil:
|
|
result = m.ast.findNode
|
|
|
|
proc execute(cmd: IdeCmd, file, dirtyfile: string, line, col: int;
|
|
graph: ModuleGraph; cache: IdentCache) =
|
|
myLog("cmd: " & $cmd & ", file: " & file & ", dirtyFile: " & dirtyfile &
|
|
"[" & $line & ":" & $col & "]")
|
|
gIdeCmd = cmd
|
|
if cmd == ideChk:
|
|
msgs.structuredErrorHook = errorHook
|
|
msgs.writelnHook = myLog
|
|
else:
|
|
msgs.structuredErrorHook = nil
|
|
msgs.writelnHook = myLog
|
|
if cmd == ideUse and suggestVersion != 0:
|
|
graph.resetAllModules()
|
|
var isKnownFile = true
|
|
let dirtyIdx = file.fileInfoIdx(isKnownFile)
|
|
|
|
if dirtyfile.len != 0: msgs.setDirtyFile(dirtyIdx, dirtyfile)
|
|
else: msgs.setDirtyFile(dirtyIdx, nil)
|
|
|
|
gTrackPos = newLineInfo(dirtyIdx, line, col)
|
|
gTrackPosAttached = false
|
|
gErrorCounter = 0
|
|
if suggestVersion == 1:
|
|
graph.usageSym = nil
|
|
if not isKnownFile:
|
|
graph.compileProject(cache)
|
|
if suggestVersion == 0 and gIdeCmd in {ideUse, ideDus} and
|
|
dirtyfile.len == 0:
|
|
discard "no need to recompile anything"
|
|
else:
|
|
let modIdx = graph.parentModule(dirtyIdx)
|
|
graph.markDirty dirtyIdx
|
|
graph.markClientsDirty dirtyIdx
|
|
if gIdeCmd != ideMod:
|
|
graph.compileProject(cache, modIdx)
|
|
if gIdeCmd in {ideUse, ideDus}:
|
|
let u = if suggestVersion != 1: graph.symFromInfo(gTrackPos) else: graph.usageSym
|
|
if u != nil:
|
|
listUsages(u)
|
|
else:
|
|
localError(gTrackPos, "found no symbol at this position " & $gTrackPos)
|
|
|
|
proc executeEpc(cmd: IdeCmd, args: SexpNode;
|
|
graph: ModuleGraph; cache: IdentCache) =
|
|
let
|
|
file = args[0].getStr
|
|
line = args[1].getNum
|
|
column = args[2].getNum
|
|
var dirtyfile = ""
|
|
if len(args) > 3:
|
|
dirtyfile = args[3].getStr(nil)
|
|
execute(cmd, file, dirtyfile, int(line), int(column), graph, cache)
|
|
|
|
proc returnEpc(socket: Socket, uid: BiggestInt, s: SexpNode|string,
|
|
return_symbol = "return") =
|
|
let response = $convertSexp([newSSymbol(return_symbol), uid, s])
|
|
socket.send(toHex(len(response), 6))
|
|
socket.send(response)
|
|
|
|
template checkSanity(client, sizeHex, size, messageBuffer: typed) =
|
|
if client.recv(sizeHex, 6) != 6:
|
|
raise newException(ValueError, "didn't get all the hexbytes")
|
|
if parseHex(sizeHex, size) == 0:
|
|
raise newException(ValueError, "invalid size hex: " & $sizeHex)
|
|
if client.recv(messageBuffer, size) != size:
|
|
raise newException(ValueError, "didn't get all the bytes")
|
|
|
|
proc toStdout() {.gcsafe.} =
|
|
while true:
|
|
let res = results.recv()
|
|
case res.section
|
|
of ideNone: break
|
|
of ideMsg: echo res.doc
|
|
of ideKnown: echo res.quality == 1
|
|
else: echo res
|
|
|
|
proc toSocket(stdoutSocket: Socket) {.gcsafe.} =
|
|
while true:
|
|
let res = results.recv()
|
|
case res.section
|
|
of ideNone: break
|
|
of ideMsg: stdoutSocket.send(res.doc & "\c\L")
|
|
of ideKnown: stdoutSocket.send($(res.quality == 1) & "\c\L")
|
|
else: stdoutSocket.send($res & "\c\L")
|
|
|
|
proc toEpc(client: Socket; uid: BiggestInt) {.gcsafe.} =
|
|
var list = newSList()
|
|
while true:
|
|
let res = results.recv()
|
|
case res.section
|
|
of ideNone: break
|
|
of ideMsg:
|
|
list.add sexp(res.doc)
|
|
of ideKnown:
|
|
list.add sexp(res.quality == 1)
|
|
else:
|
|
list.add sexp(res)
|
|
returnEpc(client, uid, list)
|
|
|
|
template setVerbosity(level: typed) =
|
|
gVerbosity = level
|
|
gNotes = NotesVerbosity[gVerbosity]
|
|
|
|
proc connectToNextFreePort(server: Socket, host: string): Port =
|
|
server.bindaddr(Port(0), host)
|
|
let (_, port) = server.getLocalAddr
|
|
result = port
|
|
|
|
type
|
|
ThreadParams = tuple[port: Port; address: string]
|
|
|
|
proc replStdinSingleCmd(line: string) =
|
|
requests.send line
|
|
toStdout()
|
|
echo ""
|
|
flushFile(stdout)
|
|
|
|
proc replStdin(x: ThreadParams) {.thread.} =
|
|
if gEmitEof:
|
|
echo DummyEof
|
|
while true:
|
|
let line = readLine(stdin)
|
|
requests.send line
|
|
if line == "quit": break
|
|
toStdout()
|
|
echo DummyEof
|
|
flushFile(stdout)
|
|
else:
|
|
echo Help
|
|
var line = ""
|
|
while readLineFromStdin("> ", line):
|
|
replStdinSingleCmd(line)
|
|
requests.send "quit"
|
|
|
|
proc replCmdline(x: ThreadParams) {.thread.} =
|
|
replStdinSingleCmd(x.address)
|
|
requests.send "quit"
|
|
|
|
proc replTcp(x: ThreadParams) {.thread.} =
|
|
var server = newSocket()
|
|
server.bindAddr(x.port, x.address)
|
|
var inp = "".TaintedString
|
|
server.listen()
|
|
while true:
|
|
var stdoutSocket = newSocket()
|
|
accept(server, stdoutSocket)
|
|
|
|
stdoutSocket.readLine(inp)
|
|
requests.send inp
|
|
toSocket(stdoutSocket)
|
|
stdoutSocket.send("\c\L")
|
|
stdoutSocket.close()
|
|
|
|
proc argsToStr(x: SexpNode): string =
|
|
if x.kind != SList: return x.getStr
|
|
doAssert x.kind == SList
|
|
doAssert x.len >= 4
|
|
let file = x[0].getStr
|
|
let line = x[1].getNum
|
|
let col = x[2].getNum
|
|
let dirty = x[3].getStr
|
|
result = x[0].getStr.escape
|
|
if dirty.len > 0:
|
|
result.add ';'
|
|
result.add dirty.escape
|
|
result.add ':'
|
|
result.add line
|
|
result.add ':'
|
|
result.add col
|
|
|
|
proc replEpc(x: ThreadParams) {.thread.} =
|
|
var server = newSocket()
|
|
let port = connectToNextFreePort(server, "localhost")
|
|
server.listen()
|
|
echo port
|
|
stdout.flushFile()
|
|
|
|
var client = newSocket()
|
|
# Wait for connection
|
|
accept(server, client)
|
|
while true:
|
|
var
|
|
sizeHex = ""
|
|
size = 0
|
|
messageBuffer = ""
|
|
checkSanity(client, sizeHex, size, messageBuffer)
|
|
let
|
|
message = parseSexp($messageBuffer)
|
|
epcApi = message[0].getSymbol
|
|
case epcApi
|
|
of "call":
|
|
let
|
|
uid = message[1].getNum
|
|
args = message[3]
|
|
|
|
gIdeCmd = parseIdeCmd(message[2].getSymbol)
|
|
case gIdeCmd
|
|
of ideSug, ideCon, ideDef, ideUse, ideDus, ideOutline, ideHighlight:
|
|
setVerbosity(0)
|
|
else: discard
|
|
let cmd = $gIdeCmd & " " & args.argsToStr
|
|
myLog "MSG CMD: " & cmd
|
|
requests.send(cmd)
|
|
toEpc(client, uid)
|
|
of "methods":
|
|
returnEpc(client, message[1].getNum, listEpc())
|
|
of "epc-error":
|
|
# an unhandled exception forces down the whole process anyway, so we
|
|
# use 'quit' here instead of 'raise'
|
|
quit("received epc error: " & $messageBuffer)
|
|
else:
|
|
let errMessage = case epcApi
|
|
of "return", "return-error":
|
|
"no return expected"
|
|
else:
|
|
"unexpected call: " & epcAPI
|
|
quit errMessage
|
|
|
|
proc execCmd(cmd: string; graph: ModuleGraph; cache: IdentCache; cachedMsgs: CachedMsgs) =
|
|
template sentinel() =
|
|
# send sentinel for the input reading thread:
|
|
results.send(Suggest(section: ideNone))
|
|
|
|
template toggle(sw) =
|
|
if sw in gGlobalOptions:
|
|
excl(gGlobalOptions, sw)
|
|
else:
|
|
incl(gGlobalOptions, sw)
|
|
sentinel()
|
|
return
|
|
|
|
template err() =
|
|
echo Help
|
|
sentinel()
|
|
return
|
|
|
|
var opc = ""
|
|
var i = parseIdent(cmd, opc, 0)
|
|
case opc.normalize
|
|
of "sug": gIdeCmd = ideSug
|
|
of "con": gIdeCmd = ideCon
|
|
of "def": gIdeCmd = ideDef
|
|
of "use": gIdeCmd = ideUse
|
|
of "dus": gIdeCmd = ideDus
|
|
of "mod": gIdeCmd = ideMod
|
|
of "chk": gIdeCmd = ideChk
|
|
of "highlight": gIdeCmd = ideHighlight
|
|
of "outline": gIdeCmd = ideOutline
|
|
of "quit":
|
|
sentinel()
|
|
quit()
|
|
of "debug": toggle optIdeDebug
|
|
of "terse": toggle optIdeTerse
|
|
of "known": gIdeCmd = ideKnown
|
|
else: err()
|
|
var dirtyfile = ""
|
|
var orig = ""
|
|
i = parseQuoted(cmd, orig, i)
|
|
if cmd[i] == ';':
|
|
i = parseQuoted(cmd, dirtyfile, i+1)
|
|
i += skipWhile(cmd, seps, i)
|
|
var line = -1
|
|
var col = 0
|
|
i += parseInt(cmd, line, i)
|
|
i += skipWhile(cmd, seps, i)
|
|
i += parseInt(cmd, col, i)
|
|
|
|
if gIdeCmd == ideKnown:
|
|
results.send(Suggest(section: ideKnown, quality: ord(fileInfoKnown(orig))))
|
|
else:
|
|
if gIdeCmd == ideChk:
|
|
for cm in cachedMsgs: errorHook(cm.info, cm.msg, cm.sev)
|
|
execute(gIdeCmd, orig, dirtyfile, line, col, graph, cache)
|
|
sentinel()
|
|
|
|
proc recompileFullProject(graph: ModuleGraph; cache: IdentCache) =
|
|
#echo "recompiling full project"
|
|
resetSystemArtifacts()
|
|
vm.globalCtx = nil
|
|
graph.resetAllModules()
|
|
GC_fullcollect()
|
|
compileProject(graph, cache)
|
|
#echo GC_getStatistics()
|
|
|
|
proc mainThread(graph: ModuleGraph; cache: IdentCache) =
|
|
if gLogging:
|
|
for it in searchPaths:
|
|
log(it)
|
|
|
|
proc wrHook(line: string) {.closure.} =
|
|
if gMode == mepc:
|
|
if gLogging: log(line)
|
|
else:
|
|
writelnToChannel(line)
|
|
|
|
msgs.writelnHook = wrHook
|
|
suggestionResultHook = sugResultHook
|
|
graph.doStopCompile = proc (): bool = requests.peek() > 0
|
|
var idle = 0
|
|
var cachedMsgs: CachedMsgs = @[]
|
|
while true:
|
|
let (hasData, req) = requests.tryRecv()
|
|
if hasData:
|
|
msgs.writelnHook = wrHook
|
|
suggestionResultHook = sugResultHook
|
|
execCmd(req, graph, cache, cachedMsgs)
|
|
idle = 0
|
|
else:
|
|
os.sleep 250
|
|
idle += 1
|
|
if idle == 20 and gRefresh:
|
|
# we use some nimsuggest activity to enable a lazy recompile:
|
|
gIdeCmd = ideChk
|
|
msgs.writelnHook = proc (s: string) = discard
|
|
cachedMsgs.setLen 0
|
|
msgs.structuredErrorHook = proc (info: TLineInfo; msg: string; sev: Severity) =
|
|
cachedMsgs.add(CachedMsg(info: info, msg: msg, sev: sev))
|
|
suggestionResultHook = proc (s: Suggest) = discard
|
|
recompileFullProject(graph, cache)
|
|
|
|
var
|
|
inputThread: Thread[ThreadParams]
|
|
|
|
proc mainCommand(graph: ModuleGraph; cache: IdentCache) =
|
|
clearPasses()
|
|
registerPass verbosePass
|
|
registerPass semPass
|
|
gCmd = cmdIdeTools
|
|
incl gGlobalOptions, optCaasEnabled
|
|
isServing = true
|
|
wantMainModule()
|
|
|
|
if not fileExists(gProjectFull):
|
|
quit "cannot find file: " & gProjectFull
|
|
|
|
add(searchPaths, options.libpath)
|
|
|
|
# do not stop after the first error:
|
|
msgs.gErrorMax = high(int)
|
|
# do not print errors, but log them
|
|
msgs.writelnHook = proc (s: string) = log(s)
|
|
msgs.structuredErrorHook = nil
|
|
|
|
# compile the project before showing any input so that we already
|
|
# can answer questions right away:
|
|
compileProject(graph, cache)
|
|
|
|
open(requests)
|
|
open(results)
|
|
|
|
case gMode
|
|
of mstdin: createThread(inputThread, replStdin, (gPort, gAddress))
|
|
of mtcp: createThread(inputThread, replTcp, (gPort, gAddress))
|
|
of mepc: createThread(inputThread, replEpc, (gPort, gAddress))
|
|
of mcmdsug: createThread(inputThread, replCmdline,
|
|
(gPort, "sug \"" & options.gProjectFull & "\":" & gAddress))
|
|
of mcmdcon: createThread(inputThread, replCmdline,
|
|
(gPort, "con \"" & options.gProjectFull & "\":" & gAddress))
|
|
mainThread(graph, cache)
|
|
joinThread(inputThread)
|
|
close(requests)
|
|
close(results)
|
|
|
|
proc processCmdLine*(pass: TCmdLinePass, cmd: string) =
|
|
var p = parseopt.initOptParser(cmd)
|
|
while true:
|
|
parseopt.next(p)
|
|
case p.kind
|
|
of cmdEnd: break
|
|
of cmdLongoption, cmdShortOption:
|
|
case p.key.normalize
|
|
of "port":
|
|
gPort = parseInt(p.val).Port
|
|
gMode = mtcp
|
|
of "address":
|
|
gAddress = p.val
|
|
gMode = mtcp
|
|
of "stdin": gMode = mstdin
|
|
of "cmdsug":
|
|
gMode = mcmdsug
|
|
gAddress = p.val
|
|
incl(gGlobalOptions, optIdeDebug)
|
|
of "cmdcon":
|
|
gMode = mcmdcon
|
|
gAddress = p.val
|
|
incl(gGlobalOptions, optIdeDebug)
|
|
of "epc":
|
|
gMode = mepc
|
|
gVerbosity = 0 # Port number gotta be first.
|
|
of "debug": incl(gGlobalOptions, optIdeDebug)
|
|
of "v2": suggestVersion = 0
|
|
of "v1": suggestVersion = 1
|
|
of "tester":
|
|
gMode = mstdin
|
|
gEmitEof = true
|
|
gRefresh = false
|
|
of "log": gLogging = true
|
|
of "refresh":
|
|
if p.val.len > 0:
|
|
gRefresh = parseBool(p.val)
|
|
else:
|
|
gRefresh = true
|
|
of "maxresults":
|
|
suggestMaxResults = parseInt(p.val)
|
|
else: processSwitch(pass, p)
|
|
of cmdArgument:
|
|
let a = unixToNativePath(p.key)
|
|
if dirExists(a) and not fileExists(a.addFileExt("nim")):
|
|
options.gProjectName = findProjectNimFile(a)
|
|
# don't make it worse, report the error the old way:
|
|
if options.gProjectName.len == 0: options.gProjectName = a
|
|
else:
|
|
options.gProjectName = a
|
|
# if processArgument(pass, p, argsCount): break
|
|
|
|
proc handleCmdLine(cache: IdentCache; config: ConfigRef) =
|
|
if paramCount() == 0:
|
|
stdout.writeline(Usage)
|
|
else:
|
|
processCmdLine(passCmd1, "")
|
|
if gMode != mstdin:
|
|
msgs.writelnHook = proc (msg: string) = discard
|
|
if gProjectName != "":
|
|
try:
|
|
gProjectFull = canonicalizePath(gProjectName)
|
|
except OSError:
|
|
gProjectFull = gProjectName
|
|
var p = splitFile(gProjectFull)
|
|
gProjectPath = canonicalizePath p.dir
|
|
gProjectName = p.name
|
|
else:
|
|
gProjectPath = canonicalizePath getCurrentDir()
|
|
|
|
# Find Nim's prefix dir.
|
|
let binaryPath = findExe("nim")
|
|
if binaryPath == "":
|
|
raise newException(IOError,
|
|
"Cannot find Nim standard library: Nim compiler not in PATH")
|
|
gPrefixDir = binaryPath.splitPath().head.parentDir()
|
|
if not dirExists(gPrefixDir / "lib"): gPrefixDir = ""
|
|
|
|
#msgs.writelnHook = proc (line: string) = log(line)
|
|
myLog("START " & gProjectFull)
|
|
|
|
loadConfigs(DefaultConfig, cache, config) # load all config files
|
|
# now process command line arguments again, because some options in the
|
|
# command line can overwite the config file's settings
|
|
options.command = "nimsuggest"
|
|
let scriptFile = gProjectFull.changeFileExt("nims")
|
|
if fileExists(scriptFile):
|
|
runNimScript(cache, scriptFile, freshDefines=false, config)
|
|
# 'nim foo.nims' means to just run the NimScript file and do nothing more:
|
|
if scriptFile == gProjectFull: return
|
|
elif fileExists(gProjectPath / "config.nims"):
|
|
# directory wide NimScript file
|
|
runNimScript(cache, gProjectPath / "config.nims", freshDefines=false, config)
|
|
|
|
extccomp.initVars()
|
|
processCmdLine(passCmd2, "")
|
|
|
|
let graph = newModuleGraph(config)
|
|
graph.suggestMode = true
|
|
mainCommand(graph, cache)
|
|
|
|
condsyms.initDefines()
|
|
defineSymbol "nimsuggest"
|
|
handleCmdline(newIdentCache(), newConfigRef())
|