v2: simplify the type system; an initial version of the V interpreter

pull/3632/head
Alexander Medvednikov 2020-01-22 21:34:38 +01:00
parent 932392a7fd
commit 9f4661391d
33 changed files with 847 additions and 517 deletions

View File

@ -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')

View File

@ -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')

View File

@ -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)
} }

View File

@ -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()
} }

View File

@ -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)
} }

104
vlib/v/eval/eval.v 100644
View File

@ -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{}
}

View File

@ -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')

View File

@ -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)

View File

@ -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;
}
}

View File

@ -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
}
}

View File

@ -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 {

View File

@ -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)
} }
*/ */
} }

View File

@ -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)
} }
} }
} }
} }
} }

View File

@ -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)

View File

@ -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{

View 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
@ -53,20 +67,50 @@ pub enum Kind {
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]
} }

View File

@ -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
} }
} }
} }
*/

View File

@ -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
} }
} }

View File

@ -1,5 +0,0 @@
module types
fn test_types() {
}