add decltests to the autodoc backend

This commit is contained in:
Jacob G-W 2022-02-18 17:39:41 -05:00 committed by Andrew Kelley
parent 5d4c88c741
commit a0ff7c8078

View File

@ -252,6 +252,8 @@ const DocData = struct {
src: usize, // index into astNodes
// typeRef: TypeRef,
value: WalkResult,
// The index in astNodes of the `test declname { }` node
decltest: ?usize = null,
};
const AstNode = struct {
@ -610,11 +612,7 @@ fn walkInstruction(
//return self.walkRef(file, parent_scope, break_operand);
//},
.block_inline => {
const pl_node = data[inst_index].pl_node;
const extra = file.zir.extraData(Zir.Inst.Block, pl_node.payload_index);
const break_index = file.zir.extra[extra.end..][extra.data.body_len - 1];
const break_operand = data[break_index].@"break".operand;
return self.walkRef(file, parent_scope, break_operand);
return self.walkRef(file, parent_scope, getBlockInlineBreak(file.zir, inst_index));
},
.func => {
const fn_info = file.zir.getFnInfo(@intCast(u32, inst_index));
@ -947,13 +945,17 @@ fn walkInstruction(
// Done to make sure that all decl refs can be resolved correctly,
// even if we haven't fully analyzed the decl yet.
{
var actual_decls_len: usize = 0;
var it = file.zir.declIterator(@intCast(u32, inst_index));
try self.decls.resize(self.arena, decls_first_index + it.decls_len);
var decls_slot_index = decls_first_index;
while (it.next()) |d| : (decls_slot_index += 1) {
const decl_name_index = file.zir.extra[d.sub_index + 5];
if (decl_name_index == 2) continue; // we don't do decltests here
actual_decls_len += 1;
try scope.insertDeclRef(self.arena, decl_name_index, decls_slot_index);
}
// we don't count decltests in our decls
try self.decls.resize(self.arena, decls_first_index + actual_decls_len);
}
extra_index = try self.walkDecls(
@ -1042,7 +1044,7 @@ fn walkDecls(
// const hash_u32s = file.zir.extra[extra_index..][0..4];
extra_index += 4;
// const line = file.zir.extra[extra_index];
const line = file.zir.extra[extra_index];
extra_index += 1;
const decl_name_index = file.zir.extra[extra_index];
extra_index += 1;
@ -1075,6 +1077,74 @@ fn walkDecls(
break :blk if (is_exported) "usingnamespace" else "comptime";
} else if (decl_name_index == 1) {
break :blk "test";
} else if (decl_name_index == 2) {
// it is a decltest
const decl_being_tested = scope.resolveDeclName(doc_comment_index);
const ast_node_index = idx: {
const idx = self.ast_nodes.items.len;
const file_source = file.getSource(self.module.gpa) catch unreachable; // TODO fix this
const source_of_decltest_function = srcloc: {
const func_index = getBlockInlineBreak(file.zir, decl_index);
// a decltest is always a function
const tag = file.zir.instructions.items(.tag)[Zir.refToIndex(func_index).?];
std.debug.assert(tag == .extended);
const extended = file.zir.instructions.items(.data)[Zir.refToIndex(func_index).?].extended;
const extra = file.zir.extraData(Zir.Inst.ExtendedFunc, extended.operand);
const small = @bitCast(Zir.Inst.ExtendedFunc.Small, extended.small);
var extra_index_for_this_func: usize = extra.end;
if (small.has_lib_name) extra_index_for_this_func += 1;
if (small.has_cc) extra_index_for_this_func += 1;
if (small.has_align) extra_index_for_this_func += 1;
const ret_ty_body = file.zir.extra[extra_index_for_this_func..][0..extra.data.ret_body_len];
extra_index_for_this_func += ret_ty_body.len;
const body = file.zir.extra[extra_index_for_this_func..][0..extra.data.body_len];
extra_index_for_this_func += body.len;
var src_locs: Zir.Inst.Func.SrcLocs = undefined;
if (body.len != 0) {
src_locs = file.zir.extraData(Zir.Inst.Func.SrcLocs, extra_index_for_this_func).data;
} else {
src_locs = .{
.lbrace_line = line,
.rbrace_line = line,
.columns = 0, // TODO get columns when body.len == 0
};
}
break :srcloc src_locs;
};
const source_slice = slice: {
var start_byte_offset: u32 = 0;
var end_byte_offset: u32 = 0;
const rbrace_col = @truncate(u16, source_of_decltest_function.columns >> 16);
var lines: u32 = 0;
for (file_source.bytes) |b, i| {
if (b == '\n') {
lines += 1;
}
if (lines == source_of_decltest_function.lbrace_line) {
start_byte_offset = @intCast(u32, i);
}
if (lines == source_of_decltest_function.rbrace_line) {
end_byte_offset = @intCast(u32, i) + rbrace_col;
break;
}
}
break :slice file_source.bytes[start_byte_offset..end_byte_offset];
};
try self.ast_nodes.append(self.arena, .{
.file = 0,
.line = line,
.col = 0,
.name = try self.arena.dupe(u8, source_slice),
});
break :idx idx;
};
self.decls.items[decl_being_tested].decltest = ast_node_index;
continue;
} else {
const raw_decl_name = file.zir.nullTerminatedString(decl_name_index);
if (raw_decl_name.len == 0) {
@ -1095,7 +1165,7 @@ fn walkDecls(
const idx = self.ast_nodes.items.len;
try self.ast_nodes.append(self.arena, .{
.file = 0,
.line = 0,
.line = line,
.col = 0,
.docs = doc_comment,
.fields = null, // walkInstruction will fill `fields` if necessary
@ -1388,3 +1458,10 @@ fn walkResultToTypeRef(wr: DocData.WalkResult) DocData.TypeRef {
//fn collectParamInfo(self: *Autodoc, file: *File, scope: *Scope, inst_idx: Zir.Index) void {
//}
fn getBlockInlineBreak(zir: Zir, inst_index: usize) Zir.Inst.Ref {
const data = zir.instructions.items(.data);
const pl_node = data[inst_index].pl_node;
const extra = zir.extraData(Zir.Inst.Block, pl_node.payload_index);
const break_index = zir.extra[extra.end..][extra.data.body_len - 1];
return data[break_index].@"break".operand;
}