core/odin: Added new file tag syntax as token. parse_file stores a list of tags that the file tag parser can use later.

This commit is contained in:
Karl Zylinski
2024-09-14 17:59:50 +02:00
parent c24e18bf10
commit b12d312408
6 changed files with 81 additions and 16 deletions
+1
View File
@@ -69,6 +69,7 @@ File :: struct {
fullpath: string,
src: string,
tags: [dynamic]tokenizer.Token,
docs: ^Comment_Group,
pkg_decl: ^Package_Decl,
+29 -10
View File
@@ -51,7 +51,7 @@ get_build_arch_from_string :: proc(str: string) -> runtime.Odin_Arch_Type {
parse_file_tags :: proc(file: ast.File, allocator := context.allocator) -> (tags: File_Tags) {
context.allocator = allocator
if file.docs == nil {
if file.docs == nil && file.tags == nil {
return
}
@@ -95,11 +95,9 @@ parse_file_tags :: proc(file: ast.File, allocator := context.allocator) -> (tags
build_project_names: [dynamic][]string
defer shrink(&build_project_names)
for comment in file.docs.list {
if len(comment.text) < 3 || comment.text[:2] != "//" {
continue
}
text := comment.text[2:]
parse_tag :: proc(text: string, tags: ^File_Tags, build_kinds: ^[dynamic]Build_Kind,
build_project_name_strings: ^[dynamic]string,
build_project_names: ^[dynamic][]string) {
i := 0
skip_whitespace(text, &i)
@@ -124,7 +122,7 @@ parse_file_tags :: proc(file: ast.File, allocator := context.allocator) -> (tags
groups_loop: for {
index_start := len(build_project_name_strings)
defer append(&build_project_names, build_project_name_strings[index_start:])
defer append(build_project_names, build_project_name_strings[index_start:])
for {
skip_whitespace(text, &i)
@@ -143,10 +141,10 @@ parse_file_tags :: proc(file: ast.File, allocator := context.allocator) -> (tags
}
scan_value(text, &i)
append(&build_project_name_strings, text[name_start:i])
append(build_project_name_strings, text[name_start:i])
}
append(&build_project_names, build_project_name_strings[index_start:])
append(build_project_names, build_project_name_strings[index_start:])
}
case "build":
kinds_loop: for {
@@ -156,7 +154,7 @@ parse_file_tags :: proc(file: ast.File, allocator := context.allocator) -> (tags
arch_positive: runtime.Odin_Arch_Types
arch_negative: runtime.Odin_Arch_Types
defer append(&build_kinds, Build_Kind{
defer append(build_kinds, Build_Kind{
os = (os_positive == {} ? runtime.ALL_ODIN_OS_TYPES : os_positive) -os_negative,
arch = (arch_positive == {} ? runtime.ALL_ODIN_ARCH_TYPES : arch_positive)-arch_negative,
})
@@ -200,6 +198,27 @@ parse_file_tags :: proc(file: ast.File, allocator := context.allocator) -> (tags
}
}
if file.docs != nil {
for comment in file.docs.list {
if len(comment.text) < 3 || comment.text[:2] != "//" {
continue
}
text := comment.text[2:]
parse_tag(text, &tags, &build_kinds, &build_project_name_strings, &build_project_names)
}
}
for tag in file.tags {
if len(tag.text) < 3 || tag.text[:2] != "#+" {
continue
}
// Only skip # because parse_tag skips the plus
text := tag.text[1:]
parse_tag(text, &tags, &build_kinds, &build_project_name_strings, &build_project_names)
}
tags.build = build_kinds[:]
tags.build_project_name = build_project_names[:]
+1 -3
View File
@@ -77,9 +77,7 @@ parse_package :: proc(pkg: ^ast.Package, p: ^Parser = nil) -> bool {
if !parse_file(p, file) {
ok = false
}
if file.pkg_decl == nil {
error(p, p.curr_tok.pos, "Expected a package declaration at the start of the file")
} else if pkg.name == "" {
if pkg.name == "" {
pkg.name = file.pkg_decl.name
} else if pkg.name != file.pkg_decl.name {
error(p, file.pkg_decl.pos, "different package name, expected '%s', got '%s'", pkg.name, file.pkg_decl.name)
+28 -3
View File
@@ -161,11 +161,36 @@ parse_file :: proc(p: ^Parser, file: ^ast.File) -> bool {
docs := p.lead_comment
p.file.pkg_token = expect_token(p, .Package)
if p.file.pkg_token.kind != .Package {
return false
invalid_pre_package_token: Maybe(tokenizer.Token)
for p.curr_tok.kind != .Package && p.curr_tok.kind != .EOF {
if p.curr_tok.kind == .Comment {
consume_comment_groups(p, p.prev_tok)
} else if p.curr_tok.kind == .File_Tag {
append(&p.file.tags, p.curr_tok)
advance_token(p)
} else {
if invalid_pre_package_token == nil {
invalid_pre_package_token = p.curr_tok
}
advance_token(p)
}
}
if p.curr_tok.kind != .Package {
t := invalid_pre_package_token.? or_else p.curr_tok
error(p, t.pos, "Expected a package declaration at the start of the file")
return false
}
p.file.pkg_token = expect_token(p, .Package)
if ippt, ok := invalid_pre_package_token.?; ok {
error(p, ippt.pos, "Expected only comments or lines starting with '#+' before the package declaration")
return false
}
pkg_name := expect_token_after(p, .Ident, "package")
if pkg_name.kind == .Ident {
switch name := pkg_name.text; {
+2
View File
@@ -32,6 +32,7 @@ Token_Kind :: enum u32 {
Invalid,
EOF,
Comment,
File_Tag,
B_Literal_Begin,
Ident, // main
@@ -166,6 +167,7 @@ tokens := [Token_Kind.COUNT]string {
"Invalid",
"EOF",
"Comment",
"FileTag",
"",
"identifier",
+20
View File
@@ -206,6 +206,23 @@ scan_comment :: proc(t: ^Tokenizer) -> string {
return string(lit)
}
scan_file_tag :: proc(t: ^Tokenizer) -> string {
offset := t.offset - 1
for t.ch != '\n' {
if t.ch == '/' {
next := peek_byte(t, 0)
if next == '/' || next == '*' {
break
}
}
advance_rune(t)
}
return string(t.src[offset : t.offset])
}
scan_identifier :: proc(t: ^Tokenizer) -> string {
offset := t.offset
@@ -636,6 +653,9 @@ scan :: proc(t: ^Tokenizer) -> Token {
if t.ch == '!' {
kind = .Comment
lit = scan_comment(t)
} else if t.ch == '+' {
kind = .File_Tag
lit = scan_file_tag(t)
}
case '/':
kind = .Quo