Implemented boolean operators

This commit is contained in:
Eduard Urbach 2024-07-08 13:38:44 +02:00
parent ee16774fe7
commit b5bb291e47
Signed by: akyoto
GPG Key ID: C874F672B1AF20C0
9 changed files with 198 additions and 24 deletions

View File

@ -1,8 +1,8 @@
main() { main() {
x := f(1) + f(2) + f(3) x := f(1) + f(2) + f(3)
if x != f(8) || x != 9 || x == 6 { if x != 9 {
exit(42) exit(1)
} }
exit(0) exit(0)

View File

@ -1,44 +1,74 @@
package core package core
import ( import (
"fmt"
"git.akyoto.dev/cli/q/src/build/asm" "git.akyoto.dev/cli/q/src/build/asm"
"git.akyoto.dev/cli/q/src/build/expression" "git.akyoto.dev/cli/q/src/build/expression"
) )
// CompileCondition inserts code to jump to the start label or end label depending on the truth of the condition. // CompileCondition inserts code to jump to the start label or end label depending on the truth of the condition.
func (f *Function) CompileCondition(condition *expression.Expression, startLabel string, endLabel string) error { func (f *Function) CompileCondition(condition *expression.Expression, successLabel string, failLabel string) error {
switch condition.Token.Text() { switch condition.Token.Text() {
case "||": case "||":
leftFailLabel := fmt.Sprintf("%s_false_%d", f.Name, f.count.subBranch)
f.count.subBranch++
// Left
left := condition.Children[0] left := condition.Children[0]
err := f.CompileCondition(left, startLabel, endLabel) err := f.CompileCondition(left, successLabel, leftFailLabel)
if err != nil { if err != nil {
return err return err
} }
f.JumpIfTrue(left.Token.Text(), startLabel) f.JumpIfTrue(left.Token.Text(), successLabel)
// Right
f.assembler.Label(asm.LABEL, leftFailLabel)
right := condition.Children[1] right := condition.Children[1]
err = f.CompileCondition(right, startLabel, endLabel) err = f.CompileCondition(right, successLabel, failLabel)
if condition.Parent != nil && condition.Parent.Token.Text() == "||" && condition != condition.Parent.LastChild() {
f.JumpIfTrue(right.Token.Text(), successLabel)
} else {
f.JumpIfFalse(right.Token.Text(), failLabel)
}
return err
case "&&":
leftSuccessLabel := fmt.Sprintf("%s_true_%d", f.Name, f.count.subBranch)
f.count.subBranch++
// Left
left := condition.Children[0]
err := f.CompileCondition(left, leftSuccessLabel, failLabel)
if err != nil { if err != nil {
return err return err
} }
if condition.Parent == nil { f.JumpIfFalse(left.Token.Text(), failLabel)
f.JumpIfFalse(right.Token.Text(), endLabel)
// Right
f.assembler.Label(asm.LABEL, leftSuccessLabel)
right := condition.Children[1]
err = f.CompileCondition(right, successLabel, failLabel)
if condition.Parent != nil && condition.Parent.Token.Text() == "||" && condition != condition.Parent.LastChild() {
f.JumpIfTrue(right.Token.Text(), successLabel)
} else { } else {
f.JumpIfTrue(right.Token.Text(), startLabel) f.JumpIfFalse(right.Token.Text(), failLabel)
} }
return nil return err
case "&&":
return nil
default: default:
err := f.Compare(condition) err := f.Compare(condition)
if condition.Parent == nil { if condition.Parent == nil {
f.JumpIfFalse(condition.Token.Text(), endLabel) f.JumpIfFalse(condition.Token.Text(), failLabel)
} }
return err return err

View File

@ -9,16 +9,16 @@ import (
// CompileIf compiles a branch instruction. // CompileIf compiles a branch instruction.
func (f *Function) CompileIf(branch *ast.If) error { func (f *Function) CompileIf(branch *ast.If) error {
startLabel := fmt.Sprintf("%s_if_start_%d", f.Name, f.count.branch) success := fmt.Sprintf("%s_if_%d_true", f.Name, f.count.branch)
endLabel := fmt.Sprintf("%s_if_end_%d", f.Name, f.count.branch) fail := fmt.Sprintf("%s_if_%d_false", f.Name, f.count.branch)
err := f.CompileCondition(branch.Condition, startLabel, endLabel) err := f.CompileCondition(branch.Condition, success, fail)
if err != nil { if err != nil {
return err return err
} }
f.assembler.Label(asm.LABEL, startLabel) f.assembler.Label(asm.LABEL, success)
defer f.assembler.Label(asm.LABEL, endLabel) defer f.assembler.Label(asm.LABEL, fail)
f.count.branch++ f.count.branch++
return f.CompileAST(branch.Body) return f.CompileAST(branch.Body)
} }

View File

@ -23,6 +23,7 @@ type state struct {
type counter struct { type counter struct {
loop int loop int
branch int branch int
subBranch int
} }
// PrintInstructions shows the assembly instructions. // PrintInstructions shows the assembly instructions.

View File

@ -0,0 +1,37 @@
main() {
x := 0
if x != x && x != x {
exit(1)
}
if x == x && x != x {
exit(1)
}
if x != x && x == x {
exit(1)
}
if x == x && x != x && x != x {
exit(1)
}
if x != x && x == x && x != x {
exit(1)
}
if x != x && x != x && x == x {
exit(1)
}
if x == x && x == x && x == x {
exit(0)
}
exit(1)
}
exit(x) {
syscall(60, x)
}

View File

@ -0,0 +1,57 @@
main() {
x := 0
if x == x && x != x || x != x && x != x {
exit(1)
}
if x != x && x == x || x != x && x != x {
exit(1)
}
if x != x && x != x || x == x && x != x {
exit(1)
}
if x != x && x != x || x != x && x == x {
exit(1)
}
if (x == x || x != x) && (x != x || x != x) {
exit(1)
}
if (x != x || x == x) && (x != x || x != x) {
exit(1)
}
if (x != x || x != x) && (x == x || x != x) {
exit(1)
}
if (x != x || x != x) && (x != x || x == x) {
exit(1)
}
if x != x || x != x || x != x || x == x || x != x {
if x + 1 == inc(x) && x - 1 == dec(x) && x == dec(inc(x)) {
if (x == x && x != x || x == x && x == x) && (x == x && x == x || x == x && x != x) {
exit(0)
}
}
}
exit(1)
}
exit(x) {
syscall(60, x)
}
inc(x) {
return x + 1
}
dec(x) {
return x - 1
}

View File

@ -0,0 +1,29 @@
main() {
x := 0
if x != x || x != x {
exit(1)
}
if x != x || x != x || x != x {
exit(1)
}
if x == x || x != x {
if x != x || x == x {
if x == x || x != x || x != x {
if x != x || x == x || x != x {
if x != x || x != x || x == x {
exit(0)
}
}
}
}
}
exit(1)
}
exit(x) {
syscall(60, x)
}

View File

@ -25,15 +25,15 @@ main() {
exit(1) exit(1)
} }
if x >= 1 || 1 <= x { if x >= 1 {
exit(1) exit(1)
} }
if x == inc(x) || x == dec(x) { if 1 <= x {
exit(1) exit(1)
} }
if inc(0) == x || dec(0) == x || inc(x) == dec(x) { if x + 1 != x + 1 {
exit(1) exit(1)
} }
@ -41,7 +41,23 @@ main() {
exit(1) exit(1)
} }
if x + 1 != x + 1 { if x - 1 != dec(x) {
exit(1)
}
if inc(x) != x + 1 {
exit(1)
}
if dec(x) != x - 1 {
exit(1)
}
if x != inc(dec(x)) {
exit(1)
}
if inc(dec(x)) != x {
exit(1) exit(1)
} }

View File

@ -23,6 +23,10 @@ var programs = []struct {
{"nested-calls.q", "", 4}, {"nested-calls.q", "", 4},
{"return.q", "", 6}, {"return.q", "", 6},
{"reassign.q", "", 2}, {"reassign.q", "", 2},
{"branch.q", "", 0},
{"branch-and.q", "", 0},
{"branch-or.q", "", 0},
{"branch-combined.q", "", 0},
} }
func TestPrograms(t *testing.T) { func TestPrograms(t *testing.T) {