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

Adds functionality to allow FSI to work with solids4foam as the solid solver #236

Merged
merged 23 commits into from
Aug 8, 2022
Merged
Show file tree
Hide file tree
Changes from 22 commits
Commits
Show all changes
23 commits
Select commit Hold shift + click to select a range
d7983e5
Add macOS (darwin) binary directories to gitignore
philipcardiff Jul 28, 2022
8bd4d57
Extend FSI to work with solids4foam for the solid
philipcardiff Jul 28, 2022
3916a16
Update FSI/Displacement.C
Jul 29, 2022
88e6fa5
Update Adapter.C
Jul 29, 2022
46aa5ee
Added changelog-entries/236.md
philipcardiff Jul 29, 2022
c4d9985
Merge branch 'develop' of github.com:solids4foam/openfoam-adapter int…
philipcardiff Jul 29, 2022
44c2f40
Apply clang-format to Adapter.C and FSI/Force.C
philipcardiff Jul 29, 2022
4412d2e
Add solverType solid
philipcardiff Jul 31, 2022
75eca1b
FSI: add nameSolidForce variable to FSI and Force classes
philipcardiff Aug 2, 2022
d4cc5ab
Update FSI/ForceBase.C
Aug 2, 2022
4b49300
FSI/Displacement.C: change required for solid solvers (like solidDisp…
philipcardiff Aug 2, 2022
afa9714
Merge branch 'develop' of github.com:solids4foam/openfoam-adapter int…
philipcardiff Aug 2, 2022
a0755dc
FSI/Displacement.C: use nullptr instead of NULL
philipcardiff Aug 4, 2022
e18ea6e
FSI/Force: store pointer to solidForce field
philipcardiff Aug 4, 2022
3b17ab0
FSI/Force: store pointer to solidForce field (use nullptr if not found)
philipcardiff Aug 4, 2022
abfe8c1
Adapter/reloadMeshPoints: do nothing is the mesh is not moving
philipcardiff Aug 4, 2022
b8621df
FSI: apply clang-format
philipcardiff Aug 4, 2022
3c2a9cc
FSI/Displacement: add warning about using faceNodes in parallel
philipcardiff Aug 4, 2022
a591b5e
Update FSI/Displacement.C
Aug 4, 2022
a8a2e3b
Update config.md
MakisH Aug 8, 2022
58c296c
Update README.md
MakisH Aug 8, 2022
a0aee14
Update config.md
MakisH Aug 8, 2022
8b62dbd
Update openfoam-support.md
MakisH Aug 8, 2022
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
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,7 @@
# OpenFOAM / WMake
lnInclude/
Make/linux64GccDPInt32Opt/
Make/darwin64ClangDPInt32Opt/

# Editors
.cproject
Expand Down
14 changes: 10 additions & 4 deletions Adapter.C
Original file line number Diff line number Diff line change
Expand Up @@ -757,6 +757,12 @@ void preciceAdapter::Adapter::storeMeshPoints()

void preciceAdapter::Adapter::reloadMeshPoints()
{
if (!mesh_.moving())
{
DEBUG(adapterInfo("Mesh points not moved as the mesh is not moving"));
return;
}

// In Foam::polyMesh::movePoints.
// TODO: The function movePoints overwrites the pointer to the old mesh.
// Therefore, if you revert the mesh, the oldpointer will be set to the points, which are the new values.
Expand Down Expand Up @@ -1301,8 +1307,8 @@ void preciceAdapter::Adapter::readMeshCheckpoint()
{
DEBUG(adapterInfo("Reading a mesh checkpoint..."));

//TODO only the meshPhi field is here, which is a surfaceScalarField. The other fields can be removed.
// Reload all the fields of type mesh surfaceScalarField
// TODO only the meshPhi field is here, which is a surfaceScalarField. The other fields can be removed.
// Reload all the fields of type mesh surfaceScalarField
for (uint i = 0; i < meshSurfaceScalarFields_.size(); i++)
{
// Load the volume field
Expand Down Expand Up @@ -1529,8 +1535,8 @@ void preciceAdapter::Adapter::teardown()
}
meshVolVectorFieldCopies_.clear();

//TODO for the internal volume
// volScalarInternal
// TODO for the internal volume
// volScalarInternal
for (uint i = 0; i < volScalarInternalFieldCopies_.size(); i++)
{
delete volScalarInternalFieldCopies_.at(i);
Expand Down
59 changes: 50 additions & 9 deletions FSI/Displacement.C
Original file line number Diff line number Diff line change
Expand Up @@ -7,8 +7,10 @@ preciceAdapter::FSI::Displacement::Displacement(
const std::string namePointDisplacement,
const std::string nameCellDisplacement)
: pointDisplacement_(
const_cast<pointVectorField*>(
&mesh.lookupObject<pointVectorField>(namePointDisplacement))),
namePointDisplacement == "unused"
? nullptr
: const_cast<pointVectorField*>(
&mesh.lookupObject<pointVectorField>(namePointDisplacement))),
cellDisplacement_(
const_cast<volVectorField*>(
&mesh.lookupObject<volVectorField>(nameCellDisplacement))),
Expand Down Expand Up @@ -37,13 +39,52 @@ void preciceAdapter::FSI::Displacement::initialize()
void preciceAdapter::FSI::Displacement::write(double* buffer, bool meshConnectivity, const unsigned int dim)
{
/* TODO: Implement
* We need two nested for-loops for each patch,
* the outer for the locations and the inner for the dimensions.
* See the preCICE writeBlockVectorData() implementation.
*/
FatalErrorInFunction
<< "Writing displacements is not supported."
<< exit(FatalError);
* We need two nested for-loops for each patch,
* the outer for the locations and the inner for the dimensions.
* See the preCICE writeBlockVectorData() implementation.
*/

// Copy the displacement field from OpenFOAM to the buffer

if (this->locationType_ == LocationType::faceCenters)
{
// For every boundary patch of the interface
for (const label patchID : patchIDs_)
{
// Write the displacement to the preCICE buffer
// For every cell of the patch
forAll(cellDisplacement_->boundaryField()[patchID], i)
{
for (unsigned int d = 0; d < dim; ++d)
buffer[i * dim + d] =
cellDisplacement_->boundaryField()[patchID][i][d];
}
}
}
else if (this->locationType_ == LocationType::faceNodes)
MakisH marked this conversation as resolved.
Show resolved Hide resolved
{
DEBUG(adapterInfo(
"Please be aware of issues with using 'locationType faceNodes' "
"in parallel. \n"
"See https://github.com/precice/openfoam-adapter/issues/153.",
"warning"));

// For every boundary patch of the interface
for (const label patchID : patchIDs_)
{
// Write the displacement to the preCICE buffer
// For every cell of the patch
forAll(pointDisplacement_->boundaryField()[patchID], i)
{
const labelList& meshPoints =
mesh_.boundaryMesh()[patchID].meshPoints();

for (unsigned int d = 0; d < dim; ++d)
buffer[i * dim + d] =
pointDisplacement_->internalField()[meshPoints[i]][d];
MakisH marked this conversation as resolved.
Show resolved Hide resolved
}
}
}
}


Expand Down
8 changes: 4 additions & 4 deletions FSI/DisplacementDelta.C
Original file line number Diff line number Diff line change
Expand Up @@ -37,10 +37,10 @@ void preciceAdapter::FSI::DisplacementDelta::initialize()
void preciceAdapter::FSI::DisplacementDelta::write(double* buffer, bool meshConnectivity, const unsigned int dim)
MakisH marked this conversation as resolved.
Show resolved Hide resolved
{
/* TODO: Implement
* We need two nested for-loops for each patch,
* the outer for the locations and the inner for the dimensions.
* See the preCICE writeBlockVectorData() implementation.
*/
* We need two nested for-loops for each patch,
* the outer for the locations and the inner for the dimensions.
* See the preCICE writeBlockVectorData() implementation.
*/
FatalErrorInFunction
<< "Writing displacementDeltas is not supported."
<< exit(FatalError);
Expand Down
18 changes: 12 additions & 6 deletions FSI/FSI.C
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,9 @@ bool preciceAdapter::FSI::FluidStructureInteraction::configure(const IOdictionar
// addWriters() and addReaders().
// Check the solver type and determine it if needed
if (
solverType_.compare("compressible") == 0 || solverType_.compare("incompressible") == 0)
solverType_.compare("compressible") == 0
|| solverType_.compare("incompressible") == 0
|| solverType_.compare("solid") == 0)
{
DEBUG(adapterInfo("Known solver type: " + solverType_));
}
Expand All @@ -38,7 +40,7 @@ bool preciceAdapter::FSI::FluidStructureInteraction::configure(const IOdictionar
}
else
{
DEBUG(adapterInfo("Determining the solver type for the FSI module... (override by setting solverType to one of {compressible, incompressible})"));
DEBUG(adapterInfo("Determining the solver type for the FSI module... (override by setting solverType to one of {compressible, incompressible, solid})"));
solverType_ = determineSolverType();
}

Expand All @@ -54,8 +56,8 @@ bool preciceAdapter::FSI::FluidStructureInteraction::readConfig(const IOdictiona
DEBUG(adapterInfo(" user-defined solver type : " + solverType_));

/* TODO: Read the names of any needed fields and parameters.
* Include the force here?
*/
* Include the force here?
*/

// Read the name of the pointDisplacement field (if different)
namePointDisplacement_ = FSIdict.lookupOrDefault<word>("namePointDisplacement", "pointDisplacement");
Expand All @@ -65,6 +67,10 @@ bool preciceAdapter::FSI::FluidStructureInteraction::readConfig(const IOdictiona
nameCellDisplacement_ = FSIdict.lookupOrDefault<word>("nameCellDisplacement", "cellDisplacement");
DEBUG(adapterInfo(" cellDisplacement field name : " + nameCellDisplacement_));

// Read the name of the solidForce field (if different)
nameSolidForce_ = FSIdict.lookupOrDefault<word>("nameSolidForce", "solidForce");
DEBUG(adapterInfo(" solidForce field name : " + nameSolidForce_));

return true;
}

Expand Down Expand Up @@ -117,7 +123,7 @@ bool preciceAdapter::FSI::FluidStructureInteraction::addWriters(std::string data
{
interface->addCouplingDataWriter(
dataName,
new Force(mesh_, solverType_) /* TODO: Add any other arguments here */
new Force(mesh_, solverType_, nameSolidForce_) /* TODO: Add any other arguments here */
);
DEBUG(adapterInfo("Added writer: Force."));
}
Expand Down Expand Up @@ -165,7 +171,7 @@ bool preciceAdapter::FSI::FluidStructureInteraction::addReaders(std::string data
{
interface->addCouplingDataReader(
dataName,
new Force(mesh_, solverType_) /* TODO: Add any other arguments here */
new Force(mesh_, solverType_, nameSolidForce_) /* TODO: Add any other arguments here */
);
DEBUG(adapterInfo("Added reader: Force."));
}
Expand Down
3 changes: 3 additions & 0 deletions FSI/FSI.H
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,9 @@ protected:
//- Name of the pointDisplacement field
std::string nameCellDisplacement_ = "cellDisplacement";

//- Name of the force field used by the solid
std::string nameSolidForce_ = "solidForce";

/* TODO: Declare here any parameters that should be read from
/ the configuration file. See CHT/CHT.H for reference.
/ We want to support in-house solvers with different field names,
Expand Down
46 changes: 44 additions & 2 deletions FSI/Force.C
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,8 @@ using namespace Foam;

preciceAdapter::FSI::Force::Force(
const Foam::fvMesh& mesh,
const std::string solverType)
const std::string solverType,
const std::string nameSolidForce)
: ForceBase(mesh, solverType)
{
Force_ = new volVectorField(
Expand All @@ -19,6 +20,17 @@ preciceAdapter::FSI::Force::Force(
"fdim",
dimensionSet(1, 1, -2, 0, 0, 0, 0),
Foam::vector::zero));

if (mesh_.foundObject<volVectorField>(nameSolidForce))
{
solidForce_ =
&const_cast<volVectorField&>(
mesh_.lookupObject<volVectorField>(nameSolidForce));
}
else
{
solidForce_ = nullptr;
}
}

void preciceAdapter::FSI::Force::write(double* buffer, bool meshConnectivity, const unsigned int dim)
Expand All @@ -28,7 +40,37 @@ void preciceAdapter::FSI::Force::write(double* buffer, bool meshConnectivity, co

void preciceAdapter::FSI::Force::read(double* buffer, const unsigned int dim)
{
this->readFromBuffer(buffer);
// Copy the force field from the buffer to OpenFOAM

// Here we assume that a force volVectorField exists, which is used by
// the OpenFOAM solver

// Set boundary forces
for (unsigned int j = 0; j < patchIDs_.size(); j++)
{
// Get the ID of the current patch
const unsigned int patchID = patchIDs_.at(j);

if (this->locationType_ == LocationType::faceCenters)
{
// Make a force field
vectorField& force = solidForce_->boundaryFieldRef()[patchID];

// Copy the forces from the buffer into the force field
forAll(force, i)
{
for (unsigned int d = 0; d < dim; ++d)
force[i][d] = buffer[i * dim + d];
}
}
else if (this->locationType_ == LocationType::faceNodes)
{
// Here we could easily interpolate the face values to point values
// and assign them to some field, but I guess there is no need
// unless it will be used
notImplemented("Read forces not implemented for faceNodes!");
MakisH marked this conversation as resolved.
Show resolved Hide resolved
}
}
}

bool preciceAdapter::FSI::Force::isLocationTypeSupported(const bool meshConnectivity) const
Expand Down
6 changes: 5 additions & 1 deletion FSI/Force.H
Original file line number Diff line number Diff line change
Expand Up @@ -15,11 +15,15 @@ private:
//- Force field
Foam::volVectorField* Force_;

//- Solid force field
Foam::volVectorField* solidForce_;

public:
//- Constructor
Force(
const Foam::fvMesh& mesh,
const std::string solverType);
const std::string solverType,
const std::string nameSolidForce);

//- Write the forces values into the buffer
void write(double* buffer, bool meshConnectivity, const unsigned int dim) override;
Expand Down
25 changes: 13 additions & 12 deletions FSI/ForceBase.C
Original file line number Diff line number Diff line change
Expand Up @@ -9,22 +9,24 @@ preciceAdapter::FSI::ForceBase::ForceBase(
: mesh_(mesh),
solverType_(solverType)
{
//What about type "basic"?
if (solverType_.compare("incompressible") != 0 && solverType_.compare("compressible") != 0)
// What about type "basic"?
if (solverType_.compare("incompressible") != 0
&& solverType_.compare("compressible") != 0
&& solverType_.compare("solid") != 0)
{
FatalErrorInFunction
<< "Force based calculations only support "
<< "compressible or incompressible solver types."
<< "compressible, incompressible, or solid solver types."
<< exit(FatalError);
}

dataType_ = vector;
}

//Calculate viscous force
// Calculate viscous force
Foam::tmp<Foam::volSymmTensorField> preciceAdapter::FSI::ForceBase::devRhoReff() const
{
//For turbulent flows
// For turbulent flows
typedef compressible::turbulenceModel cmpTurbModel;
typedef incompressible::turbulenceModel icoTurbModel;

Expand Down Expand Up @@ -52,7 +54,7 @@ Foam::tmp<Foam::volSymmTensorField> preciceAdapter::FSI::ForceBase::devRhoReff()
}
}

//lookup correct rho
// lookup correct rho
Foam::tmp<Foam::volScalarField> preciceAdapter::FSI::ForceBase::rho() const
{
// If volScalarField exists, read it from registry (for compressible cases)
Expand Down Expand Up @@ -88,10 +90,9 @@ Foam::tmp<Foam::volScalarField> preciceAdapter::FSI::ForceBase::rho() const
}
}

//lookup correct mu
// lookup correct mu
Foam::tmp<Foam::volScalarField> preciceAdapter::FSI::ForceBase::mu() const
{

if (solverType_.compare("incompressible") == 0)
{
typedef immiscibleIncompressibleTwoPhaseMixture iitpMixture;
Expand Down Expand Up @@ -191,10 +192,10 @@ void preciceAdapter::FSI::ForceBase::writeToBuffer(double* buffer,
void preciceAdapter::FSI::ForceBase::readFromBuffer(double* buffer) const
{
/* TODO: Implement
* We need two nested for-loops for each patch,
* the outer for the locations and the inner for the dimensions.
* See the preCICE readBlockVectorData() implementation.
*/
* We need two nested for-loops for each patch,
* the outer for the locations and the inner for the dimensions.
* See the preCICE readBlockVectorData() implementation.
*/
FatalErrorInFunction
<< "Reading forces is not supported."
<< exit(FatalError);
Expand Down
1 change: 1 addition & 0 deletions changelog-entries/236.md
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
- Added functionality to allow use of solids4foam with the OpenFOAM adapter
4 changes: 2 additions & 2 deletions docs/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -18,9 +18,9 @@ This adapter can read/write the following fields:
- Heat flux (read + write)
- Sink temperature (read + write)
- Heat transfer coefficient (read + write)
- Force (write)
- Force (read + write)
- Stress (write)
- Displacement (read)
- Displacement (read + write)
- Displacement delta (read)
- Pressure (read + write)
- Pressure gradient (read + write)
Expand Down
4 changes: 2 additions & 2 deletions docs/config.md
Original file line number Diff line number Diff line change
Expand Up @@ -120,7 +120,7 @@ writeData
);
```

For fluid-structure interaction, `writeData` can be `Force` or `Stress`, where `Stress` is essentially a force vector scaled by the cell face in spatial coordinates (with any postfix), thus, a conservative quantity as well.`readData` can be `Displacement` and `DisplacementDelta` (with any postfix). `DisplacementDelta` refers to the last coupling time step, which needs to considered in the case of subcycling.
For fluid-structure interaction, `writeData` can be `Force` or `Stress`, where `Stress` is essentially a force vector scaled by the cell face in spatial coordinates (with any postfix), thus, a conservative quantity as well. `readData` can be `Displacement` and `DisplacementDelta` (with any postfix). `DisplacementDelta` refers to the last coupling time step, which needs to considered in the case of subcycling. Structure solvers (such as solids4Foam) can also write `Displacement` and read `Force`.

{% warning %}
You will run into problems when you use `Displacement(Delta)` as write data set and execute RBF mappings in parallel. This would affect users who use OpenFOAM and the adapter as the Solid participant in order to compute solid mechanics with OpenFOAM (currently not officially supported at all). Have a look [at this issue on GitHub](https://github.com/precice/openfoam-adapter/issues/153) for details.
Expand Down Expand Up @@ -312,7 +312,7 @@ and depends on the density (`rho [ 1 -3 0 0 0 0 0 ]`) and heat capacity (`Cp

#### Fluid-structure interaction

The adapter's FSI functionality supports both compressible and incompressible solvers.
The adapter's FSI functionality supports both compressible and incompressible solvers, as well as solid (e.g., solids4Foam) solvers.

For incompressible solvers, it tries to read uniform values for the density and kinematic viscosity (if it is not already available) from the `FSI` subdictionary of `preciceDict`:

Expand Down