Skip to content

Build your own adapter

David Schneider edited this page Jun 12, 2019 · 14 revisions

The dealii-adapter provides a minimal example of a deal.II code, which has been coupled using preCICE. Since most of the users will probably have their own code for a preCICE coupling, this section should give an explanation and overview for the preCICE related code changes. A more general adapter example with a step-by-step tutorial for own adapter buildings is also available in the preCICE wiki.

Contact us if you have any questions. Even if you don't have any questions, please let us know about your experience when your adapter is ready!

Which information is needed by preCICE?

preCICE uses a black-box coupling approach, which means the solver only need to provide a minimal set of information. In the most simple case, this includes configuration information e.g. name of the participant and the coordinates of the data you want to exchange e.g. the mesh vertices. If you want to use a nearest-projection mapping, you need to specify additionally mesh connectivity between the coordinates, which is currently not included in this adapter example.

About fluid-structure coupling

For every multi-physical coupling, proper coupling data needs to be exchanged between all participants. In our example case, the Fluid participant calculates forces (per cell face), which are passed to the Solid participant. Using the forces for the structural calculations, the Solid participant calculates displacements, which are then passed back to the Fluid participant. As outlined above, preCICE needs coordinates of the data points you want to exchange. Since the forces live on the cell faces and the displacements live on the cell vertices, we define two coupling meshes in this example: A face based mesh for the forces: Fluid -> Solid and a vertex based mesh for the displacements: Solid -> Fluid. This is not necessarily the case and depends on your spatial discretization method and the coupling data. If both data sets live on the same mesh element, defining one mesh would be sufficient.

Step-by-step guidance

This guideline can be used to couple your own code using preCICE. All code snippets have been taken from the coupled_elasto_dynamics.cc file.

Step 1: Prepare your solver

As a first preparation step, you need to include the preCICE library headers, which is simply the SolverInterface.hpp

#include "precice/SolverInterface.hpp"

The preCICE API is explained in the source code.

Step 2: Set up API

The API is located in precice::SolverInterface, and its constructor requires the participant's name, and the rank and size of the current thread. Since it is typically for deal.II, to create a class for each code and call the run() function so start the simulation, the precice::SolverInterface object is generated in the constructor of the CoupledElastoDynamics class. Once the basic preCICE interface is set up, we can steer the behaviour of preCICE.

Step 3: Initialize preCICE

Initializing preCICE is done in the respective initialize_precice() function, which is located in the run() function. Here, the configuration file is read and the number of interface nodes and the interface coordinates are passed to preCICE. Have a look in the source code and the comments for the implementation. Depending on the location of your coupling data (cf. fluid-structure section), you might be able to copy this function completely.

Note 1: In order to exchange data between participants, preCICE needs to know, which data is actually in the data array. Therefore, the data names in the precice-config.xml are converted to preCICE specific IDs. Make sure the naming in the respective functions is the same in the source code and in the configuration file e.g. if you name the coupling data force Force in your precice-config.xml, you need to pass the same naming Force in the getDataID(Force,...) function.

Note 2: Several declarations for preCICE related functions or variables have been made in the constructor of the main class.

Note 3: Apart from the coordinates of the data points, we need to pass the number of data points to preCICE. In case of the interface faces, this step is included at the end of the make_grid() function, where the boundary_IDs are set as well, in order to avoid code duplication.

Note 4: For each information you want to pass to preCICE, you need to provide the information in a specific data format. Therefore, all vector components need to be listed in an array as shown in the following example. Think of the data coordinates: Each coordinate is a dim dimensional vector with v1 = [x1; y1; z1] , v2 = [x2; y2; z2] ... For preCICE, all components are collected in a single array and arranged like: coordinates = [x1 y1 z1 x2 y2 z2 x3...]

Step 4: Modify the time loop

Coupled problems are usually time dependent. We set up a separate Time class, which handles all time related tasks. preCICE uses specific functions in the time loop, which steer the simulation. Therefore, have a look at the following simplified basic version of our example program:

     while(precice.isCouplingOngoing() &&
           time.get_timestep() < time.get_n_timesteps())  // ask preCICE for the coupling process
     {
         time.increment();       // increment time 

         assemble_rhs();         // apply coupling data in the rhs 

         solve();                // solve the system as ususal

         update_displacement();  // update time dependent variables as usual

         advance_precice();      // exchange coupling data

         output_results(time.get_timestep()); // store result files as usual
     }
Clone this wiki locally