138 lines
3.8 KiB
Lua
138 lines
3.8 KiB
Lua
local M = {}
|
|
M.clang_filetypes = { 'c', 'cpp', 'objc', 'objcpp', 'cuda' }
|
|
M.auto_watch_enabled = true
|
|
|
|
local lspconfig = require 'lspconfig'
|
|
|
|
local function find_compile_commands()
|
|
local results = vim.fn.systemlist { 'fd', '-u', '-t', 'f', 'compile_commands.json' }
|
|
table.sort(results, function(a, b)
|
|
return a:match 'debug' and not b:match 'debug'
|
|
end)
|
|
return vim.fn.fnamemodify(results[1] or '', ':h')
|
|
end
|
|
|
|
function M.stop_clangd()
|
|
for _, client in ipairs(vim.lsp.get_clients()) do
|
|
if client.name == 'clangd' then
|
|
pcall(function()
|
|
client.stop { force = true }
|
|
end)
|
|
vim.notify '[clangd] Stopped clangd'
|
|
end
|
|
end
|
|
end
|
|
|
|
function M.start_clangd(dir)
|
|
M.stop_clangd()
|
|
|
|
local cmd = {
|
|
'clangd',
|
|
'--background-index',
|
|
'--clang-tidy',
|
|
'--header-insertion=never',
|
|
'--query-driver=' .. vim.fn.exepath 'clang++',
|
|
'--resource-dir=' .. vim.fn.systemlist({ 'clang++', '--print-resource-dir' })[1],
|
|
}
|
|
if dir and dir ~= '' then
|
|
vim.notify('[clangd] Setting up with: ' .. dir)
|
|
table.insert(cmd, '--compile-commands-dir=' .. dir)
|
|
else
|
|
vim.notify '[clangd] No compile_commands.json found.\nUse <leader>cc to manually set location'
|
|
table.insert(cmd, '--compile-commands-dir=.')
|
|
end
|
|
|
|
lspconfig.clangd.setup {
|
|
cmd = cmd,
|
|
filetypes = M.clang_filetypes,
|
|
root_dir = lspconfig.util.root_pattern '.git',
|
|
capabilities = require('blink.cmp').get_lsp_capabilities(),
|
|
single_file_support = true,
|
|
}
|
|
end
|
|
|
|
local watcher, debounce_timer
|
|
|
|
function M.watch_compile_commands(dir)
|
|
local uv = vim.uv or vim.loop
|
|
|
|
if watcher then
|
|
watcher:stop()
|
|
watcher:close()
|
|
watcher = nil
|
|
end
|
|
|
|
if debounce_timer then
|
|
debounce_timer:stop()
|
|
debounce_timer:close()
|
|
debounce_timer = nil
|
|
end
|
|
|
|
local watch_path = dir or vim.fn.getcwd()
|
|
local watch_file = watch_path .. '/compile_commands.json'
|
|
|
|
-- if not M.auto_watch_enabled or not vim.fn.filereadable(watch_file) then
|
|
-- return
|
|
-- end
|
|
|
|
watcher = uv.new_fs_event()
|
|
watcher:start(
|
|
watch_path,
|
|
{ recursive = true },
|
|
vim.schedule_wrap(function(err, fname, status)
|
|
if err then
|
|
vim.notify('[clangd] Watcher error: ' .. err, vim.log.levels.ERROR)
|
|
return
|
|
end
|
|
|
|
if fname and fname:match '.*/compile_commands%.json$' and status.change then
|
|
debounce_timer = uv.new_timer()
|
|
debounce_timer:start(200, 0, function()
|
|
vim.schedule(function()
|
|
vim.notify '[clangd] Detected compile_commands.json change. Reloading ...'
|
|
watch_path = vim.fn.fnamemodify(fname, ':h')
|
|
M.start_clangd(watch_path)
|
|
M.watch_compile_commands(watch_path)
|
|
end)
|
|
end)
|
|
end
|
|
end)
|
|
)
|
|
end
|
|
|
|
function M.pick_commands_dir()
|
|
local pickers = require 'telescope.pickers'
|
|
local finders = require 'telescope.finders'
|
|
local conf = require('telescope.config').values
|
|
pickers
|
|
.new({}, {
|
|
prompt_title = 'Pick compile_commands.json dir',
|
|
finder = finders.new_oneshot_job { 'fd', '-u', 'compile_commands.json', '-x', 'dirname', '{}' },
|
|
sorter = conf.generic_sorter {},
|
|
attach_mappings = function(_, map)
|
|
map('i', '<CR>', function(prompt_bufnr)
|
|
local entry = require('telescope.actions.state').get_selected_entry()
|
|
require('telescope.actions').close(prompt_bufnr)
|
|
vim.defer_fn(function()
|
|
M.start_clangd(entry[1])
|
|
end, 100)
|
|
end)
|
|
return true
|
|
end,
|
|
})
|
|
:find()
|
|
end
|
|
|
|
return {
|
|
'neovim/nvim-lspconfig',
|
|
ft = M.clang_filetypes,
|
|
config = function()
|
|
vim.notify('[clangd] find_compile_commands() returned: "' .. dir .. '"')
|
|
local dir = find_compile_commands()
|
|
M.start_clangd(dir)
|
|
M.watch_compile_commands(dir)
|
|
|
|
vim.keymap.set('n', '<leader>cc', M.pick_commands_dir, { desc = 'Pick location of compile_commands.json for clangd' })
|
|
end,
|
|
}
|