cgen: fix optional pointers
parent
54c382f6f1
commit
fecf4ddf65
|
@ -18,7 +18,6 @@ const (
|
||||||
'vlib/v/tests/enum_bitfield_test.v',
|
'vlib/v/tests/enum_bitfield_test.v',
|
||||||
'vlib/v/tests/fixed_array_test.v',
|
'vlib/v/tests/fixed_array_test.v',
|
||||||
'vlib/v/tests/num_lit_call_method_test.v',
|
'vlib/v/tests/num_lit_call_method_test.v',
|
||||||
'vlib/v/tests/option_test.v',
|
|
||||||
'vlib/v/tests/pointers_test.v',
|
'vlib/v/tests/pointers_test.v',
|
||||||
'vlib/v/tests/string_interpolation_variadic_test.v',
|
'vlib/v/tests/string_interpolation_variadic_test.v',
|
||||||
'vlib/v/tests/type_test.v',
|
'vlib/v/tests/type_test.v',
|
||||||
|
|
|
@ -363,7 +363,7 @@ pub fn sys_munmap(addr voidptr, len u64) errno {
|
||||||
}
|
}
|
||||||
|
|
||||||
// 22 sys_pipe int *filedes
|
// 22 sys_pipe int *filedes
|
||||||
pub fn sys_pipe(filedes intptr) errno {
|
pub fn sys_pipe(filedes &int) errno {
|
||||||
return errno(sys_call1(22, u64(filedes)))
|
return errno(sys_call1(22, u64(filedes)))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -415,7 +415,7 @@ pub fn sys_getuid() int {
|
||||||
}
|
}
|
||||||
|
|
||||||
// 247 sys_waitid int which pid_t upid struct siginfo *infop int options struct rusage *ru
|
// 247 sys_waitid int which pid_t upid struct siginfo *infop int options struct rusage *ru
|
||||||
pub fn sys_waitid (which wi_which, pid int, infop intptr, options wp_sys, ru voidptr) errno {
|
pub fn sys_waitid (which wi_which, pid int, infop &int, options wp_sys, ru voidptr) errno {
|
||||||
return errno(sys_call5(247, u64(which), u64(pid), u64(infop), u64(options), u64(ru)))
|
return errno(sys_call5(247, u64(which), u64(pid), u64(infop), u64(options), u64(ru)))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -17,7 +17,7 @@ pub fn mm_alloc(size u64) (byteptr, errno) {
|
||||||
|
|
||||||
a, e := sys_mmap(0, n_bytes, mem_prot, mem_flags, -1, 0)
|
a, e := sys_mmap(0, n_bytes, mem_prot, mem_flags, -1, 0)
|
||||||
if e == .enoerror {
|
if e == .enoerror {
|
||||||
mut ap := intptr(a)
|
mut ap := &int(a)
|
||||||
*ap = pages
|
*ap = pages
|
||||||
return byteptr(a+4), e
|
return byteptr(a+4), e
|
||||||
}
|
}
|
||||||
|
@ -25,7 +25,7 @@ pub fn mm_alloc(size u64) (byteptr, errno) {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn mm_free(addr byteptr) errno {
|
pub fn mm_free(addr byteptr) errno {
|
||||||
ap := intptr(addr-4)
|
ap := &int(addr-4)
|
||||||
size := u64(*ap) * u64(linux_mem.page_size)
|
size := u64(*ap) * u64(linux_mem.page_size)
|
||||||
|
|
||||||
return sys_munmap(ap, size)
|
return sys_munmap(ap, size)
|
||||||
|
|
|
@ -228,6 +228,24 @@ pub fn (mut g Gen) write_typeof_functions() {
|
||||||
|
|
||||||
// V type to C type
|
// V type to C type
|
||||||
pub fn (mut g Gen) typ(t table.Type) string {
|
pub fn (mut g Gen) typ(t table.Type) string {
|
||||||
|
mut styp := g.base_typ(t)
|
||||||
|
if table.type_is(t, .optional) {
|
||||||
|
// Register an optional
|
||||||
|
styp = 'Option_' + styp
|
||||||
|
if table.type_is_ptr(t) {
|
||||||
|
styp = styp.replace('*', '_ptr')
|
||||||
|
}
|
||||||
|
if !(styp in g.optionals) {
|
||||||
|
// println(styp)
|
||||||
|
x := styp // .replace('*', '_ptr') // handle option ptrs
|
||||||
|
g.typedefs2.writeln('typedef Option $x;')
|
||||||
|
g.optionals << styp
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return styp
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn (mut g Gen) base_typ(t table.Type) string {
|
||||||
nr_muls := table.type_nr_muls(t)
|
nr_muls := table.type_nr_muls(t)
|
||||||
sym := g.table.get_type_symbol(t)
|
sym := g.table.get_type_symbol(t)
|
||||||
mut styp := sym.name.replace('.', '__')
|
mut styp := sym.name.replace('.', '__')
|
||||||
|
@ -243,19 +261,6 @@ pub fn (mut g Gen) typ(t table.Type) string {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if table.type_is(t, .optional) {
|
|
||||||
// Register an optional
|
|
||||||
styp = 'Option_' + styp
|
|
||||||
if table.type_is_ptr(t) {
|
|
||||||
styp = styp.replace('*', '_ptr')
|
|
||||||
}
|
|
||||||
if !(styp in g.optionals) {
|
|
||||||
// println(styp)
|
|
||||||
x := styp // .replace('*', '_ptr') // handle option ptrs
|
|
||||||
g.typedefs2.writeln('typedef Option $x;')
|
|
||||||
g.optionals << styp
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return styp
|
return styp
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -733,12 +738,10 @@ fn (mut g Gen) gen_assign_stmt(assign_stmt ast.AssignStmt) {
|
||||||
// multi return
|
// multi return
|
||||||
mut or_stmts := []ast.Stmt
|
mut or_stmts := []ast.Stmt
|
||||||
mut return_type := table.void_type
|
mut return_type := table.void_type
|
||||||
match assign_stmt.right[0] {
|
if assign_stmt.right[0] is ast.CallExpr {
|
||||||
ast.CallExpr {
|
it := assign_stmt.right[0] as ast.CallExpr
|
||||||
or_stmts = it.or_block.stmts
|
or_stmts = it.or_block.stmts
|
||||||
return_type = it.return_type
|
return_type = it.return_type
|
||||||
}
|
|
||||||
else {}
|
|
||||||
}
|
}
|
||||||
is_optional := table.type_is(return_type, .optional)
|
is_optional := table.type_is(return_type, .optional)
|
||||||
mr_var_name := 'mr_$assign_stmt.pos.pos'
|
mr_var_name := 'mr_$assign_stmt.pos.pos'
|
||||||
|
@ -762,8 +765,8 @@ fn (mut g Gen) gen_assign_stmt(assign_stmt ast.AssignStmt) {
|
||||||
}
|
}
|
||||||
g.expr(ident)
|
g.expr(ident)
|
||||||
if is_optional {
|
if is_optional {
|
||||||
mr_styp2 := mr_styp[7..] // remove Option_
|
mr_base_styp := g.base_typ(return_type)
|
||||||
g.writeln(' = (*(${mr_styp2}*)${mr_var_name}.data).arg$i;')
|
g.writeln(' = (*(${mr_base_styp}*)${mr_var_name}.data).arg$i;')
|
||||||
} else {
|
} else {
|
||||||
g.writeln(' = ${mr_var_name}.arg$i;')
|
g.writeln(' = ${mr_var_name}.arg$i;')
|
||||||
}
|
}
|
||||||
|
@ -819,7 +822,7 @@ fn (mut g Gen) gen_assign_stmt(assign_stmt ast.AssignStmt) {
|
||||||
if is_decl {
|
if is_decl {
|
||||||
g.write('$styp ')
|
g.write('$styp ')
|
||||||
}
|
}
|
||||||
g.expr(ident)
|
g.ident(ident)
|
||||||
if g.autofree && right_sym.kind in [.array, .string] {
|
if g.autofree && right_sym.kind in [.array, .string] {
|
||||||
if g.gen_clone_assignment(val, right_sym, true) {
|
if g.gen_clone_assignment(val, right_sym, true) {
|
||||||
g.writeln(';')
|
g.writeln(';')
|
||||||
|
@ -1554,21 +1557,18 @@ fn (mut g Gen) ident(node ast.Ident) {
|
||||||
g.write('_const_')
|
g.write('_const_')
|
||||||
}
|
}
|
||||||
name := c_name(node.name)
|
name := c_name(node.name)
|
||||||
// TODO `is`
|
if node.info is ast.IdentVar {
|
||||||
match node.info {
|
ident_var := node.info as ast.IdentVar
|
||||||
ast.IdentVar {
|
// x ?int
|
||||||
// x ?int
|
// `x = 10` => `x.data = 10` (g.right_is_opt == false)
|
||||||
// `x = 10` => `x.data = 10` (g.right_is_opt == false)
|
// `x = new_opt()` => `x = new_opt()` (g.right_is_opt == true)
|
||||||
// `x = new_opt()` => `x = new_opt()` (g.right_is_opt == true)
|
// `println(x)` => `println(*(int*)x.data)`
|
||||||
// `println(x)` => `println(*(int*)x.data)`
|
if ident_var.is_optional && !(g.is_assign_lhs && g.right_is_opt) {
|
||||||
if it.is_optional && !(g.is_assign_lhs && g.right_is_opt) {
|
g.write('/*opt*/')
|
||||||
g.write('/*opt*/')
|
styp := g.base_typ(ident_var.typ)
|
||||||
styp := g.typ(it.typ)[7..] // Option_int => int TODO perf?
|
g.write('(*($styp*)${name}.data)')
|
||||||
g.write('(*($styp*)${name}.data)')
|
return
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else {}
|
|
||||||
}
|
}
|
||||||
g.write(name)
|
g.write(name)
|
||||||
}
|
}
|
||||||
|
@ -1821,14 +1821,12 @@ fn (mut g Gen) return_statement(node ast.Return) {
|
||||||
g.write(' ')
|
g.write(' ')
|
||||||
// typ_sym := g.table.get_type_symbol(g.fn_decl.return_type)
|
// typ_sym := g.table.get_type_symbol(g.fn_decl.return_type)
|
||||||
// mr_info := typ_sym.info as table.MultiReturn
|
// mr_info := typ_sym.info as table.MultiReturn
|
||||||
mut styp := g.typ(g.fn_decl.return_type)
|
mut styp := ''
|
||||||
if fn_return_is_optional { // && !table.type_is(node.types[0], .optional) && node.types[0] !=
|
if fn_return_is_optional { // && !table.type_is(node.types[0], .optional) && node.types[0] !=
|
||||||
styp = styp[7..] // remove 'Option_'
|
styp = g.base_typ(g.fn_decl.return_type)
|
||||||
mut x := styp
|
g.write('opt_ok(&($styp/*X*/[]) { ')
|
||||||
if x.ends_with('_ptr') {
|
} else {
|
||||||
x = x.replace('_ptr', '*')
|
styp = g.typ(g.fn_decl.return_type)
|
||||||
}
|
|
||||||
g.write('opt_ok(&($x/*X*/[]) { ')
|
|
||||||
}
|
}
|
||||||
g.write('($styp){')
|
g.write('($styp){')
|
||||||
for i, expr in node.exprs {
|
for i, expr in node.exprs {
|
||||||
|
@ -1864,14 +1862,10 @@ fn (mut g Gen) return_statement(node ast.Return) {
|
||||||
else {}
|
else {}
|
||||||
}
|
}
|
||||||
if !is_none && !is_error {
|
if !is_none && !is_error {
|
||||||
styp := g.typ(g.fn_decl.return_type)[7..] // remove 'Option_'
|
styp := g.base_typ(g.fn_decl.return_type)
|
||||||
mut x := styp
|
g.write('/*:)$return_sym.name*/opt_ok(&($styp[]) { ')
|
||||||
if x.ends_with('_ptr') {
|
|
||||||
x = x.replace('_ptr', '*')
|
|
||||||
}
|
|
||||||
g.write('/*:)$return_sym.name*/opt_ok(&($x[]) { ')
|
|
||||||
g.expr(node.exprs[0])
|
g.expr(node.exprs[0])
|
||||||
g.writeln(' }, sizeof($x));')
|
g.writeln(' }, sizeof($styp));')
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
// g.write('/*OPTIONAL*/')
|
// g.write('/*OPTIONAL*/')
|
||||||
|
@ -2454,8 +2448,7 @@ fn (mut g Gen) insert_before(s string) {
|
||||||
// to access its fields (`.ok`, `.error` etc)
|
// to access its fields (`.ok`, `.error` etc)
|
||||||
// `os.cp(...)` => `Option bool tmp = os__cp(...); if (!tmp.ok) { ... }`
|
// `os.cp(...)` => `Option bool tmp = os__cp(...); if (!tmp.ok) { ... }`
|
||||||
fn (mut g Gen) or_block(var_name string, stmts []ast.Stmt, return_type table.Type) {
|
fn (mut g Gen) or_block(var_name string, stmts []ast.Stmt, return_type table.Type) {
|
||||||
mr_styp := g.typ(return_type)
|
mr_styp := g.base_typ(return_type)
|
||||||
mr_styp2 := mr_styp[7..] // remove Option_
|
|
||||||
g.writeln(';') // or')
|
g.writeln(';') // or')
|
||||||
g.writeln('if (!${var_name}.ok) {')
|
g.writeln('if (!${var_name}.ok) {')
|
||||||
g.writeln('\tstring err = ${var_name}.v_error;')
|
g.writeln('\tstring err = ${var_name}.v_error;')
|
||||||
|
@ -2466,7 +2459,7 @@ fn (mut g Gen) or_block(var_name string, stmts []ast.Stmt, return_type table.Typ
|
||||||
for i, stmt in stmts {
|
for i, stmt in stmts {
|
||||||
if i == stmts.len - 1 {
|
if i == stmts.len - 1 {
|
||||||
g.indent--
|
g.indent--
|
||||||
g.write('\t*(${mr_styp2}*) ${var_name}.data = ')
|
g.write('\t*(${mr_styp}*) ${var_name}.data = ')
|
||||||
}
|
}
|
||||||
g.stmt(stmt)
|
g.stmt(stmt)
|
||||||
}
|
}
|
||||||
|
|
|
@ -335,7 +335,6 @@ typedef uint32_t rune;
|
||||||
typedef float f32;
|
typedef float f32;
|
||||||
typedef double f64;
|
typedef double f64;
|
||||||
typedef unsigned char* byteptr;
|
typedef unsigned char* byteptr;
|
||||||
typedef int* intptr;
|
|
||||||
typedef void* voidptr;
|
typedef void* voidptr;
|
||||||
typedef char* charptr;
|
typedef char* charptr;
|
||||||
typedef struct array array;
|
typedef struct array array;
|
||||||
|
|
|
@ -184,7 +184,8 @@ fn (g mut JsGen) to_js_typ(typ string) string {
|
||||||
}
|
}
|
||||||
'charptr' {
|
'charptr' {
|
||||||
styp = 'string'
|
styp = 'string'
|
||||||
} else {
|
}
|
||||||
|
else {
|
||||||
if typ.starts_with('array_') {
|
if typ.starts_with('array_') {
|
||||||
styp = g.to_js_typ(typ.replace('array_', '')) + '[]'
|
styp = g.to_js_typ(typ.replace('array_', '')) + '[]'
|
||||||
} else if typ.starts_with('map_') {
|
} else if typ.starts_with('map_') {
|
||||||
|
|
|
@ -38,16 +38,14 @@ fn ret_none() ?int {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn test_option_for_base_type_without_variable() {
|
fn test_option_for_base_type_without_variable() {
|
||||||
val := err_call(true) or {
|
mut val := err_call(true) or {
|
||||||
panic(err)
|
assert false
|
||||||
|
0
|
||||||
}
|
}
|
||||||
assert val == 42
|
assert val == 42
|
||||||
println('hm')
|
val = ret_none() or {
|
||||||
val2 := ret_none() or {
|
|
||||||
println('yep')
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
println('$val2 should have been `none`')
|
|
||||||
assert false
|
assert false
|
||||||
// This is invalid:
|
// This is invalid:
|
||||||
// x := 5 or {
|
// x := 5 or {
|
||||||
|
@ -60,7 +58,6 @@ fn test_if_opt() {
|
||||||
assert val == 42
|
assert val == 42
|
||||||
}
|
}
|
||||||
assert 1 == 1
|
assert 1 == 1
|
||||||
println('nice')
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn for_opt_default() ?string {
|
fn for_opt_default() ?string {
|
||||||
|
@ -135,24 +132,19 @@ fn test_opt_field() {
|
||||||
fn opt_ptr(a &int) ?&int {
|
fn opt_ptr(a &int) ?&int {
|
||||||
if isnil(a) {
|
if isnil(a) {
|
||||||
return none
|
return none
|
||||||
}
|
|
||||||
//
|
|
||||||
else {
|
|
||||||
|
|
||||||
}
|
}
|
||||||
return a
|
return a
|
||||||
}
|
}
|
||||||
|
|
||||||
fn test_opt_ptr() {
|
fn test_opt_ptr() {
|
||||||
a := 3
|
a := 3
|
||||||
r1 := opt_ptr(&a) or {
|
mut r := opt_ptr(&a) or {
|
||||||
&int(0)
|
&int(0)
|
||||||
}
|
}
|
||||||
assert r1 == &a
|
assert r == &a
|
||||||
r2 := opt_ptr(&int(0)) or {
|
r = opt_ptr(&int(0)) or {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
println('`$r2` should be none')
|
|
||||||
assert false
|
assert false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue