Skip to content

Commit

Permalink
Implement possibility to have empty discrete domains
Browse files Browse the repository at this point in the history
  • Loading branch information
tpadioleau committed Jul 7, 2022
1 parent fcac1f7 commit 5dd84a2
Show file tree
Hide file tree
Showing 3 changed files with 36 additions and 26 deletions.
50 changes: 26 additions & 24 deletions include/ddc/discrete_domain.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -29,9 +29,9 @@ class DiscreteDomain
using mlength_type = DiscreteVector<DDims...>;

private:
DiscreteElement<DDims...> m_lbound;
DiscreteElement<DDims...> m_element_begin;

DiscreteElement<DDims...> m_ubound;
DiscreteElement<DDims...> m_element_end;

public:
static constexpr std::size_t rank()
Expand All @@ -44,17 +44,17 @@ class DiscreteDomain
/// Construct a DiscreteDomain from a reordered copy of `domain`
template <class... ODDims>
explicit constexpr DiscreteDomain(DiscreteDomain<ODDims...> const& domain)
: m_lbound(domain.front())
, m_ubound(domain.back())
: m_element_begin(domain.front())
, m_element_end(domain.front() + domain.extents())
{
}

// Use SFINAE to disambiguate with the copy constructor.
// Note that SFINAE may be redundant because a template constructor should not be selected as a copy constructor.
template <std::size_t N = sizeof...(DDims), class = std::enable_if_t<(N != 1)>>
explicit constexpr DiscreteDomain(DiscreteDomain<DDims> const&... domains)
: m_lbound(domains.front()...)
, m_ubound(domains.back()...)
: m_element_begin(domains.front()...)
, m_element_end((domains.front() + domains.extents())...)
{
}

Expand All @@ -64,18 +64,19 @@ class DiscreteDomain
* @deprecated use the version with explicit lower bound instead
*/
[[deprecated]] constexpr DiscreteDomain(mlength_type const& size)
: m_lbound((get<DDims>(size) - get<DDims>(size))...) // Hack to have expansion of zero
, m_ubound((get<DDims>(size) - 1)...)
: m_element_begin(
(get<DDims>(size) - get<DDims>(size))...) // Hack to have expansion of zero
, m_element_end(get<DDims>(size)...)
{
}

/** Construct a DiscreteDomain starting from lbound with size points.
* @param lbound the lower bound in each direction
/** Construct a DiscreteDomain starting from element_begin with size points.
* @param element_begin the lower bound in each direction
* @param size the number of points in each direction
*/
constexpr DiscreteDomain(discrete_element_type const& lbound, mlength_type const& size)
: m_lbound(lbound)
, m_ubound((uid<DDims>(lbound) + get<DDims>(size) - 1)...)
constexpr DiscreteDomain(discrete_element_type const& element_begin, mlength_type const& size)
: m_element_begin(element_begin)
, m_element_end(element_begin + size)
{
}

Expand All @@ -92,7 +93,7 @@ class DiscreteDomain
template <class... ODims>
constexpr bool operator==(DiscreteDomain<ODims...> const& other) const
{
return m_lbound == other.m_lbound && m_ubound == other.m_ubound;
return m_element_begin == other.m_element_begin && m_element_end == other.m_element_end;
}

#if __cplusplus <= 201703L
Expand All @@ -106,28 +107,29 @@ class DiscreteDomain

std::size_t size() const
{
return (1ul * ... * (uid<DDims>(m_ubound) + 1 - uid<DDims>(m_lbound)));
return (1ul * ... * (uid<DDims>(m_element_end) - uid<DDims>(m_element_begin)));
}

constexpr mlength_type extents() const noexcept
{
return mlength_type((uid<DDims>(m_ubound) + 1 - uid<DDims>(m_lbound))...);
return mlength_type((uid<DDims>(m_element_end) - uid<DDims>(m_element_begin))...);
}

template <class QueryDDim>
inline constexpr DiscreteVector<QueryDDim> extent() const noexcept
{
return DiscreteVector<QueryDDim>(uid<QueryDDim>(m_ubound) + 1 - uid<QueryDDim>(m_lbound));
return DiscreteVector<QueryDDim>(
uid<QueryDDim>(m_element_end) - uid<QueryDDim>(m_element_begin));
}

constexpr discrete_element_type front() const noexcept
{
return m_lbound;
return m_element_begin;
}

constexpr discrete_element_type back() const noexcept
{
return m_ubound;
return discrete_element_type((uid<DDims>(m_element_end) - 1)...);
}

constexpr DiscreteDomain take_first(mlength_type n) const
Expand Down Expand Up @@ -158,13 +160,13 @@ class DiscreteDomain
template <class... ODDims>
constexpr auto restrict(DiscreteDomain<ODDims...> const& odomain) const
{
assert(((uid<ODDims>(m_lbound) <= uid<ODDims>(odomain.m_lbound)) && ...));
assert(((uid<ODDims>(m_ubound) >= uid<ODDims>(odomain.m_ubound)) && ...));
assert(((uid<ODDims>(m_element_begin) <= uid<ODDims>(odomain.m_element_begin)) && ...));
assert(((uid<ODDims>(m_element_end) >= uid<ODDims>(odomain.m_element_end)) && ...));
const DiscreteVector<DDims...> myextents = extents();
const DiscreteVector<ODDims...> oextents = odomain.extents();
return DiscreteDomain(
DiscreteElement<DDims...>(
(uid_or<DDims>(odomain.m_lbound, uid<DDims>(m_lbound)))...),
(uid_or<DDims>(odomain.m_element_begin, uid<DDims>(m_element_begin)))...),
DiscreteVector<DDims...>((get_or<DDims>(oextents, get<DDims>(myextents)))...));
}

Expand All @@ -191,7 +193,7 @@ class DiscreteDomain
class DDim0 = std::enable_if_t<N == 1, std::tuple_element_t<0, std::tuple<DDims...>>>>
auto end() const
{
return DiscreteDomainIterator<DDim0>(DiscreteElement<DDims...>(back() + 1));
return DiscreteDomainIterator<DDim0>(m_element_end);
}

template <
Expand All @@ -207,7 +209,7 @@ class DiscreteDomain
class DDim0 = std::enable_if_t<N == 1, std::tuple_element_t<0, std::tuple<DDims...>>>>
auto cend() const
{
return DiscreteDomainIterator<DDim0>(DiscreteElement<DDims...>(back() + 1));
return DiscreteDomainIterator<DDim0>(m_element_end);
}

template <
Expand Down
4 changes: 2 additions & 2 deletions tests/chunk.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -456,7 +456,7 @@ TEST(Chunk2DTest, SliceDomainXTooearly)
// the error message is checked with clang & gcc only
ASSERT_DEATH(
chunk[subdomain_x],
R"rgx([Aa]ssert.*uid<ODDims>\(m_lbound\).*uid<ODDims>\(odomain\.m_lbound\))rgx");
R"rgx([Aa]ssert.*uid<ODDims>\(m_elem_begin\).*uid<ODDims>\(odomain\.m_elem_begin\))rgx");
#endif
}

Expand All @@ -469,7 +469,7 @@ TEST(Chunk2DTest, SliceDomainXToolate)
// the error message is checked with clang & gcc only
ASSERT_DEATH(
chunk[subdomain_x],
R"rgx([Aa]ssert.*uid<ODDims>\(m_ubound\).*uid<ODDims>\(odomain\.m_ubound\).*)rgx");
R"rgx([Aa]ssert.*uid<ODDims>\(m_elem_end\).*uid<ODDims>\(odomain\.m_elem_end\).*)rgx");
#endif
}

Expand Down
8 changes: 8 additions & 0 deletions tests/discrete_domain.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -75,6 +75,14 @@ TEST(ProductMDomainTest, Constructor)
EXPECT_EQ(empty_domain[0], lbound_x);
}

TEST(ProductMDomainTest, EmptyDomain)
{
DDomXY const dom_x_y = DDomXY();
EXPECT_EQ(dom_x_y.extents(), DVectXY(0, 0));
EXPECT_EQ(dom_x_y.size(), 0);
EXPECT_TRUE(dom_x_y.empty());
}

TEST(ProductMDomainTest, Subdomain)
{
DDomXY const dom_x_y = DDomXY(lbound_x_y, nelems_x_y);
Expand Down

0 comments on commit 5dd84a2

Please sign in to comment.