mul.c revision 1.1.1.2 1 1.1 mrg /* mpf_mul -- Multiply two floats.
2 1.1 mrg
3 1.1 mrg Copyright 1993, 1994, 1996, 2001, 2005 Free Software Foundation, Inc.
4 1.1 mrg
5 1.1 mrg This file is part of the GNU MP Library.
6 1.1 mrg
7 1.1 mrg The GNU MP Library is free software; you can redistribute it and/or modify
8 1.1.1.2 mrg it under the terms of either:
9 1.1.1.2 mrg
10 1.1.1.2 mrg * the GNU Lesser General Public License as published by the Free
11 1.1.1.2 mrg Software Foundation; either version 3 of the License, or (at your
12 1.1.1.2 mrg option) any later version.
13 1.1.1.2 mrg
14 1.1.1.2 mrg or
15 1.1.1.2 mrg
16 1.1.1.2 mrg * the GNU General Public License as published by the Free Software
17 1.1.1.2 mrg Foundation; either version 2 of the License, or (at your option) any
18 1.1.1.2 mrg later version.
19 1.1.1.2 mrg
20 1.1.1.2 mrg or both in parallel, as here.
21 1.1 mrg
22 1.1 mrg The GNU MP Library is distributed in the hope that it will be useful, but
23 1.1 mrg WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
24 1.1.1.2 mrg or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
25 1.1.1.2 mrg for more details.
26 1.1 mrg
27 1.1.1.2 mrg You should have received copies of the GNU General Public License and the
28 1.1.1.2 mrg GNU Lesser General Public License along with the GNU MP Library. If not,
29 1.1.1.2 mrg see https://www.gnu.org/licenses/. */
30 1.1 mrg
31 1.1 mrg #include "gmp.h"
32 1.1 mrg #include "gmp-impl.h"
33 1.1 mrg
34 1.1 mrg void
35 1.1 mrg mpf_mul (mpf_ptr r, mpf_srcptr u, mpf_srcptr v)
36 1.1 mrg {
37 1.1 mrg mp_srcptr up, vp;
38 1.1 mrg mp_size_t usize, vsize;
39 1.1 mrg mp_size_t sign_product;
40 1.1 mrg mp_size_t prec = r->_mp_prec;
41 1.1 mrg
42 1.1 mrg usize = u->_mp_size;
43 1.1 mrg vsize = v->_mp_size;
44 1.1 mrg sign_product = usize ^ vsize;
45 1.1 mrg
46 1.1 mrg usize = ABS (usize);
47 1.1 mrg vsize = ABS (vsize);
48 1.1 mrg
49 1.1 mrg up = u->_mp_d;
50 1.1 mrg vp = v->_mp_d;
51 1.1 mrg if (usize > prec)
52 1.1 mrg {
53 1.1 mrg up += usize - prec;
54 1.1 mrg usize = prec;
55 1.1 mrg }
56 1.1 mrg if (vsize > prec)
57 1.1 mrg {
58 1.1 mrg vp += vsize - prec;
59 1.1 mrg vsize = prec;
60 1.1 mrg }
61 1.1 mrg
62 1.1 mrg if (usize == 0 || vsize == 0)
63 1.1 mrg {
64 1.1 mrg r->_mp_size = 0;
65 1.1 mrg r->_mp_exp = 0; /* ??? */
66 1.1 mrg }
67 1.1 mrg else
68 1.1 mrg {
69 1.1 mrg mp_size_t rsize;
70 1.1 mrg mp_limb_t cy_limb;
71 1.1 mrg mp_ptr rp, tp;
72 1.1 mrg mp_size_t adj;
73 1.1.1.2 mrg TMP_DECL;
74 1.1 mrg
75 1.1.1.2 mrg TMP_MARK;
76 1.1 mrg rsize = usize + vsize;
77 1.1 mrg tp = TMP_ALLOC_LIMBS (rsize);
78 1.1 mrg cy_limb = (usize >= vsize
79 1.1 mrg ? mpn_mul (tp, up, usize, vp, vsize)
80 1.1 mrg : mpn_mul (tp, vp, vsize, up, usize));
81 1.1 mrg
82 1.1 mrg adj = cy_limb == 0;
83 1.1 mrg rsize -= adj;
84 1.1 mrg prec++;
85 1.1 mrg if (rsize > prec)
86 1.1 mrg {
87 1.1 mrg tp += rsize - prec;
88 1.1 mrg rsize = prec;
89 1.1 mrg }
90 1.1 mrg rp = r->_mp_d;
91 1.1 mrg MPN_COPY (rp, tp, rsize);
92 1.1 mrg r->_mp_exp = u->_mp_exp + v->_mp_exp - adj;
93 1.1 mrg r->_mp_size = sign_product >= 0 ? rsize : -rsize;
94 1.1.1.2 mrg
95 1.1.1.2 mrg TMP_FREE;
96 1.1 mrg }
97 1.1 mrg }
98