VirtualBox

source: vbox/trunk/include/iprt/uint128.h@ 62288

Last change on this file since 62288 was 60482, checked in by vboxsync, 9 years ago

iprt/uint64.h: Converted uint128.h to RTUINT64U 64-bit for ancient compilers.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 39.2 KB
Line 
1/** @file
2 * IPRT - RTUINT128U & uint128_t methods.
3 */
4
5/*
6 * Copyright (C) 2011-2015 Oracle Corporation
7 *
8 * This file is part of VirtualBox Open Source Edition (OSE), as
9 * available from http://www.virtualbox.org. This file is free software;
10 * you can redistribute it and/or modify it under the terms of the GNU
11 * General Public License (GPL) as published by the Free Software
12 * Foundation, in version 2 as it comes in the "COPYING" file of the
13 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15 *
16 * The contents of this file may alternatively be used under the terms
17 * of the Common Development and Distribution License Version 1.0
18 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
19 * VirtualBox OSE distribution, in which case the provisions of the
20 * CDDL are applicable instead of those of the GPL.
21 *
22 * You may elect to license modified versions of this file under the
23 * terms and conditions of either the GPL or the CDDL or both.
24 */
25
26#ifndef ___iprt_uint128_h
27#define ___iprt_uint128_h
28
29#include <iprt/cdefs.h>
30#include <iprt/types.h>
31#include <iprt/asm.h>
32#ifdef RT_ARCH_AMD64
33# include <iprt/asm-math.h>
34#endif
35
36RT_C_DECLS_BEGIN
37
38/** @defgroup grp_rt_uint128 RTUInt128 - 128-bit Unsigned Integer Methods
39 * @ingroup grp_rt
40 * @{
41 */
42
43
44/**
45 * Test if a 128-bit unsigned integer value is zero.
46 *
47 * @returns true if they are, false if they aren't.
48 * @param pValue The input and output value.
49 */
50DECLINLINE(bool) RTUInt128IsZero(PRTUINT128U pValue)
51{
52#if ARCH_BITS >= 64
53 return pValue->s.Hi == 0
54 && pValue->s.Lo == 0;
55#else
56 return pValue->DWords.dw0 == 0
57 && pValue->DWords.dw1 == 0
58 && pValue->DWords.dw2 == 0
59 && pValue->DWords.dw3 == 0;
60#endif
61}
62
63
64/**
65 * Set a 128-bit unsigned integer value to zero.
66 *
67 * @returns pResult
68 * @param pResult The result variable.
69 */
70DECLINLINE(PRTUINT128U) RTUInt128SetZero(PRTUINT128U pResult)
71{
72#if ARCH_BITS >= 64
73 pResult->s.Hi = 0;
74 pResult->s.Lo = 0;
75#else
76 pResult->DWords.dw0 = 0;
77 pResult->DWords.dw1 = 0;
78 pResult->DWords.dw2 = 0;
79 pResult->DWords.dw3 = 0;
80#endif
81 return pResult;
82}
83
84
85/**
86 * Set a 128-bit unsigned integer value to the maximum value.
87 *
88 * @returns pResult
89 * @param pResult The result variable.
90 */
91DECLINLINE(PRTUINT128U) RTUInt128SetMax(PRTUINT128U pResult)
92{
93#if ARCH_BITS >= 64
94 pResult->s.Hi = UINT64_MAX;
95 pResult->s.Lo = UINT64_MAX;
96#else
97 pResult->DWords.dw0 = UINT32_MAX;
98 pResult->DWords.dw1 = UINT32_MAX;
99 pResult->DWords.dw2 = UINT32_MAX;
100 pResult->DWords.dw3 = UINT32_MAX;
101#endif
102 return pResult;
103}
104
105
106
107
108/**
109 * Adds two 128-bit unsigned integer values.
110 *
111 * @returns pResult
112 * @param pResult The result variable.
113 * @param pValue1 The first value.
114 * @param pValue2 The second value.
115 */
116DECLINLINE(PRTUINT128U) RTUInt128Add(PRTUINT128U pResult, PCRTUINT128U pValue1, PCRTUINT128U pValue2)
117{
118 pResult->s.Hi = pValue1->s.Hi + pValue2->s.Hi;
119 pResult->s.Lo = pValue1->s.Lo + pValue2->s.Lo;
120 if (pResult->s.Lo < pValue1->s.Lo)
121 pResult->s.Hi++;
122 return pResult;
123}
124
125
126/**
127 * Adds a 128-bit and a 64-bit unsigned integer values.
128 *
129 * @returns pResult
130 * @param pResult The result variable.
131 * @param pValue1 The first value.
132 * @param uValue2 The second value, 64-bit.
133 */
134DECLINLINE(PRTUINT128U) RTUInt128AddU64(PRTUINT128U pResult, PCRTUINT128U pValue1, uint64_t uValue2)
135{
136 pResult->s.Hi = pValue1->s.Hi;
137 pResult->s.Lo = pValue1->s.Lo + uValue2;
138 if (pResult->s.Lo < pValue1->s.Lo)
139 pResult->s.Hi++;
140 return pResult;
141}
142
143
144/**
145 * Subtracts a 128-bit unsigned integer value from another.
146 *
147 * @returns pResult
148 * @param pResult The result variable.
149 * @param pValue1 The minuend value.
150 * @param pValue2 The subtrahend value.
151 */
152DECLINLINE(PRTUINT128U) RTUInt128Sub(PRTUINT128U pResult, PCRTUINT128U pValue1, PCRTUINT128U pValue2)
153{
154 pResult->s.Lo = pValue1->s.Lo - pValue2->s.Lo;
155 pResult->s.Hi = pValue1->s.Hi - pValue2->s.Hi;
156 if (pResult->s.Lo > pValue1->s.Lo)
157 pResult->s.Hi--;
158 return pResult;
159}
160
161
162/**
163 * Multiplies two 128-bit unsigned integer values.
164 *
165 * @returns pResult
166 * @param pResult The result variable.
167 * @param pValue1 The first value.
168 * @param pValue2 The second value.
169 */
170DECLINLINE(PRTUINT128U) RTUInt128Mul(PRTUINT128U pResult, PCRTUINT128U pValue1, PCRTUINT128U pValue2)
171{
172 RTUINT64U uTmp;
173
174 /* multiply all dwords in v1 by v2.dw0. */
175 pResult->s.Lo = (uint64_t)pValue1->DWords.dw0 * pValue2->DWords.dw0;
176
177 uTmp.u = (uint64_t)pValue1->DWords.dw1 * pValue2->DWords.dw0;
178 pResult->DWords.dw3 = 0;
179 pResult->DWords.dw2 = uTmp.DWords.dw1;
180 pResult->DWords.dw1 += uTmp.DWords.dw0;
181 if (pResult->DWords.dw1 < uTmp.DWords.dw0)
182 if (pResult->DWords.dw2++ == UINT32_MAX)
183 pResult->DWords.dw3++;
184
185 pResult->s.Hi += (uint64_t)pValue1->DWords.dw2 * pValue2->DWords.dw0;
186 pResult->DWords.dw3 += pValue1->DWords.dw3 * pValue2->DWords.dw0;
187
188 /* multiply dw0, dw1 & dw2 in v1 by v2.dw1. */
189 uTmp.u = (uint64_t)pValue1->DWords.dw0 * pValue2->DWords.dw1;
190 pResult->DWords.dw1 += uTmp.DWords.dw0;
191 if (pResult->DWords.dw1 < uTmp.DWords.dw0)
192 if (pResult->DWords.dw2++ == UINT32_MAX)
193 pResult->DWords.dw3++;
194
195 pResult->DWords.dw2 += uTmp.DWords.dw1;
196 if (pResult->DWords.dw2 < uTmp.DWords.dw1)
197 pResult->DWords.dw3++;
198
199 pResult->s.Hi += (uint64_t)pValue1->DWords.dw1 * pValue2->DWords.dw1;
200 pResult->DWords.dw3 += pValue1->DWords.dw2 * pValue2->DWords.dw1;
201
202 /* multiply dw0 & dw1 in v1 by v2.dw2. */
203 pResult->s.Hi += (uint64_t)pValue1->DWords.dw0 * pValue2->DWords.dw2;
204 pResult->DWords.dw3 += pValue1->DWords.dw1 * pValue2->DWords.dw2;
205
206 /* multiply dw0 in v1 by v2.dw3. */
207 pResult->DWords.dw3 += pValue1->DWords.dw0 * pValue2->DWords.dw3;
208
209 return pResult;
210}
211
212
213/**
214 * Multiplies an 128-bit unsigned integer by a 64-bit unsigned integer value.
215 *
216 * @returns pResult
217 * @param pResult The result variable.
218 * @param pValue1 The first value.
219 * @param uValue2 The second value, 64-bit.
220 */
221#if defined(RT_ARCH_AMD64)
222RTDECL(PRTUINT128U) RTUInt128MulByU64(PRTUINT128U pResult, PCRTUINT128U pValue1, uint64_t uValue2);
223#else
224DECLINLINE(PRTUINT128U) RTUInt128MulByU64(PRTUINT128U pResult, PCRTUINT128U pValue1, uint64_t uValue2)
225{
226 uint32_t const uLoValue2 = (uint32_t)uValue2;
227 uint32_t const uHiValue2 = (uint32_t)(uValue2 >> 32);
228 RTUINT64U uTmp;
229
230 /* multiply all dwords in v1 by uLoValue1. */
231 pResult->s.Lo = (uint64_t)pValue1->DWords.dw0 * uLoValue2;
232
233 uTmp.u = (uint64_t)pValue1->DWords.dw1 * uLoValue2;
234 pResult->DWords.dw3 = 0;
235 pResult->DWords.dw2 = uTmp.DWords.dw1;
236 pResult->DWords.dw1 += uTmp.DWords.dw0;
237 if (pResult->DWords.dw1 < uTmp.DWords.dw0)
238 if (pResult->DWords.dw2++ == UINT32_MAX)
239 pResult->DWords.dw3++;
240
241 pResult->s.Hi += (uint64_t)pValue1->DWords.dw2 * uLoValue2;
242 pResult->DWords.dw3 += pValue1->DWords.dw3 * uLoValue2;
243
244 /* multiply dw0, dw1 & dw2 in v1 by uHiValue2. */
245 uTmp.u = (uint64_t)pValue1->DWords.dw0 * uHiValue2;
246 pResult->DWords.dw1 += uTmp.DWords.dw0;
247 if (pResult->DWords.dw1 < uTmp.DWords.dw0)
248 if (pResult->DWords.dw2++ == UINT32_MAX)
249 pResult->DWords.dw3++;
250
251 pResult->DWords.dw2 += uTmp.DWords.dw1;
252 if (pResult->DWords.dw2 < uTmp.DWords.dw1)
253 pResult->DWords.dw3++;
254
255 pResult->s.Hi += (uint64_t)pValue1->DWords.dw1 * uHiValue2;
256 pResult->DWords.dw3 += pValue1->DWords.dw2 * uHiValue2;
257
258 return pResult;
259}
260#endif
261
262
263/**
264 * Multiplies two 64-bit unsigned integer values with 128-bit precision.
265 *
266 * @returns pResult
267 * @param pResult The result variable.
268 * @param uValue1 The first value. 64-bit.
269 * @param uValue2 The second value, 64-bit.
270 */
271DECLINLINE(PRTUINT128U) RTUInt128MulU64ByU64(PRTUINT128U pResult, uint64_t uValue1, uint64_t uValue2)
272{
273#ifdef RT_ARCH_AMD64
274 pResult->s.Lo = ASMMult2xU64Ret2xU64(uValue1, uValue2, &pResult->s.Hi);
275#else
276 uint32_t const uLoValue1 = (uint32_t)uValue1;
277 uint32_t const uHiValue1 = (uint32_t)(uValue1 >> 32);
278 uint32_t const uLoValue2 = (uint32_t)uValue2;
279 uint32_t const uHiValue2 = (uint32_t)(uValue2 >> 32);
280 RTUINT64U uTmp;
281
282 /* Multiply uLoValue1 and uHiValue1 by uLoValue1. */
283 pResult->s.Lo = (uint64_t)uLoValue1 * uLoValue2;
284
285 uTmp.u = (uint64_t)uHiValue1 * uLoValue2;
286 pResult->DWords.dw3 = 0;
287 pResult->DWords.dw2 = uTmp.DWords.dw1;
288 pResult->DWords.dw1 += uTmp.DWords.dw0;
289 if (pResult->DWords.dw1 < uTmp.DWords.dw0)
290 if (pResult->DWords.dw2++ == UINT32_MAX)
291 pResult->DWords.dw3++;
292
293 /* Multiply uLoValue1 and uHiValue1 by uHiValue2. */
294 uTmp.u = (uint64_t)uLoValue1 * uHiValue2;
295 pResult->DWords.dw1 += uTmp.DWords.dw0;
296 if (pResult->DWords.dw1 < uTmp.DWords.dw0)
297 if (pResult->DWords.dw2++ == UINT32_MAX)
298 pResult->DWords.dw3++;
299
300 pResult->DWords.dw2 += uTmp.DWords.dw1;
301 if (pResult->DWords.dw2 < uTmp.DWords.dw1)
302 pResult->DWords.dw3++;
303
304 pResult->s.Hi += (uint64_t)uHiValue1 * uHiValue2;
305#endif
306 return pResult;
307}
308
309
310DECLINLINE(PRTUINT128U) RTUInt128DivRem(PRTUINT128U pQuotient, PRTUINT128U pRemainder, PCRTUINT128U pValue1, PCRTUINT128U pValue2);
311
312/**
313 * Divides a 128-bit unsigned integer value by another.
314 *
315 * @returns pResult
316 * @param pResult The result variable.
317 * @param pValue1 The dividend value.
318 * @param pValue2 The divisor value.
319 */
320DECLINLINE(PRTUINT128U) RTUInt128Div(PRTUINT128U pResult, PCRTUINT128U pValue1, PCRTUINT128U pValue2)
321{
322 RTUINT128U Ignored;
323 return RTUInt128DivRem(pResult, &Ignored, pValue1, pValue2);
324}
325
326
327/**
328 * Divides a 128-bit unsigned integer value by another, returning the remainder.
329 *
330 * @returns pResult
331 * @param pResult The result variable (remainder).
332 * @param pValue1 The dividend value.
333 * @param pValue2 The divisor value.
334 */
335DECLINLINE(PRTUINT128U) RTUInt128Mod(PRTUINT128U pResult, PCRTUINT128U pValue1, PCRTUINT128U pValue2)
336{
337 RTUINT128U Ignored;
338 RTUInt128DivRem(&Ignored, pResult, pValue1, pValue2);
339 return pResult;
340}
341
342
343/**
344 * Bitwise AND of two 128-bit unsigned integer values.
345 *
346 * @returns pResult
347 * @param pResult The result variable.
348 * @param pValue1 The first value.
349 * @param pValue2 The second value.
350 */
351DECLINLINE(PRTUINT128U) RTUInt128And(PRTUINT128U pResult, PCRTUINT128U pValue1, PCRTUINT128U pValue2)
352{
353 pResult->s.Hi = pValue1->s.Hi & pValue2->s.Hi;
354 pResult->s.Lo = pValue1->s.Lo & pValue2->s.Lo;
355 return pResult;
356}
357
358
359/**
360 * Bitwise OR of two 128-bit unsigned integer values.
361 *
362 * @returns pResult
363 * @param pResult The result variable.
364 * @param pValue1 The first value.
365 * @param pValue2 The second value.
366 */
367DECLINLINE(PRTUINT128U) RTUInt128Or( PRTUINT128U pResult, PCRTUINT128U pValue1, PCRTUINT128U pValue2)
368{
369 pResult->s.Hi = pValue1->s.Hi | pValue2->s.Hi;
370 pResult->s.Lo = pValue1->s.Lo | pValue2->s.Lo;
371 return pResult;
372}
373
374
375/**
376 * Bitwise XOR of two 128-bit unsigned integer values.
377 *
378 * @returns pResult
379 * @param pResult The result variable.
380 * @param pValue1 The first value.
381 * @param pValue2 The second value.
382 */
383DECLINLINE(PRTUINT128U) RTUInt128Xor(PRTUINT128U pResult, PCRTUINT128U pValue1, PCRTUINT128U pValue2)
384{
385 pResult->s.Hi = pValue1->s.Hi ^ pValue2->s.Hi;
386 pResult->s.Lo = pValue1->s.Lo ^ pValue2->s.Lo;
387 return pResult;
388}
389
390
391/**
392 * Shifts a 128-bit unsigned integer value @a cBits to the left.
393 *
394 * @returns pResult
395 * @param pResult The result variable.
396 * @param pValue The value to shift.
397 * @param cBits The number of bits to shift it.
398 */
399DECLINLINE(PRTUINT128U) RTUInt128ShiftLeft(PRTUINT128U pResult, PCRTUINT128U pValue, int cBits)
400{
401 cBits &= 127;
402 if (cBits < 64)
403 {
404 pResult->s.Lo = pValue->s.Lo << cBits;
405 pResult->s.Hi = (pValue->s.Hi << cBits) | (pValue->s.Lo >> (64 - cBits));
406 }
407 else
408 {
409 pResult->s.Lo = 0;
410 pResult->s.Hi = pValue->s.Lo << (cBits - 64);
411 }
412 return pResult;
413}
414
415
416/**
417 * Shifts a 128-bit unsigned integer value @a cBits to the right.
418 *
419 * @returns pResult
420 * @param pResult The result variable.
421 * @param pValue The value to shift.
422 * @param cBits The number of bits to shift it.
423 */
424DECLINLINE(PRTUINT128U) RTUInt128ShiftRight(PRTUINT128U pResult, PCRTUINT128U pValue, int cBits)
425{
426 cBits &= 127;
427 if (cBits < 64)
428 {
429 pResult->s.Hi = pValue->s.Hi >> cBits;
430 pResult->s.Lo = (pValue->s.Lo >> cBits) | (pValue->s.Hi << (64 - cBits));
431 }
432 else
433 {
434 pResult->s.Hi = 0;
435 pResult->s.Lo = pValue->s.Hi >> (cBits - 64);
436 }
437 return pResult;
438}
439
440
441/**
442 * Boolean not (result 0 or 1).
443 *
444 * @returns pResult.
445 * @param pResult The result variable.
446 * @param pValue The value.
447 */
448DECLINLINE(PRTUINT128U) RTUInt128BooleanNot(PRTUINT128U pResult, PCRTUINT128U pValue)
449{
450 pResult->s.Lo = pValue->s.Lo || pValue->s.Hi ? 0 : 1;
451 pResult->s.Hi = 0;
452 return pResult;
453}
454
455
456/**
457 * Bitwise not (flips each bit of the 128 bits).
458 *
459 * @returns pResult.
460 * @param pResult The result variable.
461 * @param pValue The value.
462 */
463DECLINLINE(PRTUINT128U) RTUInt128BitwiseNot(PRTUINT128U pResult, PCRTUINT128U pValue)
464{
465 pResult->s.Hi = ~pValue->s.Hi;
466 pResult->s.Lo = ~pValue->s.Lo;
467 return pResult;
468}
469
470
471/**
472 * Assigns one 128-bit unsigned integer value to another.
473 *
474 * @returns pResult
475 * @param pResult The result variable.
476 * @param pValue The value to assign.
477 */
478DECLINLINE(PRTUINT128U) RTUInt128Assign(PRTUINT128U pResult, PCRTUINT128U pValue)
479{
480#if ARCH_BITS >= 64
481 pResult->s.Hi = pValue->s.Hi;
482 pResult->s.Lo = pValue->s.Lo;
483#else
484 pResult->DWords.dw0 = pValue->DWords.dw0;
485 pResult->DWords.dw1 = pValue->DWords.dw1;
486 pResult->DWords.dw2 = pValue->DWords.dw2;
487 pResult->DWords.dw3 = pValue->DWords.dw3;
488#endif
489 return pResult;
490}
491
492
493/**
494 * Assigns a boolean value to 128-bit unsigned integer.
495 *
496 * @returns pValueResult
497 * @param pValueResult The result variable.
498 * @param fValue The boolean value.
499 */
500DECLINLINE(PRTUINT128U) RTUInt128AssignBoolean(PRTUINT128U pValueResult, bool fValue)
501{
502#if ARCH_BITS >= 64
503 pValueResult->s.Lo = fValue;
504 pValueResult->s.Hi = 0;
505#else
506 pValueResult->DWords.dw0 = fValue;
507 pValueResult->DWords.dw1 = 0;
508 pValueResult->DWords.dw2 = 0;
509 pValueResult->DWords.dw3 = 0;
510#endif
511 return pValueResult;
512}
513
514
515/**
516 * Assigns a 8-bit unsigned integer value to 128-bit unsigned integer.
517 *
518 * @returns pValueResult
519 * @param pValueResult The result variable.
520 * @param u8Value The 8-bit unsigned integer value.
521 */
522DECLINLINE(PRTUINT128U) RTUInt128AssignU8(PRTUINT128U pValueResult, uint8_t u8Value)
523{
524#if ARCH_BITS >= 64
525 pValueResult->s.Lo = u8Value;
526 pValueResult->s.Hi = 0;
527#else
528 pValueResult->DWords.dw0 = u8Value;
529 pValueResult->DWords.dw1 = 0;
530 pValueResult->DWords.dw2 = 0;
531 pValueResult->DWords.dw3 = 0;
532#endif
533 return pValueResult;
534}
535
536
537/**
538 * Assigns a 16-bit unsigned integer value to 128-bit unsigned integer.
539 *
540 * @returns pValueResult
541 * @param pValueResult The result variable.
542 * @param u16Value The 16-bit unsigned integer value.
543 */
544DECLINLINE(PRTUINT128U) RTUInt128AssignU16(PRTUINT128U pValueResult, uint16_t u16Value)
545{
546#if ARCH_BITS >= 64
547 pValueResult->s.Lo = u16Value;
548 pValueResult->s.Hi = 0;
549#else
550 pValueResult->DWords.dw0 = u16Value;
551 pValueResult->DWords.dw1 = 0;
552 pValueResult->DWords.dw2 = 0;
553 pValueResult->DWords.dw3 = 0;
554#endif
555 return pValueResult;
556}
557
558
559/**
560 * Assigns a 32-bit unsigned integer value to 128-bit unsigned integer.
561 *
562 * @returns pValueResult
563 * @param pValueResult The result variable.
564 * @param u32Value The 32-bit unsigned integer value.
565 */
566DECLINLINE(PRTUINT128U) RTUInt128AssignU32(PRTUINT128U pValueResult, uint32_t u32Value)
567{
568#if ARCH_BITS >= 64
569 pValueResult->s.Lo = u32Value;
570 pValueResult->s.Hi = 0;
571#else
572 pValueResult->DWords.dw0 = u32Value;
573 pValueResult->DWords.dw1 = 0;
574 pValueResult->DWords.dw2 = 0;
575 pValueResult->DWords.dw3 = 0;
576#endif
577 return pValueResult;
578}
579
580
581/**
582 * Assigns a 64-bit unsigned integer value to 128-bit unsigned integer.
583 *
584 * @returns pValueResult
585 * @param pValueResult The result variable.
586 * @param u64Value The 64-bit unsigned integer value.
587 */
588DECLINLINE(PRTUINT128U) RTUInt128AssignU64(PRTUINT128U pValueResult, uint64_t u64Value)
589{
590 pValueResult->s.Lo = u64Value;
591 pValueResult->s.Hi = 0;
592 return pValueResult;
593}
594
595
596/**
597 * Adds two 128-bit unsigned integer values, storing the result in the first.
598 *
599 * @returns pValue1Result.
600 * @param pValue1Result The first value and result.
601 * @param pValue2 The second value.
602 */
603DECLINLINE(PRTUINT128U) RTUInt128AssignAdd(PRTUINT128U pValue1Result, PCRTUINT128U pValue2)
604{
605 uint64_t const uTmp = pValue1Result->s.Lo;
606 pValue1Result->s.Lo += pValue2->s.Lo;
607 if (pValue1Result->s.Lo < uTmp)
608 pValue1Result->s.Hi++;
609 pValue1Result->s.Hi += pValue2->s.Hi;
610 return pValue1Result;
611}
612
613
614/**
615 * Adds a 64-bit unsigned integer value to a 128-bit unsigned integer values,
616 * storing the result in the 128-bit one.
617 *
618 * @returns pValue1Result.
619 * @param pValue1Result The first value and result.
620 * @param uValue2 The second value, 64-bit.
621 */
622DECLINLINE(PRTUINT128U) RTUInt128AssignAddU64(PRTUINT128U pValue1Result, uint64_t uValue2)
623{
624 pValue1Result->s.Lo += uValue2;
625 if (pValue1Result->s.Lo < uValue2)
626 pValue1Result->s.Hi++;
627 return pValue1Result;
628}
629
630
631/**
632 * Subtracts two 128-bit unsigned integer values, storing the result in the
633 * first.
634 *
635 * @returns pValue1Result.
636 * @param pValue1Result The minuend value and result.
637 * @param pValue2 The subtrahend value.
638 */
639DECLINLINE(PRTUINT128U) RTUInt128AssignSub(PRTUINT128U pValue1Result, PCRTUINT128U pValue2)
640{
641 uint64_t const uTmp = pValue1Result->s.Lo;
642 pValue1Result->s.Lo -= pValue2->s.Lo;
643 if (pValue1Result->s.Lo > uTmp)
644 pValue1Result->s.Hi--;
645 pValue1Result->s.Hi -= pValue2->s.Hi;
646 return pValue1Result;
647}
648
649
650/**
651 * Multiplies two 128-bit unsigned integer values, storing the result in the
652 * first.
653 *
654 * @returns pValue1Result.
655 * @param pValue1Result The first value and result.
656 * @param pValue2 The second value.
657 */
658DECLINLINE(PRTUINT128U) RTUInt128AssignMul(PRTUINT128U pValue1Result, PCRTUINT128U pValue2)
659{
660 RTUINT128U Result;
661 RTUInt128Mul(&Result, pValue1Result, pValue2);
662 *pValue1Result = Result;
663 return pValue1Result;
664}
665
666
667/**
668 * Divides a 128-bit unsigned integer value by another, storing the result in
669 * the first.
670 *
671 * @returns pValue1Result.
672 * @param pValue1Result The dividend value and result.
673 * @param pValue2 The divisor value.
674 */
675DECLINLINE(PRTUINT128U) RTUInt128AssignDiv(PRTUINT128U pValue1Result, PCRTUINT128U pValue2)
676{
677 RTUINT128U Result;
678 RTUINT128U Ignored;
679 RTUInt128DivRem(&Result, &Ignored, pValue1Result, pValue2);
680 *pValue1Result = Result;
681 return pValue1Result;
682}
683
684
685/**
686 * Divides a 128-bit unsigned integer value by another, storing the remainder in
687 * the first.
688 *
689 * @returns pValue1Result.
690 * @param pValue1Result The dividend value and result (remainder).
691 * @param pValue2 The divisor value.
692 */
693DECLINLINE(PRTUINT128U) RTUInt128AssignMod(PRTUINT128U pValue1Result, PCRTUINT128U pValue2)
694{
695 RTUINT128U Ignored;
696 RTUINT128U Result;
697 RTUInt128DivRem(&Ignored, &Result, pValue1Result, pValue2);
698 *pValue1Result = Result;
699 return pValue1Result;
700}
701
702
703/**
704 * Performs a bitwise AND of two 128-bit unsigned integer values and assigned
705 * the result to the first one.
706 *
707 * @returns pValue1Result.
708 * @param pValue1Result The first value and result.
709 * @param pValue2 The second value.
710 */
711DECLINLINE(PRTUINT128U) RTUInt128AssignAnd(PRTUINT128U pValue1Result, PCRTUINT128U pValue2)
712{
713#if ARCH_BITS >= 64
714 pValue1Result->s.Hi &= pValue2->s.Hi;
715 pValue1Result->s.Lo &= pValue2->s.Lo;
716#else
717 pValue1Result->DWords.dw0 &= pValue2->DWords.dw0;
718 pValue1Result->DWords.dw1 &= pValue2->DWords.dw1;
719 pValue1Result->DWords.dw2 &= pValue2->DWords.dw2;
720 pValue1Result->DWords.dw3 &= pValue2->DWords.dw3;
721#endif
722 return pValue1Result;
723}
724
725
726/**
727 * Performs a bitwise AND of a 128-bit unsigned integer value and a mask made
728 * up of the first N bits, assigning the result to the the 128-bit value.
729 *
730 * @returns pValueResult.
731 * @param pValueResult The value and result.
732 * @param cBits The number of bits to AND (counting from the first
733 * bit).
734 */
735DECLINLINE(PRTUINT128U) RTUInt128AssignAndNFirstBits(PRTUINT128U pValueResult, unsigned cBits)
736{
737 if (cBits <= 64)
738 {
739 if (cBits != 64)
740 pValueResult->s.Lo &= (RT_BIT_64(cBits) - 1);
741 pValueResult->s.Hi = 0;
742 }
743 else if (cBits < 128)
744 pValueResult->s.Hi &= (RT_BIT_64(cBits - 64) - 1);
745/** @todo \#if ARCH_BITS >= 64 */
746 return pValueResult;
747}
748
749
750/**
751 * Performs a bitwise OR of two 128-bit unsigned integer values and assigned
752 * the result to the first one.
753 *
754 * @returns pValue1Result.
755 * @param pValue1Result The first value and result.
756 * @param pValue2 The second value.
757 */
758DECLINLINE(PRTUINT128U) RTUInt128AssignOr(PRTUINT128U pValue1Result, PCRTUINT128U pValue2)
759{
760#if ARCH_BITS >= 64
761 pValue1Result->s.Hi |= pValue2->s.Hi;
762 pValue1Result->s.Lo |= pValue2->s.Lo;
763#else
764 pValue1Result->DWords.dw0 |= pValue2->DWords.dw0;
765 pValue1Result->DWords.dw1 |= pValue2->DWords.dw1;
766 pValue1Result->DWords.dw2 |= pValue2->DWords.dw2;
767 pValue1Result->DWords.dw3 |= pValue2->DWords.dw3;
768#endif
769 return pValue1Result;
770}
771
772
773/**
774 * ORs in a bit and assign the result to the input value.
775 *
776 * @returns pValue1Result.
777 * @param pValue1Result The first value and result.
778 * @param iBit The bit to set (0 based).
779 */
780DECLINLINE(PRTUINT128U) RTUInt128AssignOrBit(PRTUINT128U pValue1Result, uint32_t iBit)
781{
782#if ARCH_BITS >= 64
783 if (iBit >= 64)
784 pValue1Result->s.Hi |= RT_BIT_64(iBit - 64);
785 else
786 pValue1Result->s.Lo |= RT_BIT_64(iBit);
787#else
788 if (iBit >= 64)
789 {
790 if (iBit >= 96)
791 pValue1Result->DWords.dw3 |= RT_BIT_32(iBit - 96);
792 else
793 pValue1Result->DWords.dw2 |= RT_BIT_32(iBit - 64);
794 }
795 else
796 {
797 if (iBit >= 32)
798 pValue1Result->DWords.dw1 |= RT_BIT_32(iBit - 32);
799 else
800 pValue1Result->DWords.dw0 |= RT_BIT_32(iBit);
801 }
802#endif
803 return pValue1Result;
804}
805
806
807
808/**
809 * Performs a bitwise XOR of two 128-bit unsigned integer values and assigned
810 * the result to the first one.
811 *
812 * @returns pValue1Result.
813 * @param pValue1Result The first value and result.
814 * @param pValue2 The second value.
815 */
816DECLINLINE(PRTUINT128U) RTUInt128AssignXor(PRTUINT128U pValue1Result, PCRTUINT128U pValue2)
817{
818#if ARCH_BITS >= 64
819 pValue1Result->s.Hi ^= pValue2->s.Hi;
820 pValue1Result->s.Lo ^= pValue2->s.Lo;
821#else
822 pValue1Result->DWords.dw0 ^= pValue2->DWords.dw0;
823 pValue1Result->DWords.dw1 ^= pValue2->DWords.dw1;
824 pValue1Result->DWords.dw2 ^= pValue2->DWords.dw2;
825 pValue1Result->DWords.dw3 ^= pValue2->DWords.dw3;
826#endif
827 return pValue1Result;
828}
829
830
831/**
832 * Performs a bitwise left shift on a 128-bit unsigned integer value, assigning
833 * the result to it.
834 *
835 * @returns pValueResult.
836 * @param pValueResult The first value and result.
837 * @param cBits The number of bits to shift.
838 */
839DECLINLINE(PRTUINT128U) RTUInt128AssignShiftLeft(PRTUINT128U pValueResult, int cBits)
840{
841 RTUINT128U const InVal = *pValueResult;
842/** @todo \#if ARCH_BITS >= 64 */
843 if (cBits > 0)
844 {
845 /* (left shift) */
846 if (cBits >= 128)
847 RTUInt128SetZero(pValueResult);
848 else if (cBits >= 64)
849 {
850 pValueResult->s.Lo = 0;
851 pValueResult->s.Hi = InVal.s.Lo << (cBits - 64);
852 }
853 else
854 {
855 pValueResult->s.Hi = InVal.s.Hi << cBits;
856 pValueResult->s.Hi |= InVal.s.Lo >> (64 - cBits);
857 pValueResult->s.Lo = InVal.s.Lo << cBits;
858 }
859 }
860 else if (cBits < 0)
861 {
862 /* (right shift) */
863 cBits = -cBits;
864 if (cBits >= 128)
865 RTUInt128SetZero(pValueResult);
866 else if (cBits >= 64)
867 {
868 pValueResult->s.Hi = 0;
869 pValueResult->s.Lo = InVal.s.Hi >> (cBits - 64);
870 }
871 else
872 {
873 pValueResult->s.Lo = InVal.s.Lo >> cBits;
874 pValueResult->s.Lo |= InVal.s.Hi << (64 - cBits);
875 pValueResult->s.Hi = InVal.s.Hi >> cBits;
876 }
877 }
878 return pValueResult;
879}
880
881
882/**
883 * Performs a bitwise left shift on a 128-bit unsigned integer value, assigning
884 * the result to it.
885 *
886 * @returns pValueResult.
887 * @param pValueResult The first value and result.
888 * @param cBits The number of bits to shift.
889 */
890DECLINLINE(PRTUINT128U) RTUInt128AssignShiftRight(PRTUINT128U pValueResult, int cBits)
891{
892 return RTUInt128AssignShiftLeft(pValueResult, -cBits);
893}
894
895
896/**
897 * Performs a bitwise NOT on a 128-bit unsigned integer value, assigning the
898 * result to it.
899 *
900 * @returns pValueResult
901 * @param pValueResult The value and result.
902 */
903DECLINLINE(PRTUINT128U) RTUInt128AssignBitwiseNot(PRTUINT128U pValueResult)
904{
905#if ARCH_BITS >= 64
906 pValueResult->s.Hi = ~pValueResult->s.Hi;
907 pValueResult->s.Lo = ~pValueResult->s.Lo;
908#else
909 pValueResult->DWords.dw0 = ~pValueResult->DWords.dw0;
910 pValueResult->DWords.dw1 = ~pValueResult->DWords.dw1;
911 pValueResult->DWords.dw2 = ~pValueResult->DWords.dw2;
912 pValueResult->DWords.dw3 = ~pValueResult->DWords.dw3;
913#endif
914 return pValueResult;
915}
916
917
918/**
919 * Performs a boolean NOT on a 128-bit unsigned integer value, assigning the
920 * result to it.
921 *
922 * @returns pValueResult
923 * @param pValueResult The value and result.
924 */
925DECLINLINE(PRTUINT128U) RTUInt128AssignBooleanNot(PRTUINT128U pValueResult)
926{
927 return RTUInt128AssignBoolean(pValueResult, RTUInt128IsZero(pValueResult));
928}
929
930
931/**
932 * Compares two 128-bit unsigned integer values.
933 *
934 * @retval 0 if equal.
935 * @retval -1 if the first value is smaller than the second.
936 * @retval 1 if the first value is larger than the second.
937 *
938 * @param pValue1 The first value.
939 * @param pValue2 The second value.
940 */
941DECLINLINE(int) RTUInt128Compare(PCRTUINT128U pValue1, PCRTUINT128U pValue2)
942{
943#if ARCH_BITS >= 64
944 if (pValue1->s.Hi != pValue2->s.Hi)
945 return pValue1->s.Hi > pValue2->s.Hi ? 1 : -1;
946 if (pValue1->s.Lo != pValue2->s.Lo)
947 return pValue1->s.Lo > pValue2->s.Lo ? 1 : -1;
948 return 0;
949#else
950 if (pValue1->DWords.dw3 != pValue2->DWords.dw3)
951 return pValue1->DWords.dw3 > pValue2->DWords.dw3 ? 1 : -1;
952 if (pValue1->DWords.dw2 != pValue2->DWords.dw2)
953 return pValue1->DWords.dw2 > pValue2->DWords.dw2 ? 1 : -1;
954 if (pValue1->DWords.dw1 != pValue2->DWords.dw1)
955 return pValue1->DWords.dw1 > pValue2->DWords.dw1 ? 1 : -1;
956 if (pValue1->DWords.dw0 != pValue2->DWords.dw0)
957 return pValue1->DWords.dw0 > pValue2->DWords.dw0 ? 1 : -1;
958 return 0;
959#endif
960}
961
962
963/**
964 * Tests if a 128-bit unsigned integer value is smaller than another.
965 *
966 * @returns true if the first value is smaller, false if not.
967 * @param pValue1 The first value.
968 * @param pValue2 The second value.
969 */
970DECLINLINE(bool) RTUInt128IsSmaller(PCRTUINT128U pValue1, PCRTUINT128U pValue2)
971{
972#if ARCH_BITS >= 64
973 return pValue1->s.Hi < pValue2->s.Hi
974 || ( pValue1->s.Hi == pValue2->s.Hi
975 && pValue1->s.Lo < pValue2->s.Lo);
976#else
977 return pValue1->DWords.dw3 < pValue2->DWords.dw3
978 || ( pValue1->DWords.dw3 == pValue2->DWords.dw3
979 && ( pValue1->DWords.dw2 < pValue2->DWords.dw2
980 || ( pValue1->DWords.dw2 == pValue2->DWords.dw2
981 && ( pValue1->DWords.dw1 < pValue2->DWords.dw1
982 || ( pValue1->DWords.dw1 == pValue2->DWords.dw1
983 && pValue1->DWords.dw0 < pValue2->DWords.dw0)))));
984#endif
985}
986
987
988/**
989 * Tests if a 128-bit unsigned integer value is larger than another.
990 *
991 * @returns true if the first value is larger, false if not.
992 * @param pValue1 The first value.
993 * @param pValue2 The second value.
994 */
995DECLINLINE(bool) RTUInt128IsLarger(PCRTUINT128U pValue1, PCRTUINT128U pValue2)
996{
997#if ARCH_BITS >= 64
998 return pValue1->s.Hi > pValue2->s.Hi
999 || ( pValue1->s.Hi == pValue2->s.Hi
1000 && pValue1->s.Lo > pValue2->s.Lo);
1001#else
1002 return pValue1->DWords.dw3 > pValue2->DWords.dw3
1003 || ( pValue1->DWords.dw3 == pValue2->DWords.dw3
1004 && ( pValue1->DWords.dw2 > pValue2->DWords.dw2
1005 || ( pValue1->DWords.dw2 == pValue2->DWords.dw2
1006 && ( pValue1->DWords.dw1 > pValue2->DWords.dw1
1007 || ( pValue1->DWords.dw1 == pValue2->DWords.dw1
1008 && pValue1->DWords.dw0 > pValue2->DWords.dw0)))));
1009#endif
1010}
1011
1012
1013/**
1014 * Tests if a 128-bit unsigned integer value is larger or equal than another.
1015 *
1016 * @returns true if the first value is larger or equal, false if not.
1017 * @param pValue1 The first value.
1018 * @param pValue2 The second value.
1019 */
1020DECLINLINE(bool) RTUInt128IsLargerOrEqual(PCRTUINT128U pValue1, PCRTUINT128U pValue2)
1021{
1022#if ARCH_BITS >= 64
1023 return pValue1->s.Hi > pValue2->s.Hi
1024 || ( pValue1->s.Hi == pValue2->s.Hi
1025 && pValue1->s.Lo >= pValue2->s.Lo);
1026#else
1027 return pValue1->DWords.dw3 > pValue2->DWords.dw3
1028 || ( pValue1->DWords.dw3 == pValue2->DWords.dw3
1029 && ( pValue1->DWords.dw2 > pValue2->DWords.dw2
1030 || ( pValue1->DWords.dw2 == pValue2->DWords.dw2
1031 && ( pValue1->DWords.dw1 > pValue2->DWords.dw1
1032 || ( pValue1->DWords.dw1 == pValue2->DWords.dw1
1033 && pValue1->DWords.dw0 >= pValue2->DWords.dw0)))));
1034#endif
1035}
1036
1037
1038/**
1039 * Tests if two 128-bit unsigned integer values not equal.
1040 *
1041 * @returns true if equal, false if not equal.
1042 * @param pValue1 The first value.
1043 * @param pValue2 The second value.
1044 */
1045DECLINLINE(bool) RTUInt128IsEqual(PCRTUINT128U pValue1, PCRTUINT128U pValue2)
1046{
1047#if ARCH_BITS >= 64
1048 return pValue1->s.Hi == pValue2->s.Hi
1049 && pValue1->s.Lo == pValue2->s.Lo;
1050#else
1051 return pValue1->DWords.dw0 == pValue2->DWords.dw0
1052 && pValue1->DWords.dw1 == pValue2->DWords.dw1
1053 && pValue1->DWords.dw2 == pValue2->DWords.dw2
1054 && pValue1->DWords.dw3 == pValue2->DWords.dw3;
1055#endif
1056}
1057
1058
1059/**
1060 * Tests if two 128-bit unsigned integer values are not equal.
1061 *
1062 * @returns true if not equal, false if equal.
1063 * @param pValue1 The first value.
1064 * @param pValue2 The second value.
1065 */
1066DECLINLINE(bool) RTUInt128IsNotEqual(PCRTUINT128U pValue1, PCRTUINT128U pValue2)
1067{
1068 return !RTUInt128IsEqual(pValue1, pValue2);
1069}
1070
1071
1072/**
1073 * Sets a bit in a 128-bit unsigned integer type.
1074 *
1075 * @returns pValueResult.
1076 * @param pValueResult The input and output value.
1077 * @param iBit The bit to set.
1078 */
1079DECLINLINE(PRTUINT128U) RTUInt128BitSet(PRTUINT128U pValueResult, unsigned iBit)
1080{
1081 if (iBit < 64)
1082 {
1083#if ARCH_BITS >= 64
1084 pValueResult->s.Lo |= RT_BIT_64(iBit);
1085#else
1086 if (iBit < 32)
1087 pValueResult->DWords.dw0 |= RT_BIT_32(iBit);
1088 else
1089 pValueResult->DWords.dw1 |= RT_BIT_32(iBit - 32);
1090#endif
1091 }
1092 else if (iBit < 128)
1093 {
1094#if ARCH_BITS >= 64
1095 pValueResult->s.Hi |= RT_BIT_64(iBit - 64);
1096#else
1097 if (iBit < 96)
1098 pValueResult->DWords.dw2 |= RT_BIT_32(iBit - 64);
1099 else
1100 pValueResult->DWords.dw3 |= RT_BIT_32(iBit - 96);
1101#endif
1102 }
1103 return pValueResult;
1104}
1105
1106
1107/**
1108 * Sets a bit in a 128-bit unsigned integer type.
1109 *
1110 * @returns pValueResult.
1111 * @param pValueResult The input and output value.
1112 * @param iBit The bit to set.
1113 */
1114DECLINLINE(PRTUINT128U) RTUInt128BitClear(PRTUINT128U pValueResult, unsigned iBit)
1115{
1116 if (iBit < 64)
1117 {
1118#if ARCH_BITS >= 64
1119 pValueResult->s.Lo &= ~RT_BIT_64(iBit);
1120#else
1121 if (iBit < 32)
1122 pValueResult->DWords.dw0 &= ~RT_BIT_32(iBit);
1123 else
1124 pValueResult->DWords.dw1 &= ~RT_BIT_32(iBit - 32);
1125#endif
1126 }
1127 else if (iBit < 128)
1128 {
1129#if ARCH_BITS >= 64
1130 pValueResult->s.Hi &= ~RT_BIT_64(iBit - 64);
1131#else
1132 if (iBit < 96)
1133 pValueResult->DWords.dw2 &= ~RT_BIT_32(iBit - 64);
1134 else
1135 pValueResult->DWords.dw3 &= ~RT_BIT_32(iBit - 96);
1136#endif
1137 }
1138 return pValueResult;
1139}
1140
1141
1142/**
1143 * Tests if a bit in a 128-bit unsigned integer value is set.
1144 *
1145 * @returns pValueResult.
1146 * @param pValueResult The input and output value.
1147 * @param iBit The bit to test.
1148 */
1149DECLINLINE(bool) RTUInt128BitTest(PRTUINT128U pValueResult, unsigned iBit)
1150{
1151 bool fRc;
1152 if (iBit < 64)
1153 {
1154#if ARCH_BITS >= 64
1155 fRc = RT_BOOL(pValueResult->s.Lo & RT_BIT_64(iBit));
1156#else
1157 if (iBit < 32)
1158 fRc = RT_BOOL(pValueResult->DWords.dw0 & RT_BIT_32(iBit));
1159 else
1160 fRc = RT_BOOL(pValueResult->DWords.dw1 & RT_BIT_32(iBit - 32));
1161#endif
1162 }
1163 else if (iBit < 128)
1164 {
1165#if ARCH_BITS >= 64
1166 fRc = RT_BOOL(pValueResult->s.Hi & RT_BIT_64(iBit - 64));
1167#else
1168 if (iBit < 96)
1169 fRc = RT_BOOL(pValueResult->DWords.dw2 & RT_BIT_32(iBit - 64));
1170 else
1171 fRc = RT_BOOL(pValueResult->DWords.dw3 & RT_BIT_32(iBit - 96));
1172#endif
1173 }
1174 else
1175 fRc = false;
1176 return fRc;
1177}
1178
1179
1180/**
1181 * Set a range of bits a 128-bit unsigned integer value.
1182 *
1183 * @returns pValueResult.
1184 * @param pValueResult The input and output value.
1185 * @param iFirstBit The first bit to test.
1186 * @param cBits The number of bits to set.
1187 */
1188DECLINLINE(PRTUINT128U) RTUInt128BitSetRange(PRTUINT128U pValueResult, unsigned iFirstBit, unsigned cBits)
1189{
1190 /* bounds check & fix. */
1191 if (iFirstBit < 128)
1192 {
1193 if (iFirstBit + cBits > 128)
1194 cBits = 128 - iFirstBit;
1195
1196#if ARCH_BITS >= 64
1197 if (iFirstBit + cBits < 64)
1198 pValueResult->s.Lo |= (RT_BIT_64(cBits) - 1) << iFirstBit;
1199 else if (iFirstBit + cBits < 128 && iFirstBit >= 64)
1200 pValueResult->s.Hi |= (RT_BIT_64(cBits) - 1) << (iFirstBit - 64);
1201 else
1202#else
1203 if (iFirstBit + cBits < 32)
1204 pValueResult->DWords.dw0 |= (RT_BIT_32(cBits) - 1) << iFirstBit;
1205 else if (iFirstBit + cBits < 64 && iFirstBit >= 32)
1206 pValueResult->DWords.dw1 |= (RT_BIT_32(cBits) - 1) << (iFirstBit - 32);
1207 else if (iFirstBit + cBits < 96 && iFirstBit >= 64)
1208 pValueResult->DWords.dw2 |= (RT_BIT_32(cBits) - 1) << (iFirstBit - 64);
1209 else if (iFirstBit + cBits < 128 && iFirstBit >= 96)
1210 pValueResult->DWords.dw3 |= (RT_BIT_32(cBits) - 1) << (iFirstBit - 96);
1211 else
1212#endif
1213 while (cBits-- > 0)
1214 RTUInt128BitSet(pValueResult, iFirstBit++);
1215 }
1216 return pValueResult;
1217}
1218
1219
1220/**
1221 * Test if all the bits of a 128-bit unsigned integer value are set.
1222 *
1223 * @returns true if they are, false if they aren't.
1224 * @param pValue The input and output value.
1225 */
1226DECLINLINE(bool) RTUInt128BitAreAllSet(PRTUINT128U pValue)
1227{
1228#if ARCH_BITS >= 64
1229 return pValue->s.Hi == UINT64_MAX
1230 && pValue->s.Lo == UINT64_MAX;
1231#else
1232 return pValue->DWords.dw0 == UINT32_MAX
1233 && pValue->DWords.dw1 == UINT32_MAX
1234 && pValue->DWords.dw2 == UINT32_MAX
1235 && pValue->DWords.dw3 == UINT32_MAX;
1236#endif
1237}
1238
1239
1240/**
1241 * Test if all the bits of a 128-bit unsigned integer value are clear.
1242 *
1243 * @returns true if they are, false if they aren't.
1244 * @param pValue The input and output value.
1245 */
1246DECLINLINE(bool) RTUInt128BitAreAllClear(PRTUINT128U pValue)
1247{
1248#if ARCH_BITS >= 64
1249 return pValue->s.Hi == 0
1250 && pValue->s.Lo == 0;
1251#else
1252 return pValue->DWords.dw0 == 0
1253 && pValue->DWords.dw1 == 0
1254 && pValue->DWords.dw2 == 0
1255 && pValue->DWords.dw3 == 0;
1256#endif
1257}
1258
1259
1260DECLINLINE(uint32_t) RTUInt128BitCount(PCRTUINT128U pValue)
1261{
1262 uint32_t cBits;
1263 if (pValue->s.Hi != 0)
1264 {
1265 if (pValue->DWords.dw3)
1266 cBits = 96 + ASMBitLastSetU32(pValue->DWords.dw3);
1267 else
1268 cBits = 64 + ASMBitLastSetU32(pValue->DWords.dw2);
1269 }
1270 else
1271 {
1272 if (pValue->DWords.dw1)
1273 cBits = 32 + ASMBitLastSetU32(pValue->DWords.dw1);
1274 else
1275 cBits = 0 + ASMBitLastSetU32(pValue->DWords.dw0);
1276 }
1277 return cBits;
1278}
1279
1280
1281/**
1282 * Divides a 128-bit unsigned integer value by another, returning both quotient
1283 * and remainder.
1284 *
1285 * @returns pQuotient, NULL if pValue2 is 0.
1286 * @param pQuotient Where to return the quotient.
1287 * @param pRemainder Where to return the remainder.
1288 * @param pValue1 The dividend value.
1289 * @param pValue2 The divisor value.
1290 */
1291DECLINLINE(PRTUINT128U) RTUInt128DivRem(PRTUINT128U pQuotient, PRTUINT128U pRemainder, PCRTUINT128U pValue1, PCRTUINT128U pValue2)
1292{
1293 int iDiff;
1294
1295 /*
1296 * Sort out all the special cases first.
1297 */
1298 /* Divide by zero or 1? */
1299 if (!pValue2->s.Hi)
1300 {
1301 if (!pValue2->s.Lo)
1302 return NULL;
1303
1304 if (pValue2->s.Lo == 1)
1305 {
1306 RTUInt128SetZero(pRemainder);
1307 *pQuotient = *pValue1;
1308 return pQuotient;
1309 }
1310 /** @todo RTUint128DivModBy64 */
1311 }
1312
1313 /* Dividend is smaller? */
1314 iDiff = RTUInt128Compare(pValue1, pValue2);
1315 if (iDiff < 0)
1316 {
1317 *pRemainder = *pValue1;
1318 RTUInt128SetZero(pQuotient);
1319 }
1320
1321 /* The values are equal? */
1322 else if (iDiff == 0)
1323 {
1324 RTUInt128SetZero(pRemainder);
1325 RTUInt128AssignU64(pQuotient, 1);
1326 }
1327 else
1328 {
1329 /*
1330 * Prepare.
1331 */
1332 uint32_t iBitAdder = RTUInt128BitCount(pValue1) - RTUInt128BitCount(pValue2);
1333 RTUINT128U NormDivisor = *pValue2;
1334 if (iBitAdder)
1335 {
1336 RTUInt128ShiftLeft(&NormDivisor, pValue2, iBitAdder);
1337 if (RTUInt128IsLarger(&NormDivisor, pValue1))
1338 {
1339 RTUInt128AssignShiftRight(&NormDivisor, 1);
1340 iBitAdder--;
1341 }
1342 }
1343 else
1344 NormDivisor = *pValue2;
1345
1346 RTUInt128SetZero(pQuotient);
1347 *pRemainder = *pValue1;
1348
1349 /*
1350 * Do the division.
1351 */
1352 if (RTUInt128IsLargerOrEqual(pRemainder, pValue2))
1353 {
1354 for (;;)
1355 {
1356 if (RTUInt128IsLargerOrEqual(pRemainder, &NormDivisor))
1357 {
1358 RTUInt128AssignSub(pRemainder, &NormDivisor);
1359 RTUInt128AssignOrBit(pQuotient, iBitAdder);
1360 }
1361 if (RTUInt128IsSmaller(pRemainder, pValue2))
1362 break;
1363 RTUInt128AssignShiftRight(&NormDivisor, 1);
1364 iBitAdder--;
1365 }
1366 }
1367 }
1368 return pQuotient;
1369}
1370
1371
1372/** @} */
1373
1374RT_C_DECLS_END
1375
1376#endif
1377
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