parent
							
								
									115a0bc47f
								
							
						
					
					
						commit
						cf4f218675
					
				
				 5 changed files with 191 additions and 3 deletions
			
			
		| @ -0,0 +1,44 @@ | |||||||
|  | ## Generate compile_commands.json for premake projects | ||||||
|  | 
 | ||||||
|  | This module implements [JSON Compilation Database Format | ||||||
|  | Specification](http://clang.llvm.org/docs/JSONCompilationDatabase.html) for | ||||||
|  | premake projects. | ||||||
|  | 
 | ||||||
|  | Install this module somewhere premake can find it, for example: | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | git clone https://github.com/tarruda/premake-export-compile-commands export-compile-commands | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | Then put this at the top of your system script(eg: ~/.premake/premake-system.lua): | ||||||
|  | 
 | ||||||
|  | ```lua | ||||||
|  | require "export-compile-commands" | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | Note that while possible, it is not recommended to put the `require` line in | ||||||
|  | project-specific premake configuration because the "export-compile-commands" | ||||||
|  | module will need to be installed everywhere your project is built. | ||||||
|  | 
 | ||||||
|  | After the above steps, the "export-compile-commands" action will be available | ||||||
|  | for your projects: | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | premake5 export-compile-commands | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | The `export-compile-commands` action will generate one json file per | ||||||
|  | config/platform combination in each workspace, all under the `compile_commands` | ||||||
|  | subdirectory. For example, say you have defined `debug` and `release` | ||||||
|  | configurations with `x32` and `x64` platforms, the output will be something | ||||||
|  | like: | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | Generated WORKSPACE_BUILD_DIR/compile_commands/debug_x32.json... | ||||||
|  | Generated WORKSPACE_BUILD_DIR/compile_commands/debug_x64.json... | ||||||
|  | Generated WORKSPACE_BUILD_DIR/compile_commands/release_x32.json... | ||||||
|  | Generated WORKSPACE_BUILD_DIR/compile_commands/release_x64.json... | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | where each file contain the compilation commands for the corresponding | ||||||
|  | config/platform combo. | ||||||
| @ -0,0 +1,4 @@ | |||||||
|  | return { | ||||||
|  |   '_preload.lua', | ||||||
|  |   'export-compile-commands.lua', | ||||||
|  | } | ||||||
| @ -0,0 +1,3 @@ | |||||||
|  | return function(cfg) | ||||||
|  |   return _ACTION == 'export-compile-commands' | ||||||
|  | end | ||||||
| @ -0,0 +1,133 @@ | |||||||
|  | local p = premake | ||||||
|  | 
 | ||||||
|  | p.modules.export_compile_commands = {} | ||||||
|  | local m = p.modules.export_compile_commands | ||||||
|  | 
 | ||||||
|  | local workspace = p.workspace | ||||||
|  | local project = p.project | ||||||
|  | 
 | ||||||
|  | function m.getToolset(cfg) | ||||||
|  |   return p.tools[cfg.toolset or 'gcc'] | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | function m.getIncludeDirs(cfg) | ||||||
|  |   local flags = {} | ||||||
|  |   for _, dir in ipairs(cfg.includedirs) do | ||||||
|  |     table.insert(flags, '-I' .. p.quoted(dir)) | ||||||
|  |   end | ||||||
|  |   for _, dir in ipairs(cfg.sysincludedir or {}) do | ||||||
|  |     table.insert(result, '-isystem ' .. p.quoted(dir)) | ||||||
|  |   end | ||||||
|  |   return flags | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | function m.getCommonFlags(cfg) | ||||||
|  |   local toolset = m.getToolset(cfg) | ||||||
|  |   local flags = toolset.getcppflags(cfg) | ||||||
|  |   flags = table.join(flags, toolset.getdefines(cfg.defines)) | ||||||
|  |   flags = table.join(flags, toolset.getundefines(cfg.undefines)) | ||||||
|  |   -- can't use toolset.getincludedirs because some tools that consume | ||||||
|  |   -- compile_commands.json have problems with relative include paths | ||||||
|  |   flags = table.join(flags, m.getIncludeDirs(cfg)) | ||||||
|  |   flags = table.join(flags, toolset.getcflags(cfg)) | ||||||
|  |   return table.join(flags, cfg.buildoptions) | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | function m.getObjectPath(prj, cfg, node) | ||||||
|  |   return path.join(cfg.objdir, path.appendExtension(node.objname, '.o')) | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | function m.getDependenciesPath(prj, cfg, node) | ||||||
|  |   return path.join(cfg.objdir, path.appendExtension(node.objname, '.d')) | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | function m.getFileFlags(prj, cfg, node) | ||||||
|  |   return table.join(m.getCommonFlags(cfg), { | ||||||
|  |     '-o', m.getObjectPath(prj, cfg, node), | ||||||
|  |     '-MF', m.getDependenciesPath(prj, cfg, node), | ||||||
|  |     '-c', node.abspath | ||||||
|  |   }) | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | function m.generateCompileCommand(prj, cfg, node) | ||||||
|  |   return { | ||||||
|  |     directory = prj.location, | ||||||
|  |     file = node.abspath, | ||||||
|  |     command = 'cc '.. table.concat(m.getFileFlags(prj, cfg, node), ' ') | ||||||
|  |   } | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | function m.includeFile(prj, node, depth) | ||||||
|  |   return path.iscppfile(node.abspath) | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | function m.getConfig(prj) | ||||||
|  |   if _OPTIONS['export-compile-commands-config'] then | ||||||
|  |     return project.getconfig(prj, _OPTIONS['export-compile-commands-config'], | ||||||
|  |       _OPTIONS['export-compile-commands-platform']) | ||||||
|  |   end | ||||||
|  |   for cfg in project.eachconfig(prj) do | ||||||
|  |     -- just use the first configuration which is usually "Debug" | ||||||
|  |     return cfg | ||||||
|  |   end | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | function m.getProjectCommands(prj, cfg) | ||||||
|  |   local tr = project.getsourcetree(prj) | ||||||
|  |   local cmds = {} | ||||||
|  |   p.tree.traverse(tr, { | ||||||
|  |     onleaf = function(node, depth) | ||||||
|  |       if not m.includeFile(prj, node, depth) then | ||||||
|  |         return | ||||||
|  |       end | ||||||
|  |       table.insert(cmds, m.generateCompileCommand(prj, cfg, node)) | ||||||
|  |     end | ||||||
|  |   }) | ||||||
|  |   return cmds | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | local function execute() | ||||||
|  |   for wks in p.global.eachWorkspace() do | ||||||
|  |     local cfgCmds = {} | ||||||
|  |     for prj in workspace.eachproject(wks) do | ||||||
|  |       for cfg in project.eachconfig(prj) do | ||||||
|  |         local cfgKey = string.format('%s', cfg.shortname) | ||||||
|  |         if not cfgCmds[cfgKey] then | ||||||
|  |           cfgCmds[cfgKey] = {} | ||||||
|  |         end | ||||||
|  |         cfgCmds[cfgKey] = table.join(cfgCmds[cfgKey], m.getProjectCommands(prj, cfg)) | ||||||
|  |       end | ||||||
|  |     end | ||||||
|  |     for cfgKey,cmds in pairs(cfgCmds) do | ||||||
|  |       local outfile = string.format('compile_commands/%s.json', cfgKey) | ||||||
|  |       p.generate(wks, outfile, function(wks) | ||||||
|  |         p.w('[') | ||||||
|  |         for i = 1, #cmds do | ||||||
|  |           local item = cmds[i] | ||||||
|  |           local command = string.format([[ | ||||||
|  |           { | ||||||
|  |             "directory": "%s", | ||||||
|  |             "file": "%s", | ||||||
|  |             "command": "%s" | ||||||
|  |           }]], | ||||||
|  |           item.directory, | ||||||
|  |           item.file, | ||||||
|  |           item.command:gsub('\\', '\\\\'):gsub('"', '\\"')) | ||||||
|  |           if i > 1 then | ||||||
|  |             p.w(',') | ||||||
|  |           end | ||||||
|  |           p.w(command) | ||||||
|  |         end | ||||||
|  |         p.w(']') | ||||||
|  |       end) | ||||||
|  |     end | ||||||
|  |   end | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | newaction { | ||||||
|  |   trigger = 'export-compile-commands', | ||||||
|  |   description = 'Export compiler commands in JSON Compilation Database Format', | ||||||
|  |   execute = execute | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | return m | ||||||
					Loading…
					
					
				
		Reference in New Issue