2019-01-11 13:27:28 +03:00
|
|
|
package runtime_test
|
2019-01-11 09:34:28 +03:00
|
|
|
|
|
|
|
import "testing"
|
|
|
|
|
|
|
|
func TestTailCall(t *testing.T) {
|
|
|
|
expect(t, `
|
2019-01-11 11:53:04 +03:00
|
|
|
fac := func(n, a) {
|
|
|
|
if n == 1 {
|
|
|
|
return a
|
|
|
|
}
|
|
|
|
return fac(n-1, n*a)
|
2019-01-11 09:34:28 +03:00
|
|
|
}
|
2019-03-18 18:15:26 +03:00
|
|
|
out = fac(5, 1)`, nil, 120)
|
2019-01-11 09:34:28 +03:00
|
|
|
|
|
|
|
expect(t, `
|
2019-01-11 11:53:04 +03:00
|
|
|
fac := func(n, a) {
|
|
|
|
if n == 1 {
|
|
|
|
return a
|
|
|
|
}
|
|
|
|
x := {foo: fac} // indirection for test
|
|
|
|
return x.foo(n-1, n*a)
|
2019-01-11 09:34:28 +03:00
|
|
|
}
|
2019-03-18 18:15:26 +03:00
|
|
|
out = fac(5, 1)`, nil, 120)
|
2019-01-11 09:34:28 +03:00
|
|
|
|
|
|
|
expect(t, `
|
2019-01-11 11:53:04 +03:00
|
|
|
fib := func(x, s) {
|
|
|
|
if x == 0 {
|
|
|
|
return 0 + s
|
|
|
|
} else if x == 1 {
|
|
|
|
return 1 + s
|
|
|
|
}
|
|
|
|
return fib(x-1, fib(x-2, s))
|
2019-01-11 09:34:28 +03:00
|
|
|
}
|
2019-03-18 18:15:26 +03:00
|
|
|
out = fib(15, 0)`, nil, 610)
|
2019-01-11 09:34:28 +03:00
|
|
|
|
|
|
|
expect(t, `
|
2019-01-11 11:53:04 +03:00
|
|
|
fib := func(n, a, b) {
|
|
|
|
if n == 0 {
|
|
|
|
return a
|
|
|
|
} else if n == 1 {
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
return fib(n-1, b, a + b)
|
2019-01-11 09:34:28 +03:00
|
|
|
}
|
2019-03-18 18:15:26 +03:00
|
|
|
out = fib(15, 0, 1)`, nil, 610)
|
2019-01-11 09:34:28 +03:00
|
|
|
|
2019-01-11 11:53:04 +03:00
|
|
|
// global variable and no return value
|
2019-01-11 09:34:28 +03:00
|
|
|
expect(t, `
|
2019-01-11 11:53:04 +03:00
|
|
|
out = 0
|
|
|
|
foo := func(a) {
|
|
|
|
if a == 0 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
out += a
|
|
|
|
foo(a-1)
|
|
|
|
}
|
2019-03-18 18:15:26 +03:00
|
|
|
foo(10)`, nil, 55)
|
2019-01-11 09:34:28 +03:00
|
|
|
|
2019-01-11 11:53:04 +03:00
|
|
|
expect(t, `
|
|
|
|
f1 := func() {
|
|
|
|
f2 := 0 // TODO: this might be fixed in the future
|
|
|
|
f2 = func(n, s) {
|
|
|
|
if n == 0 { return s }
|
|
|
|
return f2(n-1, n + s)
|
|
|
|
}
|
|
|
|
return f2(5, 0)
|
|
|
|
}
|
2019-03-18 18:15:26 +03:00
|
|
|
out = f1()`, nil, 15)
|
2019-01-13 13:24:32 +03:00
|
|
|
|
|
|
|
// tail-call replacing loop
|
|
|
|
// without tail-call optimization, this code will cause stack overflow
|
|
|
|
expect(t, `
|
|
|
|
iter := func(n, max) {
|
|
|
|
if n == max {
|
|
|
|
return n
|
|
|
|
}
|
|
|
|
|
|
|
|
return iter(n+1, max)
|
|
|
|
}
|
|
|
|
out = iter(0, 9999)
|
2019-03-18 18:15:26 +03:00
|
|
|
`, nil, 9999)
|
2019-02-21 03:26:11 +03:00
|
|
|
expect(t, `
|
|
|
|
c := 0
|
|
|
|
iter := func(n, max) {
|
|
|
|
if n == max {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
c++
|
|
|
|
iter(n+1, max)
|
|
|
|
}
|
|
|
|
iter(0, 9999)
|
|
|
|
out = c
|
2019-03-18 18:15:26 +03:00
|
|
|
`, nil, 9999)
|
2019-01-11 09:34:28 +03:00
|
|
|
}
|
|
|
|
|
2019-01-11 11:53:04 +03:00
|
|
|
// tail call with free vars
|
|
|
|
func TestTailCallFreeVars(t *testing.T) {
|
|
|
|
expect(t, `
|
|
|
|
func() {
|
|
|
|
a := 10
|
|
|
|
f2 := 0
|
|
|
|
f2 = func(n, s) {
|
|
|
|
if n == 0 {
|
|
|
|
return s + a
|
|
|
|
}
|
|
|
|
return f2(n-1, n+s)
|
|
|
|
}
|
|
|
|
out = f2(5, 0)
|
2019-03-18 18:15:26 +03:00
|
|
|
}()`, nil, 25)
|
2019-01-11 09:34:28 +03:00
|
|
|
}
|