1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94
// This file is part of Gear.
// Copyright (C) 2021-2024 Gear Technologies Inc.
// SPDX-License-Identifier: GPL-3.0-or-later WITH Classpath-exception-2.0
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
//! Utility functions.
pub use scale_info::MetaType;
use scale_info::{
scale::{Encode, Output},
PortableRegistry, Registry,
};
use crate::prelude::{
mem::{transmute, MaybeUninit},
Box, String, Vec,
};
/// An auxiliary function that reduces gas consumption during payload encoding.
pub(crate) fn with_optimized_encode<T, E: Encode>(payload: E, f: impl FnOnce(&[u8]) -> T) -> T {
struct ExternalBufferOutput<'a> {
buffer: &'a mut [MaybeUninit<u8>],
offset: usize,
}
impl<'a> Output for ExternalBufferOutput<'a> {
fn write(&mut self, bytes: &[u8]) {
// SAFETY: same as
// `MaybeUninit::write_slice(&mut self.buffer[self.offset..end_offset], bytes)`.
// This code transmutes `bytes: &[T]` to `bytes: &[MaybeUninit<T>]`. These types
// can be safely transmuted since they have the same layout. Then `bytes:
// &[MaybeUninit<T>]` is written to uninitialized memory via `copy_from_slice`.
let end_offset = self.offset + bytes.len();
let this = unsafe { self.buffer.get_unchecked_mut(self.offset..end_offset) };
this.copy_from_slice(unsafe {
transmute::<&[u8], &[core::mem::MaybeUninit<u8>]>(bytes)
});
self.offset = end_offset;
}
}
gcore::stack_buffer::with_byte_buffer(payload.encoded_size(), |buffer| {
let mut output = ExternalBufferOutput { buffer, offset: 0 };
payload.encode_to(&mut output);
let ExternalBufferOutput { buffer, offset } = output;
// SAFETY: same as `MaybeUninit::slice_assume_init_ref(&buffer[..offset])`.
// `ExternalBufferOutput` writes data to uninitialized memory. So we can take
// slice `&buffer[..offset]` and say that it was initialized earlier
// because the buffer from `0` to `offset` was initialized.
let payload = unsafe { &*(&buffer[..offset] as *const _ as *const [u8]) };
f(payload)
})
}
/// Generate a registry from given meta types and encode it to hex.
pub fn to_hex_registry(meta_types: Vec<MetaType>) -> String {
let mut registry = Registry::new();
registry.register_types(meta_types);
let registry: PortableRegistry = registry.into();
hex::encode(registry.encode())
}
/// Convert a given reference to a raw pointer.
pub fn to_wasm_ptr<T: AsRef<[u8]>>(bytes: T) -> *mut [i32; 2] {
Box::into_raw(Box::new([
bytes.as_ref().as_ptr() as _,
bytes.as_ref().len() as _,
]))
}
/// Convert a given vector to a raw pointer and prevent its deallocating.
///
/// It operates similarly to [`to_wasm_ptr`] except that it consumes the input
/// and make it leak by calling [`core::mem::forget`].
pub fn to_leak_ptr(bytes: impl Into<Vec<u8>>) -> *mut [i32; 2] {
let bytes = bytes.into();
let ptr = Box::into_raw(Box::new([bytes.as_ptr() as _, bytes.len() as _]));
core::mem::forget(bytes);
ptr
}