Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

SIMD-0075: Precompile for Secp256r1 #3152

Merged
merged 75 commits into from
Nov 15, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
75 commits
Select commit Hold shift + click to select a range
226eeef
feat: secp256r1 precompile
0xRigel Oct 13, 2024
4c753ff
add: num_signatures == 0 check from SIMD-0152
0xRigel Oct 13, 2024
b0ec063
rm: unnecessary comment
0xRigel Oct 13, 2024
6d9d46e
fix: legacy numeric constant
0xRigel Oct 13, 2024
943bd7c
Merge branch 'master' into secp256r1-precompile
0xRigel Oct 15, 2024
f74009a
CI/fix: compilation for wasm32 target
0xRigel Oct 15, 2024
46e15a1
Merge branch 'master' into secp256r1-precompile
0xRigel Oct 15, 2024
6011adf
Merge branch 'secp256r1-precompile' of https://github.com/iceomatic/a…
0xRigel Oct 15, 2024
bc83ba6
Merge branch 'master' into secp256r1-precompile
0xRigel Oct 16, 2024
86a25ea
Extract secp256r1 crate
0xRigel Oct 16, 2024
2598735
rm: unnecessary import
0xRigel Oct 16, 2024
eb3c492
update: sbf/Cargo.lock
0xRigel Oct 16, 2024
4d5b213
rm: unnecesary re-exports
0xRigel Oct 16, 2024
337433c
add: secp256r1 precompile to docs
0xRigel Oct 16, 2024
eeb5023
add: docs/description to sdk/program/src/lib.rs
0xRigel Oct 16, 2024
4b257a0
fix: alpha sort deps
0xRigel Oct 23, 2024
372be6c
fixes
0xRigel Oct 23, 2024
30dbde6
docs fixes
0xRigel Oct 23, 2024
2d64c1f
add: solana-instruction std feature to deps
0xRigel Oct 23, 2024
efe084b
Merge branch 'master' into secp256r1-precompile
0xRigel Oct 23, 2024
e266020
fix: lockfile from rebase
0xRigel Oct 23, 2024
4cd7bc0
fix: target architecture
0xRigel Oct 23, 2024
ce302a7
fix: workflow for client_target android
0xRigel Oct 23, 2024
897eace
add: sudo to workflow perl install
0xRigel Oct 24, 2024
c6c2249
fix: Cargo toml workspace member
0xRigel Oct 24, 2024
930db7c
Merge branch 'master' into secp256r1-precompile
0xRigel Oct 24, 2024
3df67bb
Merge branch 'secp256r1-precompile' of https://github.com/0xRigel/aga…
0xRigel Oct 24, 2024
3a73e79
modify: ranlib path in client-targets.yaml
0xRigel Oct 24, 2024
52fb88a
fix: secp256r1/Cargo.toml formatting
0xRigel Oct 24, 2024
632118c
add: openssl feature
0xRigel Oct 31, 2024
5437fc5
fixes
0xRigel Oct 31, 2024
fdf7673
add: precompile signature range error
0xRigel Nov 1, 2024
d3d24ea
more adjustments
0xRigel Nov 1, 2024
cbc7219
change: feature id
0xRigel Nov 2, 2024
3659bd0
Merge remote-tracking branch 'origin/master' into secp256r1-precompile
0xRigel Nov 6, 2024
0d2f171
fix: cargo format
0xRigel Nov 6, 2024
7e8a3ea
Revert "add: precompile signature range error"
0xRigel Nov 6, 2024
f27ce17
Merge branch 'anza-xyz:master' into secp256r1-precompile
0xRigel Nov 6, 2024
64291f6
Merge branch 'master' into secp256r1-precompile
0xRigel Nov 6, 2024
b44c02f
fix: cargo sanity
0xRigel Nov 6, 2024
8629123
Merge branch 'master' into secp256r1-precompile
0xRigel Nov 6, 2024
26649ff
fix: client target openssl dep
0xRigel Nov 6, 2024
e539f5c
fix: 31 byte r,s support in new_secp256r1_instruction
0xRigel Nov 6, 2024
fd7bbb6
update: Cargo.lock
0xRigel Nov 6, 2024
1522fcc
fix: unchecked math in new_secp256r1_instruction
0xRigel Nov 6, 2024
7f54c62
Merge branch 'master' into secp256r1-precompile
0xRigel Nov 6, 2024
a765f92
fixes & increased test coverage
0xRigel Nov 7, 2024
51ed60e
Merge branch 'master' into secp256r1-precompile
0xRigel Nov 7, 2024
f0cfecd
Merge branch 'master' into secp256r1-precompile
0xRigel Nov 7, 2024
6b92a71
Merge branch 'secp256r1-precompile' of https://github.com/0xRigel/aga…
0xRigel Nov 7, 2024
5c66b50
add: solana-sdk/openssl to all release binaries
0xRigel Nov 13, 2024
38dbbf5
update: comment to make openssl feature more clear
0xRigel Nov 13, 2024
7c2941d
add: solana-sdk/openssl feature to dependencies
0xRigel Nov 13, 2024
86710f8
add: solana-sdk/openssl feature to dependencies
0xRigel Nov 13, 2024
818af19
merge: master into secp256r1-precompile
0xRigel Nov 13, 2024
0d1815a
Merge branch 'master' into secp256r1-precompile
0xRigel Nov 13, 2024
99f22fa
fix: test-validator formatting
0xRigel Nov 13, 2024
81350f7
Revert "add: solana-sdk/openssl to all release binaries"
0xRigel Nov 14, 2024
5fc3feb
add: reserved key for secp256r1 program
0xRigel Nov 14, 2024
cc04f9f
modify: client-targets.yaml
0xRigel Nov 14, 2024
42f9402
modify: client/Cargo.toml solana-sdk dep
0xRigel Nov 14, 2024
d6120fd
modify: ledger-tool/Cargo.toml solana-sdk
0xRigel Nov 14, 2024
bd263bb
modify: test-validator/Cargo.toml solana-sdk dep
0xRigel Nov 14, 2024
a2a400d
modify: validator/Cargo.toml solana-sdk dep
0xRigel Nov 14, 2024
cbb6b61
change: openssl feature to openssl-vendored
0xRigel Nov 14, 2024
cc7af13
remove: solana-sdk dep from sdk/program
0xRigel Nov 14, 2024
ddb07fb
refactor: secp256r1 directory name
0xRigel Nov 14, 2024
bd138d2
fmt
0xRigel Nov 14, 2024
3174739
cargo.lock files
0xRigel Nov 14, 2024
e134144
revert: rustc-demangle bump
0xRigel Nov 15, 2024
df11363
Merge upstream master and resolve conflicts
0xRigel Nov 15, 2024
55cb982
cargo lock sanity
0xRigel Nov 15, 2024
517ae86
Merge remote-tracking branch 'upstream/master' into secp256r1-precompile
0xRigel Nov 15, 2024
63d86f0
fix: faulty feature-set merge
0xRigel Nov 15, 2024
265f392
fix: reserved keys pending feature id
0xRigel Nov 15, 2024
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
5 changes: 5 additions & 0 deletions .github/workflows/client-targets.yml
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,11 @@ jobs:
steps:
- uses: actions/checkout@v4

# This can be removed once cargo-ndk >= 3.5.4 is used.
- name: Setup environment for Android NDK
run: |
echo "RANLIB=$ANDROID_NDK_ROOT/toolchains/llvm/prebuilt/linux-x86_64/bin/llvm-ranlib" >> $GITHUB_ENV
0xRigel marked this conversation as resolved.
Show resolved Hide resolved
- run: cargo install [email protected]

- name: Setup Rust
Expand Down
16 changes: 16 additions & 0 deletions Cargo.lock

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

2 changes: 2 additions & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -147,6 +147,7 @@ members = [
"sdk/reserved-account-keys",
"sdk/sanitize",
"sdk/sdk-ids",
"sdk/secp256r1-program",
"sdk/seed-derivable",
"sdk/seed-phrase",
"sdk/serde",
Expand Down Expand Up @@ -504,6 +505,7 @@ solana-remote-wallet = { path = "remote-wallet", version = "=2.2.0", default-fea
solana-rent = { path = "sdk/rent", version = "=2.2.0", default-features = false }
solana-reserved-account-keys = { path = "sdk/reserved-account-keys", version = "=2.2.0", default-features = false }
solana-reward-info = { path = "sdk/reward-info", version = "=2.2.0" }
solana-secp256r1-program = { path = "sdk/secp256r1-program", version = "=2.2.0", default-features = false }
solana-sanitize = { path = "sdk/sanitize", version = "=2.2.0" }
solana-seed-derivable = { path = "sdk/seed-derivable", version = "=2.2.0" }
solana-seed-phrase = { path = "sdk/seed-phrase", version = "=2.2.0" }
Expand Down
2 changes: 1 addition & 1 deletion client/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@ solana-quic-client = { workspace = true }
solana-rpc-client = { workspace = true, features = ["default"] }
solana-rpc-client-api = { workspace = true }
solana-rpc-client-nonce-utils = { workspace = true }
solana-sdk = { workspace = true }
solana-sdk = { workspace = true, features = ["openssl-vendored"] }
solana-streamer = { workspace = true }
solana-thin-client = { workspace = true }
solana-tpu-client = { workspace = true, features = ["default"] }
Expand Down
65 changes: 65 additions & 0 deletions docs/src/runtime/programs.md
Original file line number Diff line number Diff line change
Expand Up @@ -168,6 +168,71 @@ also receive data from the transaction itself.
Cost of the transaction will count the number of signatures to verify multiplied
by the signature cost verify multiplier.

## Secp256r1 Program

The program for verifying secp256r1 signatures. It takes a secp256r1 signature,
a public key, and a message. Up to 8 signatures can be verified. If any of the
signatures fail to verify, an error is returned.

- Program id: `Secp256r1SigVerify1111111111111111111111111`
- Instructions: [secp256r1_instruction](https://docs.rs/solana-secp256r1)

The secp256r1 program processes an instruction. The first `u8` is a count of the number of signatures to check, followed by a single byte padding. After that, the following struct is serialized, one for each signature to check:

```rust
struct Secp256r1SignatureOffsets {
signature_offset: u16, // offset to compact secp256r1 signature of 64 bytes
signature_instruction_index: u16, // instruction index to find signature
public_key_offset: u16, // offset to compressed public key of 33 bytes
public_key_instruction_index: u16, // instruction index to find public key
message_data_offset: u16, // offset to start of message data
message_data_size: u16, // size of message data
message_instruction_index: u16, // index of instruction data to get message data
}

```

The pseudo code of the signature verification:
```
process_instruction() {
if data.len() < SIGNATURE_OFFSETS_START {
return Error
}
num_signatures = data[0] as usize
if num_signatures == 0 || num_signatures > 8 {
return Error
}
expected_data_size = num_signatures * SIGNATURE_OFFSETS_SERIALIZED_SIZE + SIGNATURE_OFFSETS_START
if data.len() < expected_data_size {
return Error
}
for i in 0..num_signatures {
offsets = parse_signature_offsets(data, i)
signature = get_data_slice(data, instruction_datas, offsets.signature_instruction_index, offsets.signature_offset, SIGNATURE_SERIALIZED_SIZE)
if s > half_curve_order {
return Error
}
pubkey = get_data_slice(data, instruction_datas, offsets.public_key_instruction_index, offsets.public_key_offset, COMPRESSED_PUBKEY_SERIALIZED_SIZE)
message = get_data_slice(data, instruction_datas, offsets.message_instruction_index, offsets.message_data_offset, offsets.message_data_size)
if !verify_signature(signature, pubkey, message) {
return Error
}
}
return Success
}
```
Note: Low S values are enforced for all signatures to avoid accidental signature
malleability.

### Optimization notes

The operation will have to take place after (at least partial) deserialization,
Expand Down
2 changes: 1 addition & 1 deletion ledger-tool/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -46,7 +46,7 @@ solana-program-runtime = { workspace = true }
solana-rpc = { workspace = true }
solana-runtime = { workspace = true, features = ["dev-context-only-utils"] }
solana-runtime-transaction = { workspace = true }
solana-sdk = { workspace = true }
solana-sdk = { workspace = true, features = ["openssl-vendored"] }
solana-stake-program = { workspace = true }
solana-storage-bigtable = { workspace = true }
solana-streamer = { workspace = true }
Expand Down
14 changes: 14 additions & 0 deletions programs/sbf/Cargo.lock

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

3 changes: 3 additions & 0 deletions sdk/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -61,6 +61,8 @@ frozen-abi = [
"solana-signature/frozen-abi",
"solana-transaction-error/frozen-abi"
]
# Enables the "vendored" feature of openssl inside of secp256r1-program
openssl-vendored = ["solana-secp256r1-program/openssl-vendored"]

[dependencies]
bincode = { workspace = true }
Expand Down Expand Up @@ -131,6 +133,7 @@ solana-sanitize = { workspace = true }
solana-sdk-ids = { workspace = true }
solana-sdk-macro = { workspace = true }
solana-secp256k1-recover = { workspace = true }
solana-secp256r1-program = { workspace = true, default-features = false }
solana-seed-derivable = { workspace = true, optional = true }
solana-seed-phrase = { workspace = true, optional = true }
solana-serde = { workspace = true }
Expand Down
4 changes: 4 additions & 0 deletions sdk/feature-set/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -880,6 +880,9 @@ pub mod disable_account_loader_special_case {
solana_pubkey::declare_id!("EQUMpNFr7Nacb1sva56xn1aLfBxppEoSBH8RRVdkcD1x");
}

pub mod enable_secp256r1_precompile {
solana_pubkey::declare_id!("sr11RdZWgbHTHxSroPALe6zgaT5A1K9LcE4nfsZS4gi");
}
pub mod accounts_lt_hash {
solana_pubkey::declare_id!("LtHaSHHsUge7EWTPVrmpuexKz6uVHZXZL6cgJa7W7Zn");
}
Expand Down Expand Up @@ -1100,6 +1103,7 @@ lazy_static! {
(lift_cpi_caller_restriction::id(), "Lift the restriction in CPI that the caller must have the callee as an instruction account #2202"),
(disable_account_loader_special_case::id(), "Disable account loader special case #3513"),
(accounts_lt_hash::id(), "enables lattice-based accounts hash #3333"),
(enable_secp256r1_precompile::id(), "Enable secp256r1 precompile SIMD-0075"),
/*************** ADD NEW FEATURES HERE ***************/
]
.iter()
Expand Down
1 change: 1 addition & 0 deletions sdk/reserved-account-keys/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ solana-frozen-abi-macro = { workspace = true, optional = true, features = [
] }
solana-pubkey = { workspace = true, default-features = false }
solana-sdk-ids = { workspace = true }
solana-secp256r1-program = { workspace = true }

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would have asked you to put the secp256r1 program ID in solana-sdk-ids so solana-reserved-account-keys wouldn't end up with a solana-secp256r1-program dependency

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Will coordinate with sam to make sure #3663 is backported together with this 👍

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks like we don't have to backport since the sdk-ids split isn't in 2.1. Will just get this merged into master 🫡


[dev-dependencies]
solana-program = { path = "../program" }
Expand Down
2 changes: 2 additions & 0 deletions sdk/reserved-account-keys/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ use {
secp256k1_program, stake, system_program, sysvar, vote, zk_elgamal_proof_program,
zk_token_proof_program,
},
solana_secp256r1_program as secp256r1_program,
std::collections::{HashMap, HashSet},
};

Expand Down Expand Up @@ -151,6 +152,7 @@ lazy_static! {
ReservedAccount::new_active(feature::id()),
ReservedAccount::new_pending(loader_v4::id(), feature_set::add_new_reserved_account_keys::id()),
ReservedAccount::new_pending(secp256k1_program::id(), feature_set::add_new_reserved_account_keys::id()),
ReservedAccount::new_pending(secp256r1_program::id(), feature_set::enable_secp256r1_precompile::id()),
#[allow(deprecated)]
ReservedAccount::new_active(stake::config::id()),
ReservedAccount::new_active(stake::id()),
Expand Down
36 changes: 36 additions & 0 deletions sdk/secp256r1-program/Cargo.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
[package]
name = "solana-secp256r1-program"
description = "Precompile implementation for the secp256r1 elliptic curve."
documentation = "https://docs.rs/solana-secp256r1"
version = { workspace = true }
authors = { workspace = true }
repository = { workspace = true }
homepage = { workspace = true }
license = { workspace = true }
edition = { workspace = true }

[dependencies]
bytemuck = { workspace = true, features = ["derive"] }
solana-feature-set = { workspace = true }
solana-precompile-error = { workspace = true }
solana-pubkey = { workspace = true }

[target.'cfg(all(not(target_arch = "wasm32"), not(target_os = "solana")))'.dependencies]
solana-instruction = { workspace = true, features = ["std"] }
openssl = { workspace = true }

[dev-dependencies]
solana-logger = { workspace = true }
solana-sdk = { path = "../" }

[features]
default = []
openssl-vendored = ["openssl/vendored"]

[package.metadata.docs.rs]
targets = ["x86_64-unknown-linux-gnu"]
all-features = true
rustdoc-args = ["--cfg=docsrs"]

[lints]
workspace = true
Loading
Loading