mirror of
https://github.com/0glabs/0g-storage-node.git
synced 2025-01-15 17:45:17 +00:00
192 lines
6.1 KiB
Rust
192 lines
6.1 KiB
Rust
use super::metrics::*;
|
|
use crate::recall_range::RecallRange;
|
|
use crate::{MineRangeConfig, PoraLoader};
|
|
use blake2::{Blake2b512, Digest};
|
|
use contract_interface::zgs_flow::MineContext;
|
|
use ethereum_types::{H256, U256};
|
|
use lighthouse_metrics::inc_counter;
|
|
use storage::log_store::MineLoadChunk;
|
|
use tiny_keccak::{Hasher, Keccak};
|
|
use zgs_spec::{BYTES_PER_SCRATCHPAD, BYTES_PER_SEAL, SECTORS_PER_LOAD, SECTORS_PER_SEAL};
|
|
|
|
pub const BLAKE2B_OUTPUT_BYTES: usize = 64;
|
|
pub const KECCAK256_OUTPUT_BYTES: usize = 32;
|
|
|
|
fn keccak(input: impl AsRef<[u8]>) -> [u8; KECCAK256_OUTPUT_BYTES] {
|
|
let mut hasher = Keccak::v256();
|
|
let mut output = [0u8; 32];
|
|
hasher.update(input.as_ref());
|
|
hasher.finalize(&mut output);
|
|
output
|
|
}
|
|
|
|
pub(crate) struct Miner<'a> {
|
|
pub range: RecallRange,
|
|
pub miner_id: &'a H256,
|
|
pub context: &'a MineContext,
|
|
pub target_quality: &'a U256,
|
|
pub loader: &'a dyn PoraLoader,
|
|
pub mine_range_config: &'a MineRangeConfig,
|
|
}
|
|
#[derive(Debug)]
|
|
pub struct AnswerWithoutProof {
|
|
pub context_digest: H256,
|
|
pub context_flow_root: H256,
|
|
pub nonce: H256,
|
|
pub miner_id: H256,
|
|
pub range: RecallRange,
|
|
pub recall_position: u64,
|
|
pub seal_offset: usize,
|
|
pub sealed_data: [u8; BYTES_PER_SEAL],
|
|
}
|
|
|
|
impl<'a> Miner<'a> {
|
|
pub async fn batch_iteration(
|
|
&self,
|
|
nonce: H256,
|
|
batch_size: usize,
|
|
) -> Option<AnswerWithoutProof> {
|
|
for i in 0..batch_size {
|
|
let bytes = i.to_ne_bytes();
|
|
let mut current_nonce = nonce;
|
|
for (pos, b) in bytes.into_iter().enumerate() {
|
|
current_nonce.0[pos] ^= b;
|
|
}
|
|
if let Some(answer) = self.iteration(current_nonce).await {
|
|
return Some(answer);
|
|
}
|
|
}
|
|
None
|
|
}
|
|
|
|
pub async fn iteration(&self, nonce: H256) -> Option<AnswerWithoutProof> {
|
|
inc_counter(&SCRATCH_PAD_ITER_COUNT);
|
|
let ScratchPad {
|
|
scratch_pad,
|
|
recall_seed,
|
|
pad_seed,
|
|
} = self.make_scratch_pad(&nonce);
|
|
|
|
let recall_position = self.range.load_position(recall_seed)?;
|
|
if !self.mine_range_config.is_covered(recall_position).unwrap() {
|
|
trace!(
|
|
"recall offset not in range: recall_offset={}",
|
|
recall_position,
|
|
);
|
|
return None;
|
|
}
|
|
|
|
inc_counter(&LOADING_COUNT);
|
|
let MineLoadChunk {
|
|
loaded_chunk,
|
|
avalibilities,
|
|
} = self
|
|
.loader
|
|
.load_sealed_data(recall_position / SECTORS_PER_LOAD as u64)
|
|
.await?;
|
|
|
|
let scratch_pad: [[u8; BYTES_PER_SEAL]; BYTES_PER_SCRATCHPAD / BYTES_PER_SEAL] =
|
|
unsafe { std::mem::transmute(scratch_pad) };
|
|
|
|
for ((idx, mut sealed_data), scratch_pad) in loaded_chunk
|
|
.into_iter()
|
|
.enumerate()
|
|
.zip(scratch_pad.iter().cycle())
|
|
.zip(avalibilities.into_iter())
|
|
.filter_map(|(data, avaliable)| avaliable.then_some(data))
|
|
{
|
|
inc_counter(&PAD_MIX_COUNT);
|
|
// Rust can optimize this loop well.
|
|
for (x, y) in sealed_data.iter_mut().zip(scratch_pad.iter()) {
|
|
*x ^= y;
|
|
}
|
|
|
|
let quality = self.pora(idx, &sealed_data, pad_seed);
|
|
let difficulty_scale_x64 = self
|
|
.range
|
|
.difficulty_scale_x64(self.context.flow_length.as_u64());
|
|
|
|
if quality <= (self.target_quality / difficulty_scale_x64) << 64 {
|
|
debug!(
|
|
"Find a PoRA valid answer, quality: {}, target_quality {}, scale {:.3}",
|
|
U256::MAX / quality,
|
|
U256::MAX / self.target_quality,
|
|
difficulty_scale_x64.as_u128() as f64 / u64::MAX as f64
|
|
);
|
|
inc_counter(&HIT_COUNT);
|
|
// Undo mix data when find a valid solition
|
|
for (x, y) in sealed_data.iter_mut().zip(scratch_pad.iter()) {
|
|
*x ^= y;
|
|
}
|
|
return Some(AnswerWithoutProof {
|
|
context_digest: H256::from(self.context.digest),
|
|
context_flow_root: self.context.flow_root.into(),
|
|
nonce,
|
|
miner_id: *self.miner_id,
|
|
range: self.range,
|
|
recall_position: recall_position + idx as u64 * SECTORS_PER_SEAL as u64,
|
|
seal_offset: idx,
|
|
sealed_data,
|
|
});
|
|
}
|
|
}
|
|
None
|
|
}
|
|
|
|
fn make_scratch_pad(&self, nonce: &H256) -> ScratchPad {
|
|
let mut digest: [u8; BLAKE2B_OUTPUT_BYTES] = {
|
|
let mut hasher = Blake2b512::new();
|
|
hasher.update(self.miner_id);
|
|
hasher.update(nonce);
|
|
hasher.update(self.context.digest);
|
|
hasher.update(self.range.digest());
|
|
hasher.finalize().into()
|
|
};
|
|
|
|
let pad_seed = digest;
|
|
|
|
let mut scratch_pad =
|
|
[[0u8; BLAKE2B_OUTPUT_BYTES]; BYTES_PER_SCRATCHPAD / BLAKE2B_OUTPUT_BYTES];
|
|
for scratch_pad_cell in scratch_pad.iter_mut() {
|
|
digest = Blake2b512::new().chain_update(digest).finalize().into();
|
|
*scratch_pad_cell = digest;
|
|
}
|
|
|
|
let scratch_pad: [u8; BYTES_PER_SCRATCHPAD] = unsafe { std::mem::transmute(scratch_pad) };
|
|
let recall_seed: [u8; KECCAK256_OUTPUT_BYTES] = keccak(digest);
|
|
|
|
ScratchPad {
|
|
scratch_pad,
|
|
recall_seed,
|
|
pad_seed,
|
|
}
|
|
}
|
|
|
|
#[inline]
|
|
fn pora(
|
|
&self,
|
|
seal_index: usize,
|
|
mixed_data: &[u8; BYTES_PER_SEAL],
|
|
pad_seed: [u8; BLAKE2B_OUTPUT_BYTES],
|
|
) -> U256 {
|
|
let mut hasher = Blake2b512::new();
|
|
hasher.update([0u8; 24]);
|
|
hasher.update(seal_index.to_be_bytes());
|
|
|
|
hasher.update(pad_seed);
|
|
hasher.update([0u8; 32]);
|
|
|
|
hasher.update(mixed_data);
|
|
|
|
let digest = hasher.finalize();
|
|
|
|
U256::from_big_endian(&digest[0..32])
|
|
}
|
|
}
|
|
|
|
struct ScratchPad {
|
|
scratch_pad: [u8; BYTES_PER_SCRATCHPAD],
|
|
recall_seed: [u8; KECCAK256_OUTPUT_BYTES],
|
|
pad_seed: [u8; BLAKE2B_OUTPUT_BYTES],
|
|
}
|