[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:
River Riddle 2022-04-20 11:58:11 -07:00
parent 09af7fefc8
commit 41d2c6df5c
7 changed files with 134 additions and 21 deletions

View File

@ -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
//===--------------------------------------------------------------------===// //===--------------------------------------------------------------------===//

View File

@ -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

View File

@ -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,

View File

@ -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);

View File

@ -119,7 +119,8 @@ void LSPServer::onInitialize(const InitializeParams &params,
">", ":", ";", ",", "+", "-", "/", "*", "%", ">", ":", ";", ",", "+", "-", "/", "*", "%",
"^", "&", "#", "?", ".", "=", "\"", "'", "|"}}, "^", "&", "#", "?", ".", "=", "\"", "'", "|"}},
{"resolveProvider", false}, {"resolveProvider", false},
{"triggerCharacters", {".", ">", "(", "{", ",", "<", ":", "[", " "}}, {"triggerCharacters",
{".", ">", "(", "{", ",", "<", ":", "[", " ", "\"", "/"}},
}}, }},
{"signatureHelpProvider", {"signatureHelpProvider",
llvm::json::Object{ llvm::json::Object{

View File

@ -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);

View File

@ -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"}