rhai/src/lib.rs

501 lines
17 KiB
Rust
Raw Normal View History

2017-10-28 22:33:00 +02:00
//! # Rhai - embedded scripting for Rust
2017-12-20 12:16:14 +01:00
//!
2021-01-30 00:19:54 +01:00
//! ![Rhai logo](https://rhai.rs/book/images/logo/rhai-banner-transparent-colour.svg)
2020-12-24 09:32:43 +01:00
//!
2020-10-10 07:41:55 +02:00
//! Rhai is a tiny, simple and fast embedded scripting language for Rust
2017-10-28 22:33:00 +02:00
//! that gives you a safe and easy way to add scripting to your applications.
2020-04-09 04:38:33 +02:00
//!
2020-12-24 09:32:43 +01:00
//! It provides a familiar syntax based on JavaScript+Rust and a simple Rust interface.
//!
//! # A Quick Example
//!
//! ## Contents of `my_script.rhai`
2017-12-20 12:16:14 +01:00
//!
2022-06-12 09:04:06 +02:00
//! ```rhai
2020-12-24 09:32:43 +01:00
//! /// Brute force factorial function
2017-10-28 22:33:00 +02:00
//! fn factorial(x) {
//! if x == 1 { return 1; }
2020-03-24 09:57:35 +01:00
//! x * factorial(x - 1)
2017-10-28 22:33:00 +02:00
//! }
//!
2020-04-09 04:38:33 +02:00
//! // Calling an external function 'compute'
//! compute(factorial(10))
2017-10-28 22:33:00 +02:00
//! ```
2017-12-20 12:16:14 +01:00
//!
2020-12-24 09:32:43 +01:00
//! ## The Rust part
2017-12-20 12:16:14 +01:00
//!
2021-03-22 04:18:09 +01:00
//! ```no_run
//! use rhai::{Engine, EvalAltResult};
2017-12-20 12:16:14 +01:00
//!
//! fn main() -> Result<(), Box<EvalAltResult>>
2020-03-09 14:09:53 +01:00
//! {
2020-04-09 04:38:33 +02:00
//! // Define external function
2020-03-09 14:09:53 +01:00
//! fn compute_something(x: i64) -> bool {
2020-03-24 09:57:35 +01:00
//! (x % 40) == 0
2020-03-09 14:09:53 +01:00
//! }
//!
2020-04-09 04:38:33 +02:00
//! // Create scripting engine
2020-03-09 14:09:53 +01:00
//! let mut engine = Engine::new();
//!
2020-04-09 04:38:33 +02:00
//! // Register external function as 'compute'
//! engine.register_fn("compute", compute_something);
2017-12-20 12:16:14 +01:00
//!
//! # #[cfg(not(feature = "no_std"))]
2022-01-12 01:12:28 +01:00
//! # #[cfg(not(target_family = "wasm"))]
2022-12-30 18:07:39 +01:00
//! #
2021-12-26 05:16:48 +01:00
//! // Evaluate the script, expecting a 'bool' result
2022-05-07 09:54:44 +02:00
//! let result: bool = engine.eval_file("my_script.rhai".into())?;
2021-12-25 16:49:14 +01:00
//!
//! assert_eq!(result, true);
2020-03-09 14:09:53 +01:00
//!
//! Ok(())
//! }
2017-10-28 22:33:00 +02:00
//! ```
//!
2022-12-22 02:33:12 +01:00
//! # Features
//!
#![cfg_attr(feature = "document-features", doc = document_features::document_features!(feature_label = "<span id=\"feature-{feature}\">**`{feature}`**</span>"))]
//!
//! # On-Line Documentation
2020-04-03 13:42:01 +02:00
//!
2021-01-30 00:19:54 +01:00
//! See [The Rhai Book](https://rhai.rs/book) for details on the Rhai scripting engine and language.
#![cfg_attr(feature = "no_std", no_std)]
2022-06-05 12:17:44 +02:00
#![deny(missing_docs)]
2022-11-23 11:51:11 +01:00
// #![warn(clippy::all)]
// #![warn(clippy::pedantic)]
// #![warn(clippy::nursery)]
2022-12-22 10:34:58 +01:00
#![warn(clippy::cargo)]
2022-11-23 11:51:11 +01:00
// #![warn(clippy::undocumented_unsafe_blocks)]
2022-07-27 10:04:24 +02:00
#![allow(clippy::unit_arg)]
2022-08-27 10:26:41 +02:00
#![allow(clippy::missing_errors_doc)]
2022-11-23 06:24:14 +01:00
#![allow(clippy::missing_panics_doc)]
2022-08-27 10:26:41 +02:00
#![allow(clippy::used_underscore_binding)]
#![allow(clippy::inline_always)]
#![allow(clippy::module_name_repetitions)]
#![allow(clippy::negative_feature_names)]
#![allow(clippy::module_inception)]
2022-11-25 02:46:13 +01:00
#![allow(clippy::box_collection)]
#![allow(clippy::too_many_arguments)]
#![allow(clippy::upper_case_acronyms)]
2022-12-22 10:34:58 +01:00
#![allow(clippy::match_same_arms)]
// The lints below can be turned off to reduce signal/noise ratio
2023-02-10 17:17:26 +01:00
#![allow(clippy::too_many_lines)]
#![allow(clippy::let_underscore_drop)]
2022-12-30 18:07:39 +01:00
#![allow(clippy::absurd_extreme_comparisons)]
#![allow(clippy::unnecessary_cast)]
2023-02-10 17:17:26 +01:00
#![allow(clippy::wildcard_imports)]
#![allow(clippy::no_effect_underscore_binding)]
#![allow(clippy::semicolon_if_nothing_returned)]
#![allow(clippy::type_complexity)]
2017-10-02 23:44:45 +02:00
#[cfg(feature = "no_std")]
2020-03-17 19:26:11 +01:00
extern crate alloc;
2021-04-17 09:15:54 +02:00
#[cfg(feature = "no_std")]
extern crate no_std_compat as std;
#[cfg(feature = "no_std")]
use std::prelude::v1::*;
// Internal modules
2022-12-04 07:06:54 +01:00
#[macro_use]
mod reify;
#[macro_use]
2022-12-11 14:50:47 +01:00
mod restore;
2022-12-04 07:06:54 +01:00
2021-11-13 15:36:23 +01:00
mod api;
2020-10-28 15:18:44 +01:00
mod ast;
pub mod config;
mod engine;
2022-01-07 04:43:47 +01:00
mod eval;
2021-11-13 15:36:23 +01:00
mod func;
mod module;
2021-11-13 15:36:23 +01:00
mod optimizer;
pub mod packages;
2021-11-13 15:36:23 +01:00
mod parser;
2022-12-11 14:50:47 +01:00
#[cfg(feature = "serde")]
pub mod serde;
2021-09-07 16:12:04 +02:00
mod tests;
2021-11-13 15:36:23 +01:00
mod tokenizer;
2023-01-06 08:43:53 +01:00
mod types;
2021-12-27 05:27:31 +01:00
/// Error encountered when parsing a script.
type PERR = ParseErrorType;
/// Evaluation result.
type ERR = EvalAltResult;
2021-12-25 16:49:14 +01:00
/// General evaluation error for Rhai scripts.
2021-12-27 05:27:31 +01:00
type RhaiError = Box<ERR>;
2021-12-25 16:49:14 +01:00
/// Generic [`Result`] type for Rhai functions.
type RhaiResultOf<T> = Result<T, RhaiError>;
/// General [`Result`] type for Rhai functions returning [`Dynamic`] values.
type RhaiResult = RhaiResultOf<Dynamic>;
2021-03-02 08:02:28 +01:00
2020-11-25 02:36:06 +01:00
/// The system integer type. It is defined as [`i64`].
2020-10-29 04:37:51 +01:00
///
2020-11-25 02:36:06 +01:00
/// If the `only_i32` feature is enabled, this will be [`i32`] instead.
2020-10-29 04:37:51 +01:00
#[cfg(not(feature = "only_i32"))]
pub type INT = i64;
/// The system integer type.
2020-11-25 02:36:06 +01:00
/// It is defined as [`i32`] since the `only_i32` feature is used.
2020-10-29 04:37:51 +01:00
///
2020-11-25 02:36:06 +01:00
/// If the `only_i32` feature is not used, this will be `i64` instead.
2020-10-29 04:37:51 +01:00
#[cfg(feature = "only_i32")]
pub type INT = i32;
2021-12-31 16:01:34 +01:00
/// The unsigned system base integer type. It is defined as [`u64`].
///
/// If the `only_i32` feature is enabled, this will be [`u32`] instead.
#[cfg(not(feature = "only_i32"))]
#[allow(non_camel_case_types)]
2022-01-13 15:51:56 +01:00
type UNSIGNED_INT = u64;
2021-12-31 16:01:34 +01:00
/// The unsigned system integer base type.
/// It is defined as [`u32`] since the `only_i32` feature is used.
///
/// If the `only_i32` feature is not used, this will be `u64` instead.
#[cfg(feature = "only_i32")]
#[allow(non_camel_case_types)]
2021-12-31 16:01:34 +01:00
type UNSIGNED_INT = u32;
2022-08-27 10:26:41 +02:00
/// The maximum integer that can fit into a [`usize`].
#[cfg(not(target_pointer_width = "32"))]
const MAX_USIZE_INT: INT = INT::MAX;
/// The maximum integer that can fit into a [`usize`].
#[cfg(not(feature = "only_i32"))]
#[cfg(target_pointer_width = "32")]
const MAX_USIZE_INT: INT = usize::MAX as INT;
/// The maximum integer that can fit into a [`usize`].
2023-02-10 11:33:22 +01:00
#[cfg(feature = "only_i32")]
#[cfg(target_pointer_width = "32")]
2022-08-27 10:26:41 +02:00
const MAX_USIZE_INT: INT = INT::MAX;
2022-03-09 02:25:55 +01:00
/// Number of bits in [`INT`].
///
/// It is 64 unless the `only_i32` feature is enabled when it will be 32.
const INT_BITS: usize = std::mem::size_of::<INT>() * 8;
/// Number of bytes that make up an [`INT`].
///
/// It is 8 unless the `only_i32` feature is enabled when it will be 4.
2022-06-08 11:06:49 +02:00
#[cfg(not(feature = "no_index"))]
2022-03-09 02:25:55 +01:00
const INT_BYTES: usize = std::mem::size_of::<INT>();
2020-11-25 02:36:06 +01:00
/// The system floating-point type. It is defined as [`f64`].
2022-01-25 16:59:35 +01:00
///
/// Not available under `no_float`.
2020-11-25 02:36:06 +01:00
///
/// If the `f32_float` feature is enabled, this will be [`f32`] instead.
2023-02-05 17:59:02 +01:00
#[cfg(not(feature = "no_float"))]
#[cfg(not(feature = "f32_float"))]
2020-10-29 04:37:51 +01:00
pub type FLOAT = f64;
2020-11-01 08:48:48 +01:00
/// The system floating-point type.
2020-11-25 02:36:06 +01:00
/// It is defined as [`f32`] since the `f32_float` feature is used.
2022-01-25 16:59:35 +01:00
///
/// Not available under `no_float`.
2020-11-25 02:36:06 +01:00
///
/// If the `f32_float` feature is not used, this will be `f64` instead.
2023-02-05 17:59:02 +01:00
#[cfg(not(feature = "no_float"))]
#[cfg(feature = "f32_float")]
2020-11-01 08:48:48 +01:00
pub type FLOAT = f32;
2022-03-09 02:25:55 +01:00
/// Number of bytes that make up a [`FLOAT`].
///
/// It is 8 unless the `f32_float` feature is enabled when it will be 4.
2023-02-05 17:59:02 +01:00
#[cfg(not(feature = "no_float"))]
#[cfg(not(feature = "no_index"))]
2022-03-09 02:25:55 +01:00
const FLOAT_BYTES: usize = std::mem::size_of::<FLOAT>();
2021-12-31 16:01:34 +01:00
/// An exclusive integer range.
type ExclusiveRange = std::ops::Range<INT>;
/// An inclusive integer range.
type InclusiveRange = std::ops::RangeInclusive<INT>;
2021-12-15 05:06:17 +01:00
2022-08-09 11:24:24 +02:00
#[allow(deprecated)]
pub use api::build_type::{CustomType, TypeBuilder};
2022-12-11 14:50:47 +01:00
#[cfg(not(feature = "no_custom_syntax"))]
pub use api::custom_syntax::Expression;
2023-02-05 17:59:02 +01:00
#[cfg(not(feature = "no_std"))]
#[cfg(not(target_family = "wasm"))]
2022-08-08 03:10:15 +02:00
pub use api::files::{eval_file, run_file};
2022-08-09 11:24:24 +02:00
pub use api::{eval::eval, events::VarDefInfo, run::run};
2021-03-07 15:10:54 +01:00
pub use ast::{FnAccess, AST};
2022-01-07 04:43:47 +01:00
pub use engine::{Engine, OP_CONTAINS, OP_EQUALS};
pub use eval::EvalContext;
2023-03-22 09:16:33 +01:00
#[cfg(not(feature = "no_function"))]
2023-03-22 09:05:25 +01:00
#[cfg(not(feature = "no_object"))]
use func::calc_typed_method_hash;
2022-12-11 14:50:47 +01:00
use func::{calc_fn_hash, calc_fn_hash_full, calc_var_hash};
pub use func::{plugin, FuncArgs, NativeCallContext, RegisterNativeFunction};
2020-11-17 05:23:53 +01:00
pub use module::{FnNamespace, Module};
2022-12-11 14:50:47 +01:00
use restore::RestoreOnDrop;
pub use rhai_codegen::*;
2022-10-15 06:37:42 +02:00
#[cfg(not(feature = "no_time"))]
2022-02-03 16:54:53 +01:00
pub use types::Instant;
2021-11-13 15:36:23 +01:00
pub use types::{
2022-12-11 14:50:47 +01:00
Dynamic, EvalAltResult, FnPtr, ImmutableString, LexError, ParseError, ParseErrorType, Position,
Scope,
2021-11-13 15:36:23 +01:00
};
2022-01-24 10:04:40 +01:00
/// _(debugging)_ Module containing types for debugging.
/// Exported under the `debugging` feature only.
#[cfg(feature = "debugging")]
pub mod debugger {
2022-01-25 07:32:07 +01:00
#[cfg(not(feature = "no_function"))]
pub use super::eval::CallStackFrame;
pub use super::eval::{BreakPoint, Debugger, DebuggerCommand, DebuggerEvent};
2022-01-24 10:04:40 +01:00
}
2021-06-16 12:58:48 +02:00
/// An identifier in Rhai. [`SmartString`](https://crates.io/crates/smartstring) is used because most
/// identifiers are ASCII and short, fewer than 23 characters, so they can be stored inline.
#[cfg(not(feature = "internals"))]
type Identifier = SmartString;
2021-06-16 12:58:48 +02:00
/// An identifier in Rhai. [`SmartString`](https://crates.io/crates/smartstring) is used because most
/// identifiers are ASCII and short, fewer than 23 characters, so they can be stored inline.
#[cfg(feature = "internals")]
pub type Identifier = SmartString;
2021-03-04 07:08:11 +01:00
/// Alias to [`Rc`][std::rc::Rc] or [`Arc`][std::sync::Arc] depending on the `sync` feature flag.
2021-11-16 05:26:37 +01:00
pub use func::Shared;
2021-03-04 07:08:11 +01:00
2021-11-13 15:36:23 +01:00
/// Alias to [`RefCell`][std::cell::RefCell] or [`RwLock`][std::sync::RwLock] depending on the `sync` feature flag.
2021-11-16 05:26:37 +01:00
pub use func::Locked;
2020-11-16 09:28:04 +01:00
2022-11-08 08:01:40 +01:00
/// A shared [`Module`].
type SharedModule = Shared<Module>;
2021-01-28 09:48:56 +01:00
#[cfg(not(feature = "no_function"))]
2021-11-27 16:20:05 +01:00
pub use func::Func;
2021-01-28 09:48:56 +01:00
2021-03-07 15:10:54 +01:00
#[cfg(not(feature = "no_function"))]
pub use ast::ScriptFnMetadata;
#[cfg(not(feature = "no_function"))]
pub use api::call_fn::CallFnOptions;
2020-11-25 02:36:06 +01:00
/// Variable-sized array of [`Dynamic`] values.
2022-01-25 16:59:35 +01:00
///
2021-01-02 16:30:10 +01:00
/// Not available under `no_index`.
#[cfg(not(feature = "no_index"))]
2021-04-17 09:15:54 +02:00
pub type Array = Vec<Dynamic>;
2021-11-23 07:58:54 +01:00
/// Variable-sized array of [`u8`] values (byte array).
2022-01-25 16:59:35 +01:00
///
2021-11-23 07:58:54 +01:00
/// Not available under `no_index`.
#[cfg(not(feature = "no_index"))]
pub type Blob = Vec<u8>;
2021-12-31 16:01:34 +01:00
/// A dictionary of [`Dynamic`] values with string keys.
2022-01-25 16:59:35 +01:00
///
2021-01-02 16:30:10 +01:00
/// Not available under `no_object`.
2021-12-31 16:01:34 +01:00
///
/// [`SmartString`](https://crates.io/crates/smartstring) is used as the key type because most
/// property names are ASCII and short, fewer than 23 characters, so they can be stored inline.
2020-03-29 17:53:35 +02:00
#[cfg(not(feature = "no_object"))]
2021-04-17 09:15:54 +02:00
pub type Map = std::collections::BTreeMap<Identifier, Dynamic>;
2020-03-29 17:53:35 +02:00
2022-04-21 06:15:21 +02:00
#[cfg(not(feature = "no_object"))]
pub use api::json::format_map_as_json;
2020-05-05 09:00:10 +02:00
#[cfg(not(feature = "no_module"))]
2020-05-13 13:21:42 +02:00
pub use module::ModuleResolver;
2020-05-05 17:57:25 +02:00
2020-05-15 15:40:54 +02:00
/// Module containing all built-in _module resolvers_ available to Rhai.
2020-05-05 17:57:25 +02:00
#[cfg(not(feature = "no_module"))]
2021-01-28 09:48:56 +01:00
pub use module::resolvers as module_resolvers;
2020-07-03 11:19:55 +02:00
#[cfg(not(feature = "no_optimize"))]
2021-11-13 15:36:23 +01:00
pub use optimizer::OptimizationLevel;
2022-02-07 14:03:39 +01:00
/// Placeholder for the optimization level.
#[cfg(feature = "no_optimize")]
pub type OptimizationLevel = ();
2021-11-13 15:36:23 +01:00
// Expose internal data structures.
2020-06-23 04:43:24 +02:00
2021-03-05 07:18:36 +01:00
#[cfg(feature = "internals")]
2021-11-13 15:36:23 +01:00
pub use types::dynamic::{AccessMode, DynamicReadLock, DynamicWriteLock, Variant};
2021-03-05 07:18:36 +01:00
2023-02-10 11:33:22 +01:00
#[cfg(feature = "internals")]
#[cfg(not(feature = "no_float"))]
2022-11-08 16:17:31 +01:00
pub use types::FloatWrapper;
2021-05-10 05:07:19 +02:00
2020-06-23 04:43:24 +02:00
#[cfg(feature = "internals")]
2022-11-28 16:24:22 +01:00
pub use types::{Span, StringsInterner};
2021-09-24 03:26:35 +02:00
#[cfg(feature = "internals")]
2021-11-13 15:36:23 +01:00
pub use tokenizer::{
2022-11-09 05:44:57 +01:00
get_next_token, is_valid_function_name, is_valid_identifier, parse_string_literal, InputStream,
2022-11-28 16:24:22 +01:00
MultiInputsStream, Token, TokenIterator, TokenizeState, TokenizerControl,
2022-11-09 05:44:57 +01:00
TokenizerControlBlock,
2021-09-24 03:26:35 +02:00
};
2021-12-27 14:56:50 +01:00
#[cfg(feature = "internals")]
pub use parser::ParseState;
2020-06-23 04:43:24 +02:00
#[cfg(feature = "internals")]
pub use ast::{
2022-12-23 07:26:06 +01:00
ASTFlags, ASTNode, BinaryExpr, ConditionalExpr, Expr, FlowControl, FnCallExpr, FnCallHashes,
Ident, OpAssignment, RangeCase, ScriptFnDef, Stmt, StmtBlock, SwitchCasesCollection,
};
2020-07-09 13:54:28 +02:00
2023-02-05 17:59:02 +01:00
#[cfg(feature = "internals")]
#[cfg(not(feature = "no_custom_syntax"))]
2022-07-05 16:59:03 +02:00
pub use ast::CustomExpr;
2023-02-05 17:59:02 +01:00
#[cfg(feature = "internals")]
#[cfg(not(feature = "no_module"))]
2022-03-05 10:57:23 +01:00
pub use ast::Namespace;
#[cfg(feature = "internals")]
pub use func::EncapsulatedEnviron;
#[cfg(feature = "internals")]
2022-04-16 10:36:53 +02:00
pub use eval::{Caches, FnResolutionCache, FnResolutionCacheEntry, GlobalRuntimeState};
2020-07-09 13:54:28 +02:00
#[cfg(feature = "internals")]
2022-11-09 05:44:57 +01:00
#[allow(deprecated)]
2022-11-17 11:05:52 +01:00
pub use func::{locked_read, locked_write, CallableFunction, NativeCallContextStore};
2020-11-27 16:37:59 +01:00
2023-02-05 17:59:02 +01:00
#[cfg(feature = "internals")]
#[cfg(feature = "metadata")]
2022-07-28 12:08:21 +02:00
pub use api::definitions::Definitions;
2022-10-14 08:04:54 +02:00
/// Number of items to keep inline for [`StaticVec`].
const STATIC_VEC_INLINE_SIZE: usize = 3;
2021-09-10 14:25:22 +02:00
/// Alias to [`smallvec::SmallVec<[T; 3]>`](https://crates.io/crates/smallvec), which is a
/// specialized [`Vec`] backed by a small, inline, fixed-size array when there are ≤ 3 items stored.
2021-03-17 06:30:47 +01:00
///
/// # History
2021-03-17 06:30:47 +01:00
///
/// And Saint Attila raised the `SmallVec` up on high, saying, "O Lord, bless this Thy `SmallVec`
/// that, with it, Thou mayest blow Thine allocation costs to tiny bits in Thy mercy."
///
/// And the Lord did grin, and the people did feast upon the lambs and sloths and carp and anchovies
/// and orangutans and breakfast cereals and fruit bats and large chu...
///
/// And the Lord spake, saying, "First shalt thou depend on the [`smallvec`](https://crates.io/crates/smallvec) crate.
2021-09-10 14:25:22 +02:00
/// Then, shalt thou keep three inline. No more. No less. Three shalt be the number thou shalt keep inline,
/// and the number to keep inline shalt be three. Four shalt thou not keep inline, nor either keep inline
/// thou two, excepting that thou then proceed to three. Five is right out. Once the number three,
/// being the third number, be reached, then, lobbest thou thy `SmallVec` towards thy heap, who,
2021-03-17 06:30:47 +01:00
/// being slow and cache-naughty in My sight, shall snuff it."
///
2021-09-11 13:40:40 +02:00
/// # Why Three
2021-03-17 06:30:47 +01:00
///
/// `StaticVec` is used frequently to keep small lists of items in inline (non-heap) storage in
/// order to improve cache friendliness and reduce indirections.
///
2021-09-10 14:25:22 +02:00
/// The number 3, other than being the holy number, is carefully chosen for a balance between
2021-03-17 06:30:47 +01:00
/// storage space and reduce allocations. That is because most function calls (and most functions,
/// for that matter) contain fewer than 4 arguments, the exception being closures that capture a
2021-03-17 06:30:47 +01:00
/// large number of external variables.
///
2021-09-11 13:40:40 +02:00
/// In addition, most script blocks either contain many statements, or just one or two lines;
2021-09-10 14:25:22 +02:00
/// most scripts load fewer than 4 external modules; most module paths contain fewer than 4 levels
2021-09-11 13:40:40 +02:00
/// (e.g. `std::collections::map::HashMap` is 4 levels and it is just about as long as they get).
2020-10-10 07:41:55 +02:00
#[cfg(not(feature = "internals"))]
2022-10-14 08:04:54 +02:00
type StaticVec<T> = smallvec::SmallVec<[T; STATIC_VEC_INLINE_SIZE]>;
2020-10-10 07:41:55 +02:00
2021-09-11 13:40:40 +02:00
/// _(internals)_ Alias to [`smallvec::SmallVec<[T; 3]>`](https://crates.io/crates/smallvec),
/// which is a [`Vec`] backed by a small, inline, fixed-size array when there are ≤ 3 items stored.
2020-10-10 07:41:55 +02:00
/// Exported under the `internals` feature only.
2021-03-17 06:30:47 +01:00
///
/// # History
2021-03-17 06:30:47 +01:00
///
/// And Saint Attila raised the `SmallVec` up on high, saying, "O Lord, bless this Thy `SmallVec`
/// that, with it, Thou mayest blow Thine allocation costs to tiny bits in Thy mercy."
///
/// And the Lord did grin, and the people did feast upon the lambs and sloths and carp and anchovies
/// and orangutans and breakfast cereals and fruit bats and large chu...
///
/// And the Lord spake, saying, "First shalt thou depend on the [`smallvec`](https://crates.io/crates/smallvec) crate.
2021-09-10 14:25:22 +02:00
/// Then, shalt thou keep three inline. No more. No less. Three shalt be the number thou shalt keep inline,
/// and the number to keep inline shalt be three. Four shalt thou not keep inline, nor either keep inline
/// thou two, excepting that thou then proceed to three. Five is right out. Once the number three,
/// being the third number, be reached, then, lobbest thou thy `SmallVec` towards thy heap, who,
2021-03-17 06:30:47 +01:00
/// being slow and cache-naughty in My sight, shall snuff it."
///
2021-09-11 13:40:40 +02:00
/// # Why Three
2021-03-17 06:30:47 +01:00
///
/// `StaticVec` is used frequently to keep small lists of items in inline (non-heap) storage in
/// order to improve cache friendliness and reduce indirections.
///
2021-09-10 14:25:22 +02:00
/// The number 3, other than being the holy number, is carefully chosen for a balance between
2021-03-17 06:30:47 +01:00
/// storage space and reduce allocations. That is because most function calls (and most functions,
/// for that matter) contain fewer than 4 arguments, the exception being closures that capture a
2021-03-17 06:30:47 +01:00
/// large number of external variables.
///
2021-09-11 13:40:40 +02:00
/// In addition, most script blocks either contain many statements, or just one or two lines;
2021-09-10 14:25:22 +02:00
/// most scripts load fewer than 4 external modules; most module paths contain fewer than 4 levels
2021-09-11 13:40:40 +02:00
/// (e.g. `std::collections::map::HashMap` is 4 levels and it is just about as long as they get).
2020-06-23 04:43:24 +02:00
#[cfg(feature = "internals")]
2022-10-14 08:04:54 +02:00
pub type StaticVec<T> = smallvec::SmallVec<[T; STATIC_VEC_INLINE_SIZE]>;
/// Number of items to keep inline for [`FnArgsVec`].
#[cfg(not(feature = "no_closure"))]
const FN_ARGS_VEC_INLINE_SIZE: usize = 5;
/// Inline arguments storage for function calls.
///
/// # Notes
///
/// Since most usage of this is during a function call to gather up arguments, this is mostly
/// allocated on the stack, so we can tolerate a larger number of values stored inline.
///
/// Most functions have few parameters, but closures with a lot of captured variables can
/// potentially have many. Having a larger inline storage for arguments reduces allocations in
/// scripts with heavy closure usage.
///
/// Under `no_closure`, this type aliases to [`StaticVec`][crate::StaticVec] instead.
#[cfg(not(feature = "no_closure"))]
2022-10-14 08:04:54 +02:00
type FnArgsVec<T> = smallvec::SmallVec<[T; FN_ARGS_VEC_INLINE_SIZE]>;
/// Inline arguments storage for function calls.
2021-12-27 10:00:21 +01:00
/// This type aliases to [`StaticVec`][crate::StaticVec].
#[cfg(feature = "no_closure")]
type FnArgsVec<T> = crate::StaticVec<T>;
type SmartString = smartstring::SmartString<smartstring::LazyCompact>;
// Compiler guards against mutually-exclusive feature flags
2023-02-05 17:59:02 +01:00
#[cfg(feature = "no_float")]
#[cfg(feature = "f32_float")]
2021-09-12 15:06:13 +02:00
compile_error!("`f32_float` cannot be used with `no_float`");
2023-02-05 17:59:02 +01:00
#[cfg(feature = "only_i32")]
#[cfg(feature = "only_i64")]
2021-09-12 15:06:13 +02:00
compile_error!("`only_i32` and `only_i64` cannot be used together");
2023-02-05 17:59:02 +01:00
#[cfg(feature = "no_std")]
#[cfg(feature = "wasm-bindgen")]
2021-09-12 15:06:13 +02:00
compile_error!("`wasm-bindgen` cannot be used with `no-std`");
2023-02-05 17:59:02 +01:00
#[cfg(feature = "no_std")]
#[cfg(feature = "stdweb")]
2021-09-12 15:06:13 +02:00
compile_error!("`stdweb` cannot be used with `no-std`");
2023-02-05 17:59:02 +01:00
#[cfg(target_family = "wasm")]
#[cfg(feature = "no_std")]
2021-09-12 15:06:13 +02:00
compile_error!("`no_std` cannot be used for WASM target");
2023-02-05 17:59:02 +01:00
#[cfg(not(target_family = "wasm"))]
#[cfg(feature = "wasm-bindgen")]
2021-09-12 15:06:13 +02:00
compile_error!("`wasm-bindgen` cannot be used for non-WASM target");
2023-02-05 17:59:02 +01:00
#[cfg(not(target_family = "wasm"))]
#[cfg(feature = "stdweb")]
2021-09-12 15:06:13 +02:00
compile_error!("`stdweb` cannot be used non-WASM target");
2023-02-05 17:59:02 +01:00
#[cfg(feature = "wasm-bindgen")]
#[cfg(feature = "stdweb")]
2021-09-12 15:06:13 +02:00
compile_error!("`wasm-bindgen` and `stdweb` cannot be used together");