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

Add tile helper functions #145

Open
wants to merge 15 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 2 additions & 2 deletions CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -88,7 +88,7 @@ endif()
set(LAZPERF_VERSION "3.0.0")
# First, see if the repo is cloned locally, and if so, compile it
# this is required for CI build of sdist
if(EXISTS "${CMAKE_SOURCE_DIR}/libs/laz-perf")
if(EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/libs/laz-perf")
set(WITH_TEST_TEMP ${WITH_TESTS})
set(WITH_TESTS OFF) # never build lazperf tests
add_subdirectory(libs/laz-perf)
Expand All @@ -106,7 +106,7 @@ else()
set(EXTRA_EXPORT_TARGETS "")
else ()
# assume lazperf is compiled in tandem
message("Lazperf package not found, including ${LAZPERF_DIR}/cpp")
message("Lazperf package not found, assuming compiled in tandem...")
set(LAZPERF_LIB_NAME "lazperf")
set(EXTRA_EXPORT_TARGETS "lazperf")
endif ()
Expand Down
2 changes: 2 additions & 0 deletions cpp/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ set(${LIBRARY_TARGET_NAME}_HDR
include/${LIBRARY_TARGET_NAME}/copc/copc_config.hpp
include/${LIBRARY_TARGET_NAME}/geometry/box.hpp
include/${LIBRARY_TARGET_NAME}/geometry/vector3.hpp
include/${LIBRARY_TARGET_NAME}/geometry/helpers.hpp
include/${LIBRARY_TARGET_NAME}/hierarchy/entry.hpp
include/${LIBRARY_TARGET_NAME}/hierarchy/key.hpp
include/${LIBRARY_TARGET_NAME}/hierarchy/node.hpp
Expand Down Expand Up @@ -37,6 +38,7 @@ set(${LIBRARY_TARGET_NAME}_SRC
src/copc/extents.cpp
src/copc/copc_config.cpp
src/geometry/box.cpp
src/geometry/helpers.cpp
src/hierarchy/key.cpp
src/hierarchy/page.cpp
src/io/base_reader.cpp
Expand Down
33 changes: 33 additions & 0 deletions cpp/include/copc-lib/geometry/helpers.hpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
#ifndef CPP_INCLUDE_COPC_LIB_GEOMETRY_HELPERS
#define CPP_INCLUDE_COPC_LIB_GEOMETRY_HELPERS

#include <cmath>
#include <cstdint>
#include <vector>

namespace copc
{

namespace las
{
class LasHeader;
}
class Box;

enum RoundStrategy
{
NEAREST,
PREFER_LARGER_TILE,
PREFER_SMALLER_TILE
};

// Returns the nearest depth to the requested tile size
const int GetNearestDepth(double tile_size, const las::LasHeader &header,
RoundStrategy rounding = RoundStrategy::NEAREST);

const std::vector<copc::Box> GetPossibleTilesAtDepth(int32_t target_depth, const las::LasHeader &header);
const std::vector<copc::Box> GetPossibleTilesWithSize(double target_tile_size, const las::LasHeader &header);

} // namespace copc

#endif /* CPP_INCLUDE_COPC_LIB_GEOMETRY_HELPERS */
2 changes: 2 additions & 0 deletions cpp/include/copc-lib/hierarchy/key.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -76,6 +76,8 @@ class VoxelKey

double Resolution(const las::LasHeader &header, const CopcInfo &copc_info) const;
static double GetResolutionAtDepth(int32_t d, const las::LasHeader &header, const CopcInfo &copc_info);
double Span(const las::LasHeader &header) const;
static double GetSpanAtDepth(int32_t d, const las::LasHeader &header);

int32_t d;
int32_t x;
Expand Down
21 changes: 18 additions & 3 deletions cpp/include/copc-lib/las/point.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -22,13 +22,28 @@ class Point
#pragma region XYZ
// XYZ Scaled
double X() const { return x_scaled_; }
void X(const double &x) { x_scaled_ = x; }
void X(const double &x)
{
if (!std::isfinite(x))
throw std::runtime_error("Tried to set non-finite X value!");
x_scaled_ = x;
}

double Y() const { return y_scaled_; }
void Y(const double &y) { y_scaled_ = y; }
void Y(const double &y)
{
if (!std::isfinite(y))
throw std::runtime_error("Tried to set non-finite Y value!");
y_scaled_ = y;
}

double Z() const { return z_scaled_; }
void Z(const double &z) { z_scaled_ = z; }
void Z(const double &z)
{
if (!std::isfinite(z))
throw std::runtime_error("Tried to set non-finite Z value!");
z_scaled_ = z;
}

#pragma endregion XYZ

Expand Down
6 changes: 6 additions & 0 deletions cpp/include/copc-lib/las/utils.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -17,8 +17,14 @@ bool FormatHasNir(const uint8_t &point_format_id);
template <typename T> double ApplyScale(T value, double scale, double offset) { return (value * scale) + offset; }
template <typename T> T RemoveScale(double value, double scale, double offset)
{
if (!std::isfinite(value))
throw std::runtime_error("The input value " + std::to_string(value) + " is not finite.");

double val = std::round((value - offset) / scale);

if (!std::isfinite(val))
throw std::runtime_error("The output value " + std::to_string(val) + " is not finite.");

if (val < std::numeric_limits<T>::min() || val > std::numeric_limits<T>::max())
throw std::runtime_error("The value " + std::to_string(value) +
" is too large to save into the requested format." +
Expand Down
98 changes: 98 additions & 0 deletions cpp/src/geometry/helpers.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,98 @@
#include "copc-lib/geometry/helpers.hpp"
#include "copc-lib/hierarchy/key.hpp"
#include "copc-lib/las/header.hpp"

#include <limits>

namespace copc
{

const int GetNearestDepth(double tile_size, const las::LasHeader &header, RoundStrategy rounding)
{
double depth = std::log2(header.Span() / tile_size);
int nearest_depth;
if (rounding == RoundStrategy::NEAREST)
nearest_depth = std::lround(depth);
else if (rounding == RoundStrategy::PREFER_LARGER_TILE)
nearest_depth = (int)std::floor(depth);
else if (rounding == RoundStrategy::PREFER_SMALLER_TILE)
nearest_depth = (int)std::ceil(depth);
return (int)std::max(0, nearest_depth);
}

// Generates a vector of Boxes that tile the dataset in its entirety
// Note that points in the same node may fall within different tiles,
// due to rounding
// Additionally, some boxes may contain no points at all, since
// we cannot know for certain whether a tile has points without actually loading it
const std::vector<copc::Box> GetPossibleTilesAtDepth(int32_t target_depth, const las::LasHeader &header)
{
std::vector<copc::Box> possible_tiles;

// We can compute the maximum VoxelKey coordinate based on the depth
int max_possible_coord = std::pow(2, target_depth);
// But then, we can limit the search space in each dimension,
// since not every dimension will use the full span
// (since the span is based on the axis with the largest range)
double tile_size = header.Span() / max_possible_coord;
int max_x_coord = (int)std::ceil((header.max.x - header.min.x) / tile_size);
int max_y_coord = (int)std::ceil((header.max.y - header.min.y) / tile_size);
int max_z_coord = (int)std::ceil((header.max.z - header.min.z) / tile_size);

for (int x = 0; x < max_x_coord; x++)
for (int y = 0; y < max_y_coord; y++)
for (int z = 0; z < max_z_coord; z++)
{
auto box = copc::Box(copc::VoxelKey(target_depth, x, y, z), header);
// To avoid the same point falling into two boxes when the point's coordinates
// are the exact same as the box's limit,
// we offset the minimum coordinate in each dimension by the smallest possible
// floating point value so that two neighboring boxes don't have the exact same max and min
// but, don't do that to the first coordinate in each dimension since there's no other box
// for that point to fall in
if (x != 0)
box.x_min = std::nextafter(box.x_min, std::numeric_limits<double>::max());
if (y != 0)
box.y_min = std::nextafter(box.y_min, std::numeric_limits<double>::max());
if (z != 0)
box.z_min = std::nextafter(box.z_min, std::numeric_limits<double>::max());
possible_tiles.push_back(box);
}
return possible_tiles;
}

const std::vector<copc::Box> GetPossibleTilesWithSize(double target_tile_size, const las::LasHeader &header)
leo-stan marked this conversation as resolved.
Show resolved Hide resolved
{
std::vector<copc::Box> possible_tiles;

int max_num_tiles = header.Span() / target_tile_size;

for (int x = 0; x < max_num_tiles; x++)
for (int y = 0; y < max_num_tiles; y++)
for (int z = 0; z < max_num_tiles; z++)
{
double x_min = target_tile_size * x + header.min.x;
double y_min = target_tile_size * y + header.min.y;
double z_min = target_tile_size * z + header.min.z;
double x_max = x_min + target_tile_size;
double y_max = y_min + target_tile_size;
double z_max = z_min + target_tile_size;
auto box = copc::Box(x_min, y_min, z_min, x_max, y_max, z_max);
// To avoid the same point falling into two boxes when the point's coordinates
// are the exact same as the box's limit,
// we offset the minimum coordinate in each dimension by the smallest possible
// floating point value so that two neighboring boxes don't have the exact same max and min
// but, don't do that to the first coordinate in each dimension since there's no other box
// for that point to fall in
if (x != 0)
box.x_min = std::nextafter(box.x_min, std::numeric_limits<double>::max());
if (y != 0)
box.y_min = std::nextafter(box.y_min, std::numeric_limits<double>::max());
if (z != 0)
box.z_min = std::nextafter(box.z_min, std::numeric_limits<double>::max());
possible_tiles.push_back(box);
}
return possible_tiles;
}

} // namespace copc
4 changes: 4 additions & 0 deletions cpp/src/hierarchy/key.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -98,6 +98,10 @@ double VoxelKey::GetResolutionAtDepth(int32_t d, const las::LasHeader &header, c
return VoxelKey(d, 0, 0, 0).Resolution(header, copc_info);
}

double VoxelKey::Span(const las::LasHeader &header) const { return header.Span() * std::pow(2, -d); }

double VoxelKey::GetSpanAtDepth(int32_t d, const las::LasHeader &header) { return VoxelKey(d, 0, 0, 0).Span(header); }

leo-stan marked this conversation as resolved.
Show resolved Hide resolved
bool VoxelKey::Intersects(const las::LasHeader &header, const Box &box) const
{
return Box(*this, header).Intersects(box);
Expand Down