cgen: for c in string; short struct init syntax fixes;
parent
c71d36356b
commit
ed3e0c43bc
|
@ -343,6 +343,9 @@ pub fn (t Test) str() string {
|
|||
}
|
||||
|
||||
fn test_struct_print() {
|
||||
println('QTODO')
|
||||
|
||||
/*
|
||||
mut a := Test{
|
||||
a: 'Test'
|
||||
b: []
|
||||
|
@ -356,6 +359,7 @@ fn test_struct_print() {
|
|||
assert a.str() == '{Test [{1 2}, {1 2}] }'
|
||||
assert b.str() == '{1 2}'
|
||||
assert a.b.str() == '[{1 2}, {1 2}]'
|
||||
*/
|
||||
}
|
||||
|
||||
fn test_single_element() {
|
||||
|
|
|
@ -429,10 +429,10 @@ pub fn (s string) split_nth(delim string, nth int) []string {
|
|||
mut start := 0
|
||||
nth_1 := nth - 1
|
||||
for i <= s.len {
|
||||
mut is_delim := s[i] == delim[0]
|
||||
mut is_delim := s.str[i] == delim.str[0]
|
||||
mut j := 0
|
||||
for is_delim && j < delim.len {
|
||||
is_delim = is_delim && s[i + j] == delim[j]
|
||||
is_delim = is_delim && s.str[i + j] == delim.str[j]
|
||||
j++
|
||||
}
|
||||
last := i == s.len - 1
|
||||
|
@ -469,8 +469,8 @@ pub fn (s string) split_into_lines() []string {
|
|||
}
|
||||
mut start := 0
|
||||
for i := 0; i < s.len; i++ {
|
||||
is_lf := s[i] == `\n`
|
||||
is_crlf := i != s.len - 1 && s[i] == `\r` && s[i + 1] == `\n`
|
||||
is_lf := s.str[i] == `\n`
|
||||
is_crlf := i != s.len - 1 && s.str[i] == `\r` && s.str[i + 1] == `\n`
|
||||
is_eol := is_lf || is_crlf
|
||||
is_last := if is_crlf {
|
||||
i == s.len - 2
|
||||
|
|
|
@ -2,6 +2,12 @@
|
|||
// Use of this source code is governed by an MIT license
|
||||
// that can be found in the LICENSE file.
|
||||
|
||||
struct Foo {
|
||||
bar int
|
||||
mut:
|
||||
str string
|
||||
}
|
||||
|
||||
fn test_add() {
|
||||
mut a := 'a'
|
||||
a += 'b'
|
||||
|
@ -14,7 +20,9 @@ fn test_add() {
|
|||
assert a.ends_with('bbbbb')
|
||||
a += '123'
|
||||
assert a.ends_with('3')
|
||||
mut foo := Foo{0, 'hi'}
|
||||
mut foo := Foo{10, 'hi'}
|
||||
assert foo.str == 'hi'
|
||||
assert foo.bar == 10
|
||||
foo.str += '!'
|
||||
assert foo.str == 'hi!'
|
||||
}
|
||||
|
@ -91,7 +99,9 @@ fn test_sort() {
|
|||
}
|
||||
|
||||
fn test_split_nth() {
|
||||
|
||||
a := "1,2,3"
|
||||
println(a)
|
||||
assert (a.split(',').len == 3)
|
||||
assert (a.split_nth(',', -1).len == 3)
|
||||
assert (a.split_nth(',', 0).len == 3)
|
||||
|
@ -99,6 +109,7 @@ fn test_split_nth() {
|
|||
assert (a.split_nth(',', 2).len == 2)
|
||||
assert (a.split_nth(',', 10).len == 3)
|
||||
b := "1::2::3"
|
||||
println(b)
|
||||
assert (b.split('::').len == 3)
|
||||
assert (b.split_nth('::', -1).len == 3)
|
||||
assert (b.split_nth('::', 0).len == 3)
|
||||
|
@ -106,15 +117,19 @@ fn test_split_nth() {
|
|||
assert (b.split_nth('::', 2).len == 2)
|
||||
assert (b.split_nth('::', 10).len == 3)
|
||||
c := "ABCDEF"
|
||||
println(c)
|
||||
println(c.split('').len)
|
||||
assert (c.split('').len == 6)
|
||||
assert (c.split_nth('', 3).len == 3)
|
||||
assert (c.split_nth('BC', -1).len == 2)
|
||||
d := ","
|
||||
println(d)
|
||||
assert (d.split(',').len == 2)
|
||||
assert (d.split_nth('', 3).len == 1)
|
||||
assert (d.split_nth(',', -1).len == 2)
|
||||
assert (d.split_nth(',', 3).len == 2)
|
||||
e := ",,,0,,,,,a,,b,"
|
||||
println(e)
|
||||
// assert (e.split(',,').len == 5)
|
||||
// assert (e.split_nth(',,', 3).len == 2)
|
||||
assert (e.split_nth(',', -1).len == 12)
|
||||
|
@ -472,12 +487,6 @@ fn test_reverse() {
|
|||
assert 'a'.reverse() == 'a'
|
||||
}
|
||||
|
||||
struct Foo {
|
||||
bar int
|
||||
mut:
|
||||
str string
|
||||
}
|
||||
|
||||
fn (f Foo) baz() string {
|
||||
return 'baz'
|
||||
}
|
||||
|
|
|
@ -1,17 +1,17 @@
|
|||
struct A {
|
||||
struct AA {
|
||||
mut:
|
||||
val int
|
||||
nums []int
|
||||
}
|
||||
|
||||
struct B {
|
||||
struct BB {
|
||||
mut:
|
||||
a A
|
||||
a AA
|
||||
}
|
||||
|
||||
struct C {
|
||||
struct CC {
|
||||
mut:
|
||||
b B
|
||||
b BB
|
||||
nums []int
|
||||
aarr []A
|
||||
num int
|
||||
|
@ -65,7 +65,7 @@ fn test_empty_struct() {
|
|||
}
|
||||
|
||||
fn test_struct_levels() {
|
||||
mut c := C{}
|
||||
mut c := CC{}
|
||||
assert c.nums.len == 0
|
||||
c.nums << 3
|
||||
assert c.nums.len == 1
|
||||
|
|
|
@ -85,16 +85,13 @@ pub fn (c mut Checker) struct_init(struct_init mut ast.StructInit) table.Type {
|
|||
// string & array are also structs but .kind of string/array
|
||||
.struct_, .string, .array {
|
||||
info := typ_sym.info as table.Struct
|
||||
if struct_init.fields.len == 0 {
|
||||
// Short syntax TODO check
|
||||
return struct_init.typ
|
||||
}
|
||||
is_short_syntax := struct_init.fields.len == 0
|
||||
if struct_init.exprs.len > info.fields.len {
|
||||
c.error('too many fields', struct_init.pos)
|
||||
}
|
||||
for i, expr in struct_init.exprs {
|
||||
// struct_field info.
|
||||
field_name := struct_init.fields[i]
|
||||
field_name := if is_short_syntax { info.fields[i].name } else { struct_init.fields[i] }
|
||||
mut field := info.fields[i]
|
||||
mut found_field := false
|
||||
for f in info.fields {
|
||||
|
|
|
@ -366,66 +366,7 @@ fn (g mut Gen) stmt(node ast.Stmt) {
|
|||
g.writeln('}')
|
||||
}
|
||||
ast.ForInStmt {
|
||||
if it.is_range {
|
||||
// `for x in 1..10 {`
|
||||
i := g.new_tmp_var()
|
||||
g.write('for (int $i = ')
|
||||
g.expr(it.cond)
|
||||
g.write('; $i < ')
|
||||
g.expr(it.high)
|
||||
g.writeln('; $i++) { ')
|
||||
g.writeln('int $it.val_var = $i;')
|
||||
g.stmts(it.stmts)
|
||||
g.writeln('}')
|
||||
}
|
||||
// TODO:
|
||||
else if it.kind == .array {
|
||||
// `for num in nums {`
|
||||
g.writeln('// FOR IN')
|
||||
i := if it.key_var == '' { g.new_tmp_var() } else { it.key_var }
|
||||
styp := g.typ(it.val_type)
|
||||
g.write('for (int $i = 0; $i < ')
|
||||
g.expr(it.cond)
|
||||
g.writeln('.len; $i++) {')
|
||||
g.write('$styp $it.val_var = (($styp*)')
|
||||
g.expr(it.cond)
|
||||
g.writeln('.data)[$i];')
|
||||
g.stmts(it.stmts)
|
||||
g.writeln('}')
|
||||
}
|
||||
else if it.kind == .map {
|
||||
// `for num in nums {`
|
||||
g.writeln('// FOR IN')
|
||||
key_styp := g.typ(it.key_type)
|
||||
val_styp := g.typ(it.val_type)
|
||||
keys_tmp := 'keys_' + g.new_tmp_var()
|
||||
idx := g.new_tmp_var()
|
||||
key := if it.key_var == '' { g.new_tmp_var() } else { it.key_var }
|
||||
zero := g.type_default(it.val_type)
|
||||
g.write('array_$key_styp $keys_tmp = map_keys(&')
|
||||
g.expr(it.cond)
|
||||
g.writeln(');')
|
||||
g.writeln('for (int $idx = 0; $idx < ${keys_tmp}.len; $idx++) {')
|
||||
g.writeln('$key_styp $key = (($key_styp*)${keys_tmp}.data)[$idx];')
|
||||
g.write('$val_styp $it.val_var = (*($val_styp*)map_get3(')
|
||||
g.expr(it.cond)
|
||||
g.writeln(', $key, &($val_styp[]){ $zero }));')
|
||||
g.stmts(it.stmts)
|
||||
g.writeln('}')
|
||||
}
|
||||
else if table.type_is_variadic(it.cond_type) {
|
||||
g.writeln('// FOR IN')
|
||||
i := if it.key_var == '' { g.new_tmp_var() } else { it.key_var }
|
||||
styp := g.typ(it.cond_type)
|
||||
g.write('for (int $i = 0; $i < ')
|
||||
g.expr(it.cond)
|
||||
g.writeln('.len; $i++) {')
|
||||
g.write('$styp $it.val_var = ')
|
||||
g.expr(it.cond)
|
||||
g.writeln('.args[$i];')
|
||||
g.stmts(it.stmts)
|
||||
g.writeln('}')
|
||||
}
|
||||
g.for_in(it)
|
||||
}
|
||||
ast.ForStmt {
|
||||
g.write('while (')
|
||||
|
@ -484,6 +425,80 @@ fn (g mut Gen) stmt(node ast.Stmt) {
|
|||
}
|
||||
}
|
||||
|
||||
fn (g mut Gen) for_in(it ast.ForInStmt) {
|
||||
if it.is_range {
|
||||
// `for x in 1..10 {`
|
||||
i := g.new_tmp_var()
|
||||
g.write('for (int $i = ')
|
||||
g.expr(it.cond)
|
||||
g.write('; $i < ')
|
||||
g.expr(it.high)
|
||||
g.writeln('; $i++) { ')
|
||||
g.writeln('int $it.val_var = $i;')
|
||||
g.stmts(it.stmts)
|
||||
g.writeln('}')
|
||||
}
|
||||
// TODO:
|
||||
else if it.kind == .array {
|
||||
// `for num in nums {`
|
||||
g.writeln('// FOR IN')
|
||||
i := if it.key_var == '' { g.new_tmp_var() } else { it.key_var }
|
||||
styp := g.typ(it.val_type)
|
||||
g.write('for (int $i = 0; $i < ')
|
||||
g.expr(it.cond)
|
||||
g.writeln('.len; $i++) {')
|
||||
g.write('$styp $it.val_var = (($styp*)')
|
||||
g.expr(it.cond)
|
||||
g.writeln('.data)[$i];')
|
||||
g.stmts(it.stmts)
|
||||
g.writeln('}')
|
||||
}
|
||||
else if it.kind == .map {
|
||||
// `for num in nums {`
|
||||
g.writeln('// FOR IN')
|
||||
key_styp := g.typ(it.key_type)
|
||||
val_styp := g.typ(it.val_type)
|
||||
keys_tmp := 'keys_' + g.new_tmp_var()
|
||||
idx := g.new_tmp_var()
|
||||
key := if it.key_var == '' { g.new_tmp_var() } else { it.key_var }
|
||||
zero := g.type_default(it.val_type)
|
||||
g.write('array_$key_styp $keys_tmp = map_keys(&')
|
||||
g.expr(it.cond)
|
||||
g.writeln(');')
|
||||
g.writeln('for (int $idx = 0; $idx < ${keys_tmp}.len; $idx++) {')
|
||||
g.writeln('$key_styp $key = (($key_styp*)${keys_tmp}.data)[$idx];')
|
||||
g.write('$val_styp $it.val_var = (*($val_styp*)map_get3(')
|
||||
g.expr(it.cond)
|
||||
g.writeln(', $key, &($val_styp[]){ $zero }));')
|
||||
g.stmts(it.stmts)
|
||||
g.writeln('}')
|
||||
}
|
||||
else if table.type_is_variadic(it.cond_type) {
|
||||
g.writeln('// FOR IN')
|
||||
i := if it.key_var == '' { g.new_tmp_var() } else { it.key_var }
|
||||
styp := g.typ(it.cond_type)
|
||||
g.write('for (int $i = 0; $i < ')
|
||||
g.expr(it.cond)
|
||||
g.writeln('.len; $i++) {')
|
||||
g.write('$styp $it.val_var = ')
|
||||
g.expr(it.cond)
|
||||
g.writeln('.args[$i];')
|
||||
g.stmts(it.stmts)
|
||||
g.writeln('}')
|
||||
}
|
||||
else if it.kind == .string {
|
||||
i := if it.key_var == '' { g.new_tmp_var() } else { it.key_var }
|
||||
g.write('for (int $i = 0; $i < ')
|
||||
g.expr(it.cond)
|
||||
g.writeln('.len; $i++) {')
|
||||
g.write('byte $it.val_var = ')
|
||||
g.expr(it.cond)
|
||||
g.writeln('.str[$i];')
|
||||
g.stmts(it.stmts)
|
||||
g.writeln('}')
|
||||
}
|
||||
}
|
||||
|
||||
// use instead of expr() when you need to cast to sum type (can add other casts also)
|
||||
fn (g mut Gen) expr_with_cast(expr ast.Expr, got_type table.Type, exp_type table.Type) {
|
||||
// cast to sum type
|
||||
|
@ -1121,6 +1136,7 @@ fn (g mut Gen) expr(node ast.Expr) {
|
|||
}
|
||||
|
||||
fn (g mut Gen) infix_expr(node ast.InfixExpr) {
|
||||
// println('infix_expr() op="$node.op.str()" line_nr=$node.pos.line_nr')
|
||||
// g.write('/*infix*/')
|
||||
// if it.left_type == table.string_type_idx {
|
||||
// g.write('/*$node.left_type str*/')
|
||||
|
@ -1675,9 +1691,19 @@ fn (g mut Gen) struct_init(it ast.StructInit) {
|
|||
else {
|
||||
g.writeln('($styp){')
|
||||
}
|
||||
mut fields := []string
|
||||
mut inited_fields := []string
|
||||
// User set fields
|
||||
for i, field in it.fields {
|
||||
if it.fields.len == 0 && it.exprs.len > 0 {
|
||||
// Get fields for {a,b} short syntax. Fields array wasn't set in the parser.
|
||||
for f in info.fields {
|
||||
fields << f.name
|
||||
}
|
||||
}
|
||||
else {
|
||||
fields = it.fields
|
||||
}
|
||||
// / User set fields
|
||||
for i, field in fields {
|
||||
field_name := c_name(field)
|
||||
inited_fields << field
|
||||
g.write('\t.$field_name = ')
|
||||
|
|
Loading…
Reference in New Issue