solang 0.3.4

Solang Solidity Compiler
Documentation
// SPDX-License-Identifier: Apache-2.0

use super::Bits;
use bitvec::prelude::BitArray;
use bitvec::prelude::Lsb0;
use num_bigint::{BigInt, Sign};
use num_traits::Signed;
use num_traits::Zero;
use std::collections::HashSet;
use std::fmt;

#[derive(Hash, Debug, Clone, PartialEq, Eq)]
pub(super) struct Value {
    // which bits are known
    pub(super) known_bits: BitArray<[u8; 32], Lsb0>,
    // value
    pub(super) value: BitArray<[u8; 32], Lsb0>,
    // type
    pub(super) bits: usize,
}

impl fmt::Display for Value {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        if self.all_known() {
            write!(
                f,
                "{}",
                BigInt::from_signed_bytes_le(&self.value.into_inner())
            )
        } else if self.all_unknown() {
            write!(f, "unknown")
        } else {
            write!(
                f,
                "{} k:{}",
                BigInt::from_signed_bytes_le(&self.value.into_inner()),
                hex::encode(self.value[0..self.bits].to_bitvec().as_raw_slice())
            )
        }
    }
}

#[allow(unused)]
fn dump_set(name: &str, set: &HashSet<Value>) {
    println!(
        "{}:{}",
        name,
        set.iter()
            .map(|v| format!("{v}"))
            .collect::<Vec<String>>()
            .join(",")
    );
}

/// Is the value set just a single constant
pub(super) fn is_single_constant(set: &HashSet<Value>) -> Option<BigInt> {
    if set.len() == 1 {
        let v = set.iter().next().unwrap();

        if v.all_known() {
            let digits = v.value[0..v.bits].to_bitvec();
            return Some(BigInt::from_signed_bytes_le(digits.as_raw_slice()));
        }
    }

    None
}

/// Get the maximum signed value in a set
pub(super) fn get_max_signed(set: &HashSet<Value>) -> Option<BigInt> {
    let mut m = BigInt::zero();

    for v in set {
        let (sign_known, sign) = v.sign();

        if !sign_known {
            return None;
        }

        let v = if sign {
            BigInt::from_signed_bytes_le(&v.get_signed_min_value().into_inner())
        } else {
            BigInt::from_signed_bytes_le(&v.get_signed_max_value().into_inner())
        };

        if v.abs() > m.abs() {
            m = v;
        }
    }

    Some(m)
}

/// Get the maximum unsigned value in a set
pub(super) fn get_max_unsigned(set: &HashSet<Value>) -> BigInt {
    let mut m = BigInt::zero();

    for v in set {
        let v = BigInt::from_bytes_le(Sign::Plus, &v.get_unsigned_max_value().into_inner());

        m = std::cmp::max(v, m);
    }

    m
}

impl Value {
    /// Calculate the unsigned min value. Higher bits than the type are 0
    pub(super) fn get_unsigned_min_value(&self) -> Bits {
        self.value & self.known_bits
    }

    /// Calculate the unsigned max value. Higher bits than the type are 0
    pub(super) fn get_unsigned_max_value(&self) -> Bits {
        (BitArray::new([!0u8; 32]) & !self.known_bits) | self.value
    }

    /// Return whether the sign is known and what value it is
    pub(super) fn sign(&self) -> (bool, bool) {
        let sign_bit = self.bits - 1;

        (self.known_bits[sign_bit], self.value[sign_bit])
    }

    /// Calculate the signed max value
    pub(super) fn get_signed_max_value(&self) -> Bits {
        let negative = match self.sign() {
            (true, sign) => sign,
            (false, _) => false,
        };

        if !negative {
            // we know the value is positive; same as unsigned
            self.get_unsigned_max_value()
        } else {
            // the value might be negative. So, we want to know which bits are zero
            let mut v = self.get_unsigned_min_value();
            v[self.bits - 1..].fill(true);
            v
        }
    }

    pub(super) fn get_signed_min_value(&self) -> Bits {
        let negative = match self.sign() {
            (true, sign) => sign,
            (false, _) => true,
        };

        if !negative {
            // we know the value is positive; same as unsigned
            self.get_unsigned_min_value()
        } else {
            // the value might be negative. So, we want to know which bits are zero
            let mut v = self.get_unsigned_max_value();
            v[self.bits - 1..].fill(true);
            v
        }
    }

    pub(super) fn all_known(&self) -> bool {
        self.known_bits[0..self.bits].all()
    }

    pub(super) fn all_unknown(&self) -> bool {
        self.known_bits[0..self.bits].not_any()
    }

    pub(super) fn unknown(bits: usize) -> Value {
        Value {
            value: BitArray::new([0u8; 32]),
            known_bits: BitArray::new([0u8; 32]),
            bits,
        }
    }
}