Skip to content

Commit

Permalink
feat(frontend): load TFHE-rs integer type from saved params (JSON)
Browse files Browse the repository at this point in the history
  • Loading branch information
youben11 committed Nov 8, 2024
1 parent 84fd038 commit 798220e
Show file tree
Hide file tree
Showing 7 changed files with 133 additions and 39 deletions.
58 changes: 58 additions & 0 deletions frontends/concrete-python/concrete/fhe/tfhers/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,9 @@
tfhers module to represent, and compute on tfhers integer values.
"""

import json
from math import log2

from .bridge import new_bridge
from .dtypes import (
CryptoParams,
Expand All @@ -18,3 +21,58 @@
)
from .tracing import from_native, to_native
from .values import TFHERSInteger


def get_type_from_params(
path_to_params_json: str, is_signed: bool, precision: int
) -> TFHERSIntegerType:
"""Get a TFHE-rs integer type from TFHE-rs parameters in JSON format.
Args:
path_to_params_json (str): path to the TFHE-rs parameters (JSON format)
is_signed (bool): sign of the result type
precision (int): precision of the result type
Returns:
TFHERSIntegerType: constructed type from the loaded parameters
"""

# Read crypto parameters from TFHE-rs in the json file
with open(path_to_params_json) as f:
crypto_param_dict = json.load(f)

lwe_dim = crypto_param_dict["lwe_dimension"]
glwe_dim = crypto_param_dict["glwe_dimension"]
poly_size = crypto_param_dict["polynomial_size"]
pbs_base_log = crypto_param_dict["pbs_base_log"]
pbs_level = crypto_param_dict["pbs_level"]
msg_width = int(log2(crypto_param_dict["message_modulus"]))
carry_width = int(log2(crypto_param_dict["carry_modulus"]))
lwe_noise_distr = crypto_param_dict["lwe_noise_distribution"]["Gaussian"]["std"]
glwe_noise_distr = crypto_param_dict["glwe_noise_distribution"]["Gaussian"]["std"]
encryption_key_choice = (
EncryptionKeyChoice.BIG
if crypto_param_dict["encryption_key_choice"] == "Big"
else EncryptionKeyChoice.SMALL
)

assert glwe_dim == 1, "glwe dim must be 1"
assert encryption_key_choice == EncryptionKeyChoice.BIG, "encryption_key_choice must be BIG"

tfhers_params = CryptoParams(
lwe_dimension=lwe_dim,
glwe_dimension=glwe_dim,
polynomial_size=poly_size,
pbs_base_log=pbs_base_log,
pbs_level=pbs_level,
lwe_noise_distribution=lwe_noise_distr,
glwe_noise_distribution=glwe_noise_distr,
encryption_key_choice=encryption_key_choice,
)
return TFHERSIntegerType(
is_signed=is_signed,
bit_width=precision,
carry_width=carry_width,
msg_width=msg_width,
params=tfhers_params,
)
6 changes: 6 additions & 0 deletions frontends/concrete-python/examples/tfhers/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,12 @@ python example.py keygen -s $TDIR/tfhers_sk -o $TDIR/concrete_sk -k $TDIR/concre
../../tests/tfhers-utils/target/release/tfhers_utils encrypt-with-key --value 73 --ciphertext $TDIR/tfhers_ct_2 --client-key $TDIR/tfhers_client_key
```

{% hint style="info" %}

If you have tensor inputs, then you can encrypt by passing your flat tensor in `--value`. Concrete will take care of reshaping the values to the corresponding shape. For example `--value=1,2,3,4` can represent a 2 by 2 tensor, or a flat vector of 4 values.

{% endhint %}

## Compute in TFHE-rs

```sh
Expand Down
38 changes: 6 additions & 32 deletions frontends/concrete-python/examples/tfhers/example.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,42 +7,16 @@
from concrete import fhe
from concrete.fhe import tfhers

########## Params #####################
LWE_DIM = 909
GLWE_DIM = 1
POLY_SIZE = 4096
PBS_BASE_LOG = 15
PBS_LEVEL = 2
MSG_WIDTH = 2
CARRY_WIDTH = 3
ENCRYPTION_KEY_CHOICE = tfhers.EncryptionKeyChoice.BIG
LWE_NOISE_DISTR = 0
GLWE_NOISE_DISTR = 2.168404344971009e-19
#######################################

assert GLWE_DIM == 1, "glwe dim must be 1"

### Options ###########################
# These parameters were saved by running the tfhers_utils utility:
# tfhers_utils save-params tfhers_params.json
TFHERS_PARAMS_FILE = "tfhers_params.json"
FHEUINT_PRECISION = 8
IS_SIGNED = False
#######################################


tfhers_params = tfhers.CryptoParams(
lwe_dimension=LWE_DIM,
glwe_dimension=GLWE_DIM,
polynomial_size=POLY_SIZE,
pbs_base_log=PBS_BASE_LOG,
pbs_level=PBS_LEVEL,
lwe_noise_distribution=LWE_NOISE_DISTR,
glwe_noise_distribution=GLWE_NOISE_DISTR,
encryption_key_choice=ENCRYPTION_KEY_CHOICE,
)
tfhers_type = tfhers.TFHERSIntegerType(
is_signed=False,
bit_width=FHEUINT_PRECISION,
carry_width=CARRY_WIDTH,
msg_width=MSG_WIDTH,
params=tfhers_params,
tfhers_type = tfhers.get_type_from_params(
TFHERS_PARAMS_FILE, IS_SIGNED, FHEUINT_PRECISION
)
tfhers_int = partial(tfhers.TFHERSInteger, tfhers_type)

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
{"lwe_dimension":902,"glwe_dimension":1,"polynomial_size":4096,"lwe_noise_distribution":{"Gaussian":{"std":1.0994794733558207e-6,"mean":0.0}},"glwe_noise_distribution":{"Gaussian":{"std":2.168404344971009e-19,"mean":0.0}},"pbs_base_log":15,"pbs_level":2,"ks_base_log":3,"ks_level":6,"message_modulus":4,"carry_modulus":8,"max_noise_level":10,"log2_p_fail":-64.084,"ciphertext_modulus":{"modulus":0,"scalar_bits":64},"encryption_key_choice":"Big"}
31 changes: 31 additions & 0 deletions frontends/concrete-python/tests/tfhers-utils/Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions frontends/concrete-python/tests/tfhers-utils/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ edition = "2021"

bincode = "1.3.3"
serde = "1"
serde_json = "1.0.128"

clap = { version = "4.5.16", features = ["derive"] }

Expand Down
37 changes: 30 additions & 7 deletions frontends/concrete-python/tests/tfhers-utils/src/main.rs
Original file line number Diff line number Diff line change
@@ -1,18 +1,23 @@
use clap::{Arg, ArgAction, Command};
use core::panic;
use serde::de::DeserializeOwned;
use std::fs;
use std::io::Write;
use std::path::Path;

use clap::{Arg, ArgAction, Command};

use tfhe::core_crypto::prelude::LweSecretKey;
use tfhe::named::Named;
use tfhe::prelude::*;
use tfhe::safe_serialization::{safe_deserialize, safe_serialize};
use tfhe::shortint::{ClassicPBSParameters, EncryptionKeyChoice};
use tfhe::{generate_keys, set_server_key, ClientKey, ConfigBuilder, FheInt8, FheUint8, ServerKey};
use tfhe::{
generate_keys, set_server_key, ClientKey, ConfigBuilder, FheInt8, FheUint8, ServerKey,
Unversionize, Versionize,
};

use serde::de::DeserializeOwned;
use serde::Serialize;
use tfhe::named::Named;
use tfhe::{Unversionize, Versionize};

use tfhe::safe_serialization::{safe_deserialize, safe_serialize};
use serde_json;

const BLOCK_PARAMS: ClassicPBSParameters = tfhe::shortint::prelude::PARAM_MESSAGE_2_CARRY_3_KS_PBS;
const SERIALIZE_SIZE_LIMIT: u64 = 1_000_000_000;
Expand Down Expand Up @@ -405,6 +410,17 @@ fn main() {
.num_args(1),
),
)
.subcommand(
Command::new("save-params")
.short_flag('p')
.long_flag("save-params")
.about("save TFHE-rs parameters used into a file (JSON)")
.arg(
Arg::new("filename")
.help("filename to save TFHE-rs parameters to")
.required(true),
),
)
.get_matches();

match matches.subcommand() {
Expand Down Expand Up @@ -482,6 +498,13 @@ fn main() {
keygen(client_key_path, server_key_path, output_lwe_path.unwrap())
}
}
Some(("save-params", save_params_mtches)) => {
let filename = save_params_mtches.get_one::<String>("filename").unwrap();
let json_string = serde_json::to_string(&BLOCK_PARAMS).unwrap();
let path = Path::new(filename);
let mut file = fs::File::create(path).unwrap();
file.write_all(json_string.as_bytes()).unwrap();
}
_ => unreachable!(), // If all subcommands are defined above, anything else is unreachable
}
}

0 comments on commit 798220e

Please sign in to comment.