From 3aebc985de4be9781e82dd817f2dcd60985dd9a2 Mon Sep 17 00:00:00 2001 From: Kirpal Grewal <45569241+KGrewal1@users.noreply.github.com> Date: Thu, 5 Dec 2024 13:39:28 +0000 Subject: [PATCH] use cargo rdme for tree arena (#769) Use `cargo rdme` for crate readme and check in CI as mentioned in https://github.com/linebender/xilem/pull/752#discussion_r1867737085 --- .github/workflows/ci.yml | 12 ++++--- tree_arena/README.md | 45 ++++++++++++++++++++++----- tree_arena/src/lib.rs | 67 ++++++++++++++++++++++++++++++++++++---- 3 files changed, 106 insertions(+), 18 deletions(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index c2e3edd2d..b7071f009 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -96,10 +96,14 @@ jobs: with: tool: cargo-rdme - - name: cargo rdme + - name: cargo rdme masonry run: | cargo rdme --check --workspace-project=masonry + - name: cargo rdme tree_arena + run: | + cargo rdme --check --heading-base-level=0 --workspace-project=tree_arena + clippy-stable: name: cargo clippy runs-on: ${{ matrix.os }} @@ -195,7 +199,7 @@ jobs: steps: - name: Checkout repository uses: actions/checkout@v4 - + - name: Cache git lfs id: lfs-cache uses: actions/cache@v4 @@ -209,7 +213,7 @@ jobs: - name: Fetch lfs data if: ${{ steps.lfs-cache.outputs.cache-hit != 'true' }} run: git lfs fetch - + test-stable: name: cargo test needs: prime-lfs-cache @@ -279,7 +283,7 @@ jobs: # because those require Vello rendering to be working # See also https://github.com/linebender/vello/pull/610 SKIP_RENDER_TESTS: ${{ matrix.skip_gpu }} - + - name: Upload test results due to failure uses: actions/upload-artifact@v4 if: failure() diff --git a/tree_arena/README.md b/tree_arena/README.md index 675401701..9c4e2817c 100644 --- a/tree_arena/README.md +++ b/tree_arena/README.md @@ -1,20 +1,35 @@ # Tree Arena -This crate contains two implementations of a tree for use in masonry, one safe and the other unsafe. The safe tree is known to work, and serves as the baseline implementation and is used by default. The unsafe tree leverages a hashmap as an arena and is designed for higher performance: it leverages unsafe code to achieve this. The unsafe tree is not yet fully tested, and is not used by default. +[![Apache 2.0 license.](https://img.shields.io/badge/license-Apache--2.0-blue.svg)](#license) +[![Linebender Zulip chat.](https://img.shields.io/badge/Linebender-%23masonry-blue?logo=Zulip)](https://xi.zulipchat.com/#narrow/stream/317477-masonry) +[![GitHub Actions CI status.](https://img.shields.io/github/actions/workflow/status/linebender/xilem/ci.yml?logo=github&label=CI)](https://github.com/linebender/xilem/actions) + + + + + +This crate contains two implementations of a tree for use in [Masonry], one safe and the other unsafe. The safe tree is known to work, and serves as the baseline implementation and is used by default. +The unsafe tree leverages a hashmap as an arena and is designed for higher performance: it leverages unsafe code to achieve this. The unsafe tree is not yet fully tested, and is not used by default. The safe tree is the priority. This means: * The safe version may have features / APIs that the unsafe version doesn't yet have. -* If both versions are at feature parity, Masonry can switch on the unsafe version for best performance. +* If both versions are at feature parity, [Masonry] can switch on the unsafe version for best performance. -* Otherwise, Masonry uses the safe version. +* Otherwise, [Masonry] uses the safe version. ## Architecture ### Safe Tree -The safe tree contains a root `TreeArena` which owns the root nodes as `Vec>`, and a`parents_map` tracking the parent of every node. Each `TreeNode` subsequently owns its own children as `Vec>`. This model of owneship is thus checked by the rust compiler, but has the downside of requiring passing through every ancestor node to access the descendant - this requires an O(depth) determination of whether the node is a descendant, followed by O(children) time at each level to traverse the path to the child. +The safe tree contains a root `TreeArena` which owns the root nodes as `Vec>`, and a`parents_map` tracking the parent of every node. +Each `TreeNode` subsequently owns its own children as `Vec>`. This model of owneship is thus checked by the rust compiler, +but has the downside of requiring passing through every ancestor node to access the descendant - +this requires an O(depth) determination of whether the node is a descendant, followed by O(children) time at each level to traverse the path to the child. ### Unsafe Tree @@ -26,19 +41,29 @@ The unsafe tree arena contains a `DataMap` which **owns** all nodes. The `DataMa * `Box>>` containing the roots of the tree -It is possible to get shared (immutable) access or exclusive (mutable) access to the tree. These return `ArenaRef<'arena, T>` or `ArenaMut<'arena, T>` respectively. We do this by leveraging a hash map to store the nodes: from this we can obtain either shared or exclusive access to nodes. To ensure that only one item is allowed to create new exclusive access to nodes, this action requires mutable access to the arena as a whole (and so is checked by the compiler) - what the compiler cannot check is that the nodes accessed mutably are distinct from one another - this is done by only allowing access to descendants of the node being accessed mutably. The aim of this is to reduce the time needed to access node, as given a node, we only need to determine whether it is a descendant of the node being accessed mutably, and do not need to iterate over the children and to flatten the overall tree graph into a hash map. +It is possible to get shared (immutable) access or exclusive (mutable) access to the tree. These return `ArenaRef<'arena, T>` or `ArenaMut<'arena, T>` respectively. +We do this by leveraging a hash map to store the nodes: from this we can obtain either shared or exclusive access to nodes. +To ensure that only one item is allowed to create new exclusive access to nodes, this action requires mutable access to the arena as a whole (and so is checked by the compiler) - +what the compiler cannot check is that the nodes accessed mutably are distinct from one another - this is done by only allowing access to descendants of the node being accessed mutably. +The aim of this is to reduce the time needed to access node, as given a node, we only need to determine whether it is a descendant of the node being accessed mutably, +and do not need to iterate over the children and to flatten the overall tree graph into a hash map. #### Shared References -`ArenaRef<'arena, T>` contains the identity of the parent node, a reference to the node data, and `ArenaRefChildren<'arena, T>`. The `ArenaRefChildren<'arena, T>` contains the ids of the children of the node, the id of the node, and a reference to the arena. From this `ArenaRefChildren<'arena, T>` it is possible to get shared access to children of the node. +`ArenaRef<'arena, T>` contains the identity of the parent node, a reference to the node data, and `ArenaRefChildren<'arena, T>`. +The `ArenaRefChildren<'arena, T>` contains the ids of the children of the node, the id of the node, and a reference to the arena. From this `ArenaRefChildren<'arena, T>` it is possible to get shared access to children of the node. #### Exclusive References -`ArenaMut<'arena, T>` contains the identity of the parent node, a mutable reference to the node data, and `ArenaMutChildren<'arena, T>`. The `ArenaMutChildren<'arena, T>` contains the ids of the children of the node, the id of the node, and a mutable reference to the arena. From this `ArenaMutChildren<'arena, T>` it is possible to get exclusive access to children of the node. +`ArenaMut<'arena, T>` contains the identity of the parent node, a mutable reference to the node data, and `ArenaMutChildren<'arena, T>`. +The `ArenaMutChildren<'arena, T>` contains the ids of the children of the node, the id of the node, and a mutable reference to the arena. +From this `ArenaMutChildren<'arena, T>` it is possible to get exclusive access to children of the node. #### Safety -From the `ArenaMutChildren<'arena, T>`, it is important that we can only access descendants of that node, such that we can only ever have exclusive mutable access to the contents of a node, and never have multiple mutable references. This invariant is not checked by the compiler and thus relies on the logic to determine whether a node is a descendant being correct. +From the `ArenaMutChildren<'arena, T>`, it is important that we can only access descendants of that node, +such that we can only ever have exclusive mutable access to the contents of a node, and never have multiple mutable references. +This invariant is not checked by the compiler and thus relies on the logic to determine whether a node is a descendant being correct. ### Complexity @@ -47,3 +72,7 @@ From the `ArenaMutChildren<'arena, T>`, it is important that we can only access |Find child | O(Children) | O(1) | |Descendant | O(Depth) | O(Depth) | |From root | O(Depth) | O(1) | + +[Masonry]: https://crates.io/crates/masonry + + diff --git a/tree_arena/src/lib.rs b/tree_arena/src/lib.rs index 069eeeb93..c5f861768 100644 --- a/tree_arena/src/lib.rs +++ b/tree_arena/src/lib.rs @@ -1,15 +1,70 @@ // Copyright 2024 the Xilem Authors // SPDX-License-Identifier: Apache-2.0 -//! This crate implements a tree data structure for use in Masonry -//! It contains both a safe implementation (that is used by default) -//! and an unsafe implementation that can be used to improve performance +//! This crate contains two implementations of a tree for use in [Masonry], one safe and the other unsafe. The safe tree is known to work, and serves as the baseline implementation and is used by default. +//! The unsafe tree leverages a hashmap as an arena and is designed for higher performance: it leverages unsafe code to achieve this. The unsafe tree is not yet fully tested, and is not used by default. //! -//! The safe version is the first class citizen +//! The safe tree is the priority. This means: //! //! * The safe version may have features / APIs that the unsafe version doesn't yet have. -//! * If both versions are at feature parity, Masonry can switch on the unsafe version for best performance. -//! * Otherwise, Masonry uses the safe version. +//! +//! * If both versions are at feature parity, [Masonry] can switch on the unsafe version for best performance. +//! +//! * Otherwise, [Masonry] uses the safe version. +//! +//! ## Architecture +//! +//! ### Safe Tree +//! +//! The safe tree contains a root `TreeArena` which owns the root nodes as `Vec>`, and a`parents_map` tracking the parent of every node. +//! Each `TreeNode` subsequently owns its own children as `Vec>`. This model of owneship is thus checked by the rust compiler, +//! but has the downside of requiring passing through every ancestor node to access the descendant - +//! this requires an O(depth) determination of whether the node is a descendant, followed by O(children) time at each level to traverse the path to the child. +//! +//! ### Unsafe Tree +//! +//! The unsafe tree arena contains a `DataMap` which **owns** all nodes. The `DataMap` contains: +//! +//! * A `HashMap` associating `NodeId` with `Box>>`, owning the node data, (boxed to prevent movement of the node when the `HashMap` is resized and `UnsafeCell` to express the interior mutability) +//! +//! * A `HashMap` associating `NodeId` with `Option`, containing the parent information for the nodes +//! +//! * `Box>>` containing the roots of the tree +//! +//! It is possible to get shared (immutable) access or exclusive (mutable) access to the tree. These return `ArenaRef<'arena, T>` or `ArenaMut<'arena, T>` respectively. +//! We do this by leveraging a hash map to store the nodes: from this we can obtain either shared or exclusive access to nodes. +//! To ensure that only one item is allowed to create new exclusive access to nodes, this action requires mutable access to the arena as a whole (and so is checked by the compiler) - +//! what the compiler cannot check is that the nodes accessed mutably are distinct from one another - this is done by only allowing access to descendants of the node being accessed mutably. +//! The aim of this is to reduce the time needed to access node, as given a node, we only need to determine whether it is a descendant of the node being accessed mutably, +//! and do not need to iterate over the children and to flatten the overall tree graph into a hash map. +//! +//! #### Shared References +//! +//! `ArenaRef<'arena, T>` contains the identity of the parent node, a reference to the node data, and `ArenaRefChildren<'arena, T>`. +//! The `ArenaRefChildren<'arena, T>` contains the ids of the children of the node, the id of the node, and a reference to the arena. From this `ArenaRefChildren<'arena, T>` it is possible to get shared access to children of the node. +//! +//! #### Exclusive References +//! +//! `ArenaMut<'arena, T>` contains the identity of the parent node, a mutable reference to the node data, and `ArenaMutChildren<'arena, T>`. +//! The `ArenaMutChildren<'arena, T>` contains the ids of the children of the node, the id of the node, and a mutable reference to the arena. +//! From this `ArenaMutChildren<'arena, T>` it is possible to get exclusive access to children of the node. +//! +//! #### Safety +//! +//! From the `ArenaMutChildren<'arena, T>`, it is important that we can only access descendants of that node, +//! such that we can only ever have exclusive mutable access to the contents of a node, and never have multiple mutable references. +//! This invariant is not checked by the compiler and thus relies on the logic to determine whether a node is a descendant being correct. +//! +//! ### Complexity +//! +//! |Operation | Safe | Unsafe | +//! | --- | --- | --- | +//! |Find child | O(Children) | O(1) | +//! |Descendant | O(Depth) | O(Depth) | +//! |From root | O(Depth) | O(1) | +//! +//! [Masonry]: https://crates.io/crates/masonry + type NodeId = u64; #[cfg(not(feature = "safe_tree"))]