2022-11-17 14:24:14 +01:00
|
|
|
pragma circom 2.1.0;
|
2022-11-17 13:49:58 +01:00
|
|
|
|
|
|
|
include "../node_modules/circomlib/circuits/sha256/sha256.circom";
|
|
|
|
include "../node_modules/circomlib/circuits/poseidon.circom";
|
2022-11-17 13:59:58 +01:00
|
|
|
|
|
|
|
include "../node_modules/circomlib/circuits/bitify.circom";
|
2022-11-17 13:49:58 +01:00
|
|
|
include "tree.circom";
|
|
|
|
|
|
|
|
template HashCheck(blockSize) {
|
|
|
|
signal input block[blockSize];
|
|
|
|
//signal input blockHash[256];
|
|
|
|
signal input blockHash;
|
|
|
|
|
|
|
|
//component hash = Sha256(blockSize);
|
|
|
|
component hash = Poseidon(blockSize);
|
|
|
|
for (var i = 0; i < blockSize; i++) {
|
|
|
|
hash.inputs[i] <== block[i];
|
|
|
|
}
|
|
|
|
hash.out === blockHash; //is this checking the whole array?
|
|
|
|
// is this enough or do we need output?
|
|
|
|
}
|
|
|
|
|
2022-11-17 14:23:31 +01:00
|
|
|
template CheckInclusion(nLevels) {
|
|
|
|
signal input index;
|
|
|
|
signal input chunkHash;
|
|
|
|
signal input treeSiblings[nLevels];
|
|
|
|
signal input root;
|
|
|
|
|
|
|
|
component num2Bits = Num2Bits(nLevels);
|
|
|
|
num2Bits.in <== index;
|
|
|
|
|
|
|
|
component inclusionProof = MerkleTreeInclusionProof(nLevels);
|
|
|
|
inclusionProof.leaf <== chunkHash;
|
|
|
|
for (var j = 0; j < nLevels; j++) {
|
|
|
|
inclusionProof.siblings[j] <== treeSiblings[j];
|
|
|
|
inclusionProof.pathIndices[j] <== num2Bits.out[j];
|
|
|
|
}
|
|
|
|
root === inclusionProof.root;
|
|
|
|
}
|
|
|
|
|
2022-11-17 13:49:58 +01:00
|
|
|
template StorageProver(blockSize, qLen, nLevels) {
|
2022-11-17 14:00:16 +01:00
|
|
|
// blockSize: size of block in bits (sha256), or in symbols (Poseidon)
|
2022-11-17 13:49:58 +01:00
|
|
|
// qLen: query length, i.e. number if indices to be proven
|
|
|
|
// nLevels: size of Merkle Tree in the manifest
|
|
|
|
signal input chunks[qLen][blockSize];
|
|
|
|
//signal input chunkHashes[qLen][256];
|
|
|
|
signal input chunkHashes[qLen];
|
|
|
|
signal input indices[qLen];
|
|
|
|
signal input treeSiblings[qLen][nLevels];
|
|
|
|
|
|
|
|
signal input root;
|
|
|
|
|
|
|
|
//check that chunks hash to given hashes
|
|
|
|
for (var i = 0; i < qLen; i++) {
|
2022-11-29 14:54:58 +01:00
|
|
|
parallel HashCheck(blockSize)(
|
|
|
|
chunks[i],
|
|
|
|
chunkHashes[i]
|
|
|
|
);
|
2022-11-17 13:49:58 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
//check that the tree is correct
|
2022-11-17 14:28:37 +01:00
|
|
|
// - check indices against limits TODO
|
2022-11-17 13:49:58 +01:00
|
|
|
// - convert indices to treePathIndices
|
|
|
|
// - check chunkHash and treeSiblings according to treePathIndices against root
|
|
|
|
|
|
|
|
for (var i = 0; i < qLen; i++) {
|
2022-11-17 14:37:02 +01:00
|
|
|
parallel CheckInclusion(nLevels)(
|
2022-11-17 14:28:16 +01:00
|
|
|
indices[i],
|
|
|
|
chunkHashes[i],
|
|
|
|
treeSiblings[i],
|
|
|
|
root);
|
2022-11-17 13:49:58 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//component main {public [blockHash]} = HashCheck(512);
|
2022-11-17 14:28:37 +01:00
|
|
|
//template StorageProver(blockSize, qLen, nLevels) {
|
2022-11-17 13:49:58 +01:00
|
|
|
//component main {public [indices]} = StorageProver(512, 1, 10);
|
2022-11-24 09:17:18 +01:00
|
|
|
component main {public [indices, root]} = StorageProver(2, 1, 20);
|