Skip to content

Commit 08dbbe5

Browse files
authored
test: add tests for negative values in math/base/special/fmod
PR-URL: #2600 Ref: 8558d86#commitcomment-144215763 Reviewed-by: Athan Reines <[email protected]> Signed-off-by: GUNJ JOSHI <[email protected]>
1 parent d04dcbd commit 08dbbe5

File tree

11 files changed

+175
-50
lines changed

11 files changed

+175
-50
lines changed

lib/node_modules/@stdlib/math/base/special/fmod/README.md

+1
Original file line numberDiff line numberDiff line change
@@ -169,6 +169,7 @@ double stdlib_base_fmod( const double x, const double y );
169169
170170
```c
171171
#include "stdlib/math/base/special/fmod.h"
172+
#include <stdlib.h>
172173
#include <stdio.h>
173174
174175
int main( void ) {

lib/node_modules/@stdlib/math/base/special/fmod/src/main.c

+43-45
Original file line numberDiff line numberDiff line change
@@ -46,7 +46,7 @@
4646
*
4747
* @param x dividend
4848
* @param y divisor
49-
* @returns remainder
49+
* @return remainder
5050
*
5151
* @example
5252
* double out = stdlib_base_fmod( 8.9, 3.0 );
@@ -86,14 +86,14 @@ double stdlib_base_fmod( const double x, const double y ) {
8686
// Purge off exception values
8787
if ( ( hy | ly ) == 0 || ( hx >= STDLIB_CONSTANT_FLOAT64_HIGH_WORD_EXPONENT_MASK ) || ( ( hy | ( ( ly | -ly ) >> 31 ) ) > STDLIB_CONSTANT_FLOAT64_HIGH_WORD_EXPONENT_MASK ) ) {
8888
// y=0, x not finite, or y is NaN
89-
return ( x * y ) / ( x * y );
89+
return ( x * y ) / ( x * y );
9090
}
9191
if ( hx <= hy ) {
92-
if ( ( hx < hy ) || ( lx < ly ) ){
92+
if ( ( hx < hy ) || ( lx < ly ) ){
9393
// |x|<|y| return x
9494
return x;
9595
}
96-
if ( lx == ly ) {
96+
if ( lx == ly ) {
9797
// |x|=|y| return x*0
9898
return ZERO[ (uint32_t)sx >> 31 ];
9999
}
@@ -102,86 +102,84 @@ double stdlib_base_fmod( const double x, const double y ) {
102102
// Determine ix = ilogb(x)
103103
if ( hx < 0x00100000 ) {
104104
// subnormal x
105-
if ( hx == 0 ) {
105+
if ( hx == 0 ) {
106106
ix = -1043;
107107
for ( i = lx; i > 0; i <<= 1 ) {
108108
ix -=1;
109109
}
110-
} else {
110+
} else {
111111
ix = STDLIB_CONSTANT_FLOAT64_MIN_BASE2_EXPONENT;
112112
for ( i = ( hx << 11 ); i > 0; i <<= 1 ) {
113113
ix -=1;
114114
}
115-
}
115+
}
116116
} else {
117117
ix = ( hx >> 20 ) - STDLIB_CONSTANT_FLOAT64_EXPONENT_BIAS;
118118
}
119119

120-
// determine iy = ilogb(y)
120+
// determine iy = ilogb(y)
121121
if ( hy < 0x00100000 ) {
122122
// subnormal y
123-
if ( hy == 0 ) {
123+
if ( hy == 0 ) {
124124
iy = -1043;
125125
for ( i = ly; i > 0; i <<= 1 ) {
126126
iy -=1;
127127
}
128-
} else {
128+
} else {
129129
iy = STDLIB_CONSTANT_FLOAT64_MIN_BASE2_EXPONENT;
130130
for ( i = ( hy << 11 ); i > 0; i <<= 1) {
131131
iy -=1;
132132
}
133-
}
133+
}
134134
} else {
135135
iy = ( hy >> 20 ) - STDLIB_CONSTANT_FLOAT64_EXPONENT_BIAS;
136136
}
137137

138-
// set up {hx,lx}, {hy,ly} and align y to x
138+
// set up {hx,lx}, {hy,ly} and align y to x
139139
if ( ix >= STDLIB_CONSTANT_FLOAT64_MIN_BASE2_EXPONENT ) {
140140
hx = 0x00100000 | ( STDLIB_CONSTANT_FLOAT64_HIGH_WORD_SIGNIFICAND_MASK & hx );
141-
}
142-
else {
141+
} else {
143142
// subnormal x, shift x to normal
144-
n = STDLIB_CONSTANT_FLOAT64_MIN_BASE2_EXPONENT - ix;
145-
if ( n <= 31 ) {
146-
hx = ( (uint32_t)hx << n ) | ( lx >> ( 32 - n ) );
147-
lx <<= n;
143+
n = STDLIB_CONSTANT_FLOAT64_MIN_BASE2_EXPONENT - ix;
144+
if ( n <= 31 ) {
145+
hx = ( (uint32_t)hx << n ) | ( lx >> ( 32 - n ) );
146+
lx <<= n;
148147
} else {
149148
hx = lx << ( n - 32 );
150149
lx = 0;
151-
}
150+
}
152151
}
153152
if ( iy >= STDLIB_CONSTANT_FLOAT64_MIN_BASE2_EXPONENT ) {
154153
hy = 0x00100000 | ( STDLIB_CONSTANT_FLOAT64_HIGH_WORD_SIGNIFICAND_MASK & hy );
155-
}
156-
else {
154+
} else {
157155
// subnormal y, shift y to normal
158-
n = STDLIB_CONSTANT_FLOAT64_MIN_BASE2_EXPONENT - iy;
159-
if ( n <= 31 ) {
160-
hy = ( (uint32_t)hy << n ) | ( ly >> ( 32 - n ) );
161-
ly <<= n;
162-
} else {
156+
n = STDLIB_CONSTANT_FLOAT64_MIN_BASE2_EXPONENT - iy;
157+
if ( n <= 31 ) {
158+
hy = ( (uint32_t)hy << n ) | ( ly >> ( 32 - n ) );
159+
ly <<= n;
160+
} else {
163161
hy = ly << ( n - 32 );
164162
ly = 0;
165-
}
163+
}
166164
}
167165
n = ix - iy;
168166
while ( n-- ) {
169-
hz = hx - hy;
167+
hz = hx - hy;
170168
lz = lx - ly;
171169
if ( lx < ly ) {
172170
hz -= 1;
173171
}
174-
if ( hz < 0 ) {
172+
if ( hz < 0 ) {
175173
hx = hx + hx + ( lx >> 31 );
176174
lx += lx;
177-
}
178-
else {
179-
if ( ( hz | lz ) == 0 )
180-
// return sign(x)*0
181-
return ZERO[ (uint32_t)sx >> 31 ];
182-
hx = hz + hz + ( lz >> 31 );
175+
} else {
176+
if ( ( hz | lz ) == 0 ) {
177+
// return sign(x)*0
178+
return ZERO[ (uint32_t)sx >> 31 ];
179+
}
180+
hx = hz + hz + ( lz >> 31 );
183181
lx = lz + lz;
184-
}
182+
}
185183
}
186184
hz = hx - hy;
187185
lz = lx - ly;
@@ -193,34 +191,34 @@ double stdlib_base_fmod( const double x, const double y ) {
193191
lx = lz;
194192
}
195193

196-
// Convert back to floating value and restore the sign
194+
// Convert back to floating value and restore the sign
197195
if ( ( hx | lx ) == 0 ) {
198196
// return sign(x)*0
199197
return ZERO[ (uint32_t)sx >> 31 ];
200198
}
201199
while ( hx < 0x00100000 ) {
202200
// normalize x
203-
hx = hx + hx + ( lx >> 31 );
201+
hx = hx + hx + ( lx >> 31 );
204202
lx += lx;
205-
iy -= 1;
203+
iy -= 1;
206204
}
207205
if ( iy >= STDLIB_CONSTANT_FLOAT64_MIN_BASE2_EXPONENT ) {
208206
// normalize output
209-
hx = ( ( hx - 0x00100000 ) | ( ( iy + STDLIB_CONSTANT_FLOAT64_EXPONENT_BIAS ) << 20 ) );
207+
hx = ( ( hx - 0x00100000 ) | ( ( iy + STDLIB_CONSTANT_FLOAT64_EXPONENT_BIAS ) << 20 ) );
210208
stdlib_base_float64_from_words( (uint32_t)( hx | sx ), lx, &xc );
211209
} else {
212210
// subnormal output
213-
n = STDLIB_CONSTANT_FLOAT64_MIN_BASE2_EXPONENT - iy;
214-
if ( n <= 20 ) {
211+
n = STDLIB_CONSTANT_FLOAT64_MIN_BASE2_EXPONENT - iy;
212+
if ( n <= 20 ) {
215213
lx = ( lx >> n ) | ( (uint32_t)hx << ( 32 - n ) );
216214
hx >>= n;
217-
} else if ( n <= 31 ) {
215+
} else if ( n <= 31 ) {
218216
lx = ( hx << ( 32 - n ) ) | ( lx >> n );
219217
hx = sx;
220-
} else {
218+
} else {
221219
lx = hx >> ( n - 32 );
222220
hx = sx;
223-
}
221+
}
224222
stdlib_base_float64_from_words( (uint32_t)( hx | sx ), lx, &xc );
225223

226224
// create necessary signal

lib/node_modules/@stdlib/math/base/special/fmod/test/fixtures/julia/large_small.json

+1-1
Large diffs are not rendered by default.

lib/node_modules/@stdlib/math/base/special/fmod/test/fixtures/julia/negative_negative.json

+1
Large diffs are not rendered by default.

lib/node_modules/@stdlib/math/base/special/fmod/test/fixtures/julia/negative_positive.json

+1
Large diffs are not rendered by default.

lib/node_modules/@stdlib/math/base/special/fmod/test/fixtures/julia/positive_negative.json

+1
Large diffs are not rendered by default.

lib/node_modules/@stdlib/math/base/special/fmod/test/fixtures/julia/runner.jl

+17-2
Original file line numberDiff line numberDiff line change
@@ -38,9 +38,9 @@ julia> gen( x, y, \"data.json\" );
3838
```
3939
"""
4040
function gen( x, y, name )
41-
z = Array{Float64}( undef, length(x) );
41+
z = Array{Float64}( undef, length( x ) );
4242
for i in eachindex(x)
43-
z[ i ] = mod(x[i], y[i])
43+
z[ i ] = rem( x[ i ], y[ i ] )
4444
end
4545

4646
# Store data to be written to file as a collection:
@@ -85,3 +85,18 @@ gen( x, y, "small_large.json" );
8585
x = rand( 5001 ) .* 5e20;
8686
y = rand( 5001 ) .* 10;
8787
gen( x, y, "large_small.json" );
88+
89+
# x positive, y negative:
90+
x = range( 1.0, stop = 709.78, length = 1000 );
91+
y = range( -709.78, stop = -1.0, length = 1000 );
92+
gen( x, y, "positive_negative.json" );
93+
94+
# x negative, y positive:
95+
x = range( -709.78, stop = -1.0, length = 1000 );
96+
y = range( 1.0, stop = 709.78, length = 1000 );
97+
gen( x, y, "negative_positive.json" );
98+
99+
# x negative, y negative:
100+
x = range( -709.78, stop = -1.0, length = 1000 );
101+
y = range( -709.78, stop = -1.0, length = 1000 );
102+
gen( x, y, "negative_negative.json" );

lib/node_modules/@stdlib/math/base/special/fmod/test/fixtures/julia/small_large.json

+1-1
Large diffs are not rendered by default.

lib/node_modules/@stdlib/math/base/special/fmod/test/fixtures/julia/small_small.json

+1-1
Large diffs are not rendered by default.

lib/node_modules/@stdlib/math/base/special/fmod/test/test.js

+54
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,9 @@ var subnormalResults = require( './fixtures/julia/subnormal_results.json' );
3131
var smallSmall = require( './fixtures/julia/small_small.json' );
3232
var smallLarge = require( './fixtures/julia/small_large.json' );
3333
var largeSmall = require( './fixtures/julia/large_small.json' );
34+
var negativePositive = require( './fixtures/julia/negative_positive.json' );
35+
var positiveNegative = require( './fixtures/julia/positive_negative.json' );
36+
var negativeNegative = require( './fixtures/julia/negative_negative.json' );
3437

3538

3639
// TESTS //
@@ -109,6 +112,57 @@ tape( 'the function evaluates the modulus function (small `x`, small `y`)', func
109112
t.end();
110113
});
111114

115+
tape( 'the function evaluates the modulus function (positive `x`, negative `y`)', function test( t ) {
116+
var expected;
117+
var actual;
118+
var x;
119+
var y;
120+
var i;
121+
122+
x = positiveNegative.x;
123+
y = positiveNegative.y;
124+
expected = positiveNegative.expected;
125+
for ( i = 0; i < x.length; i++ ) {
126+
actual = fmod( x[ i ], y[ i ] );
127+
t.equal( actual, expected[ i ], 'fmod('+x[ i ]+','+y[ i ]+') returns '+expected[ i ] );
128+
}
129+
t.end();
130+
});
131+
132+
tape( 'the function evaluates the modulus function (negative `x`, positive `y`)', function test( t ) {
133+
var expected;
134+
var actual;
135+
var x;
136+
var y;
137+
var i;
138+
139+
x = negativePositive.x;
140+
y = negativePositive.y;
141+
expected = negativePositive.expected;
142+
for ( i = 0; i < x.length; i++ ) {
143+
actual = fmod( x[ i ], y[ i ] );
144+
t.equal( actual, expected[ i ], 'fmod('+x[ i ]+','+y[ i ]+') returns '+expected[ i ] );
145+
}
146+
t.end();
147+
});
148+
149+
tape( 'the function evaluates the modulus function (negative `x`, negative `y`)', function test( t ) {
150+
var expected;
151+
var actual;
152+
var x;
153+
var y;
154+
var i;
155+
156+
x = negativeNegative.x;
157+
y = negativeNegative.y;
158+
expected = negativeNegative.expected;
159+
for ( i = 0; i < x.length; i++ ) {
160+
actual = fmod( x[ i ], y[ i ] );
161+
t.equal( actual, expected[ i ], 'fmod('+x[ i ]+','+y[ i ]+') returns '+expected[ i ] );
162+
}
163+
t.end();
164+
});
165+
112166
tape( 'the function returns `NaN` if provided `NaN` for `y`', function test( t ) {
113167
var v;
114168

lib/node_modules/@stdlib/math/base/special/fmod/test/test.native.js

+54
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,9 @@ var subnormalResults = require( './fixtures/julia/subnormal_results.json' );
4040
var smallSmall = require( './fixtures/julia/small_small.json' );
4141
var smallLarge = require( './fixtures/julia/small_large.json' );
4242
var largeSmall = require( './fixtures/julia/large_small.json' );
43+
var negativePositive = require( './fixtures/julia/negative_positive.json' );
44+
var positiveNegative = require( './fixtures/julia/positive_negative.json' );
45+
var negativeNegative = require( './fixtures/julia/negative_negative.json' );
4346

4447

4548
// TESTS //
@@ -118,6 +121,57 @@ tape( 'the function evaluates the modulus function (small `x`, small `y`)', opts
118121
t.end();
119122
});
120123

124+
tape( 'the function evaluates the modulus function (positive `x`, negative `y`)', opts, function test( t ) {
125+
var expected;
126+
var actual;
127+
var x;
128+
var y;
129+
var i;
130+
131+
x = positiveNegative.x;
132+
y = positiveNegative.y;
133+
expected = positiveNegative.expected;
134+
for ( i = 0; i < x.length; i++ ) {
135+
actual = fmod( x[ i ], y[ i ] );
136+
t.equal( actual, expected[ i ], 'fmod('+x[ i ]+','+y[ i ]+') returns '+expected[ i ] );
137+
}
138+
t.end();
139+
});
140+
141+
tape( 'the function evaluates the modulus function (negative `x`, positive `y`)', opts, function test( t ) {
142+
var expected;
143+
var actual;
144+
var x;
145+
var y;
146+
var i;
147+
148+
x = negativePositive.x;
149+
y = negativePositive.y;
150+
expected = negativePositive.expected;
151+
for ( i = 0; i < x.length; i++ ) {
152+
actual = fmod( x[ i ], y[ i ] );
153+
t.equal( actual, expected[ i ], 'fmod('+x[ i ]+','+y[ i ]+') returns '+expected[ i ] );
154+
}
155+
t.end();
156+
});
157+
158+
tape( 'the function evaluates the modulus function (negative `x`, negative `y`)', opts, function test( t ) {
159+
var expected;
160+
var actual;
161+
var x;
162+
var y;
163+
var i;
164+
165+
x = negativeNegative.x;
166+
y = negativeNegative.y;
167+
expected = negativeNegative.expected;
168+
for ( i = 0; i < x.length; i++ ) {
169+
actual = fmod( x[ i ], y[ i ] );
170+
t.equal( actual, expected[ i ], 'fmod('+x[ i ]+','+y[ i ]+') returns '+expected[ i ] );
171+
}
172+
t.end();
173+
});
174+
121175
tape( 'the function returns `NaN` if provided `NaN` for `y`', opts, function test( t ) {
122176
var v;
123177

0 commit comments

Comments
 (0)