Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add xdmffile read function #3536

Open
wants to merge 23 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
23 commits
Select commit Hold shift + click to select a range
35a65b4
Safe default value for unbound variables in dolfinx.conf
mleoni-pf Oct 14, 2024
dfca9ce
Merge branch 'FEniCS:main' into main
mleoni-pf Nov 21, 2024
8de82a7
Merge branch 'main' of github.com:FEniCS/dolfinx
mleoni-pf Nov 22, 2024
a095f16
Merge branch 'FEniCS:main' into main
mleoni-pf Nov 24, 2024
f892d40
Merge branch 'main' of github.com:FEniCS/dolfinx
mleoni-pf Nov 25, 2024
2700ed7
Make read_meshtags a template function
mleoni-pf Nov 25, 2024
c53d5d3
Add XDMFFile::read_function to read P1 functions
mleoni-pf Nov 25, 2024
20924b1
Merge branch 'main' into mleoni/make_read_meshtags_a_template_function
mleoni-pf Nov 25, 2024
9d89990
Merge branch 'mleoni/make_read_meshtags_a_template_function' into mle…
mleoni-pf Nov 25, 2024
8d7c60c
Fix signature of explicit template instantiation
mleoni-pf Nov 25, 2024
0c5c541
Merge branch 'main' into mleoni/make_read_meshtags_a_template_function
mleoni-pf Nov 25, 2024
85daf1f
Add XDMFFile::read_function to read P1 functions
mleoni-pf Nov 25, 2024
6b8dfd7
Merge branch 'mleoni/add_xdmffile_read_function' of github.com:mleoni…
mleoni-pf Nov 25, 2024
38c82f7
Improve comments
mleoni-pf Nov 25, 2024
c453138
Fix loop variable type
mleoni-pf Nov 25, 2024
147ae03
Fix docstring
mleoni-pf Nov 25, 2024
08e3332
Fix docstring
mleoni-pf Nov 25, 2024
c300e3e
Fix random typo
mleoni-pf Nov 26, 2024
3284905
Add optional function name to read
mleoni-pf Nov 26, 2024
745161a
Merge branch 'main' into mleoni/add_xdmffile_read_function
mleoni-pf Nov 26, 2024
6c2d344
Merge branch 'main' into mleoni/add_xdmffile_read_function
mleoni-pf Nov 28, 2024
3e50555
Merge branch 'main' into mleoni/add_xdmffile_read_function
mleoni-pf Dec 2, 2024
bd1fc79
Merge branch 'main' into mleoni/add_xdmffile_read_function
mleoni-pf Jan 7, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion cpp/dolfinx/fem/assembler.h
Original file line number Diff line number Diff line change
Expand Up @@ -309,7 +309,7 @@ void assemble_matrix(
/// @param[in] mat_add The function for adding values into the matrix
/// @param[in] a The bilinear from to assemble
/// @param[in] bcs Boundary conditions to apply. For boundary condition
/// dofs the row and column are zeroed. The diagonal entry is not set.
/// dofs the row and column are zeroed. The diagonal entry is not set.
template <dolfinx::scalar T, std::floating_point U>
void assemble_matrix(
auto mat_add, const Form<T, U>& a,
Expand Down
163 changes: 157 additions & 6 deletions cpp/dolfinx/io/XDMFFile.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -297,6 +297,144 @@ XDMFFile::write_function(const fem::Function<std::complex<double>, double>&,
double, std::string);
/// @endcond
//-----------------------------------------------------------------------------
void XDMFFile::read_function(const mesh::Mesh<double>& mesh, std::string name,
fem::Function<double, double>& u,
std::optional<std::string> function_name,
std::string xpath)
{
/*
* This routine reads a function from file. The implementation closely
* follows `XDMFFile::read_meshtags` below.
*
* For reference, we are reading an xdmf file whose header is akin to
*
* <Xdmf Version="3.0">
* <Domain>
* <Grid Name="Grid">
* <Geometry GeometryType="XYZ">
* <DataItem DataType="Float" Dimensions="23103 3" Format="HDF"
* Precision="4"> power.h5:/data0</DataItem>
* </Geometry>
* <Topology TopologyType="Hexahedron" NumberOfElements="15000"
* NodesPerElement="8"> <DataItem DataType="Int" Dimensions="15000 8"
* Format="HDF" Precision="8"> power.h5:/data1</DataItem>
* </Topology>
* <Attribute Name="wall power density" AttributeType="Scalar"
* Center="Node"> <DataItem DataType="Float" Dimensions="23103" Format="HDF"
* Precision="8"> power.h5:/data2</DataItem>
* </Attribute>
* </Grid>
* </Domain>
* </Xdmf>
*
* and that was generated saving a vtu file from Paraview and converting it
* to xdmf with meshio.
*
* The goal for now is to read a P1 function, so the degrees of freedom are
* the vertexes of the mesh.
*/
spdlog::info("XDMF read function ({})", name);
pugi::xml_node node = _xml_doc->select_node(xpath.c_str()).node();
if (!node)
throw std::runtime_error("XML node '" + xpath + "' not found.");
pugi::xml_node grid_node
= node.select_node(("Grid[@Name='" + name + "']").c_str()).node();
if (!grid_node)
throw std::runtime_error("<Grid> with name '" + name + "' not found.");

pugi::xml_node values_data_node
= grid_node.child("Attribute").child("DataItem");
if (function_name)
{
// Search for a child that contains an attribute with the requested name
pugi::xml_node function_node = grid_node.find_child(
[fun_name = *function_name](auto n)
{ return n.attribute("Name").value() == fun_name; });
if (!function_node)
{
throw std::runtime_error("Function with name '" + *function_name
+ "' not found.");
}
else
values_data_node = function_node.child("DataItem");
}
const std::vector values
= xdmf_utils::get_dataset<double>(_comm.comm(), values_data_node, _h5_id);

/*
* Reading the cell type would read "hexahedron", so we set this
* manually to "point" instead.
*/
mesh::CellType cell_type = mesh::CellType::point;

/*
* The `entities1` vector contains the global indexes of the entities
* [aka points] that this process owns.
*/
std::int64_t num_xnodes = mesh.geometry().index_map()->size_global();
auto range
= dolfinx::MPI::local_range(dolfinx::MPI::rank(mesh.comm()), num_xnodes,
dolfinx::MPI::size(mesh.comm()));
std::vector<std::int64_t> entities1(range[1] - range[0]);
std::iota(entities1.begin(), entities1.end(), range[0]);

std::array<std::size_t, 2> shape
= {static_cast<std::size_t>(range[1] - range[0]), 1};

MDSPAN_IMPL_STANDARD_NAMESPACE::mdspan<
const std::int64_t,
MDSPAN_IMPL_STANDARD_NAMESPACE::dextents<std::size_t, 2>>
entities_span(entities1.data(), shape);

std::pair<std::vector<std::int32_t>, std::vector<double>> entities_values
= xdmf_utils::distribute_entity_data<double>(
*mesh.topology(), mesh.geometry().input_global_indices(),
mesh.geometry().index_map()->size_global(),
mesh.geometry().cmap().create_dof_layout(), mesh.geometry().dofmap(),
mesh::cell_dim(cell_type), entities_span, values);

auto num_vertices_per_cell
= dolfinx::mesh::num_cell_vertices(mesh.topology()->cell_type());
std::vector<std::int32_t> local_vertex_map(num_vertices_per_cell);

for (int i = 0; i < num_vertices_per_cell; ++i)
{
const auto v_to_d
= u.function_space()->dofmap()->element_dof_layout().entity_dofs(0, i);
assert(v_to_d.size() == 1);
local_vertex_map[i] = v_to_d.front();
}

const auto tdim = mesh.topology()->dim();
const auto c_to_v = mesh.topology()->connectivity(tdim, 0);
std::vector<std::int32_t> vertex_to_dofmap(
mesh.topology()->index_map(0)->size_local()
+ mesh.topology()->index_map(0)->num_ghosts());

for (int i = 0; i < mesh.topology()->index_map(tdim)->size_local(); ++i)
{
const auto local_vertices = c_to_v->links(i);
const auto local_dofs = u.function_space()->dofmap()->cell_dofs(i);
for (int j = 0; j < num_vertices_per_cell; ++j)
{
vertex_to_dofmap[local_vertices[j]] = local_dofs[local_vertex_map[j]];
}
}

/*
* After the data is read and distributed, we need to place the
* retrieved values in the correct position in the function's array,
* reading values and positions from `entities_values`.
*/
for (size_t i = 0; i < entities_values.first.size(); ++i)
{
u.x()->mutable_array()[vertex_to_dofmap[entities_values.first[i]]]
= entities_values.second[i];
}

u.x()->scatter_fwd();
}
//-----------------------------------------------------------------------------
template <std::floating_point T>
void XDMFFile::write_meshtags(const mesh::MeshTags<std::int32_t>& meshtags,
const mesh::Geometry<T>& x,
Expand Down Expand Up @@ -333,7 +471,8 @@ template void XDMFFile::write_meshtags(const mesh::MeshTags<std::int32_t>&,
std::string, std::string);
/// @endcond
//-----------------------------------------------------------------------------
mesh::MeshTags<std::int32_t>
template <typename T>
mesh::MeshTags<T>
XDMFFile::read_meshtags(const mesh::Mesh<double>& mesh, std::string name,
std::optional<std::string> attribute_name,
std::string xpath)
Expand Down Expand Up @@ -365,8 +504,8 @@ XDMFFile::read_meshtags(const mesh::Mesh<double>& mesh, std::string name,
else
values_data_node = attribute_node.child("DataItem");
}
const std::vector values = xdmf_utils::get_dataset<std::int32_t>(
_comm.comm(), values_data_node, _h5_id);
const std::vector values
= xdmf_utils::get_dataset<T>(_comm.comm(), values_data_node, _h5_id);

const std::pair<std::string, int> cell_type_str
= xdmf_utils::get_cell_type(grid_node.child("Topology"));
Expand All @@ -380,8 +519,8 @@ XDMFFile::read_meshtags(const mesh::Mesh<double>& mesh, std::string name,
const std::int64_t,
MDSPAN_IMPL_STANDARD_NAMESPACE::dextents<std::size_t, 2>>
entities_span(entities1.data(), eshape);
std::pair<std::vector<std::int32_t>, std::vector<std::int32_t>>
entities_values = xdmf_utils::distribute_entity_data<std::int32_t>(
std::pair<std::vector<std::int32_t>, std::vector<T>> entities_values
= xdmf_utils::distribute_entity_data<T>(
*mesh.topology(), mesh.geometry().input_global_indices(),
mesh.geometry().index_map()->size_global(),
mesh.geometry().cmap().create_dof_layout(), mesh.geometry().dofmap(),
Expand All @@ -397,12 +536,24 @@ XDMFFile::read_meshtags(const mesh::Mesh<double>& mesh, std::string name,
num_vertices_per_entity);
mesh::MeshTags meshtags = mesh::create_meshtags(
mesh.topology(), mesh::cell_dim(cell_type), entities_adj,
std::span<const std::int32_t>(entities_values.second));
std::span<const T>(entities_values.second));
meshtags.name = name;

return meshtags;
}
//-----------------------------------------------------------------------------
// Instantiation for different types
/// @cond
template mesh::MeshTags<std::int32_t>
XDMFFile::read_meshtags(const mesh::Mesh<double>& mesh, std::string name,
std::optional<std::string> attribute_name,
std::string xpath);
template mesh::MeshTags<double>
XDMFFile::read_meshtags(const mesh::Mesh<double>& mesh, std::string name,
std::optional<std::string> attribute_name,
std::string xpath);
/// @endcond
//-----------------------------------------------------------------------------
std::pair<mesh::CellType, int> XDMFFile::read_cell_type(std::string grid_name,
std::string xpath)
{
Expand Down
19 changes: 15 additions & 4 deletions cpp/dolfinx/io/XDMFFile.h
Original file line number Diff line number Diff line change
Expand Up @@ -152,6 +152,16 @@ class XDMFFile
std::string mesh_xpath
= "/Xdmf/Domain/Grid[@GridType='Uniform'][1]");

/// Read Function
/// @param[in] mesh The Mesh that the data is defined on
/// @param[in] name
/// @param[out] u The function into which to read data
/// @param[in] function_name The (optinal) name of the function to read from file
/// @param[in] xpath XPath where MeshFunction Grid is stored in file
void read_function(const mesh::Mesh<double>& mesh, std::string name,
fem::Function<double, double>& u, std::optional<std::string> function_name,
std::string xpath = "/Xdmf/Domain");

/// Write MeshTags
/// @param[in] meshtags
/// @param[in] x Mesh geometry
Expand All @@ -170,10 +180,11 @@ class XDMFFile
/// GridType="Uniform"
/// @param[in] attribute_name Name of the attribute to read
/// @param[in] xpath XPath where MeshTags Grid is stored in file
mesh::MeshTags<std::int32_t>
read_meshtags(const mesh::Mesh<double>& mesh, std::string name,
std::optional<std::string> attribute_name,
std::string xpath = "/Xdmf/Domain");
template <typename T = std::int32_t>
mesh::MeshTags<T> read_meshtags(const mesh::Mesh<double>& mesh,
std::string name,
std::optional<std::string> attribute_name,
std::string xpath = "/Xdmf/Domain");

/// Write Information
/// @param[in] name
Expand Down
2 changes: 1 addition & 1 deletion python/dolfinx/wrappers/io.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -283,7 +283,7 @@ void io(nb::module_& m)
nb::arg("name") = "mesh", nb::arg("xpath") = "/Xdmf/Domain")
.def("read_cell_type", &dolfinx::io::XDMFFile::read_cell_type,
nb::arg("name") = "mesh", nb::arg("xpath") = "/Xdmf/Domain")
.def("read_meshtags", &dolfinx::io::XDMFFile::read_meshtags,
.def("read_meshtags", &dolfinx::io::XDMFFile::read_meshtags<std::int32_t>,
nb::arg("mesh"), nb::arg("name"), nb::arg("attribute_name").none(),
nb::arg("xpath"))
.def("write_information", &dolfinx::io::XDMFFile::write_information,
Expand Down
Loading