Simplified CPU state

This commit is contained in:
2025-03-31 14:36:42 +02:00
parent 0bc52fb673
commit 008f097186
7 changed files with 23 additions and 40 deletions

View File

@ -37,12 +37,17 @@ func (f *Function) CompileAssign(node *ast.Assign) error {
switch leftValue := leftValue.(type) {
case *eval.Register:
// TODO: Reservation needs to be canceled on defer
f.CurrentScope().Reserve(leftValue.Register)
if !f.RegisterIsUsed(leftValue.Register) {
f.UseRegister(leftValue.Register)
defer f.FreeRegister(leftValue.Register)
}
f.Execute(operation, leftValue.Register, right)
case *eval.Memory:
// TODO: Reservation needs to be canceled on defer
f.CurrentScope().Reserve(leftValue.Memory.Base)
if !f.RegisterIsUsed(leftValue.Memory.Base) {
f.UseRegister(leftValue.Memory.Base)
defer f.FreeRegister(leftValue.Memory.Base)
}
if operation.Kind == token.Assign {
rightValue, err := f.Evaluate(right)

View File

@ -1,41 +1,27 @@
package cpu
// State contains information about which registers are currently in use.
type State struct {
Reserved uint64
Used uint64
}
type State uint64
// Free will reset the reserved and used status which means the register can be allocated again.
func (s *State) Free(reg Register) {
s.Reserved &= ^(1 << reg)
s.Used &= ^(1 << reg)
}
// IsReserved returns true if the register was marked for future use.
func (s *State) IsReserved(reg Register) bool {
return s.Reserved&(1<<reg) != 0
*s &= ^(1 << reg)
}
// IsUsed returns true if the register is currently in use and holds a value.
func (s *State) IsUsed(reg Register) bool {
return s.Used&(1<<reg) != 0
}
// Reserve reserves a register for future use.
func (s *State) Reserve(reg Register) {
s.Reserved |= (1 << reg)
return *s&(1<<reg) != 0
}
// Use marks a register to be currently in use which means it must be preserved across function calls.
func (s *State) Use(reg Register) {
s.Used |= (1 << reg)
*s |= (1 << reg)
}
// FindFree tries to find a free register in the given slice of registers.
func (s *State) FindFree(registers []Register) (Register, bool) {
for _, reg := range registers {
if !s.IsReserved(reg) && !s.IsUsed(reg) {
if !s.IsUsed(reg) {
return reg, true
}
}

View File

@ -8,23 +8,17 @@ import (
)
func TestRegisterState(t *testing.T) {
s := cpu.State{}
assert.False(t, s.IsReserved(0))
assert.False(t, s.IsUsed(0))
s.Reserve(0)
assert.True(t, s.IsReserved(0))
var s cpu.State
assert.False(t, s.IsUsed(0))
s.Use(0)
assert.True(t, s.IsReserved(0))
assert.True(t, s.IsUsed(0))
s.Free(0)
assert.False(t, s.IsReserved(0))
assert.False(t, s.IsUsed(0))
}
func TestFindFree(t *testing.T) {
s := cpu.State{}
s.Reserve(0)
var s cpu.State
s.Use(0)
s.Use(1)
reg, found := s.FindFree([]cpu.Register{0, 1, 2, 3})
@ -36,8 +30,8 @@ func TestFindFree(t *testing.T) {
}
func TestMustFindFree(t *testing.T) {
s := cpu.State{}
s.Reserve(0)
var s cpu.State
s.Use(0)
s.Use(1)
reg := s.MustFindFree([]cpu.Register{0, 1, 2, 3})

View File

@ -11,5 +11,5 @@ type Machine struct {
scope.Stack
Assembler asm.Assembler
CPU *cpu.CPU
RegisterHistory []uint64
RegisterHistory []cpu.State
}

View File

@ -5,6 +5,6 @@ import "git.urbach.dev/cli/q/src/cpu"
// NewRegister reserves a register.
func (f *Machine) NewRegister() cpu.Register {
register := f.CurrentScope().MustFindFree(f.CPU.General)
f.CurrentScope().Reserve(register)
f.CurrentScope().Use(register)
return register
}

View File

@ -6,8 +6,6 @@ func (f *Machine) Return() {
}
f.Assembler.Return()
scope := f.CurrentScope()
scope.Reserved = 0
scope.Used = 0
f.CurrentScope().State = 0
f.postInstruction()
}

View File

@ -7,5 +7,5 @@ func (f *Machine) postInstruction() {
return
}
f.RegisterHistory = append(f.RegisterHistory, f.CurrentScope().Used)
f.RegisterHistory = append(f.RegisterHistory, f.CurrentScope().State)
}