Caution
Following implementation of FrodoPIR private information retrieval scheme is designed from scratch, having zero third-party dependencies. It's not yet audited, avoid using it in production!
FrodoPIR: Simple, Scalable, Single-Server Private Information Retrieval
FrodoPIR is a very simple, stateful, single-server index-based Private Information Retrieval (PIR) scheme, built on top of Learning With Error (LWE) problem, proposed in https://ia.cr/2022/981.
FrodoPIR protocol can be split into offline and online phases s.t. offline phase can solely be performed by the server, doesn't require any input from clients. As soon as public parameters become available from server, client can begin preprocessing queries, making them ready for quick future use. A simplified description of the protocol is given below. See figure 1 of https://ia.cr/2022/981 for more details.
-
Offline Phase
-
server_setup
: Server samples pseudo-random matrix$A$ , from seed$\mu$ and sets up database as matrix$D$ , which has blowup factor of <3.5x, over the original database size. Server prepares public parameter$(\mu, M)$ . -
client_setup
: Client downloads public parameter$(\mu, M)$ , setups up internal state. -
client_prepare_query
: Client preprocesses a query by storing$(b, c)$ s.t.$b$ is a randomly distributed LWE sample vector.
-
-
Online Phase
-
client_query
: When client is ready to make a query, it pops a pair of$(b, c)$ from it's internal cache and sends slightly mutated vector$b$ , as query to the server. -
server_respond
: Server responds to client's query, returning back response vector$\tilde{c}$ . -
client_process_response
: Client decodes server response, obtaining content of queried database row.
-
To paint a more practical picture, imagine, we have a database with
Machine Type | Machine | Kernel | Compiler | Memory Read Speed |
---|---|---|---|---|
aarch64 server | AWS EC2 m8g.8xlarge |
Linux 6.8.0-1018-aws aarch64 |
GCC 13.2.0 |
28.25 GB/s |
x86_64 server | AWS EC2 m7i.8xlarge |
Linux 6.8.0-1018-aws x86_64 |
GCC 13.2.0 |
10.33 GB/s |
and this implementation of FrodoPIR is compiled with specified compiler, while also passing -O3 -march=native -flto
compiler optimization flags.
Note
Memory read speed is measured using $ sysbench memory --memory-block-size=1G --memory-total-size=20G --memory-oper=read run
command.
Step | (a) Time Taken on aarch64 server |
(b) Time Taken on x86_64 server |
Ratio a / b |
---|---|---|---|
server_setup |
41 seconds | 60.1 seconds | 0.68 |
client_setup |
21.7 seconds | 20.5 seconds | 1.05 |
client_preprocess_query |
39.4 milliseconds | 65.6 milliseconds | 0.6 |
client_query |
146 microseconds | 454 microseconds | 0.32 |
server_respond |
41.6 milliseconds | 150 milliseconds | 0.27 |
client_process_response |
782 microseconds | 1257 microseconds | 0.62 |
So, bandwidth of the server_respond
algorithm, which needs to traverse through the whole processed database, is
- (a) For
aarch64
server: 24.03 GB/s - (b) For
x86_64
server: 6.66 GB/s
Here I'm maintaining a zero-dependency, header-only C++20 library implementation of FrodoPIR scheme, supporting all parameter sets, as suggested in table 5 of https://ia.cr/2022/981. Using this library is very easy, follow here.
- A C++ compiler with support for compiling C++20 code.
$ g++ --version
g++ (Ubuntu 14-20240412-0ubuntu1) 14.0.1 20240412 (experimental) [master r14-9935-g67e1433a94f]
- System development utility programs such as
make
andcmake
. - For testing functional correctness of this PIR scheme, you need to globally install
google-test
library and its headers. Follow guide @ https://github.com/google/googletest/tree/main/googletest#standalone-cmake-project, if you don't have it installed. - For benchmarking this PIR scheme, you must have
google-benchmark
header and library globally installed. I found guide @ https://github.com/google/benchmark#installation helpful.
Note
If you are on a machine running GNU/Linux kernel and you want to obtain CPU cycle count for digital signature scheme routines, you should consider building google-benchmark
library with libPFM support, following https://gist.github.com/itzmeanjan/05dc3e946f635d00c5e0b21aae6203a7, a step-by-step guide. Find more about libPFM @ https://perfmon2.sourceforge.net.
Tip
Git submodule based dependencies will generally be imported automatically, but in case that doesn't work, you can manually initialize and update them by issuing $ git submodule update --init
from inside the root of this repository.
For ensuring functional correctness of this implementation of FrodoPIR scheme, issue
make test -j # Run tests without any sort of sanitizers, with default C++ compiler.
CXX=clang++ make test -j # Switch to non-default compiler, by setting variable `CXX`.
make debug_asan_test -j # Run tests with AddressSanitizer enabled, with `-O1`.
make release_asan_test -j # Run tests with AddressSanitizer enabled, with `-O3 -march=native`.
make debug_ubsan_test -j # Run tests with UndefinedBehaviourSanitizer enabled, with `-O1`.
make release_ubsan_test -j # Run tests with UndefinedBehaviourSanitizer enabled, with `-O3 -march=native`.
PASSED TESTS (4/4):
162 ms: build/test/test.out FrodoPIR.MatrixOperations
3256 ms: build/test/test.out FrodoPIR.ClientQueryCacheStateTransition
7399 ms: build/test/test.out FrodoPIR.ParsingDatabaseAndSerializingDatabaseMatrix
66962 ms: build/test/test.out FrodoPIR.PrivateInformationRetrieval
Note
There is a help menu, which introduces you to all available commands; just run make
from the root directory of this project.
Benchmarking of all 6 algorithms of FrodoPIR scheme can be done, by issuing
# For switching to non-default compiler, set `CXX` variable, right before invoking following command.
make benchmark -j # If you haven't built google-benchmark library with libPFM support.
make perf -j # If you have built google-benchmark library with libPFM support.
Caution
You must put all the CPU cores on performance mode before running benchmark program, follow guide @ https://github.com/google/benchmark/blob/main/docs/reducing_variance.md.
- On AWS EC2 Instance
m8g.8xlarge
: Benchmark result in JSON format @ bench_result_on_Linux_6.8.0-1018-aws_aarch64_with_g++_13. - On AWS EC2 Instance
m7i.8xlarge
: Benchmark result in JSON format @ bench_result_on_Linux_6.8.0-1018-aws_x86_64_with_g++_13.
Note
More about AWS EC2 instances @ https://aws.amazon.com/ec2/instance-types.
FrodoPIR is a header-only C++20 library implementing all recommended variants (see table 5) in https://ia.cr/2022/98. FrodoPIR header files live ./include
directory, while additional dependency sha3
and RandomShake
header files live under sha3/include
and RandomShake/include
, respectively.
- Let's begin by cloning the repository.
# Just clones FrodoPIR source tree, but not its submodule -based dependencies.
git clone https://github.com/itzmeanjan/frodoPIR.git
# Clones FrodoPIR source tree and also imports submodule -based dependencies.
git clone https://github.com/itzmeanjan/frodoPIR.git --recurse-submodules
- Move inside the directory holding the cloned repository and import all git submodule -based dependencies.
pushd frodoPIR
make test -j # Also runs tests
popd
- Now that we've all the dependencies to use FrodoPIR header-only library, let's run our example program, by issuing
make example
.
Original database row bytes : bf71e04189fff486c062cf1e814bedfc2205422807da319d4ac6f5a956d63e48
PIR decoded database row bytes : bf71e04189fff486c062cf1e814bedfc2205422807da319d4ac6f5a956d63e48