forked from OSchip/llvm-project
[mlir][PDLL-LSP] Add code completion for include file paths
This allows for providing completion results for include directive file paths by searching the set of include directories for the current file. Differential Revision: https://reviews.llvm.org/D124112
This commit is contained in:
parent
09af7fefc8
commit
41d2c6df5c
|
@ -66,6 +66,9 @@ public:
|
||||||
/// Signal code completion for Pattern metadata.
|
/// Signal code completion for Pattern metadata.
|
||||||
virtual void codeCompletePatternMetadata() {}
|
virtual void codeCompletePatternMetadata() {}
|
||||||
|
|
||||||
|
/// Signal code completion for an include filename.
|
||||||
|
virtual void codeCompleteIncludeFilename(StringRef curPath) {}
|
||||||
|
|
||||||
//===--------------------------------------------------------------------===//
|
//===--------------------------------------------------------------------===//
|
||||||
// Signature Hooks
|
// Signature Hooks
|
||||||
//===--------------------------------------------------------------------===//
|
//===--------------------------------------------------------------------===//
|
||||||
|
|
|
@ -22,11 +22,15 @@ using namespace mlir::pdll;
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
|
|
||||||
std::string Token::getStringValue() const {
|
std::string Token::getStringValue() const {
|
||||||
assert(getKind() == string || getKind() == string_block);
|
assert(getKind() == string || getKind() == string_block ||
|
||||||
|
getKind() == code_complete_string);
|
||||||
|
|
||||||
// Start by dropping the quotes.
|
// Start by dropping the quotes.
|
||||||
StringRef bytes = getSpelling().drop_front().drop_back();
|
StringRef bytes = getSpelling();
|
||||||
if (is(string_block)) bytes = bytes.drop_front().drop_back();
|
if (is(string))
|
||||||
|
bytes = bytes.drop_front().drop_back();
|
||||||
|
else if (is(string_block))
|
||||||
|
bytes = bytes.drop_front(2).drop_back(2);
|
||||||
|
|
||||||
std::string result;
|
std::string result;
|
||||||
result.reserve(bytes.size());
|
result.reserve(bytes.size());
|
||||||
|
@ -337,6 +341,16 @@ Token Lexer::lexNumber(const char *tokStart) {
|
||||||
|
|
||||||
Token Lexer::lexString(const char *tokStart, bool isStringBlock) {
|
Token Lexer::lexString(const char *tokStart, bool isStringBlock) {
|
||||||
while (true) {
|
while (true) {
|
||||||
|
// Check to see if there is a code completion location within the string. In
|
||||||
|
// these cases we generate a completion location and place the currently
|
||||||
|
// lexed string within the token (without the quotes). This allows for the
|
||||||
|
// parser to use the partially lexed string when computing the completion
|
||||||
|
// results.
|
||||||
|
if (curPtr == codeCompletionLocation) {
|
||||||
|
return formToken(Token::code_complete_string,
|
||||||
|
tokStart + (isStringBlock ? 2 : 1));
|
||||||
|
}
|
||||||
|
|
||||||
switch (*curPtr++) {
|
switch (*curPtr++) {
|
||||||
case '"':
|
case '"':
|
||||||
// If this is a string block, we only end the string when we encounter a
|
// If this is a string block, we only end the string when we encounter a
|
||||||
|
|
|
@ -34,22 +34,25 @@ class DiagnosticEngine;
|
||||||
class Token {
|
class Token {
|
||||||
public:
|
public:
|
||||||
enum Kind {
|
enum Kind {
|
||||||
// Markers.
|
/// Markers.
|
||||||
eof,
|
eof,
|
||||||
error,
|
error,
|
||||||
|
/// Token signifying a code completion location.
|
||||||
code_complete,
|
code_complete,
|
||||||
|
/// Token signifying a code completion location within a string.
|
||||||
|
code_complete_string,
|
||||||
|
|
||||||
// Keywords.
|
/// Keywords.
|
||||||
KW_BEGIN,
|
KW_BEGIN,
|
||||||
// Dependent keywords, i.e. those that are treated as keywords depending on
|
/// Dependent keywords, i.e. those that are treated as keywords depending on
|
||||||
// the current parser context.
|
/// the current parser context.
|
||||||
KW_DEPENDENT_BEGIN,
|
KW_DEPENDENT_BEGIN,
|
||||||
kw_attr,
|
kw_attr,
|
||||||
kw_op,
|
kw_op,
|
||||||
kw_type,
|
kw_type,
|
||||||
KW_DEPENDENT_END,
|
KW_DEPENDENT_END,
|
||||||
|
|
||||||
// General keywords.
|
/// General keywords.
|
||||||
kw_Attr,
|
kw_Attr,
|
||||||
kw_erase,
|
kw_erase,
|
||||||
kw_let,
|
kw_let,
|
||||||
|
@ -68,7 +71,7 @@ public:
|
||||||
kw_with,
|
kw_with,
|
||||||
KW_END,
|
KW_END,
|
||||||
|
|
||||||
// Punctuation.
|
/// Punctuation.
|
||||||
arrow,
|
arrow,
|
||||||
colon,
|
colon,
|
||||||
comma,
|
comma,
|
||||||
|
@ -76,7 +79,7 @@ public:
|
||||||
equal,
|
equal,
|
||||||
equal_arrow,
|
equal_arrow,
|
||||||
semicolon,
|
semicolon,
|
||||||
// Paired punctuation.
|
/// Paired punctuation.
|
||||||
less,
|
less,
|
||||||
greater,
|
greater,
|
||||||
l_brace,
|
l_brace,
|
||||||
|
@ -87,7 +90,7 @@ public:
|
||||||
r_square,
|
r_square,
|
||||||
underscore,
|
underscore,
|
||||||
|
|
||||||
// Tokens.
|
/// Tokens.
|
||||||
directive,
|
directive,
|
||||||
identifier,
|
identifier,
|
||||||
integer,
|
integer,
|
||||||
|
|
|
@ -424,6 +424,7 @@ private:
|
||||||
LogicalResult codeCompleteDialectName();
|
LogicalResult codeCompleteDialectName();
|
||||||
LogicalResult codeCompleteOperationName(StringRef dialectName);
|
LogicalResult codeCompleteOperationName(StringRef dialectName);
|
||||||
LogicalResult codeCompletePatternMetadata();
|
LogicalResult codeCompletePatternMetadata();
|
||||||
|
LogicalResult codeCompleteIncludeFilename(StringRef curPath);
|
||||||
|
|
||||||
void codeCompleteCallSignature(ast::Node *parent, unsigned currentNumArgs);
|
void codeCompleteCallSignature(ast::Node *parent, unsigned currentNumArgs);
|
||||||
void codeCompleteOperationOperandsSignature(Optional<StringRef> opName,
|
void codeCompleteOperationOperandsSignature(Optional<StringRef> opName,
|
||||||
|
@ -680,6 +681,10 @@ LogicalResult Parser::parseInclude(SmallVectorImpl<ast::Decl *> &decls) {
|
||||||
SMRange loc = curToken.getLoc();
|
SMRange loc = curToken.getLoc();
|
||||||
consumeToken(Token::directive);
|
consumeToken(Token::directive);
|
||||||
|
|
||||||
|
// Handle code completion of the include file path.
|
||||||
|
if (curToken.is(Token::code_complete_string))
|
||||||
|
return codeCompleteIncludeFilename(curToken.getStringValue());
|
||||||
|
|
||||||
// Parse the file being included.
|
// Parse the file being included.
|
||||||
if (!curToken.isString())
|
if (!curToken.isString())
|
||||||
return emitError(loc,
|
return emitError(loc,
|
||||||
|
@ -2922,6 +2927,11 @@ LogicalResult Parser::codeCompletePatternMetadata() {
|
||||||
return failure();
|
return failure();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
LogicalResult Parser::codeCompleteIncludeFilename(StringRef curPath) {
|
||||||
|
codeCompleteContext->codeCompleteIncludeFilename(curPath);
|
||||||
|
return failure();
|
||||||
|
}
|
||||||
|
|
||||||
void Parser::codeCompleteCallSignature(ast::Node *parent,
|
void Parser::codeCompleteCallSignature(ast::Node *parent,
|
||||||
unsigned currentNumArgs) {
|
unsigned currentNumArgs) {
|
||||||
ast::CallableDecl *callableDecl = tryExtractCallableDecl(parent);
|
ast::CallableDecl *callableDecl = tryExtractCallableDecl(parent);
|
||||||
|
|
|
@ -119,7 +119,8 @@ void LSPServer::onInitialize(const InitializeParams ¶ms,
|
||||||
">", ":", ";", ",", "+", "-", "/", "*", "%",
|
">", ":", ";", ",", "+", "-", "/", "*", "%",
|
||||||
"^", "&", "#", "?", ".", "=", "\"", "'", "|"}},
|
"^", "&", "#", "?", ".", "=", "\"", "'", "|"}},
|
||||||
{"resolveProvider", false},
|
{"resolveProvider", false},
|
||||||
{"triggerCharacters", {".", ">", "(", "{", ",", "<", ":", "[", " "}},
|
{"triggerCharacters",
|
||||||
|
{".", ">", "(", "{", ",", "<", ":", "[", " ", "\"", "/"}},
|
||||||
}},
|
}},
|
||||||
{"signatureHelpProvider",
|
{"signatureHelpProvider",
|
||||||
llvm::json::Object{
|
llvm::json::Object{
|
||||||
|
|
|
@ -22,7 +22,9 @@
|
||||||
#include "mlir/Tools/PDLL/Parser/Parser.h"
|
#include "mlir/Tools/PDLL/Parser/Parser.h"
|
||||||
#include "llvm/ADT/IntervalMap.h"
|
#include "llvm/ADT/IntervalMap.h"
|
||||||
#include "llvm/ADT/StringMap.h"
|
#include "llvm/ADT/StringMap.h"
|
||||||
|
#include "llvm/ADT/StringSet.h"
|
||||||
#include "llvm/ADT/TypeSwitch.h"
|
#include "llvm/ADT/TypeSwitch.h"
|
||||||
|
#include "llvm/Support/FileSystem.h"
|
||||||
#include "llvm/Support/Path.h"
|
#include "llvm/Support/Path.h"
|
||||||
|
|
||||||
using namespace mlir;
|
using namespace mlir;
|
||||||
|
@ -665,9 +667,10 @@ namespace {
|
||||||
class LSPCodeCompleteContext : public CodeCompleteContext {
|
class LSPCodeCompleteContext : public CodeCompleteContext {
|
||||||
public:
|
public:
|
||||||
LSPCodeCompleteContext(SMLoc completeLoc, lsp::CompletionList &completionList,
|
LSPCodeCompleteContext(SMLoc completeLoc, lsp::CompletionList &completionList,
|
||||||
ods::Context &odsContext)
|
ods::Context &odsContext,
|
||||||
|
ArrayRef<std::string> includeDirs)
|
||||||
: CodeCompleteContext(completeLoc), completionList(completionList),
|
: CodeCompleteContext(completeLoc), completionList(completionList),
|
||||||
odsContext(odsContext) {}
|
odsContext(odsContext), includeDirs(includeDirs) {}
|
||||||
|
|
||||||
void codeCompleteTupleMemberAccess(ast::TupleType tupleType) final {
|
void codeCompleteTupleMemberAccess(ast::TupleType tupleType) final {
|
||||||
ArrayRef<ast::Type> elementTypes = tupleType.getElementTypes();
|
ArrayRef<ast::Type> elementTypes = tupleType.getElementTypes();
|
||||||
|
@ -901,9 +904,68 @@ public:
|
||||||
"The pattern properly handles recursive application.");
|
"The pattern properly handles recursive application.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void codeCompleteIncludeFilename(StringRef curPath) final {
|
||||||
|
// Normalize the path to allow for interacting with the file system
|
||||||
|
// utilities.
|
||||||
|
SmallString<128> nativeRelDir(llvm::sys::path::convert_to_slash(curPath));
|
||||||
|
llvm::sys::path::native(nativeRelDir);
|
||||||
|
|
||||||
|
// Set of already included completion paths.
|
||||||
|
StringSet<> seenResults;
|
||||||
|
|
||||||
|
// Functor used to add a single include completion item.
|
||||||
|
auto addIncludeCompletion = [&](StringRef path, bool isDirectory) {
|
||||||
|
lsp::CompletionItem item;
|
||||||
|
item.label = (path + (isDirectory ? "/" : "")).str();
|
||||||
|
item.kind = isDirectory ? lsp::CompletionItemKind::Folder
|
||||||
|
: lsp::CompletionItemKind::File;
|
||||||
|
if (seenResults.insert(item.label).second)
|
||||||
|
completionList.items.emplace_back(item);
|
||||||
|
};
|
||||||
|
|
||||||
|
// Process the include directories for this file, adding any potential
|
||||||
|
// nested include files or directories.
|
||||||
|
for (StringRef includeDir : includeDirs) {
|
||||||
|
llvm::SmallString<128> dir = includeDir;
|
||||||
|
if (!nativeRelDir.empty())
|
||||||
|
llvm::sys::path::append(dir, nativeRelDir);
|
||||||
|
|
||||||
|
std::error_code errorCode;
|
||||||
|
for (auto it = llvm::sys::fs::directory_iterator(dir, errorCode),
|
||||||
|
e = llvm::sys::fs::directory_iterator();
|
||||||
|
!errorCode && it != e; it.increment(errorCode)) {
|
||||||
|
StringRef filename = llvm::sys::path::filename(it->path());
|
||||||
|
|
||||||
|
// To know whether a symlink should be treated as file or a directory,
|
||||||
|
// we have to stat it. This should be cheap enough as there shouldn't be
|
||||||
|
// many symlinks.
|
||||||
|
llvm::sys::fs::file_type fileType = it->type();
|
||||||
|
if (fileType == llvm::sys::fs::file_type::symlink_file) {
|
||||||
|
if (auto fileStatus = it->status())
|
||||||
|
fileType = fileStatus->type();
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (fileType) {
|
||||||
|
case llvm::sys::fs::file_type::directory_file:
|
||||||
|
addIncludeCompletion(filename, /*isDirectory=*/true);
|
||||||
|
break;
|
||||||
|
case llvm::sys::fs::file_type::regular_file: {
|
||||||
|
// Only consider concrete files that can actually be included by PDLL.
|
||||||
|
if (filename.endswith(".pdll") || filename.endswith(".td"))
|
||||||
|
addIncludeCompletion(filename, /*isDirectory=*/false);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
lsp::CompletionList &completionList;
|
lsp::CompletionList &completionList;
|
||||||
ods::Context &odsContext;
|
ods::Context &odsContext;
|
||||||
|
ArrayRef<std::string> includeDirs;
|
||||||
};
|
};
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
|
@ -921,8 +983,8 @@ PDLDocument::getCodeCompletion(const lsp::URIForFile &uri,
|
||||||
// code completion context provided.
|
// code completion context provided.
|
||||||
ods::Context tmpODSContext;
|
ods::Context tmpODSContext;
|
||||||
lsp::CompletionList completionList;
|
lsp::CompletionList completionList;
|
||||||
LSPCodeCompleteContext lspCompleteContext(posLoc, completionList,
|
LSPCodeCompleteContext lspCompleteContext(
|
||||||
tmpODSContext);
|
posLoc, completionList, tmpODSContext, sourceMgr.getIncludeDirs());
|
||||||
|
|
||||||
ast::Context tmpContext(tmpODSContext);
|
ast::Context tmpContext(tmpODSContext);
|
||||||
(void)parsePDLAST(tmpContext, sourceMgr, &lspCompleteContext);
|
(void)parsePDLAST(tmpContext, sourceMgr, &lspCompleteContext);
|
||||||
|
|
|
@ -1,16 +1,16 @@
|
||||||
// RUN: mlir-pdll-lsp-server -lit-test < %s | FileCheck -strict-whitespace %s
|
// RUN: mlir-pdll-lsp-server -pdll-extra-dir %S -pdll-extra-dir %S/../../include -lit-test < %s | FileCheck -strict-whitespace %s
|
||||||
{"jsonrpc":"2.0","id":0,"method":"initialize","params":{"processId":123,"rootPath":"pdll","capabilities":{},"trace":"off"}}
|
{"jsonrpc":"2.0","id":0,"method":"initialize","params":{"processId":123,"rootPath":"pdll","capabilities":{},"trace":"off"}}
|
||||||
// -----
|
// -----
|
||||||
{"jsonrpc":"2.0","method":"textDocument/didOpen","params":{"textDocument":{
|
{"jsonrpc":"2.0","method":"textDocument/didOpen","params":{"textDocument":{
|
||||||
"uri":"test:///foo.pdll",
|
"uri":"test:///foo.pdll",
|
||||||
"languageId":"pdll",
|
"languageId":"pdll",
|
||||||
"version":1,
|
"version":1,
|
||||||
"text":"Constraint ValueCst(value: Value);\nConstraint Cst();\nPattern FooPattern with benefit(1) {\nlet tuple = (value1 = _: Op, _: Op<test.op>);\nerase tuple.value1;\n}"
|
"text":"#include \"include/included.pdll\"\nConstraint ValueCst(value: Value);\nConstraint Cst();\nPattern FooPattern with benefit(1) {\nlet tuple = (value1 = _: Op, _: Op<test.op>);\nerase tuple.value1;\n}"
|
||||||
}}}
|
}}}
|
||||||
// -----
|
// -----
|
||||||
{"jsonrpc":"2.0","id":1,"method":"textDocument/completion","params":{
|
{"jsonrpc":"2.0","id":1,"method":"textDocument/completion","params":{
|
||||||
"textDocument":{"uri":"test:///foo.pdll"},
|
"textDocument":{"uri":"test:///foo.pdll"},
|
||||||
"position":{"line":4,"character":12}
|
"position":{"line":5,"character":12}
|
||||||
}}
|
}}
|
||||||
// CHECK: "id": 1
|
// CHECK: "id": 1
|
||||||
// CHECK-NEXT: "jsonrpc": "2.0",
|
// CHECK-NEXT: "jsonrpc": "2.0",
|
||||||
|
@ -49,7 +49,7 @@
|
||||||
// -----
|
// -----
|
||||||
{"jsonrpc":"2.0","id":1,"method":"textDocument/completion","params":{
|
{"jsonrpc":"2.0","id":1,"method":"textDocument/completion","params":{
|
||||||
"textDocument":{"uri":"test:///foo.pdll"},
|
"textDocument":{"uri":"test:///foo.pdll"},
|
||||||
"position":{"line":2,"character":23}
|
"position":{"line":3,"character":23}
|
||||||
}}
|
}}
|
||||||
// CHECK: "id": 1
|
// CHECK: "id": 1
|
||||||
// CHECK-NEXT: "jsonrpc": "2.0",
|
// CHECK-NEXT: "jsonrpc": "2.0",
|
||||||
|
@ -82,7 +82,7 @@
|
||||||
// -----
|
// -----
|
||||||
{"jsonrpc":"2.0","id":1,"method":"textDocument/completion","params":{
|
{"jsonrpc":"2.0","id":1,"method":"textDocument/completion","params":{
|
||||||
"textDocument":{"uri":"test:///foo.pdll"},
|
"textDocument":{"uri":"test:///foo.pdll"},
|
||||||
"position":{"line":3,"character":24}
|
"position":{"line":4,"character":24}
|
||||||
}}
|
}}
|
||||||
// CHECK: "id": 1
|
// CHECK: "id": 1
|
||||||
// CHECK-NEXT: "jsonrpc": "2.0",
|
// CHECK-NEXT: "jsonrpc": "2.0",
|
||||||
|
@ -200,6 +200,26 @@
|
||||||
// CHECK-NEXT: ]
|
// CHECK-NEXT: ]
|
||||||
// CHECK-NEXT: }
|
// CHECK-NEXT: }
|
||||||
// -----
|
// -----
|
||||||
|
{"jsonrpc":"2.0","id":1,"method":"textDocument/completion","params":{
|
||||||
|
"textDocument":{"uri":"test:///foo.pdll"},
|
||||||
|
"position":{"line":0,"character":18}
|
||||||
|
}}
|
||||||
|
// CHECK: "id": 1
|
||||||
|
// CHECK-NEXT: "jsonrpc": "2.0",
|
||||||
|
// CHECK-NEXT: "result": {
|
||||||
|
// CHECK-NEXT: "isIncomplete": false,
|
||||||
|
// CHECK-NEXT: "items": [
|
||||||
|
// CHECK-NEXT: {
|
||||||
|
// CHECK-NEXT: "kind": 17,
|
||||||
|
// CHECK-NEXT: "label": "included.td"
|
||||||
|
// CHECK-NEXT: },
|
||||||
|
// CHECK-NEXT: {
|
||||||
|
// CHECK-NEXT: "kind": 17,
|
||||||
|
// CHECK-NEXT: "label": "included.pdll"
|
||||||
|
// CHECK-NEXT: }
|
||||||
|
// CHECK-NEXT: ]
|
||||||
|
// CHECK-NEXT: }
|
||||||
|
// -----
|
||||||
{"jsonrpc":"2.0","id":3,"method":"shutdown"}
|
{"jsonrpc":"2.0","id":3,"method":"shutdown"}
|
||||||
// -----
|
// -----
|
||||||
{"jsonrpc":"2.0","method":"exit"}
|
{"jsonrpc":"2.0","method":"exit"}
|
||||||
|
|
Loading…
Reference in New Issue