[clang] [clang][Interp] Add IntegralAP for arbitrary-precision integers (PR #65844)
Timm Baeder via cfe-commits
cfe-commits at lists.llvm.org
Tue Sep 26 06:41:57 PDT 2023
Timm =?utf-8?q?Bäder?= <tbaeder at redhat.com>,
Timm =?utf-8?q?Bäder?= <tbaeder at redhat.com>,
Timm =?utf-8?q?Bäder?= <tbaeder at redhat.com>
Message-ID:
In-Reply-To: <llvm/llvm-project/pull/65844/clang at github.com>
================
@@ -0,0 +1,249 @@
+//===--- Integral.h - Wrapper for numeric types for the VM ------*- C++ -*-===//
+//
+// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
+// See https://llvm.org/LICENSE.txt for license information.
+// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
+//
+//===----------------------------------------------------------------------===//
+//
+// Defines the VM types and helpers operating on types.
+//
+//===----------------------------------------------------------------------===//
+
+#ifndef LLVM_CLANG_AST_INTERP_INTEGRAL_AP_H
+#define LLVM_CLANG_AST_INTERP_INTEGRAL_AP_H
+
+#include "clang/AST/APValue.h"
+#include "clang/AST/ComparisonCategories.h"
+#include "llvm/ADT/APSInt.h"
+#include "llvm/Support/MathExtras.h"
+#include "llvm/Support/raw_ostream.h"
+#include <cstddef>
+#include <cstdint>
+
+#include "Primitives.h"
+
+namespace clang {
+namespace interp {
+
+using APInt = llvm::APInt;
+using APSInt = llvm::APSInt;
+template <unsigned Bits, bool Signed> class Integral;
+class Boolean;
+
+template <bool Signed> class IntegralAP final {
+public:
+ APSInt V;
+
+public:
+ using AsUnsigned = IntegralAP<false>;
+
+ template <typename T>
+ IntegralAP(T Value) : V(APInt(sizeof(T) * 8, Value, std::is_signed_v<T>)) {}
+
+ IntegralAP(APInt V) : V(V) {}
+ IntegralAP(APSInt V) : V(V) {}
+ /// Arbitrary value for initialized variables.
+ IntegralAP() : V(APSInt::getMaxValue(1024, Signed)) {}
+
+ IntegralAP operator-() const { return IntegralAP(-V); }
+ bool operator>(IntegralAP RHS) const { return V > RHS.V; }
+ bool operator>=(IntegralAP RHS) const { return V >= RHS.V; }
+ bool operator<(IntegralAP RHS) const { return V < RHS.V; }
+ bool operator<=(IntegralAP RHS) const { return V <= RHS.V; }
+
+ explicit operator bool() const { return !V.isZero(); }
+ explicit operator int8_t() const { return V.getSExtValue(); }
+ explicit operator uint8_t() const { return V.getZExtValue(); }
+ explicit operator int16_t() const { return V.getSExtValue(); }
+ explicit operator uint16_t() const { return V.getZExtValue(); }
+ explicit operator int32_t() const { return V.getSExtValue(); }
+ explicit operator uint32_t() const { return V.getZExtValue(); }
+ explicit operator int64_t() const { return V.getSExtValue(); }
+ explicit operator uint64_t() const { return V.getZExtValue(); }
+
+ template <typename T> static IntegralAP from(T Value, unsigned NumBits = 0) {
+ assert(NumBits > 0);
+ APSInt Copy = APSInt(APInt(NumBits, Value, Signed), !Signed);
+
+ return IntegralAP<Signed>(Copy);
+ }
+
+ template <bool InputSigned>
+ static IntegralAP from(IntegralAP<InputSigned> V, unsigned NumBits = 0) {
+ if constexpr (Signed == InputSigned)
+ return V;
+
+ APSInt Copy = V.V;
+ Copy.setIsSigned(Signed);
+
+ return IntegralAP<Signed>(Copy);
+ }
+
+ template <unsigned Bits, bool InputSigned>
+ static IntegralAP from(Integral<Bits, InputSigned> I) {
+ // FIXME: Take bits parameter.
+ APSInt Copy =
+ APSInt(APInt(128, static_cast<int64_t>(I), InputSigned), !Signed);
+ Copy.setIsSigned(Signed);
+
+ assert(Copy.isSigned() == Signed);
+ return IntegralAP<Signed>(Copy);
+ }
+ static IntegralAP from(const Boolean &B) {
+ assert(false);
+ return IntegralAP::zero();
+ }
+
+ static IntegralAP zero() {
+ assert(false);
+ return IntegralAP(0);
+ }
+
+ static constexpr unsigned bitWidth() { return 128; }
+
+ APSInt toAPSInt(unsigned Bits = 0) const { return V; }
+ APValue toAPValue() const { return APValue(V); }
+
+ bool isZero() const { return V.isZero(); }
+ bool isPositive() const { return V.isNonNegative(); }
+ bool isNegative() const { return !V.isNonNegative(); }
+ bool isMin() const { return V.isMinValue(); }
+ bool isMax() const { return V.isMaxValue(); }
+ static bool isSigned() { return Signed; }
+ bool isMinusOne() const { return Signed && V == -1; }
+
+ unsigned countLeadingZeros() const { return V.countl_zero(); }
+
+ void print(llvm::raw_ostream &OS) const { OS << V; }
+
+ IntegralAP truncate(unsigned bitWidth) const { return V; }
+ IntegralAP<false> toUnsigned() const {
+ APSInt Copy = V;
+ Copy.setIsSigned(false);
+ return IntegralAP<false>(Copy);
+ }
+
+ ComparisonCategoryResult compare(const IntegralAP &RHS) const {
+ return Compare(V, RHS.V);
+ }
+
+ static bool increment(IntegralAP A, IntegralAP *R) {
+ *R = IntegralAP(A.V - 1);
----------------
tbaederr wrote:
Yes, definitely. I'll fix this but it's not important until I implement increment operators.
https://github.com/llvm/llvm-project/pull/65844
More information about the cfe-commits
mailing list