#![cfg_attr(not(feature = "std"), no_std)]
use account::SYSTEM_ACCOUNT_SIZE;
use fp_evm::PrecompileHandle;
use frame_support::dispatch::{GetDispatchInfo, PostDispatchInfo};
use frame_support::traits::{Currency, Polling};
use frame_system::pallet_prelude::BlockNumberFor;
use pallet_conviction_voting::Call as ConvictionVotingCall;
use pallet_conviction_voting::{
AccountVote, Casting, ClassLocksFor, Conviction, Delegating, Tally, TallyOf, Vote, Voting,
VotingFor,
};
use pallet_evm::{AddressMapping, Log};
use precompile_utils::prelude::*;
use sp_core::{Get, MaxEncodedLen, H160, H256, U256};
use sp_runtime::traits::{Dispatchable, StaticLookup};
use sp_std::marker::PhantomData;
use sp_std::vec::Vec;
#[cfg(test)]
mod mock;
#[cfg(test)]
mod tests;
type BalanceOf<Runtime> = <<Runtime as pallet_conviction_voting::Config>::Currency as Currency<
<Runtime as frame_system::Config>::AccountId,
>>::Balance;
type IndexOf<Runtime> = <<Runtime as pallet_conviction_voting::Config>::Polls as Polling<
Tally<
<<Runtime as pallet_conviction_voting::Config>::Currency as Currency<
<Runtime as frame_system::Config>::AccountId,
>>::Balance,
<Runtime as pallet_conviction_voting::Config>::MaxTurnout,
>,
>>::Index;
type ClassOf<Runtime> = <<Runtime as pallet_conviction_voting::Config>::Polls as Polling<
Tally<
<<Runtime as pallet_conviction_voting::Config>::Currency as Currency<
<Runtime as frame_system::Config>::AccountId,
>>::Balance,
<Runtime as pallet_conviction_voting::Config>::MaxTurnout,
>,
>>::Class;
type VotingOf<Runtime> = Voting<
BalanceOf<Runtime>,
<Runtime as frame_system::Config>::AccountId,
BlockNumberFor<Runtime>,
<<Runtime as pallet_conviction_voting::Config>::Polls as Polling<TallyOf<Runtime>>>::Index,
<Runtime as pallet_conviction_voting::Config>::MaxVotes,
>;
pub(crate) const SELECTOR_LOG_VOTED: [u8; 32] =
keccak256!("Voted(uint32,address,bool,uint256,uint8)");
pub(crate) const SELECTOR_LOG_VOTE_SPLIT: [u8; 32] =
keccak256!("VoteSplit(uint32,address,uint256,uint256)");
pub(crate) const SELECTOR_LOG_VOTE_SPLIT_ABSTAINED: [u8; 32] =
keccak256!("VoteSplitAbstained(uint32,address,uint256,uint256,uint256)");
pub(crate) const SELECTOR_LOG_VOTE_REMOVED: [u8; 32] = keccak256!("VoteRemoved(uint32,address)");
pub(crate) const SELECTOR_LOG_VOTE_REMOVED_FOR_TRACK: [u8; 32] =
keccak256!("VoteRemovedForTrack(uint32,uint16,address)");
pub(crate) const SELECTOR_LOG_VOTE_REMOVED_OTHER: [u8; 32] =
keccak256!("VoteRemovedOther(uint32,address,address,uint16)");
pub(crate) const SELECTOR_LOG_DELEGATED: [u8; 32] =
keccak256!("Delegated(uint16,address,address,uint256,uint8)");
pub(crate) const SELECTOR_LOG_UNDELEGATED: [u8; 32] = keccak256!("Undelegated(uint16,address)");
pub(crate) const SELECTOR_LOG_UNLOCKED: [u8; 32] = keccak256!("Unlocked(uint16,address)");
pub struct ConvictionVotingPrecompile<Runtime>(PhantomData<Runtime>);
#[precompile_utils::precompile]
impl<Runtime> ConvictionVotingPrecompile<Runtime>
where
Runtime: pallet_conviction_voting::Config + pallet_evm::Config + frame_system::Config,
BalanceOf<Runtime>: TryFrom<U256> + Into<U256>,
<Runtime as frame_system::Config>::RuntimeCall:
Dispatchable<PostInfo = PostDispatchInfo> + GetDispatchInfo,
<<Runtime as frame_system::Config>::RuntimeCall as Dispatchable>::RuntimeOrigin:
From<Option<Runtime::AccountId>>,
Runtime::AccountId: Into<H160>,
<Runtime as frame_system::Config>::RuntimeCall: From<ConvictionVotingCall<Runtime>>,
IndexOf<Runtime>: TryFrom<u32> + TryInto<u32>,
ClassOf<Runtime>: TryFrom<u16> + TryInto<u16>,
<Runtime as pallet_conviction_voting::Config>::Polls: Polling<
Tally<
<<Runtime as pallet_conviction_voting::Config>::Currency as Currency<
<Runtime as frame_system::Config>::AccountId,
>>::Balance,
<Runtime as pallet_conviction_voting::Config>::MaxTurnout,
>,
>,
<Runtime as pallet_evm::Config>::AddressMapping: AddressMapping<Runtime::AccountId>,
{
fn vote(
handle: &mut impl PrecompileHandle,
poll_index: u32,
vote: AccountVote<U256>,
) -> EvmResult {
let caller = handle.context().caller;
let (poll_index, vote, event) = Self::log_vote_event(handle, poll_index, vote)?;
let origin = Runtime::AddressMapping::into_account_id(caller);
let call = ConvictionVotingCall::<Runtime>::vote { poll_index, vote }.into();
<RuntimeHelper<Runtime>>::try_dispatch(handle, Some(origin).into(), call, 0)?;
event.record(handle)?;
Ok(())
}
#[precompile::public("voteYes(uint32,uint256,uint8)")]
fn vote_yes(
handle: &mut impl PrecompileHandle,
poll_index: u32,
vote_amount: U256,
conviction: u8,
) -> EvmResult {
Self::vote(
handle,
poll_index,
AccountVote::Standard {
vote: Vote {
aye: true,
conviction: Self::u8_to_conviction(conviction).in_field("conviction")?,
},
balance: vote_amount,
},
)
}
#[precompile::public("voteNo(uint32,uint256,uint8)")]
fn vote_no(
handle: &mut impl PrecompileHandle,
poll_index: u32,
vote_amount: U256,
conviction: u8,
) -> EvmResult {
Self::vote(
handle,
poll_index,
AccountVote::Standard {
vote: Vote {
aye: false,
conviction: Self::u8_to_conviction(conviction).in_field("conviction")?,
},
balance: vote_amount,
},
)
}
#[precompile::public("voteSplit(uint32,uint256,uint256)")]
fn vote_split(
handle: &mut impl PrecompileHandle,
poll_index: u32,
aye: U256,
nay: U256,
) -> EvmResult {
Self::vote(handle, poll_index, AccountVote::Split { aye, nay })
}
#[precompile::public("voteSplitAbstain(uint32,uint256,uint256,uint256)")]
fn vote_split_abstain(
handle: &mut impl PrecompileHandle,
poll_index: u32,
aye: U256,
nay: U256,
abstain: U256,
) -> EvmResult {
Self::vote(
handle,
poll_index,
AccountVote::SplitAbstain { aye, nay, abstain },
)
}
#[precompile::public("removeVote(uint32)")]
fn remove_vote(handle: &mut impl PrecompileHandle, poll_index: u32) -> EvmResult {
Self::rm_vote(handle, poll_index, None)
}
#[precompile::public("removeVoteForTrack(uint32,uint16)")]
fn remove_vote_for_track(
handle: &mut impl PrecompileHandle,
poll_index: u32,
track_id: u16,
) -> EvmResult {
Self::rm_vote(handle, poll_index, Some(track_id))
}
fn rm_vote(
handle: &mut impl PrecompileHandle,
poll_index: u32,
maybe_track_id: Option<u16>,
) -> EvmResult {
let caller = handle.context().caller;
let index = Self::u32_to_index(poll_index).in_field("pollIndex")?;
let (event, class) = if let Some(track_id) = maybe_track_id {
log::trace!(
target: "conviction-voting-precompile",
"Removing vote from poll {:?} for track {:?}",
index,
track_id,
);
(
log2(
handle.context().address,
SELECTOR_LOG_VOTE_REMOVED_FOR_TRACK,
H256::from_low_u64_be(poll_index as u64),
solidity::encode_event_data((track_id, Address(caller))),
),
Some(Self::u16_to_track_id(track_id).in_field("trackId")?),
)
} else {
log::trace!(
target: "conviction-voting-precompile",
"Removing vote from poll {:?}",
index,
);
(
log2(
handle.context().address,
SELECTOR_LOG_VOTE_REMOVED,
H256::from_low_u64_be(poll_index as u64),
solidity::encode_event_data(Address(caller)),
),
None,
)
};
let origin = Runtime::AddressMapping::into_account_id(caller);
let call = ConvictionVotingCall::<Runtime>::remove_vote { class, index };
RuntimeHelper::<Runtime>::try_dispatch(handle, Some(origin).into(), call, 0)?;
event.record(handle)?;
Ok(())
}
#[precompile::public("removeOtherVote(address,uint16,uint32)")]
fn remove_other_vote(
handle: &mut impl PrecompileHandle,
target: Address,
track_id: u16,
poll_index: u32,
) -> EvmResult {
let caller = handle.context().caller;
let event = log2(
handle.context().address,
SELECTOR_LOG_VOTE_REMOVED_OTHER,
H256::from_low_u64_be(poll_index as u64), solidity::encode_event_data((Address(caller), target, track_id)),
);
handle.record_log_costs(&[&event])?;
let class = Self::u16_to_track_id(track_id).in_field("trackId")?;
let index = Self::u32_to_index(poll_index).in_field("pollIndex")?;
let target = Runtime::AddressMapping::into_account_id(target.into());
let target: <Runtime::Lookup as StaticLookup>::Source =
Runtime::Lookup::unlookup(target.clone());
log::trace!(
target: "conviction-voting-precompile",
"Removing other vote from poll {:?}",
index
);
let origin = Runtime::AddressMapping::into_account_id(caller);
let call = ConvictionVotingCall::<Runtime>::remove_other_vote {
target,
class,
index,
};
RuntimeHelper::<Runtime>::try_dispatch(handle, Some(origin).into(), call, 0)?;
event.record(handle)?;
Ok(())
}
#[precompile::public("delegate(uint16,address,uint8,uint256)")]
fn delegate(
handle: &mut impl PrecompileHandle,
track_id: u16,
representative: Address,
conviction: u8,
amount: U256,
) -> EvmResult {
let caller = handle.context().caller;
let event = log2(
handle.context().address,
SELECTOR_LOG_DELEGATED,
H256::from_low_u64_be(track_id as u64), solidity::encode_event_data((Address(caller), representative, amount, conviction)),
);
handle.record_log_costs(&[&event])?;
let class = Self::u16_to_track_id(track_id).in_field("trackId")?;
let amount = Self::u256_to_amount(amount).in_field("amount")?;
let conviction = Self::u8_to_conviction(conviction).in_field("conviction")?;
log::trace!(target: "conviction-voting-precompile",
"Delegating vote to {:?} with balance {:?} and conviction {:?}",
representative, amount, conviction
);
let representative = Runtime::AddressMapping::into_account_id(representative.into());
let to: <Runtime::Lookup as StaticLookup>::Source =
Runtime::Lookup::unlookup(representative.clone());
let origin = Runtime::AddressMapping::into_account_id(caller);
let call = ConvictionVotingCall::<Runtime>::delegate {
class,
to,
conviction,
balance: amount,
};
RuntimeHelper::<Runtime>::try_dispatch(
handle,
Some(origin).into(),
call,
SYSTEM_ACCOUNT_SIZE,
)?;
event.record(handle)?;
Ok(())
}
#[precompile::public("undelegate(uint16)")]
fn undelegate(handle: &mut impl PrecompileHandle, track_id: u16) -> EvmResult {
let caller = handle.context().caller;
let event = log2(
handle.context().address,
SELECTOR_LOG_UNDELEGATED,
H256::from_low_u64_be(track_id as u64), solidity::encode_event_data(Address(caller)),
);
handle.record_log_costs(&[&event])?;
let class = Self::u16_to_track_id(track_id).in_field("trackId")?;
let origin = Runtime::AddressMapping::into_account_id(caller);
let call = ConvictionVotingCall::<Runtime>::undelegate { class };
RuntimeHelper::<Runtime>::try_dispatch(handle, Some(origin).into(), call, 0)?;
event.record(handle)?;
Ok(())
}
#[precompile::public("unlock(uint16,address)")]
fn unlock(handle: &mut impl PrecompileHandle, track_id: u16, target: Address) -> EvmResult {
let class = Self::u16_to_track_id(track_id).in_field("trackId")?;
let event = log2(
handle.context().address,
SELECTOR_LOG_UNLOCKED,
H256::from_low_u64_be(track_id as u64), solidity::encode_event_data(target),
);
handle.record_log_costs(&[&event])?;
let target: H160 = target.into();
let target = Runtime::AddressMapping::into_account_id(target);
let target: <Runtime::Lookup as StaticLookup>::Source =
Runtime::Lookup::unlookup(target.clone());
log::trace!(
target: "conviction-voting-precompile",
"Unlocking conviction-voting tokens for {:?}", target
);
let origin = Runtime::AddressMapping::into_account_id(handle.context().caller);
let call = ConvictionVotingCall::<Runtime>::unlock { class, target };
RuntimeHelper::<Runtime>::try_dispatch(handle, Some(origin).into(), call, 0)?;
event.record(handle)?;
Ok(())
}
#[precompile::public("votingFor(address,uint16)")]
#[precompile::view]
fn voting_for(
handle: &mut impl PrecompileHandle,
who: Address,
track_id: u16,
) -> EvmResult<OutputVotingFor> {
handle.record_db_read::<Runtime>(38 + VotingOf::<Runtime>::max_encoded_len())?;
let who = Runtime::AddressMapping::into_account_id(who.into());
let class = Self::u16_to_track_id(track_id).in_field("trackId")?;
let voting = <VotingFor<Runtime>>::get(&who, &class);
Ok(Self::voting_to_output(voting)?)
}
#[precompile::public("classLocksFor(address)")]
#[precompile::view]
fn class_locks_for(
handle: &mut impl PrecompileHandle,
who: Address,
) -> EvmResult<Vec<OutputClassLock>> {
handle.record_db_read::<Runtime>(
28 + ((2 * frame_support::traits::ClassCountOf::<
<Runtime as pallet_conviction_voting::Config>::Polls,
Tally<
<<Runtime as pallet_conviction_voting::Config>::Currency as Currency<
<Runtime as frame_system::Config>::AccountId,
>>::Balance,
<Runtime as pallet_conviction_voting::Config>::MaxTurnout,
>,
>::get()) as usize),
)?;
let who = Runtime::AddressMapping::into_account_id(who.into());
let class_locks_for = <ClassLocksFor<Runtime>>::get(&who);
let mut output = Vec::new();
for (track_id, amount) in class_locks_for {
output.push(OutputClassLock {
track: Self::track_id_to_u16(track_id)?,
amount: amount.into(),
});
}
Ok(output)
}
fn u8_to_conviction(conviction: u8) -> MayRevert<Conviction> {
conviction
.try_into()
.map_err(|_| RevertReason::custom("Must be an integer between 0 and 6 included").into())
}
fn u32_to_index(index: u32) -> MayRevert<IndexOf<Runtime>> {
index
.try_into()
.map_err(|_| RevertReason::value_is_too_large("index type").into())
}
fn u16_to_track_id(class: u16) -> MayRevert<ClassOf<Runtime>> {
class
.try_into()
.map_err(|_| RevertReason::value_is_too_large("trackId type").into())
}
fn track_id_to_u16(class: ClassOf<Runtime>) -> MayRevert<u16> {
class
.try_into()
.map_err(|_| RevertReason::value_is_too_large("trackId type").into())
}
fn u256_to_amount(value: U256) -> MayRevert<BalanceOf<Runtime>> {
value
.try_into()
.map_err(|_| RevertReason::value_is_too_large("balance type").into())
}
fn log_vote_event(
handle: &mut impl PrecompileHandle,
poll_index: u32,
vote: AccountVote<U256>,
) -> EvmResult<(IndexOf<Runtime>, AccountVote<BalanceOf<Runtime>>, Log)> {
let (contract_addr, caller) = (handle.context().address, handle.context().caller);
let (vote, event) = match vote {
AccountVote::Standard { vote, balance } => {
let event = log2(
contract_addr,
SELECTOR_LOG_VOTED,
H256::from_low_u64_be(poll_index as u64),
solidity::encode_event_data((
Address(caller),
vote.aye,
balance,
u8::from(vote.conviction),
)),
);
(
AccountVote::Standard {
vote,
balance: Self::u256_to_amount(balance).in_field("voteAmount")?,
},
event,
)
}
AccountVote::Split { aye, nay } => {
let event = log2(
contract_addr,
SELECTOR_LOG_VOTE_SPLIT,
H256::from_low_u64_be(poll_index as u64),
solidity::encode_event_data((Address(caller), aye, nay)),
);
(
AccountVote::Split {
aye: Self::u256_to_amount(aye).in_field("aye")?,
nay: Self::u256_to_amount(nay).in_field("nay")?,
},
event,
)
}
AccountVote::SplitAbstain { aye, nay, abstain } => {
let event = log2(
contract_addr,
SELECTOR_LOG_VOTE_SPLIT_ABSTAINED,
H256::from_low_u64_be(poll_index as u64),
solidity::encode_event_data((Address(caller), aye, nay, abstain)),
);
(
AccountVote::SplitAbstain {
aye: Self::u256_to_amount(aye).in_field("aye")?,
nay: Self::u256_to_amount(nay).in_field("nay")?,
abstain: Self::u256_to_amount(abstain).in_field("abstain")?,
},
event,
)
}
};
handle.record_log_costs(&[&event])?;
Ok((Self::u32_to_index(poll_index)?, vote, event))
}
fn voting_to_output(voting: VotingOf<Runtime>) -> MayRevert<OutputVotingFor> {
let output = match voting {
Voting::Casting(Casting {
votes,
delegations,
prior,
}) => {
let mut output_votes = Vec::new();
for (poll_index, account_vote) in votes {
let poll_index: u32 = poll_index
.try_into()
.map_err(|_| RevertReason::value_is_too_large("index type"))?;
let account_vote = match account_vote {
AccountVote::Standard { vote, balance } => OutputAccountVote {
is_standard: true,
standard: StandardVote {
vote: OutputVote {
aye: vote.aye,
conviction: vote.conviction.into(),
},
balance: balance.into(),
},
..Default::default()
},
AccountVote::Split { aye, nay } => OutputAccountVote {
is_split: true,
split: SplitVote {
aye: aye.into(),
nay: nay.into(),
},
..Default::default()
},
AccountVote::SplitAbstain { aye, nay, abstain } => OutputAccountVote {
is_split_abstain: true,
split_abstain: SplitAbstainVote {
aye: aye.into(),
nay: nay.into(),
abstain: abstain.into(),
},
..Default::default()
},
};
output_votes.push(PollAccountVote {
poll_index,
account_vote,
});
}
OutputVotingFor {
is_casting: true,
casting: OutputCasting {
votes: output_votes,
delegations: Delegations {
votes: delegations.votes.into(),
capital: delegations.capital.into(),
},
prior: PriorLock {
balance: prior.locked().into(),
},
},
..Default::default()
}
}
Voting::Delegating(Delegating {
balance,
target,
conviction,
delegations,
prior,
}) => OutputVotingFor {
is_delegating: true,
delegating: OutputDelegating {
balance: balance.into(),
target: Address(target.into()),
conviction: conviction.into(),
delegations: Delegations {
votes: delegations.votes.into(),
capital: delegations.capital.into(),
},
prior: PriorLock {
balance: prior.locked().into(),
},
},
..Default::default()
},
};
Ok(output)
}
}
#[derive(Default, solidity::Codec)]
pub struct OutputClassLock {
track: u16,
amount: U256,
}
#[derive(Default, solidity::Codec)]
pub struct OutputVotingFor {
is_casting: bool,
is_delegating: bool,
casting: OutputCasting,
delegating: OutputDelegating,
}
#[derive(Default, solidity::Codec)]
pub struct OutputCasting {
votes: Vec<PollAccountVote>,
delegations: Delegations,
prior: PriorLock,
}
#[derive(Default, solidity::Codec)]
pub struct PollAccountVote {
poll_index: u32,
account_vote: OutputAccountVote,
}
#[derive(Default, solidity::Codec)]
pub struct OutputDelegating {
balance: U256,
target: Address,
conviction: u8,
delegations: Delegations,
prior: PriorLock,
}
#[derive(Default, solidity::Codec)]
pub struct OutputAccountVote {
is_standard: bool,
is_split: bool,
is_split_abstain: bool,
standard: StandardVote,
split: SplitVote,
split_abstain: SplitAbstainVote,
}
#[derive(Default, solidity::Codec)]
pub struct StandardVote {
vote: OutputVote,
balance: U256,
}
#[derive(Default, solidity::Codec)]
pub struct OutputVote {
aye: bool,
conviction: u8,
}
#[derive(Default, solidity::Codec)]
pub struct SplitVote {
aye: U256,
nay: U256,
}
#[derive(Default, solidity::Codec)]
pub struct SplitAbstainVote {
aye: U256,
nay: U256,
abstain: U256,
}
#[derive(Default, solidity::Codec)]
pub struct Delegations {
votes: U256,
capital: U256,
}
#[derive(Default, solidity::Codec)]
pub struct PriorLock {
balance: U256,
}