Skip to content

Commit

Permalink
Update the Skymap/MOM code, taking into account Kostya (@hombit) feed…
Browse files Browse the repository at this point in the history
…back
  • Loading branch information
fxpineau committed Nov 5, 2024
1 parent d70cfae commit a7f9b7a
Show file tree
Hide file tree
Showing 6 changed files with 113 additions and 99 deletions.
3 changes: 2 additions & 1 deletion Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,8 @@
name = "cdshealpix"
version = "0.7.0"
authors = ["F.-X. Pineau <francois-xavier.pineau@astro.unistra.fr>"]
edition = "2018"
edition = "2021"
rust-version = "1.81"
license = "Apache-2.0 OR MIT"
readme = "README.md"
categories = ["algorithms", "science"]
Expand Down
10 changes: 10 additions & 0 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1858,4 +1858,14 @@ mod tests {
assert!((d5_mas - 620.3647193082197).abs() < 1e-12);
assert!((d6_mas - 591.2475292479544).abs() < 1e-12);
}

#[test]
fn testok_pos_at_depth29() {
let ra_deg = 45.00432028915398_f64;
let de_deg = 0.021047763781174733_f64;
assert_eq!(
29_640_498_453,
nested::hash(29, ra_deg.to_radians(), de_deg.to_radians())
);
}
}
4 changes: 2 additions & 2 deletions src/nested/map/mom/impls/mod.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
//! Module containing MOM implementations in sub-modules
pub mod zvec;
pub mod ported_from_mom_builder;
pub mod ported_from_mom_builder;
pub mod zvec;
124 changes: 53 additions & 71 deletions src/nested/map/mom/impls/zvec.rs
Original file line number Diff line number Diff line change
@@ -1,9 +1,8 @@
use std::cmp::Ordering;
use std::{iter::Map, slice::Iter, vec::IntoIter};
use std::{cmp::Ordering, iter::Map, slice::Iter, vec::IntoIter};

use super::super::{
super::skymap::{SkyMap, SkyMapValue},
Mom, ZUniqHashT,
LhsRhsBoth, Mom, ZUniqHashT,
};

/// Implementation of a MOM in an ordered vector of `(zuniq, values)` tuples.
Expand Down Expand Up @@ -113,7 +112,7 @@ where
fn from_skymap_ref<'s, S, M>(skymap: &'s S, merger: M) -> Self
where
S: SkyMap<'s, HashType = Z, ValueType = V>,
M: Fn(&V, &V, &V, &V) -> Option<V>,
M: Fn(u8, Z, [&V; 4]) -> Option<V>,
V: 's,
{
let depth = skymap.depth();
Expand All @@ -128,19 +127,25 @@ where
// (among the 4 possible values 0, 1, 2 and 3).
if h == expected_next_hash && h & Z::LAST_LAYER_MASK == Z::LAST_LAYER_MASK {
let n = entries.len();
let parent_depth = depth - 1;
let parent_h = h >> 2;
if let Some(combined_value) = merger(
&entries[n - 4].1, // sibling 0
&entries[n - 3].1, // sibling 1
&entries[n - 2].1, // sibling 2
&entries[n - 1].1, // sibling 3
parent_depth,
parent_h,
[
&entries[n - 4].1, // sibling 0
&entries[n - 3].1, // sibling 1
&entries[n - 2].1, // sibling 2
&entries[n - 1].1, // sibling 3
],
) {
let _ = entries.pop();
let _ = entries.pop();
let _ = entries.pop();
// Unwrap ok here since we are sure that the array contained at least 4 entries
// (we access them just above).
let _ = entries.pop().unwrap();
let new_zuniq = Z::to_zuniq(depth - 1, h >> 2);
let new_zuniq = Z::to_zuniq(parent_depth, parent_h);
entries.push((new_zuniq, combined_value));
Self::from_skymap_ref_recursive(&mut entries, &merger);
}
Expand All @@ -155,20 +160,7 @@ where
fn from_skymap<'s, S, M>(skymap: S, merger: M) -> Self
where
S: SkyMap<'s, HashType = Self::ZUniqHType, ValueType = Self::ValueType>,
M: Fn(
Self::ValueType,
Self::ValueType,
Self::ValueType,
Self::ValueType,
) -> Result<
Self::ValueType,
(
Self::ValueType,
Self::ValueType,
Self::ValueType,
Self::ValueType,
),
>,
M: Fn(u8, Z, [V; 4]) -> Result<V, [V; 4]>,
Self::ValueType: 's,
{
let depth = skymap.depth();
Expand All @@ -184,13 +176,15 @@ where
let (z2, v2) = entries.pop().unwrap();
let (z1, v1) = entries.pop().unwrap();
let (z0, v0) = entries.pop().unwrap();
match merger(v0, v1, v2, v3) {
let parent_depth = depth - 1;
let parent_h = h >> 2;
match merger(parent_depth, parent_h, [v0, v1, v2, v3]) {
Ok(v_merged) => {
let z_merged = Z::to_zuniq(depth - 1, h >> 2);
let z_merged = Z::to_zuniq(parent_depth, parent_h);
entries.push((z_merged, v_merged));
Self::from_skymap_recursive(&mut entries, &merger);
}
Err((v0, v1, v2, v3)) => {
Err([v0, v1, v2, v3]) => {
entries.push((z0, v0));
entries.push((z1, v1));
entries.push((z2, v2));
Expand All @@ -210,29 +204,9 @@ where
where
L: Mom<'s, ZUniqHType = Z, ValueType = V>,
R: Mom<'s, ZUniqHType = Z, ValueType = V>,
S: Fn(
Self::ValueType,
) -> (
Self::ValueType,
Self::ValueType,
Self::ValueType,
Self::ValueType,
),
O: Fn(Option<Self::ValueType>, Option<Self::ValueType>) -> Option<Self::ValueType>,
M: Fn(
Self::ValueType,
Self::ValueType,
Self::ValueType,
Self::ValueType,
) -> Result<
Self::ValueType,
(
Self::ValueType,
Self::ValueType,
Self::ValueType,
Self::ValueType,
),
>,
S: Fn(u8, Z, V) -> [V; 4],
O: Fn(LhsRhsBoth<V>) -> Option<V>,
M: Fn(u8, Z, [V; 4]) -> Result<V, [V; 4]>,
{
#[derive(Clone)]
struct DHZ<ZZ: ZUniqHashT> {
Expand Down Expand Up @@ -282,7 +256,7 @@ where
(Some(l), Some(r)) => match l.d.cmp(&r.d) {
Ordering::Equal => match l.h.cmp(&r.h) {
Ordering::Equal => {
if let Some(v) = op(Some(l.v), Some(r.v)) {
if let Some(v) = op(LhsRhsBoth::Both(l.v, r.v)) {
last_dhz_added = DHZ::new(l.d, l.h, l.z);
stack.push((l.z, v));
}
Expand All @@ -298,7 +272,7 @@ where
};
}
Ordering::Less => {
if let Some(v) = op(Some(l.v), None) {
if let Some(v) = op(LhsRhsBoth::Left(l.v)) {
last_dhz_added = DHZ::new(l.d, l.h, l.z);
stack.push((l.z, v));
}
Expand All @@ -310,7 +284,7 @@ where
right = Some(r);
}
Ordering::Greater => {
if let Some(v) = op(None, Some(r.v)) {
if let Some(v) = op(LhsRhsBoth::Right(r.v)) {
last_dhz_added = DHZ::new(r.d, r.h, r.z);
stack.push((r.z, v));
}
Expand All @@ -328,7 +302,7 @@ where
let r_hash_at_l_depth = r.h >> (((r.d - l.d) << 1) as usize);
match l.h.cmp(&r_hash_at_l_depth) {
Ordering::Equal => {
let (v0, v1, v2, v3) = split(l.v);
let [v0, v1, v2, v3] = split(l.d, l.h, l.v);
let new_d = l.d + 1;
let new_h = l.h << 2;
lstack.push(DHZV::new(
Expand All @@ -353,7 +327,7 @@ where
right = Some(r);
}
Ordering::Less => {
if let Some(v) = op(Some(l.v), None) {
if let Some(v) = op(LhsRhsBoth::Left(l.v)) {
last_dhz_added = DHZ::new(l.d, l.h, l.z);
stack.push((l.z, v));
}
Expand All @@ -365,7 +339,7 @@ where
right = Some(r);
}
Ordering::Greater => {
if let Some(v) = op(None, Some(r.v)) {
if let Some(v) = op(LhsRhsBoth::Right(r.v)) {
last_dhz_added = DHZ::new(r.d, r.h, r.z);
stack.push((r.z, v));
}
Expand All @@ -382,7 +356,7 @@ where
let l_has_at_r_depth = l.h >> (((l.d - r.d) << 1) as usize);
match l_has_at_r_depth.cmp(&r.h) {
Ordering::Equal => {
let (v0, v1, v2, v3) = split(r.v);
let [v0, v1, v2, v3] = split(r.d, r.h, r.v);
let new_d = r.d + 1;
let new_h = r.h << 2;
rstack.push(DHZV::new(
Expand All @@ -407,7 +381,7 @@ where
right = Some(DHZV::new(new_d, new_h, Z::to_zuniq(new_d, new_h), v0));
}
Ordering::Less => {
if let Some(v) = op(Some(l.v), None) {
if let Some(v) = op(LhsRhsBoth::Left(l.v)) {
last_dhz_added = DHZ::new(l.d, l.h, l.z);
stack.push((l.z, v));
}
Expand All @@ -419,7 +393,7 @@ where
right = Some(r);
}
Ordering::Greater => {
if let Some(v) = op(None, Some(r.v)) {
if let Some(v) = op(LhsRhsBoth::Right(r.v)) {
last_dhz_added = DHZ::new(r.d, r.h, r.z);
stack.push((r.z, v));
}
Expand All @@ -436,7 +410,7 @@ where
(None, None) => break, // Important to let this test here!!!
(mut left, None) => {
while let Some(l) = left {
if let Some(v) = op(Some(l.v), None) {
if let Some(v) = op(LhsRhsBoth::Left(l.v)) {
last_dhz_added = DHZ::new(l.d, l.h, l.z);
stack.push((l.z, v));
}
Expand All @@ -459,7 +433,7 @@ where
}
(None, mut right) => {
while let Some(r) = right {
if let Some(v) = op(None, Some(r.v)) {
if let Some(v) = op(LhsRhsBoth::Right(r.v)) {
last_dhz_added = DHZ::new(r.d, r.h, r.z);
stack.push((r.z, v));
}
Expand Down Expand Up @@ -515,7 +489,7 @@ where

fn from_skymap_ref_recursive<'s, M>(stack: &mut Vec<(Z, V)>, merger: &M)
where
M: Fn(&V, &V, &V, &V) -> Option<V>,
M: Fn(u8, Z, [&V; 4]) -> Option<V>,
V: 's,
{
let n = stack.len();
Expand All @@ -530,17 +504,23 @@ where
&& e2.0 == Z::to_zuniq(d0, h0 + Z::two())
&& e3.0 == Z::to_zuniq(d0, h0 + Z::three())
{
let parent_depth = d0 - 1;
let parent_h = h0 >> 2;
if let Some(combined_value) = merger(
&e0.1, // sibling 0
&e1.1, // sibling 1
&e2.1, // sibling 2
&e3.1, // sibling 3
parent_depth,
parent_h,
[
&e0.1, // sibling 0
&e1.1, // sibling 1
&e2.1, // sibling 2
&e3.1, // sibling 3
],
) {
let _ = stack.pop().unwrap();
let _ = stack.pop().unwrap();
let _ = stack.pop().unwrap();
let _ = stack.pop().unwrap();
let new_zuniq = Z::to_zuniq(d0 - 1, h0 >> 2);
let new_zuniq = Z::to_zuniq(parent_depth, parent_h);
stack.push((new_zuniq, combined_value));
Self::from_skymap_ref_recursive(stack, merger);
}
Expand All @@ -551,7 +531,7 @@ where

fn from_skymap_recursive<'s, M>(stack: &mut Vec<(Z, V)>, merger: &M)
where
M: Fn(V, V, V, V) -> Result<V, (V, V, V, V)>,
M: Fn(u8, Z, [V; 4]) -> Result<V, [V; 4]>,
V: 's,
{
let n = stack.len();
Expand All @@ -570,13 +550,15 @@ where
let (_, v2) = stack.pop().unwrap();
let (_, v1) = stack.pop().unwrap();
let (_, v0) = stack.pop().unwrap();
match merger(v0, v1, v2, v3) {
let parent_depth = d0 - 1;
let parent_h = h0 >> 2;
match merger(parent_depth, parent_h, [v0, v1, v2, v3]) {
Ok(v_merged) => {
let z_merged = Z::to_zuniq(d0 - 1, h0 >> 2);
stack.push((z_merged, v_merged));
Self::from_skymap_recursive(stack, merger);
}
Err((v0, v1, v2, v3)) => {
Err([v0, v1, v2, v3]) => {
stack.push((z0, v0));
stack.push((z1, v1));
stack.push((z2, v2));
Expand Down Expand Up @@ -612,7 +594,7 @@ mod tests {
let skymap = SkyMapEnum::from_fits_file(path).unwrap();
match skymap {
SkyMapEnum::ImplicitU64I32(skymap) => {
let merger = |n0: &i32, n1: &i32, n2: &i32, n3: &i32| -> Option<i32> {
let merger = |_depth: u8, _hash: u64, [n0, n1, n2, n3]: [&i32; 4]| -> Option<i32> {
let sum = *n0 + *n1 + *n2 + *n3;
if sum < 1_000_000 {
Some(sum)
Expand Down Expand Up @@ -670,7 +652,7 @@ mod tests {
SkyMapEnum::ImplicitU64I32(skymap) => {
// println!("Skymap size: {}", skymap.len());

let merger = |n0: &i32, n1: &i32, n2: &i32, n3: &i32| -> Option<i32> {
let merger = |_depth: u8, _hash: u64, [n0, n1, n2, n3]: [&i32; 4]| -> Option<i32> {
let mu0 = *n0 as f64;
// let sig0 = mu0.sqrt();
let mu1 = *n1 as f64;
Expand Down
Loading

0 comments on commit a7f9b7a

Please sign in to comment.