VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/math/bignum.cpp@ 63451

Last change on this file since 63451 was 63451, checked in by vboxsync, 8 years ago

Runtime: warnings

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 94.1 KB
Line 
1/* $Id: bignum.cpp 63451 2016-08-15 00:39:40Z vboxsync $ */
2/** @file
3 * IPRT - Big Integer Numbers.
4 */
5
6/*
7 * Copyright (C) 2006-2016 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31/*#ifdef IN_RING3
32# define RTMEM_WRAP_TO_EF_APIS
33#endif*/
34#include "internal/iprt.h"
35#include <iprt/bignum.h>
36
37#include <iprt/asm.h>
38#include <iprt/asm-math.h>
39#include <iprt/err.h>
40#include <iprt/mem.h>
41#include <iprt/memsafer.h>
42#include <iprt/string.h>
43#if RTBIGNUM_ELEMENT_BITS == 64
44# include <iprt/uint128.h>
45#endif
46
47
48/*********************************************************************************************************************************
49* Defined Constants And Macros *
50*********************************************************************************************************************************/
51/** Allocation alignment in elements. */
52#ifndef RTMEM_WRAP_TO_EF_APIS
53# define RTBIGNUM_ALIGNMENT 4U
54#else
55# define RTBIGNUM_ALIGNMENT 1U
56#endif
57
58/** The max size (in bytes) of an elements array. */
59#define RTBIGNUM_MAX_SIZE _4M
60
61
62/** Assert the validity of a big number structure pointer in strict builds. */
63#ifdef RT_STRICT
64# define RTBIGNUM_ASSERT_VALID(a_pBigNum) \
65 do { \
66 AssertPtr(a_pBigNum); \
67 Assert(!(a_pBigNum)->fCurScrambled); \
68 Assert( (a_pBigNum)->cUsed == (a_pBigNum)->cAllocated \
69 || ASMMemIsZero(&(a_pBigNum)->pauElements[(a_pBigNum)->cUsed], \
70 ((a_pBigNum)->cAllocated - (a_pBigNum)->cUsed) * RTBIGNUM_ELEMENT_SIZE)); \
71 } while (0)
72#else
73# define RTBIGNUM_ASSERT_VALID(a_pBigNum) do {} while (0)
74#endif
75
76
77/** Enable assembly optimizations. */
78#if defined(RT_ARCH_AMD64) || defined(RT_ARCH_X86)
79# define IPRT_BIGINT_WITH_ASM
80#endif
81
82
83/** @def RTBIGNUM_ZERO_ALIGN
84 * For calculating the rtBigNumEnsureExtraZeroElements argument from cUsed.
85 * This has to do with 64-bit assembly instruction operating as RTBIGNUMELEMENT
86 * was 64-bit on some hosts.
87 */
88#if defined(IPRT_BIGINT_WITH_ASM) && ARCH_BITS == 64 && RTBIGNUM_ELEMENT_SIZE == 4 && defined(RT_LITTLE_ENDIAN)
89# define RTBIGNUM_ZERO_ALIGN(a_cUsed) RT_ALIGN_32(a_cUsed, 2)
90#elif defined(IPRT_BIGINT_WITH_ASM)
91# define RTBIGNUM_ZERO_ALIGN(a_cUsed) (a_cUsed)
92#else
93# define RTBIGNUM_ZERO_ALIGN(a_cUsed) (a_cUsed)
94#endif
95
96#define RTBIGNUMELEMENT_HALF_MASK ( ((RTBIGNUMELEMENT)1 << (RTBIGNUM_ELEMENT_BITS / 2)) - (RTBIGNUMELEMENT)1)
97#define RTBIGNUMELEMENT_LO_HALF(a_uElement) ( (RTBIGNUMELEMENT_HALF_MASK) & (a_uElement) )
98#define RTBIGNUMELEMENT_HI_HALF(a_uElement) ( (a_uElement) >> (RTBIGNUM_ELEMENT_BITS / 2) )
99
100
101/*********************************************************************************************************************************
102* Structures and Typedefs *
103*********************************************************************************************************************************/
104/** Type the size of two elements. */
105#if RTBIGNUM_ELEMENT_BITS == 64
106typedef RTUINT128U RTBIGNUMELEMENT2X;
107#else
108typedef RTUINT64U RTBIGNUMELEMENT2X;
109#endif
110
111
112/*********************************************************************************************************************************
113* Internal Functions *
114*********************************************************************************************************************************/
115DECLINLINE(int) rtBigNumSetUsed(PRTBIGNUM pBigNum, uint32_t cNewUsed);
116
117#ifdef IPRT_BIGINT_WITH_ASM
118/* bignum-amd64-x86.asm: */
119DECLASM(void) rtBigNumMagnitudeSubAssemblyWorker(RTBIGNUMELEMENT *pauResult, RTBIGNUMELEMENT const *pauMinuend,
120 RTBIGNUMELEMENT const *pauSubtrahend, uint32_t cUsed);
121DECLASM(void) rtBigNumMagnitudeSubThisAssemblyWorker(RTBIGNUMELEMENT *pauMinuendResult, RTBIGNUMELEMENT const *pauSubtrahend,
122 uint32_t cUsed);
123DECLASM(RTBIGNUMELEMENT) rtBigNumMagnitudeShiftLeftOneAssemblyWorker(RTBIGNUMELEMENT *pauElements, uint32_t cUsed,
124 RTBIGNUMELEMENT uCarry);
125DECLASM(void) rtBigNumElement2xDiv2xBy1x(RTBIGNUMELEMENT2X *puQuotient, RTBIGNUMELEMENT *puRemainder,
126 RTBIGNUMELEMENT uDividendHi, RTBIGNUMELEMENT uDividendLo, RTBIGNUMELEMENT uDivisor);
127DECLASM(void) rtBigNumMagnitudeMultiplyAssemblyWorker(PRTBIGNUMELEMENT pauResult,
128 PCRTBIGNUMELEMENT pauMultiplier, uint32_t cMultiplier,
129 PCRTBIGNUMELEMENT pauMultiplicand, uint32_t cMultiplicand);
130#endif
131
132
133
134
135
136/** @name Functions working on one element.
137 * @{ */
138
139DECLINLINE(uint32_t) rtBigNumElementBitCount(RTBIGNUMELEMENT uElement)
140{
141#if RTBIGNUM_ELEMENT_SIZE == 8
142 if (uElement >> 32)
143 return ASMBitLastSetU32((uint32_t)(uElement >> 32)) + 32;
144 return ASMBitLastSetU32((uint32_t)uElement);
145#elif RTBIGNUM_ELEMENT_SIZE == 4
146 return ASMBitLastSetU32(uElement);
147#else
148# error "Bad RTBIGNUM_ELEMENT_SIZE value"
149#endif
150}
151
152
153/**
154 * Does addition with carry.
155 *
156 * This is a candidate for inline assembly on some platforms.
157 *
158 * @returns The result (the sum)
159 * @param uAugend What to add to.
160 * @param uAddend What to add to it.
161 * @param pfCarry Where to read the input carry and return the output
162 * carry.
163 */
164DECLINLINE(RTBIGNUMELEMENT) rtBigNumElementAddWithCarry(RTBIGNUMELEMENT uAugend, RTBIGNUMELEMENT uAddend,
165 RTBIGNUMELEMENT *pfCarry)
166{
167 RTBIGNUMELEMENT uRet = uAugend + uAddend;
168 if (!*pfCarry)
169 *pfCarry = uRet < uAugend;
170 else
171 {
172 uRet += 1;
173 *pfCarry = uRet <= uAugend;
174 }
175 return uRet;
176}
177
178
179#if !defined(IPRT_BIGINT_WITH_ASM) || defined(RT_STRICT)
180/**
181 * Does addition with borrow.
182 *
183 * This is a candidate for inline assembly on some platforms.
184 *
185 * @returns The result (the sum)
186 * @param uMinuend What to subtract from.
187 * @param uSubtrahend What to subtract.
188 * @param pfBorrow Where to read the input borrow and return the output
189 * borrow.
190 */
191DECLINLINE(RTBIGNUMELEMENT) rtBigNumElementSubWithBorrow(RTBIGNUMELEMENT uMinuend, RTBIGNUMELEMENT uSubtrahend,
192 RTBIGNUMELEMENT *pfBorrow)
193{
194 RTBIGNUMELEMENT uRet = uMinuend - uSubtrahend - *pfBorrow;
195
196 /* Figure out if we borrowed. */
197 *pfBorrow = !*pfBorrow ? uMinuend < uSubtrahend : uMinuend <= uSubtrahend;
198 return uRet;
199}
200#endif
201
202/** @} */
203
204
205
206
207/** @name Double element primitives.
208 * @{ */
209
210static int rtBigNumElement2xCopyToMagnitude(RTBIGNUMELEMENT2X const *pValue2x, PRTBIGNUM pDst)
211{
212 int rc;
213 if (pValue2x->s.Hi)
214 {
215 rc = rtBigNumSetUsed(pDst, 2);
216 if (RT_SUCCESS(rc))
217 {
218 pDst->pauElements[0] = pValue2x->s.Lo;
219 pDst->pauElements[1] = pValue2x->s.Hi;
220 }
221 }
222 else if (pValue2x->s.Lo)
223 {
224 rc = rtBigNumSetUsed(pDst, 1);
225 if (RT_SUCCESS(rc))
226 pDst->pauElements[0] = pValue2x->s.Lo;
227 }
228 else
229 rc = rtBigNumSetUsed(pDst, 0);
230 return rc;
231}
232
233static void rtBigNumElement2xDiv(RTBIGNUMELEMENT2X *puQuotient, RTBIGNUMELEMENT2X *puRemainder,
234 RTBIGNUMELEMENT uDividendHi, RTBIGNUMELEMENT uDividendLo,
235 RTBIGNUMELEMENT uDivisorHi, RTBIGNUMELEMENT uDivisorLo)
236{
237 RTBIGNUMELEMENT2X uDividend;
238 uDividend.s.Lo = uDividendLo;
239 uDividend.s.Hi = uDividendHi;
240
241 RTBIGNUMELEMENT2X uDivisor;
242 uDivisor.s.Lo = uDivisorLo;
243 uDivisor.s.Hi = uDivisorHi;
244
245#if RTBIGNUM_ELEMENT_BITS == 64
246 RTUInt128DivRem(puQuotient, puRemainder, &uDividend, &uDivisor);
247#else
248 puQuotient->u = uDividend.u / uDivisor.u;
249 puRemainder->u = uDividend.u % uDivisor.u;
250#endif
251}
252
253#ifndef IPRT_BIGINT_WITH_ASM
254static void rtBigNumElement2xDiv2xBy1x(RTBIGNUMELEMENT2X *puQuotient, RTBIGNUMELEMENT *puRemainder,
255 RTBIGNUMELEMENT uDividendHi, RTBIGNUMELEMENT uDividendLo, RTBIGNUMELEMENT uDivisor)
256{
257 RTBIGNUMELEMENT2X uDividend;
258 uDividend.s.Lo = uDividendLo;
259 uDividend.s.Hi = uDividendHi;
260
261# if RTBIGNUM_ELEMENT_BITS == 64
262 RTBIGNUMELEMENT2X uRemainder2x;
263 RTBIGNUMELEMENT2X uDivisor2x;
264 uDivisor2x.s.Hi = 0;
265 uDivisor2x.s.Lo = uDivisor;
266 /** @todo optimize this. */
267 RTUInt128DivRem(puQuotient, &uRemainder2x, &uDividend, &uDivisor2x);
268 *puRemainder = uRemainder2x.s.Lo;
269# else
270 puQuotient->u = uDividend.u / uDivisor;
271 puRemainder->u = uDividend.u % uDivisor;
272# endif
273}
274#endif
275
276DECLINLINE(void) rtBigNumElement2xDec(RTBIGNUMELEMENT2X *puValue)
277{
278#if RTBIGNUM_ELEMENT_BITS == 64
279 if (puValue->s.Lo-- == 0)
280 puValue->s.Hi--;
281#else
282 puValue->u -= 1;
283#endif
284}
285
286#if 0 /* unused */
287DECLINLINE(void) rtBigNumElement2xAdd1x(RTBIGNUMELEMENT2X *puValue, RTBIGNUMELEMENT uAdd)
288{
289#if RTBIGNUM_ELEMENT_BITS == 64
290 RTUInt128AssignAddU64(puValue, uAdd);
291#else
292 puValue->u += uAdd;
293#endif
294}
295#endif /* unused */
296
297/** @} */
298
299
300
301
302
303/**
304 * Scrambles a big number if required.
305 *
306 * @param pBigNum The big number.
307 */
308DECLINLINE(void) rtBigNumScramble(PRTBIGNUM pBigNum)
309{
310 if (pBigNum->fSensitive)
311 {
312 AssertReturnVoid(!pBigNum->fCurScrambled);
313 if (pBigNum->pauElements)
314 {
315 int rc = RTMemSaferScramble(pBigNum->pauElements, pBigNum->cAllocated * RTBIGNUM_ELEMENT_SIZE); AssertRC(rc);
316 pBigNum->fCurScrambled = RT_SUCCESS(rc);
317 }
318 else
319 pBigNum->fCurScrambled = true;
320 }
321}
322
323
324/**
325 * Unscrambles a big number if required.
326 *
327 * @returns IPRT status code.
328 * @param pBigNum The big number.
329 */
330DECLINLINE(int) rtBigNumUnscramble(PRTBIGNUM pBigNum)
331{
332 if (pBigNum->fSensitive)
333 {
334 AssertReturn(pBigNum->fCurScrambled, VERR_INTERNAL_ERROR_2);
335 if (pBigNum->pauElements)
336 {
337 int rc = RTMemSaferUnscramble(pBigNum->pauElements, pBigNum->cAllocated * RTBIGNUM_ELEMENT_SIZE); AssertRC(rc);
338 pBigNum->fCurScrambled = !RT_SUCCESS(rc);
339 return rc;
340 }
341 else
342 pBigNum->fCurScrambled = false;
343 }
344 return VINF_SUCCESS;
345}
346
347
348/**
349 * Getter function for pauElements which extends the array to infinity.
350 *
351 * @returns The element value.
352 * @param pBigNum The big number.
353 * @param iElement The element index.
354 */
355DECLINLINE(RTBIGNUMELEMENT) rtBigNumGetElement(PCRTBIGNUM pBigNum, uint32_t iElement)
356{
357 if (iElement < pBigNum->cUsed)
358 return pBigNum->pauElements[iElement];
359 return 0;
360}
361
362
363/**
364 * Grows the pauElements array so it can fit at least @a cNewUsed entries.
365 *
366 * @returns IPRT status code.
367 * @param pBigNum The big number.
368 * @param cNewUsed The new cUsed value.
369 * @param cMinElements The minimum number of elements.
370 */
371static int rtBigNumGrow(PRTBIGNUM pBigNum, uint32_t cNewUsed, uint32_t cMinElements)
372{
373 Assert(cMinElements >= cNewUsed);
374 uint32_t const cbOld = pBigNum->cAllocated * RTBIGNUM_ELEMENT_SIZE;
375 uint32_t const cNew = RT_ALIGN_32(cMinElements, RTBIGNUM_ALIGNMENT);
376 uint32_t const cbNew = cNew * RTBIGNUM_ELEMENT_SIZE;
377 Assert(cbNew > cbOld);
378 if (cbNew <= RTBIGNUM_MAX_SIZE && cbNew > cbOld)
379 {
380 void *pvNew;
381 if (pBigNum->fSensitive)
382 pvNew = RTMemSaferReallocZ(cbOld, pBigNum->pauElements, cbNew);
383 else
384 pvNew = RTMemRealloc(pBigNum->pauElements, cbNew);
385 if (RT_LIKELY(pvNew))
386 {
387 if (cbNew > cbOld)
388 RT_BZERO((char *)pvNew + cbOld, cbNew - cbOld);
389 if (pBigNum->cUsed > cNewUsed)
390 RT_BZERO((RTBIGNUMELEMENT *)pvNew + cNewUsed, (pBigNum->cUsed - cNewUsed) * RTBIGNUM_ELEMENT_SIZE);
391
392 pBigNum->pauElements = (RTBIGNUMELEMENT *)pvNew;
393 pBigNum->cUsed = cNewUsed;
394 pBigNum->cAllocated = cNew;
395 return VINF_SUCCESS;
396 }
397 return VERR_NO_MEMORY;
398 }
399 return VERR_OUT_OF_RANGE;
400}
401
402
403/**
404 * Changes the cUsed member, growing the pauElements array if necessary.
405 *
406 * Any elements added to the array will be initialized to zero.
407 *
408 * @returns IPRT status code.
409 * @param pBigNum The big number.
410 * @param cNewUsed The new cUsed value.
411 */
412DECLINLINE(int) rtBigNumSetUsed(PRTBIGNUM pBigNum, uint32_t cNewUsed)
413{
414 if (pBigNum->cAllocated >= cNewUsed)
415 {
416 if (pBigNum->cUsed > cNewUsed)
417 RT_BZERO(&pBigNum->pauElements[cNewUsed], (pBigNum->cUsed - cNewUsed) * RTBIGNUM_ELEMENT_SIZE);
418#ifdef RT_STRICT
419 else if (pBigNum->cUsed != cNewUsed)
420 Assert(ASMMemIsZero(&pBigNum->pauElements[pBigNum->cUsed], (cNewUsed - pBigNum->cUsed) * RTBIGNUM_ELEMENT_SIZE));
421#endif
422 pBigNum->cUsed = cNewUsed;
423 return VINF_SUCCESS;
424 }
425 return rtBigNumGrow(pBigNum, cNewUsed, cNewUsed);
426}
427
428
429/**
430 * Extended version of rtBigNumSetUsed that also allow specifying the number of
431 * zero elements required.
432 *
433 * @returns IPRT status code.
434 * @param pBigNum The big number.
435 * @param cNewUsed The new cUsed value.
436 * @param cMinElements The minimum number of elements allocated. The
437 * difference between @a cNewUsed and @a cMinElements
438 * is initialized to zero because all free elements are
439 * zero.
440 */
441DECLINLINE(int) rtBigNumSetUsedEx(PRTBIGNUM pBigNum, uint32_t cNewUsed, uint32_t cMinElements)
442{
443 if (pBigNum->cAllocated >= cMinElements)
444 {
445 if (pBigNum->cUsed > cNewUsed)
446 RT_BZERO(&pBigNum->pauElements[cNewUsed], (pBigNum->cUsed - cNewUsed) * RTBIGNUM_ELEMENT_SIZE);
447#ifdef RT_STRICT
448 else if (pBigNum->cUsed != cNewUsed)
449 Assert(ASMMemIsZero(&pBigNum->pauElements[pBigNum->cUsed], (cNewUsed - pBigNum->cUsed) * RTBIGNUM_ELEMENT_SIZE));
450#endif
451 pBigNum->cUsed = cNewUsed;
452 return VINF_SUCCESS;
453 }
454 return rtBigNumGrow(pBigNum, cNewUsed, cMinElements);
455}
456
457
458/**
459 * For ensuring zero padding of pauElements for sub/add with carry assembly
460 * operations.
461 *
462 * @returns IPRT status code.
463 * @param pBigNum The big number.
464 * @param cElements The number of elements that must be in the elements
465 * array array, where those after pBigNum->cUsed must
466 * be zero.
467 */
468DECLINLINE(int) rtBigNumEnsureExtraZeroElements(PRTBIGNUM pBigNum, uint32_t cElements)
469{
470 if (pBigNum->cAllocated >= cElements)
471 {
472 Assert( pBigNum->cAllocated == pBigNum->cUsed
473 || ASMMemIsZero(&pBigNum->pauElements[pBigNum->cUsed],
474 (pBigNum->cAllocated - pBigNum->cUsed) * RTBIGNUM_ELEMENT_SIZE));
475 return VINF_SUCCESS;
476 }
477 return rtBigNumGrow(pBigNum, pBigNum->cUsed, cElements);
478}
479
480
481/**
482 * The slow part of rtBigNumEnsureElementPresent where we need to do actual zero
483 * extending.
484 *
485 * @returns IPRT status code.
486 * @param pBigNum The big number.
487 * @param iElement The element we wish to access.
488 */
489static int rtBigNumEnsureElementPresentSlow(PRTBIGNUM pBigNum, uint32_t iElement)
490{
491 uint32_t const cOldUsed = pBigNum->cUsed;
492 int rc = rtBigNumSetUsed(pBigNum, iElement + 1);
493 if (RT_SUCCESS(rc))
494 {
495 RT_BZERO(&pBigNum->pauElements[cOldUsed], (iElement + 1 - cOldUsed) * RTBIGNUM_ELEMENT_SIZE);
496 return VINF_SUCCESS;
497 }
498 return rc;
499}
500
501
502/**
503 * Zero extends the element array to make sure a the specified element index is
504 * accessible.
505 *
506 * This is typically used with bit operations and self modifying methods. Any
507 * new elements added will be initialized to zero. The caller is responsible
508 * for there not being any trailing zero elements.
509 *
510 * The number must be unscrambled.
511 *
512 * @returns IPRT status code.
513 * @param pBigNum The big number.
514 * @param iElement The element we wish to access.
515 */
516DECLINLINE(int) rtBigNumEnsureElementPresent(PRTBIGNUM pBigNum, uint32_t iElement)
517{
518 if (iElement < pBigNum->cUsed)
519 return VINF_SUCCESS;
520 return rtBigNumEnsureElementPresentSlow(pBigNum, iElement);
521}
522
523
524/**
525 * Strips zero elements from the magnitude value.
526 *
527 * @param pBigNum The big number to strip.
528 */
529static void rtBigNumStripTrailingZeros(PRTBIGNUM pBigNum)
530{
531 uint32_t i = pBigNum->cUsed;
532 while (i > 0 && pBigNum->pauElements[i - 1] == 0)
533 i--;
534 pBigNum->cUsed = i;
535}
536
537
538/**
539 * Initialize the big number to zero.
540 *
541 * @returns @a pBigNum
542 * @param pBigNum The big number.
543 * @param fFlags The flags.
544 * @internal
545 */
546DECLINLINE(PRTBIGNUM) rtBigNumInitZeroInternal(PRTBIGNUM pBigNum, uint32_t fFlags)
547{
548 RT_ZERO(*pBigNum);
549 pBigNum->fSensitive = RT_BOOL(fFlags & RTBIGNUMINIT_F_SENSITIVE);
550 return pBigNum;
551}
552
553
554/**
555 * Initialize the big number to zero from a template variable.
556 *
557 * @returns @a pBigNum
558 * @param pBigNum The big number.
559 * @param pTemplate The template big number.
560 * @internal
561 */
562DECLINLINE(PRTBIGNUM) rtBigNumInitZeroTemplate(PRTBIGNUM pBigNum, PCRTBIGNUM pTemplate)
563{
564 RT_ZERO(*pBigNum);
565 pBigNum->fSensitive = pTemplate->fSensitive;
566 return pBigNum;
567}
568
569
570RTDECL(int) RTBigNumInit(PRTBIGNUM pBigNum, uint32_t fFlags, void const *pvRaw, size_t cbRaw)
571{
572 /*
573 * Validate input.
574 */
575 AssertPtrReturn(pBigNum, VERR_INVALID_POINTER);
576 AssertReturn(RT_BOOL(fFlags & RTBIGNUMINIT_F_ENDIAN_BIG) ^ RT_BOOL(fFlags & RTBIGNUMINIT_F_ENDIAN_LITTLE),
577 VERR_INVALID_PARAMETER);
578 AssertReturn(RT_BOOL(fFlags & RTBIGNUMINIT_F_UNSIGNED) ^ RT_BOOL(fFlags & RTBIGNUMINIT_F_SIGNED), VERR_INVALID_PARAMETER);
579 if (cbRaw)
580 AssertPtrReturn(pvRaw, VERR_INVALID_POINTER);
581
582 /*
583 * Initalize the big number to zero.
584 */
585 rtBigNumInitZeroInternal(pBigNum, fFlags);
586
587 /*
588 * Strip the input and figure the sign flag.
589 */
590 uint8_t const *pb = (uint8_t const *)pvRaw;
591 if (cbRaw)
592 {
593 if (fFlags & RTBIGNUMINIT_F_ENDIAN_LITTLE)
594 {
595 if (fFlags & RTBIGNUMINIT_F_UNSIGNED)
596 {
597 while (cbRaw > 0 && pb[cbRaw - 1] == 0)
598 cbRaw--;
599 }
600 else
601 {
602 if (pb[cbRaw - 1] >> 7)
603 {
604 pBigNum->fNegative = 1;
605 while (cbRaw > 1 && pb[cbRaw - 1] == 0xff)
606 cbRaw--;
607 }
608 else
609 while (cbRaw > 0 && pb[cbRaw - 1] == 0)
610 cbRaw--;
611 }
612 }
613 else
614 {
615 if (fFlags & RTBIGNUMINIT_F_UNSIGNED)
616 {
617 while (cbRaw > 0 && *pb == 0)
618 pb++, cbRaw--;
619 }
620 else
621 {
622 if (*pb >> 7)
623 {
624 pBigNum->fNegative = 1;
625 while (cbRaw > 1 && *pb == 0xff)
626 pb++, cbRaw--;
627 }
628 else
629 while (cbRaw > 0 && *pb == 0)
630 pb++, cbRaw--;
631 }
632 }
633 }
634
635 /*
636 * Allocate memory for the elements.
637 */
638 size_t cbAligned = RT_ALIGN_Z(cbRaw, RTBIGNUM_ELEMENT_SIZE);
639 if (RT_UNLIKELY(cbAligned >= RTBIGNUM_MAX_SIZE))
640 return VERR_OUT_OF_RANGE;
641 pBigNum->cUsed = (uint32_t)cbAligned / RTBIGNUM_ELEMENT_SIZE;
642 if (pBigNum->cUsed)
643 {
644 pBigNum->cAllocated = RT_ALIGN_32(pBigNum->cUsed, RTBIGNUM_ALIGNMENT);
645 if (pBigNum->fSensitive)
646 pBigNum->pauElements = (RTBIGNUMELEMENT *)RTMemSaferAllocZ(pBigNum->cAllocated * RTBIGNUM_ELEMENT_SIZE);
647 else
648 pBigNum->pauElements = (RTBIGNUMELEMENT *)RTMemAlloc(pBigNum->cAllocated * RTBIGNUM_ELEMENT_SIZE);
649 if (RT_UNLIKELY(!pBigNum->pauElements))
650 return VERR_NO_MEMORY;
651
652 /*
653 * Initialize the array.
654 */
655 uint32_t i = 0;
656 if (fFlags & RTBIGNUMINIT_F_ENDIAN_LITTLE)
657 {
658 while (cbRaw >= RTBIGNUM_ELEMENT_SIZE)
659 {
660#if RTBIGNUM_ELEMENT_SIZE == 8
661 pBigNum->pauElements[i] = RT_MAKE_U64_FROM_U8(pb[0], pb[1], pb[2], pb[3], pb[4], pb[5], pb[6], pb[7]);
662#elif RTBIGNUM_ELEMENT_SIZE == 4
663 pBigNum->pauElements[i] = RT_MAKE_U32_FROM_U8(pb[0], pb[1], pb[2], pb[3]);
664#else
665# error "Bad RTBIGNUM_ELEMENT_SIZE value"
666#endif
667 i++;
668 pb += RTBIGNUM_ELEMENT_SIZE;
669 cbRaw -= RTBIGNUM_ELEMENT_SIZE;
670 }
671
672 if (cbRaw > 0)
673 {
674 RTBIGNUMELEMENT uLast = pBigNum->fNegative ? ~(RTBIGNUMELEMENT)0 : 0;
675 switch (cbRaw)
676 {
677 default: AssertFailed();
678#if RTBIGNUM_ELEMENT_SIZE == 8
679 case 7: uLast = (uLast << 8) | pb[6];
680 case 6: uLast = (uLast << 8) | pb[5];
681 case 5: uLast = (uLast << 8) | pb[4];
682 case 4: uLast = (uLast << 8) | pb[3];
683#endif
684 case 3: uLast = (uLast << 8) | pb[2];
685 case 2: uLast = (uLast << 8) | pb[1];
686 case 1: uLast = (uLast << 8) | pb[0];
687 }
688 pBigNum->pauElements[i] = uLast;
689 }
690 }
691 else
692 {
693 pb += cbRaw;
694 while (cbRaw >= RTBIGNUM_ELEMENT_SIZE)
695 {
696 pb -= RTBIGNUM_ELEMENT_SIZE;
697#if RTBIGNUM_ELEMENT_SIZE == 8
698 pBigNum->pauElements[i] = RT_MAKE_U64_FROM_U8(pb[7], pb[6], pb[5], pb[4], pb[3], pb[2], pb[1], pb[0]);
699#elif RTBIGNUM_ELEMENT_SIZE == 4
700 pBigNum->pauElements[i] = RT_MAKE_U32_FROM_U8(pb[3], pb[2], pb[1], pb[0]);
701#else
702# error "Bad RTBIGNUM_ELEMENT_SIZE value"
703#endif
704 i++;
705 cbRaw -= RTBIGNUM_ELEMENT_SIZE;
706 }
707
708 if (cbRaw > 0)
709 {
710 RTBIGNUMELEMENT uLast = pBigNum->fNegative ? ~(RTBIGNUMELEMENT)0 : 0;
711 pb -= cbRaw;
712 switch (cbRaw)
713 {
714 default: AssertFailed();
715#if RTBIGNUM_ELEMENT_SIZE == 8
716 case 7: uLast = (uLast << 8) | *pb++;
717 case 6: uLast = (uLast << 8) | *pb++;
718 case 5: uLast = (uLast << 8) | *pb++;
719 case 4: uLast = (uLast << 8) | *pb++;
720#endif
721 case 3: uLast = (uLast << 8) | *pb++;
722 case 2: uLast = (uLast << 8) | *pb++;
723 case 1: uLast = (uLast << 8) | *pb++;
724 }
725 pBigNum->pauElements[i] = uLast;
726 }
727 }
728
729 /*
730 * If negative, negate it so we get a positive magnitude value in pauElements.
731 */
732 if (pBigNum->fNegative)
733 {
734 pBigNum->pauElements[0] = 0U - pBigNum->pauElements[0];
735 for (i = 1; i < pBigNum->cUsed; i++)
736 pBigNum->pauElements[i] = 0U - pBigNum->pauElements[i] - 1U;
737 }
738
739 /*
740 * Clear unused elements.
741 */
742 if (pBigNum->cUsed != pBigNum->cAllocated)
743 {
744 RTBIGNUMELEMENT *puUnused = &pBigNum->pauElements[pBigNum->cUsed];
745 AssertCompile(RTBIGNUM_ALIGNMENT <= 4);
746 switch (pBigNum->cAllocated - pBigNum->cUsed)
747 {
748 default: AssertFailed();
749 case 3: *puUnused++ = 0;
750 case 2: *puUnused++ = 0;
751 case 1: *puUnused++ = 0;
752 }
753 }
754 RTBIGNUM_ASSERT_VALID(pBigNum);
755 }
756
757 rtBigNumScramble(pBigNum);
758 return VINF_SUCCESS;
759}
760
761
762RTDECL(int) RTBigNumInitZero(PRTBIGNUM pBigNum, uint32_t fFlags)
763{
764 AssertReturn(!(fFlags & ~RTBIGNUMINIT_F_SENSITIVE), VERR_INVALID_PARAMETER);
765 AssertPtrReturn(pBigNum, VERR_INVALID_POINTER);
766
767 rtBigNumInitZeroInternal(pBigNum, fFlags);
768 rtBigNumScramble(pBigNum);
769 return VINF_SUCCESS;
770}
771
772
773/**
774 * Internal clone function that assumes the caller takes care of scrambling.
775 *
776 * @returns IPRT status code.
777 * @param pBigNum The target number.
778 * @param pSrc The source number.
779 */
780static int rtBigNumCloneInternal(PRTBIGNUM pBigNum, PCRTBIGNUM pSrc)
781{
782 Assert(!pSrc->fCurScrambled);
783 int rc = VINF_SUCCESS;
784
785 /*
786 * Copy over the data.
787 */
788 RT_ZERO(*pBigNum);
789 pBigNum->fNegative = pSrc->fNegative;
790 pBigNum->fSensitive = pSrc->fSensitive;
791 pBigNum->cUsed = pSrc->cUsed;
792 if (pSrc->cUsed)
793 {
794 /* Duplicate the element array. */
795 pBigNum->cAllocated = RT_ALIGN_32(pBigNum->cUsed, RTBIGNUM_ALIGNMENT);
796 if (pBigNum->fSensitive)
797 pBigNum->pauElements = (RTBIGNUMELEMENT *)RTMemSaferAllocZ(pBigNum->cAllocated * RTBIGNUM_ELEMENT_SIZE);
798 else
799 pBigNum->pauElements = (RTBIGNUMELEMENT *)RTMemAlloc(pBigNum->cAllocated * RTBIGNUM_ELEMENT_SIZE);
800 if (RT_LIKELY(pBigNum->pauElements))
801 {
802 memcpy(pBigNum->pauElements, pSrc->pauElements, pBigNum->cUsed * RTBIGNUM_ELEMENT_SIZE);
803 if (pBigNum->cUsed != pBigNum->cAllocated)
804 RT_BZERO(&pBigNum->pauElements[pBigNum->cUsed], (pBigNum->cAllocated - pBigNum->cUsed) * RTBIGNUM_ELEMENT_SIZE);
805 }
806 else
807 {
808 RT_ZERO(*pBigNum);
809 rc = VERR_NO_MEMORY;
810 }
811 }
812 return rc;
813}
814
815
816RTDECL(int) RTBigNumClone(PRTBIGNUM pBigNum, PCRTBIGNUM pSrc)
817{
818 int rc = rtBigNumUnscramble((PRTBIGNUM)pSrc);
819 if (RT_SUCCESS(rc))
820 {
821 RTBIGNUM_ASSERT_VALID(pSrc);
822 rc = rtBigNumCloneInternal(pBigNum, pSrc);
823 if (RT_SUCCESS(rc))
824 rtBigNumScramble(pBigNum);
825 rtBigNumScramble((PRTBIGNUM)pSrc);
826 }
827 return rc;
828}
829
830
831RTDECL(int) RTBigNumDestroy(PRTBIGNUM pBigNum)
832{
833 if (pBigNum)
834 {
835 if (pBigNum->pauElements)
836 {
837 Assert(pBigNum->cAllocated > 0);
838 if (pBigNum->fSensitive)
839 {
840 RTMemSaferFree(pBigNum->pauElements, pBigNum->cAllocated * RTBIGNUM_ELEMENT_SIZE);
841 RT_ZERO(*pBigNum);
842 }
843 RTMemFree(pBigNum->pauElements);
844 pBigNum->pauElements = NULL;
845 }
846 }
847 return VINF_SUCCESS;
848}
849
850
851RTDECL(int) RTBigNumAssign(PRTBIGNUM pDst, PCRTBIGNUM pSrc)
852{
853 AssertReturn(pDst->fSensitive >= pSrc->fSensitive, VERR_BIGNUM_SENSITIVE_INPUT);
854 int rc = rtBigNumUnscramble(pDst);
855 if (RT_SUCCESS(rc))
856 {
857 RTBIGNUM_ASSERT_VALID(pDst);
858 rc = rtBigNumUnscramble((PRTBIGNUM)pSrc);
859 if (RT_SUCCESS(rc))
860 {
861 RTBIGNUM_ASSERT_VALID(pSrc);
862 if ( pDst->fSensitive == pSrc->fSensitive
863 || pDst->fSensitive)
864 {
865 if (pDst->cAllocated >= pSrc->cUsed)
866 {
867 if (pDst->cUsed > pSrc->cUsed)
868 RT_BZERO(&pDst->pauElements[pSrc->cUsed], (pDst->cUsed - pSrc->cUsed) * RTBIGNUM_ELEMENT_SIZE);
869 pDst->cUsed = pSrc->cUsed;
870 pDst->fNegative = pSrc->fNegative;
871 memcpy(pDst->pauElements, pSrc->pauElements, pSrc->cUsed * RTBIGNUM_ELEMENT_SIZE);
872 }
873 else
874 {
875 rc = rtBigNumGrow(pDst, pSrc->cUsed, pSrc->cUsed);
876 if (RT_SUCCESS(rc))
877 {
878 pDst->fNegative = pSrc->fNegative;
879 memcpy(pDst->pauElements, pSrc->pauElements, pSrc->cUsed * RTBIGNUM_ELEMENT_SIZE);
880 }
881 }
882 }
883 else
884 rc = VERR_BIGNUM_SENSITIVE_INPUT;
885 rtBigNumScramble((PRTBIGNUM)pSrc);
886 }
887 rtBigNumScramble(pDst);
888 }
889 return rc;
890}
891
892
893/**
894 * Same as RTBigNumBitWidth, except that it ignore the signed bit.
895 *
896 * The number must be unscrambled.
897 *
898 * @returns The effective width of the magnitude, in bits. Returns 0 if the
899 * value is zero.
900 * @param pBigNum The bit number.
901 */
902static uint32_t rtBigNumMagnitudeBitWidth(PCRTBIGNUM pBigNum)
903{
904 uint32_t idxLast = pBigNum->cUsed;
905 if (idxLast)
906 {
907 idxLast--;
908 RTBIGNUMELEMENT uLast = pBigNum->pauElements[idxLast]; Assert(uLast);
909 return rtBigNumElementBitCount(uLast) + idxLast * RTBIGNUM_ELEMENT_BITS;
910 }
911 return 0;
912}
913
914
915RTDECL(uint32_t) RTBigNumBitWidth(PCRTBIGNUM pBigNum)
916{
917 uint32_t idxLast = pBigNum->cUsed;
918 if (idxLast)
919 {
920 idxLast--;
921 rtBigNumUnscramble((PRTBIGNUM)pBigNum);
922 RTBIGNUMELEMENT uLast = pBigNum->pauElements[idxLast]; Assert(uLast);
923 rtBigNumScramble((PRTBIGNUM)pBigNum);
924 return rtBigNumElementBitCount(uLast) + idxLast * RTBIGNUM_ELEMENT_BITS + pBigNum->fNegative;
925 }
926 return 0;
927}
928
929
930RTDECL(uint32_t) RTBigNumByteWidth(PCRTBIGNUM pBigNum)
931{
932 uint32_t cBits = RTBigNumBitWidth(pBigNum);
933 return (cBits + 7) / 8;
934}
935
936
937RTDECL(int) RTBigNumToBytesBigEndian(PCRTBIGNUM pBigNum, void *pvBuf, size_t cbWanted)
938{
939 AssertPtrReturn(pvBuf, VERR_INVALID_POINTER);
940 AssertReturn(cbWanted > 0, VERR_INVALID_PARAMETER);
941
942 int rc = rtBigNumUnscramble((PRTBIGNUM)pBigNum);
943 if (RT_SUCCESS(rc))
944 {
945 RTBIGNUM_ASSERT_VALID(pBigNum);
946 rc = VINF_SUCCESS;
947 if (pBigNum->cUsed != 0)
948 {
949 uint8_t *pbDst = (uint8_t *)pvBuf;
950 pbDst += cbWanted - 1;
951 for (uint32_t i = 0; i < pBigNum->cUsed; i++)
952 {
953 RTBIGNUMELEMENT uElement = pBigNum->pauElements[i];
954 if (pBigNum->fNegative)
955 uElement = (RTBIGNUMELEMENT)0 - uElement - (i > 0);
956 if (cbWanted >= sizeof(uElement))
957 {
958 *pbDst-- = (uint8_t)uElement;
959 uElement >>= 8;
960 *pbDst-- = (uint8_t)uElement;
961 uElement >>= 8;
962 *pbDst-- = (uint8_t)uElement;
963 uElement >>= 8;
964 *pbDst-- = (uint8_t)uElement;
965#if RTBIGNUM_ELEMENT_SIZE == 8
966 uElement >>= 8;
967 *pbDst-- = (uint8_t)uElement;
968 uElement >>= 8;
969 *pbDst-- = (uint8_t)uElement;
970 uElement >>= 8;
971 *pbDst-- = (uint8_t)uElement;
972 uElement >>= 8;
973 *pbDst-- = (uint8_t)uElement;
974#elif RTBIGNUM_ELEMENT_SIZE != 4
975# error "Bad RTBIGNUM_ELEMENT_SIZE value"
976#endif
977 cbWanted -= sizeof(uElement);
978 }
979 else
980 {
981
982 uint32_t cBitsLeft = RTBIGNUM_ELEMENT_BITS;
983 while (cbWanted > 0)
984 {
985 *pbDst-- = (uint8_t)uElement;
986 uElement >>= 8;
987 cBitsLeft -= 8;
988 cbWanted--;
989 }
990 Assert(cBitsLeft > 0); Assert(cBitsLeft < RTBIGNUM_ELEMENT_BITS);
991 if ( i + 1 < pBigNum->cUsed
992 || ( !pBigNum->fNegative
993 ? uElement != 0
994 : uElement != ((RTBIGNUMELEMENT)1 << cBitsLeft) - 1U ) )
995 rc = VERR_BUFFER_OVERFLOW;
996 break;
997 }
998 }
999
1000 /* Sign extend the number to the desired output size. */
1001 if (cbWanted > 0)
1002 memset(pbDst - cbWanted, pBigNum->fNegative ? 0 : 0xff, cbWanted);
1003 }
1004 else
1005 RT_BZERO(pvBuf, cbWanted);
1006 rtBigNumScramble((PRTBIGNUM)pBigNum);
1007 }
1008 return rc;
1009}
1010
1011
1012RTDECL(int) RTBigNumCompare(PRTBIGNUM pLeft, PRTBIGNUM pRight)
1013{
1014 int rc = rtBigNumUnscramble(pLeft);
1015 if (RT_SUCCESS(rc))
1016 {
1017 RTBIGNUM_ASSERT_VALID(pLeft);
1018 rc = rtBigNumUnscramble(pRight);
1019 if (RT_SUCCESS(rc))
1020 {
1021 RTBIGNUM_ASSERT_VALID(pRight);
1022 if (pLeft->fNegative == pRight->fNegative)
1023 {
1024 if (pLeft->cUsed == pRight->cUsed)
1025 {
1026 rc = 0;
1027 uint32_t i = pLeft->cUsed;
1028 while (i-- > 0)
1029 if (pLeft->pauElements[i] != pRight->pauElements[i])
1030 {
1031 rc = pLeft->pauElements[i] < pRight->pauElements[i] ? -1 : 1;
1032 break;
1033 }
1034 if (pLeft->fNegative)
1035 rc = -rc;
1036 }
1037 else
1038 rc = !pLeft->fNegative
1039 ? pLeft->cUsed < pRight->cUsed ? -1 : 1
1040 : pLeft->cUsed < pRight->cUsed ? 1 : -1;
1041 }
1042 else
1043 rc = pLeft->fNegative ? -1 : 1;
1044
1045 rtBigNumScramble(pRight);
1046 }
1047 rtBigNumScramble(pLeft);
1048 }
1049 return rc;
1050}
1051
1052
1053RTDECL(int) RTBigNumCompareWithU64(PRTBIGNUM pLeft, uint64_t uRight)
1054{
1055 int rc = rtBigNumUnscramble(pLeft);
1056 if (RT_SUCCESS(rc))
1057 {
1058 RTBIGNUM_ASSERT_VALID(pLeft);
1059 if (!pLeft->fNegative)
1060 {
1061 if (pLeft->cUsed * RTBIGNUM_ELEMENT_SIZE <= sizeof(uRight))
1062 {
1063 if (pLeft->cUsed == 0)
1064 rc = uRight == 0 ? 0 : -1;
1065 else
1066 {
1067#if RTBIGNUM_ELEMENT_SIZE == 8
1068 uint64_t uLeft = rtBigNumGetElement(pLeft, 0);
1069 if (uLeft < uRight)
1070 rc = -1;
1071 else
1072 rc = uLeft == uRight ? 0 : 1;
1073#elif RTBIGNUM_ELEMENT_SIZE == 4
1074 uint32_t uSubLeft = rtBigNumGetElement(pLeft, 1);
1075 uint32_t uSubRight = uRight >> 32;
1076 if (uSubLeft == uSubRight)
1077 {
1078 uSubLeft = rtBigNumGetElement(pLeft, 0);
1079 uSubRight = (uint32_t)uRight;
1080 }
1081 if (uSubLeft < uSubRight)
1082 rc = -1;
1083 else
1084 rc = uSubLeft == uSubRight ? 0 : 1;
1085#else
1086# error "Bad RTBIGNUM_ELEMENT_SIZE value"
1087#endif
1088 }
1089 }
1090 else
1091 rc = 1;
1092 }
1093 else
1094 rc = -1;
1095 rtBigNumScramble(pLeft);
1096 }
1097 return rc;
1098}
1099
1100
1101RTDECL(int) RTBigNumCompareWithS64(PRTBIGNUM pLeft, int64_t iRight)
1102{
1103 int rc = rtBigNumUnscramble(pLeft);
1104 if (RT_SUCCESS(rc))
1105 {
1106 RTBIGNUM_ASSERT_VALID(pLeft);
1107 if (pLeft->fNegative == (unsigned)(iRight < 0)) /* (unsigned cast is for MSC weirdness) */
1108 {
1109 AssertCompile(RTBIGNUM_ELEMENT_SIZE <= sizeof(iRight));
1110 if (pLeft->cUsed * RTBIGNUM_ELEMENT_SIZE <= sizeof(iRight))
1111 {
1112 uint64_t uRightMagn = !pLeft->fNegative ? (uint64_t)iRight : (uint64_t)-iRight;
1113#if RTBIGNUM_ELEMENT_SIZE == 8
1114 uint64_t uLeft = rtBigNumGetElement(pLeft, 0);
1115 if (uLeft < uRightMagn)
1116 rc = -1;
1117 else
1118 rc = uLeft == (uint64_t)uRightMagn ? 0 : 1;
1119#elif RTBIGNUM_ELEMENT_SIZE == 4
1120 uint32_t uSubLeft = rtBigNumGetElement(pLeft, 1);
1121 uint32_t uSubRight = uRightMagn >> 32;
1122 if (uSubLeft == uSubRight)
1123 {
1124 uSubLeft = rtBigNumGetElement(pLeft, 0);
1125 uSubRight = (uint32_t)uRightMagn;
1126 }
1127 if (uSubLeft < uSubRight)
1128 rc = -1;
1129 else
1130 rc = uSubLeft == uSubRight ? 0 : 1;
1131#else
1132# error "Bad RTBIGNUM_ELEMENT_SIZE value"
1133#endif
1134 if (pLeft->fNegative)
1135 rc = -rc;
1136 }
1137 else
1138 rc = pLeft->fNegative ? -1 : 1;
1139 }
1140 else
1141 rc = pLeft->fNegative ? -1 : 1;
1142 rtBigNumScramble(pLeft);
1143 }
1144 return rc;
1145}
1146
1147
1148/**
1149 * Compares the magnitude values of two big numbers.
1150 *
1151 * @retval -1 if pLeft is smaller than pRight.
1152 * @retval 0 if pLeft is equal to pRight.
1153 * @retval 1 if pLeft is larger than pRight.
1154 * @param pLeft The left side number.
1155 * @param pRight The right side number.
1156 */
1157static int rtBigNumMagnitudeCompare(PCRTBIGNUM pLeft, PCRTBIGNUM pRight)
1158{
1159 Assert(!pLeft->fCurScrambled); Assert(!pRight->fCurScrambled);
1160 int rc;
1161 uint32_t i = pLeft->cUsed;
1162 if (i == pRight->cUsed)
1163 {
1164 rc = 0;
1165 while (i-- > 0)
1166 if (pLeft->pauElements[i] != pRight->pauElements[i])
1167 {
1168 rc = pLeft->pauElements[i] < pRight->pauElements[i] ? -1 : 1;
1169 break;
1170 }
1171 }
1172 else
1173 rc = i < pRight->cUsed ? -1 : 1;
1174 return rc;
1175}
1176
1177
1178/**
1179 * Copies the magnitude of on number (@a pSrc) to another (@a pBigNum).
1180 *
1181 * The variables must be unscrambled. The sign flag is not considered nor
1182 * touched.
1183 *
1184 * @returns IPRT status code.
1185 * @param pDst The destination number.
1186 * @param pSrc The source number.
1187 */
1188DECLINLINE(int) rtBigNumMagnitudeCopy(PRTBIGNUM pDst, PCRTBIGNUM pSrc)
1189{
1190 int rc = rtBigNumSetUsed(pDst, pSrc->cUsed);
1191 if (RT_SUCCESS(rc))
1192 memcpy(pDst->pauElements, pSrc->pauElements, pSrc->cUsed * RTBIGNUM_ELEMENT_SIZE);
1193 return rc;
1194}
1195
1196
1197
1198/**
1199 * Adds two magnitudes and stores them into a third.
1200 *
1201 * All variables must be unscrambled. The sign flag is not considered nor
1202 * touched.
1203 *
1204 * @returns IPRT status code.
1205 * @param pResult The resultant.
1206 * @param pAugend To whom it shall be addede.
1207 * @param pAddend The nombre to addede.
1208 */
1209static int rtBigNumMagnitudeAdd(PRTBIGNUM pResult, PCRTBIGNUM pAugend, PCRTBIGNUM pAddend)
1210{
1211 Assert(!pResult->fCurScrambled); Assert(!pAugend->fCurScrambled); Assert(!pAddend->fCurScrambled);
1212 Assert(pResult != pAugend); Assert(pResult != pAddend);
1213
1214 uint32_t cElements = RT_MAX(pAugend->cUsed, pAddend->cUsed);
1215 int rc = rtBigNumSetUsed(pResult, cElements);
1216 if (RT_SUCCESS(rc))
1217 {
1218 /*
1219 * The primitive way, requires at least two additions for each entry
1220 * without machine code help.
1221 */
1222 RTBIGNUMELEMENT fCarry = 0;
1223 for (uint32_t i = 0; i < cElements; i++)
1224 pResult->pauElements[i] = rtBigNumElementAddWithCarry(rtBigNumGetElement(pAugend, i),
1225 rtBigNumGetElement(pAddend, i),
1226 &fCarry);
1227 if (fCarry)
1228 {
1229 rc = rtBigNumSetUsed(pResult, cElements + 1);
1230 if (RT_SUCCESS(rc))
1231 pResult->pauElements[cElements++] = 1;
1232 }
1233 Assert(pResult->cUsed == cElements || RT_FAILURE_NP(rc));
1234 }
1235
1236 return rc;
1237}
1238
1239
1240/**
1241 * Substracts a smaller (or equal) magnitude from another one and stores it into
1242 * a third.
1243 *
1244 * All variables must be unscrambled. The sign flag is not considered nor
1245 * touched. For this reason, the @a pMinuend must be larger or equal to @a
1246 * pSubtrahend.
1247 *
1248 * @returns IPRT status code.
1249 * @param pResult There to store the result.
1250 * @param pMinuend What to subtract from.
1251 * @param pSubtrahend What to subtract.
1252 */
1253static int rtBigNumMagnitudeSub(PRTBIGNUM pResult, PCRTBIGNUM pMinuend, PCRTBIGNUM pSubtrahend)
1254{
1255 Assert(!pResult->fCurScrambled); Assert(!pMinuend->fCurScrambled); Assert(!pSubtrahend->fCurScrambled);
1256 Assert(pResult != pMinuend); Assert(pResult != pSubtrahend);
1257 Assert(pMinuend->cUsed >= pSubtrahend->cUsed);
1258
1259 int rc;
1260 if (pSubtrahend->cUsed)
1261 {
1262 /*
1263 * Resize the result. In the assembly case, ensure that all three arrays
1264 * has the same number of used entries, possibly with an extra zero
1265 * element on 64-bit systems.
1266 */
1267 rc = rtBigNumSetUsedEx(pResult, pMinuend->cUsed, RTBIGNUM_ZERO_ALIGN(pMinuend->cUsed));
1268#ifdef IPRT_BIGINT_WITH_ASM
1269 if (RT_SUCCESS(rc))
1270 rc = rtBigNumEnsureExtraZeroElements((PRTBIGNUM)pMinuend, RTBIGNUM_ZERO_ALIGN(pMinuend->cUsed));
1271 if (RT_SUCCESS(rc))
1272 rc = rtBigNumEnsureExtraZeroElements((PRTBIGNUM)pSubtrahend, RTBIGNUM_ZERO_ALIGN(pMinuend->cUsed));
1273#endif
1274 if (RT_SUCCESS(rc))
1275 {
1276#ifdef IPRT_BIGINT_WITH_ASM
1277 /*
1278 * Call assembly to do the work.
1279 */
1280 rtBigNumMagnitudeSubAssemblyWorker(pResult->pauElements, pMinuend->pauElements,
1281 pSubtrahend->pauElements, pMinuend->cUsed);
1282# ifdef RT_STRICT
1283 RTBIGNUMELEMENT fBorrow = 0;
1284 for (uint32_t i = 0; i < pMinuend->cUsed; i++)
1285 {
1286 RTBIGNUMELEMENT uCorrect = rtBigNumElementSubWithBorrow(pMinuend->pauElements[i], rtBigNumGetElement(pSubtrahend, i), &fBorrow);
1287 AssertMsg(pResult->pauElements[i] == uCorrect, ("[%u]=%#x, expected %#x\n", i, pResult->pauElements[i], uCorrect));
1288 }
1289# endif
1290#else
1291 /*
1292 * The primitive C way.
1293 */
1294 RTBIGNUMELEMENT fBorrow = 0;
1295 for (uint32_t i = 0; i < pMinuend->cUsed; i++)
1296 pResult->pauElements[i] = rtBigNumElementSubWithBorrow(pMinuend->pauElements[i],
1297 rtBigNumGetElement(pSubtrahend, i),
1298 &fBorrow);
1299 Assert(fBorrow == 0);
1300#endif
1301
1302 /*
1303 * Trim the result.
1304 */
1305 rtBigNumStripTrailingZeros(pResult);
1306 }
1307 }
1308 /*
1309 * Special case: Subtrahend is zero.
1310 */
1311 else
1312 rc = rtBigNumMagnitudeCopy(pResult, pMinuend);
1313
1314 return rc;
1315}
1316
1317
1318/**
1319 * Substracts a smaller (or equal) magnitude from another one and stores the
1320 * result into the first.
1321 *
1322 * All variables must be unscrambled. The sign flag is not considered nor
1323 * touched. For this reason, the @a pMinuendResult must be larger or equal to
1324 * @a pSubtrahend.
1325 *
1326 * @returns IPRT status code (memory alloc error).
1327 * @param pMinuendResult What to subtract from and return as result.
1328 * @param pSubtrahend What to subtract.
1329 */
1330static int rtBigNumMagnitudeSubThis(PRTBIGNUM pMinuendResult, PCRTBIGNUM pSubtrahend)
1331{
1332 Assert(!pMinuendResult->fCurScrambled); Assert(!pSubtrahend->fCurScrambled);
1333 Assert(pMinuendResult != pSubtrahend);
1334 Assert(pMinuendResult->cUsed >= pSubtrahend->cUsed);
1335
1336#ifdef IPRT_BIGINT_WITH_ASM
1337 /*
1338 * Use the assembly worker. Requires same sized element arrays, so zero extend them.
1339 */
1340 int rc = rtBigNumEnsureExtraZeroElements(pMinuendResult, RTBIGNUM_ZERO_ALIGN(pMinuendResult->cUsed));
1341 if (RT_SUCCESS(rc))
1342 rc = rtBigNumEnsureExtraZeroElements((PRTBIGNUM)pSubtrahend, RTBIGNUM_ZERO_ALIGN(pMinuendResult->cUsed));
1343 if (RT_FAILURE(rc))
1344 return rc;
1345 rtBigNumMagnitudeSubThisAssemblyWorker(pMinuendResult->pauElements, pSubtrahend->pauElements, pMinuendResult->cUsed);
1346#else
1347 /*
1348 * The primitive way, as usual.
1349 */
1350 RTBIGNUMELEMENT fBorrow = 0;
1351 for (uint32_t i = 0; i < pMinuendResult->cUsed; i++)
1352 pMinuendResult->pauElements[i] = rtBigNumElementSubWithBorrow(pMinuendResult->pauElements[i],
1353 rtBigNumGetElement(pSubtrahend, i),
1354 &fBorrow);
1355 Assert(fBorrow == 0);
1356#endif
1357
1358 /*
1359 * Trim the result.
1360 */
1361 rtBigNumStripTrailingZeros(pMinuendResult);
1362
1363 return VINF_SUCCESS;
1364}
1365
1366
1367RTDECL(int) RTBigNumAdd(PRTBIGNUM pResult, PCRTBIGNUM pAugend, PCRTBIGNUM pAddend)
1368{
1369 Assert(pResult != pAugend); Assert(pResult != pAddend);
1370 AssertReturn(pResult->fSensitive >= (pAugend->fSensitive | pAddend->fSensitive), VERR_BIGNUM_SENSITIVE_INPUT);
1371
1372 int rc = rtBigNumUnscramble(pResult);
1373 if (RT_SUCCESS(rc))
1374 {
1375 RTBIGNUM_ASSERT_VALID(pResult);
1376 rc = rtBigNumUnscramble((PRTBIGNUM)pAugend);
1377 if (RT_SUCCESS(rc))
1378 {
1379 RTBIGNUM_ASSERT_VALID(pAugend);
1380 rc = rtBigNumUnscramble((PRTBIGNUM)pAddend);
1381 if (RT_SUCCESS(rc))
1382 {
1383 RTBIGNUM_ASSERT_VALID(pAddend);
1384
1385 /*
1386 * Same sign: Add magnitude, keep sign.
1387 * 1 + 1 = 2
1388 * (-1) + (-1) = -2
1389 */
1390 if (pAugend->fNegative == pAddend->fNegative)
1391 {
1392 pResult->fNegative = pAugend->fNegative;
1393 rc = rtBigNumMagnitudeAdd(pResult, pAugend, pAddend);
1394 }
1395 /*
1396 * Different sign: Subtract smaller from larger, keep sign of larger.
1397 * (-5) + 3 = -2
1398 * 5 + (-3) = 2
1399 * (-1) + 3 = 2
1400 * 1 + (-3) = -2
1401 */
1402 else if (rtBigNumMagnitudeCompare(pAugend, pAddend) >= 0)
1403 {
1404 pResult->fNegative = pAugend->fNegative;
1405 rc = rtBigNumMagnitudeSub(pResult, pAugend, pAddend);
1406 if (!pResult->cUsed)
1407 pResult->fNegative = 0;
1408 }
1409 else
1410 {
1411 pResult->fNegative = pAddend->fNegative;
1412 rc = rtBigNumMagnitudeSub(pResult, pAddend, pAugend);
1413 }
1414 rtBigNumScramble((PRTBIGNUM)pAddend);
1415 }
1416 rtBigNumScramble((PRTBIGNUM)pAugend);
1417 }
1418 rtBigNumScramble(pResult);
1419 }
1420 return rc;
1421}
1422
1423
1424RTDECL(int) RTBigNumSubtract(PRTBIGNUM pResult, PCRTBIGNUM pMinuend, PCRTBIGNUM pSubtrahend)
1425{
1426 Assert(pResult != pMinuend); Assert(pResult != pSubtrahend);
1427 AssertReturn(pResult->fSensitive >= (pMinuend->fSensitive | pSubtrahend->fSensitive), VERR_BIGNUM_SENSITIVE_INPUT);
1428
1429 int rc = rtBigNumUnscramble(pResult);
1430 if (RT_SUCCESS(rc))
1431 {
1432 RTBIGNUM_ASSERT_VALID(pResult);
1433 if (pMinuend != pSubtrahend)
1434 {
1435 rc = rtBigNumUnscramble((PRTBIGNUM)pMinuend);
1436 if (RT_SUCCESS(rc))
1437 {
1438 RTBIGNUM_ASSERT_VALID(pMinuend);
1439 rc = rtBigNumUnscramble((PRTBIGNUM)pSubtrahend);
1440 if (RT_SUCCESS(rc))
1441 {
1442 RTBIGNUM_ASSERT_VALID(pSubtrahend);
1443
1444 /*
1445 * Different sign: Add magnitude, keep sign of first.
1446 * 1 - (-2) == 3
1447 * -1 - 2 == -3
1448 */
1449 if (pMinuend->fNegative != pSubtrahend->fNegative)
1450 {
1451 pResult->fNegative = pMinuend->fNegative;
1452 rc = rtBigNumMagnitudeAdd(pResult, pMinuend, pSubtrahend);
1453 }
1454 /*
1455 * Same sign, minuend has greater or equal absolute value: Subtract, keep sign of first.
1456 * 10 - 7 = 3
1457 */
1458 else if (rtBigNumMagnitudeCompare(pMinuend, pSubtrahend) >= 0)
1459 {
1460 pResult->fNegative = pMinuend->fNegative;
1461 rc = rtBigNumMagnitudeSub(pResult, pMinuend, pSubtrahend);
1462 }
1463 /*
1464 * Same sign, subtrahend is larger: Reverse and subtract, invert sign of first.
1465 * 7 - 10 = -3
1466 * -1 - (-3) = 2
1467 */
1468 else
1469 {
1470 pResult->fNegative = !pMinuend->fNegative;
1471 rc = rtBigNumMagnitudeSub(pResult, pSubtrahend, pMinuend);
1472 }
1473 rtBigNumScramble((PRTBIGNUM)pSubtrahend);
1474 }
1475 rtBigNumScramble((PRTBIGNUM)pMinuend);
1476 }
1477 }
1478 else
1479 {
1480 /* zero. */
1481 pResult->fNegative = 0;
1482 rtBigNumSetUsed(pResult, 0);
1483 }
1484 rtBigNumScramble(pResult);
1485 }
1486 return rc;
1487}
1488
1489
1490RTDECL(int) RTBigNumNegateThis(PRTBIGNUM pThis)
1491{
1492 pThis->fNegative = !pThis->fNegative;
1493 return VINF_SUCCESS;
1494}
1495
1496
1497RTDECL(int) RTBigNumNegate(PRTBIGNUM pResult, PCRTBIGNUM pBigNum)
1498{
1499 int rc = RTBigNumAssign(pResult, pBigNum);
1500 if (RT_SUCCESS(rc))
1501 rc = RTBigNumNegateThis(pResult);
1502 return rc;
1503}
1504
1505
1506/**
1507 * Multiplies the magnitudes of two values, letting the caller care about the
1508 * sign bit.
1509 *
1510 * @returns IPRT status code.
1511 * @param pResult Where to store the result.
1512 * @param pMultiplicand The first value.
1513 * @param pMultiplier The second value.
1514 */
1515static int rtBigNumMagnitudeMultiply(PRTBIGNUM pResult, PCRTBIGNUM pMultiplicand, PCRTBIGNUM pMultiplier)
1516{
1517 Assert(pResult != pMultiplicand); Assert(pResult != pMultiplier);
1518 Assert(!pResult->fCurScrambled); Assert(!pMultiplicand->fCurScrambled); Assert(!pMultiplier->fCurScrambled);
1519
1520 /*
1521 * Multiplication involving zero is zero.
1522 */
1523 if (!pMultiplicand->cUsed || !pMultiplier->cUsed)
1524 {
1525 pResult->fNegative = 0;
1526 rtBigNumSetUsed(pResult, 0);
1527 return VINF_SUCCESS;
1528 }
1529
1530 /*
1531 * Allocate a result array that is the sum of the two factors, initialize
1532 * it to zero.
1533 */
1534 uint32_t cMax = pMultiplicand->cUsed + pMultiplier->cUsed;
1535 int rc = rtBigNumSetUsed(pResult, cMax);
1536 if (RT_SUCCESS(rc))
1537 {
1538 RT_BZERO(pResult->pauElements, pResult->cUsed * RTBIGNUM_ELEMENT_SIZE);
1539
1540#ifdef IPRT_BIGINT_WITH_ASM
1541 rtBigNumMagnitudeMultiplyAssemblyWorker(pResult->pauElements,
1542 pMultiplier->pauElements, pMultiplier->cUsed,
1543 pMultiplicand->pauElements, pMultiplicand->cUsed);
1544#else
1545 for (uint32_t i = 0; i < pMultiplier->cUsed; i++)
1546 {
1547 RTBIGNUMELEMENT uMultiplier = pMultiplier->pauElements[i];
1548 for (uint32_t j = 0; j < pMultiplicand->cUsed; j++)
1549 {
1550 RTBIGNUMELEMENT uHi;
1551 RTBIGNUMELEMENT uLo;
1552#if RTBIGNUM_ELEMENT_SIZE == 4
1553 uint64_t u64 = ASMMult2xU32RetU64(pMultiplicand->pauElements[j], uMultiplier);
1554 uLo = (uint32_t)u64;
1555 uHi = u64 >> 32;
1556#elif RTBIGNUM_ELEMENT_SIZE == 8
1557 uLo = ASMMult2xU64Ret2xU64(pMultiplicand->pauElements[j], uMultiplier, &uHi);
1558#else
1559# error "Invalid RTBIGNUM_ELEMENT_SIZE value"
1560#endif
1561 RTBIGNUMELEMENT fCarry = 0;
1562 uint64_t k = i + j;
1563 pResult->pauElements[k] = rtBigNumElementAddWithCarry(pResult->pauElements[k], uLo, &fCarry);
1564 k++;
1565 pResult->pauElements[k] = rtBigNumElementAddWithCarry(pResult->pauElements[k], uHi, &fCarry);
1566 while (fCarry)
1567 {
1568 k++;
1569 pResult->pauElements[k] = rtBigNumElementAddWithCarry(pResult->pauElements[k], 0, &fCarry);
1570 }
1571 Assert(k < cMax);
1572 }
1573 }
1574#endif
1575
1576 /* It's possible we overestimated the output size by 1 element. */
1577 rtBigNumStripTrailingZeros(pResult);
1578 }
1579 return rc;
1580}
1581
1582
1583RTDECL(int) RTBigNumMultiply(PRTBIGNUM pResult, PCRTBIGNUM pMultiplicand, PCRTBIGNUM pMultiplier)
1584{
1585 Assert(pResult != pMultiplicand); Assert(pResult != pMultiplier);
1586 AssertReturn(pResult->fSensitive >= (pMultiplicand->fSensitive | pMultiplier->fSensitive), VERR_BIGNUM_SENSITIVE_INPUT);
1587
1588 int rc = rtBigNumUnscramble(pResult);
1589 if (RT_SUCCESS(rc))
1590 {
1591 RTBIGNUM_ASSERT_VALID(pResult);
1592 rc = rtBigNumUnscramble((PRTBIGNUM)pMultiplicand);
1593 if (RT_SUCCESS(rc))
1594 {
1595 RTBIGNUM_ASSERT_VALID(pMultiplicand);
1596 rc = rtBigNumUnscramble((PRTBIGNUM)pMultiplier);
1597 if (RT_SUCCESS(rc))
1598 {
1599 RTBIGNUM_ASSERT_VALID(pMultiplier);
1600
1601 /*
1602 * The sign values follow XOR rules:
1603 * -1 * 1 = -1; 1 ^ 0 = 1
1604 * 1 * -1 = -1; 1 ^ 0 = 1
1605 * -1 * -1 = 1; 1 ^ 1 = 0
1606 * 1 * 1 = 1; 0 ^ 0 = 0
1607 */
1608 pResult->fNegative = pMultiplicand->fNegative ^ pMultiplier->fNegative;
1609 rc = rtBigNumMagnitudeMultiply(pResult, pMultiplicand, pMultiplier);
1610
1611 rtBigNumScramble((PRTBIGNUM)pMultiplier);
1612 }
1613 rtBigNumScramble((PRTBIGNUM)pMultiplicand);
1614 }
1615 rtBigNumScramble(pResult);
1616 }
1617 return rc;
1618}
1619
1620
1621#if 0 /* unused */
1622/**
1623 * Clears a bit in the magnitude of @a pBigNum.
1624 *
1625 * The variables must be unscrambled.
1626 *
1627 * @param pBigNum The big number.
1628 * @param iBit The bit to clear (0-based).
1629 */
1630DECLINLINE(void) rtBigNumMagnitudeClearBit(PRTBIGNUM pBigNum, uint32_t iBit)
1631{
1632 uint32_t iElement = iBit / RTBIGNUM_ELEMENT_BITS;
1633 if (iElement < pBigNum->cUsed)
1634 {
1635 iBit &= RTBIGNUM_ELEMENT_BITS - 1;
1636 pBigNum->pauElements[iElement] &= ~RTBIGNUM_ELEMENT_BIT(iBit);
1637 if (iElement + 1 == pBigNum->cUsed && !pBigNum->pauElements[iElement])
1638 rtBigNumStripTrailingZeros(pBigNum);
1639 }
1640}
1641#endif /* unused */
1642
1643
1644/**
1645 * Sets a bit in the magnitude of @a pBigNum.
1646 *
1647 * The variables must be unscrambled.
1648 *
1649 * @returns IPRT status code.
1650 * @param pBigNum The big number.
1651 * @param iBit The bit to clear (0-based).
1652 */
1653DECLINLINE(int) rtBigNumMagnitudeSetBit(PRTBIGNUM pBigNum, uint32_t iBit)
1654{
1655 uint32_t iElement = iBit / RTBIGNUM_ELEMENT_BITS;
1656 int rc = rtBigNumEnsureElementPresent(pBigNum, iElement);
1657 if (RT_SUCCESS(rc))
1658 {
1659 iBit &= RTBIGNUM_ELEMENT_BITS - 1;
1660 pBigNum->pauElements[iElement] |= RTBIGNUM_ELEMENT_BIT(iBit);
1661 return VINF_SUCCESS;
1662 }
1663 return rc;
1664}
1665
1666
1667#if 0 /* unused */
1668/**
1669 * Writes a bit in the magnitude of @a pBigNum.
1670 *
1671 * The variables must be unscrambled.
1672 *
1673 * @returns IPRT status code.
1674 * @param pBigNum The big number.
1675 * @param iBit The bit to write (0-based).
1676 * @param fValue The bit value.
1677 */
1678DECLINLINE(int) rtBigNumMagnitudeWriteBit(PRTBIGNUM pBigNum, uint32_t iBit, bool fValue)
1679{
1680 if (fValue)
1681 return rtBigNumMagnitudeSetBit(pBigNum, iBit);
1682 rtBigNumMagnitudeClearBit(pBigNum, iBit);
1683 return VINF_SUCCESS;
1684}
1685#endif
1686
1687
1688/**
1689 * Returns the given magnitude bit.
1690 *
1691 * The variables must be unscrambled.
1692 *
1693 * @returns The bit value (1 or 0).
1694 * @param pBigNum The big number.
1695 * @param iBit The bit to return (0-based).
1696 */
1697DECLINLINE(RTBIGNUMELEMENT) rtBigNumMagnitudeGetBit(PCRTBIGNUM pBigNum, uint32_t iBit)
1698{
1699 uint32_t iElement = iBit / RTBIGNUM_ELEMENT_BITS;
1700 if (iElement < pBigNum->cUsed)
1701 {
1702 iBit &= RTBIGNUM_ELEMENT_BITS - 1;
1703 return (pBigNum->pauElements[iElement] >> iBit) & 1;
1704 }
1705 return 0;
1706}
1707
1708
1709/**
1710 * Shifts the magnitude left by one.
1711 *
1712 * The variables must be unscrambled.
1713 *
1714 * @returns IPRT status code.
1715 * @param pBigNum The big number.
1716 * @param uCarry The value to shift in at the bottom.
1717 */
1718DECLINLINE(int) rtBigNumMagnitudeShiftLeftOne(PRTBIGNUM pBigNum, RTBIGNUMELEMENT uCarry)
1719{
1720 Assert(uCarry <= 1);
1721
1722 /* Do the shifting. */
1723 uint32_t cUsed = pBigNum->cUsed;
1724#ifdef IPRT_BIGINT_WITH_ASM
1725 uCarry = rtBigNumMagnitudeShiftLeftOneAssemblyWorker(pBigNum->pauElements, cUsed, uCarry);
1726#else
1727 for (uint32_t i = 0; i < cUsed; i++)
1728 {
1729 RTBIGNUMELEMENT uTmp = pBigNum->pauElements[i];
1730 pBigNum->pauElements[i] = (uTmp << 1) | uCarry;
1731 uCarry = uTmp >> (RTBIGNUM_ELEMENT_BITS - 1);
1732 }
1733#endif
1734
1735 /* If we still carry a bit, we need to increase the size. */
1736 if (uCarry)
1737 {
1738 int rc = rtBigNumSetUsed(pBigNum, cUsed + 1);
1739 AssertRCReturn(rc, rc);
1740 pBigNum->pauElements[cUsed] = uCarry;
1741 }
1742
1743 return VINF_SUCCESS;
1744}
1745
1746
1747/**
1748 * Shifts the magnitude left by @a cBits.
1749 *
1750 * The variables must be unscrambled.
1751 *
1752 * @returns IPRT status code.
1753 * @param pResult Where to store the result.
1754 * @param pValue The value to shift.
1755 * @param cBits The shift count.
1756 */
1757static int rtBigNumMagnitudeShiftLeft(PRTBIGNUM pResult, PCRTBIGNUM pValue, uint32_t cBits)
1758{
1759 int rc;
1760 if (cBits)
1761 {
1762 uint32_t cBitsNew = rtBigNumMagnitudeBitWidth(pValue);
1763 if (cBitsNew > 0)
1764 {
1765 if (cBitsNew + cBits > cBitsNew)
1766 {
1767 cBitsNew += cBits;
1768 rc = rtBigNumSetUsedEx(pResult, 0, RT_ALIGN_32(cBitsNew, RTBIGNUM_ELEMENT_BITS) / RTBIGNUM_ELEMENT_BITS);
1769 if (RT_SUCCESS(rc))
1770 rc = rtBigNumSetUsed(pResult, RT_ALIGN_32(cBitsNew, RTBIGNUM_ELEMENT_BITS) / RTBIGNUM_ELEMENT_BITS);
1771 if (RT_SUCCESS(rc))
1772 {
1773 uint32_t const cLeft = pValue->cUsed;
1774 PCRTBIGNUMELEMENT pauSrc = pValue->pauElements;
1775 PRTBIGNUMELEMENT pauDst = pResult->pauElements;
1776
1777 Assert(ASMMemIsZero(pauDst, (cBits / RTBIGNUM_ELEMENT_BITS) * RTBIGNUM_ELEMENT_SIZE));
1778 pauDst += cBits / RTBIGNUM_ELEMENT_BITS;
1779
1780 cBits &= RTBIGNUM_ELEMENT_BITS - 1;
1781 if (cBits)
1782 {
1783 RTBIGNUMELEMENT uPrev = 0;
1784 for (uint32_t i = 0; i < cLeft; i++)
1785 {
1786 RTBIGNUMELEMENT uCur = pauSrc[i];
1787 pauDst[i] = (uCur << cBits) | (uPrev >> (RTBIGNUM_ELEMENT_BITS - cBits));
1788 uPrev = uCur;
1789 }
1790 uPrev >>= RTBIGNUM_ELEMENT_BITS - cBits;
1791 if (uPrev)
1792 pauDst[pValue->cUsed] = uPrev;
1793 }
1794 else
1795 memcpy(pauDst, pauSrc, cLeft * RTBIGNUM_ELEMENT_SIZE);
1796 }
1797 }
1798 else
1799 rc = VERR_OUT_OF_RANGE;
1800 }
1801 /* Shifting zero always yields a zero result. */
1802 else
1803 rc = rtBigNumSetUsed(pResult, 0);
1804 }
1805 else
1806 rc = rtBigNumMagnitudeCopy(pResult, pValue);
1807 return rc;
1808}
1809
1810
1811RTDECL(int) RTBigNumShiftLeft(PRTBIGNUM pResult, PCRTBIGNUM pValue, uint32_t cBits)
1812{
1813 Assert(pResult != pValue);
1814 AssertReturn(pResult->fSensitive >= pValue->fSensitive, VERR_BIGNUM_SENSITIVE_INPUT);
1815
1816 int rc = rtBigNumUnscramble(pResult);
1817 if (RT_SUCCESS(rc))
1818 {
1819 RTBIGNUM_ASSERT_VALID(pResult);
1820 rc = rtBigNumUnscramble((PRTBIGNUM)pValue);
1821 if (RT_SUCCESS(rc))
1822 {
1823 RTBIGNUM_ASSERT_VALID(pValue);
1824
1825 pResult->fNegative = pValue->fNegative;
1826 rc = rtBigNumMagnitudeShiftLeft(pResult, pValue, cBits);
1827
1828 rtBigNumScramble((PRTBIGNUM)pValue);
1829 }
1830 rtBigNumScramble(pResult);
1831 }
1832 return rc;
1833}
1834
1835
1836/**
1837 * Shifts the magnitude right by @a cBits.
1838 *
1839 * The variables must be unscrambled.
1840 *
1841 * @returns IPRT status code.
1842 * @param pResult Where to store the result.
1843 * @param pValue The value to shift.
1844 * @param cBits The shift count.
1845 */
1846static int rtBigNumMagnitudeShiftRight(PRTBIGNUM pResult, PCRTBIGNUM pValue, uint32_t cBits)
1847{
1848 int rc;
1849 if (cBits)
1850 {
1851 uint32_t cBitsNew = rtBigNumMagnitudeBitWidth(pValue);
1852 if (cBitsNew > cBits)
1853 {
1854 cBitsNew -= cBits;
1855 uint32_t cElementsNew = RT_ALIGN_32(cBitsNew, RTBIGNUM_ELEMENT_BITS) / RTBIGNUM_ELEMENT_BITS;
1856 rc = rtBigNumSetUsed(pResult, cElementsNew);
1857 if (RT_SUCCESS(rc))
1858 {
1859 uint32_t i = cElementsNew;
1860 PCRTBIGNUMELEMENT pauSrc = pValue->pauElements;
1861 PRTBIGNUMELEMENT pauDst = pResult->pauElements;
1862
1863 pauSrc += cBits / RTBIGNUM_ELEMENT_BITS;
1864
1865 cBits &= RTBIGNUM_ELEMENT_BITS - 1;
1866 if (cBits)
1867 {
1868 RTBIGNUMELEMENT uPrev = &pauSrc[i] == &pValue->pauElements[pValue->cUsed] ? 0 : pauSrc[i];
1869 while (i-- > 0)
1870 {
1871 RTBIGNUMELEMENT uCur = pauSrc[i];
1872 pauDst[i] = (uCur >> cBits) | (uPrev << (RTBIGNUM_ELEMENT_BITS - cBits));
1873 uPrev = uCur;
1874 }
1875 }
1876 else
1877 memcpy(pauDst, pauSrc, i * RTBIGNUM_ELEMENT_SIZE);
1878 }
1879 }
1880 else
1881 rc = rtBigNumSetUsed(pResult, 0);
1882 }
1883 else
1884 rc = rtBigNumMagnitudeCopy(pResult, pValue);
1885 return rc;
1886}
1887
1888
1889RTDECL(int) RTBigNumShiftRight(PRTBIGNUM pResult, PCRTBIGNUM pValue, uint32_t cBits)
1890{
1891 Assert(pResult != pValue);
1892 AssertReturn(pResult->fSensitive >= pValue->fSensitive, VERR_BIGNUM_SENSITIVE_INPUT);
1893
1894 int rc = rtBigNumUnscramble(pResult);
1895 if (RT_SUCCESS(rc))
1896 {
1897 RTBIGNUM_ASSERT_VALID(pResult);
1898 rc = rtBigNumUnscramble((PRTBIGNUM)pValue);
1899 if (RT_SUCCESS(rc))
1900 {
1901 RTBIGNUM_ASSERT_VALID(pValue);
1902
1903 pResult->fNegative = pValue->fNegative;
1904 rc = rtBigNumMagnitudeShiftRight(pResult, pValue, cBits);
1905 if (!pResult->cUsed)
1906 pResult->fNegative = 0;
1907
1908 rtBigNumScramble((PRTBIGNUM)pValue);
1909 }
1910 rtBigNumScramble(pResult);
1911 }
1912 return rc;
1913}
1914
1915
1916/**
1917 * Implements the D3 test for Qhat decrementation.
1918 *
1919 * @returns True if Qhat should be decremented.
1920 * @param puQhat Pointer to Qhat.
1921 * @param uRhat The remainder.
1922 * @param uDivisorY The penultimate divisor element.
1923 * @param uDividendJMinus2 The j-2 dividend element.
1924 */
1925DECLINLINE(bool) rtBigNumKnuthD3_ShouldDecrementQhat(RTBIGNUMELEMENT2X const *puQhat, RTBIGNUMELEMENT uRhat,
1926 RTBIGNUMELEMENT uDivisorY, RTBIGNUMELEMENT uDividendJMinus2)
1927{
1928 if (puQhat->s.Lo == RTBIGNUM_ELEMENT_MAX && puQhat->s.Hi == 0)
1929 return true;
1930#if RTBIGNUM_ELEMENT_BITS == 64
1931 RTBIGNUMELEMENT2X TmpLeft;
1932 RTUInt128MulByU64(&TmpLeft, puQhat, uDivisorY);
1933
1934 RTBIGNUMELEMENT2X TmpRight;
1935 TmpRight.s.Lo = 0;
1936 TmpRight.s.Hi = uRhat;
1937 RTUInt128AssignAddU64(&TmpRight, uDividendJMinus2);
1938
1939 if (RTUInt128Compare(&TmpLeft, &TmpRight) > 0)
1940 return true;
1941#else
1942 if (puQhat->u * uDivisorY > ((uint64_t)uRhat << 32) + uDividendJMinus2)
1943 return true;
1944#endif
1945 return false;
1946}
1947
1948
1949/**
1950 * C implementation of the D3 step of Knuth's division algorithm.
1951 *
1952 * This estimates a value Qhat that will be used as quotient "digit" (element)
1953 * at the current level of the division (j).
1954 *
1955 * @returns The Qhat value we've estimated.
1956 * @param pauDividendJN Pointer to the j+n (normalized) dividend element.
1957 * Will access up to two elements prior to this.
1958 * @param uDivZ The last element in the (normalized) divisor.
1959 * @param uDivY The penultimate element in the (normalized) divisor.
1960 */
1961DECLINLINE(RTBIGNUMELEMENT) rtBigNumKnuthD3_EstimateQhat(PCRTBIGNUMELEMENT pauDividendJN,
1962 RTBIGNUMELEMENT uDivZ, RTBIGNUMELEMENT uDivY)
1963{
1964 RTBIGNUMELEMENT2X uQhat;
1965 RTBIGNUMELEMENT uRhat;
1966 RTBIGNUMELEMENT uDividendJN = pauDividendJN[0];
1967 Assert(uDividendJN <= uDivZ);
1968 if (uDividendJN != uDivZ)
1969 rtBigNumElement2xDiv2xBy1x(&uQhat, &uRhat, uDividendJN, pauDividendJN[-1], uDivZ);
1970 else
1971 {
1972 /*
1973 * This is the case where we end up with an initial Qhat that's all Fs.
1974 */
1975 /* Calc the remainder for max Qhat value. */
1976 RTBIGNUMELEMENT2X uTmp1; /* (v[j+n] << bits) + v[J+N-1] */
1977 uTmp1.s.Hi = uDivZ;
1978 uTmp1.s.Lo = pauDividendJN[-1];
1979
1980 RTBIGNUMELEMENT2X uTmp2; /* uQhat * uDividendJN */
1981 uTmp2.s.Hi = uDivZ - 1;
1982 uTmp2.s.Lo = 0 - uDivZ;
1983#if RTBIGNUM_ELEMENT_BITS == 64
1984 RTUInt128AssignSub(&uTmp1, &uTmp2);
1985#else
1986 uTmp1.u -= uTmp2.u;
1987#endif
1988 /* If we overflowed the remainder, don't bother trying to adjust. */
1989 if (uTmp1.s.Hi)
1990 return RTBIGNUM_ELEMENT_MAX;
1991
1992 uRhat = uTmp1.s.Lo;
1993 uQhat.s.Lo = RTBIGNUM_ELEMENT_MAX;
1994 uQhat.s.Hi = 0;
1995 }
1996
1997 /*
1998 * Adjust Q to eliminate all cases where it's two to large and most cases
1999 * where it's one too large.
2000 */
2001 while (rtBigNumKnuthD3_ShouldDecrementQhat(&uQhat, uRhat, uDivY, pauDividendJN[-2]))
2002 {
2003 rtBigNumElement2xDec(&uQhat);
2004 uRhat += uDivZ;
2005 if (uRhat < uDivZ /* overflow */ || uRhat == RTBIGNUM_ELEMENT_MAX)
2006 break;
2007 }
2008
2009 return uQhat.s.Lo;
2010}
2011
2012
2013#ifdef IPRT_BIGINT_WITH_ASM
2014DECLASM(bool) rtBigNumKnuthD4_MulSub(PRTBIGNUMELEMENT pauDividendJ, PRTBIGNUMELEMENT pauDivisor,
2015 uint32_t cDivisor, RTBIGNUMELEMENT uQhat);
2016#else
2017/**
2018 * C implementation of the D4 step of Knuth's division algorithm.
2019 *
2020 * This subtracts Divisor * Qhat from the dividend at the current J index.
2021 *
2022 * @returns true if negative result (unlikely), false if positive.
2023 * @param pauDividendJ Pointer to the j-th (normalized) dividend element.
2024 * Will access up to two elements prior to this.
2025 * @param uDivZ The last element in the (normalized) divisor.
2026 * @param uDivY The penultimate element in the (normalized) divisor.
2027 */
2028DECLINLINE(bool) rtBigNumKnuthD4_MulSub(PRTBIGNUMELEMENT pauDividendJ, PRTBIGNUMELEMENT pauDivisor,
2029 uint32_t cDivisor, RTBIGNUMELEMENT uQhat)
2030{
2031 uint32_t i;
2032 bool fBorrow = false;
2033 RTBIGNUMELEMENT uMulCarry = 0;
2034 for (i = 0; i < cDivisor; i++)
2035 {
2036 RTBIGNUMELEMENT2X uSub;
2037# if RTBIGNUM_ELEMENT_BITS == 64
2038 RTUInt128MulU64ByU64(&uSub, uQhat, pauDivisor[i]);
2039 RTUInt128AssignAddU64(&uSub, uMulCarry);
2040# else
2041 uSub.u = (uint64_t)uQhat * pauDivisor[i] + uMulCarry;
2042# endif
2043 uMulCarry = uSub.s.Hi;
2044
2045 RTBIGNUMELEMENT uDividendI = pauDividendJ[i];
2046 if (!fBorrow)
2047 {
2048 fBorrow = uDividendI < uSub.s.Lo;
2049 uDividendI -= uSub.s.Lo;
2050 }
2051 else
2052 {
2053 fBorrow = uDividendI <= uSub.s.Lo;
2054 uDividendI -= uSub.s.Lo + 1;
2055 }
2056 pauDividendJ[i] = uDividendI;
2057 }
2058
2059 /* Carry and borrow into the final dividend element. */
2060 RTBIGNUMELEMENT uDividendI = pauDividendJ[i];
2061 if (!fBorrow)
2062 {
2063 fBorrow = uDividendI < uMulCarry;
2064 pauDividendJ[i] = uDividendI - uMulCarry;
2065 }
2066 else
2067 {
2068 fBorrow = uDividendI <= uMulCarry;
2069 pauDividendJ[i] = uDividendI - uMulCarry - 1;
2070 }
2071
2072 return fBorrow;
2073}
2074#endif /* !IPRT_BIGINT_WITH_ASM */
2075
2076
2077/**
2078 * C implementation of the D6 step of Knuth's division algorithm.
2079 *
2080 * This adds the divisor to the dividend to undo the negative value step D4
2081 * produced. This is not very frequent occurence.
2082 *
2083 * @param pauDividendJ Pointer to the j-th (normalized) dividend element.
2084 * Will access up to two elements prior to this.
2085 * @param pauDivisor The last element in the (normalized) divisor.
2086 * @param cDivisor The penultimate element in the (normalized) divisor.
2087 */
2088DECLINLINE(void) rtBigNumKnuthD6_AddBack(PRTBIGNUMELEMENT pauDividendJ, PRTBIGNUMELEMENT pauDivisor, uint32_t cDivisor)
2089{
2090 RTBIGNUMELEMENT2X uTmp;
2091 uTmp.s.Lo = 0;
2092
2093 uint32_t i;
2094 for (i = 0; i < cDivisor; i++)
2095 {
2096 uTmp.s.Hi = 0;
2097#if RTBIGNUM_ELEMENT_BITS == 64
2098 RTUInt128AssignAddU64(&uTmp, pauDivisor[i]);
2099 RTUInt128AssignAddU64(&uTmp, pauDividendJ[i]);
2100#else
2101 uTmp.u += pauDivisor[i];
2102 uTmp.u += pauDividendJ[i];
2103#endif
2104 pauDividendJ[i] = uTmp.s.Lo;
2105 uTmp.s.Lo = uTmp.s.Hi;
2106 }
2107
2108 /* The final dividend entry. */
2109 Assert(pauDividendJ[i] + uTmp.s.Lo < uTmp.s.Lo);
2110 pauDividendJ[i] += uTmp.s.Lo;
2111}
2112
2113
2114/**
2115 * Knuth's division (core).
2116 *
2117 * @returns IPRT status code.
2118 * @param pQuotient Where to return the quotient. Can be NULL.
2119 * @param pRemainder Where to return the remainder.
2120 * @param pDividend What to divide.
2121 * @param pDivisor What to divide by.
2122 */
2123static int rtBigNumMagnitudeDivideKnuth(PRTBIGNUM pQuotient, PRTBIGNUM pRemainder, PCRTBIGNUM pDividend, PCRTBIGNUM pDivisor)
2124{
2125 Assert(pDivisor->cUsed > 1);
2126 uint32_t const cDivisor = pDivisor->cUsed;
2127 Assert(pDividend->cUsed >= cDivisor);
2128
2129 /*
2130 * Make sure we've got enough space in the quotient, so we can build it
2131 * without any trouble come step D5.
2132 */
2133 int rc;
2134 if (pQuotient)
2135 {
2136 rc = rtBigNumSetUsedEx(pQuotient, 0, pDividend->cUsed - cDivisor + 1);
2137 if (RT_SUCCESS(rc))
2138 rc = rtBigNumSetUsed(pQuotient, pDividend->cUsed - cDivisor + 1);
2139 if (RT_FAILURE(rc))
2140 return rc;
2141 }
2142
2143 /*
2144 * D1. Normalize. The goal here is to make sure the last element in the
2145 * divisor is greater than RTBIGNUMELEMENTS_MAX/2. We must also make sure
2146 * we can access element pDividend->cUsed of the normalized dividend.
2147 */
2148 RTBIGNUM NormDividend;
2149 RTBIGNUM NormDivisor;
2150 PCRTBIGNUM pNormDivisor = &NormDivisor;
2151 rtBigNumInitZeroTemplate(&NormDivisor, pDividend);
2152
2153 uint32_t cNormShift = (RTBIGNUM_ELEMENT_BITS - rtBigNumMagnitudeBitWidth(pDivisor)) & (RTBIGNUM_ELEMENT_BITS - 1);
2154 if (cNormShift)
2155 {
2156 rtBigNumInitZeroTemplate(&NormDividend, pDividend);
2157 rc = rtBigNumMagnitudeShiftLeft(&NormDividend, pDividend, cNormShift);
2158 if (RT_SUCCESS(rc))
2159 rc = rtBigNumMagnitudeShiftLeft(&NormDivisor, pDivisor, cNormShift);
2160 }
2161 else
2162 {
2163 pNormDivisor = pDivisor;
2164 rc = rtBigNumCloneInternal(&NormDividend, pDividend);
2165 }
2166 if (RT_SUCCESS(rc) && pDividend->cUsed == NormDividend.cUsed)
2167 rc = rtBigNumEnsureExtraZeroElements(&NormDividend, NormDividend.cUsed + 1);
2168 if (RT_SUCCESS(rc))
2169 {
2170 /*
2171 * D2. Initialize the j index so we can loop thru the elements in the
2172 * dividend that makes it larger than the divisor.
2173 */
2174 uint32_t j = pDividend->cUsed - cDivisor;
2175
2176 RTBIGNUMELEMENT const DivZ = pNormDivisor->pauElements[cDivisor - 1];
2177 RTBIGNUMELEMENT const DivY = pNormDivisor->pauElements[cDivisor - 2];
2178 for (;;)
2179 {
2180 /*
2181 * D3. Estimate a Q' by dividing the j and j-1 dividen elements by
2182 * the last divisor element, then adjust against the next elements.
2183 */
2184 RTBIGNUMELEMENT uQhat = rtBigNumKnuthD3_EstimateQhat(&NormDividend.pauElements[j + cDivisor], DivZ, DivY);
2185
2186 /*
2187 * D4. Multiply and subtract.
2188 */
2189 bool fNegative = rtBigNumKnuthD4_MulSub(&NormDividend.pauElements[j], pNormDivisor->pauElements, cDivisor, uQhat);
2190
2191 /*
2192 * D5. Test remainder.
2193 * D6. Add back.
2194 */
2195 if (fNegative)
2196 {
2197//__debugbreak();
2198 rtBigNumKnuthD6_AddBack(&NormDividend.pauElements[j], pNormDivisor->pauElements, cDivisor);
2199 uQhat--;
2200 }
2201
2202 if (pQuotient)
2203 pQuotient->pauElements[j] = uQhat;
2204
2205 /*
2206 * D7. Loop on j.
2207 */
2208 if (j == 0)
2209 break;
2210 j--;
2211 }
2212
2213 /*
2214 * D8. Unnormalize the remainder.
2215 */
2216 rtBigNumStripTrailingZeros(&NormDividend);
2217 if (cNormShift)
2218 rc = rtBigNumMagnitudeShiftRight(pRemainder, &NormDividend, cNormShift);
2219 else
2220 rc = rtBigNumMagnitudeCopy(pRemainder, &NormDividend);
2221 if (pQuotient)
2222 rtBigNumStripTrailingZeros(pQuotient);
2223 }
2224
2225 /*
2226 * Delete temporary variables.
2227 */
2228 RTBigNumDestroy(&NormDividend);
2229 if (pDivisor == &NormDivisor)
2230 RTBigNumDestroy(&NormDivisor);
2231 return rc;
2232}
2233
2234
2235static int rtBigNumMagnitudeDivideSlowLong(PRTBIGNUM pQuotient, PRTBIGNUM pRemainder, PCRTBIGNUM pDividend, PCRTBIGNUM pDivisor)
2236{
2237 /*
2238 * Do very simple long division. This ain't fast, but it does the trick.
2239 */
2240 int rc = VINF_SUCCESS;
2241 uint32_t iBit = rtBigNumMagnitudeBitWidth(pDividend);
2242 while (iBit-- > 0)
2243 {
2244 rc = rtBigNumMagnitudeShiftLeftOne(pRemainder, rtBigNumMagnitudeGetBit(pDividend, iBit));
2245 AssertRCBreak(rc);
2246 int iDiff = rtBigNumMagnitudeCompare(pRemainder, pDivisor);
2247 if (iDiff >= 0)
2248 {
2249 if (iDiff != 0)
2250 {
2251 rc = rtBigNumMagnitudeSubThis(pRemainder, pDivisor);
2252 AssertRCBreak(rc);
2253 }
2254 else
2255 rtBigNumSetUsed(pRemainder, 0);
2256 rc = rtBigNumMagnitudeSetBit(pQuotient, iBit);
2257 AssertRCBreak(rc);
2258 }
2259 }
2260
2261 /* This shouldn't be necessary. */
2262 rtBigNumStripTrailingZeros(pQuotient);
2263 rtBigNumStripTrailingZeros(pRemainder);
2264
2265 return rc;
2266}
2267
2268
2269/**
2270 * Divides the magnitudes of two values, letting the caller care about the sign
2271 * bit.
2272 *
2273 * All variables must be unscrambled. The sign flag is not considered nor
2274 * touched, this means the caller have to check for zero outputs.
2275 *
2276 * @returns IPRT status code.
2277 * @param pQuotient Where to return the quotient.
2278 * @param pRemainder Where to return the remainder.
2279 * @param pDividend What to divide.
2280 * @param pDivisor What to divide by.
2281 * @param fForceLong Force long division.
2282 */
2283static int rtBigNumMagnitudeDivide(PRTBIGNUM pQuotient, PRTBIGNUM pRemainder, PCRTBIGNUM pDividend, PCRTBIGNUM pDivisor,
2284 bool fForceLong)
2285{
2286 Assert(pQuotient != pDividend); Assert(pQuotient != pDivisor); Assert(pRemainder != pDividend); Assert(pRemainder != pDivisor); Assert(pRemainder != pQuotient);
2287 Assert(!pQuotient->fCurScrambled); Assert(!pRemainder->fCurScrambled); Assert(!pDividend->fCurScrambled); Assert(!pDivisor->fCurScrambled);
2288
2289 /*
2290 * Just set both output values to zero as that's the return for several
2291 * special case and the initial state of the general case.
2292 */
2293 rtBigNumSetUsed(pQuotient, 0);
2294 rtBigNumSetUsed(pRemainder, 0);
2295
2296 /*
2297 * Dividing something by zero is undefined.
2298 * Diving zero by something is zero, unless the divsor is also zero.
2299 */
2300 if (!pDivisor->cUsed || !pDividend->cUsed)
2301 return pDivisor->cUsed ? VINF_SUCCESS : VERR_BIGNUM_DIV_BY_ZERO;
2302
2303 /*
2304 * Dividing by one? Quotient = dividend, no remainder.
2305 */
2306 if (pDivisor->cUsed == 1 && pDivisor->pauElements[0] == 1)
2307 return rtBigNumMagnitudeCopy(pQuotient, pDividend);
2308
2309 /*
2310 * Dividend smaller than the divisor. Zero quotient, all divisor.
2311 */
2312 int iDiff = rtBigNumMagnitudeCompare(pDividend, pDivisor);
2313 if (iDiff < 0)
2314 return rtBigNumMagnitudeCopy(pRemainder, pDividend);
2315
2316 /*
2317 * Since we already have done the compare, check if the two values are the
2318 * same. The result is 1 and no remainder then.
2319 */
2320 if (iDiff == 0)
2321 {
2322 int rc = rtBigNumSetUsed(pQuotient, 1);
2323 if (RT_SUCCESS(rc))
2324 pQuotient->pauElements[0] = 1;
2325 return rc;
2326 }
2327
2328 /*
2329 * Sort out special cases before going to the preferred or select algorithm.
2330 */
2331 int rc;
2332 if (pDividend->cUsed <= 2 && !fForceLong)
2333 {
2334 if (pDividend->cUsed < 2)
2335 {
2336 /*
2337 * Single element division.
2338 */
2339 RTBIGNUMELEMENT uQ = pDividend->pauElements[0] / pDivisor->pauElements[0];
2340 RTBIGNUMELEMENT uR = pDividend->pauElements[0] % pDivisor->pauElements[0];
2341 rc = VINF_SUCCESS;
2342 if (uQ)
2343 {
2344 rc = rtBigNumSetUsed(pQuotient, 1);
2345 if (RT_SUCCESS(rc))
2346 pQuotient->pauElements[0] = uQ;
2347 }
2348 if (uR && RT_SUCCESS(rc))
2349 {
2350 rc = rtBigNumSetUsed(pRemainder, 1);
2351 if (RT_SUCCESS(rc))
2352 pRemainder->pauElements[0] = uR;
2353 }
2354 }
2355 else
2356 {
2357 /*
2358 * Two elements dividend by a one or two element divisor.
2359 */
2360 RTBIGNUMELEMENT2X uQ, uR;
2361 if (pDivisor->cUsed == 1)
2362 {
2363 rtBigNumElement2xDiv2xBy1x(&uQ, &uR.s.Lo, pDividend->pauElements[1], pDividend->pauElements[0],
2364 pDivisor->pauElements[0]);
2365 uR.s.Hi = 0;
2366 }
2367 else
2368 rtBigNumElement2xDiv(&uQ, &uR, pDividend->pauElements[1], pDividend->pauElements[0],
2369 pDivisor->pauElements[1], pDivisor->pauElements[0]);
2370 rc = rtBigNumElement2xCopyToMagnitude(&uQ, pQuotient);
2371 if (RT_SUCCESS(rc))
2372 rc = rtBigNumElement2xCopyToMagnitude(&uR, pRemainder);
2373 }
2374 }
2375 /*
2376 * Decide upon which algorithm to use. Knuth requires a divisor that's at
2377 * least 2 elements big.
2378 */
2379 else if (pDivisor->cUsed < 2 || fForceLong)
2380 rc = rtBigNumMagnitudeDivideSlowLong(pQuotient, pRemainder, pDividend, pDivisor);
2381 else
2382 rc = rtBigNumMagnitudeDivideKnuth(pQuotient, pRemainder, pDividend, pDivisor);
2383 return rc;
2384}
2385
2386
2387static int rtBigNumDivideCommon(PRTBIGNUM pQuotient, PRTBIGNUM pRemainder,
2388 PCRTBIGNUM pDividend, PCRTBIGNUM pDivisor, bool fForceLong)
2389{
2390 Assert(pQuotient != pDividend); Assert(pQuotient != pDivisor); Assert(pRemainder != pDividend); Assert(pRemainder != pDivisor); Assert(pRemainder != pQuotient);
2391 AssertReturn(pQuotient->fSensitive >= (pDividend->fSensitive | pDivisor->fSensitive), VERR_BIGNUM_SENSITIVE_INPUT);
2392 AssertReturn(pRemainder->fSensitive >= (pDividend->fSensitive | pDivisor->fSensitive), VERR_BIGNUM_SENSITIVE_INPUT);
2393
2394 int rc = rtBigNumUnscramble(pQuotient);
2395 if (RT_SUCCESS(rc))
2396 {
2397 RTBIGNUM_ASSERT_VALID(pQuotient);
2398 rc = rtBigNumUnscramble(pRemainder);
2399 if (RT_SUCCESS(rc))
2400 {
2401 RTBIGNUM_ASSERT_VALID(pRemainder);
2402 rc = rtBigNumUnscramble((PRTBIGNUM)pDividend);
2403 if (RT_SUCCESS(rc))
2404 {
2405 RTBIGNUM_ASSERT_VALID(pDividend);
2406 rc = rtBigNumUnscramble((PRTBIGNUM)pDivisor);
2407 if (RT_SUCCESS(rc))
2408 {
2409 RTBIGNUM_ASSERT_VALID(pDivisor);
2410
2411 /*
2412 * The sign value of the remainder is the same as the dividend.
2413 * The sign values of the quotient follow XOR rules, just like multiplication:
2414 * -3 / 2 = -1; r=-1; 1 ^ 0 = 1
2415 * 3 / -2 = -1; r= 1; 1 ^ 0 = 1
2416 * -3 / -2 = 1; r=-1; 1 ^ 1 = 0
2417 * 3 / 2 = 1; r= 1; 0 ^ 0 = 0
2418 */
2419 pQuotient->fNegative = pDividend->fNegative ^ pDivisor->fNegative;
2420 pRemainder->fNegative = pDividend->fNegative;
2421
2422 rc = rtBigNumMagnitudeDivide(pQuotient, pRemainder, pDividend, pDivisor, fForceLong);
2423
2424 if (pQuotient->cUsed == 0)
2425 pQuotient->fNegative = 0;
2426 if (pRemainder->cUsed == 0)
2427 pRemainder->fNegative = 0;
2428
2429 rtBigNumScramble((PRTBIGNUM)pDivisor);
2430 }
2431 rtBigNumScramble((PRTBIGNUM)pDividend);
2432 }
2433 rtBigNumScramble(pRemainder);
2434 }
2435 rtBigNumScramble(pQuotient);
2436 }
2437 return rc;
2438}
2439
2440
2441RTDECL(int) RTBigNumDivide(PRTBIGNUM pQuotient, PRTBIGNUM pRemainder, PCRTBIGNUM pDividend, PCRTBIGNUM pDivisor)
2442{
2443 return rtBigNumDivideCommon(pQuotient, pRemainder, pDividend, pDivisor, false /*fForceLong*/);
2444}
2445
2446
2447RTDECL(int) RTBigNumDivideLong(PRTBIGNUM pQuotient, PRTBIGNUM pRemainder, PCRTBIGNUM pDividend, PCRTBIGNUM pDivisor)
2448{
2449 return rtBigNumDivideCommon(pQuotient, pRemainder, pDividend, pDivisor, true /*fForceLong*/);
2450}
2451
2452
2453/**
2454 * Calculates the modulus of a magnitude value, leaving the sign bit to the
2455 * caller.
2456 *
2457 * All variables must be unscrambled. The sign flag is not considered nor
2458 * touched, this means the caller have to check for zero outputs.
2459 *
2460 * @returns IPRT status code.
2461 * @param pRemainder Where to return the remainder.
2462 * @param pDividend What to divide.
2463 * @param pDivisor What to divide by.
2464 */
2465static int rtBigNumMagnitudeModulo(PRTBIGNUM pRemainder, PCRTBIGNUM pDividend, PCRTBIGNUM pDivisor)
2466{
2467 Assert(pRemainder != pDividend); Assert(pRemainder != pDivisor);
2468 Assert(!pRemainder->fCurScrambled); Assert(!pDividend->fCurScrambled); Assert(!pDivisor->fCurScrambled);
2469
2470 /*
2471 * Just set the output value to zero as that's the return for several
2472 * special case and the initial state of the general case.
2473 */
2474 rtBigNumSetUsed(pRemainder, 0);
2475
2476 /*
2477 * Dividing something by zero is undefined.
2478 * Diving zero by something is zero, unless the divsor is also zero.
2479 */
2480 if (!pDivisor->cUsed || !pDividend->cUsed)
2481 return pDivisor->cUsed ? VINF_SUCCESS : VERR_BIGNUM_DIV_BY_ZERO;
2482
2483 /*
2484 * Dividing by one? Quotient = dividend, no remainder.
2485 */
2486 if (pDivisor->cUsed == 1 && pDivisor->pauElements[0] == 1)
2487 return VINF_SUCCESS;
2488
2489 /*
2490 * Dividend smaller than the divisor. Zero quotient, all divisor.
2491 */
2492 int iDiff = rtBigNumMagnitudeCompare(pDividend, pDivisor);
2493 if (iDiff < 0)
2494 return rtBigNumMagnitudeCopy(pRemainder, pDividend);
2495
2496 /*
2497 * Since we already have done the compare, check if the two values are the
2498 * same. The result is 1 and no remainder then.
2499 */
2500 if (iDiff == 0)
2501 return VINF_SUCCESS;
2502
2503 /** @todo optimize small numbers. */
2504 int rc = VINF_SUCCESS;
2505 if (pDivisor->cUsed < 2)
2506 {
2507 /*
2508 * Do very simple long division. This ain't fast, but it does the trick.
2509 */
2510 uint32_t iBit = rtBigNumMagnitudeBitWidth(pDividend);
2511 while (iBit-- > 0)
2512 {
2513 rc = rtBigNumMagnitudeShiftLeftOne(pRemainder, rtBigNumMagnitudeGetBit(pDividend, iBit));
2514 AssertRCBreak(rc);
2515 iDiff = rtBigNumMagnitudeCompare(pRemainder, pDivisor);
2516 if (iDiff >= 0)
2517 {
2518 if (iDiff != 0)
2519 {
2520 rc = rtBigNumMagnitudeSubThis(pRemainder, pDivisor);
2521 AssertRCBreak(rc);
2522 }
2523 else
2524 rtBigNumSetUsed(pRemainder, 0);
2525 }
2526 }
2527 }
2528 else
2529 {
2530 /*
2531 * Join paths with division.
2532 */
2533 rc = rtBigNumMagnitudeDivideKnuth(NULL, pRemainder, pDividend, pDivisor);
2534 }
2535
2536 /* This shouldn't be necessary. */
2537 rtBigNumStripTrailingZeros(pRemainder);
2538 return rc;
2539}
2540
2541
2542RTDECL(int) RTBigNumModulo(PRTBIGNUM pRemainder, PCRTBIGNUM pDividend, PCRTBIGNUM pDivisor)
2543{
2544 Assert(pRemainder != pDividend); Assert(pRemainder != pDivisor);
2545 AssertReturn(pRemainder->fSensitive >= (pDividend->fSensitive | pDivisor->fSensitive), VERR_BIGNUM_SENSITIVE_INPUT);
2546
2547 int rc = rtBigNumUnscramble(pRemainder);
2548 if (RT_SUCCESS(rc))
2549 {
2550 RTBIGNUM_ASSERT_VALID(pRemainder);
2551 rc = rtBigNumUnscramble((PRTBIGNUM)pDividend);
2552 if (RT_SUCCESS(rc))
2553 {
2554 RTBIGNUM_ASSERT_VALID(pDividend);
2555 rc = rtBigNumUnscramble((PRTBIGNUM)pDivisor);
2556 if (RT_SUCCESS(rc))
2557 {
2558 RTBIGNUM_ASSERT_VALID(pDivisor);
2559
2560 /*
2561 * The sign value of the remainder is the same as the dividend.
2562 */
2563 pRemainder->fNegative = pDividend->fNegative;
2564
2565 rc = rtBigNumMagnitudeModulo(pRemainder, pDividend, pDivisor);
2566
2567 if (pRemainder->cUsed == 0)
2568 pRemainder->fNegative = 0;
2569
2570 rtBigNumScramble((PRTBIGNUM)pDivisor);
2571 }
2572 rtBigNumScramble((PRTBIGNUM)pDividend);
2573 }
2574 rtBigNumScramble(pRemainder);
2575 }
2576 return rc;
2577}
2578
2579
2580
2581/**
2582 * Exponentiate the magnitude.
2583 *
2584 * All variables must be unscrambled. The sign flag is not considered nor
2585 * touched, this means the caller have to reject negative exponents.
2586 *
2587 * @returns IPRT status code.
2588 * @param pResult Where to return power.
2589 * @param pBase The base value.
2590 * @param pExponent The exponent (assumed positive or zero).
2591 */
2592static int rtBigNumMagnitudeExponentiate(PRTBIGNUM pResult, PCRTBIGNUM pBase, PCRTBIGNUM pExponent)
2593{
2594 Assert(pResult != pBase); Assert(pResult != pExponent);
2595 Assert(!pResult->fCurScrambled); Assert(!pBase->fCurScrambled); Assert(!pExponent->fCurScrambled);
2596
2597 /*
2598 * A couple of special cases.
2599 */
2600 int rc;
2601 /* base ^ 0 => 1. */
2602 if (pExponent->cUsed == 0)
2603 {
2604 rc = rtBigNumSetUsed(pResult, 1);
2605 if (RT_SUCCESS(rc))
2606 pResult->pauElements[0] = 1;
2607 return rc;
2608 }
2609
2610 /* base ^ 1 => base. */
2611 if (pExponent->cUsed == 1 && pExponent->pauElements[0] == 1)
2612 return rtBigNumMagnitudeCopy(pResult, pBase);
2613
2614 /*
2615 * Set up.
2616 */
2617 /* Init temporary power-of-two variable to base. */
2618 RTBIGNUM Pow2;
2619 rc = rtBigNumCloneInternal(&Pow2, pBase);
2620 if (RT_SUCCESS(rc))
2621 {
2622 /* Init result to 1. */
2623 rc = rtBigNumSetUsed(pResult, 1);
2624 if (RT_SUCCESS(rc))
2625 {
2626 pResult->pauElements[0] = 1;
2627
2628 /* Make a temporary variable that we can use for temporary storage of the result. */
2629 RTBIGNUM TmpMultiplicand;
2630 rc = rtBigNumCloneInternal(&TmpMultiplicand, pResult);
2631 if (RT_SUCCESS(rc))
2632 {
2633 /*
2634 * Exponentiation by squaring. Reduces the number of
2635 * multiplications to: NumBitsSet(Exponent) + BitWidth(Exponent).
2636 */
2637 uint32_t const cExpBits = rtBigNumMagnitudeBitWidth(pExponent);
2638 uint32_t iBit = 0;
2639 for (;;)
2640 {
2641 if (rtBigNumMagnitudeGetBit(pExponent, iBit) != 0)
2642 {
2643 rc = rtBigNumMagnitudeCopy(&TmpMultiplicand, pResult);
2644 if (RT_SUCCESS(rc))
2645 rc = rtBigNumMagnitudeMultiply(pResult, &TmpMultiplicand, &Pow2);
2646 if (RT_FAILURE(rc))
2647 break;
2648 }
2649
2650 /* Done? */
2651 iBit++;
2652 if (iBit >= cExpBits)
2653 break;
2654
2655 /* Not done yet, square the base again. */
2656 rc = rtBigNumMagnitudeCopy(&TmpMultiplicand, &Pow2);
2657 if (RT_SUCCESS(rc))
2658 rc = rtBigNumMagnitudeMultiply(&Pow2, &TmpMultiplicand, &TmpMultiplicand);
2659 if (RT_FAILURE(rc))
2660 break;
2661 }
2662 }
2663 }
2664 RTBigNumDestroy(&Pow2);
2665 }
2666 return rc;
2667}
2668
2669
2670RTDECL(int) RTBigNumExponentiate(PRTBIGNUM pResult, PCRTBIGNUM pBase, PCRTBIGNUM pExponent)
2671{
2672 Assert(pResult != pBase); Assert(pResult != pExponent);
2673 AssertReturn(pResult->fSensitive >= (pBase->fSensitive | pExponent->fSensitive), VERR_BIGNUM_SENSITIVE_INPUT);
2674
2675 int rc = rtBigNumUnscramble(pResult);
2676 if (RT_SUCCESS(rc))
2677 {
2678 RTBIGNUM_ASSERT_VALID(pResult);
2679 rc = rtBigNumUnscramble((PRTBIGNUM)pBase);
2680 if (RT_SUCCESS(rc))
2681 {
2682 RTBIGNUM_ASSERT_VALID(pBase);
2683 rc = rtBigNumUnscramble((PRTBIGNUM)pExponent);
2684 if (RT_SUCCESS(rc))
2685 {
2686 RTBIGNUM_ASSERT_VALID(pExponent);
2687 if (!pExponent->fNegative)
2688 {
2689 pResult->fNegative = pBase->fNegative; /* sign unchanged. */
2690 rc = rtBigNumMagnitudeExponentiate(pResult, pBase, pExponent);
2691 }
2692 else
2693 rc = VERR_BIGNUM_NEGATIVE_EXPONENT;
2694
2695 rtBigNumScramble((PRTBIGNUM)pExponent);
2696 }
2697 rtBigNumScramble((PRTBIGNUM)pBase);
2698 }
2699 rtBigNumScramble(pResult);
2700 }
2701 return rc;
2702}
2703
2704
2705/**
2706 * Modular exponentiation, magnitudes only.
2707 *
2708 * All variables must be unscrambled. The sign flag is not considered nor
2709 * touched, this means the caller have to reject negative exponents and do any
2710 * other necessary sign bit fiddling.
2711 *
2712 * @returns IPRT status code.
2713 * @param pResult Where to return the remainder of the power.
2714 * @param pBase The base value.
2715 * @param pExponent The exponent (assumed positive or zero).
2716 * @param pModulus The modulus value (or divisor if you like).
2717 */
2718static int rtBigNumMagnitudeModExp(PRTBIGNUM pResult, PRTBIGNUM pBase, PRTBIGNUM pExponent, PRTBIGNUM pModulus)
2719{
2720 Assert(pResult != pBase); Assert(pResult != pBase); Assert(pResult != pExponent); Assert(pResult != pModulus);
2721 Assert(!pResult->fCurScrambled); Assert(!pBase->fCurScrambled); Assert(!pExponent->fCurScrambled); Assert(!pModulus->fCurScrambled);
2722 int rc;
2723
2724 /*
2725 * Check some special cases to get them out of the way.
2726 */
2727 /* Div by 0 => invalid. */
2728 if (pModulus->cUsed == 0)
2729 return VERR_BIGNUM_DIV_BY_ZERO;
2730
2731 /* Div by 1 => no remainder. */
2732 if (pModulus->cUsed == 1 && pModulus->pauElements[0] == 1)
2733 {
2734 rtBigNumSetUsed(pResult, 0);
2735 return VINF_SUCCESS;
2736 }
2737
2738 /* base ^ 0 => 1. */
2739 if (pExponent->cUsed == 0)
2740 {
2741 rc = rtBigNumSetUsed(pResult, 1);
2742 if (RT_SUCCESS(rc))
2743 pResult->pauElements[0] = 1;
2744 return rc;
2745 }
2746
2747 /* base ^ 1 => base. */
2748 if (pExponent->cUsed == 1 && pExponent->pauElements[0] == 1)
2749 return rtBigNumMagnitudeModulo(pResult, pBase, pModulus);
2750
2751 /*
2752 * Set up.
2753 */
2754 /* Result = 1; preallocate space for the result while at it. */
2755 rc = rtBigNumSetUsed(pResult, pModulus->cUsed + 1);
2756 if (RT_SUCCESS(rc))
2757 rc = rtBigNumSetUsed(pResult, 1);
2758 if (RT_SUCCESS(rc))
2759 {
2760 pResult->pauElements[0] = 1;
2761
2762 /* ModBase = pBase or pBase % pModulus depending on the difference in size. */
2763 RTBIGNUM Pow2;
2764 if (pBase->cUsed <= pModulus->cUsed + pModulus->cUsed / 2)
2765 rc = rtBigNumCloneInternal(&Pow2, pBase);
2766 else
2767 rc = rtBigNumMagnitudeModulo(rtBigNumInitZeroTemplate(&Pow2, pBase), pBase, pModulus);
2768
2769 /* Need a couple of temporary variables. */
2770 RTBIGNUM TmpMultiplicand;
2771 rtBigNumInitZeroTemplate(&TmpMultiplicand, pResult);
2772
2773 RTBIGNUM TmpProduct;
2774 rtBigNumInitZeroTemplate(&TmpProduct, pResult);
2775
2776 /*
2777 * We combine the exponentiation by squaring with the fact that:
2778 * (a*b) mod n = ( (a mod n) * (b mod n) ) mod n
2779 *
2780 * Thus, we can reduce the size of intermediate results by mod'ing them
2781 * in each step.
2782 */
2783 uint32_t const cExpBits = rtBigNumMagnitudeBitWidth(pExponent);
2784 uint32_t iBit = 0;
2785 for (;;)
2786 {
2787 if (rtBigNumMagnitudeGetBit(pExponent, iBit) != 0)
2788 {
2789 rc = rtBigNumMagnitudeCopy(&TmpMultiplicand, pResult);
2790 if (RT_SUCCESS(rc))
2791 rc = rtBigNumMagnitudeMultiply(&TmpProduct, &TmpMultiplicand, &Pow2);
2792 if (RT_SUCCESS(rc))
2793 rc = rtBigNumMagnitudeModulo(pResult, &TmpProduct, pModulus);
2794 if (RT_FAILURE(rc))
2795 break;
2796 }
2797
2798 /* Done? */
2799 iBit++;
2800 if (iBit >= cExpBits)
2801 break;
2802
2803 /* Not done yet, square and mod the base again. */
2804 rc = rtBigNumMagnitudeCopy(&TmpMultiplicand, &Pow2);
2805 if (RT_SUCCESS(rc))
2806 rc = rtBigNumMagnitudeMultiply(&TmpProduct, &TmpMultiplicand, &TmpMultiplicand);
2807 if (RT_SUCCESS(rc))
2808 rc = rtBigNumMagnitudeModulo(&Pow2, &TmpProduct, pModulus);
2809 if (RT_FAILURE(rc))
2810 break;
2811 }
2812
2813 RTBigNumDestroy(&TmpMultiplicand);
2814 RTBigNumDestroy(&TmpProduct);
2815 RTBigNumDestroy(&Pow2);
2816 }
2817 return rc;
2818}
2819
2820
2821RTDECL(int) RTBigNumModExp(PRTBIGNUM pResult, PRTBIGNUM pBase, PRTBIGNUM pExponent, PRTBIGNUM pModulus)
2822{
2823 Assert(pResult != pBase); Assert(pResult != pBase); Assert(pResult != pExponent); Assert(pResult != pModulus);
2824 AssertReturn(pResult->fSensitive >= (pBase->fSensitive | pExponent->fSensitive | pModulus->fSensitive),
2825 VERR_BIGNUM_SENSITIVE_INPUT);
2826
2827 int rc = rtBigNumUnscramble(pResult);
2828 if (RT_SUCCESS(rc))
2829 {
2830 RTBIGNUM_ASSERT_VALID(pResult);
2831 rc = rtBigNumUnscramble((PRTBIGNUM)pBase);
2832 if (RT_SUCCESS(rc))
2833 {
2834 RTBIGNUM_ASSERT_VALID(pBase);
2835 rc = rtBigNumUnscramble((PRTBIGNUM)pExponent);
2836 if (RT_SUCCESS(rc))
2837 {
2838 RTBIGNUM_ASSERT_VALID(pExponent);
2839 rc = rtBigNumUnscramble((PRTBIGNUM)pModulus);
2840 if (RT_SUCCESS(rc))
2841 {
2842 RTBIGNUM_ASSERT_VALID(pModulus);
2843 if (!pExponent->fNegative)
2844 {
2845 pResult->fNegative = pModulus->fNegative; /* pBase ^ pExponent / pModulus; result = remainder. */
2846 rc = rtBigNumMagnitudeModExp(pResult, pBase, pExponent, pModulus);
2847 }
2848 else
2849 rc = VERR_BIGNUM_NEGATIVE_EXPONENT;
2850 rtBigNumScramble((PRTBIGNUM)pModulus);
2851 }
2852 rtBigNumScramble((PRTBIGNUM)pExponent);
2853 }
2854 rtBigNumScramble((PRTBIGNUM)pBase);
2855 }
2856 rtBigNumScramble(pResult);
2857 }
2858 return rc;
2859}
2860
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette