From ae91f9fcc95486144297e30bfebb824c7a72080e Mon Sep 17 00:00:00 2001 From: Jon C Date: Wed, 6 Dec 2023 14:34:28 +0100 Subject: [PATCH 01/10] sdk: Update to borsh 1, revert borsh 0.9 / 0.10 --- Cargo.lock | 117 ++++++++++++++++--------------- Cargo.toml | 2 +- sdk/program/Cargo.toml | 3 +- sdk/program/src/borsh.rs | 72 ++++++++++--------- sdk/program/src/borsh0_10.rs | 41 +++++++++-- sdk/program/src/borsh0_9.rs | 22 ++++-- sdk/program/src/borsh1.rs | 20 ++++++ sdk/program/src/instruction.rs | 7 +- sdk/program/src/lib.rs | 5 +- sdk/program/src/program_error.rs | 2 +- sdk/program/src/pubkey.rs | 46 ++++++++++++ sdk/program/src/stake/state.rs | 10 +-- sdk/src/compute_budget.rs | 4 +- sdk/src/lib.rs | 2 +- 14 files changed, 235 insertions(+), 118 deletions(-) create mode 100644 sdk/program/src/borsh1.rs diff --git a/Cargo.lock b/Cargo.lock index dd74ea32642cf1..41b9c951b75de7 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -706,16 +706,6 @@ version = "0.2.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "8d696c370c750c948ada61c69a0ee2cbbb9c50b1019ddb86d9317157a99c2cae" -[[package]] -name = "borsh" -version = "0.9.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "15bf3650200d8bffa99015595e10f1fbd17de07abbc25bb067da79e769939bfa" -dependencies = [ - "borsh-derive 0.9.3", - "hashbrown 0.11.2", -] - [[package]] name = "borsh" version = "0.10.3" @@ -727,16 +717,13 @@ dependencies = [ ] [[package]] -name = "borsh-derive" -version = "0.9.3" +name = "borsh" +version = "1.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6441c552f230375d18e3cc377677914d2ca2b0d36e52129fe15450a2dce46775" +checksum = "bf617fabf5cdbdc92f774bfe5062d870f228b80056d41180797abf48bed4056e" dependencies = [ - "borsh-derive-internal 0.9.3", - "borsh-schema-derive-internal 0.9.3", - "proc-macro-crate 0.1.5", - "proc-macro2", - "syn 1.0.109", + "borsh-derive 1.2.0", + "cfg_aliases", ] [[package]] @@ -745,22 +732,25 @@ version = "0.10.3" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "0754613691538d51f329cce9af41d7b7ca150bc973056f1156611489475f54f7" dependencies = [ - "borsh-derive-internal 0.10.3", - "borsh-schema-derive-internal 0.10.3", + "borsh-derive-internal", + "borsh-schema-derive-internal", "proc-macro-crate 0.1.5", "proc-macro2", "syn 1.0.109", ] [[package]] -name = "borsh-derive-internal" -version = "0.9.3" +name = "borsh-derive" +version = "1.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5449c28a7b352f2d1e592a8a28bf139bc71afb0764a14f3c02500935d8c44065" +checksum = "f404657a7ea7b5249e36808dff544bc88a28f26e0ac40009f674b7a009d14be3" dependencies = [ + "once_cell", + "proc-macro-crate 2.0.0", "proc-macro2", "quote", - "syn 1.0.109", + "syn 2.0.39", + "syn_derive", ] [[package]] @@ -774,17 +764,6 @@ dependencies = [ "syn 1.0.109", ] -[[package]] -name = "borsh-schema-derive-internal" -version = "0.9.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cdbd5696d8bfa21d53d9fe39a714a18538bad11492a42d066dbbc395fb1951c0" -dependencies = [ - "proc-macro2", - "quote", - "syn 1.0.109", -] - [[package]] name = "borsh-schema-derive-internal" version = "0.10.3" @@ -1029,6 +1008,12 @@ version = "1.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" +[[package]] +name = "cfg_aliases" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fd16c4719339c4530435d38e511904438d07cce7950afa3718a84ac36c10e89e" + [[package]] name = "chrono" version = "0.4.31" @@ -1530,7 +1515,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "978747c1d849a7d2ee5e8adc0159961c48fb7e5db2f06af6723b80123bb53856" dependencies = [ "cfg-if 1.0.0", - "hashbrown 0.14.1", + "hashbrown 0.14.3", "lock_api", "once_cell", "parking_lot_core 0.9.8", @@ -2312,15 +2297,6 @@ dependencies = [ "byteorder", ] -[[package]] -name = "hashbrown" -version = "0.11.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ab5ef0d4909ef3724cc8cce6ccc8572c5c817592e9285f5464f8e86f8bd3726e" -dependencies = [ - "ahash 0.7.6", -] - [[package]] name = "hashbrown" version = "0.12.3" @@ -2341,9 +2317,9 @@ dependencies = [ [[package]] name = "hashbrown" -version = "0.14.1" +version = "0.14.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7dfda62a12f55daeae5015f81b0baea145391cb4520f86c248fc615d72640d12" +checksum = "290f1a1d9242c78d09ce40a5e87e7554ee637af1351968159f4952f028f75604" [[package]] name = "headers" @@ -2670,7 +2646,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d530e1a18b1cb4c484e6e34556a0d948706958449fca0cab753d649f2bce3d1f" dependencies = [ "equivalent", - "hashbrown 0.14.1", + "hashbrown 0.14.3", "rayon", ] @@ -3488,7 +3464,7 @@ version = "0.7.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "6c11e44798ad209ccdd91fc192f0526a369a01234f7373e1b141c96d7cee4f0e" dependencies = [ - "proc-macro-crate 1.1.0", + "proc-macro-crate 2.0.0", "proc-macro2", "quote", "syn 2.0.39", @@ -4026,6 +4002,15 @@ dependencies = [ "toml 0.5.8", ] +[[package]] +name = "proc-macro-crate" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7e8366a6159044a37876a2b9817124296703c586a5c92e2c53751fa06d8d43e8" +dependencies = [ + "toml_edit 0.20.7", +] + [[package]] name = "proc-macro-error" version = "1.0.4" @@ -5395,7 +5380,7 @@ dependencies = [ name = "solana-banks-client" version = "1.18.0" dependencies = [ - "borsh 0.10.3", + "borsh 1.2.0", "futures 0.3.29", "solana-banks-interface", "solana-banks-server", @@ -6650,8 +6635,7 @@ dependencies = [ "bincode", "bitflags 2.4.1", "blake3", - "borsh 0.10.3", - "borsh 0.9.3", + "borsh 1.2.0", "bs58", "bv", "bytemuck", @@ -7089,7 +7073,7 @@ dependencies = [ "base64 0.21.5", "bincode", "bitflags 2.4.1", - "borsh 0.10.3", + "borsh 1.2.0", "bs58", "bytemuck", "byteorder", @@ -7436,7 +7420,7 @@ dependencies = [ "Inflector", "base64 0.21.5", "bincode", - "borsh 0.10.3", + "borsh 1.2.0", "bs58", "lazy_static", "log", @@ -8073,6 +8057,18 @@ dependencies = [ "unicode-ident", ] +[[package]] +name = "syn_derive" +version = "0.1.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1329189c02ff984e9736652b1631330da25eaa6bc639089ed4915d25446cbe7b" +dependencies = [ + "proc-macro-error", + "proc-macro2", + "quote", + "syn 2.0.39", +] + [[package]] name = "sync_wrapper" version = "0.1.1" @@ -8542,7 +8538,7 @@ dependencies = [ "serde", "serde_spanned", "toml_datetime", - "toml_edit", + "toml_edit 0.21.0", ] [[package]] @@ -8554,6 +8550,17 @@ dependencies = [ "serde", ] +[[package]] +name = "toml_edit" +version = "0.20.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "70f427fce4d84c72b5b732388bf4a9f4531b53f74e2887e3ecb2481f68f66d81" +dependencies = [ + "indexmap 2.1.0", + "toml_datetime", + "winnow", +] + [[package]] name = "toml_edit" version = "0.21.0" diff --git a/Cargo.toml b/Cargo.toml index 122258fd8680d7..b35315e8fdff4d 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -155,7 +155,7 @@ bincode = "1.3.3" bitflags = { version = "2.3.3", features = ["serde"] } blake3 = "1.5.0" block-buffer = "0.10.4" -borsh = "0.10.3" +borsh = { version = "1.2.0", features = ["derive", "unstable__schema"], default-features = false } bs58 = "0.4.0" bv = "0.11.1" byte-unit = "4.0.19" diff --git a/sdk/program/Cargo.toml b/sdk/program/Cargo.toml index f608ed61943826..ee99477b80c3ef 100644 --- a/sdk/program/Cargo.toml +++ b/sdk/program/Cargo.toml @@ -15,7 +15,8 @@ rust-version = "1.72.0" # solana platform-tools rust version bincode = { workspace = true } blake3 = { workspace = true, features = ["digest", "traits-preview"] } borsh = { workspace = true } -borsh0-9 = { package = "borsh", version = "0.9.3" } +#borsh0-10 = { package = "borsh", version = "0.10.3" } +#borsh0-9 = { package = "borsh", version = "0.9.3" } bs58 = { workspace = true } bv = { workspace = true, features = ["serde"] } bytemuck = { workspace = true, features = ["derive"] } diff --git a/sdk/program/src/borsh.rs b/sdk/program/src/borsh.rs index 90ce42f661f82f..2c19798964f48e 100644 --- a/sdk/program/src/borsh.rs +++ b/sdk/program/src/borsh.rs @@ -8,7 +8,8 @@ //! be removed in a future release //! //! [borsh]: https://borsh.io/ -use borsh::{maybestd::io::Error, BorshDeserialize, BorshSchema, BorshSerialize}; +/* +use borsh0_10::{maybestd::io::Error, BorshDeserialize, BorshSchema, BorshSerialize}; /// Get the worst-case packed length for the given BorshSchema /// @@ -19,6 +20,7 @@ use borsh::{maybestd::io::Error, BorshDeserialize, BorshSchema, BorshSerialize}; note = "Please use `borsh0_10::get_packed_len` instead" )] pub fn get_packed_len() -> usize { + #[allow(deprecated)] crate::borsh0_10::get_packed_len::() } @@ -36,6 +38,7 @@ pub fn get_packed_len() -> usize { note = "Please use `borsh0_10::try_from_slice_unchecked` instead" )] pub fn try_from_slice_unchecked(data: &[u8]) -> Result { + #[allow(deprecated)] crate::borsh0_10::try_from_slice_unchecked::(data) } @@ -50,8 +53,10 @@ pub fn try_from_slice_unchecked(data: &[u8]) -> Result(instance: &T) -> Result { + #[allow(deprecated)] crate::borsh0_10::get_instance_packed_len(instance) } +*/ macro_rules! impl_get_packed_len { ($borsh:ident $(,#[$meta:meta])?) => { @@ -61,45 +66,45 @@ macro_rules! impl_get_packed_len { /// be used on-chain in the Solana SBF execution environment. $(#[$meta])? pub fn get_packed_len() -> usize { - let $borsh::schema::BorshSchemaContainer { declaration, definitions } = - &S::schema_container(); - get_declaration_packed_len(declaration, definitions) + let container = $borsh::schema_container_of::(); + get_declaration_packed_len(container.declaration(), &container) } /// Get packed length for the given BorshSchema Declaration fn get_declaration_packed_len( declaration: &str, - definitions: &std::collections::HashMap<$borsh::schema::Declaration, $borsh::schema::Definition>, + container: &$borsh::schema::BorshSchemaContainer, ) -> usize { - match definitions.get(declaration) { - Some($borsh::schema::Definition::Array { length, elements }) => { - *length as usize * get_declaration_packed_len(elements, definitions) + match container.get_definition(declaration) { + Some($borsh::schema::Definition::Sequence { length_width, length_range, elements }) if *length_width == 0 => { + *length_range.end() as usize * get_declaration_packed_len(elements, container) } - Some($borsh::schema::Definition::Enum { variants }) => { - 1 + variants + Some($borsh::schema::Definition::Enum { tag_width, variants }) => { + (*tag_width as usize) + variants .iter() - .map(|(_, declaration)| get_declaration_packed_len(declaration, definitions)) + .map(|(_, _, declaration)| get_declaration_packed_len(declaration, container)) .max() .unwrap_or(0) } Some($borsh::schema::Definition::Struct { fields }) => match fields { $borsh::schema::Fields::NamedFields(named_fields) => named_fields .iter() - .map(|(_, declaration)| get_declaration_packed_len(declaration, definitions)) + .map(|(_, declaration)| get_declaration_packed_len(declaration, container)) .sum(), $borsh::schema::Fields::UnnamedFields(declarations) => declarations .iter() - .map(|declaration| get_declaration_packed_len(declaration, definitions)) + .map(|declaration| get_declaration_packed_len(declaration, container)) .sum(), $borsh::schema::Fields::Empty => 0, }, Some($borsh::schema::Definition::Sequence { - elements: _elements, + .. }) => panic!("Missing support for Definition::Sequence"), Some($borsh::schema::Definition::Tuple { elements }) => elements .iter() - .map(|element| get_declaration_packed_len(element, definitions)) + .map(|element| get_declaration_packed_len(element, container)) .sum(), + Some($borsh::schema::Definition::Primitive(size)) => *size as usize, None => match declaration { "bool" | "u8" | "i8" => 1, "u16" | "i16" => 2, @@ -116,7 +121,7 @@ macro_rules! impl_get_packed_len { pub(crate) use impl_get_packed_len; macro_rules! impl_try_from_slice_unchecked { - ($borsh:ident $(,#[$meta:meta])?) => { + ($borsh:ident, $borsh_io:ident $(,#[$meta:meta])?) => { /// Deserializes without checking that the entire slice has been consumed /// /// Normally, `try_from_slice` checks the length of the final slice to ensure @@ -127,7 +132,7 @@ macro_rules! impl_try_from_slice_unchecked { /// user passes a buffer destined for a different type, the error won't get caught /// as easily. $(#[$meta])? - pub fn try_from_slice_unchecked(data: &[u8]) -> Result { + pub fn try_from_slice_unchecked(data: &[u8]) -> Result { let mut data_mut = data; let result = T::deserialize(&mut data_mut)?; Ok(result) @@ -137,21 +142,21 @@ macro_rules! impl_try_from_slice_unchecked { pub(crate) use impl_try_from_slice_unchecked; macro_rules! impl_get_instance_packed_len { - ($borsh:ident $(,#[$meta:meta])?) => { + ($borsh:ident, $borsh_io:ident $(,#[$meta:meta])?) => { /// Helper struct which to count how much data would be written during serialization #[derive(Default)] struct WriteCounter { count: usize, } - impl $borsh::maybestd::io::Write for WriteCounter { - fn write(&mut self, data: &[u8]) -> Result { + impl $borsh_io::Write for WriteCounter { + fn write(&mut self, data: &[u8]) -> Result { let amount = data.len(); self.count += amount; Ok(amount) } - fn flush(&mut self) -> Result<(), $borsh::maybestd::io::Error> { + fn flush(&mut self) -> Result<(), $borsh_io::Error> { Ok(()) } } @@ -163,7 +168,7 @@ macro_rules! impl_get_instance_packed_len { /// length only from the type's schema, this can be used when an instance already /// exists, to figure out how much space to allocate in an account. $(#[$meta])? - pub fn get_instance_packed_len(instance: &T) -> Result { + pub fn get_instance_packed_len(instance: &T) -> Result { let mut counter = WriteCounter::default(); instance.serialize(&mut counter)?; Ok(counter.count) @@ -174,11 +179,14 @@ pub(crate) use impl_get_instance_packed_len; #[cfg(test)] macro_rules! impl_tests { - ($borsh:ident) => { + ($borsh:ident, $borsh_io:ident) => { + extern crate alloc; use { super::*, - std::{collections::HashMap, mem::size_of}, - $borsh::{maybestd::io::ErrorKind, BorshDeserialize, BorshSerialize}, + alloc::{collections::BTreeMap, vec::Vec}, + std::mem::size_of, + $borsh::{BorshDeserialize, BorshSerialize}, + $borsh_io::ErrorKind, }; type Child = [u8; 64]; @@ -264,13 +272,13 @@ macro_rules! impl_tests { #[test] fn instance_packed_len_with_varying_sizes_in_hashmap() { - let mut data = HashMap::new(); - let key1 = "the first string, it's actually really really long".to_string(); - let value1 = "".to_string(); - let key2 = "second string, shorter".to_string(); - let value2 = "a real value".to_string(); - let key3 = "third".to_string(); - let value3 = "an even longer value".to_string(); + let mut data = BTreeMap::new(); + let key1 = Vec::from("the first string, it's actually really really long"); + let value1 = Vec::from(""); + let key2 = Vec::from("second string, shorter"); + let value2 = Vec::from("a real value"); + let key3 = Vec::from("third"); + let value3 = Vec::from("an even longer value"); data.insert(key1.clone(), value1.clone()); data.insert(key2.clone(), value2.clone()); data.insert(key3.clone(), value3.clone()); diff --git a/sdk/program/src/borsh0_10.rs b/sdk/program/src/borsh0_10.rs index f29640885e14d6..7a1c1bdd3782dc 100644 --- a/sdk/program/src/borsh0_10.rs +++ b/sdk/program/src/borsh0_10.rs @@ -2,16 +2,43 @@ //! Utilities for the [borsh] serialization format, version 0.10. //! //! [borsh]: https://borsh.io/ -use crate::borsh::{ - impl_get_instance_packed_len, impl_get_packed_len, impl_try_from_slice_unchecked, +use { + crate::borsh::{ + impl_get_instance_packed_len, impl_get_packed_len, impl_try_from_slice_unchecked, + }, + borsh0_10::maybestd::io, }; -impl_get_packed_len!(borsh); -impl_try_from_slice_unchecked!(borsh); -impl_get_instance_packed_len!(borsh); +impl_get_packed_len!( + borsh0_10, + #[deprecated( + since = "1.18.0", + note = "Please upgrade to Borsh 1.X and use `borsh1::get_packed_len` instead" + )] +); +impl_try_from_slice_unchecked!( + borsh0_10, + io, + #[deprecated( + since = "1.18.0", + note = "Please upgrade to Borsh 1.X and use `borsh1::try_from_slice_unchecked` instead" + )] +); +impl_get_instance_packed_len!( + borsh0_10, + io, + #[deprecated( + since = "1.18.0", + note = "Please upgrade to Borsh 1.X and use `borsh1::get_instance_packed_len` instead" + )] +); #[cfg(test)] +#[allow(deprecated)] mod tests { - use crate::borsh::impl_tests; - impl_tests!(borsh); + use { + crate::borsh::impl_tests, + borsh0_10::maybestd::io, + }; + impl_tests!(borsh0_10, io); } diff --git a/sdk/program/src/borsh0_9.rs b/sdk/program/src/borsh0_9.rs index dd9e401db189c9..61d5975dd7707e 100644 --- a/sdk/program/src/borsh0_9.rs +++ b/sdk/program/src/borsh0_9.rs @@ -5,35 +5,43 @@ //! borsh 0.9, even though this crate canonically uses borsh 0.10. //! //! [borsh]: https://borsh.io/ -use crate::borsh::{ - impl_get_instance_packed_len, impl_get_packed_len, impl_try_from_slice_unchecked, +use { + crate::borsh::{ + impl_get_instance_packed_len, impl_get_packed_len, impl_try_from_slice_unchecked, + }, + borsh0_9::maybestd::io, }; impl_get_packed_len!( borsh0_9, #[deprecated( since = "1.17.0", - note = "Please upgrade to Borsh 0.10 and use `borsh0_10::get_packed_len` instead" + note = "Please upgrade to Borsh 1.X and use `borsh1::get_packed_len` instead" )] ); impl_try_from_slice_unchecked!( borsh0_9, + io, #[deprecated( since = "1.17.0", - note = "Please upgrade to Borsh 0.10 and use `borsh0_10::try_from_slice_unchecked` instead" + note = "Please upgrade to Borsh 1.X and use `borsh1::try_from_slice_unchecked` instead" )] ); impl_get_instance_packed_len!( borsh0_9, + io, #[deprecated( since = "1.17.0", - note = "Please upgrade to Borsh 0.10 and use `borsh0_10::get_instance_packed_len` instead" + note = "Please upgrade to Borsh 1.X and use `borsh1::get_instance_packed_len` instead" )] ); #[cfg(test)] #[allow(deprecated)] mod tests { - use crate::borsh::impl_tests; - impl_tests!(borsh0_9); + use { + crate::borsh::impl_tests, + borsh0_9::maybestd::io, + }; + impl_tests!(borsh0_9, io); } diff --git a/sdk/program/src/borsh1.rs b/sdk/program/src/borsh1.rs new file mode 100644 index 00000000000000..a44ea522494232 --- /dev/null +++ b/sdk/program/src/borsh1.rs @@ -0,0 +1,20 @@ +#![allow(clippy::arithmetic_side_effects)] +//! Utilities for the [borsh] serialization format, version 1. +//! +//! [borsh]: https://borsh.io/ +use { + crate::borsh::{ + impl_get_instance_packed_len, impl_get_packed_len_v1, impl_try_from_slice_unchecked, + }, + borsh::io, +}; + +impl_get_packed_len_v1!(borsh); +impl_try_from_slice_unchecked!(borsh, io); +impl_get_instance_packed_len!(borsh, io); + +#[cfg(test)] +mod tests { + use {crate::borsh::impl_tests, borsh::io}; + impl_tests!(borsh, io); +} diff --git a/sdk/program/src/instruction.rs b/sdk/program/src/instruction.rs index e68fc198a36642..f7c782ba72fb21 100644 --- a/sdk/program/src/instruction.rs +++ b/sdk/program/src/instruction.rs @@ -391,7 +391,7 @@ impl Instruction { data: &T, accounts: Vec, ) -> Self { - let data = data.try_to_vec().unwrap(); + let data = borsh::to_vec(data).unwrap(); Self { program_id, accounts, @@ -466,8 +466,7 @@ impl Instruction { /// # pubkey::Pubkey, /// # instruction::{AccountMeta, Instruction}, /// # }; - /// # use borsh::{BorshSerialize, BorshDeserialize}; - /// # use anyhow::Result; + /// # use borsh::{io::Error, BorshSerialize, BorshDeserialize}; /// # /// #[derive(BorshSerialize, BorshDeserialize)] /// pub struct MyInstruction { @@ -479,7 +478,7 @@ impl Instruction { /// from: &Pubkey, /// to: &Pubkey, /// lamports: u64, - /// ) -> Result { + /// ) -> Result { /// let instr = MyInstruction { lamports }; /// /// let mut instr_in_bytes: Vec = Vec::new(); diff --git a/sdk/program/src/lib.rs b/sdk/program/src/lib.rs index 0dfc8c3247cc03..f7322816ce60fa 100644 --- a/sdk/program/src/lib.rs +++ b/sdk/program/src/lib.rs @@ -477,8 +477,9 @@ pub(crate) mod atomic_u64; pub mod big_mod_exp; pub mod blake3; pub mod borsh; -pub mod borsh0_10; -pub mod borsh0_9; +pub mod borsh1; +//pub mod borsh0_10; +//pub mod borsh0_9; pub mod bpf_loader; pub mod bpf_loader_deprecated; pub mod bpf_loader_upgradeable; diff --git a/sdk/program/src/program_error.rs b/sdk/program/src/program_error.rs index 6eb7e9ecd71981..0840ee16b901d7 100644 --- a/sdk/program/src/program_error.rs +++ b/sdk/program/src/program_error.rs @@ -3,7 +3,7 @@ #![allow(clippy::arithmetic_side_effects)] use { crate::{decode_error::DecodeError, instruction::InstructionError, msg, pubkey::PubkeyError}, - borsh::maybestd::io::Error as BorshIoError, + borsh::io::Error as BorshIoError, num_traits::{FromPrimitive, ToPrimitive}, std::convert::TryFrom, thiserror::Error, diff --git a/sdk/program/src/pubkey.rs b/sdk/program/src/pubkey.rs index ebbe5295036fc0..5248f85d257f1c 100644 --- a/sdk/program/src/pubkey.rs +++ b/sdk/program/src/pubkey.rs @@ -668,6 +668,51 @@ impl fmt::Display for Pubkey { } } +/* +impl borsh0_10::de::BorshDeserialize for Pubkey { + fn deserialize(buf: &mut &[u8]) -> ::core::result::Result { + Ok(Self(borsh0_10::BorshDeserialize::deserialize(buf)?)) + } +} +impl borsh0_10::BorshSchema for Pubkey +where + [u8; 32]: borsh0_10::BorshSchema, +{ + fn declaration() -> borsh0_10::schema::Declaration { + "Pubkey".to_string() + } + fn add_definitions_recursively( + definitions: &mut borsh0_10::maybestd::collections::HashMap< + borsh0_10::schema::Declaration, + borsh0_10::schema::Definition, + >, + ) { + let fields = borsh0_10::schema::Fields::UnnamedFields(<[_]>::into_vec( + borsh0_10::maybestd::boxed::Box::new([ + <[u8; 32] as borsh0_10::BorshSchema>::declaration(), + ]), + )); + let definition = borsh0_10::schema::Definition::Struct { fields }; + ::add_definition( + ::declaration(), + definition, + definitions, + ); + <[u8; 32] as borsh0_10::BorshSchema>::add_definitions_recursively(definitions); + } +} +impl borsh0_10::ser::BorshSerialize for Pubkey { + fn serialize( + &self, + writer: &mut W, + ) -> ::core::result::Result<(), borsh0_10::maybestd::io::Error> { + borsh0_10::BorshSerialize::serialize(&self.0, writer)?; + Ok(()) + } +} +*/ + +/* impl borsh0_9::de::BorshDeserialize for Pubkey { fn deserialize(buf: &mut &[u8]) -> ::core::result::Result { Ok(Self(borsh0_9::BorshDeserialize::deserialize(buf)?)) @@ -709,6 +754,7 @@ impl borsh0_9::ser::BorshSerialize for Pubkey { Ok(()) } } +*/ #[cfg(test)] mod tests { diff --git a/sdk/program/src/stake/state.rs b/sdk/program/src/stake/state.rs index 4f94f73b3f2dd5..3bcdc839b64506 100644 --- a/sdk/program/src/stake/state.rs +++ b/sdk/program/src/stake/state.rs @@ -14,7 +14,7 @@ use { }, stake_history::{StakeHistory, StakeHistoryEntry}, }, - borsh::{maybestd::io, BorshDeserialize, BorshSchema, BorshSerialize}, + borsh::{io, BorshDeserialize, BorshSchema, BorshSerialize}, std::collections::HashSet, }; @@ -707,7 +707,7 @@ impl Stake { #[cfg(test)] mod test { use { - super::*, crate::borsh0_10::try_from_slice_unchecked, assert_matches::assert_matches, + super::*, crate::borsh1::try_from_slice_unchecked, assert_matches::assert_matches, bincode::serialize, }; @@ -719,7 +719,7 @@ mod test { fn check_borsh_serialization(stake: StakeStateV2) { let bincode_serialized = serialize(&stake).unwrap(); - let borsh_serialized = StakeStateV2::try_to_vec(&stake).unwrap(); + let borsh_serialized = borsh::to_vec(&stake).unwrap(); assert_eq!(bincode_serialized, borsh_serialized); } @@ -850,7 +850,7 @@ mod test { ); let bincode_serialized = serialize(&stake).unwrap(); - let borsh_serialized = StakeStateV2::try_to_vec(&stake).unwrap(); + let borsh_serialized = borsh::to_vec(&stake).unwrap(); assert_eq!(bincode_serialized[FLAG_OFFSET], expected); assert_eq!(borsh_serialized[FLAG_OFFSET], expected); @@ -872,7 +872,7 @@ mod test { fn check_borsh_serialization(stake: StakeState) { let bincode_serialized = serialize(&stake).unwrap(); - let borsh_serialized = StakeState::try_to_vec(&stake).unwrap(); + let borsh_serialized = borsh::to_vec(&stake).unwrap(); assert_eq!(bincode_serialized, borsh_serialized); } diff --git a/sdk/src/compute_budget.rs b/sdk/src/compute_budget.rs index 84d0c3766023c6..357ae787a029f8 100644 --- a/sdk/src/compute_budget.rs +++ b/sdk/src/compute_budget.rs @@ -64,8 +64,8 @@ impl ComputeBudgetInstruction { /// Serialize Instruction using borsh, this is only used in runtime::cost_model::tests but compilation /// can't be restricted as it's used across packages // #[cfg(test)] - pub fn pack(self) -> Result, std::io::Error> { - self.try_to_vec() + pub fn pack(self) -> Result, borsh::io::Error> { + borsh::to_vec(&self) } /// Create a `ComputeBudgetInstruction::SetLoadedAccountsDataSizeLimit` `Instruction` diff --git a/sdk/src/lib.rs b/sdk/src/lib.rs index eaaea2d481cbd9..0359d5f49b93de 100644 --- a/sdk/src/lib.rs +++ b/sdk/src/lib.rs @@ -45,7 +45,7 @@ pub use solana_program::address_lookup_table_account; #[cfg(not(target_os = "solana"))] pub use solana_program::program_stubs; pub use solana_program::{ - account_info, address_lookup_table, alt_bn128, big_mod_exp, blake3, borsh, borsh0_10, borsh0_9, + account_info, address_lookup_table, alt_bn128, big_mod_exp, blake3, borsh, //borsh0_10, borsh0_9, bpf_loader, bpf_loader_deprecated, bpf_loader_upgradeable, clock, config, custom_heap_default, custom_panic_default, debug_account_data, declare_deprecated_sysvar_id, declare_sysvar_id, decode_error, ed25519_program, epoch_rewards, epoch_schedule, fee_calculator, impl_sysvar_get, From 541340351750fcda7d6fd6be3078b14f0c6c436d Mon Sep 17 00:00:00 2001 From: Jon C Date: Wed, 6 Dec 2023 15:29:52 +0100 Subject: [PATCH 02/10] Restore borsh 0.10 and 0.9 support --- Cargo.lock | 62 +++++++- Cargo.toml | 2 +- cost-model/src/cost_model.rs | 2 +- .../src/compute_budget_processor.rs | 2 +- sdk/program/Cargo.toml | 4 +- sdk/program/src/blake3.rs | 1 + sdk/program/src/borsh.rs | 68 ++++++++- sdk/program/src/borsh0_10.rs | 9 +- sdk/program/src/borsh0_9.rs | 9 +- sdk/program/src/hash.rs | 1 + sdk/program/src/instruction.rs | 4 + sdk/program/src/keccak.rs | 1 + sdk/program/src/lib.rs | 4 +- sdk/program/src/message/legacy.rs | 3 + sdk/program/src/pubkey.rs | 134 ++++++++---------- sdk/program/src/secp256k1_recover.rs | 3 + sdk/program/src/stake/stake_flags.rs | 1 + sdk/program/src/stake/state.rs | 5 + sdk/program/src/system_instruction.rs | 5 + sdk/src/lib.rs | 21 +-- transaction-status/Cargo.toml | 2 +- 21 files changed, 229 insertions(+), 114 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 41b9c951b75de7..7cd0e832ea03cf 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -706,6 +706,16 @@ version = "0.2.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "8d696c370c750c948ada61c69a0ee2cbbb9c50b1019ddb86d9317157a99c2cae" +[[package]] +name = "borsh" +version = "0.9.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "15bf3650200d8bffa99015595e10f1fbd17de07abbc25bb067da79e769939bfa" +dependencies = [ + "borsh-derive 0.9.3", + "hashbrown 0.11.2", +] + [[package]] name = "borsh" version = "0.10.3" @@ -726,14 +736,27 @@ dependencies = [ "cfg_aliases", ] +[[package]] +name = "borsh-derive" +version = "0.9.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6441c552f230375d18e3cc377677914d2ca2b0d36e52129fe15450a2dce46775" +dependencies = [ + "borsh-derive-internal 0.9.3", + "borsh-schema-derive-internal 0.9.3", + "proc-macro-crate 0.1.5", + "proc-macro2", + "syn 1.0.109", +] + [[package]] name = "borsh-derive" version = "0.10.3" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "0754613691538d51f329cce9af41d7b7ca150bc973056f1156611489475f54f7" dependencies = [ - "borsh-derive-internal", - "borsh-schema-derive-internal", + "borsh-derive-internal 0.10.3", + "borsh-schema-derive-internal 0.10.3", "proc-macro-crate 0.1.5", "proc-macro2", "syn 1.0.109", @@ -753,6 +776,17 @@ dependencies = [ "syn_derive", ] +[[package]] +name = "borsh-derive-internal" +version = "0.9.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5449c28a7b352f2d1e592a8a28bf139bc71afb0764a14f3c02500935d8c44065" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] + [[package]] name = "borsh-derive-internal" version = "0.10.3" @@ -764,6 +798,17 @@ dependencies = [ "syn 1.0.109", ] +[[package]] +name = "borsh-schema-derive-internal" +version = "0.9.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cdbd5696d8bfa21d53d9fe39a714a18538bad11492a42d066dbbc395fb1951c0" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] + [[package]] name = "borsh-schema-derive-internal" version = "0.10.3" @@ -2297,6 +2342,15 @@ dependencies = [ "byteorder", ] +[[package]] +name = "hashbrown" +version = "0.11.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ab5ef0d4909ef3724cc8cce6ccc8572c5c817592e9285f5464f8e86f8bd3726e" +dependencies = [ + "ahash 0.7.6", +] + [[package]] name = "hashbrown" version = "0.12.3" @@ -6635,6 +6689,8 @@ dependencies = [ "bincode", "bitflags 2.4.1", "blake3", + "borsh 0.10.3", + "borsh 0.9.3", "borsh 1.2.0", "bs58", "bv", @@ -7420,7 +7476,7 @@ dependencies = [ "Inflector", "base64 0.21.5", "bincode", - "borsh 1.2.0", + "borsh 0.10.3", "bs58", "lazy_static", "log", diff --git a/Cargo.toml b/Cargo.toml index b35315e8fdff4d..6d28f02bc2a7fc 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -155,7 +155,7 @@ bincode = "1.3.3" bitflags = { version = "2.3.3", features = ["serde"] } blake3 = "1.5.0" block-buffer = "0.10.4" -borsh = { version = "1.2.0", features = ["derive", "unstable__schema"], default-features = false } +borsh = { version = "1.2.0", features = ["derive", "unstable__schema"] } bs58 = "0.4.0" bv = "0.11.1" byte-unit = "4.0.19" diff --git a/cost-model/src/cost_model.rs b/cost-model/src/cost_model.rs index bb3e296d6dcbe0..1232e5c3c9be6e 100644 --- a/cost-model/src/cost_model.rs +++ b/cost-model/src/cost_model.rs @@ -16,7 +16,7 @@ use { }, }, solana_sdk::{ - borsh0_10::try_from_slice_unchecked, + borsh1::try_from_slice_unchecked, compute_budget::{self, ComputeBudgetInstruction}, feature_set::{include_loaded_accounts_data_size_in_fee_calculation, FeatureSet}, fee::FeeStructure, diff --git a/program-runtime/src/compute_budget_processor.rs b/program-runtime/src/compute_budget_processor.rs index b2c3a892493d41..3b705d334d5bc9 100644 --- a/program-runtime/src/compute_budget_processor.rs +++ b/program-runtime/src/compute_budget_processor.rs @@ -5,7 +5,7 @@ use { prioritization_fee::{PrioritizationFeeDetails, PrioritizationFeeType}, }, solana_sdk::{ - borsh0_10::try_from_slice_unchecked, + borsh1::try_from_slice_unchecked, compute_budget::{self, ComputeBudgetInstruction}, entrypoint::HEAP_LENGTH as MIN_HEAP_FRAME_BYTES, feature_set::{ diff --git a/sdk/program/Cargo.toml b/sdk/program/Cargo.toml index ee99477b80c3ef..ccd18701eefcc4 100644 --- a/sdk/program/Cargo.toml +++ b/sdk/program/Cargo.toml @@ -15,8 +15,8 @@ rust-version = "1.72.0" # solana platform-tools rust version bincode = { workspace = true } blake3 = { workspace = true, features = ["digest", "traits-preview"] } borsh = { workspace = true } -#borsh0-10 = { package = "borsh", version = "0.10.3" } -#borsh0-9 = { package = "borsh", version = "0.9.3" } +borsh0-10 = { package = "borsh", version = "0.10.3" } +borsh0-9 = { package = "borsh", version = "0.9.3" } bs58 = { workspace = true } bv = { workspace = true, features = ["serde"] } bytemuck = { workspace = true, features = ["derive"] } diff --git a/sdk/program/src/blake3.rs b/sdk/program/src/blake3.rs index d8351b06c6ad0d..cc50318e336c41 100644 --- a/sdk/program/src/blake3.rs +++ b/sdk/program/src/blake3.rs @@ -31,6 +31,7 @@ const MAX_BASE58_LEN: usize = 44; Hash, AbiExample, )] +#[borsh(crate = "borsh")] #[repr(transparent)] pub struct Hash(pub [u8; HASH_BYTES]); diff --git a/sdk/program/src/borsh.rs b/sdk/program/src/borsh.rs index 2c19798964f48e..8d36b536da8db6 100644 --- a/sdk/program/src/borsh.rs +++ b/sdk/program/src/borsh.rs @@ -8,7 +8,6 @@ //! be removed in a future release //! //! [borsh]: https://borsh.io/ -/* use borsh0_10::{maybestd::io::Error, BorshDeserialize, BorshSchema, BorshSerialize}; /// Get the worst-case packed length for the given BorshSchema @@ -56,9 +55,70 @@ pub fn get_instance_packed_len(instance: &T) -> Result { + /// Get the worst-case packed length for the given BorshSchema + /// + /// Note: due to the serializer currently used by Borsh, this function cannot + /// be used on-chain in the Solana SBF execution environment. + $(#[$meta])? + pub fn get_packed_len() -> usize { + let $borsh::schema::BorshSchemaContainer { declaration, definitions } = + &S::schema_container(); + get_declaration_packed_len(declaration, definitions) + } + + /// Get packed length for the given BorshSchema Declaration + fn get_declaration_packed_len( + declaration: &str, + definitions: &std::collections::HashMap<$borsh::schema::Declaration, $borsh::schema::Definition>, + ) -> usize { + match definitions.get(declaration) { + Some($borsh::schema::Definition::Array { length, elements }) => { + *length as usize * get_declaration_packed_len(elements, definitions) + } + Some($borsh::schema::Definition::Enum { variants }) => { + 1 + variants + .iter() + .map(|(_, declaration)| get_declaration_packed_len(declaration, definitions)) + .max() + .unwrap_or(0) + } + Some($borsh::schema::Definition::Struct { fields }) => match fields { + $borsh::schema::Fields::NamedFields(named_fields) => named_fields + .iter() + .map(|(_, declaration)| get_declaration_packed_len(declaration, definitions)) + .sum(), + $borsh::schema::Fields::UnnamedFields(declarations) => declarations + .iter() + .map(|declaration| get_declaration_packed_len(declaration, definitions)) + .sum(), + $borsh::schema::Fields::Empty => 0, + }, + Some($borsh::schema::Definition::Sequence { + elements: _elements, + }) => panic!("Missing support for Definition::Sequence"), + Some($borsh::schema::Definition::Tuple { elements }) => elements + .iter() + .map(|element| get_declaration_packed_len(element, definitions)) + .sum(), + None => match declaration { + "bool" | "u8" | "i8" => 1, + "u16" | "i16" => 2, + "u32" | "i32" => 4, + "u64" | "i64" => 8, + "u128" | "i128" => 16, + "nil" => 0, + _ => panic!("Missing primitive type: {declaration}"), + }, + } + } + } +} +pub(crate) use impl_get_packed_len_v0; + +macro_rules! impl_get_packed_len_v1 { ($borsh:ident $(,#[$meta:meta])?) => { /// Get the worst-case packed length for the given BorshSchema /// @@ -118,7 +178,7 @@ macro_rules! impl_get_packed_len { } } } -pub(crate) use impl_get_packed_len; +pub(crate) use impl_get_packed_len_v1; macro_rules! impl_try_from_slice_unchecked { ($borsh:ident, $borsh_io:ident $(,#[$meta:meta])?) => { diff --git a/sdk/program/src/borsh0_10.rs b/sdk/program/src/borsh0_10.rs index 7a1c1bdd3782dc..c7d190f820b366 100644 --- a/sdk/program/src/borsh0_10.rs +++ b/sdk/program/src/borsh0_10.rs @@ -4,12 +4,12 @@ //! [borsh]: https://borsh.io/ use { crate::borsh::{ - impl_get_instance_packed_len, impl_get_packed_len, impl_try_from_slice_unchecked, + impl_get_instance_packed_len, impl_get_packed_len_v0, impl_try_from_slice_unchecked, }, borsh0_10::maybestd::io, }; -impl_get_packed_len!( +impl_get_packed_len_v0!( borsh0_10, #[deprecated( since = "1.18.0", @@ -36,9 +36,6 @@ impl_get_instance_packed_len!( #[cfg(test)] #[allow(deprecated)] mod tests { - use { - crate::borsh::impl_tests, - borsh0_10::maybestd::io, - }; + use {crate::borsh::impl_tests, borsh0_10::maybestd::io}; impl_tests!(borsh0_10, io); } diff --git a/sdk/program/src/borsh0_9.rs b/sdk/program/src/borsh0_9.rs index 61d5975dd7707e..d7d1e97013f898 100644 --- a/sdk/program/src/borsh0_9.rs +++ b/sdk/program/src/borsh0_9.rs @@ -7,12 +7,12 @@ //! [borsh]: https://borsh.io/ use { crate::borsh::{ - impl_get_instance_packed_len, impl_get_packed_len, impl_try_from_slice_unchecked, + impl_get_instance_packed_len, impl_get_packed_len_v0, impl_try_from_slice_unchecked, }, borsh0_9::maybestd::io, }; -impl_get_packed_len!( +impl_get_packed_len_v0!( borsh0_9, #[deprecated( since = "1.17.0", @@ -39,9 +39,6 @@ impl_get_instance_packed_len!( #[cfg(test)] #[allow(deprecated)] mod tests { - use { - crate::borsh::impl_tests, - borsh0_9::maybestd::io, - }; + use {crate::borsh::impl_tests, borsh0_9::maybestd::io}; impl_tests!(borsh0_9, io); } diff --git a/sdk/program/src/hash.rs b/sdk/program/src/hash.rs index 27d481b62b5441..288f696df31b93 100644 --- a/sdk/program/src/hash.rs +++ b/sdk/program/src/hash.rs @@ -46,6 +46,7 @@ const MAX_BASE58_LEN: usize = 44; Pod, Zeroable, )] +#[borsh(crate = "borsh")] #[repr(transparent)] pub struct Hash(pub(crate) [u8; HASH_BYTES]); diff --git a/sdk/program/src/instruction.rs b/sdk/program/src/instruction.rs index f7c782ba72fb21..21b3e774ae0b22 100644 --- a/sdk/program/src/instruction.rs +++ b/sdk/program/src/instruction.rs @@ -364,6 +364,7 @@ impl Instruction { /// # use borsh::{BorshSerialize, BorshDeserialize}; /// # /// #[derive(BorshSerialize, BorshDeserialize)] + /// # #[borsh(crate = "borsh")] /// pub struct MyInstruction { /// pub lamports: u64, /// } @@ -469,6 +470,7 @@ impl Instruction { /// # use borsh::{io::Error, BorshSerialize, BorshDeserialize}; /// # /// #[derive(BorshSerialize, BorshDeserialize)] + /// # #[borsh(crate = "borsh")] /// pub struct MyInstruction { /// pub lamports: u64, /// } @@ -557,6 +559,7 @@ impl AccountMeta { /// # use borsh::{BorshSerialize, BorshDeserialize}; /// # /// # #[derive(BorshSerialize, BorshDeserialize)] + /// # #[borsh(crate = "borsh")] /// # pub struct MyInstruction; /// # /// # let instruction = MyInstruction; @@ -592,6 +595,7 @@ impl AccountMeta { /// # use borsh::{BorshSerialize, BorshDeserialize}; /// # /// # #[derive(BorshSerialize, BorshDeserialize)] + /// # #[borsh(crate = "borsh")] /// # pub struct MyInstruction; /// # /// # let instruction = MyInstruction; diff --git a/sdk/program/src/keccak.rs b/sdk/program/src/keccak.rs index 17829485c2bdac..6a1cfaf1113b7b 100644 --- a/sdk/program/src/keccak.rs +++ b/sdk/program/src/keccak.rs @@ -29,6 +29,7 @@ const MAX_BASE58_LEN: usize = 44; Hash, AbiExample, )] +#[borsh(crate = "borsh")] #[repr(transparent)] pub struct Hash(pub [u8; HASH_BYTES]); diff --git a/sdk/program/src/lib.rs b/sdk/program/src/lib.rs index f7322816ce60fa..50438337436702 100644 --- a/sdk/program/src/lib.rs +++ b/sdk/program/src/lib.rs @@ -477,9 +477,9 @@ pub(crate) mod atomic_u64; pub mod big_mod_exp; pub mod blake3; pub mod borsh; +pub mod borsh0_10; +pub mod borsh0_9; pub mod borsh1; -//pub mod borsh0_10; -//pub mod borsh0_9; pub mod bpf_loader; pub mod bpf_loader_deprecated; pub mod bpf_loader_upgradeable; diff --git a/sdk/program/src/message/legacy.rs b/sdk/program/src/message/legacy.rs index e81c7c485ff5f6..868cb9bc94df49 100644 --- a/sdk/program/src/message/legacy.rs +++ b/sdk/program/src/message/legacy.rs @@ -193,6 +193,7 @@ impl Message { /// // another crate so it can be shared between the on-chain program and /// // the client. /// #[derive(BorshSerialize, BorshDeserialize)] + /// # #[borsh(crate = "borsh")] /// enum BankInstruction { /// Initialize, /// Deposit { lamports: u64 }, @@ -264,6 +265,7 @@ impl Message { /// // another crate so it can be shared between the on-chain program and /// // the client. /// #[derive(BorshSerialize, BorshDeserialize)] + /// # #[borsh(crate = "borsh")] /// enum BankInstruction { /// Initialize, /// Deposit { lamports: u64 }, @@ -363,6 +365,7 @@ impl Message { /// // another crate so it can be shared between the on-chain program and /// // the client. /// #[derive(BorshSerialize, BorshDeserialize)] + /// # #[borsh(crate = "borsh")] /// enum BankInstruction { /// Initialize, /// Deposit { lamports: u64 }, diff --git a/sdk/program/src/pubkey.rs b/sdk/program/src/pubkey.rs index 5248f85d257f1c..122c74dbeffdba 100644 --- a/sdk/program/src/pubkey.rs +++ b/sdk/program/src/pubkey.rs @@ -84,6 +84,7 @@ impl From for PubkeyError { Serialize, Zeroable, )] +#[borsh(crate = "borsh")] pub struct Pubkey(pub(crate) [u8; 32]); impl crate::sanitize::Sanitize for Pubkey {} @@ -328,6 +329,7 @@ impl Pubkey { /// // The computed address of the PDA will be passed to this program via /// // the `accounts` vector of the `Instruction` type. /// #[derive(BorshSerialize, BorshDeserialize, Debug)] + /// # #[borsh(crate = "borsh")] /// pub struct InstructionData { /// pub vault_bump_seed: u8, /// pub lamports: u64, @@ -409,6 +411,7 @@ impl Pubkey { /// # use anyhow::Result; /// # /// # #[derive(BorshSerialize, BorshDeserialize, Debug)] + /// # #[borsh(crate = "borsh")] /// # struct InstructionData { /// # pub vault_bump_seed: u8, /// # pub lamports: u64, @@ -668,93 +671,70 @@ impl fmt::Display for Pubkey { } } -/* impl borsh0_10::de::BorshDeserialize for Pubkey { - fn deserialize(buf: &mut &[u8]) -> ::core::result::Result { - Ok(Self(borsh0_10::BorshDeserialize::deserialize(buf)?)) + fn deserialize_reader( + reader: &mut R, + ) -> ::core::result::Result { + Ok(Self(borsh0_10::BorshDeserialize::deserialize_reader( + reader, + )?)) } } -impl borsh0_10::BorshSchema for Pubkey -where - [u8; 32]: borsh0_10::BorshSchema, -{ - fn declaration() -> borsh0_10::schema::Declaration { - "Pubkey".to_string() - } - fn add_definitions_recursively( - definitions: &mut borsh0_10::maybestd::collections::HashMap< - borsh0_10::schema::Declaration, - borsh0_10::schema::Definition, - >, - ) { - let fields = borsh0_10::schema::Fields::UnnamedFields(<[_]>::into_vec( - borsh0_10::maybestd::boxed::Box::new([ - <[u8; 32] as borsh0_10::BorshSchema>::declaration(), - ]), - )); - let definition = borsh0_10::schema::Definition::Struct { fields }; - ::add_definition( - ::declaration(), - definition, - definitions, - ); - <[u8; 32] as borsh0_10::BorshSchema>::add_definitions_recursively(definitions); - } -} -impl borsh0_10::ser::BorshSerialize for Pubkey { - fn serialize( - &self, - writer: &mut W, - ) -> ::core::result::Result<(), borsh0_10::maybestd::io::Error> { - borsh0_10::BorshSerialize::serialize(&self.0, writer)?; - Ok(()) - } -} -*/ - -/* impl borsh0_9::de::BorshDeserialize for Pubkey { fn deserialize(buf: &mut &[u8]) -> ::core::result::Result { Ok(Self(borsh0_9::BorshDeserialize::deserialize(buf)?)) } } -impl borsh0_9::BorshSchema for Pubkey -where - [u8; 32]: borsh0_9::BorshSchema, -{ - fn declaration() -> borsh0_9::schema::Declaration { - "Pubkey".to_string() - } - fn add_definitions_recursively( - definitions: &mut borsh0_9::maybestd::collections::HashMap< - borsh0_9::schema::Declaration, - borsh0_9::schema::Definition, - >, - ) { - let fields = borsh0_9::schema::Fields::UnnamedFields(<[_]>::into_vec( - borsh0_9::maybestd::boxed::Box::new([ - <[u8; 32] as borsh0_9::BorshSchema>::declaration(), - ]), - )); - let definition = borsh0_9::schema::Definition::Struct { fields }; - ::add_definition( - ::declaration(), - definition, - definitions, - ); - <[u8; 32] as borsh0_9::BorshSchema>::add_definitions_recursively(definitions); - } + +macro_rules! impl_borsh_schema { + ($borsh:ident) => { + impl $borsh::BorshSchema for Pubkey + where + [u8; 32]: $borsh::BorshSchema, + { + fn declaration() -> $borsh::schema::Declaration { + "Pubkey".to_string() + } + fn add_definitions_recursively( + definitions: &mut $borsh::maybestd::collections::HashMap< + $borsh::schema::Declaration, + $borsh::schema::Definition, + >, + ) { + let fields = $borsh::schema::Fields::UnnamedFields(<[_]>::into_vec( + $borsh::maybestd::boxed::Box::new([ + <[u8; 32] as $borsh::BorshSchema>::declaration(), + ]), + )); + let definition = $borsh::schema::Definition::Struct { fields }; + ::add_definition( + ::declaration(), + definition, + definitions, + ); + <[u8; 32] as $borsh::BorshSchema>::add_definitions_recursively(definitions); + } + } + }; } -impl borsh0_9::ser::BorshSerialize for Pubkey { - fn serialize( - &self, - writer: &mut W, - ) -> ::core::result::Result<(), borsh0_9::maybestd::io::Error> { - borsh0_9::BorshSerialize::serialize(&self.0, writer)?; - Ok(()) - } +impl_borsh_schema!(borsh0_10); +impl_borsh_schema!(borsh0_9); + +macro_rules! impl_borsh_serialize { + ($borsh:ident) => { + impl $borsh::ser::BorshSerialize for Pubkey { + fn serialize( + &self, + writer: &mut W, + ) -> ::core::result::Result<(), $borsh::maybestd::io::Error> { + $borsh::BorshSerialize::serialize(&self.0, writer)?; + Ok(()) + } + } + }; } -*/ +impl_borsh_serialize!(borsh0_10); +impl_borsh_serialize!(borsh0_9); #[cfg(test)] mod tests { diff --git a/sdk/program/src/secp256k1_recover.rs b/sdk/program/src/secp256k1_recover.rs index 5bca285c2f8849..8e2e3be058536f 100644 --- a/sdk/program/src/secp256k1_recover.rs +++ b/sdk/program/src/secp256k1_recover.rs @@ -78,6 +78,7 @@ pub const SECP256K1_PUBLIC_KEY_LENGTH: usize = 64; Hash, AbiExample, )] +#[borsh(crate = "borsh")] pub struct Secp256k1Pubkey(pub [u8; SECP256K1_PUBLIC_KEY_LENGTH]); impl Secp256k1Pubkey { @@ -254,6 +255,7 @@ impl Secp256k1Pubkey { /// use borsh::{BorshDeserialize, BorshSerialize}; /// /// #[derive(BorshSerialize, BorshDeserialize, Debug)] +/// # #[borsh(crate = "borsh")] /// pub struct DemoSecp256k1RecoverInstruction { /// pub message: Vec, /// pub signature: [u8; 64], @@ -348,6 +350,7 @@ impl Secp256k1Pubkey { /// }; /// # use borsh::{BorshDeserialize, BorshSerialize}; /// # #[derive(BorshSerialize, BorshDeserialize, Debug)] +/// # #[borsh(crate = "borsh")] /// # pub struct DemoSecp256k1RecoverInstruction { /// # pub message: Vec, /// # pub signature: [u8; 64], diff --git a/sdk/program/src/stake/stake_flags.rs b/sdk/program/src/stake/stake_flags.rs index 1d56c77fb9e0c0..3bb788d0f79013 100644 --- a/sdk/program/src/stake/stake_flags.rs +++ b/sdk/program/src/stake/stake_flags.rs @@ -17,6 +17,7 @@ use borsh::{BorshDeserialize, BorshSchema, BorshSerialize}; Hash, Debug, )] +#[borsh(crate = "borsh")] pub struct StakeFlags { bits: u8, } diff --git a/sdk/program/src/stake/state.rs b/sdk/program/src/stake/state.rs index 3bcdc839b64506..91786f3b93676d 100644 --- a/sdk/program/src/stake/state.rs +++ b/sdk/program/src/stake/state.rs @@ -241,6 +241,7 @@ pub enum StakeAuthorize { BorshSchema, BorshSerialize, )] +#[borsh(crate = "borsh")] pub struct Lockup { /// UnixTimestamp at which this stake will allow withdrawal, unless the /// transaction is signed by the custodian @@ -276,6 +277,7 @@ impl Lockup { BorshSchema, BorshSerialize, )] +#[borsh(crate = "borsh")] pub struct Authorized { pub staker: Pubkey, pub withdrawer: Pubkey, @@ -356,6 +358,7 @@ impl Authorized { BorshSchema, BorshSerialize, )] +#[borsh(crate = "borsh")] pub struct Meta { pub rent_exempt_reserve: u64, pub authorized: Authorized, @@ -411,6 +414,7 @@ impl Meta { BorshSchema, BorshSerialize, )] +#[borsh(crate = "borsh")] pub struct Delegation { /// to whom the stake is delegated pub voter_pubkey: Pubkey, @@ -658,6 +662,7 @@ impl Delegation { BorshSchema, BorshSerialize, )] +#[borsh(crate = "borsh")] pub struct Stake { pub delegation: Delegation, /// credits observed is credits from vote account state when delegated or redeemed diff --git a/sdk/program/src/system_instruction.rs b/sdk/program/src/system_instruction.rs index 74646f7fb7d331..da2065ccce9d69 100644 --- a/sdk/program/src/system_instruction.rs +++ b/sdk/program/src/system_instruction.rs @@ -378,6 +378,7 @@ pub enum SystemInstruction { /// }; /// /// #[derive(BorshSerialize, BorshDeserialize, Debug)] +/// # #[borsh(crate = "borsh")] /// pub struct CreateAccountInstruction { /// /// The PDA seed used to distinguish the new account from other PDAs /// pub new_account_seed: [u8; 16], @@ -594,6 +595,7 @@ pub fn create_account_with_seed( /// }; /// /// #[derive(BorshSerialize, BorshDeserialize, Debug)] +/// # #[borsh(crate = "borsh")] /// pub struct CreateAccountInstruction { /// /// The PDA seed used to distinguish the new account from other PDAs /// pub new_account_seed: [u8; 16], @@ -804,6 +806,7 @@ pub fn assign_with_seed( /// }; /// /// #[derive(BorshSerialize, BorshDeserialize, Debug)] +/// # #[borsh(crate = "borsh")] /// pub struct CreateAccountInstruction { /// /// The PDA seed used to distinguish the new account from other PDAs /// pub new_account_seed: [u8; 16], @@ -1023,6 +1026,7 @@ pub fn transfer_with_seed( /// }; /// /// #[derive(BorshSerialize, BorshDeserialize, Debug)] +/// # #[borsh(crate = "borsh")] /// pub struct CreateAccountInstruction { /// /// The PDA seed used to distinguish the new account from other PDAs /// pub new_account_seed: [u8; 16], @@ -1220,6 +1224,7 @@ pub fn allocate_with_seed( /// /// - 1: system_program - executable /// /// - *: to - writable /// #[derive(BorshSerialize, BorshDeserialize, Debug)] +/// # #[borsh(crate = "borsh")] /// pub struct TransferLamportsToManyInstruction { /// pub bank_pda_bump_seed: u8, /// pub amount_list: Vec, diff --git a/sdk/src/lib.rs b/sdk/src/lib.rs index 0359d5f49b93de..1b6763f8ea145f 100644 --- a/sdk/src/lib.rs +++ b/sdk/src/lib.rs @@ -45,16 +45,17 @@ pub use solana_program::address_lookup_table_account; #[cfg(not(target_os = "solana"))] pub use solana_program::program_stubs; pub use solana_program::{ - account_info, address_lookup_table, alt_bn128, big_mod_exp, blake3, borsh, //borsh0_10, borsh0_9, - bpf_loader, bpf_loader_deprecated, bpf_loader_upgradeable, clock, config, custom_heap_default, - custom_panic_default, debug_account_data, declare_deprecated_sysvar_id, declare_sysvar_id, - decode_error, ed25519_program, epoch_rewards, epoch_schedule, fee_calculator, impl_sysvar_get, - incinerator, instruction, keccak, lamports, loader_instruction, loader_upgradeable_instruction, - loader_v4, loader_v4_instruction, message, msg, native_token, nonce, poseidon, program, - program_error, program_memory, program_option, program_pack, rent, sanitize, sdk_ids, - secp256k1_program, secp256k1_recover, serde_varint, serialize_utils, short_vec, slot_hashes, - slot_history, stable_layout, stake, stake_history, syscalls, system_instruction, - system_program, sysvar, unchecked_div_by_const, vote, wasm_bindgen, + account_info, address_lookup_table, alt_bn128, big_mod_exp, blake3, borsh, borsh0_10, borsh0_9, + borsh1, bpf_loader, bpf_loader_deprecated, bpf_loader_upgradeable, clock, config, + custom_heap_default, custom_panic_default, debug_account_data, declare_deprecated_sysvar_id, + declare_sysvar_id, decode_error, ed25519_program, epoch_rewards, epoch_schedule, + fee_calculator, impl_sysvar_get, incinerator, instruction, keccak, lamports, + loader_instruction, loader_upgradeable_instruction, loader_v4, loader_v4_instruction, message, + msg, native_token, nonce, poseidon, program, program_error, program_memory, program_option, + program_pack, rent, sanitize, sdk_ids, secp256k1_program, secp256k1_recover, serde_varint, + serialize_utils, short_vec, slot_hashes, slot_history, stable_layout, stake, stake_history, + syscalls, system_instruction, system_program, sysvar, unchecked_div_by_const, vote, + wasm_bindgen, }; pub mod account; diff --git a/transaction-status/Cargo.toml b/transaction-status/Cargo.toml index 3c830f591403fe..b9d6c8c543c022 100644 --- a/transaction-status/Cargo.toml +++ b/transaction-status/Cargo.toml @@ -13,7 +13,7 @@ edition = { workspace = true } Inflector = { workspace = true } base64 = { workspace = true } bincode = { workspace = true } -borsh = { workspace = true } +borsh = { version = "0.10.3" } bs58 = { workspace = true } lazy_static = { workspace = true } log = { workspace = true } From 7e09ed4d9c8e1bc647ad05774146ab1f324cc7bb Mon Sep 17 00:00:00 2001 From: Jon C Date: Wed, 6 Dec 2023 15:30:38 +0100 Subject: [PATCH 03/10] Update sbf lockfile --- programs/sbf/Cargo.lock | 83 ++++++++++++++++++++++++++++++++++++++++- 1 file changed, 81 insertions(+), 2 deletions(-) diff --git a/programs/sbf/Cargo.lock b/programs/sbf/Cargo.lock index d429df77b5f1c2..cf31b2084d28c9 100644 --- a/programs/sbf/Cargo.lock +++ b/programs/sbf/Cargo.lock @@ -686,6 +686,16 @@ dependencies = [ "hashbrown 0.13.2", ] +[[package]] +name = "borsh" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bf617fabf5cdbdc92f774bfe5062d870f228b80056d41180797abf48bed4056e" +dependencies = [ + "borsh-derive 1.2.0", + "cfg_aliases", +] + [[package]] name = "borsh-derive" version = "0.9.3" @@ -712,6 +722,20 @@ dependencies = [ "syn 1.0.109", ] +[[package]] +name = "borsh-derive" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f404657a7ea7b5249e36808dff544bc88a28f26e0ac40009f674b7a009d14be3" +dependencies = [ + "once_cell", + "proc-macro-crate 2.0.1", + "proc-macro2", + "quote", + "syn 2.0.39", + "syn_derive", +] + [[package]] name = "borsh-derive-internal" version = "0.9.3" @@ -914,6 +938,12 @@ version = "1.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" +[[package]] +name = "cfg_aliases" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fd16c4719339c4530435d38e511904438d07cce7950afa3718a84ac36c10e89e" + [[package]] name = "chrono" version = "0.4.31" @@ -3616,6 +3646,16 @@ dependencies = [ "toml", ] +[[package]] +name = "proc-macro-crate" +version = "2.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "97dc5fea232fc28d2f597b37c4876b348a40e33f3b02cc975c8d006d78d94b1a" +dependencies = [ + "toml_datetime", + "toml_edit", +] + [[package]] name = "proc-macro-error" version = "1.0.4" @@ -4718,7 +4758,7 @@ dependencies = [ name = "solana-banks-client" version = "1.18.0" dependencies = [ - "borsh 0.10.3", + "borsh 1.2.0", "futures 0.3.29", "solana-banks-interface", "solana-program", @@ -5415,6 +5455,7 @@ dependencies = [ "blake3", "borsh 0.10.3", "borsh 0.9.3", + "borsh 1.2.0", "bs58", "bv", "bytemuck", @@ -6197,7 +6238,7 @@ dependencies = [ "base64 0.21.5", "bincode", "bitflags 2.4.1", - "borsh 0.10.3", + "borsh 1.2.0", "bs58", "bytemuck", "byteorder 1.5.0", @@ -6993,6 +7034,18 @@ dependencies = [ "unicode-ident", ] +[[package]] +name = "syn_derive" +version = "0.1.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1329189c02ff984e9736652b1631330da25eaa6bc639089ed4915d25446cbe7b" +dependencies = [ + "proc-macro-error", + "proc-macro2", + "quote", + "syn 2.0.39", +] + [[package]] name = "sync_wrapper" version = "0.1.1" @@ -7423,6 +7476,23 @@ dependencies = [ "serde", ] +[[package]] +name = "toml_datetime" +version = "0.6.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7cda73e2f1397b1262d6dfdcef8aafae14d1de7748d66822d3bfeeb6d03e5e4b" + +[[package]] +name = "toml_edit" +version = "0.20.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "396e4d48bbb2b7554c944bde63101b5ae446cff6ec4a24227428f15eb72ef338" +dependencies = [ + "indexmap 2.1.0", + "toml_datetime", + "winnow", +] + [[package]] name = "tonic" version = "0.9.2" @@ -8058,6 +8128,15 @@ version = "0.48.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "1a515f5799fe4961cb532f983ce2b23082366b898e52ffbce459c86f67c8378a" +[[package]] +name = "winnow" +version = "0.5.25" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b7e87b8dfbe3baffbe687eef2e164e32286eff31a5ee16463ce03d991643ec94" +dependencies = [ + "memchr", +] + [[package]] name = "winreg" version = "0.50.0" From acfd2c3cd4c2aa27c08393ee80c9802a0f95f4d3 Mon Sep 17 00:00:00 2001 From: Jon C Date: Thu, 7 Dec 2023 01:35:55 +0100 Subject: [PATCH 04/10] Add borsh 0.10 implementations for stake types --- sdk/program/src/stake/stake_flags.rs | 43 +++ sdk/program/src/stake/state.rs | 461 ++++++++++++++++++++++----- 2 files changed, 426 insertions(+), 78 deletions(-) diff --git a/sdk/program/src/stake/stake_flags.rs b/sdk/program/src/stake/stake_flags.rs index 3bb788d0f79013..aa044ff928acb7 100644 --- a/sdk/program/src/stake/stake_flags.rs +++ b/sdk/program/src/stake/stake_flags.rs @@ -21,6 +21,49 @@ use borsh::{BorshDeserialize, BorshSchema, BorshSerialize}; pub struct StakeFlags { bits: u8, } +impl borsh0_10::de::BorshDeserialize for StakeFlags { + fn deserialize_reader( + reader: &mut R, + ) -> ::core::result::Result { + Ok(Self { + bits: borsh0_10::BorshDeserialize::deserialize_reader(reader)?, + }) + } +} +impl borsh0_10::BorshSchema for StakeFlags { + fn declaration() -> borsh0_10::schema::Declaration { + "StakeFlags".to_string() + } + fn add_definitions_recursively( + definitions: &mut borsh0_10::maybestd::collections::HashMap< + borsh0_10::schema::Declaration, + borsh0_10::schema::Definition, + >, + ) { + let fields = borsh0_10::schema::Fields::NamedFields(<[_]>::into_vec( + borsh0_10::maybestd::boxed::Box::new([( + "bits".to_string(), + ::declaration(), + )]), + )); + let definition = borsh0_10::schema::Definition::Struct { fields }; + Self::add_definition( + ::declaration(), + definition, + definitions, + ); + ::add_definitions_recursively(definitions); + } +} +impl borsh0_10::ser::BorshSerialize for StakeFlags { + fn serialize( + &self, + writer: &mut W, + ) -> ::core::result::Result<(), borsh0_10::maybestd::io::Error> { + borsh0_10::BorshSerialize::serialize(&self.bits, writer)?; + Ok(()) + } +} /// Currently, only bit 1 is used. The other 7 bits are reserved for future usage. impl StakeFlags { diff --git a/sdk/program/src/stake/state.rs b/sdk/program/src/stake/state.rs index 91786f3b93676d..11652446bbd2ee 100644 --- a/sdk/program/src/stake/state.rs +++ b/sdk/program/src/stake/state.rs @@ -34,6 +34,49 @@ pub fn warmup_cooldown_rate(current_epoch: Epoch, new_rate_activation_epoch: Opt } } +macro_rules! impl_borsh_stake_state { + ($borsh:ident) => { + impl $borsh::BorshDeserialize for StakeState { + fn deserialize_reader(reader: &mut R) -> io::Result { + let enum_value: u32 = $borsh::BorshDeserialize::deserialize_reader(reader)?; + match enum_value { + 0 => Ok(StakeState::Uninitialized), + 1 => { + let meta: Meta = $borsh::BorshDeserialize::deserialize_reader(reader)?; + Ok(StakeState::Initialized(meta)) + } + 2 => { + let meta: Meta = $borsh::BorshDeserialize::deserialize_reader(reader)?; + let stake: Stake = $borsh::BorshDeserialize::deserialize_reader(reader)?; + Ok(StakeState::Stake(meta, stake)) + } + 3 => Ok(StakeState::RewardsPool), + _ => Err(io::Error::new( + io::ErrorKind::InvalidData, + "Invalid enum value", + )), + } + } + } + impl $borsh::BorshSerialize for StakeState { + fn serialize(&self, writer: &mut W) -> io::Result<()> { + match self { + StakeState::Uninitialized => writer.write_all(&0u32.to_le_bytes()), + StakeState::Initialized(meta) => { + writer.write_all(&1u32.to_le_bytes())?; + $borsh::BorshSerialize::serialize(&meta, writer) + } + StakeState::Stake(meta, stake) => { + writer.write_all(&2u32.to_le_bytes())?; + $borsh::BorshSerialize::serialize(&meta, writer)?; + $borsh::BorshSerialize::serialize(&stake, writer) + } + StakeState::RewardsPool => writer.write_all(&3u32.to_le_bytes()), + } + } + } + }; +} #[derive(Debug, Default, Serialize, Deserialize, PartialEq, Clone, Copy, AbiExample)] #[allow(clippy::large_enum_variant)] #[deprecated( @@ -47,45 +90,8 @@ pub enum StakeState { Stake(Meta, Stake), RewardsPool, } -impl BorshDeserialize for StakeState { - fn deserialize_reader(reader: &mut R) -> io::Result { - let enum_value = u32::deserialize_reader(reader)?; - match enum_value { - 0 => Ok(StakeState::Uninitialized), - 1 => { - let meta = Meta::deserialize_reader(reader)?; - Ok(StakeState::Initialized(meta)) - } - 2 => { - let meta: Meta = BorshDeserialize::deserialize_reader(reader)?; - let stake: Stake = BorshDeserialize::deserialize_reader(reader)?; - Ok(StakeState::Stake(meta, stake)) - } - 3 => Ok(StakeState::RewardsPool), - _ => Err(io::Error::new( - io::ErrorKind::InvalidData, - "Invalid enum value", - )), - } - } -} -impl BorshSerialize for StakeState { - fn serialize(&self, writer: &mut W) -> io::Result<()> { - match self { - StakeState::Uninitialized => writer.write_all(&0u32.to_le_bytes()), - StakeState::Initialized(meta) => { - writer.write_all(&1u32.to_le_bytes())?; - meta.serialize(writer) - } - StakeState::Stake(meta, stake) => { - writer.write_all(&2u32.to_le_bytes())?; - meta.serialize(writer)?; - stake.serialize(writer) - } - StakeState::RewardsPool => writer.write_all(&3u32.to_le_bytes()), - } - } -} +impl_borsh_stake_state!(borsh); +impl_borsh_stake_state!(borsh0_10); impl StakeState { /// The fixed number of bytes used to serialize each stake account pub const fn size_of() -> usize { @@ -136,49 +142,54 @@ pub enum StakeStateV2 { Stake(Meta, Stake, StakeFlags), RewardsPool, } - -impl BorshDeserialize for StakeStateV2 { - fn deserialize_reader(reader: &mut R) -> io::Result { - let enum_value = u32::deserialize_reader(reader)?; - match enum_value { - 0 => Ok(StakeStateV2::Uninitialized), - 1 => { - let meta = Meta::deserialize_reader(reader)?; - Ok(StakeStateV2::Initialized(meta)) - } - 2 => { - let meta: Meta = BorshDeserialize::deserialize_reader(reader)?; - let stake: Stake = BorshDeserialize::deserialize_reader(reader)?; - let stake_flags: StakeFlags = BorshDeserialize::deserialize_reader(reader)?; - Ok(StakeStateV2::Stake(meta, stake, stake_flags)) +macro_rules! impl_borsh_stake_state_v2 { + ($borsh:ident) => { + impl $borsh::BorshDeserialize for StakeStateV2 { + fn deserialize_reader(reader: &mut R) -> io::Result { + let enum_value: u32 = $borsh::BorshDeserialize::deserialize_reader(reader)?; + match enum_value { + 0 => Ok(StakeStateV2::Uninitialized), + 1 => { + let meta: Meta = $borsh::BorshDeserialize::deserialize_reader(reader)?; + Ok(StakeStateV2::Initialized(meta)) + } + 2 => { + let meta: Meta = $borsh::BorshDeserialize::deserialize_reader(reader)?; + let stake: Stake = $borsh::BorshDeserialize::deserialize_reader(reader)?; + let stake_flags: StakeFlags = + $borsh::BorshDeserialize::deserialize_reader(reader)?; + Ok(StakeStateV2::Stake(meta, stake, stake_flags)) + } + 3 => Ok(StakeStateV2::RewardsPool), + _ => Err(io::Error::new( + io::ErrorKind::InvalidData, + "Invalid enum value", + )), + } } - 3 => Ok(StakeStateV2::RewardsPool), - _ => Err(io::Error::new( - io::ErrorKind::InvalidData, - "Invalid enum value", - )), } - } -} - -impl BorshSerialize for StakeStateV2 { - fn serialize(&self, writer: &mut W) -> io::Result<()> { - match self { - StakeStateV2::Uninitialized => writer.write_all(&0u32.to_le_bytes()), - StakeStateV2::Initialized(meta) => { - writer.write_all(&1u32.to_le_bytes())?; - meta.serialize(writer) - } - StakeStateV2::Stake(meta, stake, stake_flags) => { - writer.write_all(&2u32.to_le_bytes())?; - meta.serialize(writer)?; - stake.serialize(writer)?; - stake_flags.serialize(writer) + impl $borsh::BorshSerialize for StakeStateV2 { + fn serialize(&self, writer: &mut W) -> io::Result<()> { + match self { + StakeStateV2::Uninitialized => writer.write_all(&0u32.to_le_bytes()), + StakeStateV2::Initialized(meta) => { + writer.write_all(&1u32.to_le_bytes())?; + $borsh::BorshSerialize::serialize(&meta, writer) + } + StakeStateV2::Stake(meta, stake, stake_flags) => { + writer.write_all(&2u32.to_le_bytes())?; + $borsh::BorshSerialize::serialize(&meta, writer)?; + $borsh::BorshSerialize::serialize(&stake, writer)?; + $borsh::BorshSerialize::serialize(&stake_flags, writer) + } + StakeStateV2::RewardsPool => writer.write_all(&3u32.to_le_bytes()), + } } - StakeStateV2::RewardsPool => writer.write_all(&3u32.to_le_bytes()), } - } + }; } +impl_borsh_stake_state_v2!(borsh); +impl_borsh_stake_state_v2!(borsh0_10); impl StakeStateV2 { /// The fixed number of bytes used to serialize each stake account @@ -253,7 +264,6 @@ pub struct Lockup { /// lockup constraints pub custodian: Pubkey, } - impl Lockup { pub fn is_in_force(&self, clock: &Clock, custodian: Option<&Pubkey>) -> bool { if custodian == Some(&self.custodian) { @@ -262,6 +272,65 @@ impl Lockup { self.unix_timestamp > clock.unix_timestamp || self.epoch > clock.epoch } } +impl borsh0_10::de::BorshDeserialize for Lockup { + fn deserialize_reader( + reader: &mut R, + ) -> ::core::result::Result { + Ok(Self { + unix_timestamp: borsh0_10::BorshDeserialize::deserialize_reader(reader)?, + epoch: borsh0_10::BorshDeserialize::deserialize_reader(reader)?, + custodian: borsh0_10::BorshDeserialize::deserialize_reader(reader)?, + }) + } +} +impl borsh0_10::BorshSchema for Lockup { + fn declaration() -> borsh0_10::schema::Declaration { + "Lockup".to_string() + } + fn add_definitions_recursively( + definitions: &mut borsh0_10::maybestd::collections::HashMap< + borsh0_10::schema::Declaration, + borsh0_10::schema::Definition, + >, + ) { + let fields = borsh0_10::schema::Fields::NamedFields(<[_]>::into_vec( + borsh0_10::maybestd::boxed::Box::new([ + ( + "unix_timestamp".to_string(), + ::declaration(), + ), + ( + "epoch".to_string(), + ::declaration(), + ), + ( + "custodian".to_string(), + ::declaration(), + ), + ]), + )); + let definition = borsh0_10::schema::Definition::Struct { fields }; + Self::add_definition( + ::declaration(), + definition, + definitions, + ); + ::add_definitions_recursively(definitions); + ::add_definitions_recursively(definitions); + ::add_definitions_recursively(definitions); + } +} +impl borsh0_10::ser::BorshSerialize for Lockup { + fn serialize( + &self, + writer: &mut W, + ) -> ::core::result::Result<(), borsh0_10::maybestd::io::Error> { + borsh0_10::BorshSerialize::serialize(&self.unix_timestamp, writer)?; + borsh0_10::BorshSerialize::serialize(&self.epoch, writer)?; + borsh0_10::BorshSerialize::serialize(&self.custodian, writer)?; + Ok(()) + } +} #[derive( Default, @@ -343,6 +412,58 @@ impl Authorized { Ok(()) } } +impl borsh0_10::de::BorshDeserialize for Authorized { + fn deserialize_reader( + reader: &mut R, + ) -> ::core::result::Result { + Ok(Self { + staker: borsh0_10::BorshDeserialize::deserialize_reader(reader)?, + withdrawer: borsh0_10::BorshDeserialize::deserialize_reader(reader)?, + }) + } +} +impl borsh0_10::BorshSchema for Authorized { + fn declaration() -> borsh0_10::schema::Declaration { + "Authorized".to_string() + } + fn add_definitions_recursively( + definitions: &mut borsh0_10::maybestd::collections::HashMap< + borsh0_10::schema::Declaration, + borsh0_10::schema::Definition, + >, + ) { + let fields = borsh0_10::schema::Fields::NamedFields(<[_]>::into_vec( + borsh0_10::maybestd::boxed::Box::new([ + ( + "staker".to_string(), + ::declaration(), + ), + ( + "withdrawer".to_string(), + ::declaration(), + ), + ]), + )); + let definition = borsh0_10::schema::Definition::Struct { fields }; + Self::add_definition( + ::declaration(), + definition, + definitions, + ); + ::add_definitions_recursively(definitions); + ::add_definitions_recursively(definitions); + } +} +impl borsh0_10::ser::BorshSerialize for Authorized { + fn serialize( + &self, + writer: &mut W, + ) -> ::core::result::Result<(), borsh0_10::maybestd::io::Error> { + borsh0_10::BorshSerialize::serialize(&self.staker, writer)?; + borsh0_10::BorshSerialize::serialize(&self.withdrawer, writer)?; + Ok(()) + } +} #[derive( Default, @@ -401,6 +522,65 @@ impl Meta { } } } +impl borsh0_10::de::BorshDeserialize for Meta { + fn deserialize_reader( + reader: &mut R, + ) -> ::core::result::Result { + Ok(Self { + rent_exempt_reserve: borsh0_10::BorshDeserialize::deserialize_reader(reader)?, + authorized: borsh0_10::BorshDeserialize::deserialize_reader(reader)?, + lockup: borsh0_10::BorshDeserialize::deserialize_reader(reader)?, + }) + } +} +impl borsh0_10::BorshSchema for Meta { + fn declaration() -> borsh0_10::schema::Declaration { + "Meta".to_string() + } + fn add_definitions_recursively( + definitions: &mut borsh0_10::maybestd::collections::HashMap< + borsh0_10::schema::Declaration, + borsh0_10::schema::Definition, + >, + ) { + let fields = borsh0_10::schema::Fields::NamedFields(<[_]>::into_vec( + borsh0_10::maybestd::boxed::Box::new([ + ( + "rent_exempt_reserve".to_string(), + ::declaration(), + ), + ( + "authorized".to_string(), + ::declaration(), + ), + ( + "lockup".to_string(), + ::declaration(), + ), + ]), + )); + let definition = borsh0_10::schema::Definition::Struct { fields }; + Self::add_definition( + ::declaration(), + definition, + definitions, + ); + ::add_definitions_recursively(definitions); + ::add_definitions_recursively(definitions); + ::add_definitions_recursively(definitions); + } +} +impl borsh0_10::ser::BorshSerialize for Meta { + fn serialize( + &self, + writer: &mut W, + ) -> ::core::result::Result<(), borsh0_10::maybestd::io::Error> { + borsh0_10::BorshSerialize::serialize(&self.rent_exempt_reserve, writer)?; + borsh0_10::BorshSerialize::serialize(&self.authorized, writer)?; + borsh0_10::BorshSerialize::serialize(&self.lockup, writer)?; + Ok(()) + } +} #[derive( Debug, @@ -648,6 +828,79 @@ impl Delegation { } } } +impl borsh0_10::de::BorshDeserialize for Delegation { + fn deserialize_reader( + reader: &mut R, + ) -> ::core::result::Result { + Ok(Self { + voter_pubkey: borsh0_10::BorshDeserialize::deserialize_reader(reader)?, + stake: borsh0_10::BorshDeserialize::deserialize_reader(reader)?, + activation_epoch: borsh0_10::BorshDeserialize::deserialize_reader(reader)?, + deactivation_epoch: borsh0_10::BorshDeserialize::deserialize_reader(reader)?, + warmup_cooldown_rate: borsh0_10::BorshDeserialize::deserialize_reader(reader)?, + }) + } +} +impl borsh0_10::BorshSchema for Delegation { + fn declaration() -> borsh0_10::schema::Declaration { + "Delegation".to_string() + } + fn add_definitions_recursively( + definitions: &mut borsh0_10::maybestd::collections::HashMap< + borsh0_10::schema::Declaration, + borsh0_10::schema::Definition, + >, + ) { + let fields = borsh0_10::schema::Fields::NamedFields(<[_]>::into_vec( + borsh0_10::maybestd::boxed::Box::new([ + ( + "voter_pubkey".to_string(), + ::declaration(), + ), + ( + "stake".to_string(), + ::declaration(), + ), + ( + "activation_epoch".to_string(), + ::declaration(), + ), + ( + "deactivation_epoch".to_string(), + ::declaration(), + ), + ( + "warmup_cooldown_rate".to_string(), + ::declaration(), + ), + ]), + )); + let definition = borsh0_10::schema::Definition::Struct { fields }; + Self::add_definition( + ::declaration(), + definition, + definitions, + ); + ::add_definitions_recursively(definitions); + ::add_definitions_recursively(definitions); + ::add_definitions_recursively(definitions); + ::add_definitions_recursively(definitions); + ::add_definitions_recursively(definitions); + } +} +impl borsh0_10::ser::BorshSerialize for Delegation { + fn serialize( + &self, + writer: &mut W, + ) -> ::core::result::Result<(), borsh0_10::maybestd::io::Error> { + borsh0_10::BorshSerialize::serialize(&self.voter_pubkey, writer)?; + borsh0_10::BorshSerialize::serialize(&self.stake, writer)?; + borsh0_10::BorshSerialize::serialize(&self.activation_epoch, writer)?; + borsh0_10::BorshSerialize::serialize(&self.deactivation_epoch, writer)?; + borsh0_10::BorshSerialize::serialize(&self.warmup_cooldown_rate, writer)?; + Ok(()) + } +} #[derive( Debug, @@ -708,6 +961,58 @@ impl Stake { } } } +impl borsh0_10::de::BorshDeserialize for Stake { + fn deserialize_reader( + reader: &mut R, + ) -> ::core::result::Result { + Ok(Self { + delegation: borsh0_10::BorshDeserialize::deserialize_reader(reader)?, + credits_observed: borsh0_10::BorshDeserialize::deserialize_reader(reader)?, + }) + } +} +impl borsh0_10::BorshSchema for Stake { + fn declaration() -> borsh0_10::schema::Declaration { + "Stake".to_string() + } + fn add_definitions_recursively( + definitions: &mut borsh0_10::maybestd::collections::HashMap< + borsh0_10::schema::Declaration, + borsh0_10::schema::Definition, + >, + ) { + let fields = borsh0_10::schema::Fields::NamedFields(<[_]>::into_vec( + borsh0_10::maybestd::boxed::Box::new([ + ( + "delegation".to_string(), + ::declaration(), + ), + ( + "credits_observed".to_string(), + ::declaration(), + ), + ]), + )); + let definition = borsh0_10::schema::Definition::Struct { fields }; + Self::add_definition( + ::declaration(), + definition, + definitions, + ); + ::add_definitions_recursively(definitions); + ::add_definitions_recursively(definitions); + } +} +impl borsh0_10::ser::BorshSerialize for Stake { + fn serialize( + &self, + writer: &mut W, + ) -> ::core::result::Result<(), borsh0_10::maybestd::io::Error> { + borsh0_10::BorshSerialize::serialize(&self.delegation, writer)?; + borsh0_10::BorshSerialize::serialize(&self.credits_observed, writer)?; + Ok(()) + } +} #[cfg(test)] mod test { From 87eb60146232465ed8328c876a1782efa36ee37f Mon Sep 17 00:00:00 2001 From: Jon C Date: Thu, 7 Dec 2023 15:32:48 +0100 Subject: [PATCH 05/10] Fix weirdness on whitespace --- sdk/program/src/borsh.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/sdk/program/src/borsh.rs b/sdk/program/src/borsh.rs index 8d36b536da8db6..95cb5ddc1297e8 100644 --- a/sdk/program/src/borsh.rs +++ b/sdk/program/src/borsh.rs @@ -69,7 +69,7 @@ macro_rules! impl_get_packed_len_v0 { get_declaration_packed_len(declaration, definitions) } - /// Get packed length for the given BorshSchema Declaration + /// Get packed length for the given BorshSchema Declaration fn get_declaration_packed_len( declaration: &str, definitions: &std::collections::HashMap<$borsh::schema::Declaration, $borsh::schema::Definition>, From dded124d1e3af0ec5ebb48225d50b1893dfce56a Mon Sep 17 00:00:00 2001 From: Jon C Date: Thu, 7 Dec 2023 15:32:55 +0100 Subject: [PATCH 06/10] Update to borsh 1.2.1 --- Cargo.lock | 16 ++++++++-------- Cargo.toml | 2 +- programs/sbf/Cargo.lock | 18 +++++++++--------- 3 files changed, 18 insertions(+), 18 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 7cd0e832ea03cf..496820cb230467 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -728,11 +728,11 @@ dependencies = [ [[package]] name = "borsh" -version = "1.2.0" +version = "1.2.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bf617fabf5cdbdc92f774bfe5062d870f228b80056d41180797abf48bed4056e" +checksum = "9897ef0f1bd2362169de6d7e436ea2237dc1085d7d1e4db75f4be34d86f309d1" dependencies = [ - "borsh-derive 1.2.0", + "borsh-derive 1.2.1", "cfg_aliases", ] @@ -764,9 +764,9 @@ dependencies = [ [[package]] name = "borsh-derive" -version = "1.2.0" +version = "1.2.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f404657a7ea7b5249e36808dff544bc88a28f26e0ac40009f674b7a009d14be3" +checksum = "478b41ff04256c5c8330f3dfdaaae2a5cc976a8e75088bafa4625b0d0208de8c" dependencies = [ "once_cell", "proc-macro-crate 2.0.0", @@ -5434,7 +5434,7 @@ dependencies = [ name = "solana-banks-client" version = "1.18.0" dependencies = [ - "borsh 1.2.0", + "borsh 1.2.1", "futures 0.3.29", "solana-banks-interface", "solana-banks-server", @@ -6691,7 +6691,7 @@ dependencies = [ "blake3", "borsh 0.10.3", "borsh 0.9.3", - "borsh 1.2.0", + "borsh 1.2.1", "bs58", "bv", "bytemuck", @@ -7129,7 +7129,7 @@ dependencies = [ "base64 0.21.5", "bincode", "bitflags 2.4.1", - "borsh 1.2.0", + "borsh 1.2.1", "bs58", "bytemuck", "byteorder", diff --git a/Cargo.toml b/Cargo.toml index 6d28f02bc2a7fc..631173aa56de59 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -155,7 +155,7 @@ bincode = "1.3.3" bitflags = { version = "2.3.3", features = ["serde"] } blake3 = "1.5.0" block-buffer = "0.10.4" -borsh = { version = "1.2.0", features = ["derive", "unstable__schema"] } +borsh = { version = "1.2.1", features = ["derive", "unstable__schema"] } bs58 = "0.4.0" bv = "0.11.1" byte-unit = "4.0.19" diff --git a/programs/sbf/Cargo.lock b/programs/sbf/Cargo.lock index cf31b2084d28c9..99eb6d6cb54742 100644 --- a/programs/sbf/Cargo.lock +++ b/programs/sbf/Cargo.lock @@ -688,11 +688,11 @@ dependencies = [ [[package]] name = "borsh" -version = "1.2.0" +version = "1.2.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bf617fabf5cdbdc92f774bfe5062d870f228b80056d41180797abf48bed4056e" +checksum = "9897ef0f1bd2362169de6d7e436ea2237dc1085d7d1e4db75f4be34d86f309d1" dependencies = [ - "borsh-derive 1.2.0", + "borsh-derive 1.2.1", "cfg_aliases", ] @@ -724,9 +724,9 @@ dependencies = [ [[package]] name = "borsh-derive" -version = "1.2.0" +version = "1.2.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f404657a7ea7b5249e36808dff544bc88a28f26e0ac40009f674b7a009d14be3" +checksum = "478b41ff04256c5c8330f3dfdaaae2a5cc976a8e75088bafa4625b0d0208de8c" dependencies = [ "once_cell", "proc-macro-crate 2.0.1", @@ -3149,7 +3149,7 @@ version = "0.7.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "6c11e44798ad209ccdd91fc192f0526a369a01234f7373e1b141c96d7cee4f0e" dependencies = [ - "proc-macro-crate 1.1.3", + "proc-macro-crate 2.0.1", "proc-macro2", "quote", "syn 2.0.39", @@ -4758,7 +4758,7 @@ dependencies = [ name = "solana-banks-client" version = "1.18.0" dependencies = [ - "borsh 1.2.0", + "borsh 1.2.1", "futures 0.3.29", "solana-banks-interface", "solana-program", @@ -5455,7 +5455,7 @@ dependencies = [ "blake3", "borsh 0.10.3", "borsh 0.9.3", - "borsh 1.2.0", + "borsh 1.2.1", "bs58", "bv", "bytemuck", @@ -6238,7 +6238,7 @@ dependencies = [ "base64 0.21.5", "bincode", "bitflags 2.4.1", - "borsh 1.2.0", + "borsh 1.2.1", "bs58", "bytemuck", "byteorder 1.5.0", From 4ffe653cdea832ac098e21f763102bc9eb428785 Mon Sep 17 00:00:00 2001 From: Jon C Date: Thu, 7 Dec 2023 15:36:13 +0100 Subject: [PATCH 07/10] Update changelog --- CHANGELOG.md | 3 +++ 1 file changed, 3 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 5e768ab8388925..2e8517a4c9e5b5 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -18,7 +18,10 @@ Release channels have their own copy of this changelog: * The default for `--use-snapshot-archives-at-startup` is now `when-newest` (#33883) * The default for `solana-ledger-tool`, however, remains `always` (#34228) * Added `central-scheduler` option for `--block-production-method` (#33890) + * Updated to Borsh v1 * Upgrade Notes + * `solana-program` and `solana-sdk` default to support for Borsh v1, with +limited backward compatibility for v0.10 and v0.9. Please upgrade to Borsh v1. ## [1.17.0] * Changes From 19429a414bd649836302851e6cd8cab2947516c9 Mon Sep 17 00:00:00 2001 From: Jon C Date: Thu, 7 Dec 2023 23:48:26 +0100 Subject: [PATCH 08/10] compute-budget: Move `pack` under dev-context-only-utils --- cost-model/Cargo.toml | 1 + sdk/src/compute_budget.rs | 2 +- 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/cost-model/Cargo.toml b/cost-model/Cargo.toml index 6142f9be876215..4a8b159bbf4cb6 100644 --- a/cost-model/Cargo.toml +++ b/cost-model/Cargo.toml @@ -32,6 +32,7 @@ name = "solana_cost_model" [dev-dependencies] solana-logger = { workspace = true } +solana-sdk = { workspace = true, features = ["dev-context-only-utils"] } static_assertions = { workspace = true } test-case = { workspace = true } diff --git a/sdk/src/compute_budget.rs b/sdk/src/compute_budget.rs index 357ae787a029f8..cf9ad7d436b929 100644 --- a/sdk/src/compute_budget.rs +++ b/sdk/src/compute_budget.rs @@ -63,7 +63,7 @@ impl ComputeBudgetInstruction { /// Serialize Instruction using borsh, this is only used in runtime::cost_model::tests but compilation /// can't be restricted as it's used across packages - // #[cfg(test)] + #[cfg(feature = "dev-context-only-utils")] pub fn pack(self) -> Result, borsh::io::Error> { borsh::to_vec(&self) } From 090efde767540d578225cff759db37863ac76bec Mon Sep 17 00:00:00 2001 From: Jon C Date: Thu, 7 Dec 2023 23:51:15 +0100 Subject: [PATCH 09/10] Revert test to use HashMap --- sdk/program/src/borsh.rs | 17 ++++++++--------- 1 file changed, 8 insertions(+), 9 deletions(-) diff --git a/sdk/program/src/borsh.rs b/sdk/program/src/borsh.rs index 95cb5ddc1297e8..0041aa80602946 100644 --- a/sdk/program/src/borsh.rs +++ b/sdk/program/src/borsh.rs @@ -243,8 +243,7 @@ macro_rules! impl_tests { extern crate alloc; use { super::*, - alloc::{collections::BTreeMap, vec::Vec}, - std::mem::size_of, + std::{collections::HashMap, mem::size_of}, $borsh::{BorshDeserialize, BorshSerialize}, $borsh_io::ErrorKind, }; @@ -332,13 +331,13 @@ macro_rules! impl_tests { #[test] fn instance_packed_len_with_varying_sizes_in_hashmap() { - let mut data = BTreeMap::new(); - let key1 = Vec::from("the first string, it's actually really really long"); - let value1 = Vec::from(""); - let key2 = Vec::from("second string, shorter"); - let value2 = Vec::from("a real value"); - let key3 = Vec::from("third"); - let value3 = Vec::from("an even longer value"); + let mut data = HashMap::new(); + let key1 = "the first string, it's actually really really long".to_string(); + let value1 = "".to_string(); + let key2 = "second string, shorter".to_string(); + let value2 = "a real value".to_string(); + let key3 = "third".to_string(); + let value3 = "an even longer value".to_string(); data.insert(key1.clone(), value1.clone()); data.insert(key2.clone(), value2.clone()); data.insert(key3.clone(), value3.clone()); From 52e2325e449f14c1405e4001bd412e037c733eba Mon Sep 17 00:00:00 2001 From: Jon C Date: Thu, 7 Dec 2023 23:53:04 +0100 Subject: [PATCH 10/10] transaction-status: Add comment about borsh version --- transaction-status/Cargo.toml | 2 ++ 1 file changed, 2 insertions(+) diff --git a/transaction-status/Cargo.toml b/transaction-status/Cargo.toml index b9d6c8c543c022..a4837acb3e107d 100644 --- a/transaction-status/Cargo.toml +++ b/transaction-status/Cargo.toml @@ -13,6 +13,8 @@ edition = { workspace = true } Inflector = { workspace = true } base64 = { workspace = true } bincode = { workspace = true } +# Update this borsh dependency to the workspace version once +# spl-associated-token-account is upgraded and used in the monorepo. borsh = { version = "0.10.3" } bs58 = { workspace = true } lazy_static = { workspace = true }