Skip to content

Commit

Permalink
- Parallelize the Part Number writing section.
Browse files Browse the repository at this point in the history
- AtomicFile: Use unique error codes and better error message formatting.

Signed-off-by: Michael Jackson <[email protected]>
  • Loading branch information
imikejackson committed Jul 29, 2024
1 parent e04b179 commit 51a1291
Show file tree
Hide file tree
Showing 4 changed files with 181 additions and 14 deletions.
7 changes: 5 additions & 2 deletions src/Plugins/SimplnxCore/docs/CombineStlFilesFilter.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,13 +2,16 @@

## Group (Subgroup)

Reader/Input (AMProcessMonitoring)
Reader/Input

## Description

This **Filter** combines all of the STL files from a given directory into a single triangle geometry. This filter will make use of the **Import STL File Filter** to read in each stl file in the given directory and then will proceed to combine each of the imported files into a single triangle geometry.

There is an option to label the triangles and vertices with the "index" of the file that was read. This would be based on the lexographical index and starts from 1. This allows for the immediate "segmentation" of the resulting triangle geometry or just as a convenience to "color by" in the visualization widget.
There is an option to label the faces and vertices with a "Part Number" that represents the index into the list of files that was used as the input. This would be based on the lexographical index and starts from 1. This allows for the immediate "segmentation" of the resulting triangle geometry or just as a convenience to "color by" in the visualization widget. This can
also be used in the "Write STL Files from Triangle Geometry" Filter if the selection for
the "File Grouping Type" is set to "Part Index" in the UI. Then use the "Part Number" array
that is created in this filter for the "Part Index".

% Auto generated parameter table will be inserted here

Expand Down
15 changes: 14 additions & 1 deletion src/Plugins/SimplnxCore/docs/WriteStlFileFilter.md
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,20 @@ IO (Output)

## Description

This **Filter** will write a binary STL File for each unique **Feature** Id in the associated **Triangle** geometry. The STL files will be named with the [Feature_Id].stl. The user can designate an optional prefix for the files.
This **Filter** can generate a single or multiple binary STL (StereoLithography)
files for a given Triangle Geometry. This is controlled by the "File Grouping Type"
parameter which is a combo box parameter type with the following choices:

- Features: The user must supply a 2 component Int32 array where the values are the 2 features that the triangle belongs to.
- Phases and Features: The user must the same kind of array as in the "Features" but also
must supply a single component array that represents another higher order of grouping, such
as a phase (in the case of microstructure data) or a part number in the base of importing
a large multipart geometry.
- Single File: The entire Triangle Geometry is saved as a single STL File
- Part Index: The user must supply a single component Int32 array at the triangle
level that represents which part the triangle belongs to. This is handy when the Triangle Geometry
has distinct, non-overlapping "parts", such as an AM build with.


% Auto generated parameter table will be inserted here

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,9 @@
#include "simplnx/DataStructure/Geometry/TriangleGeom.hpp"
#include "simplnx/Utilities/FilterUtilities.hpp"
#include "simplnx/Utilities/Math/MatrixMath.hpp"
#include "simplnx/Utilities/ParallelAlgorithmUtilities.hpp"
#include "simplnx/Utilities/ParallelDataAlgorithm.hpp"
#include "simplnx/Utilities/ParallelTaskAlgorithm.hpp"
#include "simplnx/Utilities/StringUtilities.hpp"

#include <filesystem>
Expand Down Expand Up @@ -243,6 +246,154 @@ Result<> MultiWriteOutStl(const fs::path& path, const IGeometry::MeshIndexType n
fclose(filePtr);
return result;
}

class MultiWriteStlFileImpl
{
public:
MultiWriteStlFileImpl(const fs::path& path, const IGeometry::MeshIndexType numTriangles, const std::string& header, const IGeometry::MeshIndexArrayType& triangles, const Float32Array& vertices,
const Int32Array& featureIds, const int32 featureId)
: m_Path(path)
, m_NumTriangles(numTriangles)
, m_Header(header)
, m_Triangles(triangles)
, m_Vertices(vertices)
, m_FeatureIds(featureIds)
, m_FeatureId(featureId)
{
}
~MultiWriteStlFileImpl() = default;

void operator()() const
{
Result<> result;

// Create output file writer in binary write out mode to ensure cross-compatibility
FILE* filePtr = fopen(m_Path.string().c_str(), "wb");

if(filePtr == nullptr)
{
fclose(filePtr);
std::cout << fmt::format("Error Opening STL File. Unable to create temp file at path '{}' for original file '{}'", m_Path.string(), m_Path.filename().string()) << "\n";
return;
// return {MakeWarningVoidResult(-27876, fmt::format("Error Opening STL File. Unable to create temp file at path '{}' for original file '{}'", m_Path.string(), m_Path.filename().string()))};
}

int32 triCount = 0;

{ // Scope header output processing to keep overhead low and increase readability
if(m_Header.size() >= 80)
{
result = MakeWarningVoidResult(-27874,
fmt::format("Warning: Writing STL File '{}'. Header was over the 80 characters supported by STL. Length of header: {}. Only the first 80 bytes will be written.",
m_Path.filename().string(), m_Header.length()));
}

std::array<char, 80> stlFileHeader = {};
stlFileHeader.fill(0);
size_t headLength = 80;
if(m_Header.length() < 80)
{
headLength = static_cast<size_t>(m_Header.length());
}

// std::string c_str = header;
memcpy(stlFileHeader.data(), m_Header.data(), headLength);
// Return the number of bytes written - which should be 80
fwrite(stlFileHeader.data(), 1, 80, filePtr);
}

fwrite(&triCount, 1, 4, filePtr);
triCount = 0; // Reset this to Zero. Increment for every triangle written

size_t totalWritten = 0;
std::array<float, 3> vecA = {0.0f, 0.0f, 0.0f};
std::array<float, 3> vecB = {0.0f, 0.0f, 0.0f};

std::array<char, 50> data = {};
nonstd::span<float32> normalPtr(reinterpret_cast<float32*>(data.data()), 3);
nonstd::span<float32> vert1Ptr(reinterpret_cast<float32*>(data.data() + 12), 3);
nonstd::span<float32> vert2Ptr(reinterpret_cast<float32*>(data.data() + 24), 3);
nonstd::span<float32> vert3Ptr(reinterpret_cast<float32*>(data.data() + 36), 3);
nonstd::span<uint16> attrByteCountPtr(reinterpret_cast<uint16*>(data.data() + 48), 2);
attrByteCountPtr[0] = 0;

const usize numComps = m_FeatureIds.getNumberOfComponents();
// Loop over all the triangles for this spin
for(IGeometry::MeshIndexType triangle = 0; triangle < m_NumTriangles; ++triangle)
{
// Get the true indices of the 3 nodes
IGeometry::MeshIndexType nId0 = m_Triangles[triangle * 3];
IGeometry::MeshIndexType nId1 = m_Triangles[triangle * 3 + 1];
IGeometry::MeshIndexType nId2 = m_Triangles[triangle * 3 + 2];

if(m_FeatureIds[triangle * numComps] == m_FeatureId)
{
// winding = 0; // 0 = Write it using forward spin
}
else if(numComps > 1 && m_FeatureIds[triangle * numComps + 1] == m_FeatureId)
{
// Switch the 2 node indices
IGeometry::MeshIndexType temp = nId1;
nId1 = nId2;
nId2 = temp;
}
else
{
continue; // We do not match either spin so move to the next triangle
}

vert1Ptr[0] = static_cast<float>(m_Vertices[nId0 * 3]);
vert1Ptr[1] = static_cast<float>(m_Vertices[nId0 * 3 + 1]);
vert1Ptr[2] = static_cast<float>(m_Vertices[nId0 * 3 + 2]);

vert2Ptr[0] = static_cast<float>(m_Vertices[nId1 * 3]);
vert2Ptr[1] = static_cast<float>(m_Vertices[nId1 * 3 + 1]);
vert2Ptr[2] = static_cast<float>(m_Vertices[nId1 * 3 + 2]);

vert3Ptr[0] = static_cast<float>(m_Vertices[nId2 * 3]);
vert3Ptr[1] = static_cast<float>(m_Vertices[nId2 * 3 + 1]);
vert3Ptr[2] = static_cast<float>(m_Vertices[nId2 * 3 + 2]);

// Compute the normal
vecA[0] = vert2Ptr[0] - vert1Ptr[0];
vecA[1] = vert2Ptr[1] - vert1Ptr[1];
vecA[2] = vert2Ptr[2] - vert1Ptr[2];

vecB[0] = vert3Ptr[0] - vert1Ptr[0];
vecB[1] = vert3Ptr[1] - vert1Ptr[1];
vecB[2] = vert3Ptr[2] - vert1Ptr[2];

MatrixMath::CrossProduct(vecA.data(), vecB.data(), normalPtr.data());
MatrixMath::Normalize3x1(normalPtr.data());

totalWritten = fwrite(data.data(), 1, 50, filePtr);
if(totalWritten != 50)
{
fclose(filePtr);
std::cout << fmt::format("Error Writing STL File '{}': Not enough bytes written for triangle {}. Only {} bytes written of 50 bytes", m_Path.filename().string(), triCount, totalWritten)
<< "\n";
break;
// return {MakeWarningVoidResult(
// -27873, fmt::format("Error Writing STL File '{}': Not enough bytes written for triangle {}. Only {} bytes written of 50 bytes", path.filename().string(), triCount,
// totalWritten))};
}
triCount++;
}

fseek(filePtr, 80L, SEEK_SET);
fwrite(reinterpret_cast<char*>(&triCount), 1, 4, filePtr);
fclose(filePtr);
}

private:
const fs::path m_Path;
const IGeometry::MeshIndexType m_NumTriangles;
const std::string m_Header;
const IGeometry::MeshIndexArrayType& m_Triangles;
const Float32Array& m_Vertices;
const Int32Array& m_FeatureIds;
const int32 m_FeatureId;
};
} // namespace

// -----------------------------------------------------------------------------
Expand Down Expand Up @@ -401,6 +552,10 @@ Result<> WriteStlFile::operator()()
std::unordered_set<int32> uniquePartNumbers(partNumbers.cbegin(), partNumbers.cend());
fileList.reserve(uniquePartNumbers.size()); // Reserved enough file names

// The writing of the files can happen in parallel as much as the Operating System will allow
ParallelTaskAlgorithm taskRunner;
taskRunner.setParallelizationEnabled(true);

// Loop over each Part Number and write a file
usize fileIndex = 0;
for(const auto currentPartNumber : uniquePartNumbers)
Expand All @@ -414,16 +569,11 @@ Result<> WriteStlFile::operator()()

m_MessageHandler(IFilter::Message::Type::Info, fmt::format("Writing STL for Part Number {}", currentPartNumber));

auto result = ::MultiWriteOutStl(fileList[fileIndex].value().tempFilePath(), nTriangles, {"DREAM3D Generated For Part Number " + StringUtilities::number(currentPartNumber)}, triangles, vertices,
partNumbers, currentPartNumber);
// If Result is invalid, report an error
if(result.invalid())
{
return result;
}

taskRunner.execute(MultiWriteStlFileImpl(fileList[fileIndex].value().tempFilePath(), nTriangles, {"DREAM3D Generated For Part Number " + StringUtilities::number(currentPartNumber)}, triangles,
vertices, partNumbers, currentPartNumber));
fileIndex++;
}
taskRunner.wait();
}

for(auto& atomicFile : fileList)
Expand Down
7 changes: 4 additions & 3 deletions src/simplnx/Common/AtomicFile.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,7 @@ Result<AtomicFile> AtomicFile::Create(fs::path filename)
atomicFile.m_FilePath = fs::absolute(atomicFile.m_FilePath);
} catch(const std::filesystem::filesystem_error& error)
{
return MakeErrorResult<AtomicFile>(-15780, fmt::format("When attempting to create an absolute path, AtomicFile encountered the following error: '{}'", error.what()));
return MakeErrorResult<AtomicFile>(-15780, fmt::format("AtomicFile Error: When attempting to create an absolute path, AtomicFile encountered the following error: '{}'", error.what()));
}
}

Expand Down Expand Up @@ -95,15 +95,16 @@ Result<> AtomicFile::commit()
{
if(!fs::exists(m_TempFilePath))
{
return MakeErrorResult(-15780, m_TempFilePath.string() + " does not exist");
return MakeErrorResult(-15781, fmt::format("AtomicFile Commit Error: {} does not exist", m_TempFilePath.string()));
}

try
{
fs::rename(m_TempFilePath, m_FilePath);
} catch(const std::filesystem::filesystem_error& error)
{
return MakeErrorResult(-15780, fmt::format("When attempting to move the temp file to the end absolute path, AtomicFile encountered the following error on rename(): '{}'", error.what()));
return MakeErrorResult(
-15782, fmt::format("AtomicFile Commit Error: When attempting to move the temp file to the end absolute path, AtomicFile encountered the following error on rename(): '{}'", error.what()));
}

return {};
Expand Down

0 comments on commit 51a1291

Please sign in to comment.