/* JPC: An x86 PC Hardware Emulator for a pure Java Virtual Machine Copyright (C) 2012-2013 Ian Preston This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License version 2 as published by the Free Software Foundation. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. Details (including contact information) can be found at: jpc.sourceforge.net or the developer website sourceforge.net/projects/jpc/ End of licence header */ package org.jpc.emulator.execution.opcodes.vm; import org.jpc.emulator.execution.*; import org.jpc.emulator.execution.decoder.*; import org.jpc.emulator.processor.*; import org.jpc.emulator.processor.fpu64.*; import static org.jpc.emulator.processor.Processor.*; public class fprem1 extends Executable { public fprem1(int blockStart, int eip, int prefices, PeekableInputStream input) { super(blockStart, eip); } public Branch execute(Processor cpu) { double freg0 = cpu.fpu.ST(0); double freg1 = cpu.fpu.ST(1); int d = Math.getExponent(freg0) - Math.getExponent(freg1); if (d < 64) { // full remainder cpu.fpu.conditionCode &= ~4; // clear C2 double z = Math.IEEEremainder(freg0, freg1); // compute least significant bits -> C0 C3 C1 long i = (long)Math.rint(freg0 / freg1); cpu.fpu.conditionCode &= 4; if ((i & 1) != 0) cpu.fpu.conditionCode |= 2; if ((i & 2) != 0) cpu.fpu.conditionCode |= 8; if ((i & 4) != 0) cpu.fpu.conditionCode |= 1; cpu.fpu.setST(0, z); } else { // partial remainder cpu.fpu.conditionCode |= 4; // set C2 int n = 63; // implementation dependent in manual double f = Math.pow(2.0, (double)(d - n)); double z = (freg0 / freg1) / f; double qq = (z < 0) ? Math.ceil(z) : Math.floor(z); cpu.fpu.setST(0, freg0 - (freg1 * qq * f)); } return Branch.None; } public boolean isBranch() { return false; } public String toString() { return this.getClass().getName(); } }