v2: simplify the type system; an initial version of the V interpreter
parent
932392a7fd
commit
9f4661391d
|
@ -589,4 +589,4 @@ fn test_clear() {
|
||||||
|
|
||||||
arr.clear()
|
arr.clear()
|
||||||
assert arr.len == 0
|
assert arr.len == 0
|
||||||
}
|
}
|
||||||
|
|
|
@ -441,4 +441,4 @@ pub fn (m map_string) str() string {
|
||||||
}
|
}
|
||||||
sb.writeln('}')
|
sb.writeln('}')
|
||||||
return sb.str()
|
return sb.str()
|
||||||
}
|
}
|
||||||
|
|
|
@ -134,13 +134,13 @@ fn (v mut V) cc() {
|
||||||
}
|
}
|
||||||
|
|
||||||
if !v.pref.is_so
|
if !v.pref.is_so
|
||||||
&& v.pref.build_mode != .build_module
|
&& v.pref.build_mode != .build_module
|
||||||
&& os.user_os() == 'windows'
|
&& os.user_os() == 'windows'
|
||||||
&& !v.out_name.ends_with('.exe')
|
&& !v.out_name.ends_with('.exe')
|
||||||
{
|
{
|
||||||
v.out_name += '.exe'
|
v.out_name += '.exe'
|
||||||
}
|
}
|
||||||
|
|
||||||
// linux_host := os.user_os() == 'linux'
|
// linux_host := os.user_os() == 'linux'
|
||||||
v.log('cc() isprod=$v.pref.is_prod outname=$v.out_name')
|
v.log('cc() isprod=$v.pref.is_prod outname=$v.out_name')
|
||||||
if v.pref.is_so {
|
if v.pref.is_so {
|
||||||
|
@ -356,13 +356,11 @@ start:
|
||||||
if res.output.len < 30 {
|
if res.output.len < 30 {
|
||||||
println(res.output)
|
println(res.output)
|
||||||
} else {
|
} else {
|
||||||
max := 50
|
q := res.output.all_after('error: ').limit(150)
|
||||||
n := if res.output.len > max { max } else { res.output.len }
|
println('==================')
|
||||||
partial_output := res.output[res.output.len-n..].trim_right('\r\n')
|
println(q)
|
||||||
print(partial_output)
|
println('==================')
|
||||||
if n < max {
|
println('...\n(Use `v -cg` to print the entire error message)\n')
|
||||||
println('...\n(Use `v -cg` to print the entire error message)\n')
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
verror('C error. This should never happen. ' + '\nPlease create a GitHub issue: https://github.com/vlang/v/issues/new/choose')
|
verror('C error. This should never happen. ' + '\nPlease create a GitHub issue: https://github.com/vlang/v/issues/new/choose')
|
||||||
|
|
|
@ -63,7 +63,7 @@ fn (p mut Parser) bool_expression() string {
|
||||||
p.gen('}')
|
p.gen('}')
|
||||||
//p.satisfies_interface(expected, typ, true)
|
//p.satisfies_interface(expected, typ, true)
|
||||||
}
|
}
|
||||||
// e.g. `return BinaryExpr{}` in a function expecting `Expr`
|
// e.g. `return InfixExpr{}` in a function expecting `Expr`
|
||||||
if expected != typ && expected in p.table.sum_types { // TODO perf
|
if expected != typ && expected in p.table.sum_types { // TODO perf
|
||||||
//p.warn('SUM CAST exp=$expected typ=$typ p.exp=$p.expected_type')
|
//p.warn('SUM CAST exp=$expected typ=$typ p.exp=$p.expected_type')
|
||||||
T := p.table.find_type(typ)
|
T := p.table.find_type(typ)
|
||||||
|
@ -87,7 +87,8 @@ fn (p mut Parser) bool_expression() string {
|
||||||
if typ in p.table.sum_types {
|
if typ in p.table.sum_types {
|
||||||
T := p.table.find_type(cast_typ)
|
T := p.table.find_type(cast_typ)
|
||||||
if T.parent != typ {
|
if T.parent != typ {
|
||||||
p.error('cannot cast `$typ` to `$cast_typ`. `$cast_typ` is not a variant of `$typ`')
|
p.error('cannot cast `$typ` to `$cast_typ`. `$cast_typ` is not a variant of `$typ`' +
|
||||||
|
'parent=$T.parent')
|
||||||
}
|
}
|
||||||
p.cgen.set_placeholder(start_ph, '*($cast_typ*)')
|
p.cgen.set_placeholder(start_ph, '*($cast_typ*)')
|
||||||
p.gen('.obj')
|
p.gen('.obj')
|
||||||
|
|
|
@ -405,7 +405,6 @@ pub fn (v mut V) compile_x64() {
|
||||||
}
|
}
|
||||||
//v.files << v.v_files_from_dir(filepath.join(v.pref.vlib_path,'builtin','bare'))
|
//v.files << v.v_files_from_dir(filepath.join(v.pref.vlib_path,'builtin','bare'))
|
||||||
v.files << v.dir
|
v.files << v.dir
|
||||||
|
|
||||||
mut b := builder.new_builder()
|
mut b := builder.new_builder()
|
||||||
b.build_x64(v.files, v.out_name)
|
b.build_x64(v.files, v.out_name)
|
||||||
}
|
}
|
||||||
|
|
|
@ -32,4 +32,4 @@ fn test_eventbus(){
|
||||||
|
|
||||||
fn on_test(sender voidptr, ev &EventData) {
|
fn on_test(sender voidptr, ev &EventData) {
|
||||||
assert ev.data == "hello"
|
assert ev.data == "hello"
|
||||||
}
|
}
|
||||||
|
|
|
@ -47,4 +47,4 @@ fn C.fonsGetTextureData(s &C.FONScontext, width &int, height &int) byteptr
|
||||||
fn C.fonsValidateTexture(s &C.FONScontext, dirty &int) int
|
fn C.fonsValidateTexture(s &C.FONScontext, dirty &int) int
|
||||||
|
|
||||||
// Draws the stash texture for debugging
|
// Draws the stash texture for debugging
|
||||||
fn C.fonsDrawDebug(s &C.FONScontext, x f32, y f32)
|
fn C.fonsDrawDebug(s &C.FONScontext, x f32, y f32)
|
||||||
|
|
|
@ -77,4 +77,4 @@ pub struct C.FONStextIter {
|
||||||
|
|
||||||
pub struct C.FONSfont {}
|
pub struct C.FONSfont {}
|
||||||
|
|
||||||
pub struct C.FONScontext {}
|
pub struct C.FONScontext {}
|
||||||
|
|
|
@ -12,4 +12,4 @@ const (
|
||||||
O_APPEND = 000002000 // append data to the file when writing.
|
O_APPEND = 000002000 // append data to the file when writing.
|
||||||
O_NONBLOCK = 000004000 // prevents blocking when opening files
|
O_NONBLOCK = 000004000 // prevents blocking when opening files
|
||||||
O_SYNC = 000010000 // open for synchronous I/O.
|
O_SYNC = 000010000 // open for synchronous I/O.
|
||||||
)
|
)
|
||||||
|
|
|
@ -301,4 +301,4 @@ pub enum Action {
|
||||||
load
|
load
|
||||||
dontcare
|
dontcare
|
||||||
_num
|
_num
|
||||||
}
|
}
|
||||||
|
|
|
@ -53,4 +53,4 @@ fn C.sg_query_buffer_defaults(desc &C.sg_buffer) C.sg_buffer_desc
|
||||||
fn C.sg_query_image_defaults(desc &C.sg_image) C.sg_image_desc
|
fn C.sg_query_image_defaults(desc &C.sg_image) C.sg_image_desc
|
||||||
fn C.sg_query_shader_defaults(desc &C.sg_shader) C.sg_shader_desc
|
fn C.sg_query_shader_defaults(desc &C.sg_shader) C.sg_shader_desc
|
||||||
fn C.sg_query_pipeline_defaults(desc &C.sg_pipeline) C.sg_pipeline_desc
|
fn C.sg_query_pipeline_defaults(desc &C.sg_pipeline) C.sg_pipeline_desc
|
||||||
fn C.sg_query_pass_defaults(desc &C.sg_pass) C.sg_pass_desc
|
fn C.sg_query_pass_defaults(desc &C.sg_pass) C.sg_pass_desc
|
||||||
|
|
|
@ -457,4 +457,4 @@ pub mut:
|
||||||
// layer int
|
// layer int
|
||||||
// slice int
|
// slice int
|
||||||
// }
|
// }
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,4 +9,4 @@ pub fn create_clear_pass(r, g, b, a f32) C.sg_pass_action {
|
||||||
mut pass_action := sg_pass_action{}
|
mut pass_action := sg_pass_action{}
|
||||||
pass_action.colors[0] = color_action
|
pass_action.colors[0] = color_action
|
||||||
return pass_action
|
return pass_action
|
||||||
}
|
}
|
||||||
|
|
|
@ -155,4 +155,4 @@ pub enum KeyCode {
|
||||||
right_alt = 346
|
right_alt = 346
|
||||||
right_super = 347
|
right_super = 347
|
||||||
menu = 348
|
menu = 348
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,4 +3,4 @@ module sfons
|
||||||
fn C.sfons_create(width int, height int, flags int) &C.FONScontext
|
fn C.sfons_create(width int, height int, flags int) &C.FONScontext
|
||||||
fn C.sfons_destroy(ctx &FONScontext)
|
fn C.sfons_destroy(ctx &FONScontext)
|
||||||
fn C.sfons_rgba(r byte, g byte, b byte, a byte) u32
|
fn C.sfons_rgba(r byte, g byte, b byte, a byte) u32
|
||||||
fn C.sfons_flush(ctx &FONScontext)
|
fn C.sfons_flush(ctx &FONScontext)
|
||||||
|
|
|
@ -86,4 +86,4 @@ fn C.sgl_v3f_t2f_c1i(x f32, y f32, z f32, u f32, v f32, rgba u32)
|
||||||
fn C.sgl_end()
|
fn C.sgl_end()
|
||||||
|
|
||||||
/* render everything */
|
/* render everything */
|
||||||
fn C.sgl_draw()
|
fn C.sgl_draw()
|
||||||
|
|
|
@ -21,4 +21,4 @@ pub struct C.sgl_desc_t {
|
||||||
depth_format C.sg_pixel_format
|
depth_format C.sg_pixel_format
|
||||||
sample_count int
|
sample_count int
|
||||||
face_winding C.sg_face_winding /* default front face winding is CCW */
|
face_winding C.sg_face_winding /* default front face winding is CCW */
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,11 +5,12 @@ module ast
|
||||||
|
|
||||||
import (
|
import (
|
||||||
v.token
|
v.token
|
||||||
v.types
|
v.table
|
||||||
)
|
)
|
||||||
|
|
||||||
pub type Expr = BinaryExpr | UnaryExpr | IfExpr | StringLiteral | IntegerLiteral |
|
pub type Expr = InfixExpr | IfExpr | StringLiteral | IntegerLiteral |
|
||||||
FloatLiteral | Ident | CallExpr | BoolLiteral | StructInit | ArrayInit | SelectorExpr | PostfixExpr | AssignExpr | PrefixExpr | MethodCallExpr | IndexExpr
|
FloatLiteral | Ident | CallExpr | BoolLiteral | StructInit | ArrayInit | SelectorExpr | PostfixExpr |
|
||||||
|
AssignExpr | PrefixExpr | MethodCallExpr | IndexExpr
|
||||||
|
|
||||||
pub type Stmt = VarDecl | FnDecl | Return | Module | Import | ExprStmt |
|
pub type Stmt = VarDecl | FnDecl | Return | Module | Import | ExprStmt |
|
||||||
ForStmt | StructDecl | ForCStmt | ForInStmt
|
ForStmt | StructDecl | ForCStmt | ForInStmt
|
||||||
|
@ -18,7 +19,7 @@ ForStmt | StructDecl | ForCStmt | ForInStmt
|
||||||
pub struct ExprStmt {
|
pub struct ExprStmt {
|
||||||
pub:
|
pub:
|
||||||
expr Expr
|
expr Expr
|
||||||
ti types.TypeIdent
|
ti table.Type
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct IntegerLiteral {
|
pub struct IntegerLiteral {
|
||||||
|
@ -62,7 +63,7 @@ pub struct Field {
|
||||||
pub:
|
pub:
|
||||||
name string
|
name string
|
||||||
// type_idx int
|
// type_idx int
|
||||||
ti types.TypeIdent
|
ti table.Type
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct StructDecl {
|
pub struct StructDecl {
|
||||||
|
@ -76,7 +77,7 @@ pub:
|
||||||
pub struct StructInit {
|
pub struct StructInit {
|
||||||
pub:
|
pub:
|
||||||
pos token.Position
|
pos token.Position
|
||||||
ti types.TypeIdent
|
ti table.Type
|
||||||
fields []string
|
fields []string
|
||||||
exprs []Expr
|
exprs []Expr
|
||||||
}
|
}
|
||||||
|
@ -92,7 +93,7 @@ pub:
|
||||||
|
|
||||||
pub struct Arg {
|
pub struct Arg {
|
||||||
pub:
|
pub:
|
||||||
ti types.TypeIdent
|
ti table.Type
|
||||||
name string
|
name string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -100,7 +101,7 @@ pub struct FnDecl {
|
||||||
pub:
|
pub:
|
||||||
name string
|
name string
|
||||||
stmts []Stmt
|
stmts []Stmt
|
||||||
ti types.TypeIdent
|
ti table.Type
|
||||||
args []Arg
|
args []Arg
|
||||||
is_pub bool
|
is_pub bool
|
||||||
receiver Field
|
receiver Field
|
||||||
|
@ -108,33 +109,33 @@ pub:
|
||||||
|
|
||||||
pub struct CallExpr {
|
pub struct CallExpr {
|
||||||
pub:
|
pub:
|
||||||
// tok token.Token
|
// tok token.Token
|
||||||
pos token.Position
|
pos token.Position
|
||||||
mut:
|
mut:
|
||||||
// func Expr
|
// func Expr
|
||||||
name string
|
name string
|
||||||
args []Expr
|
args []Expr
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct MethodCallExpr {
|
pub struct MethodCallExpr {
|
||||||
pub:
|
pub:
|
||||||
// tok token.Token
|
// tok token.Token
|
||||||
pos token.Position
|
pos token.Position
|
||||||
expr Expr
|
expr Expr
|
||||||
name string
|
name string
|
||||||
args []Expr
|
args []Expr
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct Return {
|
pub struct Return {
|
||||||
pub:
|
pub:
|
||||||
pos token.Position
|
pos token.Position
|
||||||
expected_ti types.TypeIdent // TODO: remove once checker updated
|
expected_ti table.Type // TODO: remove once checker updated
|
||||||
exprs []Expr
|
exprs []Expr
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
pub enum Expr {
|
pub enum Expr {
|
||||||
Binary(BinaryExpr)
|
Binary(InfixExpr)
|
||||||
If(IfExpr)
|
If(IfExpr)
|
||||||
Integer(IntegerExpr)
|
Integer(IntegerExpr)
|
||||||
}
|
}
|
||||||
|
@ -150,12 +151,12 @@ pub struct Stmt {
|
||||||
|
|
||||||
pub struct VarDecl {
|
pub struct VarDecl {
|
||||||
pub:
|
pub:
|
||||||
name string
|
name string
|
||||||
expr Expr
|
expr Expr
|
||||||
is_mut bool
|
is_mut bool
|
||||||
mut:
|
mut:
|
||||||
ti types.TypeIdent
|
typ table.Type
|
||||||
pos token.Position
|
pos token.Position
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct File {
|
pub struct File {
|
||||||
|
@ -168,7 +169,8 @@ pub:
|
||||||
pub struct IdentVar {
|
pub struct IdentVar {
|
||||||
pub:
|
pub:
|
||||||
expr Expr
|
expr Expr
|
||||||
ti types.TypeIdent
|
typ table.Type
|
||||||
|
name string
|
||||||
}
|
}
|
||||||
|
|
||||||
type IdentInfo = IdentVar
|
type IdentInfo = IdentVar
|
||||||
|
@ -190,17 +192,19 @@ mut:
|
||||||
info IdentInfo
|
info IdentInfo
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct BinaryExpr {
|
pub struct InfixExpr {
|
||||||
pub:
|
pub:
|
||||||
// op BinaryOp
|
// op BinaryOp
|
||||||
op token.Kind
|
op token.Kind
|
||||||
pos token.Position
|
pos token.Position
|
||||||
left Expr
|
left Expr
|
||||||
// left_ti types.TypeIdent
|
left_type table.Type
|
||||||
right Expr
|
right Expr
|
||||||
// right_ti types.TypeIdent
|
right_type table.Type
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
// renamed to PrefixExpr
|
||||||
pub struct UnaryExpr {
|
pub struct UnaryExpr {
|
||||||
pub:
|
pub:
|
||||||
// tok_kind token.Kind
|
// tok_kind token.Kind
|
||||||
|
@ -208,6 +212,8 @@ pub:
|
||||||
op token.Kind
|
op token.Kind
|
||||||
left Expr
|
left Expr
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
pub struct PostfixExpr {
|
pub struct PostfixExpr {
|
||||||
pub:
|
pub:
|
||||||
|
@ -225,7 +231,7 @@ pub struct IndexExpr {
|
||||||
pub:
|
pub:
|
||||||
// op token.Kind
|
// op token.Kind
|
||||||
left Expr
|
left Expr
|
||||||
index Expr
|
index Expr // [0], [start..end] etc
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct IfExpr {
|
pub struct IfExpr {
|
||||||
|
@ -234,7 +240,7 @@ pub:
|
||||||
cond Expr
|
cond Expr
|
||||||
stmts []Stmt
|
stmts []Stmt
|
||||||
else_stmts []Stmt
|
else_stmts []Stmt
|
||||||
ti types.TypeIdent
|
ti table.Type
|
||||||
left Expr // `a` in `a := if ...`
|
left Expr // `a` in `a := if ...`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -261,7 +267,7 @@ pub:
|
||||||
|
|
||||||
pub struct ReturnStmt {
|
pub struct ReturnStmt {
|
||||||
tok_kind token.Kind // or pos
|
tok_kind token.Kind // or pos
|
||||||
pos token.Position
|
pos token.Position
|
||||||
results []Expr
|
results []Expr
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -287,18 +293,21 @@ pub struct ArrayInit {
|
||||||
pub:
|
pub:
|
||||||
pos token.Position
|
pos token.Position
|
||||||
exprs []Expr
|
exprs []Expr
|
||||||
ti types.TypeIdent
|
ti table.Type
|
||||||
}
|
}
|
||||||
|
|
||||||
// string representaiton of expr
|
// string representaiton of expr
|
||||||
pub fn (x Expr) str() string {
|
pub fn (x Expr) str() string {
|
||||||
match x {
|
match x {
|
||||||
BinaryExpr {
|
InfixExpr {
|
||||||
return '(${it.left.str()} $it.op.str() ${it.right.str()})'
|
return '(${it.left.str()} $it.op.str() ${it.right.str()})'
|
||||||
}
|
}
|
||||||
UnaryExpr {
|
/*
|
||||||
|
PrefixExpr {
|
||||||
return it.left.str() + it.op.str()
|
return it.left.str() + it.op.str()
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
IntegerLiteral {
|
IntegerLiteral {
|
||||||
return it.val.str()
|
return it.val.str()
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,14 +6,13 @@ module checker
|
||||||
import (
|
import (
|
||||||
v.ast
|
v.ast
|
||||||
v.table
|
v.table
|
||||||
v.types
|
|
||||||
v.token
|
v.token
|
||||||
)
|
)
|
||||||
|
|
||||||
pub struct Checker {
|
pub struct Checker {
|
||||||
table &table.Table
|
table &table.Table
|
||||||
mut:
|
mut:
|
||||||
file_name string
|
file_name string
|
||||||
// TODO: resolved
|
// TODO: resolved
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -36,7 +35,7 @@ pub fn (c mut Checker) check_files(v_files []string, ast_files []ast.File) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (c &Checker) check_struct_init(struct_init ast.StructInit) {
|
pub fn (c &Checker) check_struct_init(struct_init ast.StructInit) table.Type {
|
||||||
typ := c.table.find_type(struct_init.ti.name) or {
|
typ := c.table.find_type(struct_init.ti.name) or {
|
||||||
c.error('unknown struct: $struct_init.ti.name', struct_init.pos)
|
c.error('unknown struct: $struct_init.ti.name', struct_init.pos)
|
||||||
panic('')
|
panic('')
|
||||||
|
@ -46,100 +45,116 @@ pub fn (c &Checker) check_struct_init(struct_init ast.StructInit) {
|
||||||
c.error('unknown struct: $struct_init.ti.name', struct_init.pos)
|
c.error('unknown struct: $struct_init.ti.name', struct_init.pos)
|
||||||
}
|
}
|
||||||
.struct_ {
|
.struct_ {
|
||||||
info := typ.info as types.Struct
|
info := typ.info as table.Struct
|
||||||
for i, expr in struct_init.exprs {
|
for i, expr in struct_init.exprs {
|
||||||
field := info.fields[i]
|
field := info.fields[i]
|
||||||
expr_ti := c.table.get_expr_ti(expr)
|
// expr_ti := c.expr(expr)
|
||||||
if !c.table.check(expr_ti, field.ti) {
|
field_type := c.expr(expr)
|
||||||
c.error('cannot assign $expr_ti.name as $field.ti.name for field $field.name', struct_init.pos)
|
if !c.table.check(field_type, field.ti) {
|
||||||
|
c.error('cannot assign $field_type.name as $field.ti.name for field $field.name', struct_init.pos)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {}
|
else {}
|
||||||
}
|
}
|
||||||
|
return struct_init.ti
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (c &Checker) check_binary_expr(binary_expr ast.BinaryExpr) {
|
pub fn (c &Checker) infix_expr(infix_expr ast.InfixExpr) table.Type {
|
||||||
left_ti := c.table.get_expr_ti(binary_expr.left)
|
left_ti := c.expr(infix_expr.left)
|
||||||
right_ti := c.table.get_expr_ti(binary_expr.right)
|
right_ti := c.expr(infix_expr.right)
|
||||||
if !c.table.check(&right_ti, &left_ti) {
|
if !c.table.check(&right_ti, &left_ti) {
|
||||||
c.error('binary expr: cannot use $right_ti.name as $left_ti.name', binary_expr.pos)
|
// if !c.table.check(&infix_expr.right_type, &infix_expr.right_type) {
|
||||||
|
// c.error('infix expr: cannot use `$infix_expr.right_type.name` as `$infix_expr.left_type.name`', infix_expr.pos)
|
||||||
|
c.error('infix expr: cannot use `$left_ti.name` as `$right_ti.name`', infix_expr.pos)
|
||||||
}
|
}
|
||||||
|
return left_ti
|
||||||
}
|
}
|
||||||
|
|
||||||
fn (c &Checker) check_assign_expr(assign_expr ast.AssignExpr) {
|
fn (c &Checker) check_assign_expr(assign_expr ast.AssignExpr) {
|
||||||
left_ti := c.table.get_expr_ti(assign_expr.left)
|
left_ti := c.expr(assign_expr.left)
|
||||||
right_ti := c.table.get_expr_ti(assign_expr.val)
|
right_ti := c.expr(assign_expr.val)
|
||||||
if !c.table.check(right_ti, left_ti) {
|
if !c.table.check(right_ti, left_ti) {
|
||||||
c.error('cannot assign $right_ti.name to $left_ti.name', assign_expr.pos)
|
c.error('cannot assign $right_ti.name to $left_ti.name', assign_expr.pos)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (c &Checker) check_call_expr(call_expr ast.CallExpr) {
|
pub fn (c &Checker) call_expr(call_expr ast.CallExpr) table.Type {
|
||||||
fn_name := call_expr.name
|
fn_name := call_expr.name
|
||||||
if f := c.table.find_fn(fn_name) {
|
if f := c.table.find_fn(fn_name) {
|
||||||
// return_ti := f.return_ti
|
// return_ti := f.return_ti
|
||||||
if call_expr.args.len < f.args.len {
|
if call_expr.args.len < f.args.len {
|
||||||
c.error('too few arguments in call to `$fn_name`', call_expr.pos)
|
c.error('too few arguments in call to `$fn_name`', call_expr.pos)
|
||||||
} else if call_expr.args.len > f.args.len {
|
}
|
||||||
|
else if call_expr.args.len > f.args.len {
|
||||||
c.error('too many arguments in call to `$fn_name`', call_expr.pos)
|
c.error('too many arguments in call to `$fn_name`', call_expr.pos)
|
||||||
}
|
}
|
||||||
for i, arg in f.args {
|
for i, arg in f.args {
|
||||||
arg_expr := call_expr.args[i]
|
arg_expr := call_expr.args[i]
|
||||||
ti := c.table.get_expr_ti(arg_expr)
|
ti := c.expr(arg_expr)
|
||||||
if !c.table.check(&ti, &arg.ti) {
|
if !c.table.check(&ti, &arg.typ) {
|
||||||
c.error('cannot use type `$ti.name` as type `$arg.ti.name` in argument to `$fn_name`', call_expr.pos)
|
c.error('!cannot use type `$ti.name` as type `$arg.typ.name` in argument to `$fn_name`', call_expr.pos)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
return f.return_type
|
||||||
|
}else{
|
||||||
c.error('unknown fn: $fn_name', call_expr.pos)
|
c.error('unknown fn: $fn_name', call_expr.pos)
|
||||||
|
exit(0)
|
||||||
// c.warn('unknown function `$fn_name`')
|
// c.warn('unknown function `$fn_name`')
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (c &Checker) check_method_call_expr(method_call_expr ast.MethodCallExpr) {
|
pub fn (c &Checker) check_method_call_expr(method_call_expr ast.MethodCallExpr) table.Type {
|
||||||
ti := c.table.get_expr_ti(method_call_expr.expr)
|
ti := c.expr(method_call_expr.expr)
|
||||||
if !c.table.has_method(ti.idx, method_call_expr.name) {
|
if !c.table.has_method(ti.idx, method_call_expr.name) {
|
||||||
c.error('type `$ti.name` has no method `$method_call_expr.name`', method_call_expr.pos)
|
c.error('type `$ti.name` has no method `$method_call_expr.name`', method_call_expr.pos)
|
||||||
}
|
}
|
||||||
|
return ti
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (c &Checker) check_selector_expr(selector_expr ast.SelectorExpr) {
|
pub fn (c &Checker) selector_expr(selector_expr ast.SelectorExpr) table.Type {
|
||||||
ti := c.table.get_expr_ti(selector_expr.expr)
|
ti := c.expr(selector_expr.expr)
|
||||||
field_name := selector_expr.field
|
field_name := selector_expr.field
|
||||||
// struct_ := c.table.types[ti.idx] as types.Struct
|
struct_ := c.table.types[ti.idx]
|
||||||
|
// struct_info := struct_.info as table.Struct
|
||||||
typ := c.table.types[ti.idx]
|
typ := c.table.types[ti.idx]
|
||||||
match typ.kind {
|
match typ.kind {
|
||||||
.struct_ {
|
.struct_ {
|
||||||
// if !c.table.struct_has_field(it, field) {
|
// if !c.table.struct_has_field(it, field) {
|
||||||
// c.error('AAA unknown field `${it.name}.$field`')
|
// c.error('AAA unknown field `${it.name}.$field`')
|
||||||
// }
|
// }
|
||||||
// TODO: fix bug
|
// TODO: fix bug
|
||||||
c.table.struct_find_field(typ, field_name) or {
|
field := c.table.struct_find_field(typ, field_name) or {
|
||||||
c.error('unknown field `${typ.name}.$field_name`', selector_expr.pos)
|
c.error('unknown field `${typ.name}.$field_name`', selector_expr.pos)
|
||||||
|
exit(0)
|
||||||
}
|
}
|
||||||
|
return field.ti
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
c.error('$ti.name is not a struct', selector_expr.pos)
|
c.error('$ti.name is not a struct', selector_expr.pos)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return table.void_type
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: non deferred
|
// TODO: non deferred
|
||||||
pub fn (c &Checker) check_return_stmt(return_stmt ast.Return) {
|
pub fn (c &Checker) return_stmt(return_stmt ast.Return) {
|
||||||
mut got_tis := []types.TypeIdent
|
mut got_tis := []table.Type
|
||||||
|
if return_stmt.exprs.len == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
for expr in return_stmt.exprs {
|
for expr in return_stmt.exprs {
|
||||||
ti := c.table.get_expr_ti(expr)
|
ti := c.expr(expr)
|
||||||
got_tis << ti
|
got_tis << ti
|
||||||
}
|
}
|
||||||
expected_ti := return_stmt.expected_ti
|
expected_ti := return_stmt.expected_ti
|
||||||
mut expected_tis := [expected_ti]
|
mut expected_tis := [expected_ti]
|
||||||
if expected_ti.kind == .multi_return {
|
if expected_ti.kind == .multi_return {
|
||||||
mr_type := c.table.types[expected_ti.idx]
|
mr_type := c.table.types[expected_ti.idx]
|
||||||
mr_info := mr_type.info as types.MultiReturn
|
mr_info := mr_type.info as table.MultiReturn
|
||||||
expected_tis = mr_info.tis
|
expected_tis = mr_info.tis
|
||||||
}
|
}
|
||||||
if expected_tis.len != got_tis.len {
|
if expected_tis.len > 0 && expected_tis.len != got_tis.len {
|
||||||
c.error('wrong number of return arguments:\n\texpected: $expected_tis.str()\n\tgot: $got_tis.str()', return_stmt.pos)
|
c.error('wrong number of return arguments:\n\texpected: $expected_tis.str()\n\tgot: $got_tis.str()', return_stmt.pos)
|
||||||
}
|
}
|
||||||
for i, exp_ti in expected_tis {
|
for i, exp_ti in expected_tis {
|
||||||
|
@ -150,11 +165,11 @@ pub fn (c &Checker) check_return_stmt(return_stmt ast.Return) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (c &Checker) check_array_init(array_init ast.ArrayInit) {
|
pub fn (c &Checker) array_init(array_init ast.ArrayInit) table.Type {
|
||||||
mut val_ti := types.void_ti
|
mut val_ti := table.void_type
|
||||||
for i, expr in array_init.exprs {
|
for i, expr in array_init.exprs {
|
||||||
c.expr(expr)
|
c.expr(expr)
|
||||||
ti := c.table.get_expr_ti(expr)
|
ti := c.expr(expr)
|
||||||
// The first element's type
|
// The first element's type
|
||||||
if i == 0 {
|
if i == 0 {
|
||||||
val_ti = ti
|
val_ti = ti
|
||||||
|
@ -164,6 +179,7 @@ pub fn (c &Checker) check_array_init(array_init ast.ArrayInit) {
|
||||||
c.error('expected array element with type `$val_ti.name`', array_init.pos)
|
c.error('expected array element with type `$val_ti.name`', array_init.pos)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return array_init.ti
|
||||||
}
|
}
|
||||||
|
|
||||||
fn (c &Checker) stmt(node ast.Stmt) {
|
fn (c &Checker) stmt(node ast.Stmt) {
|
||||||
|
@ -174,10 +190,15 @@ fn (c &Checker) stmt(node ast.Stmt) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ast.Return {
|
ast.Return {
|
||||||
c.check_return_stmt(it)
|
c.return_stmt(it)
|
||||||
}
|
}
|
||||||
ast.VarDecl {
|
ast.VarDecl {
|
||||||
c.expr(it.expr)
|
typ := c.expr(it.expr)
|
||||||
|
println('1111var decl $typ.name it.typ=$it.typ.name $it.pos.line_nr')
|
||||||
|
if it.typ.kind == .unresolved {
|
||||||
|
// it.ti = typ
|
||||||
|
println('VAR DECL UN!!!')
|
||||||
|
}
|
||||||
}
|
}
|
||||||
ast.ForStmt {
|
ast.ForStmt {
|
||||||
c.expr(it.cond)
|
c.expr(it.cond)
|
||||||
|
@ -201,42 +222,58 @@ fn (c &Checker) stmt(node ast.Stmt) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn (c &Checker) expr(node ast.Expr) {
|
pub fn (c &Checker) expr(node ast.Expr) table.Type {
|
||||||
match node {
|
match node {
|
||||||
ast.AssignExpr {
|
ast.AssignExpr {
|
||||||
c.check_assign_expr(it)
|
c.check_assign_expr(it)
|
||||||
}
|
}
|
||||||
// ast.IntegerLiteral {}
|
ast.IntegerLiteral {
|
||||||
|
return table.int_type
|
||||||
|
}
|
||||||
// ast.FloatLiteral {}
|
// ast.FloatLiteral {}
|
||||||
ast.PostfixExpr {
|
ast.PostfixExpr {
|
||||||
c.expr(it.expr)
|
return c.expr(it.expr)
|
||||||
}
|
}
|
||||||
|
/*
|
||||||
ast.UnaryExpr {
|
ast.UnaryExpr {
|
||||||
c.expr(it.left)
|
c.expr(it.left)
|
||||||
}
|
}
|
||||||
// ast.StringLiteral {}
|
*/
|
||||||
ast.PrefixExpr {
|
|
||||||
c.expr(it.right)
|
ast.StringLiteral {
|
||||||
|
return table.string_type
|
||||||
}
|
}
|
||||||
ast.BinaryExpr {
|
ast.PrefixExpr {
|
||||||
c.check_binary_expr(it)
|
return c.expr(it.right)
|
||||||
|
}
|
||||||
|
ast.InfixExpr {
|
||||||
|
return c.infix_expr(it)
|
||||||
}
|
}
|
||||||
ast.StructInit {
|
ast.StructInit {
|
||||||
c.check_struct_init(it)
|
return c.check_struct_init(it)
|
||||||
}
|
}
|
||||||
ast.CallExpr {
|
ast.CallExpr {
|
||||||
c.check_call_expr(it)
|
return c.call_expr(it)
|
||||||
}
|
}
|
||||||
ast.MethodCallExpr {
|
ast.MethodCallExpr {
|
||||||
c.check_method_call_expr(it)
|
return c.check_method_call_expr(it)
|
||||||
}
|
}
|
||||||
ast.ArrayInit {
|
ast.ArrayInit {
|
||||||
c.check_array_init(it)
|
return c.array_init(it)
|
||||||
|
}
|
||||||
|
ast.Ident {
|
||||||
|
if it.kind == .variable {
|
||||||
|
info := it.info as ast.IdentVar
|
||||||
|
if info.typ.kind != .unresolved {
|
||||||
|
return info.typ
|
||||||
|
}
|
||||||
|
return c.expr(info.expr)
|
||||||
|
}
|
||||||
|
return table.void_type
|
||||||
}
|
}
|
||||||
// ast.Ident {}
|
|
||||||
// ast.BoolLiteral {}
|
// ast.BoolLiteral {}
|
||||||
ast.SelectorExpr {
|
ast.SelectorExpr {
|
||||||
c.check_selector_expr(it)
|
return c.selector_expr(it)
|
||||||
}
|
}
|
||||||
ast.IndexExpr {
|
ast.IndexExpr {
|
||||||
c.expr(it.left)
|
c.expr(it.left)
|
||||||
|
@ -255,6 +292,7 @@ fn (c &Checker) expr(node ast.Expr) {
|
||||||
}
|
}
|
||||||
else {}
|
else {}
|
||||||
}
|
}
|
||||||
|
return table.void_type
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (c &Checker) error(s string, pos token.Position) {
|
pub fn (c &Checker) error(s string, pos token.Position) {
|
||||||
|
@ -268,5 +306,6 @@ pub fn (c &Checker) error(s string, pos token.Position) {
|
||||||
eprintln(final_msg_line)
|
eprintln(final_msg_line)
|
||||||
}
|
}
|
||||||
*/
|
*/
|
||||||
|
|
||||||
exit(1)
|
exit(1)
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,104 @@
|
||||||
|
// Copyright (c) 2019 Alexander Medvednikov. All rights reserved.
|
||||||
|
// Use of this source code is governed by an MIT license
|
||||||
|
// that can be found in the LICENSE file.
|
||||||
|
module eval
|
||||||
|
|
||||||
|
import (
|
||||||
|
v.ast
|
||||||
|
v.checker
|
||||||
|
v.table
|
||||||
|
)
|
||||||
|
|
||||||
|
pub type Object = int | string
|
||||||
|
|
||||||
|
pub struct Eval {
|
||||||
|
mut:
|
||||||
|
checker checker.Checker
|
||||||
|
vars map[string]Var
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct Var {
|
||||||
|
value Object
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn (e mut Eval) eval(file ast.File, table &table.Table) string {
|
||||||
|
mut res := ''
|
||||||
|
e.checker = checker.new_checker(table)
|
||||||
|
for stmt in file.stmts {
|
||||||
|
res += e.stmt(stmt) + '\n'
|
||||||
|
}
|
||||||
|
return res.trim_space()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn print_object(o Object) {
|
||||||
|
match o {
|
||||||
|
int {
|
||||||
|
println(it)
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
println('unknown object')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn (o Object) str() string {
|
||||||
|
match o {
|
||||||
|
int {
|
||||||
|
return it.str()
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
println('unknown object')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ''
|
||||||
|
}
|
||||||
|
|
||||||
|
fn (e mut Eval) stmt(node ast.Stmt) string {
|
||||||
|
match node {
|
||||||
|
ast.ExprStmt {
|
||||||
|
o := e.expr(it.expr)
|
||||||
|
print('out: ')
|
||||||
|
print_object(o)
|
||||||
|
return o.str()
|
||||||
|
}
|
||||||
|
ast.VarDecl {
|
||||||
|
e.vars[it.name] = Var{
|
||||||
|
value: e.expr(it.expr)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {}
|
||||||
|
}
|
||||||
|
return '>>'
|
||||||
|
}
|
||||||
|
|
||||||
|
fn (e mut Eval) expr(node ast.Expr) Object {
|
||||||
|
match node {
|
||||||
|
ast.IntegerLiteral {
|
||||||
|
return it.val
|
||||||
|
}
|
||||||
|
ast.Ident {
|
||||||
|
print_object(it.value)
|
||||||
|
// Find the variable
|
||||||
|
v := e.vars[it.name]
|
||||||
|
return v.value
|
||||||
|
}
|
||||||
|
ast.InfixExpr {
|
||||||
|
e.checker.infix_expr(it)
|
||||||
|
// println('bin $it.op')
|
||||||
|
left := e.expr(it.left) as int
|
||||||
|
right := e.expr(it.right) as int
|
||||||
|
match it.op {
|
||||||
|
.plus {
|
||||||
|
return left + right
|
||||||
|
}
|
||||||
|
.mul {
|
||||||
|
return left * right
|
||||||
|
}
|
||||||
|
else {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {}
|
||||||
|
}
|
||||||
|
return 0
|
||||||
|
// return Object{}
|
||||||
|
}
|
|
@ -4,6 +4,7 @@ import (
|
||||||
strings
|
strings
|
||||||
v.ast
|
v.ast
|
||||||
v.table
|
v.table
|
||||||
|
v.checker
|
||||||
// v.types
|
// v.types
|
||||||
term
|
term
|
||||||
)
|
)
|
||||||
|
@ -12,6 +13,7 @@ struct Gen {
|
||||||
out strings.Builder
|
out strings.Builder
|
||||||
definitions strings.Builder // typedefs, defines etc (everything that goes to the top of the file)
|
definitions strings.Builder // typedefs, defines etc (everything that goes to the top of the file)
|
||||||
table &table.Table
|
table &table.Table
|
||||||
|
checker checker.Checker
|
||||||
mut:
|
mut:
|
||||||
fn_decl &ast.FnDecl // pointer to the FnDecl we are currently inside otherwise 0
|
fn_decl &ast.FnDecl // pointer to the FnDecl we are currently inside otherwise 0
|
||||||
}
|
}
|
||||||
|
@ -22,6 +24,8 @@ pub fn cgen(files []ast.File, table &table.Table) string {
|
||||||
out: strings.new_builder(100)
|
out: strings.new_builder(100)
|
||||||
definitions: strings.new_builder(100)
|
definitions: strings.new_builder(100)
|
||||||
table: table
|
table: table
|
||||||
|
checker: checker.new_checker(table) // checker
|
||||||
|
|
||||||
fn_decl: 0
|
fn_decl: 0
|
||||||
}
|
}
|
||||||
for file in files {
|
for file in files {
|
||||||
|
@ -63,10 +67,11 @@ fn (g mut Gen) stmt(node ast.Stmt) {
|
||||||
// t := g.table.get_type(arg.ti.idx)
|
// t := g.table.get_type(arg.ti.idx)
|
||||||
ti := g.table.refresh_ti(arg.ti)
|
ti := g.table.refresh_ti(arg.ti)
|
||||||
g.write(ti.name + ' ' + arg.name)
|
g.write(ti.name + ' ' + arg.name)
|
||||||
|
g.definitions.write(ti.name + ' ' + arg.name)
|
||||||
if i < it.args.len - 1 {
|
if i < it.args.len - 1 {
|
||||||
g.write(', ')
|
g.write(', ')
|
||||||
|
g.definitions.write(', ')
|
||||||
}
|
}
|
||||||
g.definitions.write(ti.name + ' ' + arg.name)
|
|
||||||
}
|
}
|
||||||
g.writeln(') { ')
|
g.writeln(') { ')
|
||||||
if !is_main {
|
if !is_main {
|
||||||
|
@ -82,12 +87,12 @@ fn (g mut Gen) stmt(node ast.Stmt) {
|
||||||
g.fn_decl = 0
|
g.fn_decl = 0
|
||||||
}
|
}
|
||||||
ast.Return {
|
ast.Return {
|
||||||
g.write('return ')
|
g.write('return')
|
||||||
// multiple returns
|
// multiple returns
|
||||||
if it.exprs.len > 1 {
|
if it.exprs.len > 1 {
|
||||||
// t := g.table.get_type(g.fn_decl.ti.idx)
|
// t := g.table.get_type(g.fn_decl.ti.idx)
|
||||||
ti := g.table.refresh_ti(g.fn_decl.ti)
|
ti := g.table.refresh_ti(g.fn_decl.ti)
|
||||||
g.write('($ti.name){')
|
g.write(' ($ti.name){')
|
||||||
for i, expr in it.exprs {
|
for i, expr in it.exprs {
|
||||||
g.write('.arg$i=')
|
g.write('.arg$i=')
|
||||||
g.expr(expr)
|
g.expr(expr)
|
||||||
|
@ -98,17 +103,20 @@ fn (g mut Gen) stmt(node ast.Stmt) {
|
||||||
g.write('}')
|
g.write('}')
|
||||||
}
|
}
|
||||||
// normal return
|
// normal return
|
||||||
else {
|
else if it.exprs.len == 1 {
|
||||||
|
g.write(' ')
|
||||||
g.expr(it.exprs[0])
|
g.expr(it.exprs[0])
|
||||||
}
|
}
|
||||||
g.writeln(';')
|
g.writeln(';')
|
||||||
}
|
}
|
||||||
ast.VarDecl {
|
ast.VarDecl {
|
||||||
mut ti := it.ti
|
mut typ := it.typ
|
||||||
if ti.kind == .unresolved {
|
if typ.kind == .unresolved {
|
||||||
ti = g.table.get_expr_ti(it.expr)
|
// g.write('/*unresolved*/')
|
||||||
|
// ti = table.void_type // g.table.get_expr_ti(it.expr)
|
||||||
|
typ = g.checker.expr(it.expr)
|
||||||
}
|
}
|
||||||
g.write('$ti.name $it.name = ')
|
g.write('$typ.name $it.name = ')
|
||||||
g.expr(it.expr)
|
g.expr(it.expr)
|
||||||
g.writeln(';')
|
g.writeln(';')
|
||||||
}
|
}
|
||||||
|
@ -177,6 +185,7 @@ fn (g mut Gen) expr(node ast.Expr) {
|
||||||
g.expr(it.expr)
|
g.expr(it.expr)
|
||||||
g.write(it.op.str())
|
g.write(it.op.str())
|
||||||
}
|
}
|
||||||
|
/*
|
||||||
ast.UnaryExpr {
|
ast.UnaryExpr {
|
||||||
// probably not :D
|
// probably not :D
|
||||||
if it.op in [.inc, .dec] {
|
if it.op in [.inc, .dec] {
|
||||||
|
@ -188,6 +197,8 @@ fn (g mut Gen) expr(node ast.Expr) {
|
||||||
g.expr(it.left)
|
g.expr(it.left)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
ast.StringLiteral {
|
ast.StringLiteral {
|
||||||
g.write('tos3("$it.val")')
|
g.write('tos3("$it.val")')
|
||||||
}
|
}
|
||||||
|
@ -195,7 +206,7 @@ fn (g mut Gen) expr(node ast.Expr) {
|
||||||
g.write(it.op.str())
|
g.write(it.op.str())
|
||||||
g.expr(it.right)
|
g.expr(it.right)
|
||||||
}
|
}
|
||||||
ast.BinaryExpr {
|
ast.InfixExpr {
|
||||||
g.expr(it.left)
|
g.expr(it.left)
|
||||||
if it.op == .dot {
|
if it.op == .dot {
|
||||||
println('!! dot')
|
println('!! dot')
|
||||||
|
|
|
@ -8,7 +8,7 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
struct JsGen {
|
struct JsGen {
|
||||||
out strings.Builder
|
out strings.Builder
|
||||||
table &table.Table
|
table &table.Table
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -51,17 +51,15 @@ fn (g mut JsGen) stmt(node ast.Stmt) {
|
||||||
}
|
}
|
||||||
ast.Return {
|
ast.Return {
|
||||||
g.write('return ')
|
g.write('return ')
|
||||||
if it.exprs.len > 0 {
|
if it.exprs.len > 0 {}
|
||||||
|
|
||||||
}
|
|
||||||
else {
|
else {
|
||||||
g.expr(it.exprs[0])
|
g.expr(it.exprs[0])
|
||||||
}
|
}
|
||||||
g.writeln(';')
|
g.writeln(';')
|
||||||
}
|
}
|
||||||
ast.VarDecl {
|
ast.VarDecl {
|
||||||
ti := g.table.refresh_ti(it.ti)
|
typ := g.table.refresh_ti(it.typ)
|
||||||
g.write('var /* $ti.name */ $it.name = ')
|
g.write('var /* $typ.name */ $it.name = ')
|
||||||
g.expr(it.expr)
|
g.expr(it.expr)
|
||||||
g.writeln(';')
|
g.writeln(';')
|
||||||
}
|
}
|
||||||
|
@ -106,14 +104,17 @@ fn (g mut JsGen) expr(node ast.Expr) {
|
||||||
ast.FloatLiteral {
|
ast.FloatLiteral {
|
||||||
g.write(it.val)
|
g.write(it.val)
|
||||||
}
|
}
|
||||||
|
/*
|
||||||
ast.UnaryExpr {
|
ast.UnaryExpr {
|
||||||
g.expr(it.left)
|
g.expr(it.left)
|
||||||
g.write(' $it.op ')
|
g.write(' $it.op ')
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
ast.StringLiteral {
|
ast.StringLiteral {
|
||||||
g.write('tos3("$it.val")')
|
g.write('tos3("$it.val")')
|
||||||
}
|
}
|
||||||
ast.BinaryExpr {
|
ast.InfixExpr {
|
||||||
g.expr(it.left)
|
g.expr(it.left)
|
||||||
g.write(' $it.op.str() ')
|
g.write(' $it.op.str() ')
|
||||||
g.expr(it.right)
|
g.expr(it.right)
|
||||||
|
|
|
@ -4,19 +4,22 @@ int get_int2();
|
||||||
void myuser();
|
void myuser();
|
||||||
multi_return_int_string multi_return();
|
multi_return_int_string multi_return();
|
||||||
void variadic(variadic_int a);
|
void variadic(variadic_int a);
|
||||||
|
void ensure_cap(int required, int cap);
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
int age;
|
int age;
|
||||||
} User;
|
} User;
|
||||||
|
|
||||||
int main() {
|
int main() {
|
||||||
int a = 10;
|
int a = 10;
|
||||||
a++;
|
a++;
|
||||||
int negative = -a;
|
int negative = -a;
|
||||||
a == 1;
|
2 < 3;
|
||||||
foo(3);
|
a == 1;
|
||||||
int ak = 10;
|
a++;
|
||||||
return 0;
|
foo(3);
|
||||||
|
int ak = 10;
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void foo(int a) {
|
void foo(int a) {
|
||||||
|
@ -31,7 +34,19 @@ i < 10; i++;
|
||||||
1, 2, 3,
|
1, 2, 3,
|
||||||
});
|
});
|
||||||
int number = nums[0];
|
int number = nums[0];
|
||||||
|
array_bool bools = new_array_from_c_array(2, 2, sizeof(array_bool), {
|
||||||
|
true, false,
|
||||||
|
});
|
||||||
|
bool b = bools[0];
|
||||||
|
array_string mystrings = new_array_from_c_array(2, 2, sizeof(array_string), {
|
||||||
|
tos3("a"), tos3("b"),
|
||||||
|
});
|
||||||
|
string s = mystrings[0];
|
||||||
|
int x = 0;
|
||||||
|
x = get_int2();
|
||||||
int n = get_int2();
|
int n = get_int2();
|
||||||
|
bool q = true || false;
|
||||||
|
bool b2 = bools[0] || true;
|
||||||
}
|
}
|
||||||
|
|
||||||
int get_int(string a) {
|
int get_int(string a) {
|
||||||
|
@ -55,8 +70,14 @@ void myuser() {
|
||||||
}
|
}
|
||||||
|
|
||||||
multi_return_int_string multi_return() {
|
multi_return_int_string multi_return() {
|
||||||
return (multi_return_int_string){.arg0=4,.arg1=tos3("four")};
|
return (multi_return_int_string){.arg0=4,.arg1=tos3("four")};
|
||||||
}
|
}
|
||||||
|
|
||||||
void variadic(variadic_int a) {
|
void variadic(variadic_int a) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ensure_cap(int required, int cap) {
|
||||||
|
if (required < cap) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -5,12 +5,14 @@ struct User {
|
||||||
age int
|
age int
|
||||||
}
|
}
|
||||||
|
|
||||||
// lol
|
|
||||||
fn main() {
|
fn main() {
|
||||||
a := 10
|
a := 10
|
||||||
|
//bb := 2 + 'hi'
|
||||||
a++
|
a++
|
||||||
negative := -a
|
negative := -a
|
||||||
|
2 < 3
|
||||||
a == 1
|
a == 1
|
||||||
|
a++
|
||||||
foo(3)
|
foo(3)
|
||||||
ak := 10
|
ak := 10
|
||||||
}
|
}
|
||||||
|
@ -32,7 +34,16 @@ fn foo(a int) {
|
||||||
}
|
}
|
||||||
nums := [1,2,3]
|
nums := [1,2,3]
|
||||||
number := nums[0]
|
number := nums[0]
|
||||||
|
bools := [true, false]
|
||||||
|
b := bools[0]
|
||||||
|
mystrings := ['a', 'b']
|
||||||
|
s := mystrings[0]
|
||||||
|
//////////
|
||||||
|
mut x := 0
|
||||||
|
x=get_int2()
|
||||||
n := get_int2()
|
n := get_int2()
|
||||||
|
q := true || false
|
||||||
|
b2 := bools[0] || true
|
||||||
}
|
}
|
||||||
|
|
||||||
fn get_int(a string) int {
|
fn get_int(a string) int {
|
||||||
|
@ -41,6 +52,7 @@ fn get_int(a string) int {
|
||||||
|
|
||||||
fn get_int2() int {
|
fn get_int2() int {
|
||||||
a := 'hello'
|
a := 'hello'
|
||||||
|
//return get_int('sdf')
|
||||||
return get_int(a)
|
return get_int(a)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -60,3 +72,10 @@ fn multi_return() (int,string) {
|
||||||
fn variadic(a ...int) {
|
fn variadic(a ...int) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn ensure_cap(required int, cap int) {
|
||||||
|
if required < cap {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -344,11 +344,14 @@ fn (g mut Gen) expr(node ast.Expr) {
|
||||||
ast.AssignExpr {}
|
ast.AssignExpr {}
|
||||||
ast.IntegerLiteral {}
|
ast.IntegerLiteral {}
|
||||||
ast.FloatLiteral {}
|
ast.FloatLiteral {}
|
||||||
|
/*
|
||||||
ast.UnaryExpr {
|
ast.UnaryExpr {
|
||||||
g.expr(it.left)
|
g.expr(it.left)
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
ast.StringLiteral {}
|
ast.StringLiteral {}
|
||||||
ast.BinaryExpr {}
|
ast.InfixExpr {}
|
||||||
// `user := User{name: 'Bob'}`
|
// `user := User{name: 'Bob'}`
|
||||||
ast.StructInit {}
|
ast.StructInit {}
|
||||||
ast.CallExpr {
|
ast.CallExpr {
|
||||||
|
|
|
@ -6,10 +6,9 @@ module parser
|
||||||
import (
|
import (
|
||||||
v.ast
|
v.ast
|
||||||
v.table
|
v.table
|
||||||
v.types
|
|
||||||
)
|
)
|
||||||
|
|
||||||
pub fn (p mut Parser) call_expr() (ast.CallExpr,types.TypeIdent) {
|
pub fn (p mut Parser) call_expr() (ast.CallExpr,table.Type) {
|
||||||
tok := p.tok
|
tok := p.tok
|
||||||
fn_name := p.check_name()
|
fn_name := p.check_name()
|
||||||
p.check(.lpar)
|
p.check(.lpar)
|
||||||
|
@ -27,16 +26,15 @@ pub fn (p mut Parser) call_expr() (ast.CallExpr,types.TypeIdent) {
|
||||||
name: fn_name
|
name: fn_name
|
||||||
args: args
|
args: args
|
||||||
// tok: tok
|
// tok: tok
|
||||||
pos: tok.position()
|
|
||||||
|
|
||||||
|
pos: tok.position()
|
||||||
}
|
}
|
||||||
mut ti := types.unresolved_ti
|
mut ti := table.unresolved_type
|
||||||
if f := p.table.find_fn(fn_name) {
|
if f := p.table.find_fn(fn_name) {
|
||||||
ti = f.return_ti
|
ti = f.return_type
|
||||||
}
|
}
|
||||||
println('adding call_expr check $fn_name')
|
println('adding call_expr check $fn_name')
|
||||||
|
return node,ti
|
||||||
return node, ti
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (p mut Parser) call_args() []ast.Expr {
|
pub fn (p mut Parser) call_args() []ast.Expr {
|
||||||
|
@ -49,7 +47,7 @@ pub fn (p mut Parser) call_args() []ast.Expr {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
p.check(.rpar)
|
p.check(.rpar)
|
||||||
return args // ,types.void_ti
|
return args // ,table.void_ti
|
||||||
}
|
}
|
||||||
|
|
||||||
fn (p mut Parser) fn_decl() ast.FnDecl {
|
fn (p mut Parser) fn_decl() ast.FnDecl {
|
||||||
|
@ -62,7 +60,7 @@ fn (p mut Parser) fn_decl() ast.FnDecl {
|
||||||
// Receiver?
|
// Receiver?
|
||||||
mut rec_name := ''
|
mut rec_name := ''
|
||||||
mut is_method := false
|
mut is_method := false
|
||||||
mut rec_ti := types.void_ti
|
mut rec_ti := table.void_type
|
||||||
if p.tok.kind == .lpar {
|
if p.tok.kind == .lpar {
|
||||||
is_method = true
|
is_method = true
|
||||||
p.next()
|
p.next()
|
||||||
|
@ -73,7 +71,7 @@ fn (p mut Parser) fn_decl() ast.FnDecl {
|
||||||
rec_ti = p.parse_ti()
|
rec_ti = p.parse_ti()
|
||||||
p.table.register_var(table.Var{
|
p.table.register_var(table.Var{
|
||||||
name: rec_name
|
name: rec_name
|
||||||
ti: rec_ti
|
typ: rec_ti
|
||||||
})
|
})
|
||||||
p.check(.rpar)
|
p.check(.rpar)
|
||||||
}
|
}
|
||||||
|
@ -94,7 +92,7 @@ fn (p mut Parser) fn_decl() ast.FnDecl {
|
||||||
for arg_name in arg_names {
|
for arg_name in arg_names {
|
||||||
arg := table.Var{
|
arg := table.Var{
|
||||||
name: arg_name
|
name: arg_name
|
||||||
ti: ti
|
typ: ti
|
||||||
}
|
}
|
||||||
args << arg
|
args << arg
|
||||||
p.table.register_var(arg)
|
p.table.register_var(arg)
|
||||||
|
@ -112,16 +110,19 @@ fn (p mut Parser) fn_decl() ast.FnDecl {
|
||||||
}
|
}
|
||||||
p.check(.rpar)
|
p.check(.rpar)
|
||||||
// Return type
|
// Return type
|
||||||
mut ti := types.void_ti
|
mut typ := table.void_type
|
||||||
if p.tok.kind in [.name, .lpar] {
|
if p.tok.kind in [.name, .lpar] {
|
||||||
ti = p.parse_ti()
|
typ = p.parse_ti()
|
||||||
p.return_ti = ti
|
p.return_type = typ
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
p.return_type = table.void_type
|
||||||
}
|
}
|
||||||
if is_method {
|
if is_method {
|
||||||
ok := p.table.register_method(rec_ti, table.Fn{
|
ok := p.table.register_method(rec_ti, table.Fn{
|
||||||
name: name
|
name: name
|
||||||
args: args
|
args: args
|
||||||
return_ti: ti
|
return_type: typ
|
||||||
})
|
})
|
||||||
if !ok {
|
if !ok {
|
||||||
p.error('expected Struct')
|
p.error('expected Struct')
|
||||||
|
@ -131,14 +132,14 @@ fn (p mut Parser) fn_decl() ast.FnDecl {
|
||||||
p.table.register_fn(table.Fn{
|
p.table.register_fn(table.Fn{
|
||||||
name: name
|
name: name
|
||||||
args: args
|
args: args
|
||||||
return_ti: ti
|
return_type: typ
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
stmts := p.parse_block()
|
stmts := p.parse_block()
|
||||||
return ast.FnDecl{
|
return ast.FnDecl{
|
||||||
name: name
|
name: name
|
||||||
stmts: stmts
|
stmts: stmts
|
||||||
ti: ti
|
ti: typ
|
||||||
args: ast_args
|
args: ast_args
|
||||||
is_pub: is_pub
|
is_pub: is_pub
|
||||||
receiver: ast.Field{
|
receiver: ast.Field{
|
||||||
|
@ -161,4 +162,5 @@ pub fn (p &Parser) check_fn_calls() {
|
||||||
// println('IN AST typ=' + call.typ.name)
|
// println('IN AST typ=' + call.typ.name)
|
||||||
}
|
}
|
||||||
*/
|
*/
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,9 +2,12 @@ module parser
|
||||||
// Copyright (c) 2019-2020 Alexander Medvednikov. All rights reserved.
|
// Copyright (c) 2019-2020 Alexander Medvednikov. All rights reserved.
|
||||||
// Use of this source code is governed by an MIT license
|
// Use of this source code is governed by an MIT license
|
||||||
// that can be found in the LICENSE file.
|
// that can be found in the LICENSE file.
|
||||||
import v.types
|
|
||||||
|
|
||||||
pub fn (p mut Parser) parse_array_ti(nr_muls int) types.TypeIdent {
|
import (
|
||||||
|
v.table
|
||||||
|
)
|
||||||
|
|
||||||
|
pub fn (p mut Parser) parse_array_ti(nr_muls int) table.Type {
|
||||||
p.check(.lsbr)
|
p.check(.lsbr)
|
||||||
// fixed array
|
// fixed array
|
||||||
if p.tok.kind == .number {
|
if p.tok.kind == .number {
|
||||||
|
@ -12,7 +15,7 @@ pub fn (p mut Parser) parse_array_ti(nr_muls int) types.TypeIdent {
|
||||||
p.check(.rsbr)
|
p.check(.rsbr)
|
||||||
elem_ti := p.parse_ti()
|
elem_ti := p.parse_ti()
|
||||||
idx,name := p.table.find_or_register_array_fixed(&elem_ti, size, 1)
|
idx,name := p.table.find_or_register_array_fixed(&elem_ti, size, 1)
|
||||||
return types.new_ti(.array_fixed, name, idx, nr_muls)
|
return table.new_type(.array_fixed, name, idx, nr_muls)
|
||||||
}
|
}
|
||||||
// array
|
// array
|
||||||
p.check(.rsbr)
|
p.check(.rsbr)
|
||||||
|
@ -24,22 +27,22 @@ pub fn (p mut Parser) parse_array_ti(nr_muls int) types.TypeIdent {
|
||||||
nr_dims++
|
nr_dims++
|
||||||
}
|
}
|
||||||
idx,name := p.table.find_or_register_array(&elem_ti, nr_dims)
|
idx,name := p.table.find_or_register_array(&elem_ti, nr_dims)
|
||||||
return types.new_ti(.array, name, idx, nr_muls)
|
return table.new_type(.array, name, idx, nr_muls)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (p mut Parser) parse_map_ti(nr_muls int) types.TypeIdent {
|
pub fn (p mut Parser) parse_map_ti(nr_muls int) table.Type {
|
||||||
p.next()
|
p.next()
|
||||||
p.check(.lsbr)
|
p.check(.lsbr)
|
||||||
key_ti := p.parse_ti()
|
key_ti := p.parse_ti()
|
||||||
p.check(.rsbr)
|
p.check(.rsbr)
|
||||||
value_ti := p.parse_ti()
|
value_ti := p.parse_ti()
|
||||||
idx,name := p.table.find_or_register_map(&key_ti, &value_ti)
|
idx,name := p.table.find_or_register_map(&key_ti, &value_ti)
|
||||||
return types.new_ti(.map, name, idx, nr_muls)
|
return table.new_type(.map, name, idx, nr_muls)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (p mut Parser) parse_multi_return_ti() types.TypeIdent {
|
pub fn (p mut Parser) parse_multi_return_ti() table.Type {
|
||||||
p.check(.lpar)
|
p.check(.lpar)
|
||||||
mut mr_tis := []types.TypeIdent
|
mut mr_tis := []table.Type
|
||||||
for {
|
for {
|
||||||
mr_ti := p.parse_ti()
|
mr_ti := p.parse_ti()
|
||||||
mr_tis << mr_ti
|
mr_tis << mr_ti
|
||||||
|
@ -52,17 +55,17 @@ pub fn (p mut Parser) parse_multi_return_ti() types.TypeIdent {
|
||||||
}
|
}
|
||||||
p.check(.rpar)
|
p.check(.rpar)
|
||||||
idx,name := p.table.find_or_register_multi_return(mr_tis)
|
idx,name := p.table.find_or_register_multi_return(mr_tis)
|
||||||
return types.new_ti(.multi_return, name, idx, 0)
|
return table.new_type(.multi_return, name, idx, 0)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (p mut Parser) parse_variadic_ti() types.TypeIdent {
|
pub fn (p mut Parser) parse_variadic_ti() table.Type {
|
||||||
p.check(.ellipsis)
|
p.check(.ellipsis)
|
||||||
variadic_ti := p.parse_ti()
|
variadic_ti := p.parse_ti()
|
||||||
idx,name := p.table.find_or_register_variadic(&variadic_ti)
|
idx,name := p.table.find_or_register_variadic(&variadic_ti)
|
||||||
return types.new_ti(.variadic, name, idx, 0)
|
return table.new_type(.variadic, name, idx, 0)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (p mut Parser) parse_ti() types.TypeIdent {
|
pub fn (p mut Parser) parse_ti() table.Type {
|
||||||
mut nr_muls := 0
|
mut nr_muls := 0
|
||||||
for p.tok.kind == .amp {
|
for p.tok.kind == .amp {
|
||||||
p.check(.amp)
|
p.check(.amp)
|
||||||
|
@ -98,65 +101,66 @@ pub fn (p mut Parser) parse_ti() types.TypeIdent {
|
||||||
return p.parse_map_ti(nr_muls)
|
return p.parse_map_ti(nr_muls)
|
||||||
}
|
}
|
||||||
'voidptr' {
|
'voidptr' {
|
||||||
return types.new_ti(.voidptr, 'voidptr', types.voidptr_type_idx, nr_muls)
|
return table.new_type(.voidptr, 'voidptr', table.voidptr_type_idx, nr_muls)
|
||||||
}
|
}
|
||||||
'byteptr' {
|
'byteptr' {
|
||||||
return types.new_ti(.byteptr, 'byteptr', types.byteptr_type_idx, nr_muls)
|
return table.new_type(.byteptr, 'byteptr', table.byteptr_type_idx, nr_muls)
|
||||||
}
|
}
|
||||||
'charptr' {
|
'charptr' {
|
||||||
return types.new_ti(.charptr, 'charptr', types.charptr_type_idx, nr_muls)
|
return table.new_type(.charptr, 'charptr', table.charptr_type_idx, nr_muls)
|
||||||
}
|
}
|
||||||
'i8' {
|
'i8' {
|
||||||
return types.new_ti(.i8, 'i8', types.i8_type_idx, nr_muls)
|
return table.new_type(.i8, 'i8', table.i8_type_idx, nr_muls)
|
||||||
}
|
}
|
||||||
'i16' {
|
'i16' {
|
||||||
return types.new_ti(.i16, 'i16', types.i16_type_idx, nr_muls)
|
return table.new_type(.i16, 'i16', table.i16_type_idx, nr_muls)
|
||||||
}
|
}
|
||||||
'int' {
|
'int' {
|
||||||
return types.new_ti(.int, 'int', types.int_type_idx, nr_muls)
|
return table.new_type(.int, 'int', table.int_type_idx, nr_muls)
|
||||||
}
|
}
|
||||||
'i64' {
|
'i64' {
|
||||||
return types.new_ti(.i64, 'i64', types.i64_type_idx, nr_muls)
|
return table.new_type(.i64, 'i64', table.i64_type_idx, nr_muls)
|
||||||
}
|
}
|
||||||
'byte' {
|
'byte' {
|
||||||
return types.new_ti(.byte, 'byte', types.byte_type_idx, nr_muls)
|
return table.new_type(.byte, 'byte', table.byte_type_idx, nr_muls)
|
||||||
}
|
}
|
||||||
'u16' {
|
'u16' {
|
||||||
return types.new_ti(.u16, 'u16', types.u16_type_idx, nr_muls)
|
return table.new_type(.u16, 'u16', table.u16_type_idx, nr_muls)
|
||||||
}
|
}
|
||||||
'u32' {
|
'u32' {
|
||||||
return types.new_ti(.u32, 'u32', types.u32_type_idx, nr_muls)
|
return table.new_type(.u32, 'u32', table.u32_type_idx, nr_muls)
|
||||||
}
|
}
|
||||||
'u64' {
|
'u64' {
|
||||||
return types.new_ti(.u64, 'u64', types.u64_type_idx, nr_muls)
|
return table.new_type(.u64, 'u64', table.u64_type_idx, nr_muls)
|
||||||
}
|
}
|
||||||
'f32' {
|
'f32' {
|
||||||
return types.new_ti(.f32, 'f32', types.f32_type_idx, nr_muls)
|
return table.new_type(.f32, 'f32', table.f32_type_idx, nr_muls)
|
||||||
}
|
}
|
||||||
'f64' {
|
'f64' {
|
||||||
return types.new_ti(.f64, 'f64', types.f64_type_idx, nr_muls)
|
return table.new_type(.f64, 'f64', table.f64_type_idx, nr_muls)
|
||||||
}
|
}
|
||||||
'string' {
|
'string' {
|
||||||
return types.new_ti(.string, 'string', types.string_type_idx, nr_muls)
|
return table.new_type(.string, 'string', table.string_type_idx, nr_muls)
|
||||||
}
|
}
|
||||||
'char' {
|
'char' {
|
||||||
return types.new_ti(.char, 'char', types.charptr_type_idx, nr_muls)
|
return table.new_type(.char, 'char', table.charptr_type_idx, nr_muls)
|
||||||
}
|
}
|
||||||
'bool' {
|
'bool' {
|
||||||
return types.new_ti(.bool, 'bool', types.bool_type_idx, nr_muls)
|
return table.new_type(.bool, 'bool', table.bool_type_idx, nr_muls)
|
||||||
}
|
}
|
||||||
// struct / enum / placeholder
|
// struct / enum / placeholder
|
||||||
else {
|
else {
|
||||||
// struct / enum
|
// struct / enum
|
||||||
mut idx := p.table.find_type_idx(name)
|
mut idx := p.table.find_type_idx(name)
|
||||||
if idx > 0 {
|
if idx > 0 {
|
||||||
return types.new_ti(p.table.types[idx].kind, name, idx, nr_muls)
|
return table.new_type(p.table.types[idx].kind, name, idx, nr_muls)
|
||||||
}
|
}
|
||||||
// not found - add placeholder
|
// not found - add placeholder
|
||||||
idx = p.table.add_placeholder_type(name)
|
idx = p.table.add_placeholder_type(name)
|
||||||
return types.new_ti(.placeholder, name, idx, nr_muls)
|
return table.new_type(.placeholder, name, idx, nr_muls)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,10 +8,10 @@ import (
|
||||||
v.ast
|
v.ast
|
||||||
v.token
|
v.token
|
||||||
v.table
|
v.table
|
||||||
v.types
|
|
||||||
term
|
term
|
||||||
os
|
os
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
colored_output = term.can_show_color_on_stderr()
|
colored_output = term.can_show_color_on_stderr()
|
||||||
)
|
)
|
||||||
|
@ -23,20 +23,21 @@ type InfixParseFn fn(e ast.Expr)ast.Expr
|
||||||
type PostfixParseFn fn()ast.Expr
|
type PostfixParseFn fn()ast.Expr
|
||||||
|
|
||||||
struct Parser {
|
struct Parser {
|
||||||
scanner &scanner.Scanner
|
scanner &scanner.Scanner
|
||||||
file_name string
|
file_name string
|
||||||
mut:
|
mut:
|
||||||
tok token.Token
|
tok token.Token
|
||||||
peek_tok token.Token
|
peek_tok token.Token
|
||||||
// vars []string
|
// vars []string
|
||||||
table &table.Table
|
table &table.Table
|
||||||
return_ti types.TypeIdent
|
return_type table.Type // current function's return type
|
||||||
is_c bool
|
is_c bool
|
||||||
//
|
//
|
||||||
// prefix_parse_fns []PrefixParseFn
|
// prefix_parse_fns []PrefixParseFn
|
||||||
inside_if bool
|
inside_if bool
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// for tests
|
||||||
pub fn parse_stmt(text string, table &table.Table) ast.Stmt {
|
pub fn parse_stmt(text string, table &table.Table) ast.Stmt {
|
||||||
s := scanner.new_scanner(text)
|
s := scanner.new_scanner(text)
|
||||||
mut p := Parser{
|
mut p := Parser{
|
||||||
|
@ -60,22 +61,15 @@ pub fn parse_file(path string, table &table.Table) ast.File {
|
||||||
file_name: path
|
file_name: path
|
||||||
}
|
}
|
||||||
p.read_first_token()
|
p.read_first_token()
|
||||||
|
|
||||||
// module decl
|
// module decl
|
||||||
module_decl := if p.tok.kind == .key_module {
|
module_decl := if p.tok.kind == .key_module { p.module_decl() } else { ast.Module{name: 'main'
|
||||||
p.module_decl()
|
} }
|
||||||
} else {
|
|
||||||
ast.Module{
|
|
||||||
name: 'main'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// imports
|
// imports
|
||||||
mut imports := []ast.Import
|
mut imports := []ast.Import
|
||||||
for p.tok.kind == .key_import {
|
for p.tok.kind == .key_import {
|
||||||
imports << p.import_stmt()
|
imports << p.import_stmt()
|
||||||
}
|
}
|
||||||
// TODO: import only mode
|
// TODO: import only mode
|
||||||
|
|
||||||
for {
|
for {
|
||||||
// res := s.scan()
|
// res := s.scan()
|
||||||
if p.tok.kind == .eof {
|
if p.tok.kind == .eof {
|
||||||
|
@ -88,9 +82,9 @@ pub fn parse_file(path string, table &table.Table) ast.File {
|
||||||
// println('nr stmts = $stmts.len')
|
// println('nr stmts = $stmts.len')
|
||||||
// println(stmts[0])
|
// println(stmts[0])
|
||||||
return ast.File{
|
return ast.File{
|
||||||
mod: module_decl
|
mod: module_decl
|
||||||
imports: imports
|
imports: imports
|
||||||
stmts: stmts
|
stmts: stmts
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -168,12 +162,12 @@ pub fn (p mut Parser) top_stmt() ast.Stmt {
|
||||||
p.error('wrong pub keyword usage')
|
p.error('wrong pub keyword usage')
|
||||||
return ast.Stmt{}
|
return ast.Stmt{}
|
||||||
}
|
}
|
||||||
// .key_enum {
|
// .key_enum {
|
||||||
// return p.enum_decl()
|
// return p.enum_decl()
|
||||||
// }
|
// }
|
||||||
// .key_type {
|
// .key_type {
|
||||||
// return p.type_decl()
|
// return p.type_decl()
|
||||||
// }
|
// }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
.key_const {
|
.key_const {
|
||||||
|
@ -227,7 +221,7 @@ pub fn (p mut Parser) stmt() ast.Stmt {
|
||||||
pub fn (p mut Parser) assign_expr(left ast.Expr) ast.AssignExpr {
|
pub fn (p mut Parser) assign_expr(left ast.Expr) ast.AssignExpr {
|
||||||
op := p.tok.kind
|
op := p.tok.kind
|
||||||
p.next()
|
p.next()
|
||||||
val, _ := p.expr(0)
|
val,_ := p.expr(0)
|
||||||
node := ast.AssignExpr{
|
node := ast.AssignExpr{
|
||||||
left: left
|
left: left
|
||||||
val: val
|
val: val
|
||||||
|
@ -270,7 +264,8 @@ pub fn (p &Parser) error(s string) {
|
||||||
final_msg_line := '$p.file_name:$p.tok.line_nr: error: $s'
|
final_msg_line := '$p.file_name:$p.tok.line_nr: error: $s'
|
||||||
if colored_output {
|
if colored_output {
|
||||||
eprintln(term.bold(term.red(final_msg_line)))
|
eprintln(term.bold(term.red(final_msg_line)))
|
||||||
}else{
|
}
|
||||||
|
else {
|
||||||
eprintln(final_msg_line)
|
eprintln(final_msg_line)
|
||||||
}
|
}
|
||||||
exit(1)
|
exit(1)
|
||||||
|
@ -280,7 +275,8 @@ pub fn (p &Parser) error_at_line(s string, line_nr int) {
|
||||||
final_msg_line := '$p.file_name:$line_nr: error: $s'
|
final_msg_line := '$p.file_name:$line_nr: error: $s'
|
||||||
if colored_output {
|
if colored_output {
|
||||||
eprintln(term.bold(term.red(final_msg_line)))
|
eprintln(term.bold(term.red(final_msg_line)))
|
||||||
}else{
|
}
|
||||||
|
else {
|
||||||
eprintln(final_msg_line)
|
eprintln(final_msg_line)
|
||||||
}
|
}
|
||||||
exit(1)
|
exit(1)
|
||||||
|
@ -290,15 +286,16 @@ pub fn (p &Parser) warn(s string) {
|
||||||
final_msg_line := '$p.file_name:$p.tok.line_nr: warning: $s'
|
final_msg_line := '$p.file_name:$p.tok.line_nr: warning: $s'
|
||||||
if colored_output {
|
if colored_output {
|
||||||
eprintln(term.bold(term.blue(final_msg_line)))
|
eprintln(term.bold(term.blue(final_msg_line)))
|
||||||
}else{
|
}
|
||||||
|
else {
|
||||||
eprintln(final_msg_line)
|
eprintln(final_msg_line)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (p mut Parser) name_expr() (ast.Expr,types.TypeIdent) {
|
pub fn (p mut Parser) name_expr() (ast.Expr,table.Type) {
|
||||||
mut node := ast.Expr{}
|
mut node := ast.Expr{}
|
||||||
// mut ti := types.void_ti
|
// mut typ := table.void_ti
|
||||||
mut ti := types.unresolved_ti
|
mut typ := table.unresolved_type
|
||||||
if p.tok.lit == 'C' {
|
if p.tok.lit == 'C' {
|
||||||
p.next()
|
p.next()
|
||||||
p.check(.dot)
|
p.check(.dot)
|
||||||
|
@ -310,14 +307,14 @@ pub fn (p mut Parser) name_expr() (ast.Expr,types.TypeIdent) {
|
||||||
// fn call
|
// fn call
|
||||||
if p.peek_tok.kind == .lpar {
|
if p.peek_tok.kind == .lpar {
|
||||||
println('calling $p.tok.lit')
|
println('calling $p.tok.lit')
|
||||||
x, ti2 := p.call_expr() // TODO `node,typ :=` should work
|
x,ti2 := p.call_expr() // TODO `node,typ :=` should work
|
||||||
node = x
|
node = x
|
||||||
ti = ti2
|
typ = ti2
|
||||||
}
|
}
|
||||||
// struct init
|
// struct init
|
||||||
else if p.peek_tok.kind == .lcbr && !p.inside_if {
|
else if p.peek_tok.kind == .lcbr && (p.tok.lit[0].is_capital() || p.tok.lit in ['array', 'string']) {
|
||||||
ti = p.parse_ti()
|
typ = p.parse_ti()
|
||||||
// println('sturct init ti=$ti.name')
|
// p.warn('struct init typ=$typ.name')
|
||||||
p.check(.lcbr)
|
p.check(.lcbr)
|
||||||
mut field_names := []string
|
mut field_names := []string
|
||||||
mut exprs := []ast.Expr
|
mut exprs := []ast.Expr
|
||||||
|
@ -329,13 +326,14 @@ pub fn (p mut Parser) name_expr() (ast.Expr,types.TypeIdent) {
|
||||||
exprs << expr
|
exprs << expr
|
||||||
}
|
}
|
||||||
node = ast.StructInit{
|
node = ast.StructInit{
|
||||||
ti: ti
|
ti: typ
|
||||||
exprs: exprs
|
exprs: exprs
|
||||||
fields: field_names
|
fields: field_names
|
||||||
pos: p.tok.position()
|
pos: p.tok.position()
|
||||||
}
|
}
|
||||||
p.check(.rcbr)
|
p.check(.rcbr)
|
||||||
}
|
}
|
||||||
|
// variable
|
||||||
else {
|
else {
|
||||||
// p.warn('name ')
|
// p.warn('name ')
|
||||||
// left := p.parse_ident()
|
// left := p.parse_ident()
|
||||||
|
@ -346,33 +344,36 @@ pub fn (p mut Parser) name_expr() (ast.Expr,types.TypeIdent) {
|
||||||
p.error('name expr unknown variable `$p.tok.lit`')
|
p.error('name expr unknown variable `$p.tok.lit`')
|
||||||
exit(0)
|
exit(0)
|
||||||
}
|
}
|
||||||
ti = var.ti
|
typ = var.typ
|
||||||
ident.kind = .variable
|
ident.kind = .variable
|
||||||
ident.info = ast.IdentVar {
|
ident.info = ast.IdentVar{
|
||||||
ti: ti
|
typ: typ
|
||||||
expr: var.expr
|
name: ident.name
|
||||||
|
// expr: p.expr(0)// var.expr
|
||||||
|
|
||||||
}
|
}
|
||||||
// ident.ti = ti
|
// ident.ti = ti
|
||||||
node = ident
|
node = ident
|
||||||
p.next()
|
p.next()
|
||||||
}
|
}
|
||||||
return node,ti
|
return node,typ
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (p mut Parser) expr(precedence int) (ast.Expr,types.TypeIdent) {
|
pub fn (p mut Parser) expr(precedence int) (ast.Expr,table.Type) {
|
||||||
mut ti := types.void_ti
|
// println('\n\nparser.expr()')
|
||||||
|
mut typ := table.void_type
|
||||||
mut node := ast.Expr{}
|
mut node := ast.Expr{}
|
||||||
// Prefix
|
// Prefix
|
||||||
match p.tok.kind {
|
match p.tok.kind {
|
||||||
.name {
|
.name {
|
||||||
node,ti = p.name_expr()
|
node,typ = p.name_expr()
|
||||||
}
|
}
|
||||||
.str {
|
.str {
|
||||||
node,ti = p.string_expr()
|
node,typ = p.string_expr()
|
||||||
}
|
}
|
||||||
// -1, -a etc
|
// -1, -a etc
|
||||||
.minus, .amp {
|
.minus, .amp {
|
||||||
node,ti = p.prefix_expr()
|
node,typ = p.prefix_expr()
|
||||||
}
|
}
|
||||||
// .amp {
|
// .amp {
|
||||||
// p.next()
|
// p.next()
|
||||||
|
@ -381,22 +382,22 @@ pub fn (p mut Parser) expr(precedence int) (ast.Expr,types.TypeIdent) {
|
||||||
node = ast.BoolLiteral{
|
node = ast.BoolLiteral{
|
||||||
val: p.tok.kind == .key_true
|
val: p.tok.kind == .key_true
|
||||||
}
|
}
|
||||||
ti = types.bool_ti
|
typ = table.bool_type
|
||||||
p.next()
|
p.next()
|
||||||
}
|
}
|
||||||
.number {
|
.number {
|
||||||
node,ti = p.parse_number_literal()
|
node,typ = p.parse_number_literal()
|
||||||
}
|
}
|
||||||
.lpar {
|
.lpar {
|
||||||
p.check(.lpar)
|
p.check(.lpar)
|
||||||
node,ti = p.expr(0)
|
node,typ = p.expr(0)
|
||||||
p.check(.rpar)
|
p.check(.rpar)
|
||||||
}
|
}
|
||||||
.key_if {
|
.key_if {
|
||||||
node,ti = p.if_expr()
|
node,typ = p.if_expr()
|
||||||
}
|
}
|
||||||
.lsbr {
|
.lsbr {
|
||||||
node,ti = p.array_init()
|
node,typ = p.array_init()
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
p.error('expr(): bad token `$p.tok.str()`')
|
p.error('expr(): bad token `$p.tok.str()`')
|
||||||
|
@ -408,13 +409,27 @@ pub fn (p mut Parser) expr(precedence int) (ast.Expr,types.TypeIdent) {
|
||||||
node = p.assign_expr(node)
|
node = p.assign_expr(node)
|
||||||
}
|
}
|
||||||
else if p.tok.kind == .dot {
|
else if p.tok.kind == .dot {
|
||||||
node,ti = p.dot_expr(node, ti)
|
node,typ = p.dot_expr(node, typ)
|
||||||
}
|
}
|
||||||
else if p.tok.kind == .lsbr {
|
else if p.tok.kind == .lsbr {
|
||||||
node,ti = p.index_expr(node)
|
// TODO
|
||||||
|
// info := ti.info as table.Array
|
||||||
|
// ti = p.table.types[info.elem_type_idx]
|
||||||
|
if typ.name.starts_with('array_') {
|
||||||
|
elm_typ := typ.name[6..]
|
||||||
|
x := p.table.find_type(elm_typ) or {
|
||||||
|
p.error(elm_typ)
|
||||||
|
exit(0)
|
||||||
|
}
|
||||||
|
typ = x
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
typ = table.int_type
|
||||||
|
}
|
||||||
|
node = p.index_expr(node)
|
||||||
}
|
}
|
||||||
else if p.tok.kind.is_infix() {
|
else if p.tok.kind.is_infix() {
|
||||||
node,ti = p.infix_expr(node)
|
node,typ = p.infix_expr(node)
|
||||||
}
|
}
|
||||||
// Postfix
|
// Postfix
|
||||||
else if p.tok.kind in [.inc, .dec] {
|
else if p.tok.kind in [.inc, .dec] {
|
||||||
|
@ -423,16 +438,16 @@ pub fn (p mut Parser) expr(precedence int) (ast.Expr,types.TypeIdent) {
|
||||||
expr: node
|
expr: node
|
||||||
}
|
}
|
||||||
p.next()
|
p.next()
|
||||||
return node,ti
|
return node,typ
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
return node,ti
|
return node,typ
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return node,ti
|
return node,typ
|
||||||
}
|
}
|
||||||
|
|
||||||
fn (p mut Parser) prefix_expr() (ast.Expr,types.TypeIdent) {
|
fn (p mut Parser) prefix_expr() (ast.Expr,table.Type) {
|
||||||
op := p.tok.kind
|
op := p.tok.kind
|
||||||
p.next()
|
p.next()
|
||||||
right,ti := p.expr(1)
|
right,ti := p.expr(1)
|
||||||
|
@ -444,27 +459,29 @@ fn (p mut Parser) prefix_expr() (ast.Expr,types.TypeIdent) {
|
||||||
return expr,ti
|
return expr,ti
|
||||||
}
|
}
|
||||||
|
|
||||||
fn (p mut Parser) index_expr(left ast.Expr) (ast.Expr,types.TypeIdent) {
|
fn (p mut Parser) index_expr(left ast.Expr) ast.Expr {
|
||||||
|
// ,table.Type) {
|
||||||
// println('index expr$p.tok.str() line=$p.tok.line_nr')
|
// println('index expr$p.tok.str() line=$p.tok.line_nr')
|
||||||
p.next()
|
p.next()
|
||||||
println('start expr')
|
println('start index expr')
|
||||||
index,_ := p.expr(0)
|
index,typ := p.expr(0)
|
||||||
println('end expr')
|
println('end expr typ=$typ.name')
|
||||||
p.check(.rsbr)
|
p.check(.rsbr)
|
||||||
println('got ]')
|
println('got ]')
|
||||||
ti := types.int_ti
|
// /ti := table.int_type
|
||||||
mut node := ast.Expr{}
|
mut node := ast.Expr{}
|
||||||
node = ast.IndexExpr{
|
node = ast.IndexExpr{
|
||||||
left: left
|
left: left
|
||||||
index: index
|
index: index
|
||||||
}
|
}
|
||||||
return node,ti
|
return node
|
||||||
|
// /return node,ti
|
||||||
}
|
}
|
||||||
|
|
||||||
fn (p mut Parser) dot_expr(left ast.Expr, left_ti &types.TypeIdent) (ast.Expr,types.TypeIdent) {
|
fn (p mut Parser) dot_expr(left ast.Expr, left_ti &table.Type) (ast.Expr,table.Type) {
|
||||||
p.next()
|
p.next()
|
||||||
field_name := p.check_name()
|
field_name := p.check_name()
|
||||||
ti := types.unresolved_ti
|
ti := table.unresolved_type
|
||||||
// Method call
|
// Method call
|
||||||
if p.tok.kind == .lpar {
|
if p.tok.kind == .lpar {
|
||||||
p.next()
|
p.next()
|
||||||
|
@ -477,9 +494,8 @@ fn (p mut Parser) dot_expr(left ast.Expr, left_ti &types.TypeIdent) (ast.Expr,ty
|
||||||
}
|
}
|
||||||
mut node := ast.Expr{}
|
mut node := ast.Expr{}
|
||||||
node = mcall_expr
|
node = mcall_expr
|
||||||
return node, ti
|
return node,ti
|
||||||
}
|
}
|
||||||
|
|
||||||
sel_expr := ast.SelectorExpr{
|
sel_expr := ast.SelectorExpr{
|
||||||
expr: left
|
expr: left
|
||||||
field: field_name
|
field: field_name
|
||||||
|
@ -487,10 +503,10 @@ fn (p mut Parser) dot_expr(left ast.Expr, left_ti &types.TypeIdent) (ast.Expr,ty
|
||||||
}
|
}
|
||||||
mut node := ast.Expr{}
|
mut node := ast.Expr{}
|
||||||
node = sel_expr
|
node = sel_expr
|
||||||
return node, ti
|
return node,ti
|
||||||
}
|
}
|
||||||
|
|
||||||
fn (p mut Parser) infix_expr(left ast.Expr) (ast.Expr,types.TypeIdent) {
|
fn (p mut Parser) infix_expr(left ast.Expr) (ast.Expr,table.Type) {
|
||||||
op := p.tok.kind
|
op := p.tok.kind
|
||||||
// mut typ := p.
|
// mut typ := p.
|
||||||
// println('infix op=$op.str()')
|
// println('infix op=$op.str()')
|
||||||
|
@ -498,12 +514,13 @@ fn (p mut Parser) infix_expr(left ast.Expr) (ast.Expr,types.TypeIdent) {
|
||||||
p.next()
|
p.next()
|
||||||
right,mut ti := p.expr(precedence)
|
right,mut ti := p.expr(precedence)
|
||||||
if op.is_relational() {
|
if op.is_relational() {
|
||||||
ti = types.bool_ti
|
ti = table.bool_type
|
||||||
}
|
}
|
||||||
mut expr := ast.Expr{}
|
mut expr := ast.Expr{}
|
||||||
expr = ast.BinaryExpr{
|
expr = ast.InfixExpr{
|
||||||
left: left
|
left: left
|
||||||
right: right
|
right: right
|
||||||
|
right_type: ti
|
||||||
op: op
|
op: op
|
||||||
pos: p.tok.position()
|
pos: p.tok.position()
|
||||||
}
|
}
|
||||||
|
@ -549,7 +566,7 @@ fn (p mut Parser) for_statement() ast.Stmt {
|
||||||
}
|
}
|
||||||
p.check(.semicolon)
|
p.check(.semicolon)
|
||||||
if p.tok.kind != .semicolon {
|
if p.tok.kind != .semicolon {
|
||||||
mut typ := types.TypeIdent{}
|
mut typ := table.Type{}
|
||||||
cond,typ = p.expr(0)
|
cond,typ = p.expr(0)
|
||||||
if typ.kind != .bool {
|
if typ.kind != .bool {
|
||||||
p.error('non-bool used as for condition')
|
p.error('non-bool used as for condition')
|
||||||
|
@ -585,7 +602,7 @@ fn (p mut Parser) for_statement() ast.Stmt {
|
||||||
}
|
}
|
||||||
// `for cond {`
|
// `for cond {`
|
||||||
cond,ti := p.expr(0)
|
cond,ti := p.expr(0)
|
||||||
if !p.table.check(types.bool_ti, ti) {
|
if !p.table.check(table.bool_type, ti) {
|
||||||
p.error('non-bool used as for condition')
|
p.error('non-bool used as for condition')
|
||||||
}
|
}
|
||||||
stmts := p.parse_block()
|
stmts := p.parse_block()
|
||||||
|
@ -595,16 +612,17 @@ fn (p mut Parser) for_statement() ast.Stmt {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn (p mut Parser) if_expr() (ast.Expr,types.TypeIdent) {
|
fn (p mut Parser) if_expr() (ast.Expr,table.Type) {
|
||||||
p.inside_if = true
|
p.inside_if = true
|
||||||
// defer {
|
// defer {
|
||||||
// }
|
// }
|
||||||
mut node := ast.Expr{}
|
mut node := ast.Expr{}
|
||||||
p.check(.key_if)
|
p.check(.key_if)
|
||||||
cond,cond_ti := p.expr(0)
|
cond,cond_ti := p.expr(0)
|
||||||
// if !p.table.check(types.bool_ti, cond_ti) {
|
p.inside_if = false
|
||||||
|
// if !p.table.check(table.bool_ti, cond_ti) {
|
||||||
if cond_ti.kind != .bool {
|
if cond_ti.kind != .bool {
|
||||||
p.error('non-bool used as if condition')
|
p.error('non-bool (`$cond_ti.name`) used as if condition')
|
||||||
}
|
}
|
||||||
stmts := p.parse_block()
|
stmts := p.parse_block()
|
||||||
mut else_stmts := []ast.Stmt
|
mut else_stmts := []ast.Stmt
|
||||||
|
@ -612,7 +630,7 @@ fn (p mut Parser) if_expr() (ast.Expr,types.TypeIdent) {
|
||||||
p.check(.key_else)
|
p.check(.key_else)
|
||||||
else_stmts = p.parse_block()
|
else_stmts = p.parse_block()
|
||||||
}
|
}
|
||||||
mut ti := types.void_ti
|
mut ti := table.void_type
|
||||||
// mut left := ast.Expr{}
|
// mut left := ast.Expr{}
|
||||||
// If the last statement is an expression, return its type
|
// If the last statement is an expression, return its type
|
||||||
match stmts[stmts.len - 1] {
|
match stmts[stmts.len - 1] {
|
||||||
|
@ -632,18 +650,17 @@ fn (p mut Parser) if_expr() (ast.Expr,types.TypeIdent) {
|
||||||
// left: left
|
// left: left
|
||||||
|
|
||||||
}
|
}
|
||||||
p.inside_if = false
|
|
||||||
return node,ti
|
return node,ti
|
||||||
}
|
}
|
||||||
|
|
||||||
fn (p mut Parser) string_expr() (ast.Expr,types.TypeIdent) {
|
fn (p mut Parser) string_expr() (ast.Expr,table.Type) {
|
||||||
mut node := ast.Expr{}
|
mut node := ast.Expr{}
|
||||||
node = ast.StringLiteral{
|
node = ast.StringLiteral{
|
||||||
val: p.tok.lit
|
val: p.tok.lit
|
||||||
}
|
}
|
||||||
if p.peek_tok.kind != .str_dollar {
|
if p.peek_tok.kind != .str_dollar {
|
||||||
p.next()
|
p.next()
|
||||||
return node,types.string_ti
|
return node,table.string_type
|
||||||
}
|
}
|
||||||
// Handle $ interpolation
|
// Handle $ interpolation
|
||||||
for p.tok.kind == .str {
|
for p.tok.kind == .str {
|
||||||
|
@ -654,14 +671,14 @@ fn (p mut Parser) string_expr() (ast.Expr,types.TypeIdent) {
|
||||||
p.check(.str_dollar)
|
p.check(.str_dollar)
|
||||||
p.expr(0)
|
p.expr(0)
|
||||||
}
|
}
|
||||||
return node,types.string_ti
|
return node,table.string_type
|
||||||
}
|
}
|
||||||
|
|
||||||
fn (p mut Parser) array_init() (ast.Expr,types.TypeIdent) {
|
fn (p mut Parser) array_init() (ast.Expr,table.Type) {
|
||||||
p.check(.lsbr)
|
p.check(.lsbr)
|
||||||
mut val_ti := types.void_ti
|
mut val_ti := table.void_type
|
||||||
mut exprs := []ast.Expr
|
mut exprs := []ast.Expr
|
||||||
for i:=0; p.tok.kind != .rsbr; i++ {
|
for i := 0; p.tok.kind != .rsbr; i++ {
|
||||||
expr,ti := p.expr(0)
|
expr,ti := p.expr(0)
|
||||||
exprs << expr
|
exprs << expr
|
||||||
if i == 0 {
|
if i == 0 {
|
||||||
|
@ -672,7 +689,7 @@ fn (p mut Parser) array_init() (ast.Expr,types.TypeIdent) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
type_idx,type_name := p.table.find_or_register_array(val_ti, 1)
|
type_idx,type_name := p.table.find_or_register_array(val_ti, 1)
|
||||||
array_ti := types.new_ti(.array, type_name, type_idx, 0)
|
array_ti := table.new_type(.array, type_name, type_idx, 0)
|
||||||
mut node := ast.Expr{}
|
mut node := ast.Expr{}
|
||||||
node = ast.ArrayInit{
|
node = ast.ArrayInit{
|
||||||
ti: array_ti
|
ti: array_ti
|
||||||
|
@ -683,23 +700,23 @@ fn (p mut Parser) array_init() (ast.Expr,types.TypeIdent) {
|
||||||
return node,array_ti
|
return node,array_ti
|
||||||
}
|
}
|
||||||
|
|
||||||
fn (p mut Parser) parse_number_literal() (ast.Expr,types.TypeIdent) {
|
fn (p mut Parser) parse_number_literal() (ast.Expr,table.Type) {
|
||||||
lit := p.tok.lit
|
lit := p.tok.lit
|
||||||
mut node := ast.Expr{}
|
mut node := ast.Expr{}
|
||||||
mut ti := types.int_ti
|
mut ti := table.int_type
|
||||||
if lit.contains('.') {
|
if lit.contains('.') {
|
||||||
node = ast.FloatLiteral{
|
node = ast.FloatLiteral{
|
||||||
// val: lit.f64()
|
// val: lit.f64()
|
||||||
val: lit
|
val: lit
|
||||||
}
|
}
|
||||||
// ti = types.new_builtin_ti(.f64, 0)
|
// ti = table.new_builtin_ti(.f64, 0)
|
||||||
ti = types.new_ti(.f64, 'f64', types.f64_type_idx, 0)
|
ti = table.new_type(.f64, 'f64', table.f64_type_idx, 0)
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
node = ast.IntegerLiteral{
|
node = ast.IntegerLiteral{
|
||||||
val: lit.int()
|
val: lit.int()
|
||||||
}
|
}
|
||||||
// ti = types.int_ti
|
// ti = table.int_ti
|
||||||
}
|
}
|
||||||
p.next()
|
p.next()
|
||||||
return node,ti
|
return node,ti
|
||||||
|
@ -736,14 +753,15 @@ fn (p mut Parser) import_stmt() []ast.Import {
|
||||||
imports << p.parse_import()
|
imports << p.parse_import()
|
||||||
}
|
}
|
||||||
p.check(.rpar)
|
p.check(.rpar)
|
||||||
} else {
|
}
|
||||||
|
else {
|
||||||
imports << p.parse_import()
|
imports << p.parse_import()
|
||||||
}
|
}
|
||||||
return imports
|
return imports
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO
|
// TODO
|
||||||
//fn (p mut Parser) const_decl() ast... {
|
// fn (p mut Parser) const_decl() ast... {
|
||||||
fn (p mut Parser) const_decl() ast.Stmt {
|
fn (p mut Parser) const_decl() ast.Stmt {
|
||||||
p.check(.key_const)
|
p.check(.key_const)
|
||||||
p.check(.lpar)
|
p.check(.lpar)
|
||||||
|
@ -751,7 +769,7 @@ fn (p mut Parser) const_decl() ast.Stmt {
|
||||||
name := p.check_name()
|
name := p.check_name()
|
||||||
println('const: $name')
|
println('const: $name')
|
||||||
p.check(.assign)
|
p.check(.assign)
|
||||||
_, _ := p.expr(0)
|
_,_ := p.expr(0)
|
||||||
// expr, ti := p.expr(0)
|
// expr, ti := p.expr(0)
|
||||||
}
|
}
|
||||||
p.check(.rpar)
|
p.check(.rpar)
|
||||||
|
@ -767,7 +785,7 @@ fn (p mut Parser) struct_decl() ast.StructDecl {
|
||||||
name := p.check_name()
|
name := p.check_name()
|
||||||
p.check(.lcbr)
|
p.check(.lcbr)
|
||||||
mut ast_fields := []ast.Field
|
mut ast_fields := []ast.Field
|
||||||
mut fields := []types.Field
|
mut fields := []table.Field
|
||||||
for p.tok.kind != .rcbr {
|
for p.tok.kind != .rcbr {
|
||||||
if p.tok.kind == .key_pub {
|
if p.tok.kind == .key_pub {
|
||||||
p.check(.key_pub)
|
p.check(.key_pub)
|
||||||
|
@ -779,9 +797,10 @@ fn (p mut Parser) struct_decl() ast.StructDecl {
|
||||||
name: field_name
|
name: field_name
|
||||||
ti: ti
|
ti: ti
|
||||||
}
|
}
|
||||||
fields << types.Field{
|
fields << table.Field{
|
||||||
name: field_name
|
name: field_name
|
||||||
// type_idx: ti.idx
|
// type_idx: ti.idx
|
||||||
|
|
||||||
ti: ti
|
ti: ti
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -789,7 +808,7 @@ fn (p mut Parser) struct_decl() ast.StructDecl {
|
||||||
p.table.register_type(table.Type{
|
p.table.register_type(table.Type{
|
||||||
kind: .struct_
|
kind: .struct_
|
||||||
name: name
|
name: name
|
||||||
info: types.Struct{
|
info: table.Struct{
|
||||||
fields: fields
|
fields: fields
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
@ -806,10 +825,15 @@ fn (p mut Parser) return_stmt() ast.Return {
|
||||||
// return expressions
|
// return expressions
|
||||||
mut exprs := []ast.Expr
|
mut exprs := []ast.Expr
|
||||||
// return type idents
|
// return type idents
|
||||||
// mut got_tis := []types.TypeIdent
|
// mut got_tis := []table.Type
|
||||||
|
if p.return_type.idx == table.void_type_idx {
|
||||||
|
return ast.Return{
|
||||||
|
pos: p.tok.position()
|
||||||
|
}
|
||||||
|
}
|
||||||
for {
|
for {
|
||||||
// expr,ti := p.expr(0)
|
// expr,ti := p.expr(0)
|
||||||
expr, _ := p.expr(0)
|
expr,_ := p.expr(0)
|
||||||
exprs << expr
|
exprs << expr
|
||||||
// got_tis << ti
|
// got_tis << ti
|
||||||
if p.tok.kind == .comma {
|
if p.tok.kind == .comma {
|
||||||
|
@ -821,7 +845,7 @@ fn (p mut Parser) return_stmt() ast.Return {
|
||||||
}
|
}
|
||||||
// TODO: consider non deferred
|
// TODO: consider non deferred
|
||||||
stmt := ast.Return{
|
stmt := ast.Return{
|
||||||
expected_ti: p.return_ti
|
expected_ti: p.return_type
|
||||||
exprs: exprs
|
exprs: exprs
|
||||||
pos: p.tok.position()
|
pos: p.tok.position()
|
||||||
}
|
}
|
||||||
|
@ -841,29 +865,30 @@ fn (p mut Parser) var_decl() ast.VarDecl {
|
||||||
}
|
}
|
||||||
name := p.tok.lit
|
name := p.tok.lit
|
||||||
p.read_first_token()
|
p.read_first_token()
|
||||||
expr, ti := p.expr(token.lowest_prec)
|
expr,typ := p.expr(token.lowest_prec)
|
||||||
if _ := p.table.find_var(name) {
|
if _ := p.table.find_var(name) {
|
||||||
p.error('redefinition of `$name`')
|
p.error('redefinition of `$name`')
|
||||||
}
|
}
|
||||||
p.table.register_var(table.Var{
|
p.table.register_var(table.Var{
|
||||||
name: name
|
name: name
|
||||||
is_mut: is_mut
|
is_mut: is_mut
|
||||||
expr: expr
|
// expr: expr
|
||||||
ti: ti
|
|
||||||
|
typ: typ
|
||||||
})
|
})
|
||||||
|
p.warn('var decl name=$name typ=$typ.name')
|
||||||
// println(p.table.names)
|
// println(p.table.names)
|
||||||
node := ast.VarDecl{
|
node := ast.VarDecl{
|
||||||
name: name
|
name: name
|
||||||
expr: expr // p.expr(token.lowest_prec)
|
expr: expr // p.expr(token.lowest_prec)
|
||||||
|
|
||||||
is_mut: is_mut
|
is_mut: is_mut
|
||||||
ti: ti
|
typ: typ
|
||||||
pos: p.tok.position()
|
pos: p.tok.position()
|
||||||
}
|
}
|
||||||
return node
|
return node
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
fn verror(s string) {
|
fn verror(s string) {
|
||||||
println(s)
|
println(s)
|
||||||
exit(1)
|
exit(1)
|
||||||
|
|
|
@ -5,9 +5,49 @@ import (
|
||||||
v.gen
|
v.gen
|
||||||
v.table
|
v.table
|
||||||
v.checker
|
v.checker
|
||||||
|
v.eval
|
||||||
term
|
term
|
||||||
)
|
)
|
||||||
|
|
||||||
|
fn test_eval() {
|
||||||
|
inputs := [
|
||||||
|
//
|
||||||
|
'2+3',
|
||||||
|
'4',
|
||||||
|
'x := 10',
|
||||||
|
'x',
|
||||||
|
'x + 1',
|
||||||
|
'y := 2',
|
||||||
|
'x * y', // 20
|
||||||
|
//
|
||||||
|
]
|
||||||
|
expected := [
|
||||||
|
//
|
||||||
|
'5',
|
||||||
|
'4',
|
||||||
|
'>>',
|
||||||
|
'10',
|
||||||
|
'11',
|
||||||
|
'>>',
|
||||||
|
'20',
|
||||||
|
//
|
||||||
|
]
|
||||||
|
table := table.new_table()
|
||||||
|
mut stmts := []ast.Stmt
|
||||||
|
for input in inputs {
|
||||||
|
stmts << parse_stmt(input, table)
|
||||||
|
}
|
||||||
|
file := ast.File{
|
||||||
|
stmts: stmts
|
||||||
|
}
|
||||||
|
mut ev := eval.Eval{}
|
||||||
|
s := ev.eval(file, table)
|
||||||
|
println('eval done')
|
||||||
|
println(s)
|
||||||
|
assert s == expected.join('\n')
|
||||||
|
exit(0)
|
||||||
|
}
|
||||||
|
|
||||||
fn test_parse_file() {
|
fn test_parse_file() {
|
||||||
if true {
|
if true {
|
||||||
return
|
return
|
||||||
|
@ -55,7 +95,9 @@ fn test_one() {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn test_parse_expr() {
|
fn test_parse_expr() {
|
||||||
|
println('SDFSDFSDF')
|
||||||
input := ['1 == 1',
|
input := ['1 == 1',
|
||||||
|
'234234',
|
||||||
'2 * 8 + 3',
|
'2 * 8 + 3',
|
||||||
'a := 3',
|
'a := 3',
|
||||||
'a++',
|
'a++',
|
||||||
|
@ -79,7 +121,8 @@ fn test_parse_expr() {
|
||||||
'1.2 + 3.4',
|
'1.2 + 3.4',
|
||||||
'4 + 4',
|
'4 + 4',
|
||||||
'1 + 2 * 5',
|
'1 + 2 * 5',
|
||||||
'-a',
|
'-a+1',
|
||||||
|
'2+2',
|
||||||
/*
|
/*
|
||||||
/*
|
/*
|
||||||
'(2 * 3) / 2',
|
'(2 * 3) / 2',
|
||||||
|
@ -93,6 +136,7 @@ fn test_parse_expr() {
|
||||||
|
|
||||||
]
|
]
|
||||||
expecting := ['1 == 1;',
|
expecting := ['1 == 1;',
|
||||||
|
'234234;',
|
||||||
'2 * 8 + 3;',
|
'2 * 8 + 3;',
|
||||||
'int a = 3;',
|
'int a = 3;',
|
||||||
'a++;',
|
'a++;',
|
||||||
|
@ -116,13 +160,14 @@ fn test_parse_expr() {
|
||||||
'1.2 + 3.4;',
|
'1.2 + 3.4;',
|
||||||
'4 + 4;',
|
'4 + 4;',
|
||||||
'1 + 2 * 5;',
|
'1 + 2 * 5;',
|
||||||
'-a;',
|
'-a + 1;',
|
||||||
|
'2 + 2;',
|
||||||
]
|
]
|
||||||
mut e := []ast.Stmt
|
mut e := []ast.Stmt
|
||||||
table := table.new_table()
|
table := table.new_table()
|
||||||
mut checker := checker.new_checker(table)
|
mut checker := checker.new_checker(table)
|
||||||
for s in input {
|
for s in input {
|
||||||
// println('\n\nst="$s"')
|
println('\n\nst="$s"')
|
||||||
e << parse_stmt(s, table)
|
e << parse_stmt(s, table)
|
||||||
}
|
}
|
||||||
program := ast.File{
|
program := ast.File{
|
||||||
|
|
|
@ -1,7 +1,21 @@
|
||||||
// Copyright (c) 2019-2020 Alexander Medvednikov. All rights reserved.
|
// Copyright (c) 2019 Alexander Medvednikov. All rights reserved.
|
||||||
// Use of this source code is governed by an MIT license
|
// Use of this source code is governed by an MIT license
|
||||||
// that can be found in the LICENSE file.
|
// that can be found in the LICENSE file.
|
||||||
module types
|
module table
|
||||||
|
|
||||||
|
pub type TypeInfo = Array | ArrayFixed | Map | Struct | MultiReturn | Variadic
|
||||||
|
|
||||||
|
pub struct Type {
|
||||||
|
pub:
|
||||||
|
idx int
|
||||||
|
parent_idx int
|
||||||
|
mut:
|
||||||
|
info TypeInfo
|
||||||
|
kind Kind
|
||||||
|
name string
|
||||||
|
methods []Fn
|
||||||
|
nr_muls int
|
||||||
|
}
|
||||||
|
|
||||||
pub const (
|
pub const (
|
||||||
void_type_idx = 1
|
void_type_idx = 1
|
||||||
|
@ -50,23 +64,53 @@ pub enum Kind {
|
||||||
map
|
map
|
||||||
multi_return
|
multi_return
|
||||||
variadic
|
variadic
|
||||||
unresolved
|
unresolved
|
||||||
}
|
}
|
||||||
|
|
||||||
pub type TypeInfo = Array | ArrayFixed | Map | Struct | MultiReturn | Variadic
|
pub const (
|
||||||
|
unresolved_type = Type{
|
||||||
|
kind: .unresolved
|
||||||
|
name: 'unresolved'
|
||||||
|
}
|
||||||
|
void_type = Type{
|
||||||
|
kind: .void
|
||||||
|
name: 'void'
|
||||||
|
idx: void_type_idx
|
||||||
|
}
|
||||||
|
int_type = Type{
|
||||||
|
kind: .int
|
||||||
|
name: 'int'
|
||||||
|
idx: int_type_idx
|
||||||
|
}
|
||||||
|
string_type = Type{
|
||||||
|
kind: .string
|
||||||
|
name: 'string'
|
||||||
|
idx: string_type_idx
|
||||||
|
}
|
||||||
|
bool_type = Type{
|
||||||
|
kind: .bool
|
||||||
|
name: 'bool'
|
||||||
|
idx: bool_type_idx
|
||||||
|
}
|
||||||
|
)
|
||||||
|
/*
|
||||||
|
pub fn (t Type) str() string {
|
||||||
|
return t.name
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
pub struct TypeIdent {
|
|
||||||
pub:
|
pub fn (t &Type) str() string {
|
||||||
idx int
|
mut muls := ''
|
||||||
mut:
|
for _ in 0 .. t.nr_muls {
|
||||||
kind Kind
|
muls += '&'
|
||||||
name string
|
}
|
||||||
nr_muls int
|
// return '$muls$ti.name'
|
||||||
|
return '$muls$t.idx'
|
||||||
}
|
}
|
||||||
|
|
||||||
[inline]
|
pub fn new_type(kind Kind, name string, idx int, nr_muls int) Type {
|
||||||
pub fn new_ti(kind Kind, name string, idx int, nr_muls int) TypeIdent {
|
return Type{
|
||||||
return TypeIdent{
|
|
||||||
idx: idx
|
idx: idx
|
||||||
kind: kind
|
kind: kind
|
||||||
name: name
|
name: name
|
||||||
|
@ -74,53 +118,103 @@ pub fn new_ti(kind Kind, name string, idx int, nr_muls int) TypeIdent {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// [inline]
|
pub fn (t mut Table) register_builtin_types() {
|
||||||
// pub fn new_builtin_ti(kind Kind, nr_muls int) TypeIdent {
|
// reserve index 0 so nothing can go there
|
||||||
// return TypeIdent{
|
// save index check, 0 will mean not found
|
||||||
// idx: -int(kind) - 1
|
t.register_type(Type{
|
||||||
// kind: kind
|
kind: .placeholder
|
||||||
// name: kind.str()
|
name: 'reserved_0'
|
||||||
// nr_muls: nr_muls
|
})
|
||||||
// }
|
t.register_type(Type{
|
||||||
// }
|
kind: .void
|
||||||
|
name: 'void'
|
||||||
pub const (
|
})
|
||||||
unresolved_ti = new_ti(.unresolved, 'unresolved', 0, 0)
|
t.register_type(Type{
|
||||||
void_ti = new_ti(.void, 'void', void_type_idx, 0)
|
kind: .voidptr
|
||||||
int_ti = new_ti(.int, 'int', int_type_idx, 0)
|
name: 'voidptr'
|
||||||
string_ti = new_ti(.string, 'string', string_type_idx, 0)
|
})
|
||||||
bool_ti = new_ti(.bool, 'bool', bool_type_idx, 0)
|
t.register_type(Type{
|
||||||
)
|
kind: .charptr
|
||||||
|
name: 'charptr'
|
||||||
|
})
|
||||||
|
t.register_type(Type{
|
||||||
|
kind: .byteptr
|
||||||
|
name: 'byteptr'
|
||||||
|
})
|
||||||
|
t.register_type(Type{
|
||||||
|
kind: .i8
|
||||||
|
name: 'i8'
|
||||||
|
})
|
||||||
|
t.register_type(Type{
|
||||||
|
kind: .i16
|
||||||
|
name: 'i16'
|
||||||
|
})
|
||||||
|
t.register_type(Type{
|
||||||
|
kind: .int
|
||||||
|
name: 'int'
|
||||||
|
})
|
||||||
|
t.register_type(Type{
|
||||||
|
kind: .i64
|
||||||
|
name: 'i64'
|
||||||
|
})
|
||||||
|
t.register_type(Type{
|
||||||
|
kind: .u16
|
||||||
|
name: 'u16'
|
||||||
|
})
|
||||||
|
t.register_type(Type{
|
||||||
|
kind: .u32
|
||||||
|
name: 'u32'
|
||||||
|
})
|
||||||
|
t.register_type(Type{
|
||||||
|
kind: .u64
|
||||||
|
name: 'u64'
|
||||||
|
})
|
||||||
|
t.register_type(Type{
|
||||||
|
kind: .f32
|
||||||
|
name: 'f32'
|
||||||
|
})
|
||||||
|
t.register_type(Type{
|
||||||
|
kind: .f64
|
||||||
|
name: 'f64'
|
||||||
|
})
|
||||||
|
t.register_type(Type{
|
||||||
|
kind: .string
|
||||||
|
name: 'string'
|
||||||
|
})
|
||||||
|
t.register_type(Type{
|
||||||
|
kind: .char
|
||||||
|
name: 'char'
|
||||||
|
})
|
||||||
|
t.register_type(Type{
|
||||||
|
kind: .byte
|
||||||
|
name: 'byte'
|
||||||
|
})
|
||||||
|
t.register_type(Type{
|
||||||
|
kind: .bool
|
||||||
|
name: 'bool'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
[inline]
|
[inline]
|
||||||
pub fn (ti &TypeIdent) is_ptr() bool {
|
pub fn (ti &Type) is_ptr() bool {
|
||||||
return ti.nr_muls > 0
|
return ti.nr_muls > 0
|
||||||
}
|
}
|
||||||
|
|
||||||
[inline]
|
[inline]
|
||||||
pub fn (ti &TypeIdent) is_int() bool {
|
pub fn (ti &Type) is_int() bool {
|
||||||
return ti.kind in [.i8, .i16, .int, .i64, .byte, .u16, .u32, .u64]
|
return ti.kind in [.i8, .i16, .int, .i64, .byte, .u16, .u32, .u64]
|
||||||
}
|
}
|
||||||
|
|
||||||
[inline]
|
[inline]
|
||||||
pub fn (ti &TypeIdent) is_float() bool {
|
pub fn (ti &Type) is_float() bool {
|
||||||
return ti.kind in [.f32, .f64]
|
return ti.kind in [.f32, .f64]
|
||||||
}
|
}
|
||||||
|
|
||||||
[inline]
|
[inline]
|
||||||
pub fn (ti &TypeIdent) is_number() bool {
|
pub fn (ti &Type) is_number() bool {
|
||||||
return ti.is_int() || ti.is_float()
|
return ti.is_int() || ti.is_float()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (ti &TypeIdent) str() string {
|
|
||||||
mut muls := ''
|
|
||||||
for _ in 0 .. ti.nr_muls {
|
|
||||||
muls += '&'
|
|
||||||
}
|
|
||||||
// return '$muls$ti.name'
|
|
||||||
return '$muls$ti.idx'
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn (k Kind) str() string {
|
pub fn (k Kind) str() string {
|
||||||
k_str := match k {
|
k_str := match k {
|
||||||
.unresolved{
|
.unresolved{
|
||||||
|
@ -142,10 +236,10 @@ pub fn (k Kind) str() string {
|
||||||
'byteptr'
|
'byteptr'
|
||||||
}
|
}
|
||||||
// .const_{
|
// .const_{
|
||||||
// 'const'
|
// 'const'
|
||||||
// }
|
// }
|
||||||
// .enum_{
|
// .enum_{
|
||||||
// 'enum'
|
// 'enum'
|
||||||
// }
|
// }
|
||||||
.struct_{
|
.struct_{
|
||||||
'struct'
|
'struct'
|
||||||
|
@ -217,38 +311,32 @@ pub fn (kinds []Kind) str() string {
|
||||||
|
|
||||||
// pub struct Const {
|
// pub struct Const {
|
||||||
// pub:
|
// pub:
|
||||||
// name string
|
// name string
|
||||||
// }
|
// }
|
||||||
|
|
||||||
// pub struct Enum {
|
// pub struct Enum {
|
||||||
// pub:
|
// pub:
|
||||||
// name string
|
// name string
|
||||||
// }
|
// }
|
||||||
|
|
||||||
pub struct Struct {
|
pub struct Struct {
|
||||||
pub mut:
|
pub mut:
|
||||||
fields []Field
|
fields []Field
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct Field {
|
pub struct Field {
|
||||||
pub:
|
pub:
|
||||||
name string
|
name string
|
||||||
ti TypeIdent
|
ti Type
|
||||||
// type_idx int
|
// type_idx int
|
||||||
}
|
}
|
||||||
|
|
||||||
// pub struct Int {
|
// pub struct Int {
|
||||||
// pub:
|
// pub:
|
||||||
// bit_size u32
|
// bit_size u32
|
||||||
// is_unsigned bool
|
// is_unsigned bool
|
||||||
// }
|
// }
|
||||||
|
|
||||||
// pub struct Float {
|
// pub struct Float {
|
||||||
// pub:
|
// pub:
|
||||||
// bit_size u32
|
// bit_size u32
|
||||||
// }
|
// }
|
||||||
|
|
||||||
|
|
||||||
pub struct Array {
|
pub struct Array {
|
||||||
pub:
|
pub:
|
||||||
elem_type_kind Kind
|
elem_type_kind Kind
|
||||||
|
@ -277,10 +365,32 @@ pub:
|
||||||
pub struct MultiReturn {
|
pub struct MultiReturn {
|
||||||
pub:
|
pub:
|
||||||
name string
|
name string
|
||||||
tis []TypeIdent
|
tis []Type
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct Variadic {
|
pub struct Variadic {
|
||||||
pub:
|
pub:
|
||||||
ti TypeIdent
|
ti Type
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn (t &Table) refresh_ti(ti Type) Type {
|
||||||
|
if ti.idx == 0 {
|
||||||
|
return ti
|
||||||
|
}
|
||||||
|
if ti.kind in [.placeholder, .unresolved] {
|
||||||
|
typ := t.types[ti.idx]
|
||||||
|
return {
|
||||||
|
ti |
|
||||||
|
kind:typ.kind,
|
||||||
|
name:typ.name
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ti
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn (t &Table) get_type(idx int) Type {
|
||||||
|
if idx == 0 {
|
||||||
|
panic('get_type: idx 0')
|
||||||
|
}
|
||||||
|
return t.types[idx]
|
||||||
}
|
}
|
|
@ -1,51 +1,37 @@
|
||||||
|
// Copyright (c) 2019 Alexander Medvednikov. All rights reserved.
|
||||||
|
// Use of this source code is governed by an MIT license
|
||||||
|
// that can be found in the LICENSE file.
|
||||||
module table
|
module table
|
||||||
|
// import (
|
||||||
import (
|
// v.ast
|
||||||
v.ast
|
// )
|
||||||
v.types
|
|
||||||
)
|
|
||||||
|
|
||||||
pub struct Table {
|
pub struct Table {
|
||||||
// struct_fields map[string][]string
|
// struct_fields map[string][]string
|
||||||
pub mut:
|
pub mut:
|
||||||
types []Type
|
types []Type
|
||||||
// type_idxs Hashmap
|
// type_idxs Hashmap
|
||||||
type_idxs map[string]int
|
type_idxs map[string]int
|
||||||
local_vars []Var
|
local_vars []Var
|
||||||
// fns Hashmap
|
// fns Hashmap
|
||||||
fns map[string]Fn
|
fns map[string]Fn
|
||||||
tmp_cnt int
|
tmp_cnt int
|
||||||
imports []string
|
imports []string
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct Fn {
|
pub struct Fn {
|
||||||
pub:
|
pub:
|
||||||
name string
|
name string
|
||||||
args []Var
|
args []Var
|
||||||
return_ti types.TypeIdent
|
return_type Type
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct Var {
|
pub struct Var {
|
||||||
pub:
|
pub:
|
||||||
name string
|
name string
|
||||||
is_mut bool
|
is_mut bool
|
||||||
expr ast.Expr
|
// expr ast.Expr
|
||||||
mut:
|
mut:
|
||||||
ti types.TypeIdent
|
typ Type
|
||||||
}
|
|
||||||
|
|
||||||
pub struct Type {
|
|
||||||
pub:
|
|
||||||
parent_idx int
|
|
||||||
mut:
|
|
||||||
info types.TypeInfo
|
|
||||||
kind types.Kind
|
|
||||||
name string
|
|
||||||
methods []Fn
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn (t Type) str() string {
|
|
||||||
return t.name
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn new_table() &Table {
|
pub fn new_table() &Table {
|
||||||
|
@ -54,52 +40,8 @@ pub fn new_table() &Table {
|
||||||
return t
|
return t
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (t mut Table) register_builtin_types() {
|
|
||||||
// reserve index 0 so nothing can go there
|
|
||||||
// save index check, 0 will mean not found
|
|
||||||
t.register_type(Type{kind: .placeholder, name: 'reserved_0'})
|
|
||||||
t.register_type(Type{kind: .void, name: 'void'})
|
|
||||||
t.register_type(Type{kind: .voidptr, name: 'voidptr'})
|
|
||||||
t.register_type(Type{kind: .charptr, name: 'charptr'})
|
|
||||||
t.register_type(Type{kind: .byteptr, name: 'byteptr'})
|
|
||||||
t.register_type(Type{kind: .i8, name: 'i8'})
|
|
||||||
t.register_type(Type{kind: .i16, name: 'i16'})
|
|
||||||
t.register_type(Type{kind: .int, name: 'int'})
|
|
||||||
t.register_type(Type{kind: .i64, name: 'i64'})
|
|
||||||
t.register_type(Type{kind: .u16, name: 'u16'})
|
|
||||||
t.register_type(Type{kind: .u32, name: 'u32'})
|
|
||||||
t.register_type(Type{kind: .u64, name: 'u64'})
|
|
||||||
t.register_type(Type{kind: .f32, name: 'f32'})
|
|
||||||
t.register_type(Type{kind: .f64, name: 'f64'})
|
|
||||||
t.register_type(Type{kind: .string, name: 'string'})
|
|
||||||
t.register_type(Type{kind: .char, name: 'char'})
|
|
||||||
t.register_type(Type{kind: .byte, name: 'byte'})
|
|
||||||
t.register_type(Type{kind: .bool, name: 'bool'})
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn (t &Table) refresh_ti(ti types.TypeIdent) types.TypeIdent {
|
|
||||||
if ti.idx == 0 {
|
|
||||||
return ti
|
|
||||||
}
|
|
||||||
if ti.kind in [.placeholder, .unresolved] {
|
|
||||||
typ := t.types[ti.idx]
|
|
||||||
return { ti|
|
|
||||||
kind: typ.kind,
|
|
||||||
name: typ.name
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return ti
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn (t &Table) get_type(idx int) Type {
|
|
||||||
if idx == 0 {
|
|
||||||
panic('get_type: idx 0')
|
|
||||||
}
|
|
||||||
return t.types[idx]
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn (t &Table) find_var_idx(name string) int {
|
pub fn (t &Table) find_var_idx(name string) int {
|
||||||
for i,var in t.local_vars {
|
for i, var in t.local_vars {
|
||||||
if var.name == name {
|
if var.name == name {
|
||||||
return i
|
return i
|
||||||
}
|
}
|
||||||
|
@ -137,7 +79,7 @@ pub fn (t mut Table) clear_vars() {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (t mut Table) register_var(v Var) {
|
pub fn (t mut Table) register_var(v Var) {
|
||||||
println('register_var: $v.name - $v.ti.name')
|
println('register_var: $v.name - $v.typ.name')
|
||||||
t.local_vars << v
|
t.local_vars << v
|
||||||
/*
|
/*
|
||||||
mut new_var := {
|
mut new_var := {
|
||||||
|
@ -175,9 +117,9 @@ pub fn (t mut Table) register_fn(new_fn Fn) {
|
||||||
t.fns[new_fn.name] = new_fn
|
t.fns[new_fn.name] = new_fn
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (t mut Table) register_method(ti &types.TypeIdent, new_fn Fn) bool {
|
pub fn (t mut Table) register_method(typ &Type, new_fn Fn) bool {
|
||||||
idx := ti.idx
|
idx := typ.idx
|
||||||
println('register method `$new_fn.name` type=$ti.name idx=$ti.idx')
|
println('register method `$new_fn.name` type=$typ.name idx=$typ.idx')
|
||||||
mut methods := t.types[idx].methods
|
mut methods := t.types[idx].methods
|
||||||
methods << new_fn
|
methods << new_fn
|
||||||
t.types[idx].methods = methods
|
t.types[idx].methods = methods
|
||||||
|
@ -216,9 +158,9 @@ pub fn (t &Table) struct_has_field(s &Type, name string) bool {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (t &Table) struct_find_field(s &Type, name string) ?types.Field {
|
pub fn (t &Table) struct_find_field(s &Type, name string) ?Field {
|
||||||
println('struct_find_field($s.name, $name) types.len=$t.types.len s.parent_idx=$s.parent_idx')
|
println('struct_find_field($s.name, $name) types.len=$t.types.len s.parent_idx=$s.parent_idx')
|
||||||
info := s.info as types.Struct
|
info := s.info as Struct
|
||||||
for field in info.fields {
|
for field in info.fields {
|
||||||
if field.name == name {
|
if field.name == name {
|
||||||
return field
|
return field
|
||||||
|
@ -226,7 +168,7 @@ pub fn (t &Table) struct_find_field(s &Type, name string) ?types.Field {
|
||||||
}
|
}
|
||||||
if s.parent_idx != 0 {
|
if s.parent_idx != 0 {
|
||||||
parent := t.types[s.parent_idx]
|
parent := t.types[s.parent_idx]
|
||||||
parent_info := s.info as types.Struct
|
parent_info := s.info as Struct
|
||||||
println('got parent $parent.name')
|
println('got parent $parent.name')
|
||||||
for field in parent_info.fields {
|
for field in parent_info.fields {
|
||||||
if field.name == name {
|
if field.name == name {
|
||||||
|
@ -260,8 +202,9 @@ pub fn (t mut Table) register_type(typ Type) int {
|
||||||
.placeholder {
|
.placeholder {
|
||||||
// override placeholder
|
// override placeholder
|
||||||
println('overriding type placeholder `$typ.name`')
|
println('overriding type placeholder `$typ.name`')
|
||||||
t.types[existing_idx] = {typ|
|
t.types[existing_idx] = {
|
||||||
methods: ex_type.methods
|
typ |
|
||||||
|
methods:ex_type.methods
|
||||||
}
|
}
|
||||||
return existing_idx
|
return existing_idx
|
||||||
}
|
}
|
||||||
|
@ -271,7 +214,7 @@ pub fn (t mut Table) register_type(typ Type) int {
|
||||||
}
|
}
|
||||||
panic('cannot register type `$typ.name`, another type with this name exists')
|
panic('cannot register type `$typ.name`, another type with this name exists')
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
idx := t.types.len
|
idx := t.types.len
|
||||||
t.types << typ
|
t.types << typ
|
||||||
|
@ -279,8 +222,8 @@ pub fn (t mut Table) register_type(typ Type) int {
|
||||||
return idx
|
return idx
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (t mut Table) find_or_register_map(key_ti &types.TypeIdent, value_ti &types.TypeIdent) (int,string) {
|
pub fn (t mut Table) find_or_register_map(key_typ &Type, value_typ &Type) (int,string) {
|
||||||
name := 'map_${key_ti.name}_${value_ti.name}'
|
name := 'map_${key_typ.name}_${value_typ.name}'
|
||||||
// existing
|
// existing
|
||||||
existing_idx := t.type_idxs[name]
|
existing_idx := t.type_idxs[name]
|
||||||
if existing_idx > 0 {
|
if existing_idx > 0 {
|
||||||
|
@ -290,17 +233,17 @@ pub fn (t mut Table) find_or_register_map(key_ti &types.TypeIdent, value_ti &typ
|
||||||
map_type := Type{
|
map_type := Type{
|
||||||
kind: .map
|
kind: .map
|
||||||
name: name
|
name: name
|
||||||
info: types.Map{
|
info: Map{
|
||||||
key_type_idx: key_ti.idx
|
key_type_idx: key_typ.idx
|
||||||
value_type_idx: value_ti.idx
|
value_type_idx: value_typ.idx
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
idx := t.register_type(map_type)
|
idx := t.register_type(map_type)
|
||||||
return idx,name
|
return idx,name
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (t mut Table) find_or_register_array(elem_ti &types.TypeIdent, nr_dims int) (int,string) {
|
pub fn (t mut Table) find_or_register_array(elem_typ &Type, nr_dims int) (int,string) {
|
||||||
name := 'array_${elem_ti.name}' + if nr_dims > 1 { '_${nr_dims}d' } else { '' }
|
name := 'array_${elem_typ.name}' + if nr_dims > 1 { '_${nr_dims}d' } else { '' }
|
||||||
// existing
|
// existing
|
||||||
existing_idx := t.type_idxs[name]
|
existing_idx := t.type_idxs[name]
|
||||||
if existing_idx > 0 {
|
if existing_idx > 0 {
|
||||||
|
@ -312,9 +255,9 @@ pub fn (t mut Table) find_or_register_array(elem_ti &types.TypeIdent, nr_dims in
|
||||||
parent_idx: parent_idx
|
parent_idx: parent_idx
|
||||||
kind: .array
|
kind: .array
|
||||||
name: name
|
name: name
|
||||||
info: types.Array{
|
info: Array{
|
||||||
elem_type_idx: elem_ti.idx
|
elem_type_idx: elem_typ.idx
|
||||||
elem_is_ptr: elem_ti.is_ptr()
|
elem_is_ptr: elem_typ.is_ptr()
|
||||||
nr_dims: nr_dims
|
nr_dims: nr_dims
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -322,8 +265,8 @@ pub fn (t mut Table) find_or_register_array(elem_ti &types.TypeIdent, nr_dims in
|
||||||
return idx,name
|
return idx,name
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (t mut Table) find_or_register_array_fixed(elem_ti &types.TypeIdent, size int, nr_dims int) (int,string) {
|
pub fn (t mut Table) find_or_register_array_fixed(elem_typ &Type, size int, nr_dims int) (int,string) {
|
||||||
name := 'array_fixed_${elem_ti.name}_${size}' + if nr_dims > 1 { '_${nr_dims}d' } else { '' }
|
name := 'array_fixed_${elem_typ.name}_${size}' + if nr_dims > 1 { '_${nr_dims}d' } else { '' }
|
||||||
// existing
|
// existing
|
||||||
existing_idx := t.type_idxs[name]
|
existing_idx := t.type_idxs[name]
|
||||||
if existing_idx > 0 {
|
if existing_idx > 0 {
|
||||||
|
@ -333,9 +276,9 @@ pub fn (t mut Table) find_or_register_array_fixed(elem_ti &types.TypeIdent, size
|
||||||
array_fixed_type := Type{
|
array_fixed_type := Type{
|
||||||
kind: .array_fixed
|
kind: .array_fixed
|
||||||
name: name
|
name: name
|
||||||
info: types.ArrayFixed{
|
info: ArrayFixed{
|
||||||
elem_type_idx: elem_ti.idx
|
elem_type_idx: elem_typ.idx
|
||||||
elem_is_ptr: elem_ti.is_ptr()
|
elem_is_ptr: elem_typ.is_ptr()
|
||||||
size: size
|
size: size
|
||||||
nr_dims: nr_dims
|
nr_dims: nr_dims
|
||||||
}
|
}
|
||||||
|
@ -344,10 +287,10 @@ pub fn (t mut Table) find_or_register_array_fixed(elem_ti &types.TypeIdent, size
|
||||||
return idx,name
|
return idx,name
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (t mut Table) find_or_register_multi_return(mr_tis []types.TypeIdent) (int,string) {
|
pub fn (t mut Table) find_or_register_multi_return(mr_typs []Type) (int,string) {
|
||||||
mut name := 'multi_return'
|
mut name := 'multi_return'
|
||||||
for mr_ti in mr_tis {
|
for mr_typ in mr_typs {
|
||||||
name += '_$mr_ti.name'
|
name += '_$mr_typ.name'
|
||||||
}
|
}
|
||||||
// existing
|
// existing
|
||||||
existing_idx := t.type_idxs[name]
|
existing_idx := t.type_idxs[name]
|
||||||
|
@ -358,16 +301,16 @@ pub fn (t mut Table) find_or_register_multi_return(mr_tis []types.TypeIdent) (in
|
||||||
mr_type := Type{
|
mr_type := Type{
|
||||||
kind: .multi_return
|
kind: .multi_return
|
||||||
name: name
|
name: name
|
||||||
info: types.MultiReturn{
|
info: MultiReturn{
|
||||||
tis: mr_tis
|
tis: mr_typs
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
idx := t.register_type(mr_type)
|
idx := t.register_type(mr_type)
|
||||||
return idx,name
|
return idx,name
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (t mut Table) find_or_register_variadic(variadic_ti &types.TypeIdent) (int,string) {
|
pub fn (t mut Table) find_or_register_variadic(variadic_typ &Type) (int,string) {
|
||||||
name := 'variadic_$variadic_ti.name'
|
name := 'variadic_$variadic_typ.name'
|
||||||
// existing
|
// existing
|
||||||
existing_idx := t.type_idxs[name]
|
existing_idx := t.type_idxs[name]
|
||||||
if existing_idx > 0 {
|
if existing_idx > 0 {
|
||||||
|
@ -377,8 +320,8 @@ pub fn (t mut Table) find_or_register_variadic(variadic_ti &types.TypeIdent) (in
|
||||||
variadic_type := Type{
|
variadic_type := Type{
|
||||||
kind: .variadic
|
kind: .variadic
|
||||||
name: name
|
name: name
|
||||||
info: types.Variadic{
|
info: Variadic{
|
||||||
ti: variadic_ti
|
ti: variadic_typ
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
idx := t.register_type(variadic_type)
|
idx := t.register_type(variadic_type)
|
||||||
|
@ -396,77 +339,76 @@ pub fn (t mut Table) add_placeholder_type(name string) int {
|
||||||
}
|
}
|
||||||
|
|
||||||
// [inline]
|
// [inline]
|
||||||
// pub fn (t &Table) update_ti(ti &types.TypeIdent) types.TypeIdent {
|
// pub fn (t &Table) update_typ(ti &types.Type) types.Type {
|
||||||
// if ti.kind == .unresolved {
|
// if typ.kind == .unresolved {
|
||||||
|
|
||||||
// }
|
|
||||||
// }
|
// }
|
||||||
|
// }
|
||||||
pub fn (t &Table) check(got, expected &types.TypeIdent) bool {
|
pub fn (t &Table) check(got, expected &Type) bool {
|
||||||
println('check: $got.name, $expected.name')
|
println('check: $got.name, $expected.name')
|
||||||
if expected.kind == .voidptr {
|
if expected.kind == .voidptr {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
//if expected.name == 'array' {
|
// if expected.name == 'array' {
|
||||||
// return true
|
// return true
|
||||||
//}
|
// }
|
||||||
if got.idx != expected.idx {
|
if got.idx != expected.idx {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
[inline]
|
[inline]
|
||||||
pub fn (t &Table) get_expr_ti(expr ast.Expr) types.TypeIdent {
|
pub fn (t &Table) get_expr_typ(expr ast.Expr) Type {
|
||||||
match expr {
|
match expr {
|
||||||
ast.ArrayInit{
|
ast.ArrayInit {
|
||||||
return it.ti
|
return it.typ
|
||||||
}
|
}
|
||||||
ast.IndexExpr{
|
ast.IndexExpr {
|
||||||
return t.get_expr_ti(it.left)
|
return t.get_expr_typ(it.left)
|
||||||
}
|
}
|
||||||
ast.CallExpr {
|
ast.CallExpr {
|
||||||
func := t.find_fn(it.name) or {
|
func := t.find_fn(it.name) or {
|
||||||
return types.void_ti
|
return void_typ
|
||||||
}
|
}
|
||||||
return func.return_ti
|
return func.return_typ
|
||||||
}
|
}
|
||||||
ast.MethodCallExpr {
|
ast.MethodCallExpr {
|
||||||
ti := t.get_expr_ti(it.expr)
|
ti := t.get_expr_typ(it.expr)
|
||||||
func := t.find_method(ti.idx, it.name) or {
|
func := t.find_method(typ.idx, it.name) or {
|
||||||
return types.void_ti
|
return void_type
|
||||||
}
|
}
|
||||||
return func.return_ti
|
return func.return_typ
|
||||||
}
|
}
|
||||||
ast.Ident {
|
ast.Ident {
|
||||||
if it.kind == .variable {
|
if it.kind == .variable {
|
||||||
info := it.info as ast.IdentVar
|
info := it.info as ast.IdentVar
|
||||||
if info.ti.kind != .unresolved {
|
if info.typ.kind != .unresolved {
|
||||||
return info.ti
|
return info.ti
|
||||||
}
|
}
|
||||||
return t.get_expr_ti(info.expr)
|
return t.get_expr_typ(info.expr)
|
||||||
}
|
}
|
||||||
return types.void_ti
|
return types.void_typ
|
||||||
}
|
}
|
||||||
ast.StructInit {
|
ast.StructInit {
|
||||||
return it.ti
|
return it.ti
|
||||||
}
|
}
|
||||||
ast.StringLiteral {
|
ast.StringLiteral {
|
||||||
return types.string_ti
|
return types.string_typ
|
||||||
}
|
}
|
||||||
ast.IntegerLiteral {
|
ast.IntegerLiteral {
|
||||||
return types.int_ti
|
return types.int_typ
|
||||||
}
|
}
|
||||||
ast.SelectorExpr {
|
ast.SelectorExpr {
|
||||||
ti := t.get_expr_ti(it.expr)
|
ti := t.get_expr_typ(it.expr)
|
||||||
kind := t.types[ti.idx].kind
|
kind := t.types[typ.idx].kind
|
||||||
if ti.kind == .placeholder {
|
if typ.kind == .placeholder {
|
||||||
println(' ##### PH $ti.name')
|
println(' ##### PH $typ.name')
|
||||||
}
|
}
|
||||||
if !(kind in [.placeholder, .struct_]) {
|
if !(kind in [.placeholder, .struct_]) {
|
||||||
return types.void_ti
|
return types.void_typ
|
||||||
}
|
}
|
||||||
struct_ := t.types[ti.idx]
|
struct_ := t.types[typ.idx]
|
||||||
struct_info := struct_.info as types.Struct
|
struct_info := struct_.info as types.Struct
|
||||||
for field in struct_info.fields {
|
for field in struct_info.fields {
|
||||||
if field.name == it.field {
|
if field.name == it.field {
|
||||||
|
@ -482,13 +424,14 @@ pub fn (t &Table) get_expr_ti(expr ast.Expr) types.TypeIdent {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return types.void_ti
|
return types.void_typ
|
||||||
}
|
}
|
||||||
ast.BinaryExpr {
|
ast.InfixExpr {
|
||||||
return t.get_expr_ti(it.left)
|
return t.get_expr_typ(it.left)
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
return types.void_ti
|
return types.void_typ
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
|
|
|
@ -12,7 +12,8 @@ pub:
|
||||||
[inline]
|
[inline]
|
||||||
pub fn (tok &Token) position() Position {
|
pub fn (tok &Token) position() Position {
|
||||||
return Position{
|
return Position{
|
||||||
line_nr: tok.line_nr
|
line_nr: tok.line_nr - 1
|
||||||
// pos: tok.pos
|
// pos: tok.pos
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +0,0 @@
|
||||||
module types
|
|
||||||
|
|
||||||
fn test_types() {
|
|
||||||
|
|
||||||
}
|
|
Loading…
Reference in New Issue