1*9356374aSAndroid Build Coastguard Worker// 2*9356374aSAndroid Build Coastguard Worker// Copyright 2017 The Abseil Authors. 3*9356374aSAndroid Build Coastguard Worker// 4*9356374aSAndroid Build Coastguard Worker// Licensed under the Apache License, Version 2.0 (the "License"); 5*9356374aSAndroid Build Coastguard Worker// you may not use this file except in compliance with the License. 6*9356374aSAndroid Build Coastguard Worker// You may obtain a copy of the License at 7*9356374aSAndroid Build Coastguard Worker// 8*9356374aSAndroid Build Coastguard Worker// https://www.apache.org/licenses/LICENSE-2.0 9*9356374aSAndroid Build Coastguard Worker// 10*9356374aSAndroid Build Coastguard Worker// Unless required by applicable law or agreed to in writing, software 11*9356374aSAndroid Build Coastguard Worker// distributed under the License is distributed on an "AS IS" BASIS, 12*9356374aSAndroid Build Coastguard Worker// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13*9356374aSAndroid Build Coastguard Worker// See the License for the specific language governing permissions and 14*9356374aSAndroid Build Coastguard Worker// limitations under the License. 15*9356374aSAndroid Build Coastguard Worker 16*9356374aSAndroid Build Coastguard Worker// This file contains :int128 implementation details that depend on internal 17*9356374aSAndroid Build Coastguard Worker// representation when ABSL_HAVE_INTRINSIC_INT128 is *not* defined. This file 18*9356374aSAndroid Build Coastguard Worker// is included by int128.h and relies on ABSL_INTERNAL_WCHAR_T being defined. 19*9356374aSAndroid Build Coastguard Worker 20*9356374aSAndroid Build Coastguard Workerconstexpr uint64_t Int128Low64(int128 v) { return v.lo_; } 21*9356374aSAndroid Build Coastguard Worker 22*9356374aSAndroid Build Coastguard Workerconstexpr int64_t Int128High64(int128 v) { return v.hi_; } 23*9356374aSAndroid Build Coastguard Worker 24*9356374aSAndroid Build Coastguard Worker#if defined(ABSL_IS_LITTLE_ENDIAN) 25*9356374aSAndroid Build Coastguard Worker 26*9356374aSAndroid Build Coastguard Workerconstexpr int128::int128(int64_t high, uint64_t low) : lo_(low), hi_(high) {} 27*9356374aSAndroid Build Coastguard Worker 28*9356374aSAndroid Build Coastguard Workerconstexpr int128::int128(int v) 29*9356374aSAndroid Build Coastguard Worker : lo_{static_cast<uint64_t>(v)}, hi_{v < 0 ? ~int64_t{0} : 0} {} 30*9356374aSAndroid Build Coastguard Workerconstexpr int128::int128(long v) // NOLINT(runtime/int) 31*9356374aSAndroid Build Coastguard Worker : lo_{static_cast<uint64_t>(v)}, hi_{v < 0 ? ~int64_t{0} : 0} {} 32*9356374aSAndroid Build Coastguard Workerconstexpr int128::int128(long long v) // NOLINT(runtime/int) 33*9356374aSAndroid Build Coastguard Worker : lo_{static_cast<uint64_t>(v)}, hi_{v < 0 ? ~int64_t{0} : 0} {} 34*9356374aSAndroid Build Coastguard Worker 35*9356374aSAndroid Build Coastguard Workerconstexpr int128::int128(unsigned int v) : lo_{v}, hi_{0} {} 36*9356374aSAndroid Build Coastguard Worker// NOLINTNEXTLINE(runtime/int) 37*9356374aSAndroid Build Coastguard Workerconstexpr int128::int128(unsigned long v) : lo_{v}, hi_{0} {} 38*9356374aSAndroid Build Coastguard Worker// NOLINTNEXTLINE(runtime/int) 39*9356374aSAndroid Build Coastguard Workerconstexpr int128::int128(unsigned long long v) : lo_{v}, hi_{0} {} 40*9356374aSAndroid Build Coastguard Worker 41*9356374aSAndroid Build Coastguard Workerconstexpr int128::int128(uint128 v) 42*9356374aSAndroid Build Coastguard Worker : lo_{Uint128Low64(v)}, hi_{static_cast<int64_t>(Uint128High64(v))} {} 43*9356374aSAndroid Build Coastguard Worker 44*9356374aSAndroid Build Coastguard Worker#elif defined(ABSL_IS_BIG_ENDIAN) 45*9356374aSAndroid Build Coastguard Worker 46*9356374aSAndroid Build Coastguard Workerconstexpr int128::int128(int64_t high, uint64_t low) : hi_{high}, lo_{low} {} 47*9356374aSAndroid Build Coastguard Worker 48*9356374aSAndroid Build Coastguard Workerconstexpr int128::int128(int v) 49*9356374aSAndroid Build Coastguard Worker : hi_{v < 0 ? ~int64_t{0} : 0}, lo_{static_cast<uint64_t>(v)} {} 50*9356374aSAndroid Build Coastguard Workerconstexpr int128::int128(long v) // NOLINT(runtime/int) 51*9356374aSAndroid Build Coastguard Worker : hi_{v < 0 ? ~int64_t{0} : 0}, lo_{static_cast<uint64_t>(v)} {} 52*9356374aSAndroid Build Coastguard Workerconstexpr int128::int128(long long v) // NOLINT(runtime/int) 53*9356374aSAndroid Build Coastguard Worker : hi_{v < 0 ? ~int64_t{0} : 0}, lo_{static_cast<uint64_t>(v)} {} 54*9356374aSAndroid Build Coastguard Worker 55*9356374aSAndroid Build Coastguard Workerconstexpr int128::int128(unsigned int v) : hi_{0}, lo_{v} {} 56*9356374aSAndroid Build Coastguard Worker// NOLINTNEXTLINE(runtime/int) 57*9356374aSAndroid Build Coastguard Workerconstexpr int128::int128(unsigned long v) : hi_{0}, lo_{v} {} 58*9356374aSAndroid Build Coastguard Worker// NOLINTNEXTLINE(runtime/int) 59*9356374aSAndroid Build Coastguard Workerconstexpr int128::int128(unsigned long long v) : hi_{0}, lo_{v} {} 60*9356374aSAndroid Build Coastguard Worker 61*9356374aSAndroid Build Coastguard Workerconstexpr int128::int128(uint128 v) 62*9356374aSAndroid Build Coastguard Worker : hi_{static_cast<int64_t>(Uint128High64(v))}, lo_{Uint128Low64(v)} {} 63*9356374aSAndroid Build Coastguard Worker 64*9356374aSAndroid Build Coastguard Worker#else // byte order 65*9356374aSAndroid Build Coastguard Worker#error "Unsupported byte order: must be little-endian or big-endian." 66*9356374aSAndroid Build Coastguard Worker#endif // byte order 67*9356374aSAndroid Build Coastguard Worker 68*9356374aSAndroid Build Coastguard Workerconstexpr int128::operator bool() const { return lo_ || hi_; } 69*9356374aSAndroid Build Coastguard Worker 70*9356374aSAndroid Build Coastguard Workerconstexpr int128::operator char() const { 71*9356374aSAndroid Build Coastguard Worker // NOLINTNEXTLINE(runtime/int) 72*9356374aSAndroid Build Coastguard Worker return static_cast<char>(static_cast<long long>(*this)); 73*9356374aSAndroid Build Coastguard Worker} 74*9356374aSAndroid Build Coastguard Worker 75*9356374aSAndroid Build Coastguard Workerconstexpr int128::operator signed char() const { 76*9356374aSAndroid Build Coastguard Worker // NOLINTNEXTLINE(runtime/int) 77*9356374aSAndroid Build Coastguard Worker return static_cast<signed char>(static_cast<long long>(*this)); 78*9356374aSAndroid Build Coastguard Worker} 79*9356374aSAndroid Build Coastguard Worker 80*9356374aSAndroid Build Coastguard Workerconstexpr int128::operator unsigned char() const { 81*9356374aSAndroid Build Coastguard Worker return static_cast<unsigned char>(lo_); 82*9356374aSAndroid Build Coastguard Worker} 83*9356374aSAndroid Build Coastguard Worker 84*9356374aSAndroid Build Coastguard Workerconstexpr int128::operator char16_t() const { 85*9356374aSAndroid Build Coastguard Worker return static_cast<char16_t>(lo_); 86*9356374aSAndroid Build Coastguard Worker} 87*9356374aSAndroid Build Coastguard Worker 88*9356374aSAndroid Build Coastguard Workerconstexpr int128::operator char32_t() const { 89*9356374aSAndroid Build Coastguard Worker return static_cast<char32_t>(lo_); 90*9356374aSAndroid Build Coastguard Worker} 91*9356374aSAndroid Build Coastguard Worker 92*9356374aSAndroid Build Coastguard Workerconstexpr int128::operator ABSL_INTERNAL_WCHAR_T() const { 93*9356374aSAndroid Build Coastguard Worker // NOLINTNEXTLINE(runtime/int) 94*9356374aSAndroid Build Coastguard Worker return static_cast<ABSL_INTERNAL_WCHAR_T>(static_cast<long long>(*this)); 95*9356374aSAndroid Build Coastguard Worker} 96*9356374aSAndroid Build Coastguard Worker 97*9356374aSAndroid Build Coastguard Workerconstexpr int128::operator short() const { // NOLINT(runtime/int) 98*9356374aSAndroid Build Coastguard Worker // NOLINTNEXTLINE(runtime/int) 99*9356374aSAndroid Build Coastguard Worker return static_cast<short>(static_cast<long long>(*this)); 100*9356374aSAndroid Build Coastguard Worker} 101*9356374aSAndroid Build Coastguard Worker 102*9356374aSAndroid Build Coastguard Workerconstexpr int128::operator unsigned short() const { // NOLINT(runtime/int) 103*9356374aSAndroid Build Coastguard Worker return static_cast<unsigned short>(lo_); // NOLINT(runtime/int) 104*9356374aSAndroid Build Coastguard Worker} 105*9356374aSAndroid Build Coastguard Worker 106*9356374aSAndroid Build Coastguard Workerconstexpr int128::operator int() const { 107*9356374aSAndroid Build Coastguard Worker // NOLINTNEXTLINE(runtime/int) 108*9356374aSAndroid Build Coastguard Worker return static_cast<int>(static_cast<long long>(*this)); 109*9356374aSAndroid Build Coastguard Worker} 110*9356374aSAndroid Build Coastguard Worker 111*9356374aSAndroid Build Coastguard Workerconstexpr int128::operator unsigned int() const { 112*9356374aSAndroid Build Coastguard Worker return static_cast<unsigned int>(lo_); 113*9356374aSAndroid Build Coastguard Worker} 114*9356374aSAndroid Build Coastguard Worker 115*9356374aSAndroid Build Coastguard Workerconstexpr int128::operator long() const { // NOLINT(runtime/int) 116*9356374aSAndroid Build Coastguard Worker // NOLINTNEXTLINE(runtime/int) 117*9356374aSAndroid Build Coastguard Worker return static_cast<long>(static_cast<long long>(*this)); 118*9356374aSAndroid Build Coastguard Worker} 119*9356374aSAndroid Build Coastguard Worker 120*9356374aSAndroid Build Coastguard Workerconstexpr int128::operator unsigned long() const { // NOLINT(runtime/int) 121*9356374aSAndroid Build Coastguard Worker return static_cast<unsigned long>(lo_); // NOLINT(runtime/int) 122*9356374aSAndroid Build Coastguard Worker} 123*9356374aSAndroid Build Coastguard Worker 124*9356374aSAndroid Build Coastguard Workerconstexpr int128::operator long long() const { // NOLINT(runtime/int) 125*9356374aSAndroid Build Coastguard Worker // We don't bother checking the value of hi_. If *this < 0, lo_'s high bit 126*9356374aSAndroid Build Coastguard Worker // must be set in order for the value to fit into a long long. Conversely, if 127*9356374aSAndroid Build Coastguard Worker // lo_'s high bit is set, *this must be < 0 for the value to fit. 128*9356374aSAndroid Build Coastguard Worker return int128_internal::BitCastToSigned(lo_); 129*9356374aSAndroid Build Coastguard Worker} 130*9356374aSAndroid Build Coastguard Worker 131*9356374aSAndroid Build Coastguard Workerconstexpr int128::operator unsigned long long() const { // NOLINT(runtime/int) 132*9356374aSAndroid Build Coastguard Worker return static_cast<unsigned long long>(lo_); // NOLINT(runtime/int) 133*9356374aSAndroid Build Coastguard Worker} 134*9356374aSAndroid Build Coastguard Worker 135*9356374aSAndroid Build Coastguard Workerinline int128::operator float() const { 136*9356374aSAndroid Build Coastguard Worker // We must convert the absolute value and then negate as needed, because 137*9356374aSAndroid Build Coastguard Worker // floating point types are typically sign-magnitude. Otherwise, the 138*9356374aSAndroid Build Coastguard Worker // difference between the high and low 64 bits when interpreted as two's 139*9356374aSAndroid Build Coastguard Worker // complement overwhelms the precision of the mantissa. 140*9356374aSAndroid Build Coastguard Worker // 141*9356374aSAndroid Build Coastguard Worker // Also check to make sure we don't negate Int128Min() 142*9356374aSAndroid Build Coastguard Worker return hi_ < 0 && *this != Int128Min() 143*9356374aSAndroid Build Coastguard Worker ? -static_cast<float>(-*this) 144*9356374aSAndroid Build Coastguard Worker : static_cast<float>(lo_) + 145*9356374aSAndroid Build Coastguard Worker std::ldexp(static_cast<float>(hi_), 64); 146*9356374aSAndroid Build Coastguard Worker} 147*9356374aSAndroid Build Coastguard Worker 148*9356374aSAndroid Build Coastguard Workerinline int128::operator double() const { 149*9356374aSAndroid Build Coastguard Worker // See comment in int128::operator float() above. 150*9356374aSAndroid Build Coastguard Worker return hi_ < 0 && *this != Int128Min() 151*9356374aSAndroid Build Coastguard Worker ? -static_cast<double>(-*this) 152*9356374aSAndroid Build Coastguard Worker : static_cast<double>(lo_) + 153*9356374aSAndroid Build Coastguard Worker std::ldexp(static_cast<double>(hi_), 64); 154*9356374aSAndroid Build Coastguard Worker} 155*9356374aSAndroid Build Coastguard Worker 156*9356374aSAndroid Build Coastguard Workerinline int128::operator long double() const { 157*9356374aSAndroid Build Coastguard Worker // See comment in int128::operator float() above. 158*9356374aSAndroid Build Coastguard Worker return hi_ < 0 && *this != Int128Min() 159*9356374aSAndroid Build Coastguard Worker ? -static_cast<long double>(-*this) 160*9356374aSAndroid Build Coastguard Worker : static_cast<long double>(lo_) + 161*9356374aSAndroid Build Coastguard Worker std::ldexp(static_cast<long double>(hi_), 64); 162*9356374aSAndroid Build Coastguard Worker} 163*9356374aSAndroid Build Coastguard Worker 164*9356374aSAndroid Build Coastguard Worker// Comparison operators. 165*9356374aSAndroid Build Coastguard Worker 166*9356374aSAndroid Build Coastguard Workerconstexpr bool operator==(int128 lhs, int128 rhs) { 167*9356374aSAndroid Build Coastguard Worker return (Int128Low64(lhs) == Int128Low64(rhs) && 168*9356374aSAndroid Build Coastguard Worker Int128High64(lhs) == Int128High64(rhs)); 169*9356374aSAndroid Build Coastguard Worker} 170*9356374aSAndroid Build Coastguard Worker 171*9356374aSAndroid Build Coastguard Workerconstexpr bool operator!=(int128 lhs, int128 rhs) { return !(lhs == rhs); } 172*9356374aSAndroid Build Coastguard Worker 173*9356374aSAndroid Build Coastguard Workerconstexpr bool operator<(int128 lhs, int128 rhs) { 174*9356374aSAndroid Build Coastguard Worker return (Int128High64(lhs) == Int128High64(rhs)) 175*9356374aSAndroid Build Coastguard Worker ? (Int128Low64(lhs) < Int128Low64(rhs)) 176*9356374aSAndroid Build Coastguard Worker : (Int128High64(lhs) < Int128High64(rhs)); 177*9356374aSAndroid Build Coastguard Worker} 178*9356374aSAndroid Build Coastguard Worker 179*9356374aSAndroid Build Coastguard Workerconstexpr bool operator>(int128 lhs, int128 rhs) { 180*9356374aSAndroid Build Coastguard Worker return (Int128High64(lhs) == Int128High64(rhs)) 181*9356374aSAndroid Build Coastguard Worker ? (Int128Low64(lhs) > Int128Low64(rhs)) 182*9356374aSAndroid Build Coastguard Worker : (Int128High64(lhs) > Int128High64(rhs)); 183*9356374aSAndroid Build Coastguard Worker} 184*9356374aSAndroid Build Coastguard Worker 185*9356374aSAndroid Build Coastguard Workerconstexpr bool operator<=(int128 lhs, int128 rhs) { return !(lhs > rhs); } 186*9356374aSAndroid Build Coastguard Worker 187*9356374aSAndroid Build Coastguard Workerconstexpr bool operator>=(int128 lhs, int128 rhs) { return !(lhs < rhs); } 188*9356374aSAndroid Build Coastguard Worker 189*9356374aSAndroid Build Coastguard Worker#ifdef __cpp_impl_three_way_comparison 190*9356374aSAndroid Build Coastguard Workerconstexpr absl::strong_ordering operator<=>(int128 lhs, int128 rhs) { 191*9356374aSAndroid Build Coastguard Worker if (int64_t lhs_high = Int128High64(lhs), rhs_high = Int128High64(rhs); 192*9356374aSAndroid Build Coastguard Worker lhs_high < rhs_high) { 193*9356374aSAndroid Build Coastguard Worker return absl::strong_ordering::less; 194*9356374aSAndroid Build Coastguard Worker } else if (lhs_high > rhs_high) { 195*9356374aSAndroid Build Coastguard Worker return absl::strong_ordering::greater; 196*9356374aSAndroid Build Coastguard Worker } else if (uint64_t lhs_low = Uint128Low64(lhs), rhs_low = Uint128Low64(rhs); 197*9356374aSAndroid Build Coastguard Worker lhs_low < rhs_low) { 198*9356374aSAndroid Build Coastguard Worker return absl::strong_ordering::less; 199*9356374aSAndroid Build Coastguard Worker } else if (lhs_low > rhs_low) { 200*9356374aSAndroid Build Coastguard Worker return absl::strong_ordering::greater; 201*9356374aSAndroid Build Coastguard Worker } else { 202*9356374aSAndroid Build Coastguard Worker return absl::strong_ordering::equal; 203*9356374aSAndroid Build Coastguard Worker } 204*9356374aSAndroid Build Coastguard Worker} 205*9356374aSAndroid Build Coastguard Worker#endif 206*9356374aSAndroid Build Coastguard Worker 207*9356374aSAndroid Build Coastguard Worker// Unary operators. 208*9356374aSAndroid Build Coastguard Worker 209*9356374aSAndroid Build Coastguard Workerconstexpr int128 operator-(int128 v) { 210*9356374aSAndroid Build Coastguard Worker return MakeInt128(~Int128High64(v) + (Int128Low64(v) == 0), 211*9356374aSAndroid Build Coastguard Worker ~Int128Low64(v) + 1); 212*9356374aSAndroid Build Coastguard Worker} 213*9356374aSAndroid Build Coastguard Worker 214*9356374aSAndroid Build Coastguard Workerconstexpr bool operator!(int128 v) { 215*9356374aSAndroid Build Coastguard Worker return !Int128Low64(v) && !Int128High64(v); 216*9356374aSAndroid Build Coastguard Worker} 217*9356374aSAndroid Build Coastguard Worker 218*9356374aSAndroid Build Coastguard Workerconstexpr int128 operator~(int128 val) { 219*9356374aSAndroid Build Coastguard Worker return MakeInt128(~Int128High64(val), ~Int128Low64(val)); 220*9356374aSAndroid Build Coastguard Worker} 221*9356374aSAndroid Build Coastguard Worker 222*9356374aSAndroid Build Coastguard Worker// Arithmetic operators. 223*9356374aSAndroid Build Coastguard Worker 224*9356374aSAndroid Build Coastguard Workernamespace int128_internal { 225*9356374aSAndroid Build Coastguard Workerconstexpr int128 SignedAddResult(int128 result, int128 lhs) { 226*9356374aSAndroid Build Coastguard Worker // check for carry 227*9356374aSAndroid Build Coastguard Worker return (Int128Low64(result) < Int128Low64(lhs)) 228*9356374aSAndroid Build Coastguard Worker ? MakeInt128(Int128High64(result) + 1, Int128Low64(result)) 229*9356374aSAndroid Build Coastguard Worker : result; 230*9356374aSAndroid Build Coastguard Worker} 231*9356374aSAndroid Build Coastguard Worker} // namespace int128_internal 232*9356374aSAndroid Build Coastguard Workerconstexpr int128 operator+(int128 lhs, int128 rhs) { 233*9356374aSAndroid Build Coastguard Worker return int128_internal::SignedAddResult( 234*9356374aSAndroid Build Coastguard Worker MakeInt128(Int128High64(lhs) + Int128High64(rhs), 235*9356374aSAndroid Build Coastguard Worker Int128Low64(lhs) + Int128Low64(rhs)), 236*9356374aSAndroid Build Coastguard Worker lhs); 237*9356374aSAndroid Build Coastguard Worker} 238*9356374aSAndroid Build Coastguard Worker 239*9356374aSAndroid Build Coastguard Workernamespace int128_internal { 240*9356374aSAndroid Build Coastguard Workerconstexpr int128 SignedSubstructResult(int128 result, int128 lhs, int128 rhs) { 241*9356374aSAndroid Build Coastguard Worker // check for carry 242*9356374aSAndroid Build Coastguard Worker return (Int128Low64(lhs) < Int128Low64(rhs)) 243*9356374aSAndroid Build Coastguard Worker ? MakeInt128(Int128High64(result) - 1, Int128Low64(result)) 244*9356374aSAndroid Build Coastguard Worker : result; 245*9356374aSAndroid Build Coastguard Worker} 246*9356374aSAndroid Build Coastguard Worker} // namespace int128_internal 247*9356374aSAndroid Build Coastguard Workerconstexpr int128 operator-(int128 lhs, int128 rhs) { 248*9356374aSAndroid Build Coastguard Worker return int128_internal::SignedSubstructResult( 249*9356374aSAndroid Build Coastguard Worker MakeInt128(Int128High64(lhs) - Int128High64(rhs), 250*9356374aSAndroid Build Coastguard Worker Int128Low64(lhs) - Int128Low64(rhs)), 251*9356374aSAndroid Build Coastguard Worker lhs, rhs); 252*9356374aSAndroid Build Coastguard Worker} 253*9356374aSAndroid Build Coastguard Worker 254*9356374aSAndroid Build Coastguard Workerinline int128 operator*(int128 lhs, int128 rhs) { 255*9356374aSAndroid Build Coastguard Worker return MakeInt128( 256*9356374aSAndroid Build Coastguard Worker int128_internal::BitCastToSigned(Uint128High64(uint128(lhs) * rhs)), 257*9356374aSAndroid Build Coastguard Worker Uint128Low64(uint128(lhs) * rhs)); 258*9356374aSAndroid Build Coastguard Worker} 259*9356374aSAndroid Build Coastguard Worker 260*9356374aSAndroid Build Coastguard Workerinline int128 int128::operator++(int) { 261*9356374aSAndroid Build Coastguard Worker int128 tmp(*this); 262*9356374aSAndroid Build Coastguard Worker *this += 1; 263*9356374aSAndroid Build Coastguard Worker return tmp; 264*9356374aSAndroid Build Coastguard Worker} 265*9356374aSAndroid Build Coastguard Worker 266*9356374aSAndroid Build Coastguard Workerinline int128 int128::operator--(int) { 267*9356374aSAndroid Build Coastguard Worker int128 tmp(*this); 268*9356374aSAndroid Build Coastguard Worker *this -= 1; 269*9356374aSAndroid Build Coastguard Worker return tmp; 270*9356374aSAndroid Build Coastguard Worker} 271*9356374aSAndroid Build Coastguard Worker 272*9356374aSAndroid Build Coastguard Workerinline int128& int128::operator++() { 273*9356374aSAndroid Build Coastguard Worker *this += 1; 274*9356374aSAndroid Build Coastguard Worker return *this; 275*9356374aSAndroid Build Coastguard Worker} 276*9356374aSAndroid Build Coastguard Worker 277*9356374aSAndroid Build Coastguard Workerinline int128& int128::operator--() { 278*9356374aSAndroid Build Coastguard Worker *this -= 1; 279*9356374aSAndroid Build Coastguard Worker return *this; 280*9356374aSAndroid Build Coastguard Worker} 281*9356374aSAndroid Build Coastguard Worker 282*9356374aSAndroid Build Coastguard Workerconstexpr int128 operator|(int128 lhs, int128 rhs) { 283*9356374aSAndroid Build Coastguard Worker return MakeInt128(Int128High64(lhs) | Int128High64(rhs), 284*9356374aSAndroid Build Coastguard Worker Int128Low64(lhs) | Int128Low64(rhs)); 285*9356374aSAndroid Build Coastguard Worker} 286*9356374aSAndroid Build Coastguard Worker 287*9356374aSAndroid Build Coastguard Workerconstexpr int128 operator&(int128 lhs, int128 rhs) { 288*9356374aSAndroid Build Coastguard Worker return MakeInt128(Int128High64(lhs) & Int128High64(rhs), 289*9356374aSAndroid Build Coastguard Worker Int128Low64(lhs) & Int128Low64(rhs)); 290*9356374aSAndroid Build Coastguard Worker} 291*9356374aSAndroid Build Coastguard Worker 292*9356374aSAndroid Build Coastguard Workerconstexpr int128 operator^(int128 lhs, int128 rhs) { 293*9356374aSAndroid Build Coastguard Worker return MakeInt128(Int128High64(lhs) ^ Int128High64(rhs), 294*9356374aSAndroid Build Coastguard Worker Int128Low64(lhs) ^ Int128Low64(rhs)); 295*9356374aSAndroid Build Coastguard Worker} 296*9356374aSAndroid Build Coastguard Worker 297*9356374aSAndroid Build Coastguard Workerconstexpr int128 operator<<(int128 lhs, int amount) { 298*9356374aSAndroid Build Coastguard Worker // int64_t shifts of >= 63 are undefined, so we need some special-casing. 299*9356374aSAndroid Build Coastguard Worker assert(amount >= 0 && amount < 127); 300*9356374aSAndroid Build Coastguard Worker if (amount <= 0) { 301*9356374aSAndroid Build Coastguard Worker return lhs; 302*9356374aSAndroid Build Coastguard Worker } else if (amount < 63) { 303*9356374aSAndroid Build Coastguard Worker return MakeInt128( 304*9356374aSAndroid Build Coastguard Worker (Int128High64(lhs) << amount) | 305*9356374aSAndroid Build Coastguard Worker static_cast<int64_t>(Int128Low64(lhs) >> (64 - amount)), 306*9356374aSAndroid Build Coastguard Worker Int128Low64(lhs) << amount); 307*9356374aSAndroid Build Coastguard Worker } else if (amount == 63) { 308*9356374aSAndroid Build Coastguard Worker return MakeInt128(((Int128High64(lhs) << 32) << 31) | 309*9356374aSAndroid Build Coastguard Worker static_cast<int64_t>(Int128Low64(lhs) >> 1), 310*9356374aSAndroid Build Coastguard Worker (Int128Low64(lhs) << 32) << 31); 311*9356374aSAndroid Build Coastguard Worker } else if (amount == 127) { 312*9356374aSAndroid Build Coastguard Worker return MakeInt128(static_cast<int64_t>(Int128Low64(lhs) << 63), 0); 313*9356374aSAndroid Build Coastguard Worker } else if (amount > 127) { 314*9356374aSAndroid Build Coastguard Worker return MakeInt128(0, 0); 315*9356374aSAndroid Build Coastguard Worker } else { 316*9356374aSAndroid Build Coastguard Worker // amount >= 64 && amount < 127 317*9356374aSAndroid Build Coastguard Worker return MakeInt128(static_cast<int64_t>(Int128Low64(lhs) << (amount - 64)), 318*9356374aSAndroid Build Coastguard Worker 0); 319*9356374aSAndroid Build Coastguard Worker } 320*9356374aSAndroid Build Coastguard Worker} 321*9356374aSAndroid Build Coastguard Worker 322*9356374aSAndroid Build Coastguard Workerconstexpr int128 operator>>(int128 lhs, int amount) { 323*9356374aSAndroid Build Coastguard Worker // int64_t shifts of >= 63 are undefined, so we need some special-casing. 324*9356374aSAndroid Build Coastguard Worker assert(amount >= 0 && amount < 127); 325*9356374aSAndroid Build Coastguard Worker if (amount <= 0) { 326*9356374aSAndroid Build Coastguard Worker return lhs; 327*9356374aSAndroid Build Coastguard Worker } else if (amount < 63) { 328*9356374aSAndroid Build Coastguard Worker return MakeInt128( 329*9356374aSAndroid Build Coastguard Worker Int128High64(lhs) >> amount, 330*9356374aSAndroid Build Coastguard Worker Int128Low64(lhs) >> amount | static_cast<uint64_t>(Int128High64(lhs)) 331*9356374aSAndroid Build Coastguard Worker << (64 - amount)); 332*9356374aSAndroid Build Coastguard Worker } else if (amount == 63) { 333*9356374aSAndroid Build Coastguard Worker return MakeInt128((Int128High64(lhs) >> 32) >> 31, 334*9356374aSAndroid Build Coastguard Worker static_cast<uint64_t>(Int128High64(lhs) << 1) | 335*9356374aSAndroid Build Coastguard Worker (Int128Low64(lhs) >> 32) >> 31); 336*9356374aSAndroid Build Coastguard Worker 337*9356374aSAndroid Build Coastguard Worker } else if (amount >= 127) { 338*9356374aSAndroid Build Coastguard Worker return MakeInt128((Int128High64(lhs) >> 32) >> 31, 339*9356374aSAndroid Build Coastguard Worker static_cast<uint64_t>((Int128High64(lhs) >> 32) >> 31)); 340*9356374aSAndroid Build Coastguard Worker } else { 341*9356374aSAndroid Build Coastguard Worker // amount >= 64 && amount < 127 342*9356374aSAndroid Build Coastguard Worker return MakeInt128( 343*9356374aSAndroid Build Coastguard Worker (Int128High64(lhs) >> 32) >> 31, 344*9356374aSAndroid Build Coastguard Worker static_cast<uint64_t>(Int128High64(lhs) >> (amount - 64))); 345*9356374aSAndroid Build Coastguard Worker } 346*9356374aSAndroid Build Coastguard Worker} 347