2017-11-03 08:34:41 +01:00
|
|
|
declare-option -docstring "options to pass to the `clang` shell command" \
|
2017-05-16 13:35:43 +02:00
|
|
|
str clang_options
|
2014-04-03 20:00:42 +02:00
|
|
|
|
2017-11-03 08:34:41 +01:00
|
|
|
declare-option -hidden str clang_tmp_dir
|
|
|
|
declare-option -hidden completions clang_completions
|
|
|
|
declare-option -hidden line-specs clang_flags
|
|
|
|
declare-option -hidden line-specs clang_errors
|
2013-03-14 14:12:14 +01:00
|
|
|
|
2017-11-03 08:34:41 +01:00
|
|
|
define-command -params ..1 \
|
2016-10-11 09:03:41 +02:00
|
|
|
-docstring %{Parse the contents of the current buffer
|
|
|
|
The syntaxic errors detected during parsing are shown when auto-diagnostics are enabled} \
|
|
|
|
clang-parse %{
|
2018-05-06 23:29:52 +02:00
|
|
|
evaluate-commands %sh{
|
2017-06-09 13:05:31 +02:00
|
|
|
dir=$(mktemp -d "${TMPDIR:-/tmp}"/kak-clang.XXXXXXXX)
|
2014-10-24 19:46:42 +02:00
|
|
|
mkfifo ${dir}/fifo
|
2017-11-03 08:34:41 +01:00
|
|
|
printf %s\\n "set-option buffer clang_tmp_dir ${dir}"
|
2018-07-04 08:55:05 +02:00
|
|
|
printf %s\\n "evaluate-commands -no-hooks write -sync ${dir}/buf"
|
2013-03-14 14:12:14 +01:00
|
|
|
}
|
|
|
|
# end the previous %sh{} so that its output gets interpreted by kakoune
|
|
|
|
# before launching the following as a background task.
|
2018-05-06 23:29:52 +02:00
|
|
|
evaluate-commands %sh{
|
2014-10-24 19:46:42 +02:00
|
|
|
dir=${kak_opt_clang_tmp_dir}
|
2017-11-03 09:09:45 +01:00
|
|
|
printf %s\\n "evaluate-commands -draft %{
|
2017-03-08 20:33:25 +01:00
|
|
|
edit! -fifo ${dir}/fifo -debug *clang-output*
|
2017-11-03 08:34:41 +01:00
|
|
|
set-option buffer filetype make
|
|
|
|
set-option buffer make_current_error_line 0
|
2018-08-19 00:04:31 +02:00
|
|
|
hook -once -always buffer BufCloseFifo .* %{ nop %sh{ rm -r ${dir} } }
|
2014-10-24 19:46:42 +02:00
|
|
|
}"
|
2014-11-06 20:17:13 +01:00
|
|
|
# this runs in a detached shell, asynchronously, so that kakoune does
|
|
|
|
# not hang while clang is running. As completions references a cursor
|
|
|
|
# position and a buffer timestamp, only valid completions should be
|
2013-03-14 14:12:14 +01:00
|
|
|
# displayed.
|
|
|
|
(
|
2014-11-21 14:27:43 +01:00
|
|
|
case ${kak_opt_filetype} in
|
2015-09-22 00:37:49 +02:00
|
|
|
c) ft=c ;;
|
2014-11-21 14:27:43 +01:00
|
|
|
cpp) ft=c++ ;;
|
|
|
|
obj-c) ft=objective-c ;;
|
|
|
|
*) ft=c++ ;;
|
|
|
|
esac
|
|
|
|
|
2016-03-14 21:59:23 +01:00
|
|
|
if [ "$1" = "-complete" ]; then
|
2015-02-03 01:42:40 +01:00
|
|
|
pos=-:${kak_cursor_line}:${kak_cursor_column}
|
|
|
|
header="${kak_cursor_line}.${kak_cursor_column}@${kak_timestamp}"
|
|
|
|
compl=$(clang++ -x ${ft} -fsyntax-only ${kak_opt_clang_options} \
|
|
|
|
-Xclang -code-completion-brief-comments -Xclang -code-completion-at=${pos} - < ${dir}/buf 2> ${dir}/stderr |
|
2016-06-21 20:00:41 +02:00
|
|
|
awk -F ': ' '
|
2015-02-03 01:42:40 +01:00
|
|
|
/^COMPLETION:/ && ! /\(Hidden\)/ {
|
2016-04-04 14:37:28 +02:00
|
|
|
id=$2
|
|
|
|
gsub(/ +$/, "", id)
|
2018-05-30 15:28:50 +02:00
|
|
|
gsub(/~/, "~~", id)
|
2016-04-04 14:37:28 +02:00
|
|
|
gsub(/\|/, "\\|", id)
|
|
|
|
|
2015-02-03 01:42:40 +01:00
|
|
|
gsub(/[[{<]#|#[}>]/, "", $3)
|
|
|
|
gsub(/#]/, " ", $3)
|
|
|
|
gsub(/:: /, "::", $3)
|
|
|
|
gsub(/ +$/, "", $3)
|
|
|
|
desc=$4 ? $3 "\\n" $4 : $3
|
2016-04-04 14:37:28 +02:00
|
|
|
|
2018-05-30 15:28:50 +02:00
|
|
|
gsub(/~/, "~~", desc)
|
2016-04-04 14:37:28 +02:00
|
|
|
gsub(/\|/, "\\|", desc)
|
2015-10-06 23:50:51 +02:00
|
|
|
if (id in docstrings)
|
2018-05-30 15:28:50 +02:00
|
|
|
docstrings[id]=docstrings[id] "\n" desc
|
2015-02-03 01:42:40 +01:00
|
|
|
else
|
2015-10-06 23:50:51 +02:00
|
|
|
docstrings[id]=desc
|
2015-02-03 01:42:40 +01:00
|
|
|
}
|
|
|
|
END {
|
2016-08-31 00:23:33 +02:00
|
|
|
for (id in docstrings) {
|
2015-10-06 14:39:09 +02:00
|
|
|
menu=id
|
2018-02-07 09:40:45 +01:00
|
|
|
gsub(/(^|[^[:alnum:]_])(operator|new|delete)($|[^{}_[:alnum:]]+)/, "{keyword}&{}", menu)
|
2015-10-07 00:22:36 +02:00
|
|
|
gsub(/(^|[[:space:]])(int|size_t|bool|char|unsigned|signed|long)($|[[:space:]])/, "{type}&{}", menu)
|
2018-02-07 09:40:45 +01:00
|
|
|
gsub(/[^{}_[:alnum:]]+/, "{operator}&{}", menu)
|
2018-05-30 15:28:50 +02:00
|
|
|
printf "%%~%s|%s|%s~ ", id, docstrings[id], menu
|
2015-10-06 14:39:09 +02:00
|
|
|
}
|
2018-05-30 15:28:50 +02:00
|
|
|
}')
|
2017-11-03 09:09:45 +01:00
|
|
|
printf %s\\n "evaluate-commands -client ${kak_client} echo 'clang completion done'
|
2018-05-30 15:28:50 +02:00
|
|
|
set-option 'buffer=${kak_buffile}' clang_completions ${header} ${compl}" | kak -p ${kak_session}
|
2014-11-27 14:42:36 +01:00
|
|
|
else
|
2015-02-03 01:42:40 +01:00
|
|
|
clang++ -x ${ft} -fsyntax-only ${kak_opt_clang_options} - < ${dir}/buf 2> ${dir}/stderr
|
2017-11-03 09:09:45 +01:00
|
|
|
printf %s\\n "evaluate-commands -client ${kak_client} echo 'clang parsing done'" | kak -p ${kak_session}
|
2014-11-27 14:42:36 +01:00
|
|
|
fi
|
2014-10-31 21:53:36 +01:00
|
|
|
|
2014-11-27 14:42:36 +01:00
|
|
|
flags=$(cat ${dir}/stderr | sed -rne "
|
2018-05-30 15:28:50 +02:00
|
|
|
/^<stdin>:[0-9]+:([0-9]+:)? (fatal )?error/ { s/^<stdin>:([0-9]+):.*/'\1|{red}█'/; p }
|
|
|
|
/^<stdin>:[0-9]+:([0-9]+:)? warning/ { s/^<stdin>:([0-9]+):.*/'\1|{yellow}█'/; p }
|
|
|
|
" | paste -s -d ' ' -)
|
2014-11-19 14:53:31 +01:00
|
|
|
|
2014-11-27 14:42:36 +01:00
|
|
|
errors=$(cat ${dir}/stderr | sed -rne "
|
2017-05-25 13:17:22 +02:00
|
|
|
/^<stdin>:[0-9]+:([0-9]+:)? ((fatal )?error|warning)/ {
|
2018-05-30 15:28:50 +02:00
|
|
|
s/'/''/g; s/^<stdin>:([0-9]+):([0-9]+:)? (.*)/'\1|\3'/; p
|
|
|
|
}" | sort -n | paste -s -d ' ' -)
|
2014-11-19 14:53:31 +01:00
|
|
|
|
2014-11-27 14:42:36 +01:00
|
|
|
sed -e "s|<stdin>|${kak_bufname}|g" < ${dir}/stderr > ${dir}/fifo
|
2014-11-19 14:53:31 +01:00
|
|
|
|
2018-05-30 15:28:50 +02:00
|
|
|
printf %s\\n "set-option 'buffer=${kak_buffile}' clang_flags ${kak_timestamp} ${flags}
|
|
|
|
set-option 'buffer=${kak_buffile}' clang_errors ${kak_timestamp} ${errors}" | kak -p ${kak_session}
|
2014-03-06 04:35:38 +01:00
|
|
|
) > /dev/null 2>&1 < /dev/null &
|
2013-03-14 14:12:14 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-03 08:34:41 +01:00
|
|
|
define-command clang-complete -docstring "Complete the current selection" %{ clang-parse -complete }
|
2014-11-27 14:42:36 +01:00
|
|
|
|
2017-11-03 08:34:41 +01:00
|
|
|
define-command -hidden clang-show-completion-info %[ try %[
|
2017-11-03 09:09:45 +01:00
|
|
|
evaluate-commands -draft %[
|
|
|
|
execute-keys <space>{( <a-k> ^\( <ret> b <a-k> \A\w+\z <ret>
|
2018-05-06 23:29:52 +02:00
|
|
|
evaluate-commands %sh[
|
2016-08-31 00:23:33 +02:00
|
|
|
desc=$(printf %s\\n "${kak_opt_clang_completions}" | sed -e "{ s/\([^\\]\):/\1\n/g }" | sed -ne "/^${kak_selection}|/ { s/^[^|]\+|//; s/|.*$//; s/\\\:/:/g; p }")
|
2015-11-30 14:56:01 +01:00
|
|
|
if [ -n "$desc" ]; then
|
2017-11-03 09:09:45 +01:00
|
|
|
printf %s\\n "evaluate-commands -client $kak_client %{info -anchor ${kak_cursor_line}.${kak_cursor_column} -placement above %{${desc}}}"
|
2015-11-30 14:56:01 +01:00
|
|
|
fi
|
|
|
|
] ]
|
|
|
|
] ]
|
|
|
|
|
2017-11-03 08:34:41 +01:00
|
|
|
define-command clang-enable-autocomplete -docstring "Enable automatic clang completion" %{
|
2018-05-30 15:28:50 +02:00
|
|
|
set-option window completers "option=clang_completions" %opt{completers}
|
2015-11-30 14:56:01 +01:00
|
|
|
hook window -group clang-autocomplete InsertIdle .* %{
|
|
|
|
try %{
|
2017-11-03 09:09:45 +01:00
|
|
|
execute-keys -draft <a-h><a-k>(\.|->|::).\z<ret>
|
2015-11-30 14:56:01 +01:00
|
|
|
echo 'completing...'
|
|
|
|
clang-complete
|
|
|
|
}
|
|
|
|
clang-show-completion-info
|
|
|
|
}
|
2014-11-18 14:55:52 +01:00
|
|
|
alias window complete clang-complete
|
2013-03-14 14:12:14 +01:00
|
|
|
}
|
2013-04-11 14:30:02 +02:00
|
|
|
|
2017-11-03 08:34:41 +01:00
|
|
|
define-command clang-disable-autocomplete -docstring "Disable automatic clang completion" %{
|
2018-06-30 01:49:57 +02:00
|
|
|
evaluate-commands %sh{ printf "set-option window completers %s\n" $(printf %s "${kak_opt_completers}" | sed -e "s/'option=clang_completions'//g") }
|
2017-01-04 01:07:45 +01:00
|
|
|
remove-hooks window clang-autocomplete
|
2014-11-18 14:55:52 +01:00
|
|
|
unalias window complete clang-complete
|
2014-03-15 04:14:23 +01:00
|
|
|
}
|
2014-11-19 14:53:31 +01:00
|
|
|
|
2017-11-03 08:34:41 +01:00
|
|
|
define-command -hidden clang-show-error-info %{
|
2017-05-25 13:17:22 +02:00
|
|
|
update-option buffer clang_errors # Ensure we are up to date with buffer changes
|
2018-05-06 23:29:52 +02:00
|
|
|
evaluate-commands %sh{
|
2018-05-30 15:28:50 +02:00
|
|
|
eval "set -- ${kak_opt_clang_errors}"
|
|
|
|
shift # skip timestamp
|
|
|
|
for error in "$@"; do
|
2018-06-21 00:46:38 +02:00
|
|
|
if [ "${error%%|*}" = "$kak_cursor_line" ]; then
|
2018-05-30 15:28:50 +02:00
|
|
|
desc=$(printf '%s%s\n' "$desc" "${error##*|}")
|
|
|
|
fi
|
|
|
|
done
|
2017-05-25 13:17:22 +02:00
|
|
|
if [ -n "$desc" ]; then
|
2018-05-30 15:28:50 +02:00
|
|
|
printf %s\\n "info -anchor ${kak_cursor_line}.${kak_cursor_column} '$desc'"
|
2017-05-25 13:17:22 +02:00
|
|
|
fi
|
|
|
|
} }
|
2014-11-19 14:53:31 +01:00
|
|
|
|
2017-11-03 08:34:41 +01:00
|
|
|
define-command clang-enable-diagnostics -docstring %{Activate automatic error reporting and diagnostics
|
2016-10-11 09:03:41 +02:00
|
|
|
Information about the analysis are showned after the buffer has been parsed with the clang-parse function} \
|
|
|
|
%{
|
2018-07-08 11:11:04 +02:00
|
|
|
add-highlighter window/clang_flags flag-lines default clang_flags
|
2014-11-19 14:53:31 +01:00
|
|
|
hook window -group clang-diagnostics NormalIdle .* %{ clang-show-error-info }
|
2016-05-12 00:57:21 +02:00
|
|
|
hook window -group clang-diagnostics WinSetOption ^clang_errors=.* %{ info; clang-show-error-info }
|
2014-11-19 14:53:31 +01:00
|
|
|
}
|
|
|
|
|
2017-11-03 08:34:41 +01:00
|
|
|
define-command clang-disable-diagnostics -docstring "Disable automatic error reporting and diagnostics" %{
|
2018-06-30 01:49:57 +02:00
|
|
|
remove-highlighter window/clang_flags
|
2017-01-04 01:07:45 +01:00
|
|
|
remove-hooks window clang-diagnostics
|
2014-11-19 14:53:31 +01:00
|
|
|
}
|
2015-02-18 14:56:13 +01:00
|
|
|
|
2017-11-03 08:34:41 +01:00
|
|
|
define-command clang-diagnostics-next -docstring "Jump to the next line that contains an error" %{
|
2017-05-25 13:17:22 +02:00
|
|
|
update-option buffer clang_errors # Ensure we are up to date with buffer changes
|
2018-05-06 23:29:52 +02:00
|
|
|
evaluate-commands %sh{
|
2018-05-30 15:28:50 +02:00
|
|
|
eval "set -- ${kak_opt_clang_errors}"
|
|
|
|
shift # skip timestamp
|
|
|
|
for error in "$@"; do
|
|
|
|
candidate=${error%%|*}
|
|
|
|
first_line=${first_line-$candidate}
|
|
|
|
if [ "$candidate" -gt $kak_cursor_line ]; then
|
|
|
|
line=$candidate
|
|
|
|
break
|
2015-08-04 19:41:56 +02:00
|
|
|
fi
|
2018-05-30 15:28:50 +02:00
|
|
|
done
|
|
|
|
line=${line-$first_line}
|
|
|
|
if [ -n "$line" ]; then
|
|
|
|
printf %s\\n "execute-keys ${line} g"
|
|
|
|
else
|
|
|
|
echo "echo -markup '{Error}no next clang diagnostic'"
|
|
|
|
fi
|
2017-05-25 13:17:22 +02:00
|
|
|
} }
|