Skip to content

Commit

Permalink
feat(client): Interop entrypoint
Browse files Browse the repository at this point in the history
  • Loading branch information
clabby committed Jan 14, 2025
1 parent 46faa59 commit f2d7447
Show file tree
Hide file tree
Showing 12 changed files with 507 additions and 214 deletions.
1 change: 1 addition & 0 deletions Cargo.lock

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

5 changes: 5 additions & 0 deletions bin/client/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ kona-driver.workspace = true
kona-preimage.workspace = true
kona-executor.workspace = true
kona-proof.workspace = true
kona-proof-interop.workspace = true
kona-std-fpvm.workspace = true
kona-std-fpvm-proc.workspace = true

Expand Down Expand Up @@ -54,3 +55,7 @@ client-tracing = ["kona-std-fpvm/tracing"]
[[bin]]
name = "kona"
path = "src/kona.rs"

[[bin]]
name = "kona-int"
path = "src/kona_interop.rs"
249 changes: 249 additions & 0 deletions bin/client/src/interop.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,249 @@
//! Multi-chain, interoperable fault proof program entrypoint.
use alloc::{string::ToString, sync::Arc};
use alloy_consensus::Sealed;
use alloy_primitives::{Bytes, B256};
use alloy_rlp::Decodable;
use core::fmt::Debug;
use kona_driver::{Driver, DriverError};
use kona_executor::{ExecutorError, KonaHandleRegister, TrieDBProvider};
use kona_preimage::{
errors::PreimageOracleError, CommsClient, HintWriterClient, PreimageKey, PreimageKeyType,
PreimageOracleClient,
};
use kona_proof::{
errors::OracleProviderError,
executor::KonaExecutor,
l1::{OracleBlobProvider, OracleL1ChainProvider, OraclePipeline},
l2::OracleL2ChainProvider,
sync::new_pipeline_cursor,
CachingOracle,
};
use kona_proof_interop::{
pre_state::{OptimisticBlock, PreState, TransitionState},
BootInfo, HintType,
};
use thiserror::Error;
use tracing::{error, info};

/// An error that can occur when running the fault proof program.
#[derive(Error, Debug)]
pub enum FaultProofProgramError {
/// The claim is invalid.
#[error("Invalid claim. Expected {0}, actual {1}")]
InvalidClaim(B256, B256),
/// An error occurred in the Oracle provider.
#[error(transparent)]
OracleProviderError(#[from] OracleProviderError),
/// An error occurred in the driver.
#[error(transparent)]
Driver(#[from] DriverError<ExecutorError>),
/// An error occurred during RLP decoding.
#[error("RLP decoding error: {0}")]
RLPDecodingError(#[from] alloy_rlp::Error),
}

/// Executes the fault proof program with the given [PreimageOracleClient] and [HintWriterClient].
#[inline]
pub async fn run<P, H>(
oracle_client: P,
hint_client: H,
handle_register: Option<
KonaHandleRegister<
OracleL2ChainProvider<CachingOracle<P, H>>,
OracleL2ChainProvider<CachingOracle<P, H>>,
>,
>,
) -> Result<(), FaultProofProgramError>
where
P: PreimageOracleClient + Send + Sync + Debug + Clone,
H: HintWriterClient + Send + Sync + Debug + Clone,
{
const ORACLE_LRU_SIZE: usize = 1024;

////////////////////////////////////////////////////////////////
// PROLOGUE //
////////////////////////////////////////////////////////////////

let oracle = Arc::new(CachingOracle::new(ORACLE_LRU_SIZE, oracle_client, hint_client));
let boot = match BootInfo::load(oracle.as_ref()).await {
Ok(boot) => Arc::new(boot),
Err(e) => {
error!(target: "client", "Failed to load boot info: {:?}", e);
return Err(e.into());
}
};

// Load in the pre-state from the preimage oracle and fetch the L2 safe head block hash.
let pre =
PreState::decode(&mut read_raw_pre_state(oracle.as_ref(), boot.as_ref()).await?.as_ref())?;
let safe_head_hash = fetch_l2_safe_head_hash(oracle.as_ref(), &pre).await?;

// Instantiate the L1 EL + CL provider and the L2 EL provider.
let mut l1_provider = OracleL1ChainProvider::new(boot.l1_head, oracle.clone());
let mut l2_provider =
OracleL2ChainProvider::new(safe_head_hash, boot.rollup_config.clone(), oracle.clone());
let beacon = OracleBlobProvider::new(oracle.clone());

// Fetch the safe head's block header.
let safe_head = l2_provider
.header_by_hash(safe_head_hash)
.map(|header| Sealed::new_unchecked(header, safe_head_hash))?;

// Translate the claimed timestamp to an L2 block number.
let claimed_l2_block_number = (boot.claimed_l2_timestamp - boot.rollup_config.genesis.l2_time) /
boot.rollup_config.block_time;

// If the claimed L2 block number is less than the safe head of the L2 chain, the claim is
// invalid.
if claimed_l2_block_number < safe_head.number {
error!(
target: "client",
"Claimed L2 block number {claimed} is less than the safe head {safe}",
claimed = claimed_l2_block_number,
safe = safe_head.number
);
return Err(FaultProofProgramError::InvalidClaim(
boot.agreed_pre_state,
boot.claimed_post_state,
));
}

// In the case where the agreed upon L2 pre-state is the same as the claimed L2 post-state,
// trace extension is detected and we can skip the derivation and execution steps.
if boot.agreed_pre_state == boot.claimed_post_state {
info!(
target: "client",
"Trace extension detected. State transition is already agreed upon.",
);
return Ok(());
}

////////////////////////////////////////////////////////////////
// DERIVATION & EXECUTION //
////////////////////////////////////////////////////////////////

// Create a new derivation driver with the given boot information and oracle.
let cursor =
new_pipeline_cursor(&boot.rollup_config, safe_head, &mut l1_provider, &mut l2_provider)
.await?;
let cfg = Arc::new(boot.rollup_config.clone());
let pipeline = OraclePipeline::new(
cfg.clone(),
cursor.clone(),
oracle.clone(),
beacon,
l1_provider.clone(),
l2_provider.clone(),
);
let executor = KonaExecutor::new(&cfg, l2_provider.clone(), l2_provider, handle_register, None);
let mut driver = Driver::new(cursor, executor, pipeline);

// Run the derivation pipeline until we are able to produce the output root of the claimed
// L2 block.
let (number, block_hash, output_root) =
driver.advance_to_target(&boot.rollup_config, Some(claimed_l2_block_number)).await?;

////////////////////////////////////////////////////////////////
// EPILOGUE //
////////////////////////////////////////////////////////////////

// TODO(interop): incl. L2 block hash.
let optimistic_block = OptimisticBlock::new(block_hash, output_root);
let transition_state = match pre {
PreState::SuperRoot(super_root) => {
TransitionState::new(super_root, alloc::vec![optimistic_block], 1)
}
PreState::TransitionState(mut ts) => {
ts.pending_progress.push(optimistic_block);
ts.step += 1;
ts
}
};

if transition_state.hash() != boot.claimed_post_state {
error!(
target: "client",
"Failed to validate L2 block #{number} with output root {output_root}",
number = number,
output_root = output_root
);
return Err(FaultProofProgramError::InvalidClaim(
transition_state.hash(),
boot.claimed_post_state,
));
}

info!(
target: "client",
"Successfully validated L2 block #{number} with output root {output_root}",
number = number,
output_root = output_root
);

Ok(())
}

/// Reads the raw pre-state from the preimage oracle.
async fn read_raw_pre_state<O>(
caching_oracle: &O,
boot_info: &BootInfo,
) -> Result<Bytes, OracleProviderError>
where
O: CommsClient,
{
caching_oracle
.write(&HintType::AgreedPreState.encode_with(&[boot_info.agreed_pre_state.as_ref()]))
.await
.map_err(OracleProviderError::Preimage)?;
let pre = caching_oracle
.get(PreimageKey::new(*boot_info.agreed_pre_state, PreimageKeyType::Keccak256))
.await
.map_err(OracleProviderError::Preimage)?;

if pre.is_empty() {
return Err(OracleProviderError::Preimage(PreimageOracleError::Other(
"Invalid pre-state preimage".to_string(),
)));
}

Ok(Bytes::from(pre))
}

/// Fetches the safe head hash of the L2 chain based on the agreed upon L2 output root in the
/// [BootInfo].
async fn fetch_l2_safe_head_hash<O>(
caching_oracle: &O,
pre: &PreState,
) -> Result<B256, OracleProviderError>
where
O: CommsClient,
{
// Fetch the output root of the safe head block for the current L2 chain.
let rich_output = match pre {
PreState::SuperRoot(super_root) => {
super_root.output_roots.first().ok_or(OracleProviderError::Preimage(
PreimageOracleError::Other("No output roots in super root".to_string()),
))?
}
PreState::TransitionState(transition_state) => {
transition_state.pre_state.output_roots.get(transition_state.step as usize).ok_or(
OracleProviderError::Preimage(PreimageOracleError::Other(
"No output roots in transition state's pending progress".to_string(),
)),
)?
}
};

caching_oracle
.write(
&HintType::L2OutputRoot.encode_with(&[rich_output.chain_id.to_be_bytes().as_slice()]),
)
.await
.map_err(OracleProviderError::Preimage)?;
let output_preimage = caching_oracle
.get(PreimageKey::new(*rich_output.output_root, PreimageKeyType::Keccak256))
.await
.map_err(OracleProviderError::Preimage)?;
output_preimage[96..128].try_into().map_err(OracleProviderError::SliceConversion)
}
2 changes: 1 addition & 1 deletion bin/client/src/kona.rs
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,7 @@ fn main() -> Result<(), String> {
.expect("Failed to set tracing subscriber");
}

kona_proof::block_on(kona_client::run(
kona_proof::block_on(kona_client::single::run(
ORACLE_READER,
HINT_WRITER,
Some(precompiles::fpvm_handle_register),
Expand Down
47 changes: 47 additions & 0 deletions bin/client/src/kona_interop.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
#![doc = include_str!("../README.md")]
#![warn(missing_debug_implementations, missing_docs, unreachable_pub, rustdoc::all)]
#![deny(unused_must_use, rust_2018_idioms)]
#![cfg_attr(docsrs, feature(doc_cfg, doc_auto_cfg))]
#![no_std]
#![cfg_attr(any(target_arch = "mips", target_arch = "riscv64"), no_main)]

extern crate alloc;

use alloc::string::String;
use kona_preimage::{HintWriter, OracleReader};
use kona_std_fpvm::{FileChannel, FileDescriptor};
use kona_std_fpvm_proc::client_entry;

mod precompiles;

/// The global preimage oracle reader pipe.
static ORACLE_READER_PIPE: FileChannel =
FileChannel::new(FileDescriptor::PreimageRead, FileDescriptor::PreimageWrite);

/// The global hint writer pipe.
static HINT_WRITER_PIPE: FileChannel =
FileChannel::new(FileDescriptor::HintRead, FileDescriptor::HintWrite);

/// The global preimage oracle reader.
static ORACLE_READER: OracleReader<FileChannel> = OracleReader::new(ORACLE_READER_PIPE);

/// The global hint writer.
static HINT_WRITER: HintWriter<FileChannel> = HintWriter::new(HINT_WRITER_PIPE);

#[client_entry(100_000_000)]
fn main() -> Result<(), String> {
#[cfg(feature = "client-tracing")]
{
use kona_std_fpvm::tracing::FpvmTracingSubscriber;

let subscriber = FpvmTracingSubscriber::new(tracing::Level::INFO);
tracing::subscriber::set_global_default(subscriber)
.expect("Failed to set tracing subscriber");
}

kona_proof::block_on(kona_client::interop::run(
ORACLE_READER,
HINT_WRITER,
Some(precompiles::fpvm_handle_register),
))
}
Loading

0 comments on commit f2d7447

Please sign in to comment.