From patchwork Sat Jun 23 02:22:58 2018 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Programmingkid X-Patchwork-Id: 10483433 Return-Path: Received: from mail.wl.linuxfoundation.org (pdx-wl-mail.web.codeaurora.org [172.30.200.125]) by pdx-korg-patchwork.web.codeaurora.org (Postfix) with ESMTP id 6308B60532 for ; Sat, 23 Jun 2018 02:24:02 +0000 (UTC) Received: from mail.wl.linuxfoundation.org (localhost [127.0.0.1]) by mail.wl.linuxfoundation.org (Postfix) with ESMTP id 51DA929067 for ; Sat, 23 Jun 2018 02:24:02 +0000 (UTC) Received: by mail.wl.linuxfoundation.org (Postfix, from userid 486) id 4692829076; Sat, 23 Jun 2018 02:24:02 +0000 (UTC) X-Spam-Checker-Version: SpamAssassin 3.3.1 (2010-03-16) on pdx-wl-mail.web.codeaurora.org X-Spam-Level: X-Spam-Status: No, score=-7.8 required=2.0 tests=BAYES_00, DKIM_ADSP_CUSTOM_MED, DKIM_SIGNED, FREEMAIL_FROM, MAILING_LIST_MULTI, RCVD_IN_DNSWL_HI, T_DKIM_INVALID autolearn=ham version=3.3.1 Received: from lists.gnu.org (lists.gnu.org [208.118.235.17]) (using TLSv1 with cipher AES256-SHA (256/256 bits)) (No client certificate requested) by mail.wl.linuxfoundation.org (Postfix) with ESMTPS id 0602A29067 for ; Sat, 23 Jun 2018 02:24:00 +0000 (UTC) Received: from localhost ([::1]:36914 helo=lists.gnu.org) by lists.gnu.org with esmtp (Exim 4.71) (envelope-from ) id 1fWYDX-0006jM-Mq for patchwork-qemu-devel@patchwork.kernel.org; Fri, 22 Jun 2018 22:23:59 -0400 Received: from eggs.gnu.org ([2001:4830:134:3::10]:50238) by lists.gnu.org with esmtp (Exim 4.71) (envelope-from ) id 1fWYCs-0006P9-1E for qemu-devel@nongnu.org; Fri, 22 Jun 2018 22:23:19 -0400 Received: from Debian-exim by eggs.gnu.org with spam-scanned (Exim 4.71) (envelope-from ) id 1fWYCo-0000mq-1p for qemu-devel@nongnu.org; Fri, 22 Jun 2018 22:23:18 -0400 Received: from mail-it0-x242.google.com ([2607:f8b0:4001:c0b::242]:40005) by eggs.gnu.org with esmtps (TLS1.0:RSA_AES_128_CBC_SHA1:16) (Exim 4.71) (envelope-from ) id 1fWYCn-0000mH-RO; Fri, 22 Jun 2018 22:23:13 -0400 Received: by mail-it0-x242.google.com with SMTP id 188-v6so5373001ita.5; Fri, 22 Jun 2018 19:23:13 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20161025; h=from:to:cc:subject:date:message-id; bh=apddxmW5Nuvv8NjhNpsD1EyInRf/aR8YOMJ89Qlm9c8=; b=dCPcyvitSJVBnOpBb6vDGo/PYXYVe8hhnxM8N5318NdZKUmelewvzbiLwkfPTVR8i7 aWgDArnnNOKjDOPqGH56cCho1yhAloJfcAe1x+H84LjB1ISYDGYmJeCoKcTUFiPfNwYo rxF6zZ/JYgidG/29l1bZjzxbGVTLQG6PKao9i2M/FkPr4ZVLHwvAtFOyL3aZML/ljaNP cNf2FWRmHKqPU9goYEeM3zfTUcCLP5Z4zMFH4pPqQm6ccfqHEQqeSefx5VhWnXj6ik9x ySB8S1jIt3XTXz7E4+b3qF2HEGSiTIH4Hp8mbPD7hwpKiF9rJUpw4RlxuI9X9LVTSOGz GRvQ== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:from:to:cc:subject:date:message-id; bh=apddxmW5Nuvv8NjhNpsD1EyInRf/aR8YOMJ89Qlm9c8=; b=AmC6F31C6GQnIkO6I9ezutfeUQjeQxe0dMN7YFOM3acmjXOG93JMTVL5gW13BcRhw9 D4GbSy1Wq/CmbzrgCQIIm+IQsDBQ4nWMAUPyc9Vl2/O2OW4aMW0p1MNj3/JMATyZZXBA oANIB6aeXCsFxn/vIFpeFZTWgUZHTII4e/g1kWuIL1UwzgSJL+bJJRGEyR5eHNO0tgnp C5OfeeWmZF3cILuCtw6gGvFNGyySQzoV5mF+fpQ5J6crW/v6BwrUKeaB/ZGU1GddGqCG H6haSVuGCyj5zTEn5D1AXZznOQqvtMHi/BMMg6MvNiTFuhbPWxKDLIAXK7b/1iMg3Xib hp1w== X-Gm-Message-State: APt69E01s8T0X84SprAMsYh9/CoItWI08E2wUZ7G653qd4thpGmvg1hQ rXG70yg3EeFC9emUPCcNB/8= X-Google-Smtp-Source: ADUXVKJnJQWp4EzXUoCbZOQcR7mM2gOWfu6B2T1KDx4gahboLFtEMcHBFrNvCyOj4/1cZ3hlIxR8YA== X-Received: by 2002:a24:6ed0:: with SMTP id w199-v6mr3509918itc.65.1529720593189; Fri, 22 Jun 2018 19:23:13 -0700 (PDT) Received: from localhost.localdomain ([69.14.184.20]) by smtp.gmail.com with ESMTPSA id d130-v6sm2719797iog.84.2018.06.22.19.23.10 (version=TLS1 cipher=AES128-SHA bits=128/128); Fri, 22 Jun 2018 19:23:12 -0700 (PDT) From: John Arbuckle To: david@gibson.dropbear.id.au, qemu-devel@nongnu.org, qemu-ppc@nongnu.org Date: Fri, 22 Jun 2018 22:22:58 -0400 Message-Id: <20180623022258.22158-1-programmingkidx@gmail.com> X-Mailer: git-send-email 2.14.3 (Apple Git-98) X-detected-operating-system: by eggs.gnu.org: Genre and OS details not recognized. X-Received-From: 2607:f8b0:4001:c0b::242 Subject: [Qemu-devel] [PATCH] fix fdiv instruction X-BeenThere: qemu-devel@nongnu.org X-Mailman-Version: 2.1.21 Precedence: list List-Id: List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Cc: John Arbuckle Errors-To: qemu-devel-bounces+patchwork-qemu-devel=patchwork.kernel.org@nongnu.org Sender: "Qemu-devel" X-Virus-Scanned: ClamAV using ClamSMTP When the fdiv instruction divides a finite number by zero, the result actually depends on the FPSCR[ZE] bit. If this bit is set, the return value is zero. If it is not set the result should be either positive or negative infinity. The sign of this result would depend on the sign of the two inputs. What currently happens is only infinity is returned even if the FPSCR[ZE] bit is set. This patch fixes this problem by actually checking the FPSCR[ZE] bit when deciding what the answer should be. fdiv is suppose to only set the FPSCR's FPRF bits during a division by zero situation when the FPSCR[ZE] is not set. What currently happens is these bits are always set. This patch fixes this problem by checking the FPSCR[ZE] bit to decide if the FPRF bits should be set. https://www.pdfdrive.net/powerpc-microprocessor-family-the-programming-environments-for-32-e3087633.html This document has the information on the fdiv. Page 133 has the information on what action is executed when a division by zero situation takes place. Signed-off-by: John Arbuckle --- target/ppc/fpu_helper.c | 16 ++++++++++++++++ target/ppc/translate/fp-impl.inc.c | 28 +++++++++++++++++++++++++++- 2 files changed, 43 insertions(+), 1 deletion(-) diff --git a/target/ppc/fpu_helper.c b/target/ppc/fpu_helper.c index 7714bfe0f9..de694604fb 100644 --- a/target/ppc/fpu_helper.c +++ b/target/ppc/fpu_helper.c @@ -658,6 +658,20 @@ uint64_t helper_fdiv(CPUPPCState *env, uint64_t arg1, uint64_t arg2) } else if (unlikely(float64_is_zero(farg1.d) && float64_is_zero(farg2.d))) { /* Division of zero by zero */ farg1.ll = float_invalid_op_excp(env, POWERPC_EXCP_FP_VXZDZ, 1); + } else if (arg2 == 0) { + /* Division by zero */ + float_zero_divide_excp(env, GETPC()); + if (fpscr_ze) { /* if zero divide exception is enabled */ + farg1.ll = 0; + } else { + uint64_t sign = (farg1.ll ^ farg2.ll) >> 63; + if (sign) { /* Negative sign bit */ + farg1.ll = 0xfff0000000000000; /* Negative Infinity */ + } else { /* Positive sign bit */ + farg1.ll = 0x7ff0000000000000; /* Positive Infinity */ + } + helper_compute_fprf_float64(env, farg1.d); + } } else { if (unlikely(float64_is_signaling_nan(farg1.d, &env->fp_status) || float64_is_signaling_nan(farg2.d, &env->fp_status))) { @@ -665,6 +679,8 @@ uint64_t helper_fdiv(CPUPPCState *env, uint64_t arg1, uint64_t arg2) float_invalid_op_excp(env, POWERPC_EXCP_FP_VXSNAN, 1); } farg1.d = float64_div(farg1.d, farg2.d, &env->fp_status); + helper_compute_fprf_float64(env, farg1.d); + helper_float_check_status(env); } return farg1.ll; diff --git a/target/ppc/translate/fp-impl.inc.c b/target/ppc/translate/fp-impl.inc.c index 2fbd4d4f38..4e20bcceb4 100644 --- a/target/ppc/translate/fp-impl.inc.c +++ b/target/ppc/translate/fp-impl.inc.c @@ -84,6 +84,32 @@ static void gen_f##name(DisasContext *ctx) \ _GEN_FLOAT_AB(name, name, 0x3F, op2, inval, 0, set_fprf, type); \ _GEN_FLOAT_AB(name##s, name, 0x3B, op2, inval, 1, set_fprf, type); + +#define _GEN_FLOAT_DIV(name, op, op1, op2, inval, isfloat, set_fprf, type) \ +static void gen_f##name(DisasContext *ctx) \ +{ \ + if (unlikely(!ctx->fpu_enabled)) { \ + gen_exception(ctx, POWERPC_EXCP_FPU); \ + return; \ + } \ + gen_reset_fpstatus(); \ + gen_helper_f##op(cpu_fpr[rD(ctx->opcode)], cpu_env, \ + cpu_fpr[rA(ctx->opcode)], \ + cpu_fpr[rB(ctx->opcode)]); \ + if (isfloat) { \ + gen_helper_frsp(cpu_fpr[rD(ctx->opcode)], cpu_env, \ + cpu_fpr[rD(ctx->opcode)]); \ + } \ + if (unlikely(Rc(ctx->opcode) != 0)) { \ + gen_set_cr1_from_fpscr(ctx); \ + } \ +} + +#define GEN_FLOAT_DIV(name, op2, inval, set_fprf, type) \ +_GEN_FLOAT_DIV(name, name, 0x3F, op2, inval, 0, set_fprf, type); \ +_GEN_FLOAT_DIV(name##s, name, 0x3B, op2, inval, 1, set_fprf, type); + + #define _GEN_FLOAT_AC(name, op, op1, op2, inval, isfloat, set_fprf, type) \ static void gen_f##name(DisasContext *ctx) \ { \ @@ -149,7 +175,7 @@ static void gen_f##name(DisasContext *ctx) \ /* fadd - fadds */ GEN_FLOAT_AB(add, 0x15, 0x000007C0, 1, PPC_FLOAT); /* fdiv - fdivs */ -GEN_FLOAT_AB(div, 0x12, 0x000007C0, 1, PPC_FLOAT); +GEN_FLOAT_DIV(div, 0x12, 0x000007C0, 1, PPC_FLOAT); /* fmul - fmuls */ GEN_FLOAT_AC(mul, 0x19, 0x0000F800, 1, PPC_FLOAT);