// Copyright 2023 The Chromium Authors // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include #include #include #include "build/rust/tests/test_rust_shared_library/src/lib.rs.h" #include "partition_alloc/buildflags.h" #include "testing/gtest/include/gtest/gtest.h" #if PA_BUILDFLAG(HAS_64_BIT_POINTERS) #include "partition_alloc/partition_address_space.h" #else #include "partition_alloc/address_pool_manager_bitmap.h" #endif TEST(RustSharedTest, CppCallingIntoRust_BasicFFI) { EXPECT_EQ(7, add_two_ints_via_rust(3, 4)); } TEST(RustSharedTest, RustComponentUsesPartitionAlloc) { // Verify that PartitionAlloc is consistently used in C++ and Rust. auto cpp_allocated_int = std::make_unique(); SomeStruct* rust_allocated_ptr = allocate_via_rust().into_raw(); EXPECT_EQ(partition_alloc::IsManagedByPartitionAlloc( reinterpret_cast(rust_allocated_ptr)), partition_alloc::IsManagedByPartitionAlloc( reinterpret_cast(cpp_allocated_int.get()))); rust::Box::from_raw(rust_allocated_ptr); } TEST(RustSharedTest, AllocAligned) { alloc_aligned(); } // TODO(crbug.com/343218479): A rust_shared_library target does not use our // allocation shim so it is not properly redirected to PartitionAlloc (though is // happens to use PA on Linux since the std allocator calls malloc). To enable // this, we need to insert the allocation shim into `rust_shared_library`, // probably by having the template produce a Rust static library and having // clang do the linking step. TEST(RustSharedTest, DISABLED_RustLargeAllocationFailure) { // If allocation succeeds, we get true back. EXPECT_TRUE(allocate_huge_via_rust(100u, 1u)); #if PA_BUILDFLAG(USE_PARTITION_ALLOC_AS_MALLOC) // We only do these tests when using PA, as the system allocator will not fail // on large allocations (unless it is really OOM). // PartitionAlloc currently limits all allocations to no more than i32::MAX // elements, so the allocation will fail. If done through normal malloc(), // PA will crash when an allocation fails rather than return null, but Rust // can be trusted to handle failure without introducing null derefs so this // should fail gracefully. size_t max_size = partition_alloc::internal::MaxDirectMapped(); EXPECT_FALSE(allocate_huge_via_rust(max_size + 1u, 4u)); // Same as above but with an alignment larger than PartitionAlloc's default // alignment, which goes down a different path. size_t big_alignment = alignof(std::max_align_t) * 2u; EXPECT_FALSE(allocate_huge_via_rust(max_size + 1u, big_alignment)); // PartitionAlloc will crash if given an alignment larger than this. The // allocation hooks handle it gracefully. size_t max_alignment = partition_alloc::internal::kMaxSupportedAlignment; EXPECT_FALSE(allocate_huge_via_rust(100u, max_alignment * 2u)); #endif }