ci: vet all files from the compiler (#5994)
parent
1086150ab9
commit
2de1437a1e
|
@ -45,12 +45,8 @@ jobs:
|
||||||
run: ./v build-tools
|
run: ./v build-tools
|
||||||
- name: v vet
|
- name: v vet
|
||||||
run: |
|
run: |
|
||||||
./v vet vlib/v/scanner
|
|
||||||
./v vet vlib/v/parser
|
|
||||||
./v vet vlib/v/ast
|
|
||||||
./v vet vlib/v/gen/cgen.v
|
|
||||||
./v vet vlib/v/checker
|
|
||||||
./v vet vlib/sqlite
|
./v vet vlib/sqlite
|
||||||
|
./v vet vlib/v
|
||||||
- name: v fmt
|
- name: v fmt
|
||||||
run: |
|
run: |
|
||||||
./v fmt -verify vlib/v/checker/checker.v
|
./v fmt -verify vlib/v/checker/checker.v
|
||||||
|
|
|
@ -48,7 +48,7 @@ pub fn (mut b Builder) build_c(v_files []string, out_file string) {
|
||||||
|
|
||||||
pub fn (mut b Builder) compile_c() {
|
pub fn (mut b Builder) compile_c() {
|
||||||
if os.user_os() != 'windows' && b.pref.ccompiler == 'msvc' {
|
if os.user_os() != 'windows' && b.pref.ccompiler == 'msvc' {
|
||||||
verror('Cannot build with msvc on ${os.user_os()}')
|
verror('Cannot build with msvc on $os.user_os()')
|
||||||
}
|
}
|
||||||
// cgen.genln('// Generated by V')
|
// cgen.genln('// Generated by V')
|
||||||
// println('compile2()')
|
// println('compile2()')
|
||||||
|
@ -57,7 +57,9 @@ pub fn (mut b Builder) compile_c() {
|
||||||
// println(files)
|
// println(files)
|
||||||
}
|
}
|
||||||
$if windows {
|
$if windows {
|
||||||
b.pref.ccompiler = b.find_win_cc() or { panic(no_compiler_error) }
|
b.pref.ccompiler = b.find_win_cc() or {
|
||||||
|
panic(no_compiler_error)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// v1 compiler files
|
// v1 compiler files
|
||||||
// v.add_v_files_to_compile()
|
// v.add_v_files_to_compile()
|
||||||
|
|
|
@ -10,10 +10,13 @@ fn (v &Builder) get_os_cflags() []cflag.CFlag {
|
||||||
ctimedefines << v.pref.compile_defines
|
ctimedefines << v.pref.compile_defines
|
||||||
}
|
}
|
||||||
for flag in v.table.cflags {
|
for flag in v.table.cflags {
|
||||||
if flag.os == '' || (flag.os == 'linux' && v.pref.os == .linux) || (flag.os == 'darwin' &&
|
if flag.os == '' ||
|
||||||
v.pref.os == .mac) || (flag.os == 'freebsd' && v.pref.os == .freebsd) || (flag.os == 'windows' &&
|
(flag.os == 'linux' && v.pref.os == .linux) ||
|
||||||
v.pref.os == .windows) || (flag.os == 'mingw' && v.pref.os == .windows && v.pref.ccompiler !=
|
(flag.os == 'darwin' && v.pref.os == .mac) ||
|
||||||
'msvc') || (flag.os == 'solaris' && v.pref.os == .solaris) {
|
(flag.os == 'freebsd' && v.pref.os == .freebsd) ||
|
||||||
|
(flag.os == 'windows' && v.pref.os == .windows) ||
|
||||||
|
(flag.os == 'mingw' && v.pref.os == .windows && v.pref.ccompiler != 'msvc') ||
|
||||||
|
(flag.os == 'solaris' && v.pref.os == .solaris) {
|
||||||
flags << flag
|
flags << flag
|
||||||
}
|
}
|
||||||
if flag.os in ctimedefines {
|
if flag.os in ctimedefines {
|
||||||
|
|
|
@ -8,7 +8,6 @@ import v.cflag
|
||||||
#flag windows -l shell32
|
#flag windows -l shell32
|
||||||
#flag windows -l dbghelp
|
#flag windows -l dbghelp
|
||||||
#flag windows -l advapi32
|
#flag windows -l advapi32
|
||||||
|
|
||||||
struct MsvcResult {
|
struct MsvcResult {
|
||||||
full_cl_exe_path string
|
full_cl_exe_path string
|
||||||
exe_path string
|
exe_path string
|
||||||
|
@ -51,7 +50,8 @@ fn find_windows_kit_internal(key RegKey, versions []string) ?string {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
//
|
//
|
||||||
else{}
|
else {
|
||||||
|
}
|
||||||
result2 := C.RegQueryValueEx(key, version.to_wide(), 0, 0, value, &alloc_length)
|
result2 := C.RegQueryValueEx(key, version.to_wide(), 0, 0, value, &alloc_length)
|
||||||
if result2 != 0 {
|
if result2 != 0 {
|
||||||
continue
|
continue
|
||||||
|
@ -143,7 +143,7 @@ fn find_vs(vswhere_dir, host_arch string) ?VsInstallation {
|
||||||
res_output := res.output.trim_right('\r\n')
|
res_output := res.output.trim_right('\r\n')
|
||||||
// println('res: "$res"')
|
// println('res: "$res"')
|
||||||
version := os.read_file('$res_output\\VC\\Auxiliary\\Build\\Microsoft.VCToolsVersion.default.txt') or {
|
version := os.read_file('$res_output\\VC\\Auxiliary\\Build\\Microsoft.VCToolsVersion.default.txt') or {
|
||||||
//println('Unable to find msvc version')
|
// println('Unable to find msvc version')
|
||||||
return error('Unable to find vs installation')
|
return error('Unable to find vs installation')
|
||||||
}
|
}
|
||||||
version2 := version // TODO remove. cgen option bug if expr
|
version2 := version // TODO remove. cgen option bug if expr
|
||||||
|
@ -214,13 +214,11 @@ pub fn (mut v Builder) cc_msvc() {
|
||||||
} else {
|
} else {
|
||||||
a << '/MDd'
|
a << '/MDd'
|
||||||
}
|
}
|
||||||
|
|
||||||
if v.pref.is_debug {
|
if v.pref.is_debug {
|
||||||
// /Zi generates a .pdb
|
// /Zi generates a .pdb
|
||||||
// /Fd sets the pdb file name (so its not just vc140 all the time)
|
// /Fd sets the pdb file name (so its not just vc140 all the time)
|
||||||
a << ['/Zi', '/Fd"$out_name_pdb"']
|
a << ['/Zi', '/Fd"$out_name_pdb"']
|
||||||
}
|
}
|
||||||
|
|
||||||
if v.pref.is_shared {
|
if v.pref.is_shared {
|
||||||
if !v.pref.out_name.ends_with('.dll') {
|
if !v.pref.out_name.ends_with('.dll') {
|
||||||
v.pref.out_name += '.dll'
|
v.pref.out_name += '.dll'
|
||||||
|
@ -261,7 +259,7 @@ pub fn (mut v Builder) cc_msvc() {
|
||||||
// Not all of these are needed (but the compiler should discard them if they are not used)
|
// Not all of these are needed (but the compiler should discard them if they are not used)
|
||||||
// these are the defaults used by msbuild and visual studio
|
// these are the defaults used by msbuild and visual studio
|
||||||
mut real_libs := ['kernel32.lib', 'user32.lib', 'advapi32.lib']
|
mut real_libs := ['kernel32.lib', 'user32.lib', 'advapi32.lib']
|
||||||
//sflags := v.get_os_cflags().msvc_string_flags()
|
// sflags := v.get_os_cflags().msvc_string_flags()
|
||||||
sflags := msvc_string_flags(v.get_os_cflags())
|
sflags := msvc_string_flags(v.get_os_cflags())
|
||||||
real_libs << sflags.real_libs
|
real_libs << sflags.real_libs
|
||||||
inc_paths := sflags.inc_paths
|
inc_paths := sflags.inc_paths
|
||||||
|
@ -324,7 +322,6 @@ pub fn (mut v Builder) cc_msvc() {
|
||||||
|
|
||||||
fn (mut v Builder) build_thirdparty_obj_file_with_msvc(path string, moduleflags []cflag.CFlag) {
|
fn (mut v Builder) build_thirdparty_obj_file_with_msvc(path string, moduleflags []cflag.CFlag) {
|
||||||
msvc := v.cached_msvc
|
msvc := v.cached_msvc
|
||||||
|
|
||||||
if msvc.valid == false {
|
if msvc.valid == false {
|
||||||
verror('Cannot find MSVC on this OS')
|
verror('Cannot find MSVC on this OS')
|
||||||
return
|
return
|
||||||
|
@ -368,7 +365,7 @@ mut:
|
||||||
other_flags []string
|
other_flags []string
|
||||||
}
|
}
|
||||||
|
|
||||||
//pub fn (cflags []CFlag) msvc_string_flags() MsvcStringFlags {
|
// pub fn (cflags []CFlag) msvc_string_flags() MsvcStringFlags {
|
||||||
pub fn msvc_string_flags(cflags []cflag.CFlag) MsvcStringFlags {
|
pub fn msvc_string_flags(cflags []cflag.CFlag) MsvcStringFlags {
|
||||||
mut real_libs := []string{}
|
mut real_libs := []string{}
|
||||||
mut inc_paths := []string{}
|
mut inc_paths := []string{}
|
||||||
|
|
|
@ -17,7 +17,6 @@ pub fn (mut b Builder) build_x64(v_files []string, out_file string) {
|
||||||
t1 := time.ticks()
|
t1 := time.ticks()
|
||||||
parse_time := t1 - t0
|
parse_time := t1 - t0
|
||||||
b.timing_message('PARSE', parse_time)
|
b.timing_message('PARSE', parse_time)
|
||||||
|
|
||||||
b.checker.check_files(b.parsed_files)
|
b.checker.check_files(b.parsed_files)
|
||||||
t2 := time.ticks()
|
t2 := time.ticks()
|
||||||
check_time := t2 - t1
|
check_time := t2 - t1
|
||||||
|
|
|
@ -18,7 +18,8 @@ fn test_all() {
|
||||||
// -prod so that warns are errors
|
// -prod so that warns are errors
|
||||||
total_errors += check_path(vexe, classic_dir, '-prod', '.out', classic_tests)
|
total_errors += check_path(vexe, classic_dir, '-prod', '.out', classic_tests)
|
||||||
total_errors += check_path(vexe, global_dir, '--enable-globals', '.out', global_tests)
|
total_errors += check_path(vexe, global_dir, '--enable-globals', '.out', global_tests)
|
||||||
total_errors += check_path(vexe, classic_dir, '--enable-globals run', '.run.out', ['globals_error.vv'])
|
total_errors += check_path(vexe, classic_dir, '--enable-globals run', '.run.out',
|
||||||
|
['globals_error.vv'])
|
||||||
total_errors += check_path(vexe, run_dir, 'run', '.run.out', run_tests)
|
total_errors += check_path(vexe, run_dir, 'run', '.run.out', run_tests)
|
||||||
total_errors += check_path(vexe, parser_dir, '-prod', '.out', parser_tests)
|
total_errors += check_path(vexe, parser_dir, '-prod', '.out', parser_tests)
|
||||||
assert total_errors == 0
|
assert total_errors == 0
|
||||||
|
@ -76,7 +77,7 @@ fn check_path(vexe, dir, voptions, result_extension string, tests []string) int
|
||||||
println('found:')
|
println('found:')
|
||||||
println(found)
|
println(found)
|
||||||
println('============\n')
|
println('============\n')
|
||||||
diff_content( expected, found )
|
diff_content(expected, found)
|
||||||
nb_fail++
|
nb_fail++
|
||||||
} else {
|
} else {
|
||||||
println(term.green('OK'))
|
println(term.green('OK'))
|
||||||
|
@ -95,9 +96,11 @@ fn clean_line_endings(s string) string {
|
||||||
return res
|
return res
|
||||||
}
|
}
|
||||||
|
|
||||||
fn diff_content(s1 string, s2 string) {
|
fn diff_content(s1, s2 string) {
|
||||||
diff_cmd := util.find_working_diff_command() or { return }
|
diff_cmd := util.find_working_diff_command() or {
|
||||||
|
return
|
||||||
|
}
|
||||||
println('diff: ')
|
println('diff: ')
|
||||||
println( util.color_compare_strings(diff_cmd, s1, s2) )
|
println(util.color_compare_strings(diff_cmd, s1, s2))
|
||||||
println('============\n')
|
println('============\n')
|
||||||
}
|
}
|
||||||
|
|
|
@ -914,7 +914,7 @@ pub fn (mut f Fmt) expr(node ast.Expr) {
|
||||||
}
|
}
|
||||||
ast.PrefixExpr {
|
ast.PrefixExpr {
|
||||||
f.write(node.op.str())
|
f.write(node.op.str())
|
||||||
f.prefix_expr_cast_expr( node.right )
|
f.prefix_expr_cast_expr(node.right)
|
||||||
}
|
}
|
||||||
ast.RangeExpr {
|
ast.RangeExpr {
|
||||||
f.expr(node.low)
|
f.expr(node.low)
|
||||||
|
@ -1274,12 +1274,8 @@ pub fn (mut f Fmt) infix_expr(node ast.InfixExpr) {
|
||||||
else {}
|
else {}
|
||||||
}
|
}
|
||||||
match node.right as right {
|
match node.right as right {
|
||||||
ast.InfixExpr {
|
ast.InfixExpr { penalty-- }
|
||||||
penalty--
|
ast.ParExpr { penalty = 1 }
|
||||||
}
|
|
||||||
ast.ParExpr {
|
|
||||||
penalty = 1
|
|
||||||
}
|
|
||||||
else {}
|
else {}
|
||||||
}
|
}
|
||||||
f.penalties << penalty
|
f.penalties << penalty
|
||||||
|
@ -1311,8 +1307,8 @@ pub fn (mut f Fmt) infix_expr(node ast.InfixExpr) {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (mut f Fmt) if_expr(it ast.IfExpr) {
|
pub fn (mut f Fmt) if_expr(it ast.IfExpr) {
|
||||||
single_line := it.branches.len == 2 && it.has_else &&
|
single_line := it.branches.len == 2 && it.has_else && it.branches[0].stmts.len == 1 &&
|
||||||
it.branches[0].stmts.len == 1 && it.branches[1].stmts.len == 1 &&
|
it.branches[1].stmts.len == 1 &&
|
||||||
(it.is_expr || f.is_assign)
|
(it.is_expr || f.is_assign)
|
||||||
f.single_line_if = single_line
|
f.single_line_if = single_line
|
||||||
for i, branch in it.branches {
|
for i, branch in it.branches {
|
||||||
|
@ -1321,11 +1317,9 @@ pub fn (mut f Fmt) if_expr(it ast.IfExpr) {
|
||||||
if branch.cond is ast.InfixExpr {
|
if branch.cond is ast.InfixExpr {
|
||||||
infix := branch.cond as ast.InfixExpr
|
infix := branch.cond as ast.InfixExpr
|
||||||
if infix.op == .key_is &&
|
if infix.op == .key_is &&
|
||||||
(infix.left is ast.Ident || infix.left is ast.SelectorExpr) &&
|
(infix.left is ast.Ident || infix.left is ast.SelectorExpr) && infix.right is ast.Type {
|
||||||
infix.right is ast.Type {
|
// right_expr := infix.right as ast.Type
|
||||||
//right_expr := infix.right as ast.Type
|
is_variable = if infix.left is ast.Ident { (infix.left as ast.Ident).kind == .variable } else { true }
|
||||||
is_variable = if infix.left is ast.Ident { (infix.left as ast.Ident).kind ==
|
|
||||||
.variable } else { true }
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if i == 0 {
|
if i == 0 {
|
||||||
|
@ -1661,15 +1655,12 @@ pub fn (mut f Fmt) array_init(it ast.ArrayInit) {
|
||||||
}
|
}
|
||||||
mut penalty := if f.array_init_break[f.array_init_depth - 1] { 0 } else { 3 }
|
mut penalty := if f.array_init_break[f.array_init_depth - 1] { 0 } else { 3 }
|
||||||
if penalty > 0 {
|
if penalty > 0 {
|
||||||
if i == 0 ||
|
if i == 0 || it.exprs[i - 1] is ast.ArrayInit || it.exprs[i - 1] is ast.StructInit ||
|
||||||
it.exprs[i - 1] is ast.ArrayInit ||
|
|
||||||
it.exprs[i - 1] is ast.StructInit ||
|
|
||||||
it.exprs[i - 1] is ast.MapInit || it.exprs[i - 1] is ast.CallExpr {
|
it.exprs[i - 1] is ast.MapInit || it.exprs[i - 1] is ast.CallExpr {
|
||||||
penalty--
|
penalty--
|
||||||
}
|
}
|
||||||
if expr is ast.ArrayInit ||
|
if expr is ast.ArrayInit ||
|
||||||
expr is ast.StructInit || expr is ast.MapInit ||
|
expr is ast.StructInit || expr is ast.MapInit || expr is ast.CallExpr {
|
||||||
expr is ast.CallExpr {
|
|
||||||
penalty--
|
penalty--
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1786,10 +1777,10 @@ pub fn (mut f Fmt) const_decl(it ast.ConstDecl) {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn (mut f Fmt) is_external_name(name string) bool {
|
fn (mut f Fmt) is_external_name(name string) bool {
|
||||||
if name.len > 2 && name[0]==`C` && name[1]==`.` {
|
if name.len > 2 && name[0] == `C` && name[1] == `.` {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
if name.len > 3 && name[0]==`J` && name[1]==`S` && name[2] == `.` {
|
if name.len > 3 && name[0] == `J` && name[1] == `S` && name[2] == `.` {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
|
|
|
@ -11,6 +11,8 @@ fn vararg_test() {
|
||||||
variadic(1, 2, 3)
|
variadic(1, 2, 3)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO Remove `fn main` once vet supports scripts
|
||||||
|
fn main() {
|
||||||
vararg_test()
|
vararg_test()
|
||||||
|
|
||||||
arr1 := ['Hello', 'JS', 'Backend']
|
arr1 := ['Hello', 'JS', 'Backend']
|
||||||
|
@ -118,3 +120,4 @@ $f1
|
||||||
$f2
|
$f2
|
||||||
$f3
|
$f3
|
||||||
$f4')
|
$f4')
|
||||||
|
}
|
||||||
|
|
|
@ -6,6 +6,8 @@ enum Test {
|
||||||
baz
|
baz
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO Remove `fn main` once vet supports scripts
|
||||||
|
fn main() {
|
||||||
mut a := hello.Ccc.a
|
mut a := hello.Ccc.a
|
||||||
a = .b
|
a = .b
|
||||||
a = .c
|
a = .c
|
||||||
|
@ -15,3 +17,4 @@ println(a)
|
||||||
mut b := Test.foo
|
mut b := Test.foo
|
||||||
b = .bar
|
b = .bar
|
||||||
println(b)
|
println(b)
|
||||||
|
}
|
||||||
|
|
|
@ -27,87 +27,73 @@ fn class(extends string, instanceof int) {
|
||||||
_ = delete
|
_ = delete
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
println('Hello from V.js!')
|
println('Hello from V.js!')
|
||||||
println(JS.Math.atan2(1, 0))
|
println(JS.Math.atan2(1, 0))
|
||||||
|
|
||||||
mut a := 1
|
mut a := 1
|
||||||
a *= 2
|
a *= 2
|
||||||
a += 3
|
a += 3
|
||||||
println(a) // TODO: Handle string interpolation
|
println(a) // TODO: Handle string interpolation
|
||||||
|
|
||||||
mut b := hl.Aaa{}
|
mut b := hl.Aaa{}
|
||||||
b.update('an update')
|
b.update('an update')
|
||||||
println(b)
|
println(b)
|
||||||
|
mut c := Foo{hl.Aaa{}}
|
||||||
mut c := Foo{ hl.Aaa{} }
|
|
||||||
c.a.update('another update')
|
c.a.update('another update')
|
||||||
println(c)
|
println(c)
|
||||||
|
_ = 'done'
|
||||||
_ = "done"
|
|
||||||
{
|
{
|
||||||
_ = "block"
|
_ = 'block'
|
||||||
}
|
}
|
||||||
|
|
||||||
_ = POSITION.go_back
|
_ = POSITION.go_back
|
||||||
_ = hl.Ccc.a
|
_ = hl.Ccc.a
|
||||||
|
|
||||||
debugger := 'JS keywords'
|
debugger := 'JS keywords'
|
||||||
// TODO: Implement interpolation
|
// TODO: Implement interpolation
|
||||||
await := '$super: $debugger'
|
await := '$super: $debugger'
|
||||||
mut finally := 'implemented'
|
mut finally := 'implemented'
|
||||||
|
|
||||||
println('$await $finally')
|
println('$await $finally')
|
||||||
|
|
||||||
dun := i_am_a_const * 20
|
dun := i_am_a_const * 20
|
||||||
dunn := hl.hello // External constant
|
dunn := hl.hello // External constant
|
||||||
_ = hl1.nested()
|
_ = hl1.nested()
|
||||||
|
for i := 0; i < 10; i++ {
|
||||||
for i := 0; i < 10; i++ {}
|
}
|
||||||
|
for i, x in 'hello' {
|
||||||
for i, x in 'hello' {}
|
}
|
||||||
|
for x in 1 .. 10 {
|
||||||
for x in 1..10 {}
|
}
|
||||||
|
arr := [1, 2, 3, 4, 5]
|
||||||
arr := [1,2,3,4,5]
|
for i in arr {
|
||||||
for i in arr {}
|
}
|
||||||
|
ma := {
|
||||||
ma := {
|
'str': 'done'
|
||||||
'str': "done"
|
'ddo': 'baba'
|
||||||
'ddo': "baba"
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// panic('foo')
|
// panic('foo')
|
||||||
for m, n in ma {
|
for m, n in ma {
|
||||||
iss := m
|
iss := m
|
||||||
}
|
}
|
||||||
|
go async(0, 'hello')
|
||||||
go async(0, "hello")
|
|
||||||
|
|
||||||
fn_in_var := fn (number int) {
|
fn_in_var := fn (number int) {
|
||||||
println("number: $number")
|
println('number: $number')
|
||||||
}
|
}
|
||||||
|
|
||||||
hl.debugger()
|
hl.debugger()
|
||||||
anon_consumer(hl.excited(), fn (message string) {
|
anon_consumer(hl.excited(), fn (message string) {
|
||||||
println(message)
|
println(message)
|
||||||
})
|
})
|
||||||
|
|
||||||
hl.raw_js_log()
|
hl.raw_js_log()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn anon_consumer (greeting string, anon fn(message string)) {
|
fn anon_consumer(greeting string, anon fn (message string)) {
|
||||||
anon(greeting)
|
anon(greeting)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn async(num int, def string) {}
|
fn async(num int, def string) {
|
||||||
|
}
|
||||||
|
|
||||||
[inline]
|
[inline]
|
||||||
[deprecated]
|
[deprecated]
|
||||||
fn hello(game_on int, dummy ...string) (int, int) {
|
fn hello(game_on int, dummy ...string) (int, int) {
|
||||||
defer {
|
defer {
|
||||||
do := "not"
|
do := 'not'
|
||||||
}
|
}
|
||||||
for dd in dummy {
|
for dd in dummy {
|
||||||
l := dd
|
l := dd
|
||||||
|
@ -116,18 +102,20 @@ fn hello(game_on int, dummy ...string) (int, int) {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn (it Companies) method() int {
|
fn (it Companies) method() int {
|
||||||
|
ss := Companies{
|
||||||
ss := Companies {
|
|
||||||
google: 2
|
google: 2
|
||||||
amazon: true
|
amazon: true
|
||||||
yahoo: "hello"
|
yahoo: 'hello'
|
||||||
}
|
}
|
||||||
|
|
||||||
a, b := hello(2, 'google', 'not google')
|
a, b := hello(2, 'google', 'not google')
|
||||||
|
glue := if a > 2 {
|
||||||
glue := if a > 2 { 'more_glue' } else if a > 5 {'more glueee'} else { 'less glue' }
|
'more_glue'
|
||||||
|
} else if a > 5 {
|
||||||
if a != 2 {}
|
'more glueee'
|
||||||
|
} else {
|
||||||
|
'less glue'
|
||||||
|
}
|
||||||
|
if a != 2 {
|
||||||
|
}
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
|
@ -51,6 +51,8 @@ fn show(game [][]bool) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO Remove `fn main` once vet supports scripts
|
||||||
|
fn main() {
|
||||||
mut game := [][]bool{ len: h, init: []bool{ len: w } }
|
mut game := [][]bool{ len: h, init: []bool{ len: w } }
|
||||||
|
|
||||||
game[11][15] = true
|
game[11][15] = true
|
||||||
|
@ -62,3 +64,4 @@ game[12][21] = true
|
||||||
game[12][22] = true
|
game[12][22] = true
|
||||||
|
|
||||||
JS.setInterval(fn () { show(game) game = step(game) }, 500)
|
JS.setInterval(fn () { show(game) game = step(game) }, 500)
|
||||||
|
}
|
||||||
|
|
|
@ -20,17 +20,21 @@ struct Config {
|
||||||
bar string
|
bar string
|
||||||
}
|
}
|
||||||
|
|
||||||
fn use_config(c Config) {}
|
fn use_config(c Config) {
|
||||||
|
}
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
mut a := Int { value: 10 }
|
mut a := Int{
|
||||||
|
value: 10
|
||||||
|
}
|
||||||
a.add(5)
|
a.add(5)
|
||||||
println(a) // 15
|
println(a) // 15
|
||||||
|
|
||||||
mut b := Int{}
|
mut b := Int{}
|
||||||
b.add(10)
|
b.add(10)
|
||||||
println(b.get()) // 10
|
println(b.get()) // 10
|
||||||
|
use_config(Config{2, 'bar'})
|
||||||
use_config(Config{ 2, 'bar' })
|
use_config({
|
||||||
use_config(foo: 2, bar: 'bar')
|
foo: 2
|
||||||
|
bar: 'bar'
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,7 +23,7 @@ fn (mut g Gen) gen_json_for_type(typ table.Type) {
|
||||||
mut enc := strings.new_builder(100)
|
mut enc := strings.new_builder(100)
|
||||||
sym := g.table.get_type_symbol(typ)
|
sym := g.table.get_type_symbol(typ)
|
||||||
styp := g.typ(typ)
|
styp := g.typ(typ)
|
||||||
if is_js_prim(sym.name) || sym.kind == .enum_{
|
if is_js_prim(sym.name) || sym.kind == .enum_ {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if sym.kind == .array {
|
if sym.kind == .array {
|
||||||
|
@ -38,7 +38,6 @@ fn (mut g Gen) gen_json_for_type(typ table.Type) {
|
||||||
// cJSON_Parse(str) call is added by the compiler
|
// cJSON_Parse(str) call is added by the compiler
|
||||||
// Code gen decoder
|
// Code gen decoder
|
||||||
dec_fn_name := js_dec_name(sym.name)
|
dec_fn_name := js_dec_name(sym.name)
|
||||||
|
|
||||||
// Make sure that this optional type actually exists
|
// Make sure that this optional type actually exists
|
||||||
g.register_optional(typ)
|
g.register_optional(typ)
|
||||||
dec_fn_dec := 'Option_$styp ${dec_fn_name}(cJSON* root)'
|
dec_fn_dec := 'Option_$styp ${dec_fn_name}(cJSON* root)'
|
||||||
|
@ -49,15 +48,14 @@ $dec_fn_dec {
|
||||||
if (!root) {
|
if (!root) {
|
||||||
const char *error_ptr = cJSON_GetErrorPtr();
|
const char *error_ptr = cJSON_GetErrorPtr();
|
||||||
if (error_ptr != NULL) {
|
if (error_ptr != NULL) {
|
||||||
// fprintf(stderr, "Error in decode() for $styp error_ptr=: %%s\\n", error_ptr);
|
// fprintf(stderr, "Error in decode() for $styp error_ptr=: %%s\\n", error_ptr);
|
||||||
// printf("\\nbad js=%%s\\n", js.str);
|
// printf("\\nbad js=%%s\\n", js.str);
|
||||||
Option err = v_error(tos2(error_ptr));
|
Option err = v_error(tos2(error_ptr));
|
||||||
return *(Option_$styp *)&err;
|
return *(Option_$styp *)&err;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
')
|
')
|
||||||
g.json_forward_decls.writeln('$dec_fn_dec;')
|
g.json_forward_decls.writeln('$dec_fn_dec;')
|
||||||
|
|
||||||
// Code gen encoder
|
// Code gen encoder
|
||||||
// encode_TYPE funcs receive an object to encode
|
// encode_TYPE funcs receive an object to encode
|
||||||
enc_fn_name := js_enc_name(sym.name)
|
enc_fn_name := js_enc_name(sym.name)
|
||||||
|
@ -107,11 +105,9 @@ $enc_fn_dec {
|
||||||
dec.writeln(' res . ${c_name(field.name)} = *($field_type*) $dec_name (js_get(root,"$name")).data;')
|
dec.writeln(' res . ${c_name(field.name)} = *($field_type*) $dec_name (js_get(root,"$name")).data;')
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
mut enc_name := js_enc_name(field_type)
|
mut enc_name := js_enc_name(field_type)
|
||||||
if g.table.get_type_symbol(field.typ).kind == .enum_ {
|
if g.table.get_type_symbol(field.typ).kind == .enum_ {
|
||||||
enc.writeln('\tcJSON_AddItemToObject(o, "$name", json__encode_u64(val.${c_name(field.name)}));')
|
enc.writeln('\tcJSON_AddItemToObject(o, "$name", json__encode_u64(val.${c_name(field.name)}));')
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
enc.writeln('\tcJSON_AddItemToObject(o, "$name", ${enc_name}(val.${c_name(field.name)}));')
|
enc.writeln('\tcJSON_AddItemToObject(o, "$name", ${enc_name}(val.${c_name(field.name)}));')
|
||||||
}
|
}
|
||||||
|
|
|
@ -40,12 +40,13 @@ fn (mut g Gen) generate_hotcode_reloader_code() {
|
||||||
mut load_code := []string{}
|
mut load_code := []string{}
|
||||||
if g.pref.os != .windows {
|
if g.pref.os != .windows {
|
||||||
for so_fn in g.hotcode_fn_names {
|
for so_fn in g.hotcode_fn_names {
|
||||||
load_code << 'impl_live_${so_fn} = dlsym(live_lib, "impl_live_${so_fn}");'
|
load_code << 'impl_live_$so_fn = dlsym(live_lib, "impl_live_$so_fn");'
|
||||||
}
|
}
|
||||||
phd = posix_hotcode_definitions_1
|
phd = posix_hotcode_definitions_1
|
||||||
} else {
|
} else {
|
||||||
for so_fn in g.hotcode_fn_names {
|
for so_fn in g.hotcode_fn_names {
|
||||||
load_code << 'impl_live_${so_fn} = (void *)GetProcAddress(live_lib, "impl_live_${so_fn}"); '
|
load_code <<
|
||||||
|
'impl_live_$so_fn = (void *)GetProcAddress(live_lib, "impl_live_$so_fn"); '
|
||||||
}
|
}
|
||||||
phd = windows_hotcode_definitions_1
|
phd = windows_hotcode_definitions_1
|
||||||
}
|
}
|
||||||
|
@ -76,10 +77,10 @@ fn (mut g Gen) generate_hotcode_reloading_main_caller() {
|
||||||
g.writeln('\t{')
|
g.writeln('\t{')
|
||||||
g.writeln('\t\t// initialization of live function pointers')
|
g.writeln('\t\t// initialization of live function pointers')
|
||||||
for fname in g.hotcode_fn_names {
|
for fname in g.hotcode_fn_names {
|
||||||
g.writeln('\t\timpl_live_${fname} = 0;')
|
g.writeln('\t\timpl_live_$fname = 0;')
|
||||||
}
|
}
|
||||||
vexe := util.cescaped_path( pref.vexe_path() )
|
vexe := util.cescaped_path(pref.vexe_path())
|
||||||
file := util.cescaped_path( g.pref.path )
|
file := util.cescaped_path(g.pref.path)
|
||||||
msvc := if g.pref.ccompiler == 'msvc' { '-cc msvc' } else { '' }
|
msvc := if g.pref.ccompiler == 'msvc' { '-cc msvc' } else { '' }
|
||||||
so_debug_flag := if g.pref.is_debug { '-cg' } else { '' }
|
so_debug_flag := if g.pref.is_debug { '-cg' } else { '' }
|
||||||
vopts := '$msvc $so_debug_flag -sharedlive -shared'
|
vopts := '$msvc $so_debug_flag -sharedlive -shared'
|
||||||
|
|
|
@ -4,6 +4,8 @@ import v.ast
|
||||||
import v.table
|
import v.table
|
||||||
|
|
||||||
fn (mut p Parser) lock_expr() ast.LockExpr {
|
fn (mut p Parser) lock_expr() ast.LockExpr {
|
||||||
|
// TODO Handle aliasing sync
|
||||||
|
p.register_used_import('sync')
|
||||||
pos := p.tok.position()
|
pos := p.tok.position()
|
||||||
is_rlock := p.tok.kind == .key_rlock
|
is_rlock := p.tok.kind == .key_rlock
|
||||||
p.next()
|
p.next()
|
||||||
|
|
|
@ -30,15 +30,16 @@ fn (mut p Parser) register_used_import(alias string) {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn (mut p Parser) check_unused_imports() {
|
fn (mut p Parser) check_unused_imports() {
|
||||||
if p.pref.is_repl {
|
if p.pref.is_repl || p.pref.is_fmt {
|
||||||
// The REPL should be much more liberal, and should not warn about
|
// The REPL should be much more liberal, and should not warn about
|
||||||
// unused imports, because they probably will be in the next few lines...
|
// unused imports, because they probably will be in the next few lines...
|
||||||
|
// vfmt doesn't care about unused imports either
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
for import_m in p.ast_imports {
|
for import_m in p.ast_imports {
|
||||||
alias := import_m.alias
|
alias := import_m.alias
|
||||||
mod := import_m.mod
|
mod := import_m.mod
|
||||||
if !p.is_used_import(alias) && !p.pref.is_fmt {
|
if !p.is_used_import(alias) {
|
||||||
mod_alias := if alias == mod { alias } else { '$alias ($mod)' }
|
mod_alias := if alias == mod { alias } else { '$alias ($mod)' }
|
||||||
p.warn_with_pos("module '$mod_alias' is imported but never used", import_m.pos)
|
p.warn_with_pos("module '$mod_alias' is imported but never used", import_m.pos)
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,8 +15,8 @@ import strings
|
||||||
|
|
||||||
pub type Type int
|
pub type Type int
|
||||||
|
|
||||||
pub type TypeInfo = Alias | Array | ArrayFixed | Enum | FnType | Interface | Map | MultiReturn |
|
pub type TypeInfo = Alias | Array | ArrayFixed | Enum | FnType | GenericStructInst | Interface |
|
||||||
Struct | GenericStructInst | SumType
|
Map | MultiReturn | Struct | SumType
|
||||||
|
|
||||||
pub enum Language {
|
pub enum Language {
|
||||||
v
|
v
|
||||||
|
@ -46,10 +46,9 @@ pub enum TypeFlag {
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
To save precious TypeFlag bits the 4 possible ShareTypes are coded in the two
|
To save precious TypeFlag bits the 4 possible ShareTypes are coded in the two
|
||||||
bits `shared` and `atomic_or_rw` (see sharetype_from_flags() below).
|
bits `shared` and `atomic_or_rw` (see sharetype_from_flags() below).
|
||||||
*/
|
*/
|
||||||
|
|
||||||
pub enum ShareType {
|
pub enum ShareType {
|
||||||
mut_t
|
mut_t
|
||||||
shared_t
|
shared_t
|
||||||
|
@ -127,7 +126,6 @@ pub fn (t Type) to_ptr() Type {
|
||||||
if nr_muls == 255 {
|
if nr_muls == 255 {
|
||||||
panic('to_ptr: nr_muls is already at max of 255')
|
panic('to_ptr: nr_muls is already at max of 255')
|
||||||
}
|
}
|
||||||
|
|
||||||
return int(t) & 0xff00ffff | ((nr_muls + 1) << 16)
|
return int(t) & 0xff00ffff | ((nr_muls + 1) << 16)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -263,24 +261,12 @@ pub const (
|
||||||
|
|
||||||
pub const (
|
pub const (
|
||||||
integer_type_idxs = [i8_type_idx, i16_type_idx, int_type_idx, i64_type_idx, byte_type_idx,
|
integer_type_idxs = [i8_type_idx, i16_type_idx, int_type_idx, i64_type_idx, byte_type_idx,
|
||||||
u16_type_idx,
|
u16_type_idx, u32_type_idx, u64_type_idx, any_int_type_idx]
|
||||||
u32_type_idx,
|
|
||||||
u64_type_idx,
|
|
||||||
any_int_type_idx
|
|
||||||
]
|
|
||||||
signed_integer_type_idxs = [i8_type_idx, i16_type_idx, int_type_idx, i64_type_idx]
|
signed_integer_type_idxs = [i8_type_idx, i16_type_idx, int_type_idx, i64_type_idx]
|
||||||
unsigned_integer_type_idxs = [byte_type_idx, u16_type_idx, u32_type_idx, u64_type_idx]
|
unsigned_integer_type_idxs = [byte_type_idx, u16_type_idx, u32_type_idx, u64_type_idx]
|
||||||
float_type_idxs = [f32_type_idx, f64_type_idx, any_flt_type_idx]
|
float_type_idxs = [f32_type_idx, f64_type_idx, any_flt_type_idx]
|
||||||
number_type_idxs = [i8_type_idx, i16_type_idx, int_type_idx,
|
number_type_idxs = [i8_type_idx, i16_type_idx, int_type_idx, i64_type_idx, byte_type_idx,
|
||||||
i64_type_idx, byte_type_idx,
|
u16_type_idx, u32_type_idx, u64_type_idx, f32_type_idx, f64_type_idx, any_int_type_idx, any_flt_type_idx]
|
||||||
u16_type_idx,
|
|
||||||
u32_type_idx,
|
|
||||||
u64_type_idx,
|
|
||||||
f32_type_idx,
|
|
||||||
f64_type_idx,
|
|
||||||
any_int_type_idx,
|
|
||||||
any_flt_type_idx
|
|
||||||
]
|
|
||||||
pointer_type_idxs = [voidptr_type_idx, byteptr_type_idx, charptr_type_idx]
|
pointer_type_idxs = [voidptr_type_idx, byteptr_type_idx, charptr_type_idx]
|
||||||
string_type_idxs = [string_type_idx, ustring_type_idx]
|
string_type_idxs = [string_type_idx, ustring_type_idx]
|
||||||
)
|
)
|
||||||
|
@ -314,13 +300,9 @@ pub const (
|
||||||
)
|
)
|
||||||
|
|
||||||
pub const (
|
pub const (
|
||||||
builtin_type_names = ['void', 'voidptr', 'charptr', 'byteptr', 'i8', 'i16', 'int', 'i64',
|
builtin_type_names = ['void', 'voidptr', 'charptr', 'byteptr', 'i8', 'i16', 'int', 'i64', 'u16',
|
||||||
'u16',
|
'u32', 'u64', 'any_int', 'f32', 'f64', 'any_float', 'string', 'ustring', 'char', 'byte', 'bool',
|
||||||
'u32',
|
'none', 'array', 'array_fixed', 'map', 'any', 'struct', 'mapnode', 'size_t']
|
||||||
'u64', 'any_int', 'f32', 'f64', 'any_float', 'string', 'ustring', 'char', 'byte',
|
|
||||||
'bool', 'none', 'array', 'array_fixed',
|
|
||||||
'map', 'any', 'struct',
|
|
||||||
'mapnode', 'size_t']
|
|
||||||
)
|
)
|
||||||
|
|
||||||
pub struct MultiReturn {
|
pub struct MultiReturn {
|
||||||
|
|
|
@ -3,85 +3,66 @@ import table
|
||||||
fn test_idx() {
|
fn test_idx() {
|
||||||
mut t := table.new_type(table.void_type_idx)
|
mut t := table.new_type(table.void_type_idx)
|
||||||
assert t.idx() == table.void_type_idx
|
assert t.idx() == table.void_type_idx
|
||||||
|
|
||||||
t = table.new_type(table.i8_type_idx)
|
t = table.new_type(table.i8_type_idx)
|
||||||
assert t.idx() == table.i8_type_idx
|
assert t.idx() == table.i8_type_idx
|
||||||
}
|
}
|
||||||
|
|
||||||
fn test_muls() {
|
fn test_muls() {
|
||||||
mut t := table.new_type(table.void_type_idx)
|
mut t := table.new_type(table.void_type_idx)
|
||||||
|
|
||||||
idx := t.idx()
|
idx := t.idx()
|
||||||
|
|
||||||
assert t.nr_muls() == 0
|
assert t.nr_muls() == 0
|
||||||
|
for i in 0 .. 32 {
|
||||||
for i in 0..32 {
|
|
||||||
t = t.set_nr_muls(i)
|
t = t.set_nr_muls(i)
|
||||||
assert t.nr_muls() == i
|
assert t.nr_muls() == i
|
||||||
}
|
}
|
||||||
|
|
||||||
t = t.set_nr_muls(0)
|
t = t.set_nr_muls(0)
|
||||||
assert t.nr_muls() == 0
|
assert t.nr_muls() == 0
|
||||||
assert t.is_ptr() == false
|
assert t.is_ptr() == false
|
||||||
|
|
||||||
t = t.to_ptr()
|
t = t.to_ptr()
|
||||||
assert t.nr_muls() == 1
|
assert t.nr_muls() == 1
|
||||||
assert t.is_ptr() == true
|
assert t.is_ptr() == true
|
||||||
|
|
||||||
t = t.to_ptr()
|
t = t.to_ptr()
|
||||||
assert t.nr_muls() == 2
|
assert t.nr_muls() == 2
|
||||||
assert t.is_ptr() == true
|
assert t.is_ptr() == true
|
||||||
|
|
||||||
t = t.deref()
|
t = t.deref()
|
||||||
assert t.nr_muls() == 1
|
assert t.nr_muls() == 1
|
||||||
assert t.is_ptr() == true
|
assert t.is_ptr() == true
|
||||||
|
|
||||||
t = t.deref()
|
t = t.deref()
|
||||||
assert t.nr_muls() == 0
|
assert t.nr_muls() == 0
|
||||||
assert t.is_ptr() == false
|
assert t.is_ptr() == false
|
||||||
|
|
||||||
assert t.idx() == idx
|
assert t.idx() == idx
|
||||||
}
|
}
|
||||||
|
|
||||||
fn test_flags() {
|
fn test_flags() {
|
||||||
mut t := table.new_type(table.void_type_idx)
|
mut t := table.new_type(table.void_type_idx)
|
||||||
|
|
||||||
idx := t.idx()
|
idx := t.idx()
|
||||||
nr_muls := t.nr_muls()
|
nr_muls := t.nr_muls()
|
||||||
|
|
||||||
t = t.set_flag(table.TypeFlag.optional)
|
t = t.set_flag(table.TypeFlag.optional)
|
||||||
assert t.has_flag(table.TypeFlag.optional) == true
|
assert t.has_flag(table.TypeFlag.optional) == true
|
||||||
assert t.has_flag(table.TypeFlag.variadic) == false
|
assert t.has_flag(table.TypeFlag.variadic) == false
|
||||||
assert t.has_flag(table.TypeFlag.generic) == false
|
assert t.has_flag(table.TypeFlag.generic) == false
|
||||||
|
|
||||||
t = t.set_flag(table.TypeFlag.variadic)
|
t = t.set_flag(table.TypeFlag.variadic)
|
||||||
assert t.has_flag(table.TypeFlag.optional) == true
|
assert t.has_flag(table.TypeFlag.optional) == true
|
||||||
assert t.has_flag(table.TypeFlag.variadic) == true
|
assert t.has_flag(table.TypeFlag.variadic) == true
|
||||||
assert t.has_flag(table.TypeFlag.generic) == false
|
assert t.has_flag(table.TypeFlag.generic) == false
|
||||||
|
|
||||||
t = t.set_flag(table.TypeFlag.generic)
|
t = t.set_flag(table.TypeFlag.generic)
|
||||||
assert t.has_flag(table.TypeFlag.optional) == true
|
assert t.has_flag(table.TypeFlag.optional) == true
|
||||||
assert t.has_flag(table.TypeFlag.variadic) == true
|
assert t.has_flag(table.TypeFlag.variadic) == true
|
||||||
assert t.has_flag(table.TypeFlag.generic) == true
|
assert t.has_flag(table.TypeFlag.generic) == true
|
||||||
|
|
||||||
assert t.idx() == idx
|
assert t.idx() == idx
|
||||||
assert t.nr_muls() == nr_muls
|
assert t.nr_muls() == nr_muls
|
||||||
|
|
||||||
t = t.clear_flag(table.TypeFlag.optional)
|
t = t.clear_flag(table.TypeFlag.optional)
|
||||||
assert t.has_flag(table.TypeFlag.optional) == false
|
assert t.has_flag(table.TypeFlag.optional) == false
|
||||||
assert t.has_flag(table.TypeFlag.variadic) == true
|
assert t.has_flag(table.TypeFlag.variadic) == true
|
||||||
assert t.has_flag(table.TypeFlag.generic) == true
|
assert t.has_flag(table.TypeFlag.generic) == true
|
||||||
|
|
||||||
t = t.clear_flag(table.TypeFlag.variadic)
|
t = t.clear_flag(table.TypeFlag.variadic)
|
||||||
assert t.has_flag(table.TypeFlag.optional) == false
|
assert t.has_flag(table.TypeFlag.optional) == false
|
||||||
assert t.has_flag(table.TypeFlag.variadic) == false
|
assert t.has_flag(table.TypeFlag.variadic) == false
|
||||||
assert t.has_flag(table.TypeFlag.generic) == true
|
assert t.has_flag(table.TypeFlag.generic) == true
|
||||||
|
|
||||||
t = t.clear_flag(table.TypeFlag.generic)
|
t = t.clear_flag(table.TypeFlag.generic)
|
||||||
assert t.has_flag(table.TypeFlag.optional) == false
|
assert t.has_flag(table.TypeFlag.optional) == false
|
||||||
assert t.has_flag(table.TypeFlag.variadic) == false
|
assert t.has_flag(table.TypeFlag.variadic) == false
|
||||||
assert t.has_flag(table.TypeFlag.generic) == false
|
assert t.has_flag(table.TypeFlag.generic) == false
|
||||||
|
|
||||||
assert t.idx() == idx
|
assert t.idx() == idx
|
||||||
assert t.nr_muls() == nr_muls
|
assert t.nr_muls() == nr_muls
|
||||||
}
|
}
|
||||||
|
@ -91,12 +72,10 @@ fn test_derive() {
|
||||||
t = t.set_flag(table.TypeFlag.generic)
|
t = t.set_flag(table.TypeFlag.generic)
|
||||||
t = t.set_flag(table.TypeFlag.variadic)
|
t = t.set_flag(table.TypeFlag.variadic)
|
||||||
t = t.set_nr_muls(10)
|
t = t.set_nr_muls(10)
|
||||||
|
|
||||||
mut t2 := table.new_type(table.i16_type_idx)
|
mut t2 := table.new_type(table.i16_type_idx)
|
||||||
t2 = t2.derive(t)
|
t2 = t2.derive(t)
|
||||||
assert t2.has_flag(table.TypeFlag.optional) == false
|
assert t2.has_flag(table.TypeFlag.optional) == false
|
||||||
assert t2.has_flag(table.TypeFlag.variadic) == true
|
assert t2.has_flag(table.TypeFlag.variadic) == true
|
||||||
assert t2.has_flag(table.TypeFlag.generic) == true
|
assert t2.has_flag(table.TypeFlag.generic) == true
|
||||||
assert t2.nr_muls() == 10
|
assert t2.nr_muls() == 10
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,7 +19,7 @@ fn test_parse_valid_cflags() {
|
||||||
make_flag('darwin', '-framework', 'Cocoa'),
|
make_flag('darwin', '-framework', 'Cocoa'),
|
||||||
make_flag('windows', '-l', 'gdi32'),
|
make_flag('windows', '-l', 'gdi32'),
|
||||||
make_flag(no_os, '-l', 'mysqlclient'),
|
make_flag(no_os, '-l', 'mysqlclient'),
|
||||||
make_flag(no_os, no_name, '-test')
|
make_flag(no_os, no_name, '-test'),
|
||||||
]
|
]
|
||||||
parse_valid_flag(mut t, '-lmysqlclient')
|
parse_valid_flag(mut t, '-lmysqlclient')
|
||||||
parse_valid_flag(mut t, '-test')
|
parse_valid_flag(mut t, '-test')
|
||||||
|
@ -54,8 +54,7 @@ fn test_parse_invalid_cflags() {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_valid_flag(mut t table.Table, flag string) {
|
fn parse_valid_flag(mut t table.Table, flag string) {
|
||||||
t.parse_cflag(flag, module_name, cdefines) or {
|
t.parse_cflag(flag, module_name, cdefines) or { }
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn assert_parse_invalid_flag(mut t table.Table, flag string) {
|
fn assert_parse_invalid_flag(mut t table.Table, flag string) {
|
||||||
|
|
|
@ -194,7 +194,7 @@ pub fn (t &Table) get_type_symbol(typ Type) &TypeSymbol {
|
||||||
return &t.types[idx]
|
return &t.types[idx]
|
||||||
}
|
}
|
||||||
// this should never happen
|
// this should never happen
|
||||||
panic('get_type_symbol: invalid type (typ=$typ idx=${idx}). Compiler bug. This should never happen')
|
panic('get_type_symbol: invalid type (typ=$typ idx=$idx). Compiler bug. This should never happen')
|
||||||
}
|
}
|
||||||
|
|
||||||
// get_final_type_symbol follows aliases until it gets to a "real" Type
|
// get_final_type_symbol follows aliases until it gets to a "real" Type
|
||||||
|
@ -210,7 +210,7 @@ pub fn (t &Table) get_final_type_symbol(typ Type) &TypeSymbol {
|
||||||
return &t.types[idx]
|
return &t.types[idx]
|
||||||
}
|
}
|
||||||
// this should never happen
|
// this should never happen
|
||||||
panic('get_final_type_symbol: invalid type (typ=$typ idx=${idx}). Compiler bug. This should never happen')
|
panic('get_final_type_symbol: invalid type (typ=$typ idx=$idx). Compiler bug. This should never happen')
|
||||||
}
|
}
|
||||||
|
|
||||||
[inline]
|
[inline]
|
||||||
|
@ -245,8 +245,7 @@ pub fn (mut t Table) register_builtin_type_symbol(typ TypeSymbol) int {
|
||||||
typ |
|
typ |
|
||||||
kind: existing_type.kind
|
kind: existing_type.kind
|
||||||
}
|
}
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
t.types[existing_idx] = typ
|
t.types[existing_idx] = typ
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -296,7 +295,7 @@ pub fn (t &Table) known_type(name string) bool {
|
||||||
[inline]
|
[inline]
|
||||||
pub fn (t &Table) array_name(elem_type Type, nr_dims int) string {
|
pub fn (t &Table) array_name(elem_type Type, nr_dims int) string {
|
||||||
elem_type_sym := t.get_type_symbol(elem_type)
|
elem_type_sym := t.get_type_symbol(elem_type)
|
||||||
return 'array_${elem_type_sym.name}' + if elem_type.is_ptr() {
|
return 'array_$elem_type_sym.name' + if elem_type.is_ptr() {
|
||||||
'_ptr'.repeat(elem_type.nr_muls())
|
'_ptr'.repeat(elem_type.nr_muls())
|
||||||
} else {
|
} else {
|
||||||
''
|
''
|
||||||
|
@ -310,7 +309,7 @@ pub fn (t &Table) array_name(elem_type Type, nr_dims int) string {
|
||||||
[inline]
|
[inline]
|
||||||
pub fn (t &Table) array_fixed_name(elem_type Type, size, nr_dims int) string {
|
pub fn (t &Table) array_fixed_name(elem_type Type, size, nr_dims int) string {
|
||||||
elem_type_sym := t.get_type_symbol(elem_type)
|
elem_type_sym := t.get_type_symbol(elem_type)
|
||||||
return 'array_fixed_${elem_type_sym.name}_${size}' + if elem_type.is_ptr() {
|
return 'array_fixed_${elem_type_sym.name}_$size' + if elem_type.is_ptr() {
|
||||||
'_ptr'
|
'_ptr'
|
||||||
} else {
|
} else {
|
||||||
''
|
''
|
||||||
|
@ -326,7 +325,7 @@ pub fn (t &Table) map_name(key_type, value_type Type) string {
|
||||||
key_type_sym := t.get_type_symbol(key_type)
|
key_type_sym := t.get_type_symbol(key_type)
|
||||||
value_type_sym := t.get_type_symbol(value_type)
|
value_type_sym := t.get_type_symbol(value_type)
|
||||||
suffix := if value_type.is_ptr() { '_ptr' } else { '' }
|
suffix := if value_type.is_ptr() { '_ptr' } else { '' }
|
||||||
return 'map_${key_type_sym.name}_${value_type_sym.name}' + suffix
|
return 'map_${key_type_sym.name}_$value_type_sym.name' + suffix
|
||||||
// return 'map_${value_type_sym.name}' + suffix
|
// return 'map_${value_type_sym.name}' + suffix
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -489,7 +488,7 @@ pub fn (t &Table) mktyp(typ Type) Type {
|
||||||
// this is not optimal
|
// this is not optimal
|
||||||
pub fn (table &Table) qualify_module(mod, file_path string) string {
|
pub fn (table &Table) qualify_module(mod, file_path string) string {
|
||||||
for m in table.imports {
|
for m in table.imports {
|
||||||
//if m.contains('gen') { println('qm=$m') }
|
// if m.contains('gen') { println('qm=$m') }
|
||||||
if m.contains('.') && m.contains(mod) {
|
if m.contains('.') && m.contains(mod) {
|
||||||
m_parts := m.split('.')
|
m_parts := m.split('.')
|
||||||
m_path := m_parts.join(os.path_separator)
|
m_path := m_parts.join(os.path_separator)
|
||||||
|
@ -512,12 +511,11 @@ pub fn (mut table Table) register_fn_gen_type(fn_name string, typ Type) {
|
||||||
table.fn_gen_types[fn_name] = a
|
table.fn_gen_types[fn_name] = a
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// TODO: there is a bug when casting sumtype the other way if its pointer
|
// TODO: there is a bug when casting sumtype the other way if its pointer
|
||||||
// so until fixed at least show v (not C) error `x(variant) = y(SumType*)`
|
// so until fixed at least show v (not C) error `x(variant) = y(SumType*)`
|
||||||
pub fn (table &Table) sumtype_has_variant(parent Type, variant Type) bool {
|
pub fn (table &Table) sumtype_has_variant(parent, variant Type) bool {
|
||||||
parent_sym := table.get_type_symbol(parent)
|
parent_sym := table.get_type_symbol(parent)
|
||||||
if parent_sym.kind ==.sum_type {
|
if parent_sym.kind == .sum_type {
|
||||||
parent_info := parent_sym.info as SumType
|
parent_info := parent_sym.info as SumType
|
||||||
for v in parent_info.variants {
|
for v in parent_info.variants {
|
||||||
if v.idx() == variant.idx() {
|
if v.idx() == variant.idx() {
|
||||||
|
@ -529,4 +527,5 @@ pub fn (table &Table) sumtype_has_variant(parent Type, variant Type) bool {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,7 +4,7 @@ const (
|
||||||
my_random_letter_const = byte(65 + rand.u32n(25))
|
my_random_letter_const = byte(65 + rand.u32n(25))
|
||||||
)
|
)
|
||||||
|
|
||||||
fn test_rand_is_initialized_before_main(){
|
fn test_rand_is_initialized_before_main() {
|
||||||
eprintln('random letter: $my_random_letter_const.str() | ASCII code: $my_random_letter_const')
|
eprintln('random letter: $my_random_letter_const.str() | ASCII code: $my_random_letter_const')
|
||||||
assert my_random_letter_const.is_capital()
|
assert my_random_letter_const.is_capital()
|
||||||
}
|
}
|
||||||
|
|
|
@ -68,7 +68,6 @@ fn mut_arg<T>(mut x T) {
|
||||||
println(x.name) // = 'foo'
|
println(x.name) // = 'foo'
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
fn mut_arg2<T>(mut x T) T {
|
fn mut_arg2<T>(mut x T) T {
|
||||||
println(x.name) // = 'foo'
|
println(x.name) // = 'foo'
|
||||||
return x
|
return x
|
||||||
|
@ -183,7 +182,6 @@ fn test_generic_fn_in_for_in_expression() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
*/
|
*/
|
||||||
|
|
||||||
// test generic struct
|
// test generic struct
|
||||||
struct DB {
|
struct DB {
|
||||||
driver string
|
driver string
|
||||||
|
@ -200,7 +198,7 @@ pub mut:
|
||||||
name string
|
name string
|
||||||
}
|
}
|
||||||
|
|
||||||
struct Repo<T,U> {
|
struct Repo<T, U> {
|
||||||
db DB
|
db DB
|
||||||
pub mut:
|
pub mut:
|
||||||
model T
|
model T
|
||||||
|
@ -211,30 +209,29 @@ pub mut:
|
||||||
// fn new_repo<T>(db DB) Repo<T,U> {
|
// fn new_repo<T>(db DB) Repo<T,U> {
|
||||||
// return Repo<T,Permission>{db: db}
|
// return Repo<T,Permission>{db: db}
|
||||||
// }
|
// }
|
||||||
|
|
||||||
|
|
||||||
fn test_generic_struct() {
|
fn test_generic_struct() {
|
||||||
mut a := Repo<User,Permission>{
|
mut a := Repo<User, Permission>{
|
||||||
model: User{name: 'joe'}
|
model: User{
|
||||||
|
name: 'joe'
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// a.model.name = 'joe'
|
// a.model.name = 'joe'
|
||||||
assert a.model.name == 'joe'
|
assert a.model.name == 'joe'
|
||||||
println('a.model.name: $a.model.name')
|
println('a.model.name: $a.model.name')
|
||||||
|
mut b := Repo<Group, Permission>{
|
||||||
mut b := Repo<Group,Permission>{
|
permission: Permission{
|
||||||
permission: Permission{name: 'superuser'}
|
name: 'superuser'
|
||||||
|
}
|
||||||
}
|
}
|
||||||
b.model.name = 'admins'
|
b.model.name = 'admins'
|
||||||
assert b.model.name == 'admins'
|
assert b.model.name == 'admins'
|
||||||
assert b.permission.name == 'superuser'
|
assert b.permission.name == 'superuser'
|
||||||
println('b.model.name: $b.model.name')
|
println('b.model.name: $b.model.name')
|
||||||
println('b.permission.name: $b.permission.name')
|
println('b.permission.name: $b.permission.name')
|
||||||
|
|
||||||
assert typeof(a.model) == 'User'
|
assert typeof(a.model) == 'User'
|
||||||
assert typeof(b.model) == 'Group'
|
assert typeof(b.model) == 'Group'
|
||||||
println('typeof(a.model): ' + typeof(a.model))
|
println('typeof(a.model): ' + typeof(a.model))
|
||||||
println('typeof(b.model): ' + typeof(b.model))
|
println('typeof(b.model): ' + typeof(b.model))
|
||||||
|
|
||||||
// mut x := new_repo<User>(DB{})
|
// mut x := new_repo<User>(DB{})
|
||||||
// x.model.name = 'joe2'
|
// x.model.name = 'joe2'
|
||||||
// println(x.model.name)
|
// println(x.model.name)
|
||||||
|
|
|
@ -217,5 +217,5 @@ fn test_optimized_in_expression_with_string() {
|
||||||
|
|
||||||
fn test_in_array_init() {
|
fn test_in_array_init() {
|
||||||
assert 1 !in []int{}
|
assert 1 !in []int{}
|
||||||
assert [1] in [[1]]
|
assert [1] in [[1], [2]]
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,7 +6,9 @@ fn test_all() {
|
||||||
mut total_errors := 0
|
mut total_errors := 0
|
||||||
vexe := os.getenv('VEXE')
|
vexe := os.getenv('VEXE')
|
||||||
vroot := os.dir(vexe)
|
vroot := os.dir(vexe)
|
||||||
diff_cmd := util.find_working_diff_command() or { '' }
|
diff_cmd := util.find_working_diff_command() or {
|
||||||
|
''
|
||||||
|
}
|
||||||
dir := 'vlib/v/tests/inout'
|
dir := 'vlib/v/tests/inout'
|
||||||
files := os.ls(dir) or {
|
files := os.ls(dir) or {
|
||||||
panic(err)
|
panic(err)
|
||||||
|
@ -70,8 +72,8 @@ fn test_all() {
|
||||||
expected = expected.trim_right('\r\n').replace('\r\n', '\n')
|
expected = expected.trim_right('\r\n').replace('\r\n', '\n')
|
||||||
if expected.contains('================ V panic ================') {
|
if expected.contains('================ V panic ================') {
|
||||||
// panic include backtraces and absolute file paths, so can't do char by char comparison
|
// panic include backtraces and absolute file paths, so can't do char by char comparison
|
||||||
n_found := normalize_panic_message( found, vroot )
|
n_found := normalize_panic_message(found, vroot)
|
||||||
n_expected := normalize_panic_message( expected, vroot )
|
n_expected := normalize_panic_message(expected, vroot)
|
||||||
if found.contains('================ V panic ================') {
|
if found.contains('================ V panic ================') {
|
||||||
if n_found.contains(n_expected) {
|
if n_found.contains(n_expected) {
|
||||||
println(term.green('OK (panic)'))
|
println(term.green('OK (panic)'))
|
||||||
|
@ -87,12 +89,12 @@ fn test_all() {
|
||||||
}
|
}
|
||||||
if expected != found {
|
if expected != found {
|
||||||
println(term.red('FAIL'))
|
println(term.red('FAIL'))
|
||||||
println(term.header('expected:','-'))
|
println(term.header('expected:', '-'))
|
||||||
println(expected)
|
println(expected)
|
||||||
println(term.header('found:','-'))
|
println(term.header('found:', '-'))
|
||||||
println(found)
|
println(found)
|
||||||
if diff_cmd != '' {
|
if diff_cmd != '' {
|
||||||
println(term.header('difference:','-'))
|
println(term.header('difference:', '-'))
|
||||||
println(util.color_compare_strings(diff_cmd, expected, found))
|
println(util.color_compare_strings(diff_cmd, expected, found))
|
||||||
} else {
|
} else {
|
||||||
println(term.h_divider('-'))
|
println(term.h_divider('-'))
|
||||||
|
@ -105,7 +107,7 @@ fn test_all() {
|
||||||
assert total_errors == 0
|
assert total_errors == 0
|
||||||
}
|
}
|
||||||
|
|
||||||
fn normalize_panic_message(message string, vroot string) string {
|
fn normalize_panic_message(message, vroot string) string {
|
||||||
mut msg := message.all_before('=========================================')
|
mut msg := message.all_before('=========================================')
|
||||||
msg = msg.replace(vroot + os.path_separator, '')
|
msg = msg.replace(vroot + os.path_separator, '')
|
||||||
msg = msg.trim_space()
|
msg = msg.trim_space()
|
||||||
|
|
|
@ -1,5 +1,3 @@
|
||||||
import os
|
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
println('hello world')
|
println('hello world')
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,3 @@
|
||||||
import os
|
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
areas := ['game', 'web', 'tools', 'science', 'systems', 'embedded']
|
areas := ['game', 'web', 'tools', 'science', 'systems', 'embedded']
|
||||||
for i :=0; i < areas.len; i++{
|
for i :=0; i < areas.len; i++{
|
||||||
|
|
|
@ -1,4 +1,3 @@
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
test := 'hello'
|
test := 'hello'
|
||||||
hello := 'world'
|
hello := 'world'
|
||||||
|
|
|
@ -27,7 +27,6 @@ not very flaky way.
|
||||||
|
|
||||||
TODO: Cleanup this when/if v has better process control/communication primitives.
|
TODO: Cleanup this when/if v has better process control/communication primitives.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const (
|
const (
|
||||||
vexe = os.getenv('VEXE')
|
vexe = os.getenv('VEXE')
|
||||||
tmp_file = os.join_path(os.temp_dir(), 'generated_live_program.tmp.v')
|
tmp_file = os.join_path(os.temp_dir(), 'generated_live_program.tmp.v')
|
||||||
|
@ -98,12 +97,12 @@ fn main() {
|
||||||
"
|
"
|
||||||
)
|
)
|
||||||
|
|
||||||
fn atomic_write_source( source string ){
|
fn atomic_write_source(source string) {
|
||||||
// NB: here wrtiting is done in 2 steps, since os.write_file can take some time,
|
// NB: here wrtiting is done in 2 steps, since os.write_file can take some time,
|
||||||
// during which the file will be modified, but it will still be not completely written.
|
// during which the file will be modified, but it will still be not completely written.
|
||||||
// The os.mv after that, guarantees that the reloader will see a complete valid V program.
|
// The os.mv after that, guarantees that the reloader will see a complete valid V program.
|
||||||
os.write_file(tmp_file, source)
|
os.write_file(tmp_file, source)
|
||||||
os.mv(tmp_file, source_file )
|
os.mv(tmp_file, source_file)
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
|
@ -114,13 +113,15 @@ fn testsuite_begin() {
|
||||||
eprintln('You can still do it by setting FORCE_LIVE_TEST=1 .')
|
eprintln('You can still do it by setting FORCE_LIVE_TEST=1 .')
|
||||||
exit(0)
|
exit(0)
|
||||||
}
|
}
|
||||||
for f in [ tmp_file, source_file, output_file, res_original_file, res_changed_file, res_another_file, res_stop_file] {
|
for f in [tmp_file, source_file, output_file, res_original_file, res_changed_file, res_another_file,
|
||||||
|
res_stop_file,
|
||||||
|
] {
|
||||||
os.rm(f)
|
os.rm(f)
|
||||||
}
|
}
|
||||||
atomic_write_source( live_program_source )
|
atomic_write_source(live_program_source)
|
||||||
}
|
}
|
||||||
|
|
||||||
[if debuglivetest]
|
[debuglivetest]
|
||||||
fn vprintln(s string) {
|
fn vprintln(s string) {
|
||||||
eprintln(s)
|
eprintln(s)
|
||||||
}
|
}
|
||||||
|
@ -132,36 +133,36 @@ fn testsuite_end() {
|
||||||
output_lines := os.read_lines(output_file) or {
|
output_lines := os.read_lines(output_file) or {
|
||||||
panic('could not read $output_file, error: $err')
|
panic('could not read $output_file, error: $err')
|
||||||
}
|
}
|
||||||
mut histogram := map[string]int
|
mut histogram := map[string]int{}
|
||||||
for line in output_lines {
|
for line in output_lines {
|
||||||
histogram[line] = histogram[line] + 1
|
histogram[line] = histogram[line] + 1
|
||||||
}
|
}
|
||||||
for k, v in histogram {
|
for k, v in histogram {
|
||||||
eprintln('> found ${v:5d} times: ${k}')
|
eprintln('> found ${v:5d} times: $k')
|
||||||
}
|
}
|
||||||
vprintln('---------------------------------------------------------------------------')
|
vprintln('---------------------------------------------------------------------------')
|
||||||
assert histogram['START'] > 0
|
assert histogram['START'] > 0
|
||||||
assert histogram['ORIGINAL'] > 0
|
assert histogram['ORIGINAL'] > 0
|
||||||
assert histogram['CHANGED'] + histogram['ANOTHER'] > 0
|
assert histogram['CHANGED'] + histogram['ANOTHER'] > 0
|
||||||
//assert histogram['END'] > 0
|
// assert histogram['END'] > 0
|
||||||
}
|
}
|
||||||
|
|
||||||
fn change_source(new string) {
|
fn change_source(new string) {
|
||||||
time.sleep_ms(100)
|
time.sleep_ms(100)
|
||||||
vprintln('> change ORIGINAL to: $new')
|
vprintln('> change ORIGINAL to: $new')
|
||||||
atomic_write_source( live_program_source.replace('ORIGINAL', new) )
|
atomic_write_source(live_program_source.replace('ORIGINAL', new))
|
||||||
wait_for_file(new)
|
wait_for_file(new)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn wait_for_file(new string){
|
fn wait_for_file(new string) {
|
||||||
time.sleep_ms(100)
|
time.sleep_ms(100)
|
||||||
expected_file := os.join_path(os.temp_dir(), new + '.txt')
|
expected_file := os.join_path(os.temp_dir(), new + '.txt')
|
||||||
eprintln('waiting for $expected_file ...')
|
eprintln('waiting for $expected_file ...')
|
||||||
for i:=0 ; i <= 400 ; i++ {
|
for i := 0; i <= 400; i++ {
|
||||||
if i % 25 == 0 {
|
if i % 25 == 0 {
|
||||||
vprintln(' checking ${i:-10d} for $expected_file ...')
|
vprintln(' checking ${i:-10d} for $expected_file ...')
|
||||||
}
|
}
|
||||||
if os.exists( expected_file ) {
|
if os.exists(expected_file) {
|
||||||
assert true
|
assert true
|
||||||
vprintln('> done.')
|
vprintln('> done.')
|
||||||
time.sleep_ms(100)
|
time.sleep_ms(100)
|
||||||
|
|
|
@ -12,13 +12,19 @@ fn test_match_integers() {
|
||||||
mut a := 3
|
mut a := 3
|
||||||
mut b := 0
|
mut b := 0
|
||||||
match a {
|
match a {
|
||||||
2 { println('two') }
|
2 {
|
||||||
|
println('two')
|
||||||
|
}
|
||||||
3 {
|
3 {
|
||||||
println('three')
|
println('three')
|
||||||
b = 3
|
b = 3
|
||||||
}
|
}
|
||||||
4 { println('four') }
|
4 {
|
||||||
else { println('???') }
|
println('four')
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
println('???')
|
||||||
|
}
|
||||||
}
|
}
|
||||||
assert b == 3
|
assert b == 3
|
||||||
assert match 2 {
|
assert match 2 {
|
||||||
|
@ -36,8 +42,12 @@ fn test_match_integers() {
|
||||||
} == 5
|
} == 5
|
||||||
a = 0
|
a = 0
|
||||||
match 2 {
|
match 2 {
|
||||||
0 { a = 1 }
|
0 {
|
||||||
1 { a = 2 }
|
a = 1
|
||||||
|
}
|
||||||
|
1 {
|
||||||
|
a = 2
|
||||||
|
}
|
||||||
else {
|
else {
|
||||||
a = 3
|
a = 3
|
||||||
println('a is $a')
|
println('a is $a')
|
||||||
|
@ -46,7 +56,9 @@ fn test_match_integers() {
|
||||||
assert a == 3
|
assert a == 3
|
||||||
a = 0
|
a = 0
|
||||||
match 1 {
|
match 1 {
|
||||||
0 { a = 1 }
|
0 {
|
||||||
|
a = 1
|
||||||
|
}
|
||||||
1 {
|
1 {
|
||||||
a = 2
|
a = 2
|
||||||
a = a + 2
|
a = a + 2
|
||||||
|
@ -83,18 +95,24 @@ fn test_match_range() {
|
||||||
fn test_match_enums() {
|
fn test_match_enums() {
|
||||||
mut b := Color.red
|
mut b := Color.red
|
||||||
match b {
|
match b {
|
||||||
.red { b = .green }
|
.red {
|
||||||
.green { b = .blue }
|
b = .green
|
||||||
|
}
|
||||||
|
.green {
|
||||||
|
b = .blue
|
||||||
|
}
|
||||||
else {
|
else {
|
||||||
println('b is ${b.str()}')
|
println('b is $b.str()')
|
||||||
b = .red
|
b = .red
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
assert b == .green
|
assert b == .green
|
||||||
match b {
|
match b {
|
||||||
.red { b = .green }
|
.red {
|
||||||
|
b = .green
|
||||||
|
}
|
||||||
else {
|
else {
|
||||||
println('b is ${b.str()}')
|
println('b is $b.str()')
|
||||||
b = .blue
|
b = .blue
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -106,23 +124,22 @@ type Sum = A1 | B1
|
||||||
struct A1 {
|
struct A1 {
|
||||||
pos int
|
pos int
|
||||||
}
|
}
|
||||||
|
|
||||||
struct B1 {
|
struct B1 {
|
||||||
val string
|
val string
|
||||||
}
|
}
|
||||||
|
|
||||||
fn f(s Sum) string {
|
fn f(s Sum) string {
|
||||||
match s {
|
match s {
|
||||||
A1 {
|
A1 { return typeof(s) }
|
||||||
return typeof(s)
|
B1 { return '' }
|
||||||
}
|
|
||||||
B1 {
|
|
||||||
return ''
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return ''
|
return ''
|
||||||
}
|
}
|
||||||
|
|
||||||
fn test_sum_type_name() {
|
fn test_sum_type_name() {
|
||||||
a := A1{pos: 22}
|
a := A1{
|
||||||
|
pos: 22
|
||||||
|
}
|
||||||
assert f(a) == 'A1'
|
assert f(a) == 'A1'
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,7 +2,6 @@ module amodule
|
||||||
|
|
||||||
// This tests whether _test.v files can be *internal* to a
|
// This tests whether _test.v files can be *internal* to a
|
||||||
// module, and thus have access to its guts.
|
// module, and thus have access to its guts.
|
||||||
|
|
||||||
// NB: the function test_private_isub() is defined both here
|
// NB: the function test_private_isub() is defined both here
|
||||||
// and inside internal_module_test.v . That is done on purpose,
|
// and inside internal_module_test.v . That is done on purpose,
|
||||||
// with the goal of ensuring that _test.v files are compiled
|
// with the goal of ensuring that _test.v files are compiled
|
||||||
|
@ -10,7 +9,6 @@ module amodule
|
||||||
//
|
//
|
||||||
// _test.v files should *only* import all the other normal .v
|
// _test.v files should *only* import all the other normal .v
|
||||||
// files from the same folder, NOT other _test.v files from it.
|
// files from the same folder, NOT other _test.v files from it.
|
||||||
|
fn test_private_isub() {
|
||||||
fn test_private_isub(){
|
assert private_isub(7, 5) == 2
|
||||||
assert private_isub(7,5) == 2
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,15 +2,14 @@ module amodule
|
||||||
|
|
||||||
// this tests whether _test.v files can be *internal*
|
// this tests whether _test.v files can be *internal*
|
||||||
// to a module, and thus have access to its guts.
|
// to a module, and thus have access to its guts.
|
||||||
|
fn test_iadd() {
|
||||||
fn test_iadd(){
|
|
||||||
assert iadd(10, 20) == 30
|
assert iadd(10, 20) == 30
|
||||||
}
|
}
|
||||||
|
|
||||||
fn test_imul(){
|
fn test_imul() {
|
||||||
assert imul(5,8) == 40
|
assert imul(5, 8) == 40
|
||||||
}
|
}
|
||||||
|
|
||||||
fn test_private_isub(){
|
fn test_private_isub() {
|
||||||
assert private_isub(10,6) == 4
|
assert private_isub(10, 6) == 4
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,15 +1,14 @@
|
||||||
module amodule
|
module amodule
|
||||||
|
|
||||||
pub fn iadd(x int, y int) int {
|
pub fn iadd(x, y int) int {
|
||||||
return x + y
|
return x + y
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn imul(x int, y int) int {
|
pub fn imul(x, y int) int {
|
||||||
return x * y
|
return x * y
|
||||||
}
|
}
|
||||||
|
|
||||||
///////////////////////////////////////
|
// /////////////////////////////////////
|
||||||
|
fn private_isub(x, y int) int {
|
||||||
fn private_isub(x int, y int) int {
|
|
||||||
return x - y
|
return x - y
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,7 +8,10 @@ fn new_st() MyStruct {
|
||||||
|
|
||||||
fn get_st() MyStruct {
|
fn get_st() MyStruct {
|
||||||
r := new_st()
|
r := new_st()
|
||||||
return {r|s:'6'}
|
return {
|
||||||
|
r |
|
||||||
|
s: '6'
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
|
|
|
@ -2,7 +2,7 @@ module main
|
||||||
|
|
||||||
import mod1
|
import mod1
|
||||||
|
|
||||||
fn main(){
|
fn main() {
|
||||||
res := mod1.vadd(1,2)
|
res := mod1.vadd(1, 2)
|
||||||
println( res )
|
println(res)
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import mod1.submodule as m
|
import mod1.submodule as m
|
||||||
|
|
||||||
fn test_mod1_can_still_be_found_through_parent_project_vmod(){
|
fn test_mod1_can_still_be_found_through_parent_project_vmod() {
|
||||||
assert 1051 == m.f()
|
assert 1051 == m.f()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,11 +1,10 @@
|
||||||
import mod1
|
import mod1
|
||||||
|
|
||||||
import mod1.submodule
|
import mod1.submodule
|
||||||
|
|
||||||
fn test_mod1(){
|
fn test_mod1() {
|
||||||
assert 1 == mod1.f()
|
assert 1 == mod1.f()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn test_mod1_submodule_can_find_and_use_all_its_sibling_submodules(){
|
fn test_mod1_submodule_can_find_and_use_all_its_sibling_submodules() {
|
||||||
assert 1051 == submodule.f()
|
assert 1051 == submodule.f()
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,9 +1,8 @@
|
||||||
|
fn iadd(x, y int) int {
|
||||||
fn iadd(x int, y int) int {
|
|
||||||
return x + y
|
return x + y
|
||||||
}
|
}
|
||||||
|
|
||||||
fn main(){
|
fn main() {
|
||||||
println('Hello world')
|
println('Hello world')
|
||||||
println('iadd: ' + iadd(1,2).str())
|
println('iadd: ' + iadd(1, 2).str())
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
module main
|
module main
|
||||||
fn test_iadd_3_4(){
|
|
||||||
a := iadd(3,4)
|
fn test_iadd_3_4() {
|
||||||
|
a := iadd(3, 4)
|
||||||
assert a == 7
|
assert a == 7
|
||||||
assert iadd(10,20) == 30
|
assert iadd(10, 20) == 30
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,9 +1,11 @@
|
||||||
module main
|
module main
|
||||||
fn test_iadd_3_4(){
|
|
||||||
a := iadd(3,4)
|
fn test_iadd_3_4() {
|
||||||
|
a := iadd(3, 4)
|
||||||
assert a == 7
|
assert a == 7
|
||||||
}
|
}
|
||||||
fn test_iadd_5_6(){
|
|
||||||
a := iadd(5,6)
|
fn test_iadd_5_6() {
|
||||||
|
a := iadd(5, 6)
|
||||||
assert a == 11
|
assert a == 11
|
||||||
}
|
}
|
||||||
|
|
|
@ -34,47 +34,41 @@ pub fn full_path_to_v(dirs_in int) string {
|
||||||
return vexec
|
return vexec
|
||||||
}
|
}
|
||||||
|
|
||||||
fn diff_files( file_result, file_expected string ) string {
|
fn diff_files(file_result, file_expected string) string {
|
||||||
diffcmd := util.find_working_diff_command() or { return err }
|
diffcmd := util.find_working_diff_command() or {
|
||||||
|
return err
|
||||||
|
}
|
||||||
return util.color_compare_files(diffcmd, file_result, file_expected)
|
return util.color_compare_files(diffcmd, file_result, file_expected)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn run_repl_file(wd string, vexec string, file string) ?string {
|
pub fn run_repl_file(wd, vexec, file string) ?string {
|
||||||
vexec_folder := os.dir(vexec) + os.path_separator
|
vexec_folder := os.dir(vexec) + os.path_separator
|
||||||
fcontent := os.read_file(file) or { return error('Could not read file ${file}') }
|
fcontent := os.read_file(file) or {
|
||||||
|
return error('Could not read file $file')
|
||||||
|
}
|
||||||
content := fcontent.replace('\r', '')
|
content := fcontent.replace('\r', '')
|
||||||
input := content.all_before('===output===\n')
|
input := content.all_before('===output===\n')
|
||||||
output := content.all_after('===output===\n').trim_right('\n\r')
|
output := content.all_after('===output===\n').trim_right('\n\r')
|
||||||
|
fname := os.file_name(file)
|
||||||
fname := os.file_name( file )
|
input_temporary_filename := os.real_path(os.join_path(wd, 'input_temporary_filename.txt'))
|
||||||
|
|
||||||
input_temporary_filename := os.real_path(os.join_path( wd, 'input_temporary_filename.txt'))
|
|
||||||
os.write_file(input_temporary_filename, input)
|
os.write_file(input_temporary_filename, input)
|
||||||
os.write_file( os.real_path(os.join_path( wd, 'original.txt' ) ), fcontent )
|
os.write_file(os.real_path(os.join_path(wd, 'original.txt')), fcontent)
|
||||||
rcmd := '"$vexec" repl -replfolder "$wd" -replprefix "${fname}." < $input_temporary_filename'
|
rcmd := '"$vexec" repl -replfolder "$wd" -replprefix "${fname}." < $input_temporary_filename'
|
||||||
r := os.exec(rcmd) or {
|
r := os.exec(rcmd) or {
|
||||||
os.rm(input_temporary_filename)
|
os.rm(input_temporary_filename)
|
||||||
return error('Could not execute: $rcmd')
|
return error('Could not execute: $rcmd')
|
||||||
}
|
}
|
||||||
os.rm(input_temporary_filename)
|
os.rm(input_temporary_filename)
|
||||||
|
result := r.output.replace('\r', '').replace('>>> ', '').replace('>>>', '').replace('... ',
|
||||||
result := r.output.replace('\r','')
|
'').all_after('Use Ctrl-C or `exit` to exit\n').replace(wd + os.path_separator, '').replace(vexec_folder,
|
||||||
.replace('>>> ', '')
|
'').replace('\\', '/').trim_right('\n\r')
|
||||||
.replace('>>>', '')
|
|
||||||
.replace('... ', '')
|
|
||||||
.all_after('Use Ctrl-C or `exit` to exit\n')
|
|
||||||
.replace(wd + os.path_separator, '' )
|
|
||||||
.replace(vexec_folder, '')
|
|
||||||
.replace('\\', '/')
|
|
||||||
.trim_right('\n\r')
|
|
||||||
|
|
||||||
if result != output {
|
if result != output {
|
||||||
file_result := '${file}.result.txt'
|
file_result := '${file}.result.txt'
|
||||||
file_expected := '${file}.expected.txt'
|
file_expected := '${file}.expected.txt'
|
||||||
os.write_file( file_result, result )
|
os.write_file(file_result, result)
|
||||||
os.write_file( file_expected, output )
|
os.write_file(file_expected, output)
|
||||||
diff := diff_files( file_result, file_expected )
|
diff := diff_files(file_result, file_expected)
|
||||||
return error('Difference found in REPL file: ${file}
|
return error('Difference found in REPL file: $file
|
||||||
====> Got :
|
====> Got :
|
||||||
|$result|
|
|$result|
|
||||||
====> Expected :
|
====> Expected :
|
||||||
|
@ -83,31 +77,29 @@ pub fn run_repl_file(wd string, vexec string, file string) ?string {
|
||||||
$diff
|
$diff
|
||||||
')
|
')
|
||||||
} else {
|
} else {
|
||||||
return 'Repl file ${file} is OK'
|
return 'Repl file $file is OK'
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn run_prod_file(wd string, vexec string, file string) ?string {
|
pub fn run_prod_file(wd, vexec, file string) ?string {
|
||||||
file_expected := '${file}.expected.txt'
|
file_expected := '${file}.expected.txt'
|
||||||
f_expected_content := os.read_file(file_expected) or { return error('Could not read file ${file}') }
|
f_expected_content := os.read_file(file_expected) or {
|
||||||
|
return error('Could not read file $file')
|
||||||
|
}
|
||||||
expected_content := f_expected_content.replace('\r', '')
|
expected_content := f_expected_content.replace('\r', '')
|
||||||
|
cmd := '"$vexec" -prod run "$file"'
|
||||||
cmd := '"$vexec" -prod run "${file}"'
|
|
||||||
r := os.exec(cmd) or {
|
r := os.exec(cmd) or {
|
||||||
return error('Could not execute: $cmd')
|
return error('Could not execute: $cmd')
|
||||||
}
|
}
|
||||||
|
|
||||||
if r.exit_code != 0 {
|
if r.exit_code != 0 {
|
||||||
return error('$cmd return exit code: $r.exit_code')
|
return error('$cmd return exit code: $r.exit_code')
|
||||||
}
|
}
|
||||||
|
result := r.output.replace('\r', '')
|
||||||
result := r.output.replace('\r','')
|
|
||||||
|
|
||||||
if result != expected_content {
|
if result != expected_content {
|
||||||
file_result := '${file}.result.txt'
|
file_result := '${file}.result.txt'
|
||||||
os.write_file( file_result, result )
|
os.write_file(file_result, result)
|
||||||
diff := diff_files( file_result, file_expected )
|
diff := diff_files(file_result, file_expected)
|
||||||
return error('Difference found in test: ${file}
|
return error('Difference found in test: $file
|
||||||
====> Got :
|
====> Got :
|
||||||
|$result|
|
|$result|
|
||||||
====> Expected :
|
====> Expected :
|
||||||
|
@ -116,7 +108,7 @@ pub fn run_prod_file(wd string, vexec string, file string) ?string {
|
||||||
$diff
|
$diff
|
||||||
')
|
')
|
||||||
} else {
|
} else {
|
||||||
return 'Prod file ${file} is OK'
|
return 'Prod file $file is OK'
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -127,11 +119,11 @@ pub fn new_options() RunnerOptions {
|
||||||
if os.args.len > 1 {
|
if os.args.len > 1 {
|
||||||
files = os.args[1..]
|
files = os.args[1..]
|
||||||
} else {
|
} else {
|
||||||
os.chdir( os.dir(vexec) )
|
os.chdir(os.dir(vexec))
|
||||||
wd = os.getwd()
|
wd = os.getwd()
|
||||||
files = os.walk_ext('.', '.repl')
|
files = os.walk_ext('.', '.repl')
|
||||||
}
|
}
|
||||||
return RunnerOptions {
|
return RunnerOptions{
|
||||||
wd: wd
|
wd: wd
|
||||||
vexec: vexec
|
vexec: vexec
|
||||||
files: files
|
files: files
|
||||||
|
@ -147,7 +139,7 @@ pub fn new_prod_options() RunnerOptions {
|
||||||
} else {
|
} else {
|
||||||
files = os.walk_ext(wd, '.prod.v')
|
files = os.walk_ext(wd, '.prod.v')
|
||||||
}
|
}
|
||||||
return RunnerOptions {
|
return RunnerOptions{
|
||||||
wd: wd
|
wd: wd
|
||||||
vexec: vexec
|
vexec: vexec
|
||||||
files: files
|
files: files
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
|
struct Zest {
|
||||||
struct Zest { val int }
|
val int
|
||||||
|
}
|
||||||
|
|
||||||
fn (t Zest) get_a_finger_to_the_moon() voidptr {
|
fn (t Zest) get_a_finger_to_the_moon() voidptr {
|
||||||
return voidptr(0)
|
return voidptr(0)
|
||||||
|
@ -10,7 +11,9 @@ fn get_the_dao_way() voidptr {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn test_returning_a_void_pointer_from_a_method() {
|
fn test_returning_a_void_pointer_from_a_method() {
|
||||||
t := &Zest{ val: 123 }
|
t := &Zest{
|
||||||
|
val: 123
|
||||||
|
}
|
||||||
z := voidptr(0)
|
z := voidptr(0)
|
||||||
assert z == t.get_a_finger_to_the_moon()
|
assert z == t.get_a_finger_to_the_moon()
|
||||||
assert t.get_a_finger_to_the_moon() == 0
|
assert t.get_a_finger_to_the_moon() == 0
|
||||||
|
|
|
@ -2,7 +2,7 @@ module main
|
||||||
|
|
||||||
struct Anything {
|
struct Anything {
|
||||||
mut:
|
mut:
|
||||||
name string = ""
|
name string = ''
|
||||||
keepo int = 0
|
keepo int = 0
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -12,80 +12,102 @@ fn (a Anything) str() string {
|
||||||
|
|
||||||
fn test_array_of_ptrs_to_structs_can_be_printed() {
|
fn test_array_of_ptrs_to_structs_can_be_printed() {
|
||||||
mut testing := []&Anything{}
|
mut testing := []&Anything{}
|
||||||
testing << &Anything{name: "Hehe"}
|
testing << &Anything{
|
||||||
testing << &Anything{name: "other"}
|
name: 'Hehe'
|
||||||
testing << &Anything{name: "test"}
|
}
|
||||||
|
testing << &Anything{
|
||||||
|
name: 'other'
|
||||||
|
}
|
||||||
|
testing << &Anything{
|
||||||
|
name: 'test'
|
||||||
|
}
|
||||||
for test in testing {
|
for test in testing {
|
||||||
println(test)
|
println(test)
|
||||||
assert true
|
assert true
|
||||||
}
|
}
|
||||||
println('testing: $testing')
|
println('testing: $testing')
|
||||||
println( testing )
|
println(testing)
|
||||||
assert true
|
assert true
|
||||||
}
|
}
|
||||||
|
|
||||||
// At the same time, this should also work:
|
// At the same time, this should also work:
|
||||||
// (note the str method defined on (a &T), instead on (a T))
|
// (note the str method defined on (a &T), instead on (a T))
|
||||||
|
|
||||||
struct PstrAnything {
|
struct PstrAnything {
|
||||||
mut:
|
mut:
|
||||||
name string = ""
|
name string = ''
|
||||||
keepo int = 0
|
keepo int = 0
|
||||||
}
|
}
|
||||||
|
|
||||||
fn (a &PstrAnything) str() string {
|
fn (a &PstrAnything) str() string {
|
||||||
return a.name
|
return a.name
|
||||||
}
|
}
|
||||||
|
|
||||||
fn test_array_of_ptrs_to_structs_can_be_printed_when_structs_have_str_with_ptr() {
|
fn test_array_of_ptrs_to_structs_can_be_printed_when_structs_have_str_with_ptr() {
|
||||||
mut testing := []&PstrAnything{}
|
mut testing := []&PstrAnything{}
|
||||||
testing << &PstrAnything{name: "abc"}
|
testing << &PstrAnything{
|
||||||
testing << &PstrAnything{name: "def"}
|
name: 'abc'
|
||||||
testing << &PstrAnything{name: "ghi"}
|
}
|
||||||
|
testing << &PstrAnything{
|
||||||
|
name: 'def'
|
||||||
|
}
|
||||||
|
testing << &PstrAnything{
|
||||||
|
name: 'ghi'
|
||||||
|
}
|
||||||
for test in testing {
|
for test in testing {
|
||||||
println(test)
|
println(test)
|
||||||
assert true
|
assert true
|
||||||
}
|
}
|
||||||
println('testing: $testing')
|
println('testing: $testing')
|
||||||
println( testing )
|
println(testing)
|
||||||
assert true
|
assert true
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
|
|
||||||
fn test_stack_array_of_structs_can_be_printed_when_structs_have_ordinary_str() {
|
fn test_stack_array_of_structs_can_be_printed_when_structs_have_ordinary_str() {
|
||||||
mut t := [3]Anything
|
mut t := [3]Anything
|
||||||
t[0] = Anything{name: "012"}
|
t[0] = Anything{
|
||||||
t[1] = Anything{name: "345"}
|
name: '012'
|
||||||
t[2] = Anything{name: "678"}
|
}
|
||||||
|
t[1] = Anything{
|
||||||
|
name: '345'
|
||||||
|
}
|
||||||
|
t[2] = Anything{
|
||||||
|
name: '678'
|
||||||
|
}
|
||||||
for test in t {
|
for test in t {
|
||||||
println(test)
|
println(test)
|
||||||
assert true
|
assert true
|
||||||
}
|
}
|
||||||
println('t: $t')
|
println('t: $t')
|
||||||
println( t )
|
println(t)
|
||||||
println( 't[0] := ${t[0]}')
|
println('t[0] := ${t[0]}')
|
||||||
assert true
|
assert true
|
||||||
}
|
}
|
||||||
|
|
||||||
fn test_stack_array_of_structs_can_be_printed_when_structs_have_str_with_ptr() {
|
fn test_stack_array_of_structs_can_be_printed_when_structs_have_str_with_ptr() {
|
||||||
// this generates a C error
|
// this generates a C error
|
||||||
mut pt := [3]PstrAnything
|
mut pt := [3]PstrAnything
|
||||||
pt[0] = PstrAnything{name: "P012"}
|
pt[0] = PstrAnything{
|
||||||
pt[1] = PstrAnything{name: "P345"}
|
name: 'P012'
|
||||||
pt[2] = PstrAnything{name: "P678"}
|
}
|
||||||
|
pt[1] = PstrAnything{
|
||||||
|
name: 'P345'
|
||||||
|
}
|
||||||
|
pt[2] = PstrAnything{
|
||||||
|
name: 'P678'
|
||||||
|
}
|
||||||
for test in pt {
|
for test in pt {
|
||||||
println(test)
|
println(test)
|
||||||
assert true
|
assert true
|
||||||
}
|
}
|
||||||
println('pt: $pt')
|
println('pt: $pt')
|
||||||
println( pt )
|
println(pt)
|
||||||
print( 'pt[0] := ')
|
print('pt[0] := ')
|
||||||
print( pt[0] )
|
print(pt[0])
|
||||||
println('')
|
println('')
|
||||||
assert true
|
assert true
|
||||||
|
|
||||||
$if debug_buggy_println ? {
|
$if debug_buggy_println ? {
|
||||||
//TODO: fix string interpolation for structs with `fn (t &T) str() string` too:
|
// TODO: fix string interpolation for structs with `fn (t &T) str() string` too:
|
||||||
println( 'pt[0] := ${pt[0]}')
|
println('pt[0] := ${pt[0]}')
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,11 +1,21 @@
|
||||||
fn test_ptr_assign() {
|
fn test_ptr_assign() {
|
||||||
v := [int(5), 6, 77, 1]
|
v := [int(5), 6, 77, 1]
|
||||||
mut p := &v[0]
|
mut p := &v[0]
|
||||||
unsafe { (*p)++ }
|
unsafe {
|
||||||
unsafe { p++ } // p now points to v[1]
|
(*p)++
|
||||||
unsafe { (*p) += 2 }
|
}
|
||||||
unsafe { p += 2 } // p now points to v[3]
|
unsafe {
|
||||||
unsafe { *p = 31 }
|
p++
|
||||||
|
} // p now points to v[1]
|
||||||
|
unsafe {
|
||||||
|
(*p) += 2
|
||||||
|
}
|
||||||
|
unsafe {
|
||||||
|
p += 2
|
||||||
|
} // p now points to v[3]
|
||||||
|
unsafe {
|
||||||
|
*p = 31
|
||||||
|
}
|
||||||
assert v[0] == 6
|
assert v[0] == 6
|
||||||
assert v[1] == 8
|
assert v[1] == 8
|
||||||
assert v[2] == 77
|
assert v[2] == 77
|
||||||
|
@ -14,18 +24,28 @@ fn test_ptr_assign() {
|
||||||
|
|
||||||
fn test_ptr_infix() {
|
fn test_ptr_infix() {
|
||||||
v := 4
|
v := 4
|
||||||
mut q := unsafe{ &v - 1 }
|
mut q := unsafe {
|
||||||
q = unsafe {q + 3}
|
&v - 1
|
||||||
|
}
|
||||||
|
|
||||||
|
q = unsafe {
|
||||||
|
q + 3
|
||||||
|
}
|
||||||
|
|
||||||
_ := q
|
_ := q
|
||||||
_ := v
|
_ := v
|
||||||
}
|
}
|
||||||
|
|
||||||
struct S1 {}
|
struct S1 {
|
||||||
|
}
|
||||||
|
|
||||||
[unsafe_fn]
|
[unsafe_fn]
|
||||||
fn (s S1) f(){}
|
fn (s S1) f() {
|
||||||
|
}
|
||||||
|
|
||||||
fn test_funcs() {
|
fn test_funcs() {
|
||||||
s := S1{}
|
s := S1{}
|
||||||
unsafe { s.f() }
|
unsafe {
|
||||||
|
s.f()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -168,11 +168,7 @@ fn get_array_content(tokens []Token, st_idx int) ?([]string, int) {
|
||||||
if tokens[idx + 1].typ !in [.comma, .rabr] {
|
if tokens[idx + 1].typ !in [.comma, .rabr] {
|
||||||
return error('vmod: invalid separator "${tokens[idx+1].val}"')
|
return error('vmod: invalid separator "${tokens[idx+1].val}"')
|
||||||
}
|
}
|
||||||
idx += if tokens[idx + 1].typ == .comma {
|
idx += if tokens[idx + 1].typ == .comma { 2 } else { 1 }
|
||||||
2
|
|
||||||
} else {
|
|
||||||
1
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
.rabr {
|
.rabr {
|
||||||
idx++
|
idx++
|
||||||
|
|
|
@ -19,14 +19,11 @@ import os
|
||||||
// => ModFileAndFolder{'vlib/v.mod', 'vlib'}
|
// => ModFileAndFolder{'vlib/v.mod', 'vlib'}
|
||||||
// ModFileCacher.get('vlib/v/test/project_with_c_code/mod1')
|
// ModFileCacher.get('vlib/v/test/project_with_c_code/mod1')
|
||||||
// => ModFileAndFolder{'vlib/v/test/project_with_c_code/mod1/v.mod', 'vlib/v/test/project_with_c_code/mod1'}
|
// => ModFileAndFolder{'vlib/v/test/project_with_c_code/mod1/v.mod', 'vlib/v/test/project_with_c_code/mod1'}
|
||||||
|
|
||||||
|
|
||||||
pub struct ModFileAndFolder {
|
pub struct ModFileAndFolder {
|
||||||
pub:
|
pub:
|
||||||
// vmod_file contains the full path of the found 'v.mod' file, or ''
|
// vmod_file contains the full path of the found 'v.mod' file, or ''
|
||||||
// if no 'v.mod' file was found in file_path_dir, or in its parent folders.
|
// if no 'v.mod' file was found in file_path_dir, or in its parent folders.
|
||||||
vmod_file string
|
vmod_file string
|
||||||
|
|
||||||
// vmod_folder contains the file_path_dir, if there is no 'v.mod' file in
|
// vmod_folder contains the file_path_dir, if there is no 'v.mod' file in
|
||||||
// *any* of the parent folders, otherwise it is the first parent folder,
|
// *any* of the parent folders, otherwise it is the first parent folder,
|
||||||
// where a v.mod file was found.
|
// where a v.mod file was found.
|
||||||
|
@ -49,35 +46,34 @@ pub fn (mcache &ModFileCacher) dump() {
|
||||||
$if debug {
|
$if debug {
|
||||||
eprintln('ModFileCacher DUMP:')
|
eprintln('ModFileCacher DUMP:')
|
||||||
eprintln(' ModFileCacher.cache:')
|
eprintln(' ModFileCacher.cache:')
|
||||||
for k,v in mcache.cache {
|
for k, v in mcache.cache {
|
||||||
eprintln(' K: ${k:-32s} | V: "${v.vmod_file:32s}" | "${v.vmod_folder:32s}" ')
|
eprintln(' K: ${k:-32s} | V: "${v.vmod_file:32s}" | "${v.vmod_folder:32s}" ')
|
||||||
}
|
}
|
||||||
eprintln(' ModFileCacher.folder_files:')
|
eprintln(' ModFileCacher.folder_files:')
|
||||||
for k,v in mcache.folder_files {
|
for k, v in mcache.folder_files {
|
||||||
eprintln(' K: ${k:-32s} | V: ${v.str()}')
|
eprintln(' K: ${k:-32s} | V: $v.str()')
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
pub fn (mut mcache ModFileCacher) get_by_file(vfile string) ModFileAndFolder {
|
pub fn (mut mcache ModFileCacher) get_by_file(vfile string) ModFileAndFolder {
|
||||||
return mcache.get_by_folder( os.dir( vfile ) )
|
return mcache.get_by_folder(os.dir(vfile))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (mut mcache ModFileCacher) get_by_folder(vfolder string) ModFileAndFolder {
|
pub fn (mut mcache ModFileCacher) get_by_folder(vfolder string) ModFileAndFolder {
|
||||||
mfolder := os.real_path( vfolder )
|
mfolder := os.real_path(vfolder)
|
||||||
if mfolder in mcache.cache {
|
if mfolder in mcache.cache {
|
||||||
return mcache.cache[ mfolder ]
|
return mcache.cache[mfolder]
|
||||||
}
|
}
|
||||||
traversed_folders, res := mcache.traverse( mfolder )
|
traversed_folders, res := mcache.traverse(mfolder)
|
||||||
for tfolder in traversed_folders {
|
for tfolder in traversed_folders {
|
||||||
mcache.add( tfolder, res )
|
mcache.add(tfolder, res)
|
||||||
}
|
}
|
||||||
return res
|
return res
|
||||||
}
|
}
|
||||||
|
|
||||||
fn (mut cacher ModFileCacher) add(path string, result ModFileAndFolder) {
|
fn (mut cacher ModFileCacher) add(path string, result ModFileAndFolder) {
|
||||||
cacher.cache[ path ] = result
|
cacher.cache[path] = result
|
||||||
}
|
}
|
||||||
|
|
||||||
fn (mut mcache ModFileCacher) traverse(mfolder string) ([]string, ModFileAndFolder) {
|
fn (mut mcache ModFileCacher) traverse(mfolder string) ([]string, ModFileAndFolder) {
|
||||||
|
@ -92,47 +88,59 @@ fn (mut mcache ModFileCacher) traverse(mfolder string) ([]string, ModFileAndFold
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
if cfolder in mcache.cache {
|
if cfolder in mcache.cache {
|
||||||
res := mcache.cache[ cfolder ]
|
res := mcache.cache[cfolder]
|
||||||
if res.vmod_file.len == 0 {
|
if res.vmod_file.len == 0 {
|
||||||
mcache.mark_folders_as_vmod_free( folders_so_far )
|
mcache.mark_folders_as_vmod_free(folders_so_far)
|
||||||
}else{
|
} else {
|
||||||
mcache.mark_folders_with_vmod( folders_so_far, res )
|
mcache.mark_folders_with_vmod(folders_so_far, res)
|
||||||
}
|
}
|
||||||
return []string{}, res
|
return []string{}, res
|
||||||
}
|
}
|
||||||
files := mcache.get_files( cfolder )
|
files := mcache.get_files(cfolder)
|
||||||
if 'v.mod' in files {
|
if 'v.mod' in files {
|
||||||
// TODO: actually read the v.mod file and parse its contents to see
|
// TODO: actually read the v.mod file and parse its contents to see
|
||||||
// if its source folder is different
|
// if its source folder is different
|
||||||
res := ModFileAndFolder{ vmod_file: os.join_path( cfolder, 'v.mod'), vmod_folder: cfolder }
|
res := ModFileAndFolder{
|
||||||
|
vmod_file: os.join_path(cfolder, 'v.mod')
|
||||||
|
vmod_folder: cfolder
|
||||||
|
}
|
||||||
return folders_so_far, res
|
return folders_so_far, res
|
||||||
}
|
}
|
||||||
if mcache.check_for_stop( cfolder, files ) {
|
if mcache.check_for_stop(cfolder, files) {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
cfolder = os.base_dir( cfolder )
|
cfolder = os.base_dir(cfolder)
|
||||||
folders_so_far << cfolder
|
folders_so_far << cfolder
|
||||||
levels++
|
levels++
|
||||||
}
|
}
|
||||||
mcache.mark_folders_as_vmod_free( folders_so_far )
|
mcache.mark_folders_as_vmod_free(folders_so_far)
|
||||||
return [mfolder], ModFileAndFolder{ vmod_file: '', vmod_folder: mfolder }
|
return [mfolder], ModFileAndFolder{
|
||||||
}
|
vmod_file: ''
|
||||||
|
vmod_folder: mfolder
|
||||||
fn (mut mcache ModFileCacher) mark_folders_with_vmod( folders_so_far []string, vmod ModFileAndFolder ) {
|
|
||||||
for f in folders_so_far {
|
|
||||||
mcache.add( f, vmod )
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn (mut mcache ModFileCacher) mark_folders_as_vmod_free( folders_so_far []string ) {
|
fn (mut mcache ModFileCacher) mark_folders_with_vmod(folders_so_far []string, vmod ModFileAndFolder) {
|
||||||
|
for f in folders_so_far {
|
||||||
|
mcache.add(f, vmod)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn (mut mcache ModFileCacher) mark_folders_as_vmod_free(folders_so_far []string) {
|
||||||
// No need to check these folders anymore,
|
// No need to check these folders anymore,
|
||||||
// because their parents do not contain v.mod files
|
// because their parents do not contain v.mod files
|
||||||
for f in folders_so_far {
|
for f in folders_so_far {
|
||||||
mcache.add( f, ModFileAndFolder{ vmod_file: '', vmod_folder: f } )
|
mcache.add(f, ModFileAndFolder{
|
||||||
|
vmod_file: ''
|
||||||
|
vmod_folder: f
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const ( mod_file_stop_paths = ['.git', '.hg', '.svn', '.v.mod.stop' ] )
|
const (
|
||||||
|
mod_file_stop_paths = ['.git', '.hg', '.svn', '.v.mod.stop']
|
||||||
|
)
|
||||||
|
|
||||||
fn (mcache &ModFileCacher) check_for_stop(cfolder string, files []string) bool {
|
fn (mcache &ModFileCacher) check_for_stop(cfolder string, files []string) bool {
|
||||||
for i in mod_file_stop_paths {
|
for i in mod_file_stop_paths {
|
||||||
if i in files {
|
if i in files {
|
||||||
|
@ -144,20 +152,23 @@ fn (mcache &ModFileCacher) check_for_stop(cfolder string, files []string) bool {
|
||||||
|
|
||||||
fn (mut mcache ModFileCacher) get_files(cfolder string) []string {
|
fn (mut mcache ModFileCacher) get_files(cfolder string) []string {
|
||||||
if cfolder in mcache.folder_files {
|
if cfolder in mcache.folder_files {
|
||||||
return mcache.folder_files[ cfolder ]
|
return mcache.folder_files[cfolder]
|
||||||
}
|
}
|
||||||
mut files := []string{}
|
mut files := []string{}
|
||||||
if os.exists( cfolder ) && os.is_dir(cfolder) {
|
if os.exists(cfolder) && os.is_dir(cfolder) {
|
||||||
if listing := os.ls(cfolder) {
|
if listing := os.ls(cfolder) {
|
||||||
files = listing
|
files = listing
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
mcache.folder_files[ cfolder ] = files
|
mcache.folder_files[cfolder] = files
|
||||||
return files
|
return files
|
||||||
}
|
}
|
||||||
|
|
||||||
// used during lookup for v.mod to support @VROOT
|
// used during lookup for v.mod to support @VROOT
|
||||||
const ( private_file_cacher = new_mod_file_cacher() )
|
const (
|
||||||
|
private_file_cacher = new_mod_file_cacher()
|
||||||
|
)
|
||||||
|
|
||||||
pub fn get_cache() &ModFileCacher {
|
pub fn get_cache() &ModFileCacher {
|
||||||
return private_file_cacher
|
return private_file_cacher
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue