tfhe 1.6.1

TFHE-rs is a fully homomorphic encryption (FHE) library that implements Zama's variant of TFHE.
Documentation
use tfhe_versionable::Versionize;

use crate::backward_compatibility::keys::KeySwitchingKeyVersions;
use crate::high_level_api::integers::{FheIntId, FheUintId};
use crate::high_level_api::re_randomization::ReRandomizationMetadata;
use crate::integer::BooleanBlock;
use crate::named::Named;
use crate::prelude::FheKeyswitch;
pub use crate::shortint::parameters::key_switching::ShortintKeySwitchingParameters;
use crate::{ClientKey, FheBool, FheInt, FheUint, ServerKey, Tag};
use std::fmt::{Display, Formatter};

#[derive(Copy, Clone, Debug)]
pub struct IncompatibleParameters;

impl Display for IncompatibleParameters {
    fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
        write!(f, "{self:?}")
    }
}

impl std::error::Error for IncompatibleParameters {}

#[derive(serde::Deserialize, serde::Serialize, Versionize)]
#[versionize(KeySwitchingKeyVersions)]
pub struct KeySwitchingKey {
    key: crate::integer::key_switching_key::KeySwitchingKey,
    tag_in: Tag,
    tag_out: Tag,
}

impl KeySwitchingKey {
    pub fn new(
        key_pair_from: (&ClientKey, &ServerKey),
        key_pair_to: (&ClientKey, &ServerKey),
    ) -> Result<Self, IncompatibleParameters> {
        let params_from = key_pair_from.0.key.block_parameters();
        let params_to = key_pair_to.0.key.block_parameters();

        if params_to != params_from {
            return Err(IncompatibleParameters);
        }

        // params_to == params_from, so we can use the parameters from params_to
        let params = ShortintKeySwitchingParameters {
            ks_base_log: params_to.ks_base_log(),
            ks_level: params_to.ks_level(),
            destination_key: params_to.encryption_key_choice(),
        };

        Ok(Self::with_parameters(key_pair_from, key_pair_to, params))
    }

    pub fn with_parameters(
        key_pair_from: (&ClientKey, &ServerKey),
        key_pair_to: (&ClientKey, &ServerKey),
        params: ShortintKeySwitchingParameters,
    ) -> Self {
        Self {
            key: crate::integer::key_switching_key::KeySwitchingKey::new(
                (&key_pair_from.0.key.key, Some(&key_pair_from.1.key.key)),
                (&key_pair_to.0.key.key, &key_pair_to.1.key.key),
                params,
            ),
            tag_in: key_pair_from.0.tag.clone(),
            tag_out: key_pair_to.0.tag.clone(),
        }
    }

    pub fn tag_in(&self) -> &Tag {
        &self.tag_in
    }

    pub fn tag_out(&self) -> &Tag {
        &self.tag_out
    }
}

impl<Id> FheKeyswitch<FheUint<Id>> for KeySwitchingKey
where
    Id: FheUintId,
{
    fn keyswitch(&self, input: &FheUint<Id>) -> FheUint<Id> {
        let radix = input.ciphertext.on_cpu();
        let casted = self.key.cast(&*radix);
        FheUint::new(
            casted,
            self.tag_out.clone(),
            ReRandomizationMetadata::default(),
        )
    }
}

impl<Id> FheKeyswitch<FheInt<Id>> for KeySwitchingKey
where
    Id: FheIntId,
{
    fn keyswitch(&self, input: &FheInt<Id>) -> FheInt<Id> {
        let radix = input.ciphertext.on_cpu();
        let casted = self.key.cast(&*radix);
        FheInt::new(
            casted,
            self.tag_out.clone(),
            ReRandomizationMetadata::default(),
        )
    }
}

impl FheKeyswitch<FheBool> for KeySwitchingKey {
    fn keyswitch(&self, input: &FheBool) -> FheBool {
        let boolean_block = input.ciphertext.on_cpu();
        let casted = self.key.key.cast(boolean_block.as_ref());
        FheBool::new(
            BooleanBlock::new_unchecked(casted),
            self.tag_out.clone(),
            ReRandomizationMetadata::default(),
        )
    }
}

impl Named for KeySwitchingKey {
    const NAME: &'static str = "high_level_api::KeySwitchingKey";
}