package core import ( "git.urbach.dev/cli/q/src/asm" "git.urbach.dev/cli/q/src/config" "git.urbach.dev/cli/q/src/cpu" "git.urbach.dev/cli/q/src/errors" "git.urbach.dev/cli/q/src/token" "git.urbach.dev/cli/q/src/x86" ) // ExecuteRegisterRegister performs an operation on two registers. func (f *Function) ExecuteRegisterRegister(operation token.Token, register cpu.Register, operand cpu.Register) error { if !operation.IsAssignment() && !operation.IsComparison() { f.SaveRegister(register) } switch operation.Kind { case token.Add, token.AddAssign: f.RegisterRegister(asm.ADD, register, operand) case token.Sub, token.SubAssign: f.RegisterRegister(asm.SUB, register, operand) case token.Mul, token.MulAssign: f.RegisterRegister(asm.MUL, register, operand) case token.Div, token.DivAssign: if config.TargetArch == config.X86 { f.SaveRegister(x86.RDX) } f.RegisterRegister(asm.DIV, register, operand) case token.Mod, token.ModAssign: if config.TargetArch == config.X86 { f.SaveRegister(x86.RDX) } f.RegisterRegister(asm.MODULO, register, operand) case token.And, token.AndAssign: f.RegisterRegister(asm.AND, register, operand) case token.Or, token.OrAssign: f.RegisterRegister(asm.OR, register, operand) case token.Xor, token.XorAssign: f.RegisterRegister(asm.XOR, register, operand) case token.Assign: f.RegisterRegister(asm.MOVE, register, operand) case token.Equal, token.NotEqual, token.Less, token.LessEqual, token.Greater, token.GreaterEqual: f.RegisterRegister(asm.COMPARE, register, operand) default: return errors.New(&errors.InvalidOperator{Operator: operation.Text(f.File.Bytes)}, f.File, operation.Position) } return nil }