Merge pull request #1215 from Techatrix/refactor-doc-scope

Refactor document scope generation
This commit is contained in:
Lee Cannon 2023-05-30 21:05:26 +01:00 committed by GitHub
commit fa0793596f
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 169 additions and 380 deletions

View File

@ -1032,7 +1032,8 @@ fn resolveTypeOfNodeUncached(analyser: *Analyser, node_handle: NodeWithHandle) e
const new_handle = analyser.store.getOrLoadHandle(builtin_uri) orelse return null; const new_handle = analyser.store.getOrLoadHandle(builtin_uri) orelse return null;
const root_scope_decls = new_handle.document_scope.scopes.items(.decls)[0]; const root_scope_decls = new_handle.document_scope.scopes.items(.decls)[0];
const decl = root_scope_decls.get("Type") orelse return null; const decl_index = root_scope_decls.get("Type") orelse return null;
const decl = new_handle.document_scope.decls.items[@enumToInt(decl_index)];
if (decl != .ast_node) return null; if (decl != .ast_node) return null;
const var_decl = new_handle.tree.fullVarDecl(decl.ast_node) orelse return null; const var_decl = new_handle.tree.fullVarDecl(decl.ast_node) orelse return null;
@ -1969,6 +1970,8 @@ pub const Declaration = union(enum) {
/// always an identifier /// always an identifier
error_token: Ast.Node.Index, error_token: Ast.Node.Index,
pub const Index = enum(u32) { _ };
pub fn eql(a: Declaration, b: Declaration) bool { pub fn eql(a: Declaration, b: Declaration) bool {
return std.meta.eql(a, b); return std.meta.eql(a, b);
} }
@ -2127,9 +2130,11 @@ pub const DeclWithHandle = struct {
const scope_index = findContainerScopeIndex(.{ .node = switch_expr_type.type.data.other, .handle = switch_expr_type.handle }) orelse return null; const scope_index = findContainerScopeIndex(.{ .node = switch_expr_type.type.data.other, .handle = switch_expr_type.handle }) orelse return null;
const scope_decls = switch_expr_type.handle.document_scope.scopes.items(.decls); const scope_decls = switch_expr_type.handle.document_scope.scopes.items(.decls);
const candidate = scope_decls[scope_index].getEntry(tree.tokenSlice(main_tokens[pay.items[0]])) orelse return null; const name = tree.tokenSlice(main_tokens[pay.items[0]]);
const decl_index = scope_decls[scope_index].get(name) orelse return null;
const decl = switch_expr_type.handle.document_scope.decls.items[@enumToInt(decl_index)];
switch (candidate.value_ptr.*) { switch (decl) {
.ast_node => |node| { .ast_node => |node| {
if (switch_expr_type.handle.tree.fullContainerField(node)) |container_field| { if (switch_expr_type.handle.tree.fullContainerField(node)) |container_field| {
if (container_field.ast.type_expr != 0) { if (container_field.ast.type_expr != 0) {
@ -2186,9 +2191,9 @@ fn iterateSymbolsContainerInternal(
const scope_uses = handle.document_scope.scopes.items(.uses); const scope_uses = handle.document_scope.scopes.items(.uses);
const container_scope_index = findContainerScopeIndex(container_handle) orelse return; const container_scope_index = findContainerScopeIndex(container_handle) orelse return;
var decl_it = scope_decls[container_scope_index].iterator(); for (scope_decls[container_scope_index].values()) |decl_index| {
while (decl_it.next()) |entry| { const decl = &handle.document_scope.decls.items[@enumToInt(decl_index)];
switch (entry.value_ptr.*) { switch (decl.*) {
.ast_node => |node| { .ast_node => |node| {
if (node_tags[node].isContainerField()) { if (node_tags[node].isContainerField()) {
if (!instance_access and !is_enum) continue; if (!instance_access and !is_enum) continue;
@ -2205,9 +2210,9 @@ fn iterateSymbolsContainerInternal(
else => {}, else => {},
} }
const decl = DeclWithHandle{ .decl = entry.value_ptr, .handle = handle }; const decl_with_handle = DeclWithHandle{ .decl = decl, .handle = handle };
if (handle != orig_handle and !decl.isPublic()) continue; if (handle != orig_handle and !decl_with_handle.isPublic()) continue;
try callback(context, decl); try callback(context, decl_with_handle);
} }
for (scope_uses[container_scope_index]) |use| { for (scope_uses[container_scope_index]) |use| {
@ -2285,13 +2290,10 @@ pub fn iterateLabels(handle: *const DocumentStore.Handle, source_index: usize, c
var scope_iterator = iterateEnclosingScopes(handle.document_scope, source_index); var scope_iterator = iterateEnclosingScopes(handle.document_scope, source_index);
while (scope_iterator.next()) |scope_index| { while (scope_iterator.next()) |scope_index| {
var decl_it = scope_decls[@enumToInt(scope_index)].iterator(); for (scope_decls[@enumToInt(scope_index)].values()) |decl_index| {
while (decl_it.next()) |entry| { const decl = &handle.document_scope.decls.items[@enumToInt(decl_index)];
switch (entry.value_ptr.*) { if (decl.* != .label_decl) continue;
.label_decl => {}, try callback(context, DeclWithHandle{ .decl = decl, .handle = handle });
else => continue,
}
try callback(context, DeclWithHandle{ .decl = entry.value_ptr, .handle = handle });
} }
} }
} }
@ -2308,12 +2310,11 @@ fn iterateSymbolsGlobalInternal(
var scope_iterator = iterateEnclosingScopes(handle.document_scope, source_index); var scope_iterator = iterateEnclosingScopes(handle.document_scope, source_index);
while (scope_iterator.next()) |scope_index| { while (scope_iterator.next()) |scope_index| {
var decl_it = scope_decls[@enumToInt(scope_index)].iterator(); for (scope_decls[@enumToInt(scope_index)].values()) |decl_index| {
while (decl_it.next()) |entry| { const decl = &handle.document_scope.decls.items[@enumToInt(decl_index)];
if (entry.value_ptr.* == .ast_node and if (decl.* == .ast_node and handle.tree.nodes.items(.tag)[decl.ast_node].isContainerField()) continue;
handle.tree.nodes.items(.tag)[entry.value_ptr.*.ast_node].isContainerField()) continue; if (decl.* == .label_decl) continue;
if (entry.value_ptr.* == .label_decl) continue; try callback(context, DeclWithHandle{ .decl = decl, .handle = handle });
try callback(context, DeclWithHandle{ .decl = entry.value_ptr, .handle = handle });
} }
for (scope_uses[@enumToInt(scope_index)]) |use| { for (scope_uses[@enumToInt(scope_index)]) |use| {
@ -2425,16 +2426,12 @@ pub fn lookupLabel(
var scope_iterator = iterateEnclosingScopes(handle.document_scope, source_index); var scope_iterator = iterateEnclosingScopes(handle.document_scope, source_index);
while (scope_iterator.next()) |scope_index| { while (scope_iterator.next()) |scope_index| {
const candidate = scope_decls[@enumToInt(scope_index)].getEntry(symbol) orelse continue; const decl_index = scope_decls[@enumToInt(scope_index)].get(symbol) orelse continue;
const decl = &handle.document_scope.decls.items[@enumToInt(decl_index)];
switch (candidate.value_ptr.*) { if (decl.* != .label_decl) continue;
.label_decl => {},
else => continue, return DeclWithHandle{ .decl = decl, .handle = handle };
}
return DeclWithHandle{
.decl = candidate.value_ptr,
.handle = handle,
};
} }
return null; return null;
} }
@ -2449,18 +2446,16 @@ pub fn lookupSymbolGlobal(analyser: *Analyser, handle: *const DocumentStore.Hand
while (current_scope != .none) { while (current_scope != .none) {
const scope_index = @enumToInt(current_scope); const scope_index = @enumToInt(current_scope);
defer current_scope = scope_parents[scope_index]; defer current_scope = scope_parents[scope_index];
if (scope_decls[scope_index].getEntry(symbol)) |candidate| { if (scope_decls[scope_index].get(symbol)) |decl_index| {
switch (candidate.value_ptr.*) { const candidate = &handle.document_scope.decls.items[@enumToInt(decl_index)];
switch (candidate.*) {
.ast_node => |node| { .ast_node => |node| {
if (handle.tree.nodes.items(.tag)[node].isContainerField()) continue; if (handle.tree.nodes.items(.tag)[node].isContainerField()) continue;
}, },
.label_decl => continue, .label_decl => continue,
else => {}, else => {},
} }
return DeclWithHandle{ return DeclWithHandle{ .decl = candidate, .handle = handle };
.decl = candidate.value_ptr,
.handle = handle,
};
} }
if (try analyser.resolveUse(scope_uses[scope_index], symbol, handle)) |result| return result; if (try analyser.resolveUse(scope_uses[scope_index], symbol, handle)) |result| return result;
} }
@ -2488,8 +2483,9 @@ pub fn lookupSymbolContainer(
const scope_uses = handle.document_scope.scopes.items(.uses); const scope_uses = handle.document_scope.scopes.items(.uses);
if (findContainerScopeIndex(container_handle)) |container_scope_index| { if (findContainerScopeIndex(container_handle)) |container_scope_index| {
if (scope_decls[container_scope_index].getEntry(symbol)) |candidate| { if (scope_decls[container_scope_index].get(symbol)) |decl_index| {
switch (candidate.value_ptr.*) { const decl = &handle.document_scope.decls.items[@enumToInt(decl_index)];
switch (decl.*) {
.ast_node => |node| { .ast_node => |node| {
if (node_tags[node].isContainerField()) { if (node_tags[node].isContainerField()) {
if (!instance_access and !is_enum) return null; if (!instance_access and !is_enum) return null;
@ -2499,7 +2495,7 @@ pub fn lookupSymbolContainer(
.label_decl => unreachable, .label_decl => unreachable,
else => {}, else => {},
} }
return DeclWithHandle{ .decl = candidate.value_ptr, .handle = handle }; return DeclWithHandle{ .decl = decl, .handle = handle };
} }
if (try analyser.resolveUse(scope_uses[container_scope_index], symbol, handle)) |result| return result; if (try analyser.resolveUse(scope_uses[container_scope_index], symbol, handle)) |result| return result;
@ -2532,9 +2528,10 @@ comptime {
} }
pub const DocumentScope = struct { pub const DocumentScope = struct {
scopes: std.MultiArrayList(Scope), scopes: std.MultiArrayList(Scope) = .{},
error_completions: CompletionSet, decls: std.ArrayListUnmanaged(Declaration) = .{},
enum_completions: CompletionSet, error_completions: CompletionSet = .{},
enum_completions: CompletionSet = .{},
pub fn deinit(self: *DocumentScope, allocator: std.mem.Allocator) void { pub fn deinit(self: *DocumentScope, allocator: std.mem.Allocator) void {
for ( for (
@ -2549,8 +2546,9 @@ pub const DocumentScope = struct {
allocator.free(uses); allocator.free(uses);
} }
self.scopes.deinit(allocator); self.scopes.deinit(allocator);
self.decls.deinit(allocator);
for (self.error_completions.entries.items(.key)) |item| { for (self.error_completions.keys()) |item| {
if (item.detail) |detail| allocator.free(detail); if (item.detail) |detail| allocator.free(detail);
switch (item.documentation orelse continue) { switch (item.documentation orelse continue) {
.string => |str| allocator.free(str), .string => |str| allocator.free(str),
@ -2558,7 +2556,7 @@ pub const DocumentScope = struct {
} }
} }
self.error_completions.deinit(allocator); self.error_completions.deinit(allocator);
for (self.enum_completions.entries.items(.key)) |item| { for (self.enum_completions.keys()) |item| {
if (item.detail) |detail| allocator.free(detail); if (item.detail) |detail| allocator.free(detail);
switch (item.documentation orelse continue) { switch (item.documentation orelse continue) {
.string => |str| allocator.free(str), .string => |str| allocator.free(str),
@ -2592,7 +2590,7 @@ pub const Scope = struct {
loc: offsets.Loc, loc: offsets.Loc,
parent: Index, parent: Index,
data: Data, data: Data,
decls: std.StringHashMapUnmanaged(Declaration) = .{}, decls: std.StringArrayHashMapUnmanaged(Declaration.Index) = .{},
child_scopes: std.ArrayListUnmanaged(Scope.Index) = .{}, child_scopes: std.ArrayListUnmanaged(Scope.Index) = .{},
tests: []const Ast.Node.Index = &.{}, tests: []const Ast.Node.Index = &.{},
uses: []const Ast.Node.Index = &.{}, uses: []const Ast.Node.Index = &.{},
@ -2602,63 +2600,64 @@ pub fn makeDocumentScope(allocator: std.mem.Allocator, tree: Ast) !DocumentScope
const tracy_zone = tracy.trace(@src()); const tracy_zone = tracy.trace(@src());
defer tracy_zone.end(); defer tracy_zone.end();
var document_scope = DocumentScope{ var document_scope = DocumentScope{};
.scopes = .{},
.error_completions = .{},
.enum_completions = .{},
};
errdefer document_scope.deinit(allocator); errdefer document_scope.deinit(allocator);
var current_scope: Scope.Index = .none; var current_scope: Scope.Index = .none;
try makeInnerScope(.{ try makeInnerScope(.{
.allocator = allocator, .allocator = allocator,
.scopes = &document_scope.scopes, .doc_scope = &document_scope,
.current_scope = &current_scope, .current_scope = &current_scope,
.tree = tree, }, tree, 0);
.errors = &document_scope.error_completions,
.enums = &document_scope.enum_completions,
}, 0);
return document_scope; return document_scope;
} }
const ScopeContext = struct { const ScopeContext = struct {
allocator: std.mem.Allocator, allocator: std.mem.Allocator,
scopes: *std.MultiArrayList(Scope), doc_scope: *DocumentScope,
current_scope: *Scope.Index, current_scope: *Scope.Index,
tree: Ast,
enums: *CompletionSet,
errors: *CompletionSet,
fn pushScope(context: ScopeContext, loc: offsets.Loc, data: Scope.Data) error{OutOfMemory}!usize { fn pushScope(context: ScopeContext, loc: offsets.Loc, data: Scope.Data) error{OutOfMemory}!Scope.Index {
try context.scopes.append(context.allocator, .{ try context.doc_scope.scopes.append(context.allocator, .{
.parent = context.current_scope.*, .parent = context.current_scope.*,
.loc = loc, .loc = loc,
.data = data, .data = data,
}); });
const new_scope = @intToEnum(Scope.Index, context.scopes.len - 1); const new_scope = @intToEnum(Scope.Index, context.doc_scope.scopes.len - 1);
if (context.current_scope.* != .none) { if (context.current_scope.* != .none) {
try context.scopes.items(.child_scopes)[@enumToInt(context.current_scope.*)].append(context.allocator, new_scope); try context.doc_scope.scopes.items(.child_scopes)[@enumToInt(context.current_scope.*)].append(context.allocator, new_scope);
} }
context.current_scope.* = new_scope; context.current_scope.* = new_scope;
return @enumToInt(new_scope); return new_scope;
} }
fn popScope(context: ScopeContext) void { fn popScope(context: ScopeContext) void {
const parent_scope = context.scopes.items(.parent)[@enumToInt(context.current_scope.*)]; const parent_scope = context.doc_scope.scopes.items(.parent)[@enumToInt(context.current_scope.*)];
context.current_scope.* = parent_scope; context.current_scope.* = parent_scope;
} }
fn putDecl(context: ScopeContext, scope: Scope.Index, name: []const u8, decl: Declaration) error{OutOfMemory}!void {
std.debug.assert(scope != .none);
try context.doc_scope.decls.append(context.allocator, decl);
errdefer _ = context.doc_scope.decls.pop();
const decl_index = @intToEnum(Declaration.Index, context.doc_scope.decls.items.len - 1);
try context.doc_scope.scopes.items(.decls)[@enumToInt(scope)].put(context.allocator, name, decl_index);
}
}; };
fn makeInnerScope(context: ScopeContext, node_idx: Ast.Node.Index) error{OutOfMemory}!void { fn makeInnerScope(context: ScopeContext, tree: Ast, node_idx: Ast.Node.Index) error{OutOfMemory}!void {
const tracy_zone = tracy.trace(@src()); const tracy_zone = tracy.trace(@src());
defer tracy_zone.end(); defer tracy_zone.end();
const allocator = context.allocator; const allocator = context.allocator;
const scopes = context.scopes; const scopes = &context.doc_scope.scopes;
const tree = context.tree;
const tags = tree.nodes.items(.tag); const tags = tree.nodes.items(.tag);
const token_tags = tree.tokens.items(.tag);
const scope_index = try context.pushScope( const scope_index = try context.pushScope(
offsets.nodeToLoc(tree, node_idx), offsets.nodeToLoc(tree, node_idx),
@ -2675,51 +2674,57 @@ fn makeInnerScope(context: ScopeContext, node_idx: Ast.Node.Index) error{OutOfMe
errdefer uses.deinit(allocator); errdefer uses.deinit(allocator);
for (container_decl.ast.members) |decl| { for (container_decl.ast.members) |decl| {
if (tags[decl] == .@"usingnamespace") { try makeScopeInternal(context, tree, decl);
switch (tags[decl]) {
.@"usingnamespace" => {
try uses.append(allocator, decl); try uses.append(allocator, decl);
continue; continue;
} },
.test_decl => {
try makeScopeInternal(context, decl);
const name = getDeclName(tree, decl) orelse continue;
if (tags[decl] == .test_decl) {
try tests.append(allocator, decl); try tests.append(allocator, decl);
continue; continue;
},
else => {},
} }
try scopes.items(.decls)[scope_index].put(allocator, name, .{ .ast_node = decl });
if (container_decl.ast.enum_token != null) { const name = getDeclName(tree, decl) orelse continue;
try context.putDecl(scope_index, name, .{ .ast_node = decl });
if ((node_idx != 0 and token_tags[container_decl.ast.main_token] == .keyword_enum) or
container_decl.ast.enum_token != null)
{
if (std.mem.eql(u8, name, "_")) return; if (std.mem.eql(u8, name, "_")) return;
const Documentation = @TypeOf(@as(types.CompletionItem, undefined).documentation);
var doc: Documentation = if (try getDocComments(allocator, tree, decl, .markdown)) |docs| .{ .MarkupContent = types.MarkupContent{ .kind = .markdown, .value = docs } } else null; const doc = try getDocComments(allocator, tree, decl, .markdown);
var gop_res = try context.enums.getOrPut(allocator, .{ errdefer if (doc) |d| allocator.free(d);
var gop_res = try context.doc_scope.enum_completions.getOrPut(allocator, .{
.label = name, .label = name,
.kind = .Constant, .kind = .Enum,
.insertText = name, .insertText = name,
.insertTextFormat = .PlainText, .insertTextFormat = .PlainText,
.documentation = doc, .documentation = if (doc) |d| .{ .MarkupContent = types.MarkupContent{ .kind = .markdown, .value = d } } else null,
}); });
if (gop_res.found_existing) { if (gop_res.found_existing) {
if (doc) |d| allocator.free(d.MarkupContent.value); if (doc) |d| allocator.free(d);
} }
} }
} }
scopes.items(.tests)[scope_index] = try tests.toOwnedSlice(allocator); scopes.items(.tests)[@enumToInt(scope_index)] = try tests.toOwnedSlice(allocator);
scopes.items(.uses)[scope_index] = try uses.toOwnedSlice(allocator); scopes.items(.uses)[@enumToInt(scope_index)] = try uses.toOwnedSlice(allocator);
} }
/// If `node_idx` is a block it's scope index will be returned /// If `node_idx` is a block it's scope index will be returned
/// Otherwise, a new scope will be created that will enclose `node_idx` /// Otherwise, a new scope will be created that will enclose `node_idx`
fn makeBlockScopeInternal(context: ScopeContext, node_idx: Ast.Node.Index) error{OutOfMemory}!?usize { fn makeBlockScopeInternal(context: ScopeContext, tree: Ast, node_idx: Ast.Node.Index) error{OutOfMemory}!?Scope.Index {
if (node_idx == 0) return null; if (node_idx == 0) return null;
const tags = context.tree.nodes.items(.tag); const tags = tree.nodes.items(.tag);
// if node_idx is a block, the next scope will be a block so we store its index here // if node_idx is a block, the next scope will be a block so we store its index here
const block_scope_index = context.scopes.len; const block_scope_index = context.doc_scope.scopes.len;
try makeScopeInternal(context, node_idx); try makeScopeInternal(context, tree, node_idx);
switch (tags[node_idx]) { switch (tags[node_idx]) {
.block, .block,
@ -2727,12 +2732,12 @@ fn makeBlockScopeInternal(context: ScopeContext, node_idx: Ast.Node.Index) error
.block_two, .block_two,
.block_two_semicolon, .block_two_semicolon,
=> { => {
std.debug.assert(context.scopes.items(.data)[block_scope_index] == .block); std.debug.assert(context.doc_scope.scopes.items(.data)[block_scope_index] == .block);
return block_scope_index; return @intToEnum(Scope.Index, block_scope_index);
}, },
else => { else => {
const new_scope = try context.pushScope( const new_scope = try context.pushScope(
offsets.nodeToLoc(context.tree, node_idx), offsets.nodeToLoc(tree, node_idx),
.other, .other,
); );
context.popScope(); context.popScope();
@ -2741,16 +2746,16 @@ fn makeBlockScopeInternal(context: ScopeContext, node_idx: Ast.Node.Index) error
} }
} }
fn makeScopeInternal(context: ScopeContext, node_idx: Ast.Node.Index) error{OutOfMemory}!void { fn makeScopeInternal(context: ScopeContext, tree: Ast, node_idx: Ast.Node.Index) error{OutOfMemory}!void {
if (node_idx == 0) return; if (node_idx == 0) return;
const allocator = context.allocator; const allocator = context.allocator;
const scopes = context.scopes;
const tree = context.tree;
const tags = tree.nodes.items(.tag); const tags = tree.nodes.items(.tag);
const token_tags = tree.tokens.items(.tag); const token_tags = tree.tokens.items(.tag);
const data = tree.nodes.items(.data); const data = tree.nodes.items(.data);
const main_tokens = tree.nodes.items(.main_token); const main_tokens = tree.nodes.items(.main_token);
const node_tag = tags[node_idx]; const node_tag = tags[node_idx];
switch (node_tag) { switch (node_tag) {
@ -2767,9 +2772,7 @@ fn makeScopeInternal(context: ScopeContext, node_idx: Ast.Node.Index) error{OutO
.tagged_union_two_trailing, .tagged_union_two_trailing,
.tagged_union_enum_tag, .tagged_union_enum_tag,
.tagged_union_enum_tag_trailing, .tagged_union_enum_tag_trailing,
=> { => try makeInnerScope(context, tree, node_idx),
try makeInnerScope(context, node_idx);
},
.error_set_decl => { .error_set_decl => {
const scope_index = try context.pushScope( const scope_index = try context.pushScope(
offsets.nodeToLoc(tree, node_idx), offsets.nodeToLoc(tree, node_idx),
@ -2784,8 +2787,8 @@ fn makeScopeInternal(context: ScopeContext, node_idx: Ast.Node.Index) error{OutO
.doc_comment, .comma => {}, .doc_comment, .comma => {},
.identifier => { .identifier => {
const name = offsets.tokenToSlice(tree, tok_i); const name = offsets.tokenToSlice(tree, tok_i);
try scopes.items(.decls)[scope_index].put(allocator, name, .{ .error_token = tok_i }); try context.putDecl(scope_index, name, .{ .error_token = tok_i });
const gop = try context.errors.getOrPut(allocator, .{ const gop = try context.doc_scope.error_completions.getOrPut(allocator, .{
.label = name, .label = name,
.kind = .Constant, .kind = .Constant,
//.detail = //.detail =
@ -2824,15 +2827,15 @@ fn makeScopeInternal(context: ScopeContext, node_idx: Ast.Node.Index) error{OutO
while (ast.nextFnParam(&it)) |param| { while (ast.nextFnParam(&it)) |param| {
// Add parameter decls // Add parameter decls
if (param.name_token) |name_token| { if (param.name_token) |name_token| {
try scopes.items(.decls)[scope_index].put( try context.putDecl(
allocator, scope_index,
tree.tokenSlice(name_token), tree.tokenSlice(name_token),
.{ .param_payload = .{ .param = param, .param_idx = @intCast(u16, param_index), .func = node_idx } }, .{ .param_payload = .{ .param = param, .param_idx = @intCast(u16, param_index), .func = node_idx } },
); );
} }
// Visit parameter types to pick up any error sets and enum // Visit parameter types to pick up any error sets and enum
// completions // completions
try makeScopeInternal(context, param.type_expr); try makeScopeInternal(context, tree, param.type_expr);
param_index += 1; param_index += 1;
} }
@ -2841,11 +2844,11 @@ fn makeScopeInternal(context: ScopeContext, node_idx: Ast.Node.Index) error{OutO
const return_type_node = data[data[node_idx].lhs].rhs; const return_type_node = data[data[node_idx].lhs].rhs;
// Visit the return type // Visit the return type
try makeScopeInternal(context, return_type_node); try makeScopeInternal(context, tree, return_type_node);
} }
// Visit the function body // Visit the function body
try makeScopeInternal(context, data[node_idx].rhs); try makeScopeInternal(context, tree, data[node_idx].rhs);
}, },
.block, .block,
.block_semicolon, .block_semicolon,
@ -2873,8 +2876,8 @@ fn makeScopeInternal(context: ScopeContext, node_idx: Ast.Node.Index) error{OutO
// if labeled block // if labeled block
if (token_tags[first_token] == .identifier) { if (token_tags[first_token] == .identifier) {
try scopes.items(.decls)[scope_index].putNoClobber( try context.putDecl(
allocator, scope_index,
tree.tokenSlice(first_token), tree.tokenSlice(first_token),
.{ .label_decl = .{ .label = first_token, .block = node_idx } }, .{ .label_decl = .{ .label = first_token, .block = node_idx } },
); );
@ -2884,10 +2887,10 @@ fn makeScopeInternal(context: ScopeContext, node_idx: Ast.Node.Index) error{OutO
const statements = ast.blockStatements(tree, node_idx, &buffer).?; const statements = ast.blockStatements(tree, node_idx, &buffer).?;
for (statements) |idx| { for (statements) |idx| {
try makeScopeInternal(context, idx); try makeScopeInternal(context, tree, idx);
if (tree.fullVarDecl(idx)) |var_decl| { if (tree.fullVarDecl(idx)) |var_decl| {
const name = tree.tokenSlice(var_decl.ast.mut_token + 1); const name = tree.tokenSlice(var_decl.ast.mut_token + 1);
try scopes.items(.decls)[scope_index].put(allocator, name, .{ .ast_node = idx }); try context.putDecl(scope_index, name, .{ .ast_node = idx });
} }
} }
}, },
@ -2896,32 +2899,32 @@ fn makeScopeInternal(context: ScopeContext, node_idx: Ast.Node.Index) error{OutO
=> { => {
const if_node = ast.fullIf(tree, node_idx).?; const if_node = ast.fullIf(tree, node_idx).?;
const then_scope = (try makeBlockScopeInternal(context, if_node.ast.then_expr)).?; const then_scope = (try makeBlockScopeInternal(context, tree, if_node.ast.then_expr)).?;
if (if_node.payload_token) |payload| { if (if_node.payload_token) |payload| {
const name_token = payload + @boolToInt(token_tags[payload] == .asterisk); const name_token = payload + @boolToInt(token_tags[payload] == .asterisk);
std.debug.assert(token_tags[name_token] == .identifier); std.debug.assert(token_tags[name_token] == .identifier);
const name = tree.tokenSlice(name_token); const name = tree.tokenSlice(name_token);
try scopes.items(.decls)[then_scope].put( try context.putDecl(
allocator, then_scope,
name, name,
.{ .pointer_payload = .{ .name = name_token, .condition = if_node.ast.cond_expr } }, .{ .pointer_payload = .{ .name = name_token, .condition = if_node.ast.cond_expr } },
); );
} }
if (if_node.ast.else_expr != 0) { if (if_node.ast.else_expr != 0) {
const else_scope = (try makeBlockScopeInternal(context, if_node.ast.else_expr)).?; const else_scope = (try makeBlockScopeInternal(context, tree, if_node.ast.else_expr)).?;
if (if_node.error_token) |err_token| { if (if_node.error_token) |err_token| {
const name = tree.tokenSlice(err_token); const name = tree.tokenSlice(err_token);
try scopes.items(.decls)[else_scope].put(allocator, name, .{ .ast_node = if_node.ast.else_expr }); try context.putDecl(else_scope, name, .{ .ast_node = if_node.ast.else_expr });
} }
} }
}, },
.@"catch" => { .@"catch" => {
try makeScopeInternal(context, data[node_idx].lhs); try makeScopeInternal(context, tree, data[node_idx].lhs);
const expr_scope = (try makeBlockScopeInternal(context, data[node_idx].rhs)).?; const expr_scope = (try makeBlockScopeInternal(context, tree, data[node_idx].rhs)).?;
const catch_token = main_tokens[node_idx] + 2; const catch_token = main_tokens[node_idx] + 2;
if (token_tags.len > catch_token and if (token_tags.len > catch_token and
@ -2929,7 +2932,7 @@ fn makeScopeInternal(context: ScopeContext, node_idx: Ast.Node.Index) error{OutO
token_tags[catch_token] == .identifier) token_tags[catch_token] == .identifier)
{ {
const name = tree.tokenSlice(catch_token); const name = tree.tokenSlice(catch_token);
try scopes.items(.decls)[expr_scope].put(allocator, name, .{ .ast_node = data[node_idx].rhs }); try context.putDecl(expr_scope, name, .{ .ast_node = data[node_idx].rhs });
} }
}, },
.@"while", .@"while",
@ -2939,27 +2942,19 @@ fn makeScopeInternal(context: ScopeContext, node_idx: Ast.Node.Index) error{OutO
// label_token: inline_token while (cond_expr) |payload_token| : (cont_expr) then_expr else else_expr // label_token: inline_token while (cond_expr) |payload_token| : (cont_expr) then_expr else else_expr
const while_node = ast.fullWhile(tree, node_idx).?; const while_node = ast.fullWhile(tree, node_idx).?;
try makeScopeInternal(context, while_node.ast.cond_expr); try makeScopeInternal(context, tree, while_node.ast.cond_expr);
const cont_scope = try makeBlockScopeInternal(context, while_node.ast.cont_expr); const cont_scope = try makeBlockScopeInternal(context, tree, while_node.ast.cont_expr);
const then_scope = (try makeBlockScopeInternal(context, while_node.ast.then_expr)).?; const then_scope = (try makeBlockScopeInternal(context, tree, while_node.ast.then_expr)).?;
const else_scope = try makeBlockScopeInternal(context, while_node.ast.else_expr); const else_scope = try makeBlockScopeInternal(context, tree, while_node.ast.else_expr);
if (while_node.label_token) |label| { if (while_node.label_token) |label| {
std.debug.assert(token_tags[label] == .identifier); std.debug.assert(token_tags[label] == .identifier);
const name = tree.tokenSlice(label); const name = tree.tokenSlice(label);
try scopes.items(.decls)[then_scope].put( try context.putDecl(then_scope, name, .{ .label_decl = .{ .label = label, .block = while_node.ast.then_expr } });
allocator,
name,
.{ .label_decl = .{ .label = label, .block = while_node.ast.then_expr } },
);
if (else_scope) |index| { if (else_scope) |index| {
try scopes.items(.decls)[index].put( try context.putDecl(index, name, .{ .label_decl = .{ .label = label, .block = while_node.ast.else_expr } });
allocator,
name,
.{ .label_decl = .{ .label = label, .block = while_node.ast.else_expr } },
);
} }
} }
@ -2975,15 +2970,15 @@ fn makeScopeInternal(context: ScopeContext, node_idx: Ast.Node.Index) error{OutO
}, },
}; };
if (cont_scope) |index| { if (cont_scope) |index| {
try scopes.items(.decls)[index].put(allocator, name, decl); try context.putDecl(index, name, decl);
} }
try scopes.items(.decls)[then_scope].put(allocator, name, decl); try context.putDecl(then_scope, name, decl);
} }
if (while_node.error_token) |err_token| { if (while_node.error_token) |err_token| {
std.debug.assert(token_tags[err_token] == .identifier); std.debug.assert(token_tags[err_token] == .identifier);
const name = tree.tokenSlice(err_token); const name = tree.tokenSlice(err_token);
try scopes.items(.decls)[else_scope.?].put(allocator, name, .{ .ast_node = while_node.ast.else_expr }); try context.putDecl(else_scope.?, name, .{ .ast_node = while_node.ast.else_expr });
} }
}, },
.@"for", .@"for",
@ -2993,11 +2988,11 @@ fn makeScopeInternal(context: ScopeContext, node_idx: Ast.Node.Index) error{OutO
const for_node = ast.fullFor(tree, node_idx).?; const for_node = ast.fullFor(tree, node_idx).?;
for (for_node.ast.inputs) |input_node| { for (for_node.ast.inputs) |input_node| {
try makeScopeInternal(context, input_node); try makeScopeInternal(context, tree, input_node);
} }
const then_scope = (try makeBlockScopeInternal(context, for_node.ast.then_expr)).?; const then_scope = (try makeBlockScopeInternal(context, tree, for_node.ast.then_expr)).?;
const else_scope = try makeBlockScopeInternal(context, for_node.ast.else_expr); const else_scope = try makeBlockScopeInternal(context, tree, for_node.ast.else_expr);
var capture_token = for_node.payload_token; var capture_token = for_node.payload_token;
for (for_node.ast.inputs) |input| { for (for_node.ast.inputs) |input| {
@ -3006,8 +3001,8 @@ fn makeScopeInternal(context: ScopeContext, node_idx: Ast.Node.Index) error{OutO
const name_token = capture_token + @boolToInt(capture_is_ref); const name_token = capture_token + @boolToInt(capture_is_ref);
capture_token = name_token + 2; capture_token = name_token + 2;
try scopes.items(.decls)[then_scope].put( try context.putDecl(
allocator, then_scope,
offsets.tokenToSlice(tree, name_token), offsets.tokenToSlice(tree, name_token),
.{ .array_payload = .{ .identifier = name_token, .array_expr = input } }, .{ .array_payload = .{ .identifier = name_token, .array_expr = input } },
); );
@ -3017,14 +3012,14 @@ fn makeScopeInternal(context: ScopeContext, node_idx: Ast.Node.Index) error{OutO
std.debug.assert(token_tags[label] == .identifier); std.debug.assert(token_tags[label] == .identifier);
const name = tree.tokenSlice(label); const name = tree.tokenSlice(label);
try scopes.items(.decls)[then_scope].put( try context.putDecl(
allocator, then_scope,
name, name,
.{ .label_decl = .{ .label = label, .block = for_node.ast.then_expr } }, .{ .label_decl = .{ .label = label, .block = for_node.ast.then_expr } },
); );
if (else_scope) |index| { if (else_scope) |index| {
try scopes.items(.decls)[index].put( try context.putDecl(
allocator, index,
name, name,
.{ .label_decl = .{ .label = label, .block = for_node.ast.else_expr } }, .{ .label_decl = .{ .label = label, .block = for_node.ast.else_expr } },
); );
@ -3042,258 +3037,32 @@ fn makeScopeInternal(context: ScopeContext, node_idx: Ast.Node.Index) error{OutO
const switch_case: Ast.full.SwitchCase = tree.fullSwitchCase(case).?; const switch_case: Ast.full.SwitchCase = tree.fullSwitchCase(case).?;
if (switch_case.payload_token) |payload| { if (switch_case.payload_token) |payload| {
const expr_index = (try makeBlockScopeInternal(context, switch_case.ast.target_expr)).?; const expr_index = (try makeBlockScopeInternal(context, tree, switch_case.ast.target_expr)).?;
// if payload is *name than get next token // if payload is *name than get next token
const name_token = payload + @boolToInt(token_tags[payload] == .asterisk); const name_token = payload + @boolToInt(token_tags[payload] == .asterisk);
const name = tree.tokenSlice(name_token); const name = tree.tokenSlice(name_token);
try scopes.items(.decls)[expr_index].put(allocator, name, .{ try context.putDecl(expr_index, name, .{
.switch_payload = .{ .switch_payload = .{ .node = name_token, .switch_expr = cond, .items = switch_case.ast.values },
.node = name_token,
.switch_expr = cond,
.items = switch_case.ast.values,
},
}); });
try makeScopeInternal(context, switch_case.ast.target_expr); try makeScopeInternal(context, tree, switch_case.ast.target_expr);
} else { } else {
try makeScopeInternal(context, switch_case.ast.target_expr); try makeScopeInternal(context, tree, switch_case.ast.target_expr);
} }
} }
}, },
.global_var_decl,
.local_var_decl,
.aligned_var_decl,
.simple_var_decl,
=> {
const var_decl = tree.fullVarDecl(node_idx).?;
if (var_decl.ast.type_node != 0) {
try makeScopeInternal(context, var_decl.ast.type_node);
}
if (var_decl.ast.init_node != 0) {
try makeScopeInternal(context, var_decl.ast.init_node);
}
},
.call,
.call_comma,
.call_one,
.call_one_comma,
.async_call,
.async_call_comma,
.async_call_one,
.async_call_one_comma,
=> {
var buf: [1]Ast.Node.Index = undefined;
const call = tree.fullCall(&buf, node_idx).?;
try makeScopeInternal(context, call.ast.fn_expr);
for (call.ast.params) |param|
try makeScopeInternal(context, param);
},
.struct_init,
.struct_init_comma,
.struct_init_dot,
.struct_init_dot_comma,
.struct_init_dot_two,
.struct_init_dot_two_comma,
.struct_init_one,
.struct_init_one_comma,
=> {
var buf: [2]Ast.Node.Index = undefined;
const struct_init: Ast.full.StructInit = tree.fullStructInit(&buf, node_idx).?;
if (struct_init.ast.type_expr != 0)
try makeScopeInternal(context, struct_init.ast.type_expr);
for (struct_init.ast.fields) |field| {
try makeScopeInternal(context, field);
}
},
.array_init,
.array_init_comma,
.array_init_dot,
.array_init_dot_comma,
.array_init_dot_two,
.array_init_dot_two_comma,
.array_init_one,
.array_init_one_comma,
=> {
var buf: [2]Ast.Node.Index = undefined;
const array_init: Ast.full.ArrayInit = tree.fullArrayInit(&buf, node_idx).?;
if (array_init.ast.type_expr != 0)
try makeScopeInternal(context, array_init.ast.type_expr);
for (array_init.ast.elements) |elem| {
try makeScopeInternal(context, elem);
}
},
.container_field,
.container_field_align,
.container_field_init,
=> {
const field = tree.fullContainerField(node_idx).?;
try makeScopeInternal(context, field.ast.type_expr);
try makeScopeInternal(context, field.ast.align_expr);
try makeScopeInternal(context, field.ast.value_expr);
},
.builtin_call,
.builtin_call_comma,
.builtin_call_two,
.builtin_call_two_comma,
=> {
var buffer: [2]Ast.Node.Index = undefined;
const params = ast.builtinCallParams(tree, node_idx, &buffer).?;
for (params) |param| {
try makeScopeInternal(context, param);
}
},
.ptr_type,
.ptr_type_aligned,
.ptr_type_bit_range,
.ptr_type_sentinel,
=> {
const ptr_type: Ast.full.PtrType = ast.fullPtrType(tree, node_idx).?;
try makeScopeInternal(context, ptr_type.ast.sentinel);
try makeScopeInternal(context, ptr_type.ast.align_node);
try makeScopeInternal(context, ptr_type.ast.child_type);
},
.array_type_sentinel => {
const array_type: Ast.full.ArrayType = tree.fullArrayType(node_idx).?;
try makeScopeInternal(context, array_type.ast.elem_count);
try makeScopeInternal(context, array_type.ast.elem_type);
try makeScopeInternal(context, array_type.ast.sentinel);
},
.slice,
.slice_open,
.slice_sentinel,
=> {
const slice: Ast.full.Slice = tree.fullSlice(node_idx).?;
try makeScopeInternal(context, slice.ast.sliced);
try makeScopeInternal(context, slice.ast.start);
try makeScopeInternal(context, slice.ast.end);
try makeScopeInternal(context, slice.ast.sentinel);
},
.@"errdefer" => { .@"errdefer" => {
const expr_scope = (try makeBlockScopeInternal(context, data[node_idx].rhs)).?; const expr_scope = (try makeBlockScopeInternal(context, tree, data[node_idx].rhs)).?;
const payload_token = data[node_idx].lhs; const payload_token = data[node_idx].lhs;
if (payload_token != 0) { if (payload_token != 0) {
const name = tree.tokenSlice(payload_token); const name = tree.tokenSlice(payload_token);
try scopes.items(.decls)[expr_scope].putNoClobber(allocator, name, .{ .ast_node = data[node_idx].rhs }); try context.putDecl(expr_scope, name, .{ .ast_node = data[node_idx].rhs });
} }
}, },
else => {
.switch_case, try ast.iterateChildren(tree, node_idx, context, error{OutOfMemory}, makeScopeInternal);
.switch_case_inline,
.switch_case_one,
.switch_case_inline_one,
.@"asm",
.asm_simple,
.asm_output,
.asm_input,
.error_value,
.multiline_string_literal,
.string_literal,
.enum_literal,
.identifier,
.anyframe_literal,
.char_literal,
.number_literal,
.unreachable_literal,
.@"continue",
=> {},
.test_decl,
.@"break",
.@"defer",
.anyframe_type,
=> {
try makeScopeInternal(context, data[node_idx].rhs);
},
.@"return",
.@"resume",
.field_access,
.@"suspend",
.deref,
.@"try",
.@"await",
.optional_type,
.@"comptime",
.@"nosuspend",
.bool_not,
.negation,
.bit_not,
.negation_wrap,
.address_of,
.grouped_expression,
.unwrap_optional,
.@"usingnamespace",
=> {
try makeScopeInternal(context, data[node_idx].lhs);
},
.equal_equal,
.bang_equal,
.less_than,
.greater_than,
.less_or_equal,
.greater_or_equal,
.assign_mul,
.assign_div,
.assign_mod,
.assign_add,
.assign_sub,
.assign_shl,
.assign_shr,
.assign_bit_and,
.assign_bit_xor,
.assign_bit_or,
.assign_mul_wrap,
.assign_add_wrap,
.assign_sub_wrap,
.assign_mul_sat,
.assign_add_sat,
.assign_sub_sat,
.assign_shl_sat,
.assign,
.merge_error_sets,
.mul,
.div,
.mod,
.array_mult,
.mul_wrap,
.mul_sat,
.add,
.sub,
.array_cat,
.add_wrap,
.sub_wrap,
.add_sat,
.sub_sat,
.shl,
.shl_sat,
.shr,
.bit_and,
.bit_xor,
.bit_or,
.@"orelse",
.bool_and,
.bool_or,
.array_type,
.array_access,
.error_union,
.for_range,
.switch_range,
=> {
try makeScopeInternal(context, data[node_idx].lhs);
try makeScopeInternal(context, data[node_idx].rhs);
}, },
} }
} }

View File

@ -300,6 +300,16 @@ test "completion - enum" {
.{ .label = "alpha", .kind = .Enum }, .{ .label = "alpha", .kind = .Enum },
.{ .label = "beta", .kind = .Enum }, .{ .label = "beta", .kind = .Enum },
}); });
try testCompletion(
\\const E = enum {
\\ alpha,
\\ beta,
\\};
\\const foo: E = .<cursor>
, &.{
.{ .label = "alpha", .kind = .Enum },
.{ .label = "beta", .kind = .Enum },
});
} }
test "completion - error union" { test "completion - error union" {
@ -463,6 +473,16 @@ test "completion - usingnamespace" {
// TODO private should not be visible // TODO private should not be visible
.{ .label = "private", .kind = .Function, .detail = "fn private() !void" }, .{ .label = "private", .kind = .Function, .detail = "fn private() !void" },
}); });
try testCompletion(
\\const S1 = struct {
\\ usingnamespace struct {
\\ pub fn inner() void {}
\\ };
\\};
\\const foo = S1.<cursor>
, &.{
.{ .label = "inner", .kind = .Function, .detail = "fn inner() void" },
});
} }
test "completion - block" { test "completion - block" {