use crate::mmtk::SFT_MAP;
use crate::plan::{ObjectQueue, VectorObjectQueue};
use crate::policy::sft::GCWorkerMutRef;
use crate::policy::sft::SFT;
use crate::policy::space::{CommonSpace, Space};
use crate::util::address::Address;
use crate::util::constants::BYTES_IN_PAGE;
use crate::util::heap::externalpageresource::{ExternalPageResource, ExternalPages};
use crate::util::heap::layout::vm_layout::BYTES_IN_CHUNK;
use crate::util::heap::PageResource;
use crate::util::metadata::mark_bit::MarkState;
#[cfg(feature = "set_unlog_bits_vm_space")]
use crate::util::metadata::MetadataSpec;
use crate::util::object_enum::ObjectEnumerator;
use crate::util::opaque_pointer::*;
use crate::util::ObjectReference;
use crate::vm::{ObjectModel, VMBinding};
use std::sync::atomic::Ordering;
pub struct VMSpace<VM: VMBinding> {
mark_state: MarkState,
common: CommonSpace<VM>,
pr: ExternalPageResource<VM>,
}
impl<VM: VMBinding> SFT for VMSpace<VM> {
fn name(&self) -> &'static str {
self.common.name
}
fn is_live(&self, _object: ObjectReference) -> bool {
true
}
fn is_reachable(&self, object: ObjectReference) -> bool {
self.mark_state.is_marked::<VM>(object)
}
#[cfg(feature = "object_pinning")]
fn pin_object(&self, _object: ObjectReference) -> bool {
false
}
#[cfg(feature = "object_pinning")]
fn unpin_object(&self, _object: ObjectReference) -> bool {
false
}
#[cfg(feature = "object_pinning")]
fn is_object_pinned(&self, _object: ObjectReference) -> bool {
true
}
fn is_movable(&self) -> bool {
false
}
#[cfg(feature = "sanity")]
fn is_sane(&self) -> bool {
true
}
fn initialize_object_metadata(&self, object: ObjectReference, _alloc: bool) {
self.mark_state
.on_object_metadata_initialization::<VM>(object);
if self.common.needs_log_bit {
VM::VMObjectModel::GLOBAL_LOG_BIT_SPEC.mark_as_unlogged::<VM>(object, Ordering::SeqCst);
}
#[cfg(feature = "vo_bit")]
crate::util::metadata::vo_bit::set_vo_bit(object);
}
#[cfg(feature = "is_mmtk_object")]
fn is_mmtk_object(&self, addr: Address) -> Option<ObjectReference> {
crate::util::metadata::vo_bit::is_vo_bit_set_for_addr(addr)
}
#[cfg(feature = "is_mmtk_object")]
fn find_object_from_internal_pointer(
&self,
ptr: Address,
max_search_bytes: usize,
) -> Option<ObjectReference> {
crate::util::metadata::vo_bit::find_object_from_internal_pointer::<VM>(
ptr,
max_search_bytes,
)
}
fn sft_trace_object(
&self,
queue: &mut VectorObjectQueue,
object: ObjectReference,
_worker: GCWorkerMutRef,
) -> ObjectReference {
self.trace_object(queue, object)
}
}
impl<VM: VMBinding> Space<VM> for VMSpace<VM> {
fn as_space(&self) -> &dyn Space<VM> {
self
}
fn as_sft(&self) -> &(dyn SFT + Sync + 'static) {
self
}
fn get_page_resource(&self) -> &dyn PageResource<VM> {
&self.pr
}
fn maybe_get_page_resource_mut(&mut self) -> Option<&mut dyn PageResource<VM>> {
Some(&mut self.pr)
}
fn common(&self) -> &CommonSpace<VM> {
&self.common
}
fn initialize_sft(&self, sft_map: &mut dyn crate::policy::sft_map::SFTMap) {
let vm_regions = self.pr.get_external_pages();
assert!(vm_regions.len() <= 1);
for external_pages in vm_regions.iter() {
let start = external_pages.start.align_down(BYTES_IN_CHUNK);
let size = external_pages.end.align_up(BYTES_IN_CHUNK) - start;
debug_assert_eq!(
sft_map.get_checked(start).name(),
crate::policy::sft::EMPTY_SFT_NAME
);
assert!(sft_map.has_sft_entry(start), "The VM space start (aligned to {}) does not have a valid SFT entry. Possibly the address range is not in the address range we use.", start);
unsafe {
sft_map.eager_initialize(self.as_sft(), start, size);
}
}
}
fn release_multiple_pages(&mut self, _start: Address) {
unreachable!()
}
fn acquire(&self, _tls: VMThread, _pages: usize) -> Address {
unreachable!()
}
fn address_in_space(&self, start: Address) -> bool {
SFT_MAP.get_checked(start).name() == self.name()
}
fn enumerate_objects(&self, enumerator: &mut dyn ObjectEnumerator) {
let external_pages = self.pr.get_external_pages();
for ep in external_pages.iter() {
enumerator.visit_address_range(ep.start, ep.end);
}
}
}
use crate::scheduler::GCWorker;
use crate::util::copy::CopySemantics;
impl<VM: VMBinding> crate::policy::gc_work::PolicyTraceObject<VM> for VMSpace<VM> {
fn trace_object<Q: ObjectQueue, const KIND: crate::policy::gc_work::TraceKind>(
&self,
queue: &mut Q,
object: ObjectReference,
_copy: Option<CopySemantics>,
_worker: &mut GCWorker<VM>,
) -> ObjectReference {
self.trace_object(queue, object)
}
fn may_move_objects<const KIND: crate::policy::gc_work::TraceKind>() -> bool {
false
}
}
impl<VM: VMBinding> VMSpace<VM> {
pub fn new(args: crate::policy::space::PlanCreateSpaceArgs<VM>) -> Self {
let (vm_space_start, vm_space_size) =
(*args.options.vm_space_start, *args.options.vm_space_size);
let space = Self {
mark_state: MarkState::new(),
pr: ExternalPageResource::new(args.vm_map),
common: CommonSpace::new(args.into_policy_args(
false,
true,
crate::util::metadata::extract_side_metadata(&[
*VM::VMObjectModel::LOCAL_MARK_BIT_SPEC,
]),
)),
};
if !vm_space_start.is_zero() {
space.set_vm_region_inner(vm_space_start, vm_space_size, false);
}
space
}
pub fn set_vm_region(&mut self, start: Address, size: usize) {
self.set_vm_region_inner(start, size, true);
}
fn set_vm_region_inner(&self, start: Address, size: usize, set_sft: bool) {
assert!(size > 0);
assert!(!start.is_zero());
let end = start + size;
let chunk_start = start.align_down(BYTES_IN_CHUNK);
let chunk_end = end.align_up(BYTES_IN_CHUNK);
let chunk_size = chunk_end - chunk_start;
assert!(Address::range_intersection(
&(chunk_start..chunk_end),
&crate::util::heap::layout::available_range()
)
.is_empty());
debug!(
"Align VM space ({}, {}) to chunk ({}, {})",
start, end, chunk_start, chunk_end
);
self.common.mmapper.mark_as_mapped(chunk_start, chunk_size);
self.common
.metadata
.try_map_metadata_space(chunk_start, chunk_size, self.get_name())
.unwrap();
if set_sft {
assert!(SFT_MAP.has_sft_entry(chunk_start), "The VM space start (aligned to {}) does not have a valid SFT entry. Possibly the address range is not in the address range we use.", chunk_start);
unsafe {
SFT_MAP.update(self.as_sft(), chunk_start, chunk_size);
}
}
self.pr.add_new_external_pages(ExternalPages {
start: start.align_down(BYTES_IN_PAGE),
end: end.align_up(BYTES_IN_PAGE),
});
#[cfg(feature = "set_unlog_bits_vm_space")]
if self.common.needs_log_bit {
if let MetadataSpec::OnSide(side) = *VM::VMObjectModel::GLOBAL_LOG_BIT_SPEC {
side.bset_metadata(start, size);
}
}
}
pub fn prepare(&mut self) {
self.mark_state.on_global_prepare::<VM>();
for external_pages in self.pr.get_external_pages().iter() {
self.mark_state.on_block_reset::<VM>(
external_pages.start,
external_pages.end - external_pages.start,
);
}
}
pub fn release(&mut self) {
self.mark_state.on_global_release::<VM>();
}
pub fn trace_object<Q: ObjectQueue>(
&self,
queue: &mut Q,
object: ObjectReference,
) -> ObjectReference {
#[cfg(feature = "vo_bit")]
debug_assert!(
crate::util::metadata::vo_bit::is_vo_bit_set(object),
"{:x}: VO bit not set",
object
);
debug_assert!(self.in_space(object));
if self.mark_state.test_and_mark::<VM>(object) {
queue.enqueue(object);
}
object
}
}