Improved division split
This commit is contained in:
parent
bb74c0cf50
commit
c2c147f1b4
@ -15,7 +15,7 @@ func Count(body AST, buffer []byte, kind token.Kind, name string) uint8 {
|
|||||||
count += node.Expression.Count(buffer, kind, name)
|
count += node.Expression.Count(buffer, kind, name)
|
||||||
|
|
||||||
case *Define:
|
case *Define:
|
||||||
count += node.Value.Count(buffer, kind, name)
|
count += node.Expression.Count(buffer, kind, name)
|
||||||
|
|
||||||
case *Return:
|
case *Return:
|
||||||
if node.Value != nil {
|
if node.Value != nil {
|
||||||
|
@ -2,11 +2,9 @@ package ast
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"git.akyoto.dev/cli/q/src/build/expression"
|
"git.akyoto.dev/cli/q/src/build/expression"
|
||||||
"git.akyoto.dev/cli/q/src/build/token"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// Define represents a variable definition.
|
// Define represents a variable definition.
|
||||||
type Define struct {
|
type Define struct {
|
||||||
Value *expression.Expression
|
Expression *expression.Expression
|
||||||
Name token.Token
|
|
||||||
}
|
}
|
||||||
|
@ -83,9 +83,7 @@ func toASTNode(tokens token.List, buffer []byte) (Node, error) {
|
|||||||
return nil, errors.New(errors.MissingOperand, nil, expr.Token.End())
|
return nil, errors.New(errors.MissingOperand, nil, expr.Token.End())
|
||||||
}
|
}
|
||||||
|
|
||||||
name := expr.Children[0].Token
|
return &Define{Expression: expr}, nil
|
||||||
value := expr.Children[1]
|
|
||||||
return &Define{Name: name, Value: value}, nil
|
|
||||||
|
|
||||||
case IsAssignment(expr):
|
case IsAssignment(expr):
|
||||||
if len(expr.Children) < 2 {
|
if len(expr.Children) < 2 {
|
||||||
|
@ -29,11 +29,14 @@ func (f *Function) CompileAssignDivision(node *ast.Assign) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
dividend := right.Children[0]
|
dividend := right.Children[0]
|
||||||
name = dividend.Token.Text(f.File.Bytes)
|
dividendRegister, err := f.Evaluate(dividend)
|
||||||
dividendVariable := f.VariableByName(name)
|
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
divisor := right.Children[1]
|
divisor := right.Children[1]
|
||||||
err := f.Execute(right.Token, dividendVariable.Register, divisor)
|
err = f.Execute(right.Token, dividendRegister, divisor)
|
||||||
f.RegisterRegister(asm.MOVE, quotientVariable.Register, x64.RAX)
|
f.RegisterRegister(asm.MOVE, quotientVariable.Register, x64.RAX)
|
||||||
f.RegisterRegister(asm.MOVE, remainderVariable.Register, x64.RDX)
|
f.RegisterRegister(asm.MOVE, remainderVariable.Register, x64.RDX)
|
||||||
return err
|
return err
|
||||||
|
@ -9,20 +9,22 @@ import (
|
|||||||
|
|
||||||
// CompileDefinition compiles a variable definition.
|
// CompileDefinition compiles a variable definition.
|
||||||
func (f *Function) CompileDefinition(node *ast.Define) error {
|
func (f *Function) CompileDefinition(node *ast.Define) error {
|
||||||
name := node.Name.Text(f.File.Bytes)
|
left := node.Expression.Children[0]
|
||||||
|
right := node.Expression.Children[1]
|
||||||
|
name := left.Token.Text(f.File.Bytes)
|
||||||
|
|
||||||
if f.IdentifierExists(name) {
|
if f.IdentifierExists(name) {
|
||||||
return errors.New(&errors.VariableAlreadyExists{Name: name}, f.File, node.Name.Position)
|
return errors.New(&errors.VariableAlreadyExists{Name: name}, f.File, left.Token.Position)
|
||||||
}
|
}
|
||||||
|
|
||||||
uses := token.Count(f.Body, f.File.Bytes, token.Identifier, name) - 1
|
uses := token.Count(f.Body, f.File.Bytes, token.Identifier, name) - 1
|
||||||
|
|
||||||
if uses == 0 {
|
if uses == 0 {
|
||||||
return errors.New(&errors.UnusedVariable{Name: name}, f.File, node.Name.Position)
|
return errors.New(&errors.UnusedVariable{Name: name}, f.File, left.Token.Position)
|
||||||
}
|
}
|
||||||
|
|
||||||
register := f.NewRegister()
|
register := f.NewRegister()
|
||||||
err := f.ExpressionToRegister(node.Value, register)
|
err := f.ExpressionToRegister(right, register)
|
||||||
|
|
||||||
f.AddVariable(&scope.Variable{
|
f.AddVariable(&scope.Variable{
|
||||||
Name: name,
|
Name: name,
|
||||||
|
24
src/build/core/Evaluate.go
Normal file
24
src/build/core/Evaluate.go
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
package core
|
||||||
|
|
||||||
|
import (
|
||||||
|
"git.akyoto.dev/cli/q/src/build/cpu"
|
||||||
|
"git.akyoto.dev/cli/q/src/build/expression"
|
||||||
|
"git.akyoto.dev/cli/q/src/build/token"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Evaluate evaluates an expression and returns a register that contains the value of the expression.
|
||||||
|
func (f *Function) Evaluate(expr *expression.Expression) (cpu.Register, error) {
|
||||||
|
if expr.Token.Kind == token.Identifier {
|
||||||
|
name := expr.Token.Text(f.File.Bytes)
|
||||||
|
variable := f.VariableByName(name)
|
||||||
|
|
||||||
|
if variable.Alive == 1 {
|
||||||
|
f.UseVariable(variable)
|
||||||
|
return variable.Register, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
tmp := f.NewRegister()
|
||||||
|
err := f.ExpressionToRegister(expr, tmp)
|
||||||
|
return tmp, err
|
||||||
|
}
|
22
tests/programs/division-split.q
Normal file
22
tests/programs/division-split.q
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
main() {
|
||||||
|
quotient := 0
|
||||||
|
remainder := 0
|
||||||
|
dividend := 256
|
||||||
|
divisor := 100
|
||||||
|
|
||||||
|
quotient, remainder = 256 / 100
|
||||||
|
assert quotient == 2
|
||||||
|
assert remainder == 56
|
||||||
|
|
||||||
|
quotient, remainder = dividend / 100
|
||||||
|
assert quotient == 2
|
||||||
|
assert remainder == 56
|
||||||
|
|
||||||
|
quotient, remainder = 256 / divisor
|
||||||
|
assert quotient == 2
|
||||||
|
assert remainder == 56
|
||||||
|
|
||||||
|
quotient, remainder = dividend / divisor
|
||||||
|
assert quotient == 2
|
||||||
|
assert remainder == 56
|
||||||
|
}
|
8
tests/programs/modulo-assign.q
Normal file
8
tests/programs/modulo-assign.q
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
main() {
|
||||||
|
x := 256
|
||||||
|
x %= 100
|
||||||
|
assert x == 56
|
||||||
|
|
||||||
|
x %= 10
|
||||||
|
assert x == 6
|
||||||
|
}
|
9
tests/programs/modulo.q
Normal file
9
tests/programs/modulo.q
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
main() {
|
||||||
|
assert 0 % 1 == 0
|
||||||
|
assert 1 % 1 == 0
|
||||||
|
assert 2 % 1 == 0
|
||||||
|
assert 0 % 2 == 0
|
||||||
|
assert 1 % 2 == 1
|
||||||
|
assert 2 % 2 == 0
|
||||||
|
assert 3 % 2 == 1
|
||||||
|
}
|
@ -1,46 +0,0 @@
|
|||||||
import sys
|
|
||||||
|
|
||||||
main() {
|
|
||||||
if 0 % 1 != 0 {
|
|
||||||
sys.exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
if 1 % 1 != 0 {
|
|
||||||
sys.exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
if 2 % 1 != 0 {
|
|
||||||
sys.exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
if 0 % 2 != 0 {
|
|
||||||
sys.exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
if 1 % 2 != 1 {
|
|
||||||
sys.exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
if 2 % 2 != 0 {
|
|
||||||
sys.exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
if 3 % 2 != 1 {
|
|
||||||
sys.exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
x := 256
|
|
||||||
x %= 100
|
|
||||||
|
|
||||||
if x != 56 {
|
|
||||||
sys.exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
x %= 10
|
|
||||||
|
|
||||||
if x != 6 {
|
|
||||||
sys.exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
sys.exit(0)
|
|
||||||
}
|
|
@ -33,14 +33,16 @@ var programs = []struct {
|
|||||||
{"branch-and", "", "", 0},
|
{"branch-and", "", "", 0},
|
||||||
{"branch-or", "", "", 0},
|
{"branch-or", "", "", 0},
|
||||||
{"branch-both", "", "", 0},
|
{"branch-both", "", "", 0},
|
||||||
|
{"jump-near", "", "", 0},
|
||||||
|
{"loop", "", "", 0},
|
||||||
|
{"loop-lifetime", "", "", 0},
|
||||||
{"bitwise-and", "", "", 0},
|
{"bitwise-and", "", "", 0},
|
||||||
{"bitwise-or", "", "", 0},
|
{"bitwise-or", "", "", 0},
|
||||||
{"bitwise-xor", "", "", 0},
|
{"bitwise-xor", "", "", 0},
|
||||||
{"shift", "", "", 0},
|
{"shift", "", "", 0},
|
||||||
{"remainder", "", "", 0},
|
{"modulo", "", "", 0},
|
||||||
{"jump-near", "", "", 0},
|
{"modulo-assign", "", "", 0},
|
||||||
{"loop", "", "", 0},
|
{"division-split", "", "", 0},
|
||||||
{"loop-lifetime", "", "", 0},
|
|
||||||
{"assert", "", "", 1},
|
{"assert", "", "", 1},
|
||||||
{"negative", "", "", 32},
|
{"negative", "", "", 32},
|
||||||
{"negation", "", "", 32},
|
{"negation", "", "", 32},
|
||||||
|
Loading…
Reference in New Issue
Block a user