Implemented boolean operators
This commit is contained in:
parent
ee16774fe7
commit
b5bb291e47
@ -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)
|
||||||
|
@ -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
|
||||||
|
@ -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)
|
||||||
}
|
}
|
||||||
|
@ -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.
|
||||||
|
37
tests/programs/branch-and.q
Normal file
37
tests/programs/branch-and.q
Normal 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)
|
||||||
|
}
|
57
tests/programs/branch-combined.q
Normal file
57
tests/programs/branch-combined.q
Normal 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
|
||||||
|
}
|
29
tests/programs/branch-or.q
Normal file
29
tests/programs/branch-or.q
Normal 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)
|
||||||
|
}
|
@ -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)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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) {
|
||||||
|
Loading…
Reference in New Issue
Block a user