From 420e4cff0bed220c16c601cd7d478c0c2af09b47 Mon Sep 17 00:00:00 2001
From: Markus Frank <Markus.Frank@cern.ch>
Date: Wed, 17 Jan 2024 13:35:02 +0100
Subject: [PATCH] Try to fix (some) CAD interface tests

---
 DDCAD/include/DDCAD/Utilities.h | 41 ++++++++++++++++++++++++++++-----
 DDCAD/src/ASSIMPReader.cpp      | 30 ++++++++++++++----------
 DDCAD/src/ASSIMPWriter.cpp      |  6 ++---
 DDCore/include/DD4hep/Handle.h  |  4 +++-
 DDCore/src/Handle.cpp           |  4 ++++
 5 files changed, 63 insertions(+), 22 deletions(-)

diff --git a/DDCAD/include/DDCAD/Utilities.h b/DDCAD/include/DDCAD/Utilities.h
index ef5a4c89b..1b0a86f08 100644
--- a/DDCAD/include/DDCAD/Utilities.h
+++ b/DDCAD/include/DDCAD/Utilities.h
@@ -15,6 +15,7 @@
 
 #include <vector>
 #include <cmath>
+#include <limits>
 
 #include <TGeoTessellated.h>
 #include <TGeoVector3.h>
@@ -25,7 +26,7 @@ namespace dd4hep {
   /// Namespace for implementation details of the AIDA detector description toolkit
   namespace cad  {
 
-    inline std::stringstream streamFacet(TGeoFacet const& facet,
+    inline std::string streamFacet(TGeoFacet const& facet,
 					 TGeoTessellated const& shape) {
       using ::operator<<;
       std::stringstream str;
@@ -40,24 +41,24 @@ namespace dd4hep {
 #else
       str << facet;
 #endif
-      return str;
+      return str.str();
     }
 
-    inline std::stringstream streamVertices(ROOT::Geom::Vertex_t const& v1,
+    inline std::string streamVertices(ROOT::Geom::Vertex_t const& v1,
 					    ROOT::Geom::Vertex_t const& v2,
 					    ROOT::Geom::Vertex_t const& v3) {
       using ::operator<<;
       std::stringstream str;
       str << "{" << v1 << ", " << v2 <<  ", " << v3 << "}";
-      return str;
+      return str.str();
     }
-
     // Determine if the facet is degenerated by calculating its determinant
     inline bool facetIsDegenerated(std::vector<ROOT::Geom::Vertex_t> const& vertices){
+#if 0
       const ROOT::Geom::Vertex_t& v1 = vertices[0];
       const ROOT::Geom::Vertex_t& v2 = vertices[1];
       const ROOT::Geom::Vertex_t& v3 = vertices[2];
-      constexpr double epsilon = 1.0e-10;
+      constexpr double epsilon = std::numeric_limits<double>::epsilon();
       // v1.x v2.x v3.x v1.x v2.x
       //
       // v1.y v2.y v3.y	v1.y v2.y
@@ -71,6 +72,34 @@ namespace dd4hep {
         - v2.z() * v3.y() * v1.x()
         - v3.z() * v1.y() * v2.x();
       return std::fabs(det) < epsilon;
+#else
+      using ROOT::Geom::Vertex_t;
+      // Compute normal using non-zero segments
+      constexpr double kTolerance = 1.e-20;
+      bool degenerated = true;
+      Vertex_t normal;
+      int fNvert = int(vertices.size());
+      for (int i = 0; i < fNvert - 1; ++i) {
+        Vertex_t e1 = vertices[i + 1] - vertices[i];
+        if (e1.Mag2() < kTolerance)
+          continue;
+        for (int j = i + 1; j < fNvert; ++j) {
+          Vertex_t e2 = vertices[(j + 1) % fNvert] - vertices[j];
+          if (e2.Mag2() < kTolerance)
+            continue;
+          normal = Vertex_t::Cross(e1, e2);
+          // e1 and e2 may be colinear
+          if (normal.Mag2() < kTolerance)
+            continue;
+          normal.Normalize();
+          degenerated = false;
+          break;
+        }
+        if (!degenerated)
+          break;
+      }
+      return degenerated;
+#endif
     }
   }
 }
diff --git a/DDCAD/src/ASSIMPReader.cpp b/DDCAD/src/ASSIMPReader.cpp
index 67a1c37a0..5a22c5b3b 100644
--- a/DDCAD/src/ASSIMPReader.cpp
+++ b/DDCAD/src/ASSIMPReader.cpp
@@ -63,9 +63,9 @@ ASSIMPReader::readShapes(const std::string& source, double unit_length)  const
         if ( dump_facets )   {
           for( size_t i=0, n=shape->GetNfacets(); i < n; ++i )   {
             const auto& facet = shape->GetFacet(i);
-            std::stringstream str = dd4hep::cad::streamFacet(facet, shape);
+            std::string str = dd4hep::cad::streamFacet(facet, shape);
             printout(ALWAYS,"ASSIMPReader","++ Facet %4ld : %s",
-                     i, str.str().c_str());
+                     i, str.c_str());
           }
         }
         shape->SetTitle(TESSELLATEDSOLID_TAG);
@@ -90,7 +90,6 @@ ASSIMPReader::readVolumes(const std::string& source, double unit_length)  const
   bool dump_facets = ((flags>>8)&0x1) == 1;
   int aiflags = aiProcess_Triangulate|aiProcess_JoinIdenticalVertices|aiProcess_CalcTangentSpace;
   auto scene = importer->ReadFile( source.c_str(), aiflags);
-  char text[1048];
   
   if ( !scene )  {
     except("ASSIMPReader","+++ FileNotFound: %s",source.c_str());
@@ -108,9 +107,20 @@ ASSIMPReader::readVolumes(const std::string& source, double unit_length)  const
         vertices.emplace_back(Vertex(v[i].x*unit, v[i].y*unit, v[i].z*unit));
       }
       TessellatedSolid shape(name,vertices);
+      if ( name.empty() )  {
+        name = _toString(result.size(), "tessellated_%ld");
+      }
+
       for(unsigned int i=0; i < mesh->mNumFaces; i++)  {
         const unsigned int* idx  = mesh->mFaces[i].mIndices;
-        shape->AddFacet(idx[0], idx[1], idx[2]);
+        bool degenerated = dd4hep::cad::facetIsDegenerated({vertices[idx[0]], vertices[idx[1]], vertices[idx[2]]});
+        if ( !degenerated )   {
+          shape->AddFacet(idx[0], idx[1], idx[2]);
+          continue;
+        }
+        printout(INFO, "ASSIMPReader", "+++ %s: Drop degenerated facet: %d %d %d",
+                 name.c_str(), idx[0], idx[1], idx[2]);
+
       }
       if ( shape->GetNfacets() > 2 )   {
         std::string mat_name;
@@ -124,14 +134,9 @@ ASSIMPReader::readVolumes(const std::string& source, double unit_length)  const
         if ( !mat.isValid() )   {
           printout(ERROR, "ASSIMPReader",
                    "+++ %s: No material named '%s' FOUND. Will use Air. [Missing material]",
-                   text, mat_name.c_str());
+                   name.c_str(), mat_name.c_str());
           mat = detector.air();
         }
-        if ( name.empty() )  {
-          ::snprintf(text,sizeof(text),"tessellated_%ld", result.size());
-          text[sizeof(text)-1] = 0;
-          name = text;
-        }
         Volume vol(name, Solid(shape.ptr()), mat);
         if ( mesh->HasVertexColors(0) )   {
           const aiColor4D* col = mesh->mColors[0];
@@ -146,6 +151,7 @@ ASSIMPReader::readVolumes(const std::string& source, double unit_length)  const
               }
             }
             if ( !vis.isValid() )   {
+              char text[1024];
               ::snprintf(text,sizeof(text),"vis_%s_%p", name.c_str(), (void*)vol.ptr());
               text[sizeof(text)-1] = 0;
               vis = VisAttr(text);
@@ -162,9 +168,9 @@ ASSIMPReader::readVolumes(const std::string& source, double unit_length)  const
         if ( dump_facets )   {
           for( size_t i=0, n=shape->GetNfacets(); i < n; ++i )   {
             const auto& facet = shape->GetFacet(i);
-            std::stringstream str = dd4hep::cad::streamFacet(facet, shape);
+            std::string str = dd4hep::cad::streamFacet(facet, shape);
             printout(ALWAYS,"ASSIMPReader","++ Facet %4ld : %s",
-                     i, str.str().c_str());
+                     i, str.c_str());
           }
         }
         result.emplace_back(std::unique_ptr<TGeoVolume>(vol.ptr()));
diff --git a/DDCAD/src/ASSIMPWriter.cpp b/DDCAD/src/ASSIMPWriter.cpp
index f31219174..f468c63e0 100644
--- a/DDCAD/src/ASSIMPWriter.cpp
+++ b/DDCAD/src/ASSIMPWriter.cpp
@@ -159,7 +159,7 @@ namespace  {
             continue;
           }
 #if ROOT_VERSION_CODE >= ROOT_VERSION(6,31,1)
-          bool degenerated = dd4hep::cad::facetIsDegenerated(vertices);
+          bool degenerated = dd4hep::cad::facetIsDegenerated({vertices[vv0],vertices[vv1],vertices[vv2]});
 #else
           bool degenerated = true;
           TGeoFacet f(&vertices, 3, vv0, vv1, vv2);
@@ -426,8 +426,8 @@ int ASSIMPWriter::write(const std::string& file_name,
           ROOT::Geom::Vertex_t v1(vv[id[0]].x, vv[id[0]].y, vv[id[0]].z);
           ROOT::Geom::Vertex_t v2(vv[id[1]].x, vv[id[1]].y, vv[id[1]].z);
           ROOT::Geom::Vertex_t v3(vv[id[2]].x, vv[id[2]].y, vv[id[2]].z);
-          std::stringstream str = dd4hep::cad::streamVertices(v1, v2, v3);
-          printout(ALWAYS,"ASSIMPWriter","++ Facet %4ld : %s", j, str.str().c_str());
+          std::string str = dd4hep::cad::streamVertices(v1, v2, v3);
+          printout(ALWAYS,"ASSIMPWriter","++ Facet %4ld : %s", j, str.c_str());
         }
       }
       else   {
diff --git a/DDCore/include/DD4hep/Handle.h b/DDCore/include/DD4hep/Handle.h
index f337140b0..8dfdbce70 100644
--- a/DDCore/include/DD4hep/Handle.h
+++ b/DDCore/include/DD4hep/Handle.h
@@ -239,10 +239,12 @@ namespace dd4hep {
 
   /// String conversions: boolean value to string  \ingroup DD4HEP_CORE
   std::string _toString(bool value);
-  /// String conversions: integer value to string  \ingroup DD4HEP_CORE
+  /// String conversions: short integer value to string  \ingroup DD4HEP_CORE
   std::string _toString(short value, const char* fmt = "%d");
   /// String conversions: integer value to string  \ingroup DD4HEP_CORE
   std::string _toString(int value, const char* fmt = "%d");
+  /// String conversions: unsigned long integer value to string  \ingroup DD4HEP_CORE
+  std::string _toString(unsigned long value, const char* fmt = "%ld");
   /// String conversions: float value to string  \ingroup DD4HEP_CORE
   std::string _toString(float value, const char* fmt = "%.17e");
   /// String conversions: double value to string  \ingroup DD4HEP_CORE
diff --git a/DDCore/src/Handle.cpp b/DDCore/src/Handle.cpp
index 7cfa46fcb..cbc9b1b7b 100644
--- a/DDCore/src/Handle.cpp
+++ b/DDCore/src/Handle.cpp
@@ -344,6 +344,10 @@ namespace dd4hep  {
     return __to_string(value, fmt);
   }
 
+  string _toString(unsigned long value, const char* fmt) {
+    return __to_string(value, fmt);
+  }
+
   string _toString(float value, const char* fmt) {
     return __to_string(value, fmt);
   }
-- 
GitLab