aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--libjava/ChangeLog11
-rw-r--r--libjava/include/i386-signal.h83
-rw-r--r--libjava/include/sparc-signal.h3
-rw-r--r--libjava/prims.cc6
-rw-r--r--libjava/testsuite/libjava.lang/Divide_1.java105
-rw-r--r--libjava/testsuite/libjava.lang/Divide_1.out11
6 files changed, 211 insertions, 8 deletions
diff --git a/libjava/ChangeLog b/libjava/ChangeLog
index b583109..1a96458 100644
--- a/libjava/ChangeLog
+++ b/libjava/ChangeLog
@@ -1,3 +1,14 @@
+1999-05-20 Andrew Haley <aph@cygnus.com>
+
+ * libjava/prims.cc (catch_fpe): Call to HANDLE_DIVIDE_OVERFLOW
+ added.
+ * include/i386-signal.h (HANDLE_DIVIDE_OVERFLOW): New macro.
+ (INIT_FPE): Exception string made more informative.
+ * include/sparc-signal.h (INIT_FPE): Exception string made more
+ informative.
+ * testsuite/libjava.lang/Divide_1.java: New file.
+ * testsuite/libjava.lang/Divide_1.out: New file.
+
1999-05-19 Tom Tromey <tromey@cygnus.com>
* aclocal.m4, configure: Rebuilt.
diff --git a/libjava/include/i386-signal.h b/libjava/include/i386-signal.h
index 73d8bbb..12ffe2a 100644
--- a/libjava/include/i386-signal.h
+++ b/libjava/include/i386-signal.h
@@ -25,16 +25,86 @@ details. */
static void _name (int _dummy)
#define MAKE_THROW_FRAME \
+do \
{ \
void **_p = (void **)&_dummy; \
struct sigcontext_struct *_regs = (struct sigcontext_struct *)++_p; \
\
register unsigned long _ebp = _regs->ebp; \
- register unsigned long _eip = _regs->eip; \
- \
+ register unsigned char *_eip = (unsigned char *)_regs->eip; \
+ \
+ asm volatile ("mov %0, (%%ebp); mov %1, 4(%%ebp)" \
+ : : "r"(_ebp), "r"(_eip)); \
+} \
+while (0)
+
+#define HANDLE_DIVIDE_OVERFLOW \
+do \
+{ \
+ void **_p = (void **)&_dummy; \
+ struct sigcontext_struct *_regs = (struct sigcontext_struct *)++_p; \
+ \
+ register unsigned long *_ebp = (unsigned long *)_regs->ebp; \
+ register unsigned char *_eip = (unsigned char *)_regs->eip; \
+ \
+ /* According to the JVM spec, "if the dividend is the negative \
+ * integer of the smallest magnitude and the divisor is -1, then \
+ * overflow occurs and the result is equal to the dividend. Despite \
+ * the overflow, no exception occurs". \
+ \
+ * We handle this by inspecting the instruction which generated the \
+ * signal and advancing eip to point to the following instruction. \
+ * As the instructions are variable length it is necessary to do a \
+ * little calculation to figure out where the following instruction \
+ * actually is. \
+ \
+ */ \
+ \
+ if (_eip[0] == 0xf7) \
+ { \
+ unsigned char _modrm = _eip[1]; \
+ \
+ if (_regs->eax == 0x80000000 \
+ && ((_modrm >> 3) & 7) == 7) /* Signed divide */ \
+ { \
+ _regs->edx = 0; /* the remainder is zero */ \
+ switch (_modrm >> 6) \
+ { \
+ case 0: \
+ if ((_modrm & 7) == 5) \
+ _eip += 4; \
+ break; \
+ case 1: \
+ _eip += 1; \
+ break; \
+ case 2: \
+ _eip += 4; \
+ break; \
+ case 3: \
+ break; \
+ } \
+ _eip += 2; \
+ _regs->eip = (unsigned long)_eip; \
+ return; \
+ } \
+ else if (((_modrm >> 3) & 7) == 6) /* Unsigned divide */ \
+ { \
+ /* We assume that unsigned divisions are in library code, so \
+ * we throw one level down the stack, which was hopefully \
+ * the place that called the library routine. This will \
+ * break if the library is ever compiled with \
+ * -fomit-frame-pointer, but at least this way we've got a \
+ * good chance of finding the exception handler. */ \
+ \
+ _eip = (unsigned char *)_ebp[1]; \
+ _ebp = (unsigned long *)_ebp[0]; \
+ } \
+ } \
+ \
asm volatile ("mov %0, (%%ebp); mov %1, 4(%%ebp)" \
: : "r"(_ebp), "r"(_eip)); \
-}
+} \
+while (0)
#define INIT_SEGV \
do \
@@ -48,10 +118,11 @@ do \
} \
while (0)
-#define INIT_FPE \
+#define INIT_FPE \
do \
- { \
- arithexception = new java::lang::ArithmeticException (); \
+ { \
+ arithexception = new java::lang::ArithmeticException \
+ (JvNewStringLatin1 ("/ by zero")); \
struct sigaction act; \
act.sa_handler = catch_fpe; \
sigemptyset (&act.sa_mask); \
diff --git a/libjava/include/sparc-signal.h b/libjava/include/sparc-signal.h
index dfe8635..03b5cdc 100644
--- a/libjava/include/sparc-signal.h
+++ b/libjava/include/sparc-signal.h
@@ -51,7 +51,8 @@ while (0)
#define INIT_FPE \
do \
{ \
- arithexception = new java::lang::ArithmeticException (); \
+ arithexception = new java::lang::ArithmeticException \
+ (JvNewStringLatin1 ("/ by zero")); \
struct sigaction act; \
act.sa_flags = SA_SIGINFO; \
act.sa_sigaction = catch_fpe; \
diff --git a/libjava/prims.cc b/libjava/prims.cc
index 4cc31cd..9909485 100644
--- a/libjava/prims.cc
+++ b/libjava/prims.cc
@@ -68,7 +68,11 @@ SIGNAL_HANDLER (catch_segv)
static java::lang::ArithmeticException *arithexception;
SIGNAL_HANDLER (catch_fpe)
{
+#ifdef HANDLE_DIVIDE_OVERFLOW
+ HANDLE_DIVIDE_OVERFLOW;
+#else
MAKE_THROW_FRAME;
+#endif
_Jv_Throw (arithexception);
}
#endif
@@ -97,7 +101,7 @@ _Jv_equalUtf8Consts (Utf8Const* a, Utf8Const *b)
}
/* True iff A is equal to STR.
- HASH is STR->hashCode().
+ HASH is STR->hashCode().
*/
jboolean
diff --git a/libjava/testsuite/libjava.lang/Divide_1.java b/libjava/testsuite/libjava.lang/Divide_1.java
new file mode 100644
index 0000000..d4e0193
--- /dev/null
+++ b/libjava/testsuite/libjava.lang/Divide_1.java
@@ -0,0 +1,105 @@
+public class Divide_1
+{
+ static int b = Integer.parseInt ("-1");
+ int b1 = Integer.parseInt ("-1");
+ static int zero = Integer.parseInt ("0");
+
+ void probe ()
+ {
+ try {
+ int a = Integer.parseInt ("-80000000", 16);
+ int c = a/b;
+ System.out.println (c);
+ } catch (Exception _) {
+ System.out.println (_);
+ }
+
+ try {
+ int a = Integer.parseInt ("-80000000", 16);
+ int c = a/-1;
+ System.out.println (c);
+ } catch (Exception _) {
+ System.out.println (_);
+ }
+
+ try {
+ int a = Integer.parseInt ("-80000000", 16);
+ int c = a%b;
+ System.out.println (c);
+ } catch (Exception _) {
+ System.out.println (_);
+ }
+
+ try {
+ int a = Integer.parseInt ("-80000000", 16);
+ int c = a%b1;
+ System.out.println (c);
+ } catch (Exception _) {
+ System.out.println (_);
+ }
+
+ try {
+ int a = Integer.parseInt ("-80000000", 16);
+ int c = a%-1;
+ System.out.println (c);
+ } catch (Exception _) {
+ System.out.println (_);
+ }
+
+ try {
+ int a = Integer.parseInt ("8000", 16);
+ int b = Integer.parseInt ("0", 16);
+ int c = a/b;
+ System.out.println (c);
+ } catch (Exception _) {
+ System.out.println (_);
+ }
+
+ try {
+ int a = Integer.parseInt ("8000", 16);
+ int b = Integer.parseInt ("0", 16);
+ int c = a%b;
+ System.out.println (c);
+ } catch (Exception _) {
+ System.out.println (_);
+ }
+
+ try {
+ long a = Long.parseLong ("-8000000000000000", 16);
+ long c = a/b;
+ System.out.println (c);
+ } catch (Exception _) {
+ System.out.println (_);
+ }
+
+ try {
+ long a = Long.parseLong ("-8000000000000000", 16);
+ long c = a%b;
+ System.out.println (c);
+ } catch (Exception _) {
+ System.out.println (_);
+ }
+
+ try {
+ long a = Long.parseLong ("8000", 16);
+ long b = Long.parseLong ("0", 16);
+ long c = a/b;
+ System.out.println (c);
+ } catch (Exception _) {
+ System.out.println (_);
+ }
+
+ try {
+ long a = Long.parseLong ("8000", 16);
+ long b = Long.parseLong ("0", 16);
+ long c = a%b;
+ System.out.println (c);
+ } catch (Exception _) {
+ System.out.println (_);
+ }
+ }
+
+ public static void main (String[] args) {
+ new Divide_1 ().probe ();
+ }
+}
diff --git a/libjava/testsuite/libjava.lang/Divide_1.out b/libjava/testsuite/libjava.lang/Divide_1.out
new file mode 100644
index 0000000..6e1f87e
--- /dev/null
+++ b/libjava/testsuite/libjava.lang/Divide_1.out
@@ -0,0 +1,11 @@
+-2147483648
+-2147483648
+0
+0
+0
+java.lang.ArithmeticException: / by zero
+java.lang.ArithmeticException: / by zero
+-9223372036854775808
+0
+java.lang.ArithmeticException: / by zero
+java.lang.ArithmeticException: / by zero