snmalloc-sys 0.7.4

rust raw bindings of snmalloc.
Documentation
#pragma once

#include "../ds_core/ds_core.h"

#include <stddef.h>

namespace snmalloc
{
  /**
   * A mixin AAL that applies CHERI to a `Base` architecture.  Gives
   * architectural teeth to the capptr_bound primitive.
   */
  template<typename Base>
  class AAL_CHERI : public Base
  {
  public:
    /**
     * CHERI pointers are not integers and come with strict provenance
     * requirements.
     */
    static constexpr uint64_t aal_features =
      (Base::aal_features & ~IntegerPointers) | StrictProvenance;

    enum AalCheriFeatures : uint64_t
    {
      /**
       * This CHERI flavor traps if the capability input to a bounds-setting
       * instruction has its tag clear, rather than just leaving the output
       * untagged.
       *
       * For example, CHERI-RISC-V's CSetBoundsExact traps in contrast to
       * Morello's SCBNDSE.
       */
      SetBoundsTrapsUntagged = (1 << 0),

      /**
       * This CHERI flavor traps if the capability input to a
       * permissions-masking instruction has its tag clear, rather than just
       * leaving the output untagged.
       *
       * For example, CHERI-RISC-V's CAndPerms traps in contrast to Morello's
       * CLRPERM.
       */
      AndPermsTrapsUntagged = (1 << 0),
    };

    /**
     * Specify "features" of the particular CHERI machine we're running on.
     */
    static constexpr uint64_t aal_cheri_features =
      /* CHERI-RISC-V prefers to trap on untagged inputs.  Morello does not. */
      (Base::aal_name == RISCV ?
         SetBoundsTrapsUntagged | AndPermsTrapsUntagged :
         0);

    /**
     * On CHERI-aware compilers, ptraddr_t is an integral type that is wide
     * enough to hold any address that may be contained within a memory
     * capability.  It does not carry provenance: it is not a capability, but
     * merely an address.
     */
    typedef ptraddr_t address_t;

    template<
      typename T,
      SNMALLOC_CONCEPT(capptr::IsBound) BOut,
      SNMALLOC_CONCEPT(capptr::IsBound) BIn,
      typename U = T>
    static SNMALLOC_FAST_PATH CapPtr<T, BOut>
    capptr_bound(CapPtr<U, BIn> a, size_t size) noexcept
    {
      static_assert(
        capptr::is_spatial_refinement<BIn, BOut>(),
        "capptr_bound must preserve non-spatial CapPtr dimensions");
      SNMALLOC_ASSERT(__builtin_cheri_tag_get(a.unsafe_ptr()));

      if constexpr (aal_cheri_features & SetBoundsTrapsUntagged)
      {
        if (a == nullptr)
        {
          return nullptr;
        }
      }

      void* pb = __builtin_cheri_bounds_set_exact(a.unsafe_ptr(), size);

      SNMALLOC_ASSERT_MSG(
        __builtin_cheri_tag_get(pb),
        "capptr_bound exactness failed. {} of size {}",
        a.unsafe_ptr(),
        size);

      return CapPtr<T, BOut>::unsafe_from(static_cast<T*>(pb));
    }

    template<
      typename T,
      SNMALLOC_CONCEPT(capptr::IsBound) BOut,
      SNMALLOC_CONCEPT(capptr::IsBound) BIn,
      typename U = T>
    static SNMALLOC_FAST_PATH CapPtr<T, BOut>
    capptr_rebound(CapPtr<T, BOut> a, CapPtr<U, BIn> b) noexcept
    {
      return CapPtr<T, BOut>::unsafe_from(static_cast<T*>(
        __builtin_cheri_address_set(a.unsafe_ptr(), address_cast(b))));
    }

    static SNMALLOC_FAST_PATH size_t capptr_size_round(size_t sz) noexcept
    {
      /*
       * Round up sz to the next representable value for the target
       * architecture's choice of CHERI Concentrate T/B mantissa width.
       *
       * On Morello specifically, this intrinsic will (soon, as of this text
       * being written) expand to a multi-instruction sequence to work around a
       * bug in which sz values satisfying $\exists_E sz = ((1 << 12) - 1) <<
       * (E + 3)$ are incorrectly increased.  If for some reason this ends up
       * being at all hot, there will also be a
       * __builtin_morello_round_representable_length_inexact, which will just
       * return too big of a size for those values (by rounding up to $1 << (E
       * + 15)$).  While technically incorrect, this behavior is probably fine
       * for snmalloc: we already slot metadata allocations into NAPOT holes
       * and then return any unused space at the top, so the over-reach would,
       * at the worst, just prevent said return, and our sizeclasses never run
       * into this issue.  That is, we're clear to use the __builtin_morello_*
       * intrinsic if the multi-instruction sequence proves slow.  See
       * https://git.morello-project.org/morello/llvm-project/-/merge_requests/199
       */
      return __builtin_cheri_round_representable_length(sz);
    }
  };
} // namespace snmalloc