From bcccb7ff7bc970ec2cf55afc5fa80b51a3f5b2b2 Mon Sep 17 00:00:00 2001 From: Ihar Hubchyk Date: Sun, 17 Mar 2024 19:09:07 +0800 Subject: [PATCH 1/2] Modernize the code (part 2) This is the second batch of changes to update the code in preparation for multi-type image code changes. --- src/blob_detection.cpp | 260 +++++++------------------------ src/blob_detection.h | 289 ++++++++++++++++++++++++++--------- src/edge_detection.cpp | 92 +++++++---- src/function_pool.cpp | 39 ++--- src/math/haar_transform.cpp | 9 +- src/math/hough_transform.cpp | 16 +- src/math/math_base.h | 2 +- src/memory/cpu_memory.h | 27 ++-- 8 files changed, 379 insertions(+), 355 deletions(-) diff --git a/src/blob_detection.cpp b/src/blob_detection.cpp index f93f9558..a9d6e3a2 100644 --- a/src/blob_detection.cpp +++ b/src/blob_detection.cpp @@ -1,6 +1,6 @@ /*************************************************************************** * penguinV: https://github.com/ihhub/penguinV * - * Copyright (C) 2017 - 2022 * + * Copyright (C) 2017 - 2024 * * * * This program is free software; you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * @@ -27,7 +27,7 @@ namespace { - enum PixelState + enum PixelState : uint8_t { EMPTY = 0u, NOT_IN_USE = 1u, @@ -39,169 +39,45 @@ namespace double getLengthFromCountour( const std::vector & contourX, const std::vector & contourY, PointBase2D & startPoint, PointBase2D & endPoint ) { - if ( contourX.size() > 1 ) { - std::vector::const_iterator x = contourX.cbegin(); - std::vector::const_iterator y = contourY.cbegin(); - std::vector::const_iterator end = contourX.cend(); + if ( contourX.size() <= 1 ) { + return 0; + } - int32_t maximumDistance = 0; + auto x = contourX.cbegin(); + auto y = contourY.cbegin(); + auto end = contourX.cend(); - for ( ; x != ( end - 1 ); ++x, ++y ) { - std::vector::const_iterator xx = x + 1; - std::vector::const_iterator yy = y + 1; + int32_t maximumDistance = 0; - for ( ; xx != end; ++xx, ++yy ) { - const int32_t distance - = static_cast( *x - *xx ) * static_cast( *x - *xx ) + static_cast( *y - *yy ) * static_cast( *y - *yy ); + for ( ; x != ( end - 1 ); ++x, ++y ) { + auto xx = x + 1; + auto yy = y + 1; - if ( maximumDistance < distance ) { - maximumDistance = distance; + for ( ; xx != end; ++xx, ++yy ) { + const int32_t distance + = static_cast( *x - *xx ) * static_cast( *x - *xx ) + static_cast( *y - *yy ) * static_cast( *y - *yy ); - startPoint.x = *x; - startPoint.y = *y; + if ( maximumDistance < distance ) { + maximumDistance = distance; - endPoint.x = *xx; - endPoint.y = *xx; - } + startPoint.x = *x; + startPoint.y = *y; + + endPoint.x = *xx; + endPoint.y = *xx; } } - return sqrt( static_cast( maximumDistance ) ); - } - else { - return 0; } + + return sqrt( static_cast( maximumDistance ) ); } } namespace Blob_Detection { - const std::vector & BlobInfo::pointX() const - { - return _pointX; - } - - const std::vector & BlobInfo::pointY() const - { - return _pointY; - } - - const std::vector & BlobInfo::contourX() const - { - return _contourX; - } - - const std::vector & BlobInfo::contourY() const - { - return _contourY; - } - - const std::vector & BlobInfo::edgeX() const - { - return _edgeX; - } - - const std::vector & BlobInfo::edgeY() const - { - return _edgeY; - } - - Area BlobInfo::area() - { - _getArea(); - - return _area.value; - } - - Area BlobInfo::area() const - { - return _area.value; - } - - Point2d BlobInfo::center() - { - _getCenter(); - - return _center.value; - } - - Point2d BlobInfo::center() const - { - return _center.value; - } - - double BlobInfo::circularity() - { - _getCircularity(); - - return _circularity.value; - } - - double BlobInfo::circularity() const - { - return _circularity.value; - } - - double BlobInfo::elongation() - { - _getElongation(); - - return _elongation.value; - } - - double BlobInfo::elongation() const - { - return _elongation.value; - } - - uint32_t BlobInfo::height() - { - _getHeight(); - - return _height.value; - } - - uint32_t BlobInfo::height() const - { - return _height.value; - } - - double BlobInfo::length() - { - _getLength(); - - return _length.value; - } - - double BlobInfo::length() const - { - return _length.value; - } - - size_t BlobInfo::size() const - { - return _pointX.size(); - } - - uint32_t BlobInfo::width() - { - _getWidth(); - - return _width.value; - } - - uint32_t BlobInfo::width() const - { - return _width.value; - } - - bool BlobInfo::isSolid() const - { - return _contourX.size() == _edgeX.size(); - } - void BlobInfo::_getArea() { - if ( !_contourX.empty() && !_contourY.empty() && !_area.found ) { + if ( !_area.found && !_contourX.empty() && !_contourY.empty() ) { _area.value.left = *( std::min_element( _contourX.begin(), _contourX.end() ) ); _area.value.right = *( std::max_element( _contourX.begin(), _contourX.end() ) ) + 1; // note that we add 1 _area.value.top = *( std::min_element( _contourY.begin(), _contourY.end() ) ); @@ -213,7 +89,7 @@ namespace Blob_Detection void BlobInfo::_getCenter() { - if ( !_pointX.empty() && !_pointY.empty() && !_center.found ) { + if ( !_center.found && !_pointX.empty() && !_pointY.empty() ) { _center.value.x = static_cast( std::accumulate( _pointX.begin(), _pointX.end(), 0 ) ) / static_cast( size() ); _center.value.y = static_cast( std::accumulate( _pointY.begin(), _pointY.end(), 0 ) ) / static_cast( size() ); @@ -223,15 +99,15 @@ namespace Blob_Detection void BlobInfo::_getCircularity() { - if ( !_contourX.empty() && !_circularity.found ) { + if ( !_circularity.found && !_contourX.empty() ) { const double radius = sqrt( static_cast( size() ) / pvmath::pi ); _getCenter(); double difference = 0; - std::vector::const_iterator x = _contourX.begin(); - std::vector::const_iterator y = _contourY.begin(); - std::vector::const_iterator end = _contourX.end(); + auto x = _contourX.begin(); + auto y = _contourY.begin(); + auto end = _contourX.end(); for ( ; x != end; ++x, ++y ) { difference += fabs( sqrt( ( *x - _center.value.x ) * ( *x - _center.value.x ) + ( *y - _center.value.y ) * ( *y - _center.value.y ) ) - radius ); @@ -245,7 +121,7 @@ namespace Blob_Detection void BlobInfo::_getElongation() { - if ( !_contourX.empty() && !_contourY.empty() && !_elongation.found ) { + if ( !_elongation.found && !_contourX.empty() && !_contourY.empty() ) { if ( _contourX.size() > 1 ) { PointBase2D startPoint; PointBase2D endPoint; @@ -257,17 +133,19 @@ namespace Blob_Detection std::vector contourYTemp( _contourY.begin(), _contourY.end() ); - std::vector::const_iterator xRotated = _contourX.begin(); - std::vector::iterator yRotated = contourYTemp.begin(); - std::vector::const_iterator endRotated = _contourX.end(); + auto xRotated = _contourX.begin(); + auto yRotated = contourYTemp.begin(); + auto endRotated = _contourX.end(); - for ( ; xRotated != endRotated; ++xRotated, ++yRotated ) + for ( ; xRotated != endRotated; ++xRotated, ++yRotated ) { ( *yRotated ) = ( *xRotated - startPoint.x ) * _sin + ( *yRotated - startPoint.y ) * _cos; + } double height = *( std::max_element( contourYTemp.begin(), contourYTemp.end() ) ) - *( std::min_element( contourYTemp.begin(), contourYTemp.end() ) ); - if ( height < 1 ) + if ( height < 1 ) { height = 1; + } _elongation.value = length / height; } @@ -281,7 +159,7 @@ namespace Blob_Detection void BlobInfo::_getHeight() { - if ( !_contourY.empty() && !_height.found ) { + if ( !_height.found && !_contourY.empty() ) { _height.value = *( std::max_element( _contourY.begin(), _contourY.end() ) ) - *( std::min_element( _contourY.begin(), _contourY.end() ) ) + 1; _height.found = true; @@ -290,7 +168,7 @@ namespace Blob_Detection void BlobInfo::_getLength() { - if ( !_contourX.empty() && !_contourY.empty() && !_length.found ) { + if ( !_length.found && !_contourX.empty() && !_contourY.empty() ) { PointBase2D startPoint; PointBase2D endPoint; _length.value = getLengthFromCountour( _contourX, _contourY, startPoint, endPoint ); @@ -301,18 +179,13 @@ namespace Blob_Detection void BlobInfo::_getWidth() { - if ( !_contourX.empty() && !_width.found ) { + if ( !_width.found && !_contourX.empty() ) { _width.value = *( std::max_element( _contourX.begin(), _contourX.end() ) ) - *( std::min_element( _contourX.begin(), _contourX.end() ) ) + 1; _width.found = true; } } - const std::vector & BlobDetection::find( const penguinV::Image & image, const BlobParameters & parameter, uint8_t threshold ) - { - return find( image, 0, 0, image.width(), image.height(), parameter, threshold ); - } - const std::vector & BlobDetection::find( const penguinV::Image & image, uint32_t x, uint32_t y, uint32_t width, uint32_t height, const BlobParameters & parameter, uint8_t threshold ) { @@ -343,8 +216,9 @@ namespace Blob_Detection uint8_t * mapValueY = mapValueX; for ( ; imageX != imageXEnd; ++imageX, ++mapValueY ) { - if ( ( *imageX ) >= threshold ) + if ( ( *imageX ) >= threshold ) { *mapValueY = NOT_IN_USE; + } } } @@ -609,71 +483,51 @@ namespace Blob_Detection return get(); } - const std::vector & BlobDetection::get() const - { - return _blob; - } - - std::vector & BlobDetection::get() - { - return _blob; - } - - const std::vector & BlobDetection::operator()() const - { - return _blob; - } - - std::vector & BlobDetection::operator()() - { - return _blob; - } - const BlobInfo & BlobDetection::getBestBlob( BlobCriterion criterion ) const { switch ( criterion ) { - case BY_CIRCULARITY: + case BlobCriterion::BY_CIRCULARITY: return *( std::max_element( _blob.begin(), _blob.end(), []( const BlobInfo & blob1, const BlobInfo & blob2 ) { return blob1.circularity() < blob2.circularity(); } ) ); - case BY_ELONGATION: + case BlobCriterion::BY_ELONGATION: return *( std::max_element( _blob.begin(), _blob.end(), []( const BlobInfo & blob1, const BlobInfo & blob2 ) { return blob1.elongation() < blob2.elongation(); } ) ); - case BY_HEIGHT: + case BlobCriterion::BY_HEIGHT: return *( std::max_element( _blob.begin(), _blob.end(), []( const BlobInfo & blob1, const BlobInfo & blob2 ) { return blob1.height() < blob2.height(); } ) ); - case BY_LENGTH: + case BlobCriterion::BY_LENGTH: return *( std::max_element( _blob.begin(), _blob.end(), []( const BlobInfo & blob1, const BlobInfo & blob2 ) { return blob1.length() < blob2.length(); } ) ); - case BY_SIZE: + case BlobCriterion::BY_SIZE: return *( std::max_element( _blob.begin(), _blob.end(), []( const BlobInfo & blob1, const BlobInfo & blob2 ) { return blob1.size() < blob2.size(); } ) ); - case BY_WIDTH: + case BlobCriterion::BY_WIDTH: return *( std::max_element( _blob.begin(), _blob.end(), []( const BlobInfo & blob1, const BlobInfo & blob2 ) { return blob1.width() < blob2.width(); } ) ); default: - throw penguinVException( "Bad criterion for blob finding" ); + throw penguinVException( "No criterion for blob sorting was set. Did you add a new criterion?" ); } } void BlobDetection::sort( BlobCriterion criterion ) { switch ( criterion ) { - case BY_CIRCULARITY: + case BlobCriterion::BY_CIRCULARITY: std::sort( _blob.begin(), _blob.end(), []( const BlobInfo & blob1, const BlobInfo & blob2 ) { return blob1.circularity() > blob2.circularity(); } ); break; - case BY_ELONGATION: + case BlobCriterion::BY_ELONGATION: std::sort( _blob.begin(), _blob.end(), []( const BlobInfo & blob1, const BlobInfo & blob2 ) { return blob1.elongation() > blob2.elongation(); } ); break; - case BY_HEIGHT: + case BlobCriterion::BY_HEIGHT: std::sort( _blob.begin(), _blob.end(), []( const BlobInfo & blob1, const BlobInfo & blob2 ) { return blob1.height() > blob2.height(); } ); break; - case BY_LENGTH: + case BlobCriterion::BY_LENGTH: std::sort( _blob.begin(), _blob.end(), []( const BlobInfo & blob1, const BlobInfo & blob2 ) { return blob1.length() > blob2.length(); } ); break; - case BY_SIZE: + case BlobCriterion::BY_SIZE: std::sort( _blob.begin(), _blob.end(), []( const BlobInfo & blob1, const BlobInfo & blob2 ) { return blob1.size() > blob2.size(); } ); break; - case BY_WIDTH: + case BlobCriterion::BY_WIDTH: std::sort( _blob.begin(), _blob.end(), []( const BlobInfo & blob1, const BlobInfo & blob2 ) { return blob1.width() > blob2.width(); } ); break; default: - throw penguinVException( "Bad criterion for blob sorting" ); + throw penguinVException( "No criterion for blob sorting was set. Did you add a new criterion?" ); } } } diff --git a/src/blob_detection.h b/src/blob_detection.h index 792deb9e..41bf86e4 100644 --- a/src/blob_detection.h +++ b/src/blob_detection.h @@ -1,6 +1,6 @@ /*************************************************************************** * penguinV: https://github.com/ihhub/penguinV * - * Copyright (C) 2017 - 2022 * + * Copyright (C) 2017 - 2024 * * * * This program is free software; you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * @@ -28,38 +28,38 @@ namespace Blob_Detection template struct Parameter { - Parameter() - : minimum( 0 ) // minimum value - , checkMinimum( false ) // set to compare a value with minimum value - , maximum( 0 ) // maximum value - , checkMaximum( false ) // set to compare a value with maximum value - {} + // Minimum value. + Data minimum{ 0 }; - Data minimum; - bool checkMinimum; + // Whether the value should be compared with the minimum value. + bool checkMinimum{ false }; - Data maximum; - bool checkMaximum; + // Maximum value. + Data maximum{ 0 }; + + // Whether the value should be compared with the maximum value. + bool checkMaximum{ false }; void verify() const { - if ( checkMaximum && checkMinimum && ( minimum > maximum ) ) + if ( checkMaximum && checkMinimum && ( minimum > maximum ) ) { throw penguinVException( "Minimum value cannot be bigger than maximum value" ); + } } - void set( Data min, Data max ) + void set( const Data min, const Data max ) { setMinimum( min ); setMaximum( max ); } - void setMinimum( Data min ) + void setMinimum( const Data min ) { checkMinimum = true; minimum = min; } - void setMaximum( Data max ) + void setMaximum( const Data max ) { checkMaximum = true; maximum = max; @@ -67,19 +67,30 @@ namespace Blob_Detection void reset() { - *this = Parameter(); + *this = {}; } }; struct BlobParameters { - Parameter circularity; // this parameter will be 1 if blob is ideal circle and will be less than 1 if it's not - // closer this value to 1 --> blob shape is closer to circle - Parameter elongation; // some people call it inertia: ratio of the minor and major axes of a blob - Parameter height; // height, in pixels - Parameter length; // maximum distance between any of 2 pixels, in pixels - Parameter size; // overall size of blobs, in pixels - Parameter width; // width, in pixels + // This parameter will be 1 if blob is ideal circle and will be less than 1 if it's not, + // Closer this value to 1 --> blob shape is closer to circle, + Parameter circularity; + + // Some people call it inertia: ratio of the minor and major axes of a blob. + Parameter elongation; + + // Width, in pixels. + Parameter width; + + // Height, in pixels. + Parameter height; + + // Maximum distance between any of 2 pixels, in pixels. + Parameter length; + + // Overall size of blobs, in pixels. + Parameter size; // this function will be called in BlobInfo class before finding blobs void _verify() const @@ -91,6 +102,7 @@ namespace Blob_Detection size.verify(); width.verify(); } + // reset all parameters to initial values void _reset() { @@ -105,12 +117,7 @@ namespace Blob_Detection struct Area { - Area() - : left( 0 ) - , right( 0 ) - , top( 0 ) - , bottom( 0 ) - {} + Area() = default; // this constructor is made to avoid 'Value' template restriction explicit Area( uint32_t value ) @@ -118,24 +125,21 @@ namespace Blob_Detection , right( value ) , top( value ) , bottom( value ) - {} + { + // Do nothing. + } - uint32_t left; - uint32_t right; - uint32_t top; - uint32_t bottom; + uint32_t left{ 0 }; + uint32_t right{ 0 }; + uint32_t top{ 0 }; + uint32_t bottom{ 0 }; }; template struct Value { - Value() - : value() - , found( false ) - {} - Data value; - bool found; + bool found{ false }; }; // This class follows an idea of lazy computations: @@ -147,33 +151,156 @@ namespace Blob_Detection public: friend class BlobDetection; - const std::vector & pointX() const; // returns an array what contains all blob's pixel X positions (unsorted) - const std::vector & pointY() const; // returns an array what contains all blob's pixel Y positions (unsorted) - const std::vector & contourX() const; // returns an array what contains all blob's contour pixel X positions (unsorted) - const std::vector & contourY() const; // returns an array what contains all blob's contour pixel Y positions (unsorted) - const std::vector & edgeX() const; // returns an array what contains all blob's edge pixel X positions (unsorted) - const std::vector & edgeY() const; // returns an array what contains all blob's edge pixel Y positions (unsorted) + // Returns an array what contains all blob's pixel X positions (unsorted). + const std::vector & pointX() const + { + return _pointX; + } + + // Returns an array what contains all blob's pixel Y positions (unsorted). + const std::vector & pointY() const + { + return _pointY; + } + + // Returns an array what contains all blob's contour pixel X positions (unsorted). + const std::vector & contourX() const + { + return _contourX; + } + + // Returns an array what contains all blob's contour pixel Y positions (unsorted). + const std::vector & contourY() const + { + return _contourY; + } + + // Returns an array what contains all blob's edge pixel X positions (unsorted). + const std::vector & edgeX() const + { + return _edgeX; + } + + // Returns an array what contains all blob's edge pixel Y positions (unsorted). + const std::vector & edgeY() const + { + return _edgeY; + } // Each function has 2 overloaded forms: - // - non-constant function check whether value was calculated, calculates it if neccessary and return value - // - constant function return value no matter a value was calculated or not - Area area(); // minimum fitting rectangle what can contain blob - Area area() const; // minimum fitting rectangle what can contain blob - Point2d center(); // gravity center of blob - Point2d center() const; // gravity center of blob - double circularity(); // circularity of blob - double circularity() const; // circularity of blob - double elongation(); // elongation of blob - double elongation() const; // elongation of blob - uint32_t height(); // height of blob - uint32_t height() const; // height of blob - double length(); // length of blob - double length() const; // length of blob - size_t size() const; // total number of pixels in blob - uint32_t width(); // width of blob - uint32_t width() const; // width of blob - - bool isSolid() const; // true if blob does not have inner edge points + // - non-constant function check whether value was calculated, calculates it, if necessary, and return value + // - constant function return value no matter a value was calculated or not. + + // Minimum fitting rectangle what can contain the blob. + Area area() + { + _getArea(); + + return _area.value; + } + + // Minimum fitting rectangle what can contain the blob. + Area area() const + { + return _area.value; + } + + // Gravity center of the blob. + Point2d center() + { + _getCenter(); + + return _center.value; + } + + // Gravity center of the blob. + Point2d center() const + { + return _center.value; + } + + // Circularity of the blob. + double circularity() + { + _getCircularity(); + + return _circularity.value; + } + + // Circularity of the blob. + double circularity() const + { + return _circularity.value; + } + + // Elongation of the blob. + double elongation() + { + _getElongation(); + + return _elongation.value; + } + + // Elongation of the blob. + double elongation() const + { + return _elongation.value; + } + + // Width of the blob. + uint32_t width() + { + _getWidth(); + + return _width.value; + } + + // Width of the blob. + uint32_t width() const + { + return _width.value; + } + + // Height of the blob. + uint32_t height() + { + _getHeight(); + + return _height.value; + } + + // Height of the blob. + uint32_t height() const + { + return _height.value; + } + + // Length of the blob. + double length() + { + _getLength(); + + return _length.value; + } + + // Length of the blob. + double length() const + { + return _length.value; + } + + // Total number of pixels in the blob. + size_t size() const + { + return _pointX.size(); + } + + // Returns true if blob does not have inner edge points. + bool isSolid() const + { + return _contourX.size() == _edgeX.size(); + } + private: std::vector _pointX; std::vector _pointY; @@ -205,17 +332,37 @@ namespace Blob_Detection // Sorting blobs will be in alphabet order of sorting criteria // Example: length and width criteria enabled. So first all blobs would be removed if they are not fitting length criterion // and then all remain blobs would be removed if they are not fitting for width criterion - const std::vector & find( const penguinV::Image & image, const BlobParameters & parameter = BlobParameters(), uint8_t threshold = 1 ); + const std::vector & find( const penguinV::Image & image, const BlobParameters & parameter = BlobParameters(), uint8_t threshold = 1 ) + { + return find( image, 0, 0, image.width(), image.height(), parameter, threshold ); + } + const std::vector & find( const penguinV::Image & image, uint32_t x, uint32_t y, uint32_t width, uint32_t height, const BlobParameters & parameter = BlobParameters(), uint8_t threshold = 1 ); // Retrieve an array of all found blobs - const std::vector & get() const; - std::vector & get(); - const std::vector & operator()() const; // these are same functions, added to simplify coding - std::vector & operator()(); + const std::vector & get() const + { + return _blob; + } + + std::vector & get() + { + return _blob; + } + + // These are same functions, added to simplify coding. + const std::vector & operator()() const + { + return _blob; + } + + std::vector & operator()() + { + return _blob; + } - enum BlobCriterion + enum class BlobCriterion : uint8_t { BY_CIRCULARITY, BY_ELONGATION, diff --git a/src/edge_detection.cpp b/src/edge_detection.cpp index 5544ebe8..03c3ac4d 100644 --- a/src/edge_detection.cpp +++ b/src/edge_detection.cpp @@ -29,8 +29,9 @@ namespace template void leaveFirstElement( std::vector<_Type> & data ) { - if ( data.size() > 1u ) + if ( data.size() > 1u ) { data.resize( 1u ); + } } template @@ -46,29 +47,33 @@ namespace template void createPositiveXEdge( const std::vector<_Type> & data, std::vector> & point, _Type x, _Type y ) { - for ( typename std::vector<_Type>::const_iterator dataX = data.cbegin(); dataX != data.cend(); ++dataX ) + for ( auto dataX = data.cbegin(); dataX != data.cend(); ++dataX ) { point.push_back( PointBase2D<_Type>( ( *dataX ) + x, y ) ); + } } template void createNegativeXEdge( const std::vector<_Type> & data, std::vector> & point, _Type x, _Type y ) { - for ( typename std::vector<_Type>::const_iterator dataX = data.cbegin(); dataX != data.cend(); ++dataX ) + for ( auto dataX = data.cbegin(); dataX != data.cend(); ++dataX ) { point.push_back( PointBase2D<_Type>( x - ( *dataX ), y ) ); + } } template void createPositiveYEdge( const std::vector<_Type> & data, std::vector> & point, _Type x, _Type y ) { - for ( typename std::vector<_Type>::const_iterator dataY = data.cbegin(); dataY != data.cend(); ++dataY ) + for ( auto dataY = data.cbegin(); dataY != data.cend(); ++dataY ) { point.push_back( PointBase2D<_Type>( x, y + ( *dataY ) ) ); + } } template void createNegativeYEdge( const std::vector<_Type> & data, std::vector> & point, _Type x, _Type y ) { - for ( typename std::vector<_Type>::const_iterator dataY = data.cbegin(); dataY != data.cend(); ++dataY ) + for ( auto dataY = data.cbegin(); dataY != data.cend(); ++dataY ) { point.push_back( PointBase2D<_Type>( x, y - ( *dataY ) ) ); + } } template @@ -78,8 +83,9 @@ namespace const int maxIntensity = *( std::max_element( data.begin() + leftSideOffset, data.begin() + rightSideOffset + 1 ) ); const int minIntensity = *( std::min_element( data.begin() + leftSideOffset, data.begin() + rightSideOffset + 1 ) ); - if ( maxIntensity - minIntensity < minimumContrast ) + if ( maxIntensity - minIntensity < minimumContrast ) { return false; + } if ( checkContrast && leftSideContrastCheck <= position && ( rightSideContrastCheck + position ) < size ) { const uint32_t blackContrastEnd = position; @@ -99,10 +105,13 @@ namespace } if ( !checkContrast ) { - if ( second[position] != second[position + 1] ) + if ( second[position] != second[position + 1] ) { edge.push_back( static_cast<_Type>( position + static_cast( second[position] ) / ( second[position] - second[position + 1u] ) ) ); - else + } + else { edge.push_back( static_cast<_Type>( position ) + 0.5f ); + } + return true; } @@ -119,13 +128,15 @@ namespace const bool checkContrast = ( edgeParameter.contrastCheckLeftSideOffset > 0u ) && ( edgeParameter.contrastCheckRightSideOffset > 0u ); for ( uint32_t i = 1u; i < dataSize - 2u; i++ ) { - if ( second[i] < 0 || second[i + 1] > 0 ) + if ( second[i] < 0 || second[i + 1] > 0 ) { continue; + } const int maxGradient = ( first[i] > first[i + 1] ) ? first[i] : first[i + 1]; - if ( maxGradient <= 0 ) + if ( maxGradient <= 0 ) { continue; + } const int minGradientValue = ( maxGradient < 3 ) ? 1 : maxGradient / 3; const int halfGradient = ( maxGradient < 2 ) ? 1 : maxGradient / 2; @@ -200,18 +211,20 @@ namespace void removeSimilarPoints( std::vector<_Type> & edge ) { for ( size_t i = 1u; i < edge.size(); ) { - if ( ( edge[i] - edge[i - 1u] ) < 1.0 ) + if ( ( edge[i] - edge[i - 1u] ) < 1.0 ) { edge.erase( edge.begin() + static_cast::difference_type>( i ) ); // it's safe to do - else + } + else { ++i; + } } } void getDerivatives( const std::vector & image, std::vector & first, std::vector & second ) { // input array range is [0; n) - // first deriviative range is [0; n - 1) - // second deriviative range is [1; n - 1) + // first derivative range is [0; n - 1) + // second derivative range is [1; n - 1) std::transform( image.begin() + 1u, image.end(), image.begin(), first.begin(), std::minus() ); std::transform( first.begin() + 1u, first.end(), first.begin(), second.begin() + 1u, std::minus() ); } @@ -251,8 +264,9 @@ namespace const bool horizontalEdgeDetectionBase = ( edgeParameter.direction == EdgeParameter::DirectionType::LEFT_TO_RIGHT || edgeParameter.direction == EdgeParameter::DirectionType::RIGHT_TO_LEFT ); - if ( ( horizontalEdgeDetectionBase && ( width < 4u ) ) || ( !horizontalEdgeDetectionBase && ( height < 4u ) ) ) + if ( ( horizontalEdgeDetectionBase && ( width < 4u ) ) || ( !horizontalEdgeDetectionBase && ( height < 4u ) ) ) { return; + } const uint32_t rowSize = image.rowSize(); @@ -275,8 +289,9 @@ namespace imageXEnd = imageX + width; dataX = data.data(); - for ( ; imageX != imageXEnd; ++imageX, ++dataX ) + for ( ; imageX != imageXEnd; ++imageX, ++dataX ) { *dataX += *imageX; + } } std::vector<_Type> edgePositive; @@ -287,18 +302,22 @@ namespace const _Type yPosition = static_cast<_Type>( y + rowId + ( edgeParameter.groupFactor - 1 ) / 2.0 ); if ( edgeParameter.direction == EdgeParameter::DirectionType::LEFT_TO_RIGHT ) { - if ( edgeParameter.gradient == EdgeParameter::GradientType::POSITIVE || edgeParameter.gradient == EdgeParameter::GradientType::ANY ) + if ( edgeParameter.gradient == EdgeParameter::GradientType::POSITIVE || edgeParameter.gradient == EdgeParameter::GradientType::ANY ) { createPositiveXEdge( edgePositive, positiveEdgePoint, static_cast<_Type>( x ), yPosition ); + } - if ( edgeParameter.gradient == EdgeParameter::GradientType::NEGATIVE || edgeParameter.gradient == EdgeParameter::GradientType::ANY ) + if ( edgeParameter.gradient == EdgeParameter::GradientType::NEGATIVE || edgeParameter.gradient == EdgeParameter::GradientType::ANY ) { createNegativeXEdge( edgeNegative, negativeEdgePoint, static_cast<_Type>( x + width - 1 ), yPosition ); + } } else { - if ( edgeParameter.gradient == EdgeParameter::GradientType::POSITIVE || edgeParameter.gradient == EdgeParameter::GradientType::ANY ) + if ( edgeParameter.gradient == EdgeParameter::GradientType::POSITIVE || edgeParameter.gradient == EdgeParameter::GradientType::ANY ) { createNegativeXEdge( edgeNegative, positiveEdgePoint, static_cast<_Type>( x + width - 1 ), yPosition ); + } - if ( edgeParameter.gradient == EdgeParameter::GradientType::NEGATIVE || edgeParameter.gradient == EdgeParameter::GradientType::ANY ) + if ( edgeParameter.gradient == EdgeParameter::GradientType::NEGATIVE || edgeParameter.gradient == EdgeParameter::GradientType::ANY ) { createPositiveXEdge( edgePositive, negativeEdgePoint, static_cast<_Type>( x ), yPosition ); + } } } } @@ -312,16 +331,18 @@ namespace const uint8_t * imageY = imageX; const uint8_t * imageYEnd = imageY + height * rowSize; int * dataX = data.data(); - for ( ; imageY != imageYEnd; imageY += rowSize, ++dataX ) + for ( ; imageY != imageYEnd; imageY += rowSize, ++dataX ) { *dataX = *imageY; + } for ( uint32_t groupId = 1u; groupId < edgeParameter.groupFactor; ++groupId ) { imageY = imageX + groupId; imageYEnd = imageY + height * rowSize; dataX = data.data(); - for ( ; imageY != imageYEnd; imageY += rowSize, ++dataX ) + for ( ; imageY != imageYEnd; imageY += rowSize, ++dataX ) { *dataX += *imageY; + } } std::vector<_Type> edgePositive; @@ -332,18 +353,22 @@ namespace const _Type xPosition = static_cast<_Type>( x + rowId + ( edgeParameter.groupFactor - 1 ) / 2.0 ); if ( edgeParameter.direction == EdgeParameter::DirectionType::TOP_TO_BOTTOM ) { - if ( edgeParameter.gradient == EdgeParameter::GradientType::POSITIVE || edgeParameter.gradient == EdgeParameter::GradientType::ANY ) + if ( edgeParameter.gradient == EdgeParameter::GradientType::POSITIVE || edgeParameter.gradient == EdgeParameter::GradientType::ANY ) { createPositiveYEdge( edgePositive, positiveEdgePoint, xPosition, static_cast<_Type>( y ) ); + } - if ( edgeParameter.gradient == EdgeParameter::GradientType::NEGATIVE || edgeParameter.gradient == EdgeParameter::GradientType::ANY ) + if ( edgeParameter.gradient == EdgeParameter::GradientType::NEGATIVE || edgeParameter.gradient == EdgeParameter::GradientType::ANY ) { createNegativeYEdge( edgeNegative, negativeEdgePoint, xPosition, static_cast<_Type>( y + height - 1 ) ); + } } else { - if ( edgeParameter.gradient == EdgeParameter::GradientType::POSITIVE || edgeParameter.gradient == EdgeParameter::GradientType::ANY ) + if ( edgeParameter.gradient == EdgeParameter::GradientType::POSITIVE || edgeParameter.gradient == EdgeParameter::GradientType::ANY ) { createNegativeYEdge( edgeNegative, positiveEdgePoint, xPosition, static_cast<_Type>( y + height - 1 ) ); + } - if ( edgeParameter.gradient == EdgeParameter::GradientType::NEGATIVE || edgeParameter.gradient == EdgeParameter::GradientType::ANY ) + if ( edgeParameter.gradient == EdgeParameter::GradientType::NEGATIVE || edgeParameter.gradient == EdgeParameter::GradientType::ANY ) { createPositiveYEdge( edgePositive, negativeEdgePoint, xPosition, static_cast<_Type>( y ) ); + } } } } @@ -360,16 +385,23 @@ EdgeParameter::EdgeParameter( DirectionType _direction, GradientType _gradient, , contrastCheckLeftSideOffset( _contrastCheckLeftSideOffset ) , contrastCheckRightSideOffset( _contrastCheckRightSideOffset ) , minimumContrast( _minimumContrast ) -{} +{ + // Do nothing. +} void EdgeParameter::verify() const { - if ( groupFactor == 0u ) + if ( groupFactor == 0u ) { throw penguinVException( "Grouping factor for edge detection cannot be 0" ); - if ( skipFactor == 0u ) + } + + if ( skipFactor == 0u ) { throw penguinVException( "Skip factor for edge detection cannot be 0" ); - if ( minimumContrast == 0u ) + } + + if ( minimumContrast == 0u ) { throw penguinVException( "Minimum contrast for edge detection cannot be 0" ); + } } void EdgeDetectionHelper::find( EdgeDetectionBase & edgeDetection, const penguinV::Image & image, uint32_t x, uint32_t y, uint32_t width, uint32_t height, diff --git a/src/function_pool.cpp b/src/function_pool.cpp index ccf68860..8a43caf0 100644 --- a/src/function_pool.cpp +++ b/src/function_pool.cpp @@ -1,6 +1,6 @@ /*************************************************************************** * penguinV: https://github.com/ihhub/penguinV * - * Copyright (C) 2017 - 2022 * + * Copyright (C) 2017 - 2024 * * * * This program is free software; you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * @@ -29,27 +29,17 @@ namespace Function_Pool // This structure holds input parameters for some specific functions struct InputInfo { - InputInfo() - : minThreshold( 0 ) - , maxThreshold( 255 ) - , horizontalProjection( false ) - , coefficientA( 1 ) - , coefficientGamma( 1 ) - , extractChannelId( 255 ) - , horizontalFlip( false ) - , verticalFlip( false ) - {} - - uint8_t minThreshold; // for Threshold() function same as threshold - uint8_t maxThreshold; // for Threshold() function - bool horizontalProjection; // for ProjectionProfile() function - double coefficientA; // for GammaCorrection() function - double coefficientGamma; // for GammaCorrection() function - uint8_t extractChannelId; // for ExtractChannel() function + uint8_t minThreshold{ 0 }; // for Threshold() function same as threshold + uint8_t maxThreshold{ 255 }; // for Threshold() function + uint8_t extractChannelId{ 255 }; // for ExtractChannel() function + bool horizontalProjection{ false }; // for ProjectionProfile() function + bool horizontalFlip{ false }; + bool verticalFlip{ false }; + double coefficientA{ 1.0 }; // for GammaCorrection() function + double coefficientGamma{ 1.0 }; // for GammaCorrection() function std::vector lookupTable; // for LookupTable() function - bool horizontalFlip; - bool verticalFlip; }; + // This structure holds output data for some specific functions struct OutputInfo { @@ -145,12 +135,13 @@ namespace Function_Pool throw penguinVException( "Returned histograms are not the same size" ); for ( size_t i = 1; i < input.size(); ++i ) { - std::vector::iterator out = output.begin(); - std::vector::const_iterator in = input[i].begin(); - std::vector::const_iterator end = input[i].end(); + auto out = output.begin(); + auto in = input[i].begin(); + auto end = input[i].end(); - for ( ; in != end; ++in, ++out ) + for ( ; in != end; ++in, ++out ) { *out += *in; + } } input.clear(); // to guarantee that no one can use it second time diff --git a/src/math/haar_transform.cpp b/src/math/haar_transform.cpp index 730aa17a..7332d8a3 100644 --- a/src/math/haar_transform.cpp +++ b/src/math/haar_transform.cpp @@ -1,6 +1,6 @@ /*************************************************************************** * penguinV: https://github.com/ihhub/penguinV * - * Copyright (C) 2017 - 2022 * + * Copyright (C) 2017 - 2024 * * * * This program is free software; you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * @@ -27,8 +27,9 @@ namespace template void directTransform( const std::vector<_Type> & input, std::vector<_Type> & output, size_t width, size_t height ) { - if ( input.empty() || input.size() != output.size() || input.size() != width * height || ( width % 2 ) != 0 || ( height % 2 ) != 0 ) - throw penguinVException( "Incorrect input parameters for Haar direct transform" ); + if ( input.empty() || input.size() != output.size() || input.size() != width * height || ( width % 2 ) != 0 || ( height % 2 ) != 0 ) { + throw penguinVException( "Incorrect input parameters for Haar direct transform." ); + } // Direct Haar Matrix // | 1/sqrt(2) 1/sqrt(2) | @@ -80,7 +81,7 @@ namespace void inverseTransform( const std::vector<_Type> & input, std::vector<_Type> & output, size_t width, size_t height ) { if ( input.empty() || input.size() != output.size() || input.size() != width * height || ( width % 2 ) != 0 || ( height % 2 ) != 0 ) - throw penguinVException( "Incorrect input parameters for Haar inverse transform" ); + throw penguinVException( "Incorrect input parameters for Haar inverse transform." ); // Inverse Haar Matrix // | 1/sqrt(2) -1/sqrt(2) | diff --git a/src/math/hough_transform.cpp b/src/math/hough_transform.cpp index 19047bcc..e1ec68cc 100644 --- a/src/math/hough_transform.cpp +++ b/src/math/hough_transform.cpp @@ -36,8 +36,9 @@ namespace std::vector> & outOnLine, std::vector> & outOffLine ) { // validate input data - if ( input.size() < 2u ) + if ( input.size() < 2u ) { return false; + } if ( angleStep < minimumAngleStep ) { angleStep = minimumAngleStep; @@ -91,8 +92,9 @@ namespace const _Type tolerance = lineToleranceRange + distanceToLine[pointId]; for ( ; endPointId < inputPointCount; ++endPointId ) { - if ( tolerance < distanceToLine[endPointId] ) + if ( tolerance < distanceToLine[endPointId] ) { break; + } } if ( onLinePointCount < endPointId - pointId ) { @@ -130,10 +132,12 @@ namespace for ( ; point != pointEnd; ++point, ++distanceVal ) { ( *distanceVal ) = point->x * sinVal + point->y * cosVal; - if ( ( ( *distanceVal ) < minDistance ) || ( ( *distanceVal ) > maxDistance ) ) - outOffLine.push_back( ( *point ) ); - else - outOnLine.push_back( ( *point ) ); + if ( ( ( *distanceVal ) < minDistance ) || ( ( *distanceVal ) > maxDistance ) ) { + outOffLine.push_back( *point ); + } + else { + outOnLine.push_back( *point ); + } } return true; diff --git a/src/math/math_base.h b/src/math/math_base.h index 57236e03..cfbedc9c 100644 --- a/src/math/math_base.h +++ b/src/math/math_base.h @@ -280,7 +280,7 @@ class LineBase2D _Type sumYY = 0; _Type sumXY = 0; - for ( typename _container>::const_iterator point = points.begin(); point != points.end(); ++point ) { + for ( auto point = points.begin(); point != points.end(); ++point ) { const _Type x = point->x; const _Type y = point->y; sumX += x; diff --git a/src/memory/cpu_memory.h b/src/memory/cpu_memory.h index d6d1526f..33eb7830 100644 --- a/src/memory/cpu_memory.h +++ b/src/memory/cpu_memory.h @@ -1,6 +1,6 @@ /*************************************************************************** * penguinV: https://github.com/ihhub/penguinV * - * Copyright (C) 2017 - 2022 * + * Copyright (C) 2017 - 2024 * * * * This program is free software; you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * @@ -30,10 +30,10 @@ namespace cpu_Memory class MemoryAllocator : public BaseMemoryAllocator { public: - MemoryAllocator() - : _data( nullptr ) - , _alignedData( nullptr ) - {} + MemoryAllocator() = default; + + MemoryAllocator( const MemoryAllocator & ) = delete; + MemoryAllocator & operator=( const MemoryAllocator & ) = delete; static MemoryAllocator & instance() { @@ -61,7 +61,7 @@ namespace cpu_Memory const uint8_t level = _getAllocationLevel( overallSize ); if ( _split( level ) ) { - std::set::iterator chunk = _freeChunk[level].begin(); + auto chunk = _freeChunk[level].begin(); _DataType * address = reinterpret_cast<_DataType *>( _alignedData + *chunk ); _allocatedChunk.insert( std::pair( *chunk, level ) ); _freeChunk[level].erase( chunk ); @@ -83,7 +83,7 @@ namespace cpu_Memory { _lock.lock(); if ( _data != nullptr && reinterpret_cast( address ) >= _alignedData ) { - std::map::iterator pos = _allocatedChunk.find( static_cast( reinterpret_cast( address ) - _alignedData ) ); + auto pos = _allocatedChunk.find( static_cast( reinterpret_cast( address ) - _alignedData ) ); if ( pos != _allocatedChunk.end() ) { _freeChunk[pos->second].insert( pos->first ); @@ -99,8 +99,8 @@ namespace cpu_Memory } private: - uint8_t * _data; // a pointer to memory allocated chunk - uint8_t * _alignedData; // aligned pointer for SIMD access + uint8_t * _data{ nullptr }; // a pointer to memory allocated chunk + uint8_t * _alignedData{ nullptr }; // aligned pointer for SIMD access std::mutex _lock; // a map which holds an information about allocated memory in preallocated memory chunk @@ -112,8 +112,9 @@ namespace cpu_Memory { _lock.lock(); if ( _size != size && size > 0 ) { - if ( !_allocatedChunk.empty() ) + if ( !_allocatedChunk.empty() ) { throw std::logic_error( "Cannot free a memory on CPU. Not all objects were previously deallocated from allocator." ); + } _free(); @@ -138,11 +139,5 @@ namespace cpu_Memory _allocatedChunk.clear(); } - - MemoryAllocator( const MemoryAllocator & ) {} - MemoryAllocator & operator=( const MemoryAllocator & ) - { - return ( *this ); - } }; } From 173a74934c62e559d09f096dcb82bff848896bf3 Mon Sep 17 00:00:00 2001 From: Ihar Hubchyk Date: Sun, 17 Mar 2024 19:12:33 +0800 Subject: [PATCH 2/2] Fix compilation --- examples/blob_detection/example_blob_detection.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/examples/blob_detection/example_blob_detection.cpp b/examples/blob_detection/example_blob_detection.cpp index 2b81e653..95690226 100644 --- a/examples/blob_detection/example_blob_detection.cpp +++ b/examples/blob_detection/example_blob_detection.cpp @@ -59,7 +59,7 @@ int main( int argc, char * argv[] ) if ( !detection().empty() ) { // okay, our image contains some blobs // extract a biggest one - const Blob_Detection::BlobInfo & blob = detection.getBestBlob( Blob_Detection::BlobDetection::BY_SIZE ); + const Blob_Detection::BlobInfo & blob = detection.getBestBlob( Blob_Detection::BlobDetection::BlobCriterion::BY_SIZE ); // clear image and draw contour of found blob image.fill( 0 );