Merge remote-tracking branch 'upstream/master'
This commit is contained in:
commit
2c5cc2b48f
@ -1,4 +1,4 @@
|
|||||||
![Zig Language Server](https://raw.githubusercontent.com/SuperAuguste/zls/master/.assets/zls.svg)
|
![Zig Language Server](https://raw.githubusercontent.com/zigtools/zls/master/.assets/zls.svg)
|
||||||
|
|
||||||
![CI](https://github.com/zigtools/zls/workflows/CI/badge.svg)
|
![CI](https://github.com/zigtools/zls/workflows/CI/badge.svg)
|
||||||
![Zig Tools](https://img.shields.io/static/v1?label=zigtools&message=for%20all%20of%20ziguanity&color=F7A41D&logo=)
|
![Zig Tools](https://img.shields.io/static/v1?label=zigtools&message=for%20all%20of%20ziguanity&color=F7A41D&logo=)
|
||||||
@ -20,7 +20,7 @@ Zig Language Server, or `zls`, is a language server for Zig. The Zig wiki states
|
|||||||
Installing `zls` is pretty simple;
|
Installing `zls` is pretty simple;
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
git clone https://github.com/SuperAuguste/zls
|
git clone https://github.com/zigtools/zls
|
||||||
cd zls
|
cd zls
|
||||||
zig build
|
zig build
|
||||||
```
|
```
|
||||||
@ -33,6 +33,8 @@ zig build
|
|||||||
|
|
||||||
Then, you can use the `zls` executable in an editor of your choice that has a Zig language server client!
|
Then, you can use the `zls` executable in an editor of your choice that has a Zig language server client!
|
||||||
|
|
||||||
|
*Note:`zls` itself must be built using the master branch of zig currently due to a bug in `std.json` which was [fixed](https://github.com/ziglang/zig/pull/5167) after 0.6.0 was released.*
|
||||||
|
|
||||||
### Configuration Options
|
### Configuration Options
|
||||||
|
|
||||||
You can configure zls by providing a zls.json file in the same directory as the executable.
|
You can configure zls by providing a zls.json file in the same directory as the executable.
|
||||||
@ -42,6 +44,7 @@ The following options are currently available.
|
|||||||
| --- | --- | --- | --- |
|
| --- | --- | --- | --- |
|
||||||
| `enable_snippets` | `bool` | `true` | Enables snippet completion, set to false for compatibility with language clients that do not support snippets (such as ale). |
|
| `enable_snippets` | `bool` | `true` | Enables snippet completion, set to false for compatibility with language clients that do not support snippets (such as ale). |
|
||||||
| `zig_lib_path` | `?[]const u8` | `null` | zig library path, used to analyze std library imports. |
|
| `zig_lib_path` | `?[]const u8` | `null` | zig library path, used to analyze std library imports. |
|
||||||
|
| `warn_style` | `bool` | `false` | Enables warnings for style *guideline* mismatches |
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
|
192
src/analysis.zig
192
src/analysis.zig
@ -79,6 +79,7 @@ pub fn getFunctionSignature(tree: *ast.Tree, func: *ast.Node.FnProto) []const u8
|
|||||||
const start = tree.tokens.at(func.firstToken()).start;
|
const start = tree.tokens.at(func.firstToken()).start;
|
||||||
const end = tree.tokens.at(switch (func.return_type) {
|
const end = tree.tokens.at(switch (func.return_type) {
|
||||||
.Explicit, .InferErrorSet => |node| node.lastToken(),
|
.Explicit, .InferErrorSet => |node| node.lastToken(),
|
||||||
|
.Invalid => |r_paren| r_paren,
|
||||||
}).end;
|
}).end;
|
||||||
return tree.source[start..end];
|
return tree.source[start..end];
|
||||||
}
|
}
|
||||||
@ -118,13 +119,12 @@ pub fn getFunctionSnippet(allocator: *std.mem.Allocator, tree: *ast.Tree, func:
|
|||||||
try buffer.appendSlice(": ");
|
try buffer.appendSlice(": ");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (param_decl.var_args_token) |_| {
|
switch (param_decl.param_type) {
|
||||||
try buffer.appendSlice("...");
|
.var_args => try buffer.appendSlice("..."),
|
||||||
continue;
|
.var_type => try buffer.appendSlice("var"),
|
||||||
}
|
.type_expr => |type_expr| {
|
||||||
|
var curr_tok = type_expr.firstToken();
|
||||||
var curr_tok = param_decl.type_node.firstToken();
|
var end_tok = type_expr.lastToken();
|
||||||
var end_tok = param_decl.type_node.lastToken();
|
|
||||||
while (curr_tok <= end_tok) : (curr_tok += 1) {
|
while (curr_tok <= end_tok) : (curr_tok += 1) {
|
||||||
const id = tree.tokens.at(curr_tok).id;
|
const id = tree.tokens.at(curr_tok).id;
|
||||||
const is_comma = tree.tokens.at(curr_tok).id == .Comma;
|
const is_comma = tree.tokens.at(curr_tok).id == .Comma;
|
||||||
@ -134,6 +134,8 @@ pub fn getFunctionSnippet(allocator: *std.mem.Allocator, tree: *ast.Tree, func:
|
|||||||
try buffer.appendSlice(tree.tokenSlice(curr_tok));
|
try buffer.appendSlice(tree.tokenSlice(curr_tok));
|
||||||
if (is_comma or id == .Keyword_const) try buffer.append(' ');
|
if (is_comma or id == .Keyword_const) try buffer.append(' ');
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
try buffer.append('}');
|
try buffer.append('}');
|
||||||
}
|
}
|
||||||
@ -146,12 +148,26 @@ pub fn getFunctionSnippet(allocator: *std.mem.Allocator, tree: *ast.Tree, func:
|
|||||||
pub fn getVariableSignature(tree: *ast.Tree, var_decl: *ast.Node.VarDecl) []const u8 {
|
pub fn getVariableSignature(tree: *ast.Tree, var_decl: *ast.Node.VarDecl) []const u8 {
|
||||||
const start = tree.tokens.at(var_decl.firstToken()).start;
|
const start = tree.tokens.at(var_decl.firstToken()).start;
|
||||||
const end = tree.tokens.at(var_decl.semicolon_token).start;
|
const end = tree.tokens.at(var_decl.semicolon_token).start;
|
||||||
// var end =
|
|
||||||
// if (var_decl.init_n) |body| tree.tokens.at(body.firstToken()).start
|
|
||||||
// else tree.tokens.at(var_decl.name_token).end;
|
|
||||||
return tree.source[start..end];
|
return tree.source[start..end];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Gets a param signature
|
||||||
|
pub fn getParamSignature(tree: *ast.Tree, param: *ast.Node.ParamDecl) []const u8 {
|
||||||
|
const start = tree.tokens.at(param.firstToken()).start;
|
||||||
|
const end = tree.tokens.at(param.lastToken()).end;
|
||||||
|
return tree.source[start..end];
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn isTypeFunction(tree: *ast.Tree, func: *ast.Node.FnProto) bool {
|
||||||
|
switch (func.return_type) {
|
||||||
|
.Explicit => |node| return if (node.cast(std.zig.ast.Node.Identifier)) |ident|
|
||||||
|
std.mem.eql(u8, tree.tokenSlice(ident.token), "type")
|
||||||
|
else
|
||||||
|
false,
|
||||||
|
.InferErrorSet, .Invalid => return false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// STYLE
|
// STYLE
|
||||||
|
|
||||||
pub fn isCamelCase(name: []const u8) bool {
|
pub fn isCamelCase(name: []const u8) bool {
|
||||||
@ -188,20 +204,59 @@ pub fn getChild(tree: *ast.Tree, node: *ast.Node, name: []const u8) ?*ast.Node {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Gets the child of slice
|
||||||
|
pub fn getChildOfSlice(tree: *ast.Tree, nodes: []*ast.Node, name: []const u8) ?*ast.Node {
|
||||||
|
// var index: usize = 0;
|
||||||
|
for (nodes) |child| {
|
||||||
|
switch (child.id) {
|
||||||
|
.VarDecl => {
|
||||||
|
const vari = child.cast(ast.Node.VarDecl).?;
|
||||||
|
if (std.mem.eql(u8, tree.tokenSlice(vari.name_token), name)) return child;
|
||||||
|
},
|
||||||
|
.ParamDecl => {
|
||||||
|
const decl = child.cast(ast.Node.ParamDecl).?;
|
||||||
|
if (decl.name_token != null and std.mem.eql(u8, tree.tokenSlice(decl.name_token.?), name)) return child;
|
||||||
|
},
|
||||||
|
.FnProto => {
|
||||||
|
const func = child.cast(ast.Node.FnProto).?;
|
||||||
|
if (func.name_token != null and std.mem.eql(u8, tree.tokenSlice(func.name_token.?), name)) return child;
|
||||||
|
},
|
||||||
|
.ContainerField => {
|
||||||
|
const field = child.cast(ast.Node.ContainerField).?;
|
||||||
|
if (std.mem.eql(u8, tree.tokenSlice(field.name_token), name)) return child;
|
||||||
|
},
|
||||||
|
else => {},
|
||||||
|
}
|
||||||
|
// index += 1;
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
/// Resolves the type of a node
|
/// Resolves the type of a node
|
||||||
pub fn resolveTypeOfNode(analysis_ctx: *AnalysisContext, node: *ast.Node) ?*ast.Node {
|
pub fn resolveTypeOfNode(analysis_ctx: *AnalysisContext, node: *ast.Node) ?*ast.Node {
|
||||||
std.debug.warn("Resolving node of type {}\n", .{node.id});
|
std.debug.warn("NODE {}\n", .{node});
|
||||||
switch (node.id) {
|
switch (node.id) {
|
||||||
.VarDecl => {
|
.VarDecl => {
|
||||||
const vari = node.cast(ast.Node.VarDecl).?;
|
const vari = node.cast(ast.Node.VarDecl).?;
|
||||||
|
|
||||||
return resolveTypeOfNode(analysis_ctx, vari.type_node orelse vari.init_node.?) orelse null;
|
return resolveTypeOfNode(analysis_ctx, vari.type_node orelse vari.init_node.?) orelse null;
|
||||||
},
|
},
|
||||||
|
.ParamDecl => {
|
||||||
|
const decl = node.cast(ast.Node.ParamDecl).?;
|
||||||
|
switch (decl.param_type) {
|
||||||
|
.var_type, .type_expr => |var_type| {
|
||||||
|
return resolveTypeOfNode(analysis_ctx, var_type) orelse null;
|
||||||
|
},
|
||||||
|
else => {},
|
||||||
|
}
|
||||||
|
},
|
||||||
.FnProto => {
|
.FnProto => {
|
||||||
return node;
|
return node;
|
||||||
},
|
},
|
||||||
.Identifier => {
|
.Identifier => {
|
||||||
if (getChild(analysis_ctx.tree, &analysis_ctx.tree.root_node.base, analysis_ctx.tree.getNodeSource(node))) |child| {
|
// std.debug.warn("IDENTIFIER {}\n", .{analysis_ctx.tree.getNodeSource(node)});
|
||||||
|
if (getChildOfSlice(analysis_ctx.tree, analysis_ctx.scope_nodes, analysis_ctx.tree.getNodeSource(node))) |child| {
|
||||||
|
// std.debug.warn("CHILD {}\n", .{child});
|
||||||
return resolveTypeOfNode(analysis_ctx, child);
|
return resolveTypeOfNode(analysis_ctx, child);
|
||||||
} else return null;
|
} else return null;
|
||||||
},
|
},
|
||||||
@ -218,13 +273,14 @@ pub fn resolveTypeOfNode(analysis_ctx: *AnalysisContext, node: *ast.Node) ?*ast.
|
|||||||
.SuffixOp => {
|
.SuffixOp => {
|
||||||
const suffix_op = node.cast(ast.Node.SuffixOp).?;
|
const suffix_op = node.cast(ast.Node.SuffixOp).?;
|
||||||
switch (suffix_op.op) {
|
switch (suffix_op.op) {
|
||||||
.Call => {
|
.Call, .StructInitializer => {
|
||||||
const func_decl = resolveTypeOfNode(analysis_ctx, suffix_op.lhs.node) orelse return null;
|
const func_decl = resolveTypeOfNode(analysis_ctx, suffix_op.lhs.node) orelse return null;
|
||||||
|
|
||||||
if (func_decl.id == .FnProto) {
|
if (func_decl.id == .FnProto) {
|
||||||
const func = node.cast(ast.Node.FnProto).?;
|
const func = node.cast(ast.Node.FnProto).?;
|
||||||
switch (func.return_type) {
|
switch (func.return_type) {
|
||||||
.Explicit, .InferErrorSet => |return_type| return resolveTypeOfNode(analysis_ctx, return_type),
|
.Explicit, .InferErrorSet => |return_type| return resolveTypeOfNode(analysis_ctx, return_type),
|
||||||
|
.Invalid => {},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
@ -246,15 +302,7 @@ pub fn resolveTypeOfNode(analysis_ctx: *AnalysisContext, node: *ast.Node) ?*ast.
|
|||||||
// Use the analysis context temporary arena to store the rhs string.
|
// Use the analysis context temporary arena to store the rhs string.
|
||||||
rhs_str = std.mem.dupe(&analysis_ctx.arena.allocator, u8, rhs_str) catch return null;
|
rhs_str = std.mem.dupe(&analysis_ctx.arena.allocator, u8, rhs_str) catch return null;
|
||||||
const left = resolveTypeOfNode(analysis_ctx, infix_op.lhs) orelse return null;
|
const left = resolveTypeOfNode(analysis_ctx, infix_op.lhs) orelse return null;
|
||||||
std.debug.warn("InfixOp left = {}\n", .{left});
|
return resolveTypeOfNode(analysis_ctx, getChild(analysis_ctx.tree, left, rhs_str) orelse return null);
|
||||||
const child = getChild(analysis_ctx.tree, left, rhs_str) orelse return null;
|
|
||||||
std.debug.warn("InfixOp child = {}\n", .{child});
|
|
||||||
|
|
||||||
const right_type = resolveTypeOfNode(analysis_ctx, child);
|
|
||||||
|
|
||||||
std.debug.warn("InfixOp rightType = {}\n", .{right_type});
|
|
||||||
|
|
||||||
return right_type;
|
|
||||||
},
|
},
|
||||||
else => {},
|
else => {},
|
||||||
}
|
}
|
||||||
@ -262,8 +310,14 @@ pub fn resolveTypeOfNode(analysis_ctx: *AnalysisContext, node: *ast.Node) ?*ast.
|
|||||||
.PrefixOp => {
|
.PrefixOp => {
|
||||||
const prefix_op = node.cast(ast.Node.PrefixOp).?;
|
const prefix_op = node.cast(ast.Node.PrefixOp).?;
|
||||||
switch (prefix_op.op) {
|
switch (prefix_op.op) {
|
||||||
|
.SliceType, .ArrayType => return node,
|
||||||
.PtrType => {
|
.PtrType => {
|
||||||
return resolveTypeOfNode(analysis_ctx, prefix_op.rhs);
|
const op_token = analysis_ctx.tree.tokens.at(prefix_op.op_token);
|
||||||
|
switch (op_token.id) {
|
||||||
|
.Asterisk => return resolveTypeOfNode(analysis_ctx, prefix_op.rhs),
|
||||||
|
.LBracket, .AsteriskAsterisk => return null,
|
||||||
|
else => unreachable,
|
||||||
|
}
|
||||||
},
|
},
|
||||||
else => {},
|
else => {},
|
||||||
}
|
}
|
||||||
@ -279,10 +333,13 @@ pub fn resolveTypeOfNode(analysis_ctx: *AnalysisContext, node: *ast.Node) ?*ast.
|
|||||||
|
|
||||||
const import_str = analysis_ctx.tree.tokenSlice(import_param.cast(ast.Node.StringLiteral).?.token);
|
const import_str = analysis_ctx.tree.tokenSlice(import_param.cast(ast.Node.StringLiteral).?.token);
|
||||||
return analysis_ctx.onImport(import_str[1 .. import_str.len - 1]) catch |err| block: {
|
return analysis_ctx.onImport(import_str[1 .. import_str.len - 1]) catch |err| block: {
|
||||||
std.debug.warn("Error {} while proessing import {}\n", .{ err, import_str });
|
std.debug.warn("Error {} while processing import {}\n", .{ err, import_str });
|
||||||
break :block null;
|
break :block null;
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
.MultilineStringLiteral, .StringLiteral => {
|
||||||
|
return node;
|
||||||
|
},
|
||||||
else => {
|
else => {
|
||||||
std.debug.warn("Type resolution case not implemented; {}\n", .{node.id});
|
std.debug.warn("Type resolution case not implemented; {}\n", .{node.id});
|
||||||
},
|
},
|
||||||
@ -347,7 +404,7 @@ pub fn getFieldAccessTypeNode(analysis_ctx: *AnalysisContext, tokenizer: *std.zi
|
|||||||
.Identifier => {
|
.Identifier => {
|
||||||
// var root = current_node.cast(ast.Node.Root).?;
|
// var root = current_node.cast(ast.Node.Root).?;
|
||||||
// current_node.
|
// current_node.
|
||||||
if (getChild(analysis_ctx.tree, current_node, tokenizer.buffer[next.start..next.end])) |child| {
|
if (getChildOfSlice(analysis_ctx.tree, analysis_ctx.scope_nodes, tokenizer.buffer[next.start..next.end])) |child| {
|
||||||
if (resolveTypeOfNode(analysis_ctx, child)) |node_type| {
|
if (resolveTypeOfNode(analysis_ctx, child)) |node_type| {
|
||||||
current_node = node_type;
|
current_node = node_type;
|
||||||
} else return null;
|
} else return null;
|
||||||
@ -378,25 +435,14 @@ pub fn isNodePublic(tree: *ast.Tree, node: *ast.Node) bool {
|
|||||||
switch (node.id) {
|
switch (node.id) {
|
||||||
.VarDecl => {
|
.VarDecl => {
|
||||||
const var_decl = node.cast(ast.Node.VarDecl).?;
|
const var_decl = node.cast(ast.Node.VarDecl).?;
|
||||||
if (var_decl.visib_token) |visib_token| {
|
return var_decl.visib_token != null;
|
||||||
return std.mem.eql(u8, tree.tokenSlice(visib_token), "pub");
|
|
||||||
} else return false;
|
|
||||||
},
|
},
|
||||||
.FnProto => {
|
.FnProto => {
|
||||||
const func = node.cast(ast.Node.FnProto).?;
|
const func = node.cast(ast.Node.FnProto).?;
|
||||||
if (func.visib_token) |visib_token| {
|
return func.visib_token != null;
|
||||||
return std.mem.eql(u8, tree.tokenSlice(visib_token), "pub");
|
|
||||||
} else return false;
|
|
||||||
},
|
|
||||||
.ContainerField => {
|
|
||||||
return true;
|
|
||||||
},
|
|
||||||
else => {
|
|
||||||
return false;
|
|
||||||
},
|
},
|
||||||
|
else => return true,
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn nodeToString(tree: *ast.Tree, node: *ast.Node) ?[]const u8 {
|
pub fn nodeToString(tree: *ast.Tree, node: *ast.Node) ?[]const u8 {
|
||||||
@ -426,3 +472,71 @@ pub fn nodeToString(tree: *ast.Tree, node: *ast.Node) ?[]const u8 {
|
|||||||
|
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn declsFromIndexInternal(allocator: *std.mem.Allocator, tree: *ast.Tree, node: *ast.Node, nodes: *std.ArrayList(*ast.Node)) anyerror!void {
|
||||||
|
switch (node.id) {
|
||||||
|
.FnProto => {
|
||||||
|
const func = node.cast(ast.Node.FnProto).?;
|
||||||
|
|
||||||
|
var param_index: usize = 0;
|
||||||
|
while (param_index < func.params.len) : (param_index += 1)
|
||||||
|
try declsFromIndexInternal(allocator, tree, func.params.at(param_index).*, nodes);
|
||||||
|
|
||||||
|
if (func.body_node) |body_node|
|
||||||
|
try declsFromIndexInternal(allocator, tree, body_node, nodes);
|
||||||
|
},
|
||||||
|
.Block => {
|
||||||
|
var index: usize = 0;
|
||||||
|
|
||||||
|
while (node.iterate(index)) |inode| {
|
||||||
|
try declsFromIndexInternal(allocator, tree, inode, nodes);
|
||||||
|
index += 1;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
.VarDecl => {
|
||||||
|
try nodes.append(node);
|
||||||
|
},
|
||||||
|
.ParamDecl => {
|
||||||
|
try nodes.append(node);
|
||||||
|
},
|
||||||
|
else => {
|
||||||
|
try nodes.appendSlice(try getCompletionsFromNode(allocator, tree, node));
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn getCompletionsFromNode(allocator: *std.mem.Allocator, tree: *ast.Tree, node: *ast.Node) ![]*ast.Node {
|
||||||
|
var nodes = std.ArrayList(*ast.Node).init(allocator);
|
||||||
|
|
||||||
|
var index: usize = 0;
|
||||||
|
while (node.iterate(index)) |child_node| {
|
||||||
|
try nodes.append(child_node);
|
||||||
|
|
||||||
|
index += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return nodes.items;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn declsFromIndex(allocator: *std.mem.Allocator, tree: *ast.Tree, index: usize) ![]*ast.Node {
|
||||||
|
var iindex: usize = 0;
|
||||||
|
|
||||||
|
var node = &tree.root_node.base;
|
||||||
|
var nodes = std.ArrayList(*ast.Node).init(allocator);
|
||||||
|
|
||||||
|
try nodes.appendSlice(try getCompletionsFromNode(allocator, tree, node));
|
||||||
|
|
||||||
|
while (node.iterate(iindex)) |inode| {
|
||||||
|
if (tree.tokens.at(inode.firstToken()).start < index and index < tree.tokens.at(inode.lastToken()).start) {
|
||||||
|
try declsFromIndexInternal(allocator, tree, inode, &nodes);
|
||||||
|
}
|
||||||
|
|
||||||
|
iindex += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tree.tokens.at(node.firstToken()).start < index and index < tree.tokens.at(node.lastToken()).start) {
|
||||||
|
return nodes.items;
|
||||||
|
}
|
||||||
|
|
||||||
|
return nodes.items;
|
||||||
|
}
|
||||||
|
@ -5,3 +5,7 @@ enable_snippets: bool = true,
|
|||||||
|
|
||||||
/// zig library path
|
/// zig library path
|
||||||
zig_lib_path: ?[]const u8 = null,
|
zig_lib_path: ?[]const u8 = null,
|
||||||
|
|
||||||
|
/// Whether to pay attention to style issues. This is opt-in since the style
|
||||||
|
/// guide explicitly states that the style info provided is a guideline only.
|
||||||
|
warn_style: bool = false,
|
||||||
|
124
src/debug_allocator.zig
Normal file
124
src/debug_allocator.zig
Normal file
@ -0,0 +1,124 @@
|
|||||||
|
//! This allocator collects information about allocation sizes
|
||||||
|
|
||||||
|
const std = @import("std");
|
||||||
|
|
||||||
|
const DebugAllocator = @This();
|
||||||
|
|
||||||
|
fn toMB(value: var) f64 {
|
||||||
|
return switch (@TypeOf(value)) {
|
||||||
|
f64 => value / (1024 * 1024),
|
||||||
|
else => @intToFloat(f64, value) / (1024 * 1024),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const Stats = struct {
|
||||||
|
mean: f64 = 0,
|
||||||
|
mean_of_squares: f64 = 0,
|
||||||
|
total: usize = 0,
|
||||||
|
count: usize = 0,
|
||||||
|
|
||||||
|
fn addSample(self: *Stats, value: usize) void {
|
||||||
|
const count_f64 = @intToFloat(f64, self.count);
|
||||||
|
self.mean = (self.mean * count_f64 + @intToFloat(f64, value)) / (count_f64 + 1);
|
||||||
|
self.mean_of_squares = (self.mean_of_squares * count_f64 + @intToFloat(f64, value * value)) / (count_f64 + 1);
|
||||||
|
self.total += value;
|
||||||
|
self.count += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn stdDev(self: Stats) f64 {
|
||||||
|
return std.math.sqrt(self.mean_of_squares - self.mean * self.mean);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
pub const AllocationInfo = struct {
|
||||||
|
allocation_stats: Stats = Stats{},
|
||||||
|
deallocation_count: usize = 0,
|
||||||
|
deallocation_total: usize = 0,
|
||||||
|
|
||||||
|
reallocation_stats: Stats = Stats{},
|
||||||
|
shrink_stats: Stats = Stats{},
|
||||||
|
|
||||||
|
pub fn format(
|
||||||
|
self: AllocationInfo,
|
||||||
|
comptime fmt: []const u8,
|
||||||
|
options: std.fmt.FormatOptions,
|
||||||
|
out_stream: var,
|
||||||
|
) !void {
|
||||||
|
@setEvalBranchQuota(2000);
|
||||||
|
|
||||||
|
// TODO: Make these behave like {Bi}, which doesnt work on floating point numbers.
|
||||||
|
return std.fmt.format(
|
||||||
|
out_stream,
|
||||||
|
\\------------------------------------------ Allocation info ------------------------------------------
|
||||||
|
\\{} total allocations (total: {d:.2} MB, mean: {d:.2} MB, std. dev: {d:.2} MB), {} deallocations
|
||||||
|
\\{} current allocations ({d:.2} MB)
|
||||||
|
\\{} reallocations (total: {d:.2} MB, mean: {d:.2} MB, std. dev: {d:.2} MB)
|
||||||
|
\\{} shrinks (total: {d:.2} MB, mean: {d:.2} MB, std. dev: {d:.2} MB)
|
||||||
|
\\-----------------------------------------------------------------------------------------------------
|
||||||
|
,
|
||||||
|
.{
|
||||||
|
self.allocation_stats.count,
|
||||||
|
toMB(self.allocation_stats.total),
|
||||||
|
toMB(self.allocation_stats.mean),
|
||||||
|
toMB(self.allocation_stats.stdDev()),
|
||||||
|
self.deallocation_count,
|
||||||
|
self.allocation_stats.count - self.deallocation_count,
|
||||||
|
toMB(self.allocation_stats.total + self.reallocation_stats.total - self.deallocation_total - self.shrink_stats.total),
|
||||||
|
self.reallocation_stats.count,
|
||||||
|
toMB(self.reallocation_stats.total),
|
||||||
|
toMB(self.reallocation_stats.mean),
|
||||||
|
toMB(self.reallocation_stats.stdDev()),
|
||||||
|
self.shrink_stats.count,
|
||||||
|
toMB(self.shrink_stats.total),
|
||||||
|
toMB(self.shrink_stats.mean),
|
||||||
|
toMB(self.shrink_stats.stdDev()),
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
base_allocator: *std.mem.Allocator,
|
||||||
|
info: AllocationInfo,
|
||||||
|
|
||||||
|
// Interface implementation
|
||||||
|
allocator: std.mem.Allocator,
|
||||||
|
|
||||||
|
pub fn init(base_allocator: *std.mem.Allocator) DebugAllocator {
|
||||||
|
return .{
|
||||||
|
.base_allocator = base_allocator,
|
||||||
|
.info = .{},
|
||||||
|
.allocator = .{
|
||||||
|
.reallocFn = realloc,
|
||||||
|
.shrinkFn = shrink,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
fn realloc(allocator: *std.mem.Allocator, old_mem: []u8, old_align: u29, new_size: usize, new_align: u29) ![]u8 {
|
||||||
|
const self = @fieldParentPtr(DebugAllocator, "allocator", allocator);
|
||||||
|
var data = try self.base_allocator.reallocFn(self.base_allocator, old_mem, old_align, new_size, new_align);
|
||||||
|
if (old_mem.len == 0) {
|
||||||
|
self.info.allocation_stats.addSample(new_size);
|
||||||
|
} else if (new_size > old_mem.len) {
|
||||||
|
self.info.reallocation_stats.addSample(new_size - old_mem.len);
|
||||||
|
} else if (new_size < old_mem.len) {
|
||||||
|
self.info.shrink_stats.addSample(old_mem.len - new_size);
|
||||||
|
}
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn shrink(allocator: *std.mem.Allocator, old_mem: []u8, old_align: u29, new_size: usize, new_align: u29) []u8 {
|
||||||
|
const self = @fieldParentPtr(DebugAllocator, "allocator", allocator);
|
||||||
|
if (new_size == 0) {
|
||||||
|
if (self.info.allocation_stats.count == self.info.deallocation_count) {
|
||||||
|
@panic("error - too many calls to free, most likely double free");
|
||||||
|
}
|
||||||
|
self.info.deallocation_total += old_mem.len;
|
||||||
|
self.info.deallocation_count += 1;
|
||||||
|
} else if (new_size < old_mem.len) {
|
||||||
|
self.info.shrink_stats.addSample(old_mem.len - new_size);
|
||||||
|
} else if (new_size > old_mem.len) {
|
||||||
|
@panic("error - trying to shrink to a bigger size");
|
||||||
|
}
|
||||||
|
return self.base_allocator.shrinkFn(self.base_allocator, old_mem, old_align, new_size, new_align);
|
||||||
|
}
|
@ -14,24 +14,10 @@ pub const Handle = struct {
|
|||||||
return handle.document.uri;
|
return handle.document.uri;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns the zig AST resulting from parsing the document's text, even
|
/// Returns a zig AST, with all its errors.
|
||||||
/// if it contains errors.
|
pub fn tree(handle: Handle, allocator: *std.mem.Allocator) !*std.zig.ast.Tree {
|
||||||
pub fn dirtyTree(handle: Handle, allocator: *std.mem.Allocator) !*std.zig.ast.Tree {
|
|
||||||
return try std.zig.parse(allocator, handle.document.text);
|
return try std.zig.parse(allocator, handle.document.text);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns a zig AST with no errors, either from the current text or
|
|
||||||
/// the stored sane text, null if no such ast exists.
|
|
||||||
pub fn saneTree(handle: Handle, allocator: *std.mem.Allocator) !?*std.zig.ast.Tree {
|
|
||||||
var tree = try std.zig.parse(allocator, handle.document.text);
|
|
||||||
if (tree.errors.len == 0) return tree;
|
|
||||||
|
|
||||||
tree.deinit();
|
|
||||||
if (handle.document.sane_text) |sane| {
|
|
||||||
return try std.zig.parse(allocator, sane);
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
allocator: *std.mem.Allocator,
|
allocator: *std.mem.Allocator,
|
||||||
@ -61,32 +47,23 @@ pub fn init(self: *DocumentStore, allocator: *std.mem.Allocator, zig_lib_path: ?
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// This function assersts the document is not open yet and takes owneship
|
/// This function asserts the document is not open yet and takes ownership
|
||||||
/// of the uri and text passed in.
|
/// of the uri and text passed in.
|
||||||
fn newDocument(self: *DocumentStore, uri: []const u8, text: []u8) !*Handle {
|
fn newDocument(self: *DocumentStore, uri: []const u8, text: []u8) !*Handle {
|
||||||
std.debug.warn("Opened document: {}\n", .{uri});
|
std.debug.warn("Opened document: {}\n", .{uri});
|
||||||
|
|
||||||
errdefer {
|
var handle = Handle{
|
||||||
self.allocator.free(uri);
|
|
||||||
self.allocator.free(text);
|
|
||||||
}
|
|
||||||
|
|
||||||
var handle = try self.allocator.create(Handle);
|
|
||||||
errdefer self.allocator.destroy(handle);
|
|
||||||
|
|
||||||
handle.* = Handle{
|
|
||||||
.count = 1,
|
.count = 1,
|
||||||
.import_uris = std.ArrayList([]const u8).init(self.allocator),
|
.import_uris = std.ArrayList([]const u8).init(self.allocator),
|
||||||
.document = .{
|
.document = .{
|
||||||
.uri = uri,
|
.uri = uri,
|
||||||
.text = text,
|
.text = text,
|
||||||
.mem = text,
|
.mem = text,
|
||||||
.sane_text = null,
|
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
try self.checkSanity(handle);
|
try self.checkSanity(&handle);
|
||||||
try self.handles.putNoClobber(uri, handle);
|
const kv = try self.handles.getOrPutValue(uri, handle);
|
||||||
return (self.handles.get(uri) orelse unreachable).value;
|
return &kv.value;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn openDocument(self: *DocumentStore, uri: []const u8, text: []const u8) !*Handle {
|
pub fn openDocument(self: *DocumentStore, uri: []const u8, text: []const u8) !*Handle {
|
||||||
@ -102,7 +79,7 @@ pub fn openDocument(self: *DocumentStore, uri: []const u8, text: []const u8) !*H
|
|||||||
const duped_uri = try std.mem.dupe(self.allocator, u8, uri);
|
const duped_uri = try std.mem.dupe(self.allocator, u8, uri);
|
||||||
errdefer self.allocator.free(duped_uri);
|
errdefer self.allocator.free(duped_uri);
|
||||||
|
|
||||||
return self.newDocument(duped_uri, duped_text);
|
return try self.newDocument(duped_uri, duped_text);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn decrementCount(self: *DocumentStore, uri: []const u8) void {
|
fn decrementCount(self: *DocumentStore, uri: []const u8) void {
|
||||||
@ -113,9 +90,6 @@ fn decrementCount(self: *DocumentStore, uri: []const u8) void {
|
|||||||
|
|
||||||
std.debug.warn("Freeing document: {}\n", .{uri});
|
std.debug.warn("Freeing document: {}\n", .{uri});
|
||||||
self.allocator.free(entry.value.document.mem);
|
self.allocator.free(entry.value.document.mem);
|
||||||
if (entry.value.document.sane_text) |sane| {
|
|
||||||
self.allocator.free(sane);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (entry.value.import_uris.items) |import_uri| {
|
for (entry.value.import_uris.items) |import_uri| {
|
||||||
self.decrementCount(import_uri);
|
self.decrementCount(import_uri);
|
||||||
@ -145,18 +119,10 @@ pub fn getHandle(self: *DocumentStore, uri: []const u8) ?*Handle {
|
|||||||
|
|
||||||
// Check if the document text is now sane, move it to sane_text if so.
|
// Check if the document text is now sane, move it to sane_text if so.
|
||||||
fn checkSanity(self: *DocumentStore, handle: *Handle) !void {
|
fn checkSanity(self: *DocumentStore, handle: *Handle) !void {
|
||||||
const dirty_tree = try handle.dirtyTree(self.allocator);
|
const tree = try handle.tree(self.allocator);
|
||||||
defer dirty_tree.deinit();
|
defer tree.deinit();
|
||||||
|
|
||||||
if (dirty_tree.errors.len > 0) return;
|
|
||||||
|
|
||||||
std.debug.warn("New sane text for document {}\n", .{handle.uri()});
|
|
||||||
if (handle.document.sane_text) |sane| {
|
|
||||||
self.allocator.free(sane);
|
|
||||||
}
|
|
||||||
|
|
||||||
handle.document.sane_text = try std.mem.dupe(self.allocator, u8, handle.document.text);
|
|
||||||
|
|
||||||
|
std.debug.warn("New text for document {}\n", .{handle.uri()});
|
||||||
// TODO: Better algorithm or data structure?
|
// TODO: Better algorithm or data structure?
|
||||||
// Removing the imports is costly since they live in an array list
|
// Removing the imports is costly since they live in an array list
|
||||||
// Perhaps we should use an AutoHashMap([]const u8, {}) ?
|
// Perhaps we should use an AutoHashMap([]const u8, {}) ?
|
||||||
@ -164,7 +130,7 @@ fn checkSanity(self: *DocumentStore, handle: *Handle) !void {
|
|||||||
// Try to detect removed imports and decrement their counts.
|
// Try to detect removed imports and decrement their counts.
|
||||||
if (handle.import_uris.items.len == 0) return;
|
if (handle.import_uris.items.len == 0) return;
|
||||||
|
|
||||||
const import_strs = try analysis.collectImports(self.allocator, dirty_tree);
|
const import_strs = try analysis.collectImports(self.allocator, tree);
|
||||||
defer self.allocator.free(import_strs);
|
defer self.allocator.free(import_strs);
|
||||||
|
|
||||||
const still_exist = try self.allocator.alloc(bool, handle.import_uris.items.len);
|
const still_exist = try self.allocator.alloc(bool, handle.import_uris.items.len);
|
||||||
@ -175,7 +141,7 @@ fn checkSanity(self: *DocumentStore, handle: *Handle) !void {
|
|||||||
}
|
}
|
||||||
|
|
||||||
for (import_strs) |str| {
|
for (import_strs) |str| {
|
||||||
const uri = (try uriFromImportStr(self, handle, str)) orelse continue;
|
const uri = (try uriFromImportStr(self, handle.*, str)) orelse continue;
|
||||||
defer self.allocator.free(uri);
|
defer self.allocator.free(uri);
|
||||||
|
|
||||||
var idx: usize = 0;
|
var idx: usize = 0;
|
||||||
@ -205,7 +171,7 @@ fn checkSanity(self: *DocumentStore, handle: *Handle) !void {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn applyChanges(self: *DocumentStore, handle: *Handle, content_changes: std.json.Array) !void {
|
pub fn applyChanges(self: *DocumentStore, handle: *Handle, content_changes: std.json.Array) !void {
|
||||||
var document = &handle.document;
|
const document = &handle.document;
|
||||||
|
|
||||||
for (content_changes.items) |change| {
|
for (content_changes.items) |change| {
|
||||||
if (change.Object.getValue("range")) |range| {
|
if (change.Object.getValue("range")) |range| {
|
||||||
@ -260,7 +226,7 @@ pub fn applyChanges(self: *DocumentStore, handle: *Handle, content_changes: std.
|
|||||||
try self.checkSanity(handle);
|
try self.checkSanity(handle);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn uriFromImportStr(store: *DocumentStore, handle: *Handle, import_str: []const u8) !?[]const u8 {
|
fn uriFromImportStr(store: *DocumentStore, handle: Handle, import_str: []const u8) !?[]const u8 {
|
||||||
return if (std.mem.eql(u8, import_str, "std"))
|
return if (std.mem.eql(u8, import_str, "std"))
|
||||||
if (store.std_uri) |std_root_uri| try std.mem.dupe(store.allocator, u8, std_root_uri) else {
|
if (store.std_uri) |std_root_uri| try std.mem.dupe(store.allocator, u8, std_root_uri) else {
|
||||||
std.debug.warn("Cannot resolve std library import, path is null.\n", .{});
|
std.debug.warn("Cannot resolve std library import, path is null.\n", .{});
|
||||||
@ -289,10 +255,11 @@ pub const AnalysisContext = struct {
|
|||||||
// not for the tree allocations.
|
// not for the tree allocations.
|
||||||
arena: *std.heap.ArenaAllocator,
|
arena: *std.heap.ArenaAllocator,
|
||||||
tree: *std.zig.ast.Tree,
|
tree: *std.zig.ast.Tree,
|
||||||
|
scope_nodes: []*std.zig.ast.Node,
|
||||||
|
|
||||||
pub fn onImport(self: *AnalysisContext, import_str: []const u8) !?*std.zig.ast.Node {
|
pub fn onImport(self: *AnalysisContext, import_str: []const u8) !?*std.zig.ast.Node {
|
||||||
const allocator = self.store.allocator;
|
const allocator = self.store.allocator;
|
||||||
const final_uri = (try uriFromImportStr(self.store, self.handle, import_str)) orelse return null;
|
const final_uri = (try uriFromImportStr(self.store, self.handle.*, import_str)) orelse return null;
|
||||||
|
|
||||||
std.debug.warn("Import final URI: {}\n", .{final_uri});
|
std.debug.warn("Import final URI: {}\n", .{final_uri});
|
||||||
var consumed_final_uri = false;
|
var consumed_final_uri = false;
|
||||||
@ -305,12 +272,9 @@ pub const AnalysisContext = struct {
|
|||||||
self.handle = self.store.getHandle(final_uri) orelse return null;
|
self.handle = self.store.getHandle(final_uri) orelse return null;
|
||||||
|
|
||||||
self.tree.deinit();
|
self.tree.deinit();
|
||||||
if (try self.handle.saneTree(allocator)) |tree| {
|
self.tree = try self.handle.tree(allocator);
|
||||||
self.tree = tree;
|
|
||||||
return &self.tree.root_node.base;
|
return &self.tree.root_node.base;
|
||||||
}
|
}
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// New import.
|
// New import.
|
||||||
@ -321,12 +285,9 @@ pub const AnalysisContext = struct {
|
|||||||
self.handle = new_handle;
|
self.handle = new_handle;
|
||||||
|
|
||||||
self.tree.deinit();
|
self.tree.deinit();
|
||||||
if (try self.handle.saneTree(allocator)) |tree| {
|
self.tree = try self.handle.tree(allocator);
|
||||||
self.tree = tree;
|
|
||||||
return &self.tree.root_node.base;
|
return &self.tree.root_node.base;
|
||||||
}
|
}
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
// New document, read the file then call into openDocument.
|
// New document, read the file then call into openDocument.
|
||||||
const file_path = try URI.parse(allocator, final_uri);
|
const file_path = try URI.parse(allocator, final_uri);
|
||||||
@ -355,32 +316,31 @@ pub const AnalysisContext = struct {
|
|||||||
|
|
||||||
// Swap handles and get new tree.
|
// Swap handles and get new tree.
|
||||||
// This takes ownership of the passed uri and text.
|
// This takes ownership of the passed uri and text.
|
||||||
self.handle = try newDocument(self.store, try std.mem.dupe(allocator, u8, final_uri), file_contents);
|
const duped_final_uri = try std.mem.dupe(allocator, u8, final_uri);
|
||||||
|
errdefer allocator.free(duped_final_uri);
|
||||||
|
self.handle = try newDocument(self.store, duped_final_uri, file_contents);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Free old tree, add new one if it exists.
|
// Free old tree, add new one if it exists.
|
||||||
// If we return null, no one should access the tree.
|
// If we return null, no one should access the tree.
|
||||||
self.tree.deinit();
|
self.tree.deinit();
|
||||||
if (try self.handle.saneTree(allocator)) |tree| {
|
self.tree = try self.handle.tree(allocator);
|
||||||
self.tree = tree;
|
|
||||||
return &self.tree.root_node.base;
|
return &self.tree.root_node.base;
|
||||||
}
|
}
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn deinit(self: *AnalysisContext) void {
|
pub fn deinit(self: *AnalysisContext) void {
|
||||||
self.tree.deinit();
|
self.tree.deinit();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
pub fn analysisContext(self: *DocumentStore, handle: *Handle, arena: *std.heap.ArenaAllocator) !?AnalysisContext {
|
pub fn analysisContext(self: *DocumentStore, handle: *Handle, arena: *std.heap.ArenaAllocator, position: types.Position) !AnalysisContext {
|
||||||
const tree = (try handle.saneTree(self.allocator)) orelse return null;
|
const tree = try handle.tree(self.allocator);
|
||||||
|
|
||||||
return AnalysisContext{
|
return AnalysisContext{
|
||||||
.store = self,
|
.store = self,
|
||||||
.handle = handle,
|
.handle = handle,
|
||||||
.arena = arena,
|
.arena = arena,
|
||||||
.tree = tree,
|
.tree = tree,
|
||||||
|
.scope_nodes = try analysis.declsFromIndex(&arena.allocator, tree, try handle.document.positionToIndex(position))
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -388,9 +348,6 @@ pub fn deinit(self: *DocumentStore) void {
|
|||||||
var entry_iterator = self.handles.iterator();
|
var entry_iterator = self.handles.iterator();
|
||||||
while (entry_iterator.next()) |entry| {
|
while (entry_iterator.next()) |entry| {
|
||||||
self.allocator.free(entry.value.document.mem);
|
self.allocator.free(entry.value.document.mem);
|
||||||
if (entry.value.document.sane_text) |sane| {
|
|
||||||
self.allocator.free(sane);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (entry.value.import_uris.items) |uri| {
|
for (entry.value.import_uris.items) |uri| {
|
||||||
self.allocator.free(uri);
|
self.allocator.free(uri);
|
||||||
|
44
src/header.zig
Normal file
44
src/header.zig
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
const std = @import("std");
|
||||||
|
const mem = std.mem;
|
||||||
|
|
||||||
|
const RequestHeader = struct {
|
||||||
|
content_length: usize,
|
||||||
|
|
||||||
|
/// null implies "application/vscode-jsonrpc; charset=utf-8"
|
||||||
|
content_type: ?[]const u8,
|
||||||
|
|
||||||
|
pub fn deinit(self: @This(), allocator: *mem.Allocator) void {
|
||||||
|
if (self.content_type) |ct| allocator.free(ct);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
pub fn readRequestHeader(allocator: *mem.Allocator, instream: var) !RequestHeader {
|
||||||
|
var r = RequestHeader{
|
||||||
|
.content_length = undefined,
|
||||||
|
.content_type = null,
|
||||||
|
};
|
||||||
|
errdefer r.deinit(allocator);
|
||||||
|
|
||||||
|
var has_content_length = false;
|
||||||
|
while (true) {
|
||||||
|
const header = try instream.readUntilDelimiterAlloc(allocator, '\n', 0x100);
|
||||||
|
defer allocator.free(header);
|
||||||
|
if (header.len == 0 or header[header.len - 1] != '\r') return error.MissingCarriageReturn;
|
||||||
|
if (header.len == 1) break;
|
||||||
|
|
||||||
|
const header_name = header[0..mem.indexOf(u8, header, ": ") orelse return error.MissingColon];
|
||||||
|
const header_value = header[header_name.len + 2..header.len-1];
|
||||||
|
if (mem.eql(u8, header_name, "Content-Length")) {
|
||||||
|
if (header_value.len == 0) return error.MissingHeaderValue;
|
||||||
|
r.content_length = std.fmt.parseInt(usize, header_value, 10) catch return error.InvalidContentLength;
|
||||||
|
has_content_length = true;
|
||||||
|
} else if (mem.eql(u8, header_name, "Content-Type")) {
|
||||||
|
r.content_type = try mem.dupe(allocator, u8, header_value);
|
||||||
|
} else {
|
||||||
|
return error.UnknownHeader;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!has_content_length) return error.MissingContentLength;
|
||||||
|
|
||||||
|
return r;
|
||||||
|
}
|
231
src/main.zig
231
src/main.zig
@ -3,6 +3,8 @@ const build_options = @import("build_options");
|
|||||||
|
|
||||||
const Config = @import("config.zig");
|
const Config = @import("config.zig");
|
||||||
const DocumentStore = @import("document_store.zig");
|
const DocumentStore = @import("document_store.zig");
|
||||||
|
const DebugAllocator = @import("debug_allocator.zig");
|
||||||
|
const readRequestHeader = @import("header.zig").readRequestHeader;
|
||||||
const data = @import("data/" ++ build_options.data_version ++ ".zig");
|
const data = @import("data/" ++ build_options.data_version ++ ".zig");
|
||||||
const types = @import("types.zig");
|
const types = @import("types.zig");
|
||||||
const analysis = @import("analysis.zig");
|
const analysis = @import("analysis.zig");
|
||||||
@ -99,7 +101,7 @@ fn astLocationToRange(loc: std.zig.ast.Tree.Location) types.Range {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn publishDiagnostics(handle: DocumentStore.Handle, config: Config) !void {
|
fn publishDiagnostics(handle: DocumentStore.Handle, config: Config) !void {
|
||||||
const tree = try handle.dirtyTree(allocator);
|
const tree = try handle.tree(allocator);
|
||||||
defer tree.deinit();
|
defer tree.deinit();
|
||||||
|
|
||||||
// Use an arena for our local memory allocations.
|
// Use an arena for our local memory allocations.
|
||||||
@ -137,16 +139,11 @@ fn publishDiagnostics(handle: DocumentStore.Handle, config: Config) !void {
|
|||||||
if (is_extern)
|
if (is_extern)
|
||||||
break :blk;
|
break :blk;
|
||||||
|
|
||||||
|
if (config.warn_style) {
|
||||||
if (func.name_token) |name_token| {
|
if (func.name_token) |name_token| {
|
||||||
const loc = tree.tokenLocation(0, name_token);
|
const loc = tree.tokenLocation(0, name_token);
|
||||||
|
|
||||||
const is_type_function = switch (func.return_type) {
|
const is_type_function = analysis.isTypeFunction(tree, func);
|
||||||
.Explicit => |node| if (node.cast(std.zig.ast.Node.Identifier)) |ident|
|
|
||||||
std.mem.eql(u8, tree.tokenSlice(ident.token), "type")
|
|
||||||
else
|
|
||||||
false,
|
|
||||||
.InferErrorSet => false,
|
|
||||||
};
|
|
||||||
|
|
||||||
const func_name = tree.tokenSlice(name_token);
|
const func_name = tree.tokenSlice(name_token);
|
||||||
if (!is_type_function and !analysis.isCamelCase(func_name)) {
|
if (!is_type_function and !analysis.isCamelCase(func_name)) {
|
||||||
@ -167,6 +164,7 @@ fn publishDiagnostics(handle: DocumentStore.Handle, config: Config) !void {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
},
|
},
|
||||||
else => {},
|
else => {},
|
||||||
}
|
}
|
||||||
@ -184,8 +182,17 @@ fn publishDiagnostics(handle: DocumentStore.Handle, config: Config) !void {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
fn nodeToCompletion(alloc: *std.mem.Allocator, tree: *std.zig.ast.Tree, decl: *std.zig.ast.Node, config: Config) !?types.CompletionItem {
|
fn containerToCompletion(list: *std.ArrayList(types.CompletionItem), tree: *std.zig.ast.Tree, container: *std.zig.ast.Node, config: Config) !void {
|
||||||
var doc = if (try analysis.getDocComments(alloc, tree, decl)) |doc_comments|
|
var index: usize = 0;
|
||||||
|
while (container.iterate(index)) |child_node| : (index+=1) {
|
||||||
|
if (analysis.isNodePublic(tree, child_node)) {
|
||||||
|
try nodeToCompletion(list, tree, child_node, config);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn nodeToCompletion(list: *std.ArrayList(types.CompletionItem), tree: *std.zig.ast.Tree, node: *std.zig.ast.Node, config: Config) error{OutOfMemory}!void {
|
||||||
|
var doc = if (try analysis.getDocComments(list.allocator, tree, node)) |doc_comments|
|
||||||
types.MarkupContent{
|
types.MarkupContent{
|
||||||
.kind = .Markdown,
|
.kind = .Markdown,
|
||||||
.value = doc_comments,
|
.value = doc_comments,
|
||||||
@ -193,48 +200,78 @@ fn nodeToCompletion(alloc: *std.mem.Allocator, tree: *std.zig.ast.Tree, decl: *s
|
|||||||
else
|
else
|
||||||
null;
|
null;
|
||||||
|
|
||||||
switch (decl.id) {
|
switch (node.id) {
|
||||||
|
.ErrorSetDecl, .Root, .ContainerDecl => {
|
||||||
|
try containerToCompletion(list, tree, node, config);
|
||||||
|
},
|
||||||
.FnProto => {
|
.FnProto => {
|
||||||
const func = decl.cast(std.zig.ast.Node.FnProto).?;
|
const func = node.cast(std.zig.ast.Node.FnProto).?;
|
||||||
if (func.name_token) |name_token| {
|
if (func.name_token) |name_token| {
|
||||||
const insert_text = if (config.enable_snippets)
|
const insert_text = if (config.enable_snippets)
|
||||||
try analysis.getFunctionSnippet(alloc, tree, func)
|
try analysis.getFunctionSnippet(list.allocator, tree, func)
|
||||||
else
|
else
|
||||||
null;
|
null;
|
||||||
|
|
||||||
return types.CompletionItem{
|
const is_type_function = analysis.isTypeFunction(tree, func);
|
||||||
|
|
||||||
|
try list.append(.{
|
||||||
.label = tree.tokenSlice(name_token),
|
.label = tree.tokenSlice(name_token),
|
||||||
.kind = .Function,
|
.kind = if (is_type_function) .Struct else .Function,
|
||||||
.documentation = doc,
|
.documentation = doc,
|
||||||
.detail = analysis.getFunctionSignature(tree, func),
|
.detail = analysis.getFunctionSignature(tree, func),
|
||||||
.insertText = insert_text,
|
.insertText = insert_text,
|
||||||
.insertTextFormat = if (config.enable_snippets) .Snippet else .PlainText,
|
.insertTextFormat = if (config.enable_snippets) .Snippet else .PlainText,
|
||||||
};
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
.VarDecl => {
|
.VarDecl => {
|
||||||
const var_decl = decl.cast(std.zig.ast.Node.VarDecl).?;
|
const var_decl = node.cast(std.zig.ast.Node.VarDecl).?;
|
||||||
return types.CompletionItem{
|
const is_const = tree.tokens.at(var_decl.mut_token).id == .Keyword_const;
|
||||||
|
try list.append(.{
|
||||||
.label = tree.tokenSlice(var_decl.name_token),
|
.label = tree.tokenSlice(var_decl.name_token),
|
||||||
.kind = .Variable,
|
.kind = if (is_const) .Constant else .Variable,
|
||||||
.documentation = doc,
|
.documentation = doc,
|
||||||
.detail = analysis.getVariableSignature(tree, var_decl),
|
.detail = analysis.getVariableSignature(tree, var_decl),
|
||||||
};
|
});
|
||||||
},
|
},
|
||||||
else => if (analysis.nodeToString(tree, decl)) |string| {
|
.ParamDecl => {
|
||||||
return types.CompletionItem{
|
const param = node.cast(std.zig.ast.Node.ParamDecl).?;
|
||||||
|
if (param.name_token) |name_token|
|
||||||
|
try list.append(.{
|
||||||
|
.label = tree.tokenSlice(name_token),
|
||||||
|
.kind = .Constant,
|
||||||
|
.documentation = doc,
|
||||||
|
.detail = analysis.getParamSignature(tree, param),
|
||||||
|
});
|
||||||
|
},
|
||||||
|
.PrefixOp => {
|
||||||
|
try list.append(.{
|
||||||
|
.label = "len",
|
||||||
|
.kind = .Field,
|
||||||
|
});
|
||||||
|
try list.append(.{
|
||||||
|
.label = "ptr",
|
||||||
|
.kind = .Field,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
.StringLiteral => {
|
||||||
|
try list.append(.{
|
||||||
|
.label = "len",
|
||||||
|
.kind = .Field,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
else => if (analysis.nodeToString(tree, node)) |string| {
|
||||||
|
try list.append(.{
|
||||||
.label = string,
|
.label = string,
|
||||||
.kind = .Field,
|
.kind = .Field,
|
||||||
.documentation = doc,
|
.documentation = doc,
|
||||||
};
|
});
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
return null;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn completeGlobal(id: i64, handle: DocumentStore.Handle, config: Config) !void {
|
fn completeGlobal(id: i64, pos_index: usize, handle: DocumentStore.Handle, config: Config) !void {
|
||||||
var tree = (try handle.saneTree(allocator)) orelse return respondGeneric(id, no_completions_response);
|
var tree = try handle.tree(allocator);
|
||||||
defer tree.deinit();
|
defer tree.deinit();
|
||||||
|
|
||||||
// We use a local arena allocator to deallocate all temporary data without iterating
|
// We use a local arena allocator to deallocate all temporary data without iterating
|
||||||
@ -243,12 +280,11 @@ fn completeGlobal(id: i64, handle: DocumentStore.Handle, config: Config) !void {
|
|||||||
// Deallocate all temporary data.
|
// Deallocate all temporary data.
|
||||||
defer arena.deinit();
|
defer arena.deinit();
|
||||||
|
|
||||||
var decls = tree.root_node.decls.iterator(0);
|
// var decls = tree.root_node.decls.iterator(0);
|
||||||
while (decls.next()) |decl_ptr| {
|
var decls = try analysis.declsFromIndex(&arena.allocator, tree, pos_index);
|
||||||
|
for (decls) |decl_ptr| {
|
||||||
var decl = decl_ptr.*;
|
var decl = decl_ptr.*;
|
||||||
if (try nodeToCompletion(&arena.allocator, tree, decl, config)) |completion| {
|
try nodeToCompletion(&completions, tree, decl_ptr, config);
|
||||||
try completions.append(completion);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
try send(types.Response{
|
try send(types.Response{
|
||||||
@ -266,26 +302,15 @@ fn completeFieldAccess(id: i64, handle: *DocumentStore.Handle, position: types.P
|
|||||||
var arena = std.heap.ArenaAllocator.init(allocator);
|
var arena = std.heap.ArenaAllocator.init(allocator);
|
||||||
defer arena.deinit();
|
defer arena.deinit();
|
||||||
|
|
||||||
var analysis_ctx = (try document_store.analysisContext(handle, &arena)) orelse {
|
var analysis_ctx = try document_store.analysisContext(handle, &arena, position);
|
||||||
return send(types.Response{
|
|
||||||
.id = .{ .Integer = id },
|
|
||||||
.result = .{
|
|
||||||
.CompletionList = .{
|
|
||||||
.isIncomplete = false,
|
|
||||||
.items = &[_]types.CompletionItem{},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
|
||||||
};
|
|
||||||
defer analysis_ctx.deinit();
|
defer analysis_ctx.deinit();
|
||||||
|
|
||||||
var completions = std.ArrayList(types.CompletionItem).init(&arena.allocator);
|
var completions = std.ArrayList(types.CompletionItem).init(&arena.allocator);
|
||||||
|
|
||||||
var line = try handle.document.getLine(@intCast(usize, position.line));
|
const line = try handle.document.getLine(@intCast(usize, position.line));
|
||||||
// handle pointer could change from underneath us, so let's copy the line
|
var tokenizer = std.zig.Tokenizer.init(line[line_start_idx..]);
|
||||||
var line_copy = try std.mem.dupe(&arena.allocator, u8, line[line_start_idx..]);
|
|
||||||
var tokenizer = std.zig.Tokenizer.init(line_copy);
|
|
||||||
|
|
||||||
|
// var decls = try analysis.declsFromIndex(&arena.allocator, analysis_ctx.tree, try handle.document.positionToIndex(position));
|
||||||
if (analysis.getFieldAccessTypeNode(&analysis_ctx, &tokenizer)) |node| {
|
if (analysis.getFieldAccessTypeNode(&analysis_ctx, &tokenizer)) |node| {
|
||||||
var index: usize = 0;
|
var index: usize = 0;
|
||||||
while (node.iterate(index)) |child_node| {
|
while (node.iterate(index)) |child_node| {
|
||||||
@ -579,7 +604,7 @@ fn processJsonRpc(parser: *std.json.Parser, json: []const u8, config: Config) !v
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
.var_access, .empty => try completeGlobal(id, handle.*, config),
|
.var_access, .empty => try completeGlobal(id, pos_index, handle.*, config),
|
||||||
.field_access => |start_idx| try completeFieldAccess(id, handle, pos, start_idx, config),
|
.field_access => |start_idx| try completeFieldAccess(id, handle, pos, start_idx, config),
|
||||||
else => try respondGeneric(id, no_completions_response),
|
else => try respondGeneric(id, no_completions_response),
|
||||||
}
|
}
|
||||||
@ -607,20 +632,20 @@ fn processJsonRpc(parser: *std.json.Parser, json: []const u8, config: Config) !v
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var debug_alloc_state: std.testing.LeakCountAllocator = undefined;
|
var debug_alloc_state: DebugAllocator = undefined;
|
||||||
// We can now use if(leak_count_alloc) |alloc| { ... } as a comptime check.
|
// We can now use if(leak_count_alloc) |alloc| { ... } as a comptime check.
|
||||||
const debug_alloc: ?*std.testing.LeakCountAllocator = if (build_options.allocation_info) &debug_alloc_state else null;
|
const debug_alloc: ?*DebugAllocator = if (build_options.allocation_info) &debug_alloc_state else null;
|
||||||
|
|
||||||
pub fn main() anyerror!void {
|
pub fn main() anyerror!void {
|
||||||
// TODO: Use a better purpose general allocator once std has one.
|
// TODO: Use a better purpose general allocator once std has one.
|
||||||
// Probably after the generic composable allocators PR?
|
// Probably after the generic composable allocators PR?
|
||||||
// This is not too bad for now since most allocations happen in local areans.
|
// This is not too bad for now since most allocations happen in local arenas.
|
||||||
allocator = std.heap.page_allocator;
|
allocator = std.heap.page_allocator;
|
||||||
|
|
||||||
if (build_options.allocation_info) {
|
if (build_options.allocation_info) {
|
||||||
// TODO: Use a better debugging allocator, track size in bytes, memory reserved etc..
|
// TODO: Use a better debugging allocator, track size in bytes, memory reserved etc..
|
||||||
// Initialize the leak counting allocator.
|
// Initialize the leak counting allocator.
|
||||||
debug_alloc_state = std.testing.LeakCountAllocator.init(allocator);
|
debug_alloc_state = DebugAllocator.init(allocator);
|
||||||
allocator = &debug_alloc_state.allocator;
|
allocator = &debug_alloc_state.allocator;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -636,9 +661,10 @@ pub fn main() anyerror!void {
|
|||||||
const stdin = std.io.getStdIn().inStream();
|
const stdin = std.io.getStdIn().inStream();
|
||||||
stdout = std.io.getStdOut().outStream();
|
stdout = std.io.getStdOut().outStream();
|
||||||
|
|
||||||
// Read he configuration, if any.
|
// Read the configuration, if any.
|
||||||
var config = Config{};
|
|
||||||
const config_parse_options = std.json.ParseOptions{ .allocator = allocator };
|
const config_parse_options = std.json.ParseOptions{ .allocator = allocator };
|
||||||
|
var config = Config{};
|
||||||
|
defer std.json.parseFree(Config, config, config_parse_options);
|
||||||
|
|
||||||
// TODO: Investigate using std.fs.Watch to detect writes to the config and reload it.
|
// TODO: Investigate using std.fs.Watch to detect writes to the config and reload it.
|
||||||
config_read: {
|
config_read: {
|
||||||
@ -648,31 +674,27 @@ pub fn main() anyerror!void {
|
|||||||
var exec_dir = std.fs.cwd().openDir(exec_dir_path, .{}) catch break :config_read;
|
var exec_dir = std.fs.cwd().openDir(exec_dir_path, .{}) catch break :config_read;
|
||||||
defer exec_dir.close();
|
defer exec_dir.close();
|
||||||
|
|
||||||
var conf_file = exec_dir.openFile("zls.json", .{}) catch break :config_read;
|
const conf_file = exec_dir.openFile("zls.json", .{}) catch break :config_read;
|
||||||
defer conf_file.close();
|
defer conf_file.close();
|
||||||
|
|
||||||
const conf_file_stat = conf_file.stat() catch break :config_read;
|
// Max 1MB
|
||||||
|
const file_buf = conf_file.inStream().readAllAlloc(allocator, 0x1000000) catch break :config_read;
|
||||||
// Allocate enough memory for the whole file.
|
|
||||||
var file_buf = try allocator.alloc(u8, conf_file_stat.size);
|
|
||||||
defer allocator.free(file_buf);
|
defer allocator.free(file_buf);
|
||||||
|
|
||||||
const bytes_read = conf_file.readAll(file_buf) catch break :config_read;
|
// TODO: Better errors? Doesn't seem like std.json can provide us positions or context.
|
||||||
if (bytes_read != conf_file_stat.size) break :config_read;
|
|
||||||
|
|
||||||
// TODO: Better errors? Doesnt seem like std.json can provide us positions or context.
|
|
||||||
config = std.json.parse(Config, &std.json.TokenStream.init(file_buf), config_parse_options) catch |err| {
|
config = std.json.parse(Config, &std.json.TokenStream.init(file_buf), config_parse_options) catch |err| {
|
||||||
std.debug.warn("Error while parsing configuration file: {}\nUsing default config.\n", .{err});
|
std.debug.warn("Error while parsing configuration file: {}\nUsing default config.\n", .{err});
|
||||||
break :config_read;
|
break :config_read;
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
defer std.json.parseFree(Config, config, config_parse_options);
|
|
||||||
|
|
||||||
if (config.zig_lib_path != null and !std.fs.path.isAbsolute(config.zig_lib_path.?)) {
|
if (config.zig_lib_path) |zig_lib_path| {
|
||||||
|
if (!std.fs.path.isAbsolute(zig_lib_path)) {
|
||||||
std.debug.warn("zig library path is not absolute, defaulting to null.\n", .{});
|
std.debug.warn("zig library path is not absolute, defaulting to null.\n", .{});
|
||||||
allocator.free(config.zig_lib_path.?);
|
allocator.free(zig_lib_path);
|
||||||
config.zig_lib_path = null;
|
config.zig_lib_path = null;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
try document_store.init(allocator, config.zig_lib_path);
|
try document_store.init(allocator, config.zig_lib_path);
|
||||||
defer document_store.deinit();
|
defer document_store.deinit();
|
||||||
@ -681,77 +703,20 @@ pub fn main() anyerror!void {
|
|||||||
var json_parser = std.json.Parser.init(allocator, false);
|
var json_parser = std.json.Parser.init(allocator, false);
|
||||||
defer json_parser.deinit();
|
defer json_parser.deinit();
|
||||||
|
|
||||||
var offset: usize = 0;
|
while (true) {
|
||||||
var bytes_read: usize = 0;
|
const headers = readRequestHeader(allocator, stdin) catch |err| {
|
||||||
|
try log("{}; exiting!", .{@errorName(err)});
|
||||||
var index: usize = 0;
|
|
||||||
var content_len: usize = 0;
|
|
||||||
|
|
||||||
stdin_poll: while (true) {
|
|
||||||
if (offset >= 16 and std.mem.startsWith(u8, buffer.items, "Content-Length: ")) {
|
|
||||||
index = 16;
|
|
||||||
while (index <= offset + 10) : (index += 1) {
|
|
||||||
const c = buffer.items[index];
|
|
||||||
if (c >= '0' and c <= '9') {
|
|
||||||
content_len = content_len * 10 + (c - '0');
|
|
||||||
}
|
|
||||||
if (c == '\r' and buffer.items[index + 1] == '\n') {
|
|
||||||
index += 2;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (buffer.items[index] == '\r') {
|
|
||||||
index += 2;
|
|
||||||
if (buffer.items.len < index + content_len) {
|
|
||||||
try buffer.resize(index + content_len);
|
|
||||||
}
|
|
||||||
|
|
||||||
body_poll: while (offset < content_len + index) {
|
|
||||||
bytes_read = try stdin.readAll(buffer.items[offset .. index + content_len]);
|
|
||||||
if (bytes_read == 0) {
|
|
||||||
try log("0 bytes read; exiting!", .{});
|
|
||||||
return;
|
return;
|
||||||
}
|
};
|
||||||
|
defer headers.deinit(allocator);
|
||||||
offset += bytes_read;
|
const buf = try allocator.alloc(u8, headers.content_length);
|
||||||
}
|
defer allocator.free(buf);
|
||||||
|
try stdin.readNoEof(buf);
|
||||||
try processJsonRpc(&json_parser, buffer.items[index .. index + content_len], config);
|
try processJsonRpc(&json_parser, buf, config);
|
||||||
json_parser.reset();
|
json_parser.reset();
|
||||||
|
|
||||||
offset = 0;
|
|
||||||
content_len = 0;
|
|
||||||
} else {
|
|
||||||
try log("\\r not found", .{});
|
|
||||||
}
|
|
||||||
} else if (offset >= 16) {
|
|
||||||
try log("Offset is greater than 16!", .{});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (offset < 16) {
|
|
||||||
bytes_read = try stdin.readAll(buffer.items[offset..25]);
|
|
||||||
} else {
|
|
||||||
if (offset == buffer.items.len) {
|
|
||||||
try buffer.resize(buffer.items.len * 2);
|
|
||||||
}
|
|
||||||
if (index + content_len > buffer.items.len) {
|
|
||||||
bytes_read = try stdin.readAll(buffer.items[offset..buffer.items.len]);
|
|
||||||
} else {
|
|
||||||
bytes_read = try stdin.readAll(buffer.items[offset .. index + content_len]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (bytes_read == 0) {
|
|
||||||
try log("0 bytes read; exiting!", .{});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
offset += bytes_read;
|
|
||||||
|
|
||||||
if (debug_alloc) |dbg| {
|
if (debug_alloc) |dbg| {
|
||||||
try log("Allocations alive: {}", .{dbg.count});
|
try log("{}", .{dbg.info});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -140,7 +140,6 @@ pub const TextDocument = struct {
|
|||||||
text: String,
|
text: String,
|
||||||
// This holds the memory that we have actually allocated.
|
// This holds the memory that we have actually allocated.
|
||||||
mem: []u8,
|
mem: []u8,
|
||||||
sane_text: ?String = null,
|
|
||||||
|
|
||||||
pub fn positionToIndex(self: TextDocument, position: Position) !usize {
|
pub fn positionToIndex(self: TextDocument, position: Position) !usize {
|
||||||
var split_iterator = std.mem.split(self.text, "\n");
|
var split_iterator = std.mem.split(self.text, "\n");
|
||||||
@ -186,11 +185,11 @@ pub const MarkupKind = enum(u1) {
|
|||||||
options: json.StringifyOptions,
|
options: json.StringifyOptions,
|
||||||
out_stream: var,
|
out_stream: var,
|
||||||
) !void {
|
) !void {
|
||||||
if (@enumToInt(value) == 0) {
|
const str = switch (value) {
|
||||||
try json.stringify("plaintext", options, out_stream);
|
.PlainText => "plaintext",
|
||||||
} else {
|
.Markdown => "markdown",
|
||||||
try json.stringify("markdown", options, out_stream);
|
};
|
||||||
}
|
try json.stringify(str, options, out_stream);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -14,7 +14,7 @@ pub fn fromPath(allocator: *std.mem.Allocator, path: []const u8) ![]const u8 {
|
|||||||
var buf = std.ArrayList(u8).init(allocator);
|
var buf = std.ArrayList(u8).init(allocator);
|
||||||
try buf.appendSlice(prefix);
|
try buf.appendSlice(prefix);
|
||||||
|
|
||||||
var out_stream = buf.outStream();
|
const out_stream = buf.outStream();
|
||||||
|
|
||||||
for (path) |char| {
|
for (path) |char| {
|
||||||
if (char == std.fs.path.sep) {
|
if (char == std.fs.path.sep) {
|
||||||
@ -55,17 +55,16 @@ fn parseHex(c: u8) !u8 {
|
|||||||
pub fn parse(allocator: *std.mem.Allocator, str: []const u8) ![]u8 {
|
pub fn parse(allocator: *std.mem.Allocator, str: []const u8) ![]u8 {
|
||||||
if (str.len < 7 or !std.mem.eql(u8, "file://", str[0..7])) return error.UriBadScheme;
|
if (str.len < 7 or !std.mem.eql(u8, "file://", str[0..7])) return error.UriBadScheme;
|
||||||
|
|
||||||
var uri = try allocator.alloc(u8, str.len - (if (std.fs.path.sep == '\\') 8 else 7));
|
const uri = try allocator.alloc(u8, str.len - (if (std.fs.path.sep == '\\') 8 else 7));
|
||||||
errdefer allocator.free(uri);
|
errdefer allocator.free(uri);
|
||||||
|
|
||||||
const path = if (std.fs.path.sep == '\\') str[8..] else str[7..];
|
const path = if (std.fs.path.sep == '\\') str[8..] else str[7..];
|
||||||
|
|
||||||
var i: usize = 0;
|
var i: usize = 0;
|
||||||
var j: usize = 0;
|
var j: usize = 0;
|
||||||
var e: usize = path.len;
|
while (j < path.len) : (i += 1) {
|
||||||
while (j < e) : (i += 1) {
|
|
||||||
if (path[j] == '%') {
|
if (path[j] == '%') {
|
||||||
if (j + 2 >= e) return error.UriBadEscape;
|
if (j + 2 >= path.len) return error.UriBadEscape;
|
||||||
const upper = try parseHex(path[j + 1]);
|
const upper = try parseHex(path[j + 1]);
|
||||||
const lower = try parseHex(path[j + 2]);
|
const lower = try parseHex(path[j + 2]);
|
||||||
uri[i] = (upper << 4) + lower;
|
uri[i] = (upper << 4) + lower;
|
||||||
|
Loading…
Reference in New Issue
Block a user