111 lines
2.3 KiB
Rust

//! Completes function abi strings.
use syntax::{
ast::{self, IsString},
AstNode, AstToken,
};
use crate::{
completions::Completions,
context::{CompletionContext, IdentContext},
CompletionItem, CompletionItemKind,
};
// Most of these are feature gated, we should filter/add feature gate completions once we have them.
const SUPPORTED_CALLING_CONVENTIONS: &[&str] = &[
"Rust",
"C",
"C-unwind",
"cdecl",
"stdcall",
"stdcall-unwind",
"fastcall",
"vectorcall",
"thiscall",
"thiscall-unwind",
"aapcs",
"win64",
"sysv64",
"ptx-kernel",
"msp430-interrupt",
"x86-interrupt",
"amdgpu-kernel",
"efiapi",
"avr-interrupt",
"avr-non-blocking-interrupt",
"C-cmse-nonsecure-call",
"wasm",
"system",
"system-unwind",
"rust-intrinsic",
"rust-call",
"platform-intrinsic",
"unadjusted",
];
pub(crate) fn complete_extern_abi(acc: &mut Completions, ctx: &CompletionContext) -> Option<()> {
let abi_str = match &ctx.ident_ctx {
IdentContext::String { expanded: Some(expanded), .. }
if expanded.syntax().parent().map_or(false, |it| ast::Abi::can_cast(it.kind())) =>
{
expanded
}
_ => return None,
};
let source_range = abi_str.text_range_between_quotes()?;
for &abi in SUPPORTED_CALLING_CONVENTIONS {
CompletionItem::new(CompletionItemKind::Keyword, source_range, abi).add_to(acc);
}
Some(())
}
#[cfg(test)]
mod tests {
use expect_test::{expect, Expect};
use crate::tests::{check_edit, completion_list_no_kw};
fn check(ra_fixture: &str, expect: Expect) {
let actual = completion_list_no_kw(ra_fixture);
expect.assert_eq(&actual);
}
#[test]
fn only_completes_in_string_literals() {
check(
r#"
$0 fn foo {}
"#,
expect![[]],
);
}
#[test]
fn requires_extern_prefix() {
check(
r#"
"$0" fn foo {}
"#,
expect![[]],
);
}
#[test]
fn works() {
check(
r#"
extern "$0" fn foo {}
"#,
expect![[]],
);
check_edit(
"Rust",
r#"
extern "$0" fn foo {}
"#,
r#"
extern "Rust" fn foo {}
"#,
);
}
}