Skip to content
This repository has been archived by the owner on Nov 15, 2023. It is now read-only.

Deprecate Currency; introduce holds and freezing into fungible traits #12951

Merged
merged 174 commits into from
Mar 18, 2023
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
174 commits
Select commit Hold shift + click to select a range
ccd2ebb
First reworking of fungibles API
gavofyork Dec 14, 2022
de93255
New API and docs
gavofyork Dec 16, 2022
31ba6f2
More fungible::* API improvements
gavofyork Dec 16, 2022
0c8a27c
New ref-counting logic for old API
gavofyork Dec 19, 2022
97f7460
Missing files
gavofyork Dec 19, 2022
5027cb1
Fixes
gavofyork Dec 19, 2022
b42a687
Use the new transfer logic
gavofyork Dec 20, 2022
77d4c8e
Use fungibles for the dispatchables
gavofyork Dec 20, 2022
4485cd6
Use shelve/restore names
gavofyork Dec 20, 2022
67d8a7d
Locking works with total balance.
gavofyork Dec 22, 2022
77152ab
Merge remote-tracking branch 'origin/master' into gav-hold-freeze
gavofyork Dec 23, 2022
04a4c75
repotting and removal
gavofyork Dec 27, 2022
e84fbb5
Separate Holds from Reserves
gavofyork Jan 6, 2023
11306ed
Introduce freezes
gavofyork Jan 16, 2023
872bbcc
Merge remote-tracking branch 'origin/master' into gav-hold-freeze
gavofyork Jan 17, 2023
3f33588
Missing files
gavofyork Jan 17, 2023
900fef3
Merge remote-tracking branch 'origin/master' into gav-hold-freeze
gavofyork Jan 18, 2023
359d858
Tests for freezing
gavofyork Jan 18, 2023
428a41c
Fix hold+freeze combo
gavofyork Jan 19, 2023
f009268
More tests
gavofyork Jan 19, 2023
1d5a2d8
Fee-free dispatchable for upgrading accounts
gavofyork Jan 19, 2023
3807bb4
Benchmarks and a few fixes
gavofyork Jan 19, 2023
db5d596
Another test
gavofyork Jan 20, 2023
c609d74
Docs and refactor to avoid blanket impls
gavofyork Jan 20, 2023
3af6d09
Repot
gavofyork Jan 20, 2023
cd9343a
Fit out ItemOf fully
gavofyork Jan 20, 2023
5f5e6d1
Add events to Balanced traits
gavofyork Jan 20, 2023
176f92d
Introduced events into Hold traits
gavofyork Jan 20, 2023
1e0219a
Fix Assets pallet tests
gavofyork Jan 20, 2023
7cfd446
Assets benchmarks pass
gavofyork Jan 20, 2023
0f92a6b
Missing files and fixes
gavofyork Jan 21, 2023
4745cb8
Fixes
gavofyork Jan 21, 2023
4ee2a44
Fixes
gavofyork Jan 23, 2023
aa0eccf
Benchmarks fixes
gavofyork Jan 23, 2023
14beb70
Merge remote-tracking branch 'origin/master' into gav-hold-freeze
gavofyork Jan 23, 2023
84049b1
Fix balance benchmarks
gavofyork Jan 23, 2023
857efce
Formatting
gavofyork Jan 23, 2023
bf24b91
Expose fungible sub modules
gavofyork Jan 23, 2023
0ab291c
Merge branch 'gav-hold-freeze' into gav-nis-hold
gavofyork Jan 23, 2023
72fce5f
Move NIS to fungible API
gavofyork Jan 23, 2023
b2cae2a
Fix broken impl and add test
gavofyork Jan 23, 2023
020d11a
Merge branch 'gav-hold-freeze' into gav-nis-hold
gavofyork Jan 23, 2023
4b28afb
Fix tests
gavofyork Jan 23, 2023
d4da1a1
API for `transfer_and_hold`
gavofyork Jan 23, 2023
49fd3fc
Merge branch 'gav-hold-freeze' into gav-nis-hold
gavofyork Jan 23, 2023
bee7038
Use composite APIs
gavofyork Jan 23, 2023
f438f86
Merge branch 'gav-nis-hold' into gav-hold-freeze
gavofyork Jan 23, 2023
947d2b3
Formatting
gavofyork Jan 23, 2023
b469b45
Upgraded event
gavofyork Jan 23, 2023
ecb1765
Fixes
gavofyork Jan 23, 2023
79cf354
Fixes
gavofyork Jan 23, 2023
4c28b9d
Fixes
gavofyork Jan 23, 2023
01caead
Fixes
gavofyork Jan 23, 2023
83f47cb
Repot tests and some fixed
gavofyork Jan 24, 2023
16bdba7
Fix some bits
gavofyork Jan 24, 2023
cad82e8
Fix dust tests
gavofyork Jan 24, 2023
7b0b55c
Rename `set_balance`
gavofyork Jan 26, 2023
30de46d
becomes
gavofyork Jan 26, 2023
81dedbd
Move dust handling to fungibles API
gavofyork Jan 27, 2023
e0c1dc5
Formatting
gavofyork Jan 27, 2023
63e1698
Merge branch 'master' into gav-hold-freeze
gavofyork Jan 27, 2023
b210e19
Fixes and more refactoring
gavofyork Jan 27, 2023
6289a64
Fixes
gavofyork Jan 27, 2023
732e102
Fixes
gavofyork Jan 27, 2023
607b36a
Fixes
gavofyork Jan 27, 2023
37b999a
Fixes
gavofyork Jan 27, 2023
be3b91a
Fixes
gavofyork Jan 27, 2023
cdbd9b1
Merge branch 'master' into gav-hold-freeze
gavofyork Jan 27, 2023
dec5e42
Fixes
gavofyork Jan 28, 2023
242a8a1
Merge branch 'gav-hold-freeze' of github.com:paritytech/substrate int…
gavofyork Jan 28, 2023
f8439c1
Fixes
gavofyork Jan 28, 2023
1beffe4
Fixes
gavofyork Jan 28, 2023
75d5924
Fixes
gavofyork Jan 28, 2023
804f776
Use reducible_balance for better correctness on fees
gavofyork Feb 7, 2023
a5b6971
Reducing hold to zero should remove entry.
gavofyork Feb 7, 2023
c982d1b
Add test
gavofyork Feb 7, 2023
696b859
Docs
gavofyork Feb 7, 2023
32ee5dc
Update frame/support/src/traits/tokens/fungibles/hold.rs
gavofyork Feb 7, 2023
576f8a1
Update frame/support/src/traits/tokens/fungibles/regular.rs
gavofyork Feb 7, 2023
4a124cb
Update frame/support/src/traits/tokens/fungible/hold.rs
gavofyork Feb 7, 2023
574cdec
Update frame/support/src/traits/tokens/fungible/regular.rs
gavofyork Feb 7, 2023
c0c320d
Docs
gavofyork Feb 7, 2023
07d8132
Merge branch 'gav-hold-freeze' of github.com:paritytech/substrate int…
gavofyork Feb 7, 2023
b632c9f
Docs
gavofyork Feb 7, 2023
436fea7
Docs
gavofyork Feb 7, 2023
49caee7
Merge branch 'master' into gav-hold-freeze
gavofyork Feb 7, 2023
91b7343
Fix NIS benchmarks
gavofyork Feb 9, 2023
53fbf7d
Doc comment
gavofyork Feb 9, 2023
c7459ae
Remove post_mutation
gavofyork Feb 9, 2023
864f46a
Merge remote-tracking branch 'origin/master' into gav-hold-freeze
gavofyork Feb 23, 2023
7360c20
Fix some tests
gavofyork Feb 23, 2023
6dbdca4
Fix some grumbles
gavofyork Feb 23, 2023
543713b
Enumify bool args to fungible(s) functions
gavofyork Feb 23, 2023
78a6b18
Fix up assets and balances
gavofyork Feb 23, 2023
35760c0
Formatting
gavofyork Feb 23, 2023
0d1a458
Fix contracts
gavofyork Feb 23, 2023
1f292ae
Merge remote-tracking branch 'origin/master' into gav-hold-freeze
gavofyork Feb 24, 2023
1415e77
Fix tests & benchmarks build
gavofyork Feb 24, 2023
dd6e45f
Typify minted boolean arg
gavofyork Feb 24, 2023
4d69fb8
Typify on_hold boolean arg; renames
gavofyork Feb 24, 2023
3e513e1
Fix numerous tests
gavofyork Feb 24, 2023
e14456e
Fix dependency issue
gavofyork Feb 24, 2023
0f1cd51
Privatize dangerous API mutate_account
gavofyork Feb 24, 2023
dd200cd
Fix contracts (@alext - please check this commit)
gavofyork Feb 24, 2023
5c1ea9d
Remove println
gavofyork Feb 24, 2023
354aa13
Fix tests for contracts
athei Feb 24, 2023
82ace53
Fix broken rename
gavofyork Feb 24, 2023
36ae88e
Merge branch 'gav-hold-freeze' of github.com:paritytech/substrate int…
gavofyork Feb 24, 2023
fcb69a8
Fix broken rename
gavofyork Feb 24, 2023
a3579af
Fix broken rename
gavofyork Feb 24, 2023
a40f3a8
Docs
gavofyork Feb 24, 2023
ba9ff68
Update frame/support/src/traits/tokens/fungible/hold.rs
gavofyork Mar 3, 2023
9d5e61f
Merge remote-tracking branch 'origin/master' into gav-hold-freeze
gavofyork Mar 3, 2023
22595ef
remove from_ref_time
gavofyork Mar 3, 2023
f9ca2ae
Update frame/executive/src/lib.rs
gavofyork Mar 6, 2023
ef22af2
Update frame/executive/src/lib.rs
gavofyork Mar 6, 2023
d6f71e3
Reenable test
gavofyork Mar 6, 2023
7ab3d21
Merge branch 'gav-hold-freeze' of github.com:paritytech/substrate int…
gavofyork Mar 6, 2023
114d623
Update frame/support/src/traits/tokens/fungibles/hold.rs
gavofyork Mar 6, 2023
b1a20d8
Update frame/support/src/traits/tokens/fungible/hold.rs
gavofyork Mar 6, 2023
5c712bd
Update frame/support/src/traits/tokens/fungible/hold.rs
gavofyork Mar 6, 2023
1607004
Update frame/support/src/traits/tokens/fungible/hold.rs
gavofyork Mar 6, 2023
a50dde8
Update frame/support/src/traits/tokens/currency.rs
gavofyork Mar 6, 2023
c3cdb70
Update frame/lottery/src/tests.rs
gavofyork Mar 6, 2023
57a4140
Update frame/support/src/traits/tokens/fungible/mod.rs
gavofyork Mar 6, 2023
176b6a1
Update frame/support/src/traits/tokens/fungible/regular.rs
gavofyork Mar 6, 2023
bbcace9
Update frame/support/src/traits/tokens/fungibles/freeze.rs
gavofyork Mar 6, 2023
a84e900
Update frame/support/src/traits/tokens/fungible/regular.rs
gavofyork Mar 6, 2023
96644e5
Update frame/support/src/traits/tokens/fungibles/hold.rs
gavofyork Mar 6, 2023
58f0a32
Update frame/support/src/traits/tokens/fungibles/hold.rs
gavofyork Mar 6, 2023
73b49d3
Update frame/support/src/traits/tokens/fungibles/hold.rs
gavofyork Mar 6, 2023
4fd8bc2
Rename UnwantedRemoval to UnwantedAccountRemoval
gavofyork Mar 6, 2023
200e274
Merge branch 'gav-hold-freeze' of github.com:paritytech/substrate int…
gavofyork Mar 6, 2023
f88ed63
Docs
gavofyork Mar 6, 2023
aa681b2
Merge remote-tracking branch 'origin/master' into gav-hold-freeze
gavofyork Mar 6, 2023
a1117e8
Formatting
gavofyork Mar 6, 2023
b5884b2
Update frame/balances/src/lib.rs
gavofyork Mar 13, 2023
d78b14a
Update primitives/runtime/src/lib.rs
gavofyork Mar 13, 2023
68f1acb
handle_raw_dust oes nothing
gavofyork Mar 13, 2023
d288318
Formatting
gavofyork Mar 14, 2023
fc5ff3d
Fixes
gavofyork Mar 14, 2023
cc1bd4f
Grumble
gavofyork Mar 14, 2023
516cc00
Fixes
gavofyork Mar 14, 2023
69fedb7
Add test
gavofyork Mar 14, 2023
4a8e549
Add test
gavofyork Mar 14, 2023
ebdc4b9
Merge remote-tracking branch 'origin/master' into gav-hold-freeze
gavofyork Mar 14, 2023
c6fdad5
Tests for reducible_balance
gavofyork Mar 14, 2023
535c918
Fixes
gavofyork Mar 14, 2023
deae58f
Fix Salary
gavofyork Mar 14, 2023
0e2f5d9
Fixes
gavofyork Mar 14, 2023
d9c3c22
Disable broken test
gavofyork Mar 14, 2023
e8ff2b2
Disable nicely
gavofyork Mar 14, 2023
a2cdc2f
Fixes
gavofyork Mar 15, 2023
39fa2fc
Fixes
gavofyork Mar 15, 2023
07d9e48
Fixes
gavofyork Mar 15, 2023
9e56f0d
Rename some events
gavofyork Mar 15, 2023
b8a13c0
Merge remote-tracking branch 'origin/master' into gav-hold-freeze
gavofyork Mar 15, 2023
68dbc16
Merge remote-tracking branch 'origin/master' into gav-hold-freeze
gavofyork Mar 16, 2023
2e30a2a
Fix nomination pools breakage
gavofyork Mar 16, 2023
f546870
Add compatibility stub for transfer tx
gavofyork Mar 16, 2023
5686ca4
Reinstate a safely compatible version of Balances set_balance
gavofyork Mar 16, 2023
590b780
Fixes
gavofyork Mar 16, 2023
181bcd5
Grumble
gavofyork Mar 16, 2023
880fac6
Update frame/nis/src/lib.rs
gavofyork Mar 16, 2023
58dca36
Merge branch 'master' of https://github.com/paritytech/substrate into…
Mar 17, 2023
1d47200
".git/.scripts/commands/bench/bench.sh" pallet dev pallet_balances
Mar 17, 2023
d025e95
Merge remote-tracking branch 'origin/master' into gav-hold-freeze
gavofyork Mar 17, 2023
be47675
disable flakey tests
gavofyork Mar 17, 2023
3f1a752
Update frame/balances/src/lib.rs
gavofyork Mar 17, 2023
66e7acc
Grumbles
gavofyork Mar 17, 2023
a730846
Merge branch 'gav-hold-freeze' of github.com:paritytech/substrate int…
gavofyork Mar 17, 2023
938db17
Grumble
gavofyork Mar 17, 2023
a76f91c
Merge branch 'master' into gav-hold-freeze
gavofyork Mar 17, 2023
d59d300
Merge remote-tracking branch 'origin/master' into gav-hold-freeze
gavofyork Mar 18, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
New ref-counting logic for old API
  • Loading branch information
gavofyork committed Dec 19, 2022
commit 0c8a27c7c3bed81fa965a26b479603fcc5aab80b
209 changes: 96 additions & 113 deletions frame/balances/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -172,8 +172,8 @@ use frame_support::{
ensure,
pallet_prelude::DispatchResult,
traits::{
tokens::{fungible, BalanceStatus as Status, DepositConsequence, WithdrawConsequence},
Currency, DefensiveSaturating, ExistenceRequirement,
tokens::{fungible, BalanceStatus as Status, DepositConsequence, WithdrawConsequence, KeepAlive::{CanKill, Keep}},
Currency, DefensiveSaturating, ExistenceRequirement, Defensive,
ExistenceRequirement::{AllowDeath, KeepAlive},
Get, Imbalance, LockIdentifier, LockableCurrency, NamedReservableCurrency, OnUnbalanced,
ReservableCurrency, SignedImbalance, StoredMap, TryDrop, WithdrawReasons,
Expand Down Expand Up @@ -202,6 +202,12 @@ pub mod pallet {
use frame_support::pallet_prelude::*;
use frame_system::pallet_prelude::*;

pub enum RefType {
Provides,
Consumes,
Sufficient,
}

#[pallet::config]
pub trait Config<I: 'static = ()>: frame_system::Config {
/// The balance of an account.
Expand Down Expand Up @@ -245,6 +251,8 @@ pub mod pallet {

/// The id type for named reserves.
type ReserveIdentifier: Parameter + Member + MaxEncodedLen + Ord + Copy;

// TODO: LockIdentifier
}

/// The current storage version.
Expand Down Expand Up @@ -318,25 +326,19 @@ pub mod pallet {
origin: OriginFor<T>,
who: AccountIdLookupOf<T>,
#[pallet::compact] new_free: T::Balance,
#[pallet::compact] new_reserved: T::Balance,
) -> DispatchResultWithPostInfo {
ensure_root(origin)?;
let who = T::Lookup::lookup(who)?;
let existential_deposit = T::ExistentialDeposit::get();

let wipeout = new_free + new_reserved < existential_deposit;
let wipeout = new_free < existential_deposit;
let new_free = if wipeout { Zero::zero() } else { new_free };
let new_reserved = if wipeout { Zero::zero() } else { new_reserved };

// First we try to modify the account's balance to the forced balance.
let (old_free, old_reserved) = Self::mutate_account(&who, |account| {
let old_free = Self::mutate_account(&who, |account| {
let old_free = account.free;
let old_reserved = account.reserved;

account.free = new_free;
account.reserved = new_reserved;

(old_free, old_reserved)
old_free
})?;

// This will adjust the total issuance, which was not done by the `mutate_account`
Expand All @@ -347,13 +349,7 @@ pub mod pallet {
mem::drop(NegativeImbalance::<T, I>::new(old_free - new_free));
}

if new_reserved > old_reserved {
mem::drop(PositiveImbalance::<T, I>::new(new_reserved - old_reserved));
} else if new_reserved < old_reserved {
mem::drop(NegativeImbalance::<T, I>::new(old_reserved - new_reserved));
}

Self::deposit_event(Event::BalanceSet { who, free: new_free, reserved: new_reserved });
Self::deposit_event(Event::BalanceSet { who, free: new_free });
Ok(().into())
}

Expand Down Expand Up @@ -428,6 +424,7 @@ pub mod pallet {
) -> DispatchResult {
use fungible::{Inspect, Transfer};
let transactor = ensure_signed(origin)?;
let keep_alive = if keep_alive { Keep } else { CanKill };
let reducible_balance = Self::reducible_balance(&transactor, keep_alive, false);
let dest = T::Lookup::lookup(dest)?;
<Self as Transfer<_>>::transfer(&transactor, &dest, reducible_balance, keep_alive)?;
Expand Down Expand Up @@ -462,7 +459,7 @@ pub mod pallet {
/// Transfer succeeded.
Transfer { from: T::AccountId, to: T::AccountId, amount: T::Balance },
/// A balance was set by root.
BalanceSet { who: T::AccountId, free: T::Balance, reserved: T::Balance },
BalanceSet { who: T::AccountId, free: T::Balance },
/// Some balance was reserved (moved from free to reserved).
Reserved { who: T::AccountId, amount: T::Balance },
/// Some balance was unreserved (moved from reserved to free).
Expand Down Expand Up @@ -608,6 +605,7 @@ pub mod pallet {
for &(ref who, free) in self.balances.iter() {
assert!(T::AccountStore::insert(who, AccountData { free, ..Default::default() })
.is_ok());
frame_system::Pallet::<T>::inc_providers(who);
}
}
}
Expand Down Expand Up @@ -782,12 +780,13 @@ impl<T: Config<I>, I: 'static> Pallet<T, I> {
_who: &T::AccountId,
new: AccountData<T::Balance>,
) -> (Option<AccountData<T::Balance>>, Option<NegativeImbalance<T, I>>) {
let total = new.total();
if total < T::ExistentialDeposit::get() {
if total.is_zero() {
// We should never be dropping if reserved is non-zero. Reserved being non-zero should imply
// that we have a consumer ref, so this is economically safe.
if new.free < T::ExistentialDeposit::get() && new.reserved.is_zero() {
if new.free.is_zero() {
(None, None)
} else {
(None, Some(NegativeImbalance::new(total)))
(None, Some(NegativeImbalance::new(new.free)))
}
} else {
(Some(new), None)
Expand Down Expand Up @@ -925,12 +924,44 @@ impl<T: Config<I>, I: 'static> Pallet<T, I> {
let result = T::AccountStore::try_mutate_exists(who, |maybe_account| {
let is_new = maybe_account.is_none();
let mut account = maybe_account.take().unwrap_or_default();
f(&mut account, is_new).map(move |result| {
let maybe_endowed = if is_new { Some(account.free) } else { None };
let maybe_account_maybe_dust = Self::post_mutation(who, account);
*maybe_account = maybe_account_maybe_dust.0;
(maybe_endowed, maybe_account_maybe_dust.1, result)
})
let did_provide = account.free >= T::ExistentialDeposit::get();
let did_consume = !is_new && !account.reserved.is_zero();

let result = f(&mut account, is_new)?;

let does_provide = account.free >= T::ExistentialDeposit::get();
let does_consume = !account.reserved.is_zero();

if !did_provide && does_provide {
frame_system::Pallet::<T>::inc_providers(who);
}
if did_consume && !does_consume {
frame_system::Pallet::<T>::dec_consumers(who);
}
if !did_consume && does_consume {
frame_system::Pallet::<T>::inc_consumers(who)?;
}
if did_provide && !does_provide {
// This could reap the account so must go last.
frame_system::Pallet::<T>::dec_providers(who).map_err(|r| {
if did_consume && !does_consume {
// best-effort revert consumer change.
let _ = frame_system::Pallet::<T>::inc_consumers(who).defensive();
}
if !did_consume && does_consume {
let _ = frame_system::Pallet::<T>::dec_consumers(who);
}
r
})?;
}

let maybe_endowed = if is_new { Some(account.free) } else { None };
let maybe_account_maybe_dust = Self::post_mutation(who, account);
*maybe_account = maybe_account_maybe_dust.0;
if let Some(ref account) = &maybe_account {
assert!(account.free.is_zero() || account.free >= T::ExistentialDeposit::get() || !account.reserved.is_zero());
}
Ok((maybe_endowed, maybe_account_maybe_dust.1, result))
});
result.map(|(maybe_endowed, maybe_dust, result)| {
if let Some(endowed) = maybe_endowed {
Expand Down Expand Up @@ -1401,61 +1432,29 @@ where
if Self::total_balance(who).is_zero() {
return (NegativeImbalance::zero(), value)
}

for attempt in 0..2 {
match Self::try_mutate_account(
who,
|account,
_is_new|
-> Result<(Self::NegativeImbalance, Self::Balance), DispatchError> {
// Best value is the most amount we can slash following liveness rules.
let best_value = match attempt {
// First attempt we try to slash the full amount, and see if liveness issues
// happen.
0 => value,
// If acting as a critical provider (i.e. first attempt failed), then slash
// as much as possible while leaving at least at ED.
_ => value.min(
(account.free + account.reserved)
.saturating_sub(T::ExistentialDeposit::get()),
),
};

let free_slash = cmp::min(account.free, best_value);
account.free -= free_slash; // Safe because of above check
let remaining_slash = best_value - free_slash; // Safe because of above check

if !remaining_slash.is_zero() {
// If we have remaining slash, take it from reserved balance.
let reserved_slash = cmp::min(account.reserved, remaining_slash);
account.reserved -= reserved_slash; // Safe because of above check
Ok((
NegativeImbalance::new(free_slash + reserved_slash),
value - free_slash - reserved_slash, /* Safe because value is gt or
* eq total slashed */
))
} else {
// Else we are done!
Ok((
NegativeImbalance::new(free_slash),
value - free_slash, // Safe because value is gt or eq to total slashed
))
}
},
) {
Ok((imbalance, not_slashed)) => {
Self::deposit_event(Event::Slashed {
who: who.clone(),
amount: value.saturating_sub(not_slashed),
});
return (imbalance, not_slashed)
},
Err(_) => (),
}
match Self::try_mutate_account(who, |account, _is_new|
-> Result<(Self::NegativeImbalance, Self::Balance), DispatchError>
{
// Best value is the most amount we can slash following liveness rules.
let ed = T::ExistentialDeposit::get();
let actual = match system::Pallet::<T>::can_dec_provider(who) {
true => value.min(account.free),
false => value.min(account.free.saturating_sub(ed)),
};
account.free.saturating_reduce(actual);
let remaining = value.saturating_sub(actual);
Ok((NegativeImbalance::new(actual), remaining))
},
) {
Ok((imbalance, remaining)) => {
Self::deposit_event(Event::Slashed {
who: who.clone(),
amount: value.saturating_sub(remaining),
});
(imbalance, remaining)
},
Err(_) => (Self::NegativeImbalance::zero(), value),
}

// Should never get here. But we'll be defensive anyway.
(Self::NegativeImbalance::zero(), value)
}

/// Deposit some `value` into the free balance of an existing target account `who`.
Expand Down Expand Up @@ -1579,7 +1578,6 @@ where
Self::deposit_event(Event::BalanceSet {
who: who.clone(),
free: account.free,
reserved: account.reserved,
});
Ok(imbalance)
},
Expand Down Expand Up @@ -1680,37 +1678,22 @@ where
// NOTE: `mutate_account` may fail if it attempts to reduce the balance to the point that an
// account is attempted to be illegally destroyed.

for attempt in 0..2 {
match Self::mutate_account(who, |account| {
let best_value = match attempt {
0 => value,
// If acting as a critical provider (i.e. first attempt failed), then ensure
// slash leaves at least the ED.
_ => value.min(
(account.free + account.reserved)
.saturating_sub(T::ExistentialDeposit::get()),
),
};
match Self::mutate_account(who, |account| {
let actual = value.min(account.reserved);
account.reserved.saturating_reduce(actual);

let actual = cmp::min(account.reserved, best_value);
account.reserved -= actual;

// underflow should never happen, but it if does, there's nothing to be done here.
(NegativeImbalance::new(actual), value - actual)
}) {
Ok((imbalance, not_slashed)) => {
Self::deposit_event(Event::Slashed {
who: who.clone(),
amount: value.saturating_sub(not_slashed),
});
return (imbalance, not_slashed)
},
Err(_) => (),
}
// underflow should never happen, but it if does, there's nothing to be done here.
(NegativeImbalance::new(actual), value.saturating_sub(actual))
}) {
Ok((imbalance, not_slashed)) => {
Self::deposit_event(Event::Slashed {
who: who.clone(),
amount: value.saturating_sub(not_slashed),
});
(imbalance, not_slashed)
},
Err(_) => (Self::NegativeImbalance::zero(), value),
}
// Should never get here as we ensure that ED is left in the second attempt.
// In case we do, though, then we fail gracefully.
(Self::NegativeImbalance::zero(), value)
}

/// Move the reserved balance of one account into the balance of another, according to `status`.
Expand Down
Loading