Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Let the VM customize object forwarding implementation #975

Draft
wants to merge 13 commits into
base: master
Choose a base branch
from
5 changes: 5 additions & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -143,6 +143,11 @@ malloc_counted_size = []
# Count the size of all live objects in GC
count_live_bytes_in_gc = []

# Let the VM implement forwarding states (forwarding bits) and forwarding pointers.
# Useful for VMs that have to use the type tag to encode forwarding states due to the lack of spare
# bits in the header.
vm_forwarding = []

# Do not modify the following line - ci-common.sh matches it
# -- Mutally exclusive features --
# Only one feature from each group can be provided. Otherwise build will fail.
Expand Down
39 changes: 17 additions & 22 deletions src/policy/copyspace.rs
Original file line number Diff line number Diff line change
Expand Up @@ -235,30 +235,25 @@ impl<VM: VMBinding> CopySpace<VM> {
);

trace!("attempting to forward");
let forwarding_status = object_forwarding::attempt_to_forward::<VM>(object);

trace!("checking if object is being forwarded");
if object_forwarding::state_is_forwarded_or_being_forwarded(forwarding_status) {
trace!("... yes it is");
let new_object =
object_forwarding::spin_and_get_forwarded_object::<VM>(object, forwarding_status);
trace!("Returning");
new_object
} else {
trace!("... no it isn't. Copying");
let new_object = object_forwarding::forward_object::<VM>(
object,
semantics.unwrap(),
worker.get_copy_context_mut(),
);
match object_forwarding::ForwardingAttempt::<VM>::attempt(object) {
object_forwarding::ForwardingAttempt::Lost(lost) => {
trace!("... Lost the forwarding race. Another thread is forwarding the object");
let new_object = lost.spin_and_get_forwarded_object();
trace!("... {} is already forwarded to {}", object, new_object);
new_object
}
object_forwarding::ForwardingAttempt::Won(won) => {
trace!("... Won the forwarding race. Forwarding...");
let new_object =
won.forward_object(semantics.unwrap(), worker.get_copy_context_mut());

#[cfg(feature = "vo_bit")]
crate::util::metadata::vo_bit::set_vo_bit::<VM>(new_object);
#[cfg(feature = "vo_bit")]
crate::util::metadata::vo_bit::set_vo_bit::<VM>(new_object);

trace!("Forwarding pointer");
queue.enqueue(new_object);
trace!("Copied [{:?} -> {:?}]", object, new_object);
new_object
queue.enqueue(new_object);
trace!("Copied [{:?} -> {:?}]", object, new_object);
new_object
}
}
}

Expand Down
111 changes: 57 additions & 54 deletions src/policy/immix/immixspace.rs
Original file line number Diff line number Diff line change
Expand Up @@ -583,75 +583,78 @@ impl<VM: VMBinding> ImmixSpace<VM> {
#[cfg(feature = "vo_bit")]
vo_bit::helper::on_trace_object::<VM>(object);

let forwarding_status = object_forwarding::attempt_to_forward::<VM>(object);
if object_forwarding::state_is_forwarded_or_being_forwarded(forwarding_status) {
// We lost the forwarding race as some other thread has set the forwarding word; wait
// until the object has been forwarded by the winner. Note that the object may not
// necessarily get forwarded since Immix opportunistically moves objects.
#[allow(clippy::let_and_return)]
let new_object =
object_forwarding::spin_and_get_forwarded_object::<VM>(object, forwarding_status);
#[cfg(debug_assertions)]
{
if new_object == object {
debug_assert!(
match object_forwarding::ForwardingAttempt::<VM>::attempt(object) {
object_forwarding::ForwardingAttempt::Lost(lost) => {
// We lost the forwarding race as some other thread has set the forwarding word; wait
// until the object has been forwarded by the winner. Note that the object may not
// necessarily get forwarded since Immix opportunistically moves objects.
#[allow(clippy::let_and_return)]
let new_object = lost.spin_and_get_forwarded_object();

#[cfg(debug_assertions)]
{
if new_object == object {
debug_assert!(
self.is_marked(object) || self.defrag.space_exhausted() || self.is_pinned(object),
"Forwarded object is the same as original object {} even though it should have been copied",
object,
);
} else {
// new_object != object
debug_assert!(
} else {
// new_object != object
debug_assert!(
!Block::containing::<VM>(new_object).is_defrag_source(),
"Block {:?} containing forwarded object {} should not be a defragmentation source",
Block::containing::<VM>(new_object),
new_object,
);
}
}
new_object
}
new_object
} else if self.is_marked(object) {
// We won the forwarding race but the object is already marked so we clear the
// forwarding status and return the unmoved object
object_forwarding::clear_forwarding_bits::<VM>(object);
object
} else {
// We won the forwarding race; actually forward and copy the object if it is not pinned
// and we have sufficient space in our copy allocator
let new_object = if self.is_pinned(object)
|| (!nursery_collection && self.defrag.space_exhausted())
{
self.attempt_mark(object, self.mark_state);
object_forwarding::clear_forwarding_bits::<VM>(object);
Block::containing::<VM>(object).set_state(BlockState::Marked);

#[cfg(feature = "vo_bit")]
vo_bit::helper::on_object_marked::<VM>(object);

object
} else {
// We are forwarding objects. When the copy allocator allocates the block, it should
// mark the block. So we do not need to explicitly mark it here.
object_forwarding::ForwardingAttempt::Won(won) => {
if self.is_marked(object) {
// We won the forwarding race but the object is already marked so we clear the
// forwarding status and return the unmoved object
won.revert();
object
} else {
// We won the forwarding race; actually forward and copy the object if it is not pinned
// and we have sufficient space in our copy allocator
let new_object = if self.is_pinned(object)
|| (!nursery_collection && self.defrag.space_exhausted())
{
self.attempt_mark(object, self.mark_state);
won.revert();
Block::containing::<VM>(object).set_state(BlockState::Marked);

#[cfg(feature = "vo_bit")]
vo_bit::helper::on_object_marked::<VM>(object);

object
} else {
// We are forwarding objects. When the copy allocator allocates the block, it should
// mark the block. So we do not need to explicitly mark it here.

// Clippy complains if the "vo_bit" feature is not enabled.
#[allow(clippy::let_and_return)]
let new_object =
object_forwarding::forward_object::<VM>(object, semantics, copy_context);
// Clippy complains if the "vo_bit" feature is not enabled.
#[allow(clippy::let_and_return)]
let new_object = won.forward_object(semantics, copy_context);

#[cfg(feature = "vo_bit")]
vo_bit::helper::on_object_forwarded::<VM>(new_object);
#[cfg(feature = "vo_bit")]
vo_bit::helper::on_object_forwarded::<VM>(new_object);

new_object
};
debug_assert_eq!(
Block::containing::<VM>(new_object).get_state(),
BlockState::Marked
);
new_object
};
debug_assert_eq!(
Block::containing::<VM>(new_object).get_state(),
BlockState::Marked
);

queue.enqueue(new_object);
debug_assert!(new_object.is_live());
self.unlog_object_if_needed(new_object);
new_object
queue.enqueue(new_object);
debug_assert!(new_object.is_live());
self.unlog_object_if_needed(new_object);
new_object
}
}
}
}

Expand Down
4 changes: 4 additions & 0 deletions src/util/copy/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ use crate::policy::copyspace::CopySpaceCopyContext;
use crate::policy::immix::ImmixSpace;
use crate::policy::immix::{ImmixCopyContext, ImmixHybridCopyContext};
use crate::policy::space::Space;
#[cfg(not(feature = "vm_forwarding"))]
use crate::util::object_forwarding;
use crate::util::opaque_pointer::VMWorkerThread;
use crate::util::{Address, ObjectReference};
Expand Down Expand Up @@ -111,6 +112,9 @@ impl<VM: VMBinding> GCWorkerCopyContext<VM> {
/// * `semantics`: The copy semantic used for the copying.
pub fn post_copy(&mut self, object: ObjectReference, bytes: usize, semantics: CopySemantics) {
// Clear forwarding bits.
// Not used when using VM-side forwarding implementation because the VM binding is supposed
// to ensure the copied object is not in the "being forwarded" or "forwarded" state.
#[cfg(not(feature = "vm_forwarding"))]
object_forwarding::clear_forwarding_bits::<VM>(object);
// If we are copying objects in mature space, we would need to mark the object as mature.
if semantics.is_mature() && self.config.constraints.needs_log_bit {
Expand Down
Loading