Skip to content

Commit

Permalink
Merge pull request #36 from ecstatic-morse/rel-join-empty-tuple
Browse files Browse the repository at this point in the history
Allow `Relation<Key>` to join with `Variable<(Key, Value)>` directly
  • Loading branch information
lqd authored Aug 30, 2021
2 parents 3d18dec + 586043f commit 31e8fa8
Showing 1 changed file with 38 additions and 17 deletions.
55 changes: 38 additions & 17 deletions src/join.rs
Original file line number Diff line number Diff line change
Expand Up @@ -51,13 +51,13 @@ fn join_delta<'me, Key: Ord, Val1: Ord, Val2: Ord>(
let recent1 = input1.recent();
let recent2 = input2.recent();

for batch2 in input2.stable().iter() {
input2.for_each_stable_set(|batch2| {
join_helper(&recent1, &batch2, &mut result);
}
});

for batch1 in input1.stable().iter() {
input1.for_each_stable_set(|batch1| {
join_helper(&batch1, &recent2, &mut result);
}
});

join_helper(&recent1, &recent2, &mut result);
}
Expand Down Expand Up @@ -164,40 +164,61 @@ pub trait JoinInput<'me, Tuple: Ord>: Copy {
/// empty slice.)
type RecentTuples: Deref<Target = [Tuple]>;

/// If we are on iteration N of the loop, these are the tuples
/// added on iteration N - 2 or before. (For a `Relation`, this is
/// just `self`.)
type StableTuples: Deref<Target = [Relation<Tuple>]>;

/// Get the set of recent tuples.
fn recent(self) -> Self::RecentTuples;

/// Get the set of stable tuples.
fn stable(self) -> Self::StableTuples;
/// Call a function for each set of stable tuples.
fn for_each_stable_set(self, f: impl FnMut(&[Tuple]));
}

impl<'me, Tuple: Ord> JoinInput<'me, Tuple> for &'me Variable<Tuple> {
type RecentTuples = Ref<'me, [Tuple]>;
type StableTuples = Ref<'me, [Relation<Tuple>]>;

fn recent(self) -> Self::RecentTuples {
Ref::map(self.recent.borrow(), |r| &r.elements[..])
}

fn stable(self) -> Self::StableTuples {
Ref::map(self.stable.borrow(), |v| &v[..])
fn for_each_stable_set(self, mut f: impl FnMut(&[Tuple])) {
for stable in self.stable.borrow().iter() {
f(stable)
}
}
}

impl<'me, Tuple: Ord> JoinInput<'me, Tuple> for &'me Relation<Tuple> {
type RecentTuples = &'me [Tuple];
type StableTuples = &'me [Relation<Tuple>];

fn recent(self) -> Self::RecentTuples {
&[]
}

fn stable(self) -> Self::StableTuples {
std::slice::from_ref(self)
fn for_each_stable_set(self, mut f: impl FnMut(&[Tuple])) {
f(&self.elements)
}
}

impl<'me, Tuple: Ord> JoinInput<'me, (Tuple, ())> for &'me Relation<Tuple> {
type RecentTuples = &'me [(Tuple, ())];

fn recent(self) -> Self::RecentTuples {
&[]
}

fn for_each_stable_set(self, mut f: impl FnMut(&[(Tuple, ())])) {
use std::mem;
assert_eq!(mem::size_of::<(Tuple, ())>(), mem::size_of::<Tuple>());
assert_eq!(mem::align_of::<(Tuple, ())>(), mem::align_of::<Tuple>());

// SAFETY: https://rust-lang.github.io/unsafe-code-guidelines/layout/structs-and-tuples.html#structs-with-1-zst-fields
// guarantees that `T` is layout compatible with `(T, ())`, since `()` is a 1-ZST. We use
// `slice::from_raw_parts` because the layout compatibility guarantee does not extend to
// containers like `&[T]`.
let elements: &'me [Tuple] = self.elements.as_slice();
let len = elements.len();

let elements: &'me [(Tuple, ())] =
unsafe { std::slice::from_raw_parts(elements.as_ptr() as *const _, len) };

f(elements)
}
}

0 comments on commit 31e8fa8

Please sign in to comment.