Line data Source code
1 : #include "tommath_private.h" 2 : #ifdef BN_MP_SUB_C 3 : /* LibTomMath, multiple-precision integer library -- Tom St Denis */ 4 : /* SPDX-License-Identifier: Unlicense */ 5 : 6 : /* high level subtraction (handles signs) */ 7 1070402 : mp_err mp_sub(const mp_int *a, const mp_int *b, mp_int *c) 8 : { 9 1070402 : mp_sign sa = a->sign, sb = b->sign; 10 73060 : mp_err err; 11 : 12 1070402 : if (sa != sb) { 13 : /* subtract a negative from a positive, OR */ 14 : /* subtract a positive from a negative. */ 15 : /* In either case, ADD their magnitudes, */ 16 : /* and use the sign of the first number. */ 17 324071 : c->sign = sa; 18 324071 : err = s_mp_add(a, b, c); 19 : } else { 20 : /* subtract a positive from a positive, OR */ 21 : /* subtract a negative from a negative. */ 22 : /* First, take the difference between their */ 23 : /* magnitudes, then... */ 24 746331 : if (mp_cmp_mag(a, b) != MP_LT) { 25 : /* Copy the sign from the first */ 26 522318 : c->sign = sa; 27 : /* The first has a larger or equal magnitude */ 28 522318 : err = s_mp_sub(a, b, c); 29 : } else { 30 : /* The result has the *opposite* sign from */ 31 : /* the first number. */ 32 224013 : c->sign = (sa == MP_ZPOS) ? MP_NEG : MP_ZPOS; 33 : /* The second has a larger magnitude */ 34 224013 : err = s_mp_sub(b, a, c); 35 : } 36 : } 37 1070402 : return err; 38 : } 39 : 40 : #endif