Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

compile_commands.json support #24798

Draft
wants to merge 2 commits into
base: devel
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
12 changes: 12 additions & 0 deletions compiler/commands.nim
Original file line number Diff line number Diff line change
Expand Up @@ -1154,6 +1154,18 @@ proc processSwitch*(switch, arg: string, pass: TCmdLinePass, info: TLineInfo;
handleStdinInput(conf)
of "nilseqs", "nilchecks", "symbol", "taintmode", "cs", "deadcodeelim": warningOptionNoop(switch)
of "nimmainprefix": conf.nimMainPrefix = arg
of "compilecmdsjson":
# TODO
# processOnOffSwitch(conf, {optCursorInference}, arg, pass, info)
conf.compileCommandsJson = true
of "projectcompilecmdjson":
# TODO
# processOnOffSwitch(conf, {optCursorInference}, arg, pass, info)
conf.projectCompileCommandJsons = true
of "rootcompilecmdsjson":
# TODO
# processOnOffSwitch(conf, {optCursorInference}, arg, pass, info)
conf.copyCompileCommandsJsonToCurDir = true
else:
if strutils.find(switch, '.') >= 0: options.setConfigVar(conf, switch, arg)
else: invalidCmdLineOption(conf, pass, switch, info)
Expand Down
29 changes: 29 additions & 0 deletions compiler/extccomp.nim
Original file line number Diff line number Diff line change
Expand Up @@ -1003,6 +1003,18 @@ proc callCCompiler*(conf: ConfigRef) =

template hashNimExe(): string = $secureHashFile(os.getAppFilename())

iterator compileCommandsJsonFiles*(conf: ConfigRef): AbsoluteFile =
var rf = if conf.projectCompileCommandJsons:
("compile_commands_" & conf.outFile.changeFileExt("json").string).RelativeFile
else:
"compile_commands".RelativeFile

yield getNimcacheDir(conf) / rf

if conf.copyCompileCommandsJsonToCurDir:
yield conf.projectPath / rf


proc jsonBuildInstructionsFile*(conf: ConfigRef): AbsoluteFile =
# `outFile` is better than `projectName`, as it allows having different json
# files for a given source file compiled with different options; it also
Expand All @@ -1027,6 +1039,16 @@ type BuildCache = object
depfiles: seq[(string, string)]
nimexe: string

proc jsonCompileCommands*(conf: ConfigRef): JsonNode =
# ref: https://clang.llvm.org/docs/JSONCompilationDatabase.html
result = %*[]
for it in conf.toCompile:
result.add %*{
"file": it.cname.string,
"command": getCompileCFileCmd(conf, it),
"directory": conf.outDir.string,
}

proc writeJsonBuildInstructions*(conf: ConfigRef; deps: StringTableRef) =
var linkFiles = collect(for it in conf.externalToLink:
var it = it
Expand Down Expand Up @@ -1060,8 +1082,15 @@ proc writeJsonBuildInstructions*(conf: ConfigRef; deps: StringTableRef) =
if fileExists(bcache.outputFile):
bcache.outputLastModificationTime = $getLastModificationTime(bcache.outputFile)
conf.jsonBuildFile = conf.jsonBuildInstructionsFile

conf.jsonBuildFile.string.writeFile(bcache.toJson.pretty)

if conf.compileCommandsJson:
# NOTE: compile_commands.json uses indent of 2 by convention
let compileCmds = conf.jsonCompileCommands.pretty(2)
for p in conf.compileCommandsJsonFiles:
p.writeFile(compileCmds)

proc changeDetectedViaJsonBuildInstructions*(conf: ConfigRef; jsonFile: AbsoluteFile): bool =
result = false
if not fileExists(jsonFile) or not fileExists(conf.absOutFile): return true
Expand Down
4 changes: 4 additions & 0 deletions compiler/options.nim
Original file line number Diff line number Diff line change
Expand Up @@ -448,6 +448,10 @@ type
currentConfigDir*: string # used for passPP only; absolute dir
clientProcessId*: int

compileCommandsJson*: bool
projectCompileCommandJsons*: bool
copyCompileCommandsJsonToCurDir*: bool



proc assignIfDefault*[T](result: var T, val: T, def = default(T)) =
Expand Down
7 changes: 7 additions & 0 deletions tests/compile_commands_json/foo.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
#include <iostream>

#ifdef ENABLE_FOO
extern "C" void foo() {
std::cout << "hello world\n";
}
#endif
6 changes: 6 additions & 0 deletions tests/compile_commands_json/foo.nim
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
{.compile("foo.cpp", "-DENABLE_FOO").}

proc foo() {.importc.}

when isMainModule:
foo()
2 changes: 2 additions & 0 deletions tests/compile_commands_json/foo.nims
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
--nimcache:"./nimcache"
--compileCmdsJson:on
Loading