Loading...
1/* mpi-add.c - MPI functions
2 * Copyright (C) 1998, 1999, 2000, 2001 Free Software Foundation, Inc.
3 * Copyright (C) 1994, 1996 Free Software Foundation, Inc.
4 *
5 * This file is part of GnuPG.
6 *
7 * GnuPG is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License, or
10 * (at your option) any later version.
11 *
12 * GnuPG is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, write to the Free Software
19 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
20 *
21 * Note: This code is heavily based on the GNU MP Library.
22 * Actually it's the same code with only minor changes in the
23 * way the data is stored; this is to support the abstraction
24 * of an optional secure memory allocation which may be used
25 * to avoid revealing of sensitive data due to paging etc.
26 * The GNU MP Library itself is published under the LGPL;
27 * however I decided to publish this code under the plain GPL.
28 */
29
30#include "mpi-internal.h"
31
32/****************
33 * Add the unsigned integer V to the mpi-integer U and store the
34 * result in W. U and V may be the same.
35 */
36int mpi_add_ui(MPI w, const MPI u, unsigned long v)
37{
38 mpi_ptr_t wp, up;
39 mpi_size_t usize, wsize;
40 int usign, wsign;
41
42 usize = u->nlimbs;
43 usign = u->sign;
44 wsign = 0;
45
46 /* If not space for W (and possible carry), increase space. */
47 wsize = usize + 1;
48 if (w->alloced < wsize)
49 if (mpi_resize(w, wsize) < 0)
50 return -ENOMEM;
51
52 /* These must be after realloc (U may be the same as W). */
53 up = u->d;
54 wp = w->d;
55
56 if (!usize) { /* simple */
57 wp[0] = v;
58 wsize = v ? 1 : 0;
59 } else if (!usign) { /* mpi is not negative */
60 mpi_limb_t cy;
61 cy = mpihelp_add_1(wp, up, usize, v);
62 wp[usize] = cy;
63 wsize = usize + cy;
64 } else { /* The signs are different. Need exact comparison to determine
65 * which operand to subtract from which. */
66 if (usize == 1 && up[0] < v) {
67 wp[0] = v - up[0];
68 wsize = 1;
69 } else {
70 mpihelp_sub_1(wp, up, usize, v);
71 /* Size can decrease with at most one limb. */
72 wsize = usize - (wp[usize - 1] == 0);
73 wsign = 1;
74 }
75 }
76
77 w->nlimbs = wsize;
78 w->sign = wsign;
79 return 0;
80}
81
82int mpi_add(MPI w, MPI u, MPI v)
83{
84 mpi_ptr_t wp, up, vp;
85 mpi_size_t usize, vsize, wsize;
86 int usign, vsign, wsign;
87
88 if (u->nlimbs < v->nlimbs) { /* Swap U and V. */
89 usize = v->nlimbs;
90 usign = v->sign;
91 vsize = u->nlimbs;
92 vsign = u->sign;
93 wsize = usize + 1;
94 if (RESIZE_IF_NEEDED(w, wsize) < 0)
95 return -ENOMEM;
96 /* These must be after realloc (u or v may be the same as w). */
97 up = v->d;
98 vp = u->d;
99 } else {
100 usize = u->nlimbs;
101 usign = u->sign;
102 vsize = v->nlimbs;
103 vsign = v->sign;
104 wsize = usize + 1;
105 if (RESIZE_IF_NEEDED(w, wsize) < 0)
106 return -ENOMEM;
107 /* These must be after realloc (u or v may be the same as w). */
108 up = u->d;
109 vp = v->d;
110 }
111 wp = w->d;
112 wsign = 0;
113
114 if (!vsize) { /* simple */
115 MPN_COPY(wp, up, usize);
116 wsize = usize;
117 wsign = usign;
118 } else if (usign != vsign) { /* different sign */
119 /* This test is right since USIZE >= VSIZE */
120 if (usize != vsize) {
121 mpihelp_sub(wp, up, usize, vp, vsize);
122 wsize = usize;
123 MPN_NORMALIZE(wp, wsize);
124 wsign = usign;
125 } else if (mpihelp_cmp(up, vp, usize) < 0) {
126 mpihelp_sub_n(wp, vp, up, usize);
127 wsize = usize;
128 MPN_NORMALIZE(wp, wsize);
129 if (!usign)
130 wsign = 1;
131 } else {
132 mpihelp_sub_n(wp, up, vp, usize);
133 wsize = usize;
134 MPN_NORMALIZE(wp, wsize);
135 if (usign)
136 wsign = 1;
137 }
138 } else { /* U and V have same sign. Add them. */
139 mpi_limb_t cy = mpihelp_add(wp, up, usize, vp, vsize);
140 wp[usize] = cy;
141 wsize = usize + cy;
142 if (usign)
143 wsign = 1;
144 }
145
146 w->nlimbs = wsize;
147 w->sign = wsign;
148 return 0;
149}
150
151/****************
152 * Subtract the unsigned integer V from the mpi-integer U and store the
153 * result in W.
154 */
155int mpi_sub_ui(MPI w, MPI u, unsigned long v)
156{
157 mpi_ptr_t wp, up;
158 mpi_size_t usize, wsize;
159 int usign, wsign;
160
161 usize = u->nlimbs;
162 usign = u->sign;
163 wsign = 0;
164
165 /* If not space for W (and possible carry), increase space. */
166 wsize = usize + 1;
167 if (w->alloced < wsize)
168 if (mpi_resize(w, wsize) < 0)
169 return -ENOMEM;
170
171 /* These must be after realloc (U may be the same as W). */
172 up = u->d;
173 wp = w->d;
174
175 if (!usize) { /* simple */
176 wp[0] = v;
177 wsize = v ? 1 : 0;
178 wsign = 1;
179 } else if (usign) { /* mpi and v are negative */
180 mpi_limb_t cy;
181 cy = mpihelp_add_1(wp, up, usize, v);
182 wp[usize] = cy;
183 wsize = usize + cy;
184 } else { /* The signs are different. Need exact comparison to determine
185 * which operand to subtract from which. */
186 if (usize == 1 && up[0] < v) {
187 wp[0] = v - up[0];
188 wsize = 1;
189 wsign = 1;
190 } else {
191 mpihelp_sub_1(wp, up, usize, v);
192 /* Size can decrease with at most one limb. */
193 wsize = usize - (wp[usize - 1] == 0);
194 }
195 }
196
197 w->nlimbs = wsize;
198 w->sign = wsign;
199 return 0;
200}
201
202int mpi_sub(MPI w, MPI u, MPI v)
203{
204 int rc;
205
206 if (w == v) {
207 MPI vv;
208 if (mpi_copy(&vv, v) < 0)
209 return -ENOMEM;
210 vv->sign = !vv->sign;
211 rc = mpi_add(w, u, vv);
212 mpi_free(vv);
213 } else {
214 /* fixme: this is not thread-save (we temp. modify v) */
215 v->sign = !v->sign;
216 rc = mpi_add(w, u, v);
217 v->sign = !v->sign;
218 }
219 return rc;
220}
221
222int mpi_addm(MPI w, MPI u, MPI v, MPI m)
223{
224 if (mpi_add(w, u, v) < 0 || mpi_fdiv_r(w, w, m) < 0)
225 return -ENOMEM;
226 return 0;
227}
228
229int mpi_subm(MPI w, MPI u, MPI v, MPI m)
230{
231 if (mpi_sub(w, u, v) < 0 || mpi_fdiv_r(w, w, m) < 0)
232 return -ENOMEM;
233 return 0;
234}
1/* mpi-add.c - MPI functions
2 * Copyright (C) 1994, 1996, 1998, 2001, 2002,
3 * 2003 Free Software Foundation, Inc.
4 *
5 * This file is part of Libgcrypt.
6 *
7 * Note: This code is heavily based on the GNU MP Library.
8 * Actually it's the same code with only minor changes in the
9 * way the data is stored; this is to support the abstraction
10 * of an optional secure memory allocation which may be used
11 * to avoid revealing of sensitive data due to paging etc.
12 */
13
14#include "mpi-internal.h"
15
16/****************
17 * Add the unsigned integer V to the mpi-integer U and store the
18 * result in W. U and V may be the same.
19 */
20void mpi_add_ui(MPI w, MPI u, unsigned long v)
21{
22 mpi_ptr_t wp, up;
23 mpi_size_t usize, wsize;
24 int usign, wsign;
25
26 usize = u->nlimbs;
27 usign = u->sign;
28 wsign = 0;
29
30 /* If not space for W (and possible carry), increase space. */
31 wsize = usize + 1;
32 if (w->alloced < wsize)
33 mpi_resize(w, wsize);
34
35 /* These must be after realloc (U may be the same as W). */
36 up = u->d;
37 wp = w->d;
38
39 if (!usize) { /* simple */
40 wp[0] = v;
41 wsize = v ? 1:0;
42 } else if (!usign) { /* mpi is not negative */
43 mpi_limb_t cy;
44 cy = mpihelp_add_1(wp, up, usize, v);
45 wp[usize] = cy;
46 wsize = usize + cy;
47 } else {
48 /* The signs are different. Need exact comparison to determine
49 * which operand to subtract from which.
50 */
51 if (usize == 1 && up[0] < v) {
52 wp[0] = v - up[0];
53 wsize = 1;
54 } else {
55 mpihelp_sub_1(wp, up, usize, v);
56 /* Size can decrease with at most one limb. */
57 wsize = usize - (wp[usize-1] == 0);
58 wsign = 1;
59 }
60 }
61
62 w->nlimbs = wsize;
63 w->sign = wsign;
64}
65
66
67void mpi_add(MPI w, MPI u, MPI v)
68{
69 mpi_ptr_t wp, up, vp;
70 mpi_size_t usize, vsize, wsize;
71 int usign, vsign, wsign;
72
73 if (u->nlimbs < v->nlimbs) { /* Swap U and V. */
74 usize = v->nlimbs;
75 usign = v->sign;
76 vsize = u->nlimbs;
77 vsign = u->sign;
78 wsize = usize + 1;
79 RESIZE_IF_NEEDED(w, wsize);
80 /* These must be after realloc (u or v may be the same as w). */
81 up = v->d;
82 vp = u->d;
83 } else {
84 usize = u->nlimbs;
85 usign = u->sign;
86 vsize = v->nlimbs;
87 vsign = v->sign;
88 wsize = usize + 1;
89 RESIZE_IF_NEEDED(w, wsize);
90 /* These must be after realloc (u or v may be the same as w). */
91 up = u->d;
92 vp = v->d;
93 }
94 wp = w->d;
95 wsign = 0;
96
97 if (!vsize) { /* simple */
98 MPN_COPY(wp, up, usize);
99 wsize = usize;
100 wsign = usign;
101 } else if (usign != vsign) { /* different sign */
102 /* This test is right since USIZE >= VSIZE */
103 if (usize != vsize) {
104 mpihelp_sub(wp, up, usize, vp, vsize);
105 wsize = usize;
106 MPN_NORMALIZE(wp, wsize);
107 wsign = usign;
108 } else if (mpihelp_cmp(up, vp, usize) < 0) {
109 mpihelp_sub_n(wp, vp, up, usize);
110 wsize = usize;
111 MPN_NORMALIZE(wp, wsize);
112 if (!usign)
113 wsign = 1;
114 } else {
115 mpihelp_sub_n(wp, up, vp, usize);
116 wsize = usize;
117 MPN_NORMALIZE(wp, wsize);
118 if (usign)
119 wsign = 1;
120 }
121 } else { /* U and V have same sign. Add them. */
122 mpi_limb_t cy = mpihelp_add(wp, up, usize, vp, vsize);
123 wp[usize] = cy;
124 wsize = usize + cy;
125 if (usign)
126 wsign = 1;
127 }
128
129 w->nlimbs = wsize;
130 w->sign = wsign;
131}
132EXPORT_SYMBOL_GPL(mpi_add);
133
134void mpi_sub(MPI w, MPI u, MPI v)
135{
136 MPI vv = mpi_copy(v);
137 vv->sign = !vv->sign;
138 mpi_add(w, u, vv);
139 mpi_free(vv);
140}
141EXPORT_SYMBOL_GPL(mpi_sub);
142
143void mpi_addm(MPI w, MPI u, MPI v, MPI m)
144{
145 mpi_add(w, u, v);
146 mpi_mod(w, w, m);
147}
148EXPORT_SYMBOL_GPL(mpi_addm);
149
150void mpi_subm(MPI w, MPI u, MPI v, MPI m)
151{
152 mpi_sub(w, u, v);
153 mpi_mod(w, w, m);
154}
155EXPORT_SYMBOL_GPL(mpi_subm);