2020-10-28 15:18:44 +01:00
|
|
|
//! Module defining the AST (abstract syntax tree).
|
|
|
|
|
2021-06-08 08:46:49 +02:00
|
|
|
use crate::dynamic::Union;
|
2020-11-19 03:41:08 +01:00
|
|
|
use crate::fn_native::shared_make_mut;
|
2021-01-08 17:40:44 +01:00
|
|
|
use crate::module::NamespaceRef;
|
2021-04-17 09:15:54 +02:00
|
|
|
use crate::token::Token;
|
|
|
|
use crate::utils::calc_fn_hash;
|
|
|
|
use crate::{
|
2021-04-20 17:40:52 +02:00
|
|
|
Dynamic, FnNamespace, Identifier, ImmutableString, Module, Position, Shared, StaticVec, INT,
|
2021-04-17 09:15:54 +02:00
|
|
|
};
|
|
|
|
#[cfg(feature = "no_std")]
|
|
|
|
use std::prelude::v1::*;
|
|
|
|
use std::{
|
2021-03-29 07:07:10 +02:00
|
|
|
collections::BTreeMap,
|
2020-10-29 04:37:51 +01:00
|
|
|
fmt,
|
2021-01-18 03:56:42 +01:00
|
|
|
hash::Hash,
|
2021-05-05 12:38:52 +02:00
|
|
|
mem,
|
2021-04-05 17:06:48 +02:00
|
|
|
num::{NonZeroU8, NonZeroUsize},
|
2021-04-16 07:15:11 +02:00
|
|
|
ops::{Add, AddAssign, Deref, DerefMut},
|
2020-11-17 05:23:53 +01:00
|
|
|
};
|
2020-11-16 16:10:14 +01:00
|
|
|
|
|
|
|
#[cfg(not(feature = "no_float"))]
|
2021-04-17 09:15:54 +02:00
|
|
|
use std::str::FromStr;
|
|
|
|
|
|
|
|
#[cfg(not(feature = "no_float"))]
|
|
|
|
use crate::FLOAT;
|
2020-11-16 16:10:14 +01:00
|
|
|
|
2021-04-06 17:18:41 +02:00
|
|
|
#[cfg(not(feature = "no_float"))]
|
|
|
|
use num_traits::Float;
|
|
|
|
|
2020-11-16 16:10:14 +01:00
|
|
|
#[cfg(not(feature = "no_index"))]
|
|
|
|
use crate::Array;
|
|
|
|
|
2020-11-17 05:23:53 +01:00
|
|
|
/// A type representing the access mode of a function.
|
2020-10-29 04:37:51 +01:00
|
|
|
#[derive(Debug, Clone, Copy, Eq, PartialEq, Hash)]
|
|
|
|
pub enum FnAccess {
|
|
|
|
/// Public function.
|
|
|
|
Public,
|
|
|
|
/// Private function.
|
|
|
|
Private,
|
|
|
|
}
|
|
|
|
|
2020-11-20 09:52:28 +01:00
|
|
|
/// _(INTERNALS)_ A type containing information on a scripted function.
|
2020-10-29 04:37:51 +01:00
|
|
|
/// Exported under the `internals` feature only.
|
|
|
|
///
|
2021-01-16 07:46:03 +01:00
|
|
|
/// # Volatile Data Structure
|
2020-10-29 04:37:51 +01:00
|
|
|
///
|
|
|
|
/// This type is volatile and may change.
|
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub struct ScriptFnDef {
|
2020-11-04 05:34:54 +01:00
|
|
|
/// Function body.
|
2021-03-10 15:12:48 +01:00
|
|
|
pub body: StmtBlock,
|
2020-11-04 05:34:54 +01:00
|
|
|
/// Encapsulated running environment, if any.
|
|
|
|
pub lib: Option<Shared<Module>>,
|
2020-11-09 14:52:23 +01:00
|
|
|
/// Encapsulated imported modules.
|
2021-05-10 05:07:19 +02:00
|
|
|
///
|
|
|
|
/// Not available under `no_module`.
|
2020-11-09 14:52:23 +01:00
|
|
|
#[cfg(not(feature = "no_module"))]
|
2020-11-16 09:28:04 +01:00
|
|
|
pub mods: crate::engine::Imports,
|
2020-10-29 04:37:51 +01:00
|
|
|
/// Function name.
|
2021-03-29 05:36:02 +02:00
|
|
|
pub name: Identifier,
|
2020-10-29 04:37:51 +01:00
|
|
|
/// Function access mode.
|
|
|
|
pub access: FnAccess,
|
|
|
|
/// Names of function parameters.
|
2021-03-29 05:36:02 +02:00
|
|
|
pub params: StaticVec<Identifier>,
|
2020-11-09 14:52:23 +01:00
|
|
|
/// Access to external variables.
|
2021-05-10 05:07:19 +02:00
|
|
|
///
|
|
|
|
/// Not available under `no_closure`.
|
2020-10-29 04:37:51 +01:00
|
|
|
#[cfg(not(feature = "no_closure"))]
|
2021-04-17 09:15:54 +02:00
|
|
|
pub externals: std::collections::BTreeSet<Identifier>,
|
2021-05-10 05:07:19 +02:00
|
|
|
/// _(METADATA)_ Function doc-comments (if any).
|
|
|
|
/// Exported under the `metadata` feature only.
|
|
|
|
///
|
|
|
|
/// Not available under `no_function`.
|
2021-04-09 16:49:47 +02:00
|
|
|
#[cfg(not(feature = "no_function"))]
|
|
|
|
#[cfg(feature = "metadata")]
|
2021-04-20 13:19:35 +02:00
|
|
|
pub comments: StaticVec<String>,
|
2020-10-29 04:37:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for ScriptFnDef {
|
2020-12-29 03:41:20 +01:00
|
|
|
#[inline(always)]
|
2020-10-29 04:37:51 +01:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
|
|
write!(
|
|
|
|
f,
|
2020-12-28 02:49:54 +01:00
|
|
|
"{}{}({})",
|
2021-03-03 04:40:27 +01:00
|
|
|
match self.access {
|
|
|
|
FnAccess::Public => "",
|
2021-03-15 14:30:55 +01:00
|
|
|
FnAccess::Private => "private ",
|
2020-10-29 04:37:51 +01:00
|
|
|
},
|
|
|
|
self.name,
|
|
|
|
self.params
|
|
|
|
.iter()
|
|
|
|
.map(|s| s.as_str())
|
2021-06-06 06:17:04 +02:00
|
|
|
.collect::<StaticVec<_>>()
|
2020-11-22 10:21:34 +01:00
|
|
|
.join(", ")
|
2020-10-29 04:37:51 +01:00
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-18 09:31:44 +01:00
|
|
|
/// A type containing the metadata of a script-defined function.
|
2021-04-17 06:03:29 +02:00
|
|
|
///
|
2021-03-24 12:27:38 +01:00
|
|
|
/// Not available under `no_function`.
|
2020-12-18 09:31:44 +01:00
|
|
|
///
|
|
|
|
/// Created by [`AST::iter_functions`].
|
2021-03-07 15:10:54 +01:00
|
|
|
#[cfg(not(feature = "no_function"))]
|
2020-12-18 09:31:44 +01:00
|
|
|
#[derive(Debug, Eq, PartialEq, Clone, Hash)]
|
|
|
|
pub struct ScriptFnMetadata<'a> {
|
2021-05-10 05:07:19 +02:00
|
|
|
/// _(METADATA)_ Function doc-comments (if any).
|
|
|
|
/// Exported under the `metadata` feature only.
|
|
|
|
///
|
|
|
|
/// Not available under `no_function`.
|
2020-12-18 09:31:44 +01:00
|
|
|
///
|
|
|
|
/// Block doc-comments are kept in a single string slice with line-breaks within.
|
|
|
|
///
|
|
|
|
/// Line doc-comments are kept in one string slice per line without the termination line-break.
|
|
|
|
///
|
|
|
|
/// Leading white-spaces are stripped, and each string slice always starts with the corresponding
|
|
|
|
/// doc-comment leader: `///` or `/**`.
|
2021-04-09 16:49:47 +02:00
|
|
|
#[cfg(not(feature = "no_function"))]
|
|
|
|
#[cfg(feature = "metadata")]
|
2020-12-18 09:31:44 +01:00
|
|
|
pub comments: Vec<&'a str>,
|
|
|
|
/// Function access mode.
|
2020-12-12 11:44:28 +01:00
|
|
|
pub access: FnAccess,
|
2020-12-18 09:31:44 +01:00
|
|
|
/// Function name.
|
|
|
|
pub name: &'a str,
|
|
|
|
/// Function parameters (if any).
|
|
|
|
pub params: Vec<&'a str>,
|
2020-12-12 11:44:28 +01:00
|
|
|
}
|
|
|
|
|
2021-03-07 15:10:54 +01:00
|
|
|
#[cfg(not(feature = "no_function"))]
|
2020-12-18 09:31:44 +01:00
|
|
|
impl fmt::Display for ScriptFnMetadata<'_> {
|
2020-12-29 03:41:20 +01:00
|
|
|
#[inline(always)]
|
2020-12-12 11:44:28 +01:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
|
|
write!(
|
|
|
|
f,
|
2020-12-28 02:49:54 +01:00
|
|
|
"{}{}({})",
|
2021-03-03 04:40:27 +01:00
|
|
|
match self.access {
|
|
|
|
FnAccess::Public => "",
|
2021-03-15 14:30:55 +01:00
|
|
|
FnAccess::Private => "private ",
|
2020-12-12 11:44:28 +01:00
|
|
|
},
|
2020-12-18 09:31:44 +01:00
|
|
|
self.name,
|
2021-06-06 06:17:04 +02:00
|
|
|
self.params
|
|
|
|
.iter()
|
|
|
|
.cloned()
|
|
|
|
.collect::<StaticVec<_>>()
|
|
|
|
.join(", ")
|
2020-12-12 11:44:28 +01:00
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-07 15:10:54 +01:00
|
|
|
#[cfg(not(feature = "no_function"))]
|
2020-12-18 09:31:44 +01:00
|
|
|
impl<'a> Into<ScriptFnMetadata<'a>> for &'a ScriptFnDef {
|
2020-12-29 03:41:20 +01:00
|
|
|
#[inline(always)]
|
2020-12-18 09:31:44 +01:00
|
|
|
fn into(self) -> ScriptFnMetadata<'a> {
|
2020-12-12 11:44:28 +01:00
|
|
|
ScriptFnMetadata {
|
2021-04-09 16:49:47 +02:00
|
|
|
#[cfg(not(feature = "no_function"))]
|
|
|
|
#[cfg(feature = "metadata")]
|
2020-12-18 09:31:44 +01:00
|
|
|
comments: self.comments.iter().map(|s| s.as_str()).collect(),
|
2020-12-12 11:44:28 +01:00
|
|
|
access: self.access,
|
2020-12-18 09:31:44 +01:00
|
|
|
name: &self.name,
|
|
|
|
params: self.params.iter().map(|s| s.as_str()).collect(),
|
2020-12-12 11:44:28 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-04 04:19:08 +02:00
|
|
|
#[cfg(not(feature = "no_function"))]
|
|
|
|
impl std::cmp::PartialOrd for ScriptFnMetadata<'_> {
|
|
|
|
fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
|
|
|
|
Some(self.cmp(other))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(not(feature = "no_function"))]
|
|
|
|
impl std::cmp::Ord for ScriptFnMetadata<'_> {
|
|
|
|
fn cmp(&self, other: &Self) -> std::cmp::Ordering {
|
|
|
|
match self.name.cmp(other.name) {
|
|
|
|
std::cmp::Ordering::Equal => self.params.len().cmp(&other.params.len()),
|
|
|
|
cmp => cmp,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-28 15:18:44 +01:00
|
|
|
/// Compiled AST (abstract syntax tree) of a Rhai script.
|
|
|
|
///
|
|
|
|
/// # Thread Safety
|
|
|
|
///
|
2020-11-20 09:52:28 +01:00
|
|
|
/// Currently, [`AST`] is neither `Send` nor `Sync`. Turn on the `sync` feature to make it `Send + Sync`.
|
2020-11-15 06:49:54 +01:00
|
|
|
#[derive(Debug, Clone)]
|
2020-12-12 08:57:55 +01:00
|
|
|
pub struct AST {
|
2020-12-21 15:04:46 +01:00
|
|
|
/// Source of the [`AST`].
|
2021-03-29 05:36:02 +02:00
|
|
|
source: Option<Identifier>,
|
2020-10-28 15:18:44 +01:00
|
|
|
/// Global statements.
|
2021-03-10 06:32:09 +01:00
|
|
|
body: StmtBlock,
|
2020-10-28 15:18:44 +01:00
|
|
|
/// Script-defined functions.
|
2020-12-12 08:57:55 +01:00
|
|
|
functions: Shared<Module>,
|
2021-01-08 17:24:55 +01:00
|
|
|
/// Embedded module resolver, if any.
|
2021-05-10 05:07:19 +02:00
|
|
|
///
|
|
|
|
/// Not available under `no_module`.
|
2021-01-08 17:40:44 +01:00
|
|
|
#[cfg(not(feature = "no_module"))]
|
|
|
|
resolver: Option<Shared<crate::module::resolvers::StaticModuleResolver>>,
|
2020-12-12 08:57:55 +01:00
|
|
|
}
|
2020-10-28 15:18:44 +01:00
|
|
|
|
2020-11-15 06:49:54 +01:00
|
|
|
impl Default for AST {
|
2020-12-29 03:41:20 +01:00
|
|
|
#[inline(always)]
|
2020-11-15 06:49:54 +01:00
|
|
|
fn default() -> Self {
|
2020-12-12 08:57:55 +01:00
|
|
|
Self {
|
2020-12-21 15:04:46 +01:00
|
|
|
source: None,
|
2021-03-10 06:32:09 +01:00
|
|
|
body: Default::default(),
|
2020-12-12 08:57:55 +01:00
|
|
|
functions: Default::default(),
|
2021-01-08 17:40:44 +01:00
|
|
|
#[cfg(not(feature = "no_module"))]
|
2021-01-08 17:24:55 +01:00
|
|
|
resolver: None,
|
2020-12-12 08:57:55 +01:00
|
|
|
}
|
2020-11-15 06:49:54 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-28 15:18:44 +01:00
|
|
|
impl AST {
|
2020-11-20 09:52:28 +01:00
|
|
|
/// Create a new [`AST`].
|
2020-10-28 15:18:44 +01:00
|
|
|
#[inline(always)]
|
2020-12-12 08:57:55 +01:00
|
|
|
pub fn new(
|
|
|
|
statements: impl IntoIterator<Item = Stmt>,
|
|
|
|
functions: impl Into<Shared<Module>>,
|
|
|
|
) -> Self {
|
|
|
|
Self {
|
2020-12-21 15:04:46 +01:00
|
|
|
source: None,
|
2021-06-06 06:17:04 +02:00
|
|
|
body: StmtBlock::new(statements.into_iter().collect(), Position::NONE),
|
2020-12-12 08:57:55 +01:00
|
|
|
functions: functions.into(),
|
2021-01-08 17:40:44 +01:00
|
|
|
#[cfg(not(feature = "no_module"))]
|
2021-01-08 17:24:55 +01:00
|
|
|
resolver: None,
|
2020-12-12 08:57:55 +01:00
|
|
|
}
|
2020-10-28 15:18:44 +01:00
|
|
|
}
|
2020-12-21 15:04:46 +01:00
|
|
|
/// Create a new [`AST`] with a source name.
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn new_with_source(
|
|
|
|
statements: impl IntoIterator<Item = Stmt>,
|
|
|
|
functions: impl Into<Shared<Module>>,
|
2021-03-29 05:36:02 +02:00
|
|
|
source: impl Into<Identifier>,
|
2020-12-21 15:04:46 +01:00
|
|
|
) -> Self {
|
|
|
|
Self {
|
|
|
|
source: Some(source.into()),
|
2021-06-06 06:17:04 +02:00
|
|
|
body: StmtBlock::new(statements.into_iter().collect(), Position::NONE),
|
2020-12-21 15:04:46 +01:00
|
|
|
functions: functions.into(),
|
2021-01-08 17:40:44 +01:00
|
|
|
#[cfg(not(feature = "no_module"))]
|
2021-01-08 17:24:55 +01:00
|
|
|
resolver: None,
|
2020-12-21 15:04:46 +01:00
|
|
|
}
|
|
|
|
}
|
2021-01-09 08:20:07 +01:00
|
|
|
/// Get the source, if any.
|
2020-12-29 03:41:20 +01:00
|
|
|
#[inline(always)]
|
2020-12-21 15:04:46 +01:00
|
|
|
pub fn source(&self) -> Option<&str> {
|
|
|
|
self.source.as_ref().map(|s| s.as_str())
|
|
|
|
}
|
2021-01-09 08:20:07 +01:00
|
|
|
/// Clone the source, if any.
|
2020-12-29 03:41:20 +01:00
|
|
|
#[inline(always)]
|
2021-03-29 05:36:02 +02:00
|
|
|
pub(crate) fn clone_source(&self) -> Option<Identifier> {
|
2020-12-21 15:04:46 +01:00
|
|
|
self.source.clone()
|
|
|
|
}
|
|
|
|
/// Set the source.
|
2020-12-29 03:41:20 +01:00
|
|
|
#[inline(always)]
|
2021-03-29 05:36:02 +02:00
|
|
|
pub fn set_source(&mut self, source: impl Into<Identifier>) -> &mut Self {
|
2021-05-25 04:54:48 +02:00
|
|
|
let source = Some(source.into());
|
|
|
|
Shared::get_mut(&mut self.functions)
|
|
|
|
.as_mut()
|
|
|
|
.map(|m| m.set_id(source.clone()));
|
|
|
|
self.source = source;
|
2021-01-09 08:20:07 +01:00
|
|
|
self
|
|
|
|
}
|
|
|
|
/// Clear the source.
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn clear_source(&mut self) -> &mut Self {
|
|
|
|
self.source = None;
|
|
|
|
self
|
2020-12-21 15:04:46 +01:00
|
|
|
}
|
2020-10-28 15:18:44 +01:00
|
|
|
/// Get the statements.
|
|
|
|
#[cfg(not(feature = "internals"))]
|
|
|
|
#[inline(always)]
|
|
|
|
pub(crate) fn statements(&self) -> &[Stmt] {
|
2021-04-16 06:04:33 +02:00
|
|
|
&self.body.0
|
2020-10-28 15:18:44 +01:00
|
|
|
}
|
2020-11-20 09:52:28 +01:00
|
|
|
/// _(INTERNALS)_ Get the statements.
|
2020-10-28 15:18:44 +01:00
|
|
|
/// Exported under the `internals` feature only.
|
|
|
|
#[cfg(feature = "internals")]
|
2020-12-23 08:30:35 +01:00
|
|
|
#[deprecated = "this method is volatile and may change"]
|
2020-10-28 15:18:44 +01:00
|
|
|
#[inline(always)]
|
|
|
|
pub fn statements(&self) -> &[Stmt] {
|
2021-04-16 07:15:11 +02:00
|
|
|
&self.body.0
|
2020-10-28 15:18:44 +01:00
|
|
|
}
|
|
|
|
/// Get a mutable reference to the statements.
|
|
|
|
#[cfg(not(feature = "no_optimize"))]
|
|
|
|
#[inline(always)]
|
2021-03-10 06:32:09 +01:00
|
|
|
pub(crate) fn statements_mut(&mut self) -> &mut StaticVec<Stmt> {
|
2021-04-16 06:04:33 +02:00
|
|
|
&mut self.body.0
|
2020-10-28 15:18:44 +01:00
|
|
|
}
|
2020-11-20 09:52:28 +01:00
|
|
|
/// Get the internal shared [`Module`] containing all script-defined functions.
|
2021-01-15 19:38:50 +01:00
|
|
|
#[cfg(not(feature = "internals"))]
|
2020-11-27 16:37:59 +01:00
|
|
|
#[cfg(not(feature = "no_module"))]
|
2020-11-22 15:15:17 +01:00
|
|
|
#[cfg(not(feature = "no_function"))]
|
2020-11-19 03:52:45 +01:00
|
|
|
#[inline(always)]
|
|
|
|
pub(crate) fn shared_lib(&self) -> Shared<Module> {
|
2020-12-12 08:57:55 +01:00
|
|
|
self.functions.clone()
|
2020-11-19 03:52:45 +01:00
|
|
|
}
|
2021-01-16 07:46:03 +01:00
|
|
|
/// _(INTERNALS)_ Get the internal shared [`Module`] containing all script-defined functions.
|
|
|
|
/// Exported under the `internals` feature only.
|
2021-04-17 06:03:29 +02:00
|
|
|
///
|
2021-05-10 05:07:19 +02:00
|
|
|
/// Not available under `no_function` or `no_module`.
|
2021-01-15 19:38:50 +01:00
|
|
|
#[cfg(feature = "internals")]
|
2021-01-16 07:46:03 +01:00
|
|
|
#[deprecated = "this method is volatile and may change"]
|
2021-01-15 19:38:50 +01:00
|
|
|
#[cfg(not(feature = "no_module"))]
|
|
|
|
#[cfg(not(feature = "no_function"))]
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn shared_lib(&self) -> Shared<Module> {
|
|
|
|
self.functions.clone()
|
|
|
|
}
|
2020-11-20 09:52:28 +01:00
|
|
|
/// Get the internal [`Module`] containing all script-defined functions.
|
2020-10-28 15:18:44 +01:00
|
|
|
#[cfg(not(feature = "internals"))]
|
|
|
|
#[inline(always)]
|
|
|
|
pub(crate) fn lib(&self) -> &Module {
|
2020-12-12 08:57:55 +01:00
|
|
|
&self.functions
|
2020-10-28 15:18:44 +01:00
|
|
|
}
|
2020-11-20 09:52:28 +01:00
|
|
|
/// _(INTERNALS)_ Get the internal [`Module`] containing all script-defined functions.
|
2020-10-28 15:18:44 +01:00
|
|
|
/// Exported under the `internals` feature only.
|
2021-04-17 06:03:29 +02:00
|
|
|
///
|
2021-03-24 12:27:38 +01:00
|
|
|
/// Not available under `no_function`.
|
2020-10-28 15:18:44 +01:00
|
|
|
#[cfg(feature = "internals")]
|
2020-12-23 08:30:35 +01:00
|
|
|
#[deprecated = "this method is volatile and may change"]
|
2020-10-28 15:18:44 +01:00
|
|
|
#[inline(always)]
|
|
|
|
pub fn lib(&self) -> &Module {
|
2020-12-12 08:57:55 +01:00
|
|
|
&self.functions
|
2020-10-28 15:18:44 +01:00
|
|
|
}
|
2021-01-08 17:24:55 +01:00
|
|
|
/// Get the embedded [module resolver][`ModuleResolver`].
|
2021-01-08 17:40:44 +01:00
|
|
|
#[cfg(not(feature = "no_module"))]
|
2021-01-08 17:49:50 +01:00
|
|
|
#[cfg(not(feature = "internals"))]
|
2021-01-08 17:24:55 +01:00
|
|
|
#[inline(always)]
|
2021-01-08 17:49:50 +01:00
|
|
|
pub(crate) fn resolver(
|
2021-01-08 17:40:44 +01:00
|
|
|
&self,
|
|
|
|
) -> Option<Shared<crate::module::resolvers::StaticModuleResolver>> {
|
2021-01-08 17:24:55 +01:00
|
|
|
self.resolver.clone()
|
|
|
|
}
|
2021-01-11 16:09:33 +01:00
|
|
|
/// _(INTERNALS)_ Get the embedded [module resolver][crate::ModuleResolver].
|
2021-01-08 17:24:55 +01:00
|
|
|
/// Exported under the `internals` feature only.
|
2021-05-10 05:07:19 +02:00
|
|
|
///
|
|
|
|
/// Not available under `no_module`.
|
2021-01-08 17:40:44 +01:00
|
|
|
#[cfg(not(feature = "no_module"))]
|
2021-01-08 17:24:55 +01:00
|
|
|
#[cfg(feature = "internals")]
|
|
|
|
#[inline(always)]
|
2021-01-08 17:49:50 +01:00
|
|
|
pub fn resolver(&self) -> Option<Shared<crate::module::resolvers::StaticModuleResolver>> {
|
|
|
|
self.resolver.clone()
|
2021-01-08 17:24:55 +01:00
|
|
|
}
|
|
|
|
/// Set the embedded [module resolver][`ModuleResolver`].
|
2021-01-08 17:40:44 +01:00
|
|
|
#[cfg(not(feature = "no_module"))]
|
2021-01-08 17:24:55 +01:00
|
|
|
#[inline(always)]
|
|
|
|
pub(crate) fn set_resolver(
|
|
|
|
&mut self,
|
2021-01-08 17:40:44 +01:00
|
|
|
resolver: impl Into<Shared<crate::module::resolvers::StaticModuleResolver>>,
|
2021-01-08 17:24:55 +01:00
|
|
|
) -> &mut Self {
|
|
|
|
self.resolver = Some(resolver.into());
|
|
|
|
self
|
|
|
|
}
|
2020-11-20 09:52:28 +01:00
|
|
|
/// Clone the [`AST`]'s functions into a new [`AST`].
|
2020-10-28 15:18:44 +01:00
|
|
|
/// No statements are cloned.
|
2021-04-17 06:03:29 +02:00
|
|
|
///
|
2021-03-24 12:27:38 +01:00
|
|
|
/// Not available under `no_function`.
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
|
|
|
/// This operation is cheap because functions are shared.
|
2020-10-29 04:37:51 +01:00
|
|
|
#[cfg(not(feature = "no_function"))]
|
2020-10-28 15:18:44 +01:00
|
|
|
#[inline(always)]
|
|
|
|
pub fn clone_functions_only(&self) -> Self {
|
2020-11-17 05:23:53 +01:00
|
|
|
self.clone_functions_only_filtered(|_, _, _, _, _| true)
|
2020-10-28 15:18:44 +01:00
|
|
|
}
|
2020-11-20 09:52:28 +01:00
|
|
|
/// Clone the [`AST`]'s functions into a new [`AST`] based on a filter predicate.
|
2020-10-28 15:18:44 +01:00
|
|
|
/// No statements are cloned.
|
2021-04-17 06:03:29 +02:00
|
|
|
///
|
2021-03-24 12:27:38 +01:00
|
|
|
/// Not available under `no_function`.
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
|
|
|
/// This operation is cheap because functions are shared.
|
2020-10-29 04:37:51 +01:00
|
|
|
#[cfg(not(feature = "no_function"))]
|
2020-10-28 15:18:44 +01:00
|
|
|
#[inline(always)]
|
|
|
|
pub fn clone_functions_only_filtered(
|
|
|
|
&self,
|
2021-02-13 03:56:09 +01:00
|
|
|
filter: impl Fn(FnNamespace, FnAccess, bool, &str, usize) -> bool,
|
2020-10-28 15:18:44 +01:00
|
|
|
) -> Self {
|
|
|
|
let mut functions: Module = Default::default();
|
2021-02-13 03:56:09 +01:00
|
|
|
functions.merge_filtered(&self.functions, &filter);
|
2020-12-12 08:57:55 +01:00
|
|
|
Self {
|
2020-12-21 15:04:46 +01:00
|
|
|
source: self.source.clone(),
|
2021-03-10 06:32:09 +01:00
|
|
|
body: Default::default(),
|
2020-12-12 08:57:55 +01:00
|
|
|
functions: functions.into(),
|
2021-01-08 17:40:44 +01:00
|
|
|
#[cfg(not(feature = "no_module"))]
|
2021-01-08 17:24:55 +01:00
|
|
|
resolver: self.resolver.clone(),
|
2020-12-12 08:57:55 +01:00
|
|
|
}
|
2020-10-28 15:18:44 +01:00
|
|
|
}
|
2020-11-20 09:52:28 +01:00
|
|
|
/// Clone the [`AST`]'s script statements into a new [`AST`].
|
2020-10-28 15:18:44 +01:00
|
|
|
/// No functions are cloned.
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn clone_statements_only(&self) -> Self {
|
2020-12-12 08:57:55 +01:00
|
|
|
Self {
|
2020-12-21 15:04:46 +01:00
|
|
|
source: self.source.clone(),
|
2021-03-10 06:32:09 +01:00
|
|
|
body: self.body.clone(),
|
2020-12-12 08:57:55 +01:00
|
|
|
functions: Default::default(),
|
2021-01-08 17:40:44 +01:00
|
|
|
#[cfg(not(feature = "no_module"))]
|
2021-01-08 17:24:55 +01:00
|
|
|
resolver: self.resolver.clone(),
|
2020-12-12 08:57:55 +01:00
|
|
|
}
|
2020-10-28 15:18:44 +01:00
|
|
|
}
|
2021-03-24 12:27:38 +01:00
|
|
|
/// Merge two [`AST`] into one. Both [`AST`]'s are untouched and a new, merged,
|
|
|
|
/// version is returned.
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
2020-11-20 09:52:28 +01:00
|
|
|
/// Statements in the second [`AST`] are simply appended to the end of the first _without any processing_.
|
|
|
|
/// Thus, the return value of the first [`AST`] (if using expression-statement syntax) is buried.
|
|
|
|
/// Of course, if the first [`AST`] uses a `return` statement at the end, then
|
|
|
|
/// the second [`AST`] will essentially be dead code.
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
2020-11-20 09:52:28 +01:00
|
|
|
/// All script-defined functions in the second [`AST`] overwrite similarly-named functions
|
|
|
|
/// in the first [`AST`] with the same number of parameters.
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
|
|
|
/// ```
|
|
|
|
/// # fn main() -> Result<(), Box<rhai::EvalAltResult>> {
|
|
|
|
/// # #[cfg(not(feature = "no_function"))]
|
|
|
|
/// # {
|
|
|
|
/// use rhai::Engine;
|
|
|
|
///
|
|
|
|
/// let engine = Engine::new();
|
|
|
|
///
|
2021-06-05 09:26:43 +02:00
|
|
|
/// let ast1 = engine.compile("
|
|
|
|
/// fn foo(x) { 42 + x }
|
|
|
|
/// foo(1)
|
|
|
|
/// ")?;
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
|
|
|
/// let ast2 = engine.compile(r#"
|
2021-06-05 09:26:43 +02:00
|
|
|
/// fn foo(n) { `hello${n}` }
|
|
|
|
/// foo("!")
|
|
|
|
/// "#)?;
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
|
|
|
/// let ast = ast1.merge(&ast2); // Merge 'ast2' into 'ast1'
|
|
|
|
///
|
|
|
|
/// // Notice that using the '+' operator also works:
|
|
|
|
/// // let ast = &ast1 + &ast2;
|
|
|
|
///
|
|
|
|
/// // 'ast' is essentially:
|
|
|
|
/// //
|
2021-04-04 17:22:45 +02:00
|
|
|
/// // fn foo(n) { `hello${n}` } // <- definition of first 'foo' is overwritten
|
2020-10-28 15:18:44 +01:00
|
|
|
/// // foo(1) // <- notice this will be "hello1" instead of 43,
|
|
|
|
/// // // but it is no longer the return value
|
|
|
|
/// // foo("!") // returns "hello!"
|
|
|
|
///
|
|
|
|
/// // Evaluate it
|
|
|
|
/// assert_eq!(engine.eval_ast::<String>(&ast)?, "hello!");
|
|
|
|
/// # }
|
|
|
|
/// # Ok(())
|
|
|
|
/// # }
|
|
|
|
/// ```
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn merge(&self, other: &Self) -> Self {
|
2020-11-17 05:23:53 +01:00
|
|
|
self.merge_filtered(other, |_, _, _, _, _| true)
|
2020-10-28 15:18:44 +01:00
|
|
|
}
|
2020-11-20 09:52:28 +01:00
|
|
|
/// Combine one [`AST`] with another. The second [`AST`] is consumed.
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
2020-11-20 09:52:28 +01:00
|
|
|
/// Statements in the second [`AST`] are simply appended to the end of the first _without any processing_.
|
|
|
|
/// Thus, the return value of the first [`AST`] (if using expression-statement syntax) is buried.
|
|
|
|
/// Of course, if the first [`AST`] uses a `return` statement at the end, then
|
|
|
|
/// the second [`AST`] will essentially be dead code.
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
2020-11-20 09:52:28 +01:00
|
|
|
/// All script-defined functions in the second [`AST`] overwrite similarly-named functions
|
|
|
|
/// in the first [`AST`] with the same number of parameters.
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
|
|
|
/// ```
|
|
|
|
/// # fn main() -> Result<(), Box<rhai::EvalAltResult>> {
|
|
|
|
/// # #[cfg(not(feature = "no_function"))]
|
|
|
|
/// # {
|
|
|
|
/// use rhai::Engine;
|
|
|
|
///
|
|
|
|
/// let engine = Engine::new();
|
|
|
|
///
|
2021-06-05 09:26:43 +02:00
|
|
|
/// let mut ast1 = engine.compile("
|
|
|
|
/// fn foo(x) { 42 + x }
|
|
|
|
/// foo(1)
|
|
|
|
/// ")?;
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
|
|
|
/// let ast2 = engine.compile(r#"
|
2021-06-05 09:26:43 +02:00
|
|
|
/// fn foo(n) { `hello${n}` }
|
|
|
|
/// foo("!")
|
|
|
|
/// "#)?;
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
|
|
|
/// ast1.combine(ast2); // Combine 'ast2' into 'ast1'
|
|
|
|
///
|
|
|
|
/// // Notice that using the '+=' operator also works:
|
|
|
|
/// // ast1 += ast2;
|
|
|
|
///
|
|
|
|
/// // 'ast1' is essentially:
|
|
|
|
/// //
|
2021-04-04 17:22:45 +02:00
|
|
|
/// // fn foo(n) { `hello${n}` } // <- definition of first 'foo' is overwritten
|
2020-10-28 15:18:44 +01:00
|
|
|
/// // foo(1) // <- notice this will be "hello1" instead of 43,
|
|
|
|
/// // // but it is no longer the return value
|
|
|
|
/// // foo("!") // returns "hello!"
|
|
|
|
///
|
|
|
|
/// // Evaluate it
|
|
|
|
/// assert_eq!(engine.eval_ast::<String>(&ast1)?, "hello!");
|
|
|
|
/// # }
|
|
|
|
/// # Ok(())
|
|
|
|
/// # }
|
|
|
|
/// ```
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn combine(&mut self, other: Self) -> &mut Self {
|
2020-11-17 05:23:53 +01:00
|
|
|
self.combine_filtered(other, |_, _, _, _, _| true)
|
2020-10-28 15:18:44 +01:00
|
|
|
}
|
2020-11-20 09:52:28 +01:00
|
|
|
/// Merge two [`AST`] into one. Both [`AST`]'s are untouched and a new, merged, version
|
2020-10-28 15:18:44 +01:00
|
|
|
/// is returned.
|
|
|
|
///
|
2020-11-20 09:52:28 +01:00
|
|
|
/// Statements in the second [`AST`] are simply appended to the end of the first _without any processing_.
|
|
|
|
/// Thus, the return value of the first [`AST`] (if using expression-statement syntax) is buried.
|
|
|
|
/// Of course, if the first [`AST`] uses a `return` statement at the end, then
|
|
|
|
/// the second [`AST`] will essentially be dead code.
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
2020-11-20 09:52:28 +01:00
|
|
|
/// All script-defined functions in the second [`AST`] are first selected based on a filter
|
|
|
|
/// predicate, then overwrite similarly-named functions in the first [`AST`] with the
|
2020-10-28 15:18:44 +01:00
|
|
|
/// same number of parameters.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
|
|
|
/// ```
|
|
|
|
/// # fn main() -> Result<(), Box<rhai::EvalAltResult>> {
|
|
|
|
/// # #[cfg(not(feature = "no_function"))]
|
|
|
|
/// # {
|
|
|
|
/// use rhai::Engine;
|
|
|
|
///
|
|
|
|
/// let engine = Engine::new();
|
|
|
|
///
|
2021-06-05 09:26:43 +02:00
|
|
|
/// let ast1 = engine.compile("
|
|
|
|
/// fn foo(x) { 42 + x }
|
|
|
|
/// foo(1)
|
|
|
|
/// ")?;
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
|
|
|
/// let ast2 = engine.compile(r#"
|
2021-06-05 09:26:43 +02:00
|
|
|
/// fn foo(n) { `hello${n}` }
|
|
|
|
/// fn error() { 0 }
|
|
|
|
/// foo("!")
|
|
|
|
/// "#)?;
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
|
|
|
/// // Merge 'ast2', picking only 'error()' but not 'foo(_)', into 'ast1'
|
2020-11-17 05:23:53 +01:00
|
|
|
/// let ast = ast1.merge_filtered(&ast2, |_, _, script, name, params|
|
|
|
|
/// script && name == "error" && params == 0);
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
|
|
|
/// // 'ast' is essentially:
|
|
|
|
/// //
|
|
|
|
/// // fn foo(n) { 42 + n } // <- definition of 'ast1::foo' is not overwritten
|
|
|
|
/// // // because 'ast2::foo' is filtered away
|
|
|
|
/// // foo(1) // <- notice this will be 43 instead of "hello1",
|
|
|
|
/// // // but it is no longer the return value
|
|
|
|
/// // fn error() { 0 } // <- this function passes the filter and is merged
|
|
|
|
/// // foo("!") // <- returns "42!"
|
|
|
|
///
|
|
|
|
/// // Evaluate it
|
|
|
|
/// assert_eq!(engine.eval_ast::<String>(&ast)?, "42!");
|
|
|
|
/// # }
|
|
|
|
/// # Ok(())
|
|
|
|
/// # }
|
|
|
|
/// ```
|
|
|
|
#[inline]
|
|
|
|
pub fn merge_filtered(
|
|
|
|
&self,
|
|
|
|
other: &Self,
|
2021-02-13 03:56:09 +01:00
|
|
|
filter: impl Fn(FnNamespace, FnAccess, bool, &str, usize) -> bool,
|
2020-10-28 15:18:44 +01:00
|
|
|
) -> Self {
|
2020-12-12 08:57:55 +01:00
|
|
|
let Self {
|
2021-03-10 06:32:09 +01:00
|
|
|
body, functions, ..
|
2020-12-12 08:57:55 +01:00
|
|
|
} = self;
|
2020-10-28 15:18:44 +01:00
|
|
|
|
2021-03-10 06:32:09 +01:00
|
|
|
let merged = match (body.is_empty(), other.body.is_empty()) {
|
2020-10-28 15:18:44 +01:00
|
|
|
(false, false) => {
|
2021-03-10 06:32:09 +01:00
|
|
|
let mut body = body.clone();
|
2021-04-16 06:04:33 +02:00
|
|
|
body.0.extend(other.body.0.iter().cloned());
|
2021-03-10 06:32:09 +01:00
|
|
|
body
|
2020-10-28 15:18:44 +01:00
|
|
|
}
|
2021-03-10 06:32:09 +01:00
|
|
|
(false, true) => body.clone(),
|
|
|
|
(true, false) => other.body.clone(),
|
|
|
|
(true, true) => Default::default(),
|
2020-10-28 15:18:44 +01:00
|
|
|
};
|
|
|
|
|
2021-01-04 04:58:24 +01:00
|
|
|
let source = other.source.clone().or_else(|| self.source.clone());
|
2020-12-21 15:04:46 +01:00
|
|
|
|
2020-11-19 03:41:08 +01:00
|
|
|
let mut functions = functions.as_ref().clone();
|
2021-02-13 03:56:09 +01:00
|
|
|
functions.merge_filtered(&other.functions, &filter);
|
2020-10-28 15:18:44 +01:00
|
|
|
|
2020-12-21 15:04:46 +01:00
|
|
|
if let Some(source) = source {
|
2021-04-16 06:04:33 +02:00
|
|
|
Self::new_with_source(merged.0, functions, source)
|
2020-12-21 15:04:46 +01:00
|
|
|
} else {
|
2021-04-16 06:04:33 +02:00
|
|
|
Self::new(merged.0, functions)
|
2020-12-21 15:04:46 +01:00
|
|
|
}
|
2020-10-28 15:18:44 +01:00
|
|
|
}
|
2020-11-20 09:52:28 +01:00
|
|
|
/// Combine one [`AST`] with another. The second [`AST`] is consumed.
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
2020-11-20 09:52:28 +01:00
|
|
|
/// Statements in the second [`AST`] are simply appended to the end of the first _without any processing_.
|
|
|
|
/// Thus, the return value of the first [`AST`] (if using expression-statement syntax) is buried.
|
|
|
|
/// Of course, if the first [`AST`] uses a `return` statement at the end, then
|
|
|
|
/// the second [`AST`] will essentially be dead code.
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
2020-11-20 09:52:28 +01:00
|
|
|
/// All script-defined functions in the second [`AST`] are first selected based on a filter
|
|
|
|
/// predicate, then overwrite similarly-named functions in the first [`AST`] with the
|
2020-10-28 15:18:44 +01:00
|
|
|
/// same number of parameters.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
|
|
|
/// ```
|
|
|
|
/// # fn main() -> Result<(), Box<rhai::EvalAltResult>> {
|
|
|
|
/// # #[cfg(not(feature = "no_function"))]
|
|
|
|
/// # {
|
|
|
|
/// use rhai::Engine;
|
|
|
|
///
|
|
|
|
/// let engine = Engine::new();
|
|
|
|
///
|
2021-06-05 09:26:43 +02:00
|
|
|
/// let mut ast1 = engine.compile("
|
|
|
|
/// fn foo(x) { 42 + x }
|
|
|
|
/// foo(1)
|
|
|
|
/// ")?;
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
|
|
|
/// let ast2 = engine.compile(r#"
|
2021-06-05 09:26:43 +02:00
|
|
|
/// fn foo(n) { `hello${n}` }
|
|
|
|
/// fn error() { 0 }
|
|
|
|
/// foo("!")
|
|
|
|
/// "#)?;
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
|
|
|
/// // Combine 'ast2', picking only 'error()' but not 'foo(_)', into 'ast1'
|
2020-11-17 05:23:53 +01:00
|
|
|
/// ast1.combine_filtered(ast2, |_, _, script, name, params|
|
|
|
|
/// script && name == "error" && params == 0);
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
|
|
|
/// // 'ast1' is essentially:
|
|
|
|
/// //
|
|
|
|
/// // fn foo(n) { 42 + n } // <- definition of 'ast1::foo' is not overwritten
|
|
|
|
/// // // because 'ast2::foo' is filtered away
|
|
|
|
/// // foo(1) // <- notice this will be 43 instead of "hello1",
|
|
|
|
/// // // but it is no longer the return value
|
|
|
|
/// // fn error() { 0 } // <- this function passes the filter and is merged
|
|
|
|
/// // foo("!") // <- returns "42!"
|
|
|
|
///
|
|
|
|
/// // Evaluate it
|
|
|
|
/// assert_eq!(engine.eval_ast::<String>(&ast1)?, "42!");
|
|
|
|
/// # }
|
|
|
|
/// # Ok(())
|
|
|
|
/// # }
|
|
|
|
/// ```
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn combine_filtered(
|
|
|
|
&mut self,
|
|
|
|
other: Self,
|
2021-02-13 03:56:09 +01:00
|
|
|
filter: impl Fn(FnNamespace, FnAccess, bool, &str, usize) -> bool,
|
2020-10-28 15:18:44 +01:00
|
|
|
) -> &mut Self {
|
2021-04-16 06:04:33 +02:00
|
|
|
self.body.0.extend(other.body.0.into_iter());
|
2021-03-10 06:32:09 +01:00
|
|
|
|
2020-12-12 08:57:55 +01:00
|
|
|
if !other.functions.is_empty() {
|
2021-02-13 03:56:09 +01:00
|
|
|
shared_make_mut(&mut self.functions).merge_filtered(&other.functions, &filter);
|
2020-11-19 03:41:08 +01:00
|
|
|
}
|
2020-10-28 15:18:44 +01:00
|
|
|
self
|
|
|
|
}
|
|
|
|
/// Filter out the functions, retaining only some based on a filter predicate.
|
|
|
|
///
|
2021-01-11 16:09:33 +01:00
|
|
|
/// Not available under `no_function`.
|
2021-01-02 16:30:10 +01:00
|
|
|
///
|
2020-10-28 15:18:44 +01:00
|
|
|
/// # Example
|
|
|
|
///
|
|
|
|
/// ```
|
|
|
|
/// # fn main() -> Result<(), Box<rhai::EvalAltResult>> {
|
|
|
|
/// # #[cfg(not(feature = "no_function"))]
|
|
|
|
/// # {
|
|
|
|
/// use rhai::Engine;
|
|
|
|
///
|
|
|
|
/// let engine = Engine::new();
|
|
|
|
///
|
|
|
|
/// let mut ast = engine.compile(r#"
|
2021-06-05 09:26:43 +02:00
|
|
|
/// fn foo(n) { n + 1 }
|
|
|
|
/// fn bar() { print("hello"); }
|
|
|
|
/// "#)?;
|
2020-10-28 15:18:44 +01:00
|
|
|
///
|
|
|
|
/// // Remove all functions except 'foo(_)'
|
2020-11-17 05:23:53 +01:00
|
|
|
/// ast.retain_functions(|_, _, name, params| name == "foo" && params == 1);
|
2020-10-28 15:18:44 +01:00
|
|
|
/// # }
|
|
|
|
/// # Ok(())
|
|
|
|
/// # }
|
|
|
|
/// ```
|
|
|
|
#[cfg(not(feature = "no_function"))]
|
|
|
|
#[inline(always)]
|
2020-11-17 05:23:53 +01:00
|
|
|
pub fn retain_functions(
|
|
|
|
&mut self,
|
2021-02-13 03:56:09 +01:00
|
|
|
filter: impl Fn(FnNamespace, FnAccess, &str, usize) -> bool,
|
2020-11-19 03:41:08 +01:00
|
|
|
) -> &mut Self {
|
2020-12-12 08:57:55 +01:00
|
|
|
if !self.functions.is_empty() {
|
|
|
|
shared_make_mut(&mut self.functions).retain_script_functions(filter);
|
2020-11-19 03:41:08 +01:00
|
|
|
}
|
|
|
|
self
|
2020-10-28 15:18:44 +01:00
|
|
|
}
|
2020-12-18 09:31:44 +01:00
|
|
|
/// Iterate through all function definitions.
|
2021-01-02 16:30:10 +01:00
|
|
|
///
|
2021-04-17 06:03:29 +02:00
|
|
|
/// Not available under `no_function`.
|
2020-10-28 15:18:44 +01:00
|
|
|
#[cfg(not(feature = "no_function"))]
|
2021-01-08 17:40:44 +01:00
|
|
|
#[cfg(not(feature = "no_module"))]
|
2020-10-28 15:18:44 +01:00
|
|
|
#[inline(always)]
|
2021-01-08 17:24:55 +01:00
|
|
|
pub(crate) fn iter_fn_def(&self) -> impl Iterator<Item = &ScriptFnDef> {
|
|
|
|
self.functions
|
|
|
|
.iter_script_fn()
|
2021-03-12 07:11:08 +01:00
|
|
|
.map(|(_, _, _, _, fn_def)| fn_def.as_ref())
|
2021-01-08 17:24:55 +01:00
|
|
|
}
|
|
|
|
/// Iterate through all function definitions.
|
|
|
|
///
|
2021-01-11 16:09:33 +01:00
|
|
|
/// Not available under `no_function`.
|
2021-01-08 17:24:55 +01:00
|
|
|
#[cfg(not(feature = "no_function"))]
|
|
|
|
#[inline(always)]
|
2020-12-12 11:44:28 +01:00
|
|
|
pub fn iter_functions<'a>(&'a self) -> impl Iterator<Item = ScriptFnMetadata> + 'a {
|
2020-12-12 09:31:13 +01:00
|
|
|
self.functions
|
|
|
|
.iter_script_fn()
|
2021-03-12 07:11:08 +01:00
|
|
|
.map(|(_, _, _, _, fn_def)| fn_def.as_ref().into())
|
2020-10-28 15:18:44 +01:00
|
|
|
}
|
2020-11-20 09:52:28 +01:00
|
|
|
/// Clear all function definitions in the [`AST`].
|
2021-01-02 16:30:10 +01:00
|
|
|
///
|
2021-01-11 16:09:33 +01:00
|
|
|
/// Not available under `no_function`.
|
2020-10-28 15:18:44 +01:00
|
|
|
#[cfg(not(feature = "no_function"))]
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn clear_functions(&mut self) {
|
2020-12-12 08:57:55 +01:00
|
|
|
self.functions = Default::default();
|
2020-10-28 15:18:44 +01:00
|
|
|
}
|
2020-11-20 09:52:28 +01:00
|
|
|
/// Clear all statements in the [`AST`], leaving only function definitions.
|
2020-10-28 15:18:44 +01:00
|
|
|
#[inline(always)]
|
|
|
|
pub fn clear_statements(&mut self) {
|
2021-03-10 06:32:09 +01:00
|
|
|
self.body = Default::default();
|
2020-10-28 15:18:44 +01:00
|
|
|
}
|
2021-01-09 09:52:22 +01:00
|
|
|
/// Recursively walk the [`AST`], including function bodies (if any).
|
2021-03-11 14:55:55 +01:00
|
|
|
/// Return `false` from the callback to terminate the walk.
|
2021-01-09 09:52:22 +01:00
|
|
|
#[cfg(not(feature = "internals"))]
|
2021-01-09 10:06:01 +01:00
|
|
|
#[cfg(not(feature = "no_module"))]
|
2021-01-09 09:52:22 +01:00
|
|
|
#[inline(always)]
|
2021-03-11 14:55:55 +01:00
|
|
|
pub(crate) fn walk(&self, on_node: &mut impl FnMut(&[ASTNode]) -> bool) -> bool {
|
|
|
|
let path = &mut Default::default();
|
|
|
|
|
|
|
|
for stmt in self.statements() {
|
|
|
|
if !stmt.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#[cfg(not(feature = "no_function"))]
|
2021-04-16 06:04:33 +02:00
|
|
|
for stmt in self.iter_fn_def().flat_map(|f| f.body.0.iter()) {
|
2021-03-11 14:55:55 +01:00
|
|
|
if !stmt.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
true
|
2021-01-09 09:52:22 +01:00
|
|
|
}
|
|
|
|
/// _(INTERNALS)_ Recursively walk the [`AST`], including function bodies (if any).
|
2021-03-11 14:55:55 +01:00
|
|
|
/// Return `false` from the callback to terminate the walk.
|
2021-01-09 09:52:22 +01:00
|
|
|
/// Exported under the `internals` feature only.
|
|
|
|
#[cfg(feature = "internals")]
|
|
|
|
#[inline(always)]
|
2021-03-11 15:27:35 +01:00
|
|
|
pub fn walk(&self, on_node: &mut impl FnMut(&[ASTNode]) -> bool) -> bool {
|
|
|
|
let path = &mut Default::default();
|
|
|
|
|
|
|
|
for stmt in self.statements() {
|
|
|
|
if !stmt.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#[cfg(not(feature = "no_function"))]
|
2021-04-16 07:15:11 +02:00
|
|
|
for stmt in self.iter_fn_def().flat_map(|f| f.body.0.iter()) {
|
2021-03-11 15:27:35 +01:00
|
|
|
if !stmt.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
true
|
2021-01-09 09:52:22 +01:00
|
|
|
}
|
2020-10-28 15:18:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<A: AsRef<AST>> Add<A> for &AST {
|
|
|
|
type Output = AST;
|
|
|
|
|
|
|
|
#[inline(always)]
|
|
|
|
fn add(self, rhs: A) -> Self::Output {
|
|
|
|
self.merge(rhs.as_ref())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<A: Into<AST>> AddAssign<A> for AST {
|
|
|
|
#[inline(always)]
|
|
|
|
fn add_assign(&mut self, rhs: A) {
|
|
|
|
self.combine(rhs.into());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl AsRef<[Stmt]> for AST {
|
|
|
|
#[inline(always)]
|
|
|
|
fn as_ref(&self) -> &[Stmt] {
|
|
|
|
self.statements()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl AsRef<Module> for AST {
|
|
|
|
#[inline(always)]
|
|
|
|
fn as_ref(&self) -> &Module {
|
|
|
|
self.lib()
|
|
|
|
}
|
|
|
|
}
|
2020-10-29 04:37:51 +01:00
|
|
|
|
2021-03-29 05:36:02 +02:00
|
|
|
/// _(INTERNALS)_ An identifier containing a name and a [position][Position].
|
2020-11-25 02:36:06 +01:00
|
|
|
/// Exported under the `internals` feature only.
|
|
|
|
///
|
2021-01-16 07:46:03 +01:00
|
|
|
/// # Volatile Data Structure
|
2020-11-25 02:36:06 +01:00
|
|
|
///
|
|
|
|
/// This type is volatile and may change.
|
2020-12-28 02:49:54 +01:00
|
|
|
#[derive(Clone, Eq, PartialEq, Hash)]
|
2020-12-22 09:45:56 +01:00
|
|
|
pub struct Ident {
|
2020-12-22 15:35:25 +01:00
|
|
|
/// Identifier name.
|
2021-03-29 05:36:02 +02:00
|
|
|
pub name: Identifier,
|
2020-12-22 15:35:25 +01:00
|
|
|
/// Declaration position.
|
2020-10-29 04:37:51 +01:00
|
|
|
pub pos: Position,
|
|
|
|
}
|
|
|
|
|
2020-12-28 02:49:54 +01:00
|
|
|
impl fmt::Debug for Ident {
|
2020-12-29 03:41:20 +01:00
|
|
|
#[inline(always)]
|
2020-12-28 02:49:54 +01:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
2021-04-22 17:02:25 +02:00
|
|
|
write!(f, "{:?}", self.name)?;
|
2021-04-23 08:24:53 +02:00
|
|
|
self.pos.debug_print(f)
|
2020-12-28 02:49:54 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-20 09:52:28 +01:00
|
|
|
/// _(INTERNALS)_ A type encapsulating the mode of a `return`/`throw` statement.
|
2020-10-29 04:37:51 +01:00
|
|
|
/// Exported under the `internals` feature only.
|
|
|
|
///
|
2021-01-16 07:46:03 +01:00
|
|
|
/// # Volatile Data Structure
|
2020-10-29 04:37:51 +01:00
|
|
|
///
|
|
|
|
/// This type is volatile and may change.
|
|
|
|
#[derive(Debug, Eq, PartialEq, Clone, Copy, Hash)]
|
|
|
|
pub enum ReturnType {
|
|
|
|
/// `return` statement.
|
|
|
|
Return,
|
|
|
|
/// `throw` statement.
|
|
|
|
Exception,
|
|
|
|
}
|
|
|
|
|
2021-01-09 09:52:22 +01:00
|
|
|
/// _(INTERNALS)_ An [`AST`] node, consisting of either an [`Expr`] or a [`Stmt`].
|
|
|
|
/// Exported under the `internals` feature only.
|
|
|
|
///
|
2021-01-16 07:46:03 +01:00
|
|
|
/// # Volatile Data Structure
|
2021-01-09 09:52:22 +01:00
|
|
|
///
|
|
|
|
/// This type is volatile and may change.
|
|
|
|
#[derive(Debug, Clone, Hash)]
|
|
|
|
pub enum ASTNode<'a> {
|
|
|
|
Stmt(&'a Stmt),
|
|
|
|
Expr(&'a Expr),
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> From<&'a Stmt> for ASTNode<'a> {
|
|
|
|
fn from(stmt: &'a Stmt) -> Self {
|
|
|
|
Self::Stmt(stmt)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> From<&'a Expr> for ASTNode<'a> {
|
|
|
|
fn from(expr: &'a Expr) -> Self {
|
|
|
|
Self::Expr(expr)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-10 05:27:10 +01:00
|
|
|
/// _(INTERNALS)_ A statements block.
|
|
|
|
/// Exported under the `internals` feature only.
|
|
|
|
///
|
|
|
|
/// # Volatile Data Structure
|
|
|
|
///
|
|
|
|
/// This type is volatile and may change.
|
2021-03-10 06:32:09 +01:00
|
|
|
#[derive(Clone, Hash, Default)]
|
2021-04-16 07:15:11 +02:00
|
|
|
pub struct StmtBlock(StaticVec<Stmt>, Position);
|
2021-03-10 05:27:10 +01:00
|
|
|
|
|
|
|
impl StmtBlock {
|
2021-04-16 07:15:11 +02:00
|
|
|
/// Create a new [`StmtBlock`].
|
2021-06-06 06:17:04 +02:00
|
|
|
pub fn new(mut statements: StaticVec<Stmt>, pos: Position) -> Self {
|
2021-05-05 12:38:52 +02:00
|
|
|
statements.shrink_to_fit();
|
|
|
|
Self(statements, pos)
|
2021-04-16 07:15:11 +02:00
|
|
|
}
|
2021-03-10 05:27:10 +01:00
|
|
|
/// Is this statements block empty?
|
2021-03-10 06:32:09 +01:00
|
|
|
#[inline(always)]
|
2021-03-10 05:27:10 +01:00
|
|
|
pub fn is_empty(&self) -> bool {
|
2021-04-16 06:04:33 +02:00
|
|
|
self.0.is_empty()
|
2021-03-10 05:27:10 +01:00
|
|
|
}
|
|
|
|
/// Number of statements in this statements block.
|
2021-03-10 06:32:09 +01:00
|
|
|
#[inline(always)]
|
2021-03-10 05:27:10 +01:00
|
|
|
pub fn len(&self) -> usize {
|
2021-04-16 06:04:33 +02:00
|
|
|
self.0.len()
|
2021-03-10 05:27:10 +01:00
|
|
|
}
|
2021-04-16 07:15:11 +02:00
|
|
|
/// Get the position of this statements block.
|
|
|
|
pub fn position(&self) -> Position {
|
|
|
|
self.1
|
|
|
|
}
|
|
|
|
/// Get the statements of this statements block.
|
|
|
|
pub fn statements(&mut self) -> &mut StaticVec<Stmt> {
|
|
|
|
&mut self.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Deref for StmtBlock {
|
|
|
|
type Target = StaticVec<Stmt>;
|
|
|
|
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
|
|
&self.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl DerefMut for StmtBlock {
|
|
|
|
fn deref_mut(&mut self) -> &mut Self::Target {
|
|
|
|
&mut self.0
|
|
|
|
}
|
2021-03-10 05:27:10 +01:00
|
|
|
}
|
|
|
|
|
2021-03-10 06:32:09 +01:00
|
|
|
impl fmt::Debug for StmtBlock {
|
|
|
|
#[inline(always)]
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
2021-05-22 13:14:24 +02:00
|
|
|
f.write_str("Block")?;
|
2021-04-16 06:04:33 +02:00
|
|
|
fmt::Debug::fmt(&self.0, f)?;
|
2021-04-23 08:24:53 +02:00
|
|
|
self.1.debug_print(f)
|
2021-03-10 06:32:09 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-16 07:15:11 +02:00
|
|
|
impl From<StmtBlock> for Stmt {
|
|
|
|
fn from(block: StmtBlock) -> Self {
|
|
|
|
let block_pos = block.position();
|
2021-05-05 12:38:52 +02:00
|
|
|
Self::Block(block.0.into_boxed_slice(), block_pos)
|
2021-04-16 07:15:11 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-20 09:52:28 +01:00
|
|
|
/// _(INTERNALS)_ A statement.
|
2020-10-29 04:37:51 +01:00
|
|
|
/// Exported under the `internals` feature only.
|
|
|
|
///
|
2021-01-16 07:46:03 +01:00
|
|
|
/// # Volatile Data Structure
|
2020-11-06 09:27:40 +01:00
|
|
|
///
|
|
|
|
/// This type is volatile and may change.
|
2021-01-04 04:58:24 +01:00
|
|
|
#[derive(Debug, Clone, Hash)]
|
2020-10-29 04:37:51 +01:00
|
|
|
pub enum Stmt {
|
|
|
|
/// No-op.
|
|
|
|
Noop(Position),
|
2020-11-20 09:52:28 +01:00
|
|
|
/// `if` expr `{` stmt `}` `else` `{` stmt `}`
|
2021-03-10 05:27:10 +01:00
|
|
|
If(Expr, Box<(StmtBlock, StmtBlock)>, Position),
|
2021-04-16 06:04:33 +02:00
|
|
|
/// `switch` expr `if` condition `{` literal or _ `=>` stmt `,` ... `}`
|
2021-03-30 17:55:29 +02:00
|
|
|
Switch(
|
|
|
|
Expr,
|
2021-04-16 06:04:33 +02:00
|
|
|
Box<(BTreeMap<u64, Box<(Option<Expr>, StmtBlock)>>, StmtBlock)>,
|
2021-03-30 17:55:29 +02:00
|
|
|
Position,
|
|
|
|
),
|
2020-11-20 09:52:28 +01:00
|
|
|
/// `while` expr `{` stmt `}`
|
2021-03-10 05:27:10 +01:00
|
|
|
While(Expr, Box<StmtBlock>, Position),
|
2020-11-20 15:23:37 +01:00
|
|
|
/// `do` `{` stmt `}` `while`|`until` expr
|
2021-03-10 05:27:10 +01:00
|
|
|
Do(Box<StmtBlock>, Expr, bool, Position),
|
2021-06-07 05:01:16 +02:00
|
|
|
/// `for` `(` id `,` counter `)` `in` expr `{` stmt `}`
|
|
|
|
For(Expr, Box<(Ident, Option<Ident>, StmtBlock)>, Position),
|
2020-11-20 09:52:28 +01:00
|
|
|
/// \[`export`\] `let` id `=` expr
|
2021-03-29 05:36:02 +02:00
|
|
|
Let(Expr, Box<Ident>, bool, Position),
|
2020-11-20 09:52:28 +01:00
|
|
|
/// \[`export`\] `const` id `=` expr
|
2021-03-29 05:36:02 +02:00
|
|
|
Const(Expr, Box<Ident>, bool, Position),
|
2020-11-20 09:52:28 +01:00
|
|
|
/// expr op`=` expr
|
2021-06-12 04:26:50 +02:00
|
|
|
Assignment(Box<(Expr, Option<OpAssignment<'static>>, Expr)>, Position),
|
2021-04-21 12:16:24 +02:00
|
|
|
/// func `(` expr `,` ... `)`
|
|
|
|
///
|
|
|
|
/// Note - this is a duplicate of [`Expr::FnCall`] to cover the very common pattern of a single
|
|
|
|
/// function call forming one statement.
|
|
|
|
FnCall(Box<FnCallExpr>, Position),
|
2020-11-20 09:52:28 +01:00
|
|
|
/// `{` stmt`;` ... `}`
|
2021-05-05 12:38:52 +02:00
|
|
|
Block(Box<[Stmt]>, Position),
|
2020-11-20 09:52:28 +01:00
|
|
|
/// `try` `{` stmt; ... `}` `catch` `(` var `)` `{` stmt; ... `}`
|
2021-03-10 05:27:10 +01:00
|
|
|
TryCatch(
|
|
|
|
Box<(StmtBlock, Option<Ident>, StmtBlock)>,
|
|
|
|
Position,
|
|
|
|
Position,
|
|
|
|
),
|
2020-11-20 09:52:28 +01:00
|
|
|
/// [expression][Expr]
|
2020-10-29 04:37:51 +01:00
|
|
|
Expr(Expr),
|
2020-11-20 09:52:28 +01:00
|
|
|
/// `continue`
|
2020-10-29 04:37:51 +01:00
|
|
|
Continue(Position),
|
2020-11-20 09:52:28 +01:00
|
|
|
/// `break`
|
2020-10-29 04:37:51 +01:00
|
|
|
Break(Position),
|
2020-11-20 09:52:28 +01:00
|
|
|
/// `return`/`throw`
|
2021-03-09 16:30:48 +01:00
|
|
|
Return(ReturnType, Option<Expr>, Position),
|
2020-11-20 09:52:28 +01:00
|
|
|
/// `import` expr `as` var
|
2021-05-10 05:07:19 +02:00
|
|
|
///
|
|
|
|
/// Not available under `no_module`.
|
2020-10-29 04:37:51 +01:00
|
|
|
#[cfg(not(feature = "no_module"))]
|
2021-03-29 05:36:02 +02:00
|
|
|
Import(Expr, Option<Box<Ident>>, Position),
|
2020-11-20 09:52:28 +01:00
|
|
|
/// `export` var `as` var `,` ...
|
2021-05-10 05:07:19 +02:00
|
|
|
///
|
|
|
|
/// Not available under `no_module`.
|
2020-10-29 04:37:51 +01:00
|
|
|
#[cfg(not(feature = "no_module"))]
|
2021-05-05 12:38:52 +02:00
|
|
|
Export(Box<[(Ident, Option<Ident>)]>, Position),
|
2020-10-29 04:37:51 +01:00
|
|
|
/// Convert a variable to shared.
|
2021-05-10 05:07:19 +02:00
|
|
|
///
|
|
|
|
/// Not available under `no_closure`.
|
2020-10-29 04:37:51 +01:00
|
|
|
#[cfg(not(feature = "no_closure"))]
|
2021-03-30 17:55:29 +02:00
|
|
|
Share(Identifier),
|
2020-10-29 04:37:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for Stmt {
|
|
|
|
#[inline(always)]
|
|
|
|
fn default() -> Self {
|
2020-11-20 09:52:28 +01:00
|
|
|
Self::Noop(Position::NONE)
|
2020-10-29 04:37:51 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-10 05:27:10 +01:00
|
|
|
impl From<Stmt> for StmtBlock {
|
2021-03-10 15:12:48 +01:00
|
|
|
#[inline(always)]
|
2021-03-10 05:27:10 +01:00
|
|
|
fn from(stmt: Stmt) -> Self {
|
|
|
|
match stmt {
|
2021-06-06 06:17:04 +02:00
|
|
|
Stmt::Block(mut block, pos) => Self(block.iter_mut().map(mem::take).collect(), pos),
|
2021-04-16 06:04:33 +02:00
|
|
|
Stmt::Noop(pos) => Self(Default::default(), pos),
|
2021-03-10 15:12:48 +01:00
|
|
|
_ => {
|
|
|
|
let pos = stmt.position();
|
2021-04-16 06:04:33 +02:00
|
|
|
Self(vec![stmt].into(), pos)
|
2021-03-10 15:12:48 +01:00
|
|
|
}
|
2021-03-10 05:27:10 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-29 04:37:51 +01:00
|
|
|
impl Stmt {
|
2020-11-20 09:52:28 +01:00
|
|
|
/// Is this statement [`Noop`][Stmt::Noop]?
|
2020-12-29 03:41:20 +01:00
|
|
|
#[inline(always)]
|
2020-10-29 04:37:51 +01:00
|
|
|
pub fn is_noop(&self) -> bool {
|
|
|
|
match self {
|
|
|
|
Self::Noop(_) => true,
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
2020-12-26 16:21:16 +01:00
|
|
|
/// Get the [position][Position] of this statement.
|
2020-10-29 04:37:51 +01:00
|
|
|
pub fn position(&self) -> Position {
|
|
|
|
match self {
|
|
|
|
Self::Noop(pos)
|
|
|
|
| Self::Continue(pos)
|
|
|
|
| Self::Break(pos)
|
|
|
|
| Self::Block(_, pos)
|
|
|
|
| Self::Assignment(_, pos)
|
2021-04-21 12:16:24 +02:00
|
|
|
| Self::FnCall(_, pos)
|
2020-11-14 15:55:23 +01:00
|
|
|
| Self::If(_, _, pos)
|
2020-11-14 16:43:36 +01:00
|
|
|
| Self::Switch(_, _, pos)
|
2020-10-29 04:37:51 +01:00
|
|
|
| Self::While(_, _, pos)
|
2020-11-20 15:23:37 +01:00
|
|
|
| Self::Do(_, _, _, pos)
|
2021-03-09 16:48:40 +01:00
|
|
|
| Self::For(_, _, pos)
|
2021-03-09 16:30:48 +01:00
|
|
|
| Self::Return(_, _, pos)
|
2021-03-10 05:27:10 +01:00
|
|
|
| Self::Let(_, _, _, pos)
|
|
|
|
| Self::Const(_, _, _, pos)
|
2020-11-09 05:21:11 +01:00
|
|
|
| Self::TryCatch(_, pos, _) => *pos,
|
2020-10-29 04:37:51 +01:00
|
|
|
|
|
|
|
Self::Expr(x) => x.position(),
|
|
|
|
|
|
|
|
#[cfg(not(feature = "no_module"))]
|
|
|
|
Self::Import(_, _, pos) => *pos,
|
|
|
|
#[cfg(not(feature = "no_module"))]
|
|
|
|
Self::Export(_, pos) => *pos,
|
|
|
|
|
|
|
|
#[cfg(not(feature = "no_closure"))]
|
2021-03-30 17:55:29 +02:00
|
|
|
Self::Share(_) => Position::NONE,
|
2020-10-29 04:37:51 +01:00
|
|
|
}
|
|
|
|
}
|
2020-12-26 16:21:16 +01:00
|
|
|
/// Override the [position][Position] of this statement.
|
2020-10-29 04:37:51 +01:00
|
|
|
pub fn set_position(&mut self, new_pos: Position) -> &mut Self {
|
|
|
|
match self {
|
|
|
|
Self::Noop(pos)
|
|
|
|
| Self::Continue(pos)
|
|
|
|
| Self::Break(pos)
|
|
|
|
| Self::Block(_, pos)
|
|
|
|
| Self::Assignment(_, pos)
|
2021-04-21 12:16:24 +02:00
|
|
|
| Self::FnCall(_, pos)
|
2020-11-14 15:55:23 +01:00
|
|
|
| Self::If(_, _, pos)
|
2020-11-14 16:43:36 +01:00
|
|
|
| Self::Switch(_, _, pos)
|
2020-10-29 04:37:51 +01:00
|
|
|
| Self::While(_, _, pos)
|
2020-11-20 15:23:37 +01:00
|
|
|
| Self::Do(_, _, _, pos)
|
2021-03-09 16:48:40 +01:00
|
|
|
| Self::For(_, _, pos)
|
2021-03-09 16:30:48 +01:00
|
|
|
| Self::Return(_, _, pos)
|
2021-03-10 05:27:10 +01:00
|
|
|
| Self::Let(_, _, _, pos)
|
|
|
|
| Self::Const(_, _, _, pos)
|
2020-11-09 05:21:11 +01:00
|
|
|
| Self::TryCatch(_, pos, _) => *pos = new_pos,
|
2020-10-29 04:37:51 +01:00
|
|
|
|
|
|
|
Self::Expr(x) => {
|
|
|
|
x.set_position(new_pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(not(feature = "no_module"))]
|
|
|
|
Self::Import(_, _, pos) => *pos = new_pos,
|
|
|
|
#[cfg(not(feature = "no_module"))]
|
|
|
|
Self::Export(_, pos) => *pos = new_pos,
|
|
|
|
|
|
|
|
#[cfg(not(feature = "no_closure"))]
|
2021-03-30 17:55:29 +02:00
|
|
|
Self::Share(_) => (),
|
2020-10-29 04:37:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
self
|
|
|
|
}
|
2021-03-11 11:29:22 +01:00
|
|
|
/// Does this statement return a value?
|
|
|
|
pub fn returns_value(&self) -> bool {
|
|
|
|
match self {
|
2021-04-21 12:16:24 +02:00
|
|
|
Self::If(_, _, _)
|
|
|
|
| Self::Switch(_, _, _)
|
|
|
|
| Self::Block(_, _)
|
|
|
|
| Self::Expr(_)
|
|
|
|
| Self::FnCall(_, _) => true,
|
2021-03-11 11:29:22 +01:00
|
|
|
|
|
|
|
Self::Noop(_)
|
|
|
|
| Self::While(_, _, _)
|
|
|
|
| Self::Do(_, _, _, _)
|
|
|
|
| Self::For(_, _, _)
|
|
|
|
| Self::TryCatch(_, _, _) => false,
|
|
|
|
|
|
|
|
Self::Let(_, _, _, _)
|
|
|
|
| Self::Const(_, _, _, _)
|
|
|
|
| Self::Assignment(_, _)
|
|
|
|
| Self::Continue(_)
|
|
|
|
| Self::Break(_)
|
|
|
|
| Self::Return(_, _, _) => false,
|
|
|
|
|
|
|
|
#[cfg(not(feature = "no_module"))]
|
|
|
|
Self::Import(_, _, _) | Self::Export(_, _) => false,
|
|
|
|
|
|
|
|
#[cfg(not(feature = "no_closure"))]
|
|
|
|
Self::Share(_) => unreachable!("Stmt::Share should not be parsed"),
|
|
|
|
}
|
|
|
|
}
|
2020-10-29 04:37:51 +01:00
|
|
|
/// Is this statement self-terminated (i.e. no need for a semicolon terminator)?
|
|
|
|
pub fn is_self_terminated(&self) -> bool {
|
|
|
|
match self {
|
2020-11-14 15:55:23 +01:00
|
|
|
Self::If(_, _, _)
|
2020-11-14 16:43:36 +01:00
|
|
|
| Self::Switch(_, _, _)
|
2020-10-29 04:37:51 +01:00
|
|
|
| Self::While(_, _, _)
|
2021-03-09 16:48:40 +01:00
|
|
|
| Self::For(_, _, _)
|
2020-10-29 04:37:51 +01:00
|
|
|
| Self::Block(_, _)
|
2020-11-06 09:27:40 +01:00
|
|
|
| Self::TryCatch(_, _, _) => true,
|
2020-10-29 04:37:51 +01:00
|
|
|
|
|
|
|
// A No-op requires a semicolon in order to know it is an empty statement!
|
|
|
|
Self::Noop(_) => false,
|
|
|
|
|
2021-03-10 05:27:10 +01:00
|
|
|
Self::Let(_, _, _, _)
|
|
|
|
| Self::Const(_, _, _, _)
|
2020-10-29 04:37:51 +01:00
|
|
|
| Self::Assignment(_, _)
|
2021-04-21 12:16:24 +02:00
|
|
|
| Self::FnCall(_, _)
|
2020-10-29 04:37:51 +01:00
|
|
|
| Self::Expr(_)
|
2020-11-20 15:23:37 +01:00
|
|
|
| Self::Do(_, _, _, _)
|
2020-10-29 04:37:51 +01:00
|
|
|
| Self::Continue(_)
|
|
|
|
| Self::Break(_)
|
2020-11-14 15:55:23 +01:00
|
|
|
| Self::Return(_, _, _) => false,
|
2020-10-29 04:37:51 +01:00
|
|
|
|
|
|
|
#[cfg(not(feature = "no_module"))]
|
|
|
|
Self::Import(_, _, _) | Self::Export(_, _) => false,
|
|
|
|
|
|
|
|
#[cfg(not(feature = "no_closure"))]
|
2020-12-28 07:21:13 +01:00
|
|
|
Self::Share(_) => unreachable!("Stmt::Share should not be parsed"),
|
2020-10-29 04:37:51 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
/// Is this statement _pure_?
|
2020-11-25 02:36:06 +01:00
|
|
|
///
|
|
|
|
/// A pure statement has no side effects.
|
2020-10-29 04:37:51 +01:00
|
|
|
pub fn is_pure(&self) -> bool {
|
|
|
|
match self {
|
|
|
|
Self::Noop(_) => true,
|
|
|
|
Self::Expr(expr) => expr.is_pure(),
|
2021-03-10 05:27:10 +01:00
|
|
|
Self::If(condition, x, _) => {
|
|
|
|
condition.is_pure()
|
2021-04-16 06:04:33 +02:00
|
|
|
&& (x.0).0.iter().all(Stmt::is_pure)
|
|
|
|
&& (x.1).0.iter().all(Stmt::is_pure)
|
2021-03-10 05:27:10 +01:00
|
|
|
}
|
2020-11-14 16:43:36 +01:00
|
|
|
Self::Switch(expr, x, _) => {
|
|
|
|
expr.is_pure()
|
2021-04-16 06:04:33 +02:00
|
|
|
&& x.0.values().all(|block| {
|
|
|
|
block.0.as_ref().map(Expr::is_pure).unwrap_or(true)
|
|
|
|
&& (block.1).0.iter().all(Stmt::is_pure)
|
|
|
|
})
|
|
|
|
&& (x.1).0.iter().all(Stmt::is_pure)
|
2020-10-29 04:37:51 +01:00
|
|
|
}
|
2021-03-09 16:30:48 +01:00
|
|
|
Self::While(condition, block, _) | Self::Do(block, condition, _, _) => {
|
2021-04-16 06:04:33 +02:00
|
|
|
condition.is_pure() && block.0.iter().all(Stmt::is_pure)
|
2021-03-10 05:27:10 +01:00
|
|
|
}
|
2021-06-07 05:01:16 +02:00
|
|
|
Self::For(iterable, x, _) => iterable.is_pure() && (x.2).0.iter().all(Stmt::is_pure),
|
2021-04-21 12:16:24 +02:00
|
|
|
Self::Let(_, _, _, _)
|
|
|
|
| Self::Const(_, _, _, _)
|
|
|
|
| Self::Assignment(_, _)
|
|
|
|
| Self::FnCall(_, _) => false,
|
2020-10-29 04:37:51 +01:00
|
|
|
Self::Block(block, _) => block.iter().all(|stmt| stmt.is_pure()),
|
2020-11-14 15:55:23 +01:00
|
|
|
Self::Continue(_) | Self::Break(_) | Self::Return(_, _, _) => false,
|
2021-03-10 05:27:10 +01:00
|
|
|
Self::TryCatch(x, _, _) => {
|
2021-04-16 06:04:33 +02:00
|
|
|
(x.0).0.iter().all(Stmt::is_pure) && (x.2).0.iter().all(Stmt::is_pure)
|
2021-03-10 05:27:10 +01:00
|
|
|
}
|
2020-10-29 04:37:51 +01:00
|
|
|
|
|
|
|
#[cfg(not(feature = "no_module"))]
|
|
|
|
Self::Import(_, _, _) => false,
|
|
|
|
#[cfg(not(feature = "no_module"))]
|
|
|
|
Self::Export(_, _) => false,
|
|
|
|
|
|
|
|
#[cfg(not(feature = "no_closure"))]
|
|
|
|
Self::Share(_) => false,
|
|
|
|
}
|
|
|
|
}
|
2021-03-11 11:29:22 +01:00
|
|
|
/// Is this statement _pure_ within the containing block?
|
|
|
|
///
|
|
|
|
/// An internally pure statement only has side effects that disappear outside the block.
|
|
|
|
///
|
|
|
|
/// Only variable declarations (i.e. `let` and `const`) and `import`/`export` statements
|
|
|
|
/// are internally pure.
|
2021-03-11 14:55:55 +01:00
|
|
|
#[inline(always)]
|
2021-03-11 11:29:22 +01:00
|
|
|
pub fn is_internally_pure(&self) -> bool {
|
|
|
|
match self {
|
|
|
|
Self::Let(expr, _, _, _) | Self::Const(expr, _, _, _) => expr.is_pure(),
|
|
|
|
|
|
|
|
#[cfg(not(feature = "no_module"))]
|
|
|
|
Self::Import(expr, _, _) => expr.is_pure(),
|
|
|
|
#[cfg(not(feature = "no_module"))]
|
|
|
|
Self::Export(_, _) => true,
|
|
|
|
|
|
|
|
_ => self.is_pure(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/// Does this statement break the current control flow through the containing block?
|
|
|
|
///
|
|
|
|
/// Currently this is only true for `return`, `throw`, `break` and `continue`.
|
|
|
|
///
|
|
|
|
/// All statements following this statement will essentially be dead code.
|
2021-03-11 14:55:55 +01:00
|
|
|
#[inline(always)]
|
2021-03-11 11:29:22 +01:00
|
|
|
pub fn is_control_flow_break(&self) -> bool {
|
|
|
|
match self {
|
|
|
|
Self::Return(_, _, _) | Self::Break(_) | Self::Continue(_) => true,
|
2021-03-11 14:55:55 +01:00
|
|
|
_ => false,
|
2021-03-11 11:29:22 +01:00
|
|
|
}
|
|
|
|
}
|
2021-01-08 17:24:55 +01:00
|
|
|
/// Recursively walk this statement.
|
2021-03-11 14:55:55 +01:00
|
|
|
/// Return `false` from the callback to terminate the walk.
|
|
|
|
pub fn walk<'a>(
|
|
|
|
&'a self,
|
|
|
|
path: &mut Vec<ASTNode<'a>>,
|
|
|
|
on_node: &mut impl FnMut(&[ASTNode]) -> bool,
|
|
|
|
) -> bool {
|
2021-05-22 13:14:24 +02:00
|
|
|
// Push the current node onto the path
|
2021-01-09 09:52:22 +01:00
|
|
|
path.push(self.into());
|
2021-03-11 14:55:55 +01:00
|
|
|
|
|
|
|
if !on_node(path) {
|
|
|
|
return false;
|
|
|
|
}
|
2021-01-08 17:24:55 +01:00
|
|
|
|
|
|
|
match self {
|
2021-03-11 14:55:55 +01:00
|
|
|
Self::Let(e, _, _, _) | Self::Const(e, _, _, _) => {
|
|
|
|
if !e.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2021-01-08 17:24:55 +01:00
|
|
|
Self::If(e, x, _) => {
|
2021-03-11 14:55:55 +01:00
|
|
|
if !e.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
2021-04-16 06:04:33 +02:00
|
|
|
for s in &(x.0).0 {
|
2021-03-11 14:55:55 +01:00
|
|
|
if !s.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2021-04-16 06:04:33 +02:00
|
|
|
for s in &(x.1).0 {
|
2021-03-11 14:55:55 +01:00
|
|
|
if !s.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2021-01-08 17:24:55 +01:00
|
|
|
}
|
|
|
|
Self::Switch(e, x, _) => {
|
2021-03-11 14:55:55 +01:00
|
|
|
if !e.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
2021-04-16 06:04:33 +02:00
|
|
|
for b in x.0.values() {
|
|
|
|
if !b.0.as_ref().map(|e| e.walk(path, on_node)).unwrap_or(true) {
|
2021-03-11 14:55:55 +01:00
|
|
|
return false;
|
|
|
|
}
|
2021-04-16 06:04:33 +02:00
|
|
|
for s in &(b.1).0 {
|
|
|
|
if !s.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2021-03-11 14:55:55 +01:00
|
|
|
}
|
2021-04-16 06:04:33 +02:00
|
|
|
for s in &(x.1).0 {
|
2021-03-11 14:55:55 +01:00
|
|
|
if !s.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2021-01-08 17:24:55 +01:00
|
|
|
}
|
2021-03-09 16:30:48 +01:00
|
|
|
Self::While(e, s, _) | Self::Do(s, e, _, _) => {
|
2021-03-11 14:55:55 +01:00
|
|
|
if !e.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
2021-04-16 06:04:33 +02:00
|
|
|
for s in &s.0 {
|
2021-03-11 14:55:55 +01:00
|
|
|
if !s.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2021-01-08 17:24:55 +01:00
|
|
|
}
|
2021-03-09 16:48:40 +01:00
|
|
|
Self::For(e, x, _) => {
|
2021-03-11 14:55:55 +01:00
|
|
|
if !e.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
2021-06-07 05:01:16 +02:00
|
|
|
for s in &(x.2).0 {
|
2021-03-11 14:55:55 +01:00
|
|
|
if !s.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2021-01-08 17:24:55 +01:00
|
|
|
}
|
|
|
|
Self::Assignment(x, _) => {
|
2021-03-11 14:55:55 +01:00
|
|
|
if !x.0.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
2021-03-30 17:55:29 +02:00
|
|
|
if !x.2.walk(path, on_node) {
|
2021-03-11 14:55:55 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2021-04-21 12:16:24 +02:00
|
|
|
Self::FnCall(x, _) => {
|
|
|
|
for s in &x.args {
|
|
|
|
if !s.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-03-11 14:55:55 +01:00
|
|
|
Self::Block(x, _) => {
|
2021-05-05 12:38:52 +02:00
|
|
|
for s in x.iter() {
|
2021-03-11 14:55:55 +01:00
|
|
|
if !s.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2021-01-08 17:24:55 +01:00
|
|
|
}
|
|
|
|
Self::TryCatch(x, _, _) => {
|
2021-04-16 06:04:33 +02:00
|
|
|
for s in &(x.0).0 {
|
2021-03-11 14:55:55 +01:00
|
|
|
if !s.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2021-04-16 06:04:33 +02:00
|
|
|
for s in &(x.2).0 {
|
2021-03-11 14:55:55 +01:00
|
|
|
if !s.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Self::Expr(e) | Self::Return(_, Some(e), _) => {
|
|
|
|
if !e.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
2021-01-08 17:24:55 +01:00
|
|
|
}
|
2021-01-08 17:40:44 +01:00
|
|
|
#[cfg(not(feature = "no_module"))]
|
2021-03-11 14:55:55 +01:00
|
|
|
Self::Import(e, _, _) => {
|
|
|
|
if !e.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2021-01-08 17:24:55 +01:00
|
|
|
_ => (),
|
|
|
|
}
|
2021-01-09 09:52:22 +01:00
|
|
|
|
2021-05-22 13:14:24 +02:00
|
|
|
path.pop()
|
|
|
|
.expect("never fails because `path` always contains the current node");
|
2021-03-11 14:55:55 +01:00
|
|
|
|
|
|
|
true
|
2021-01-08 17:24:55 +01:00
|
|
|
}
|
2020-10-29 04:37:51 +01:00
|
|
|
}
|
|
|
|
|
2021-01-14 12:07:03 +01:00
|
|
|
/// _(INTERNALS)_ A custom syntax expression.
|
2020-10-29 04:37:51 +01:00
|
|
|
/// Exported under the `internals` feature only.
|
|
|
|
///
|
2021-01-16 07:46:03 +01:00
|
|
|
/// # Volatile Data Structure
|
2020-10-29 04:37:51 +01:00
|
|
|
///
|
|
|
|
/// This type is volatile and may change.
|
2021-01-14 12:07:03 +01:00
|
|
|
#[derive(Debug, Clone, Hash)]
|
2020-10-29 04:37:51 +01:00
|
|
|
pub struct CustomExpr {
|
2020-12-13 07:31:24 +01:00
|
|
|
/// List of keywords.
|
2020-12-14 16:05:13 +01:00
|
|
|
pub keywords: StaticVec<Expr>,
|
2021-05-11 04:58:28 +02:00
|
|
|
/// Is the current [`Scope`][crate::Scope] modified?
|
|
|
|
pub scope_changed: bool,
|
2021-05-05 12:38:52 +02:00
|
|
|
/// List of tokens actually parsed.
|
|
|
|
pub tokens: StaticVec<Identifier>,
|
2020-10-29 04:37:51 +01:00
|
|
|
}
|
|
|
|
|
2020-11-20 09:52:28 +01:00
|
|
|
/// _(INTERNALS)_ A binary expression.
|
2020-11-06 09:27:40 +01:00
|
|
|
/// Exported under the `internals` feature only.
|
|
|
|
///
|
2021-01-16 07:46:03 +01:00
|
|
|
/// # Volatile Data Structure
|
2020-11-06 09:27:40 +01:00
|
|
|
///
|
|
|
|
/// This type is volatile and may change.
|
2021-04-13 09:52:45 +02:00
|
|
|
#[derive(Clone, Hash)]
|
2020-10-29 04:37:51 +01:00
|
|
|
pub struct BinaryExpr {
|
2020-10-31 07:13:45 +01:00
|
|
|
/// LHS expression.
|
2020-10-29 04:37:51 +01:00
|
|
|
pub lhs: Expr,
|
2020-10-31 07:13:45 +01:00
|
|
|
/// RHS expression.
|
2020-10-29 04:37:51 +01:00
|
|
|
pub rhs: Expr,
|
2020-10-31 07:13:45 +01:00
|
|
|
}
|
|
|
|
|
2021-03-08 08:30:32 +01:00
|
|
|
/// _(INTERNALS)_ An op-assignment operator.
|
|
|
|
/// Exported under the `internals` feature only.
|
|
|
|
///
|
|
|
|
/// # Volatile Data Structure
|
|
|
|
///
|
|
|
|
/// This type is volatile and may change.
|
2021-06-12 04:26:50 +02:00
|
|
|
#[derive(Debug, Clone, Copy, Eq, PartialEq, Hash)]
|
|
|
|
pub struct OpAssignment<'a> {
|
2021-03-08 08:30:32 +01:00
|
|
|
pub hash_op_assign: u64,
|
|
|
|
pub hash_op: u64,
|
2021-06-12 04:26:50 +02:00
|
|
|
pub op: &'a str,
|
2021-03-08 08:30:32 +01:00
|
|
|
}
|
|
|
|
|
2021-06-12 04:26:50 +02:00
|
|
|
impl OpAssignment<'_> {
|
2021-04-24 05:55:40 +02:00
|
|
|
/// Create a new [`OpAssignment`].
|
|
|
|
///
|
|
|
|
/// # Panics
|
|
|
|
///
|
|
|
|
/// Panics if the operator name is not an op-assignment operator.
|
|
|
|
pub fn new(op: Token) -> Self {
|
|
|
|
let op_raw = op
|
|
|
|
.map_op_assignment()
|
2021-05-22 13:14:24 +02:00
|
|
|
.expect("never fails because token must be an op-assignment operator")
|
2021-04-24 05:55:40 +02:00
|
|
|
.keyword_syntax();
|
|
|
|
let op_assignment = op.keyword_syntax();
|
2021-04-04 07:13:07 +02:00
|
|
|
|
|
|
|
Self {
|
2021-05-19 14:26:11 +02:00
|
|
|
hash_op_assign: calc_fn_hash(op_assignment, 2),
|
|
|
|
hash_op: calc_fn_hash(op_raw, 2),
|
2021-04-24 05:55:40 +02:00
|
|
|
op: op_assignment,
|
2021-04-04 07:13:07 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-08 08:30:32 +01:00
|
|
|
/// _(INTERNALS)_ An set of function call hashes.
|
|
|
|
/// Exported under the `internals` feature only.
|
|
|
|
///
|
2021-03-17 06:30:47 +01:00
|
|
|
/// Two separate hashes are pre-calculated because of the following pattern:
|
|
|
|
///
|
2021-03-22 04:18:09 +01:00
|
|
|
/// ```ignore
|
2021-03-17 06:30:47 +01:00
|
|
|
/// func(a, b, c); // Native: func(a, b, c) - 3 parameters
|
|
|
|
/// // Script: func(a, b, c) - 3 parameters
|
|
|
|
///
|
|
|
|
/// a.func(b, c); // Native: func(&mut a, b, c) - 3 parameters
|
|
|
|
/// // Script: func(b, c) - 2 parameters
|
|
|
|
/// ```
|
|
|
|
///
|
|
|
|
/// For normal function calls, the native hash equals the script hash.
|
|
|
|
/// For method-style calls, the script hash contains one fewer parameter.
|
|
|
|
///
|
|
|
|
/// Function call hashes are used in the following manner:
|
|
|
|
///
|
|
|
|
/// * First, the script hash is tried, which contains only the called function's name plus the
|
|
|
|
/// of parameters.
|
|
|
|
///
|
|
|
|
/// * Next, the actual types of arguments are hashed and _combined_ with the native hash, which is
|
|
|
|
/// then used to search for a native function.
|
|
|
|
/// In other words, a native function call hash always contains the called function's name plus
|
|
|
|
/// the types of the arguments. This is to due to possible function overloading for different parameter types.
|
|
|
|
///
|
2021-03-08 08:30:32 +01:00
|
|
|
/// # Volatile Data Structure
|
|
|
|
///
|
|
|
|
/// This type is volatile and may change.
|
2021-03-09 07:00:21 +01:00
|
|
|
#[derive(Clone, Copy, Eq, PartialEq, Hash, Default)]
|
2021-04-20 16:26:08 +02:00
|
|
|
pub struct FnCallHashes {
|
2021-03-08 08:30:32 +01:00
|
|
|
/// Pre-calculated hash for a script-defined function ([`None`] if native functions only).
|
2021-03-17 06:30:47 +01:00
|
|
|
pub script: Option<u64>,
|
2021-03-08 08:30:32 +01:00
|
|
|
/// Pre-calculated hash for a native Rust function with no parameter types.
|
2021-03-17 06:30:47 +01:00
|
|
|
pub native: u64,
|
2021-03-08 08:30:32 +01:00
|
|
|
}
|
|
|
|
|
2021-04-20 16:26:08 +02:00
|
|
|
impl fmt::Debug for FnCallHashes {
|
2021-03-09 07:00:21 +01:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
|
|
if let Some(script) = self.script {
|
|
|
|
if script == self.native {
|
|
|
|
write!(f, "({}=={})", script, self.native)
|
|
|
|
} else {
|
|
|
|
write!(f, "({}, {})", script, self.native)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
write!(f, "{}", self.native)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-20 16:26:08 +02:00
|
|
|
impl FnCallHashes {
|
2021-05-02 17:57:35 +02:00
|
|
|
/// Create a [`FnCallHashes`] with only the native Rust hash.
|
2021-03-08 08:30:32 +01:00
|
|
|
#[inline(always)]
|
|
|
|
pub fn from_native(hash: u64) -> Self {
|
|
|
|
Self {
|
|
|
|
script: None,
|
|
|
|
native: hash,
|
|
|
|
}
|
|
|
|
}
|
2021-05-02 17:57:35 +02:00
|
|
|
/// Create a [`FnCallHashes`] with both native Rust and script function hashes set to the same value.
|
2021-03-08 08:30:32 +01:00
|
|
|
#[inline(always)]
|
|
|
|
pub fn from_script(hash: u64) -> Self {
|
|
|
|
Self {
|
|
|
|
script: Some(hash),
|
|
|
|
native: hash,
|
|
|
|
}
|
|
|
|
}
|
2021-05-02 17:57:35 +02:00
|
|
|
/// Create a [`FnCallHashes`] with both native Rust and script function hashes.
|
2021-03-08 08:30:32 +01:00
|
|
|
#[inline(always)]
|
|
|
|
pub fn from_script_and_native(script: u64, native: u64) -> Self {
|
|
|
|
Self {
|
|
|
|
script: Some(script),
|
|
|
|
native,
|
|
|
|
}
|
|
|
|
}
|
2021-05-02 17:57:35 +02:00
|
|
|
/// Is this [`FnCallHashes`] native Rust only?
|
2021-03-08 08:30:32 +01:00
|
|
|
#[inline(always)]
|
|
|
|
pub fn is_native_only(&self) -> bool {
|
|
|
|
self.script.is_none()
|
|
|
|
}
|
2021-05-02 17:57:35 +02:00
|
|
|
/// Get the script function hash from this [`FnCallHashes`].
|
2021-03-08 08:30:32 +01:00
|
|
|
#[inline(always)]
|
2021-05-22 13:14:24 +02:00
|
|
|
pub fn script_hash(&self) -> Option<u64> {
|
|
|
|
self.script
|
2021-03-08 08:30:32 +01:00
|
|
|
}
|
2021-05-02 17:57:35 +02:00
|
|
|
/// Get the naive Rust function hash from this [`FnCallHashes`].
|
2021-03-08 08:30:32 +01:00
|
|
|
#[inline(always)]
|
|
|
|
pub fn native_hash(&self) -> u64 {
|
|
|
|
self.native
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-20 09:52:28 +01:00
|
|
|
/// _(INTERNALS)_ A function call.
|
2020-11-06 09:27:40 +01:00
|
|
|
/// Exported under the `internals` feature only.
|
|
|
|
///
|
2021-01-16 07:46:03 +01:00
|
|
|
/// # Volatile Data Structure
|
2020-11-06 09:27:40 +01:00
|
|
|
///
|
|
|
|
/// This type is volatile and may change.
|
2021-04-21 12:16:24 +02:00
|
|
|
#[derive(Debug, Clone, Default, Hash)]
|
2020-11-10 16:26:50 +01:00
|
|
|
pub struct FnCallExpr {
|
2021-04-20 17:28:04 +02:00
|
|
|
/// Namespace of the function, if any.
|
|
|
|
pub namespace: Option<NamespaceRef>,
|
2021-04-20 16:26:08 +02:00
|
|
|
/// Pre-calculated hashes.
|
|
|
|
pub hashes: FnCallHashes,
|
2021-03-28 17:06:59 +02:00
|
|
|
/// List of function call argument expressions.
|
2021-03-08 08:55:26 +01:00
|
|
|
pub args: StaticVec<Expr>,
|
2021-03-28 13:04:25 +02:00
|
|
|
/// List of function call arguments that are constants.
|
2021-06-08 08:46:49 +02:00
|
|
|
pub constants: smallvec::SmallVec<[Dynamic; 2]>,
|
2020-11-02 16:54:19 +01:00
|
|
|
/// Function name.
|
2021-03-29 05:36:02 +02:00
|
|
|
pub name: Identifier,
|
2021-04-20 17:28:04 +02:00
|
|
|
/// Does this function call capture the parent scope?
|
|
|
|
pub capture: bool,
|
2020-10-29 04:37:51 +01:00
|
|
|
}
|
|
|
|
|
2021-03-28 13:04:25 +02:00
|
|
|
impl FnCallExpr {
|
2021-04-20 17:28:04 +02:00
|
|
|
/// Does this function call contain a qualified namespace?
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn is_qualified(&self) -> bool {
|
|
|
|
self.namespace.is_some()
|
|
|
|
}
|
2021-03-28 13:04:25 +02:00
|
|
|
}
|
|
|
|
|
2021-04-06 17:18:41 +02:00
|
|
|
/// A type that wraps a floating-point number and implements [`Hash`].
|
2021-05-10 05:07:19 +02:00
|
|
|
///
|
|
|
|
/// Not available under `no_float`.
|
2021-01-04 04:58:24 +01:00
|
|
|
#[cfg(not(feature = "no_float"))]
|
2021-02-12 16:07:28 +01:00
|
|
|
#[derive(Clone, Copy, PartialEq, PartialOrd)]
|
2021-04-06 17:18:41 +02:00
|
|
|
pub struct FloatWrapper<F>(F);
|
2021-01-04 04:58:24 +01:00
|
|
|
|
|
|
|
#[cfg(not(feature = "no_float"))]
|
2021-04-06 17:18:41 +02:00
|
|
|
impl Hash for FloatWrapper<FLOAT> {
|
2021-03-10 06:32:09 +01:00
|
|
|
#[inline(always)]
|
2021-04-17 09:15:54 +02:00
|
|
|
fn hash<H: std::hash::Hasher>(&self, state: &mut H) {
|
2021-03-05 02:31:56 +01:00
|
|
|
self.0.to_ne_bytes().hash(state);
|
2021-01-04 04:58:24 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(not(feature = "no_float"))]
|
2021-04-06 17:18:41 +02:00
|
|
|
impl<F: Float> AsRef<F> for FloatWrapper<F> {
|
2021-03-10 06:32:09 +01:00
|
|
|
#[inline(always)]
|
2021-04-06 17:18:41 +02:00
|
|
|
fn as_ref(&self) -> &F {
|
2021-01-04 04:58:24 +01:00
|
|
|
&self.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(not(feature = "no_float"))]
|
2021-04-06 17:18:41 +02:00
|
|
|
impl<F: Float> AsMut<F> for FloatWrapper<F> {
|
2021-03-10 06:32:09 +01:00
|
|
|
#[inline(always)]
|
2021-04-06 17:18:41 +02:00
|
|
|
fn as_mut(&mut self) -> &mut F {
|
2021-01-04 04:58:24 +01:00
|
|
|
&mut self.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(not(feature = "no_float"))]
|
2021-04-17 09:15:54 +02:00
|
|
|
impl<F: Float> std::ops::Deref for FloatWrapper<F> {
|
2021-04-06 17:18:41 +02:00
|
|
|
type Target = F;
|
2021-01-04 04:58:24 +01:00
|
|
|
|
2021-03-10 06:32:09 +01:00
|
|
|
#[inline(always)]
|
2021-01-04 04:58:24 +01:00
|
|
|
fn deref(&self) -> &Self::Target {
|
|
|
|
&self.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(not(feature = "no_float"))]
|
2021-04-17 09:15:54 +02:00
|
|
|
impl<F: Float> std::ops::DerefMut for FloatWrapper<F> {
|
2021-03-10 06:32:09 +01:00
|
|
|
#[inline(always)]
|
2021-01-04 04:58:24 +01:00
|
|
|
fn deref_mut(&mut self) -> &mut Self::Target {
|
|
|
|
&mut self.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(not(feature = "no_float"))]
|
2021-04-11 05:20:34 +02:00
|
|
|
impl<F: Float + fmt::Debug> fmt::Debug for FloatWrapper<F> {
|
2021-03-10 06:32:09 +01:00
|
|
|
#[inline(always)]
|
2021-01-04 04:58:24 +01:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
2021-04-11 05:20:34 +02:00
|
|
|
fmt::Debug::fmt(&self.0, f)
|
2021-01-04 04:58:24 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(not(feature = "no_float"))]
|
2021-04-06 17:18:41 +02:00
|
|
|
impl<F: Float + fmt::Display + fmt::LowerExp + From<f32>> fmt::Display for FloatWrapper<F> {
|
2021-03-10 06:32:09 +01:00
|
|
|
#[inline(always)]
|
2021-01-04 04:58:24 +01:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
2021-02-11 14:28:07 +01:00
|
|
|
let abs = self.0.abs();
|
2021-04-06 17:18:41 +02:00
|
|
|
if abs > Self::MAX_NATURAL_FLOAT_FOR_DISPLAY.into()
|
|
|
|
|| abs < Self::MIN_NATURAL_FLOAT_FOR_DISPLAY.into()
|
|
|
|
{
|
2021-02-11 14:28:07 +01:00
|
|
|
write!(f, "{:e}", self.0)
|
|
|
|
} else {
|
2021-04-06 17:18:41 +02:00
|
|
|
fmt::Display::fmt(&self.0, f)?;
|
|
|
|
if abs.fract().is_zero() {
|
|
|
|
f.write_str(".0")?;
|
|
|
|
}
|
|
|
|
Ok(())
|
2021-02-11 14:28:07 +01:00
|
|
|
}
|
2021-01-04 04:58:24 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(not(feature = "no_float"))]
|
2021-04-06 17:18:41 +02:00
|
|
|
impl<F: Float> From<F> for FloatWrapper<F> {
|
2021-03-10 06:32:09 +01:00
|
|
|
#[inline(always)]
|
2021-04-06 17:18:41 +02:00
|
|
|
fn from(value: F) -> Self {
|
2021-01-04 04:58:24 +01:00
|
|
|
Self::new(value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-12 16:07:28 +01:00
|
|
|
#[cfg(not(feature = "no_float"))]
|
2021-04-06 17:18:41 +02:00
|
|
|
impl<F: Float + FromStr> FromStr for FloatWrapper<F> {
|
|
|
|
type Err = <F as FromStr>::Err;
|
2021-02-12 16:07:28 +01:00
|
|
|
|
2021-03-10 06:32:09 +01:00
|
|
|
#[inline(always)]
|
2021-02-12 16:07:28 +01:00
|
|
|
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
2021-04-06 17:18:41 +02:00
|
|
|
F::from_str(s).map(Into::<Self>::into)
|
2021-02-12 16:07:28 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-04 04:58:24 +01:00
|
|
|
#[cfg(not(feature = "no_float"))]
|
2021-04-06 17:18:41 +02:00
|
|
|
impl<F: Float> FloatWrapper<F> {
|
|
|
|
/// Maximum floating-point number for natural display before switching to scientific notation.
|
|
|
|
pub const MAX_NATURAL_FLOAT_FOR_DISPLAY: f32 = 10000000000000.0;
|
|
|
|
|
|
|
|
/// Minimum floating-point number for natural display before switching to scientific notation.
|
|
|
|
pub const MIN_NATURAL_FLOAT_FOR_DISPLAY: f32 = 0.0000000000001;
|
|
|
|
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn new(value: F) -> Self {
|
|
|
|
Self(value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-06 17:28:22 +02:00
|
|
|
#[cfg(not(feature = "no_float"))]
|
2021-04-06 17:18:41 +02:00
|
|
|
impl FloatWrapper<FLOAT> {
|
2021-03-10 06:32:09 +01:00
|
|
|
#[inline(always)]
|
2021-04-06 17:18:41 +02:00
|
|
|
pub(crate) const fn const_new(value: FLOAT) -> Self {
|
2021-01-04 04:58:24 +01:00
|
|
|
Self(value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-20 09:52:28 +01:00
|
|
|
/// _(INTERNALS)_ An expression sub-tree.
|
2020-10-29 04:37:51 +01:00
|
|
|
/// Exported under the `internals` feature only.
|
|
|
|
///
|
2021-01-16 07:46:03 +01:00
|
|
|
/// # Volatile Data Structure
|
2020-10-29 04:37:51 +01:00
|
|
|
///
|
|
|
|
/// This type is volatile and may change.
|
2021-04-06 17:18:41 +02:00
|
|
|
#[derive(Clone, Hash)]
|
2020-10-29 04:37:51 +01:00
|
|
|
pub enum Expr {
|
2020-11-14 12:04:49 +01:00
|
|
|
/// Dynamic constant.
|
2021-03-24 03:02:50 +01:00
|
|
|
/// Used to hold either an [`Array`] or [`Map`][crate::Map] literal for quick cloning.
|
2020-11-14 15:43:56 +01:00
|
|
|
/// All other primitive data types should use the appropriate variants for better speed.
|
2020-11-14 12:04:49 +01:00
|
|
|
DynamicConstant(Box<Dynamic>, Position),
|
2020-11-19 15:29:50 +01:00
|
|
|
/// Boolean constant.
|
|
|
|
BoolConstant(bool, Position),
|
2020-10-29 04:37:51 +01:00
|
|
|
/// Integer constant.
|
2020-10-31 07:13:45 +01:00
|
|
|
IntegerConstant(INT, Position),
|
2020-10-29 04:37:51 +01:00
|
|
|
/// Floating-point constant.
|
2021-05-10 05:07:19 +02:00
|
|
|
///
|
|
|
|
/// Not available under `no_float`.
|
2020-10-29 04:37:51 +01:00
|
|
|
#[cfg(not(feature = "no_float"))]
|
2021-04-06 17:18:41 +02:00
|
|
|
FloatConstant(FloatWrapper<FLOAT>, Position),
|
2020-10-29 04:37:51 +01:00
|
|
|
/// Character constant.
|
2020-10-31 07:13:45 +01:00
|
|
|
CharConstant(char, Position),
|
2020-11-20 09:52:28 +01:00
|
|
|
/// [String][ImmutableString] constant.
|
2020-11-13 03:43:54 +01:00
|
|
|
StringConstant(ImmutableString, Position),
|
2021-04-04 07:13:07 +02:00
|
|
|
/// An interpolated [string][ImmutableString].
|
|
|
|
InterpolatedString(Box<StaticVec<Expr>>),
|
2020-11-13 11:32:18 +01:00
|
|
|
/// [ expr, ... ]
|
|
|
|
Array(Box<StaticVec<Expr>>, Position),
|
|
|
|
/// #{ name:expr, ... }
|
2021-03-23 11:25:40 +01:00
|
|
|
Map(
|
2021-03-29 11:14:22 +02:00
|
|
|
Box<(StaticVec<(Ident, Expr)>, BTreeMap<Identifier, Dynamic>)>,
|
2021-03-23 11:25:40 +01:00
|
|
|
Position,
|
|
|
|
),
|
2020-11-13 11:32:18 +01:00
|
|
|
/// ()
|
|
|
|
Unit(Position),
|
2021-04-05 17:59:15 +02:00
|
|
|
/// Variable access - optional short index, position, (optional index, optional (hash, modules), variable name)
|
2021-04-05 17:06:48 +02:00
|
|
|
///
|
|
|
|
/// The short index is [`u8`] which is used when the index is <= 255, which should be the vast
|
|
|
|
/// majority of cases (unless there are more than 255 variables defined!).
|
|
|
|
/// This is to avoid reading a pointer redirection during each variable access.
|
|
|
|
Variable(
|
|
|
|
Option<NonZeroU8>,
|
2021-04-05 17:59:15 +02:00
|
|
|
Position,
|
|
|
|
Box<(
|
|
|
|
Option<NonZeroUsize>,
|
|
|
|
Option<(u64, NamespaceRef)>,
|
|
|
|
Identifier,
|
|
|
|
)>,
|
2021-04-05 17:06:48 +02:00
|
|
|
),
|
2021-03-10 15:12:48 +01:00
|
|
|
/// Property access - ((getter, hash), (setter, hash), prop)
|
2021-05-18 14:12:30 +02:00
|
|
|
Property(
|
|
|
|
Box<(
|
|
|
|
(Identifier, u64),
|
|
|
|
(Identifier, u64),
|
|
|
|
(ImmutableString, Position),
|
|
|
|
)>,
|
|
|
|
),
|
2021-06-08 08:46:49 +02:00
|
|
|
/// Stack slot
|
|
|
|
Stack(usize, Position),
|
2021-03-09 06:44:54 +01:00
|
|
|
/// { [statement][Stmt] ... }
|
2021-03-10 15:12:48 +01:00
|
|
|
Stmt(Box<StmtBlock>),
|
2020-11-20 09:52:28 +01:00
|
|
|
/// func `(` expr `,` ... `)`
|
2020-11-10 16:26:50 +01:00
|
|
|
FnCall(Box<FnCallExpr>, Position),
|
2020-11-20 09:52:28 +01:00
|
|
|
/// lhs `.` rhs
|
2020-10-31 16:26:21 +01:00
|
|
|
Dot(Box<BinaryExpr>, Position),
|
2020-11-20 09:52:28 +01:00
|
|
|
/// expr `[` expr `]`
|
2020-10-31 16:26:21 +01:00
|
|
|
Index(Box<BinaryExpr>, Position),
|
2020-11-20 09:52:28 +01:00
|
|
|
/// lhs `&&` rhs
|
2020-10-31 16:26:21 +01:00
|
|
|
And(Box<BinaryExpr>, Position),
|
2020-11-20 09:52:28 +01:00
|
|
|
/// lhs `||` rhs
|
2020-10-31 16:26:21 +01:00
|
|
|
Or(Box<BinaryExpr>, Position),
|
2020-10-29 04:37:51 +01:00
|
|
|
/// Custom syntax
|
2020-10-31 16:26:21 +01:00
|
|
|
Custom(Box<CustomExpr>, Position),
|
2020-10-29 04:37:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for Expr {
|
|
|
|
#[inline(always)]
|
|
|
|
fn default() -> Self {
|
2020-11-20 09:52:28 +01:00
|
|
|
Self::Unit(Position::NONE)
|
2020-10-29 04:37:51 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-06 17:18:41 +02:00
|
|
|
impl fmt::Debug for Expr {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
2021-04-23 08:24:53 +02:00
|
|
|
let mut display_pos = self.position();
|
|
|
|
|
2021-04-06 17:18:41 +02:00
|
|
|
match self {
|
2021-04-22 17:02:25 +02:00
|
|
|
Self::DynamicConstant(value, _) => write!(f, "{:?}", value),
|
|
|
|
Self::BoolConstant(value, _) => write!(f, "{:?}", value),
|
|
|
|
Self::IntegerConstant(value, _) => write!(f, "{:?}", value),
|
|
|
|
#[cfg(not(feature = "no_float"))]
|
|
|
|
Self::FloatConstant(value, _) => write!(f, "{:?}", value),
|
|
|
|
Self::CharConstant(value, _) => write!(f, "{:?}", value),
|
|
|
|
Self::StringConstant(value, _) => write!(f, "{:?}", value),
|
|
|
|
Self::Unit(_) => f.write_str("()"),
|
|
|
|
|
2021-04-06 17:18:41 +02:00
|
|
|
Self::InterpolatedString(x) => {
|
|
|
|
f.write_str("InterpolatedString")?;
|
2021-04-23 08:24:53 +02:00
|
|
|
return f.debug_list().entries(x.iter()).finish();
|
2021-04-06 17:18:41 +02:00
|
|
|
}
|
2021-04-23 08:24:53 +02:00
|
|
|
Self::Array(x, _) => {
|
2021-04-06 17:18:41 +02:00
|
|
|
f.write_str("Array")?;
|
2021-04-23 08:24:53 +02:00
|
|
|
f.debug_list().entries(x.iter()).finish()
|
2021-04-06 17:18:41 +02:00
|
|
|
}
|
2021-04-23 08:24:53 +02:00
|
|
|
Self::Map(x, _) => {
|
2021-04-06 17:18:41 +02:00
|
|
|
f.write_str("Map")?;
|
|
|
|
f.debug_map()
|
|
|
|
.entries(x.0.iter().map(|(k, v)| (k, v)))
|
2021-04-23 08:24:53 +02:00
|
|
|
.finish()
|
2021-04-06 17:18:41 +02:00
|
|
|
}
|
2021-04-23 08:24:53 +02:00
|
|
|
Self::Variable(i, _, x) => {
|
2021-04-06 17:18:41 +02:00
|
|
|
f.write_str("Variable(")?;
|
|
|
|
match x.1 {
|
2021-04-13 09:52:45 +02:00
|
|
|
Some((_, ref namespace)) => write!(f, "{}", namespace)?,
|
2021-04-06 17:18:41 +02:00
|
|
|
_ => (),
|
|
|
|
}
|
2021-06-04 08:23:40 +02:00
|
|
|
f.write_str(&x.2)?;
|
2021-04-06 17:18:41 +02:00
|
|
|
match i.map_or_else(|| x.0, |n| NonZeroUsize::new(n.get() as usize)) {
|
2021-06-07 05:01:16 +02:00
|
|
|
Some(n) => write!(f, " #{}", n)?,
|
2021-04-06 17:18:41 +02:00
|
|
|
_ => (),
|
|
|
|
}
|
2021-04-23 08:24:53 +02:00
|
|
|
f.write_str(")")
|
2021-04-06 17:18:41 +02:00
|
|
|
}
|
2021-05-18 14:12:30 +02:00
|
|
|
Self::Property(x) => write!(f, "Property({})", (x.2).0),
|
2021-06-08 08:46:49 +02:00
|
|
|
Self::Stack(x, _) => write!(f, "StackSlot({})", x),
|
2021-04-06 17:18:41 +02:00
|
|
|
Self::Stmt(x) => {
|
2021-05-22 13:14:24 +02:00
|
|
|
f.write_str("ExprStmtBlock")?;
|
2021-04-23 08:24:53 +02:00
|
|
|
f.debug_list().entries(x.0.iter()).finish()
|
2021-04-06 17:18:41 +02:00
|
|
|
}
|
2021-04-23 08:24:53 +02:00
|
|
|
Self::FnCall(x, _) => {
|
2021-04-13 09:52:45 +02:00
|
|
|
let mut ff = f.debug_struct("FnCall");
|
2021-05-25 04:54:48 +02:00
|
|
|
x.namespace.as_ref().map(|ns| ff.field("namespace", ns));
|
2021-04-13 09:52:45 +02:00
|
|
|
ff.field("name", &x.name)
|
2021-04-20 16:26:08 +02:00
|
|
|
.field("hash", &x.hashes)
|
2021-04-13 09:52:45 +02:00
|
|
|
.field("args", &x.args);
|
2021-06-08 08:46:49 +02:00
|
|
|
if !x.constants.is_empty() {
|
|
|
|
ff.field("constants", &x.constants);
|
2021-04-13 09:52:45 +02:00
|
|
|
}
|
|
|
|
if x.capture {
|
|
|
|
ff.field("capture", &x.capture);
|
|
|
|
}
|
2021-04-23 08:24:53 +02:00
|
|
|
ff.finish()
|
2021-04-06 17:18:41 +02:00
|
|
|
}
|
2021-04-23 08:24:53 +02:00
|
|
|
Self::Dot(x, pos) | Self::Index(x, pos) | Self::And(x, pos) | Self::Or(x, pos) => {
|
2021-04-13 09:52:45 +02:00
|
|
|
let op_name = match self {
|
2021-04-06 17:18:41 +02:00
|
|
|
Self::Dot(_, _) => "Dot",
|
|
|
|
Self::Index(_, _) => "Index",
|
|
|
|
Self::And(_, _) => "And",
|
|
|
|
Self::Or(_, _) => "Or",
|
|
|
|
_ => unreachable!(),
|
|
|
|
};
|
|
|
|
|
2021-04-23 08:24:53 +02:00
|
|
|
display_pos = *pos;
|
|
|
|
|
2021-04-13 09:52:45 +02:00
|
|
|
f.debug_struct(op_name)
|
2021-04-06 17:18:41 +02:00
|
|
|
.field("lhs", &x.lhs)
|
|
|
|
.field("rhs", &x.rhs)
|
2021-04-23 08:24:53 +02:00
|
|
|
.finish()
|
2021-04-06 17:18:41 +02:00
|
|
|
}
|
2021-04-23 08:24:53 +02:00
|
|
|
Self::Custom(x, _) => f.debug_tuple("Custom").field(x).finish(),
|
|
|
|
}?;
|
|
|
|
|
|
|
|
display_pos.debug_print(f)
|
2021-04-06 17:18:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-29 04:37:51 +01:00
|
|
|
impl Expr {
|
2021-06-10 04:16:39 +02:00
|
|
|
/// Get the [`Dynamic`] value of a literal constant expression.
|
2020-10-29 04:37:51 +01:00
|
|
|
///
|
2021-06-10 04:16:39 +02:00
|
|
|
/// Returns [`None`] if the expression is not a literal constant.
|
2021-03-10 06:32:09 +01:00
|
|
|
#[inline]
|
2021-06-10 04:16:39 +02:00
|
|
|
pub fn get_literal_value(&self) -> Option<Dynamic> {
|
2020-10-29 04:37:51 +01:00
|
|
|
Some(match self {
|
2020-11-14 12:04:49 +01:00
|
|
|
Self::DynamicConstant(x, _) => x.as_ref().clone(),
|
2020-10-31 07:13:45 +01:00
|
|
|
Self::IntegerConstant(x, _) => (*x).into(),
|
2020-10-29 04:37:51 +01:00
|
|
|
#[cfg(not(feature = "no_float"))]
|
2020-11-13 11:32:18 +01:00
|
|
|
Self::FloatConstant(x, _) => (*x).into(),
|
2020-10-31 07:13:45 +01:00
|
|
|
Self::CharConstant(x, _) => (*x).into(),
|
2020-11-13 03:43:54 +01:00
|
|
|
Self::StringConstant(x, _) => x.clone().into(),
|
2020-11-19 15:29:50 +01:00
|
|
|
Self::BoolConstant(x, _) => (*x).into(),
|
2021-01-02 16:30:10 +01:00
|
|
|
Self::Unit(_) => Dynamic::UNIT,
|
2020-10-29 04:37:51 +01:00
|
|
|
|
|
|
|
#[cfg(not(feature = "no_index"))]
|
2020-11-15 05:07:35 +01:00
|
|
|
Self::Array(x, _) if self.is_constant() => {
|
2021-03-23 05:13:53 +01:00
|
|
|
let mut arr = Array::with_capacity(x.len());
|
2021-05-22 13:14:24 +02:00
|
|
|
arr.extend(x.iter().map(|v| {
|
2021-06-10 04:16:39 +02:00
|
|
|
v.get_literal_value()
|
2021-05-22 13:14:24 +02:00
|
|
|
.expect("never fails because a constant array always has a constant value")
|
|
|
|
}));
|
2021-04-20 16:21:51 +02:00
|
|
|
Dynamic::from_array(arr)
|
2020-11-15 05:07:35 +01:00
|
|
|
}
|
2020-10-29 04:37:51 +01:00
|
|
|
|
|
|
|
#[cfg(not(feature = "no_object"))]
|
2020-11-15 05:07:35 +01:00
|
|
|
Self::Map(x, _) if self.is_constant() => {
|
2021-03-23 11:25:40 +01:00
|
|
|
let mut map = x.1.clone();
|
|
|
|
x.0.iter().for_each(|(k, v)| {
|
2021-05-22 13:14:24 +02:00
|
|
|
*map.get_mut(k.name.as_str())
|
|
|
|
.expect("never fails because the template should contain all the keys") = v
|
2021-06-10 04:16:39 +02:00
|
|
|
.get_literal_value()
|
2021-05-22 13:14:24 +02:00
|
|
|
.expect("never fails because a constant map always has a constant value")
|
2021-03-23 11:25:40 +01:00
|
|
|
});
|
2021-04-20 16:21:51 +02:00
|
|
|
Dynamic::from_map(map)
|
2020-10-29 04:37:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
_ => return None,
|
|
|
|
})
|
|
|
|
}
|
2021-06-08 08:46:49 +02:00
|
|
|
/// Create an [`Expr`] from a [`Dynamic`] value.
|
|
|
|
#[inline]
|
|
|
|
pub fn from_dynamic(value: Dynamic, pos: Position) -> Self {
|
|
|
|
match value.0 {
|
|
|
|
Union::Unit(_, _, _) => Self::Unit(pos),
|
|
|
|
Union::Bool(b, _, _) => Self::BoolConstant(b, pos),
|
|
|
|
Union::Str(s, _, _) => Self::StringConstant(s, pos),
|
|
|
|
Union::Char(c, _, _) => Self::CharConstant(c, pos),
|
|
|
|
Union::Int(i, _, _) => Self::IntegerConstant(i, pos),
|
|
|
|
|
|
|
|
#[cfg(not(feature = "no_float"))]
|
|
|
|
Union::Float(f, _, _) => Self::FloatConstant(f, pos),
|
|
|
|
|
|
|
|
_ => Self::DynamicConstant(Box::new(value), pos),
|
|
|
|
}
|
|
|
|
}
|
2021-04-05 17:59:15 +02:00
|
|
|
/// Is the expression a simple variable access?
|
|
|
|
#[inline(always)]
|
|
|
|
pub(crate) fn is_variable_access(&self, non_qualified: bool) -> bool {
|
|
|
|
match self {
|
|
|
|
Self::Variable(_, _, x) => !non_qualified || x.1.is_none(),
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
2021-02-03 12:14:26 +01:00
|
|
|
/// Return the variable name if the expression a simple variable access.
|
2020-12-29 03:41:20 +01:00
|
|
|
#[inline(always)]
|
2021-04-05 17:59:15 +02:00
|
|
|
pub(crate) fn get_variable_name(&self, non_qualified: bool) -> Option<&str> {
|
2020-10-29 04:37:51 +01:00
|
|
|
match self {
|
2021-04-05 17:59:15 +02:00
|
|
|
Self::Variable(_, _, x) if !non_qualified || x.1.is_none() => Some(x.2.as_str()),
|
2020-10-29 04:37:51 +01:00
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
2020-12-26 16:21:16 +01:00
|
|
|
/// Get the [position][Position] of the expression.
|
2021-03-10 06:32:09 +01:00
|
|
|
#[inline]
|
2020-10-29 04:37:51 +01:00
|
|
|
pub fn position(&self) -> Position {
|
|
|
|
match self {
|
|
|
|
#[cfg(not(feature = "no_float"))]
|
2020-10-31 07:13:45 +01:00
|
|
|
Self::FloatConstant(_, pos) => *pos,
|
2020-10-29 04:37:51 +01:00
|
|
|
|
2021-04-23 08:24:53 +02:00
|
|
|
Self::DynamicConstant(_, pos)
|
|
|
|
| Self::BoolConstant(_, pos)
|
|
|
|
| Self::IntegerConstant(_, pos)
|
|
|
|
| Self::CharConstant(_, pos)
|
|
|
|
| Self::Unit(pos)
|
|
|
|
| Self::StringConstant(_, pos)
|
|
|
|
| Self::Array(_, pos)
|
|
|
|
| Self::Map(_, pos)
|
|
|
|
| Self::Variable(_, pos, _)
|
2021-06-08 08:46:49 +02:00
|
|
|
| Self::Stack(_, pos)
|
2021-04-23 08:24:53 +02:00
|
|
|
| Self::FnCall(_, pos)
|
|
|
|
| Self::Custom(_, pos) => *pos,
|
|
|
|
|
2021-05-22 13:14:24 +02:00
|
|
|
Self::InterpolatedString(x) => x
|
|
|
|
.first()
|
|
|
|
.expect(
|
|
|
|
"never fails because an interpolated string always contains at least one item",
|
|
|
|
)
|
|
|
|
.position(),
|
2021-04-23 08:24:53 +02:00
|
|
|
|
2021-05-18 14:12:30 +02:00
|
|
|
Self::Property(x) => (x.2).1,
|
2021-04-16 06:04:33 +02:00
|
|
|
Self::Stmt(x) => x.1,
|
2020-10-29 04:37:51 +01:00
|
|
|
|
2021-04-23 08:24:53 +02:00
|
|
|
Self::And(x, _) | Self::Or(x, _) | Self::Dot(x, _) | Self::Index(x, _) => {
|
|
|
|
x.lhs.position()
|
|
|
|
}
|
2020-10-29 04:37:51 +01:00
|
|
|
}
|
|
|
|
}
|
2020-12-26 16:21:16 +01:00
|
|
|
/// Override the [position][Position] of the expression.
|
2021-03-10 06:32:09 +01:00
|
|
|
#[inline]
|
2020-10-29 04:37:51 +01:00
|
|
|
pub fn set_position(&mut self, new_pos: Position) -> &mut Self {
|
|
|
|
match self {
|
|
|
|
#[cfg(not(feature = "no_float"))]
|
2020-10-31 07:13:45 +01:00
|
|
|
Self::FloatConstant(_, pos) => *pos = new_pos,
|
2020-10-29 04:37:51 +01:00
|
|
|
|
2021-04-23 08:24:53 +02:00
|
|
|
Self::DynamicConstant(_, pos)
|
|
|
|
| Self::BoolConstant(_, pos)
|
|
|
|
| Self::IntegerConstant(_, pos)
|
|
|
|
| Self::CharConstant(_, pos)
|
|
|
|
| Self::Unit(pos)
|
|
|
|
| Self::StringConstant(_, pos)
|
|
|
|
| Self::Array(_, pos)
|
|
|
|
| Self::Map(_, pos)
|
|
|
|
| Self::And(_, pos)
|
|
|
|
| Self::Or(_, pos)
|
|
|
|
| Self::Dot(_, pos)
|
|
|
|
| Self::Index(_, pos)
|
|
|
|
| Self::Variable(_, pos, _)
|
2021-06-08 08:46:49 +02:00
|
|
|
| Self::Stack(_, pos)
|
2021-04-23 08:24:53 +02:00
|
|
|
| Self::FnCall(_, pos)
|
|
|
|
| Self::Custom(_, pos) => *pos = new_pos,
|
|
|
|
|
2021-04-04 07:13:07 +02:00
|
|
|
Self::InterpolatedString(x) => {
|
2021-05-22 13:14:24 +02:00
|
|
|
x.first_mut()
|
|
|
|
.expect("never fails because an interpolated string always contains at least one item")
|
|
|
|
.set_position(new_pos);
|
2021-04-04 07:13:07 +02:00
|
|
|
}
|
2021-04-23 08:24:53 +02:00
|
|
|
|
2021-05-18 14:12:30 +02:00
|
|
|
Self::Property(x) => (x.2).1 = new_pos,
|
2021-04-16 06:04:33 +02:00
|
|
|
Self::Stmt(x) => x.1 = new_pos,
|
2020-10-29 04:37:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
self
|
|
|
|
}
|
|
|
|
/// Is the expression pure?
|
|
|
|
///
|
|
|
|
/// A pure expression has no side effects.
|
2021-03-10 06:32:09 +01:00
|
|
|
#[inline]
|
2020-10-29 04:37:51 +01:00
|
|
|
pub fn is_pure(&self) -> bool {
|
|
|
|
match self {
|
2021-04-04 07:13:07 +02:00
|
|
|
Self::InterpolatedString(x) | Self::Array(x, _) => x.iter().all(Self::is_pure),
|
2020-10-31 16:26:21 +01:00
|
|
|
|
2021-03-23 11:25:40 +01:00
|
|
|
Self::Map(x, _) => x.0.iter().map(|(_, v)| v).all(Self::is_pure),
|
2020-10-29 04:37:51 +01:00
|
|
|
|
2021-04-24 07:42:30 +02:00
|
|
|
Self::And(x, _) | Self::Or(x, _) => x.lhs.is_pure() && x.rhs.is_pure(),
|
2020-10-29 04:37:51 +01:00
|
|
|
|
2021-04-16 06:04:33 +02:00
|
|
|
Self::Stmt(x) => x.0.iter().all(Stmt::is_pure),
|
2020-10-29 04:37:51 +01:00
|
|
|
|
2021-06-08 08:46:49 +02:00
|
|
|
Self::Variable(_, _, _) | Self::Stack(_, _) => true,
|
2020-10-29 04:37:51 +01:00
|
|
|
|
|
|
|
_ => self.is_constant(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/// Is the expression the unit `()` literal?
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn is_unit(&self) -> bool {
|
|
|
|
match self {
|
|
|
|
Self::Unit(_) => true,
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/// Is the expression a constant?
|
2021-03-10 06:32:09 +01:00
|
|
|
#[inline]
|
2020-10-29 04:37:51 +01:00
|
|
|
pub fn is_constant(&self) -> bool {
|
|
|
|
match self {
|
|
|
|
#[cfg(not(feature = "no_float"))]
|
2020-10-31 07:13:45 +01:00
|
|
|
Self::FloatConstant(_, _) => true,
|
2020-10-29 04:37:51 +01:00
|
|
|
|
2020-11-14 12:04:49 +01:00
|
|
|
Self::DynamicConstant(_, _)
|
2020-11-19 15:29:50 +01:00
|
|
|
| Self::BoolConstant(_, _)
|
2020-11-14 12:04:49 +01:00
|
|
|
| Self::IntegerConstant(_, _)
|
2020-10-31 07:13:45 +01:00
|
|
|
| Self::CharConstant(_, _)
|
2020-11-13 03:43:54 +01:00
|
|
|
| Self::StringConstant(_, _)
|
2021-06-08 08:46:49 +02:00
|
|
|
| Self::Unit(_)
|
|
|
|
| Self::Stack(_, _) => true,
|
2020-10-29 04:37:51 +01:00
|
|
|
|
2021-04-04 07:13:07 +02:00
|
|
|
Self::InterpolatedString(x) | Self::Array(x, _) => x.iter().all(Self::is_constant),
|
2020-10-29 04:37:51 +01:00
|
|
|
|
2021-03-23 11:25:40 +01:00
|
|
|
Self::Map(x, _) => x.0.iter().map(|(_, expr)| expr).all(Self::is_constant),
|
2020-10-29 04:37:51 +01:00
|
|
|
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
2020-11-25 02:36:06 +01:00
|
|
|
/// Is a particular [token][Token] allowed as a postfix operator to this expression?
|
2021-03-10 06:32:09 +01:00
|
|
|
#[inline]
|
2020-10-29 04:37:51 +01:00
|
|
|
pub fn is_valid_postfix(&self, token: &Token) -> bool {
|
2020-12-27 09:50:48 +01:00
|
|
|
match token {
|
|
|
|
#[cfg(not(feature = "no_object"))]
|
|
|
|
Token::Period => return true,
|
|
|
|
_ => (),
|
|
|
|
}
|
|
|
|
|
2020-10-29 04:37:51 +01:00
|
|
|
match self {
|
|
|
|
#[cfg(not(feature = "no_float"))]
|
2020-10-31 07:13:45 +01:00
|
|
|
Self::FloatConstant(_, _) => false,
|
2020-10-29 04:37:51 +01:00
|
|
|
|
2020-11-14 12:04:49 +01:00
|
|
|
Self::DynamicConstant(_, _)
|
2020-11-19 15:29:50 +01:00
|
|
|
| Self::BoolConstant(_, _)
|
2020-10-31 07:13:45 +01:00
|
|
|
| Self::CharConstant(_, _)
|
2020-10-31 16:26:21 +01:00
|
|
|
| Self::And(_, _)
|
|
|
|
| Self::Or(_, _)
|
2020-10-29 04:37:51 +01:00
|
|
|
| Self::Unit(_) => false,
|
|
|
|
|
2021-06-02 08:29:18 +02:00
|
|
|
Self::IntegerConstant(_, _)
|
|
|
|
| Self::StringConstant(_, _)
|
2021-04-04 07:13:07 +02:00
|
|
|
| Self::InterpolatedString(_)
|
2020-10-31 16:26:21 +01:00
|
|
|
| Self::FnCall(_, _)
|
2021-03-10 15:12:48 +01:00
|
|
|
| Self::Stmt(_)
|
2020-10-31 16:26:21 +01:00
|
|
|
| Self::Dot(_, _)
|
|
|
|
| Self::Index(_, _)
|
|
|
|
| Self::Array(_, _)
|
|
|
|
| Self::Map(_, _) => match token {
|
2020-10-29 04:37:51 +01:00
|
|
|
#[cfg(not(feature = "no_index"))]
|
|
|
|
Token::LeftBracket => true,
|
|
|
|
_ => false,
|
|
|
|
},
|
|
|
|
|
2021-04-05 17:59:15 +02:00
|
|
|
Self::Variable(_, _, _) => match token {
|
2020-10-29 04:37:51 +01:00
|
|
|
#[cfg(not(feature = "no_index"))]
|
|
|
|
Token::LeftBracket => true,
|
|
|
|
Token::LeftParen => true,
|
|
|
|
Token::Bang => true,
|
|
|
|
Token::DoubleColon => true,
|
|
|
|
_ => false,
|
|
|
|
},
|
|
|
|
|
|
|
|
Self::Property(_) => match token {
|
|
|
|
#[cfg(not(feature = "no_index"))]
|
|
|
|
Token::LeftBracket => true,
|
|
|
|
Token::LeftParen => true,
|
|
|
|
_ => false,
|
|
|
|
},
|
|
|
|
|
2020-10-31 16:26:21 +01:00
|
|
|
Self::Custom(_, _) => false,
|
2021-06-08 08:46:49 +02:00
|
|
|
|
|
|
|
Self::Stack(_, _) => unreachable!("Expr::Stack should not occur naturally"),
|
2020-10-29 04:37:51 +01:00
|
|
|
}
|
|
|
|
}
|
2021-01-08 17:24:55 +01:00
|
|
|
/// Recursively walk this expression.
|
2021-03-11 14:55:55 +01:00
|
|
|
/// Return `false` from the callback to terminate the walk.
|
|
|
|
pub fn walk<'a>(
|
|
|
|
&'a self,
|
|
|
|
path: &mut Vec<ASTNode<'a>>,
|
|
|
|
on_node: &mut impl FnMut(&[ASTNode]) -> bool,
|
|
|
|
) -> bool {
|
2021-05-22 13:14:24 +02:00
|
|
|
// Push the current node onto the path
|
2021-01-09 09:52:22 +01:00
|
|
|
path.push(self.into());
|
2021-03-11 14:55:55 +01:00
|
|
|
|
|
|
|
if !on_node(path) {
|
|
|
|
return false;
|
|
|
|
}
|
2021-01-08 17:24:55 +01:00
|
|
|
|
|
|
|
match self {
|
2021-03-11 14:55:55 +01:00
|
|
|
Self::Stmt(x) => {
|
2021-04-16 06:04:33 +02:00
|
|
|
for s in &x.0 {
|
2021-03-11 14:55:55 +01:00
|
|
|
if !s.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-04-04 07:13:07 +02:00
|
|
|
Self::InterpolatedString(x) | Self::Array(x, _) => {
|
2021-03-11 14:55:55 +01:00
|
|
|
for e in x.as_ref() {
|
|
|
|
if !e.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Self::Map(x, _) => {
|
2021-03-23 11:25:40 +01:00
|
|
|
for (_, e) in &x.0 {
|
2021-03-11 14:55:55 +01:00
|
|
|
if !e.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-03-09 06:44:54 +01:00
|
|
|
Self::Index(x, _) | Self::Dot(x, _) | Expr::And(x, _) | Expr::Or(x, _) => {
|
2021-03-11 14:55:55 +01:00
|
|
|
if !x.lhs.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if !x.rhs.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Self::FnCall(x, _) => {
|
|
|
|
for e in &x.args {
|
|
|
|
if !e.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Self::Custom(x, _) => {
|
|
|
|
for e in &x.keywords {
|
|
|
|
if !e.walk(path, on_node) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2021-01-08 17:24:55 +01:00
|
|
|
}
|
|
|
|
_ => (),
|
|
|
|
}
|
2021-01-09 09:52:22 +01:00
|
|
|
|
2021-05-22 13:14:24 +02:00
|
|
|
path.pop()
|
|
|
|
.expect("never fails because `path` always contains the current node");
|
2021-03-11 14:55:55 +01:00
|
|
|
|
|
|
|
true
|
2021-01-08 17:24:55 +01:00
|
|
|
}
|
2020-10-29 04:37:51 +01:00
|
|
|
}
|
2020-11-04 05:34:54 +01:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
/// This test is to make sure no code changes increase the sizes of critical data structures.
|
|
|
|
#[test]
|
|
|
|
fn check_struct_sizes() {
|
2021-01-04 04:58:24 +01:00
|
|
|
use crate::*;
|
2021-04-17 09:15:54 +02:00
|
|
|
use std::mem::size_of;
|
2021-01-04 04:58:24 +01:00
|
|
|
|
|
|
|
assert_eq!(size_of::<Dynamic>(), 16);
|
|
|
|
assert_eq!(size_of::<Option<Dynamic>>(), 16);
|
2021-04-22 17:02:25 +02:00
|
|
|
#[cfg(not(feature = "no_position"))]
|
2021-01-04 04:58:24 +01:00
|
|
|
assert_eq!(size_of::<Position>(), 4);
|
|
|
|
assert_eq!(size_of::<ast::Expr>(), 16);
|
|
|
|
assert_eq!(size_of::<Option<ast::Expr>>(), 16);
|
2021-03-29 07:07:10 +02:00
|
|
|
assert_eq!(size_of::<ast::Stmt>(), 32);
|
|
|
|
assert_eq!(size_of::<Option<ast::Stmt>>(), 32);
|
2021-05-14 13:21:28 +02:00
|
|
|
assert_eq!(
|
|
|
|
size_of::<FnPtr>(),
|
|
|
|
if cfg!(feature = "no_smartstring") {
|
|
|
|
80
|
|
|
|
} else {
|
|
|
|
96
|
|
|
|
}
|
|
|
|
);
|
2021-05-18 06:40:40 +02:00
|
|
|
assert_eq!(size_of::<Scope>(), 160);
|
2021-01-04 04:58:24 +01:00
|
|
|
assert_eq!(size_of::<LexError>(), 56);
|
2021-04-22 17:02:25 +02:00
|
|
|
assert_eq!(
|
|
|
|
size_of::<ParseError>(),
|
|
|
|
if cfg!(feature = "no_position") { 8 } else { 16 }
|
|
|
|
);
|
2021-01-04 04:58:24 +01:00
|
|
|
assert_eq!(size_of::<EvalAltResult>(), 72);
|
2020-11-04 05:34:54 +01:00
|
|
|
}
|
|
|
|
}
|