Skip to content

Commit

Permalink
E57Simple: Simplify & clarify node types when setting up Data3D (#178)
Browse files Browse the repository at this point in the history
Note that Data3D's "intensity" has been changed to a double so we can allow writing them as doubles.

Addresses some of #157 and #160
  • Loading branch information
asmaloney authored Nov 22, 2022
1 parent f554fc3 commit 0c8cbd5
Show file tree
Hide file tree
Showing 9 changed files with 548 additions and 252 deletions.
2 changes: 2 additions & 0 deletions include/E57Exception.h
Original file line number Diff line number Diff line change
Expand Up @@ -95,6 +95,8 @@ namespace e57
ErrorBadConfiguration = 49, //!< bad configuration string
ErrorInvarianceViolation = 50, //!< class invariance constraint violation in debug mode

ErrorInvalidNodeType = 51, ///< an invalid note type was passed in Data3D pointFields

/// @deprecated Will be removed in 4.0. Use e57::Success.
E57_SUCCESS [[deprecated( "Will be removed in 4.0. Use Success." )]] = Success,
/// @deprecated Will be removed in 4.0. Use e57::ErrorBadCVHeader.
Expand Down
134 changes: 76 additions & 58 deletions include/E57SimpleData.h
Original file line number Diff line number Diff line change
Expand Up @@ -37,12 +37,6 @@

namespace e57
{
//! Indicates to use FloatNode instead of ScaledIntegerNode in fields that can use both.
constexpr double E57_NOT_SCALED_USE_FLOAT = 0.0;

//! Indicates to use ScaledIntegerNode instead of FloatNode in fields that can use both.
constexpr double E57_NOT_SCALED_USE_INTEGER = -1.0;

//! @cond documentNonPublic The following isn't part of the API, and isn't documented.
class ReaderImpl;
class WriterImpl;
Expand Down Expand Up @@ -305,6 +299,15 @@ namespace e57
GroupingByLine groupingByLine; //!< Grouping information by row or column index
};

//! @brief Used to set the type of node in some PointStandardizedFieldsAvailable fields.
enum class NumericalNodeType
{
Integer = 0, ///< Use IntegerNode
ScaledInteger, ///< Use ScaledIntegerNode
Float, ///< Use FloatNode with floats
Double, ///< Use FloatNode with doubles
};

//! @brief Used to interrogate if standardized fields are available
struct E57_DLL PointStandardizedFieldsAvailable
{
Expand All @@ -326,14 +329,15 @@ namespace e57
//! E57_FLOAT_MAX or E57_DOUBLE_MAX. If using a ScaledIntegerNode then this needs to be a maximum range value.
double pointRangeMaximum = DOUBLE_MAX;

//! @brief Controls the type of Node used for the PointRecord cartesian and range fields
//! @details The value determines which type of Node to use and whether to use floats or doubles.
//! Value | Node Type
//! -- | --
//! &lt; 0.0 | FloatNode using doubles
//! == 0.0 (e57::E57_NOT_SCALED_USE_FLOAT) | FloatNode using floats (@em default)
//! &gt; 0.0 | ScaledIntegerNode with the value as the scale setting
double pointRangeScaledInteger = E57_NOT_SCALED_USE_FLOAT;
/// @brief Controls the type of Node used for the PointRecord cartesian and range fields
/// @details Accepts NumericalNodeType::ScaledInteger, NumericalNodeType::Float, and
/// NumericalNodeType::Double.
NumericalNodeType pointRangeNodeType = NumericalNodeType::Float;

/// @brief Sets the scale if using scaled integers for point fields
/// @details If pointRangeNodeType == NumericalNodeType::ScaledInteger, it will use this value
/// to scale the numbers and it must be > 0.0.
double pointRangeScale = 0.0;

//! Indicates that the PointRecord angle fields should be configured with this minimum value E57_FLOAT_MIN or
//! E57_DOUBLE_MIN. If using a ScaledIntegerNode then this needs to be a minimum angle value.
Expand All @@ -343,14 +347,15 @@ namespace e57
//! E57_DOUBLE_MAX. If using a ScaledIntegerNode then this needs to be a maximum angle value.
double angleMaximum = DOUBLE_MAX;

//! @brief Controls the type of Node used for the PointRecord angle fields
//! @details The value determines which type of Node to use and whether to use floats or doubles.
//! Value | Node Type
//! -- | --
//! &lt; 0.0 | FloatNode using doubles
//! == 0.0 (e57::E57_NOT_SCALED_USE_FLOAT) | FloatNode using floats (@em default)
//! &gt; 0.0 | ScaledIntegerNode with the value as the scale setting
double angleScaledInteger = E57_NOT_SCALED_USE_FLOAT;
/// @brief Controls the type of Node used for the PointRecord angle fields
/// @details Accepts NumericalNodeType::ScaledInteger, NumericalNodeType::Float, and
/// NumericalNodeType::Double.
NumericalNodeType angleNodeType = NumericalNodeType::Float;

/// @brief Sets the scale if using scaled integers for angle fields
/// @details If angleNodeType == NumericalNodeType::ScaledInteger, it will use this value
/// to scale the numbers and it must be > 0.0.
double angleScale = 0.0;

bool rowIndexField = false; //!< Indicates that the PointRecord @a rowIndex field is active

Expand Down Expand Up @@ -381,36 +386,46 @@ namespace e57
//! E57_UINT32_MAX, E57_DOUBLE_MAX or E57_DOUBLE_MAX.
double timeMaximum = DOUBLE_MAX;

//! @brief Controls the type of Node used for the PointRecord @a timeStamp fields
//! @details The value determines which type of Node to use and whether to use floats or doubles.
//! Value | Node Type
//! -- | --
//! &lt; 0.0 | IntegerNode
//! == 0.0 (e57::E57_NOT_SCALED_USE_FLOAT) | FloatNode using floats if (::timeMaximum == E57_FLOAT_MAX)
//! == 0.0 | FloatNode using doubles if (::timeMaximum == E57_DOUBLE_MAX) (@em default)
//! &gt; 0.0 | ScaledIntegerNode with the value as the scale setting
double timeScaledInteger = E57_NOT_SCALED_USE_FLOAT;

bool intensityField = false; //!< Indicates that the PointRecord @a intensity field is active
bool isIntensityInvalidField = false; //!< Indicates that the PointRecord @a isIntensityInvalid field is active

//! @brief Controls the type of Node used for the PointRecord @a intensity fields
//! @details The value determines which type of Node to use.
//! Value | Node Type
//! -- | --
//! &lt; 0.0 | IntegerNode
//! == 0.0 (e57::E57_NOT_SCALED_USE_FLOAT) | FloatNode using floats (@em default)
//! &gt; 0.0 | ScaledIntegerNode with the value as the scale setting
double intensityScaledInteger = E57_NOT_SCALED_USE_INTEGER;

bool colorRedField = false; //!< Indicates that the PointRecord @a colorRed field is active
bool colorGreenField = false; //!< Indicates that the PointRecord @a colorGreen field is active
bool colorBlueField = false; //!< Indicates that the PointRecord @a colorBlue field is active
bool isColorInvalidField = false; //!< Indicates that the PointRecord @a isColorInvalid field is active

bool normalXField = false; //!< Indicates that the PointRecord @a nor:normalX field is active
bool normalYField = false; //!< Indicates that the PointRecord @a nor:normalY field is active
bool normalZField = false; //!< Indicates that the PointRecord @a nor:normalZ field is active
/// @brief Controls the type of Node used for the PointRecord time fields
/// @details Accepts NumericalNodeType::Integer, NumericalNodeType::ScaledInteger, NumericalNodeType::Float, and
/// NumericalNodeType::Double.
NumericalNodeType timeNodeType = NumericalNodeType::Float;

/// @brief Sets the scale if using scaled integers for time fields
/// @details If timeNodeType == NumericalNodeType::ScaledInteger, it will use this value
/// to scale the numbers and it must be > 0.0.
double timeScale = 0.0;

/// Indicates that the PointRecord @a intensity field is active
bool intensityField = false;
/// Indicates that the PointRecord @a isIntensityInvalid field is active
bool isIntensityInvalidField = false;

/// @brief Controls the type of Node used for the PointRecord intensity fields
/// @details Accepts NumericalNodeType::Integer, NumericalNodeType::ScaledInteger, NumericalNodeType::Float, and
/// NumericalNodeType::Double.
NumericalNodeType intensityNodeType = NumericalNodeType::Float;

/// @brief Sets the scale if using scaled integers for intensity fields
/// @details If intensityNodeType == NumericalNodeType::ScaledInteger, it will use this value
/// to scale the numbers and it must be > 0.0.
double intensityScale = 0.0;

/// Indicates that the PointRecord @a colorRed field is active
bool colorRedField = false;
/// Indicates that the PointRecord @a colorGreen field is active
bool colorGreenField = false;
/// Indicates that the PointRecord @a colorBlue field is active
bool colorBlueField = false;
/// Indicates that the PointRecord @a isColorInvalid field is active
bool isColorInvalidField = false;

/// Indicates that the PointRecord @a nor:normalX field is active
bool normalXField = false;
/// Indicates that the PointRecord @a nor:normalY field is active
bool normalYField = false;
/// Indicates that the PointRecord @a nor:normalZ field is active
bool normalZField = false;
};

//! @brief Stores the top-level information for a single lidar scan
Expand Down Expand Up @@ -478,12 +493,15 @@ namespace e57
{
static_assert( std::is_floating_point<COORDTYPE>::value, "Floating point type required." );

//! @brief Default constructor does not manage any memory or adjust min/max for floats.
//! @brief Default constructor does not manage any memory, adjust min/max for floats, or validate data.
Data3DPointsData_t() = default;

//! @brief Constructor which allocates buffers for all valid fields in the given Data3D header
//! This constructor will also adjust the min/max fields in the data3D pointFields if we are using floats.
//! @param [in] data3D Completed header which indicates the fields we are using
/// @brief Constructor which allocates buffers for all valid fields in the given Data3D header.
/// @details This constructor will also adjust the min/max fields in the data3D pointFields if we are using
/// floats, and run some validation on the Data3D.
/// @param [in] data3D Completed header which indicates the fields we are using
/// @throw ::ErrorValueOutOfBounds
/// @throw ::ErrorInvalidNodeType
explicit Data3DPointsData_t( e57::Data3D &data3D );

//! @brief Destructor will delete any memory allocated using the Data3DPointsData_t( const e57::Data3D & )
Expand All @@ -502,8 +520,8 @@ namespace e57
//! Value = 0 if the point is considered valid, 1 otherwise
int8_t *cartesianInvalidState = nullptr;

//! Pointer to a buffer with the Point response intensity. Unit is unspecified.
float *intensity = nullptr;
/// @brief Pointer to a buffer with the Point response intensity. Unit is unspecified.
double *intensity = nullptr;

//! Value = 0 if the intensity is considered valid, 1 otherwise
int8_t *isIntensityInvalid = nullptr;
Expand Down
3 changes: 3 additions & 0 deletions src/E57Exception.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -447,6 +447,9 @@ namespace e57
case ErrorInvarianceViolation:
return "class invariance constraint violation in debug mode "
"(ErrorInvarianceViolation)";
case ErrorInvalidNodeType:
return "an invalid node type was passed in Data3D pointFields";

default:
return "unknown error (" + std::to_string( ecode ) + ")";
}
Expand Down
46 changes: 39 additions & 7 deletions src/E57SimpleData.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,25 @@

namespace e57
{
/// Validates a Data3D and throws on error.
void _validateData3D( const Data3D &inData3D )
{
if ( inData3D.pointCount < 1 )
{
throw E57_EXCEPTION2( ErrorValueOutOfBounds, "pointCount=" + toString( inData3D.pointCount ) + " minimum=1" );
}

if ( inData3D.pointFields.pointRangeNodeType == NumericalNodeType::Integer )
{
throw E57_EXCEPTION2( ErrorInvalidNodeType, "pointRangeNodeType cannot be Integer" );
}

if ( inData3D.pointFields.angleNodeType == NumericalNodeType::Integer )
{
throw E57_EXCEPTION2( ErrorInvalidNodeType, "angleNodeType cannot be Integer" );
}
}

// To avoid exposing M_PI, we define the constructor here.
SphericalBounds::SphericalBounds()
{
Expand All @@ -31,15 +50,12 @@ namespace e57
template <typename COORDTYPE>
Data3DPointsData_t<COORDTYPE>::Data3DPointsData_t( Data3D &data3D ) : _selfAllocated( true )
{
const auto cPointCount = data3D.pointCount;
_validateData3D( data3D );

if ( cPointCount < 1 )
{
throw E57_EXCEPTION2( ErrorValueOutOfBounds, "pointCount=" + toString( cPointCount ) + " minimum=1" );
}
constexpr bool cIsFloat = std::is_same<COORDTYPE, float>::value;

// We need to adjust min/max for floats.
if ( std::is_same<COORDTYPE, float>::value )
if ( cIsFloat )
{
data3D.pointFields.pointRangeMinimum = FLOAT_MIN;
data3D.pointFields.pointRangeMaximum = FLOAT_MAX;
Expand All @@ -49,6 +65,22 @@ namespace e57
data3D.pointFields.timeMaximum = FLOAT_MAX;
}

// IF point range node type is not ScaledInteger
// THEN make sure the proper floating point type is set
if ( data3D.pointFields.pointRangeNodeType != NumericalNodeType::ScaledInteger )
{
data3D.pointFields.pointRangeNodeType = ( cIsFloat ? NumericalNodeType::Float : NumericalNodeType::Double );
}

// IF angle node type is not ScaledInteger
// THEN make sure the proper floating point type is set
if ( data3D.pointFields.angleNodeType != NumericalNodeType::ScaledInteger )
{
data3D.pointFields.angleNodeType = ( cIsFloat ? NumericalNodeType::Float : NumericalNodeType::Double );
}

const auto cPointCount = data3D.pointCount;

if ( data3D.pointFields.cartesianXField )
{
cartesianX = new COORDTYPE[cPointCount];
Expand All @@ -71,7 +103,7 @@ namespace e57

if ( data3D.pointFields.intensityField )
{
intensity = new float[cPointCount];
intensity = new double[cPointCount];
}

if ( data3D.pointFields.isIntensityInvalidField )
Expand Down
13 changes: 7 additions & 6 deletions src/E57SimpleWriter.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -54,7 +54,7 @@ namespace
auto pointRangeMinimum = cMax;
auto pointRangeMaximum = cMin;

const bool writePointRange = ( pointFields.pointRangeScaledInteger != 0.0 ) &&
const bool writePointRange = ( pointFields.pointRangeNodeType == e57::NumericalNodeType::ScaledInteger ) &&
( pointFields.pointRangeMinimum == cMin ) &&
( pointFields.pointRangeMaximum == cMax );

Expand All @@ -64,14 +64,14 @@ namespace
auto angleMinimum = cMax;
auto angleMaximum = cMin;

const bool writeAngle = ( pointFields.angleScaledInteger != 0.0 ) && ( pointFields.angleMinimum == cMin ) &&
( pointFields.angleMaximum == cMax );
const bool writeAngle = ( pointFields.angleNodeType == e57::NumericalNodeType::ScaledInteger ) &&
( pointFields.angleMinimum == cMin ) && ( pointFields.angleMaximum == cMax );

// IF we are using intesity
// AND we haven't set either min or max
// THEN calculate them from the points
float intensityMinimum = std::numeric_limits<float>::max();
float intensityMaximum = std::numeric_limits<float>::lowest();
double intensityMinimum = std::numeric_limits<double>::max();
double intensityMaximum = std::numeric_limits<double>::lowest();

const bool writeIntensity =
pointFields.intensityField && ( ioData3DHeader.intensityLimits == e57::IntensityLimits{} );
Expand All @@ -82,7 +82,8 @@ namespace
double timeMinimum = std::numeric_limits<double>::max();
double timeMaximum = std::numeric_limits<double>::lowest();

const bool writeTimeStamp = pointFields.timeStampField && ( pointFields.timeScaledInteger != 0.0 ) &&
const bool writeTimeStamp = pointFields.timeStampField &&
( pointFields.timeNodeType == e57::NumericalNodeType::ScaledInteger ) &&
( pointFields.timeMinimum == cMin ) && ( pointFields.timeMaximum == cMax );

// Now run through the points and set the things we need to
Expand Down
Loading

0 comments on commit 0c8cbd5

Please sign in to comment.