ab_system_contract_simple_wallet_base/payload/
builder.rs1#[cfg(test)]
4mod tests;
5
6extern crate alloc;
7
8use crate::payload::{
9 FfiDataSizeCapacityRo, TransactionInput, TransactionMethodContext, TransactionSlot,
10};
11use ab_contracts_common::MAX_TOTAL_METHOD_ARGS;
12use ab_contracts_common::metadata::decode::{
13 ArgumentKind, MetadataDecodingError, MethodMetadataDecoder, MethodMetadataItem,
14 MethodsContainerKind,
15};
16use ab_contracts_common::method::{ExternalArgs, MethodFingerprint};
17use ab_core_primitives::address::Address;
18use ab_io_type::MAX_ALIGNMENT;
19use ab_io_type::metadata::IoTypeDetails;
20use ab_io_type::trivial_type::TrivialType;
21use alloc::vec::Vec;
22use core::ffi::c_void;
23use core::mem::MaybeUninit;
24use core::num::NonZeroU8;
25use core::ptr::NonNull;
26use core::{ptr, slice};
27
28const {
29 assert!(MAX_TOTAL_METHOD_ARGS as u32 == u8::BITS);
31}
32
33#[inline(always)]
39unsafe fn read_external_args<T>(external_args: &mut NonNull<c_void>) -> T {
40 unsafe {
42 let value = external_args.cast::<T>().read();
43 *external_args = external_args.byte_add(size_of::<T>());
44 value
45 }
46}
47
48#[derive(Debug, thiserror::Error)]
50pub enum TransactionPayloadBuilderError<'a> {
51 #[error("Metadata decoding error: {0}")]
53 MetadataDecodingError(MetadataDecodingError<'a>),
54 #[error("Too many arguments")]
56 TooManyArguments(u8),
57 #[error("Invalid alignment: {0}")]
59 InvalidAlignment(NonZeroU8),
60 #[error("Invalid output index: {0}")]
62 InvalidOutputIndex(u8),
63}
64
65#[derive(Debug, Clone)]
73pub struct TransactionPayloadBuilder {
74 payload: Vec<u8>,
75}
76
77impl Default for TransactionPayloadBuilder {
78 #[inline]
79 fn default() -> Self {
80 Self {
81 payload: Vec::with_capacity(1024),
82 }
83 }
84}
85
86impl TransactionPayloadBuilder {
87 #[cfg_attr(feature = "no-panic", no_panic::no_panic)]
95 pub fn with_method_call<Args>(
96 &mut self,
97 contract: &Address,
98 external_args: &Args,
99 method_context: TransactionMethodContext,
100 slot_output_index: &[Option<u8>],
101 input_output_index: &[Option<u8>],
102 ) -> Result<(), TransactionPayloadBuilderError<'static>>
103 where
104 Args: ExternalArgs,
105 {
106 let external_args = NonNull::from_ref(external_args).cast::<*const c_void>();
107
108 unsafe {
110 self.with_method_call_untyped(
111 contract,
112 external_args,
113 Args::METADATA,
114 &Args::FINGERPRINT,
115 method_context,
116 slot_output_index,
117 input_output_index,
118 )
119 }
120 }
121
122 #[expect(
129 clippy::too_many_arguments,
130 reason = "Only exceeds the limit due to being untyped, while above typed version is not"
131 )]
132 #[cfg_attr(feature = "no-panic", no_panic::no_panic)]
133 pub unsafe fn with_method_call_untyped<'a>(
134 &mut self,
135 contract: &Address,
136 external_args: NonNull<*const c_void>,
137 mut method_metadata: &'a [u8],
138 method_fingerprint: &MethodFingerprint,
139 method_context: TransactionMethodContext,
140 slot_output_index: &[Option<u8>],
141 input_output_index: &[Option<u8>],
142 ) -> Result<(), TransactionPayloadBuilderError<'a>> {
143 let external_args = &mut external_args.cast::<c_void>();
144
145 let (metadata_decoder, method_metadata_item) =
146 MethodMetadataDecoder::new(&mut method_metadata, MethodsContainerKind::Unknown)
147 .decode_next()
148 .map_err(TransactionPayloadBuilderError::MetadataDecodingError)?;
149 let mut metadata_decoder = metadata_decoder.without_auto_drain();
150
151 let MethodMetadataItem {
152 method_kind,
153 num_arguments,
154 ..
155 } = method_metadata_item;
156 let number_of_arguments = num_arguments.saturating_add(u8::from(method_kind.has_self()));
157
158 if number_of_arguments > MAX_TOTAL_METHOD_ARGS {
159 return Err(TransactionPayloadBuilderError::TooManyArguments(
160 number_of_arguments,
161 ));
162 }
163
164 self.extend_payload_with_alignment(contract.as_bytes(), align_of_val(contract));
165 self.extend_payload_with_alignment(
166 method_fingerprint.as_bytes(),
167 align_of_val(method_fingerprint),
168 );
169 self.push_payload_byte(method_context as u8);
170
171 let mut num_slot_arguments = 0u8;
172 let mut num_input_arguments = 0u8;
173 let mut num_output_arguments = 0u8;
174
175 let mut input_output_type_details =
176 [MaybeUninit::<IoTypeDetails>::uninit(); MAX_TOTAL_METHOD_ARGS as usize];
177 while let Some(item) = metadata_decoder
179 .decode_next()
180 .transpose()
181 .map_err(TransactionPayloadBuilderError::MetadataDecodingError)?
182 {
183 match item.argument_kind {
184 ArgumentKind::EnvRo
185 | ArgumentKind::EnvRw
186 | ArgumentKind::TmpRo
187 | ArgumentKind::TmpRw => {
188 }
190 ArgumentKind::SlotRo | ArgumentKind::SlotRw => {
191 num_slot_arguments += 1;
192 }
193 ArgumentKind::Input => {
194 input_output_type_details[usize::from(num_input_arguments)]
195 .write(item.type_details.unwrap_or(IoTypeDetails::bytes(0)));
196 num_input_arguments += 1;
197 }
198 ArgumentKind::Output | ArgumentKind::Return => {
199 input_output_type_details
200 [usize::from(num_input_arguments + num_output_arguments)]
201 .write(item.type_details.unwrap_or(IoTypeDetails::bytes(0)));
202 num_output_arguments += 1;
203 }
204 }
205 }
206 let (input_type_details, output_type_details) = unsafe {
208 let (input_type_details, output_type_details) =
209 input_output_type_details.split_at_unchecked(usize::from(num_input_arguments));
210 let output_type_details =
211 output_type_details.get_unchecked(..usize::from(num_output_arguments));
212
213 (
214 input_type_details.assume_init_ref(),
215 output_type_details.assume_init_ref(),
216 )
217 };
218
219 self.push_payload_byte(num_slot_arguments);
221 for slot_offset in 0..usize::from(num_slot_arguments) {
222 let slot_type = if let Some(&Some(output_index)) = slot_output_index.get(slot_offset) {
223 TransactionSlot::new_output_index(output_index).ok_or(
224 TransactionPayloadBuilderError::InvalidOutputIndex(output_index),
225 )?
226 } else {
227 TransactionSlot::new_address()
228 };
229 self.push_payload_byte(slot_type.into_u8());
230 }
231
232 self.push_payload_byte(num_input_arguments);
234 for (input_offset, type_details) in input_type_details.iter().enumerate() {
235 let input_type = if let Some(&Some(output_index)) = input_output_index.get(input_offset)
236 {
237 TransactionInput::new_output_index(output_index).ok_or(
238 TransactionPayloadBuilderError::InvalidOutputIndex(output_index),
239 )?
240 } else {
241 TransactionInput::new_value(type_details.alignment).ok_or(
242 TransactionPayloadBuilderError::InvalidAlignment(type_details.alignment),
243 )?
244 };
245 self.push_payload_byte(input_type.into_u8());
246 }
247
248 self.push_payload_byte(num_output_arguments);
250
251 for slot_offset in 0..usize::from(num_slot_arguments) {
252 let address = unsafe { read_external_args::<NonNull<Address>>(external_args).as_ref() };
254
255 if slot_output_index
256 .get(slot_offset)
257 .copied()
258 .flatten()
259 .is_none()
260 {
261 self.extend_payload_with_alignment(address.as_bytes(), align_of_val(address));
262 }
263 }
264
265 for (input_offset, type_details) in input_type_details.iter().enumerate() {
266 let (size, data) = unsafe {
268 let FfiDataSizeCapacityRo {
269 data_ptr,
270 size,
271 capacity: _,
272 } = read_external_args(external_args);
273
274 let data = slice::from_raw_parts(data_ptr.as_ptr().cast_const(), size as usize);
275
276 (size, data)
277 };
278
279 if input_output_index
280 .get(input_offset)
281 .copied()
282 .flatten()
283 .is_none()
284 {
285 self.extend_payload_with_alignment(&size.to_le_bytes(), align_of_val(&size));
286 self.extend_payload_with_alignment(data, type_details.alignment.get() as usize);
287 }
288 }
289
290 for type_details in output_type_details {
291 self.extend_payload_with_alignment(
292 &type_details.recommended_capacity.to_le_bytes(),
293 align_of_val(&type_details.recommended_capacity),
294 );
295 self.extend_payload_with_alignment(
296 &[type_details.alignment.ilog2() as u8],
297 align_of::<u8>(),
298 );
299 }
300
301 Ok(())
302 }
303
304 pub fn into_aligned_bytes(mut self) -> Vec<u128> {
333 self.ensure_alignment(usize::from(MAX_ALIGNMENT));
335
336 let output_len = self.payload.len() / size_of::<u128>();
337 let mut output = Vec::<u128>::with_capacity(output_len);
338
339 unsafe {
341 ptr::copy_nonoverlapping(
342 self.payload.as_ptr(),
343 output.as_mut_ptr().cast::<u8>(),
344 self.payload.len(),
345 );
346 output.set_len(output_len);
347 }
348
349 debug_assert_eq!(align_of_val(output.as_slice()), usize::from(MAX_ALIGNMENT));
350
351 output
352 }
353
354 #[cfg_attr(feature = "no-panic", no_panic::no_panic)]
355 fn extend_payload_with_alignment(&mut self, bytes: &[u8], alignment: usize) {
356 self.ensure_alignment(alignment);
357
358 self.payload.extend_from_slice(bytes);
359 }
360
361 fn ensure_alignment(&mut self, alignment: usize) {
363 debug_assert!(alignment <= usize::from(MAX_ALIGNMENT));
364
365 let unaligned_by = self.payload.len() & (alignment - 1);
368 if unaligned_by > 0 {
369 let padding_bytes = unsafe { alignment.unchecked_sub(unaligned_by) };
371 self.payload.resize(self.payload.len() + padding_bytes, 0);
372 }
373 }
374
375 #[cfg_attr(feature = "no-panic", no_panic::no_panic)]
376 fn push_payload_byte(&mut self, byte: u8) {
377 self.payload.push(byte);
378 }
379}