]> git.rm.cloudns.org Git - xonotic/darkplaces.git/commitdiff
Make division by 0 return NAN when output is a floating point type, not 0.
authorRudolf Polzer <divVerent@gmail.com>
Sun, 5 Jan 2025 01:33:14 +0000 (20:33 -0500)
committerdivVerent <divVerent@gmail.com>
Tue, 4 Feb 2025 21:08:11 +0000 (16:08 -0500)
For now the division by zero warning is kept.

The old behavior is behind a cvar `prvm_gameplayfix_div0is0` to enable in
case this causes any trouble for anyone.

Fixed #240.

prvm_edict.c
prvm_exec.c
prvm_execprogram.h

index c6fadd1665a2d4e75bcb6405bf443e11724d29eb..5f96e19c792fd929fb29e57910ffe58f7c1415b5 100644 (file)
@@ -54,6 +54,7 @@ cvar_t prvm_garbagecollection_scan_limit = {CF_CLIENT | CF_SERVER, "prvm_garbage
 cvar_t prvm_garbagecollection_strings = {CF_CLIENT | CF_SERVER, "prvm_garbagecollection_strings", "1", "automatically call strunzone() on strings that are not referenced"};
 cvar_t prvm_stringdebug = {CF_CLIENT | CF_SERVER, "prvm_stringdebug", "0", "Print debug and warning messages related to strings"};
 cvar_t sv_entfields_noescapes = {CF_SERVER, "sv_entfields_noescapes", "wad", "Space-separated list of fields in which backslashes won't be parsed as escapes when loading entities from .bsp or .ent files. This is a workaround for buggy maps with unescaped backslashes used as path separators (only forward slashes are allowed in Quake VFS paths)."};
+cvar_t prvm_gameplayfix_div0is0 = {CF_SERVER, "prvm_gameplayfix_div0is0", "0", "When set to 1, floating point division by 0 will return zero instead of returning the IEEE standardized result (likely nan or inf). Other ways of getting non-finite values are not affected, and the warning will still print."};
 
 static double prvm_reuseedicts_always_allow = 0;
 qbool prvm_runawaycheck = true;
@@ -3238,6 +3239,7 @@ void PRVM_Init (void)
        Cvar_RegisterVariable (&prvm_garbagecollection_strings);
        Cvar_RegisterVariable (&prvm_stringdebug);
        Cvar_RegisterVariable (&sv_entfields_noescapes);
+       Cvar_RegisterVariable (&prvm_gameplayfix_div0is0);
 
        // COMMANDLINEOPTION: PRVM: -norunaway disables the runaway loop check (it might be impossible to exit DarkPlaces if used!)
        prvm_runawaycheck = !Sys_CheckParm("-norunaway");
index fc1e6b207dcb247585716517181f0d22f30e2948..75b31440ba06d658ce32aa8c507ca522d9f07aee 100644 (file)
@@ -946,6 +946,7 @@ static void PRVM_StatementCoverageEvent(prvm_prog_t *prog, mfunction_t *func, in
 extern cvar_t prvm_traceqc;
 extern cvar_t prvm_statementprofiling;
 extern qbool prvm_runawaycheck;
+extern cvar_t prvm_gameplayfix_div0is0;
 
 #define PRVM_GLOBALSBASE 0x80000000
 
index b26c6db2e891d3c7211c2177823a310d70584003..92206704c8eaa705b776fa4d2e16cfd7c81d574b 100644 (file)
@@ -394,16 +394,17 @@ prvm_eval_t *src;
                                OPC->vector[2] = tempfloat * OPA->vector[2];
                                DISPATCH_OPCODE();
                        HANDLE_OPCODE(OP_DIV_F):
-                               if( OPB->_float != 0.0f )
-                               {
-                                       OPC->_float = OPA->_float / OPB->_float;
-                               }
-                               else
+                               if( OPB->_float == 0.0f )
                                {
                                        PRE_ERROR();
                                        VM_Warning(prog, "Attempted division of %f by zero\n", OPA->_float);
-                                       OPC->_float = 0.0f;
+                                       if (prvm_gameplayfix_div0is0.integer)
+                                       {
+                                               OPC->_float = 0;
+                                               DISPATCH_OPCODE();
+                                       }
                                }
+                               OPC->_float = OPA->_float / OPB->_float;
                                DISPATCH_OPCODE();
                        HANDLE_OPCODE(OP_BITAND_F):
                                OPC->_float = (prvm_int_t)OPA->_float & (prvm_int_t)OPB->_float;
@@ -884,21 +885,22 @@ prvm_eval_t *src;
                                OPC->vector[2] = (prvm_vec_t) OPB->_int * OPA->vector[2];
                                DISPATCH_OPCODE();
                        HANDLE_OPCODE(OP_DIV_VF):
-                               if( OPB->_float != 0.0f )
-                               {
-                                       tempfloat = OPB->_float;
-                                       OPC->vector[0] = OPA->vector[0] / tempfloat;
-                                       OPC->vector[1] = OPA->vector[1] / tempfloat;
-                                       OPC->vector[2] = OPA->vector[2] / tempfloat;
-                               }
-                               else
+                               if( OPB->_float == 0.0f )
                                {
                                        PRE_ERROR();
                                        VM_Warning(prog, "Attempted division of '%f %f %f' by zero\n", OPA->vector[0], OPA->vector[1], OPA->vector[2]);
-                                       OPC->vector[0] = 0.0f;
-                                       OPC->vector[1] = 0.0f;
-                                       OPC->vector[2] = 0.0f;
+                                       if (prvm_gameplayfix_div0is0.integer)
+                                       {
+                                               OPC->vector[0] = 0;
+                                               OPC->vector[1] = 0;
+                                               OPC->vector[2] = 0;
+                                               DISPATCH_OPCODE();
+                                       }
                                }
+                               tempfloat = OPB->_float;
+                               OPC->vector[0] = OPA->vector[0] / tempfloat;
+                               OPC->vector[1] = OPA->vector[1] / tempfloat;
+                               OPC->vector[2] = OPA->vector[2] / tempfloat;
                                DISPATCH_OPCODE();
                        HANDLE_OPCODE(OP_DIV_I):
                                // NOTE: This also catches the second kind of division that can trap, namely, -2147483648 / -1,
@@ -915,28 +917,30 @@ prvm_eval_t *src;
                                }
                                DISPATCH_OPCODE();
                        HANDLE_OPCODE(OP_DIV_IF):
-                               if( OPB->_float != 0.0f )
-                               {
-                                       OPC->_float = ((prvm_vec_t) OPA->_int) / OPB->_float;
-                               }
-                               else
+                               if( OPB->_float == 0.0f )
                                {
                                        PRE_ERROR();
                                        VM_Warning(prog, "Attempted division of %"PRVM_PRIi" by zero\n", OPA->_int);
-                                       OPC->_float = 0;
+                                       if (prvm_gameplayfix_div0is0.integer)
+                                       {
+                                               OPC->_float = 0;
+                                               DISPATCH_OPCODE();
+                                       }
                                }
+                               OPC->_float = ((prvm_vec_t) OPA->_int) / OPB->_float;
                                DISPATCH_OPCODE();
                        HANDLE_OPCODE(OP_DIV_FI):
-                               if( OPB->_int != 0 )
-                               {
-                                       OPC->_float = OPA->_float / (prvm_vec_t) OPB->_int;
-                               }
-                               else
+                               if( OPB->_int == 0 )
                                {
                                        PRE_ERROR();
                                        VM_Warning(prog, "Attempted division of %f by zero\n", OPA->_float);
-                                       OPC->_float = 0;
+                                       if (prvm_gameplayfix_div0is0.integer)
+                                       {
+                                               OPC->_float = 0;
+                                               DISPATCH_OPCODE();
+                                       }
                                }
+                               OPC->_float = OPA->_float / (prvm_vec_t) OPB->_int;
                                DISPATCH_OPCODE();
                        HANDLE_OPCODE(OP_CONV_ITOF):
                                OPC->_float = OPA->_int;