wasmtime 42.0.2

High-level API to expose the Wasmtime runtime
Documentation
//! Construction of the call-graph, for the purposes of inlining.
//!
//! These call graphs are not necessarily complete or accurate, and Wasmtime's
//! soundness does not rely on those properties. First off, we do not attempt to
//! understand indirect calls, which at their worst must force any call analysis
//! give up and say "the callee could be absolutely any function". More
//! interestingly, these call graphs are only used for scheduling bottom-up
//! inlining, so the worst that inaccurate information can do is cause us to
//! miss inlining opportunities or lose potential parallelism in our
//! schedule. For best results, however, every direct call that is potentially
//! inlinable should be reported when constructing these call graphs.

use super::*;
use core::{
    fmt::{self, Debug},
    ops::Range,
};
use wasmtime_environ::{EntityRef, SecondaryMap};

/// A call graph reified into a densely packed and quickly accessible
/// representation.
///
/// In a call graph, nodes are functions, and an edge `f --> g` means that the
/// function `f` calls the function `g`.
pub struct CallGraph<Node>
where
    Node: EntityRef + Debug,
{
    /// A map from each node to the subslice of `self.edge_elems` that are its
    /// edges.
    edges: SecondaryMap<Node, Range<u32>>,

    /// Densely packed edge elements for `self.edges`.
    edge_elems: Vec<Node>,
}

impl<Node> Debug for CallGraph<Node>
where
    Node: EntityRef + Debug,
{
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        struct Edges<'a, Node: EntityRef + Debug>(&'a CallGraph<Node>);

        impl<'a, Node: EntityRef + Debug> Debug for Edges<'a, Node> {
            fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
                f.debug_map()
                    .entries(self.0.nodes().map(|n| (n, self.0.edges(n))))
                    .finish()
            }
        }

        f.debug_struct("CallGraph")
            .field("edges", &Edges(self))
            .finish()
    }
}

impl<Node> CallGraph<Node>
where
    Node: EntityRef + Debug,
{
    /// Construct a new call graph.
    ///
    /// `funcs` should be an iterator over all function nodes in this call
    /// graph's translation unit.
    ///
    /// The `get_calls` function should yield (by pushing onto the given `Vec`)
    /// all of the callee function nodes that the given caller function node
    /// calls.
    pub fn new(
        funcs: impl IntoIterator<Item = Node>,
        mut get_calls: impl FnMut(Node, &mut Vec<Node>) -> Result<()>,
    ) -> Result<Self> {
        let funcs = funcs.into_iter();

        let (min, max) = funcs.size_hint();
        let capacity = max.unwrap_or_else(|| 2 * min);
        let mut edges = SecondaryMap::with_capacity(capacity);
        let mut edge_elems = vec![];

        let mut calls = vec![];
        for caller in funcs {
            debug_assert!(calls.is_empty());
            get_calls(caller, &mut calls)?;

            debug_assert_eq!(edges[caller], Range::default());
            edges[caller] = extend_with_range(&mut edge_elems, calls.drain(..));
        }

        Ok(CallGraph { edges, edge_elems })
    }

    /// Get the function nodes in this call graph.
    pub fn nodes(&self) -> impl ExactSizeIterator<Item = Node> {
        self.edges.keys()
    }

    /// Get the callee function nodes that the given caller function node calls.
    pub fn edges(&self, node: Node) -> &[Node] {
        let Range { start, end } = self.edges[node].clone();
        let start = usize::try_from(start).unwrap();
        let end = usize::try_from(end).unwrap();
        &self.edge_elems[start..end]
    }
}