maint.mk: avoid spurious failure of _sc_search_regexp-using tests
[gnulib.git] / lib / remainder.c
1 /* Remainder.
2    Copyright (C) 2012 Free Software Foundation, Inc.
3
4    This program is free software: you can redistribute it and/or modify
5    it under the terms of the GNU General Public License as published by
6    the Free Software Foundation; either version 3 of the License, or
7    (at your option) any later version.
8
9    This program is distributed in the hope that it will be useful,
10    but WITHOUT ANY WARRANTY; without even the implied warranty of
11    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12    GNU General Public License for more details.
13
14    You should have received a copy of the GNU General Public License
15    along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
16
17 #include <config.h>
18
19 /* Specification.  */
20 #include <math.h>
21
22 double
23 remainder (double x, double y)
24 {
25   double q = - round (x / y);
26   double r = fma (q, y, x); /* = x + q * y, computed in one step */
27   /* Correct possible rounding errors in the quotient x / y.  */
28   double half_y = 0.5L * y;
29   if (y >= 0)
30     {
31       /* Expect -y/2 <= r <= y/2.  */
32       if (r > half_y)
33         q -= 1, r = fma (q, y, x);
34       else if (r < - half_y)
35         q += 1, r = fma (q, y, x);
36     }
37   else
38     {
39       /* Expect y/2 <= r <= -y/2.  */
40       if (r > - half_y)
41         q += 1, r = fma (q, y, x);
42       else if (r < half_y)
43         q -= 1, r = fma (q, y, x);
44     }
45   return r;
46 }