mirror of
https://github.com/Cuprate/cuprate.git
synced 2025-01-11 05:15:24 +00:00
initial add_aux_pow
impl
This commit is contained in:
parent
fd84940ee6
commit
46bf19861a
1 changed files with 152 additions and 16 deletions
|
@ -13,7 +13,7 @@ use cuprate_constants::{
|
||||||
rpc::{RESTRICTED_BLOCK_COUNT, RESTRICTED_BLOCK_HEADER_RANGE},
|
rpc::{RESTRICTED_BLOCK_COUNT, RESTRICTED_BLOCK_HEADER_RANGE},
|
||||||
};
|
};
|
||||||
use cuprate_helper::{
|
use cuprate_helper::{
|
||||||
cast::{u64_to_usize, usize_to_u64},
|
cast::{u32_to_usize, u64_to_usize, usize_to_u64},
|
||||||
map::split_u128_into_low_high_bits,
|
map::split_u128_into_low_high_bits,
|
||||||
};
|
};
|
||||||
use cuprate_p2p_core::{client::handshaker::builder::DummyAddressBook, ClearNet, Network};
|
use cuprate_p2p_core::{client::handshaker::builder::DummyAddressBook, ClearNet, Network};
|
||||||
|
@ -403,7 +403,7 @@ async fn get_info(
|
||||||
let (database_size, free_space) = blockchain::database_size(&mut state.blockchain_read).await?;
|
let (database_size, free_space) = blockchain::database_size(&mut state.blockchain_read).await?;
|
||||||
let (database_size, free_space) = if restricted {
|
let (database_size, free_space) = if restricted {
|
||||||
// <https://github.com/monero-project/monero/blob/cc73fe71162d564ffda8e549b79a350bca53c454/src/rpc/core_rpc_server.cpp#L131-L134>
|
// <https://github.com/monero-project/monero/blob/cc73fe71162d564ffda8e549b79a350bca53c454/src/rpc/core_rpc_server.cpp#L131-L134>
|
||||||
fn round_up(value: u64, quantum: u64) -> u64 {
|
const fn round_up(value: u64, quantum: u64) -> u64 {
|
||||||
(value + quantum - 1) / quantum * quantum
|
(value + quantum - 1) / quantum * quantum
|
||||||
}
|
}
|
||||||
let database_size = round_up(database_size, 5 * 1024 * 1024 * 1024);
|
let database_size = round_up(database_size, 5 * 1024 * 1024 * 1024);
|
||||||
|
@ -964,8 +964,9 @@ async fn add_aux_pow(
|
||||||
mut state: CupratedRpcHandler,
|
mut state: CupratedRpcHandler,
|
||||||
request: AddAuxPowRequest,
|
request: AddAuxPowRequest,
|
||||||
) -> Result<AddAuxPowResponse, Error> {
|
) -> Result<AddAuxPowResponse, Error> {
|
||||||
let hex = hex::decode(request.blocktemplate_blob)?;
|
if request.aux_pow.is_empty() {
|
||||||
let block_template = Block::read(&mut hex.as_slice())?;
|
return Err(anyhow!("Empty `aux_pow` vector"));
|
||||||
|
}
|
||||||
|
|
||||||
let aux_pow = request
|
let aux_pow = request
|
||||||
.aux_pow
|
.aux_pow
|
||||||
|
@ -975,31 +976,166 @@ async fn add_aux_pow(
|
||||||
let hash = helper::hex_to_hash(aux.hash)?;
|
let hash = helper::hex_to_hash(aux.hash)?;
|
||||||
Ok(cuprate_types::AuxPow { id, hash })
|
Ok(cuprate_types::AuxPow { id, hash })
|
||||||
})
|
})
|
||||||
.collect::<Result<Vec<_>, Error>>()?;
|
.collect::<Result<Box<[_]>, Error>>()?;
|
||||||
|
// Some of the code below requires that the
|
||||||
|
// `.len()` of certain containers are the same.
|
||||||
|
// Boxed slices are used over `Vec` to slightly
|
||||||
|
// safe-guard against accidently pushing to it.
|
||||||
|
|
||||||
let resp = todo!();
|
// --- BEGIN AUX POW IMPL ---
|
||||||
// let resp =
|
|
||||||
// blockchain_manager::add_aux_pow(&mut state.blockchain_manager, block_template, aux_pow)
|
|
||||||
// .await?;
|
|
||||||
|
|
||||||
let blocktemplate_blob = hex::encode(resp.blocktemplate_blob);
|
// Original impl:
|
||||||
let blockhashing_blob = hex::encode(resp.blockhashing_blob);
|
// <https://github.com/monero-project/monero/blob/cc73fe71162d564ffda8e549b79a350bca53c454/src/rpc/core_rpc_server.cpp#L2110-L2206>
|
||||||
let merkle_root = hex::encode(resp.merkle_root);
|
|
||||||
let aux_pow = resp
|
let len = aux_pow.len();
|
||||||
.aux_pow
|
|
||||||
.into_iter()
|
let mut path_domain = 1_usize;
|
||||||
|
while 1 << path_domain < len {
|
||||||
|
path_domain += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
let nonce = 0_u32;
|
||||||
|
let mut collision = true;
|
||||||
|
let mut slots: Box<[u32]> = vec![0; len].into_boxed_slice(); // INVARIANT: this must be the same `.len()` as `aux_pow`
|
||||||
|
|
||||||
|
for nonce in 0..u32::MAX {
|
||||||
|
let slot_seen: Vec<bool> = vec![false; len];
|
||||||
|
|
||||||
|
collision = false;
|
||||||
|
|
||||||
|
slots.iter_mut().for_each(|i| *i = u32::MAX);
|
||||||
|
|
||||||
|
for i in &mut slots {
|
||||||
|
let slot_u32: u32 = todo!("const uint32_t slot = cryptonote::get_aux_slot(aux_pow[idx].first, nonce, aux_pow.size());");
|
||||||
|
let slot = u32_to_usize(slot_u32);
|
||||||
|
|
||||||
|
if slot >= len {
|
||||||
|
return Err(anyhow!("Computed slot is out of range"));
|
||||||
|
}
|
||||||
|
|
||||||
|
if slot_seen[slot] {
|
||||||
|
collision = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
slot_seen[slot] = true;
|
||||||
|
*i = slot_u32;
|
||||||
|
}
|
||||||
|
|
||||||
|
if !collision {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if collision {
|
||||||
|
return Err(anyhow!("Failed to find a suitable nonce"));
|
||||||
|
}
|
||||||
|
|
||||||
|
let slots = slots;
|
||||||
|
|
||||||
|
// FIXME: use iterator version.
|
||||||
|
let (aux_pow_id_raw, aux_pow_raw) = {
|
||||||
|
let mut aux_pow_id_raw = Vec::<[u8; 32]>::with_capacity(len);
|
||||||
|
let mut aux_pow_raw = Vec::<[u8; 32]>::with_capacity(len);
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
aux_pow.len(),
|
||||||
|
slots.len(),
|
||||||
|
"these need to be the same or else the below .zip() doesn't make sense"
|
||||||
|
);
|
||||||
|
|
||||||
|
for (aux_pow, slot) in aux_pow.iter().zip(&slots) {
|
||||||
|
if u32_to_usize(*slot) >= len {
|
||||||
|
return Err(anyhow!("Slot value out of range"));
|
||||||
|
}
|
||||||
|
|
||||||
|
aux_pow_id_raw.push(aux_pow.id);
|
||||||
|
aux_pow_raw.push(aux_pow.hash);
|
||||||
|
}
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
slots.len(),
|
||||||
|
aux_pow_raw.len(),
|
||||||
|
"these need to be the same or else the below .zip() doesn't make sense"
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
aux_pow_raw.len(),
|
||||||
|
aux_pow_id_raw.len(),
|
||||||
|
"these need to be the same or else the below .zip() doesn't make sense"
|
||||||
|
);
|
||||||
|
|
||||||
|
for (slot, aux_pow) in slots.iter().zip(&aux_pow) {
|
||||||
|
let slot = u32_to_usize(*slot);
|
||||||
|
|
||||||
|
if slot >= len {
|
||||||
|
return Err(anyhow!("Slot value out of range"));
|
||||||
|
}
|
||||||
|
|
||||||
|
aux_pow_raw[slot] = aux_pow.hash;
|
||||||
|
aux_pow_id_raw[slot] = aux_pow.id;
|
||||||
|
}
|
||||||
|
|
||||||
|
(
|
||||||
|
aux_pow_id_raw.into_boxed_slice(),
|
||||||
|
aux_pow_raw.into_boxed_slice(),
|
||||||
|
)
|
||||||
|
};
|
||||||
|
|
||||||
|
// let crypto_tree_hash = || todo!(&aux_pow_raw, aux_pow_raw.len());
|
||||||
|
let crypto_tree_hash = todo!();
|
||||||
|
|
||||||
|
let block_template = {
|
||||||
|
let hex = hex::decode(request.blocktemplate_blob)?;
|
||||||
|
Block::read(&mut hex.as_slice())?
|
||||||
|
};
|
||||||
|
|
||||||
|
fn remove_field_from_tx_extra() -> Result<(), ()> {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
if remove_field_from_tx_extra().is_err() {
|
||||||
|
return Err(anyhow!("Error removing existing merkle root"));
|
||||||
|
}
|
||||||
|
|
||||||
|
fn add_mm_merkle_root_to_tx_extra() -> Result<(), ()> {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
if add_mm_merkle_root_to_tx_extra().is_err() {
|
||||||
|
return Err(anyhow!("Error adding merkle root"));
|
||||||
|
}
|
||||||
|
|
||||||
|
fn invalidate_hashes() {
|
||||||
|
// block_template.invalidate_hashes();
|
||||||
|
// block_template.miner_tx.invalidate_hashes();
|
||||||
|
todo!();
|
||||||
|
}
|
||||||
|
|
||||||
|
invalidate_hashes();
|
||||||
|
|
||||||
|
let blocktemplate_blob = block_template.serialize();
|
||||||
|
let blockhashing_blob = block_template.serialize_pow_hash();
|
||||||
|
let merkle_root: [u8; 32] = todo!();
|
||||||
|
let merkle_tree_depth = todo!();
|
||||||
|
|
||||||
|
let blocktemplate_blob = hex::encode(blocktemplate_blob);
|
||||||
|
let blockhashing_blob = hex::encode(blockhashing_blob);
|
||||||
|
let merkle_root = hex::encode(merkle_root);
|
||||||
|
let aux_pow = IntoIterator::into_iter(aux_pow) // must be explicit due to `boxed_slice_into_iter`
|
||||||
.map(|aux| AuxPow {
|
.map(|aux| AuxPow {
|
||||||
id: hex::encode(aux.id),
|
id: hex::encode(aux.id),
|
||||||
hash: hex::encode(aux.hash),
|
hash: hex::encode(aux.hash),
|
||||||
})
|
})
|
||||||
.collect::<Vec<AuxPow>>();
|
.collect::<Vec<AuxPow>>();
|
||||||
|
|
||||||
|
// --- END AUX POW IMPL ---
|
||||||
|
|
||||||
Ok(AddAuxPowResponse {
|
Ok(AddAuxPowResponse {
|
||||||
base: ResponseBase::OK,
|
base: ResponseBase::OK,
|
||||||
blocktemplate_blob,
|
blocktemplate_blob,
|
||||||
blockhashing_blob,
|
blockhashing_blob,
|
||||||
merkle_root,
|
merkle_root,
|
||||||
merkle_tree_depth: resp.merkle_tree_depth,
|
merkle_tree_depth,
|
||||||
aux_pow,
|
aux_pow,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue