[VM] Fixed floating point arithmetic opcodes.

They were broken because they weren't actually using floats.
This commit is contained in:
jtuu 2019-10-17 22:11:59 +03:00
parent 2906c8b577
commit 48f525a3bd

View File

@ -308,74 +308,87 @@ export class VirtualMachine {
exec.push_arg(this.string_arg_store.address, Kind.String); exec.push_arg(this.string_arg_store.address, Kind.String);
} }
break; break;
// arithmetic operations // integer arithmetic operations
case OP_ADD.code: case OP_ADD.code:
case OP_FADD.code: this.do_integer_op_with_register(arg0, arg1, numeric_ops.add);
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.add);
break; break;
case OP_ADDI.code: case OP_ADDI.code:
case OP_FADDI.code: this.do_integer_op_with_literal(arg0, arg1, numeric_ops.add);
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.add);
break; break;
case OP_SUB.code: case OP_SUB.code:
case OP_FSUB.code: this.do_integer_op_with_register(arg0, arg1, numeric_ops.sub);
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.sub);
break; break;
case OP_SUBI.code: case OP_SUBI.code:
case OP_FSUBI.code: this.do_integer_op_with_literal(arg0, arg1, numeric_ops.sub);
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.sub);
break; break;
case OP_MUL.code: case OP_MUL.code:
case OP_FMUL.code: this.do_integer_op_with_register(arg0, arg1, numeric_ops.mul);
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.mul);
break; break;
case OP_MULI.code: case OP_MULI.code:
case OP_FMULI.code: this.do_integer_op_with_literal(arg0, arg1, numeric_ops.mul);
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.mul);
break; break;
case OP_DIV.code: case OP_DIV.code:
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.idiv); this.do_integer_op_with_register(arg0, arg1, numeric_ops.idiv);
break;
case OP_FDIV.code:
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.div);
break; break;
case OP_DIVI.code: case OP_DIVI.code:
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.idiv); this.do_integer_op_with_literal(arg0, arg1, numeric_ops.idiv);
break;
case OP_FDIVI.code:
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.div);
break; break;
case OP_MOD.code: case OP_MOD.code:
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.mod); this.do_integer_op_with_register(arg0, arg1, numeric_ops.mod);
break; break;
case OP_MODI.code: case OP_MODI.code:
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.mod); this.do_integer_op_with_literal(arg0, arg1, numeric_ops.mod);
break;
// float arithmetic operations
case OP_FADD.code:
this.do_float_op_with_register(arg0, arg1, numeric_ops.add);
break;
case OP_FADDI.code:
this.do_float_op_with_literal(arg0, arg1, numeric_ops.add);
break;
case OP_FSUB.code:
this.do_float_op_with_register(arg0, arg1, numeric_ops.sub);
break;
case OP_FSUBI.code:
this.do_float_op_with_literal(arg0, arg1, numeric_ops.sub);
break;
case OP_FMUL.code:
this.do_float_op_with_register(arg0, arg1, numeric_ops.mul);
break;
case OP_FMULI.code:
this.do_float_op_with_literal(arg0, arg1, numeric_ops.mul);
break;
case OP_FDIV.code:
this.do_float_op_with_register(arg0, arg1, numeric_ops.div);
break;
case OP_FDIVI.code:
this.do_float_op_with_literal(arg0, arg1, numeric_ops.div);
break; break;
// bit operations // bit operations
case OP_AND.code: case OP_AND.code:
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.and); this.do_integer_op_with_register(arg0, arg1, numeric_ops.and);
break; break;
case OP_ANDI.code: case OP_ANDI.code:
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.and); this.do_integer_op_with_literal(arg0, arg1, numeric_ops.and);
break; break;
case OP_OR.code: case OP_OR.code:
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.or); this.do_integer_op_with_register(arg0, arg1, numeric_ops.or);
break; break;
case OP_ORI.code: case OP_ORI.code:
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.or); this.do_integer_op_with_literal(arg0, arg1, numeric_ops.or);
break; break;
case OP_XOR.code: case OP_XOR.code:
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.xor); this.do_integer_op_with_register(arg0, arg1, numeric_ops.xor);
break; break;
case OP_XORI.code: case OP_XORI.code:
this.do_numeric_op_with_literal(arg0, arg1, numeric_ops.xor); this.do_integer_op_with_literal(arg0, arg1, numeric_ops.xor);
break; break;
// shift operations // shift operations
case OP_SHIFT_LEFT.code: case OP_SHIFT_LEFT.code:
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.shl); this.do_integer_op_with_register(arg0, arg1, numeric_ops.shl);
break; break;
case OP_SHIFT_RIGHT.code: case OP_SHIFT_RIGHT.code:
this.do_numeric_op_with_register(arg0, arg1, numeric_ops.shr); this.do_integer_op_with_register(arg0, arg1, numeric_ops.shr);
break; break;
// conditional jumps // conditional jumps
case OP_JMP_ON.code: case OP_JMP_ON.code:
@ -623,15 +636,15 @@ export class VirtualMachine {
this.registers.write_f32_at(REGISTER_SIZE * reg, value); this.registers.write_f32_at(REGISTER_SIZE * reg, value);
} }
private do_numeric_op_with_register( private do_integer_op_with_register(
reg1: number, reg1: number,
reg2: number, reg2: number,
op: BinaryNumericOperation, op: BinaryNumericOperation,
): void { ): void {
this.do_numeric_op_with_literal(reg1, this.get_register_signed(reg2), op); this.do_integer_op_with_literal(reg1, this.get_register_signed(reg2), op);
} }
private do_numeric_op_with_literal( private do_integer_op_with_literal(
reg: number, reg: number,
literal: number, literal: number,
op: BinaryNumericOperation, op: BinaryNumericOperation,
@ -642,6 +655,25 @@ export class VirtualMachine {
this.set_register_signed(reg, op(this.get_register_signed(reg), literal)); this.set_register_signed(reg, op(this.get_register_signed(reg), literal));
} }
private do_float_op_with_register(
reg1: number,
reg2: number,
op: BinaryNumericOperation,
): void {
this.do_float_op_with_literal(reg1, this.get_register_float(reg2), op);
}
private do_float_op_with_literal(
reg: number,
literal: number,
op: BinaryNumericOperation,
): void {
if ((op === numeric_ops.div || op === numeric_ops.idiv) && literal === 0) {
throw new Error("Division by zero");
}
this.set_register_float(reg, op(this.get_register_float(reg), literal));
}
private push_call_stack(exec: Thread, label: number): void { private push_call_stack(exec: Thread, label: number): void {
const seg_idx = this.label_to_seg_idx.get(label); const seg_idx = this.label_to_seg_idx.get(label);