Skip to content

Commit 349dc3e

Browse files
committed
solana: Configure clippy to deny possible truncation
Adds a clippy rule to block the inclusion of code that can result in truncation (i.e. `as T` conversions) Adds linting directive to existing instances in the code and add comments about why they are safe Add a unit test on bitmap length to confirm that the maximum possible number of votes (i.e. maximum "length" of the bitmap) is within the limits of u8.
1 parent 76211b2 commit 349dc3e

File tree

3 files changed

+13
-7
lines changed

3 files changed

+13
-7
lines changed

.github/workflows/solana.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -70,7 +70,7 @@ jobs:
7070
run: cargo check --workspace --tests --manifest-path solana/Cargo.toml
7171

7272
- name: Run `cargo clippy`
73-
run: cargo clippy --workspace --tests --manifest-path solana/Cargo.toml
73+
run: cargo clippy --workspace --tests --manifest-path solana/Cargo.toml -- -Dclippy::cast_possible_truncation
7474

7575
- name: Cache solana tools
7676
id: cache-solana

solana/programs/example-native-token-transfers/src/bitmap.rs

+1-2
Original file line numberDiff line numberDiff line change
@@ -41,10 +41,9 @@ impl Bitmap {
4141
Ok(BM::<128>::from_value(self.map).get(usize::from(index)))
4242
}
4343

44+
#[allow(clippy::cast_possible_truncation)]
4445
pub fn count_enabled_votes(&self, enabled: Bitmap) -> u8 {
4546
let bm = BM::<128>::from_value(self.map) & BM::<128>::from_value(enabled.map);
46-
// Conversion from usize to u8 is safe here. The Bitmap uses u128, so its maximum length
47-
// (number of true bits) is 128.
4847
bm.len()
4948
.try_into()
5049
.expect("Bitmap length must not exceed the bounds of u8")

solana/programs/example-native-token-transfers/src/queue/rate_limit.rs

+11-4
Original file line numberDiff line numberDiff line change
@@ -46,12 +46,15 @@ impl RateLimitState {
4646
/// Returns the capacity of the rate limiter.
4747
/// On-chain programs and unit tests should always use [`capacity`].
4848
/// This function is useful in solana-program-test, where the clock sysvar
49-
/// SECURITY: Integer division is OK here. We are not that concerned with precision. Removing
50-
/// the remainder in this case is arguably more secure as it reduces the available capacity.
51-
/// SECURITY: Sign loss is OK here. It is a conversion performed on a timestamp that must always be
52-
/// positive.
49+
// SECURITY: Integer division is OK here. We are not that concerned with precision. Removing
50+
// the remainder in this case is arguably more secure as it reduces the available capacity.
51+
// SECURITY: Sign loss is OK here. It is a conversion performed on a timestamp that must always be
52+
// positive.
53+
// SECURITY: Truncation is allowed here. Clippy warns about the final returned expression, but it is
54+
// safe.
5355
#[allow(clippy::integer_division)]
5456
#[allow(clippy::cast_sign_loss)]
57+
#[allow(clippy::cast_possible_truncation)]
5558
pub fn capacity_at(&self, now: UnixTimestamp) -> u64 {
5659
assert!(self.last_tx_timestamp <= now);
5760

@@ -76,6 +79,10 @@ impl RateLimitState {
7679
+ time_passed as u128 * limit / (Self::RATE_LIMIT_DURATION as u128)
7780
};
7881

82+
// The use of `min` here prevents truncation.
83+
// The value of `limit` is u64 in reality. If both `calculated_capacity` and `limit` are at
84+
// their maxiumum possible values (u128::MAX and u64::MAX), then u64::MAX will be chosen by
85+
// `min`. So truncation is not possible.
7986
calculated_capacity.min(limit) as u64
8087
}
8188

0 commit comments

Comments
 (0)