diff --git a/Applications/FileIO/CMakeLists.txt b/Applications/FileIO/CMakeLists.txt index c83d1009486b8296e85b58f1cc4f84436a5a06c1..91b4dfc6a13b055d0c8f5940b2248b0699f2175e 100644 --- a/Applications/FileIO/CMakeLists.txt +++ b/Applications/FileIO/CMakeLists.txt @@ -6,6 +6,10 @@ if(NOT Shapelib_FOUND) list(REMOVE_ITEM SOURCES SHPInterface.h SHPInterface.cpp) endif() +# GO2OGS +GET_SOURCE_FILES(SOURCES_GO2OGS GocadIO) +set(SOURCES ${SOURCES} ${SOURCES_GO2OGS}) + if(Qt5XmlPatterns_FOUND) APPEND_SOURCE_FILES(SOURCES XmlIO/Qt) APPEND_SOURCE_FILES(SOURCES FEFLOW) diff --git a/Applications/FileIO/GocadIO/CMakeLists.txt b/Applications/FileIO/GocadIO/CMakeLists.txt new file mode 100644 index 0000000000000000000000000000000000000000..7ca90dceddc1db1c45a524b9f51cf6db2fc3ea36 --- /dev/null +++ b/Applications/FileIO/GocadIO/CMakeLists.txt @@ -0,0 +1,6 @@ +include_directories( + ${CMAKE_SOURCE_DIR}/BaseLib + ${CMAKE_SOURCE_DIR}/FileIO + ${CMAKE_SOURCE_DIR}/GeoLib + ${CMAKE_SOURCE_DIR}/MeshLib +) diff --git a/Applications/FileIO/GocadIO/GenerateFaceSetMeshes.cpp b/Applications/FileIO/GocadIO/GenerateFaceSetMeshes.cpp new file mode 100644 index 0000000000000000000000000000000000000000..d0c068710aef03ea084660a6b565a98a340c6c72 --- /dev/null +++ b/Applications/FileIO/GocadIO/GenerateFaceSetMeshes.cpp @@ -0,0 +1,40 @@ +/** + * + * @copyright + * Copyright (c) 2012-2019, OpenGeoSys Community (http://www.opengeosys.org) + * Distributed under a Modified BSD License. + * See accompanying file LICENSE.txt or + * http://www.opengeosys.org/LICENSE.txt + */ + +#include "GenerateFaceSetMeshes.h" +#include "MeshLib/Elements/Quad.h" +#include "MeshLib/IO/writeMeshToFile.h" + +namespace FileIO +{ +namespace Gocad +{ + +void generateFaceSets(GocadSGridReader const& reader, std::string const& path) +{ + for (std::size_t l(0); l < 128; l++) + { + std::unique_ptr<MeshLib::Mesh> face_set(reader.getFaceSetMesh(l)); + + if (!face_set) + { + continue; + } + INFO("Face set mesh created. #nodes: %d, #elements: %d", + face_set->getNumberOfNodes(), + face_set->getNumberOfElements()); + + std::string const mesh_out_fname(path + face_set->getName() + ".vtu"); + INFO("Writing face set mesh to '%s'.", mesh_out_fname.c_str()); + MeshLib::IO::writeMeshToFile(*face_set, mesh_out_fname); + } +} + +} // namespace Gocad +} // namespace FileIO diff --git a/Applications/FileIO/GocadIO/GenerateFaceSetMeshes.h b/Applications/FileIO/GocadIO/GenerateFaceSetMeshes.h new file mode 100644 index 0000000000000000000000000000000000000000..ec197c85661e31a7a8d0158c351341ac783664cf --- /dev/null +++ b/Applications/FileIO/GocadIO/GenerateFaceSetMeshes.h @@ -0,0 +1,27 @@ + +/** + * + * @copyright + * Copyright (c) 2012-2019, OpenGeoSys Community (http://www.opengeosys.org) + * Distributed under a Modified BSD License. + * See accompanying file LICENSE.txt or + * http://www.opengeosys.org/LICENSE.txt + */ + +#pragma once + +#include <string> + +#include "MeshLib/Mesh.h" + +#include "GocadSGridReader.h" + +namespace FileIO +{ +namespace Gocad +{ + +void generateFaceSets(GocadSGridReader const& reader, std::string const& path); + +} // namespace Gocad +} // namespace FileIO diff --git a/Applications/FileIO/GocadIO/GocadSGridReader.cpp b/Applications/FileIO/GocadIO/GocadSGridReader.cpp new file mode 100644 index 0000000000000000000000000000000000000000..008db648a3274d5b54cbcef325d856a3a98773e3 --- /dev/null +++ b/Applications/FileIO/GocadIO/GocadSGridReader.cpp @@ -0,0 +1,856 @@ +/** + * + * @copyright + * Copyright (c) 2012-2019, OpenGeoSys Community (http://www.opengeosys.org) + * Distributed under a Modified BSD License. + * See accompanying file LICENSE.txt or + * http://www.opengeosys.org/LICENSE.txt + */ + +#include "GocadSGridReader.h" + +#include <algorithm> +#include <cstddef> +#include <fstream> +#include <iostream> +#include <sstream> + +#include <boost/tokenizer.hpp> + +#include "MeshLib/Elements/Hex.h" +#include "MeshLib/Elements/Quad.h" +#include "MeshLib/Mesh.h" + +namespace FileIO +{ +namespace Gocad +{ +using Bitset = boost::dynamic_bitset<>; + +GocadSGridReader::GocadSGridReader(std::string const& fname) + : _fname(fname), + _path(BaseLib::extractPath(fname)), + _n_face_sets(0), + _double_precision_binary(false), + _bin_pnts_in_double_precision(false) +{ + // check if file exists + std::ifstream in(_fname.c_str()); + if (!in) + { + ERR("Could not open '%s'.", _fname.c_str()); + in.close(); + return; + } + + bool pnts_read(false); + + // read information in the stratigraphic grid file + std::string line; + while (std::getline(in, line)) + { + if (line.empty()) // skip empty lines + { + continue; + } + if (line.back() == '\r') // check dos line ending + { + line.pop_back(); + } + if (line.substr(0, 8) == "HEADER {") + { + parseHeader(in); + } + if (line.substr(0, 32) == "GOCAD_ORIGINAL_COORDINATE_SYSTEM") + { + _coordinate_system.parse(in); + continue; + } + if (line.substr(0, 7) == "AXIS_N ") + { + parseDims(line); + } + else if (line.substr(0, 12) == "POINTS_FILE ") + { + parseFileName(line, _pnts_fname); + } + else if (line.substr(0, 9) == "PROPERTY ") + { + _property_meta_data_vecs.push_back( + Gocad::parseGocadPropertyMetaData(line, in, _path)); + } + else if (line.substr(0, 35) == "BINARY_POINTS_IN_DOUBLE_PRECISION 1") + { + _bin_pnts_in_double_precision = true; + } + else if (line.substr(0, 11) == "FLAGS_FILE ") + { + parseFileName(line, _flags_fname); + } + else if (line.substr(0, 18) == "REGION_FLAGS_FILE ") + { + parseFileName(line, _region_flags_fname); + } + else if (line.substr(0, 7) == "REGION " || + line.substr(0, 13) == "MODEL_REGION ") + { + regions.push_back(Gocad::parseRegion(line)); + } + else if (line.substr(0, 12) == "MODEL_LAYER ") + { + layers.push_back(Gocad::parseLayer(line, regions)); + } + else if (line.substr(0, 24) == "REGION_FLAGS_BIT_LENGTH ") + { + std::istringstream iss(line); + std::istream_iterator<std::string> it(iss); + it++; + std::size_t bit_length = std::atoi(it->c_str()); + if (regions.size() != bit_length) + { + ERR("%d regions read but %d expected.\n", regions.size(), + bit_length); + throw std::runtime_error( + "Number of read regions differs from expected.\n"); + } + } + else if (line.substr(0, 9) == "FACE_SET ") + { + // first read the points + if (!pnts_read) + { + readNodesBinary(); + pnts_read = true; + } + parseFaceSet(line, in); + } + } + +#ifndef NDEBUG + std::stringstream regions_ss; + regions_ss << regions.size() << " regions read:\n"; + std::copy(regions.begin(), regions.end(), + std::ostream_iterator<Gocad::Region>(regions_ss, "\t")); + DBUG("%s", regions_ss.str().c_str()); + + std::stringstream layers_ss; + layers_ss << layers.size() << " layers read:\n"; + std::copy(layers.begin(), layers.end(), + std::ostream_iterator<Gocad::Layer>(layers_ss, "\n")); + DBUG("%s", layers_ss.str().c_str()); + + std::stringstream properties_ss; + properties_ss << "meta data for " << _property_meta_data_vecs.size() + << " properties read:\n"; + std::copy(_property_meta_data_vecs.begin(), _property_meta_data_vecs.end(), + std::ostream_iterator<Gocad::Property>(properties_ss, "\n")); + DBUG("%s", properties_ss.str().c_str()); +#endif + + // if not done already read the points + if (!pnts_read) + { + readNodesBinary(); + pnts_read = true; + } + readElementPropertiesBinary(); + std::vector<Bitset> region_flags = readRegionFlagsBinary(); + mapRegionFlagsToCellProperties(region_flags); + + readSplitInformation(); + + in.close(); +} + +GocadSGridReader::~GocadSGridReader() +{ + for (auto node : _nodes) + { + delete node; + } + for (auto node : _split_nodes) + { + delete node; + } +} + +std::unique_ptr<MeshLib::Mesh> GocadSGridReader::getMesh() const +{ + std::vector<MeshLib::Node*> nodes; + std::transform(_nodes.cbegin(), _nodes.cend(), std::back_inserter(nodes), + [](MeshLib::Node const* const node) { + return new MeshLib::Node(*node); + }); + + std::vector<MeshLib::Element*> elements(createElements(nodes)); + applySplitInformation(nodes, elements); + + DBUG("Creating mesh from Gocad SGrid."); + std::unique_ptr<MeshLib::Mesh> mesh(new MeshLib::Mesh( + BaseLib::extractBaseNameWithoutExtension(_fname), nodes, elements)); + addGocadPropertiesToMesh(*mesh); + DBUG("Mesh created."); + + return mesh; +} + +void GocadSGridReader::addGocadPropertiesToMesh(MeshLib::Mesh& mesh) const +{ + std::vector<std::string> const& prop_names(getPropertyNames()); + for (auto const& name : prop_names) + { + boost::optional<Gocad::Property const&> prop(getProperty(name)); + if (!prop) + { + continue; + } + + DBUG("Adding Gocad property '%s' with %d values.", name.c_str(), + prop->_property_data.size()); + + auto pv = MeshLib::getOrCreateMeshProperty<double>( + mesh, name, MeshLib::MeshItemType::Cell, 1); + if (pv == nullptr) + { + ERR("Could not create mesh property '%s'.", name.c_str()); + continue; + } + + pv->resize(prop->_property_data.size()); + std::copy(prop->_property_data.cbegin(), prop->_property_data.cend(), + pv->begin()); + } +} + +void GocadSGridReader::parseHeader(std::istream& in) +{ + std::string line; + while (std::getline(in, line)) + { + if (line.front() == '}') + { + return; + } + if (line.substr(0, 27) == "double_precision_binary: on") + { + _double_precision_binary = true; + } + } + if (_double_precision_binary) + { + DBUG( + "GocadSGridReader::parseHeader(): _double_precision_binary == " + "true."); + } +} + +void GocadSGridReader::parseDims(std::string const& line) +{ + std::size_t x_dim(0), y_dim(0), z_dim(0); + boost::tokenizer<> tok(line); + auto it(tok.begin()); + it++; // overread token "AXIS" + it++; // overread "N" + std::stringstream ssx(*(it), + std::stringstream::in | std::stringstream::out); + ssx >> x_dim; + it++; + std::stringstream ssy(*it, std::stringstream::in | std::stringstream::out); + ssy >> y_dim; + it++; + std::stringstream ssz(*it, std::stringstream::in | std::stringstream::out); + ssz >> z_dim; + _index_calculator = Gocad::IndexCalculator(x_dim, y_dim, z_dim); + DBUG("x_dim = %d, y_dim = %d, z_dim = %d => #nodes = %d, #cells = %d", + x_dim, y_dim, z_dim, _index_calculator._n_nodes, + _index_calculator._n_cells); +} + +void GocadSGridReader::parseFileName(std::string const& line, + std::string& result_string) const +{ + boost::char_separator<char> sep(" \r"); + boost::tokenizer<boost::char_separator<char>> tok(line, sep); + auto it(tok.begin()); + ++it; // overread POINTS_FILE or FLAGS_FILE or REGION_FLAGS_FILE + result_string = _path + *it; +} + +/** + * @param line input/output + * @param in input stream containing the face set + */ +void GocadSGridReader::parseFaceSet(std::string& line, std::istream& in) +{ + // create and initialize a Gocad::Property object for storing face set data + Gocad::Property face_set_property; + face_set_property._property_id = _n_face_sets; + face_set_property._property_name = "FaceSet"; + face_set_property._property_class_name = "FaceSetData"; + face_set_property._property_unit = "unitless"; + face_set_property._property_data_type = "double"; + face_set_property._property_data_fname = ""; + face_set_property._property_no_data_value = -1.0; + face_set_property._property_data.resize(_index_calculator._n_cells); + std::fill(face_set_property._property_data.begin(), + face_set_property._property_data.end(), + face_set_property._property_no_data_value); + + std::istringstream iss(line); + std::istream_iterator<std::string> it(iss); + // Check first word is FACE_SET + if (*it != std::string("FACE_SET")) + { + ERR("Expected FACE_SET keyword but '%s' found.", it->c_str()); + throw std::runtime_error( + "In GocadSGridReader::parseFaceSet() expected FACE_SET keyword not " + "found."); + } + ++it; + face_set_property._property_name += *it; + ++it; + std::size_t const n_of_face_set_ids( + static_cast<std::size_t>(std::atoi(it->c_str()))); + std::size_t face_set_id_cnt(0); + + while (std::getline(in, line) && face_set_id_cnt < n_of_face_set_ids) + { + if (line.back() == '\r') + { + line.pop_back(); + } + boost::char_separator<char> sep("\t "); + boost::tokenizer<boost::char_separator<char>> tokens(line, sep); + + for (auto tok_it = tokens.begin(); tok_it != tokens.end();) + { + std::size_t const id( + static_cast<std::size_t>(std::atoi(tok_it->c_str()))); + tok_it++; + std::size_t const face_indicator( + static_cast<std::size_t>(std::atoi(tok_it->c_str()))); + tok_it++; + + if (id >= _index_calculator._n_nodes) + { + ERR("Face set id %d is greater than the number of nodes (%d).", + id, _index_calculator._n_nodes); + } + else + { + static_cast<GocadNode*>(_nodes[id]) + ->setFaceSet(_n_face_sets, face_indicator); + std::array<std::size_t, 3> const c( + _index_calculator.getCoordsForID(id)); + if (c[0] >= _index_calculator._x_dim - 1) + ERR("****** i coord %d to big for id %d.", c[0], id); + if (c[1] >= _index_calculator._y_dim - 1) + ERR("****** j coord %d to big for id %d.", c[1], id); + if (c[2] >= _index_calculator._z_dim - 1) + ERR("****** k coord %d to big for id %d.", c[2], id); + std::size_t const cell_id( + _index_calculator.getCellIdx(c[0], c[1], c[2])); + face_set_property._property_data[cell_id] = face_indicator; + } + face_set_id_cnt++; + } + } + + if (face_set_id_cnt != n_of_face_set_ids) + { + ERR("Expected %d number of face set ids, read %d.", n_of_face_set_ids, + face_set_id_cnt); + throw std::runtime_error( + "Expected number of face set points does not match number of read " + "points."); + } + _n_face_sets++; + + // pre condition: split nodes are read already + for (auto split_node : _split_nodes) + { + std::size_t const id(_index_calculator(split_node->getGridCoords())); + split_node->transmitFaceDirections(*_nodes[id]); + } + + _property_meta_data_vecs.push_back(face_set_property); +} + +// Reads given number of bits (rounded up to next byte) into a bitset. +// Used for reading region information which can be represented by some +// number of bits. +Bitset readBits(std::ifstream& in, const std::size_t bits) +{ + using block_t = Bitset::block_type; + std::size_t const bytes = static_cast<std::size_t>(std::ceil(bits / 8.)); + std::size_t const blocks = + bytes + 1 < sizeof(block_t) ? 1 : (bytes + 1) / sizeof(block_t); + + std::vector<block_t> data; + data.resize(blocks); + std::fill_n(data.data(), blocks, 0); + in.read(reinterpret_cast<char*>(data.data()), bytes); + + return Bitset(data.begin(), data.end()); +} + +void GocadSGridReader::readNodesBinary() +{ + std::ifstream in(_pnts_fname.c_str(), std::ios::binary); + if (!in) + { + ERR("Could not open points file '%s'.", _pnts_fname.c_str()); + throw std::runtime_error("Could not open points file."); + } + + std::size_t const n = _index_calculator._n_nodes; + _nodes.resize(n); + + double coords[3]; + + std::size_t k = 0; + while (in && k < n * 3) + { + if (_bin_pnts_in_double_precision) + { + coords[k % 3] = + BaseLib::swapEndianness(BaseLib::readBinaryValue<double>(in)); + } + else + { + coords[k % 3] = + BaseLib::swapEndianness(BaseLib::readBinaryValue<float>(in)); + } + if ((k + 1) % 3 == 0) + { + const std::size_t layer_transition_idx( + _index_calculator.getCoordsForID(k / 3)[2]); + _nodes[k / 3] = + new GocadNode(coords, k / 3, layer_transition_idx); + } + k++; + } + if (k != n * 3 && !in.eof()) + ERR("Read different number of points. Expected %d floats, got %d.\n", + n * 3, k); +} + +void GocadSGridReader::mapRegionFlagsToCellProperties( + std::vector<Bitset> const& rf) +{ + DBUG( + "GocadSGridReader::mapRegionFlagsToCellProperties region_flags.size: " + "%d", + rf.size()); + + Gocad::Property region_flags; + region_flags._property_id = 0; + region_flags._property_name = "RegionFlags"; + region_flags._property_class_name = "RegionFlags"; + region_flags._property_unit = "unitless"; + region_flags._property_data_type = "int"; + region_flags._property_data_fname = ""; + region_flags._property_no_data_value = -1; + std::size_t const n = _index_calculator._n_cells; + region_flags._property_data.resize(n); + std::fill(region_flags._property_data.begin(), + region_flags._property_data.end(), -1); + + // region flags are stored in each node ijk and give the region index for + // the ijk-th cell. + for (std::size_t i(0); i < _index_calculator._x_dim - 1; i++) + { + for (std::size_t j(0); j < _index_calculator._y_dim - 1; j++) + { + for (std::size_t k(0); k < _index_calculator._z_dim - 1; k++) + { + std::size_t const cell_id( + _index_calculator.getCellIdx(i, j, k)); + std::size_t const node_id(_index_calculator({i, j, k})); + for (auto& region : regions) + { + if (rf[node_id].test(region.bit)) + { + region_flags._property_data[cell_id] += region.bit + 1; + } + } + } + } + } + + _property_meta_data_vecs.push_back(region_flags); +} + +void GocadSGridReader::readElementPropertiesBinary() +{ + for (auto prop_it(_property_meta_data_vecs.begin()); + prop_it != _property_meta_data_vecs.end(); + prop_it++) + { + std::string const& fname(prop_it->_property_data_fname); + if (prop_it->_property_data_fname.empty()) + { + WARN("Empty filename for property %s.", + prop_it->_property_name.c_str()); + continue; + } + std::vector<float> float_properties = + BaseLib::readBinaryArray<float>(fname, _index_calculator._n_cells); + DBUG( + "GocadSGridReader::readElementPropertiesBinary(): Read %d float " + "properties from binary file.", + _index_calculator._n_cells); + + std::transform(float_properties.cbegin(), float_properties.cend(), + float_properties.begin(), [](float const& val) { + return BaseLib::swapEndianness(val); + }); + + prop_it->_property_data.resize(float_properties.size()); + std::copy(float_properties.begin(), float_properties.end(), + prop_it->_property_data.begin()); + if (prop_it->_property_data.empty()) + { + ERR("Reading of element properties file '%s' failed.", + fname.c_str()); + } + } +} + +std::vector<int> GocadSGridReader::readFlagsBinary() const +{ + std::vector<int> result; + if (!_double_precision_binary) + { + result = BaseLib::readBinaryArray<int32_t>(_flags_fname, + _index_calculator._n_nodes); + std::for_each(result.begin(), result.end(), + [](int32_t& val) { BaseLib::swapEndianness(val); }); + } + else + { + result = BaseLib::readBinaryArray<int>(_flags_fname, + _index_calculator._n_nodes); + std::for_each(result.begin(), result.end(), + [](int& val) { BaseLib::swapEndianness(val); }); + } + + if (result.empty()) + ERR("Reading of flags file '%s' failed.", _flags_fname.c_str()); + + return result; +} + +std::vector<Bitset> GocadSGridReader::readRegionFlagsBinary() const +{ + std::vector<Bitset> result; + + std::ifstream in(_region_flags_fname.c_str()); + if (!in) + { + ERR("readRegionFlagsBinary(): Could not open file '%s' for input.\n", + _region_flags_fname.c_str()); + in.close(); + return result; + } + + std::size_t const n = _index_calculator._n_nodes; + result.resize(n); + + std::size_t k = 0; + while (in && k < n) + { + result[k++] = readBits(in, regions.size()); + } + if (k != n && !in.eof()) + ERR("Read different number of values. Expected %d, got %d.\n", n, k); + + return result; +} +std::vector<MeshLib::Element*> GocadSGridReader::createElements( + std::vector<MeshLib::Node*> const& nodes) const +{ + std::vector<MeshLib::Element*> elements; + elements.resize(_index_calculator._n_cells); + std::array<MeshLib::Node*, 8> element_nodes{}; + std::size_t cnt(0); + for (std::size_t k(0); k < _index_calculator._z_dim - 1; k++) + { + for (std::size_t j(0); j < _index_calculator._y_dim - 1; j++) + { + for (std::size_t i(0); i < _index_calculator._x_dim - 1; i++) + { + element_nodes[0] = nodes[_index_calculator({i, j, k})]; + element_nodes[1] = nodes[_index_calculator({i + 1, j, k})]; + element_nodes[2] = nodes[_index_calculator({i + 1, j + 1, k})]; + element_nodes[3] = nodes[_index_calculator({i, j + 1, k})]; + element_nodes[4] = nodes[_index_calculator({i, j, k + 1})]; + element_nodes[5] = nodes[_index_calculator({i + 1, j, k + 1})]; + element_nodes[6] = + nodes[_index_calculator({i + 1, j + 1, k + 1})]; + element_nodes[7] = nodes[_index_calculator({i, j + 1, k + 1})]; + elements[cnt] = new MeshLib::Hex( + element_nodes, _index_calculator.getCellIdx(i, j, k)); + cnt++; + } + } + } + return elements; +} + +void GocadSGridReader::readSplitInformation() +{ + std::ifstream in(_fname.c_str()); + if (!in) + { + ERR("Could not open '%s'.", _fname.c_str()); + in.close(); + return; + } + + // read split information from the stratigraphic grid file + std::string line; + std::stringstream ss; + while (std::getline(in, line)) + { + std::size_t pos(line.find("SPLIT ")); + if (pos != std::string::npos) + { + ss << line.substr(pos + 6, line.size() - (pos + 6)); + // read position in grid + std::array<std::size_t, 3> grid_coords{}; + ss >> grid_coords[0]; + ss >> grid_coords[1]; + ss >> grid_coords[2]; + // read coordinates for the split node + double coords[3]; + ss >> coords[0]; + ss >> coords[1]; + ss >> coords[2]; + // read the id + std::size_t id; + ss >> id; + // read the affected cells + std::bitset<8> affected_cells{}; + for (std::size_t ac = 0; ac < 8; ++ac) + { + char bit; + ss >> bit; + affected_cells[ac] = bit == '0' ? false : true; + } + const std::size_t layer_transition_index( + _nodes[id]->getLayerTransitionIndex()); + _split_nodes.push_back(new GocadSplitNode(coords, id, grid_coords, + affected_cells, + layer_transition_index)); + } + } +} + +void GocadSGridReader::applySplitInformation( + std::vector<MeshLib::Node*>& nodes, + std::vector<MeshLib::Element*>& elements) const +{ + for (auto split_node : _split_nodes) + { + std::size_t const new_node_pos(nodes.size()); + nodes.push_back( + new MeshLib::Node(split_node->getCoords(), new_node_pos)); + + // get grid coordinates + std::array<std::size_t, 3> const& gc(split_node->getGridCoords()); + // get affected cells + auto const& affected_cells(split_node->getAffectedCells()); + // get mesh node to substitute in elements + MeshLib::Node const* const node2sub(nodes[_index_calculator(gc)]); + + if (affected_cells[0] && gc[0] < _index_calculator._x_dim - 1 && + gc[1] < _index_calculator._y_dim - 1 && + gc[2] < _index_calculator._z_dim - 1) + { + const std::size_t idx( + _index_calculator.getCellIdx(gc[0], gc[1], gc[2])); + modifyElement(elements[idx], node2sub, nodes[new_node_pos]); + } + if (affected_cells[1] && gc[0] > 0 && + gc[1] < _index_calculator._y_dim - 1 && + gc[2] < _index_calculator._z_dim - 1) + { + const std::size_t idx( + _index_calculator.getCellIdx(gc[0] - 1, gc[1], gc[2])); + modifyElement(elements[idx], node2sub, nodes[new_node_pos]); + } + if (affected_cells[2] && gc[1] > 0 && + gc[0] < _index_calculator._x_dim - 1 && + gc[2] < _index_calculator._z_dim - 1) + { + const std::size_t idx( + _index_calculator.getCellIdx(gc[0], gc[1] - 1, gc[2])); + modifyElement(elements[idx], node2sub, nodes[new_node_pos]); + } + if (affected_cells[3] && gc[0] > 0 && gc[1] > 0 && + gc[2] < _index_calculator._z_dim - 1) + { + const std::size_t idx( + _index_calculator.getCellIdx(gc[0] - 1, gc[1] - 1, gc[2])); + modifyElement(elements[idx], node2sub, nodes[new_node_pos]); + } + if (affected_cells[4] && gc[2] > 0 && + gc[0] < _index_calculator._x_dim - 1 && + gc[1] < _index_calculator._y_dim - 1) + { + const std::size_t idx( + _index_calculator.getCellIdx(gc[0], gc[1], gc[2] - 1)); + modifyElement(elements[idx], node2sub, nodes[new_node_pos]); + } + if (affected_cells[5] && gc[0] > 0 && gc[2] > 0 && + gc[1] < _index_calculator._y_dim - 1) + { + const std::size_t idx( + _index_calculator.getCellIdx(gc[0] - 1, gc[1], gc[2] - 1)); + modifyElement(elements[idx], node2sub, nodes[new_node_pos]); + } + if (affected_cells[6] && gc[1] > 0 && gc[2] > 0 && + gc[0] < _index_calculator._x_dim - 1) + { + const std::size_t idx( + _index_calculator.getCellIdx(gc[0], gc[1] - 1, gc[2] - 1)); + modifyElement(elements[idx], node2sub, nodes[new_node_pos]); + } + if (affected_cells[7] && gc[0] > 0 && gc[1] > 0 && gc[2] > 0) + { + const std::size_t idx( + _index_calculator.getCellIdx(gc[0] - 1, gc[1] - 1, gc[2] - 1)); + modifyElement(elements[idx], node2sub, nodes[new_node_pos]); + } + } +} + +void GocadSGridReader::modifyElement(MeshLib::Element* hex, + MeshLib::Node const* node2sub, + MeshLib::Node* substitute_node) const +{ + // get the node pointers of the cell + MeshLib::Node* const* hex_nodes(hex->getNodes()); + // search for the position the split node will be set to + MeshLib::Node* const* node_pos( + std::find(hex_nodes, hex_nodes + 8, node2sub)); + // set the split node instead of the node2sub + if (node_pos != hex_nodes + 8) + { + const_cast<MeshLib::Node**>( + hex_nodes)[std::distance(hex_nodes, node_pos)] = substitute_node; + } +} + +boost::optional<Gocad::Property const&> GocadSGridReader::getProperty( + std::string const& name) const +{ + auto const it(std::find_if(_property_meta_data_vecs.begin(), + _property_meta_data_vecs.end(), + [&name](Gocad::Property const& p) { + return p._property_name == name; + })); + if (it == _property_meta_data_vecs.end()) + { + return boost::optional<Gocad::Property const&>(); + } + return boost::optional<Gocad::Property const&>(*it); +} + +std::vector<std::string> GocadSGridReader::getPropertyNames() const +{ + std::vector<std::string> names; + std::transform(_property_meta_data_vecs.begin(), + _property_meta_data_vecs.end(), + std::back_inserter(names), + [](Gocad::Property const& p) { return p._property_name; }); + return names; +} + +std::unique_ptr<MeshLib::Mesh> GocadSGridReader::getFaceSetMesh( + std::size_t const face_set_number) const +{ + std::vector<MeshLib::Node*> face_set_nodes; + std::vector<MeshLib::Element*> face_set_elements; + + for (auto const node : _nodes) + { + if (node->isMemberOfFaceSet(face_set_number)) + { + addFaceSetQuad(node, face_set_number, face_set_nodes, + face_set_elements); + } + } + + if (face_set_nodes.empty()) + { + return nullptr; + } + + for (auto const node : _split_nodes) + { + if (node->isMemberOfFaceSet(face_set_number)) + { + if (node->getAffectedCells()[0]) + { + addFaceSetQuad(node, face_set_number, face_set_nodes, + face_set_elements); + } + } + } + + std::string const mesh_name("FaceSet-" + std::to_string(face_set_number)); + return std::make_unique<MeshLib::Mesh>(mesh_name, face_set_nodes, + face_set_elements); +} + +void GocadSGridReader::addFaceSetQuad( + GocadNode* face_set_node, std::size_t face_set_number, + std::vector<MeshLib::Node*>& face_set_nodes, + std::vector<MeshLib::Element*>& face_set_elements) const +{ + std::array<MeshLib::Node*, 4> quad_nodes{}; + quad_nodes[0] = new GocadNode(*face_set_node); + const std::size_t id(face_set_node->getID()); + std::array<std::size_t, 3> const c(_index_calculator.getCoordsForID(id)); + + const FaceDirection dir(face_set_node->getFaceDirection(face_set_number)); + switch (dir) + { + case FaceDirection::U: + quad_nodes[1] = new GocadNode( + *_nodes[_index_calculator({c[0], c[1] + 1, c[2]})]); + quad_nodes[2] = new GocadNode( + *_nodes[_index_calculator({c[0], c[1] + 1, c[2] + 1})]); + quad_nodes[3] = new GocadNode( + *_nodes[_index_calculator({c[0], c[1], c[2] + 1})]); + break; + case FaceDirection::V: + quad_nodes[1] = new GocadNode( + *_nodes[_index_calculator({c[0] + 1, c[1], c[2]})]); + quad_nodes[2] = new GocadNode( + *_nodes[_index_calculator({c[0] + 1, c[1], c[2] + 1})]); + quad_nodes[3] = new GocadNode( + *_nodes[_index_calculator({c[0], c[1], c[2] + 1})]); + break; + case FaceDirection::W: + quad_nodes[1] = new GocadNode( + *_nodes[_index_calculator({c[0] + 1, c[1], c[2]})]); + quad_nodes[2] = new GocadNode( + *_nodes[_index_calculator({c[0] + 1, c[1] + 1, c[2]})]); + quad_nodes[3] = new GocadNode( + *_nodes[_index_calculator({c[0], c[1] + 1, c[2]})]); + break; + default: + ERR("Could not create face for node with id %d.", id); + } + for (auto quad_node : quad_nodes) + { + face_set_nodes.push_back(quad_node); + } + face_set_elements.push_back(new MeshLib::Quad(quad_nodes)); +} + +} // end namespace Gocad +} // end namespace FileIO diff --git a/Applications/FileIO/GocadIO/GocadSGridReader.h b/Applications/FileIO/GocadIO/GocadSGridReader.h new file mode 100644 index 0000000000000000000000000000000000000000..1c21b8a84cac782bf0d154837b871c942cbb51e6 --- /dev/null +++ b/Applications/FileIO/GocadIO/GocadSGridReader.h @@ -0,0 +1,115 @@ +/** + * + * @copyright + * Copyright (c) 2012-2019, OpenGeoSys Community (http://www.opengeosys.org) + * Distributed under a Modified BSD License. + * See accompanying file LICENSE.txt or + * http://www.opengeosys.org/LICENSE.txt + */ + +#pragma once + +#include <cstddef> +#include <limits> +#include <numeric> +#include <string> +#include <vector> + +#include <boost/dynamic_bitset.hpp> + +#include <logog/include/logog.hpp> + +#include "MeshLib/Elements/Element.h" + +#include "CoordinateSystem.h" +#include "GocadNode.h" +#include "IndexCalculator.h" +#include "Layer.h" +#include "Property.h" +#include "Region.h" + +namespace FileIO +{ +namespace Gocad +{ +class GocadSGridReader final +{ +public: + /** + * Constructor takes as argument the Gocad .sg text file. + * @param fname file name + */ + explicit GocadSGridReader(std::string const& fname); + ~GocadSGridReader(); + + GocadSGridReader() = delete; + GocadSGridReader(GocadSGridReader&& src) = delete; + GocadSGridReader(GocadSGridReader const& src) = delete; + GocadSGridReader& operator=(GocadSGridReader&& rhs) = delete; + GocadSGridReader& operator=(GocadSGridReader const& rhs) = delete; + + std::unique_ptr<MeshLib::Mesh> getMesh() const; + std::unique_ptr<MeshLib::Mesh> getFaceSetMesh( + std::size_t const face_set_number) const; + + std::vector<std::string> getPropertyNames() const; + +private: + using Bitset = boost::dynamic_bitset<>; + + void parseDims(std::string const& line); + void parseFileName(std::string const& line, + std::string& result_string) const; + void parseHeader(std::istream& in); + void parseFaceSet(std::string& line, std::istream& in); + + void readNodesBinary(); + std::vector<int> readFlagsBinary() const; + std::vector<Bitset> readRegionFlagsBinary() const; + void readElementPropertiesBinary(); + void mapRegionFlagsToCellProperties(std::vector<Bitset> const& rf); + + std::vector<MeshLib::Element*> createElements( + std::vector<MeshLib::Node*> const& nodes) const; + + // split handling + void readSplitInformation(); + void applySplitInformation(std::vector<MeshLib::Node*>& nodes, + std::vector<MeshLib::Element*>& elements) const; + void modifyElement(MeshLib::Element* hex, MeshLib::Node const* node2sub, + MeshLib::Node* substitute_node) const; + + void addFaceSetQuad( + GocadNode* face_set_node, std::size_t face_set_number, + std::vector<MeshLib::Node*>& face_set_nodes, + std::vector<MeshLib::Element*>& face_set_elements) const; + + boost::optional<Gocad::Property const&> getProperty( + std::string const& name) const; + void addGocadPropertiesToMesh(MeshLib::Mesh& mesh) const; + + std::string const& _fname; + std::string const _path; + // data read from sg file + Gocad::IndexCalculator _index_calculator; + Gocad::CoordinateSystem _coordinate_system; + std::string _pnts_fname; + std::string _flags_fname; + std::string _region_flags_fname; + + std::vector<Gocad::Region> regions; + std::vector<Gocad::Layer> layers; + std::size_t _n_face_sets; + + bool _double_precision_binary; + bool _bin_pnts_in_double_precision; + + // data read from binary points file + std::vector<GocadNode*> _nodes; + std::vector<GocadSplitNode*> _split_nodes; + // properties + std::vector<Gocad::Property> _property_meta_data_vecs; +}; // end class GocadSGridReader + +} // end namespace Gocad +} // end namespace FileIO diff --git a/Applications/Utils/FileConverter/CMakeLists.txt b/Applications/Utils/FileConverter/CMakeLists.txt index 34f7437ba3ba830dd7489b917c11213b238f5306..8db49249a1b3908de206a4fd0417e26890d23e8e 100644 --- a/Applications/Utils/FileConverter/CMakeLists.txt +++ b/Applications/Utils/FileConverter/CMakeLists.txt @@ -43,11 +43,19 @@ add_executable(TecPlotTools TecPlotTools.cpp) set_target_properties(TecPlotTools PROPERTIES FOLDER Utilities) target_link_libraries(TecPlotTools GeoLib MeshLib) +add_executable(GocadSGridReader GocadSGridReaderMain.cpp) +set_target_properties(GocadSGridReader PROPERTIES FOLDER Utilities) +target_link_libraries(GocadSGridReader + GeoLib + MeshLib + ApplicationsFileIO + ${Boost_LIBRARIES} +) #################### ### Installation ### #################### -install(TARGETS generateMatPropsFromMatID GMSH2OGS OGS2VTK VTK2OGS VTK2TIN +install(TARGETS generateMatPropsFromMatID GMSH2OGS OGS2VTK VTK2OGS VTK2TIN GocadSGridReader RUNTIME DESTINATION bin COMPONENT ogs_converter) if(Qt5XmlPatterns_FOUND) diff --git a/Applications/Utils/FileConverter/GocadSGridReaderMain.cpp b/Applications/Utils/FileConverter/GocadSGridReaderMain.cpp new file mode 100644 index 0000000000000000000000000000000000000000..3fbf80f92eae00354097f15a950f33e31cbadf59 --- /dev/null +++ b/Applications/Utils/FileConverter/GocadSGridReaderMain.cpp @@ -0,0 +1,95 @@ +/** + * + * @copyright + * Copyright (c) 2012-2019, OpenGeoSys Community (http://www.opengeosys.org) + * Distributed under a Modified BSD License. + * See accompanying file LICENSE.txt or + * http://www.opengeosys.org/LICENSE.txt + */ + +#include <fstream> +#include <sstream> +#include <string> + +#include <logog/include/logog.hpp> +#include <tclap/CmdLine.h> + +#include "Applications/ApplicationsLib/LogogSetup.h" +#include "Applications/FileIO/GocadIO/GenerateFaceSetMeshes.h" +#include "Applications/FileIO/GocadIO/GocadSGridReader.h" +#include "BaseLib/BuildInfo.h" +#include "BaseLib/FileTools.h" +#include "MeshLib/Elements/Element.h" +#include "MeshLib/IO/writeMeshToFile.h" +#include "MeshLib/Mesh.h" +#include "MeshLib/Node.h" + +int main(int argc, char* argv[]) +{ + ApplicationsLib::LogogSetup logog_setup; + + TCLAP::CmdLine cmd( + "Reads a Gocad stratigraphic grid file (file ending sg) and writes the " + "data in the vtk unstructured grid file format.\n\n OpenGeoSys-6 " + "software, version " + + BaseLib::BuildInfo::git_describe + + ".\n" + "Copyright (c) 2012-2019, OpenGeoSys Community " + "(http://www.opengeosys.org)", + ' ', BaseLib::BuildInfo::git_describe); + + TCLAP::ValueArg<bool> face_set_arg( + "f", "generate-face-sets", + "generate face sets; default false, i.e. do not generate face sets", + false, false, "true/false"); + cmd.add(face_set_arg); + + TCLAP::ValueArg<std::string> mesh_output_arg( + "o", "output-mesh", "vtk unstructured grid file name", true, "", + "file_name.vtu"); + cmd.add(mesh_output_arg); + + TCLAP::ValueArg<std::string> sg_file_arg( + "s", "sg", "Gocad stratigraphic grid file name", true, "", + "file_name.sg"); + cmd.add(sg_file_arg); + + cmd.parse(argc, argv); + + // read the Gocad SGrid + INFO("Start reading Gocad SGrid."); + FileIO::Gocad::GocadSGridReader reader(sg_file_arg.getValue()); + INFO("End reading Gocad SGrid."); + + if (face_set_arg.getValue()) + { + INFO("Generating a mesh for every face set."); + FileIO::Gocad::generateFaceSets( + reader, BaseLib::extractPath(sg_file_arg.getValue())); + } + + std::unique_ptr<MeshLib::Mesh> mesh(reader.getMesh()); + + INFO("The mesh contains the following PropertyVectors:"); + auto property_names(mesh->getProperties().getPropertyVectorNames()); + for (auto const& property_name : property_names) + { + INFO("- %s (#values: %d)", property_name.c_str(), + mesh->getProperties() + .getPropertyVector<double>(property_name) + ->size()); + auto bounds( + std::minmax_element(mesh->getProperties() + .getPropertyVector<double>(property_name) + ->cbegin(), + mesh->getProperties() + .getPropertyVector<double>(property_name) + ->cend())); + INFO("\tvalues in range [%e, %e].", *(bounds.first), *(bounds.second)); + } + + INFO("Writing mesh to '%s'.", mesh_output_arg.getValue().c_str()); + MeshLib::IO::writeMeshToFile(*mesh, mesh_output_arg.getValue()); + + return 0; +}