diff --git a/previews/PR514/.documenter-siteinfo.json b/previews/PR514/.documenter-siteinfo.json index 5842a6dab..a5c134f78 100644 --- a/previews/PR514/.documenter-siteinfo.json +++ b/previews/PR514/.documenter-siteinfo.json @@ -1 +1 @@ -{"documenter":{"julia_version":"1.11.1","generation_timestamp":"2024-11-19T16:53:59","documenter_version":"1.8.0"}} \ No newline at end of file +{"documenter":{"julia_version":"1.11.1","generation_timestamp":"2024-11-19T17:09:01","documenter_version":"1.8.0"}} \ No newline at end of file diff --git a/previews/PR514/authors/index.html b/previews/PR514/authors/index.html index 9c10fde56..f15bff091 100644 --- a/previews/PR514/authors/index.html +++ b/previews/PR514/authors/index.html @@ -1,2 +1,2 @@ -Authors · TrixiParticles.jl

Authors

TrixiParticles.jl's development is coordinated by a group of principal developers, who are also its main contributors and who can be contacted in case of questions about TrixiParticles.jl. In addition, there are contributors who have provided substantial additions or modifications. Together, these two groups form "The TrixiParticles.jl Authors" as mentioned under License.

Principal Developers

Contributors

The following people contributed major additions or modifications to TrixiParticles.jl and are listed in alphabetical order:

  • Sven Berger
  • Erik Faulhaber
  • Gregor Gassner
  • Niklas Neher
  • Hendrik Ranocha
  • Michael Schlottke-Lakemper
+Authors · TrixiParticles.jl

Authors

TrixiParticles.jl's development is coordinated by a group of principal developers, who are also its main contributors and who can be contacted in case of questions about TrixiParticles.jl. In addition, there are contributors who have provided substantial additions or modifications. Together, these two groups form "The TrixiParticles.jl Authors" as mentioned under License.

Principal Developers

Contributors

The following people contributed major additions or modifications to TrixiParticles.jl and are listed in alphabetical order:

  • Sven Berger
  • Erik Faulhaber
  • Gregor Gassner
  • Niklas Neher
  • Hendrik Ranocha
  • Michael Schlottke-Lakemper
diff --git a/previews/PR514/callbacks/index.html b/previews/PR514/callbacks/index.html index b860239cd..57b7b4560 100644 --- a/previews/PR514/callbacks/index.html +++ b/previews/PR514/callbacks/index.html @@ -1,11 +1,11 @@ -Callbacks · TrixiParticles.jl

Callbacks

TrixiParticles.DensityReinitializationCallbackType
DensityReinitializationCallback(; interval::Integer=0, dt=0.0)

Callback to reinitialize the density field when using ContinuityDensity [42].

Keywords

  • interval=0: Reinitialize the density every interval time steps.
  • dt: Reinitialize the density in regular intervals of dt in terms of integration time.
  • reinit_initial_solution: Reinitialize the initial solution (default=false)
source
TrixiParticles.InfoCallbackMethod
InfoCallback()

Create and return a callback that prints a human-readable summary of the simulation setup at the beginning of a simulation and then resets the timer. When the returned callback is executed directly, the current timer values are shown.

source
TrixiParticles.PostprocessCallbackType
PostprocessCallback(; interval::Integer=0, dt=0.0, exclude_boundary=true, filename="values",
+Callbacks · TrixiParticles.jl

Callbacks

TrixiParticles.DensityReinitializationCallbackType
DensityReinitializationCallback(; interval::Integer=0, dt=0.0)

Callback to reinitialize the density field when using ContinuityDensity [42].

Keywords

  • interval=0: Reinitialize the density every interval time steps.
  • dt: Reinitialize the density in regular intervals of dt in terms of integration time.
  • reinit_initial_solution: Reinitialize the initial solution (default=false)
source
TrixiParticles.InfoCallbackMethod
InfoCallback()

Create and return a callback that prints a human-readable summary of the simulation setup at the beginning of a simulation and then resets the timer. When the returned callback is executed directly, the current timer values are shown.

source
TrixiParticles.PostprocessCallbackType
PostprocessCallback(; interval::Integer=0, dt=0.0, exclude_boundary=true, filename="values",
                     output_directory="out", append_timestamp=false, write_csv=true,
                     write_json=true, write_file_interval=1, funcs...)

Create a callback to post-process simulation data at regular intervals. This callback allows for the execution of a user-defined function func at specified intervals during the simulation. The function is applied to the current state of the simulation, and its results can be saved or used for further analysis. The provided function cannot be anonymous as the function name will be used as part of the name of the value.

The callback can be triggered either by a fixed number of time steps (interval) or by a fixed interval of simulation time (dt).

Keywords

  • funcs...: Functions to be executed at specified intervals during the simulation. Each function must have the arguments (v, u, t, system), and will be called for every system, where v and u are the wrapped solution arrays for the corresponding system and t is the current simulation time. Note that working with these v and u arrays requires undocumented internal functions of TrixiParticles. See Custom Quantities for a list of pre-defined functions that can be used here.
  • interval=0: Specifies the number of time steps between each invocation of the callback. If set to 0, the callback will not be triggered based on time steps. Either interval or dt must be set to something larger than 0.
  • dt=0.0: Specifies the simulation time interval between each invocation of the callback. If set to 0.0, the callback will not be triggered based on simulation time. Either interval or dt must be set to something larger than 0.
  • exclude_boundary=true: If set to true, boundary particles will be excluded from the post-processing.
  • filename="values": The filename of the postprocessing files to be saved.
  • output_directory="out": The path where the results of the post-processing will be saved.
  • write_csv=true: If set to true, write a csv file.
  • write_json=true: If set to true, write a json file.
  • append_timestep=false: If set to true, the current timestamp will be added to the filename.
  • write_file_interval=1: Files will be written after every write_file_interval number of postprocessing execution steps. A value of 0 indicates that files are only written at the end of the simulation, eliminating I/O overhead.

Examples

# Create a callback that is triggered every 100 time steps
 postprocess_callback = PostprocessCallback(interval=100, example_quantity=kinetic_energy)
 
 # Create a callback that is triggered every 0.1 simulation time units
-postprocess_callback = PostprocessCallback(dt=0.1, example_quantity=kinetic_energy)
source
TrixiParticles.SolutionSavingCallbackType
SolutionSavingCallback(; interval::Integer=0, dt=0.0, save_times=Array{Float64, 1}([]),
+postprocess_callback = PostprocessCallback(dt=0.1, example_quantity=kinetic_energy)
source
TrixiParticles.SolutionSavingCallbackType
SolutionSavingCallback(; interval::Integer=0, dt=0.0, save_times=Array{Float64, 1}([]),
                        save_initial_solution=true, save_final_solution=true,
                        output_directory="out", append_timestamp=false, prefix="",
                        verbose=false, write_meta_data=true, max_coordinates=2^15,
@@ -16,6 +16,6 @@
 saving_callback = SolutionSavingCallback(dt=0.1)
 
 # Additionally store the kinetic energy of each system as "my_custom_quantity"
-saving_callback = SolutionSavingCallback(dt=0.1, my_custom_quantity=kinetic_energy)
source
TrixiParticles.SteadyStateReachedCallbackType
SteadyStateReachedCallback(; interval::Integer=0, dt=0.0,
-                           interval_size::Integer=10, abstol=1.0e-8, reltol=1.0e-6)

Terminates the integration when the change of kinetic energy between time steps falls below the threshold specified by abstol + reltol * ekin, where ekin is the total kinetic energy of the simulation.

Keywords

  • interval=0: Check steady state condition every interval time steps.
  • dt=0.0: Check steady state condition in regular intervals of dt in terms of integration time by adding additional tstops (note that this may change the solution).
  • interval_size: The interval in which the change of the kinetic energy is considered. interval_size is a (integer) multiple of interval or dt.
  • abstol: Absolute tolerance.
  • reltol: Relative tolerance.
source
TrixiParticles.StepsizeCallbackMethod
StepsizeCallback(; cfl::Real)

Set the time step size according to a CFL condition if the time integration method isn't adaptive itself.

The current implementation is using the simplest form of CFL condition, which chooses a time step size that is constant during the simulation. The step size is therefore only applied once at the beginning of the simulation.

The step size $\Delta t$ is chosen as the minimum

\[ \Delta t = \min(\Delta t_\eta, \Delta t_a, \Delta t_c),\]

where

\[ \Delta t_\eta = 0.125 \, h^2 / \eta, \quad \Delta t_a = 0.25 \sqrt{h / \lVert g \rVert}, - \quad \Delta t_c = \text{CFL} \, h / c,\]

with $\nu = \alpha h c / (2n + 4)$, where $\alpha$ is the parameter of the viscosity and $n$ is the number of dimensions.

Experimental implementation

This is an experimental feature and may change in future releases.

References

[21], [14], [43], [44]

source
TrixiParticles.UpdateCallbackMethod
UpdateCallback(; interval::Integer, dt=0.0)

Callback to update quantities either at the end of every interval time steps or in intervals of dt in terms of integration time by adding additional tstops (note that this may change the solution).

Keywords

  • interval=1: Update quantities at the end of every interval time steps.
  • dt: Update quantities in regular intervals of dt in terms of integration time by adding additional tstops (note that this may change the solution).
source

Custom Quantities

The following pre-defined custom quantities can be used with the SolutionSavingCallback and PostprocessCallback.

+saving_callback = SolutionSavingCallback(dt=0.1, my_custom_quantity=kinetic_energy)
source
TrixiParticles.SteadyStateReachedCallbackType
SteadyStateReachedCallback(; interval::Integer=0, dt=0.0,
+                           interval_size::Integer=10, abstol=1.0e-8, reltol=1.0e-6)

Terminates the integration when the change of kinetic energy between time steps falls below the threshold specified by abstol + reltol * ekin, where ekin is the total kinetic energy of the simulation.

Keywords

  • interval=0: Check steady state condition every interval time steps.
  • dt=0.0: Check steady state condition in regular intervals of dt in terms of integration time by adding additional tstops (note that this may change the solution).
  • interval_size: The interval in which the change of the kinetic energy is considered. interval_size is a (integer) multiple of interval or dt.
  • abstol: Absolute tolerance.
  • reltol: Relative tolerance.
source
TrixiParticles.StepsizeCallbackMethod
StepsizeCallback(; cfl::Real)

Set the time step size according to a CFL condition if the time integration method isn't adaptive itself.

The current implementation is using the simplest form of CFL condition, which chooses a time step size that is constant during the simulation. The step size is therefore only applied once at the beginning of the simulation.

The step size $\Delta t$ is chosen as the minimum

\[ \Delta t = \min(\Delta t_\eta, \Delta t_a, \Delta t_c),\]

where

\[ \Delta t_\eta = 0.125 \, h^2 / \eta, \quad \Delta t_a = 0.25 \sqrt{h / \lVert g \rVert}, + \quad \Delta t_c = \text{CFL} \, h / c,\]

with $\nu = \alpha h c / (2n + 4)$, where $\alpha$ is the parameter of the viscosity and $n$ is the number of dimensions.

Experimental implementation

This is an experimental feature and may change in future releases.

References

[21], [14], [43], [44]

source
TrixiParticles.UpdateCallbackMethod
UpdateCallback(; interval::Integer, dt=0.0)

Callback to update quantities either at the end of every interval time steps or in intervals of dt in terms of integration time by adding additional tstops (note that this may change the solution).

Keywords

  • interval=1: Update quantities at the end of every interval time steps.
  • dt: Update quantities in regular intervals of dt in terms of integration time by adding additional tstops (note that this may change the solution).
source

Custom Quantities

The following pre-defined custom quantities can be used with the SolutionSavingCallback and PostprocessCallback.

diff --git a/previews/PR514/code_of_conduct/index.html b/previews/PR514/code_of_conduct/index.html index 52d5cbc6a..3ddfe30f4 100644 --- a/previews/PR514/code_of_conduct/index.html +++ b/previews/PR514/code_of_conduct/index.html @@ -1,2 +1,2 @@ -Code of Conduct · TrixiParticles.jl

Code of Conduct

Contributor Covenant Code of Conduct

Our Pledge

We as members, contributors, and leaders pledge to make participation in our community a harassment-free experience for everyone, regardless of age, body size, visible or invisible disability, ethnicity, sex characteristics, gender identity and expression, level of experience, education, socio-economic status, nationality, personal appearance, race, religion, or sexual identity and orientation.

We pledge to act and interact in ways that contribute to an open, welcoming, diverse, inclusive, and healthy community.

Our Standards

Examples of behavior that contributes to a positive environment for our community include:

  • Demonstrating empathy and kindness toward other people
  • Being respectful of differing opinions, viewpoints, and experiences
  • Giving and gracefully accepting constructive feedback
  • Accepting responsibility and apologizing to those affected by our mistakes, and learning from the experience
  • Focusing on what is best not just for us as individuals, but for the overall community

Examples of unacceptable behavior include:

  • The use of sexualized language or imagery, and sexual attention or advances of any kind
  • Trolling, insulting or derogatory comments, and personal or political attacks
  • Public or private harassment
  • Publishing others' private information, such as a physical or email address, without their explicit permission
  • Other conduct which could reasonably be considered inappropriate in a professional setting

Enforcement Responsibilities

Community leaders are responsible for clarifying and enforcing our standards of acceptable behavior and will take appropriate and fair corrective action in response to any behavior that they deem inappropriate, threatening, offensive, or harmful.

Community leaders have the right and responsibility to remove, edit, or reject comments, commits, code, wiki edits, issues, and other contributions that are not aligned to this Code of Conduct, and will communicate reasons for moderation decisions when appropriate.

Scope

This Code of Conduct applies within all community spaces, and also applies when an individual is officially representing the community in public spaces. Examples of representing our community include using an official e-mail address, posting via an official social media account, or acting as an appointed representative at an online or offline event.

Enforcement

Instances of abusive, harassing, or otherwise unacceptable behavior may be reported to Michael Schlottke-Lakemper, Sven Berger, or any other of the principal developers responsible for enforcement listed in Authors. All complaints will be reviewed and investigated promptly and fairly.

All community leaders are obligated to respect the privacy and security of the reporter of any incident.

Enforcement Guidelines

Community leaders will follow these Community Impact Guidelines in determining the consequences for any action they deem in violation of this Code of Conduct:

1. Correction

Community Impact: Use of inappropriate language or other behavior deemed unprofessional or unwelcome in the community.

Consequence: A private, written warning from community leaders, providing clarity around the nature of the violation and an explanation of why the behavior was inappropriate. A public apology may be requested.

2. Warning

Community Impact: A violation through a single incident or series of actions.

Consequence: A warning with consequences for continued behavior. No interaction with the people involved, including unsolicited interaction with those enforcing the Code of Conduct, for a specified period of time. This includes avoiding interactions in community spaces as well as external channels like social media. Violating these terms may lead to a temporary or permanent ban.

3. Temporary Ban

Community Impact: A serious violation of community standards, including sustained inappropriate behavior.

Consequence: A temporary ban from any sort of interaction or public communication with the community for a specified period of time. No public or private interaction with the people involved, including unsolicited interaction with those enforcing the Code of Conduct, is allowed during this period. Violating these terms may lead to a permanent ban.

4. Permanent Ban

Community Impact: Demonstrating a pattern of violation of community standards, including sustained inappropriate behavior, harassment of an individual, or aggression toward or disparagement of classes of individuals.

Consequence: A permanent ban from any sort of public interaction within the community.

Attribution

This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 2.0, available at https://www.contributor-covenant.org/version/2/0/codeofconduct.html.

Community Impact Guidelines were inspired by Mozilla's code of conduct enforcement ladder.

[homepage]: https://www.contributor-covenant.org

For answers to common questions about this code of conduct, see the FAQ at https://www.contributor-covenant.org/faq. Translations are available at https://www.contributor-covenant.org/translations.

+Code of Conduct · TrixiParticles.jl

Code of Conduct

Contributor Covenant Code of Conduct

Our Pledge

We as members, contributors, and leaders pledge to make participation in our community a harassment-free experience for everyone, regardless of age, body size, visible or invisible disability, ethnicity, sex characteristics, gender identity and expression, level of experience, education, socio-economic status, nationality, personal appearance, race, religion, or sexual identity and orientation.

We pledge to act and interact in ways that contribute to an open, welcoming, diverse, inclusive, and healthy community.

Our Standards

Examples of behavior that contributes to a positive environment for our community include:

  • Demonstrating empathy and kindness toward other people
  • Being respectful of differing opinions, viewpoints, and experiences
  • Giving and gracefully accepting constructive feedback
  • Accepting responsibility and apologizing to those affected by our mistakes, and learning from the experience
  • Focusing on what is best not just for us as individuals, but for the overall community

Examples of unacceptable behavior include:

  • The use of sexualized language or imagery, and sexual attention or advances of any kind
  • Trolling, insulting or derogatory comments, and personal or political attacks
  • Public or private harassment
  • Publishing others' private information, such as a physical or email address, without their explicit permission
  • Other conduct which could reasonably be considered inappropriate in a professional setting

Enforcement Responsibilities

Community leaders are responsible for clarifying and enforcing our standards of acceptable behavior and will take appropriate and fair corrective action in response to any behavior that they deem inappropriate, threatening, offensive, or harmful.

Community leaders have the right and responsibility to remove, edit, or reject comments, commits, code, wiki edits, issues, and other contributions that are not aligned to this Code of Conduct, and will communicate reasons for moderation decisions when appropriate.

Scope

This Code of Conduct applies within all community spaces, and also applies when an individual is officially representing the community in public spaces. Examples of representing our community include using an official e-mail address, posting via an official social media account, or acting as an appointed representative at an online or offline event.

Enforcement

Instances of abusive, harassing, or otherwise unacceptable behavior may be reported to Michael Schlottke-Lakemper, Sven Berger, or any other of the principal developers responsible for enforcement listed in Authors. All complaints will be reviewed and investigated promptly and fairly.

All community leaders are obligated to respect the privacy and security of the reporter of any incident.

Enforcement Guidelines

Community leaders will follow these Community Impact Guidelines in determining the consequences for any action they deem in violation of this Code of Conduct:

1. Correction

Community Impact: Use of inappropriate language or other behavior deemed unprofessional or unwelcome in the community.

Consequence: A private, written warning from community leaders, providing clarity around the nature of the violation and an explanation of why the behavior was inappropriate. A public apology may be requested.

2. Warning

Community Impact: A violation through a single incident or series of actions.

Consequence: A warning with consequences for continued behavior. No interaction with the people involved, including unsolicited interaction with those enforcing the Code of Conduct, for a specified period of time. This includes avoiding interactions in community spaces as well as external channels like social media. Violating these terms may lead to a temporary or permanent ban.

3. Temporary Ban

Community Impact: A serious violation of community standards, including sustained inappropriate behavior.

Consequence: A temporary ban from any sort of interaction or public communication with the community for a specified period of time. No public or private interaction with the people involved, including unsolicited interaction with those enforcing the Code of Conduct, is allowed during this period. Violating these terms may lead to a permanent ban.

4. Permanent Ban

Community Impact: Demonstrating a pattern of violation of community standards, including sustained inappropriate behavior, harassment of an individual, or aggression toward or disparagement of classes of individuals.

Consequence: A permanent ban from any sort of public interaction within the community.

Attribution

This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 2.0, available at https://www.contributor-covenant.org/version/2/0/codeofconduct.html.

Community Impact Guidelines were inspired by Mozilla's code of conduct enforcement ladder.

[homepage]: https://www.contributor-covenant.org

For answers to common questions about this code of conduct, see the FAQ at https://www.contributor-covenant.org/faq. Translations are available at https://www.contributor-covenant.org/translations.

diff --git a/previews/PR514/contributing/index.html b/previews/PR514/contributing/index.html index bb6a7785e..be0427c75 100644 --- a/previews/PR514/contributing/index.html +++ b/previews/PR514/contributing/index.html @@ -35,4 +35,4 @@ are public and that a record of the contribution (including all personal information I submit with it, including my sign-off) is maintained indefinitely and may be redistributed consistent with - this project or the open source license(s) involved. + this project or the open source license(s) involved. diff --git a/previews/PR514/development/index.html b/previews/PR514/development/index.html index e07970463..c4f69e723 100644 --- a/previews/PR514/development/index.html +++ b/previews/PR514/development/index.html @@ -1,2 +1,2 @@ -Development · TrixiParticles.jl

Development

Preview of the documentation

To generate the Documentation, first instantiate the docs environment by executing the following command from the TrixiParticles.jl root directory:

julia --project=docs -e "using Pkg; Pkg.develop(PackageSpec(path=pwd())); Pkg.instantiate()"

This command only has to be run once. After that, maintain the docs environment as described under Installation.

With an instantiated docs environment, generate the docs with the following command (again from the TrixiParticles.jl root directory):

julia --project=docs --color=yes docs/make.jl

You can then open the generated files in docs/build with your webbrowser. Alternatively, run

python3 -m http.server -d docs/build

and open localhost:8000 in your webbrowser.

Release management

To create a new release for TrixiParticles.jl, perform the following steps:

  1. Make sure that all PRs and changes that you want to go into the release are merged to main and that the latest commit on main has passed all CI tests.
  2. Determine the currently released version of TrixiParticles.jl, e.g., on the release page. For this manual, we will assume that the latest release was v0.2.3.
  3. Decide on the next version number. We follow semantic versioning, thus each version is of the form vX.Y.Z where X is the major version, Y the minor version, and Z the patch version. In this manual, we assume that the major version is always 0, thus the decision process on the new version is as follows:
    • If the new release contains breaking changes (i.e., user code might not work as before without modifications), increase the minor version by one and set the patch version to zero. In our example, the new version should thus be v0.3.0.
    • If the new release only contains minor modifications and/or bug fixes, the minor version is kept as-is and the patch version is increased by one. In our example, the new version should thus be v0.2.4.
  4. Edit the version string in the Project.toml and set it to the new version. Push/merge this change to main.
  5. Go to GitHub and add a comment to the commit that you would like to become the new release (typically this will be the commit where you just updated the version). You can comment on a commit by going to the commit overview and clicking on the title of the commit. The comment should contain the following text:
    @JuliaRegistrator register
  6. Wait for the magic to happen! Specifically, JuliaRegistrator will create a new PR to the Julia registry with the new release information. After a grace period of ~15 minutes, this PR will be merged automatically. A short while after, TagBot will create a new release of TrixiParticles.jl in our GitHub repository.
  7. Once the new release has been created, the new version can be obtained through the Julia package manager as usual.
  8. To make sure people do not mistake the latest state of main as the latest release, we set the version in the Project.toml to a development version. The development version should be the latest released version, with the patch version incremented by one, and the -dev suffix added. For example, if you just released v0.3.0, the new development version should be v0.3.1-dev. If you just released v0.2.4, the new development version should be v0.2.5-dev.
+Development · TrixiParticles.jl

Development

Preview of the documentation

To generate the Documentation, first instantiate the docs environment by executing the following command from the TrixiParticles.jl root directory:

julia --project=docs -e "using Pkg; Pkg.develop(PackageSpec(path=pwd())); Pkg.instantiate()"

This command only has to be run once. After that, maintain the docs environment as described under Installation.

With an instantiated docs environment, generate the docs with the following command (again from the TrixiParticles.jl root directory):

julia --project=docs --color=yes docs/make.jl

You can then open the generated files in docs/build with your webbrowser. Alternatively, run

python3 -m http.server -d docs/build

and open localhost:8000 in your webbrowser.

Release management

To create a new release for TrixiParticles.jl, perform the following steps:

  1. Make sure that all PRs and changes that you want to go into the release are merged to main and that the latest commit on main has passed all CI tests.
  2. Determine the currently released version of TrixiParticles.jl, e.g., on the release page. For this manual, we will assume that the latest release was v0.2.3.
  3. Decide on the next version number. We follow semantic versioning, thus each version is of the form vX.Y.Z where X is the major version, Y the minor version, and Z the patch version. In this manual, we assume that the major version is always 0, thus the decision process on the new version is as follows:
    • If the new release contains breaking changes (i.e., user code might not work as before without modifications), increase the minor version by one and set the patch version to zero. In our example, the new version should thus be v0.3.0.
    • If the new release only contains minor modifications and/or bug fixes, the minor version is kept as-is and the patch version is increased by one. In our example, the new version should thus be v0.2.4.
  4. Edit the version string in the Project.toml and set it to the new version. Push/merge this change to main.
  5. Go to GitHub and add a comment to the commit that you would like to become the new release (typically this will be the commit where you just updated the version). You can comment on a commit by going to the commit overview and clicking on the title of the commit. The comment should contain the following text:
    @JuliaRegistrator register
  6. Wait for the magic to happen! Specifically, JuliaRegistrator will create a new PR to the Julia registry with the new release information. After a grace period of ~15 minutes, this PR will be merged automatically. A short while after, TagBot will create a new release of TrixiParticles.jl in our GitHub repository.
  7. Once the new release has been created, the new version can be obtained through the Julia package manager as usual.
  8. To make sure people do not mistake the latest state of main as the latest release, we set the version in the Project.toml to a development version. The development version should be the latest released version, with the patch version incremented by one, and the -dev suffix added. For example, if you just released v0.3.0, the new development version should be v0.3.1-dev. If you just released v0.2.4, the new development version should be v0.2.5-dev.
diff --git a/previews/PR514/examples/index.html b/previews/PR514/examples/index.html index 84f83b718..a57852273 100644 --- a/previews/PR514/examples/index.html +++ b/previews/PR514/examples/index.html @@ -1,2 +1,2 @@ -Examples · TrixiParticles.jl

Examples

Fluid

Accelerated Tank 2D (fluid/accelerated_tank_2d.jl)

Dam Break 2D (fluid/dam_break_2d.jl)

Dam Break 3D (fluid/dam_break_3d.jl)

Falling Water Column (fluid/falling_water_column_2d.jl)

Hydrostatic Water Column (fluid/hydrostatic_water_column_*.jl)

Moving Wall (fluid/moving_wall_2d.jl)

Oscillating Drop (fluid/oscillating_drop_2d.jl)

Periodic Channel (fluid/periodic_channel_2d.jl)

Fluid Structure Interaction

Dam Break with Elastic Plate (fsi/dam_break_plate_2d.jl)

Falling Sphere 2D (fsi/falling_sphere_2d.jl)

Falling Spheres 2D (fsi/falling_spheres_2d.jl)

Structure Mechanics

Oscillating Beam (solid/oscillating_beam_2d.jl)

+Examples · TrixiParticles.jl

Examples

Fluid

Accelerated Tank 2D (fluid/accelerated_tank_2d.jl)

Dam Break 2D (fluid/dam_break_2d.jl)

Dam Break 3D (fluid/dam_break_3d.jl)

Falling Water Column (fluid/falling_water_column_2d.jl)

Hydrostatic Water Column (fluid/hydrostatic_water_column_*.jl)

Moving Wall (fluid/moving_wall_2d.jl)

Oscillating Drop (fluid/oscillating_drop_2d.jl)

Periodic Channel (fluid/periodic_channel_2d.jl)

Fluid Structure Interaction

Dam Break with Elastic Plate (fsi/dam_break_plate_2d.jl)

Falling Sphere 2D (fsi/falling_sphere_2d.jl)

Falling Spheres 2D (fsi/falling_spheres_2d.jl)

Structure Mechanics

Oscillating Beam (solid/oscillating_beam_2d.jl)

diff --git a/previews/PR514/general/density_calculators/index.html b/previews/PR514/general/density_calculators/index.html index 069eedd70..efcd8d2fd 100644 --- a/previews/PR514/general/density_calculators/index.html +++ b/previews/PR514/general/density_calculators/index.html @@ -1,2 +1,2 @@ -Density Calculators · TrixiParticles.jl

Density Calculators

TrixiParticles.ContinuityDensityType
ContinuityDensity()

Density calculator to integrate the density from the continuity equation

\[\frac{\mathrm{d}\rho_a}{\mathrm{d}t} = \sum_{b} m_b v_{ab} \cdot \nabla_{r_a} W(\Vert r_a - r_b \Vert, h),\]

where $\rho_a$ denotes the density of particle $a$ and $r_{ab} = r_a - r_b$ is the difference of the coordinates, $v_{ab} = v_a - v_b$ of the velocities of particles $a$ and $b$.

source
TrixiParticles.SummationDensityType
SummationDensity()

Density calculator to use the summation formula

\[\rho(r) = \sum_{b} m_b W(\Vert r - r_b \Vert, h),\]

for the density estimation, where $r_b$ denotes the coordinates and $m_b$ the mass of particle $b$.

source
+Density Calculators · TrixiParticles.jl

Density Calculators

TrixiParticles.ContinuityDensityType
ContinuityDensity()

Density calculator to integrate the density from the continuity equation

\[\frac{\mathrm{d}\rho_a}{\mathrm{d}t} = \sum_{b} m_b v_{ab} \cdot \nabla_{r_a} W(\Vert r_a - r_b \Vert, h),\]

where $\rho_a$ denotes the density of particle $a$ and $r_{ab} = r_a - r_b$ is the difference of the coordinates, $v_{ab} = v_a - v_b$ of the velocities of particles $a$ and $b$.

source
TrixiParticles.SummationDensityType
SummationDensity()

Density calculator to use the summation formula

\[\rho(r) = \sum_{b} m_b W(\Vert r - r_b \Vert, h),\]

for the density estimation, where $r_b$ denotes the coordinates and $m_b$ the mass of particle $b$.

source
diff --git a/previews/PR514/general/initial_condition/index.html b/previews/PR514/general/initial_condition/index.html index a378a1d6c..858c78a33 100644 --- a/previews/PR514/general/initial_condition/index.html +++ b/previews/PR514/general/initial_condition/index.html @@ -28,13 +28,13 @@ initial_condition = InitialCondition(; coordinates, velocity, mass, density) # With functions -initial_condition = InitialCondition(; coordinates, velocity=x -> 2x, mass=1.0, density=1000.0)source

Setups

TrixiParticles.ComplexShapeMethod
ComplexShape(geometry::Union{TriangleMesh, Polygon}; particle_spacing, density,
+initial_condition = InitialCondition(; coordinates, velocity=x -> 2x, mass=1.0, density=1000.0)
source

Setups

TrixiParticles.ComplexShapeMethod
ComplexShape(geometry::Union{TriangleMesh, Polygon}; particle_spacing, density,
              pressure=0.0, mass=nothing, velocity=zeros(ndims(geometry)),
              point_in_geometry_algorithm=WindingNumberJacobson(; geometry,
                                                                hierarchical_winding=false,
                                                                winding_number_factor=sqrt(eps())),
              grid_offset::Real=0.0, max_nparticles=10^7,
-             pad_initial_particle_grid=2particle_spacing)

Sample a complex geometry with particles. Returns an InitialCondition. Note that an initial particle grid is generated inside the bounding box of the geometry. A point_in_geometry_algorithm checks if particles are inside the geometry or not. For more information about the method see WindingNumberJacobson or WindingNumberHormann.

Arguments

Keywords

  • particle_spacing: Spacing between the particles.
  • density: Either a function mapping each particle's coordinates to its density, or a scalar for a constant density over all particles.
  • velocity: Either a function mapping each particle's coordinates to its velocity, or, for a constant fluid velocity, a vector holding this velocity. Velocity is constant zero by default.
  • mass: Either nothing (default) to automatically compute particle mass from particle density and spacing, or a function mapping each particle's coordinates to its mass, or a scalar for a constant mass over all particles.
  • pressure: Scalar to set the pressure of all particles to this value. This is only used by the EntropicallyDampedSPHSystem and will be overwritten when using an initial pressure function in the system.
  • point_in_geometry_algorithm: Algorithm for sampling the complex geometry with particles. It basically checks whether a particle is inside an object or not. For more information see WindingNumberJacobson or WindingNumberHormann
  • grid_offset: Offset of the initial particle grid of the bounding box of the geometry.
  • max_nparticles: Maximum number of particles in the initial particle grid. This is only used to avoid accidentally choosing a particle_spacing that is too small for the scale of the geometry.
  • pad_initial_particle_grid: Padding of the initial particle grid.
Experimental Implementation

This is an experimental feature and may change in any future releases.

source
TrixiParticles.extrude_geometryMethod
extrude_geometry(geometry; particle_spacing, direction, n_extrude::Integer,
+             pad_initial_particle_grid=2particle_spacing)

Sample a complex geometry with particles. Returns an InitialCondition. Note that an initial particle grid is generated inside the bounding box of the geometry. A point_in_geometry_algorithm checks if particles are inside the geometry or not. For more information about the method see WindingNumberJacobson or WindingNumberHormann.

Arguments

Keywords

  • particle_spacing: Spacing between the particles.
  • density: Either a function mapping each particle's coordinates to its density, or a scalar for a constant density over all particles.
  • velocity: Either a function mapping each particle's coordinates to its velocity, or, for a constant fluid velocity, a vector holding this velocity. Velocity is constant zero by default.
  • mass: Either nothing (default) to automatically compute particle mass from particle density and spacing, or a function mapping each particle's coordinates to its mass, or a scalar for a constant mass over all particles.
  • pressure: Scalar to set the pressure of all particles to this value. This is only used by the EntropicallyDampedSPHSystem and will be overwritten when using an initial pressure function in the system.
  • point_in_geometry_algorithm: Algorithm for sampling the complex geometry with particles. It basically checks whether a particle is inside an object or not. For more information see WindingNumberJacobson or WindingNumberHormann
  • grid_offset: Offset of the initial particle grid of the bounding box of the geometry.
  • max_nparticles: Maximum number of particles in the initial particle grid. This is only used to avoid accidentally choosing a particle_spacing that is too small for the scale of the geometry.
  • pad_initial_particle_grid: Padding of the initial particle grid.
Experimental Implementation

This is an experimental feature and may change in any future releases.

source
TrixiParticles.extrude_geometryMethod
extrude_geometry(geometry; particle_spacing, direction, n_extrude::Integer,
                  velocity=zeros(length(direction)),
                  mass=nothing, density=nothing, pressure=0.0)

Extrude either a line, a plane or a shape along a specific direction. Returns an InitialCondition.

Arguments

  • geometry: Either particle coordinates or an InitialCondition defining a 2D shape to extrude to a 3D volume, or two 2D points $(A, B)$ defining the interval $[A, B]$ to extrude to a plane in 2D, or three 3D points $(A, B, C)$ defining the parallelogram spanned by the vectors $\widehat{AB}$ and $\widehat {AC}$ to extrude to a parallelepiped.

Keywords

  • particle_spacing: Spacing between the particles. Can be omitted when geometry is an InitialCondition (unless geometry.particle_spacing == -1).
  • direction: A vector that specifies the direction in which to extrude.
  • n_extrude: Number of layers of particles created in the direction of extrusion.
  • velocity: Either a function mapping each particle's coordinates to its velocity, or, for a constant fluid velocity, a vector holding this velocity. Velocity is constant zero by default.
  • mass: Either nothing (default) to automatically compute particle mass from particle density and spacing, or a function mapping each particle's coordinates to its mass, or a scalar for a constant mass over all particles.
  • density: Either a function mapping each particle's coordinates to its density, or a scalar for a constant density over all particles.
  • pressure: Scalar to set the pressure of all particles to this value. This is only used by the EntropicallyDampedSPHSystem and will be overwritten when using an initial pressure function in the system.
  • tlsph: With the TotalLagrangianSPHSystem, particles need to be placed on the boundary of the shape and not one particle radius away, as for fluids. When tlsph=true, particles will be placed on the boundary of the shape.

Examples

# Extrude a line in 2D to a plane in 2D
 p1 = [0.0, 0.0]
@@ -59,7 +59,7 @@
 
 direction = [0.0, 0.0, 1.0]
 
-shape = extrude_geometry(shape; direction, particle_spacing=0.1, n_extrude=4, density=1000.0)
Experimental Implementation

This is an experimental feature and may change in any future releases.

source
TrixiParticles.RectangularShapeMethod
RectangularShape(particle_spacing, n_particles_per_dimension, min_coordinates;
+shape = extrude_geometry(shape; direction, particle_spacing=0.1, n_extrude=4, density=1000.0)
Experimental Implementation

This is an experimental feature and may change in any future releases.

source
TrixiParticles.RectangularShapeMethod
RectangularShape(particle_spacing, n_particles_per_dimension, min_coordinates;
                  velocity=zeros(length(n_particles_per_dimension)),
                  mass=nothing, density=nothing, pressure=0.0,
                  acceleration=nothing, state_equation=nothing,
@@ -73,7 +73,7 @@
                                acceleration=(0.0, -9.81), state_equation=state_equation)
 
 # 3D
-rectangular = RectangularShape(particle_spacing, (5, 4, 7), (1.0, 2.0, 3.0), density=1000.0)
source
TrixiParticles.RectangularTankType
RectangularTank(particle_spacing, fluid_size, tank_size, fluid_density;
+rectangular = RectangularShape(particle_spacing, (5, 4, 7), (1.0, 2.0, 3.0), density=1000.0)
source
TrixiParticles.RectangularTankType
RectangularTank(particle_spacing, fluid_size, tank_size, fluid_density;
                 velocity=zeros(length(fluid_size)), fluid_mass=nothing,
                 pressure=0.0,
                 acceleration=nothing, state_equation=nothing,
@@ -95,7 +95,7 @@
 # 3D
 setup = RectangularTank(particle_spacing, (water_width, water_height, water_depth),
                         (container_width, container_height, container_depth), fluid_density,
-                        n_layers=2)

See also: reset_wall!.

source
TrixiParticles.reset_wall!Method
reset_wall!(rectangular_tank::RectangularTank, reset_faces, positions)

The selected walls of the tank will be placed at the new positions.

Arguments

  • reset_faces: Boolean tuple of 4 (in 2D) or 6 (in 3D) dimensions, similar to faces in RectangularTank.
  • positions: Tuple of new positions
Warning

There are overlapping particles when adjacent walls are moved inwards simultaneously.

source
TrixiParticles.RoundSphereType
RoundSphere(; start_angle=0.0, end_angle=2π)

Construct a sphere (or sphere segment) by nesting perfectly round concentric spheres. The resulting ball will be perfectly round, but will not have a regular inner structure.

Keywords

  • start_angle: The starting angle of the sphere segment in radians. It determines the beginning point of the segment. The default is set to 0.0 representing the positive x-axis.
  • end_angle: The ending angle of the sphere segment in radians. It defines the termination point of the segment. The default is set to 2pi, completing a full sphere.
Usage

See SphereShape on how to use this.

Warning

The sphere segment is intended for 2D geometries and hollow spheres. If used for filled spheres or in a 3D context, results may not be accurate.

source
TrixiParticles.VoxelSphereType
VoxelSphere()

Construct a sphere of voxels (where particles are placed in the voxel center) with a regular inner structure but corners on the surface. Essentially, a grid of particles is generated and all particles outside the sphere are removed. The resulting sphere will have a perfect inner structure, but is not perfectly round, as it will have corners (like a sphere in Minecraft).

Usage

See SphereShape on how to use this.

source
TrixiParticles.SphereShapeMethod
SphereShape(particle_spacing, radius, center_position, density;
+                        n_layers=2)

See also: reset_wall!.

source
TrixiParticles.reset_wall!Method
reset_wall!(rectangular_tank::RectangularTank, reset_faces, positions)

The selected walls of the tank will be placed at the new positions.

Arguments

  • reset_faces: Boolean tuple of 4 (in 2D) or 6 (in 3D) dimensions, similar to faces in RectangularTank.
  • positions: Tuple of new positions
Warning

There are overlapping particles when adjacent walls are moved inwards simultaneously.

source
TrixiParticles.RoundSphereType
RoundSphere(; start_angle=0.0, end_angle=2π)

Construct a sphere (or sphere segment) by nesting perfectly round concentric spheres. The resulting ball will be perfectly round, but will not have a regular inner structure.

Keywords

  • start_angle: The starting angle of the sphere segment in radians. It determines the beginning point of the segment. The default is set to 0.0 representing the positive x-axis.
  • end_angle: The ending angle of the sphere segment in radians. It defines the termination point of the segment. The default is set to 2pi, completing a full sphere.
Usage

See SphereShape on how to use this.

Warning

The sphere segment is intended for 2D geometries and hollow spheres. If used for filled spheres or in a 3D context, results may not be accurate.

source
TrixiParticles.VoxelSphereType
VoxelSphere()

Construct a sphere of voxels (where particles are placed in the voxel center) with a regular inner structure but corners on the surface. Essentially, a grid of particles is generated and all particles outside the sphere are removed. The resulting sphere will have a perfect inner structure, but is not perfectly round, as it will have corners (like a sphere in Minecraft).

Usage

See SphereShape on how to use this.

source
TrixiParticles.SphereShapeMethod
SphereShape(particle_spacing, radius, center_position, density;
             sphere_type=VoxelSphere(), n_layers=-1, layer_outwards=false,
             cutout_min=(0.0, 0.0), cutout_max=(0.0, 0.0), tlsph=false,
             velocity=zeros(length(center_position)), mass=nothing, pressure=0.0)

Generate a sphere that is either completely filled (by default) or hollow (by passing n_layers).

With the sphere type VoxelSphere, a sphere of voxels (where particles are placed in the voxel center) with a regular inner structure but corners on the surface is created. Essentially, a grid of particles is generated and all particles outside the sphere are removed. With the sphere type RoundSphere, a perfectly round sphere with an imperfect inner structure is created.

A cuboid can be cut out of the sphere by specifying the two corners in negative and positive coordinate directions as cutout_min and cutout_max.

Arguments

  • particle_spacing: Spacing between the particles.
  • radius: Radius of the sphere.
  • center_position: The coordinates of the center of the sphere.
  • density: Either a function mapping each particle's coordinates to its density, or a scalar for a constant density over all particles.

Keywords

  • sphere_type: Either VoxelSphere or RoundSphere (see explanation above).
  • n_layers: Set to an integer greater than zero to generate a hollow sphere, where the shell consists of n_layers layers.
  • layer_outwards: When set to false (by default), radius is the outer radius of the sphere. When set to true, radius is the inner radius of the sphere. This is only used when n_layers > 0.
  • cutout_min: Corner in negative coordinate directions of a cuboid that is to be cut out of the sphere.
  • cutout_max: Corner in positive coordinate directions of a cuboid that is to be cut out of the sphere.
  • tlsph: With the TotalLagrangianSPHSystem, particles need to be placed on the boundary of the shape and not one particle radius away, as for fluids. When tlsph=true, particles will be placed on the boundary of the shape.
  • velocity: Either a function mapping each particle's coordinates to its velocity, or, for a constant fluid velocity, a vector holding this velocity. Velocity is constant zero by default.
  • mass: Either nothing (default) to automatically compute particle mass from particle density and spacing, or a function mapping each particle's coordinates to its mass, or a scalar for a constant mass over all particles.
  • pressure: Either a function mapping each particle's coordinates to its pressure, or a scalar for a constant pressure over all particles. This is optional and only needed when using the EntropicallyDampedSPHSystem.

Examples

# Filled circle with radius 0.5, center in (0.2, 0.4) and a particle spacing of 0.1
@@ -123,4 +123,4 @@
 SphereShape(0.1, 0.5, (0.2, 0.4, 0.3), 1000.0)
 
 # Same as before, but perfectly round
-SphereShape(0.1, 0.5, (0.2, 0.4, 0.3), 1000.0, sphere_type=RoundSphere())
source
+SphereShape(0.1, 0.5, (0.2, 0.4, 0.3), 1000.0, sphere_type=RoundSphere())source diff --git a/previews/PR514/general/interpolation/index.html b/previews/PR514/general/interpolation/index.html index 7e750d499..4c19d503e 100644 --- a/previews/PR514/general/interpolation/index.html +++ b/previews/PR514/general/interpolation/index.html @@ -2,17 +2,17 @@ Interpolation · TrixiParticles.jl

Interpolation

TrixiParticles.interpolate_lineMethod
interpolate_line(start, end_, n_points, semi, ref_system, sol; endpoint=true,
                  smoothing_length=ref_system.smoothing_length, cut_off_bnd=true,
                  clip_negative_pressure=false)

Interpolates properties along a line in a TrixiParticles simulation. The line interpolation is accomplished by generating a series of evenly spaced points between start and end_. If endpoint is false, the line is interpolated between the start and end points, but does not include these points.

See also: interpolate_point, interpolate_plane_2d, interpolate_plane_2d_vtk, interpolate_plane_3d.

Arguments

  • start: The starting point of the line.
  • end_: The ending point of the line.
  • n_points: The number of points to interpolate along the line.
  • semi: The semidiscretization used for the simulation.
  • ref_system: The reference system for the interpolation.
  • sol: The solution state from which the properties are interpolated.

Keywords

  • endpoint=true: A boolean to include (true) or exclude (false) the end point in the interpolation.
  • smoothing_length=ref_system.smoothing_length: The smoothing length used in the interpolation.
  • cut_off_bnd=true: Boolean to indicate if quantities should be set to NaN when the point is "closer" to the boundary than to the fluid in a kernel-weighted sense. Or, in more detail, when the boundary has more influence than the fluid on the density summation in this point, i.e., when the boundary particles add more kernel-weighted mass than the fluid particles.
  • clip_negative_pressure=false: One common approach in SPH models is to clip negative pressure values, but this is unphysical. Instead we clip here during interpolation thus only impacting the local interpolated value.

Returns

  • A NamedTuple of arrays containing interpolated properties at each point along the line.
Note
  • This function is particularly useful for analyzing gradients or creating visualizations along a specified line in the SPH simulation domain.
  • The interpolation accuracy is subject to the density of particles and the chosen smoothing length.
  • With cut_off_bnd, a density-based estimation of the surface is used which is not as accurate as a real surface reconstruction.

Examples

# Interpolating along a line from [1.0, 0.0] to [1.0, 1.0] with 5 points
-results = interpolate_line([1.0, 0.0], [1.0, 1.0], 5, semi, ref_system, sol)
source
TrixiParticles.interpolate_plane_2dMethod
interpolate_plane_2d(min_corner, max_corner, resolution, semi, ref_system, sol;
+results = interpolate_line([1.0, 0.0], [1.0, 1.0], 5, semi, ref_system, sol)
source
TrixiParticles.interpolate_plane_2dMethod
interpolate_plane_2d(min_corner, max_corner, resolution, semi, ref_system, sol;
                      smoothing_length=ref_system.smoothing_length, cut_off_bnd=true,
                      clip_negative_pressure=false)

Interpolates properties along a plane in a TrixiParticles simulation. The region for interpolation is defined by its lower left and top right corners, with a specified resolution determining the density of the interpolation points.

The function generates a grid of points within the defined region, spaced uniformly according to the given resolution.

See also: interpolate_plane_2d_vtk, interpolate_plane_3d, interpolate_line, interpolate_point.

Arguments

  • min_corner: The lower left corner of the interpolation region.
  • max_corner: The top right corner of the interpolation region.
  • resolution: The distance between adjacent interpolation points in the grid.
  • semi: The semidiscretization used for the simulation.
  • ref_system: The reference system for the interpolation.
  • sol: The solution state from which the properties are interpolated.

Keywords

  • smoothing_length=ref_system.smoothing_length: The smoothing length used in the interpolation.
  • cut_off_bnd=true: Boolean to indicate if quantities should be set to NaN when the point is "closer" to the boundary than to the fluid in a kernel-weighted sense. Or, in more detail, when the boundary has more influence than the fluid on the density summation in this point, i.e., when the boundary particles add more kernel-weighted mass than the fluid particles.
  • clip_negative_pressure=false: One common approach in SPH models is to clip negative pressure values, but this is unphysical. Instead we clip here during interpolation thus only impacting the local interpolated value.

Returns

  • A NamedTuple of arrays containing interpolated properties at each point within the plane.
Note
  • The interpolation accuracy is subject to the density of particles and the chosen smoothing length.
  • With cut_off_bnd, a density-based estimation of the surface is used, which is not as accurate as a real surface reconstruction.

Examples

# Interpolating across a plane from [0.0, 0.0] to [1.0, 1.0] with a resolution of 0.2
-results = interpolate_plane_2d([0.0, 0.0], [1.0, 1.0], 0.2, semi, ref_system, sol)
source
TrixiParticles.interpolate_plane_2d_vtkMethod
interpolate_plane_2d_vtk(min_corner, max_corner, resolution, semi, ref_system, sol;
                          smoothing_length=ref_system.smoothing_length, cut_off_bnd=true,
                          clip_negative_pressure=false, output_directory="out", filename="plane")

Interpolates properties along a plane in a TrixiParticles simulation and exports the result as a VTI file. The region for interpolation is defined by its lower left and top right corners, with a specified resolution determining the density of the interpolation points.

The function generates a grid of points within the defined region, spaced uniformly according to the given resolution.

See also: interpolate_plane_2d, interpolate_plane_3d, interpolate_line, interpolate_point.

Arguments

  • min_corner: The lower left corner of the interpolation region.
  • max_corner: The top right corner of the interpolation region.
  • resolution: The distance between adjacent interpolation points in the grid.
  • semi: The semidiscretization used for the simulation.
  • ref_system: The reference system for the interpolation.
  • sol: The solution state from which the properties are interpolated.

Keywords

  • smoothing_length=ref_system.smoothing_length: The smoothing length used in the interpolation.
  • output_directory="out": Directory to save the VTI file.
  • filename="plane": Name of the VTI file.
  • cut_off_bnd=true: Boolean to indicate if quantities should be set to NaN when the point is "closer" to the boundary than to the fluid in a kernel-weighted sense. Or, in more detail, when the boundary has more influence than the fluid on the density summation in this point, i.e., when the boundary particles add more kernel-weighted mass than the fluid particles.
  • clip_negative_pressure=false: One common approach in SPH models is to clip negative pressure values, but this is unphysical. Instead we clip here during interpolation thus only impacting the local interpolated value.
Note
  • The interpolation accuracy is subject to the density of particles and the chosen smoothing length.
  • With cut_off_bnd, a density-based estimation of the surface is used, which is not as accurate as a real surface reconstruction.

Examples

# Interpolating across a plane from [0.0, 0.0] to [1.0, 1.0] with a resolution of 0.2
-results = interpolate_plane_2d([0.0, 0.0], [1.0, 1.0], 0.2, semi, ref_system, sol)
source
TrixiParticles.interpolate_plane_3dMethod
interpolate_plane_3d(point1, point2, point3, resolution, semi, ref_system, sol;
+results = interpolate_plane_2d([0.0, 0.0], [1.0, 1.0], 0.2, semi, ref_system, sol)
source
TrixiParticles.interpolate_plane_3dMethod
interpolate_plane_3d(point1, point2, point3, resolution, semi, ref_system, sol;
                      smoothing_length=ref_system.smoothing_length, cut_off_bnd=true,
                      clip_negative_pressure=false)

Interpolates properties along a plane in a 3D space in a TrixiParticles simulation. The plane for interpolation is defined by three points in 3D space, with a specified resolution determining the density of the interpolation points.

The function generates a grid of points on a parallelogram within the plane defined by the three points, spaced uniformly according to the given resolution.

See also: interpolate_plane_2d, interpolate_plane_2d_vtk, interpolate_line, interpolate_point.

Arguments

  • point1: The first point defining the plane.
  • point2: The second point defining the plane.
  • point3: The third point defining the plane. The points must not be collinear.
  • resolution: The distance between adjacent interpolation points in the grid.
  • semi: The semidiscretization used for the simulation.
  • ref_system: The reference system for the interpolation.
  • sol: The solution state from which the properties are interpolated.

Keywords

  • smoothing_length=ref_system.smoothing_length: The smoothing length used in the interpolation.
  • cut_off_bnd=true: Boolean to indicate if quantities should be set to NaN when the point is "closer" to the boundary than to the fluid in a kernel-weighted sense. Or, in more detail, when the boundary has more influence than the fluid on the density summation in this point, i.e., when the boundary particles add more kernel-weighted mass than the fluid particles.
  • clip_negative_pressure=false: One common approach in SPH models is to clip negative pressure values, but this is unphysical. Instead we clip here during interpolation thus only impacting the local interpolated value.

Returns

  • A NamedTuple of arrays containing interpolated properties at each point within the plane.
Note
  • The interpolation accuracy is subject to the density of particles and the chosen smoothing length.
  • With cut_off_bnd, a density-based estimation of the surface is used which is not as accurate as a real surface reconstruction.

Examples

# Interpolating across a plane defined by points [0.0, 0.0, 0.0], [1.0, 0.0, 0.0], and [0.0, 1.0, 0.0]
 # with a resolution of 0.1
-results = interpolate_plane_3d([0.0, 0.0, 0.0], [1.0, 0.0, 0.0], [0.0, 1.0, 0.0], 0.1, semi, ref_system, sol)
source
TrixiParticles.interpolate_pointMethod
interpolate_point(points_coords::Array{Array{Float64,1},1}, semi, ref_system, sol;
+results = interpolate_plane_3d([0.0, 0.0, 0.0], [1.0, 0.0, 0.0], [0.0, 1.0, 0.0], 0.1, semi, ref_system, sol)
source
TrixiParticles.interpolate_pointMethod
interpolate_point(points_coords::Array{Array{Float64,1},1}, semi, ref_system, sol;
                   smoothing_length=ref_system.smoothing_length, cut_off_bnd=true,
                   clip_negative_pressure=false)
 
@@ -23,4 +23,4 @@
 
 # For multiple points
 points = [[1.0, 0.5], [1.0, 0.6], [1.0, 0.7]]
-results = interpolate_point(points, semi, ref_system, sol)
Note
  • This function is particularly useful for analyzing gradients or creating visualizations along a specified line in the SPH simulation domain.
  • The interpolation accuracy is subject to the density of particles and the chosen smoothing length.
  • With cut_off_bnd, a density-based estimation of the surface is used which is not as

accurate as a real surface reconstruction.

source
+results = interpolate_point(points, semi, ref_system, sol)
Note

accurate as a real surface reconstruction.

source diff --git a/previews/PR514/general/neighborhood_search/index.html b/previews/PR514/general/neighborhood_search/index.html index 5b265a779..539d65936 100644 --- a/previews/PR514/general/neighborhood_search/index.html +++ b/previews/PR514/general/neighborhood_search/index.html @@ -2,4 +2,4 @@ Neighborhood Search · TrixiParticles.jl

Neighborhood Search

The neighborhood search is the most essential component for performance. We provide several implementations in the package PointNeighbors.jl. See the docs of this package for an overview and a comparison of different implementations.

Usage

To run a simulation with a neighborhood search implementation, pass a template of the neighborhood search to the constructor of the Semidiscretization. A template is just an empty neighborhood search with search radius 0.0. See copy_neighborhood_search and the examples below for more details.

semi = Semidiscretization(system1, system2,
                           neighborhood_search=PrecomputedNeighborhoodSearch{2}())

The keyword argument periodic_box in the neighborhood search constructors can be used to define a periodic domain. See the PointNeighbors.jl docs for more details.

periodic_box = PeriodicBox(min_corner=[0.0, -0.25], max_corner=[1.0, 0.75])
 semi = Semidiscretization(system1, system2,
-                          neighborhood_search=GridNeighborhoodSearch{2}(; periodic_box))
+ neighborhood_search=GridNeighborhoodSearch{2}(; periodic_box)) diff --git a/previews/PR514/general/semidiscretization/index.html b/previews/PR514/general/semidiscretization/index.html index 5c407d068..f27b2e205 100644 --- a/previews/PR514/general/semidiscretization/index.html +++ b/previews/PR514/general/semidiscretization/index.html @@ -12,6 +12,6 @@ neighborhood_search=PrecomputedNeighborhoodSearch{2}()) semi = Semidiscretization(fluid_system, boundary_system, - neighborhood_search=nothing)source
TrixiParticles.SourceTermDampingType
SourceTermDamping(; damping_coefficient)

A source term to be used when a damping step is required before running a full simulation. The term $-c \cdot v_a$ is added to the acceleration $\frac{\mathrm{d}v_a}{\mathrm{d}t}$ of particle $a$, where $c$ is the damping coefficient and $v_a$ is the velocity of particle $a$.

Keywords

  • damping_coefficient: The coefficient $d$ above. A higher coefficient means more damping. A coefficient of 1e-4 is a good starting point for damping a fluid at rest.

Examples

source_terms = SourceTermDamping(; damping_coefficient=1e-4)
source
TrixiParticles.restart_with!Method
restart_with!(semi, sol)

Set the initial coordinates and velocities of all systems in semi to the final values in the solution sol. semidiscretize has to be called again afterwards, or another Semidiscretization can be created with the updated systems.

Arguments

  • semi: The semidiscretization
  • sol: The ODESolution returned by solve of OrdinaryDiffEq
source
TrixiParticles.semidiscretizeMethod
semidiscretize(semi, tspan; reset_threads=true)

Create an ODEProblem from the semidiscretization with the specified tspan.

Arguments

  • semi: A Semidiscretization holding the systems involved in the simulation.
  • tspan: The time span over which the simulation will be run.

Keywords

  • reset_threads: A boolean flag to reset Polyester.jl threads before the simulation (default: true). After an error within a threaded loop, threading might be disabled. Resetting the threads before the simulation ensures that threading is enabled again for the simulation. See also trixi-framework/Trixi.jl#1583.

Returns

A DynamicalODEProblem (see the OrdinaryDiffEq.jl docs) to be integrated with OrdinaryDiffEq.jl. Note that this is not a true DynamicalODEProblem where the acceleration does not depend on the velocity. Therefore, not all integrators designed for DynamicalODEProblems will work properly. However, all integrators designed for ODEProblems can be used.

Examples

semi = Semidiscretization(fluid_system, boundary_system)
+                          neighborhood_search=nothing)
source
TrixiParticles.SourceTermDampingType
SourceTermDamping(; damping_coefficient)

A source term to be used when a damping step is required before running a full simulation. The term $-c \cdot v_a$ is added to the acceleration $\frac{\mathrm{d}v_a}{\mathrm{d}t}$ of particle $a$, where $c$ is the damping coefficient and $v_a$ is the velocity of particle $a$.

Keywords

  • damping_coefficient: The coefficient $d$ above. A higher coefficient means more damping. A coefficient of 1e-4 is a good starting point for damping a fluid at rest.

Examples

source_terms = SourceTermDamping(; damping_coefficient=1e-4)
source
TrixiParticles.restart_with!Method
restart_with!(semi, sol)

Set the initial coordinates and velocities of all systems in semi to the final values in the solution sol. semidiscretize has to be called again afterwards, or another Semidiscretization can be created with the updated systems.

Arguments

  • semi: The semidiscretization
  • sol: The ODESolution returned by solve of OrdinaryDiffEq
source
TrixiParticles.semidiscretizeMethod
semidiscretize(semi, tspan; reset_threads=true)

Create an ODEProblem from the semidiscretization with the specified tspan.

Arguments

  • semi: A Semidiscretization holding the systems involved in the simulation.
  • tspan: The time span over which the simulation will be run.

Keywords

  • reset_threads: A boolean flag to reset Polyester.jl threads before the simulation (default: true). After an error within a threaded loop, threading might be disabled. Resetting the threads before the simulation ensures that threading is enabled again for the simulation. See also trixi-framework/Trixi.jl#1583.

Returns

A DynamicalODEProblem (see the OrdinaryDiffEq.jl docs) to be integrated with OrdinaryDiffEq.jl. Note that this is not a true DynamicalODEProblem where the acceleration does not depend on the velocity. Therefore, not all integrators designed for DynamicalODEProblems will work properly. However, all integrators designed for ODEProblems can be used.

Examples

semi = Semidiscretization(fluid_system, boundary_system)
 tspan = (0.0, 1.0)
-ode_problem = semidiscretize(semi, tspan)
source
+ode_problem = semidiscretize(semi, tspan)source diff --git a/previews/PR514/general/smoothing_kernels/index.html b/previews/PR514/general/smoothing_kernels/index.html index 6c356ecb7..a4d02edb8 100644 --- a/previews/PR514/general/smoothing_kernels/index.html +++ b/previews/PR514/general/smoothing_kernels/index.html @@ -1,33 +1,33 @@ -Smoothing Kernels · TrixiParticles.jl

Smoothing Kernels

The following smoothing kernels are currently available:

Smoothing KernelCompact SupportTyp. Smoothing LengthRecommended ApplicationStability
SchoenbergCubicSplineKernel$[0, 2h]$$1.1$ to $1.3$General + sharp waves++
SchoenbergQuarticSplineKernel$[0, 2.5h]$$1.1$ to $1.5$General+++
SchoenbergQuinticSplineKernel$[0, 3h]$$1.1$ to $1.5$General++++
GaussianKernel$[0, 3h]$$1.0$ to $1.5$Literature+++++
WendlandC2Kernel$[0, 1h]$$2.5$ to $4.0$General (recommended)++++
WendlandC4Kernel$[0, 1h]$$3.0$ to $4.5$General+++++
WendlandC6Kernel$[0, 1h]$$3.5$ to $5.0$General+++++
Poly6Kernel$[0, 1h]$$1.5$ to $2.5$Literature+
SpikyKernel$[0, 1h]$$1.5$ to $3.0$Sharp corners + waves+

We recommend to use the WendlandC2Kernel for most applications. If less smoothing is needed, try SchoenbergCubicSplineKernel, for more smoothing try WendlandC6Kernel.

Usage

The kernel can be called as

TrixiParticles.kernel(smoothing_kernel, r, h)

The length of the compact support can be obtained as

TrixiParticles.compact_support(smoothing_kernel, h)

Note that $r$ has to be a scalar, so in the context of SPH, the kernel should be used as

\[W(\Vert r_a - r_b \Vert, h).\]

The gradient required in SPH,

\[ \nabla_{r_a} W(\Vert r_a - r_b \Vert, h)\]

can be called as

TrixiParticles.kernel_grad(smoothing_kernel, pos_diff, distance, h)

where pos_diff is $r_a - r_b$ and distance is $\Vert r_a - r_b \Vert$.

TrixiParticles.GaussianKernelType
GaussianKernel{NDIMS}()

Gaussian kernel given by

\[W(r, h) = \frac{\sigma_d}{h^d} e^{-r^2/h^2}\]

where $d$ is the number of dimensions and

  • $\sigma_2 = \frac{1}{\pi}$ for 2D,
  • $\sigma_3 = \frac{1}{\pi^{3/2}}$ for 3D.

This kernel function has an infinite support, but in practice, it's often truncated at a certain multiple of $h$, such as $3h$.

In this implementation, the kernel is truncated at $3h$, so this kernel function has a compact support of $[0, 3h]$.

The smoothing length is typically in the range $[1.0\delta, 1.5\delta]$, where $\delta$ is the typical particle spacing.

For general information and usage see Smoothing Kernels.

Note: This truncation makes this Kernel not conservative, which is beneficial in regards to stability but makes it less accurate.

source
TrixiParticles.Poly6KernelType
Poly6Kernel{NDIMS}()

Poly6 kernel, a commonly used kernel in SPH literature [3], especially in computer graphics contexts. It is defined as

\[W(r, h) = \frac{1}{h^d} w(r/h)\]

with

\[w(q) = \sigma \begin{cases} +Smoothing Kernels · TrixiParticles.jl

Smoothing Kernels

The following smoothing kernels are currently available:

Smoothing KernelCompact SupportTyp. Smoothing LengthRecommended ApplicationStability
SchoenbergCubicSplineKernel$[0, 2h]$$1.1$ to $1.3$General + sharp waves++
SchoenbergQuarticSplineKernel$[0, 2.5h]$$1.1$ to $1.5$General+++
SchoenbergQuinticSplineKernel$[0, 3h]$$1.1$ to $1.5$General++++
GaussianKernel$[0, 3h]$$1.0$ to $1.5$Literature+++++
WendlandC2Kernel$[0, 1h]$$2.5$ to $4.0$General (recommended)++++
WendlandC4Kernel$[0, 1h]$$3.0$ to $4.5$General+++++
WendlandC6Kernel$[0, 1h]$$3.5$ to $5.0$General+++++
Poly6Kernel$[0, 1h]$$1.5$ to $2.5$Literature+
SpikyKernel$[0, 1h]$$1.5$ to $3.0$Sharp corners + waves+

We recommend to use the WendlandC2Kernel for most applications. If less smoothing is needed, try SchoenbergCubicSplineKernel, for more smoothing try WendlandC6Kernel.

Usage

The kernel can be called as

TrixiParticles.kernel(smoothing_kernel, r, h)

The length of the compact support can be obtained as

TrixiParticles.compact_support(smoothing_kernel, h)

Note that $r$ has to be a scalar, so in the context of SPH, the kernel should be used as

\[W(\Vert r_a - r_b \Vert, h).\]

The gradient required in SPH,

\[ \nabla_{r_a} W(\Vert r_a - r_b \Vert, h)\]

can be called as

TrixiParticles.kernel_grad(smoothing_kernel, pos_diff, distance, h)

where pos_diff is $r_a - r_b$ and distance is $\Vert r_a - r_b \Vert$.

TrixiParticles.GaussianKernelType
GaussianKernel{NDIMS}()

Gaussian kernel given by

\[W(r, h) = \frac{\sigma_d}{h^d} e^{-r^2/h^2}\]

where $d$ is the number of dimensions and

  • $\sigma_2 = \frac{1}{\pi}$ for 2D,
  • $\sigma_3 = \frac{1}{\pi^{3/2}}$ for 3D.

This kernel function has an infinite support, but in practice, it's often truncated at a certain multiple of $h$, such as $3h$.

In this implementation, the kernel is truncated at $3h$, so this kernel function has a compact support of $[0, 3h]$.

The smoothing length is typically in the range $[1.0\delta, 1.5\delta]$, where $\delta$ is the typical particle spacing.

For general information and usage see Smoothing Kernels.

Note: This truncation makes this Kernel not conservative, which is beneficial in regards to stability but makes it less accurate.

source
TrixiParticles.Poly6KernelType
Poly6Kernel{NDIMS}()

Poly6 kernel, a commonly used kernel in SPH literature [3], especially in computer graphics contexts. It is defined as

\[W(r, h) = \frac{1}{h^d} w(r/h)\]

with

\[w(q) = \sigma \begin{cases} (1 - q^2)^3 & \text{if } 0 \leq q < 1, \\ 0 & \text{if } q \geq 1, -\end{cases}\]

where $d$ is the number of dimensions and $\sigma$ is a normalization factor that depends on the dimension. The normalization factor $\sigma$ is $4 / \pi$ in two dimensions or $315 / 64\pi$ in three dimensions.

This kernel function has a compact support of $[0, h]$.

Poly6 is well-known for its computational simplicity, though it's worth noting that there are other kernels that might offer better accuracy for hydrodynamic simulations. Furthermore, its derivatives are not that smooth, which can lead to stability problems. It is also susceptible to clumping.

The smoothing length is typically in the range $[1.5\delta, 2.5\delta]$, where $\delta$ is the typical particle spacing.

For general information and usage see Smoothing Kernels.

source
TrixiParticles.SchoenbergCubicSplineKernelType
SchoenbergCubicSplineKernel{NDIMS}()

Cubic spline kernel by Schoenberg (1946), given by

\[ W(r, h) = \frac{1}{h^d} w(r/h)\]

with

\[w(q) = \sigma \begin{cases} +\end{cases}\]

where $d$ is the number of dimensions and $\sigma$ is a normalization factor that depends on the dimension. The normalization factor $\sigma$ is $4 / \pi$ in two dimensions or $315 / 64\pi$ in three dimensions.

This kernel function has a compact support of $[0, h]$.

Poly6 is well-known for its computational simplicity, though it's worth noting that there are other kernels that might offer better accuracy for hydrodynamic simulations. Furthermore, its derivatives are not that smooth, which can lead to stability problems. It is also susceptible to clumping.

The smoothing length is typically in the range $[1.5\delta, 2.5\delta]$, where $\delta$ is the typical particle spacing.

For general information and usage see Smoothing Kernels.

source
TrixiParticles.SchoenbergCubicSplineKernelType
SchoenbergCubicSplineKernel{NDIMS}()

Cubic spline kernel by Schoenberg (1946), given by

\[ W(r, h) = \frac{1}{h^d} w(r/h)\]

with

\[w(q) = \sigma \begin{cases} \frac{1}{4} (2 - q)^3 - (1 - q)^3 & \text{if } 0 \leq q < 1, \\ \frac{1}{4} (2 - q)^3 & \text{if } 1 \leq q < 2, \\ 0 & \text{if } q \geq 2, \\ -\end{cases}\]

where $d$ is the number of dimensions and $\sigma$ is a normalization constant given by $\sigma =[\frac{2}{3}, \frac{10}{7 \pi}, \frac{1}{\pi}]$ in $[1, 2, 3]$ dimensions.

This kernel function has a compact support of $[0, 2h]$.

For an overview of Schoenberg cubic, quartic and quintic spline kernels including normalization factors, see Price (2012). For an analytic formula for higher order Schoenberg kernels, see Monaghan (1985). The largest disadvantage of Schoenberg Spline Kernel is the rather non-smooth first derivative, which can lead to increased noise compared to other kernel variants.

The smoothing length is typically in the range $[1.1\delta, 1.3\delta]$, where $\delta$ is the typical particle spacing.

For general information and usage see Smoothing Kernels.

source
TrixiParticles.SchoenbergQuarticSplineKernelType
SchoenbergQuarticSplineKernel{NDIMS}()

Quartic spline kernel by Schoenberg (1946), given by

\[ W(r, h) = \frac{1}{h^d} w(r/h)\]

with

\[w(q) = \sigma \begin{cases} +\end{cases}\]

where $d$ is the number of dimensions and $\sigma$ is a normalization constant given by $\sigma =[\frac{2}{3}, \frac{10}{7 \pi}, \frac{1}{\pi}]$ in $[1, 2, 3]$ dimensions.

This kernel function has a compact support of $[0, 2h]$.

For an overview of Schoenberg cubic, quartic and quintic spline kernels including normalization factors, see Price (2012). For an analytic formula for higher order Schoenberg kernels, see Monaghan (1985). The largest disadvantage of Schoenberg Spline Kernel is the rather non-smooth first derivative, which can lead to increased noise compared to other kernel variants.

The smoothing length is typically in the range $[1.1\delta, 1.3\delta]$, where $\delta$ is the typical particle spacing.

For general information and usage see Smoothing Kernels.

source
TrixiParticles.SchoenbergQuarticSplineKernelType
SchoenbergQuarticSplineKernel{NDIMS}()

Quartic spline kernel by Schoenberg (1946), given by

\[ W(r, h) = \frac{1}{h^d} w(r/h)\]

with

\[w(q) = \sigma \begin{cases} \left(5/2 - q \right)^4 - 5\left(3/2 - q \right)^4 + 10\left(1/2 - q \right)^4 & \text{if } 0 \leq q < \frac{1}{2}, \\ \left(5/2 - q \right)^4 - 5\left(3/2 - q \right)^4 & \text{if } \frac{1}{2} \leq q < \frac{3}{2}, \\ \left(5/2 - q \right)^4 & \text{if } \frac{3}{2} \leq q < \frac{5}{2}, \\ 0 & \text{if } q \geq \frac{5}{2}, -\end{cases}\]

where $d$ is the number of dimensions and $\sigma$ is a normalization constant given by $\sigma =[\frac{1}{24}, \frac{96}{1199 \pi}, \frac{1}{20\pi}]$ in $[1, 2, 3]$ dimensions.

This kernel function has a compact support of $[0, 2.5h]$.

For an overview of Schoenberg cubic, quartic and quintic spline kernels including normalization factors, see Price (2012). For an analytic formula for higher order Schoenberg kernels, see Monaghan (1985).

The largest disadvantage of Schoenberg Spline Kernel are the rather non-smooth first derivative, which can lead to increased noise compared to other kernel variants.

The smoothing length is typically in the range $[1.1\delta, 1.5\delta]$, where $\delta$ is the typical particle spacing.

For general information and usage see Smoothing Kernels.

source
TrixiParticles.SchoenbergQuinticSplineKernelType
SchoenbergQuinticSplineKernel{NDIMS}()

Quintic spline kernel by Schoenberg (1946), given by

\[ W(r, h) = \frac{1}{h^d} w(r/h)\]

with

\[w(q) = \sigma \begin{cases} +\end{cases}\]

where $d$ is the number of dimensions and $\sigma$ is a normalization constant given by $\sigma =[\frac{1}{24}, \frac{96}{1199 \pi}, \frac{1}{20\pi}]$ in $[1, 2, 3]$ dimensions.

This kernel function has a compact support of $[0, 2.5h]$.

For an overview of Schoenberg cubic, quartic and quintic spline kernels including normalization factors, see Price (2012). For an analytic formula for higher order Schoenberg kernels, see Monaghan (1985).

The largest disadvantage of Schoenberg Spline Kernel are the rather non-smooth first derivative, which can lead to increased noise compared to other kernel variants.

The smoothing length is typically in the range $[1.1\delta, 1.5\delta]$, where $\delta$ is the typical particle spacing.

For general information and usage see Smoothing Kernels.

source
TrixiParticles.SchoenbergQuinticSplineKernelType
SchoenbergQuinticSplineKernel{NDIMS}()

Quintic spline kernel by Schoenberg (1946), given by

\[ W(r, h) = \frac{1}{h^d} w(r/h)\]

with

\[w(q) = \sigma \begin{cases} (3 - q)^5 - 6(2 - q)^5 + 15(1 - q)^5 & \text{if } 0 \leq q < 1, \\ (3 - q)^5 - 6(2 - q)^5 & \text{if } 1 \leq q < 2, \\ (3 - q)^5 & \text{if } 2 \leq q < 3, \\ 0 & \text{if } q \geq 3, -\end{cases}\]

where $d$ is the number of dimensions and $\sigma$ is a normalization constant given by $\sigma =[\frac{1}{120}, \frac{7}{478 \pi}, \frac{1}{120\pi}]$ in $[1, 2, 3]$ dimensions.

This kernel function has a compact support of $[0, 3h]$.

For an overview of Schoenberg cubic, quartic and quintic spline kernels including normalization factors, see Price (2012). For an analytic formula for higher order Schoenberg kernels, see Monaghan (1985).

The largest disadvantage of Schoenberg Spline Kernel are the rather non-smooth first derivative, which can lead to increased noise compared to other kernel variants.

The smoothing length is typically in the range $[1.1\delta, 1.5\delta]$, where $\delta$ is the typical particle spacing.

For general information and usage see Smoothing Kernels.

source
TrixiParticles.SpikyKernelType
SpikyKernel{NDIMS}()

The Spiky kernel is another frequently used kernel in SPH, especially due to its desirable properties in preserving features near boundaries in fluid simulations [3]. It is defined as:

\[ W(r, h) = \frac{1}{h^d} w(r/h)\]

with:

\[w(q) = \sigma \begin{cases} +\end{cases}\]

where $d$ is the number of dimensions and $\sigma$ is a normalization constant given by $\sigma =[\frac{1}{120}, \frac{7}{478 \pi}, \frac{1}{120\pi}]$ in $[1, 2, 3]$ dimensions.

This kernel function has a compact support of $[0, 3h]$.

For an overview of Schoenberg cubic, quartic and quintic spline kernels including normalization factors, see Price (2012). For an analytic formula for higher order Schoenberg kernels, see Monaghan (1985).

The largest disadvantage of Schoenberg Spline Kernel are the rather non-smooth first derivative, which can lead to increased noise compared to other kernel variants.

The smoothing length is typically in the range $[1.1\delta, 1.5\delta]$, where $\delta$ is the typical particle spacing.

For general information and usage see Smoothing Kernels.

source
TrixiParticles.SpikyKernelType
SpikyKernel{NDIMS}()

The Spiky kernel is another frequently used kernel in SPH, especially due to its desirable properties in preserving features near boundaries in fluid simulations [3]. It is defined as:

\[ W(r, h) = \frac{1}{h^d} w(r/h)\]

with:

\[w(q) = \sigma \begin{cases} (1 - q)^3 & \text{if } 0 \leq q < 1, \\ 0 & \text{if } q \geq 1, -\end{cases}\]

where $d$ is the number of dimensions and the normalization factor $\sigma$ is $10 / \pi$ in two dimensions or $15 / \pi$ in three dimensions.

This kernel function has a compact support of $[0, h]$.

The Spiky kernel is particularly known for its sharp gradients, which can help to preserve sharp features in fluid simulations, especially near solid boundaries. These sharp gradients at the boundary are also the largest disadvantage as they can lead to instability.

The smoothing length is typically in the range $[1.5\delta, 3.0\delta]$, where $\delta$ is the typical particle spacing.

For general information and usage see Smoothing Kernels.

source
TrixiParticles.WendlandC2KernelType
WendlandC2Kernel{NDIMS}()

Wendland C2 kernel [7], a piecewise polynomial function designed to have compact support and to be twice continuously differentiable everywhere. Given by

\[ W(r, h) = \frac{1}{h^d} w(r/h)\]

with

\[w(q) = \sigma \begin{cases} +\end{cases}\]

where $d$ is the number of dimensions and the normalization factor $\sigma$ is $10 / \pi$ in two dimensions or $15 / \pi$ in three dimensions.

This kernel function has a compact support of $[0, h]$.

The Spiky kernel is particularly known for its sharp gradients, which can help to preserve sharp features in fluid simulations, especially near solid boundaries. These sharp gradients at the boundary are also the largest disadvantage as they can lead to instability.

The smoothing length is typically in the range $[1.5\delta, 3.0\delta]$, where $\delta$ is the typical particle spacing.

For general information and usage see Smoothing Kernels.

source
TrixiParticles.WendlandC2KernelType
WendlandC2Kernel{NDIMS}()

Wendland C2 kernel [7], a piecewise polynomial function designed to have compact support and to be twice continuously differentiable everywhere. Given by

\[ W(r, h) = \frac{1}{h^d} w(r/h)\]

with

\[w(q) = \sigma \begin{cases} (1 - q)^4 (4q + 1) & \text{if } 0 \leq q < 1, \\ 0 & \text{if } q \geq 1, -\end{cases}\]

where $d$ is the number of dimensions and $\sigma$ is a normalization factor dependent on the dimension. The normalization factor $\sigma$ is $40/7\pi$ in two dimensions or $21/2\pi$ in three dimensions.

This kernel function has a compact support of $[0, h]$.

For a detailed discussion on Wendland functions and their applications in SPH, see Dehnen (2012). The smoothness of these functions is also the largest disadvantage as they lose details at sharp corners.

The smoothing length is typically in the range $[2.5\delta, 4.0\delta]$, where $\delta$ is the typical particle spacing.

For general information and usage see Smoothing Kernels.

source
TrixiParticles.WendlandC4KernelType
WendlandC4Kernel{NDIMS}()

Wendland C4 kernel [7], a piecewise polynomial function designed to have compact support and to be four times continuously differentiable everywhere. Given by

\[ W(r, h) = \frac{1}{h^d} w(r/h)\]

with

\[w(q) = \sigma \begin{cases} +\end{cases}\]

where $d$ is the number of dimensions and $\sigma$ is a normalization factor dependent on the dimension. The normalization factor $\sigma$ is $40/7\pi$ in two dimensions or $21/2\pi$ in three dimensions.

This kernel function has a compact support of $[0, h]$.

For a detailed discussion on Wendland functions and their applications in SPH, see Dehnen (2012). The smoothness of these functions is also the largest disadvantage as they lose details at sharp corners.

The smoothing length is typically in the range $[2.5\delta, 4.0\delta]$, where $\delta$ is the typical particle spacing.

For general information and usage see Smoothing Kernels.

source
TrixiParticles.WendlandC4KernelType
WendlandC4Kernel{NDIMS}()

Wendland C4 kernel [7], a piecewise polynomial function designed to have compact support and to be four times continuously differentiable everywhere. Given by

\[ W(r, h) = \frac{1}{h^d} w(r/h)\]

with

\[w(q) = \sigma \begin{cases} (1 - q)^6 (35q^2 / 3 + 6q + 1) & \text{if } 0 \leq q < 1, \\ 0 & \text{if } q \geq 1, -\end{cases}\]

where $d$ is the number of dimensions and $\sigma$ is a normalization factor dependent on the dimension. The normalization factor $\sigma$ is $9 / \pi$ in two dimensions or $495 / 32\pi$ in three dimensions.

This kernel function has a compact support of $[0, h]$.

For a detailed discussion on Wendland functions and their applications in SPH, see Dehnen (2012). The smoothness of these functions is also the largest disadvantage as they loose details at sharp corners.

The smoothing length is typically in the range $[3.0\delta, 4.5\delta]$, where $\delta$ is the typical particle spacing.

For general information and usage see Smoothing Kernels.

source
TrixiParticles.WendlandC6KernelType
WendlandC6Kernel{NDIMS}()

Wendland C6 kernel [7], a piecewise polynomial function designed to have compact support and to be six times continuously differentiable everywhere. Given by:

\[W(r, h) = \frac{1}{h^d} w(r/h)\]

with:

\[w(q) = \sigma \begin{cases} +\end{cases}\]

where $d$ is the number of dimensions and $\sigma$ is a normalization factor dependent on the dimension. The normalization factor $\sigma$ is $9 / \pi$ in two dimensions or $495 / 32\pi$ in three dimensions.

This kernel function has a compact support of $[0, h]$.

For a detailed discussion on Wendland functions and their applications in SPH, see Dehnen (2012). The smoothness of these functions is also the largest disadvantage as they loose details at sharp corners.

The smoothing length is typically in the range $[3.0\delta, 4.5\delta]$, where $\delta$ is the typical particle spacing.

For general information and usage see Smoothing Kernels.

source
TrixiParticles.WendlandC6KernelType
WendlandC6Kernel{NDIMS}()

Wendland C6 kernel [7], a piecewise polynomial function designed to have compact support and to be six times continuously differentiable everywhere. Given by:

\[W(r, h) = \frac{1}{h^d} w(r/h)\]

with:

\[w(q) = \sigma \begin{cases} (1 - q)^8 (32q^3 + 25q^2 + 8q + 1) & \text{if } 0 \leq q < 1, \\ 0 & \text{if } q \geq 1, -\end{cases}\]

where $d$ is the number of dimensions and $\sigma$ is a normalization factor dependent on the dimension. The normalization factor $\sigma$ is $78 / 7 \pi$ in two dimensions or $1365 / 64\pi$ in three dimensions.

This kernel function has a compact support of $[0, h]$.

For a detailed discussion on Wendland functions and their applications in SPH, Dehnen (2012). The smoothness of these functions is also the largest disadvantage as they loose details at sharp corners.

The smoothing length is typically in the range $[3.5\delta, 5.0\delta]$, where $\delta$ is the typical particle spacing.

For general information and usage see Smoothing Kernels.

source
+\end{cases}\]

where $d$ is the number of dimensions and $\sigma$ is a normalization factor dependent on the dimension. The normalization factor $\sigma$ is $78 / 7 \pi$ in two dimensions or $1365 / 64\pi$ in three dimensions.

This kernel function has a compact support of $[0, h]$.

For a detailed discussion on Wendland functions and their applications in SPH, Dehnen (2012). The smoothness of these functions is also the largest disadvantage as they loose details at sharp corners.

The smoothing length is typically in the range $[3.5\delta, 5.0\delta]$, where $\delta$ is the typical particle spacing.

For general information and usage see Smoothing Kernels.

source
diff --git a/previews/PR514/general/util/index.html b/previews/PR514/general/util/index.html index 6493e2469..a90e243ba 100644 --- a/previews/PR514/general/util/index.html +++ b/previews/PR514/general/util/index.html @@ -1,3 +1,3 @@ -Util · TrixiParticles.jl

Util

TrixiParticles.examples_dirMethod
examples_dir()

Return the directory where the example files provided with TrixiParticles.jl are located. If TrixiParticles is installed as a regular package (with ]add TrixiParticles), these files are read-only and should not be modified. To find out which files are available, use, e.g., readdir.

Copied from Trixi.jl.

Examples

readdir(examples_dir())
source
TrixiParticles.validation_dirMethod
validation_dir()

Return the directory where the validation files provided with TrixiParticles.jl are located. If TrixiParticles is installed as a regular package (with ]add TrixiParticles), these files are read-only and should not be modified. To find out which files are available, use, e.g., readdir.

Copied from Trixi.jl.

Examples

readdir(validation_dir())
source
TrixiParticles.@autoinfiltrateMacro
@autoinfiltrate
-@autoinfiltrate condition::Bool

Invoke the @infiltrate macro of the package Infiltrator.jl to create a breakpoint for ad-hoc interactive debugging in the REPL. If the optional argument condition is given, the breakpoint is only enabled if condition evaluates to true.

As opposed to using Infiltrator.@infiltrate directly, this macro does not require Infiltrator.jl to be added as a dependency to TrixiParticles.jl. As a bonus, the macro will also attempt to load the Infiltrator module if it has not yet been loaded manually.

Note: For this macro to work, the Infiltrator.jl package needs to be installed in your current Julia environment stack.

See also: Infiltrator.jl

Internal use only

Please note that this macro is intended for internal use only. It is not part of the public API of TrixiParticles.jl, and it thus can altered (or be removed) at any time without it being considered a breaking change.

source
+Util · TrixiParticles.jl

Util

TrixiParticles.examples_dirMethod
examples_dir()

Return the directory where the example files provided with TrixiParticles.jl are located. If TrixiParticles is installed as a regular package (with ]add TrixiParticles), these files are read-only and should not be modified. To find out which files are available, use, e.g., readdir.

Copied from Trixi.jl.

Examples

readdir(examples_dir())
source
TrixiParticles.validation_dirMethod
validation_dir()

Return the directory where the validation files provided with TrixiParticles.jl are located. If TrixiParticles is installed as a regular package (with ]add TrixiParticles), these files are read-only and should not be modified. To find out which files are available, use, e.g., readdir.

Copied from Trixi.jl.

Examples

readdir(validation_dir())
source
TrixiParticles.@autoinfiltrateMacro
@autoinfiltrate
+@autoinfiltrate condition::Bool

Invoke the @infiltrate macro of the package Infiltrator.jl to create a breakpoint for ad-hoc interactive debugging in the REPL. If the optional argument condition is given, the breakpoint is only enabled if condition evaluates to true.

As opposed to using Infiltrator.@infiltrate directly, this macro does not require Infiltrator.jl to be added as a dependency to TrixiParticles.jl. As a bonus, the macro will also attempt to load the Infiltrator module if it has not yet been loaded manually.

Note: For this macro to work, the Infiltrator.jl package needs to be installed in your current Julia environment stack.

See also: Infiltrator.jl

Internal use only

Please note that this macro is intended for internal use only. It is not part of the public API of TrixiParticles.jl, and it thus can altered (or be removed) at any time without it being considered a breaking change.

source
diff --git a/previews/PR514/getting_started/index.html b/previews/PR514/getting_started/index.html index d65e1f92f..45aea0383 100644 --- a/previews/PR514/getting_started/index.html +++ b/previews/PR514/getting_started/index.html @@ -1,2 +1,2 @@ -Getting started · TrixiParticles.jl

Getting started

If you have not installed TrixiParticles.jl, please follow the instructions given here.

In the following sections, we will give a short introduction. For a more thorough discussion, take a look at our Tutorials.

Running an Example

The easiest way to run a simulation is to run one of our predefined example files. We will run the file examples/fluid/hydrostatic_water_column_2d.jl, which simulates a fluid resting in a rectangular tank. Since TrixiParticles.jl uses multithreading, you should start Julia with the flag --threads auto (or, e.g. --threads 4 for 4 threads).

In the Julia REPL, first load the package TrixiParticles.jl.

julia> using TrixiParticles

Then start the simulation by executing

julia> trixi_include(joinpath(examples_dir(), "fluid", "hydrostatic_water_column_2d.jl"))

The easiest way to quickly visualize the result is to use Plots.jl:

julia> using Plots; plot(sol)

This will open a new window with a 2D visualization of the final solution: plot_hydrostatic_water_column

For more information about visualization, see Visualization.

Running other Examples

You can find a list of our other predefined examples under Examples. Execute them as follows from the Julia REPL by replacing subfolder and example_name

julia> trixi_include(joinpath(examples_dir(), "subfolder", "example_name.jl"))

Modifying an example

You can pass keyword arguments to the function trixi_include to overwrite assignments in the file.

With trixi_include, we can overwrite variables defined in the example file to run a different simulation without modifying the example file.

julia> trixi_include(joinpath(examples_dir(), "fluid", "hydrostatic_water_column_2d.jl"), initial_fluid_size=(1.0, 0.5))

This for example, will change the fluid size from $(0.9, 1.0)$ to $(1.0, 0.5)$.

To understand why, take a look into the file hydrostatic_water_column_2d.jl in the subfolder fluid inside the examples directory, which is the file that we executed earlier. You can see that the initial size of the fluid is defined in the variable initial_fluid_size, which we could overwrite with the trixi_include call above. Another variable that is worth experimenting with is fluid_particle_spacing, which controls the resolution of the simulation in this case. A lower value will increase the resolution and the runtime.

Set up you first simulation from scratch

See Set up your first simulation.

Find an overview over the available tutorials under Tutorials.

+Getting started · TrixiParticles.jl

Getting started

If you have not installed TrixiParticles.jl, please follow the instructions given here.

In the following sections, we will give a short introduction. For a more thorough discussion, take a look at our Tutorials.

Running an Example

The easiest way to run a simulation is to run one of our predefined example files. We will run the file examples/fluid/hydrostatic_water_column_2d.jl, which simulates a fluid resting in a rectangular tank. Since TrixiParticles.jl uses multithreading, you should start Julia with the flag --threads auto (or, e.g. --threads 4 for 4 threads).

In the Julia REPL, first load the package TrixiParticles.jl.

julia> using TrixiParticles

Then start the simulation by executing

julia> trixi_include(joinpath(examples_dir(), "fluid", "hydrostatic_water_column_2d.jl"))

The easiest way to quickly visualize the result is to use Plots.jl:

julia> using Plots; plot(sol)

This will open a new window with a 2D visualization of the final solution: plot_hydrostatic_water_column

For more information about visualization, see Visualization.

Running other Examples

You can find a list of our other predefined examples under Examples. Execute them as follows from the Julia REPL by replacing subfolder and example_name

julia> trixi_include(joinpath(examples_dir(), "subfolder", "example_name.jl"))

Modifying an example

You can pass keyword arguments to the function trixi_include to overwrite assignments in the file.

With trixi_include, we can overwrite variables defined in the example file to run a different simulation without modifying the example file.

julia> trixi_include(joinpath(examples_dir(), "fluid", "hydrostatic_water_column_2d.jl"), initial_fluid_size=(1.0, 0.5))

This for example, will change the fluid size from $(0.9, 1.0)$ to $(1.0, 0.5)$.

To understand why, take a look into the file hydrostatic_water_column_2d.jl in the subfolder fluid inside the examples directory, which is the file that we executed earlier. You can see that the initial size of the fluid is defined in the variable initial_fluid_size, which we could overwrite with the trixi_include call above. Another variable that is worth experimenting with is fluid_particle_spacing, which controls the resolution of the simulation in this case. A lower value will increase the resolution and the runtime.

Set up you first simulation from scratch

See Set up your first simulation.

Find an overview over the available tutorials under Tutorials.

diff --git a/previews/PR514/gpu/index.html b/previews/PR514/gpu/index.html index 459ddfc9f..44eb70903 100644 --- a/previews/PR514/gpu/index.html +++ b/previews/PR514/gpu/index.html @@ -5,4 +5,4 @@ cell_list = TrixiParticles.PointNeighbors.FullGridCellList(; min_corner, max_corner)

We then need to pass this cell list to the neighborhood search and the neighborhood search to the Semidiscretization.

semi = Semidiscretization(fluid_system, boundary_system,
                           neighborhood_search=GridNeighborhoodSearch{2}(; cell_list))

At this point, we should run the simulation and make sure that it still works and that the bounding box is large enough. For some simulations where particles move outside the initial tank coordinates, for example when the tank is not closed or when the tank is moving, an appropriate bounding box has to be specified.

Then, we only need to specify the data type that is used for the simulation. On an Nvidia GPU, we specify:

using CUDA
 ode = semidiscretize(semi, tspan, data_type=CuArray)

On an AMD GPU, we use:

using AMDGPU
-ode = semidiscretize(semi, tspan, data_type=ROCArray)

Then, we can run the simulation as usual. All data is transferred to the GPU during initialization and all loops over particles and their neighbors will be executed on the GPU as kernels generated by KernelAbstractions.jl. Data is only copied to the CPU for saving VTK files via the SolutionSavingCallback.

+ode = semidiscretize(semi, tspan, data_type=ROCArray)

Then, we can run the simulation as usual. All data is transferred to the GPU during initialization and all loops over particles and their neighbors will be executed on the GPU as kernels generated by KernelAbstractions.jl. Data is only copied to the CPU for saving VTK files via the SolutionSavingCallback.

diff --git a/previews/PR514/index.html b/previews/PR514/index.html index cae81f263..baca8b814 100644 --- a/previews/PR514/index.html +++ b/previews/PR514/index.html @@ -16,4 +16,4 @@
Dam Break with Elastic Plate
-

Quickstart

  1. Installation
  2. Getting started

If you have any questions concerning TrixiParticles.jl you can join our community on Slack or open an issue with your question.

Start with development

To get started with development have a look at these pages:

  1. Installation
  2. Development
  3. Contributing
+

Quickstart

  1. Installation
  2. Getting started

If you have any questions concerning TrixiParticles.jl you can join our community on Slack or open an issue with your question.

Start with development

To get started with development have a look at these pages:

  1. Installation
  2. Development
  3. Contributing
diff --git a/previews/PR514/install/index.html b/previews/PR514/install/index.html index f16435ebb..f8864adb6 100644 --- a/previews/PR514/install/index.html +++ b/previews/PR514/install/index.html @@ -11,4 +11,4 @@ julia> Pkg.resolve() -julia> Pkg.instantiate() +julia> Pkg.instantiate() diff --git a/previews/PR514/license/index.html b/previews/PR514/license/index.html index 116a45308..57e2bda69 100644 --- a/previews/PR514/license/index.html +++ b/previews/PR514/license/index.html @@ -1,2 +1,2 @@ -License · TrixiParticles.jl

License

MIT License

Copyright (c) 2023-present The TrixiParticles.jl Authors (see Authors)
Copyright (c) 2023-present Helmholtz-Zentrum hereon GmbH, Institute of Surface Science

Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

+License · TrixiParticles.jl

License

MIT License

Copyright (c) 2023-present The TrixiParticles.jl Authors (see Authors)
Copyright (c) 2023-present Helmholtz-Zentrum hereon GmbH, Institute of Surface Science

Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

diff --git a/previews/PR514/news/index.html b/previews/PR514/news/index.html index 3af85b9ef..3f3a88f2e 100644 --- a/previews/PR514/news/index.html +++ b/previews/PR514/news/index.html @@ -1,2 +1,2 @@ -News · TrixiParticles.jl

Changelog

TrixiParticles.jl follows the interpretation of semantic versioning (semver) used in the Julia ecosystem. Notable changes will be documented in this file for human readability.

Version 0.2.3

Highlights

Transport Velocity Formulation (TVF) based on the work of Ramachandran et al. "Entropically damped artificial compressibility for SPH" (2019) was added.

Version 0.2.2

Highlights

Hotfix for threaded sampling of complex geometries.

Version 0.2.1

Highlights

Particle sampling of complex geometries from .stl and .asc files.

Version 0.2.0

Removed

Use of the internal neighborhood search has been removed and replaced with PointNeighbors.jl.

Development Cycle 0.1

Highlights

Discrete Element Method

A basic implementation of the discrete element method was added.

Surface Tension and Adhesion Model

A surface tension and adhesion model based on the work by Akinci et al., "Versatile Surface Tension and Adhesion for SPH Fluids" (2013) was added to WCSPH.

Support for Open Boundaries

Open boundaries using the method of characteristics based on the work of Lastiwka et al., "Permeable and non-reflecting boundary conditions in SPH" (2009) were added for WCSPH and EDAC.

Pre Initial Release (v0.1.0)

This section summarizes the initial features that TrixiParticles.jl was released with.

Highlights

EDAC

An implementation of EDAC (Entropically Damped Artificial Compressibility) was added, which allows for more stable simulations compared to basic WCSPH and reduces spurious pressure oscillations.

WCSPH

An implementation of WCSPH (Weakly Compressible Smoothed Particle Hydrodynamics), which is the classical SPH approach.

Features:

  • Correction schemes (Shepard (0. Order) ... MixedKernelGradient (1. Order))
  • Density reinitialization
  • Kernel summation and Continuity equation density formulations
  • Flexible boundary conditions e.g. dummy particles with Adami pressure extrapolation, pressure zeroing, pressure mirroring...
  • Moving boundaries
  • Density diffusion based on the models by Molteni & Colagrossi (2009), Ferrari et al. (2009) and Antuono et al. (2010).

TLSPH

An implementation of TLSPH (Total Lagrangian Smoothed Particle Hydrodynamics) for solid bodies enabling FSI (Fluid Structure Interactions).

+News · TrixiParticles.jl

Changelog

TrixiParticles.jl follows the interpretation of semantic versioning (semver) used in the Julia ecosystem. Notable changes will be documented in this file for human readability.

Version 0.2.3

Highlights

Transport Velocity Formulation (TVF) based on the work of Ramachandran et al. "Entropically damped artificial compressibility for SPH" (2019) was added.

Version 0.2.2

Highlights

Hotfix for threaded sampling of complex geometries.

Version 0.2.1

Highlights

Particle sampling of complex geometries from .stl and .asc files.

Version 0.2.0

Removed

Use of the internal neighborhood search has been removed and replaced with PointNeighbors.jl.

Development Cycle 0.1

Highlights

Discrete Element Method

A basic implementation of the discrete element method was added.

Surface Tension and Adhesion Model

A surface tension and adhesion model based on the work by Akinci et al., "Versatile Surface Tension and Adhesion for SPH Fluids" (2013) was added to WCSPH.

Support for Open Boundaries

Open boundaries using the method of characteristics based on the work of Lastiwka et al., "Permeable and non-reflecting boundary conditions in SPH" (2009) were added for WCSPH and EDAC.

Pre Initial Release (v0.1.0)

This section summarizes the initial features that TrixiParticles.jl was released with.

Highlights

EDAC

An implementation of EDAC (Entropically Damped Artificial Compressibility) was added, which allows for more stable simulations compared to basic WCSPH and reduces spurious pressure oscillations.

WCSPH

An implementation of WCSPH (Weakly Compressible Smoothed Particle Hydrodynamics), which is the classical SPH approach.

Features:

  • Correction schemes (Shepard (0. Order) ... MixedKernelGradient (1. Order))
  • Density reinitialization
  • Kernel summation and Continuity equation density formulations
  • Flexible boundary conditions e.g. dummy particles with Adami pressure extrapolation, pressure zeroing, pressure mirroring...
  • Moving boundaries
  • Density diffusion based on the models by Molteni & Colagrossi (2009), Ferrari et al. (2009) and Antuono et al. (2010).

TLSPH

An implementation of TLSPH (Total Lagrangian Smoothed Particle Hydrodynamics) for solid bodies enabling FSI (Fluid Structure Interactions).

diff --git a/previews/PR514/overview/index.html b/previews/PR514/overview/index.html index 5ea15138e..9dbf4e9c2 100644 --- a/previews/PR514/overview/index.html +++ b/previews/PR514/overview/index.html @@ -1,2 +1,2 @@ -Overview · TrixiParticles.jl

Overview

The actual API reference is not listed on a single page, like in most Julia packages, but instead is split into multiple sections that follow a similar structure as the code files themselves. In these sections, API docs are combined with explanations of the theoretical background of these methods.

The following page gives a rough overview of important parts of the code.

Program flow

To initiate a simulation, the goal is to solve an ordinary differential equation, for example, by employing the time integration schemes provided by OrdinaryDiffEq.jl. These schemes are then utilized to integrate $\mathrm{d}u/\mathrm{d}t$ and $\mathrm{d}v/\mathrm{d}t$, where $u$ represents the particles' positions and $v$ their properties such as velocity and density. During a single time step or an intermediate step of the time integration scheme, the functions drift! and kick! are invoked, followed by the functions depicted in this diagram (with key parts highlighted in orange/yellow).

Main Program Flow

Structure

What we refer to as schemes are various models such as Weakly Compressible Smoothed Particle Hydrodynamics (WCSPH) or Total Lagrangian Smoothed Particle Hydrodynamics (TLSPH). These schemes are categorized based on the applicable physical regimes, namely fluid, solid, gas, and others. Each scheme comprises at least two files: a system.jl file and an rhs.jl file. The system.jl file provides the data structure holding the particles of this scheme and some routines, particularly those for allocation and the main update routines, excluding system interactions. The interactions between particles of this scheme (and with particles of other schemes) are handled in the rhs.jl file.

+Overview · TrixiParticles.jl

Overview

The actual API reference is not listed on a single page, like in most Julia packages, but instead is split into multiple sections that follow a similar structure as the code files themselves. In these sections, API docs are combined with explanations of the theoretical background of these methods.

The following page gives a rough overview of important parts of the code.

Program flow

To initiate a simulation, the goal is to solve an ordinary differential equation, for example, by employing the time integration schemes provided by OrdinaryDiffEq.jl. These schemes are then utilized to integrate $\mathrm{d}u/\mathrm{d}t$ and $\mathrm{d}v/\mathrm{d}t$, where $u$ represents the particles' positions and $v$ their properties such as velocity and density. During a single time step or an intermediate step of the time integration scheme, the functions drift! and kick! are invoked, followed by the functions depicted in this diagram (with key parts highlighted in orange/yellow).

Main Program Flow

Structure

What we refer to as schemes are various models such as Weakly Compressible Smoothed Particle Hydrodynamics (WCSPH) or Total Lagrangian Smoothed Particle Hydrodynamics (TLSPH). These schemes are categorized based on the applicable physical regimes, namely fluid, solid, gas, and others. Each scheme comprises at least two files: a system.jl file and an rhs.jl file. The system.jl file provides the data structure holding the particles of this scheme and some routines, particularly those for allocation and the main update routines, excluding system interactions. The interactions between particles of this scheme (and with particles of other schemes) are handled in the rhs.jl file.

diff --git a/previews/PR514/preprocessing/preprocessing/index.html b/previews/PR514/preprocessing/preprocessing/index.html index 73e770a42..d511e2b86 100644 --- a/previews/PR514/preprocessing/preprocessing/index.html +++ b/previews/PR514/preprocessing/preprocessing/index.html @@ -48,5 +48,5 @@ correct evaluation intersecting

The evaluation then looks as follows.

correct evaluation intersecting 2 -
TrixiParticles.WindingNumberHormannType
WindingNumberHormann()

Algorithm for inside-outside segmentation of a complex geometry proposed by Hormann (2001). It is only supported for 2D geometries. WindingNumberHormann might handle edge cases a bit better, since the winding number is an integer value.

Experimental Implementation

This is an experimental feature and may change in any future releases.

source
TrixiParticles.WindingNumberJacobsonType
WindingNumberJacobson(; geometry=nothing, winding_number_factor=sqrt(eps()),
-                      hierarchical_winding=false)

Algorithm for inside-outside segmentation of a complex geometry proposed by [2].

Keywords

  • geometry: Complex geometry returned by load_geometry and is only required when using hierarchical_winding=true.
  • hierarchical_winding: If set to true, an optimized hierarchical approach will be used, which gives a significant speedup. For further information see Hierarchical Winding.
  • winding_number_factor: For leaky geometries, a factor of 0.4 will give a better inside-outside segmentation.
Experimental Implementation

This is an experimental feature and may change in any future releases.

source
TrixiParticles.load_geometryMethod
load_geometry(filename; element_type=Float64)

Load file and return corresponding type for ComplexShape. Supported file formats are .stl and .asc.

Arguments

  • filename: Name of the file to be loaded.

Keywords

  • element_type: Element type (default is Float64)
source
+
TrixiParticles.WindingNumberHormannType
WindingNumberHormann()

Algorithm for inside-outside segmentation of a complex geometry proposed by Hormann (2001). It is only supported for 2D geometries. WindingNumberHormann might handle edge cases a bit better, since the winding number is an integer value.

Experimental Implementation

This is an experimental feature and may change in any future releases.

source
TrixiParticles.WindingNumberJacobsonType
WindingNumberJacobson(; geometry=nothing, winding_number_factor=sqrt(eps()),
+                      hierarchical_winding=false)

Algorithm for inside-outside segmentation of a complex geometry proposed by [2].

Keywords

  • geometry: Complex geometry returned by load_geometry and is only required when using hierarchical_winding=true.
  • hierarchical_winding: If set to true, an optimized hierarchical approach will be used, which gives a significant speedup. For further information see Hierarchical Winding.
  • winding_number_factor: For leaky geometries, a factor of 0.4 will give a better inside-outside segmentation.
Experimental Implementation

This is an experimental feature and may change in any future releases.

source
TrixiParticles.load_geometryMethod
load_geometry(filename; element_type=Float64)

Load file and return corresponding type for ComplexShape. Supported file formats are .stl and .asc.

Arguments

  • filename: Name of the file to be loaded.

Keywords

  • element_type: Element type (default is Float64)
source
diff --git a/previews/PR514/reference-pointneighbors/index.html b/previews/PR514/reference-pointneighbors/index.html index 8d8051dfb..46cd6e0c8 100644 --- a/previews/PR514/reference-pointneighbors/index.html +++ b/previews/PR514/reference-pointneighbors/index.html @@ -16,4 +16,4 @@ # output GridNeighborhoodSearch{2, Float64, ...}(...)source
PointNeighbors.foreach_point_neighborMethod
foreach_point_neighbor(f, system_coords, neighbor_coords, neighborhood_search;
-                       points = axes(system_coords, 2), parallel = true)

Loop for each point in system_coords over all points in neighbor_coords whose distances to that point are smaller than the search radius and execute the function f(i, j, x, y, d), where

  • i is the column index of the point in system_coords,
  • j the column index of the neighbor in neighbor_coords,
  • x an SVector of the coordinates of the point (system_coords[:, i]),
  • y an SVector of the coordinates of the neighbor (neighbor_coords[:, j]),
  • d the distance between x and y.

The neighborhood_search must have been initialized or updated with system_coords as first coordinate array and neighbor_coords as second coordinate array.

Note that system_coords and neighbor_coords can be identical.

Arguments

  • f: The function explained above.
  • system_coords: A matrix where the i-th column contains the coordinates of point i.
  • neighbor_coords: A matrix where the j-th column contains the coordinates of point j.
  • neighborhood_search: A neighborhood search initialized or updated with system_coords as first coordinate array and neighbor_coords as second coordinate array.

Keywords

  • points: Loop over these point indices. By default all columns of system_coords.
  • parallel=true: Run the outer loop over points thread-parallel.

See also initialize!, update!.

source
PointNeighbors.initialize!Method
initialize!(search::AbstractNeighborhoodSearch, x, y)

Initialize a neighborhood search with the two coordinate arrays x and y.

In general, the purpose of a neighborhood search is to find for one point in x all points in y whose distances to that point are smaller than the search radius. x and y are expected to be matrices, where the i-th column contains the coordinates of point i. Note that x and y can be identical.

See also update!.

source
PointNeighbors.update!Method
update!(search::AbstractNeighborhoodSearch, x, y; points_moving = (true, true))

Update an already initialized neighborhood search with the two coordinate arrays x and y.

Like initialize!, but reusing the existing data structures of the already initialized neighborhood search. When the points only moved a small distance since the last update! or initialize!, this is significantly faster than initialize!.

Not all implementations support incremental updates. If incremental updates are not possible for an implementation, update! will fall back to a regular initialize!.

Some neighborhood searches might not need to update when only x changed since the last update! or initialize! and y did not change. Pass points_moving = (true, false) in this case to avoid unnecessary updates. The first flag in points_moving indicates if points in x are moving. The second flag indicates if points in y are moving.

Experimental Feature: Backend Specification

The keyword argument parallelization_backend allows users to specify the multithreading backend. This feature is currently considered experimental!

Possible parallelization backends are:

See also initialize!.

source
PointNeighbors.@threadedMacro
@threaded x for ... end

Run either a threaded CPU loop or launch a kernel on the GPU, depending on the type of x. Semantically the same as Threads.@threads when iterating over a AbstractUnitRange but without guarantee that the underlying implementation uses Threads.@threads or works for more general for loops.

The first argument must either be a parallelization backend (see below) or an array from which the backend can be derived to determine if the loop must be run threaded on the CPU or launched as a kernel on the GPU. Passing KernelAbstractions.CPU() will run the GPU kernel on the CPU.

Possible parallelization backends are:

In particular, the underlying threading capabilities might be provided by other packages such as Polyester.jl.

Warning

This macro does not necessarily work for general for loops. For example, it does not necessarily support general iterables such as eachline(filename).

source
+ points = axes(system_coords, 2), parallel = true)

Loop for each point in system_coords over all points in neighbor_coords whose distances to that point are smaller than the search radius and execute the function f(i, j, x, y, d), where

The neighborhood_search must have been initialized or updated with system_coords as first coordinate array and neighbor_coords as second coordinate array.

Note that system_coords and neighbor_coords can be identical.

Arguments

Keywords

See also initialize!, update!.

source
PointNeighbors.initialize!Method
initialize!(search::AbstractNeighborhoodSearch, x, y)

Initialize a neighborhood search with the two coordinate arrays x and y.

In general, the purpose of a neighborhood search is to find for one point in x all points in y whose distances to that point are smaller than the search radius. x and y are expected to be matrices, where the i-th column contains the coordinates of point i. Note that x and y can be identical.

See also update!.

source
PointNeighbors.update!Method
update!(search::AbstractNeighborhoodSearch, x, y; points_moving = (true, true))

Update an already initialized neighborhood search with the two coordinate arrays x and y.

Like initialize!, but reusing the existing data structures of the already initialized neighborhood search. When the points only moved a small distance since the last update! or initialize!, this is significantly faster than initialize!.

Not all implementations support incremental updates. If incremental updates are not possible for an implementation, update! will fall back to a regular initialize!.

Some neighborhood searches might not need to update when only x changed since the last update! or initialize! and y did not change. Pass points_moving = (true, false) in this case to avoid unnecessary updates. The first flag in points_moving indicates if points in x are moving. The second flag indicates if points in y are moving.

Experimental Feature: Backend Specification

The keyword argument parallelization_backend allows users to specify the multithreading backend. This feature is currently considered experimental!

Possible parallelization backends are:

See also initialize!.

source
PointNeighbors.@threadedMacro
@threaded x for ... end

Run either a threaded CPU loop or launch a kernel on the GPU, depending on the type of x. Semantically the same as Threads.@threads when iterating over a AbstractUnitRange but without guarantee that the underlying implementation uses Threads.@threads or works for more general for loops.

The first argument must either be a parallelization backend (see below) or an array from which the backend can be derived to determine if the loop must be run threaded on the CPU or launched as a kernel on the GPU. Passing KernelAbstractions.CPU() will run the GPU kernel on the CPU.

Possible parallelization backends are:

In particular, the underlying threading capabilities might be provided by other packages such as Polyester.jl.

Warning

This macro does not necessarily work for general for loops. For example, it does not necessarily support general iterables such as eachline(filename).

source
diff --git a/previews/PR514/reference-trixibase/index.html b/previews/PR514/reference-trixibase/index.html index 8295ae87b..4dae8f358 100644 --- a/previews/PR514/reference-trixibase/index.html +++ b/previews/PR514/reference-trixibase/index.html @@ -7,4 +7,4 @@ sol.t[end] end [ Info: You just called `trixi_include`. Julia may now compile the code, please be patient. -0.1source
TrixiBase.@trixi_timeitMacro
@trixi_timeit timer() "some label" expression

Basically the same as a special case of @timeit_debug from TimerOutputs.jl, but without try ... finally ... end block. Thus, it's not exception-safe, but it also avoids some related performance problems. Since we do not use exception handling in Trixi.jl, that's not really an issue.

All @trixi_timeit timings can be disabled with disable_debug_timings. The timings should then be optimized away, allowing for truly zero-overhead.

See also disable_debug_timings, enable_debug_timings.

source
+0.1source
TrixiBase.@trixi_timeitMacro
@trixi_timeit timer() "some label" expression

Basically the same as a special case of @timeit_debug from TimerOutputs.jl, but without try ... finally ... end block. Thus, it's not exception-safe, but it also avoids some related performance problems. Since we do not use exception handling in Trixi.jl, that's not really an issue.

All @trixi_timeit timings can be disabled with disable_debug_timings. The timings should then be optimized away, allowing for truly zero-overhead.

See also disable_debug_timings, enable_debug_timings.

source
diff --git a/previews/PR514/references/index.html b/previews/PR514/references/index.html index d1140afc5..be0b190a0 100644 --- a/previews/PR514/references/index.html +++ b/previews/PR514/references/index.html @@ -1,2 +1,2 @@ -References · TrixiParticles.jl
[1]
K. Hormann and A. Agathos. The point in polygon problem for arbitrary polygons. Computational Geometry 20, 131–144 (2001).
[2]
A. Jacobson, L. Kavan and O. Sorkine-Hornung. Robust inside-outside segmentation using generalized winding numbers. ACM Transactions on Graphics 32, 1–12 (2013).
[3]
M. Müller, D. Charypar and M. Gross. Particle-Based Fluid Simulation for Interactive Applications. In: Proceedings of the 2003 ACM SIGGRAPH/Eurographics Symposium on Computer Animation (Eurographics Association, 07 2003); pp. 154–159.
[4]
I. J. Schoenberg. Contributions to the problem of approximation of equidistant data by analytic functions. Part B. On the problem of osculatory interpolation. A second class of analytic approximation formulae. Quarterly of Applied Mathematics 4, 112–141 (1946).
[5]
D. J. Price. Smoothed particle hydrodynamics and magnetohydrodynamics. Journal of Computational Physics 231, 759–794 (2012).
[6]
J. Monaghan. Particle methods for hydrodynamics. Computer Physics Reports 3, 71–124 (1985).
[7]
H. Wendland. Piecewise polynomial, positive definite and compactly supported radial functions of minimal degree. Advances in Computational Mathematics 4, 389–396 (1995).
[8]
W. Dehnen and H. Aly. Improving convergence in smoothed particle hydrodynamics simulations without pairing instability: SPH without pairing instability. Monthly Notices of the Royal Astronomical Society 425, 1068–1082 (2012).
[9]
N. Bićanić. Discrete Element Methods. In: Discrete Element Methods (Wiley, 2004).
[10]
P. A. Cundall and O. D. Strack. A discrete numerical model for granular assemblies. Géotechnique 29, 47–65 (1979).
[11]
A. Di Renzo and F. P. Di Maio. Comparison of contact-force models for the simulation of collisions in DEM-based granular flow codes. Chemical Engineering Science 59, 525–541 (2004).
[12]
J. Monaghan. Simulating Free Surface Flows with SPH. Journal of Computational Physics 110, 399–406 (1994).
[13]
R. H. Cole and R. Weller. Underwater Explosions. Physics Today 1, 35–35 (1948).
[14]
S. Adami, X. Hu and N. Adams. A generalized wall boundary condition for smoothed particle hydrodynamics. Journal of Computational Physics 231, 7057–7075 (2012).
[15]
J. P. Morris, P. J. Fox and Y. Zhu. Modeling Low Reynolds Number Incompressible Flows Using SPH. Journal of Computational Physics 136, 214–226 (1997).
[16]
J. J. Monaghan. Smoothed Particle Hydrodynamics. Annual Review of Astronomy and Astrophysics 30, 543–574 (1992).
[17]
[18]
J. J. Monaghan. Smoothed particle hydrodynamics. Reports on Progress in Physics 68, 1703–1759 (2005).
[19]
P. Ramachandran and K. Puri. Entropically damped artificial compressibility for SPH. Computers & Fluids 179, 579–594 (2019).
[20]
G. Fourtakas, J. M. Dominguez, R. Vacondio and B. D. Rogers. Local uniform stencil (LUST) boundary condition for arbitrary 3-D boundaries in parallel smoothed particle hydrodynamics (SPH) models. Computers & Fluids 190, 346–361 (2019).
[21]
M. Antuono, A. Colagrossi and S. Marrone. Numerical diffusive terms in weakly-compressible SPH schemes. Computer Physics Communications 183, 2570–2580 (2012).
[22]
M. Antuono, A. Colagrossi, S. Marrone and D. Molteni. Free-surface flows solved by means of SPH schemes with numerical diffusive terms. Computer Physics Communications 181, 532–549 (2010).
[23]
D. Molteni and A. Colagrossi. A simple procedure to improve the pressure evaluation in hydrodynamic context using the SPH. Computer Physics Communications 180, 861–872 (2009).
[24]
A. Ferrari, M. Dumbser, E. F. Toro and A. Armanini. A new 3D parallel SPH scheme for free surface flows. Computers & Fluids 38, 1203–1217 (2009).
[25]
N. Akinci, G. Akinci and M. Teschner. Versatile surface tension and adhesion for SPH fluids. ACM Transactions on Graphics 32, 1–8 (2013).
[26]
J. Bonet and T.-S. Lok. Variational and momentum preservation aspects of Smooth Particle Hydrodynamic formulations. Computer Methods in Applied Mechanics and Engineering 180, 97–115 (1999).
[27]
M. Basa, N. J. Quinlan and M. Lastiwka. Robustness and accuracy of SPH formulations for viscous flow. International Journal for Numerical Methods in Fluids 60, 1127–1148 (2008).
[28]
S. Li and W. K. Liu. Moving least-square reproducing kernel method Part II: Fourier analysis. Computer Methods in Applied Mechanics and Engineering 139, 159–193 (1996).
[29]
J. R. Clausen. Entropically damped form of artificial compressibility for explicit simulation of incompressible flow. Physical Review E 87, 013309 (2013).
[30]
S. Adami, X. Hu and N. Adams. A transport-velocity formulation for smoothed particle hydrodynamics. Journal of Computational Physics 241, 292–307 (2013).
[31]
J. O’Connor and B. D. Rogers. A fluid–structure interaction model for free-surface flows and flexible structures using smoothed particle hydrodynamics on a GPU. Journal of Fluids and Structures 104, 103312 (2021).
[32]
T. Belytschko, Y. Guo, W. K. Liu and S. P. Xiao. A unified stability analysis of meshless particle methods. International Journal for Numerical Methods in Engineering 48, 1359–1400 (2000).
[33]
G. C. Ganzenmüller. An hourglass control algorithm for Lagrangian Smooth Particle Hydrodynamics. Computer Methods in Applied Mechanics and Engineering 286, 87–106 (2015).
[34]
A. Valizadeh and J. J. Monaghan. A study of solid wall models for weakly compressible SPH. Journal of Computational Physics 300, 5–19 (2015).
[35]
N. Akinci, M. Ihmsen, G. Akinci, B. Solenthaler and M. Teschner. Versatile rigid-fluid coupling for incompressible SPH. ACM Transactions on Graphics 31, 1–8 (2012).
[36]
A. J. Crespo, M. Gomez-Gesteira and R. A. Dalrymple. Boundary Conditions Generated by Dynamic Particles in SPH Methods. Computers, Materials and Continua 5, 173–184 (2007).
[37]
S. Band, C. Gissler, A. Peer and M. Teschner. MLS pressure boundaries for divergence-free and viscous SPH fluids. Computers & Graphics 76, 37–46 (2018).
[38]
J. Monaghan and J. Kajtar. SPH particle boundary forces for arbitrary boundaries. Computer Physics Communications 180, 1811–1820 (2009).
[39]
M. B. Giles. Nonreflecting boundary conditions for Euler equation calculations. AIAA Journal 28, 2050–2058 (1990).
[40]
M. Lastiwka, M. Basa and N. J. Quinlan. Permeable and non‐reflecting boundary conditions in SPH. International Journal for Numerical Methods in Fluids 61, 709–724 (2008).
[41]
P. Negi, P. Ramachandran and A. Haftu. An improved non-reflecting outlet boundary condition for weakly-compressible SPH. Computer Methods in Applied Mechanics and Engineering 367, 113119 (2020).
[42]
A. Panizzo, G. Cuomo and R. A. Dalrymple. 3D-SPH SIMULATION OF LANDSLIDE GENERATED WAVES. In: Coastal Engineering 2006 (World Scientific Publishing Company, Apr 2007).
[43]
P. Sun, A. Colagrossi, S. Marrone and A. Zhang. Delta-SPH model: Simple procedures for a further improvement of the SPH scheme. Computer Methods in Applied Mechanics and Engineering 315, 25–49 (2017).
[44]
M. Antuono, S. Marrone, A. Colagrossi and B. Bouscasse. Energy balance in the Delta-SPH scheme. Computer Methods in Applied Mechanics and Engineering 289, 209–226 (2015).
+References · TrixiParticles.jl
[1]
K. Hormann and A. Agathos. The point in polygon problem for arbitrary polygons. Computational Geometry 20, 131–144 (2001).
[2]
A. Jacobson, L. Kavan and O. Sorkine-Hornung. Robust inside-outside segmentation using generalized winding numbers. ACM Transactions on Graphics 32, 1–12 (2013).
[3]
M. Müller, D. Charypar and M. Gross. Particle-Based Fluid Simulation for Interactive Applications. In: Proceedings of the 2003 ACM SIGGRAPH/Eurographics Symposium on Computer Animation (Eurographics Association, 07 2003); pp. 154–159.
[4]
I. J. Schoenberg. Contributions to the problem of approximation of equidistant data by analytic functions. Part B. On the problem of osculatory interpolation. A second class of analytic approximation formulae. Quarterly of Applied Mathematics 4, 112–141 (1946).
[5]
D. J. Price. Smoothed particle hydrodynamics and magnetohydrodynamics. Journal of Computational Physics 231, 759–794 (2012).
[6]
J. Monaghan. Particle methods for hydrodynamics. Computer Physics Reports 3, 71–124 (1985).
[7]
H. Wendland. Piecewise polynomial, positive definite and compactly supported radial functions of minimal degree. Advances in Computational Mathematics 4, 389–396 (1995).
[8]
W. Dehnen and H. Aly. Improving convergence in smoothed particle hydrodynamics simulations without pairing instability: SPH without pairing instability. Monthly Notices of the Royal Astronomical Society 425, 1068–1082 (2012).
[9]
N. Bićanić. Discrete Element Methods. In: Discrete Element Methods (Wiley, 2004).
[10]
P. A. Cundall and O. D. Strack. A discrete numerical model for granular assemblies. Géotechnique 29, 47–65 (1979).
[11]
A. Di Renzo and F. P. Di Maio. Comparison of contact-force models for the simulation of collisions in DEM-based granular flow codes. Chemical Engineering Science 59, 525–541 (2004).
[12]
J. Monaghan. Simulating Free Surface Flows with SPH. Journal of Computational Physics 110, 399–406 (1994).
[13]
R. H. Cole and R. Weller. Underwater Explosions. Physics Today 1, 35–35 (1948).
[14]
S. Adami, X. Hu and N. Adams. A generalized wall boundary condition for smoothed particle hydrodynamics. Journal of Computational Physics 231, 7057–7075 (2012).
[15]
J. P. Morris, P. J. Fox and Y. Zhu. Modeling Low Reynolds Number Incompressible Flows Using SPH. Journal of Computational Physics 136, 214–226 (1997).
[16]
J. J. Monaghan. Smoothed Particle Hydrodynamics. Annual Review of Astronomy and Astrophysics 30, 543–574 (1992).
[17]
[18]
J. J. Monaghan. Smoothed particle hydrodynamics. Reports on Progress in Physics 68, 1703–1759 (2005).
[19]
P. Ramachandran and K. Puri. Entropically damped artificial compressibility for SPH. Computers & Fluids 179, 579–594 (2019).
[20]
G. Fourtakas, J. M. Dominguez, R. Vacondio and B. D. Rogers. Local uniform stencil (LUST) boundary condition for arbitrary 3-D boundaries in parallel smoothed particle hydrodynamics (SPH) models. Computers & Fluids 190, 346–361 (2019).
[21]
M. Antuono, A. Colagrossi and S. Marrone. Numerical diffusive terms in weakly-compressible SPH schemes. Computer Physics Communications 183, 2570–2580 (2012).
[22]
M. Antuono, A. Colagrossi, S. Marrone and D. Molteni. Free-surface flows solved by means of SPH schemes with numerical diffusive terms. Computer Physics Communications 181, 532–549 (2010).
[23]
D. Molteni and A. Colagrossi. A simple procedure to improve the pressure evaluation in hydrodynamic context using the SPH. Computer Physics Communications 180, 861–872 (2009).
[24]
A. Ferrari, M. Dumbser, E. F. Toro and A. Armanini. A new 3D parallel SPH scheme for free surface flows. Computers & Fluids 38, 1203–1217 (2009).
[25]
N. Akinci, G. Akinci and M. Teschner. Versatile surface tension and adhesion for SPH fluids. ACM Transactions on Graphics 32, 1–8 (2013).
[26]
J. Bonet and T.-S. Lok. Variational and momentum preservation aspects of Smooth Particle Hydrodynamic formulations. Computer Methods in Applied Mechanics and Engineering 180, 97–115 (1999).
[27]
M. Basa, N. J. Quinlan and M. Lastiwka. Robustness and accuracy of SPH formulations for viscous flow. International Journal for Numerical Methods in Fluids 60, 1127–1148 (2008).
[28]
S. Li and W. K. Liu. Moving least-square reproducing kernel method Part II: Fourier analysis. Computer Methods in Applied Mechanics and Engineering 139, 159–193 (1996).
[29]
J. R. Clausen. Entropically damped form of artificial compressibility for explicit simulation of incompressible flow. Physical Review E 87, 013309 (2013).
[30]
S. Adami, X. Hu and N. Adams. A transport-velocity formulation for smoothed particle hydrodynamics. Journal of Computational Physics 241, 292–307 (2013).
[31]
J. O’Connor and B. D. Rogers. A fluid–structure interaction model for free-surface flows and flexible structures using smoothed particle hydrodynamics on a GPU. Journal of Fluids and Structures 104, 103312 (2021).
[32]
T. Belytschko, Y. Guo, W. K. Liu and S. P. Xiao. A unified stability analysis of meshless particle methods. International Journal for Numerical Methods in Engineering 48, 1359–1400 (2000).
[33]
G. C. Ganzenmüller. An hourglass control algorithm for Lagrangian Smooth Particle Hydrodynamics. Computer Methods in Applied Mechanics and Engineering 286, 87–106 (2015).
[34]
A. Valizadeh and J. J. Monaghan. A study of solid wall models for weakly compressible SPH. Journal of Computational Physics 300, 5–19 (2015).
[35]
N. Akinci, M. Ihmsen, G. Akinci, B. Solenthaler and M. Teschner. Versatile rigid-fluid coupling for incompressible SPH. ACM Transactions on Graphics 31, 1–8 (2012).
[36]
A. J. Crespo, M. Gomez-Gesteira and R. A. Dalrymple. Boundary Conditions Generated by Dynamic Particles in SPH Methods. Computers, Materials and Continua 5, 173–184 (2007).
[37]
S. Band, C. Gissler, A. Peer and M. Teschner. MLS pressure boundaries for divergence-free and viscous SPH fluids. Computers & Graphics 76, 37–46 (2018).
[38]
J. Monaghan and J. Kajtar. SPH particle boundary forces for arbitrary boundaries. Computer Physics Communications 180, 1811–1820 (2009).
[39]
M. B. Giles. Nonreflecting boundary conditions for Euler equation calculations. AIAA Journal 28, 2050–2058 (1990).
[40]
M. Lastiwka, M. Basa and N. J. Quinlan. Permeable and non‐reflecting boundary conditions in SPH. International Journal for Numerical Methods in Fluids 61, 709–724 (2008).
[41]
P. Negi, P. Ramachandran and A. Haftu. An improved non-reflecting outlet boundary condition for weakly-compressible SPH. Computer Methods in Applied Mechanics and Engineering 367, 113119 (2020).
[42]
A. Panizzo, G. Cuomo and R. A. Dalrymple. 3D-SPH SIMULATION OF LANDSLIDE GENERATED WAVES. In: Coastal Engineering 2006 (World Scientific Publishing Company, Apr 2007).
[43]
P. Sun, A. Colagrossi, S. Marrone and A. Zhang. Delta-SPH model: Simple procedures for a further improvement of the SPH scheme. Computer Methods in Applied Mechanics and Engineering 315, 25–49 (2017).
[44]
M. Antuono, S. Marrone, A. Colagrossi and B. Bouscasse. Energy balance in the Delta-SPH scheme. Computer Methods in Applied Mechanics and Engineering 289, 209–226 (2015).
diff --git a/previews/PR514/search_index.js b/previews/PR514/search_index.js index 33a67569e..e69f800d9 100644 --- a/previews/PR514/search_index.js +++ b/previews/PR514/search_index.js @@ -1,3 +1,3 @@ var documenterSearchIndex = {"docs": -[{"location":"references/","page":"References","title":"References","text":"K. Hormann and A. Agathos. The point in polygon problem for arbitrary polygons. Computational Geometry 20, 131–144 (2001).\n\n\n\nA. Jacobson, L. Kavan and O. Sorkine-Hornung. Robust inside-outside segmentation using generalized winding numbers. ACM Transactions on Graphics 32, 1–12 (2013).\n\n\n\nM. Müller, D. Charypar and M. Gross. Particle-Based Fluid Simulation for Interactive Applications. In: Proceedings of the 2003 ACM SIGGRAPH/Eurographics Symposium on Computer Animation (Eurographics Association, 07 2003); pp. 154–159.\n\n\n\nI. J. Schoenberg. Contributions to the problem of approximation of equidistant data by analytic functions. Part B. On the problem of osculatory interpolation. A second class of analytic approximation formulae. Quarterly of Applied Mathematics 4, 112–141 (1946).\n\n\n\nD. J. Price. Smoothed particle hydrodynamics and magnetohydrodynamics. Journal of Computational Physics 231, 759–794 (2012).\n\n\n\nJ. Monaghan. Particle methods for hydrodynamics. Computer Physics Reports 3, 71–124 (1985).\n\n\n\nH. Wendland. Piecewise polynomial, positive definite and compactly supported radial functions of minimal degree. Advances in Computational Mathematics 4, 389–396 (1995).\n\n\n\nW. Dehnen and H. Aly. Improving convergence in smoothed particle hydrodynamics simulations without pairing instability: SPH without pairing instability. Monthly Notices of the Royal Astronomical Society 425, 1068–1082 (2012).\n\n\n\nN. Bićanić. Discrete Element Methods. In: Discrete Element Methods (Wiley, 2004).\n\n\n\nP. A. Cundall and O. D. Strack. A discrete numerical model for granular assemblies. Géotechnique 29, 47–65 (1979).\n\n\n\nA. Di Renzo and F. P. Di Maio. Comparison of contact-force models for the simulation of collisions in DEM-based granular flow codes. Chemical Engineering Science 59, 525–541 (2004).\n\n\n\nJ. Monaghan. Simulating Free Surface Flows with SPH. Journal of Computational Physics 110, 399–406 (1994).\n\n\n\nR. H. Cole and R. Weller. Underwater Explosions. Physics Today 1, 35–35 (1948).\n\n\n\nS. Adami, X. Hu and N. Adams. A generalized wall boundary condition for smoothed particle hydrodynamics. Journal of Computational Physics 231, 7057–7075 (2012).\n\n\n\nJ. P. Morris, P. J. Fox and Y. Zhu. Modeling Low Reynolds Number Incompressible Flows Using SPH. Journal of Computational Physics 136, 214–226 (1997).\n\n\n\nJ. J. Monaghan. Smoothed Particle Hydrodynamics. Annual Review of Astronomy and Astrophysics 30, 543–574 (1992).\n\n\n\nJ. Monaghan. On the problem of penetration in particle methods. Journal of Computational Physics 82, 1–15 (1989).\n\n\n\nJ. J. Monaghan. Smoothed particle hydrodynamics. Reports on Progress in Physics 68, 1703–1759 (2005).\n\n\n\nP. Ramachandran and K. Puri. Entropically damped artificial compressibility for SPH. Computers & Fluids 179, 579–594 (2019).\n\n\n\nG. Fourtakas, J. M. Dominguez, R. Vacondio and B. D. Rogers. Local uniform stencil (LUST) boundary condition for arbitrary 3-D boundaries in parallel smoothed particle hydrodynamics (SPH) models. Computers & Fluids 190, 346–361 (2019).\n\n\n\nM. Antuono, A. Colagrossi and S. Marrone. Numerical diffusive terms in weakly-compressible SPH schemes. Computer Physics Communications 183, 2570–2580 (2012).\n\n\n\nM. Antuono, A. Colagrossi, S. Marrone and D. Molteni. Free-surface flows solved by means of SPH schemes with numerical diffusive terms. Computer Physics Communications 181, 532–549 (2010).\n\n\n\nD. Molteni and A. Colagrossi. A simple procedure to improve the pressure evaluation in hydrodynamic context using the SPH. Computer Physics Communications 180, 861–872 (2009).\n\n\n\nA. Ferrari, M. Dumbser, E. F. Toro and A. Armanini. A new 3D parallel SPH scheme for free surface flows. Computers & Fluids 38, 1203–1217 (2009).\n\n\n\nN. Akinci, G. Akinci and M. Teschner. Versatile surface tension and adhesion for SPH fluids. ACM Transactions on Graphics 32, 1–8 (2013).\n\n\n\nJ. Bonet and T.-S. Lok. Variational and momentum preservation aspects of Smooth Particle Hydrodynamic formulations. Computer Methods in Applied Mechanics and Engineering 180, 97–115 (1999).\n\n\n\nM. Basa, N. J. Quinlan and M. Lastiwka. Robustness and accuracy of SPH formulations for viscous flow. International Journal for Numerical Methods in Fluids 60, 1127–1148 (2008).\n\n\n\nS. Li and W. K. Liu. Moving least-square reproducing kernel method Part II: Fourier analysis. Computer Methods in Applied Mechanics and Engineering 139, 159–193 (1996).\n\n\n\nJ. R. Clausen. Entropically damped form of artificial compressibility for explicit simulation of incompressible flow. Physical Review E 87, 013309 (2013).\n\n\n\nS. Adami, X. Hu and N. Adams. A transport-velocity formulation for smoothed particle hydrodynamics. Journal of Computational Physics 241, 292–307 (2013).\n\n\n\nJ. O’Connor and B. D. Rogers. A fluid–structure interaction model for free-surface flows and flexible structures using smoothed particle hydrodynamics on a GPU. Journal of Fluids and Structures 104, 103312 (2021).\n\n\n\nT. Belytschko, Y. Guo, W. K. Liu and S. P. Xiao. A unified stability analysis of meshless particle methods. International Journal for Numerical Methods in Engineering 48, 1359–1400 (2000).\n\n\n\nG. C. Ganzenmüller. An hourglass control algorithm for Lagrangian Smooth Particle Hydrodynamics. Computer Methods in Applied Mechanics and Engineering 286, 87–106 (2015).\n\n\n\nA. Valizadeh and J. J. Monaghan. A study of solid wall models for weakly compressible SPH. Journal of Computational Physics 300, 5–19 (2015).\n\n\n\nN. Akinci, M. Ihmsen, G. Akinci, B. Solenthaler and M. Teschner. Versatile rigid-fluid coupling for incompressible SPH. ACM Transactions on Graphics 31, 1–8 (2012).\n\n\n\nA. J. Crespo, M. Gomez-Gesteira and R. A. Dalrymple. Boundary Conditions Generated by Dynamic Particles in SPH Methods. Computers, Materials and Continua 5, 173–184 (2007).\n\n\n\nS. Band, C. Gissler, A. Peer and M. Teschner. MLS pressure boundaries for divergence-free and viscous SPH fluids. Computers & Graphics 76, 37–46 (2018).\n\n\n\nJ. Monaghan and J. Kajtar. SPH particle boundary forces for arbitrary boundaries. Computer Physics Communications 180, 1811–1820 (2009).\n\n\n\nM. B. Giles. Nonreflecting boundary conditions for Euler equation calculations. AIAA Journal 28, 2050–2058 (1990).\n\n\n\nM. Lastiwka, M. Basa and N. J. Quinlan. Permeable and non‐reflecting boundary conditions in SPH. International Journal for Numerical Methods in Fluids 61, 709–724 (2008).\n\n\n\nP. Negi, P. Ramachandran and A. Haftu. An improved non-reflecting outlet boundary condition for weakly-compressible SPH. Computer Methods in Applied Mechanics and Engineering 367, 113119 (2020).\n\n\n\nA. Panizzo, G. Cuomo and R. A. Dalrymple. 3D-SPH SIMULATION OF LANDSLIDE GENERATED WAVES. In: Coastal Engineering 2006 (World Scientific Publishing Company, Apr 2007).\n\n\n\nP. Sun, A. Colagrossi, S. Marrone and A. Zhang. Delta-SPH model: Simple procedures for a further improvement of the SPH scheme. Computer Methods in Applied Mechanics and Engineering 315, 25–49 (2017).\n\n\n\nM. Antuono, S. Marrone, A. Colagrossi and B. Bouscasse. Energy balance in the Delta-SPH scheme. Computer Methods in Applied Mechanics and Engineering 289, 209–226 (2015).\n\n\n\n","category":"page"},{"location":"preprocessing/preprocessing/#Sampling-of-Geometries","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"","category":"section"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Generating the initial configuration of a simulation requires filling volumes (3D) or surfaces (2D) of complex geometries with particles. The algorithm to sample a complex geometry should be robust and fast, since for large problems (large numbers of particles) or complex geometries (many geometry faces), generating the initial configuration is not trivial and can be very expensive in terms of computational cost. We therefore use a winding number approach for an inside-outside segmentation of an object. The winding number w(mathbfp) is a signed integer-valued function of a point mathbfp and is defined as","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"w(mathbfp) = frac12 pi sum^n_i=1 Theta_i","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Here, Theta_i is the signed angle between mathbfc_i - mathbfp and mathbfc_i+1 - mathbfp where mathbfc_i and mathbfc_i+1 are two consecutive vertices on a curve. In 3D, we refer to the solid angle of an oriented triangle with respect to mathbfp.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"We provide the following methods to calculate w(mathbfp):","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Hormann et al. (2001) evaluate the winding number combined with an even-odd rule, but only for 2D polygons (see WindingNumberHormann).\nNaive winding: Jacobson et al. (2013) generalized the winding number so that the algorithm can be applied for both 2D and 3D geometries (see WindingNumberJacobson).\nHierarchical winding: Jacobson et al. (2013) also introduced a fast hierarchical evaluation of the winding number. For further information see the description below.","category":"page"},{"location":"preprocessing/preprocessing/#hierarchical_winding","page":"Sampling of Geometries","title":"Hierarchical Winding","text":"","category":"section"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"According to Jacobson et al. (2013) the winding number with respect to a polygon (2D) or triangle mesh (3D) is the sum of the winding numbers with respect to each edge (2D) or face (3D). We can show this with the following example in which we determine the winding number for each edge of a triangle separately and sum them up:","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"using TrixiParticles\nusing Plots\n\ntriangle = [125.0 375.0 250.0 125.0;\n 175.0 175.0 350.0 175.0]\n\n# Delete all edges but one\nedge1 = deleteat!(TrixiParticles.Polygon(triangle), [2, 3])\nedge2 = deleteat!(TrixiParticles.Polygon(triangle), [1, 3])\nedge3 = deleteat!(TrixiParticles.Polygon(triangle), [1, 2])\n\nalgorithm = WindingNumberJacobson()\n\ngrid = hcat(([x, y] for x in 1:500, y in 1:500)...)\n\n_, w1 = algorithm(edge1, grid; store_winding_number=true)\n_, w2 = algorithm(edge2, grid; store_winding_number=true)\n_, w3 = algorithm(edge3, grid; store_winding_number=true)\n\nw = w1 + w2 + w3\n\nheatmap(1:500, 1:500, reshape(w1, 500, 500)', color=:coolwarm, showaxis=false,\n tickfontsize=12, size=(570, 500), margin=6 * Plots.mm)\nheatmap(1:500, 1:500, reshape(w2, 500, 500)', color=:coolwarm, showaxis=false,\n tickfontsize=12, size=(570, 500), margin=6 * Plots.mm)\nheatmap(1:500, 1:500, reshape(w3, 500, 500)', color=:coolwarm, showaxis=false,\n tickfontsize=12, size=(570, 500), margin=6 * Plots.mm)\nheatmap(1:500, 1:500, reshape(w, 500, 500)', color=:coolwarm, showaxis=false,\n tickfontsize=12, size=(570, 500), margin=6 * Plots.mm, clims=(-1, 1))\n","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"
\n \"triangle\"/\n
","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"This summation property has some interesting consequences that we can utilize for an efficient computation of the winding number. Let mathcalS be an open surface and barmathcalS an arbitrary closing surface, such that","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"partial barmathcalS = partial mathcalS","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"and mathcalB = barmathcalS cup mathcalS is some closed oriented surface. For any query point mathbfp outside of mathcalB, we know that","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"w_mathcalS(mathbfp) + w_barmathcalS(mathbfp) = w_mathcalB(mathbfp) = 0","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"This means","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"w_mathcalS(mathbfp) = - w_barmathcalS(mathbfp)","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"regardless of how barmathcalS is constructed (as long as mathbfp is outside of mathcalB).","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"We can use this property in the discrete case to efficiently compute the winding number of a query point by partitioning the polygon or mesh in a \"small\" part (as in consisting of a small number of edges/faces) and a \"large\" part. For the small part we just compute the winding number, and for the large part we construct a small closing and compute its winding number. The partitioning is based on a hierarchical construction of bounding boxes.","category":"page"},{"location":"preprocessing/preprocessing/#Bounding-volume-hierarchy","page":"Sampling of Geometries","title":"Bounding volume hierarchy","text":"","category":"section"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"To efficiently find a \"small part\" and a \"large part\" as mentioned above, we construct a hierarchy of bounding boxes by starting with the whole domain and recursively splitting it in two equally sized boxes. The resulting hierarchy is a binary tree.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"The algorithm by Jacobsen et al. (Algorithm 2, p. 5) traverses this binary tree recursively until we find the leaf in which the query point is located. The recursion stops with the following criteria:","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"if the bounding box T is a leaf then TmathcalS = mathcalS cap T, the part of mathcalS that lies inside T, is the \"small part\" mentioned above, so evaluate the winding number naively as w(mathbfp TmathcalS).\nelse if mathbfp is outside T then TmathcalS is the \"large part\", so evaluate the winding number naively as -w(mathbfp TbarmathcalS), where TbarmathcalS is the closing surface of TmathcalS.","category":"page"},{"location":"preprocessing/preprocessing/#Continuous-example","page":"Sampling of Geometries","title":"Continuous example","text":"","category":"section"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Now consider the following continuous (not discretized to a polygon) 2D example. We compute the winding number of the point mathbfp with respect to mathcalS using the depicted hierarchy of bounding boxes.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"
\n \"continuous\n
","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"(1):","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Recurse left: w_textleft = texttexttthierarchical_winding (mathbfp Ttextleft)\nRecurse right: w_textright = texttexttthierarchical_winding (mathbfpTtextright)","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"(2):","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Query point mathbfp is outside bounding box T, so don't recurse deeper.\nCompute w_mathcalS(mathbfp) = - w_barmathcalS(mathbfp) with the closure TbarmathcalS, which is generally much smaller (fewer edges in the discrete version) than TmathcalS:","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"w_textleft = -texttextttnaive_winding (mathbfp TbarmathcalS)","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"(3):","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Bounding box T is a leaf. Use open surface TmathcalS:","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"w_textright = texttextttnaive_winding (mathbfp TmathcalS)","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"The reconstructed surface will then look as in the following image.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"
\n \"reconstructed\n
","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"We finally sum up the winding numbers","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"w = w_textleft + w_textright = -w_T_textleftbarmathcalS + w_T_textrightmathcalS","category":"page"},{"location":"preprocessing/preprocessing/#Discrete-example","page":"Sampling of Geometries","title":"Discrete example","text":"","category":"section"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"We will now go through the discrete version of the example above.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"
\n \"discrete\n
","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"To construct the hierarchy for the discrete piecewise-linear example in (1), we have to do the following.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"(2): Each edge is distributed to the child whose box contains the edge's barycenter (red dots in (2)). Splitting stops when the number of a box's edges slips below a threshold (usually approx 100 faces in 3D, here: 6 edges).","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"(3): For the closure, Jacobson et al. (2013) define exterior vertices (exterior edges in 3D) as boundary vertices of such a segmentation (red dots in (3)). To find them, we traverse around each edge (face in 3D) in order, and increment or decrement for each vertex (edge) a specific counter.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"v1 = edge_vertices_ids[edge][1]\nv2 = edge_vertices_ids[edge][2]\n\nvertex_count[v1] += 1\nvertex_count[v2] -= 1","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"In 2D, a vertex is declared as exterior if vertex_count(vertex) != 0, so there is not the same amount of edges in this box going into versus out of the vertex. To construct the closing surface, the exterior vertices are then connected to one arbitrary exterior vertex using appropriately oriented line segments:","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"edge = vertex_count[v] > 0 ? (closing_vertex, v) : (v, closing_vertex)","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"The resulting closed surface TS cup TbarS then has the same number of edges going into and out of each vertex.","category":"page"},{"location":"preprocessing/preprocessing/#Incorrect-evaluation","page":"Sampling of Geometries","title":"Incorrect evaluation","text":"","category":"section"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"If we follow the algorithm, we know that recursion stops if","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"the bounding box T is a leaf or\nthe query point mathbfp is outside the box.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"
\n \"incorrect\n
","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"(1): The query point mathbfp is outside the box, so we calculate the winding number with the (red) closure of the box.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"(2): The query point mathbfp is inside the box, so we use the (blue) edges distributed to the box.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"(3): In this case, it leads to an incorrect evaluation of the winding number. The query point is clearly inside the box, but not inside the reconstructed surface. This is because the property w_mathcalS(mathbfp) = - w_barmathcalS(mathbfp) only holds when mathbfp is outside of mathcalB, which is not the case here.","category":"page"},{"location":"preprocessing/preprocessing/#Correct-evaluation","page":"Sampling of Geometries","title":"Correct evaluation","text":"","category":"section"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Jacobson et al. (2013) don't mention this problem or provide a solution to it. We contacted the authors and found that they know about this problem and solve it by resizing the bounding box to fully include the closing surface of the neighboring box, since it doesn't matter if the boxes overlap.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"
\n \"correct\n
","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"To avoid resizing, we take a different approach and calculate the closure of the bounding box differently:","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Exclude intersecting edges in the calculation of the exterior vertices.\nThis way, all exterior vertices are inside the bounding box, and so will be the closing surface.\nThe intersecting edges are later added with flipped orientation, so that the closing is actually a closing of the exterior plus intersecting edges.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"
\n \"correct\n
","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"The evaluation then looks as follows.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"
\n \"correct\n
","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"preprocessing\", \"point_in_poly\", \"winding_number_hormann.jl\")]","category":"page"},{"location":"preprocessing/preprocessing/#TrixiParticles.WindingNumberHormann","page":"Sampling of Geometries","title":"TrixiParticles.WindingNumberHormann","text":"WindingNumberHormann()\n\nAlgorithm for inside-outside segmentation of a complex geometry proposed by Hormann (2001). It is only supported for 2D geometries. WindingNumberHormann might handle edge cases a bit better, since the winding number is an integer value.\n\nwarning: Experimental Implementation\nThis is an experimental feature and may change in any future releases.\n\n\n\n\n\n","category":"type"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"preprocessing\", \"point_in_poly\", \"winding_number_jacobson.jl\")]","category":"page"},{"location":"preprocessing/preprocessing/#TrixiParticles.WindingNumberJacobson","page":"Sampling of Geometries","title":"TrixiParticles.WindingNumberJacobson","text":"WindingNumberJacobson(; geometry=nothing, winding_number_factor=sqrt(eps()),\n hierarchical_winding=false)\n\nAlgorithm for inside-outside segmentation of a complex geometry proposed by [2].\n\nKeywords\n\ngeometry: Complex geometry returned by load_geometry and is only required when using hierarchical_winding=true.\nhierarchical_winding: If set to true, an optimized hierarchical approach will be used, which gives a significant speedup. For further information see Hierarchical Winding.\nwinding_number_factor: For leaky geometries, a factor of 0.4 will give a better inside-outside segmentation.\n\nwarning: Experimental Implementation\nThis is an experimental feature and may change in any future releases.\n\n\n\n\n\n","category":"type"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"preprocessing\", \"geometries\", \"io.jl\")]","category":"page"},{"location":"preprocessing/preprocessing/#TrixiParticles.load_geometry-Tuple{Any}","page":"Sampling of Geometries","title":"TrixiParticles.load_geometry","text":"load_geometry(filename; element_type=Float64)\n\nLoad file and return corresponding type for ComplexShape. Supported file formats are .stl and .asc.\n\nArguments\n\nfilename: Name of the file to be loaded.\n\nKeywords\n\nelement_type: Element type (default is Float64)\n\n\n\n\n\n","category":"method"},{"location":"systems/dem/#dem","page":"Discrete Element Method (Solid)","title":"Discrete Element Method","text":"","category":"section"},{"location":"systems/dem/","page":"Discrete Element Method (Solid)","title":"Discrete Element Method (Solid)","text":"The Discrete Element Method (DEM) is a computational technique widely used in physics, engineering, and applied mathematics for simulating the mechanical behavior of granular materials, such as powders, sand, soil, or rock, as well as other discontinua. Unlike continuum mechanics that treats materials as continuous, DEM considers individual particles or elements and their interactions. This approach provides detailed insights into the micro-mechanical behavior of materials, making it particularly valuable in fields such as geomechanics, material science, and mechanical engineering.","category":"page"},{"location":"systems/dem/#Fundamental-Principles","page":"Discrete Element Method (Solid)","title":"Fundamental Principles","text":"","category":"section"},{"location":"systems/dem/","page":"Discrete Element Method (Solid)","title":"Discrete Element Method (Solid)","text":"The core idea behind DEM is the discretization of a material system into a finite set of distinct, interacting mass elements (particles). These elements (particles) can vary in shape, size, and properties, and they interact with each other and possibly with their boundaries through contact forces and potential fields. The motion and behavior of each mass element are governed by Newton's laws of motion, accounting for the forces and moments acting upon them.","category":"page"},{"location":"systems/dem/","page":"Discrete Element Method (Solid)","title":"Discrete Element Method (Solid)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"solid\", \"discrete_element_method\", \"system.jl\")]","category":"page"},{"location":"systems/dem/#TrixiParticles.DEMSystem","page":"Discrete Element Method (Solid)","title":"TrixiParticles.DEMSystem","text":"DEMSystem(initial_condition, normal_stiffness, elastic_modulus, poissons_ratio;\n damping_coefficient=0.0001, acceleration=ntuple(_ -> 0.0, NDIMS), source_terms=nothing)\n\nConstructs a Discrete Element Method (DEM) system for numerically simulating the dynamics of granular and particulate matter. DEM is employed to simulate and analyze the motion, interactions, and collective behavior of assemblies of discrete, solid particles, typically under mechanical loading. The model accounts for individual particle characteristics and implements interaction laws that govern contact forces (normal and tangential), based on specified material properties and contact mechanics.\n\nArguments\n\ninitial_condition: Initial condition of the system, encapsulating the initial positions, velocities, masses, and radii of particles.\nnormal_stiffness: Normal stiffness coefficient for particle-particle and particle-wall contacts.\nelastic_modulus: Elastic modulus for this particle system.\npoissons_ratio: Poisson ratio for this particle system.\n\nKeywords\n\nacceleration: Global acceleration vector applied to the system, such as gravity. Specified as an SVector of length NDIMS, with a default of zero in each dimension.\nsource_terms: Optional; additional forces or modifications to particle dynamics not captured by standard DEM interactions, such as electromagnetic forces or user-defined perturbations.\ndamping_coefficient=0.0001: Set a damping coefficient for the collision interactions.\n\nwarning: Experimental Implementation\nThis is an experimental feature and may change in a future releases.\n\nReferences\n\n[9], [10], [11]\n\n\n\n\n\n","category":"type"},{"location":"general/smoothing_kernels/#smoothing_kernel","page":"Smoothing Kernels","title":"Smoothing Kernels","text":"","category":"section"},{"location":"general/smoothing_kernels/","page":"Smoothing Kernels","title":"Smoothing Kernels","text":"The following smoothing kernels are currently available:","category":"page"},{"location":"general/smoothing_kernels/","page":"Smoothing Kernels","title":"Smoothing Kernels","text":"Smoothing Kernel Compact Support Typ. Smoothing Length Recommended Application Stability\nSchoenbergCubicSplineKernel 0 2h 11 to 13 General + sharp waves ++\nSchoenbergQuarticSplineKernel 0 25h 11 to 15 General +++\nSchoenbergQuinticSplineKernel 0 3h 11 to 15 General ++++\nGaussianKernel 0 3h 10 to 15 Literature +++++\nWendlandC2Kernel 0 1h 25 to 40 General (recommended) ++++\nWendlandC4Kernel 0 1h 30 to 45 General +++++\nWendlandC6Kernel 0 1h 35 to 50 General +++++\nPoly6Kernel 0 1h 15 to 25 Literature +\nSpikyKernel 0 1h 15 to 30 Sharp corners + waves +","category":"page"},{"location":"general/smoothing_kernels/","page":"Smoothing Kernels","title":"Smoothing Kernels","text":"We recommend to use the WendlandC2Kernel for most applications. If less smoothing is needed, try SchoenbergCubicSplineKernel, for more smoothing try WendlandC6Kernel.","category":"page"},{"location":"general/smoothing_kernels/","page":"Smoothing Kernels","title":"Smoothing Kernels","text":"note: Usage\nThe kernel can be called asTrixiParticles.kernel(smoothing_kernel, r, h)The length of the compact support can be obtained asTrixiParticles.compact_support(smoothing_kernel, h)Note that r has to be a scalar, so in the context of SPH, the kernel should be used asW(Vert r_a - r_b Vert h)The gradient required in SPH, nabla_r_a W(Vert r_a - r_b Vert h)can be called asTrixiParticles.kernel_grad(smoothing_kernel, pos_diff, distance, h)where pos_diff is r_a - r_b and distance is Vert r_a - r_b Vert.","category":"page"},{"location":"general/smoothing_kernels/","page":"Smoothing Kernels","title":"Smoothing Kernels","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"general\", \"smoothing_kernels.jl\")]","category":"page"},{"location":"general/smoothing_kernels/#TrixiParticles.GaussianKernel","page":"Smoothing Kernels","title":"TrixiParticles.GaussianKernel","text":"GaussianKernel{NDIMS}()\n\nGaussian kernel given by\n\nW(r h) = fracsigma_dh^d e^-r^2h^2\n\nwhere d is the number of dimensions and\n\nsigma_2 = frac1pi for 2D,\nsigma_3 = frac1pi^32 for 3D.\n\nThis kernel function has an infinite support, but in practice, it's often truncated at a certain multiple of h, such as 3h.\n\nIn this implementation, the kernel is truncated at 3h, so this kernel function has a compact support of 0 3h.\n\nThe smoothing length is typically in the range 10delta 15delta, where delta is the typical particle spacing.\n\nFor general information and usage see Smoothing Kernels.\n\nNote: This truncation makes this Kernel not conservative, which is beneficial in regards to stability but makes it less accurate.\n\n\n\n\n\n","category":"type"},{"location":"general/smoothing_kernels/#TrixiParticles.Poly6Kernel","page":"Smoothing Kernels","title":"TrixiParticles.Poly6Kernel","text":"Poly6Kernel{NDIMS}()\n\nPoly6 kernel, a commonly used kernel in SPH literature [3], especially in computer graphics contexts. It is defined as\n\nW(r h) = frac1h^d w(rh)\n\nwith\n\nw(q) = sigma begincases\n (1 - q^2)^3 textif 0 leq q 1 \n 0 textif q geq 1\nendcases\n\nwhere d is the number of dimensions and sigma is a normalization factor that depends on the dimension. The normalization factor sigma is 4 pi in two dimensions or 315 64pi in three dimensions.\n\nThis kernel function has a compact support of 0 h.\n\nPoly6 is well-known for its computational simplicity, though it's worth noting that there are other kernels that might offer better accuracy for hydrodynamic simulations. Furthermore, its derivatives are not that smooth, which can lead to stability problems. It is also susceptible to clumping.\n\nThe smoothing length is typically in the range 15delta 25delta, where delta is the typical particle spacing.\n\nFor general information and usage see Smoothing Kernels.\n\n\n\n\n\n","category":"type"},{"location":"general/smoothing_kernels/#TrixiParticles.SchoenbergCubicSplineKernel","page":"Smoothing Kernels","title":"TrixiParticles.SchoenbergCubicSplineKernel","text":"SchoenbergCubicSplineKernel{NDIMS}()\n\nCubic spline kernel by Schoenberg (1946), given by\n\n W(r h) = frac1h^d w(rh)\n\nwith\n\nw(q) = sigma begincases\n frac14 (2 - q)^3 - (1 - q)^3 textif 0 leq q 1 \n frac14 (2 - q)^3 textif 1 leq q 2 \n 0 textif q geq 2 \nendcases\n\nwhere d is the number of dimensions and sigma is a normalization constant given by sigma =frac23 frac107 pi frac1pi in 1 2 3 dimensions.\n\nThis kernel function has a compact support of 0 2h.\n\nFor an overview of Schoenberg cubic, quartic and quintic spline kernels including normalization factors, see Price (2012). For an analytic formula for higher order Schoenberg kernels, see Monaghan (1985). The largest disadvantage of Schoenberg Spline Kernel is the rather non-smooth first derivative, which can lead to increased noise compared to other kernel variants.\n\nThe smoothing length is typically in the range 11delta 13delta, where delta is the typical particle spacing.\n\nFor general information and usage see Smoothing Kernels.\n\n\n\n\n\n","category":"type"},{"location":"general/smoothing_kernels/#TrixiParticles.SchoenbergQuarticSplineKernel","page":"Smoothing Kernels","title":"TrixiParticles.SchoenbergQuarticSplineKernel","text":"SchoenbergQuarticSplineKernel{NDIMS}()\n\nQuartic spline kernel by Schoenberg (1946), given by\n\n W(r h) = frac1h^d w(rh)\n\nwith\n\nw(q) = sigma begincases\n left(52 - q right)^4 - 5left(32 - q right)^4\n + 10left(12 - q right)^4 textif 0 leq q frac12 \n left(52 - q right)^4 - 5left(32 - q right)^4\n textif frac12 leq q frac32 \n left(52 - q right)^4 textif frac32 leq q frac52 \n 0 textif q geq frac52\nendcases\n\nwhere d is the number of dimensions and sigma is a normalization constant given by sigma =frac124 frac961199 pi frac120pi in 1 2 3 dimensions.\n\nThis kernel function has a compact support of 0 25h.\n\nFor an overview of Schoenberg cubic, quartic and quintic spline kernels including normalization factors, see Price (2012). For an analytic formula for higher order Schoenberg kernels, see Monaghan (1985).\n\nThe largest disadvantage of Schoenberg Spline Kernel are the rather non-smooth first derivative, which can lead to increased noise compared to other kernel variants.\n\nThe smoothing length is typically in the range 11delta 15delta, where delta is the typical particle spacing.\n\nFor general information and usage see Smoothing Kernels.\n\n\n\n\n\n","category":"type"},{"location":"general/smoothing_kernels/#TrixiParticles.SchoenbergQuinticSplineKernel","page":"Smoothing Kernels","title":"TrixiParticles.SchoenbergQuinticSplineKernel","text":"SchoenbergQuinticSplineKernel{NDIMS}()\n\nQuintic spline kernel by Schoenberg (1946), given by\n\n W(r h) = frac1h^d w(rh)\n\nwith\n\nw(q) = sigma begincases\n (3 - q)^5 - 6(2 - q)^5 + 15(1 - q)^5 textif 0 leq q 1 \n (3 - q)^5 - 6(2 - q)^5 textif 1 leq q 2 \n (3 - q)^5 textif 2 leq q 3 \n 0 textif q geq 3\nendcases\n\nwhere d is the number of dimensions and sigma is a normalization constant given by sigma =frac1120 frac7478 pi frac1120pi in 1 2 3 dimensions.\n\nThis kernel function has a compact support of 0 3h.\n\nFor an overview of Schoenberg cubic, quartic and quintic spline kernels including normalization factors, see Price (2012). For an analytic formula for higher order Schoenberg kernels, see Monaghan (1985).\n\nThe largest disadvantage of Schoenberg Spline Kernel are the rather non-smooth first derivative, which can lead to increased noise compared to other kernel variants.\n\nThe smoothing length is typically in the range 11delta 15delta, where delta is the typical particle spacing.\n\nFor general information and usage see Smoothing Kernels.\n\n\n\n\n\n","category":"type"},{"location":"general/smoothing_kernels/#TrixiParticles.SpikyKernel","page":"Smoothing Kernels","title":"TrixiParticles.SpikyKernel","text":"SpikyKernel{NDIMS}()\n\nThe Spiky kernel is another frequently used kernel in SPH, especially due to its desirable properties in preserving features near boundaries in fluid simulations [3]. It is defined as:\n\n W(r h) = frac1h^d w(rh)\n\nwith:\n\nw(q) = sigma begincases\n (1 - q)^3 textif 0 leq q 1 \n 0 textif q geq 1\nendcases\n\nwhere d is the number of dimensions and the normalization factor sigma is 10 pi in two dimensions or 15 pi in three dimensions.\n\nThis kernel function has a compact support of 0 h.\n\nThe Spiky kernel is particularly known for its sharp gradients, which can help to preserve sharp features in fluid simulations, especially near solid boundaries. These sharp gradients at the boundary are also the largest disadvantage as they can lead to instability.\n\nThe smoothing length is typically in the range 15delta 30delta, where delta is the typical particle spacing.\n\nFor general information and usage see Smoothing Kernels.\n\n\n\n\n\n","category":"type"},{"location":"general/smoothing_kernels/#TrixiParticles.WendlandC2Kernel","page":"Smoothing Kernels","title":"TrixiParticles.WendlandC2Kernel","text":"WendlandC2Kernel{NDIMS}()\n\nWendland C2 kernel [7], a piecewise polynomial function designed to have compact support and to be twice continuously differentiable everywhere. Given by\n\n W(r h) = frac1h^d w(rh)\n\nwith\n\nw(q) = sigma begincases\n (1 - q)^4 (4q + 1) textif 0 leq q 1 \n 0 textif q geq 1\nendcases\n\nwhere d is the number of dimensions and sigma is a normalization factor dependent on the dimension. The normalization factor sigma is 407pi in two dimensions or 212pi in three dimensions.\n\nThis kernel function has a compact support of 0 h.\n\nFor a detailed discussion on Wendland functions and their applications in SPH, see Dehnen (2012). The smoothness of these functions is also the largest disadvantage as they lose details at sharp corners.\n\nThe smoothing length is typically in the range 25delta 40delta, where delta is the typical particle spacing.\n\nFor general information and usage see Smoothing Kernels.\n\n\n\n\n\n","category":"type"},{"location":"general/smoothing_kernels/#TrixiParticles.WendlandC4Kernel","page":"Smoothing Kernels","title":"TrixiParticles.WendlandC4Kernel","text":"WendlandC4Kernel{NDIMS}()\n\nWendland C4 kernel [7], a piecewise polynomial function designed to have compact support and to be four times continuously differentiable everywhere. Given by\n\n W(r h) = frac1h^d w(rh)\n\nwith\n\nw(q) = sigma begincases\n (1 - q)^6 (35q^2 3 + 6q + 1) textif 0 leq q 1 \n 0 textif q geq 1\nendcases\n\nwhere d is the number of dimensions and sigma is a normalization factor dependent on the dimension. The normalization factor sigma is 9 pi in two dimensions or 495 32pi in three dimensions.\n\nThis kernel function has a compact support of 0 h.\n\nFor a detailed discussion on Wendland functions and their applications in SPH, see Dehnen (2012). The smoothness of these functions is also the largest disadvantage as they loose details at sharp corners.\n\nThe smoothing length is typically in the range 30delta 45delta, where delta is the typical particle spacing.\n\nFor general information and usage see Smoothing Kernels.\n\n\n\n\n\n","category":"type"},{"location":"general/smoothing_kernels/#TrixiParticles.WendlandC6Kernel","page":"Smoothing Kernels","title":"TrixiParticles.WendlandC6Kernel","text":"WendlandC6Kernel{NDIMS}()\n\nWendland C6 kernel [7], a piecewise polynomial function designed to have compact support and to be six times continuously differentiable everywhere. Given by:\n\nW(r h) = frac1h^d w(rh)\n\nwith:\n\nw(q) = sigma begincases\n (1 - q)^8 (32q^3 + 25q^2 + 8q + 1) textif 0 leq q 1 \n 0 textif q geq 1\nendcases\n\nwhere d is the number of dimensions and sigma is a normalization factor dependent on the dimension. The normalization factor sigma is 78 7 pi in two dimensions or 1365 64pi in three dimensions.\n\nThis kernel function has a compact support of 0 h.\n\nFor a detailed discussion on Wendland functions and their applications in SPH, Dehnen (2012). The smoothness of these functions is also the largest disadvantage as they loose details at sharp corners.\n\nThe smoothing length is typically in the range 35delta 50delta, where delta is the typical particle spacing.\n\nFor general information and usage see Smoothing Kernels.\n\n\n\n\n\n","category":"type"},{"location":"reference-trixibase/#TrixiBase.jl-API","page":"TrixiBase.jl API Reference","title":"TrixiBase.jl API","text":"","category":"section"},{"location":"reference-trixibase/","page":"TrixiBase.jl API Reference","title":"TrixiBase.jl API Reference","text":"CurrentModule = TrixiBase","category":"page"},{"location":"reference-trixibase/","page":"TrixiBase.jl API Reference","title":"TrixiBase.jl API Reference","text":"Modules = [TrixiBase]","category":"page"},{"location":"reference-trixibase/#TrixiBase.disable_debug_timings-Tuple{}","page":"TrixiBase.jl API Reference","title":"TrixiBase.disable_debug_timings","text":"disable_debug_timings()\n\nDisable all @trixi_timeit timings. The timings should be optimized away, allowing for truly zero-overhead. Enable timings again with enable_debug_timings.\n\nSee also enable_debug_timings, @trixi_timeit.\n\n\n\n\n\n","category":"method"},{"location":"reference-trixibase/#TrixiBase.enable_debug_timings-Tuple{}","page":"TrixiBase.jl API Reference","title":"TrixiBase.enable_debug_timings","text":"enable_debug_timings()\n\nEnable all @trixi_timeit timings (default behavior).\n\nSee also disable_debug_timings, @trixi_timeit.\n\n\n\n\n\n","category":"method"},{"location":"reference-trixibase/#TrixiBase.timer-Tuple{}","page":"TrixiBase.jl API Reference","title":"TrixiBase.timer","text":"timer()\n\nMain timer for global timing, e.g., to be used with @trixi_timeit.\n\n\n\n\n\n","category":"method"},{"location":"reference-trixibase/#TrixiBase.trixi_include-Tuple{Module, AbstractString}","page":"TrixiBase.jl API Reference","title":"TrixiBase.trixi_include","text":"trixi_include([mod::Module=Main,] elixir::AbstractString; kwargs...)\n\ninclude the file elixir and evaluate its content in the global scope of module mod. You can override specific assignments in elixir by supplying keyword arguments. Its basic purpose is to make it easier to modify some parameters while running simulations from the REPL. Additionally, this is used in tests to reduce the computational burden for CI while still providing examples with sensible default values for users.\n\nBefore replacing assignments in elixir, the keyword argument maxiters is inserted into calls to solve with it's default value used in the SciML ecosystem for ODEs, see the \"Miscellaneous\" section of the documentation.\n\nExamples\n\njulia> using TrixiBase, Trixi\n\njulia> redirect_stdout(devnull) do\n trixi_include(@__MODULE__, joinpath(examples_dir(), \"tree_1d_dgsem\", \"elixir_advection_extended.jl\"),\n tspan=(0.0, 0.1))\n sol.t[end]\n end\n[ Info: You just called `trixi_include`. Julia may now compile the code, please be patient.\n0.1\n\n\n\n\n\n","category":"method"},{"location":"reference-trixibase/#TrixiBase.@trixi_timeit-Tuple{Any, Any, Any}","page":"TrixiBase.jl API Reference","title":"TrixiBase.@trixi_timeit","text":"@trixi_timeit timer() \"some label\" expression\n\nBasically the same as a special case of @timeit_debug from TimerOutputs.jl, but without try ... finally ... end block. Thus, it's not exception-safe, but it also avoids some related performance problems. Since we do not use exception handling in Trixi.jl, that's not really an issue.\n\nAll @trixi_timeit timings can be disabled with disable_debug_timings. The timings should then be optimized away, allowing for truly zero-overhead.\n\nSee also disable_debug_timings, enable_debug_timings.\n\n\n\n\n\n","category":"macro"},{"location":"tutorials_template/tut_beam/#Example-file","page":"Example file","title":"Example file","text":"","category":"section"},{"location":"tutorials_template/tut_beam/","page":"Example file","title":"Example file","text":"!!include:examples/solid/oscillating_beam_2d.jl!!\n","category":"page"},{"location":"contributing/","page":"Contributing","title":"Contributing","text":"EditURL = \"https://github.com/trixi-framework/TrixiParticles.jl/blob/main/CONTRIBUTING.md\"","category":"page"},{"location":"contributing/#Contributing","page":"Contributing","title":"Contributing","text":"","category":"section"},{"location":"contributing/","page":"Contributing","title":"Contributing","text":"TrixiParticles.jl is an open-source project and we are very happy to accept contributions from the community. Please feel free to open issues or submit patches (preferably as pull requests) any time. For planned larger contributions, it is often beneficial to get in contact with one of the principal developers first (see Authors).","category":"page"},{"location":"contributing/","page":"Contributing","title":"Contributing","text":"TrixiParticles.jl and its contributions are licensed under the MIT license (see License). As a contributor, you certify that all your contributions are in conformance with the Developer Certificate of Origin (Version 1.1), which is reproduced below.","category":"page"},{"location":"contributing/#Developer-Certificate-of-Origin-(Version-1.1)","page":"Contributing","title":"Developer Certificate of Origin (Version 1.1)","text":"","category":"section"},{"location":"contributing/","page":"Contributing","title":"Contributing","text":"The following text was taken from https://developercertificate.org:","category":"page"},{"location":"contributing/","page":"Contributing","title":"Contributing","text":"Developer Certificate of Origin\nVersion 1.1\n\nCopyright (C) 2004, 2006 The Linux Foundation and its contributors.\n1 Letterman Drive\nSuite D4700\nSan Francisco, CA, 94129\n\nEveryone is permitted to copy and distribute verbatim copies of this\nlicense document, but changing it is not allowed.\n\n\nDeveloper's Certificate of Origin 1.1\n\nBy making a contribution to this project, I certify that:\n\n(a) The contribution was created in whole or in part by me and I\n have the right to submit it under the open source license\n indicated in the file; or\n\n(b) The contribution is based upon previous work that, to the best\n of my knowledge, is covered under an appropriate open source\n license and I have the right under that license to submit that\n work with modifications, whether created in whole or in part\n by me, under the same open source license (unless I am\n permitted to submit under a different license), as indicated\n in the file; or\n\n(c) The contribution was provided directly to me by some other\n person who certified (a), (b) or (c) and I have not modified\n it.\n\n(d) I understand and agree that this project and the contribution\n are public and that a record of the contribution (including all\n personal information I submit with it, including my sign-off) is\n maintained indefinitely and may be redistributed consistent with\n this project or the open source license(s) involved.","category":"page"},{"location":"development/#development","page":"Development","title":"Development","text":"","category":"section"},{"location":"development/#Preview-of-the-documentation","page":"Development","title":"Preview of the documentation","text":"","category":"section"},{"location":"development/","page":"Development","title":"Development","text":"To generate the Documentation, first instantiate the docs environment by executing the following command from the TrixiParticles.jl root directory:","category":"page"},{"location":"development/","page":"Development","title":"Development","text":"julia --project=docs -e \"using Pkg; Pkg.develop(PackageSpec(path=pwd())); Pkg.instantiate()\"","category":"page"},{"location":"development/","page":"Development","title":"Development","text":"This command only has to be run once. After that, maintain the docs environment as described under Installation.","category":"page"},{"location":"development/","page":"Development","title":"Development","text":"With an instantiated docs environment, generate the docs with the following command (again from the TrixiParticles.jl root directory):","category":"page"},{"location":"development/","page":"Development","title":"Development","text":"julia --project=docs --color=yes docs/make.jl","category":"page"},{"location":"development/","page":"Development","title":"Development","text":"You can then open the generated files in docs/build with your webbrowser. Alternatively, run","category":"page"},{"location":"development/","page":"Development","title":"Development","text":"python3 -m http.server -d docs/build","category":"page"},{"location":"development/","page":"Development","title":"Development","text":"and open localhost:8000 in your webbrowser.","category":"page"},{"location":"development/#Release-management","page":"Development","title":"Release management","text":"","category":"section"},{"location":"development/","page":"Development","title":"Development","text":"To create a new release for TrixiParticles.jl, perform the following steps:","category":"page"},{"location":"development/","page":"Development","title":"Development","text":"Make sure that all PRs and changes that you want to go into the release are merged to main and that the latest commit on main has passed all CI tests.\nDetermine the currently released version of TrixiParticles.jl, e.g., on the release page. For this manual, we will assume that the latest release was v0.2.3.\nDecide on the next version number. We follow semantic versioning, thus each version is of the form vX.Y.Z where X is the major version, Y the minor version, and Z the patch version. In this manual, we assume that the major version is always 0, thus the decision process on the new version is as follows:\nIf the new release contains breaking changes (i.e., user code might not work as before without modifications), increase the minor version by one and set the patch version to zero. In our example, the new version should thus be v0.3.0.\nIf the new release only contains minor modifications and/or bug fixes, the minor version is kept as-is and the patch version is increased by one. In our example, the new version should thus be v0.2.4.\nEdit the version string in the Project.toml and set it to the new version. Push/merge this change to main.\nGo to GitHub and add a comment to the commit that you would like to become the new release (typically this will be the commit where you just updated the version). You can comment on a commit by going to the commit overview and clicking on the title of the commit. The comment should contain the following text:\n@JuliaRegistrator register\nWait for the magic to happen! Specifically, JuliaRegistrator will create a new PR to the Julia registry with the new release information. After a grace period of ~15 minutes, this PR will be merged automatically. A short while after, TagBot will create a new release of TrixiParticles.jl in our GitHub repository.\nOnce the new release has been created, the new version can be obtained through the Julia package manager as usual.\nTo make sure people do not mistake the latest state of main as the latest release, we set the version in the Project.toml to a development version. The development version should be the latest released version, with the patch version incremented by one, and the -dev suffix added. For example, if you just released v0.3.0, the new development version should be v0.3.1-dev. If you just released v0.2.4, the new development version should be v0.2.5-dev.","category":"page"},{"location":"install/#installation","page":"Installation","title":"Installation","text":"","category":"section"},{"location":"install/#Setting-up-Julia","page":"Installation","title":"Setting up Julia","text":"","category":"section"},{"location":"install/","page":"Installation","title":"Installation","text":"If you have not yet installed Julia, please follow the instructions on the official website. TrixiParticles.jl works with Julia v1.9 and newer. We recommend using the latest stable release of Julia.","category":"page"},{"location":"install/#For-users","page":"Installation","title":"For users","text":"","category":"section"},{"location":"install/","page":"Installation","title":"Installation","text":"TrixiParticles.jl is a registered Julia package. You can install TrixiParticles.jl, OrdinaryDiffEq.jl (used for time integration) and Plots.jl by executing the following commands in the Julia REPL:","category":"page"},{"location":"install/","page":"Installation","title":"Installation","text":"julia> using Pkg\n\njulia> Pkg.add([\"TrixiParticles\", \"OrdinaryDiffEq\", \"Plots\"])","category":"page"},{"location":"install/#for-developers","page":"Installation","title":"For developers","text":"","category":"section"},{"location":"install/","page":"Installation","title":"Installation","text":"If you plan on editing TrixiParticles.jl itself, you can download TrixiParticles.jl to a local folder and use the code from the cloned directory:","category":"page"},{"location":"install/","page":"Installation","title":"Installation","text":"git clone git@github.com:trixi-framework/TrixiParticles.jl.git\ncd TrixiParticles.jl\nmkdir run\njulia --project=run -e 'using Pkg; Pkg.develop(PackageSpec(path=\".\"))' # Add TrixiParticles.jl to `run` project\njulia --project=run -e 'using Pkg; Pkg.add([\"OrdinaryDiffEq\", \"Plots\"])' # Add additional packages","category":"page"},{"location":"install/","page":"Installation","title":"Installation","text":"If you installed TrixiParticles.jl this way, you always have to start Julia with the --project flag set to your run directory, e.g.,","category":"page"},{"location":"install/","page":"Installation","title":"Installation","text":"julia --project=run","category":"page"},{"location":"install/","page":"Installation","title":"Installation","text":"from the TrixiParticles.jl root directory.","category":"page"},{"location":"install/","page":"Installation","title":"Installation","text":"The advantage of using a separate run directory is that you can also add other related packages (e.g., OrdinaryDiffEq.jl, see above) to the project in the run folder and always have a reproducible environment at hand to share with others.","category":"page"},{"location":"install/#Optional-software/packages","page":"Installation","title":"Optional software/packages","text":"","category":"section"},{"location":"install/","page":"Installation","title":"Installation","text":"OrdinaryDiffEq.jl – A Julia package of ordinary differential equation solvers that is used in the examples\nPlots.jl – Julia Plotting library that is used in some examples\nPythonPlot.jl – Plotting library that can be used instead of Plots.jl\nParaView – Software that can be used for visualization of results","category":"page"},{"location":"install/#installation-issues","page":"Installation","title":"Common issues","text":"","category":"section"},{"location":"install/","page":"Installation","title":"Installation","text":"If you followed the installation instructions for developers and you run into any problems with packages when pulling the latest version of TrixiParticles.jl, start Julia with the project in the run folder,","category":"page"},{"location":"install/","page":"Installation","title":"Installation","text":" julia --project=run","category":"page"},{"location":"install/","page":"Installation","title":"Installation","text":"update all packages in that project, resolve all conflicts in the project, and install all new dependencies:","category":"page"},{"location":"install/","page":"Installation","title":"Installation","text":"julia> using Pkg\n\njulia> Pkg.update()\n\njulia> Pkg.resolve()\n\njulia> Pkg.instantiate()","category":"page"},{"location":"visualization/#Visualization","page":"Visualization","title":"Visualization","text":"","category":"section"},{"location":"visualization/#Export-VTK-files","page":"Visualization","title":"Export VTK files","text":"","category":"section"},{"location":"visualization/","page":"Visualization","title":"Visualization","text":"You can export particle data as VTK files by using the SolutionSavingCallback. All our predefined examples are already using this callback to export VTK files to the out directory (relative to the directory that you are running Julia from). VTK files can be read by visualization tools like ParaView and VisIt.","category":"page"},{"location":"visualization/#ParaView","page":"Visualization","title":"ParaView","text":"","category":"section"},{"location":"visualization/","page":"Visualization","title":"Visualization","text":"Follow these steps to view the exported VTK files in ParaView:","category":"page"},{"location":"visualization/","page":"Visualization","title":"Visualization","text":"Click File -> Open.\nNavigate to the out directory (relative to the directory that you are running Julia from).\nOpen both boundary_1.pvd and fluid_1.pvd.\nClick \"Apply\", which by default is on the left pane below the \"Pipeline Browser\".\nHold the left mouse button to move the solution around.","category":"page"},{"location":"visualization/","page":"Visualization","title":"Visualization","text":"You will now see the following: (Image: image)","category":"page"},{"location":"visualization/","page":"Visualization","title":"Visualization","text":"It is useful to make the particles larger. For this, first make sure you have \"fluid_1.pvd\" highlighted in the \"Pipeline Browser\" then in the \"Properties\" window in the bottom left change \"Point Size\" to a larger value. (Image: image)","category":"page"},{"location":"visualization/","page":"Visualization","title":"Visualization","text":"To now view the result variables first make sure you have \"fluid_1.pvd\" highlighted in the \"Pipeline Browser\" then select them in the variable selection combo box (see picture below). Let's, for example, pick \"density\". To now view the time progression of the result hit the \"play button\" (see picture below). (Image: image)","category":"page"},{"location":"visualization/#API","page":"Visualization","title":"API","text":"","category":"section"},{"location":"visualization/","page":"Visualization","title":"Visualization","text":"Modules = [TrixiParticles]\nPages = map(file -> joinpath(\"visualization\", file), readdir(joinpath(\"..\", \"src\", \"visualization\")))","category":"page"},{"location":"visualization/#TrixiParticles.trixi2vtk-Tuple{Any, Any, Any}","page":"Visualization","title":"TrixiParticles.trixi2vtk","text":"trixi2vtk(vu_ode, semi, t; iter=nothing, output_directory=\"out\", prefix=\"\",\n write_meta_data=true, max_coordinates=Inf, custom_quantities...)\n\nConvert Trixi simulation data to VTK format.\n\nArguments\n\nvu_ode: Solution of the TrixiParticles ODE system at one time step. This expects an ArrayPartition as returned in the examples as sol.u[end].\nsemi: Semidiscretization of the TrixiParticles simulation.\nt: Current time of the simulation.\n\nKeywords\n\niter=nothing: Iteration number when multiple iterations are to be stored in separate files. This number is just appended to the filename.\noutput_directory=\"out\": Output directory path.\nprefix=\"\": Prefix for output files.\nwrite_meta_data=true: Write meta data.\nmax_coordinates=Inf The coordinates of particles will be clipped if their absolute values exceed this threshold.\ncustom_quantities...: Additional custom quantities to include in the VTK output. Each custom quantity must be a function of (v, u, t, system), which will be called for every system, where v and u are the wrapped solution arrays for the corresponding system and t is the current simulation time. Note that working with these v and u arrays requires undocumented internal functions of TrixiParticles. See Custom Quantities for a list of pre-defined custom quantities that can be used here.\n\nExample\n\ntrixi2vtk(sol.u[end], semi, 0.0, iter=1, output_directory=\"output\", prefix=\"solution\")\n\n# Additionally store the kinetic energy of each system as \"my_custom_quantity\"\ntrixi2vtk(sol.u[end], semi, 0.0, iter=1, my_custom_quantity=kinetic_energy)\n\n\n\n\n\n","category":"method"},{"location":"visualization/#TrixiParticles.trixi2vtk-Tuple{Any}","page":"Visualization","title":"TrixiParticles.trixi2vtk","text":"trixi2vtk(coordinates; output_directory=\"out\", prefix=\"\", filename=\"coordinates\",\n custom_quantities...)\n\nConvert coordinate data to VTK format.\n\nArguments\n\ncoordinates: Coordinates to be saved.\n\nKeywords\n\noutput_directory=\"out\": Output directory path.\nprefix=\"\": Prefix for the output file.\nfilename=\"coordinates\": Name of the output file.\ncustom_quantities...: Additional custom quantities to include in the VTK output.\n\nReturns\n\nfile::AbstractString: Path to the generated VTK file.\n\n\n\n\n\n","category":"method"},{"location":"visualization/#TrixiParticles.trixi2vtk-Tuple{InitialCondition}","page":"Visualization","title":"TrixiParticles.trixi2vtk","text":"trixi2vtk(initial_condition::InitialCondition; output_directory=\"out\",\n prefix=\"\", filename=\"initial_condition\", custom_quantities...)\n\nConvert InitialCondition data to VTK format.\n\nArguments\n\ninitial_condition: InitialCondition to be saved.\n\nKeywords\n\noutput_directory=\"out\": Output directory path.\nprefix=\"\": Prefix for the output file.\nfilename=\"coordinates\": Name of the output file.\ncustom_quantities...: Additional custom quantities to include in the VTK output.\n\nReturns\n\nfile::AbstractString: Path to the generated VTK file.\n\n\n\n\n\n","category":"method"},{"location":"general/initial_condition/#initial_condition","page":"Initial Condition and Setups","title":"Initial Condition","text":"","category":"section"},{"location":"general/initial_condition/","page":"Initial Condition and Setups","title":"Initial Condition and Setups","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"general\", \"initial_condition.jl\")]","category":"page"},{"location":"general/initial_condition/#TrixiParticles.InitialCondition","page":"Initial Condition and Setups","title":"TrixiParticles.InitialCondition","text":"InitialCondition(; coordinates, density, velocity=zeros(size(coordinates, 1)),\n mass=nothing, pressure=0.0, particle_spacing=-1.0)\n\nStruct to hold the initial configuration of the particles.\n\nThe following setups return InitialConditions for commonly used setups:\n\nRectangularShape\nSphereShape\nRectangularTank\nComplexShape\nextrude_geometry\n\nInitialConditions support the set operations union, setdiff and intersect in order to build more complex geometries.\n\nArguments\n\ncoordinates: An array where the i-th column holds the coordinates of particle i.\ndensity: Either a vector holding the density of each particle, or a function mapping each particle's coordinates to its density, or a scalar for a constant density over all particles.\n\nKeywords\n\nvelocity: Either an array where the i-th column holds the velocity of particle i, or a function mapping each particle's coordinates to its velocity, or, for a constant fluid velocity, a vector holding this velocity. Velocity is constant zero by default.\nmass: Either nothing (default) to automatically compute particle mass from particle density and spacing, or a vector holding the mass of each particle, or a function mapping each particle's coordinates to its mass, or a scalar for a constant mass over all particles.\npressure: Either a vector holding the pressure of each particle, or a function mapping each particle's coordinates to its pressure, or a scalar for a constant pressure over all particles. This is optional and only needed when using the EntropicallyDampedSPHSystem.\nparticle_spacing: The spacing between the particles. This is a scalar, as the spacing is assumed to be uniform. This is only needed when using set operations on the InitialCondition or for automatic mass calculation.\n\nExamples\n\n# Rectangle filled with particles\ninitial_condition = RectangularShape(0.1, (3, 4), (-1.0, 1.0), density=1.0)\n\n# Two spheres in one initial condition\ninitial_condition = union(SphereShape(0.15, 0.5, (-1.0, 1.0), 1.0),\n SphereShape(0.15, 0.2, (0.0, 1.0), 1.0))\n\n# Rectangle with a spherical hole\nshape1 = RectangularShape(0.1, (16, 13), (-0.8, 0.0), density=1.0)\nshape2 = SphereShape(0.1, 0.35, (0.0, 0.6), 1.0, sphere_type=RoundSphere())\ninitial_condition = setdiff(shape1, shape2)\n\n# Intersect of a rectangle with a sphere. Note that this keeps the particles of the\n# rectangle that are in the intersect, while `intersect(shape2, shape1)` would consist of\n# the particles of the sphere that are in the intersect.\nshape1 = RectangularShape(0.1, (16, 13), (-0.8, 0.0), density=1.0)\nshape2 = SphereShape(0.1, 0.35, (0.0, 0.6), 1.0, sphere_type=RoundSphere())\ninitial_condition = intersect(shape1, shape2)\n\n# Build `InitialCondition` manually\ncoordinates = [0.0 1.0 1.0\n 0.0 0.0 1.0]\nvelocity = zero(coordinates)\nmass = ones(3)\ndensity = 1000 * ones(3)\ninitial_condition = InitialCondition(; coordinates, velocity, mass, density)\n\n# With functions\ninitial_condition = InitialCondition(; coordinates, velocity=x -> 2x, mass=1.0, density=1000.0)\n\n\n\n\n\n","category":"type"},{"location":"general/initial_condition/#Setups","page":"Initial Condition and Setups","title":"Setups","text":"","category":"section"},{"location":"general/initial_condition/","page":"Initial Condition and Setups","title":"Initial Condition and Setups","text":"Modules = [TrixiParticles]\nPages = map(file -> joinpath(\"setups\", file), readdir(joinpath(\"..\", \"src\", \"setups\")))","category":"page"},{"location":"general/initial_condition/#TrixiParticles.ComplexShape-Tuple{Union{TrixiParticles.Polygon, TrixiParticles.TriangleMesh}}","page":"Initial Condition and Setups","title":"TrixiParticles.ComplexShape","text":"ComplexShape(geometry::Union{TriangleMesh, Polygon}; particle_spacing, density,\n pressure=0.0, mass=nothing, velocity=zeros(ndims(geometry)),\n point_in_geometry_algorithm=WindingNumberJacobson(; geometry,\n hierarchical_winding=false,\n winding_number_factor=sqrt(eps())),\n grid_offset::Real=0.0, max_nparticles=10^7,\n pad_initial_particle_grid=2particle_spacing)\n\nSample a complex geometry with particles. Returns an InitialCondition. Note that an initial particle grid is generated inside the bounding box of the geometry. A point_in_geometry_algorithm checks if particles are inside the geometry or not. For more information about the method see WindingNumberJacobson or WindingNumberHormann.\n\nArguments\n\ngeometry: Geometry returned by load_geometry.\n\nKeywords\n\nparticle_spacing: Spacing between the particles.\ndensity: Either a function mapping each particle's coordinates to its density, or a scalar for a constant density over all particles.\nvelocity: Either a function mapping each particle's coordinates to its velocity, or, for a constant fluid velocity, a vector holding this velocity. Velocity is constant zero by default.\nmass: Either nothing (default) to automatically compute particle mass from particle density and spacing, or a function mapping each particle's coordinates to its mass, or a scalar for a constant mass over all particles.\npressure: Scalar to set the pressure of all particles to this value. This is only used by the EntropicallyDampedSPHSystem and will be overwritten when using an initial pressure function in the system.\npoint_in_geometry_algorithm: Algorithm for sampling the complex geometry with particles. It basically checks whether a particle is inside an object or not. For more information see WindingNumberJacobson or WindingNumberHormann\ngrid_offset: Offset of the initial particle grid of the bounding box of the geometry.\nmax_nparticles: Maximum number of particles in the initial particle grid. This is only used to avoid accidentally choosing a particle_spacing that is too small for the scale of the geometry.\npad_initial_particle_grid: Padding of the initial particle grid.\n\nwarning: Experimental Implementation\nThis is an experimental feature and may change in any future releases.\n\n\n\n\n\n","category":"method"},{"location":"general/initial_condition/#TrixiParticles.extrude_geometry-Tuple{Any}","page":"Initial Condition and Setups","title":"TrixiParticles.extrude_geometry","text":"extrude_geometry(geometry; particle_spacing, direction, n_extrude::Integer,\n velocity=zeros(length(direction)),\n mass=nothing, density=nothing, pressure=0.0)\n\nExtrude either a line, a plane or a shape along a specific direction. Returns an InitialCondition.\n\nArguments\n\ngeometry: Either particle coordinates or an InitialCondition defining a 2D shape to extrude to a 3D volume, or two 2D points (A B) defining the interval A B to extrude to a plane in 2D, or three 3D points (A B C) defining the parallelogram spanned by the vectors widehatAB and widehat AC to extrude to a parallelepiped.\n\nKeywords\n\nparticle_spacing: Spacing between the particles. Can be omitted when geometry is an InitialCondition (unless geometry.particle_spacing == -1).\ndirection: A vector that specifies the direction in which to extrude.\nn_extrude: Number of layers of particles created in the direction of extrusion.\nvelocity: Either a function mapping each particle's coordinates to its velocity, or, for a constant fluid velocity, a vector holding this velocity. Velocity is constant zero by default.\nmass: Either nothing (default) to automatically compute particle mass from particle density and spacing, or a function mapping each particle's coordinates to its mass, or a scalar for a constant mass over all particles.\ndensity: Either a function mapping each particle's coordinates to its density, or a scalar for a constant density over all particles.\npressure: Scalar to set the pressure of all particles to this value. This is only used by the EntropicallyDampedSPHSystem and will be overwritten when using an initial pressure function in the system.\ntlsph: With the TotalLagrangianSPHSystem, particles need to be placed on the boundary of the shape and not one particle radius away, as for fluids. When tlsph=true, particles will be placed on the boundary of the shape.\n\nExamples\n\n# Extrude a line in 2D to a plane in 2D\np1 = [0.0, 0.0]\np2 = [1.0, 1.0]\n\ndirection = [-1.0, 1.0]\n\nshape = extrude_geometry((p1, p2); direction, particle_spacing=0.1, n_extrude=4, density=1000.0)\n\n# Extrude a parallelogram in 3D to a parallelepiped in 3D\np1 = [0.0, 0.0, 0.0]\np2 = [0.5, 1.0, 0.0]\np3 = [1.0, 0.2, 0.0]\n\ndirection = [0.0, 0.0, 1.0]\n\nshape = extrude_geometry((p1, p2, p3); direction, particle_spacing=0.1, n_extrude=4, density=1000.0)\n\n# Extrude a 2D shape (here: a disc) to a 3D shape (here: a cylinder)\nshape = SphereShape(0.1, 0.5, (0.2, 0.4), 1000.0, n_layers=3,\n sphere_type=RoundSphere(end_angle=pi))\n\ndirection = [0.0, 0.0, 1.0]\n\nshape = extrude_geometry(shape; direction, particle_spacing=0.1, n_extrude=4, density=1000.0)\n\nwarning: Experimental Implementation\nThis is an experimental feature and may change in any future releases.\n\n\n\n\n\n","category":"method"},{"location":"general/initial_condition/#TrixiParticles.RectangularShape-Tuple{Any, Any, Any}","page":"Initial Condition and Setups","title":"TrixiParticles.RectangularShape","text":"RectangularShape(particle_spacing, n_particles_per_dimension, min_coordinates;\n velocity=zeros(length(n_particles_per_dimension)),\n mass=nothing, density=nothing, pressure=0.0,\n acceleration=nothing, state_equation=nothing,\n tlsph=false, loop_order=nothing)\n\nRectangular shape filled with particles. Returns an InitialCondition.\n\nArguments\n\nparticle_spacing: Spacing between the particles.\nn_particles_per_dimension: Tuple containing the number of particles in x, y and z (only 3D) direction, respectively.\nmin_coordinates: Coordinates of the corner in negative coordinate directions.\n\nKeywords\n\nvelocity: Either a function mapping each particle's coordinates to its velocity, or, for a constant fluid velocity, a vector holding this velocity. Velocity is constant zero by default.\nmass: Either nothing (default) to automatically compute particle mass from particle density and spacing, or a function mapping each particle's coordinates to its mass, or a scalar for a constant mass over all particles.\ndensity: Either a function mapping each particle's coordinates to its density, or a scalar for a constant density over all particles. Obligatory when not using a state equation. Cannot be used together with state_equation.\npressure: Scalar to set the pressure of all particles to this value. This is only used by the EntropicallyDampedSPHSystem and will be overwritten when using an initial pressure function in the system. Cannot be used together with hydrostatic pressure gradient.\nacceleration: In order to initialize particles with a hydrostatic pressure gradient, an acceleration vector can be passed. Note that only accelerations in one coordinate direction and no diagonal accelerations are supported. This will only change the pressure of the particles. When using the WeaklyCompressibleSPHSystem, pass a state_equation as well to initialize the particles with the corresponding density and mass. When using the EntropicallyDampedSPHSystem, the pressure will be overwritten when using an initial pressure function in the system. This cannot be used together with the pressure keyword argument.\nstate_equation: When calculating a hydrostatic pressure gradient by setting acceleration, the state_equation will be used to set the corresponding density. Cannot be used together with density.\ntlsph: With the TotalLagrangianSPHSystem, particles need to be placed on the boundary of the shape and not one particle radius away, as for fluids. When tlsph=true, particles will be placed on the boundary of the shape.\ncoordinates_perturbation: Add a small random displacement to the particle positions, where the amplitude is coordinates_perturbation * particle_spacing.\n\nExamples\n\n# 2D\nrectangular = RectangularShape(particle_spacing, (5, 4), (1.0, 2.0), density=1000.0)\n\n# 2D with hydrostatic pressure gradient.\n# `state_equation` has to be the same as for the WCSPH system.\nstate_equation = StateEquationCole(sound_speed=20.0, exponent=7, reference_density=1000.0)\nrectangular = RectangularShape(particle_spacing, (5, 4), (1.0, 2.0),\n acceleration=(0.0, -9.81), state_equation=state_equation)\n\n# 3D\nrectangular = RectangularShape(particle_spacing, (5, 4, 7), (1.0, 2.0, 3.0), density=1000.0)\n\n\n\n\n\n","category":"method"},{"location":"general/initial_condition/#TrixiParticles.RectangularTank","page":"Initial Condition and Setups","title":"TrixiParticles.RectangularTank","text":"RectangularTank(particle_spacing, fluid_size, tank_size, fluid_density;\n velocity=zeros(length(fluid_size)), fluid_mass=nothing,\n pressure=0.0,\n acceleration=nothing, state_equation=nothing,\n boundary_density=fluid_density,\n n_layers=1, spacing_ratio=1.0,\n min_coordinates=zeros(length(fluid_size)),\n faces=Tuple(trues(2 * length(fluid_size))))\n\nRectangular tank filled with a fluid to set up dam-break-style simulations.\n\nArguments\n\nparticle_spacing: Spacing between the fluid particles.\nfluid_size: The dimensions of the fluid as (x, y) (or (x, y, z) in 3D).\ntank_size: The dimensions of the tank as (x, y) (or (x, y, z) in 3D).\nfluid_density: The rest density of the fluid. Will only be used as default for boundary_density when using a state equation.\n\nKeywords\n\nvelocity: Either a function mapping each particle's coordinates to its velocity, or, for a constant fluid velocity, a vector holding this velocity. Velocity is constant zero by default.\nfluid_mass: Either nothing (default) to automatically compute particle mass from particle density and spacing, or a function mapping each particle's coordinates to its mass, or a scalar for a constant mass over all particles.\npressure: Scalar to set the pressure of all particles to this value. This is only used by the EntropicallyDampedSPHSystem and will be overwritten when using an initial pressure function in the system. Cannot be used together with hydrostatic pressure gradient.\nacceleration: In order to initialize particles with a hydrostatic pressure gradient, an acceleration vector can be passed. Note that only accelerations in one coordinate direction and no diagonal accelerations are supported. This will only change the pressure of the particles. When using the WeaklyCompressibleSPHSystem, pass a state_equation as well to initialize the particles with the corresponding density and mass. When using the EntropicallyDampedSPHSystem, the pressure will be overwritten when using an initial pressure function in the system. This cannot be used together with the pressure keyword argument.\nstate_equation: When calculating a hydrostatic pressure gradient by setting acceleration, the state_equation will be used to set the corresponding density. Cannot be used together with density.\nboundary_density: Density of each boundary particle (by default set to the fluid density)\nn_layers: Number of boundary layers.\nspacing_ratio: Ratio of particle_spacing to boundary particle spacing. A value of 2 means that the boundary particle spacing will be half the fluid particle spacing.\nmin_coordinates: Coordinates of the corner in negative coordinate directions.\nfaces: By default all faces are generated. Set faces by passing a bit-array of length 4 (2D) or 6 (3D) to generate the faces in the normal direction: -x,+x,-y,+y,-z,+z.\n\nFields\n\nfluid::InitialCondition: InitialCondition for the fluid.\nboundary::InitialCondition: InitialCondition for the boundary.\nfluid_size::Tuple: Tuple containing the size of the fluid in each dimension after rounding.\ntank_size::Tuple: Tuple containing the size of the tank in each dimension after rounding.\n\nExamples\n\n# 2D\nsetup = RectangularTank(particle_spacing, (water_width, water_height),\n (container_width, container_height), fluid_density,\n n_layers=2, spacing_ratio=3)\n\n# 2D with hydrostatic pressure gradient.\n# `state_equation` has to be the same as for the WCSPH system.\nstate_equation = StateEquationCole(sound_speed=10.0, exponent=1, reference_density=1000.0)\nsetup = RectangularTank(particle_spacing, (water_width, water_height),\n (container_width, container_height), fluid_density,\n acceleration=(0.0, -9.81), state_equation=state_equation)\n\n# 3D\nsetup = RectangularTank(particle_spacing, (water_width, water_height, water_depth),\n (container_width, container_height, container_depth), fluid_density,\n n_layers=2)\n\nSee also: reset_wall!.\n\n\n\n\n\n","category":"type"},{"location":"general/initial_condition/#TrixiParticles.reset_wall!-Tuple{Any, Any, Any}","page":"Initial Condition and Setups","title":"TrixiParticles.reset_wall!","text":"reset_wall!(rectangular_tank::RectangularTank, reset_faces, positions)\n\nThe selected walls of the tank will be placed at the new positions.\n\nArguments\n\nreset_faces: Boolean tuple of 4 (in 2D) or 6 (in 3D) dimensions, similar to faces in RectangularTank.\npositions: Tuple of new positions\n\nwarning: Warning\nThere are overlapping particles when adjacent walls are moved inwards simultaneously.\n\n\n\n\n\n","category":"method"},{"location":"general/initial_condition/#TrixiParticles.RoundSphere","page":"Initial Condition and Setups","title":"TrixiParticles.RoundSphere","text":"RoundSphere(; start_angle=0.0, end_angle=2π)\n\nConstruct a sphere (or sphere segment) by nesting perfectly round concentric spheres. The resulting ball will be perfectly round, but will not have a regular inner structure.\n\nKeywords\n\nstart_angle: The starting angle of the sphere segment in radians. It determines the beginning point of the segment. The default is set to 0.0 representing the positive x-axis.\nend_angle: The ending angle of the sphere segment in radians. It defines the termination point of the segment. The default is set to 2pi, completing a full sphere.\n\nnote: Usage\nSee SphereShape on how to use this.\n\nwarning: Warning\nThe sphere segment is intended for 2D geometries and hollow spheres. If used for filled spheres or in a 3D context, results may not be accurate.\n\n\n\n\n\n","category":"type"},{"location":"general/initial_condition/#TrixiParticles.VoxelSphere","page":"Initial Condition and Setups","title":"TrixiParticles.VoxelSphere","text":"VoxelSphere()\n\nConstruct a sphere of voxels (where particles are placed in the voxel center) with a regular inner structure but corners on the surface. Essentially, a grid of particles is generated and all particles outside the sphere are removed. The resulting sphere will have a perfect inner structure, but is not perfectly round, as it will have corners (like a sphere in Minecraft).\n\nnote: Usage\nSee SphereShape on how to use this.\n\n\n\n\n\n","category":"type"},{"location":"general/initial_condition/#TrixiParticles.SphereShape-NTuple{4, Any}","page":"Initial Condition and Setups","title":"TrixiParticles.SphereShape","text":"SphereShape(particle_spacing, radius, center_position, density;\n sphere_type=VoxelSphere(), n_layers=-1, layer_outwards=false,\n cutout_min=(0.0, 0.0), cutout_max=(0.0, 0.0), tlsph=false,\n velocity=zeros(length(center_position)), mass=nothing, pressure=0.0)\n\nGenerate a sphere that is either completely filled (by default) or hollow (by passing n_layers).\n\nWith the sphere type VoxelSphere, a sphere of voxels (where particles are placed in the voxel center) with a regular inner structure but corners on the surface is created. Essentially, a grid of particles is generated and all particles outside the sphere are removed. With the sphere type RoundSphere, a perfectly round sphere with an imperfect inner structure is created.\n\nA cuboid can be cut out of the sphere by specifying the two corners in negative and positive coordinate directions as cutout_min and cutout_max.\n\nArguments\n\nparticle_spacing: Spacing between the particles.\nradius: Radius of the sphere.\ncenter_position: The coordinates of the center of the sphere.\ndensity: Either a function mapping each particle's coordinates to its density, or a scalar for a constant density over all particles.\n\nKeywords\n\nsphere_type: Either VoxelSphere or RoundSphere (see explanation above).\nn_layers: Set to an integer greater than zero to generate a hollow sphere, where the shell consists of n_layers layers.\nlayer_outwards: When set to false (by default), radius is the outer radius of the sphere. When set to true, radius is the inner radius of the sphere. This is only used when n_layers > 0.\ncutout_min: Corner in negative coordinate directions of a cuboid that is to be cut out of the sphere.\ncutout_max: Corner in positive coordinate directions of a cuboid that is to be cut out of the sphere.\ntlsph: With the TotalLagrangianSPHSystem, particles need to be placed on the boundary of the shape and not one particle radius away, as for fluids. When tlsph=true, particles will be placed on the boundary of the shape.\nvelocity: Either a function mapping each particle's coordinates to its velocity, or, for a constant fluid velocity, a vector holding this velocity. Velocity is constant zero by default.\nmass: Either nothing (default) to automatically compute particle mass from particle density and spacing, or a function mapping each particle's coordinates to its mass, or a scalar for a constant mass over all particles.\npressure: Either a function mapping each particle's coordinates to its pressure, or a scalar for a constant pressure over all particles. This is optional and only needed when using the EntropicallyDampedSPHSystem.\n\nExamples\n\n# Filled circle with radius 0.5, center in (0.2, 0.4) and a particle spacing of 0.1\nSphereShape(0.1, 0.5, (0.2, 0.4), 1000.0)\n\n# Same as before, but perfectly round\nSphereShape(0.1, 0.5, (0.2, 0.4), 1000.0, sphere_type=RoundSphere())\n\n# Hollow circle with ~3 layers, outer radius 0.5, center in (0.2, 0.4) and a particle\n# spacing of 0.1.\nSphereShape(0.1, 0.5, (0.2, 0.4), 1000.0, n_layers=3)\n\n# Same as before, but perfectly round\nSphereShape(0.1, 0.5, (0.2, 0.4), 1000.0, n_layers=3, sphere_type=RoundSphere())\n\n# Hollow circle with 3 layers, inner radius 0.5, center in (0.2, 0.4) and a particle spacing\n# of 0.1.\nSphereShape(0.1, 0.5, (0.2, 0.4), 1000.0, n_layers=3, layer_outwards=true)\n\n# Filled circle with radius 0.1, center in (0.0, 0.0), particle spacing 0.1, but the\n# rectangle [0, 1] x [-0.2, 0.2] is cut out.\nSphereShape(0.1, 1.0, (0.0, 0.0), 1000.0, cutout_min=(0.0, -0.2), cutout_max=(1.0, 0.2))\n\n# Filled 3D sphere with radius 0.5, center in (0.2, 0.4, 0.3) and a particle spacing of 0.1\nSphereShape(0.1, 0.5, (0.2, 0.4, 0.3), 1000.0)\n\n# Same as before, but perfectly round\nSphereShape(0.1, 0.5, (0.2, 0.4, 0.3), 1000.0, sphere_type=RoundSphere())\n\n\n\n\n\n","category":"method"},{"location":"systems/weakly_compressible_sph/#wcsph","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"Weakly compressible SPH as introduced by Monaghan (1994). This formulation relies on a stiff equation of state that generates large pressure changes for small density variations.","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"fluid\", \"weakly_compressible_sph\", \"system.jl\")]","category":"page"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.WeaklyCompressibleSPHSystem","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.WeaklyCompressibleSPHSystem","text":"WeaklyCompressibleSPHSystem(initial_condition,\n density_calculator, state_equation,\n smoothing_kernel, smoothing_length;\n viscosity=nothing, density_diffusion=nothing,\n acceleration=ntuple(_ -> 0.0, NDIMS),\n buffer_size=nothing,\n correction=nothing, source_terms=nothing)\n\nSystem for particles of a fluid. The weakly compressible SPH (WCSPH) scheme is used, wherein a stiff equation of state generates large pressure changes for small density variations. See Weakly Compressible SPH for more details on the method.\n\nArguments\n\ninitial_condition: InitialCondition representing the system's particles.\ndensity_calculator: Density calculator for the system. See ContinuityDensity and SummationDensity.\nstate_equation: Equation of state for the system. See StateEquationCole.\nsmoothing_kernel: Smoothing kernel to be used for this system. See Smoothing Kernels.\nsmoothing_length: Smoothing length to be used for this system. See Smoothing Kernels.\n\nKeyword Arguments\n\nviscosity: Viscosity model for this system (default: no viscosity). See ArtificialViscosityMonaghan or ViscosityAdami.\ndensity_diffusion: Density diffusion terms for this system. See DensityDiffusion.\nacceleration: Acceleration vector for the system. (default: zero vector)\nbuffer_size: Number of buffer particles. This is needed when simulating with OpenBoundarySPHSystem.\ncorrection: Correction method used for this system. (default: no correction, see Corrections)\nsource_terms: Additional source terms for this system. Has to be either nothing (by default), or a function of (coords, velocity, density, pressure, t) (which are the quantities of a single particle), returning a Tuple or SVector that is to be added to the acceleration of that particle. See, for example, SourceTermDamping. Note that these source terms will not be used in the calculation of the boundary pressure when using a boundary with BoundaryModelDummyParticles and AdamiPressureExtrapolation. The keyword argument acceleration should be used instead for gravity-like source terms.\nsurface_tension: Surface tension model used for this SPH system. (default: no surface tension)\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#equation_of_state","page":"Weakly Compressible SPH (Fluid)","title":"Equation of State","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"The equation of state is used to relate fluid density to pressure and thus allow an explicit simulation of the WCSPH system. The equation in the following formulation was introduced by Cole (1948) (pp. 39 and 43). The pressure p is calculated as","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":" p = B left(left(fracrhorho_0right)^gamma - 1right) + p_textbackground","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"where rho denotes the density, rho_0 the reference density, and p_textbackground the background pressure, which is set to zero when applied to free-surface flows (Adami et al., 2012).","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"The bulk modulus, B = fracrho_0 c^2gamma, is calculated from the artificial speed of sound c and the isentropic exponent gamma.","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"An ideal gas equation of state with a linear relationship between pressure and density can be obtained by choosing exponent=1, i.e.","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":" p = B left( fracrhorho_0 -1 right) = c^2(rho - rho_0)","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"For higher Reynolds numbers, exponent=7 is recommended, whereas at lower Reynolds numbers exponent=1 yields more accurate pressure estimates since pressure and density are proportional (see Morris, 1997).","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"When using SummationDensity (or DensityReinitializationCallback) and free surfaces, initializing particles with equal spacing will cause underestimated density and therefore strong attractive forces between particles at the free surface. Setting clip_negative_pressure=true can avoid this.","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"fluid\", \"weakly_compressible_sph\", \"state_equations.jl\")]","category":"page"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.StateEquationCole","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.StateEquationCole","text":"StateEquationCole(; sound_speed, reference_density, exponent,\n background_pressure=0.0, clip_negative_pressure=false)\n\nEquation of state to describe the relationship between pressure and density of water up to high pressures.\n\nKeywords\n\nsound_speed: Artificial speed of sound.\nreference_density: Reference density of the fluid.\nexponent: A value of 7 is usually used for most simulations.\nbackground_pressure=0.0: Background pressure.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#viscosity_wcsph","page":"Weakly Compressible SPH (Fluid)","title":"Viscosity","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"TODO: Explain viscosity.","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"fluid\", \"viscosity.jl\")]","category":"page"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.ArtificialViscosityMonaghan","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.ArtificialViscosityMonaghan","text":"ArtificialViscosityMonaghan(; alpha, beta=0.0, epsilon=0.01)\n\nArtificial viscosity by Monaghan ([16], [17]), given by\n\nPi_ab =\nbegincases\n -(alpha c mu_ab + beta mu_ab^2) barrho_ab textif v_ab cdot r_ab 0 \n 0 textotherwise\nendcases\n\nwith\n\nmu_ab = frach v_ab cdot r_abVert r_ab Vert^2 + epsilon h^2\n\nwhere alpha beta epsilon are parameters, c is the speed of sound, h is the smoothing length, r_ab = r_a - r_b is the difference of the coordinates of particles a and b, v_ab = v_a - v_b is the difference of their velocities, and barrho_ab is the arithmetic mean of their densities.\n\nNote that alpha needs to adjusted for different resolutions to maintain a specific Reynolds Number. To do so, Monaghan (2005) defined an equivalent effective physical kinematic viscosity nu by\n\n nu = fracalpha h c 2d + 4\n\nwhere d is the dimension.\n\nKeywords\n\nalpha: A value of 0.02 is usually used for most simulations. For a relation with the kinematic viscosity, see description above.\nbeta=0.0: A value of 0.0 works well for most fluid simulations and simulations with shocks of moderate strength. In simulations where the Mach number can be very high, eg. astrophysical calculation, good results can be obtained by choosing a value of beta=2.0 and alpha=1.0.\nepsilon=0.01: Parameter to prevent singularities.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.ViscosityAdami","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.ViscosityAdami","text":"ViscosityAdami(; nu, epsilon=0.01)\n\nViscosity by Adami (2012). The viscous interaction is calculated with the shear force for incompressible flows given by\n\nf_ab = sum_w bareta_ab left( V_a^2 + V_b^2 right) fracv_abr_ab^2+epsilon h_ab^2 nabla W_ab cdot r_ab\n\nwhere r_ab = r_a - r_b is the difference of the coordinates of particles a and b, v_ab = v_a - v_b is the difference of their velocities, h is the smoothing length and V is the particle volume. The parameter epsilon prevents singularities (see Ramachandran (2019)). The inter-particle-averaged shear stress is\n\n bareta_ab =frac2 eta_a eta_beta_a + eta_b\n\nwhere eta_a = rho_a nu_a with nu as the kinematic viscosity.\n\nKeywords\n\nnu: Kinematic viscosity\nepsilon=0.01: Parameter to prevent singularities\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.ViscosityMorris","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.ViscosityMorris","text":"ViscosityMorris(; nu, epsilon=0.01)\n\nViscosity by Morris (1997) also used by Fourtakas (2019).\n\nTo the force f_ab between two particles a and b due to pressure gradients, an additional force term tildef_ab is added with\n\ntildef_ab = m_a m_b frac(mu_a + mu_b) r_ab cdot nabla W_abrho_a rho_b (Vert r_ab Vert^2 + epsilon h^2) v_ab\n\nwhere mu_a = rho_a nu and mu_b = rho_b nu denote the dynamic viscosity of particle a and b respectively, and nu is the kinematic viscosity.\n\nKeywords\n\nnu: Kinematic viscosity\nepsilon=0.01: Parameter to prevent singularities\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#Density-Diffusion","page":"Weakly Compressible SPH (Fluid)","title":"Density Diffusion","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"Density diffusion can be used with ContinuityDensity to remove the noise in the pressure field. It is highly recommended to use density diffusion when using WCSPH.","category":"page"},{"location":"systems/weakly_compressible_sph/#Formulation","page":"Weakly Compressible SPH (Fluid)","title":"Formulation","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"All density diffusion terms extend the continuity equation (see ContinuityDensity) by an additional term","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"fracmathrmdrho_amathrmdt = sum_b m_b v_ab cdot nabla_r_a W(Vert r_ab Vert h)\n + delta h c sum_b V_b psi_ab cdot nabla_r_a W(Vert r_ab Vert h)","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"where V_b = m_b rho_b is the volume of particle b and psi_ab depends on the density diffusion method (see DensityDiffusion for available terms). Also, rho_a denotes the density of particle a and r_ab = r_a - r_b is the difference of the coordinates, v_ab = v_a - v_b of the velocities of particles a and b.","category":"page"},{"location":"systems/weakly_compressible_sph/#Numerical-Results","page":"Weakly Compressible SPH (Fluid)","title":"Numerical Results","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"All density diffusion terms remove numerical noise in the pressure field and produce more accurate results than weakly commpressible SPH without density diffusion. This can be demonstrated with dam break examples in 2D and 3D. Here, δ = 01 has been used for all terms. Note that, due to added stability, the adaptive time integration method that was used here can choose higher time steps in the simulations with density diffusion. For the cheap DensityDiffusionMolteniColagrossi, this results in reduced runtime.","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"
\n \"density_diffusion_2d\"/\n
Dam break in 2D with different density diffusion terms
\n
","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"
\n \"density_diffusion_3d\"/\n
Dam break in 3D with different density diffusion terms
\n
","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"The simpler terms DensityDiffusionMolteniColagrossi and DensityDiffusionFerrari do not solve the hydrostatic problem and lead to incorrect solutions in long-running steady-state hydrostatic simulations with free surfaces (Antuono et al., 2012). This can be seen when running the simple rectangular tank example until t = 40 (again using δ = 01):","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"
\n \"density_diffusion_tank\"/\n
Tank in rest under gravity in 3D with different density diffusion terms
\n
","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"DensityDiffusionAntuono adds a correction term to solve this problem, but this term is very expensive and adds about 40–50% of computational cost.","category":"page"},{"location":"systems/weakly_compressible_sph/#API","page":"Weakly Compressible SPH (Fluid)","title":"API","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"fluid\", \"weakly_compressible_sph\", \"density_diffusion.jl\")]","category":"page"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.DensityDiffusion","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.DensityDiffusion","text":"DensityDiffusion\n\nAn abstract supertype of all density diffusion formulations.\n\nCurrently, the following formulations are available:\n\nFormulation Suitable for Steady-State Simulations Low Computational Cost\nDensityDiffusionMolteniColagrossi ❌ ✅\nDensityDiffusionFerrari ❌ ✅\nDensityDiffusionAntuono ✅ ❌\n\nSee Density Diffusion for a comparison and more details.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.DensityDiffusionAntuono","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.DensityDiffusionAntuono","text":"DensityDiffusionAntuono(initial_condition; delta)\n\nThe commonly used density diffusion terms by Antuono (2010), also referred to as δ-SPH. The density diffusion term by Molteni (2009) is extended by a second term, which is nicely written down by Antuono (2012).\n\nThe term psi_ab in the continuity equation in DensityDiffusion is defined by\n\npsi_ab = 2left(rho_a - rho_b - frac12big(nablarho^L_a + nablarho^L_bbig) cdot r_abright)\n fracr_abVert r_ab Vert^2\n\nwhere rho_a and rho_b denote the densities of particles a and b respectively and r_ab = r_a - r_b is the difference of the coordinates of particles a and b. The symbol nablarho^L_a denotes the renormalized density gradient defined as\n\nnablarho^L_a = -sum_b (rho_a - rho_b) V_b L_a nabla_r_a W(Vert r_ab Vert h)\n\nwith\n\nL_a = left( -sum_b V_b r_ab otimes nabla_r_a W(Vert r_ab Vert h) right)^-1 in R^d times d\n\nwhere d is the number of dimensions.\n\nSee DensityDiffusion for an overview and comparison of implemented density diffusion terms.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.DensityDiffusionFerrari","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.DensityDiffusionFerrari","text":"DensityDiffusionFerrari()\n\nA density diffusion term by Ferrari (2009).\n\nThe term psi_ab in the continuity equation in DensityDiffusion is defined by\n\npsi_ab = fracrho_a - rho_b2h fracr_abVert r_ab Vert\n\nwhere rho_a and rho_b denote the densities of particles a and b respectively, r_ab = r_a - r_b is the difference of the coordinates of particles a and b and h is the smoothing length.\n\nSee DensityDiffusion for an overview and comparison of implemented density diffusion terms.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.DensityDiffusionMolteniColagrossi","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.DensityDiffusionMolteniColagrossi","text":"DensityDiffusionMolteniColagrossi(; delta)\n\nThe commonly used density diffusion term by Molteni (2009).\n\nThe term psi_ab in the continuity equation in DensityDiffusion is defined by\n\npsi_ab = 2(rho_a - rho_b) fracr_abVert r_ab Vert^2\n\nwhere rho_a and rho_b denote the densities of particles a and b respectively and r_ab = r_a - r_b is the difference of the coordinates of particles a and b.\n\nSee DensityDiffusion for an overview and comparison of implemented density diffusion terms.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#corrections","page":"Weakly Compressible SPH (Fluid)","title":"Corrections","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"general\", \"corrections.jl\")]","category":"page"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.AkinciFreeSurfaceCorrection","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.AkinciFreeSurfaceCorrection","text":"AkinciFreeSurfaceCorrection(rho0)\n\nFree surface correction according to Akinci et al. (2013). At a free surface, the mean density is typically lower than the reference density, resulting in reduced surface tension and viscosity forces. The free surface correction adjusts the viscosity, pressure, and surface tension forces near free surfaces to counter this effect. It's important to note that this correlation is unphysical and serves as an approximation. The computation time added by this method is about 2–3%.\n\nMathematically the idea is quite simple. If we have an SPH particle in the middle of a volume at rest, its density will be identical to the rest density rho_0. If we now consider an SPH particle at a free surface at rest, it will have neighbors missing in the direction normal to the surface, which will result in a lower density. If we calculate the correction factor\n\nk = rho_0rho_textmean\n\nthis value will be about ~1.5 for particles at the free surface and can then be used to increase the pressure and viscosity accordingly.\n\nArguments\n\nrho0: Rest density.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.BlendedGradientCorrection","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.BlendedGradientCorrection","text":"BlendedGradientCorrection()\n\nCalculate a blended gradient to reduce the stability issues of the GradientCorrection as explained by Bonet (1999).\n\nThis calculates the following,\n\ntildenabla A_i = (1-lambda) nabla A_i + lambda L_i nabla A_i\n\nwith 0 leq lambda leq 1 being the blending factor.\n\nArguments\n\nblending_factor: Blending factor between corrected and regular SPH gradient.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.GradientCorrection","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.GradientCorrection","text":"GradientCorrection()\n\nCompute the corrected gradient of particle interactions based on their relative positions (see Bonet, 1999).\n\nMathematical Details\n\nGiven the standard SPH representation, the gradient of a field A at particle a is given by\n\nnabla A_a = sum_b m_b fracA_b - A_arho_b nabla_r_a W(Vert r_a - r_b Vert h)\n\nwhere m_b is the mass of particle b and rho_b is the density of particle b.\n\nThe gradient correction, as commonly proposed, involves multiplying this gradient with a correction matrix L:\n\ntildenabla A_a = bmL_a nabla A_a\n\nThe correction matrix bmL_a is computed based on the provided particle configuration, aiming to make the corrected gradient more accurate, especially near domain boundaries.\n\nTo satisfy\n\nsum_b V_b r_ba otimes tildenablaW_b(r_a) = left( sum_b V_b r_ba otimes nabla W_b(r_a) right) bmL_a^T = bmI\n\nthe correction matrix bmL_a is evaluated explicitly as\n\nbmL_a = left( sum_b V_b nabla W_b(r_a) otimes r_ba right)^-1\n\nnote: Note\nStability issues arise, especially when particles separate into small clusters.\nDoubles the computational effort.\nBetter stability with smoother smoothing Kernels with larger support, e.g. SchoenbergQuinticSplineKernel or WendlandC6Kernel.\nSet dt_max =< 1e-3 for stability.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.KernelCorrection","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.KernelCorrection","text":"KernelCorrection()\n\nKernel correction, as explained by Bonet (1999), uses Shepard interpolation to obtain a 0-th order accurate result, which was first proposed by Li et al. This can be further extended to obtain a kernel corrected gradient as shown by Basa et al. (2008).\n\nThe kernel correction coefficient is determined by\n\nc(x) = sum_b=1 V_b W_b(x)\n\nThe gradient of corrected kernel is determined by\n\nnabla tildeW_b(r) =fracnabla W_b(r) - W_b(r) gamma(r)sum_b=1 V_b W_b(r) quad textwhere quad\ngamma(r) = fracsum_b=1 V_b nabla W_b(r)sum_b=1 V_b W_b(r)\n\nThis correction can be applied with SummationDensity and ContinuityDensity, which leads to an improvement, especially at free surfaces.\n\nnote: Note\nThis only works when the boundary model uses SummationDensity (yet).\nIt is also referred to as \"0th order correction\".\nIn 2D, we can expect an increase of about 10–15% in computation time.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.MixedKernelGradientCorrection","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.MixedKernelGradientCorrection","text":"MixedKernelGradientCorrection()\n\nCombines GradientCorrection and KernelCorrection, which results in a 1st-order-accurate SPH method (see Bonet, 1999).\n\nNotes:\n\nStability issues, especially when particles separate into small clusters.\nDoubles the computational effort.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.ShepardKernelCorrection","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.ShepardKernelCorrection","text":"ShepardKernelCorrection()\n\nKernel correction, as explained by Bonet (1999), uses Shepard interpolation to obtain a 0-th order accurate result, which was first proposed by Li et al. (1996).\n\nThe kernel correction coefficient is determined by\n\nc(x) = sum_b=1 V_b W_b(x)\n\nwhere V_b = m_b rho_b is the volume of particle b.\n\nThis correction is applied with SummationDensity to correct the density and leads to an improvement, especially at free surfaces.\n\nnote: Note\nIt is also referred to as \"0th order correction\".\nIn 2D, we can expect an increase of about 5–6% in computation time.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#surface_tension","page":"Weakly Compressible SPH (Fluid)","title":"Surface Tension","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/#Akinci-based-intra-particle-force-surface-tension-and-wall-adhesion-model","page":"Weakly Compressible SPH (Fluid)","title":"Akinci-based intra-particle force surface tension and wall adhesion model","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"The work by Akinci proposes three forces:","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"a cohesion force\na surface area minimization force\na wall adhesion force","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"The classical model is composed of the curvature minimization and cohesion force.","category":"page"},{"location":"systems/weakly_compressible_sph/#Cohesion-force","page":"Weakly Compressible SPH (Fluid)","title":"Cohesion force","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"The model calculates the cohesion force based on the distance between particles and the support radius h_c. This force is determined using two distinct regimes within the support radius:","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"For particles closer than half the support radius, a repulsive force is calculated to prevent particle clustering too tightly, enhancing the simulation's stability and realism.\nBeyond half the support radius and within the full support radius, an attractive force is computed, simulating the effects of surface tension that draw particles together.","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"The cohesion force, F_textcohesion, for a pair of particles is given by:","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"F_textcohesion = -sigma m_b C(r) fracrVert r Vert","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"where:","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"sigma represents the surface tension coefficient, adjusting the overall strength of the cohesion effect.\nC is a scalar function of the distance between particles.","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"The cohesion kernel C is defined as","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"C(r)=frac32pi h_c^9\nbegincases\n(h_c-r)^3 r^3 textif 2r h_c \n2(h_c-r)^3 r^3 - frach^664 textif r 0 text and 2r leq h_c \n0 textotherwise\nendcases","category":"page"},{"location":"systems/weakly_compressible_sph/#Surface-area-minimization-force","page":"Weakly Compressible SPH (Fluid)","title":"Surface area minimization force","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"To model the minimization of the surface area and curvature of the fluid, a curvature force is used, which is calculated as","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"F_textcurvature = -sigma (n_a - n_b)","category":"page"},{"location":"systems/weakly_compressible_sph/#Wall-adhesion-force","page":"Weakly Compressible SPH (Fluid)","title":"Wall adhesion force","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"The wall adhesion model proposed by Akinci et al. is based on a kernel function which is 0 from 0.0 to 0.5 support radiia with a maximum at 0.75. With the force calculated with an adhesion coefficient beta as","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"F_textadhesion = -beta m_b A(r) fracrVert r Vert","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"with A being the adhesion kernel defined as","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"A(r)= frac0007h_c^325\nbegincases\nsqrt4- frac4r^2h_c + 6r - 2h_c textif 2r h_c text and r leq h_c \n0 textotherwise\nendcases","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"fluid\", \"surface_tension.jl\")]","category":"page"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.CohesionForceAkinci","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.CohesionForceAkinci","text":"CohesionForceAkinci(surface_tension_coefficient=1.0)\n\nThis model only implements the cohesion force of the [25] surface tension model.\n\nKeywords\n\nsurface_tension_coefficient=1.0: Modifies the intensity of the surface tension-induced force, enabling the tuning of the fluid's surface tension properties within the simulation.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.SurfaceTensionAkinci","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.SurfaceTensionAkinci","text":"SurfaceTensionAkinci(surface_tension_coefficient=1.0)\n\nImplements a model for surface tension and adhesion effects drawing upon the principles outlined by [25]. This model is instrumental in capturing the nuanced behaviors of fluid surfaces, such as droplet formation and the dynamics of merging or separation, by utilizing intra-particle forces.\n\nKeywords\n\nsurface_tension_coefficient=1.0: A parameter to adjust the magnitude of surface tension forces, facilitating the fine-tuning of how surface tension phenomena are represented in the simulation.\n\n\n\n\n\n","category":"type"},{"location":"tutorials/tut_dam_break/","page":"Example file","title":"Example file","text":"EditURL = \"https://github.com/trixi-framework/TrixiParticles.jl/blob/main/docs/src/tutorials_template/tut_dam_break.md\"","category":"page"},{"location":"tutorials/tut_dam_break/#Example-file","page":"Example file","title":"Example file","text":"","category":"section"},{"location":"tutorials/tut_dam_break/","page":"Example file","title":"Example file","text":"# 2D dam break simulation based on\n#\n# S. Marrone, M. Antuono, A. Colagrossi, G. Colicchio, D. le Touzé, G. Graziani.\n# \"δ-SPH model for simulating violent impact flows\".\n# In: Computer Methods in Applied Mechanics and Engineering, Volume 200, Issues 13–16 (2011), pages 1526–1542.\n# https://doi.org/10.1016/J.CMA.2010.12.016\n\nusing TrixiParticles\nusing OrdinaryDiffEq\n\n# Size parameters\nH = 0.6\nW = 2 * H\n\n# ==========================================================================================\n# ==== Resolution\nfluid_particle_spacing = H / 40\n\n# Change spacing ratio to 3 and boundary layers to 1 when using Monaghan-Kajtar boundary model\nboundary_layers = 4\nspacing_ratio = 1\n\nboundary_particle_spacing = fluid_particle_spacing / spacing_ratio\n\n# ==========================================================================================\n# ==== Experiment Setup\ngravity = 9.81\n\ntspan = (0.0, 5.7 / sqrt(gravity))\n\n# Boundary geometry and initial fluid particle positions\ninitial_fluid_size = (W, H)\ntank_size = (floor(5.366 * H / boundary_particle_spacing) * boundary_particle_spacing, 4.0)\n\nfluid_density = 1000.0\nsound_speed = 20 * sqrt(gravity * H)\nstate_equation = StateEquationCole(; sound_speed, reference_density=fluid_density,\n exponent=1, clip_negative_pressure=false)\n\ntank = RectangularTank(fluid_particle_spacing, initial_fluid_size, tank_size, fluid_density,\n n_layers=boundary_layers, spacing_ratio=spacing_ratio,\n acceleration=(0.0, -gravity), state_equation=state_equation)\n\n# ==========================================================================================\n# ==== Fluid\nsmoothing_length = 3.5 * fluid_particle_spacing\nsmoothing_kernel = WendlandC2Kernel{2}()\n\nfluid_density_calculator = ContinuityDensity()\nviscosity = ArtificialViscosityMonaghan(alpha=0.02, beta=0.0)\n# nu = 0.02 * smoothing_length * sound_speed/8\n# viscosity = ViscosityMorris(nu=nu)\n# viscosity = ViscosityAdami(nu=nu)\n# Alternatively the density diffusion model by Molteni & Colagrossi can be used,\n# which will run faster.\n# density_diffusion = DensityDiffusionMolteniColagrossi(delta=0.1)\ndensity_diffusion = DensityDiffusionAntuono(tank.fluid, delta=0.1)\n\nfluid_system = WeaklyCompressibleSPHSystem(tank.fluid, fluid_density_calculator,\n state_equation, smoothing_kernel,\n smoothing_length, viscosity=viscosity,\n density_diffusion=density_diffusion,\n acceleration=(0.0, -gravity), correction=nothing,\n surface_tension=nothing)\n\n# ==========================================================================================\n# ==== Boundary\nboundary_density_calculator = AdamiPressureExtrapolation()\nboundary_model = BoundaryModelDummyParticles(tank.boundary.density, tank.boundary.mass,\n state_equation=state_equation,\n boundary_density_calculator,\n smoothing_kernel, smoothing_length,\n correction=nothing)\n\nboundary_system = BoundarySPHSystem(tank.boundary, boundary_model, adhesion_coefficient=0.0)\n\n# ==========================================================================================\n# ==== Simulation\n# `nothing` will automatically choose the best update strategy. This is only to be able\n# to change this with `trixi_include`.\nsemi = Semidiscretization(fluid_system, boundary_system,\n neighborhood_search=GridNeighborhoodSearch{2}(update_strategy=nothing))\node = semidiscretize(semi, tspan, data_type=nothing)\n\ninfo_callback = InfoCallback(interval=100)\n\nsolution_prefix = \"\"\nsaving_callback = SolutionSavingCallback(dt=0.02, prefix=solution_prefix)\n\n# Save at certain timepoints which allows comparison to the results of Marrone et al.,\n# i.e. (1.5, 2.36, 3.0, 5.7, 6.45).\n# Please note that the images in Marrone et al. are obtained at a particle_spacing = H/320,\n# which takes between 2 and 4 hours.\nsaving_paper = SolutionSavingCallback(save_times=[0.0, 0.371, 0.584, 0.743, 1.411, 1.597],\n prefix=\"marrone_times\")\n\n# This can be overwritten with `trixi_include`\nextra_callback = nothing\n\nuse_reinit = false\ndensity_reinit_cb = use_reinit ?\n DensityReinitializationCallback(semi.systems[1], interval=10) :\n nothing\nstepsize_callback = StepsizeCallback(cfl=0.9)\n\ncallbacks = CallbackSet(info_callback, saving_callback, stepsize_callback, extra_callback,\n density_reinit_cb, saving_paper)\n\nsol = solve(ode, CarpenterKennedy2N54(williamson_condition=false),\n dt=1.0, # This is overwritten by the stepsize callback\n save_everystep=false, callback=callbacks);\n\n","category":"page"},{"location":"general/neighborhood_search/#Neighborhood-Search","page":"Neighborhood Search","title":"Neighborhood Search","text":"","category":"section"},{"location":"general/neighborhood_search/","page":"Neighborhood Search","title":"Neighborhood Search","text":"The neighborhood search is the most essential component for performance. We provide several implementations in the package PointNeighbors.jl. See the docs of this package for an overview and a comparison of different implementations.","category":"page"},{"location":"general/neighborhood_search/","page":"Neighborhood Search","title":"Neighborhood Search","text":"note: Usage\nTo run a simulation with a neighborhood search implementation, pass a template of the neighborhood search to the constructor of the Semidiscretization. A template is just an empty neighborhood search with search radius 0.0. See copy_neighborhood_search and the examples below for more details.semi = Semidiscretization(system1, system2,\n neighborhood_search=PrecomputedNeighborhoodSearch{2}())The keyword argument periodic_box in the neighborhood search constructors can be used to define a periodic domain. See the PointNeighbors.jl docs for more details.periodic_box = PeriodicBox(min_corner=[0.0, -0.25], max_corner=[1.0, 0.75])\nsemi = Semidiscretization(system1, system2,\n neighborhood_search=GridNeighborhoodSearch{2}(; periodic_box))","category":"page"},{"location":"tutorials/tut_falling/","page":"Example file","title":"Example file","text":"EditURL = \"https://github.com/trixi-framework/TrixiParticles.jl/blob/main/docs/src/tutorials_template/tut_falling.md\"","category":"page"},{"location":"tutorials/tut_falling/#Example-file","page":"Example file","title":"Example file","text":"","category":"section"},{"location":"tutorials/tut_falling/","page":"Example file","title":"Example file","text":"using TrixiParticles\nusing OrdinaryDiffEq\n\n# ==========================================================================================\n# ==== Resolution\nfluid_particle_spacing = 0.02\nsolid_particle_spacing = fluid_particle_spacing\n\n# Change spacing ratio to 3 and boundary layers to 1 when using Monaghan-Kajtar boundary model\nboundary_layers = 3\nspacing_ratio = 1\n\n# ==========================================================================================\n# ==== Experiment Setup\ngravity = 9.81\ntspan = (0.0, 1.0)\n\n# Boundary geometry and initial fluid particle positions\ninitial_fluid_size = (2.0, 0.9)\ntank_size = (2.0, 1.0)\n\nfluid_density = 1000.0\nsound_speed = 10 * sqrt(gravity * initial_fluid_size[2])\nstate_equation = StateEquationCole(; sound_speed, reference_density=fluid_density,\n exponent=1)\n\ntank = RectangularTank(fluid_particle_spacing, initial_fluid_size, tank_size, fluid_density,\n n_layers=boundary_layers, spacing_ratio=spacing_ratio,\n faces=(true, true, true, false),\n acceleration=(0.0, -gravity), state_equation=state_equation)\n\nsphere1_radius = 0.3\nsphere2_radius = 0.2\nsphere1_density = 500.0\nsphere2_density = 1100.0\n\n# Young's modulus and Poisson ratio\nsphere1_E = 7e4\nsphere2_E = 1e5\nnu = 0.0\n\nsphere1_center = (0.5, 1.6)\nsphere2_center = (1.5, 1.6)\nsphere1 = SphereShape(solid_particle_spacing, sphere1_radius, sphere1_center,\n sphere1_density, sphere_type=VoxelSphere())\nsphere2 = SphereShape(solid_particle_spacing, sphere2_radius, sphere2_center,\n sphere2_density, sphere_type=VoxelSphere())\n\n# ==========================================================================================\n# ==== Fluid\nfluid_smoothing_length = 3.0 * fluid_particle_spacing\nfluid_smoothing_kernel = WendlandC2Kernel{2}()\n\nfluid_density_calculator = ContinuityDensity()\nviscosity = ArtificialViscosityMonaghan(alpha=0.02, beta=0.0)\ndensity_diffusion = DensityDiffusionMolteniColagrossi(delta=0.1)\n\nfluid_system = WeaklyCompressibleSPHSystem(tank.fluid, fluid_density_calculator,\n state_equation, fluid_smoothing_kernel,\n fluid_smoothing_length, viscosity=viscosity,\n density_diffusion=density_diffusion,\n acceleration=(0.0, -gravity))\n\n# ==========================================================================================\n# ==== Boundary\nboundary_density_calculator = BernoulliPressureExtrapolation()\nboundary_model = BoundaryModelDummyParticles(tank.boundary.density, tank.boundary.mass,\n state_equation=state_equation,\n boundary_density_calculator,\n fluid_smoothing_kernel, fluid_smoothing_length)\n\nboundary_system = BoundarySPHSystem(tank.boundary, boundary_model)\n\n# ==========================================================================================\n# ==== Solid\nsolid_smoothing_length = 2 * sqrt(2) * solid_particle_spacing\nsolid_smoothing_kernel = WendlandC2Kernel{2}()\n\n# For the FSI we need the hydrodynamic masses and densities in the solid boundary model\nhydrodynamic_densites_1 = fluid_density * ones(size(sphere1.density))\nhydrodynamic_masses_1 = hydrodynamic_densites_1 * solid_particle_spacing^ndims(fluid_system)\n\nsolid_boundary_model_1 = BoundaryModelDummyParticles(hydrodynamic_densites_1,\n hydrodynamic_masses_1,\n state_equation=state_equation,\n boundary_density_calculator,\n fluid_smoothing_kernel,\n fluid_smoothing_length)\n\nhydrodynamic_densites_2 = fluid_density * ones(size(sphere2.density))\nhydrodynamic_masses_2 = hydrodynamic_densites_2 * solid_particle_spacing^ndims(fluid_system)\n\nsolid_boundary_model_2 = BoundaryModelDummyParticles(hydrodynamic_densites_2,\n hydrodynamic_masses_2,\n state_equation=state_equation,\n boundary_density_calculator,\n fluid_smoothing_kernel,\n fluid_smoothing_length)\n\nsolid_system_1 = TotalLagrangianSPHSystem(sphere1,\n solid_smoothing_kernel, solid_smoothing_length,\n sphere1_E, nu,\n acceleration=(0.0, -gravity),\n boundary_model=solid_boundary_model_1,\n penalty_force=PenaltyForceGanzenmueller(alpha=0.3))\n\nsolid_system_2 = TotalLagrangianSPHSystem(sphere2,\n solid_smoothing_kernel, solid_smoothing_length,\n sphere2_E, nu,\n acceleration=(0.0, -gravity),\n boundary_model=solid_boundary_model_2,\n penalty_force=PenaltyForceGanzenmueller(alpha=0.3))\n\n# ==========================================================================================\n# ==== Simulation\nsemi = Semidiscretization(fluid_system, boundary_system, solid_system_1, solid_system_2)\node = semidiscretize(semi, tspan)\n\ninfo_callback = InfoCallback(interval=10)\nsaving_callback = SolutionSavingCallback(dt=0.02, output_directory=\"out\", prefix=\"\",\n write_meta_data=true)\n\ncallbacks = CallbackSet(info_callback, saving_callback)\n\n# Use a Runge-Kutta method with automatic (error based) time step size control.\nsol = solve(ode, RDPK3SpFSAL49(),\n abstol=1e-6, # Default abstol is 1e-6\n reltol=1e-3, # Default reltol is 1e-3\n save_everystep=false, callback=callbacks);\n\n","category":"page"},{"location":"tutorials/tut_beam/","page":"Example file","title":"Example file","text":"EditURL = \"https://github.com/trixi-framework/TrixiParticles.jl/blob/main/docs/src/tutorials_template/tut_beam.md\"","category":"page"},{"location":"tutorials/tut_beam/#Example-file","page":"Example file","title":"Example file","text":"","category":"section"},{"location":"tutorials/tut_beam/","page":"Example file","title":"Example file","text":"using TrixiParticles\nusing OrdinaryDiffEq\n\n# ==========================================================================================\n# ==== Resolution\nn_particles_y = 5\n\n# ==========================================================================================\n# ==== Experiment Setup\ngravity = 2.0\ntspan = (0.0, 5.0)\n\nelastic_beam = (length=0.35, thickness=0.02)\nmaterial = (density=1000.0, E=1.4e6, nu=0.4)\nclamp_radius = 0.05\n\n# The structure starts at the position of the first particle and ends\n# at the position of the last particle.\nparticle_spacing = elastic_beam.thickness / (n_particles_y - 1)\n\n# Add particle_spacing/2 to the clamp_radius to ensure that particles are also placed on the radius\nfixed_particles = SphereShape(particle_spacing, clamp_radius + particle_spacing / 2,\n (0.0, elastic_beam.thickness / 2), material.density,\n cutout_min=(0.0, 0.0),\n cutout_max=(clamp_radius, elastic_beam.thickness),\n tlsph=true)\n\nn_particles_clamp_x = round(Int, clamp_radius / particle_spacing)\n\n# Beam and clamped particles\nn_particles_per_dimension = (round(Int, elastic_beam.length / particle_spacing) +\n n_particles_clamp_x + 1, n_particles_y)\n\n# Note that the `RectangularShape` puts the first particle half a particle spacing away\n# from the boundary, which is correct for fluids, but not for solids.\n# We therefore need to pass `tlsph=true`.\nbeam = RectangularShape(particle_spacing, n_particles_per_dimension,\n (0.0, 0.0), density=material.density, tlsph=true)\n\nsolid = union(beam, fixed_particles)\n\n# ==========================================================================================\n# ==== Solid\n# The kernel in the reference uses a differently scaled smoothing length,\n# so this is equivalent to the smoothing length of `sqrt(2) * particle_spacing` used in the paper.\nsmoothing_length = 2 * sqrt(2) * particle_spacing\nsmoothing_kernel = WendlandC2Kernel{2}()\n\nsolid_system = TotalLagrangianSPHSystem(solid, smoothing_kernel, smoothing_length,\n material.E, material.nu,\n n_fixed_particles=nparticles(fixed_particles),\n acceleration=(0.0, -gravity),\n penalty_force=nothing)\n\n# ==========================================================================================\n# ==== Simulation\nsemi = Semidiscretization(solid_system,\n neighborhood_search=PrecomputedNeighborhoodSearch{2}())\node = semidiscretize(semi, tspan)\n\ninfo_callback = InfoCallback(interval=100)\n\n# Track the position of the particle in the middle of the tip of the beam.\nmiddle_particle_id = Int(n_particles_per_dimension[1] * (n_particles_per_dimension[2] + 1) /\n 2)\nstartposition_x = beam.coordinates[1, middle_particle_id]\nstartposition_y = beam.coordinates[2, middle_particle_id]\n\nfunction deflection_x(v, u, t, system)\n return system.current_coordinates[1, middle_particle_id] - startposition_x\nend\n\nfunction deflection_y(v, u, t, system)\n return system.current_coordinates[2, middle_particle_id] - startposition_y\nend\n\nsaving_callback = SolutionSavingCallback(dt=0.02, prefix=\"\",\n deflection_x=deflection_x,\n deflection_y=deflection_y)\n\ncallbacks = CallbackSet(info_callback, saving_callback)\n\n# Use a Runge-Kutta method with automatic (error based) time step size control\nsol = solve(ode, RDPK3SpFSAL49(), save_everystep=false, callback=callbacks);\n\n","category":"page"},{"location":"general/density_calculators/#density_calculator","page":"Density Calculators","title":"Density Calculators","text":"","category":"section"},{"location":"general/density_calculators/","page":"Density Calculators","title":"Density Calculators","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"general\", \"density_calculators.jl\")]","category":"page"},{"location":"general/density_calculators/#TrixiParticles.ContinuityDensity","page":"Density Calculators","title":"TrixiParticles.ContinuityDensity","text":"ContinuityDensity()\n\nDensity calculator to integrate the density from the continuity equation\n\nfracmathrmdrho_amathrmdt = sum_b m_b v_ab cdot nabla_r_a W(Vert r_a - r_b Vert h)\n\nwhere rho_a denotes the density of particle a and r_ab = r_a - r_b is the difference of the coordinates, v_ab = v_a - v_b of the velocities of particles a and b.\n\n\n\n\n\n","category":"type"},{"location":"general/density_calculators/#TrixiParticles.SummationDensity","page":"Density Calculators","title":"TrixiParticles.SummationDensity","text":"SummationDensity()\n\nDensity calculator to use the summation formula\n\nrho(r) = sum_b m_b W(Vert r - r_b Vert h)\n\nfor the density estimation, where r_b denotes the coordinates and m_b the mass of particle b.\n\n\n\n\n\n","category":"type"},{"location":"general/semidiscretization/#Semidiscretization","page":"Semidiscretization","title":"Semidiscretization","text":"","category":"section"},{"location":"general/semidiscretization/","page":"Semidiscretization","title":"Semidiscretization","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"general\", \"semidiscretization.jl\")]","category":"page"},{"location":"general/semidiscretization/#TrixiParticles.Semidiscretization","page":"Semidiscretization","title":"TrixiParticles.Semidiscretization","text":"Semidiscretization(systems...; neighborhood_search=GridNeighborhoodSearch{NDIMS}())\n\nThe semidiscretization couples the passed systems to one simulation.\n\nArguments\n\nsystems: Systems to be coupled in this semidiscretization\n\nKeywords\n\nneighborhood_search: The neighborhood search to be used in the simulation. By default, the GridNeighborhoodSearch is used. Use nothing to loop over all particles (no neighborhood search). To use other neighborhood search implementations, pass a template of a neighborhood search. See copy_neighborhood_search and the examples below for more details. To use a periodic domain, pass a PeriodicBox to the neighborhood search.\nthreaded_nhs_update=true: Can be used to deactivate thread parallelization in the neighborhood search update. This can be one of the largest sources of variations between simulations with different thread numbers due to particle ordering changes.\n\nExamples\n\nsemi = Semidiscretization(fluid_system, boundary_system)\n\nsemi = Semidiscretization(fluid_system, boundary_system,\n neighborhood_search=GridNeighborhoodSearch{2}(update_strategy=SerialUpdate()))\n\nperiodic_box = PeriodicBox(min_corner = [0.0, 0.0], max_corner = [1.0, 1.0])\nsemi = Semidiscretization(fluid_system, boundary_system,\n neighborhood_search=GridNeighborhoodSearch{2}(; periodic_box))\n\nsemi = Semidiscretization(fluid_system, boundary_system,\n neighborhood_search=PrecomputedNeighborhoodSearch{2}())\n\nsemi = Semidiscretization(fluid_system, boundary_system,\n neighborhood_search=nothing)\n\n\n\n\n\n","category":"type"},{"location":"general/semidiscretization/#TrixiParticles.SourceTermDamping","page":"Semidiscretization","title":"TrixiParticles.SourceTermDamping","text":"SourceTermDamping(; damping_coefficient)\n\nA source term to be used when a damping step is required before running a full simulation. The term -c cdot v_a is added to the acceleration fracmathrmdv_amathrmdt of particle a, where c is the damping coefficient and v_a is the velocity of particle a.\n\nKeywords\n\ndamping_coefficient: The coefficient d above. A higher coefficient means more damping. A coefficient of 1e-4 is a good starting point for damping a fluid at rest.\n\nExamples\n\nsource_terms = SourceTermDamping(; damping_coefficient=1e-4)\n\n\n\n\n\n","category":"type"},{"location":"general/semidiscretization/#TrixiParticles.restart_with!-Tuple{Any, Any}","page":"Semidiscretization","title":"TrixiParticles.restart_with!","text":"restart_with!(semi, sol)\n\nSet the initial coordinates and velocities of all systems in semi to the final values in the solution sol. semidiscretize has to be called again afterwards, or another Semidiscretization can be created with the updated systems.\n\nArguments\n\nsemi: The semidiscretization\nsol: The ODESolution returned by solve of OrdinaryDiffEq\n\n\n\n\n\n","category":"method"},{"location":"general/semidiscretization/#TrixiParticles.semidiscretize-Tuple{Any, Any}","page":"Semidiscretization","title":"TrixiParticles.semidiscretize","text":"semidiscretize(semi, tspan; reset_threads=true)\n\nCreate an ODEProblem from the semidiscretization with the specified tspan.\n\nArguments\n\nsemi: A Semidiscretization holding the systems involved in the simulation.\ntspan: The time span over which the simulation will be run.\n\nKeywords\n\nreset_threads: A boolean flag to reset Polyester.jl threads before the simulation (default: true). After an error within a threaded loop, threading might be disabled. Resetting the threads before the simulation ensures that threading is enabled again for the simulation. See also trixi-framework/Trixi.jl#1583.\n\nReturns\n\nA DynamicalODEProblem (see the OrdinaryDiffEq.jl docs) to be integrated with OrdinaryDiffEq.jl. Note that this is not a true DynamicalODEProblem where the acceleration does not depend on the velocity. Therefore, not all integrators designed for DynamicalODEProblems will work properly. However, all integrators designed for ODEProblems can be used.\n\nExamples\n\nsemi = Semidiscretization(fluid_system, boundary_system)\ntspan = (0.0, 1.0)\node_problem = semidiscretize(semi, tspan)\n\n\n\n\n\n","category":"method"},{"location":"systems/boundary/#Boundary-System","page":"Boundary","title":"Boundary System","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":" BoundarySPHSystem","category":"page"},{"location":"systems/boundary/#TrixiParticles.BoundarySPHSystem","page":"Boundary","title":"TrixiParticles.BoundarySPHSystem","text":"BoundarySPHSystem(initial_condition, boundary_model; movement=nothing, adhesion_coefficient=0.0)\n\nSystem for boundaries modeled by boundary particles. The interaction between fluid and boundary particles is specified by the boundary model.\n\nArguments\n\ninitial_condition: Initial condition (see InitialCondition)\nboundary_model: Boundary model (see Boundary Models)\n\nKeyword Arguments\n\nmovement: For moving boundaries, a BoundaryMovement can be passed.\nadhesion_coefficient: Coefficient specifying the adhesion of a fluid to the surface. Note: currently it is assumed that all fluids have the same adhesion coefficient.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":" BoundaryDEMSystem","category":"page"},{"location":"systems/boundary/#TrixiParticles.BoundaryDEMSystem","page":"Boundary","title":"TrixiParticles.BoundaryDEMSystem","text":"BoundaryDEMSystem(initial_condition, normal_stiffness)\n\nSystem for boundaries modeled by boundary particles. The interaction between fluid and boundary particles is specified by the boundary model.\n\nwarning: Experimental Implementation\nThis is an experimental feature and may change in a future releases.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":" BoundaryMovement","category":"page"},{"location":"systems/boundary/#TrixiParticles.BoundaryMovement","page":"Boundary","title":"TrixiParticles.BoundaryMovement","text":"BoundaryMovement(movement_function, is_moving; moving_particles=nothing)\n\nArguments\n\nmovement_function: Time-dependent function returning an SVector of d dimensions for a d-dimensional problem.\nis_moving: Function to determine in each timestep if the particles are moving or not. Its boolean return value is mandatory to determine if the neighborhood search will be updated.\n\nKeyword Arguments\n\nmoving_particles: Indices of moving particles. Default is each particle in BoundarySPHSystem.\n\nIn the example below, movement describes particles moving in a circle as long as the time is lower than 1.5.\n\nExamples\n\nmovement_function(t) = SVector(cos(2pi*t), sin(2pi*t))\nis_moving(t) = t < 1.5\n\nmovement = BoundaryMovement(movement_function, is_moving)\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/#boundary_models","page":"Boundary","title":"Boundary Models","text":"","category":"section"},{"location":"systems/boundary/#Dummy-Particles","page":"Boundary","title":"Dummy Particles","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Boundaries modeled as dummy particles, which are treated like fluid particles, but their positions and velocities are not evolved in time. Since the force towards the fluid should not change with the material density when used with a TotalLagrangianSPHSystem, the dummy particles need to have a mass corresponding to the fluid's rest density, which we call \"hydrodynamic mass\", as opposed to mass corresponding to the material density of a TotalLagrangianSPHSystem.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Here, initial_density and hydrodynamic_mass are vectors that contains the initial density and the hydrodynamic mass respectively for each boundary particle. Note that when used with SummationDensity (see below), this is only used to determine the element type and the number of boundary particles.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"To establish a relationship between density and pressure, a state_equation has to be passed, which should be the same as for the adjacent fluid systems. To sum over neighboring particles, a smoothing_kernel and smoothing_length needs to be passed. This should be the same as for the adjacent fluid system with the largest smoothing length.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"In the literature, this kind of boundary particles is referred to as \"dummy particles\" (Adami et al., 2012 and Valizadeh & Monaghan, 2015), \"frozen fluid particles\" (Akinci et al., 2012) or \"dynamic boundaries Crespo et al., 2007. The key detail of this boundary condition and the only difference between the boundary models in these references is the way the density and pressure of boundary particles is computed.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Since boundary particles are treated like fluid particles, the force on fluid particle a due to boundary particle b is given by","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"f_ab = m_a m_b left( fracp_arho_a^2 + fracp_brho_b^2 right) nabla_r_a W(Vert r_a - r_b Vert h)","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"The quantities to be defined here are the density rho_b and pressure p_b of the boundary particle b.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":" BoundaryModelDummyParticles","category":"page"},{"location":"systems/boundary/#TrixiParticles.BoundaryModelDummyParticles","page":"Boundary","title":"TrixiParticles.BoundaryModelDummyParticles","text":"BoundaryModelDummyParticles(initial_density, hydrodynamic_mass,\n density_calculator, smoothing_kernel,\n smoothing_length; viscosity=nothing,\n state_equation=nothing, correction=nothing)\n\nBoundary model for BoundarySPHSystem.\n\nArguments\n\ninitial_density: Vector holding the initial density of each boundary particle.\nhydrodynamic_mass: Vector holding the \"hydrodynamic mass\" of each boundary particle. See description above for more information.\ndensity_calculator: Strategy to compute the hydrodynamic density of the boundary particles. See description below for more information.\nsmoothing_kernel: Smoothing kernel should be the same as for the adjacent fluid system.\nsmoothing_length: Smoothing length should be the same as for the adjacent fluid system.\n\nKeywords\n\nstate_equation: This should be the same as for the adjacent fluid system (see e.g. StateEquationCole).\ncorrection: Correction method of the adjacent fluid system (see Corrections).\nviscosity: Slip (default) or no-slip condition. See description below for further information.\n\nExamples\n\n# Free-slip condition\nboundary_model = BoundaryModelDummyParticles(densities, masses, AdamiPressureExtrapolation(),\n smoothing_kernel, smoothing_length)\n\n# No-slip condition\nboundary_model = BoundaryModelDummyParticles(densities, masses, AdamiPressureExtrapolation(),\n smoothing_kernel, smoothing_length,\n viscosity=ViscosityAdami(nu=1e-6))\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/#Hydrodynamic-density-of-dummy-particles","page":"Boundary","title":"Hydrodynamic density of dummy particles","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"We provide six options to compute the boundary density and pressure, determined by the density_calculator:","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"(Recommended) With AdamiPressureExtrapolation, the pressure is extrapolated from the pressure of the fluid according to Adami et al., 2012, and the density is obtained by applying the inverse of the state equation. This option usually yields the best results of the options listed here.\n(Only relevant for FSI) With BernoulliPressureExtrapolation, the pressure is extrapolated from the pressure similar to the AdamiPressureExtrapolation, but a relative velocity-dependent pressure part is calculated between moving solids and fluids, which increases the boundary pressure in areas prone to penetrations.\nWith SummationDensity, the density is calculated by summation over the neighboring particles, and the pressure is computed from the density with the state equation.\nWith ContinuityDensity, the density is integrated from the continuity equation, and the pressure is computed from the density with the state equation. Note that this causes a gap between fluid and boundary where the boundary is initialized without any contact to the fluid. This is due to overestimation of the boundary density as soon as the fluid comes in contact with boundary particles that initially did not have contact to the fluid. Therefore, in dam break simulations, there is a visible \"step\", even though the boundary is supposed to be flat. See also dual.sphysics.org/faq/#Q_13.\nWith PressureZeroing, the density is set to the reference density and the pressure is computed from the density with the state equation. This option is not recommended. The other options yield significantly better results.\nWith PressureMirroring, the density is set to the reference density. The pressure is not used. Instead, the fluid pressure is mirrored as boundary pressure in the momentum equation. This option is not recommended due to stability issues. See PressureMirroring for more details.","category":"page"},{"location":"systems/boundary/#1.-[AdamiPressureExtrapolation](@ref)","page":"Boundary","title":"1. AdamiPressureExtrapolation","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"The pressure of the boundary particles is obtained by extrapolating the pressure of the fluid according to Adami et al., 2012. The pressure of a boundary particle b is given by","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"p_b = fracsum_f (p_f + rho_f (bmg - bma_b) cdot bmr_bf) W(Vert r_bf Vert h)sum_f W(Vert r_bf Vert h)","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"where the sum is over all fluid particles, rho_f and p_f denote the density and pressure of fluid particle f, respectively, r_bf = r_b - r_f denotes the difference of the coordinates of particles b and f, bmg denotes the gravitational acceleration acting on the fluid, and bma_b denotes the acceleration of the boundary particle b.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":" AdamiPressureExtrapolation","category":"page"},{"location":"systems/boundary/#TrixiParticles.AdamiPressureExtrapolation","page":"Boundary","title":"TrixiParticles.AdamiPressureExtrapolation","text":"AdamiPressureExtrapolation(; pressure_offset=0.0)\n\ndensity_calculator for BoundaryModelDummyParticles.\n\nKeywords\n\npressure_offset=0.0: Sometimes it is necessary to artificially increase the boundary pressure to prevent penetration, which is possible by increasing this value.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/#2.-[BernoulliPressureExtrapolation](@ref)","page":"Boundary","title":"2. BernoulliPressureExtrapolation","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Identical to the pressure p_b calculated via AdamiPressureExtrapolation, but it adds the dynamic pressure component of the Bernoulli equation:","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"p_b = fracsum_f (p_f + frac12 rho_textneighbor left( frac (mathbfv_f - mathbfv_textbody) cdot (mathbfx_f - mathbfx_textneighbor) left mathbfx_f - mathbfx_textneighbor right right)^2 times textfactor +rho_f (bmg - bma_b) cdot bmr_bf) W(Vert r_bf Vert h)sum_f W(Vert r_bf Vert h) ","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"where mathbfv_f is the velocity of the fluid and mathbfv_textbody is the velocity of the body. This adjustment provides a higher boundary pressure for solid bodies moving with a relative velocity to the fluid to prevent penetration. This modification is original and not derived from any literature source.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":" BernoulliPressureExtrapolation","category":"page"},{"location":"systems/boundary/#TrixiParticles.BernoulliPressureExtrapolation","page":"Boundary","title":"TrixiParticles.BernoulliPressureExtrapolation","text":"BernoulliPressureExtrapolation(; pressure_offset=0.0, factor=1.0)\n\ndensity_calculator for BoundaryModelDummyParticles.\n\nKeywords\n\npressure_offset=0.0: Sometimes it is necessary to artificially increase the boundary pressure to prevent penetration, which is possible by increasing this value.\nfactor=1.0 : Setting factor allows to just increase the strength of the dynamic pressure part.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/#5.-[PressureZeroing](@ref)","page":"Boundary","title":"5. PressureZeroing","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"This is the simplest way to implement dummy boundary particles. The density of each particle is set to the reference density and the pressure to the reference pressure (the corresponding pressure to the reference density by the state equation).","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":" PressureZeroing","category":"page"},{"location":"systems/boundary/#TrixiParticles.PressureZeroing","page":"Boundary","title":"TrixiParticles.PressureZeroing","text":"PressureZeroing()\n\ndensity_calculator for BoundaryModelDummyParticles.\n\nnote: Note\nThis boundary model produces significantly worse results than all other models and is only included for research purposes.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/#6.-[PressureMirroring](@ref)","page":"Boundary","title":"6. PressureMirroring","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Instead of calculating density and pressure for each boundary particle, we modify the momentum equation,","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"fracmathrmdv_amathrmdt = -sum_b m_b left( fracp_arho_a^2 + fracp_brho_b^2 right) nabla_a W_ab","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"to replace the unknown density rho_b if b is a boundary particle by the reference density and the unknown pressure p_b if b is a boundary particle by the pressure p_a of the interacting fluid particle. The momentum equation therefore becomes","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"fracmathrmdv_amathrmdt = -sum_f m_f left( fracp_arho_a^2 + fracp_frho_f^2 right) nabla_a W_af\n-sum_b m_b left( fracp_arho_a^2 + fracp_arho_0^2 right) nabla_a W_ab","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"where the first sum is over all fluid particles and the second over all boundary particles.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"This approach was first mentioned by Akinci et al. (2012) and written down in this form by Band et al. (2018).","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":" PressureMirroring","category":"page"},{"location":"systems/boundary/#TrixiParticles.PressureMirroring","page":"Boundary","title":"TrixiParticles.PressureMirroring","text":"PressureMirroring()\n\ndensity_calculator for BoundaryModelDummyParticles.\n\nnote: Note\nThis boundary model requires high viscosity for stability with WCSPH. It also produces significantly worse results than AdamiPressureExtrapolation and is not more efficient because smaller time steps are required due to more noise in the pressure. We added this model only for research purposes and for comparison with SPlisHSPlasH.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/#No-slip-conditions","page":"Boundary","title":"No-slip conditions","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"For the interaction of dummy particles and fluid particles, Adami et al. (2012) impose a no-slip boundary condition by assigning a wall velocity v_w to the dummy particle.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"The wall velocity of particle a is calculated from the prescribed boundary particle velocity v_a and the smoothed velocity field","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"v_w = 2 v_a - fracsum_b v_b W_absum_b W_ab","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"where the sum is over all fluid particles.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"By choosing the viscosity model ViscosityAdami for viscosity, a no-slip condition is imposed. It is recommended to choose nu in the order of either the kinematic viscosity parameter of the adjacent fluid or the equivalent from the artificial parameter alpha of the adjacent fluid (nu = fracalpha h c 2d + 4). When omitting the viscous interaction (default viscosity=nothing), a free-slip wall boundary condition is applied.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"warning: Warning\nThe viscosity model ArtificialViscosityMonaghan for BoundaryModelDummyParticles has not been verified yet.","category":"page"},{"location":"systems/boundary/#Repulsive-Particles","page":"Boundary","title":"Repulsive Particles","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Boundaries modeled as boundary particles which exert forces on the fluid particles (Monaghan, Kajtar, 2009). The force on fluid particle a due to boundary particle b is given by","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"f_ab = m_a left(tildef_ab - m_b Pi_ab nabla_r_a W(Vert r_a - r_b Vert h)right)","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"with","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"tildef_ab = fracKbeta^n-1 fracr_abVert r_ab Vert (Vert r_ab Vert - d) Phi(Vert r_ab Vert h)\nfrac2 m_bm_a + m_b","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"where m_a and m_b are the masses of fluid particle a and boundary particle b respectively, r_ab = r_a - r_b is the difference of the coordinates of particles a and b, d denotes the boundary particle spacing and n denotes the number of dimensions (see Monaghan & Kajtar, 2009, Equation (3.1) and Valizadeh & Monaghan, 2015). Note that the repulsive acceleration tildef_ab does not depend on the masses of the boundary particles. Here, Phi denotes the 1D Wendland C4 kernel, normalized to 177 for q=0 (Monaghan & Kajtar, 2009, Section 4), with Phi(r h) = w(rh) and","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"w(q) =\nbegincases\n (17732) (1 + (52)q + 2q^2)(2 - q)^5 textif 0 leq q 2 \n 0 textif q geq 2\nendcases","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"The boundary particles are assumed to have uniform spacing by the factor beta smaller than the expected fluid particle spacing. For example, if the fluid particles have an expected spacing of 03 and the boundary particles have a uniform spacing of 01, then this parameter should be set to beta = 3. According to Monaghan & Kajtar (2009), a value of beta = 3 for the Wendland C4 that we use here is reasonable for most computing purposes.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"The parameter K is used to scale the force exerted by the boundary particles. In Monaghan & Kajtar (2009), a value of gD is used for static tank simulations, where g is the gravitational acceleration and D is the depth of the fluid.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"The viscosity Pi_ab is calculated according to the viscosity used in the simulation, where the density of the boundary particle if needed is assumed to be identical to the density of the fluid particle.","category":"page"},{"location":"systems/boundary/#No-slip-condition","page":"Boundary","title":"No-slip condition","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"By choosing the viscosity model ArtificialViscosityMonaghan for viscosity, a no-slip condition is imposed. When omitting the viscous interaction (default viscosity=nothing), a free-slip wall boundary condition is applied.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"warning: Warning\nThe no-slip conditions for BoundaryModelMonaghanKajtar have not been verified yet.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"boundary\", \"monaghan_kajtar\", \"monaghan_kajtar.jl\")]","category":"page"},{"location":"systems/boundary/#TrixiParticles.BoundaryModelMonaghanKajtar","page":"Boundary","title":"TrixiParticles.BoundaryModelMonaghanKajtar","text":"BoundaryModelMonaghanKajtar(K, beta, boundary_particle_spacing, mass;\n viscosity=nothing)\n\nBoundary model for BoundarySPHSystem.\n\nArguments\n\nK: Scaling factor for repulsive force.\nbeta: Ratio of fluid particle spacing to boundary particle spacing.\nboundary_particle_spacing: Boundary particle spacing.\nmass: Vector holding the mass of each boundary particle.\n\nKeywords\n\nviscosity: Free-slip (default) or no-slip condition. See description above for further information.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/#open_boundary","page":"Boundary","title":"Open Boundaries","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"boundary\", \"open_boundary\", \"system.jl\")]","category":"page"},{"location":"systems/boundary/#TrixiParticles.OpenBoundarySPHSystem","page":"Boundary","title":"TrixiParticles.OpenBoundarySPHSystem","text":"OpenBoundarySPHSystem(boundary_zone::Union{InFlow, OutFlow};\n fluid_system::FluidSystem, buffer_size::Integer,\n boundary_model,\n reference_velocity=nothing,\n reference_pressure=nothing,\n reference_density=nothing)\n\nOpen boundary system for in- and outflow particles.\n\nArguments\n\nboundary_zone: Use InFlow for an inflow and OutFlow for an outflow boundary.\n\nKeywords\n\nfluid_system: The corresponding fluid system\nboundary_model: Boundary model (see Open Boundary Models)\nbuffer_size: Number of buffer particles.\nreference_velocity: Reference velocity is either a function mapping each particle's coordinates and time to its velocity, an array where the i-th column holds the velocity of particle i or, for a constant fluid velocity, a vector holding this velocity.\nreference_pressure: Reference pressure is either a function mapping each particle's coordinates and time to its pressure, a vector holding the pressure of each particle, or a scalar for a constant pressure over all particles.\nreference_density: Reference density is either a function mapping each particle's coordinates and time to its density, a vector holding the density of each particle, or a scalar for a constant density over all particles.\n\nwarning: Experimental Implementation\nThis is an experimental feature and may change in any future releases.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"boundary\", \"open_boundary\", \"boundary_zones.jl\")]","category":"page"},{"location":"systems/boundary/#TrixiParticles.InFlow","page":"Boundary","title":"TrixiParticles.InFlow","text":"InFlow(; plane, flow_direction, density, particle_spacing,\n initial_condition=nothing, extrude_geometry=nothing,\n open_boundary_layers::Integer)\n\nInflow boundary zone for OpenBoundarySPHSystem.\n\nThe specified plane (line in 2D or rectangle in 3D) will be extruded in upstream direction (the direction opposite to flow_direction) to create a box for the boundary zone. There are three ways to specify the actual shape of the inflow:\n\nDon't pass initial_condition or extrude_geometry. The boundary zone box will then be filled with inflow particles (default).\nSpecify extrude_geometry by passing a 1D shape in 2D or a 2D shape in 3D, which is then extruded in upstream direction to create the inflow particles.\nIn 2D, the shape must be either an initial condition with 2D coordinates, which lies on the line specified by plane, or an initial condition with 1D coordinates, which lies on the line specified by plane when a y-coordinate of 0 is added.\nIn 3D, the shape must be either an initial condition with 3D coordinates, which lies in the rectangle specified by plane, or an initial condition with 2D coordinates, which lies in the rectangle specified by plane when a z-coordinate of 0 is added.\nSpecify initial_condition by passing a 2D initial condition in 2D or a 3D initial condition in 3D, which will be used for the inflow particles.\n\nnote: Note\nParticles outside the boundary zone box will be removed.\n\nKeywords\n\nplane: Tuple of points defining a part of the surface of the domain. The points must either span a line in 2D or a rectangle in 3D. This line or rectangle is then extruded in upstream direction to obtain the boundary zone. In 2D, pass two points (A B), so that the interval A B is the inflow surface. In 3D, pass three points (A B C), so that the rectangular inflow surface is spanned by the vectors widehatAB and widehatAC. These two vectors must be orthogonal.\nflow_direction: Vector defining the flow direction.\nopen_boundary_layers: Number of particle layers in upstream direction.\nparticle_spacing: The spacing between the particles (see InitialCondition).\ndensity: Particle density (see InitialCondition).\ninitial_condition=nothing: InitialCondition for the inflow particles. Particles outside the boundary zone will be removed. Do not use together with extrude_geometry.\nextrude_geometry=nothing: 1D shape in 2D or 2D shape in 3D, which lies on the plane and is extruded upstream to obtain the inflow particles. See point 2 above for more details.\n\nExamples\n\n# 2D\nplane_points = ([0.0, 0.0], [0.0, 1.0])\nflow_direction=[1.0, 0.0]\n\ninflow = InFlow(; plane=plane_points, particle_spacing=0.1, flow_direction, density=1.0,\n open_boundary_layers=4)\n\n# 3D\nplane_points = ([0.0, 0.0, 0.0], [1.0, 0.0, 0.0], [0.0, 1.0, 0.0])\nflow_direction=[0.0, 0.0, 1.0]\n\ninflow = InFlow(; plane=plane_points, particle_spacing=0.1, flow_direction, density=1.0,\n open_boundary_layers=4)\n\n# 3D particles sampled as cylinder\ncircle = SphereShape(0.1, 0.5, (0.5, 0.5), 1.0, sphere_type=RoundSphere())\n\ninflow = InFlow(; plane=plane_points, particle_spacing=0.1, flow_direction, density=1.0,\n extrude_geometry=circle, open_boundary_layers=4)\n\nwarning: Experimental Implementation\nThis is an experimental feature and may change in any future releases.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/#TrixiParticles.OutFlow","page":"Boundary","title":"TrixiParticles.OutFlow","text":"OutFlow(; plane, flow_direction, density, particle_spacing,\n initial_condition=nothing, extrude_geometry=nothing,\n open_boundary_layers::Integer)\n\nOutflow boundary zone for OpenBoundarySPHSystem.\n\nThe specified plane (line in 2D or rectangle in 3D) will be extruded in downstream direction (the direction in flow_direction) to create a box for the boundary zone. There are three ways to specify the actual shape of the outflow:\n\nDon't pass initial_condition or extrude_geometry. The boundary zone box will then be filled with outflow particles (default).\nSpecify extrude_geometry by passing a 1D shape in 2D or a 2D shape in 3D, which is then extruded in downstream direction to create the outflow particles.\nIn 2D, the shape must be either an initial condition with 2D coordinates, which lies on the line specified by plane, or an initial condition with 1D coordinates, which lies on the line specified by plane when a y-coordinate of 0 is added.\nIn 3D, the shape must be either an initial condition with 3D coordinates, which lies in the rectangle specified by plane, or an initial condition with 2D coordinates, which lies in the rectangle specified by plane when a z-coordinate of 0 is added.\nSpecify initial_condition by passing a 2D initial condition in 2D or a 3D initial condition in 3D, which will be used for the outflow particles.\n\nnote: Note\nParticles outside the boundary zone box will be removed.\n\nKeywords\n\nplane: Tuple of points defining a part of the surface of the domain. The points must either span a line in 2D or a rectangle in 3D. This line or rectangle is then extruded in downstream direction to obtain the boundary zone. In 2D, pass two points (A B), so that the interval A B is the outflow surface. In 3D, pass three points (A B C), so that the rectangular outflow surface is spanned by the vectors widehatAB and widehatAC. These two vectors must be orthogonal.\nflow_direction: Vector defining the flow direction.\nopen_boundary_layers: Number of particle layers in downstream direction.\nparticle_spacing: The spacing between the particles (see InitialCondition).\ndensity: Particle density (see InitialCondition).\ninitial_condition=nothing: InitialCondition for the outflow particles. Particles outside the boundary zone will be removed. Do not use together with extrude_geometry.\nextrude_geometry=nothing: 1D shape in 2D or 2D shape in 3D, which lies on the plane and is extruded downstream to obtain the outflow particles. See point 2 above for more details.\n\nExamples\n\n# 2D\nplane_points = ([0.0, 0.0], [0.0, 1.0])\nflow_direction = [1.0, 0.0]\n\noutflow = OutFlow(; plane=plane_points, particle_spacing=0.1, flow_direction, density=1.0,\n open_boundary_layers=4)\n\n# 3D\nplane_points = ([0.0, 0.0, 0.0], [1.0, 0.0, 0.0], [0.0, 1.0, 0.0])\nflow_direction = [0.0, 0.0, 1.0]\n\noutflow = OutFlow(; plane=plane_points, particle_spacing=0.1, flow_direction, density=1.0,\n open_boundary_layers=4)\n\n# 3D particles sampled as cylinder\ncircle = SphereShape(0.1, 0.5, (0.5, 0.5), 1.0, sphere_type=RoundSphere())\n\noutflow = OutFlow(; plane=plane_points, particle_spacing=0.1, flow_direction, density=1.0,\n extrude_geometry=circle, open_boundary_layers=4)\n\nwarning: Experimental Implementation\nThis is an experimental feature and may change in any future releases.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/#open_boundary_models","page":"Boundary","title":"Open Boundary Models","text":"","category":"section"},{"location":"systems/boundary/#method_of_characteristics","page":"Boundary","title":"Method of characteristics","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"boundary\", \"open_boundary\", \"method_of_characteristics.jl\")]","category":"page"},{"location":"systems/boundary/#TrixiParticles.BoundaryModelLastiwka","page":"Boundary","title":"TrixiParticles.BoundaryModelLastiwka","text":"BoundaryModelLastiwka()\n\nBoundary model for OpenBoundarySPHSystem. This model uses the characteristic variables to propagate the appropriate values to the outlet or inlet and have been proposed by Lastiwka et al. (2009). For more information about the method see description below.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"The difficulty in non-reflecting boundary conditions, also called open boundaries, is to determine the appropriate boundary values of the exact characteristics of the Euler equations. Assuming the flow near the boundaries is normal to the boundary and free of shock waves and significant viscous effects, it can be shown that three characteristic variables exist:","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"J_1, associated with convection of entropy and propagates at flow velocity,\nJ_2, downstream-running characteristics,\nJ_3, upstream-running characteristics.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Giles (1990) derived those variables based on a linearized set of governing equations:","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"J_1 = -c_s^2 (rho - rho_textref) + (p - p_textref)","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"J_2 = rho c_s (v - v_textref) + (p - p_textref)","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"J_3 = - rho c_s (v - v_textref) + (p - p_textref)","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"where the subscript \"ref\" denotes the reference flow near the boundaries, which can be prescribed.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Specifying the reference variables is not equivalent to prescription of rho, v and p directly, since the perturbation from the reference flow is allowed.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Lastiwka et al. (2009) applied the method of characteristic to SPH and determined the number of variables that should be prescribed at the boundary and the number which should be propagated from the fluid domain to the boundary:","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"For an inflow boundary:\nPrescribe downstream-running characteristics J_1 and J_2\nTransmit J_3 from the fluid domain (allow J_3 to propagate upstream to the boundary).\nFor an outflow boundary:\nPrescribe upstream-running characteristic J_3\nTransmit J_1 and J_2 from the fluid domain.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Prescribing is done by simply setting the characteristics to zero. To transmit the characteristics from the fluid domain, or in other words, to carry the information of the fluid to the boundaries, Negi et al. (2020) use a Shepard Interpolation","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"f_i = fracsum_j^N f_j W_ijsum_j^N W_ij","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"where the i-th particle is a boundary particle, f is either J_1, J_2 or J_3 and N is the set of neighboring fluid particles.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"To express pressure p, density rho and velocity v as functions of the characteristic variables, the system of equations from the characteristic variables is inverted and gives","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":" rho - rho_textref = frac1c_s^2 left( -J_1 + frac12 J_2 + frac12 J_3 right)","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"u - u_textref= frac12rho c_s left( J_2 - J_3 right)","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"p - p_textref = frac12 left( J_2 + J_3 right)","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"With J_1, J_2 and J_3 determined, we can easily solve for the actual variables for each particle.","category":"page"},{"location":"tutorials_template/tut_setup/#Setting-up-your-simulation-from-scratch","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"","category":"section"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"In this tutorial, we will guide you through the general structure of simulation files. We will set up a simulation similar to the example simulation examples/fluid/hydrostatic_water_column_2d.jl, which is one of our simplest example simulations. In the second part of this tutorial, we will show how to replace components of TrixiParticles.jl by custom implementations from within a simulation file, without ever cloning the repository.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"For different setups and physics, have a look at our other example files.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"First, we import TrixiParticles.jl and OrdinaryDiffEq.jl, which we will use at the very end for the time integration.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"using TrixiParticles\nusing OrdinaryDiffEq","category":"page"},{"location":"tutorials_template/tut_setup/#Resolution","page":"Setting up your simulation from scratch","title":"Resolution","text":"","category":"section"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"Now, we define the particle spacing, which is our numerical resolution. We usually call the variable fluid_particle_spacing, so that we can easily change the resolution of an example file by overwriting this variable with trixi_include. In 2D, the number of particles will grow quadratically, in 3D cubically with the spacing.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"We also set the number of boundary layers, which need to be sufficiently large, depending on the smoothing kernel and smoothing length, so that the compact support of the smoothing kernel is fully sampled with particles for a fluid particle close to a boundary. In particular, we require boundary_layers >= compact_support. The value for the compact support for each kernel can be found in the smoothing kernel overview.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"fluid_particle_spacing = 0.05\nboundary_layers = 3\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/#Experiment-setup","page":"Setting up your simulation from scratch","title":"Experiment setup","text":"","category":"section"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"We want to simulate a water column resting under hydrostatic pressure inside a rectangular tank. (Image: Experiment Setup) First, we define the physical parameters gravitational acceleration, simulation time, initial fluid size, tank size and fluid density.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"gravity = 9.81\ntspan = (0.0, 1.0)\ninitial_fluid_size = (1.0, 0.9)\ntank_size = (1.0, 1.0)\nfluid_density = 1000.0\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"In order to have the initial particle mass and density correspond to the hydrostatic pressure gradient, we need to define a state equation, which relates the fluid density to pressure. Note that we could also skip this part here and define the state equation later when we define the fluid system, but then the fluid would be initialized with constant density, which would cause it to oscillate under gravity.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"sound_speed = 10.0\nstate_equation = StateEquationCole(; sound_speed, reference_density=fluid_density,\n exponent=7)\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"The speed of sound here is numerical and not physical. We artificially lower the speed of sound, since the physical speed of sound in water would lead to prohibitively small time steps. The speed of sound in Weakly Compressible SPH should be chosen as small as possible for numerical efficiency, but large enough to limit density fluctuations to about 1%.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"TrixiParticles.jl requires the initial particle positions and quantities in form of an InitialCondition. Instead of manually defining particle positions, you can work with our pre-defined setups. Among others, we provide setups for rectangular shapes, circles, and spheres. Initial conditions can also be combined with common set operations. See this page for a list of pre-defined setups and details on set operations on initial conditions.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"Here, we use the RectangularTank setup, which generates a rectangular fluid inside a rectangular tank, and supports a hydrostatic pressure gradient by passing a gravitational acceleration and a state equation (see above).","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"tank = RectangularTank(fluid_particle_spacing, initial_fluid_size, tank_size,\n fluid_density, n_layers=boundary_layers,\n acceleration=(0.0, -gravity), state_equation=state_equation)\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/#Fluid-system","page":"Setting up your simulation from scratch","title":"Fluid system","text":"","category":"section"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"To model the water column, we use the Weakly Compressible Smoothed Particle Hydrodynamics (WCSPH) method. This method requires a smoothing kernel and a corresponding smoothing length, which should be chosen in relation to the particle spacing.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"smoothing_length = 1.2 * fluid_particle_spacing\nsmoothing_kernel = SchoenbergCubicSplineKernel{2}()\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"You can find an overview over smoothing kernels and corresponding smoothing lengths here.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"For stability, we need numerical dissipation in form of an artificial viscosity term. Other viscosity models offer a physical approach based on the kinematic viscosity of the fluid.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"viscosity = ArtificialViscosityMonaghan(alpha=0.02, beta=0.0)\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"We choose the parameters as small as possible to avoid non-physical behavior, but as large as possible to stabilize the simulation.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"The WCSPH method can either compute the particle density directly with a kernel summation over all neighboring particles (see SummationDensity) or by making the particle density a variable in the ODE system and integrating its change over time. We choose the latter approach here by using the density calculator ContinuityDensity, which is more efficient and handles free surfaces without the need for additional correction terms.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"fluid_density_calculator = ContinuityDensity()\nfluid_system = WeaklyCompressibleSPHSystem(tank.fluid, fluid_density_calculator,\n state_equation, smoothing_kernel,\n smoothing_length, viscosity=viscosity,\n acceleration=(0.0, -gravity))\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/#Boundary-system","page":"Setting up your simulation from scratch","title":"Boundary system","text":"","category":"section"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"To model the boundary, we use particle-based boundary conditions, in which particles are sampled in the boundary that interact with the fluid particles to avoid penetration. In order to define a boundary system, we first have to choose a boundary model, which defines how the fluid interacts with boundary particles. We will use the BoundaryModelDummyParticles with AdamiPressureExtrapolation. See here for a comprehensive overview over boundary models.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"boundary_model = BoundaryModelDummyParticles(tank.boundary.density, tank.boundary.mass,\n state_equation=state_equation,\n AdamiPressureExtrapolation(),\n smoothing_kernel, smoothing_length)\nboundary_system = BoundarySPHSystem(tank.boundary, boundary_model)\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/#Semidiscretization","page":"Setting up your simulation from scratch","title":"Semidiscretization","text":"","category":"section"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"The key component of every simulation is the Semidiscretization, which couples all systems of the simulation. All methods in TrixiParticles.jl are semidiscretizations, which discretize the equations in time to provide an ordinary differential equation that still has to be solved in time. By providing a simulation time span, we can call semidiscretize, which returns an ODEProblem that can be solved with a time integration method.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"semi = Semidiscretization(fluid_system, boundary_system)\node = semidiscretize(semi, tspan)\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/#Time-integration","page":"Setting up your simulation from scratch","title":"Time integration","text":"","category":"section"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"We use the methods provided by OrdinaryDiffEq.jl, but note that other packages or custom implementations can also be used.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"OrdinaryDiffEq.jl supports callbacks, which are executed during the simulation. For this simulation, we use the InfoCallback, which prints information about the simulation setup at the beginning of the simulation, information about the current simulation time and runtime during the simulation, and a performance summary at the end of the simulation. We also want to save the current solution in regular intervals in terms of simulation time as VTK, so that we can look at the solution in ParaView. The SolutionSavingCallback provides this functionality. To pass the callbacks to OrdinaryDiffEq.jl, we have to bundle them into a CallbackSet.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"info_callback = InfoCallback(interval=50)\nsaving_callback = SolutionSavingCallback(dt=0.02)\n\ncallbacks = CallbackSet(info_callback, saving_callback)\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"Finally, we can start the simulation by solving the ODEProblem. We use the method RDPK3SpFSAL35 of OrdinaryDiffEq.jl, which is a Runge-Kutta method with automatic (error based) time step size control. This method is usually a good choice for prototyping, since we do not have to worry about choosing a stable step size and can just run the simulation. For better performance, it might be beneficial to tweak the tolerances of this method or choose a different method that is more efficient for the respective simulation. You can find both approaches in our example files. Here, we just use the method with the default parameters, and only disable save_everystep to avoid expensive saving of the solution in every time step.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"sol = solve(ode, RDPK3SpFSAL35(), save_everystep=false, callback=callbacks);\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"See Visualization for how to visualize the solution. For the simplest visualization, we can use Plots.jl:","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"using Plots\nplot(sol)\nplot!(dpi=200); savefig(\"tut_setup_plot.png\"); nothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"(Image: plot)","category":"page"},{"location":"tutorials_template/tut_setup/#Replacing-components-with-custom-implementations","page":"Setting up your simulation from scratch","title":"Replacing components with custom implementations","text":"","category":"section"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"If we would like to use an implementation of a component that is not available in TrixiParticles.jl, we can implement it ourselves within the simulation file, without ever cloning the TrixiParticles.jl repository. A good starting point is to check out the available implementations in TrixiParticles.jl, then copy the relevant functions to the simulation file and modify them as needed.","category":"page"},{"location":"tutorials_template/tut_setup/#Custom-smoothing-kernel","page":"Setting up your simulation from scratch","title":"Custom smoothing kernel","text":"","category":"section"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"To implement a custom smoothing kernel, we define a struct extending TrixiParticles.SmoothingKernel. This abstract struct has a type parameter for the number of dimensions, which we set to 2 in this case.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"struct MyGaussianKernel <: TrixiParticles.SmoothingKernel{2} end","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"This kernel is going to be an implementation of the Gaussian kernel with a cutoff for compact support. Note that the same kernel in a more optimized version is already implemented in TrixiParticles.jl as GaussianKernel.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"In order to use our new kernel, we have to define three functions. TrixiParticles.kernel, which is the kernel function itself, TrixiParticles.kernel_deriv, which is the derivative of the kernel function, and TrixiParticles.compact_support, which defines the compact support of the kernel in relation to the smoothing length. The latter is relevant for determining the search radius of the neighborhood search.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"function TrixiParticles.kernel(kernel::MyGaussianKernel, r, h)\n q = r / h\n\n if q < 2\n return 1 / (pi * h^2) * exp(-q^2)\n end\n\n return 0.0\nend\n\nfunction TrixiParticles.kernel_deriv(kernel::MyGaussianKernel, r, h)\n q = r * h\n\n if q < 2\n return 1 / (pi * h^2) * (-2 * q) * exp(-q^2) / h\n end\n\n return 0.0\nend\n\nTrixiParticles.compact_support(::MyGaussianKernel, h) = 3 * h","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"For this kernel, we use a different smoothing length, which yields a similar kernel to the SchoenbergCubicSplineKernel that we used earlier.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"smoothing_length_gauss = 1.0 * fluid_particle_spacing","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"We can compare these kernels in a plot.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"using Plots\nx = range(-0.2, 0.2, length=500)\nplot(x, r -> TrixiParticles.kernel(SchoenbergCubicSplineKernel{2}(), abs(r), smoothing_length),\n label=\"SchoenbergCubicSplineKernel\", xlabel=\"r\")\nplot!(x, r -> TrixiParticles.kernel(MyGaussianKernel(), abs(r), smoothing_length_gauss),\n label=\"MyGaussianKernel\")\nplot!(dpi=200); savefig(\"tut_setup_plot2.png\"); nothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"(Image: plot)","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"This is all we need to use our custom kernel implementation in a simulation. We only need to replace the definition above by","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"smoothing_kernel = MyGaussianKernel()\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"and run the simulation file again.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"In order to use our kernel in a pre-defined example file, we can use the function trixi_include to replace the definition of the variable smoothing_kernel. The following will run the example simulation examples/fluid/hydrostatic_water_column_2d.jl with our custom kernel.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"trixi_include(joinpath(examples_dir(), \"fluid\", \"hydrostatic_water_column_2d.jl\"),\n smoothing_kernel=MyGaussianKernel());\nnothing # hide","category":"page"},{"location":"news/","page":"News","title":"News","text":"EditURL = \"https://github.com/trixi-framework/TrixiParticles.jl/blob/main/NEWS.md\"","category":"page"},{"location":"news/#Changelog","page":"News","title":"Changelog","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"TrixiParticles.jl follows the interpretation of semantic versioning (semver) used in the Julia ecosystem. Notable changes will be documented in this file for human readability.","category":"page"},{"location":"news/#Version-0.2.3","page":"News","title":"Version 0.2.3","text":"","category":"section"},{"location":"news/#Highlights","page":"News","title":"Highlights","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"Transport Velocity Formulation (TVF) based on the work of Ramachandran et al. \"Entropically damped artificial compressibility for SPH\" (2019) was added.","category":"page"},{"location":"news/#Version-0.2.2","page":"News","title":"Version 0.2.2","text":"","category":"section"},{"location":"news/#Highlights-2","page":"News","title":"Highlights","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"Hotfix for threaded sampling of complex geometries.","category":"page"},{"location":"news/#Version-0.2.1","page":"News","title":"Version 0.2.1","text":"","category":"section"},{"location":"news/#Highlights-3","page":"News","title":"Highlights","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"Particle sampling of complex geometries from .stl and .asc files.","category":"page"},{"location":"news/#Version-0.2.0","page":"News","title":"Version 0.2.0","text":"","category":"section"},{"location":"news/#Removed","page":"News","title":"Removed","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"Use of the internal neighborhood search has been removed and replaced with PointNeighbors.jl.","category":"page"},{"location":"news/#Development-Cycle-0.1","page":"News","title":"Development Cycle 0.1","text":"","category":"section"},{"location":"news/#Highlights-4","page":"News","title":"Highlights","text":"","category":"section"},{"location":"news/#Discrete-Element-Method","page":"News","title":"Discrete Element Method","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"A basic implementation of the discrete element method was added.","category":"page"},{"location":"news/#Surface-Tension-and-Adhesion-Model","page":"News","title":"Surface Tension and Adhesion Model","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"A surface tension and adhesion model based on the work by Akinci et al., \"Versatile Surface Tension and Adhesion for SPH Fluids\" (2013) was added to WCSPH.","category":"page"},{"location":"news/#Support-for-Open-Boundaries","page":"News","title":"Support for Open Boundaries","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"Open boundaries using the method of characteristics based on the work of Lastiwka et al., \"Permeable and non-reflecting boundary conditions in SPH\" (2009) were added for WCSPH and EDAC.","category":"page"},{"location":"news/#Pre-Initial-Release-(v0.1.0)","page":"News","title":"Pre Initial Release (v0.1.0)","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"This section summarizes the initial features that TrixiParticles.jl was released with.","category":"page"},{"location":"news/#Highlights-5","page":"News","title":"Highlights","text":"","category":"section"},{"location":"news/#EDAC","page":"News","title":"EDAC","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"An implementation of EDAC (Entropically Damped Artificial Compressibility) was added, which allows for more stable simulations compared to basic WCSPH and reduces spurious pressure oscillations.","category":"page"},{"location":"news/#WCSPH","page":"News","title":"WCSPH","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"An implementation of WCSPH (Weakly Compressible Smoothed Particle Hydrodynamics), which is the classical SPH approach.","category":"page"},{"location":"news/","page":"News","title":"News","text":"Features:","category":"page"},{"location":"news/","page":"News","title":"News","text":"Correction schemes (Shepard (0. Order) ... MixedKernelGradient (1. Order))\nDensity reinitialization\nKernel summation and Continuity equation density formulations\nFlexible boundary conditions e.g. dummy particles with Adami pressure extrapolation, pressure zeroing, pressure mirroring...\nMoving boundaries\nDensity diffusion based on the models by Molteni & Colagrossi (2009), Ferrari et al. (2009) and Antuono et al. (2010).","category":"page"},{"location":"news/#TLSPH","page":"News","title":"TLSPH","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"An implementation of TLSPH (Total Lagrangian Smoothed Particle Hydrodynamics) for solid bodies enabling FSI (Fluid Structure Interactions).","category":"page"},{"location":"general/util/#Util","page":"Util","title":"Util","text":"","category":"section"},{"location":"general/util/","page":"Util","title":"Util","text":"Modules = [TrixiParticles]\nPages = [\"util.jl\"]","category":"page"},{"location":"general/util/#TrixiParticles.examples_dir-Tuple{}","page":"Util","title":"TrixiParticles.examples_dir","text":"examples_dir()\n\nReturn the directory where the example files provided with TrixiParticles.jl are located. If TrixiParticles is installed as a regular package (with ]add TrixiParticles), these files are read-only and should not be modified. To find out which files are available, use, e.g., readdir.\n\nCopied from Trixi.jl.\n\nExamples\n\nreaddir(examples_dir())\n\n\n\n\n\n","category":"method"},{"location":"general/util/#TrixiParticles.validation_dir-Tuple{}","page":"Util","title":"TrixiParticles.validation_dir","text":"validation_dir()\n\nReturn the directory where the validation files provided with TrixiParticles.jl are located. If TrixiParticles is installed as a regular package (with ]add TrixiParticles), these files are read-only and should not be modified. To find out which files are available, use, e.g., readdir.\n\nCopied from Trixi.jl.\n\nExamples\n\nreaddir(validation_dir())\n\n\n\n\n\n","category":"method"},{"location":"general/util/#TrixiParticles.@autoinfiltrate","page":"Util","title":"TrixiParticles.@autoinfiltrate","text":"@autoinfiltrate\n@autoinfiltrate condition::Bool\n\nInvoke the @infiltrate macro of the package Infiltrator.jl to create a breakpoint for ad-hoc interactive debugging in the REPL. If the optional argument condition is given, the breakpoint is only enabled if condition evaluates to true.\n\nAs opposed to using Infiltrator.@infiltrate directly, this macro does not require Infiltrator.jl to be added as a dependency to TrixiParticles.jl. As a bonus, the macro will also attempt to load the Infiltrator module if it has not yet been loaded manually.\n\nNote: For this macro to work, the Infiltrator.jl package needs to be installed in your current Julia environment stack.\n\nSee also: Infiltrator.jl\n\nwarning: Internal use only\nPlease note that this macro is intended for internal use only. It is not part of the public API of TrixiParticles.jl, and it thus can altered (or be removed) at any time without it being considered a breaking change.\n\n\n\n\n\n","category":"macro"},{"location":"tutorials_template/tut_falling/#Example-file","page":"Example file","title":"Example file","text":"","category":"section"},{"location":"tutorials_template/tut_falling/","page":"Example file","title":"Example file","text":"!!include:examples/fsi/falling_spheres_2d.jl!!\n","category":"page"},{"location":"code_of_conduct/","page":"Code of Conduct","title":"Code of Conduct","text":"EditURL = \"https://github.com/trixi-framework/TrixiParticles.jl/blob/main/CODE_OF_CONDUCT.md\"","category":"page"},{"location":"code_of_conduct/#Code-of-Conduct","page":"Code of Conduct","title":"Code of Conduct","text":"","category":"section"},{"location":"code_of_conduct/","page":"Code of Conduct","title":"Code of Conduct","text":"Contributor Covenant Code of ConductOur PledgeWe as members, contributors, and leaders pledge to make participation in our community a harassment-free experience for everyone, regardless of age, body size, visible or invisible disability, ethnicity, sex characteristics, gender identity and expression, level of experience, education, socio-economic status, nationality, personal appearance, race, religion, or sexual identity and orientation.We pledge to act and interact in ways that contribute to an open, welcoming, diverse, inclusive, and healthy community.Our StandardsExamples of behavior that contributes to a positive environment for our community include:Demonstrating empathy and kindness toward other people\nBeing respectful of differing opinions, viewpoints, and experiences\nGiving and gracefully accepting constructive feedback\nAccepting responsibility and apologizing to those affected by our mistakes, and learning from the experience\nFocusing on what is best not just for us as individuals, but for the overall communityExamples of unacceptable behavior include:The use of sexualized language or imagery, and sexual attention or advances of any kind\nTrolling, insulting or derogatory comments, and personal or political attacks\nPublic or private harassment\nPublishing others' private information, such as a physical or email address, without their explicit permission\nOther conduct which could reasonably be considered inappropriate in a professional settingEnforcement ResponsibilitiesCommunity leaders are responsible for clarifying and enforcing our standards of acceptable behavior and will take appropriate and fair corrective action in response to any behavior that they deem inappropriate, threatening, offensive, or harmful.Community leaders have the right and responsibility to remove, edit, or reject comments, commits, code, wiki edits, issues, and other contributions that are not aligned to this Code of Conduct, and will communicate reasons for moderation decisions when appropriate.ScopeThis Code of Conduct applies within all community spaces, and also applies when an individual is officially representing the community in public spaces. Examples of representing our community include using an official e-mail address, posting via an official social media account, or acting as an appointed representative at an online or offline event.EnforcementInstances of abusive, harassing, or otherwise unacceptable behavior may be reported to Michael Schlottke-Lakemper, Sven Berger, or any other of the principal developers responsible for enforcement listed in Authors. All complaints will be reviewed and investigated promptly and fairly.All community leaders are obligated to respect the privacy and security of the reporter of any incident.Enforcement GuidelinesCommunity leaders will follow these Community Impact Guidelines in determining the consequences for any action they deem in violation of this Code of Conduct:1. CorrectionCommunity Impact: Use of inappropriate language or other behavior deemed unprofessional or unwelcome in the community.Consequence: A private, written warning from community leaders, providing clarity around the nature of the violation and an explanation of why the behavior was inappropriate. A public apology may be requested.2. WarningCommunity Impact: A violation through a single incident or series of actions.Consequence: A warning with consequences for continued behavior. No interaction with the people involved, including unsolicited interaction with those enforcing the Code of Conduct, for a specified period of time. This includes avoiding interactions in community spaces as well as external channels like social media. Violating these terms may lead to a temporary or permanent ban.3. Temporary BanCommunity Impact: A serious violation of community standards, including sustained inappropriate behavior.Consequence: A temporary ban from any sort of interaction or public communication with the community for a specified period of time. No public or private interaction with the people involved, including unsolicited interaction with those enforcing the Code of Conduct, is allowed during this period. Violating these terms may lead to a permanent ban.4. Permanent BanCommunity Impact: Demonstrating a pattern of violation of community standards, including sustained inappropriate behavior, harassment of an individual, or aggression toward or disparagement of classes of individuals.Consequence: A permanent ban from any sort of public interaction within the community.AttributionThis Code of Conduct is adapted from the [Contributor Covenant][homepage], version 2.0, available at https://www.contributor-covenant.org/version/2/0/codeofconduct.html.Community Impact Guidelines were inspired by Mozilla's code of conduct enforcement ladder.[homepage]: https://www.contributor-covenant.orgFor answers to common questions about this code of conduct, see the FAQ at https://www.contributor-covenant.org/faq. Translations are available at https://www.contributor-covenant.org/translations.","category":"page"},{"location":"systems/entropically_damped_sph/#edac","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility (EDAC) for SPH","text":"","category":"section"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"As opposed to the weakly compressible SPH scheme, which uses an equation of state, this scheme uses a pressure evolution equation to calculate the pressure","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"fracmathrmd p_amathrmdt = - rho c_s^2 nabla cdot v + nu nabla^2 p","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"which is derived by Clausen (2013). This equation is similar to the continuity equation (first term, see ContinuityDensity), but also contains a pressure damping term (second term, similar to density diffusion see DensityDiffusion), which reduces acoustic pressure waves through an entropy-generation mechanism.","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"The pressure evolution is discretized with the SPH method by Ramachandran (2019) as following:","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"The first term is equivalent to the classical artificial compressible methods, which are commonly motivated by assuming the artificial equation of state (StateEquationCole with exponent=1) and is discretized as","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"- rho c_s^2 nabla cdot v = sum_b m_b fracrho_arho_b c_s^2 v_ab cdot nabla_r_a W(Vert r_a - r_b Vert h)","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"where rho_a, rho_b, r_a, r_b, denote the density and coordinates of particles a and b respectively, c_s is the speed of sound and v_ab = v_a - v_b is the difference in the velocity.","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"The second term smooths the pressure through the introduction of entropy and is discretized as","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"nu nabla^2 p = fracV_a^2 + V_b^2m_a tildeeta_ab fracp_abVert r_ab^2 Vert + eta h_ab^2 nabla_r_a\nW(Vert r_a - r_b Vert h) cdot r_ab","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"where V_a, V_b denote the volume of particles a and b respectively and p_ab= p_a -p_b is the difference in the pressure.","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"The viscosity parameter eta_a for a particle a is given as","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"eta_a = rho_a fracalpha h c_s8","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"where it is found in the numerical experiments of Ramachandran (2019) that alpha = 05 is a good choice for a wide range of Reynolds numbers (0.0125 to 10000).","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"note: Note\nThe EDAC formulation keeps the density constant and this eliminates the need for the continuity equation or the use of a summation density to find the pressure. However, in SPH discretizations, mrho is typically used as a proxy for the particle volume. The density of the fluids can therefore be computed using the summation density approach. [19]","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"fluid\", \"entropically_damped_sph\", \"system.jl\")]","category":"page"},{"location":"systems/entropically_damped_sph/#TrixiParticles.EntropicallyDampedSPHSystem","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"TrixiParticles.EntropicallyDampedSPHSystem","text":"EntropicallyDampedSPHSystem(initial_condition, smoothing_kernel,\n smoothing_length, sound_speed;\n pressure_acceleration=inter_particle_averaged_pressure,\n density_calculator=SummationDensity(),\n transport_velocity=nothing,\n alpha=0.5, viscosity=nothing,\n acceleration=ntuple(_ -> 0.0, NDIMS), buffer_size=nothing,\n source_terms=nothing)\n\nSystem for particles of a fluid. As opposed to the weakly compressible SPH scheme, which uses an equation of state, this scheme uses a pressure evolution equation to calculate the pressure. See Entropically Damped Artificial Compressibility for SPH for more details on the method.\n\nArguments\n\ninitial_condition: Initial condition representing the system's particles.\nsound_speed: Speed of sound.\nsmoothing_kernel: Smoothing kernel to be used for this system. See Smoothing Kernels.\nsmoothing_length: Smoothing length to be used for this system. See Smoothing Kernels.\n\nKeyword Arguments\n\nviscosity: Viscosity model for this system (default: no viscosity). Recommended: ViscosityAdami.\nacceleration: Acceleration vector for the system. (default: zero vector)\npressure_acceleration: Pressure acceleration formulation (default: inter-particle averaged pressure). When set to nothing, the pressure acceleration formulation for the corresponding density calculator is chosen.\ndensity_calculator: Density calculator (default: SummationDensity)\ntransport_velocity: Transport Velocity Formulation (TVF). Default is no TVF.\nbuffer_size: Number of buffer particles. This is needed when simulating with OpenBoundarySPHSystem.\nsource_terms: Additional source terms for this system. Has to be either nothing (by default), or a function of (coords, velocity, density, pressure, t) (which are the quantities of a single particle), returning a Tuple or SVector that is to be added to the acceleration of that particle. See, for example, SourceTermDamping. Note that these source terms will not be used in the calculation of the boundary pressure when using a boundary with BoundaryModelDummyParticles and AdamiPressureExtrapolation. The keyword argument acceleration should be used instead for gravity-like source terms.\n\n\n\n\n\n","category":"type"},{"location":"systems/entropically_damped_sph/#transport_velocity_formulation","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Transport Velocity Formulation (TVF)","text":"","category":"section"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"Standard SPH suffers from problems like tensile instability or the creation of void regions in the flow. To address these problems, Adami (2013) modified the advection velocity and added an extra term to the momentum equation. The authors introduced the so-called Transport Velocity Formulation (TVF) for WCSPH. Ramachandran (2019) applied the TVF also for the EDAC scheme.","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"The transport velocity tildev_a of particle a is used to evolve the position of the particle r_a from one time step to the next by","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"fracmathrmd r_amathrmdt = tildev_a","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"and is obtained at every time-step Delta t from","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"tildev_a (t + Delta t) = v_a (t) + Delta t left(fractildemathrmd v_amathrmdt - frac1rho_a nabla p_textbackground right)","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"where rho_a is the density of particle a and p_textbackground is a constant background pressure field. The tilde in the second term of the right hand side indicates that the material derivative has an advection part.","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"The discretized form of the last term is","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":" -frac1rho_a nabla p_textbackground approx -fracp_textbackgroundm_a sum_b left(V_a^2 + V_b^2 right) nabla_a W_ab","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"where V_a, V_b denote the volume of particles a and b respectively. Note that although in the continuous case nabla p_textbackground = 0, the discretization is not 0th-order consistent for non-uniform particle distribution, which means that there is a non-vanishing contribution only when particles are disordered. That also means that p_textbackground occurs as prefactor to correct the trajectory of a particle resulting in uniform pressure distributions. Suggested is a background pressure which is in the order of the reference pressure but can be chosen arbitrarily large when the time-step criterion is adjusted.","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"The inviscid momentum equation with an additional convection term for a particle moving with tildev is","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"fractildemathrmd left( rho v right)mathrmdt = -nabla p + nabla cdot bmA","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"where the tensor bmA = rho vleft(tildev-vright)^T is a consequence of the modified advection velocity and can be interpreted as the convection of momentum with the relative velocity tildev-v.","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"The discretized form of the momentum equation for a particle a reads as","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"fractildemathrmd v_amathrmdt = frac1m_a sum_b left(V_a^2 + V_b^2 right) left -tildep_ab nabla_a W_ab + frac12 left(bmA_a + bmA_b right) cdot nabla_a W_ab right","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"Here, tildep_ab is the density-weighted pressure","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"tildep_ab = fracrho_b p_a + rho_a p_brho_a + rho_b","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"with the density rho_a, rho_b and the pressure p_a, p_b of particles a and b respectively. bmA_a and bmA_b are the convection tensors for particle a and b respectively and are given, e.g. for particle a, as bmA_a = rho v_aleft(tildev_a-v_aright)^T.","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"fluid\", \"transport_velocity.jl\")]","category":"page"},{"location":"systems/entropically_damped_sph/#TrixiParticles.TransportVelocityAdami","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"TrixiParticles.TransportVelocityAdami","text":"TransportVelocityAdami(background_pressure::Real)\n\nTransport Velocity Formulation (TVF) to suppress pairing and tensile instability. See TVF for more details of the method.\n\nArguments\n\nbackground_pressure: Background pressure. Suggested is a background pressure which is on the order of the reference pressure.\n\nnote: Note\nThere is no need for an artificial viscosity to suppress tensile instability when using TransportVelocityAdami. Thus, it is highly recommended to use ViscosityAdami as viscosity model, since ArtificialViscosityMonaghan leads to bad results.\n\n\n\n\n\n","category":"type"},{"location":"authors/","page":"Authors","title":"Authors","text":"EditURL = \"https://github.com/trixi-framework/TrixiParticles.jl/blob/main/AUTHORS.md\"","category":"page"},{"location":"authors/#Authors","page":"Authors","title":"Authors","text":"","category":"section"},{"location":"authors/","page":"Authors","title":"Authors","text":"TrixiParticles.jl's development is coordinated by a group of principal developers, who are also its main contributors and who can be contacted in case of questions about TrixiParticles.jl. In addition, there are contributors who have provided substantial additions or modifications. Together, these two groups form \"The TrixiParticles.jl Authors\" as mentioned under License.","category":"page"},{"location":"authors/#Principal-Developers","page":"Authors","title":"Principal Developers","text":"","category":"section"},{"location":"authors/","page":"Authors","title":"Authors","text":"Erik Faulhaber, University of Cologne, Germany\nNiklas Neher, High-Performance Computing Center Stuttgart (HLRS), Germany\nSven Berger, Helmholtz Center Hereon, Germany","category":"page"},{"location":"authors/#Contributors","page":"Authors","title":"Contributors","text":"","category":"section"},{"location":"authors/","page":"Authors","title":"Authors","text":"The following people contributed major additions or modifications to TrixiParticles.jl and are listed in alphabetical order:","category":"page"},{"location":"authors/","page":"Authors","title":"Authors","text":"Sven Berger\nErik Faulhaber\nGregor Gassner\nNiklas Neher\nHendrik Ranocha\nMichael Schlottke-Lakemper","category":"page"},{"location":"tutorials_template/tut_dam_break/#Example-file","page":"Example file","title":"Example file","text":"","category":"section"},{"location":"tutorials_template/tut_dam_break/","page":"Example file","title":"Example file","text":"!!include:examples/fluid/dam_break_2d.jl!!\n","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"EditURL = \"https://github.com/trixi-framework/TrixiParticles.jl/blob/main/docs/src/tutorials_template/tut_setup.md\"","category":"page"},{"location":"tutorials/tut_setup/#Setting-up-your-simulation-from-scratch","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"","category":"section"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"In this tutorial, we will guide you through the general structure of simulation files. We will set up a simulation similar to the example simulation examples/fluid/hydrostatic_water_column_2d.jl, which is one of our simplest example simulations. In the second part of this tutorial, we will show how to replace components of TrixiParticles.jl by custom implementations from within a simulation file, without ever cloning the repository.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"For different setups and physics, have a look at our other example files.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"First, we import TrixiParticles.jl and OrdinaryDiffEq.jl, which we will use at the very end for the time integration.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"using TrixiParticles\nusing OrdinaryDiffEq","category":"page"},{"location":"tutorials/tut_setup/#Resolution","page":"Setting up your simulation from scratch","title":"Resolution","text":"","category":"section"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"Now, we define the particle spacing, which is our numerical resolution. We usually call the variable fluid_particle_spacing, so that we can easily change the resolution of an example file by overwriting this variable with trixi_include. In 2D, the number of particles will grow quadratically, in 3D cubically with the spacing.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"We also set the number of boundary layers, which need to be sufficiently large, depending on the smoothing kernel and smoothing length, so that the compact support of the smoothing kernel is fully sampled with particles for a fluid particle close to a boundary. In particular, we require boundary_layers >= compact_support. The value for the compact support for each kernel can be found in the smoothing kernel overview.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"fluid_particle_spacing = 0.05\nboundary_layers = 3\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/#Experiment-setup","page":"Setting up your simulation from scratch","title":"Experiment setup","text":"","category":"section"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"We want to simulate a water column resting under hydrostatic pressure inside a rectangular tank. (Image: Experiment Setup) First, we define the physical parameters gravitational acceleration, simulation time, initial fluid size, tank size and fluid density.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"gravity = 9.81\ntspan = (0.0, 1.0)\ninitial_fluid_size = (1.0, 0.9)\ntank_size = (1.0, 1.0)\nfluid_density = 1000.0\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"In order to have the initial particle mass and density correspond to the hydrostatic pressure gradient, we need to define a state equation, which relates the fluid density to pressure. Note that we could also skip this part here and define the state equation later when we define the fluid system, but then the fluid would be initialized with constant density, which would cause it to oscillate under gravity.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"sound_speed = 10.0\nstate_equation = StateEquationCole(; sound_speed, reference_density=fluid_density,\n exponent=7)\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"The speed of sound here is numerical and not physical. We artificially lower the speed of sound, since the physical speed of sound in water would lead to prohibitively small time steps. The speed of sound in Weakly Compressible SPH should be chosen as small as possible for numerical efficiency, but large enough to limit density fluctuations to about 1%.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"TrixiParticles.jl requires the initial particle positions and quantities in form of an InitialCondition. Instead of manually defining particle positions, you can work with our pre-defined setups. Among others, we provide setups for rectangular shapes, circles, and spheres. Initial conditions can also be combined with common set operations. See this page for a list of pre-defined setups and details on set operations on initial conditions.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"Here, we use the RectangularTank setup, which generates a rectangular fluid inside a rectangular tank, and supports a hydrostatic pressure gradient by passing a gravitational acceleration and a state equation (see above).","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"tank = RectangularTank(fluid_particle_spacing, initial_fluid_size, tank_size,\n fluid_density, n_layers=boundary_layers,\n acceleration=(0.0, -gravity), state_equation=state_equation)\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/#Fluid-system","page":"Setting up your simulation from scratch","title":"Fluid system","text":"","category":"section"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"To model the water column, we use the Weakly Compressible Smoothed Particle Hydrodynamics (WCSPH) method. This method requires a smoothing kernel and a corresponding smoothing length, which should be chosen in relation to the particle spacing.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"smoothing_length = 1.2 * fluid_particle_spacing\nsmoothing_kernel = SchoenbergCubicSplineKernel{2}()\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"You can find an overview over smoothing kernels and corresponding smoothing lengths here.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"For stability, we need numerical dissipation in form of an artificial viscosity term. Other viscosity models offer a physical approach based on the kinematic viscosity of the fluid.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"viscosity = ArtificialViscosityMonaghan(alpha=0.02, beta=0.0)\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"We choose the parameters as small as possible to avoid non-physical behavior, but as large as possible to stabilize the simulation.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"The WCSPH method can either compute the particle density directly with a kernel summation over all neighboring particles (see SummationDensity) or by making the particle density a variable in the ODE system and integrating its change over time. We choose the latter approach here by using the density calculator ContinuityDensity, which is more efficient and handles free surfaces without the need for additional correction terms.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"fluid_density_calculator = ContinuityDensity()\nfluid_system = WeaklyCompressibleSPHSystem(tank.fluid, fluid_density_calculator,\n state_equation, smoothing_kernel,\n smoothing_length, viscosity=viscosity,\n acceleration=(0.0, -gravity))\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/#Boundary-system","page":"Setting up your simulation from scratch","title":"Boundary system","text":"","category":"section"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"To model the boundary, we use particle-based boundary conditions, in which particles are sampled in the boundary that interact with the fluid particles to avoid penetration. In order to define a boundary system, we first have to choose a boundary model, which defines how the fluid interacts with boundary particles. We will use the BoundaryModelDummyParticles with AdamiPressureExtrapolation. See here for a comprehensive overview over boundary models.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"boundary_model = BoundaryModelDummyParticles(tank.boundary.density, tank.boundary.mass,\n state_equation=state_equation,\n AdamiPressureExtrapolation(),\n smoothing_kernel, smoothing_length)\nboundary_system = BoundarySPHSystem(tank.boundary, boundary_model)\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/#Semidiscretization","page":"Setting up your simulation from scratch","title":"Semidiscretization","text":"","category":"section"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"The key component of every simulation is the Semidiscretization, which couples all systems of the simulation. All methods in TrixiParticles.jl are semidiscretizations, which discretize the equations in time to provide an ordinary differential equation that still has to be solved in time. By providing a simulation time span, we can call semidiscretize, which returns an ODEProblem that can be solved with a time integration method.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"semi = Semidiscretization(fluid_system, boundary_system)\node = semidiscretize(semi, tspan)\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/#Time-integration","page":"Setting up your simulation from scratch","title":"Time integration","text":"","category":"section"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"We use the methods provided by OrdinaryDiffEq.jl, but note that other packages or custom implementations can also be used.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"OrdinaryDiffEq.jl supports callbacks, which are executed during the simulation. For this simulation, we use the InfoCallback, which prints information about the simulation setup at the beginning of the simulation, information about the current simulation time and runtime during the simulation, and a performance summary at the end of the simulation. We also want to save the current solution in regular intervals in terms of simulation time as VTK, so that we can look at the solution in ParaView. The SolutionSavingCallback provides this functionality. To pass the callbacks to OrdinaryDiffEq.jl, we have to bundle them into a CallbackSet.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"info_callback = InfoCallback(interval=50)\nsaving_callback = SolutionSavingCallback(dt=0.02)\n\ncallbacks = CallbackSet(info_callback, saving_callback)\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"Finally, we can start the simulation by solving the ODEProblem. We use the method RDPK3SpFSAL35 of OrdinaryDiffEq.jl, which is a Runge-Kutta method with automatic (error based) time step size control. This method is usually a good choice for prototyping, since we do not have to worry about choosing a stable step size and can just run the simulation. For better performance, it might be beneficial to tweak the tolerances of this method or choose a different method that is more efficient for the respective simulation. You can find both approaches in our example files. Here, we just use the method with the default parameters, and only disable save_everystep to avoid expensive saving of the solution in every time step.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"sol = solve(ode, RDPK3SpFSAL35(), save_everystep=false, callback=callbacks);\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"See Visualization for how to visualize the solution. For the simplest visualization, we can use Plots.jl:","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"using Plots\nplot(sol)\nplot!(dpi=200); savefig(\"tut_setup_plot.png\"); nothing # hide","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"(Image: plot)","category":"page"},{"location":"tutorials/tut_setup/#Replacing-components-with-custom-implementations","page":"Setting up your simulation from scratch","title":"Replacing components with custom implementations","text":"","category":"section"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"If we would like to use an implementation of a component that is not available in TrixiParticles.jl, we can implement it ourselves within the simulation file, without ever cloning the TrixiParticles.jl repository. A good starting point is to check out the available implementations in TrixiParticles.jl, then copy the relevant functions to the simulation file and modify them as needed.","category":"page"},{"location":"tutorials/tut_setup/#Custom-smoothing-kernel","page":"Setting up your simulation from scratch","title":"Custom smoothing kernel","text":"","category":"section"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"To implement a custom smoothing kernel, we define a struct extending TrixiParticles.SmoothingKernel. This abstract struct has a type parameter for the number of dimensions, which we set to 2 in this case.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"struct MyGaussianKernel <: TrixiParticles.SmoothingKernel{2} end","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"This kernel is going to be an implementation of the Gaussian kernel with a cutoff for compact support. Note that the same kernel in a more optimized version is already implemented in TrixiParticles.jl as GaussianKernel.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"In order to use our new kernel, we have to define three functions. TrixiParticles.kernel, which is the kernel function itself, TrixiParticles.kernel_deriv, which is the derivative of the kernel function, and TrixiParticles.compact_support, which defines the compact support of the kernel in relation to the smoothing length. The latter is relevant for determining the search radius of the neighborhood search.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"function TrixiParticles.kernel(kernel::MyGaussianKernel, r, h)\n q = r / h\n\n if q < 2\n return 1 / (pi * h^2) * exp(-q^2)\n end\n\n return 0.0\nend\n\nfunction TrixiParticles.kernel_deriv(kernel::MyGaussianKernel, r, h)\n q = r * h\n\n if q < 2\n return 1 / (pi * h^2) * (-2 * q) * exp(-q^2) / h\n end\n\n return 0.0\nend\n\nTrixiParticles.compact_support(::MyGaussianKernel, h) = 3 * h","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"For this kernel, we use a different smoothing length, which yields a similar kernel to the SchoenbergCubicSplineKernel that we used earlier.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"smoothing_length_gauss = 1.0 * fluid_particle_spacing","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"We can compare these kernels in a plot.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"using Plots\nx = range(-0.2, 0.2, length=500)\nplot(x, r -> TrixiParticles.kernel(SchoenbergCubicSplineKernel{2}(), abs(r), smoothing_length),\n label=\"SchoenbergCubicSplineKernel\", xlabel=\"r\")\nplot!(x, r -> TrixiParticles.kernel(MyGaussianKernel(), abs(r), smoothing_length_gauss),\n label=\"MyGaussianKernel\")\nplot!(dpi=200); savefig(\"tut_setup_plot2.png\"); nothing # hide","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"(Image: plot)","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"This is all we need to use our custom kernel implementation in a simulation. We only need to replace the definition above by","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"smoothing_kernel = MyGaussianKernel()\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"and run the simulation file again.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"In order to use our kernel in a pre-defined example file, we can use the function trixi_include to replace the definition of the variable smoothing_kernel. The following will run the example simulation examples/fluid/hydrostatic_water_column_2d.jl with our custom kernel.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"trixi_include(joinpath(examples_dir(), \"fluid\", \"hydrostatic_water_column_2d.jl\"),\n smoothing_kernel=MyGaussianKernel());\nnothing # hide","category":"page"},{"location":"systems/total_lagrangian_sph/#tlsph","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH","text":"","category":"section"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"A Total Lagrangian framework is used wherein the governing equations are formulated such that all relevant quantities and operators are measured with respect to the initial configuration (O’Connor & Rogers, 2021, Belytschko et al., 2000).","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"The governing equations with respect to the initial configuration are given by:","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"fracmathrmDbmvmathrmDt = frac1rho_0 nabla_0 cdot bmP + bmg","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"where the zero subscript denotes a derivative with respect to the initial configuration and bmP is the first Piola-Kirchhoff (PK1) stress tensor.","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"The discretized version of this equation is given by O’Connor & Rogers (2021):","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"fracmathrmdbmv_amathrmdt = sum_b m_0b\n left( fracbmP_a bmL_0arho_0a^2 + fracbmP_b bmL_0brho_0b^2 right)\n nabla_0a W(bmX_ab) + fracbmf_a^PFm_0a + bmg","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"with the correction matrix (see also GradientCorrection)","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"bmL_0a = left( -sum_b fracm_0brho_0b nabla_0a W(bmX_ab) bmX_ab^T right)^-1 in R^d times d","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"The subscripts a and b denote quantities of particle a and b, respectively. The zero subscript on quantities denotes that the quantity is to be measured in the initial configuration. The difference in the initial coordinates is denoted by bmX_ab = bmX_a - bmX_b, the difference in the current coordinates is denoted by bmx_ab = bmx_a - bmx_b.","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"For the computation of the PK1 stress tensor, the deformation gradient bmF is computed per particle as","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"bmF_a = sum_b fracm_0brho_0b bmx_ba (bmL_0anabla_0a W(bmX_ab))^T \n qquad = -left(sum_b fracm_0brho_0b bmx_ab (nabla_0a W(bmX_ab))^T right) bmL_0a^T","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"with 1 leq ij leq d. From the deformation gradient, the Green-Lagrange strain","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"bmE = frac12(bmF^TbmF - bmI)","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"and the second Piola-Kirchhoff stress tensor","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"bmS = lambda operatornametr(bmE) bmI + 2mu bmE","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"are computed to obtain the PK1 stress tensor as","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"bmP = bmFbmS","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"Here,","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"mu = fracE2(1 + nu)","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"and","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"lambda = fracEnu(1 + nu)(1 - 2nu)","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"are the Lamé coefficients, where E is the Young's modulus and nu is the Poisson ratio.","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"The term bmf_a^PF is an optional penalty force. See e.g. PenaltyForceGanzenmueller.","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"solid\", \"total_lagrangian_sph\", \"system.jl\")]","category":"page"},{"location":"systems/total_lagrangian_sph/#TrixiParticles.TotalLagrangianSPHSystem","page":"Total Lagrangian SPH (Elastic Structure)","title":"TrixiParticles.TotalLagrangianSPHSystem","text":"TotalLagrangianSPHSystem(initial_condition,\n smoothing_kernel, smoothing_length,\n young_modulus, poisson_ratio;\n n_fixed_particles=0, boundary_model=nothing,\n acceleration=ntuple(_ -> 0.0, NDIMS),\n penalty_force=nothing, source_terms=nothing)\n\nSystem for particles of an elastic structure.\n\nA Total Lagrangian framework is used wherein the governing equations are formulated such that all relevant quantities and operators are measured with respect to the initial configuration (O’Connor & Rogers 2021, Belytschko et al. 2000). See Total Lagrangian SPH for more details on the method.\n\nArguments\n\ninitial_condition: Initial condition representing the system's particles.\nyoung_modulus: Young's modulus.\npoisson_ratio: Poisson ratio.\nsmoothing_kernel: Smoothing kernel to be used for this system. See Smoothing Kernels.\nsmoothing_length: Smoothing length to be used for this system. See Smoothing Kernels.\n\nKeyword Arguments\n\nn_fixed_particles: Number of fixed particles which are used to clamp the structure particles. Note that the fixed particles must be the last particles in the InitialCondition. See the info box below.\nboundary_model: Boundary model to compute the hydrodynamic density and pressure for fluid-structure interaction (see Boundary Models).\npenalty_force: Penalty force to ensure regular particle position under large deformations (see PenaltyForceGanzenmueller).\nacceleration: Acceleration vector for the system. (default: zero vector)\nsource_terms: Additional source terms for this system. Has to be either nothing (by default), or a function of (coords, velocity, density, pressure) (which are the quantities of a single particle), returning a Tuple or SVector that is to be added to the acceleration of that particle. See, for example, SourceTermDamping.\n\nnote: Note\nThe fixed particles must be the last particles in the InitialCondition. To do so, e.g. use the union function:solid = union(beam, fixed_particles)where beam and fixed_particles are of type InitialCondition.\n\n\n\n\n\n","category":"type"},{"location":"systems/total_lagrangian_sph/#Penalty-Force","page":"Total Lagrangian SPH (Elastic Structure)","title":"Penalty Force","text":"","category":"section"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"In FEM, underintegrated elements can deform without an associated increase of energy. This is caused by the stiffness matrix having zero eigenvalues (so-called hourglass modes). The name \"hourglass modes\" comes from the fact that elements can deform into an hourglass shape.","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"Similar effects can occur in SPH as well. Particles can change positions without changing the SPH approximation of the deformation gradient bmF, thus, without causing an increase of energy. To ensure regular particle positions, we can apply similar correction forces as are used in FEM.","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"Ganzenmüller (2015) introduced a so-called hourglass correction force or penalty force f^PF, which is given by","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"bmf_a^PF = frac12 alpha sum_b fracm_0a m_0b W_0abrho_0arho_0b bmX_ab^2\n left( E delta_ab^a + E delta_ba^b right) fracbmx_abbmx_ab","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"The subscripts a and b denote quantities of particle a and b, respectively. The zero subscript on quantities denotes that the quantity is to be measured in the initial configuration. The difference in the initial coordinates is denoted by bmX_ab = bmX_a - bmX_b, the difference in the current coordinates is denoted by bmx_ab = bmx_a - bmx_b. Note that Ganzenmüller (2015) has a flipped sign here because they define bmx_ab the other way around.","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"This correction force is based on the potential energy density of a Hookean material. Thus, E is the Young's modulus and alpha is a dimensionless coefficient that controls the amplitude of hourglass correction. The separation vector delta_ab^a indicates the change of distance which the particle separation should attain in order to minimize the error and is given by","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":" delta_ab^a = fracbmepsilon_ab^a cdot bmx_abbmx_ab","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"where the error vector is defined as","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":" bmepsilon_ab^a = bmF_a bmX_ab - bmx_ab","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"solid\", \"total_lagrangian_sph\", \"penalty_force.jl\")]","category":"page"},{"location":"systems/total_lagrangian_sph/#TrixiParticles.PenaltyForceGanzenmueller","page":"Total Lagrangian SPH (Elastic Structure)","title":"TrixiParticles.PenaltyForceGanzenmueller","text":"PenaltyForceGanzenmueller(; alpha=0.1)\n\nPenalty force to ensure regular particle positions under large deformations.\n\nKeywords\n\nalpha: Coefficient to control the amplitude of hourglass correction.\n\n\n\n\n\n","category":"type"},{"location":"license/","page":"License","title":"License","text":"EditURL = \"https://github.com/trixi-framework/TrixiParticles.jl/blob/main/LICENSE.md\"","category":"page"},{"location":"license/#License","page":"License","title":"License","text":"","category":"section"},{"location":"license/","page":"License","title":"License","text":"MIT LicenseCopyright (c) 2023-present The TrixiParticles.jl Authors (see Authors) \nCopyright (c) 2023-present Helmholtz-Zentrum hereon GmbH, Institute of Surface Science \n \nPermission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the \"Software\"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.","category":"page"},{"location":"getting_started/#getting_started","page":"Getting started","title":"Getting started","text":"","category":"section"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"If you have not installed TrixiParticles.jl, please follow the instructions given here.","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"In the following sections, we will give a short introduction. For a more thorough discussion, take a look at our Tutorials.","category":"page"},{"location":"getting_started/#Running-an-Example","page":"Getting started","title":"Running an Example","text":"","category":"section"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"The easiest way to run a simulation is to run one of our predefined example files. We will run the file examples/fluid/hydrostatic_water_column_2d.jl, which simulates a fluid resting in a rectangular tank. Since TrixiParticles.jl uses multithreading, you should start Julia with the flag --threads auto (or, e.g. --threads 4 for 4 threads).","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"In the Julia REPL, first load the package TrixiParticles.jl.","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"julia> using TrixiParticles","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"Then start the simulation by executing","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"julia> trixi_include(joinpath(examples_dir(), \"fluid\", \"hydrostatic_water_column_2d.jl\"))","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"The easiest way to quickly visualize the result is to use Plots.jl:","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"julia> using Plots; plot(sol)","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"This will open a new window with a 2D visualization of the final solution: (Image: plot_hydrostatic_water_column)","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"For more information about visualization, see Visualization.","category":"page"},{"location":"getting_started/#Running-other-Examples","page":"Getting started","title":"Running other Examples","text":"","category":"section"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"You can find a list of our other predefined examples under Examples. Execute them as follows from the Julia REPL by replacing subfolder and example_name","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"julia> trixi_include(joinpath(examples_dir(), \"subfolder\", \"example_name.jl\"))","category":"page"},{"location":"getting_started/#Modifying-an-example","page":"Getting started","title":"Modifying an example","text":"","category":"section"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"You can pass keyword arguments to the function trixi_include to overwrite assignments in the file.","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"With trixi_include, we can overwrite variables defined in the example file to run a different simulation without modifying the example file.","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"julia> trixi_include(joinpath(examples_dir(), \"fluid\", \"hydrostatic_water_column_2d.jl\"), initial_fluid_size=(1.0, 0.5))","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"This for example, will change the fluid size from (09 10) to (10 05).","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"To understand why, take a look into the file hydrostatic_water_column_2d.jl in the subfolder fluid inside the examples directory, which is the file that we executed earlier. You can see that the initial size of the fluid is defined in the variable initial_fluid_size, which we could overwrite with the trixi_include call above. Another variable that is worth experimenting with is fluid_particle_spacing, which controls the resolution of the simulation in this case. A lower value will increase the resolution and the runtime.","category":"page"},{"location":"getting_started/#Set-up-you-first-simulation-from-scratch","page":"Getting started","title":"Set up you first simulation from scratch","text":"","category":"section"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"See Set up your first simulation.","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"Find an overview over the available tutorials under Tutorials.","category":"page"},{"location":"examples/#examples","page":"Examples","title":"Examples","text":"","category":"section"},{"location":"examples/#Fluid","page":"Examples","title":"Fluid","text":"","category":"section"},{"location":"examples/#Accelerated-Tank-2D-(fluid/accelerated_tank_2d.jl)","page":"Examples","title":"Accelerated Tank 2D (fluid/accelerated_tank_2d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Dam-Break-2D-(fluid/dam_break_2d.jl)","page":"Examples","title":"Dam Break 2D (fluid/dam_break_2d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Dam-Break-3D-(fluid/dam_break_3d.jl)","page":"Examples","title":"Dam Break 3D (fluid/dam_break_3d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Falling-Water-Column-(fluid/falling_water_column_2d.jl)","page":"Examples","title":"Falling Water Column (fluid/falling_water_column_2d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Hydrostatic-Water-Column-(fluid/hydrostatic_water_column_*.jl)","page":"Examples","title":"Hydrostatic Water Column (fluid/hydrostatic_water_column_*.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Moving-Wall-(fluid/moving_wall_2d.jl)","page":"Examples","title":"Moving Wall (fluid/moving_wall_2d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Oscillating-Drop-(fluid/oscillating_drop_2d.jl)","page":"Examples","title":"Oscillating Drop (fluid/oscillating_drop_2d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Periodic-Channel-(fluid/periodic_channel_2d.jl)","page":"Examples","title":"Periodic Channel (fluid/periodic_channel_2d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Fluid-Structure-Interaction","page":"Examples","title":"Fluid Structure Interaction","text":"","category":"section"},{"location":"examples/#Dam-Break-with-Elastic-Plate-(fsi/dam_break_plate_2d.jl)","page":"Examples","title":"Dam Break with Elastic Plate (fsi/dam_break_plate_2d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Falling-Sphere-2D-(fsi/falling_sphere_2d.jl)","page":"Examples","title":"Falling Sphere 2D (fsi/falling_sphere_2d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Falling-Spheres-2D-(fsi/falling_spheres_2d.jl)","page":"Examples","title":"Falling Spheres 2D (fsi/falling_spheres_2d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Structure-Mechanics","page":"Examples","title":"Structure Mechanics","text":"","category":"section"},{"location":"examples/#Oscillating-Beam-(solid/oscillating_beam_2d.jl)","page":"Examples","title":"Oscillating Beam (solid/oscillating_beam_2d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"overview/#Overview","page":"Overview","title":"Overview","text":"","category":"section"},{"location":"overview/","page":"Overview","title":"Overview","text":"The actual API reference is not listed on a single page, like in most Julia packages, but instead is split into multiple sections that follow a similar structure as the code files themselves. In these sections, API docs are combined with explanations of the theoretical background of these methods.","category":"page"},{"location":"overview/","page":"Overview","title":"Overview","text":"The following page gives a rough overview of important parts of the code.","category":"page"},{"location":"overview/#Program-flow","page":"Overview","title":"Program flow","text":"","category":"section"},{"location":"overview/","page":"Overview","title":"Overview","text":"To initiate a simulation, the goal is to solve an ordinary differential equation, for example, by employing the time integration schemes provided by OrdinaryDiffEq.jl. These schemes are then utilized to integrate mathrmdumathrmdt and mathrmdvmathrmdt, where u represents the particles' positions and v their properties such as velocity and density. During a single time step or an intermediate step of the time integration scheme, the functions drift! and kick! are invoked, followed by the functions depicted in this diagram (with key parts highlighted in orange/yellow).","category":"page"},{"location":"overview/","page":"Overview","title":"Overview","text":"(Image: Main Program Flow)","category":"page"},{"location":"overview/#Structure","page":"Overview","title":"Structure","text":"","category":"section"},{"location":"overview/","page":"Overview","title":"Overview","text":"What we refer to as schemes are various models such as Weakly Compressible Smoothed Particle Hydrodynamics (WCSPH) or Total Lagrangian Smoothed Particle Hydrodynamics (TLSPH). These schemes are categorized based on the applicable physical regimes, namely fluid, solid, gas, and others. Each scheme comprises at least two files: a system.jl file and an rhs.jl file. The system.jl file provides the data structure holding the particles of this scheme and some routines, particularly those for allocation and the main update routines, excluding system interactions. The interactions between particles of this scheme (and with particles of other schemes) are handled in the rhs.jl file.","category":"page"},{"location":"callbacks/#Callbacks","page":"Callbacks","title":"Callbacks","text":"","category":"section"},{"location":"callbacks/","page":"Callbacks","title":"Callbacks","text":"Modules = [TrixiParticles]\nPages = map(file -> joinpath(\"callbacks\", file), readdir(joinpath(\"..\", \"src\", \"callbacks\")))","category":"page"},{"location":"callbacks/#TrixiParticles.DensityReinitializationCallback","page":"Callbacks","title":"TrixiParticles.DensityReinitializationCallback","text":"DensityReinitializationCallback(; interval::Integer=0, dt=0.0)\n\nCallback to reinitialize the density field when using ContinuityDensity [42].\n\nKeywords\n\ninterval=0: Reinitialize the density every interval time steps.\ndt: Reinitialize the density in regular intervals of dt in terms of integration time.\nreinit_initial_solution: Reinitialize the initial solution (default=false)\n\n\n\n\n\n","category":"type"},{"location":"callbacks/#TrixiParticles.InfoCallback-Tuple{}","page":"Callbacks","title":"TrixiParticles.InfoCallback","text":"InfoCallback()\n\nCreate and return a callback that prints a human-readable summary of the simulation setup at the beginning of a simulation and then resets the timer. When the returned callback is executed directly, the current timer values are shown.\n\n\n\n\n\n","category":"method"},{"location":"callbacks/#TrixiParticles.PostprocessCallback","page":"Callbacks","title":"TrixiParticles.PostprocessCallback","text":"PostprocessCallback(; interval::Integer=0, dt=0.0, exclude_boundary=true, filename=\"values\",\n output_directory=\"out\", append_timestamp=false, write_csv=true,\n write_json=true, write_file_interval=1, funcs...)\n\nCreate a callback to post-process simulation data at regular intervals. This callback allows for the execution of a user-defined function func at specified intervals during the simulation. The function is applied to the current state of the simulation, and its results can be saved or used for further analysis. The provided function cannot be anonymous as the function name will be used as part of the name of the value.\n\nThe callback can be triggered either by a fixed number of time steps (interval) or by a fixed interval of simulation time (dt).\n\nKeywords\n\nfuncs...: Functions to be executed at specified intervals during the simulation. Each function must have the arguments (v, u, t, system), and will be called for every system, where v and u are the wrapped solution arrays for the corresponding system and t is the current simulation time. Note that working with these v and u arrays requires undocumented internal functions of TrixiParticles. See Custom Quantities for a list of pre-defined functions that can be used here.\ninterval=0: Specifies the number of time steps between each invocation of the callback. If set to 0, the callback will not be triggered based on time steps. Either interval or dt must be set to something larger than 0.\ndt=0.0: Specifies the simulation time interval between each invocation of the callback. If set to 0.0, the callback will not be triggered based on simulation time. Either interval or dt must be set to something larger than 0.\nexclude_boundary=true: If set to true, boundary particles will be excluded from the post-processing.\nfilename=\"values\": The filename of the postprocessing files to be saved.\noutput_directory=\"out\": The path where the results of the post-processing will be saved.\nwrite_csv=true: If set to true, write a csv file.\nwrite_json=true: If set to true, write a json file.\nappend_timestep=false: If set to true, the current timestamp will be added to the filename.\nwrite_file_interval=1: Files will be written after every write_file_interval number of postprocessing execution steps. A value of 0 indicates that files are only written at the end of the simulation, eliminating I/O overhead.\n\nExamples\n\n# Create a callback that is triggered every 100 time steps\npostprocess_callback = PostprocessCallback(interval=100, example_quantity=kinetic_energy)\n\n# Create a callback that is triggered every 0.1 simulation time units\npostprocess_callback = PostprocessCallback(dt=0.1, example_quantity=kinetic_energy)\n\n\n\n\n\n","category":"type"},{"location":"callbacks/#TrixiParticles.SolutionSavingCallback","page":"Callbacks","title":"TrixiParticles.SolutionSavingCallback","text":"SolutionSavingCallback(; interval::Integer=0, dt=0.0, save_times=Array{Float64, 1}([]),\n save_initial_solution=true, save_final_solution=true,\n output_directory=\"out\", append_timestamp=false, prefix=\"\",\n verbose=false, write_meta_data=true, max_coordinates=2^15,\n custom_quantities...)\n\nCallback to save the current numerical solution in VTK format in regular intervals. Either pass interval to save every interval time steps, or pass dt to save in intervals of dt in terms of integration time by adding additional tstops (note that this may change the solution).\n\nAdditional user-defined quantities can be saved by passing functions as keyword arguments, which map (v, u, t, system) to an Array where the columns represent the particles in the same order as in u. To ignore a custom quantity for a specific system, return nothing.\n\nKeywords\n\ninterval=0: Save the solution every interval time steps.\ndt: Save the solution in regular intervals of dt in terms of integration time by adding additional tstops (note that this may change the solution).\nsave_times=[] List of times at which to save a solution.\nsave_initial_solution=true: Save the initial solution.\nsave_final_solution=true: Save the final solution.\noutput_directory=\"out\": Directory to save the VTK files.\nappend_timestamp=false: Append current timestamp to the output directory.\n'prefix=\"\"': Prefix added to the filename.\ncustom_quantities...: Additional user-defined quantities.\nwrite_meta_data=true: Write meta data.\nverbose=false: Print to standard IO when a file is written.\nmax_coordinates=2^15: The coordinates of particles will be clipped if their absolute values exceed this threshold.\ncustom_quantities...: Additional custom quantities to include in the VTK output. Each custom quantity must be a function of (v, u, t, system), which will be called for every system, where v and u are the wrapped solution arrays for the corresponding system and t is the current simulation time. Note that working with these v and u arrays requires undocumented internal functions of TrixiParticles. See Custom Quantities for a list of pre-defined custom quantities that can be used here.\n\nExamples\n\n# Save every 100 time steps\nsaving_callback = SolutionSavingCallback(interval=100)\n\n# Save in intervals of 0.1 in terms of simulation time\nsaving_callback = SolutionSavingCallback(dt=0.1)\n\n# Additionally store the kinetic energy of each system as \"my_custom_quantity\"\nsaving_callback = SolutionSavingCallback(dt=0.1, my_custom_quantity=kinetic_energy)\n\n\n\n\n\n","category":"type"},{"location":"callbacks/#TrixiParticles.SteadyStateReachedCallback","page":"Callbacks","title":"TrixiParticles.SteadyStateReachedCallback","text":"SteadyStateReachedCallback(; interval::Integer=0, dt=0.0,\n interval_size::Integer=10, abstol=1.0e-8, reltol=1.0e-6)\n\nTerminates the integration when the change of kinetic energy between time steps falls below the threshold specified by abstol + reltol * ekin, where ekin is the total kinetic energy of the simulation.\n\nKeywords\n\ninterval=0: Check steady state condition every interval time steps.\ndt=0.0: Check steady state condition in regular intervals of dt in terms of integration time by adding additional tstops (note that this may change the solution).\ninterval_size: The interval in which the change of the kinetic energy is considered. interval_size is a (integer) multiple of interval or dt.\nabstol: Absolute tolerance.\nreltol: Relative tolerance.\n\n\n\n\n\n","category":"type"},{"location":"callbacks/#TrixiParticles.StepsizeCallback-Tuple{}","page":"Callbacks","title":"TrixiParticles.StepsizeCallback","text":"StepsizeCallback(; cfl::Real)\n\nSet the time step size according to a CFL condition if the time integration method isn't adaptive itself.\n\nThe current implementation is using the simplest form of CFL condition, which chooses a time step size that is constant during the simulation. The step size is therefore only applied once at the beginning of the simulation.\n\nThe step size Delta t is chosen as the minimum\n\n Delta t = min(Delta t_eta Delta t_a Delta t_c)\n\nwhere\n\n Delta t_eta = 0125 h^2 eta quad Delta t_a = 025 sqrth lVert g rVert\n quad Delta t_c = textCFL h c\n\nwith nu = alpha h c (2n + 4), where alpha is the parameter of the viscosity and n is the number of dimensions.\n\nwarning: Experimental implementation\nThis is an experimental feature and may change in future releases.\n\nReferences\n\n[21], [14], [43], [44]\n\n\n\n\n\n","category":"method"},{"location":"callbacks/#TrixiParticles.UpdateCallback-Tuple{}","page":"Callbacks","title":"TrixiParticles.UpdateCallback","text":"UpdateCallback(; interval::Integer, dt=0.0)\n\nCallback to update quantities either at the end of every interval time steps or in intervals of dt in terms of integration time by adding additional tstops (note that this may change the solution).\n\nKeywords\n\ninterval=1: Update quantities at the end of every interval time steps.\ndt: Update quantities in regular intervals of dt in terms of integration time by adding additional tstops (note that this may change the solution).\n\n\n\n\n\n","category":"method"},{"location":"callbacks/#custom_quantities","page":"Callbacks","title":"Custom Quantities","text":"","category":"section"},{"location":"callbacks/","page":"Callbacks","title":"Callbacks","text":"The following pre-defined custom quantities can be used with the SolutionSavingCallback and PostprocessCallback.","category":"page"},{"location":"callbacks/","page":"Callbacks","title":"Callbacks","text":"Modules = [TrixiParticles]\nPages = [\"general/custom_quantities.jl\"]","category":"page"},{"location":"callbacks/#TrixiParticles.avg_density-Tuple{Any, Any, Any, TrixiParticles.FluidSystem}","page":"Callbacks","title":"TrixiParticles.avg_density","text":"avg_density\n\nReturns the average_density over all particles in a system.\n\n\n\n\n\n","category":"method"},{"location":"callbacks/#TrixiParticles.avg_pressure-Tuple{Any, Any, Any, TrixiParticles.FluidSystem}","page":"Callbacks","title":"TrixiParticles.avg_pressure","text":"avg_pressure\n\nReturns the average pressure over all particles in a system.\n\n\n\n\n\n","category":"method"},{"location":"callbacks/#TrixiParticles.kinetic_energy-NTuple{4, Any}","page":"Callbacks","title":"TrixiParticles.kinetic_energy","text":"kinetic_energy\n\nReturns the total kinetic energy of all particles in a system.\n\n\n\n\n\n","category":"method"},{"location":"callbacks/#TrixiParticles.max_density-Tuple{Any, Any, Any, TrixiParticles.FluidSystem}","page":"Callbacks","title":"TrixiParticles.max_density","text":"max_density\n\nReturns the maximum density over all particles in a system.\n\n\n\n\n\n","category":"method"},{"location":"callbacks/#TrixiParticles.max_pressure-Tuple{Any, Any, Any, TrixiParticles.FluidSystem}","page":"Callbacks","title":"TrixiParticles.max_pressure","text":"max_pressure\n\nReturns the maximum pressure over all particles in a system.\n\n\n\n\n\n","category":"method"},{"location":"callbacks/#TrixiParticles.min_density-Tuple{Any, Any, Any, TrixiParticles.FluidSystem}","page":"Callbacks","title":"TrixiParticles.min_density","text":"min_density\n\nReturns the minimum density over all particles in a system.\n\n\n\n\n\n","category":"method"},{"location":"callbacks/#TrixiParticles.min_pressure-Tuple{Any, Any, Any, TrixiParticles.FluidSystem}","page":"Callbacks","title":"TrixiParticles.min_pressure","text":"min_pressure\n\nReturns the minimum pressure over all particles in a system.\n\n\n\n\n\n","category":"method"},{"location":"callbacks/#TrixiParticles.total_mass-NTuple{4, Any}","page":"Callbacks","title":"TrixiParticles.total_mass","text":"total_mass\n\nReturns the total mass of all particles in a system.\n\n\n\n\n\n","category":"method"},{"location":"gpu/#GPU-Support","page":"GPU Support","title":"GPU Support","text":"","category":"section"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"GPU support is still an experimental feature that is actively being worked on. As of now, the WeaklyCompressibleSPHSystem and the BoundarySPHSystem are supported on GPUs. We have tested this on GPUs by Nvidia and AMD.","category":"page"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"To run a simulation on a GPU, we need to use the FullGridCellList as cell list for the GridNeighborhoodSearch. This cell list requires a bounding box for the domain, unlike the default cell list, which uses an unbounded domain. For simulations that are bounded by a closed tank, we can use the boundary of the tank to obtain the bounding box as follows.","category":"page"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"search_radius = TrixiParticles.compact_support(smoothing_kernel, smoothing_length)\nmin_corner = minimum(tank.boundary.coordinates, dims=2) .- search_radius\nmax_corner = maximum(tank.boundary.coordinates, dims=2) .+ search_radius\ncell_list = TrixiParticles.PointNeighbors.FullGridCellList(; min_corner, max_corner)","category":"page"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"We then need to pass this cell list to the neighborhood search and the neighborhood search to the Semidiscretization.","category":"page"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"semi = Semidiscretization(fluid_system, boundary_system,\n neighborhood_search=GridNeighborhoodSearch{2}(; cell_list))","category":"page"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"At this point, we should run the simulation and make sure that it still works and that the bounding box is large enough. For some simulations where particles move outside the initial tank coordinates, for example when the tank is not closed or when the tank is moving, an appropriate bounding box has to be specified.","category":"page"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"Then, we only need to specify the data type that is used for the simulation. On an Nvidia GPU, we specify:","category":"page"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"using CUDA\node = semidiscretize(semi, tspan, data_type=CuArray)","category":"page"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"On an AMD GPU, we use:","category":"page"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"using AMDGPU\node = semidiscretize(semi, tspan, data_type=ROCArray)","category":"page"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"Then, we can run the simulation as usual. All data is transferred to the GPU during initialization and all loops over particles and their neighbors will be executed on the GPU as kernels generated by KernelAbstractions.jl. Data is only copied to the CPU for saving VTK files via the SolutionSavingCallback.","category":"page"},{"location":"general/interpolation/#Interpolation","page":"Interpolation","title":"Interpolation","text":"","category":"section"},{"location":"general/interpolation/","page":"Interpolation","title":"Interpolation","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"general\", \"interpolation.jl\")]","category":"page"},{"location":"general/interpolation/#TrixiParticles.interpolate_line-Tuple{Any, Any, Any, Any, Any, ODESolution}","page":"Interpolation","title":"TrixiParticles.interpolate_line","text":"interpolate_line(start, end_, n_points, semi, ref_system, sol; endpoint=true,\n smoothing_length=ref_system.smoothing_length, cut_off_bnd=true,\n clip_negative_pressure=false)\n\nInterpolates properties along a line in a TrixiParticles simulation. The line interpolation is accomplished by generating a series of evenly spaced points between start and end_. If endpoint is false, the line is interpolated between the start and end points, but does not include these points.\n\nSee also: interpolate_point, interpolate_plane_2d, interpolate_plane_2d_vtk, interpolate_plane_3d.\n\nArguments\n\nstart: The starting point of the line.\nend_: The ending point of the line.\nn_points: The number of points to interpolate along the line.\nsemi: The semidiscretization used for the simulation.\nref_system: The reference system for the interpolation.\nsol: The solution state from which the properties are interpolated.\n\nKeywords\n\nendpoint=true: A boolean to include (true) or exclude (false) the end point in the interpolation.\nsmoothing_length=ref_system.smoothing_length: The smoothing length used in the interpolation.\ncut_off_bnd=true: Boolean to indicate if quantities should be set to NaN when the point is \"closer\" to the boundary than to the fluid in a kernel-weighted sense. Or, in more detail, when the boundary has more influence than the fluid on the density summation in this point, i.e., when the boundary particles add more kernel-weighted mass than the fluid particles.\nclip_negative_pressure=false: One common approach in SPH models is to clip negative pressure values, but this is unphysical. Instead we clip here during interpolation thus only impacting the local interpolated value.\n\nReturns\n\nA NamedTuple of arrays containing interpolated properties at each point along the line.\n\nnote: Note\nThis function is particularly useful for analyzing gradients or creating visualizations along a specified line in the SPH simulation domain.\nThe interpolation accuracy is subject to the density of particles and the chosen smoothing length.\nWith cut_off_bnd, a density-based estimation of the surface is used which is not as accurate as a real surface reconstruction.\n\nExamples\n\n# Interpolating along a line from [1.0, 0.0] to [1.0, 1.0] with 5 points\nresults = interpolate_line([1.0, 0.0], [1.0, 1.0], 5, semi, ref_system, sol)\n\n\n\n\n\n","category":"method"},{"location":"general/interpolation/#TrixiParticles.interpolate_plane_2d-Tuple{Any, Any, Any, Any, Any, ODESolution}","page":"Interpolation","title":"TrixiParticles.interpolate_plane_2d","text":"interpolate_plane_2d(min_corner, max_corner, resolution, semi, ref_system, sol;\n smoothing_length=ref_system.smoothing_length, cut_off_bnd=true,\n clip_negative_pressure=false)\n\nInterpolates properties along a plane in a TrixiParticles simulation. The region for interpolation is defined by its lower left and top right corners, with a specified resolution determining the density of the interpolation points.\n\nThe function generates a grid of points within the defined region, spaced uniformly according to the given resolution.\n\nSee also: interpolate_plane_2d_vtk, interpolate_plane_3d, interpolate_line, interpolate_point.\n\nArguments\n\nmin_corner: The lower left corner of the interpolation region.\nmax_corner: The top right corner of the interpolation region.\nresolution: The distance between adjacent interpolation points in the grid.\nsemi: The semidiscretization used for the simulation.\nref_system: The reference system for the interpolation.\nsol: The solution state from which the properties are interpolated.\n\nKeywords\n\nsmoothing_length=ref_system.smoothing_length: The smoothing length used in the interpolation.\ncut_off_bnd=true: Boolean to indicate if quantities should be set to NaN when the point is \"closer\" to the boundary than to the fluid in a kernel-weighted sense. Or, in more detail, when the boundary has more influence than the fluid on the density summation in this point, i.e., when the boundary particles add more kernel-weighted mass than the fluid particles.\nclip_negative_pressure=false: One common approach in SPH models is to clip negative pressure values, but this is unphysical. Instead we clip here during interpolation thus only impacting the local interpolated value.\n\nReturns\n\nA NamedTuple of arrays containing interpolated properties at each point within the plane.\n\nnote: Note\nThe interpolation accuracy is subject to the density of particles and the chosen smoothing length.\nWith cut_off_bnd, a density-based estimation of the surface is used, which is not as accurate as a real surface reconstruction.\n\nExamples\n\n# Interpolating across a plane from [0.0, 0.0] to [1.0, 1.0] with a resolution of 0.2\nresults = interpolate_plane_2d([0.0, 0.0], [1.0, 1.0], 0.2, semi, ref_system, sol)\n\n\n\n\n\n","category":"method"},{"location":"general/interpolation/#TrixiParticles.interpolate_plane_2d_vtk-Tuple{Any, Any, Any, Any, Any, ODESolution}","page":"Interpolation","title":"TrixiParticles.interpolate_plane_2d_vtk","text":"interpolate_plane_2d_vtk(min_corner, max_corner, resolution, semi, ref_system, sol;\n smoothing_length=ref_system.smoothing_length, cut_off_bnd=true,\n clip_negative_pressure=false, output_directory=\"out\", filename=\"plane\")\n\nInterpolates properties along a plane in a TrixiParticles simulation and exports the result as a VTI file. The region for interpolation is defined by its lower left and top right corners, with a specified resolution determining the density of the interpolation points.\n\nThe function generates a grid of points within the defined region, spaced uniformly according to the given resolution.\n\nSee also: interpolate_plane_2d, interpolate_plane_3d, interpolate_line, interpolate_point.\n\nArguments\n\nmin_corner: The lower left corner of the interpolation region.\nmax_corner: The top right corner of the interpolation region.\nresolution: The distance between adjacent interpolation points in the grid.\nsemi: The semidiscretization used for the simulation.\nref_system: The reference system for the interpolation.\nsol: The solution state from which the properties are interpolated.\n\nKeywords\n\nsmoothing_length=ref_system.smoothing_length: The smoothing length used in the interpolation.\noutput_directory=\"out\": Directory to save the VTI file.\nfilename=\"plane\": Name of the VTI file.\ncut_off_bnd=true: Boolean to indicate if quantities should be set to NaN when the point is \"closer\" to the boundary than to the fluid in a kernel-weighted sense. Or, in more detail, when the boundary has more influence than the fluid on the density summation in this point, i.e., when the boundary particles add more kernel-weighted mass than the fluid particles.\nclip_negative_pressure=false: One common approach in SPH models is to clip negative pressure values, but this is unphysical. Instead we clip here during interpolation thus only impacting the local interpolated value.\n\nnote: Note\nThe interpolation accuracy is subject to the density of particles and the chosen smoothing length.\nWith cut_off_bnd, a density-based estimation of the surface is used, which is not as accurate as a real surface reconstruction.\n\nExamples\n\n# Interpolating across a plane from [0.0, 0.0] to [1.0, 1.0] with a resolution of 0.2\nresults = interpolate_plane_2d([0.0, 0.0], [1.0, 1.0], 0.2, semi, ref_system, sol)\n\n\n\n\n\n","category":"method"},{"location":"general/interpolation/#TrixiParticles.interpolate_plane_3d-Tuple{Any, Any, Any, Any, Any, Any, ODESolution}","page":"Interpolation","title":"TrixiParticles.interpolate_plane_3d","text":"interpolate_plane_3d(point1, point2, point3, resolution, semi, ref_system, sol;\n smoothing_length=ref_system.smoothing_length, cut_off_bnd=true,\n clip_negative_pressure=false)\n\nInterpolates properties along a plane in a 3D space in a TrixiParticles simulation. The plane for interpolation is defined by three points in 3D space, with a specified resolution determining the density of the interpolation points.\n\nThe function generates a grid of points on a parallelogram within the plane defined by the three points, spaced uniformly according to the given resolution.\n\nSee also: interpolate_plane_2d, interpolate_plane_2d_vtk, interpolate_line, interpolate_point.\n\nArguments\n\npoint1: The first point defining the plane.\npoint2: The second point defining the plane.\npoint3: The third point defining the plane. The points must not be collinear.\nresolution: The distance between adjacent interpolation points in the grid.\nsemi: The semidiscretization used for the simulation.\nref_system: The reference system for the interpolation.\nsol: The solution state from which the properties are interpolated.\n\nKeywords\n\nsmoothing_length=ref_system.smoothing_length: The smoothing length used in the interpolation.\ncut_off_bnd=true: Boolean to indicate if quantities should be set to NaN when the point is \"closer\" to the boundary than to the fluid in a kernel-weighted sense. Or, in more detail, when the boundary has more influence than the fluid on the density summation in this point, i.e., when the boundary particles add more kernel-weighted mass than the fluid particles.\nclip_negative_pressure=false: One common approach in SPH models is to clip negative pressure values, but this is unphysical. Instead we clip here during interpolation thus only impacting the local interpolated value.\n\nReturns\n\nA NamedTuple of arrays containing interpolated properties at each point within the plane.\n\nnote: Note\nThe interpolation accuracy is subject to the density of particles and the chosen smoothing length.\nWith cut_off_bnd, a density-based estimation of the surface is used which is not as accurate as a real surface reconstruction.\n\nExamples\n\n# Interpolating across a plane defined by points [0.0, 0.0, 0.0], [1.0, 0.0, 0.0], and [0.0, 1.0, 0.0]\n# with a resolution of 0.1\nresults = interpolate_plane_3d([0.0, 0.0, 0.0], [1.0, 0.0, 0.0], [0.0, 1.0, 0.0], 0.1, semi, ref_system, sol)\n\n\n\n\n\n","category":"method"},{"location":"general/interpolation/#TrixiParticles.interpolate_point-Tuple{Any, Any, Any, ODESolution}","page":"Interpolation","title":"TrixiParticles.interpolate_point","text":"interpolate_point(points_coords::Array{Array{Float64,1},1}, semi, ref_system, sol;\n smoothing_length=ref_system.smoothing_length, cut_off_bnd=true,\n clip_negative_pressure=false)\n\ninterpolate_point(point_coords, semi, ref_system, sol;\n smoothing_length=ref_system.smoothing_length, cut_off_bnd=true,\n clip_negative_pressure=false)\n\nPerforms interpolation of properties at specified points or an array of points in a TrixiParticles simulation.\n\nWhen given an array of points (points_coords), it iterates over each point and applies interpolation individually. For a single point (point_coords), it performs the interpolation at that specific location. The interpolation utilizes the same kernel function of the SPH simulation to weigh contributions from nearby particles.\n\nSee also: interpolate_line, interpolate_plane_2d, interpolate_plane_2d_vtk, interpolate_plane_3d, .\n\nArguments\n\npoints_coords: An array of point coordinates, for which to interpolate properties.\npoint_coords: The coordinates of a single point for interpolation.\nsemi: The semidiscretization used in the SPH simulation.\nref_system: The reference system defining the properties of the SPH particles.\nsol: The current solution state from which properties are interpolated.\n\nKeywords\n\nsmoothing_length=ref_system.smoothing_length: The smoothing length used in the interpolation.\ncut_off_bnd=true: Boolean to indicate if quantities should be set to NaN when the point is \"closer\" to the boundary than to the fluid in a kernel-weighted sense. Or, in more detail, when the boundary has more influence than the fluid on the density summation in this point, i.e., when the boundary particles add more kernel-weighted mass than the fluid particles.\nclip_negative_pressure=false: One common approach in SPH models is to clip negative pressure values, but this is unphysical. Instead we clip here during interpolation thus only impacting the local interpolated value.\n\nReturns\n\nFor multiple points: A NamedTuple of arrays containing interpolated properties at each point.\nFor a single point: A NamedTuple of interpolated properties at the point.\n\nExamples\n\n# For a single point\nresult = interpolate_point([1.0, 0.5], semi, ref_system, sol)\n\n# For multiple points\npoints = [[1.0, 0.5], [1.0, 0.6], [1.0, 0.7]]\nresults = interpolate_point(points, semi, ref_system, sol)\n\nnote: Note\nThis function is particularly useful for analyzing gradients or creating visualizations along a specified line in the SPH simulation domain.\nThe interpolation accuracy is subject to the density of particles and the chosen smoothing length.\nWith cut_off_bnd, a density-based estimation of the surface is used which is not asaccurate as a real surface reconstruction.\n\n\n\n\n\n","category":"method"},{"location":"#TrixiParticles.jl","page":"Home","title":"TrixiParticles.jl","text":"","category":"section"},{"location":"","page":"Home","title":"Home","text":"TrixiParticles.jl is a high-performance particle simulation framework designed to overcome challenges of particle-based numerical methods in multiphysics applications. Existing frameworks often lack user-friendliness, involve complex configuration, and are not easily extensible for development of new methods. In the future we also want to provide seamless scalability from CPU to Exascale-level computing with GPU support. TrixiParticles.jl addresses these limitations with an intuitive interface, straightforward configuration, and an extensible design, facilitating efficient simulation setup and execution.","category":"page"},{"location":"","page":"Home","title":"Home","text":"TrixiParticles.jl focuses on the following use cases:","category":"page"},{"location":"","page":"Home","title":"Home","text":"Development of new particle-based methods and models. By providing an extensible architecture to incorporate additional particle methods easily and not focusing on a single model or numerical method.\nAccurate, reliable and efficient physics-based modelling of complex multiphysics problems by providing a flexible configuration system, tools, high performance and a wide range of validation and test cases.\nEasy setup of accessible simulations for educational purposes, including student projects, coursework, and thesis work through extensive documentation, community engagement and readable configuration files.","category":"page"},{"location":"","page":"Home","title":"Home","text":"Its features include:","category":"page"},{"location":"#Features","page":"Home","title":"Features","text":"","category":"section"},{"location":"","page":"Home","title":"Home","text":"Incompressible Navier-Stokes\nMethods: Weakly Compressible Smoothed Particle Hydrodynamics (WCSPH), Entropically Damped Artificial Compressibility (EDAC)\nModels: Surface Tension\nSolid-body mechanics\nMethods: Total Lagrangian SPH (TLSPH)\nFluid-Structure Interaction\nOutput formats:\nVTK","category":"page"},{"location":"#Examples","page":"Home","title":"Examples","text":"","category":"section"},{"location":"","page":"Home","title":"Home","text":"\n \n \n \n \n \n \n \n \n
\n
2D Dam Break
\n
\n
Moving Wall
\n
\n
Oscillating Beam
\n
\n
Dam Break with Elastic Plate
\n
","category":"page"},{"location":"#Quickstart","page":"Home","title":"Quickstart","text":"","category":"section"},{"location":"","page":"Home","title":"Home","text":"Installation\nGetting started","category":"page"},{"location":"","page":"Home","title":"Home","text":"If you have any questions concerning TrixiParticles.jl you can join our community on Slack or open an issue with your question.","category":"page"},{"location":"#Start-with-development","page":"Home","title":"Start with development","text":"","category":"section"},{"location":"","page":"Home","title":"Home","text":"To get started with development have a look at these pages:","category":"page"},{"location":"","page":"Home","title":"Home","text":"Installation\nDevelopment\nContributing","category":"page"},{"location":"tutorial/#Tutorial","page":"Tutorial","title":"Tutorial","text":"","category":"section"},{"location":"tutorial/#Fluid","page":"Tutorial","title":"Fluid","text":"","category":"section"},{"location":"tutorial/","page":"Tutorial","title":"Tutorial","text":"Setting up your simulation from scratch\nSetting up a dam break simulation","category":"page"},{"location":"tutorial/#Mechanics","page":"Tutorial","title":"Mechanics","text":"","category":"section"},{"location":"tutorial/","page":"Tutorial","title":"Tutorial","text":"Deforming a beam","category":"page"},{"location":"tutorial/#Fluid-Structure-Interaction","page":"Tutorial","title":"Fluid-Structure Interaction","text":"","category":"section"},{"location":"tutorial/","page":"Tutorial","title":"Tutorial","text":"Setting up a falling structure","category":"page"},{"location":"reference-pointneighbors/#pointneighbors","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.jl API","text":"","category":"section"},{"location":"reference-pointneighbors/","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.jl API Reference","text":"CurrentModule = PointNeighbors","category":"page"},{"location":"reference-pointneighbors/","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.jl API Reference","text":"Modules = [PointNeighbors]","category":"page"},{"location":"reference-pointneighbors/#PointNeighbors.DictionaryCellList","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.DictionaryCellList","text":"DictionaryCellList{NDIMS}()\n\nA simple cell list implementation where a cell index (i, j) or (i, j, k) is mapped to a Vector{Int} by a Dict. By using a dictionary, which only stores non-empty cells, the domain is potentially infinite.\n\nThis implementation is very simple, but it neither uses an optimized hash function for integer tuples, nor does it use a contiguous memory layout. Consequently, this cell list is not GPU-compatible.\n\nArguments\n\nNDIMS: Number of dimensions.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.FullGridCellList","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.FullGridCellList","text":"FullGridCellList(; min_corner, max_corner, search_radius = 0.0,\n periodicity = false, backend = DynamicVectorOfVectors{Int32},\n max_points_per_cell = 100)\n\nA simple cell list implementation where each (empty or non-empty) cell of a rectangular (axis-aligned) domain is assigned a list of points. This cell list only works when all points are inside the specified domain at all times.\n\nOnly set min_corner and max_corner and use the default values for the other arguments to create an empty \"template\" cell list that can be used to create an empty \"template\" neighborhood search. See copy_neighborhood_search for more details.\n\nKeywords\n\nmin_corner: Coordinates of the domain corner in negative coordinate directions.\nmax_corner: Coordinates of the domain corner in positive coordinate directions.\nsearch_radius = 0.0: Search radius of the neighborhood search, which will determine the cell size. Use the default of 0.0 to create a template (see above).\nperiodicity = false: Set to true when using a PeriodicBox with the neighborhood search. When using copy_neighborhood_search, this option can be ignored an will be set automatically depending on the periodicity of the neighborhood search.\nbackend = DynamicVectorOfVectors{Int32}: Type of the data structure to store the actual cell lists. Can be\nVector{Vector{Int32}}: Scattered memory, but very memory-efficient.\nDynamicVectorOfVectors{Int32}: Contiguous memory, optimizing cache-hits.\nmax_points_per_cell = 100: Maximum number of points per cell. This will be used to allocate the DynamicVectorOfVectors. It is not used with the Vector{Vector{Int32}} backend.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.GridNeighborhoodSearch","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.GridNeighborhoodSearch","text":"GridNeighborhoodSearch{NDIMS}(; search_radius = 0.0, n_points = 0,\n periodic_box = nothing,\n cell_list = DictionaryCellList{NDIMS}(),\n update_strategy = nothing)\n\nSimple grid-based neighborhood search with uniform search radius. The domain is divided into a regular grid. For each (non-empty) grid cell, a list of points in this cell is stored. Instead of representing a finite domain by an array of cells, a potentially infinite domain is represented by storing cell lists in a hash table (using Julia's Dict data structure), indexed by the cell index tuple\n\nleft( leftlfloor fracxd rightrfloor leftlfloor fracyd rightrfloor right) quad textor quad\nleft( leftlfloor fracxd rightrfloor leftlfloor fracyd rightrfloor leftlfloor fraczd rightrfloor right)\n\nwhere x y z are the space coordinates and d is the search radius.\n\nTo find points within the search radius around a position, only points in the neighboring cells are considered.\n\nSee also (Chalela et al., 2021), (Ihmsen et al. 2011, Section 4.4).\n\nAs opposed to (Ihmsen et al. 2011), we do not sort the points in any way, since not sorting makes our implementation a lot faster (although less parallelizable).\n\nArguments\n\nNDIMS: Number of dimensions.\n\nKeywords\n\nsearch_radius = 0.0: The fixed search radius. The default of 0.0 is useful together with copy_neighborhood_search.\nn_points = 0: Total number of points. The default of 0 is useful together with copy_neighborhood_search.\nperiodic_box = nothing: In order to use a (rectangular) periodic domain, pass a PeriodicBox.\ncell_list: The cell list that maps a cell index to a list of points inside the cell. By default, a DictionaryCellList is used.\nupdate_strategy = nothing: Strategy to parallelize update!. Available options are:\nnothing: Automatically choose the best available option.\nParallelUpdate(): This is not available for all cell list implementations, but is the default when available.\nSemiParallelUpdate(): This is available for all cell list implementations and is the default when ParallelUpdate is not available.\nSerialUpdate()\n\nReferences\n\nM. Chalela, E. Sillero, L. Pereyra, M.A. Garcia, J.B. Cabral, M. Lares, M. Merchán. \"GriSPy: A Python package for fixed-radius nearest neighbors search\". In: Astronomy and Computing 34 (2021). doi: 10.1016/j.ascom.2020.100443\nMarkus Ihmsen, Nadir Akinci, Markus Becker, Matthias Teschner. \"A Parallel SPH Implementation on Multi-Core CPUs\". In: Computer Graphics Forum 30.1 (2011), pages 99–112. doi: 10.1111/J.1467-8659.2010.01832.X\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.ParallelUpdate","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.ParallelUpdate","text":"ParallelUpdate()\n\nFully parallel update by using atomic operations to avoid race conditions when adding points into the same cell. This is not available for all cell list implementations, but is the default when available.\n\nSee GridNeighborhoodSearch for usage information.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.PeriodicBox","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.PeriodicBox","text":"PeriodicBox(; min_corner, max_corner)\n\nDefine a rectangular (axis-aligned) periodic domain.\n\nKeywords\n\nmin_corner: Coordinates of the domain corner in negative coordinate directions.\nmax_corner: Coordinates of the domain corner in positive coordinate directions.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.PolyesterBackend","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.PolyesterBackend","text":"PolyesterBackend()\n\nPass as first argument to the @threaded macro to make the loop multithreaded with Polyester.@batch.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.PrecomputedNeighborhoodSearch","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.PrecomputedNeighborhoodSearch","text":"PrecomputedNeighborhoodSearch{NDIMS}(; search_radius = 0.0, n_points = 0,\n periodic_box = nothing, update_strategy = nothing)\n\nNeighborhood search with precomputed neighbor lists. A list of all neighbors is computed for each point during initialization and update. This neighborhood search maximizes the performance of neighbor loops at the cost of a much slower update!.\n\nA GridNeighborhoodSearch is used internally to compute the neighbor lists during initialization and update.\n\nArguments\n\nNDIMS: Number of dimensions.\n\nKeywords\n\nsearch_radius = 0.0: The fixed search radius. The default of 0.0 is useful together with copy_neighborhood_search.\nn_points = 0: Total number of points. The default of 0 is useful together with copy_neighborhood_search.\nperiodic_box = nothing: In order to use a (rectangular) periodic domain, pass a PeriodicBox.\nupdate_strategy: Strategy to parallelize update! of the internally used GridNeighborhoodSearch. See GridNeighborhoodSearch for available options.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.SemiParallelUpdate","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.SemiParallelUpdate","text":"SemiParallelUpdate()\n\nLoop over all cells in parallel to mark cells with points that now belong to a different cell. Then, move points of affected cells serially to avoid race conditions. This is available for all cell list implementations and is the default when ParallelUpdate is not available.\n\nSee GridNeighborhoodSearch for usage information.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.SerialUpdate","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.SerialUpdate","text":"SerialUpdate()\n\nDeactivate parallelization in the neighborhood search update. Parallel neighborhood search update can be one of the largest sources of error variations between simulations with different thread numbers due to neighbor ordering changes.\n\nSee GridNeighborhoodSearch for usage information.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.ThreadsDynamicBackend","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.ThreadsDynamicBackend","text":"ThreadsDynamicBackend()\n\nPass as first argument to the @threaded macro to make the loop multithreaded with Threads.@threads :dynamic.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.ThreadsStaticBackend","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.ThreadsStaticBackend","text":"ThreadsStaticBackend()\n\nPass as first argument to the @threaded macro to make the loop multithreaded with Threads.@threads :static.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.TrivialNeighborhoodSearch","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.TrivialNeighborhoodSearch","text":"TrivialNeighborhoodSearch{NDIMS}(; search_radius = 0.0, eachpoint = 1:0,\n periodic_box = nothing)\n\nTrivial neighborhood search that simply loops over all points.\n\nArguments\n\nNDIMS: Number of dimensions.\n\nKeywords\n\nsearch_radius = 0.0: The fixed search radius. The default of 0.0 is useful together with copy_neighborhood_search.\neachpoint = 1:0: Iterator for all point indices. Usually just 1:n_points. The default of 1:0 is useful together with copy_neighborhood_search.\nperiodic_box = nothing: In order to use a (rectangular) periodic domain, pass a PeriodicBox.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.copy_neighborhood_search-Tuple{PointNeighbors.AbstractNeighborhoodSearch, Any, Any}","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.copy_neighborhood_search","text":"copy_neighborhood_search(search::AbstractNeighborhoodSearch, search_radius, n_points;\n eachpoint = 1:n_points)\n\nCreate a new uninitialized neighborhood search of the same type and with the same configuration options as search, but with a different search radius and number of points.\n\nThe TrivialNeighborhoodSearch also requires an iterator eachpoint, which most of the time will be 1:n_points. If the TrivialNeighborhoodSearch is never going to be used, the keyword argument eachpoint can be ignored.\n\nThis is useful when a simulation code requires multiple neighborhood searches of the same kind. One can then just pass an empty neighborhood search as a template and use this function inside the simulation code to generate similar neighborhood searches with different search radii and different numbers of points.\n\n# Template\nnhs = GridNeighborhoodSearch{2}()\n\n# Inside the simulation code, generate similar neighborhood searches\nnhs1 = copy_neighborhood_search(nhs, 1.0, 100)\n\n# output\nGridNeighborhoodSearch{2, Float64, ...}(...)\n\n\n\n\n\n","category":"method"},{"location":"reference-pointneighbors/#PointNeighbors.foreach_point_neighbor-Union{Tuple{T}, Tuple{T, Any, Any, Any}} where T","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.foreach_point_neighbor","text":"foreach_point_neighbor(f, system_coords, neighbor_coords, neighborhood_search;\n points = axes(system_coords, 2), parallel = true)\n\nLoop for each point in system_coords over all points in neighbor_coords whose distances to that point are smaller than the search radius and execute the function f(i, j, x, y, d), where\n\ni is the column index of the point in system_coords,\nj the column index of the neighbor in neighbor_coords,\nx an SVector of the coordinates of the point (system_coords[:, i]),\ny an SVector of the coordinates of the neighbor (neighbor_coords[:, j]),\nd the distance between x and y.\n\nThe neighborhood_search must have been initialized or updated with system_coords as first coordinate array and neighbor_coords as second coordinate array.\n\nNote that system_coords and neighbor_coords can be identical.\n\nArguments\n\nf: The function explained above.\nsystem_coords: A matrix where the i-th column contains the coordinates of point i.\nneighbor_coords: A matrix where the j-th column contains the coordinates of point j.\nneighborhood_search: A neighborhood search initialized or updated with system_coords as first coordinate array and neighbor_coords as second coordinate array.\n\nKeywords\n\npoints: Loop over these point indices. By default all columns of system_coords.\nparallel=true: Run the outer loop over points thread-parallel.\n\nSee also initialize!, update!.\n\n\n\n\n\n","category":"method"},{"location":"reference-pointneighbors/#PointNeighbors.initialize!-Tuple{PointNeighbors.AbstractNeighborhoodSearch, Any, Any}","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.initialize!","text":"initialize!(search::AbstractNeighborhoodSearch, x, y)\n\nInitialize a neighborhood search with the two coordinate arrays x and y.\n\nIn general, the purpose of a neighborhood search is to find for one point in x all points in y whose distances to that point are smaller than the search radius. x and y are expected to be matrices, where the i-th column contains the coordinates of point i. Note that x and y can be identical.\n\nSee also update!.\n\n\n\n\n\n","category":"method"},{"location":"reference-pointneighbors/#PointNeighbors.update!-Tuple{PointNeighbors.AbstractNeighborhoodSearch, Any, Any}","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.update!","text":"update!(search::AbstractNeighborhoodSearch, x, y; points_moving = (true, true))\n\nUpdate an already initialized neighborhood search with the two coordinate arrays x and y.\n\nLike initialize!, but reusing the existing data structures of the already initialized neighborhood search. When the points only moved a small distance since the last update! or initialize!, this is significantly faster than initialize!.\n\nNot all implementations support incremental updates. If incremental updates are not possible for an implementation, update! will fall back to a regular initialize!.\n\nSome neighborhood searches might not need to update when only x changed since the last update! or initialize! and y did not change. Pass points_moving = (true, false) in this case to avoid unnecessary updates. The first flag in points_moving indicates if points in x are moving. The second flag indicates if points in y are moving.\n\nwarning: Experimental Feature: Backend Specification\nThe keyword argument parallelization_backend allows users to specify the multithreading backend. This feature is currently considered experimental!Possible parallelization backends are:ThreadsDynamicBackend to use Threads.@threads :dynamic\nThreadsStaticBackend to use Threads.@threads :static\nPolyesterBackend to use Polyester.@batch\nKernelAbstractions.Backend to launch a GPU kernel\n\nSee also initialize!.\n\n\n\n\n\n","category":"method"},{"location":"reference-pointneighbors/#PointNeighbors.@threaded-Tuple{Any, Any}","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.@threaded","text":"@threaded x for ... end\n\nRun either a threaded CPU loop or launch a kernel on the GPU, depending on the type of x. Semantically the same as Threads.@threads when iterating over a AbstractUnitRange but without guarantee that the underlying implementation uses Threads.@threads or works for more general for loops.\n\nThe first argument must either be a parallelization backend (see below) or an array from which the backend can be derived to determine if the loop must be run threaded on the CPU or launched as a kernel on the GPU. Passing KernelAbstractions.CPU() will run the GPU kernel on the CPU.\n\nPossible parallelization backends are:\n\nPolyesterBackend to use Polyester.@batch\nThreadsDynamicBackend to use Threads.@threads :dynamic\nThreadsStaticBackend to use Threads.@threads :static\nKernelAbstractions.Backend to execute the loop as a GPU kernel\n\nIn particular, the underlying threading capabilities might be provided by other packages such as Polyester.jl.\n\nwarning: Warning\nThis macro does not necessarily work for general for loops. For example, it does not necessarily support general iterables such as eachline(filename).\n\n\n\n\n\n","category":"macro"}] +[{"location":"references/","page":"References","title":"References","text":"K. Hormann and A. Agathos. The point in polygon problem for arbitrary polygons. Computational Geometry 20, 131–144 (2001).\n\n\n\nA. Jacobson, L. Kavan and O. Sorkine-Hornung. Robust inside-outside segmentation using generalized winding numbers. ACM Transactions on Graphics 32, 1–12 (2013).\n\n\n\nM. Müller, D. Charypar and M. Gross. Particle-Based Fluid Simulation for Interactive Applications. In: Proceedings of the 2003 ACM SIGGRAPH/Eurographics Symposium on Computer Animation (Eurographics Association, 07 2003); pp. 154–159.\n\n\n\nI. J. Schoenberg. Contributions to the problem of approximation of equidistant data by analytic functions. Part B. On the problem of osculatory interpolation. A second class of analytic approximation formulae. Quarterly of Applied Mathematics 4, 112–141 (1946).\n\n\n\nD. J. Price. Smoothed particle hydrodynamics and magnetohydrodynamics. Journal of Computational Physics 231, 759–794 (2012).\n\n\n\nJ. Monaghan. Particle methods for hydrodynamics. Computer Physics Reports 3, 71–124 (1985).\n\n\n\nH. Wendland. Piecewise polynomial, positive definite and compactly supported radial functions of minimal degree. Advances in Computational Mathematics 4, 389–396 (1995).\n\n\n\nW. Dehnen and H. Aly. Improving convergence in smoothed particle hydrodynamics simulations without pairing instability: SPH without pairing instability. Monthly Notices of the Royal Astronomical Society 425, 1068–1082 (2012).\n\n\n\nN. Bićanić. Discrete Element Methods. In: Discrete Element Methods (Wiley, 2004).\n\n\n\nP. A. Cundall and O. D. Strack. A discrete numerical model for granular assemblies. Géotechnique 29, 47–65 (1979).\n\n\n\nA. Di Renzo and F. P. Di Maio. Comparison of contact-force models for the simulation of collisions in DEM-based granular flow codes. Chemical Engineering Science 59, 525–541 (2004).\n\n\n\nJ. Monaghan. Simulating Free Surface Flows with SPH. Journal of Computational Physics 110, 399–406 (1994).\n\n\n\nR. H. Cole and R. Weller. Underwater Explosions. Physics Today 1, 35–35 (1948).\n\n\n\nS. Adami, X. Hu and N. Adams. A generalized wall boundary condition for smoothed particle hydrodynamics. Journal of Computational Physics 231, 7057–7075 (2012).\n\n\n\nJ. P. Morris, P. J. Fox and Y. Zhu. Modeling Low Reynolds Number Incompressible Flows Using SPH. Journal of Computational Physics 136, 214–226 (1997).\n\n\n\nJ. J. Monaghan. Smoothed Particle Hydrodynamics. Annual Review of Astronomy and Astrophysics 30, 543–574 (1992).\n\n\n\nJ. Monaghan. On the problem of penetration in particle methods. Journal of Computational Physics 82, 1–15 (1989).\n\n\n\nJ. J. Monaghan. Smoothed particle hydrodynamics. Reports on Progress in Physics 68, 1703–1759 (2005).\n\n\n\nP. Ramachandran and K. Puri. Entropically damped artificial compressibility for SPH. Computers & Fluids 179, 579–594 (2019).\n\n\n\nG. Fourtakas, J. M. Dominguez, R. Vacondio and B. D. Rogers. Local uniform stencil (LUST) boundary condition for arbitrary 3-D boundaries in parallel smoothed particle hydrodynamics (SPH) models. Computers & Fluids 190, 346–361 (2019).\n\n\n\nM. Antuono, A. Colagrossi and S. Marrone. Numerical diffusive terms in weakly-compressible SPH schemes. Computer Physics Communications 183, 2570–2580 (2012).\n\n\n\nM. Antuono, A. Colagrossi, S. Marrone and D. Molteni. Free-surface flows solved by means of SPH schemes with numerical diffusive terms. Computer Physics Communications 181, 532–549 (2010).\n\n\n\nD. Molteni and A. Colagrossi. A simple procedure to improve the pressure evaluation in hydrodynamic context using the SPH. Computer Physics Communications 180, 861–872 (2009).\n\n\n\nA. Ferrari, M. Dumbser, E. F. Toro and A. Armanini. A new 3D parallel SPH scheme for free surface flows. Computers & Fluids 38, 1203–1217 (2009).\n\n\n\nN. Akinci, G. Akinci and M. Teschner. Versatile surface tension and adhesion for SPH fluids. ACM Transactions on Graphics 32, 1–8 (2013).\n\n\n\nJ. Bonet and T.-S. Lok. Variational and momentum preservation aspects of Smooth Particle Hydrodynamic formulations. Computer Methods in Applied Mechanics and Engineering 180, 97–115 (1999).\n\n\n\nM. Basa, N. J. Quinlan and M. Lastiwka. Robustness and accuracy of SPH formulations for viscous flow. International Journal for Numerical Methods in Fluids 60, 1127–1148 (2008).\n\n\n\nS. Li and W. K. Liu. Moving least-square reproducing kernel method Part II: Fourier analysis. Computer Methods in Applied Mechanics and Engineering 139, 159–193 (1996).\n\n\n\nJ. R. Clausen. Entropically damped form of artificial compressibility for explicit simulation of incompressible flow. Physical Review E 87, 013309 (2013).\n\n\n\nS. Adami, X. Hu and N. Adams. A transport-velocity formulation for smoothed particle hydrodynamics. Journal of Computational Physics 241, 292–307 (2013).\n\n\n\nJ. O’Connor and B. D. Rogers. A fluid–structure interaction model for free-surface flows and flexible structures using smoothed particle hydrodynamics on a GPU. Journal of Fluids and Structures 104, 103312 (2021).\n\n\n\nT. Belytschko, Y. Guo, W. K. Liu and S. P. Xiao. A unified stability analysis of meshless particle methods. International Journal for Numerical Methods in Engineering 48, 1359–1400 (2000).\n\n\n\nG. C. Ganzenmüller. An hourglass control algorithm for Lagrangian Smooth Particle Hydrodynamics. Computer Methods in Applied Mechanics and Engineering 286, 87–106 (2015).\n\n\n\nA. Valizadeh and J. J. Monaghan. A study of solid wall models for weakly compressible SPH. Journal of Computational Physics 300, 5–19 (2015).\n\n\n\nN. Akinci, M. Ihmsen, G. Akinci, B. Solenthaler and M. Teschner. Versatile rigid-fluid coupling for incompressible SPH. ACM Transactions on Graphics 31, 1–8 (2012).\n\n\n\nA. J. Crespo, M. Gomez-Gesteira and R. A. Dalrymple. Boundary Conditions Generated by Dynamic Particles in SPH Methods. Computers, Materials and Continua 5, 173–184 (2007).\n\n\n\nS. Band, C. Gissler, A. Peer and M. Teschner. MLS pressure boundaries for divergence-free and viscous SPH fluids. Computers & Graphics 76, 37–46 (2018).\n\n\n\nJ. Monaghan and J. Kajtar. SPH particle boundary forces for arbitrary boundaries. Computer Physics Communications 180, 1811–1820 (2009).\n\n\n\nM. B. Giles. Nonreflecting boundary conditions for Euler equation calculations. AIAA Journal 28, 2050–2058 (1990).\n\n\n\nM. Lastiwka, M. Basa and N. J. Quinlan. Permeable and non‐reflecting boundary conditions in SPH. International Journal for Numerical Methods in Fluids 61, 709–724 (2008).\n\n\n\nP. Negi, P. Ramachandran and A. Haftu. An improved non-reflecting outlet boundary condition for weakly-compressible SPH. Computer Methods in Applied Mechanics and Engineering 367, 113119 (2020).\n\n\n\nA. Panizzo, G. Cuomo and R. A. Dalrymple. 3D-SPH SIMULATION OF LANDSLIDE GENERATED WAVES. In: Coastal Engineering 2006 (World Scientific Publishing Company, Apr 2007).\n\n\n\nP. Sun, A. Colagrossi, S. Marrone and A. Zhang. Delta-SPH model: Simple procedures for a further improvement of the SPH scheme. Computer Methods in Applied Mechanics and Engineering 315, 25–49 (2017).\n\n\n\nM. Antuono, S. Marrone, A. Colagrossi and B. Bouscasse. Energy balance in the Delta-SPH scheme. Computer Methods in Applied Mechanics and Engineering 289, 209–226 (2015).\n\n\n\n","category":"page"},{"location":"preprocessing/preprocessing/#Sampling-of-Geometries","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"","category":"section"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Generating the initial configuration of a simulation requires filling volumes (3D) or surfaces (2D) of complex geometries with particles. The algorithm to sample a complex geometry should be robust and fast, since for large problems (large numbers of particles) or complex geometries (many geometry faces), generating the initial configuration is not trivial and can be very expensive in terms of computational cost. We therefore use a winding number approach for an inside-outside segmentation of an object. The winding number w(mathbfp) is a signed integer-valued function of a point mathbfp and is defined as","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"w(mathbfp) = frac12 pi sum^n_i=1 Theta_i","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Here, Theta_i is the signed angle between mathbfc_i - mathbfp and mathbfc_i+1 - mathbfp where mathbfc_i and mathbfc_i+1 are two consecutive vertices on a curve. In 3D, we refer to the solid angle of an oriented triangle with respect to mathbfp.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"We provide the following methods to calculate w(mathbfp):","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Hormann et al. (2001) evaluate the winding number combined with an even-odd rule, but only for 2D polygons (see WindingNumberHormann).\nNaive winding: Jacobson et al. (2013) generalized the winding number so that the algorithm can be applied for both 2D and 3D geometries (see WindingNumberJacobson).\nHierarchical winding: Jacobson et al. (2013) also introduced a fast hierarchical evaluation of the winding number. For further information see the description below.","category":"page"},{"location":"preprocessing/preprocessing/#hierarchical_winding","page":"Sampling of Geometries","title":"Hierarchical Winding","text":"","category":"section"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"According to Jacobson et al. (2013) the winding number with respect to a polygon (2D) or triangle mesh (3D) is the sum of the winding numbers with respect to each edge (2D) or face (3D). We can show this with the following example in which we determine the winding number for each edge of a triangle separately and sum them up:","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"using TrixiParticles\nusing Plots\n\ntriangle = [125.0 375.0 250.0 125.0;\n 175.0 175.0 350.0 175.0]\n\n# Delete all edges but one\nedge1 = deleteat!(TrixiParticles.Polygon(triangle), [2, 3])\nedge2 = deleteat!(TrixiParticles.Polygon(triangle), [1, 3])\nedge3 = deleteat!(TrixiParticles.Polygon(triangle), [1, 2])\n\nalgorithm = WindingNumberJacobson()\n\ngrid = hcat(([x, y] for x in 1:500, y in 1:500)...)\n\n_, w1 = algorithm(edge1, grid; store_winding_number=true)\n_, w2 = algorithm(edge2, grid; store_winding_number=true)\n_, w3 = algorithm(edge3, grid; store_winding_number=true)\n\nw = w1 + w2 + w3\n\nheatmap(1:500, 1:500, reshape(w1, 500, 500)', color=:coolwarm, showaxis=false,\n tickfontsize=12, size=(570, 500), margin=6 * Plots.mm)\nheatmap(1:500, 1:500, reshape(w2, 500, 500)', color=:coolwarm, showaxis=false,\n tickfontsize=12, size=(570, 500), margin=6 * Plots.mm)\nheatmap(1:500, 1:500, reshape(w3, 500, 500)', color=:coolwarm, showaxis=false,\n tickfontsize=12, size=(570, 500), margin=6 * Plots.mm)\nheatmap(1:500, 1:500, reshape(w, 500, 500)', color=:coolwarm, showaxis=false,\n tickfontsize=12, size=(570, 500), margin=6 * Plots.mm, clims=(-1, 1))\n","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"
\n \"triangle\"/\n
","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"This summation property has some interesting consequences that we can utilize for an efficient computation of the winding number. Let mathcalS be an open surface and barmathcalS an arbitrary closing surface, such that","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"partial barmathcalS = partial mathcalS","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"and mathcalB = barmathcalS cup mathcalS is some closed oriented surface. For any query point mathbfp outside of mathcalB, we know that","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"w_mathcalS(mathbfp) + w_barmathcalS(mathbfp) = w_mathcalB(mathbfp) = 0","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"This means","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"w_mathcalS(mathbfp) = - w_barmathcalS(mathbfp)","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"regardless of how barmathcalS is constructed (as long as mathbfp is outside of mathcalB).","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"We can use this property in the discrete case to efficiently compute the winding number of a query point by partitioning the polygon or mesh in a \"small\" part (as in consisting of a small number of edges/faces) and a \"large\" part. For the small part we just compute the winding number, and for the large part we construct a small closing and compute its winding number. The partitioning is based on a hierarchical construction of bounding boxes.","category":"page"},{"location":"preprocessing/preprocessing/#Bounding-volume-hierarchy","page":"Sampling of Geometries","title":"Bounding volume hierarchy","text":"","category":"section"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"To efficiently find a \"small part\" and a \"large part\" as mentioned above, we construct a hierarchy of bounding boxes by starting with the whole domain and recursively splitting it in two equally sized boxes. The resulting hierarchy is a binary tree.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"The algorithm by Jacobsen et al. (Algorithm 2, p. 5) traverses this binary tree recursively until we find the leaf in which the query point is located. The recursion stops with the following criteria:","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"if the bounding box T is a leaf then TmathcalS = mathcalS cap T, the part of mathcalS that lies inside T, is the \"small part\" mentioned above, so evaluate the winding number naively as w(mathbfp TmathcalS).\nelse if mathbfp is outside T then TmathcalS is the \"large part\", so evaluate the winding number naively as -w(mathbfp TbarmathcalS), where TbarmathcalS is the closing surface of TmathcalS.","category":"page"},{"location":"preprocessing/preprocessing/#Continuous-example","page":"Sampling of Geometries","title":"Continuous example","text":"","category":"section"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Now consider the following continuous (not discretized to a polygon) 2D example. We compute the winding number of the point mathbfp with respect to mathcalS using the depicted hierarchy of bounding boxes.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"
\n \"continuous\n
","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"(1):","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Recurse left: w_textleft = texttexttthierarchical_winding (mathbfp Ttextleft)\nRecurse right: w_textright = texttexttthierarchical_winding (mathbfpTtextright)","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"(2):","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Query point mathbfp is outside bounding box T, so don't recurse deeper.\nCompute w_mathcalS(mathbfp) = - w_barmathcalS(mathbfp) with the closure TbarmathcalS, which is generally much smaller (fewer edges in the discrete version) than TmathcalS:","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"w_textleft = -texttextttnaive_winding (mathbfp TbarmathcalS)","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"(3):","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Bounding box T is a leaf. Use open surface TmathcalS:","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"w_textright = texttextttnaive_winding (mathbfp TmathcalS)","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"The reconstructed surface will then look as in the following image.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"
\n \"reconstructed\n
","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"We finally sum up the winding numbers","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"w = w_textleft + w_textright = -w_T_textleftbarmathcalS + w_T_textrightmathcalS","category":"page"},{"location":"preprocessing/preprocessing/#Discrete-example","page":"Sampling of Geometries","title":"Discrete example","text":"","category":"section"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"We will now go through the discrete version of the example above.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"
\n \"discrete\n
","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"To construct the hierarchy for the discrete piecewise-linear example in (1), we have to do the following.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"(2): Each edge is distributed to the child whose box contains the edge's barycenter (red dots in (2)). Splitting stops when the number of a box's edges slips below a threshold (usually approx 100 faces in 3D, here: 6 edges).","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"(3): For the closure, Jacobson et al. (2013) define exterior vertices (exterior edges in 3D) as boundary vertices of such a segmentation (red dots in (3)). To find them, we traverse around each edge (face in 3D) in order, and increment or decrement for each vertex (edge) a specific counter.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"v1 = edge_vertices_ids[edge][1]\nv2 = edge_vertices_ids[edge][2]\n\nvertex_count[v1] += 1\nvertex_count[v2] -= 1","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"In 2D, a vertex is declared as exterior if vertex_count(vertex) != 0, so there is not the same amount of edges in this box going into versus out of the vertex. To construct the closing surface, the exterior vertices are then connected to one arbitrary exterior vertex using appropriately oriented line segments:","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"edge = vertex_count[v] > 0 ? (closing_vertex, v) : (v, closing_vertex)","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"The resulting closed surface TS cup TbarS then has the same number of edges going into and out of each vertex.","category":"page"},{"location":"preprocessing/preprocessing/#Incorrect-evaluation","page":"Sampling of Geometries","title":"Incorrect evaluation","text":"","category":"section"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"If we follow the algorithm, we know that recursion stops if","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"the bounding box T is a leaf or\nthe query point mathbfp is outside the box.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"
\n \"incorrect\n
","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"(1): The query point mathbfp is outside the box, so we calculate the winding number with the (red) closure of the box.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"(2): The query point mathbfp is inside the box, so we use the (blue) edges distributed to the box.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"(3): In this case, it leads to an incorrect evaluation of the winding number. The query point is clearly inside the box, but not inside the reconstructed surface. This is because the property w_mathcalS(mathbfp) = - w_barmathcalS(mathbfp) only holds when mathbfp is outside of mathcalB, which is not the case here.","category":"page"},{"location":"preprocessing/preprocessing/#Correct-evaluation","page":"Sampling of Geometries","title":"Correct evaluation","text":"","category":"section"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Jacobson et al. (2013) don't mention this problem or provide a solution to it. We contacted the authors and found that they know about this problem and solve it by resizing the bounding box to fully include the closing surface of the neighboring box, since it doesn't matter if the boxes overlap.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"
\n \"correct\n
","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"To avoid resizing, we take a different approach and calculate the closure of the bounding box differently:","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Exclude intersecting edges in the calculation of the exterior vertices.\nThis way, all exterior vertices are inside the bounding box, and so will be the closing surface.\nThe intersecting edges are later added with flipped orientation, so that the closing is actually a closing of the exterior plus intersecting edges.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"
\n \"correct\n
","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"The evaluation then looks as follows.","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"
\n \"correct\n
","category":"page"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"preprocessing\", \"point_in_poly\", \"winding_number_hormann.jl\")]","category":"page"},{"location":"preprocessing/preprocessing/#TrixiParticles.WindingNumberHormann","page":"Sampling of Geometries","title":"TrixiParticles.WindingNumberHormann","text":"WindingNumberHormann()\n\nAlgorithm for inside-outside segmentation of a complex geometry proposed by Hormann (2001). It is only supported for 2D geometries. WindingNumberHormann might handle edge cases a bit better, since the winding number is an integer value.\n\nwarning: Experimental Implementation\nThis is an experimental feature and may change in any future releases.\n\n\n\n\n\n","category":"type"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"preprocessing\", \"point_in_poly\", \"winding_number_jacobson.jl\")]","category":"page"},{"location":"preprocessing/preprocessing/#TrixiParticles.WindingNumberJacobson","page":"Sampling of Geometries","title":"TrixiParticles.WindingNumberJacobson","text":"WindingNumberJacobson(; geometry=nothing, winding_number_factor=sqrt(eps()),\n hierarchical_winding=false)\n\nAlgorithm for inside-outside segmentation of a complex geometry proposed by [2].\n\nKeywords\n\ngeometry: Complex geometry returned by load_geometry and is only required when using hierarchical_winding=true.\nhierarchical_winding: If set to true, an optimized hierarchical approach will be used, which gives a significant speedup. For further information see Hierarchical Winding.\nwinding_number_factor: For leaky geometries, a factor of 0.4 will give a better inside-outside segmentation.\n\nwarning: Experimental Implementation\nThis is an experimental feature and may change in any future releases.\n\n\n\n\n\n","category":"type"},{"location":"preprocessing/preprocessing/","page":"Sampling of Geometries","title":"Sampling of Geometries","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"preprocessing\", \"geometries\", \"io.jl\")]","category":"page"},{"location":"preprocessing/preprocessing/#TrixiParticles.load_geometry-Tuple{Any}","page":"Sampling of Geometries","title":"TrixiParticles.load_geometry","text":"load_geometry(filename; element_type=Float64)\n\nLoad file and return corresponding type for ComplexShape. Supported file formats are .stl and .asc.\n\nArguments\n\nfilename: Name of the file to be loaded.\n\nKeywords\n\nelement_type: Element type (default is Float64)\n\n\n\n\n\n","category":"method"},{"location":"systems/dem/#dem","page":"Discrete Element Method (Solid)","title":"Discrete Element Method","text":"","category":"section"},{"location":"systems/dem/","page":"Discrete Element Method (Solid)","title":"Discrete Element Method (Solid)","text":"The Discrete Element Method (DEM) is a computational technique widely used in physics, engineering, and applied mathematics for simulating the mechanical behavior of granular materials, such as powders, sand, soil, or rock, as well as other discontinua. Unlike continuum mechanics that treats materials as continuous, DEM considers individual particles or elements and their interactions. This approach provides detailed insights into the micro-mechanical behavior of materials, making it particularly valuable in fields such as geomechanics, material science, and mechanical engineering.","category":"page"},{"location":"systems/dem/#Fundamental-Principles","page":"Discrete Element Method (Solid)","title":"Fundamental Principles","text":"","category":"section"},{"location":"systems/dem/","page":"Discrete Element Method (Solid)","title":"Discrete Element Method (Solid)","text":"The core idea behind DEM is the discretization of a material system into a finite set of distinct, interacting mass elements (particles). These elements (particles) can vary in shape, size, and properties, and they interact with each other and possibly with their boundaries through contact forces and potential fields. The motion and behavior of each mass element are governed by Newton's laws of motion, accounting for the forces and moments acting upon them.","category":"page"},{"location":"systems/dem/","page":"Discrete Element Method (Solid)","title":"Discrete Element Method (Solid)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"solid\", \"discrete_element_method\", \"system.jl\")]","category":"page"},{"location":"systems/dem/#TrixiParticles.DEMSystem","page":"Discrete Element Method (Solid)","title":"TrixiParticles.DEMSystem","text":"DEMSystem(initial_condition, normal_stiffness, elastic_modulus, poissons_ratio;\n damping_coefficient=0.0001, acceleration=ntuple(_ -> 0.0, NDIMS), source_terms=nothing)\n\nConstructs a Discrete Element Method (DEM) system for numerically simulating the dynamics of granular and particulate matter. DEM is employed to simulate and analyze the motion, interactions, and collective behavior of assemblies of discrete, solid particles, typically under mechanical loading. The model accounts for individual particle characteristics and implements interaction laws that govern contact forces (normal and tangential), based on specified material properties and contact mechanics.\n\nArguments\n\ninitial_condition: Initial condition of the system, encapsulating the initial positions, velocities, masses, and radii of particles.\nnormal_stiffness: Normal stiffness coefficient for particle-particle and particle-wall contacts.\nelastic_modulus: Elastic modulus for this particle system.\npoissons_ratio: Poisson ratio for this particle system.\n\nKeywords\n\nacceleration: Global acceleration vector applied to the system, such as gravity. Specified as an SVector of length NDIMS, with a default of zero in each dimension.\nsource_terms: Optional; additional forces or modifications to particle dynamics not captured by standard DEM interactions, such as electromagnetic forces or user-defined perturbations.\ndamping_coefficient=0.0001: Set a damping coefficient for the collision interactions.\n\nwarning: Experimental Implementation\nThis is an experimental feature and may change in a future releases.\n\nReferences\n\n[9], [10], [11]\n\n\n\n\n\n","category":"type"},{"location":"general/smoothing_kernels/#smoothing_kernel","page":"Smoothing Kernels","title":"Smoothing Kernels","text":"","category":"section"},{"location":"general/smoothing_kernels/","page":"Smoothing Kernels","title":"Smoothing Kernels","text":"The following smoothing kernels are currently available:","category":"page"},{"location":"general/smoothing_kernels/","page":"Smoothing Kernels","title":"Smoothing Kernels","text":"Smoothing Kernel Compact Support Typ. Smoothing Length Recommended Application Stability\nSchoenbergCubicSplineKernel 0 2h 11 to 13 General + sharp waves ++\nSchoenbergQuarticSplineKernel 0 25h 11 to 15 General +++\nSchoenbergQuinticSplineKernel 0 3h 11 to 15 General ++++\nGaussianKernel 0 3h 10 to 15 Literature +++++\nWendlandC2Kernel 0 1h 25 to 40 General (recommended) ++++\nWendlandC4Kernel 0 1h 30 to 45 General +++++\nWendlandC6Kernel 0 1h 35 to 50 General +++++\nPoly6Kernel 0 1h 15 to 25 Literature +\nSpikyKernel 0 1h 15 to 30 Sharp corners + waves +","category":"page"},{"location":"general/smoothing_kernels/","page":"Smoothing Kernels","title":"Smoothing Kernels","text":"We recommend to use the WendlandC2Kernel for most applications. If less smoothing is needed, try SchoenbergCubicSplineKernel, for more smoothing try WendlandC6Kernel.","category":"page"},{"location":"general/smoothing_kernels/","page":"Smoothing Kernels","title":"Smoothing Kernels","text":"note: Usage\nThe kernel can be called asTrixiParticles.kernel(smoothing_kernel, r, h)The length of the compact support can be obtained asTrixiParticles.compact_support(smoothing_kernel, h)Note that r has to be a scalar, so in the context of SPH, the kernel should be used asW(Vert r_a - r_b Vert h)The gradient required in SPH, nabla_r_a W(Vert r_a - r_b Vert h)can be called asTrixiParticles.kernel_grad(smoothing_kernel, pos_diff, distance, h)where pos_diff is r_a - r_b and distance is Vert r_a - r_b Vert.","category":"page"},{"location":"general/smoothing_kernels/","page":"Smoothing Kernels","title":"Smoothing Kernels","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"general\", \"smoothing_kernels.jl\")]","category":"page"},{"location":"general/smoothing_kernels/#TrixiParticles.GaussianKernel","page":"Smoothing Kernels","title":"TrixiParticles.GaussianKernel","text":"GaussianKernel{NDIMS}()\n\nGaussian kernel given by\n\nW(r h) = fracsigma_dh^d e^-r^2h^2\n\nwhere d is the number of dimensions and\n\nsigma_2 = frac1pi for 2D,\nsigma_3 = frac1pi^32 for 3D.\n\nThis kernel function has an infinite support, but in practice, it's often truncated at a certain multiple of h, such as 3h.\n\nIn this implementation, the kernel is truncated at 3h, so this kernel function has a compact support of 0 3h.\n\nThe smoothing length is typically in the range 10delta 15delta, where delta is the typical particle spacing.\n\nFor general information and usage see Smoothing Kernels.\n\nNote: This truncation makes this Kernel not conservative, which is beneficial in regards to stability but makes it less accurate.\n\n\n\n\n\n","category":"type"},{"location":"general/smoothing_kernels/#TrixiParticles.Poly6Kernel","page":"Smoothing Kernels","title":"TrixiParticles.Poly6Kernel","text":"Poly6Kernel{NDIMS}()\n\nPoly6 kernel, a commonly used kernel in SPH literature [3], especially in computer graphics contexts. It is defined as\n\nW(r h) = frac1h^d w(rh)\n\nwith\n\nw(q) = sigma begincases\n (1 - q^2)^3 textif 0 leq q 1 \n 0 textif q geq 1\nendcases\n\nwhere d is the number of dimensions and sigma is a normalization factor that depends on the dimension. The normalization factor sigma is 4 pi in two dimensions or 315 64pi in three dimensions.\n\nThis kernel function has a compact support of 0 h.\n\nPoly6 is well-known for its computational simplicity, though it's worth noting that there are other kernels that might offer better accuracy for hydrodynamic simulations. Furthermore, its derivatives are not that smooth, which can lead to stability problems. It is also susceptible to clumping.\n\nThe smoothing length is typically in the range 15delta 25delta, where delta is the typical particle spacing.\n\nFor general information and usage see Smoothing Kernels.\n\n\n\n\n\n","category":"type"},{"location":"general/smoothing_kernels/#TrixiParticles.SchoenbergCubicSplineKernel","page":"Smoothing Kernels","title":"TrixiParticles.SchoenbergCubicSplineKernel","text":"SchoenbergCubicSplineKernel{NDIMS}()\n\nCubic spline kernel by Schoenberg (1946), given by\n\n W(r h) = frac1h^d w(rh)\n\nwith\n\nw(q) = sigma begincases\n frac14 (2 - q)^3 - (1 - q)^3 textif 0 leq q 1 \n frac14 (2 - q)^3 textif 1 leq q 2 \n 0 textif q geq 2 \nendcases\n\nwhere d is the number of dimensions and sigma is a normalization constant given by sigma =frac23 frac107 pi frac1pi in 1 2 3 dimensions.\n\nThis kernel function has a compact support of 0 2h.\n\nFor an overview of Schoenberg cubic, quartic and quintic spline kernels including normalization factors, see Price (2012). For an analytic formula for higher order Schoenberg kernels, see Monaghan (1985). The largest disadvantage of Schoenberg Spline Kernel is the rather non-smooth first derivative, which can lead to increased noise compared to other kernel variants.\n\nThe smoothing length is typically in the range 11delta 13delta, where delta is the typical particle spacing.\n\nFor general information and usage see Smoothing Kernels.\n\n\n\n\n\n","category":"type"},{"location":"general/smoothing_kernels/#TrixiParticles.SchoenbergQuarticSplineKernel","page":"Smoothing Kernels","title":"TrixiParticles.SchoenbergQuarticSplineKernel","text":"SchoenbergQuarticSplineKernel{NDIMS}()\n\nQuartic spline kernel by Schoenberg (1946), given by\n\n W(r h) = frac1h^d w(rh)\n\nwith\n\nw(q) = sigma begincases\n left(52 - q right)^4 - 5left(32 - q right)^4\n + 10left(12 - q right)^4 textif 0 leq q frac12 \n left(52 - q right)^4 - 5left(32 - q right)^4\n textif frac12 leq q frac32 \n left(52 - q right)^4 textif frac32 leq q frac52 \n 0 textif q geq frac52\nendcases\n\nwhere d is the number of dimensions and sigma is a normalization constant given by sigma =frac124 frac961199 pi frac120pi in 1 2 3 dimensions.\n\nThis kernel function has a compact support of 0 25h.\n\nFor an overview of Schoenberg cubic, quartic and quintic spline kernels including normalization factors, see Price (2012). For an analytic formula for higher order Schoenberg kernels, see Monaghan (1985).\n\nThe largest disadvantage of Schoenberg Spline Kernel are the rather non-smooth first derivative, which can lead to increased noise compared to other kernel variants.\n\nThe smoothing length is typically in the range 11delta 15delta, where delta is the typical particle spacing.\n\nFor general information and usage see Smoothing Kernels.\n\n\n\n\n\n","category":"type"},{"location":"general/smoothing_kernels/#TrixiParticles.SchoenbergQuinticSplineKernel","page":"Smoothing Kernels","title":"TrixiParticles.SchoenbergQuinticSplineKernel","text":"SchoenbergQuinticSplineKernel{NDIMS}()\n\nQuintic spline kernel by Schoenberg (1946), given by\n\n W(r h) = frac1h^d w(rh)\n\nwith\n\nw(q) = sigma begincases\n (3 - q)^5 - 6(2 - q)^5 + 15(1 - q)^5 textif 0 leq q 1 \n (3 - q)^5 - 6(2 - q)^5 textif 1 leq q 2 \n (3 - q)^5 textif 2 leq q 3 \n 0 textif q geq 3\nendcases\n\nwhere d is the number of dimensions and sigma is a normalization constant given by sigma =frac1120 frac7478 pi frac1120pi in 1 2 3 dimensions.\n\nThis kernel function has a compact support of 0 3h.\n\nFor an overview of Schoenberg cubic, quartic and quintic spline kernels including normalization factors, see Price (2012). For an analytic formula for higher order Schoenberg kernels, see Monaghan (1985).\n\nThe largest disadvantage of Schoenberg Spline Kernel are the rather non-smooth first derivative, which can lead to increased noise compared to other kernel variants.\n\nThe smoothing length is typically in the range 11delta 15delta, where delta is the typical particle spacing.\n\nFor general information and usage see Smoothing Kernels.\n\n\n\n\n\n","category":"type"},{"location":"general/smoothing_kernels/#TrixiParticles.SpikyKernel","page":"Smoothing Kernels","title":"TrixiParticles.SpikyKernel","text":"SpikyKernel{NDIMS}()\n\nThe Spiky kernel is another frequently used kernel in SPH, especially due to its desirable properties in preserving features near boundaries in fluid simulations [3]. It is defined as:\n\n W(r h) = frac1h^d w(rh)\n\nwith:\n\nw(q) = sigma begincases\n (1 - q)^3 textif 0 leq q 1 \n 0 textif q geq 1\nendcases\n\nwhere d is the number of dimensions and the normalization factor sigma is 10 pi in two dimensions or 15 pi in three dimensions.\n\nThis kernel function has a compact support of 0 h.\n\nThe Spiky kernel is particularly known for its sharp gradients, which can help to preserve sharp features in fluid simulations, especially near solid boundaries. These sharp gradients at the boundary are also the largest disadvantage as they can lead to instability.\n\nThe smoothing length is typically in the range 15delta 30delta, where delta is the typical particle spacing.\n\nFor general information and usage see Smoothing Kernels.\n\n\n\n\n\n","category":"type"},{"location":"general/smoothing_kernels/#TrixiParticles.WendlandC2Kernel","page":"Smoothing Kernels","title":"TrixiParticles.WendlandC2Kernel","text":"WendlandC2Kernel{NDIMS}()\n\nWendland C2 kernel [7], a piecewise polynomial function designed to have compact support and to be twice continuously differentiable everywhere. Given by\n\n W(r h) = frac1h^d w(rh)\n\nwith\n\nw(q) = sigma begincases\n (1 - q)^4 (4q + 1) textif 0 leq q 1 \n 0 textif q geq 1\nendcases\n\nwhere d is the number of dimensions and sigma is a normalization factor dependent on the dimension. The normalization factor sigma is 407pi in two dimensions or 212pi in three dimensions.\n\nThis kernel function has a compact support of 0 h.\n\nFor a detailed discussion on Wendland functions and their applications in SPH, see Dehnen (2012). The smoothness of these functions is also the largest disadvantage as they lose details at sharp corners.\n\nThe smoothing length is typically in the range 25delta 40delta, where delta is the typical particle spacing.\n\nFor general information and usage see Smoothing Kernels.\n\n\n\n\n\n","category":"type"},{"location":"general/smoothing_kernels/#TrixiParticles.WendlandC4Kernel","page":"Smoothing Kernels","title":"TrixiParticles.WendlandC4Kernel","text":"WendlandC4Kernel{NDIMS}()\n\nWendland C4 kernel [7], a piecewise polynomial function designed to have compact support and to be four times continuously differentiable everywhere. Given by\n\n W(r h) = frac1h^d w(rh)\n\nwith\n\nw(q) = sigma begincases\n (1 - q)^6 (35q^2 3 + 6q + 1) textif 0 leq q 1 \n 0 textif q geq 1\nendcases\n\nwhere d is the number of dimensions and sigma is a normalization factor dependent on the dimension. The normalization factor sigma is 9 pi in two dimensions or 495 32pi in three dimensions.\n\nThis kernel function has a compact support of 0 h.\n\nFor a detailed discussion on Wendland functions and their applications in SPH, see Dehnen (2012). The smoothness of these functions is also the largest disadvantage as they loose details at sharp corners.\n\nThe smoothing length is typically in the range 30delta 45delta, where delta is the typical particle spacing.\n\nFor general information and usage see Smoothing Kernels.\n\n\n\n\n\n","category":"type"},{"location":"general/smoothing_kernels/#TrixiParticles.WendlandC6Kernel","page":"Smoothing Kernels","title":"TrixiParticles.WendlandC6Kernel","text":"WendlandC6Kernel{NDIMS}()\n\nWendland C6 kernel [7], a piecewise polynomial function designed to have compact support and to be six times continuously differentiable everywhere. Given by:\n\nW(r h) = frac1h^d w(rh)\n\nwith:\n\nw(q) = sigma begincases\n (1 - q)^8 (32q^3 + 25q^2 + 8q + 1) textif 0 leq q 1 \n 0 textif q geq 1\nendcases\n\nwhere d is the number of dimensions and sigma is a normalization factor dependent on the dimension. The normalization factor sigma is 78 7 pi in two dimensions or 1365 64pi in three dimensions.\n\nThis kernel function has a compact support of 0 h.\n\nFor a detailed discussion on Wendland functions and their applications in SPH, Dehnen (2012). The smoothness of these functions is also the largest disadvantage as they loose details at sharp corners.\n\nThe smoothing length is typically in the range 35delta 50delta, where delta is the typical particle spacing.\n\nFor general information and usage see Smoothing Kernels.\n\n\n\n\n\n","category":"type"},{"location":"reference-trixibase/#TrixiBase.jl-API","page":"TrixiBase.jl API Reference","title":"TrixiBase.jl API","text":"","category":"section"},{"location":"reference-trixibase/","page":"TrixiBase.jl API Reference","title":"TrixiBase.jl API Reference","text":"CurrentModule = TrixiBase","category":"page"},{"location":"reference-trixibase/","page":"TrixiBase.jl API Reference","title":"TrixiBase.jl API Reference","text":"Modules = [TrixiBase]","category":"page"},{"location":"reference-trixibase/#TrixiBase.disable_debug_timings-Tuple{}","page":"TrixiBase.jl API Reference","title":"TrixiBase.disable_debug_timings","text":"disable_debug_timings()\n\nDisable all @trixi_timeit timings. The timings should be optimized away, allowing for truly zero-overhead. Enable timings again with enable_debug_timings.\n\nSee also enable_debug_timings, @trixi_timeit.\n\n\n\n\n\n","category":"method"},{"location":"reference-trixibase/#TrixiBase.enable_debug_timings-Tuple{}","page":"TrixiBase.jl API Reference","title":"TrixiBase.enable_debug_timings","text":"enable_debug_timings()\n\nEnable all @trixi_timeit timings (default behavior).\n\nSee also disable_debug_timings, @trixi_timeit.\n\n\n\n\n\n","category":"method"},{"location":"reference-trixibase/#TrixiBase.timer-Tuple{}","page":"TrixiBase.jl API Reference","title":"TrixiBase.timer","text":"timer()\n\nMain timer for global timing, e.g., to be used with @trixi_timeit.\n\n\n\n\n\n","category":"method"},{"location":"reference-trixibase/#TrixiBase.trixi_include-Tuple{Module, AbstractString}","page":"TrixiBase.jl API Reference","title":"TrixiBase.trixi_include","text":"trixi_include([mod::Module=Main,] elixir::AbstractString; kwargs...)\n\ninclude the file elixir and evaluate its content in the global scope of module mod. You can override specific assignments in elixir by supplying keyword arguments. Its basic purpose is to make it easier to modify some parameters while running simulations from the REPL. Additionally, this is used in tests to reduce the computational burden for CI while still providing examples with sensible default values for users.\n\nBefore replacing assignments in elixir, the keyword argument maxiters is inserted into calls to solve with it's default value used in the SciML ecosystem for ODEs, see the \"Miscellaneous\" section of the documentation.\n\nExamples\n\njulia> using TrixiBase, Trixi\n\njulia> redirect_stdout(devnull) do\n trixi_include(@__MODULE__, joinpath(examples_dir(), \"tree_1d_dgsem\", \"elixir_advection_extended.jl\"),\n tspan=(0.0, 0.1))\n sol.t[end]\n end\n[ Info: You just called `trixi_include`. Julia may now compile the code, please be patient.\n0.1\n\n\n\n\n\n","category":"method"},{"location":"reference-trixibase/#TrixiBase.@trixi_timeit-Tuple{Any, Any, Any}","page":"TrixiBase.jl API Reference","title":"TrixiBase.@trixi_timeit","text":"@trixi_timeit timer() \"some label\" expression\n\nBasically the same as a special case of @timeit_debug from TimerOutputs.jl, but without try ... finally ... end block. Thus, it's not exception-safe, but it also avoids some related performance problems. Since we do not use exception handling in Trixi.jl, that's not really an issue.\n\nAll @trixi_timeit timings can be disabled with disable_debug_timings. The timings should then be optimized away, allowing for truly zero-overhead.\n\nSee also disable_debug_timings, enable_debug_timings.\n\n\n\n\n\n","category":"macro"},{"location":"tutorials_template/tut_beam/#Example-file","page":"Example file","title":"Example file","text":"","category":"section"},{"location":"tutorials_template/tut_beam/","page":"Example file","title":"Example file","text":"!!include:examples/solid/oscillating_beam_2d.jl!!\n","category":"page"},{"location":"contributing/","page":"Contributing","title":"Contributing","text":"EditURL = \"https://github.com/trixi-framework/TrixiParticles.jl/blob/main/CONTRIBUTING.md\"","category":"page"},{"location":"contributing/#Contributing","page":"Contributing","title":"Contributing","text":"","category":"section"},{"location":"contributing/","page":"Contributing","title":"Contributing","text":"TrixiParticles.jl is an open-source project and we are very happy to accept contributions from the community. Please feel free to open issues or submit patches (preferably as pull requests) any time. For planned larger contributions, it is often beneficial to get in contact with one of the principal developers first (see Authors).","category":"page"},{"location":"contributing/","page":"Contributing","title":"Contributing","text":"TrixiParticles.jl and its contributions are licensed under the MIT license (see License). As a contributor, you certify that all your contributions are in conformance with the Developer Certificate of Origin (Version 1.1), which is reproduced below.","category":"page"},{"location":"contributing/#Developer-Certificate-of-Origin-(Version-1.1)","page":"Contributing","title":"Developer Certificate of Origin (Version 1.1)","text":"","category":"section"},{"location":"contributing/","page":"Contributing","title":"Contributing","text":"The following text was taken from https://developercertificate.org:","category":"page"},{"location":"contributing/","page":"Contributing","title":"Contributing","text":"Developer Certificate of Origin\nVersion 1.1\n\nCopyright (C) 2004, 2006 The Linux Foundation and its contributors.\n1 Letterman Drive\nSuite D4700\nSan Francisco, CA, 94129\n\nEveryone is permitted to copy and distribute verbatim copies of this\nlicense document, but changing it is not allowed.\n\n\nDeveloper's Certificate of Origin 1.1\n\nBy making a contribution to this project, I certify that:\n\n(a) The contribution was created in whole or in part by me and I\n have the right to submit it under the open source license\n indicated in the file; or\n\n(b) The contribution is based upon previous work that, to the best\n of my knowledge, is covered under an appropriate open source\n license and I have the right under that license to submit that\n work with modifications, whether created in whole or in part\n by me, under the same open source license (unless I am\n permitted to submit under a different license), as indicated\n in the file; or\n\n(c) The contribution was provided directly to me by some other\n person who certified (a), (b) or (c) and I have not modified\n it.\n\n(d) I understand and agree that this project and the contribution\n are public and that a record of the contribution (including all\n personal information I submit with it, including my sign-off) is\n maintained indefinitely and may be redistributed consistent with\n this project or the open source license(s) involved.","category":"page"},{"location":"development/#development","page":"Development","title":"Development","text":"","category":"section"},{"location":"development/#Preview-of-the-documentation","page":"Development","title":"Preview of the documentation","text":"","category":"section"},{"location":"development/","page":"Development","title":"Development","text":"To generate the Documentation, first instantiate the docs environment by executing the following command from the TrixiParticles.jl root directory:","category":"page"},{"location":"development/","page":"Development","title":"Development","text":"julia --project=docs -e \"using Pkg; Pkg.develop(PackageSpec(path=pwd())); Pkg.instantiate()\"","category":"page"},{"location":"development/","page":"Development","title":"Development","text":"This command only has to be run once. After that, maintain the docs environment as described under Installation.","category":"page"},{"location":"development/","page":"Development","title":"Development","text":"With an instantiated docs environment, generate the docs with the following command (again from the TrixiParticles.jl root directory):","category":"page"},{"location":"development/","page":"Development","title":"Development","text":"julia --project=docs --color=yes docs/make.jl","category":"page"},{"location":"development/","page":"Development","title":"Development","text":"You can then open the generated files in docs/build with your webbrowser. Alternatively, run","category":"page"},{"location":"development/","page":"Development","title":"Development","text":"python3 -m http.server -d docs/build","category":"page"},{"location":"development/","page":"Development","title":"Development","text":"and open localhost:8000 in your webbrowser.","category":"page"},{"location":"development/#Release-management","page":"Development","title":"Release management","text":"","category":"section"},{"location":"development/","page":"Development","title":"Development","text":"To create a new release for TrixiParticles.jl, perform the following steps:","category":"page"},{"location":"development/","page":"Development","title":"Development","text":"Make sure that all PRs and changes that you want to go into the release are merged to main and that the latest commit on main has passed all CI tests.\nDetermine the currently released version of TrixiParticles.jl, e.g., on the release page. For this manual, we will assume that the latest release was v0.2.3.\nDecide on the next version number. We follow semantic versioning, thus each version is of the form vX.Y.Z where X is the major version, Y the minor version, and Z the patch version. In this manual, we assume that the major version is always 0, thus the decision process on the new version is as follows:\nIf the new release contains breaking changes (i.e., user code might not work as before without modifications), increase the minor version by one and set the patch version to zero. In our example, the new version should thus be v0.3.0.\nIf the new release only contains minor modifications and/or bug fixes, the minor version is kept as-is and the patch version is increased by one. In our example, the new version should thus be v0.2.4.\nEdit the version string in the Project.toml and set it to the new version. Push/merge this change to main.\nGo to GitHub and add a comment to the commit that you would like to become the new release (typically this will be the commit where you just updated the version). You can comment on a commit by going to the commit overview and clicking on the title of the commit. The comment should contain the following text:\n@JuliaRegistrator register\nWait for the magic to happen! Specifically, JuliaRegistrator will create a new PR to the Julia registry with the new release information. After a grace period of ~15 minutes, this PR will be merged automatically. A short while after, TagBot will create a new release of TrixiParticles.jl in our GitHub repository.\nOnce the new release has been created, the new version can be obtained through the Julia package manager as usual.\nTo make sure people do not mistake the latest state of main as the latest release, we set the version in the Project.toml to a development version. The development version should be the latest released version, with the patch version incremented by one, and the -dev suffix added. For example, if you just released v0.3.0, the new development version should be v0.3.1-dev. If you just released v0.2.4, the new development version should be v0.2.5-dev.","category":"page"},{"location":"install/#installation","page":"Installation","title":"Installation","text":"","category":"section"},{"location":"install/#Setting-up-Julia","page":"Installation","title":"Setting up Julia","text":"","category":"section"},{"location":"install/","page":"Installation","title":"Installation","text":"If you have not yet installed Julia, please follow the instructions on the official website. TrixiParticles.jl works with Julia v1.9 and newer. We recommend using the latest stable release of Julia.","category":"page"},{"location":"install/#For-users","page":"Installation","title":"For users","text":"","category":"section"},{"location":"install/","page":"Installation","title":"Installation","text":"TrixiParticles.jl is a registered Julia package. You can install TrixiParticles.jl, OrdinaryDiffEq.jl (used for time integration) and Plots.jl by executing the following commands in the Julia REPL:","category":"page"},{"location":"install/","page":"Installation","title":"Installation","text":"julia> using Pkg\n\njulia> Pkg.add([\"TrixiParticles\", \"OrdinaryDiffEq\", \"Plots\"])","category":"page"},{"location":"install/#for-developers","page":"Installation","title":"For developers","text":"","category":"section"},{"location":"install/","page":"Installation","title":"Installation","text":"If you plan on editing TrixiParticles.jl itself, you can download TrixiParticles.jl to a local folder and use the code from the cloned directory:","category":"page"},{"location":"install/","page":"Installation","title":"Installation","text":"git clone git@github.com:trixi-framework/TrixiParticles.jl.git\ncd TrixiParticles.jl\nmkdir run\njulia --project=run -e 'using Pkg; Pkg.develop(PackageSpec(path=\".\"))' # Add TrixiParticles.jl to `run` project\njulia --project=run -e 'using Pkg; Pkg.add([\"OrdinaryDiffEq\", \"Plots\"])' # Add additional packages","category":"page"},{"location":"install/","page":"Installation","title":"Installation","text":"If you installed TrixiParticles.jl this way, you always have to start Julia with the --project flag set to your run directory, e.g.,","category":"page"},{"location":"install/","page":"Installation","title":"Installation","text":"julia --project=run","category":"page"},{"location":"install/","page":"Installation","title":"Installation","text":"from the TrixiParticles.jl root directory.","category":"page"},{"location":"install/","page":"Installation","title":"Installation","text":"The advantage of using a separate run directory is that you can also add other related packages (e.g., OrdinaryDiffEq.jl, see above) to the project in the run folder and always have a reproducible environment at hand to share with others.","category":"page"},{"location":"install/#Optional-software/packages","page":"Installation","title":"Optional software/packages","text":"","category":"section"},{"location":"install/","page":"Installation","title":"Installation","text":"OrdinaryDiffEq.jl – A Julia package of ordinary differential equation solvers that is used in the examples\nPlots.jl – Julia Plotting library that is used in some examples\nPythonPlot.jl – Plotting library that can be used instead of Plots.jl\nParaView – Software that can be used for visualization of results","category":"page"},{"location":"install/#installation-issues","page":"Installation","title":"Common issues","text":"","category":"section"},{"location":"install/","page":"Installation","title":"Installation","text":"If you followed the installation instructions for developers and you run into any problems with packages when pulling the latest version of TrixiParticles.jl, start Julia with the project in the run folder,","category":"page"},{"location":"install/","page":"Installation","title":"Installation","text":" julia --project=run","category":"page"},{"location":"install/","page":"Installation","title":"Installation","text":"update all packages in that project, resolve all conflicts in the project, and install all new dependencies:","category":"page"},{"location":"install/","page":"Installation","title":"Installation","text":"julia> using Pkg\n\njulia> Pkg.update()\n\njulia> Pkg.resolve()\n\njulia> Pkg.instantiate()","category":"page"},{"location":"visualization/#Visualization","page":"Visualization","title":"Visualization","text":"","category":"section"},{"location":"visualization/#Export-VTK-files","page":"Visualization","title":"Export VTK files","text":"","category":"section"},{"location":"visualization/","page":"Visualization","title":"Visualization","text":"You can export particle data as VTK files by using the SolutionSavingCallback. All our predefined examples are already using this callback to export VTK files to the out directory (relative to the directory that you are running Julia from). VTK files can be read by visualization tools like ParaView and VisIt.","category":"page"},{"location":"visualization/#ParaView","page":"Visualization","title":"ParaView","text":"","category":"section"},{"location":"visualization/","page":"Visualization","title":"Visualization","text":"Follow these steps to view the exported VTK files in ParaView:","category":"page"},{"location":"visualization/","page":"Visualization","title":"Visualization","text":"Click File -> Open.\nNavigate to the out directory (relative to the directory that you are running Julia from).\nOpen both boundary_1.pvd and fluid_1.pvd.\nClick \"Apply\", which by default is on the left pane below the \"Pipeline Browser\".\nHold the left mouse button to move the solution around.","category":"page"},{"location":"visualization/","page":"Visualization","title":"Visualization","text":"You will now see the following: (Image: image)","category":"page"},{"location":"visualization/","page":"Visualization","title":"Visualization","text":"It is useful to make the particles larger. For this, first make sure you have \"fluid_1.pvd\" highlighted in the \"Pipeline Browser\" then in the \"Properties\" window in the bottom left change \"Point Size\" to a larger value. (Image: image)","category":"page"},{"location":"visualization/","page":"Visualization","title":"Visualization","text":"To now view the result variables first make sure you have \"fluid_1.pvd\" highlighted in the \"Pipeline Browser\" then select them in the variable selection combo box (see picture below). Let's, for example, pick \"density\". To now view the time progression of the result hit the \"play button\" (see picture below). (Image: image)","category":"page"},{"location":"visualization/#API","page":"Visualization","title":"API","text":"","category":"section"},{"location":"visualization/","page":"Visualization","title":"Visualization","text":"Modules = [TrixiParticles]\nPages = map(file -> joinpath(\"visualization\", file), readdir(joinpath(\"..\", \"src\", \"visualization\")))","category":"page"},{"location":"visualization/#TrixiParticles.trixi2vtk-Tuple{Any, Any, Any}","page":"Visualization","title":"TrixiParticles.trixi2vtk","text":"trixi2vtk(vu_ode, semi, t; iter=nothing, output_directory=\"out\", prefix=\"\",\n write_meta_data=true, max_coordinates=Inf, custom_quantities...)\n\nConvert Trixi simulation data to VTK format.\n\nArguments\n\nvu_ode: Solution of the TrixiParticles ODE system at one time step. This expects an ArrayPartition as returned in the examples as sol.u[end].\nsemi: Semidiscretization of the TrixiParticles simulation.\nt: Current time of the simulation.\n\nKeywords\n\niter=nothing: Iteration number when multiple iterations are to be stored in separate files. This number is just appended to the filename.\noutput_directory=\"out\": Output directory path.\nprefix=\"\": Prefix for output files.\nwrite_meta_data=true: Write meta data.\nmax_coordinates=Inf The coordinates of particles will be clipped if their absolute values exceed this threshold.\ncustom_quantities...: Additional custom quantities to include in the VTK output. Each custom quantity must be a function of (v, u, t, system), which will be called for every system, where v and u are the wrapped solution arrays for the corresponding system and t is the current simulation time. Note that working with these v and u arrays requires undocumented internal functions of TrixiParticles. See Custom Quantities for a list of pre-defined custom quantities that can be used here.\n\nExample\n\ntrixi2vtk(sol.u[end], semi, 0.0, iter=1, output_directory=\"output\", prefix=\"solution\")\n\n# Additionally store the kinetic energy of each system as \"my_custom_quantity\"\ntrixi2vtk(sol.u[end], semi, 0.0, iter=1, my_custom_quantity=kinetic_energy)\n\n\n\n\n\n","category":"method"},{"location":"visualization/#TrixiParticles.trixi2vtk-Tuple{Any}","page":"Visualization","title":"TrixiParticles.trixi2vtk","text":"trixi2vtk(coordinates; output_directory=\"out\", prefix=\"\", filename=\"coordinates\",\n custom_quantities...)\n\nConvert coordinate data to VTK format.\n\nArguments\n\ncoordinates: Coordinates to be saved.\n\nKeywords\n\noutput_directory=\"out\": Output directory path.\nprefix=\"\": Prefix for the output file.\nfilename=\"coordinates\": Name of the output file.\ncustom_quantities...: Additional custom quantities to include in the VTK output.\n\nReturns\n\nfile::AbstractString: Path to the generated VTK file.\n\n\n\n\n\n","category":"method"},{"location":"visualization/#TrixiParticles.trixi2vtk-Tuple{InitialCondition}","page":"Visualization","title":"TrixiParticles.trixi2vtk","text":"trixi2vtk(initial_condition::InitialCondition; output_directory=\"out\",\n prefix=\"\", filename=\"initial_condition\", custom_quantities...)\n\nConvert InitialCondition data to VTK format.\n\nArguments\n\ninitial_condition: InitialCondition to be saved.\n\nKeywords\n\noutput_directory=\"out\": Output directory path.\nprefix=\"\": Prefix for the output file.\nfilename=\"coordinates\": Name of the output file.\ncustom_quantities...: Additional custom quantities to include in the VTK output.\n\nReturns\n\nfile::AbstractString: Path to the generated VTK file.\n\n\n\n\n\n","category":"method"},{"location":"general/initial_condition/#initial_condition","page":"Initial Condition and Setups","title":"Initial Condition","text":"","category":"section"},{"location":"general/initial_condition/","page":"Initial Condition and Setups","title":"Initial Condition and Setups","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"general\", \"initial_condition.jl\")]","category":"page"},{"location":"general/initial_condition/#TrixiParticles.InitialCondition","page":"Initial Condition and Setups","title":"TrixiParticles.InitialCondition","text":"InitialCondition(; coordinates, density, velocity=zeros(size(coordinates, 1)),\n mass=nothing, pressure=0.0, particle_spacing=-1.0)\n\nStruct to hold the initial configuration of the particles.\n\nThe following setups return InitialConditions for commonly used setups:\n\nRectangularShape\nSphereShape\nRectangularTank\nComplexShape\nextrude_geometry\n\nInitialConditions support the set operations union, setdiff and intersect in order to build more complex geometries.\n\nArguments\n\ncoordinates: An array where the i-th column holds the coordinates of particle i.\ndensity: Either a vector holding the density of each particle, or a function mapping each particle's coordinates to its density, or a scalar for a constant density over all particles.\n\nKeywords\n\nvelocity: Either an array where the i-th column holds the velocity of particle i, or a function mapping each particle's coordinates to its velocity, or, for a constant fluid velocity, a vector holding this velocity. Velocity is constant zero by default.\nmass: Either nothing (default) to automatically compute particle mass from particle density and spacing, or a vector holding the mass of each particle, or a function mapping each particle's coordinates to its mass, or a scalar for a constant mass over all particles.\npressure: Either a vector holding the pressure of each particle, or a function mapping each particle's coordinates to its pressure, or a scalar for a constant pressure over all particles. This is optional and only needed when using the EntropicallyDampedSPHSystem.\nparticle_spacing: The spacing between the particles. This is a scalar, as the spacing is assumed to be uniform. This is only needed when using set operations on the InitialCondition or for automatic mass calculation.\n\nExamples\n\n# Rectangle filled with particles\ninitial_condition = RectangularShape(0.1, (3, 4), (-1.0, 1.0), density=1.0)\n\n# Two spheres in one initial condition\ninitial_condition = union(SphereShape(0.15, 0.5, (-1.0, 1.0), 1.0),\n SphereShape(0.15, 0.2, (0.0, 1.0), 1.0))\n\n# Rectangle with a spherical hole\nshape1 = RectangularShape(0.1, (16, 13), (-0.8, 0.0), density=1.0)\nshape2 = SphereShape(0.1, 0.35, (0.0, 0.6), 1.0, sphere_type=RoundSphere())\ninitial_condition = setdiff(shape1, shape2)\n\n# Intersect of a rectangle with a sphere. Note that this keeps the particles of the\n# rectangle that are in the intersect, while `intersect(shape2, shape1)` would consist of\n# the particles of the sphere that are in the intersect.\nshape1 = RectangularShape(0.1, (16, 13), (-0.8, 0.0), density=1.0)\nshape2 = SphereShape(0.1, 0.35, (0.0, 0.6), 1.0, sphere_type=RoundSphere())\ninitial_condition = intersect(shape1, shape2)\n\n# Build `InitialCondition` manually\ncoordinates = [0.0 1.0 1.0\n 0.0 0.0 1.0]\nvelocity = zero(coordinates)\nmass = ones(3)\ndensity = 1000 * ones(3)\ninitial_condition = InitialCondition(; coordinates, velocity, mass, density)\n\n# With functions\ninitial_condition = InitialCondition(; coordinates, velocity=x -> 2x, mass=1.0, density=1000.0)\n\n\n\n\n\n","category":"type"},{"location":"general/initial_condition/#Setups","page":"Initial Condition and Setups","title":"Setups","text":"","category":"section"},{"location":"general/initial_condition/","page":"Initial Condition and Setups","title":"Initial Condition and Setups","text":"Modules = [TrixiParticles]\nPages = map(file -> joinpath(\"setups\", file), readdir(joinpath(\"..\", \"src\", \"setups\")))","category":"page"},{"location":"general/initial_condition/#TrixiParticles.ComplexShape-Tuple{Union{TrixiParticles.Polygon, TrixiParticles.TriangleMesh}}","page":"Initial Condition and Setups","title":"TrixiParticles.ComplexShape","text":"ComplexShape(geometry::Union{TriangleMesh, Polygon}; particle_spacing, density,\n pressure=0.0, mass=nothing, velocity=zeros(ndims(geometry)),\n point_in_geometry_algorithm=WindingNumberJacobson(; geometry,\n hierarchical_winding=false,\n winding_number_factor=sqrt(eps())),\n grid_offset::Real=0.0, max_nparticles=10^7,\n pad_initial_particle_grid=2particle_spacing)\n\nSample a complex geometry with particles. Returns an InitialCondition. Note that an initial particle grid is generated inside the bounding box of the geometry. A point_in_geometry_algorithm checks if particles are inside the geometry or not. For more information about the method see WindingNumberJacobson or WindingNumberHormann.\n\nArguments\n\ngeometry: Geometry returned by load_geometry.\n\nKeywords\n\nparticle_spacing: Spacing between the particles.\ndensity: Either a function mapping each particle's coordinates to its density, or a scalar for a constant density over all particles.\nvelocity: Either a function mapping each particle's coordinates to its velocity, or, for a constant fluid velocity, a vector holding this velocity. Velocity is constant zero by default.\nmass: Either nothing (default) to automatically compute particle mass from particle density and spacing, or a function mapping each particle's coordinates to its mass, or a scalar for a constant mass over all particles.\npressure: Scalar to set the pressure of all particles to this value. This is only used by the EntropicallyDampedSPHSystem and will be overwritten when using an initial pressure function in the system.\npoint_in_geometry_algorithm: Algorithm for sampling the complex geometry with particles. It basically checks whether a particle is inside an object or not. For more information see WindingNumberJacobson or WindingNumberHormann\ngrid_offset: Offset of the initial particle grid of the bounding box of the geometry.\nmax_nparticles: Maximum number of particles in the initial particle grid. This is only used to avoid accidentally choosing a particle_spacing that is too small for the scale of the geometry.\npad_initial_particle_grid: Padding of the initial particle grid.\n\nwarning: Experimental Implementation\nThis is an experimental feature and may change in any future releases.\n\n\n\n\n\n","category":"method"},{"location":"general/initial_condition/#TrixiParticles.extrude_geometry-Tuple{Any}","page":"Initial Condition and Setups","title":"TrixiParticles.extrude_geometry","text":"extrude_geometry(geometry; particle_spacing, direction, n_extrude::Integer,\n velocity=zeros(length(direction)),\n mass=nothing, density=nothing, pressure=0.0)\n\nExtrude either a line, a plane or a shape along a specific direction. Returns an InitialCondition.\n\nArguments\n\ngeometry: Either particle coordinates or an InitialCondition defining a 2D shape to extrude to a 3D volume, or two 2D points (A B) defining the interval A B to extrude to a plane in 2D, or three 3D points (A B C) defining the parallelogram spanned by the vectors widehatAB and widehat AC to extrude to a parallelepiped.\n\nKeywords\n\nparticle_spacing: Spacing between the particles. Can be omitted when geometry is an InitialCondition (unless geometry.particle_spacing == -1).\ndirection: A vector that specifies the direction in which to extrude.\nn_extrude: Number of layers of particles created in the direction of extrusion.\nvelocity: Either a function mapping each particle's coordinates to its velocity, or, for a constant fluid velocity, a vector holding this velocity. Velocity is constant zero by default.\nmass: Either nothing (default) to automatically compute particle mass from particle density and spacing, or a function mapping each particle's coordinates to its mass, or a scalar for a constant mass over all particles.\ndensity: Either a function mapping each particle's coordinates to its density, or a scalar for a constant density over all particles.\npressure: Scalar to set the pressure of all particles to this value. This is only used by the EntropicallyDampedSPHSystem and will be overwritten when using an initial pressure function in the system.\ntlsph: With the TotalLagrangianSPHSystem, particles need to be placed on the boundary of the shape and not one particle radius away, as for fluids. When tlsph=true, particles will be placed on the boundary of the shape.\n\nExamples\n\n# Extrude a line in 2D to a plane in 2D\np1 = [0.0, 0.0]\np2 = [1.0, 1.0]\n\ndirection = [-1.0, 1.0]\n\nshape = extrude_geometry((p1, p2); direction, particle_spacing=0.1, n_extrude=4, density=1000.0)\n\n# Extrude a parallelogram in 3D to a parallelepiped in 3D\np1 = [0.0, 0.0, 0.0]\np2 = [0.5, 1.0, 0.0]\np3 = [1.0, 0.2, 0.0]\n\ndirection = [0.0, 0.0, 1.0]\n\nshape = extrude_geometry((p1, p2, p3); direction, particle_spacing=0.1, n_extrude=4, density=1000.0)\n\n# Extrude a 2D shape (here: a disc) to a 3D shape (here: a cylinder)\nshape = SphereShape(0.1, 0.5, (0.2, 0.4), 1000.0, n_layers=3,\n sphere_type=RoundSphere(end_angle=pi))\n\ndirection = [0.0, 0.0, 1.0]\n\nshape = extrude_geometry(shape; direction, particle_spacing=0.1, n_extrude=4, density=1000.0)\n\nwarning: Experimental Implementation\nThis is an experimental feature and may change in any future releases.\n\n\n\n\n\n","category":"method"},{"location":"general/initial_condition/#TrixiParticles.RectangularShape-Tuple{Any, Any, Any}","page":"Initial Condition and Setups","title":"TrixiParticles.RectangularShape","text":"RectangularShape(particle_spacing, n_particles_per_dimension, min_coordinates;\n velocity=zeros(length(n_particles_per_dimension)),\n mass=nothing, density=nothing, pressure=0.0,\n acceleration=nothing, state_equation=nothing,\n tlsph=false, loop_order=nothing)\n\nRectangular shape filled with particles. Returns an InitialCondition.\n\nArguments\n\nparticle_spacing: Spacing between the particles.\nn_particles_per_dimension: Tuple containing the number of particles in x, y and z (only 3D) direction, respectively.\nmin_coordinates: Coordinates of the corner in negative coordinate directions.\n\nKeywords\n\nvelocity: Either a function mapping each particle's coordinates to its velocity, or, for a constant fluid velocity, a vector holding this velocity. Velocity is constant zero by default.\nmass: Either nothing (default) to automatically compute particle mass from particle density and spacing, or a function mapping each particle's coordinates to its mass, or a scalar for a constant mass over all particles.\ndensity: Either a function mapping each particle's coordinates to its density, or a scalar for a constant density over all particles. Obligatory when not using a state equation. Cannot be used together with state_equation.\npressure: Scalar to set the pressure of all particles to this value. This is only used by the EntropicallyDampedSPHSystem and will be overwritten when using an initial pressure function in the system. Cannot be used together with hydrostatic pressure gradient.\nacceleration: In order to initialize particles with a hydrostatic pressure gradient, an acceleration vector can be passed. Note that only accelerations in one coordinate direction and no diagonal accelerations are supported. This will only change the pressure of the particles. When using the WeaklyCompressibleSPHSystem, pass a state_equation as well to initialize the particles with the corresponding density and mass. When using the EntropicallyDampedSPHSystem, the pressure will be overwritten when using an initial pressure function in the system. This cannot be used together with the pressure keyword argument.\nstate_equation: When calculating a hydrostatic pressure gradient by setting acceleration, the state_equation will be used to set the corresponding density. Cannot be used together with density.\ntlsph: With the TotalLagrangianSPHSystem, particles need to be placed on the boundary of the shape and not one particle radius away, as for fluids. When tlsph=true, particles will be placed on the boundary of the shape.\ncoordinates_perturbation: Add a small random displacement to the particle positions, where the amplitude is coordinates_perturbation * particle_spacing.\n\nExamples\n\n# 2D\nrectangular = RectangularShape(particle_spacing, (5, 4), (1.0, 2.0), density=1000.0)\n\n# 2D with hydrostatic pressure gradient.\n# `state_equation` has to be the same as for the WCSPH system.\nstate_equation = StateEquationCole(sound_speed=20.0, exponent=7, reference_density=1000.0)\nrectangular = RectangularShape(particle_spacing, (5, 4), (1.0, 2.0),\n acceleration=(0.0, -9.81), state_equation=state_equation)\n\n# 3D\nrectangular = RectangularShape(particle_spacing, (5, 4, 7), (1.0, 2.0, 3.0), density=1000.0)\n\n\n\n\n\n","category":"method"},{"location":"general/initial_condition/#TrixiParticles.RectangularTank","page":"Initial Condition and Setups","title":"TrixiParticles.RectangularTank","text":"RectangularTank(particle_spacing, fluid_size, tank_size, fluid_density;\n velocity=zeros(length(fluid_size)), fluid_mass=nothing,\n pressure=0.0,\n acceleration=nothing, state_equation=nothing,\n boundary_density=fluid_density,\n n_layers=1, spacing_ratio=1.0,\n min_coordinates=zeros(length(fluid_size)),\n faces=Tuple(trues(2 * length(fluid_size))))\n\nRectangular tank filled with a fluid to set up dam-break-style simulations.\n\nArguments\n\nparticle_spacing: Spacing between the fluid particles.\nfluid_size: The dimensions of the fluid as (x, y) (or (x, y, z) in 3D).\ntank_size: The dimensions of the tank as (x, y) (or (x, y, z) in 3D).\nfluid_density: The rest density of the fluid. Will only be used as default for boundary_density when using a state equation.\n\nKeywords\n\nvelocity: Either a function mapping each particle's coordinates to its velocity, or, for a constant fluid velocity, a vector holding this velocity. Velocity is constant zero by default.\nfluid_mass: Either nothing (default) to automatically compute particle mass from particle density and spacing, or a function mapping each particle's coordinates to its mass, or a scalar for a constant mass over all particles.\npressure: Scalar to set the pressure of all particles to this value. This is only used by the EntropicallyDampedSPHSystem and will be overwritten when using an initial pressure function in the system. Cannot be used together with hydrostatic pressure gradient.\nacceleration: In order to initialize particles with a hydrostatic pressure gradient, an acceleration vector can be passed. Note that only accelerations in one coordinate direction and no diagonal accelerations are supported. This will only change the pressure of the particles. When using the WeaklyCompressibleSPHSystem, pass a state_equation as well to initialize the particles with the corresponding density and mass. When using the EntropicallyDampedSPHSystem, the pressure will be overwritten when using an initial pressure function in the system. This cannot be used together with the pressure keyword argument.\nstate_equation: When calculating a hydrostatic pressure gradient by setting acceleration, the state_equation will be used to set the corresponding density. Cannot be used together with density.\nboundary_density: Density of each boundary particle (by default set to the fluid density)\nn_layers: Number of boundary layers.\nspacing_ratio: Ratio of particle_spacing to boundary particle spacing. A value of 2 means that the boundary particle spacing will be half the fluid particle spacing.\nmin_coordinates: Coordinates of the corner in negative coordinate directions.\nfaces: By default all faces are generated. Set faces by passing a bit-array of length 4 (2D) or 6 (3D) to generate the faces in the normal direction: -x,+x,-y,+y,-z,+z.\n\nFields\n\nfluid::InitialCondition: InitialCondition for the fluid.\nboundary::InitialCondition: InitialCondition for the boundary.\nfluid_size::Tuple: Tuple containing the size of the fluid in each dimension after rounding.\ntank_size::Tuple: Tuple containing the size of the tank in each dimension after rounding.\n\nExamples\n\n# 2D\nsetup = RectangularTank(particle_spacing, (water_width, water_height),\n (container_width, container_height), fluid_density,\n n_layers=2, spacing_ratio=3)\n\n# 2D with hydrostatic pressure gradient.\n# `state_equation` has to be the same as for the WCSPH system.\nstate_equation = StateEquationCole(sound_speed=10.0, exponent=1, reference_density=1000.0)\nsetup = RectangularTank(particle_spacing, (water_width, water_height),\n (container_width, container_height), fluid_density,\n acceleration=(0.0, -9.81), state_equation=state_equation)\n\n# 3D\nsetup = RectangularTank(particle_spacing, (water_width, water_height, water_depth),\n (container_width, container_height, container_depth), fluid_density,\n n_layers=2)\n\nSee also: reset_wall!.\n\n\n\n\n\n","category":"type"},{"location":"general/initial_condition/#TrixiParticles.reset_wall!-Tuple{Any, Any, Any}","page":"Initial Condition and Setups","title":"TrixiParticles.reset_wall!","text":"reset_wall!(rectangular_tank::RectangularTank, reset_faces, positions)\n\nThe selected walls of the tank will be placed at the new positions.\n\nArguments\n\nreset_faces: Boolean tuple of 4 (in 2D) or 6 (in 3D) dimensions, similar to faces in RectangularTank.\npositions: Tuple of new positions\n\nwarning: Warning\nThere are overlapping particles when adjacent walls are moved inwards simultaneously.\n\n\n\n\n\n","category":"method"},{"location":"general/initial_condition/#TrixiParticles.RoundSphere","page":"Initial Condition and Setups","title":"TrixiParticles.RoundSphere","text":"RoundSphere(; start_angle=0.0, end_angle=2π)\n\nConstruct a sphere (or sphere segment) by nesting perfectly round concentric spheres. The resulting ball will be perfectly round, but will not have a regular inner structure.\n\nKeywords\n\nstart_angle: The starting angle of the sphere segment in radians. It determines the beginning point of the segment. The default is set to 0.0 representing the positive x-axis.\nend_angle: The ending angle of the sphere segment in radians. It defines the termination point of the segment. The default is set to 2pi, completing a full sphere.\n\nnote: Usage\nSee SphereShape on how to use this.\n\nwarning: Warning\nThe sphere segment is intended for 2D geometries and hollow spheres. If used for filled spheres or in a 3D context, results may not be accurate.\n\n\n\n\n\n","category":"type"},{"location":"general/initial_condition/#TrixiParticles.VoxelSphere","page":"Initial Condition and Setups","title":"TrixiParticles.VoxelSphere","text":"VoxelSphere()\n\nConstruct a sphere of voxels (where particles are placed in the voxel center) with a regular inner structure but corners on the surface. Essentially, a grid of particles is generated and all particles outside the sphere are removed. The resulting sphere will have a perfect inner structure, but is not perfectly round, as it will have corners (like a sphere in Minecraft).\n\nnote: Usage\nSee SphereShape on how to use this.\n\n\n\n\n\n","category":"type"},{"location":"general/initial_condition/#TrixiParticles.SphereShape-NTuple{4, Any}","page":"Initial Condition and Setups","title":"TrixiParticles.SphereShape","text":"SphereShape(particle_spacing, radius, center_position, density;\n sphere_type=VoxelSphere(), n_layers=-1, layer_outwards=false,\n cutout_min=(0.0, 0.0), cutout_max=(0.0, 0.0), tlsph=false,\n velocity=zeros(length(center_position)), mass=nothing, pressure=0.0)\n\nGenerate a sphere that is either completely filled (by default) or hollow (by passing n_layers).\n\nWith the sphere type VoxelSphere, a sphere of voxels (where particles are placed in the voxel center) with a regular inner structure but corners on the surface is created. Essentially, a grid of particles is generated and all particles outside the sphere are removed. With the sphere type RoundSphere, a perfectly round sphere with an imperfect inner structure is created.\n\nA cuboid can be cut out of the sphere by specifying the two corners in negative and positive coordinate directions as cutout_min and cutout_max.\n\nArguments\n\nparticle_spacing: Spacing between the particles.\nradius: Radius of the sphere.\ncenter_position: The coordinates of the center of the sphere.\ndensity: Either a function mapping each particle's coordinates to its density, or a scalar for a constant density over all particles.\n\nKeywords\n\nsphere_type: Either VoxelSphere or RoundSphere (see explanation above).\nn_layers: Set to an integer greater than zero to generate a hollow sphere, where the shell consists of n_layers layers.\nlayer_outwards: When set to false (by default), radius is the outer radius of the sphere. When set to true, radius is the inner radius of the sphere. This is only used when n_layers > 0.\ncutout_min: Corner in negative coordinate directions of a cuboid that is to be cut out of the sphere.\ncutout_max: Corner in positive coordinate directions of a cuboid that is to be cut out of the sphere.\ntlsph: With the TotalLagrangianSPHSystem, particles need to be placed on the boundary of the shape and not one particle radius away, as for fluids. When tlsph=true, particles will be placed on the boundary of the shape.\nvelocity: Either a function mapping each particle's coordinates to its velocity, or, for a constant fluid velocity, a vector holding this velocity. Velocity is constant zero by default.\nmass: Either nothing (default) to automatically compute particle mass from particle density and spacing, or a function mapping each particle's coordinates to its mass, or a scalar for a constant mass over all particles.\npressure: Either a function mapping each particle's coordinates to its pressure, or a scalar for a constant pressure over all particles. This is optional and only needed when using the EntropicallyDampedSPHSystem.\n\nExamples\n\n# Filled circle with radius 0.5, center in (0.2, 0.4) and a particle spacing of 0.1\nSphereShape(0.1, 0.5, (0.2, 0.4), 1000.0)\n\n# Same as before, but perfectly round\nSphereShape(0.1, 0.5, (0.2, 0.4), 1000.0, sphere_type=RoundSphere())\n\n# Hollow circle with ~3 layers, outer radius 0.5, center in (0.2, 0.4) and a particle\n# spacing of 0.1.\nSphereShape(0.1, 0.5, (0.2, 0.4), 1000.0, n_layers=3)\n\n# Same as before, but perfectly round\nSphereShape(0.1, 0.5, (0.2, 0.4), 1000.0, n_layers=3, sphere_type=RoundSphere())\n\n# Hollow circle with 3 layers, inner radius 0.5, center in (0.2, 0.4) and a particle spacing\n# of 0.1.\nSphereShape(0.1, 0.5, (0.2, 0.4), 1000.0, n_layers=3, layer_outwards=true)\n\n# Filled circle with radius 0.1, center in (0.0, 0.0), particle spacing 0.1, but the\n# rectangle [0, 1] x [-0.2, 0.2] is cut out.\nSphereShape(0.1, 1.0, (0.0, 0.0), 1000.0, cutout_min=(0.0, -0.2), cutout_max=(1.0, 0.2))\n\n# Filled 3D sphere with radius 0.5, center in (0.2, 0.4, 0.3) and a particle spacing of 0.1\nSphereShape(0.1, 0.5, (0.2, 0.4, 0.3), 1000.0)\n\n# Same as before, but perfectly round\nSphereShape(0.1, 0.5, (0.2, 0.4, 0.3), 1000.0, sphere_type=RoundSphere())\n\n\n\n\n\n","category":"method"},{"location":"systems/weakly_compressible_sph/#wcsph","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"Weakly compressible SPH as introduced by Monaghan (1994). This formulation relies on a stiff equation of state that generates large pressure changes for small density variations.","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"fluid\", \"weakly_compressible_sph\", \"system.jl\")]","category":"page"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.WeaklyCompressibleSPHSystem","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.WeaklyCompressibleSPHSystem","text":"WeaklyCompressibleSPHSystem(initial_condition,\n density_calculator, state_equation,\n smoothing_kernel, smoothing_length;\n viscosity=nothing, density_diffusion=nothing,\n acceleration=ntuple(_ -> 0.0, NDIMS),\n buffer_size=nothing,\n correction=nothing, source_terms=nothing)\n\nSystem for particles of a fluid. The weakly compressible SPH (WCSPH) scheme is used, wherein a stiff equation of state generates large pressure changes for small density variations. See Weakly Compressible SPH for more details on the method.\n\nArguments\n\ninitial_condition: InitialCondition representing the system's particles.\ndensity_calculator: Density calculator for the system. See ContinuityDensity and SummationDensity.\nstate_equation: Equation of state for the system. See StateEquationCole.\nsmoothing_kernel: Smoothing kernel to be used for this system. See Smoothing Kernels.\nsmoothing_length: Smoothing length to be used for this system. See Smoothing Kernels.\n\nKeyword Arguments\n\nviscosity: Viscosity model for this system (default: no viscosity). See ArtificialViscosityMonaghan or ViscosityAdami.\ndensity_diffusion: Density diffusion terms for this system. See DensityDiffusion.\nacceleration: Acceleration vector for the system. (default: zero vector)\nbuffer_size: Number of buffer particles. This is needed when simulating with OpenBoundarySPHSystem.\ncorrection: Correction method used for this system. (default: no correction, see Corrections)\nsource_terms: Additional source terms for this system. Has to be either nothing (by default), or a function of (coords, velocity, density, pressure, t) (which are the quantities of a single particle), returning a Tuple or SVector that is to be added to the acceleration of that particle. See, for example, SourceTermDamping. Note that these source terms will not be used in the calculation of the boundary pressure when using a boundary with BoundaryModelDummyParticles and AdamiPressureExtrapolation. The keyword argument acceleration should be used instead for gravity-like source terms.\nsurface_tension: Surface tension model used for this SPH system. (default: no surface tension)\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#equation_of_state","page":"Weakly Compressible SPH (Fluid)","title":"Equation of State","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"The equation of state is used to relate fluid density to pressure and thus allow an explicit simulation of the WCSPH system. The equation in the following formulation was introduced by Cole (1948) (pp. 39 and 43). The pressure p is calculated as","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":" p = B left(left(fracrhorho_0right)^gamma - 1right) + p_textbackground","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"where rho denotes the density, rho_0 the reference density, and p_textbackground the background pressure, which is set to zero when applied to free-surface flows (Adami et al., 2012).","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"The bulk modulus, B = fracrho_0 c^2gamma, is calculated from the artificial speed of sound c and the isentropic exponent gamma.","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"An ideal gas equation of state with a linear relationship between pressure and density can be obtained by choosing exponent=1, i.e.","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":" p = B left( fracrhorho_0 -1 right) = c^2(rho - rho_0)","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"For higher Reynolds numbers, exponent=7 is recommended, whereas at lower Reynolds numbers exponent=1 yields more accurate pressure estimates since pressure and density are proportional (see Morris, 1997).","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"When using SummationDensity (or DensityReinitializationCallback) and free surfaces, initializing particles with equal spacing will cause underestimated density and therefore strong attractive forces between particles at the free surface. Setting clip_negative_pressure=true can avoid this.","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"fluid\", \"weakly_compressible_sph\", \"state_equations.jl\")]","category":"page"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.StateEquationCole","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.StateEquationCole","text":"StateEquationCole(; sound_speed, reference_density, exponent,\n background_pressure=0.0, clip_negative_pressure=false)\n\nEquation of state to describe the relationship between pressure and density of water up to high pressures.\n\nKeywords\n\nsound_speed: Artificial speed of sound.\nreference_density: Reference density of the fluid.\nexponent: A value of 7 is usually used for most simulations.\nbackground_pressure=0.0: Background pressure.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#viscosity_wcsph","page":"Weakly Compressible SPH (Fluid)","title":"Viscosity","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"TODO: Explain viscosity.","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"fluid\", \"viscosity.jl\")]","category":"page"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.ArtificialViscosityMonaghan","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.ArtificialViscosityMonaghan","text":"ArtificialViscosityMonaghan(; alpha, beta=0.0, epsilon=0.01)\n\nArtificial viscosity by Monaghan ([16], [17]), given by\n\nPi_ab =\nbegincases\n -(alpha c mu_ab + beta mu_ab^2) barrho_ab textif v_ab cdot r_ab 0 \n 0 textotherwise\nendcases\n\nwith\n\nmu_ab = frach v_ab cdot r_abVert r_ab Vert^2 + epsilon h^2\n\nwhere alpha beta epsilon are parameters, c is the speed of sound, h is the smoothing length, r_ab = r_a - r_b is the difference of the coordinates of particles a and b, v_ab = v_a - v_b is the difference of their velocities, and barrho_ab is the arithmetic mean of their densities.\n\nNote that alpha needs to adjusted for different resolutions to maintain a specific Reynolds Number. To do so, Monaghan (2005) defined an equivalent effective physical kinematic viscosity nu by\n\n nu = fracalpha h c 2d + 4\n\nwhere d is the dimension.\n\nKeywords\n\nalpha: A value of 0.02 is usually used for most simulations. For a relation with the kinematic viscosity, see description above.\nbeta=0.0: A value of 0.0 works well for most fluid simulations and simulations with shocks of moderate strength. In simulations where the Mach number can be very high, eg. astrophysical calculation, good results can be obtained by choosing a value of beta=2.0 and alpha=1.0.\nepsilon=0.01: Parameter to prevent singularities.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.ViscosityAdami","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.ViscosityAdami","text":"ViscosityAdami(; nu, epsilon=0.01)\n\nViscosity by Adami (2012). The viscous interaction is calculated with the shear force for incompressible flows given by\n\nf_ab = sum_w bareta_ab left( V_a^2 + V_b^2 right) fracv_abr_ab^2+epsilon h_ab^2 nabla W_ab cdot r_ab\n\nwhere r_ab = r_a - r_b is the difference of the coordinates of particles a and b, v_ab = v_a - v_b is the difference of their velocities, h is the smoothing length and V is the particle volume. The parameter epsilon prevents singularities (see Ramachandran (2019)). The inter-particle-averaged shear stress is\n\n bareta_ab =frac2 eta_a eta_beta_a + eta_b\n\nwhere eta_a = rho_a nu_a with nu as the kinematic viscosity.\n\nKeywords\n\nnu: Kinematic viscosity\nepsilon=0.01: Parameter to prevent singularities\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.ViscosityMorris","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.ViscosityMorris","text":"ViscosityMorris(; nu, epsilon=0.01)\n\nViscosity by Morris (1997) also used by Fourtakas (2019).\n\nTo the force f_ab between two particles a and b due to pressure gradients, an additional force term tildef_ab is added with\n\ntildef_ab = m_a m_b frac(mu_a + mu_b) r_ab cdot nabla W_abrho_a rho_b (Vert r_ab Vert^2 + epsilon h^2) v_ab\n\nwhere mu_a = rho_a nu and mu_b = rho_b nu denote the dynamic viscosity of particle a and b respectively, and nu is the kinematic viscosity.\n\nKeywords\n\nnu: Kinematic viscosity\nepsilon=0.01: Parameter to prevent singularities\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#Density-Diffusion","page":"Weakly Compressible SPH (Fluid)","title":"Density Diffusion","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"Density diffusion can be used with ContinuityDensity to remove the noise in the pressure field. It is highly recommended to use density diffusion when using WCSPH.","category":"page"},{"location":"systems/weakly_compressible_sph/#Formulation","page":"Weakly Compressible SPH (Fluid)","title":"Formulation","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"All density diffusion terms extend the continuity equation (see ContinuityDensity) by an additional term","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"fracmathrmdrho_amathrmdt = sum_b m_b v_ab cdot nabla_r_a W(Vert r_ab Vert h)\n + delta h c sum_b V_b psi_ab cdot nabla_r_a W(Vert r_ab Vert h)","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"where V_b = m_b rho_b is the volume of particle b and psi_ab depends on the density diffusion method (see DensityDiffusion for available terms). Also, rho_a denotes the density of particle a and r_ab = r_a - r_b is the difference of the coordinates, v_ab = v_a - v_b of the velocities of particles a and b.","category":"page"},{"location":"systems/weakly_compressible_sph/#Numerical-Results","page":"Weakly Compressible SPH (Fluid)","title":"Numerical Results","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"All density diffusion terms remove numerical noise in the pressure field and produce more accurate results than weakly commpressible SPH without density diffusion. This can be demonstrated with dam break examples in 2D and 3D. Here, δ = 01 has been used for all terms. Note that, due to added stability, the adaptive time integration method that was used here can choose higher time steps in the simulations with density diffusion. For the cheap DensityDiffusionMolteniColagrossi, this results in reduced runtime.","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"
\n \"density_diffusion_2d\"/\n
Dam break in 2D with different density diffusion terms
\n
","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"
\n \"density_diffusion_3d\"/\n
Dam break in 3D with different density diffusion terms
\n
","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"The simpler terms DensityDiffusionMolteniColagrossi and DensityDiffusionFerrari do not solve the hydrostatic problem and lead to incorrect solutions in long-running steady-state hydrostatic simulations with free surfaces (Antuono et al., 2012). This can be seen when running the simple rectangular tank example until t = 40 (again using δ = 01):","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"
\n \"density_diffusion_tank\"/\n
Tank in rest under gravity in 3D with different density diffusion terms
\n
","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"DensityDiffusionAntuono adds a correction term to solve this problem, but this term is very expensive and adds about 40–50% of computational cost.","category":"page"},{"location":"systems/weakly_compressible_sph/#API","page":"Weakly Compressible SPH (Fluid)","title":"API","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"fluid\", \"weakly_compressible_sph\", \"density_diffusion.jl\")]","category":"page"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.DensityDiffusion","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.DensityDiffusion","text":"DensityDiffusion\n\nAn abstract supertype of all density diffusion formulations.\n\nCurrently, the following formulations are available:\n\nFormulation Suitable for Steady-State Simulations Low Computational Cost\nDensityDiffusionMolteniColagrossi ❌ ✅\nDensityDiffusionFerrari ❌ ✅\nDensityDiffusionAntuono ✅ ❌\n\nSee Density Diffusion for a comparison and more details.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.DensityDiffusionAntuono","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.DensityDiffusionAntuono","text":"DensityDiffusionAntuono(initial_condition; delta)\n\nThe commonly used density diffusion terms by Antuono (2010), also referred to as δ-SPH. The density diffusion term by Molteni (2009) is extended by a second term, which is nicely written down by Antuono (2012).\n\nThe term psi_ab in the continuity equation in DensityDiffusion is defined by\n\npsi_ab = 2left(rho_a - rho_b - frac12big(nablarho^L_a + nablarho^L_bbig) cdot r_abright)\n fracr_abVert r_ab Vert^2\n\nwhere rho_a and rho_b denote the densities of particles a and b respectively and r_ab = r_a - r_b is the difference of the coordinates of particles a and b. The symbol nablarho^L_a denotes the renormalized density gradient defined as\n\nnablarho^L_a = -sum_b (rho_a - rho_b) V_b L_a nabla_r_a W(Vert r_ab Vert h)\n\nwith\n\nL_a = left( -sum_b V_b r_ab otimes nabla_r_a W(Vert r_ab Vert h) right)^-1 in R^d times d\n\nwhere d is the number of dimensions.\n\nSee DensityDiffusion for an overview and comparison of implemented density diffusion terms.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.DensityDiffusionFerrari","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.DensityDiffusionFerrari","text":"DensityDiffusionFerrari()\n\nA density diffusion term by Ferrari (2009).\n\nThe term psi_ab in the continuity equation in DensityDiffusion is defined by\n\npsi_ab = fracrho_a - rho_b2h fracr_abVert r_ab Vert\n\nwhere rho_a and rho_b denote the densities of particles a and b respectively, r_ab = r_a - r_b is the difference of the coordinates of particles a and b and h is the smoothing length.\n\nSee DensityDiffusion for an overview and comparison of implemented density diffusion terms.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.DensityDiffusionMolteniColagrossi","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.DensityDiffusionMolteniColagrossi","text":"DensityDiffusionMolteniColagrossi(; delta)\n\nThe commonly used density diffusion term by Molteni (2009).\n\nThe term psi_ab in the continuity equation in DensityDiffusion is defined by\n\npsi_ab = 2(rho_a - rho_b) fracr_abVert r_ab Vert^2\n\nwhere rho_a and rho_b denote the densities of particles a and b respectively and r_ab = r_a - r_b is the difference of the coordinates of particles a and b.\n\nSee DensityDiffusion for an overview and comparison of implemented density diffusion terms.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#corrections","page":"Weakly Compressible SPH (Fluid)","title":"Corrections","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"general\", \"corrections.jl\")]","category":"page"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.AkinciFreeSurfaceCorrection","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.AkinciFreeSurfaceCorrection","text":"AkinciFreeSurfaceCorrection(rho0)\n\nFree surface correction according to Akinci et al. (2013). At a free surface, the mean density is typically lower than the reference density, resulting in reduced surface tension and viscosity forces. The free surface correction adjusts the viscosity, pressure, and surface tension forces near free surfaces to counter this effect. It's important to note that this correlation is unphysical and serves as an approximation. The computation time added by this method is about 2–3%.\n\nMathematically the idea is quite simple. If we have an SPH particle in the middle of a volume at rest, its density will be identical to the rest density rho_0. If we now consider an SPH particle at a free surface at rest, it will have neighbors missing in the direction normal to the surface, which will result in a lower density. If we calculate the correction factor\n\nk = rho_0rho_textmean\n\nthis value will be about ~1.5 for particles at the free surface and can then be used to increase the pressure and viscosity accordingly.\n\nArguments\n\nrho0: Rest density.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.BlendedGradientCorrection","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.BlendedGradientCorrection","text":"BlendedGradientCorrection()\n\nCalculate a blended gradient to reduce the stability issues of the GradientCorrection as explained by Bonet (1999).\n\nThis calculates the following,\n\ntildenabla A_i = (1-lambda) nabla A_i + lambda L_i nabla A_i\n\nwith 0 leq lambda leq 1 being the blending factor.\n\nArguments\n\nblending_factor: Blending factor between corrected and regular SPH gradient.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.GradientCorrection","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.GradientCorrection","text":"GradientCorrection()\n\nCompute the corrected gradient of particle interactions based on their relative positions (see Bonet, 1999).\n\nMathematical Details\n\nGiven the standard SPH representation, the gradient of a field A at particle a is given by\n\nnabla A_a = sum_b m_b fracA_b - A_arho_b nabla_r_a W(Vert r_a - r_b Vert h)\n\nwhere m_b is the mass of particle b and rho_b is the density of particle b.\n\nThe gradient correction, as commonly proposed, involves multiplying this gradient with a correction matrix L:\n\ntildenabla A_a = bmL_a nabla A_a\n\nThe correction matrix bmL_a is computed based on the provided particle configuration, aiming to make the corrected gradient more accurate, especially near domain boundaries.\n\nTo satisfy\n\nsum_b V_b r_ba otimes tildenablaW_b(r_a) = left( sum_b V_b r_ba otimes nabla W_b(r_a) right) bmL_a^T = bmI\n\nthe correction matrix bmL_a is evaluated explicitly as\n\nbmL_a = left( sum_b V_b nabla W_b(r_a) otimes r_ba right)^-1\n\nnote: Note\nStability issues arise, especially when particles separate into small clusters.\nDoubles the computational effort.\nBetter stability with smoother smoothing Kernels with larger support, e.g. SchoenbergQuinticSplineKernel or WendlandC6Kernel.\nSet dt_max =< 1e-3 for stability.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.KernelCorrection","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.KernelCorrection","text":"KernelCorrection()\n\nKernel correction, as explained by Bonet (1999), uses Shepard interpolation to obtain a 0-th order accurate result, which was first proposed by Li et al. This can be further extended to obtain a kernel corrected gradient as shown by Basa et al. (2008).\n\nThe kernel correction coefficient is determined by\n\nc(x) = sum_b=1 V_b W_b(x)\n\nThe gradient of corrected kernel is determined by\n\nnabla tildeW_b(r) =fracnabla W_b(r) - W_b(r) gamma(r)sum_b=1 V_b W_b(r) quad textwhere quad\ngamma(r) = fracsum_b=1 V_b nabla W_b(r)sum_b=1 V_b W_b(r)\n\nThis correction can be applied with SummationDensity and ContinuityDensity, which leads to an improvement, especially at free surfaces.\n\nnote: Note\nThis only works when the boundary model uses SummationDensity (yet).\nIt is also referred to as \"0th order correction\".\nIn 2D, we can expect an increase of about 10–15% in computation time.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.MixedKernelGradientCorrection","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.MixedKernelGradientCorrection","text":"MixedKernelGradientCorrection()\n\nCombines GradientCorrection and KernelCorrection, which results in a 1st-order-accurate SPH method (see Bonet, 1999).\n\nNotes:\n\nStability issues, especially when particles separate into small clusters.\nDoubles the computational effort.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.ShepardKernelCorrection","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.ShepardKernelCorrection","text":"ShepardKernelCorrection()\n\nKernel correction, as explained by Bonet (1999), uses Shepard interpolation to obtain a 0-th order accurate result, which was first proposed by Li et al. (1996).\n\nThe kernel correction coefficient is determined by\n\nc(x) = sum_b=1 V_b W_b(x)\n\nwhere V_b = m_b rho_b is the volume of particle b.\n\nThis correction is applied with SummationDensity to correct the density and leads to an improvement, especially at free surfaces.\n\nnote: Note\nIt is also referred to as \"0th order correction\".\nIn 2D, we can expect an increase of about 5–6% in computation time.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#surface_tension","page":"Weakly Compressible SPH (Fluid)","title":"Surface Tension","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/#Akinci-based-intra-particle-force-surface-tension-and-wall-adhesion-model","page":"Weakly Compressible SPH (Fluid)","title":"Akinci-based intra-particle force surface tension and wall adhesion model","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"The work by Akinci proposes three forces:","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"a cohesion force\na surface area minimization force\na wall adhesion force","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"The classical model is composed of the curvature minimization and cohesion force.","category":"page"},{"location":"systems/weakly_compressible_sph/#Cohesion-force","page":"Weakly Compressible SPH (Fluid)","title":"Cohesion force","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"The model calculates the cohesion force based on the distance between particles and the support radius h_c. This force is determined using two distinct regimes within the support radius:","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"For particles closer than half the support radius, a repulsive force is calculated to prevent particle clustering too tightly, enhancing the simulation's stability and realism.\nBeyond half the support radius and within the full support radius, an attractive force is computed, simulating the effects of surface tension that draw particles together.","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"The cohesion force, F_textcohesion, for a pair of particles is given by:","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"F_textcohesion = -sigma m_b C(r) fracrVert r Vert","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"where:","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"sigma represents the surface tension coefficient, adjusting the overall strength of the cohesion effect.\nC is a scalar function of the distance between particles.","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"The cohesion kernel C is defined as","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"C(r)=frac32pi h_c^9\nbegincases\n(h_c-r)^3 r^3 textif 2r h_c \n2(h_c-r)^3 r^3 - frach^664 textif r 0 text and 2r leq h_c \n0 textotherwise\nendcases","category":"page"},{"location":"systems/weakly_compressible_sph/#Surface-area-minimization-force","page":"Weakly Compressible SPH (Fluid)","title":"Surface area minimization force","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"To model the minimization of the surface area and curvature of the fluid, a curvature force is used, which is calculated as","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"F_textcurvature = -sigma (n_a - n_b)","category":"page"},{"location":"systems/weakly_compressible_sph/#Wall-adhesion-force","page":"Weakly Compressible SPH (Fluid)","title":"Wall adhesion force","text":"","category":"section"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"The wall adhesion model proposed by Akinci et al. is based on a kernel function which is 0 from 0.0 to 0.5 support radiia with a maximum at 0.75. With the force calculated with an adhesion coefficient beta as","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"F_textadhesion = -beta m_b A(r) fracrVert r Vert","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"with A being the adhesion kernel defined as","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"A(r)= frac0007h_c^325\nbegincases\nsqrt4- frac4r^2h_c + 6r - 2h_c textif 2r h_c text and r leq h_c \n0 textotherwise\nendcases","category":"page"},{"location":"systems/weakly_compressible_sph/","page":"Weakly Compressible SPH (Fluid)","title":"Weakly Compressible SPH (Fluid)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"fluid\", \"surface_tension.jl\")]","category":"page"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.CohesionForceAkinci","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.CohesionForceAkinci","text":"CohesionForceAkinci(surface_tension_coefficient=1.0)\n\nThis model only implements the cohesion force of the [25] surface tension model.\n\nKeywords\n\nsurface_tension_coefficient=1.0: Modifies the intensity of the surface tension-induced force, enabling the tuning of the fluid's surface tension properties within the simulation.\n\n\n\n\n\n","category":"type"},{"location":"systems/weakly_compressible_sph/#TrixiParticles.SurfaceTensionAkinci","page":"Weakly Compressible SPH (Fluid)","title":"TrixiParticles.SurfaceTensionAkinci","text":"SurfaceTensionAkinci(surface_tension_coefficient=1.0)\n\nImplements a model for surface tension and adhesion effects drawing upon the principles outlined by [25]. This model is instrumental in capturing the nuanced behaviors of fluid surfaces, such as droplet formation and the dynamics of merging or separation, by utilizing intra-particle forces.\n\nKeywords\n\nsurface_tension_coefficient=1.0: A parameter to adjust the magnitude of surface tension forces, facilitating the fine-tuning of how surface tension phenomena are represented in the simulation.\n\n\n\n\n\n","category":"type"},{"location":"tutorials/tut_dam_break/","page":"Example file","title":"Example file","text":"EditURL = \"https://github.com/trixi-framework/TrixiParticles.jl/blob/main/docs/src/tutorials_template/tut_dam_break.md\"","category":"page"},{"location":"tutorials/tut_dam_break/#Example-file","page":"Example file","title":"Example file","text":"","category":"section"},{"location":"tutorials/tut_dam_break/","page":"Example file","title":"Example file","text":"# 2D dam break simulation based on\n#\n# S. Marrone, M. Antuono, A. Colagrossi, G. Colicchio, D. le Touzé, G. Graziani.\n# \"δ-SPH model for simulating violent impact flows\".\n# In: Computer Methods in Applied Mechanics and Engineering, Volume 200, Issues 13–16 (2011), pages 1526–1542.\n# https://doi.org/10.1016/J.CMA.2010.12.016\n\nusing TrixiParticles\nusing OrdinaryDiffEq\n\n# Size parameters\nH = 0.6\nW = 2 * H\n\n# ==========================================================================================\n# ==== Resolution\nfluid_particle_spacing = H / 40\n\n# Change spacing ratio to 3 and boundary layers to 1 when using Monaghan-Kajtar boundary model\nboundary_layers = 4\nspacing_ratio = 1\n\nboundary_particle_spacing = fluid_particle_spacing / spacing_ratio\n\n# ==========================================================================================\n# ==== Experiment Setup\ngravity = 9.81\n\ntspan = (0.0, 5.7 / sqrt(gravity))\n\n# Boundary geometry and initial fluid particle positions\ninitial_fluid_size = (W, H)\ntank_size = (floor(5.366 * H / boundary_particle_spacing) * boundary_particle_spacing, 4.0)\n\nfluid_density = 1000.0\nsound_speed = 20 * sqrt(gravity * H)\nstate_equation = StateEquationCole(; sound_speed, reference_density=fluid_density,\n exponent=1, clip_negative_pressure=false)\n\ntank = RectangularTank(fluid_particle_spacing, initial_fluid_size, tank_size, fluid_density,\n n_layers=boundary_layers, spacing_ratio=spacing_ratio,\n acceleration=(0.0, -gravity), state_equation=state_equation)\n\n# ==========================================================================================\n# ==== Fluid\nsmoothing_length = 3.5 * fluid_particle_spacing\nsmoothing_kernel = WendlandC2Kernel{2}()\n\nfluid_density_calculator = ContinuityDensity()\nviscosity = ArtificialViscosityMonaghan(alpha=0.02, beta=0.0)\n# nu = 0.02 * smoothing_length * sound_speed/8\n# viscosity = ViscosityMorris(nu=nu)\n# viscosity = ViscosityAdami(nu=nu)\n# Alternatively the density diffusion model by Molteni & Colagrossi can be used,\n# which will run faster.\n# density_diffusion = DensityDiffusionMolteniColagrossi(delta=0.1)\ndensity_diffusion = DensityDiffusionAntuono(tank.fluid, delta=0.1)\n\nfluid_system = WeaklyCompressibleSPHSystem(tank.fluid, fluid_density_calculator,\n state_equation, smoothing_kernel,\n smoothing_length, viscosity=viscosity,\n density_diffusion=density_diffusion,\n acceleration=(0.0, -gravity), correction=nothing,\n surface_tension=nothing)\n\n# ==========================================================================================\n# ==== Boundary\nboundary_density_calculator = AdamiPressureExtrapolation()\nboundary_model = BoundaryModelDummyParticles(tank.boundary.density, tank.boundary.mass,\n state_equation=state_equation,\n boundary_density_calculator,\n smoothing_kernel, smoothing_length,\n correction=nothing)\n\nboundary_system = BoundarySPHSystem(tank.boundary, boundary_model, adhesion_coefficient=0.0)\n\n# ==========================================================================================\n# ==== Simulation\n# `nothing` will automatically choose the best update strategy. This is only to be able\n# to change this with `trixi_include`.\nsemi = Semidiscretization(fluid_system, boundary_system,\n neighborhood_search=GridNeighborhoodSearch{2}(update_strategy=nothing))\node = semidiscretize(semi, tspan, data_type=nothing)\n\ninfo_callback = InfoCallback(interval=100)\n\nsolution_prefix = \"\"\nsaving_callback = SolutionSavingCallback(dt=0.02, prefix=solution_prefix)\n\n# Save at certain timepoints which allows comparison to the results of Marrone et al.,\n# i.e. (1.5, 2.36, 3.0, 5.7, 6.45).\n# Please note that the images in Marrone et al. are obtained at a particle_spacing = H/320,\n# which takes between 2 and 4 hours.\nsaving_paper = SolutionSavingCallback(save_times=[0.0, 0.371, 0.584, 0.743, 1.411, 1.597],\n prefix=\"marrone_times\")\n\n# This can be overwritten with `trixi_include`\nextra_callback = nothing\n\nuse_reinit = false\ndensity_reinit_cb = use_reinit ?\n DensityReinitializationCallback(semi.systems[1], interval=10) :\n nothing\nstepsize_callback = StepsizeCallback(cfl=0.9)\n\ncallbacks = CallbackSet(info_callback, saving_callback, stepsize_callback, extra_callback,\n density_reinit_cb, saving_paper)\n\nsol = solve(ode, CarpenterKennedy2N54(williamson_condition=false),\n dt=1.0, # This is overwritten by the stepsize callback\n save_everystep=false, callback=callbacks);\n\n","category":"page"},{"location":"general/neighborhood_search/#Neighborhood-Search","page":"Neighborhood Search","title":"Neighborhood Search","text":"","category":"section"},{"location":"general/neighborhood_search/","page":"Neighborhood Search","title":"Neighborhood Search","text":"The neighborhood search is the most essential component for performance. We provide several implementations in the package PointNeighbors.jl. See the docs of this package for an overview and a comparison of different implementations.","category":"page"},{"location":"general/neighborhood_search/","page":"Neighborhood Search","title":"Neighborhood Search","text":"note: Usage\nTo run a simulation with a neighborhood search implementation, pass a template of the neighborhood search to the constructor of the Semidiscretization. A template is just an empty neighborhood search with search radius 0.0. See copy_neighborhood_search and the examples below for more details.semi = Semidiscretization(system1, system2,\n neighborhood_search=PrecomputedNeighborhoodSearch{2}())The keyword argument periodic_box in the neighborhood search constructors can be used to define a periodic domain. See the PointNeighbors.jl docs for more details.periodic_box = PeriodicBox(min_corner=[0.0, -0.25], max_corner=[1.0, 0.75])\nsemi = Semidiscretization(system1, system2,\n neighborhood_search=GridNeighborhoodSearch{2}(; periodic_box))","category":"page"},{"location":"tutorials/tut_falling/","page":"Example file","title":"Example file","text":"EditURL = \"https://github.com/trixi-framework/TrixiParticles.jl/blob/main/docs/src/tutorials_template/tut_falling.md\"","category":"page"},{"location":"tutorials/tut_falling/#Example-file","page":"Example file","title":"Example file","text":"","category":"section"},{"location":"tutorials/tut_falling/","page":"Example file","title":"Example file","text":"using TrixiParticles\nusing OrdinaryDiffEq\n\n# ==========================================================================================\n# ==== Resolution\nfluid_particle_spacing = 0.02\nsolid_particle_spacing = fluid_particle_spacing\n\n# Change spacing ratio to 3 and boundary layers to 1 when using Monaghan-Kajtar boundary model\nboundary_layers = 3\nspacing_ratio = 1\n\n# ==========================================================================================\n# ==== Experiment Setup\ngravity = 9.81\ntspan = (0.0, 1.0)\n\n# Boundary geometry and initial fluid particle positions\ninitial_fluid_size = (2.0, 0.9)\ntank_size = (2.0, 1.0)\n\nfluid_density = 1000.0\nsound_speed = 10 * sqrt(gravity * initial_fluid_size[2])\nstate_equation = StateEquationCole(; sound_speed, reference_density=fluid_density,\n exponent=1)\n\ntank = RectangularTank(fluid_particle_spacing, initial_fluid_size, tank_size, fluid_density,\n n_layers=boundary_layers, spacing_ratio=spacing_ratio,\n faces=(true, true, true, false),\n acceleration=(0.0, -gravity), state_equation=state_equation)\n\nsphere1_radius = 0.3\nsphere2_radius = 0.2\nsphere1_density = 500.0\nsphere2_density = 1100.0\n\n# Young's modulus and Poisson ratio\nsphere1_E = 7e4\nsphere2_E = 1e5\nnu = 0.0\n\nsphere1_center = (0.5, 1.6)\nsphere2_center = (1.5, 1.6)\nsphere1 = SphereShape(solid_particle_spacing, sphere1_radius, sphere1_center,\n sphere1_density, sphere_type=VoxelSphere())\nsphere2 = SphereShape(solid_particle_spacing, sphere2_radius, sphere2_center,\n sphere2_density, sphere_type=VoxelSphere())\n\n# ==========================================================================================\n# ==== Fluid\nfluid_smoothing_length = 3.0 * fluid_particle_spacing\nfluid_smoothing_kernel = WendlandC2Kernel{2}()\n\nfluid_density_calculator = ContinuityDensity()\nviscosity = ArtificialViscosityMonaghan(alpha=0.02, beta=0.0)\ndensity_diffusion = DensityDiffusionMolteniColagrossi(delta=0.1)\n\nfluid_system = WeaklyCompressibleSPHSystem(tank.fluid, fluid_density_calculator,\n state_equation, fluid_smoothing_kernel,\n fluid_smoothing_length, viscosity=viscosity,\n density_diffusion=density_diffusion,\n acceleration=(0.0, -gravity))\n\n# ==========================================================================================\n# ==== Boundary\nboundary_density_calculator = BernoulliPressureExtrapolation()\nboundary_model = BoundaryModelDummyParticles(tank.boundary.density, tank.boundary.mass,\n state_equation=state_equation,\n boundary_density_calculator,\n fluid_smoothing_kernel, fluid_smoothing_length)\n\nboundary_system = BoundarySPHSystem(tank.boundary, boundary_model)\n\n# ==========================================================================================\n# ==== Solid\nsolid_smoothing_length = 2 * sqrt(2) * solid_particle_spacing\nsolid_smoothing_kernel = WendlandC2Kernel{2}()\n\n# For the FSI we need the hydrodynamic masses and densities in the solid boundary model\nhydrodynamic_densites_1 = fluid_density * ones(size(sphere1.density))\nhydrodynamic_masses_1 = hydrodynamic_densites_1 * solid_particle_spacing^ndims(fluid_system)\n\nsolid_boundary_model_1 = BoundaryModelDummyParticles(hydrodynamic_densites_1,\n hydrodynamic_masses_1,\n state_equation=state_equation,\n boundary_density_calculator,\n fluid_smoothing_kernel,\n fluid_smoothing_length)\n\nhydrodynamic_densites_2 = fluid_density * ones(size(sphere2.density))\nhydrodynamic_masses_2 = hydrodynamic_densites_2 * solid_particle_spacing^ndims(fluid_system)\n\nsolid_boundary_model_2 = BoundaryModelDummyParticles(hydrodynamic_densites_2,\n hydrodynamic_masses_2,\n state_equation=state_equation,\n boundary_density_calculator,\n fluid_smoothing_kernel,\n fluid_smoothing_length)\n\nsolid_system_1 = TotalLagrangianSPHSystem(sphere1,\n solid_smoothing_kernel, solid_smoothing_length,\n sphere1_E, nu,\n acceleration=(0.0, -gravity),\n boundary_model=solid_boundary_model_1,\n penalty_force=PenaltyForceGanzenmueller(alpha=0.3))\n\nsolid_system_2 = TotalLagrangianSPHSystem(sphere2,\n solid_smoothing_kernel, solid_smoothing_length,\n sphere2_E, nu,\n acceleration=(0.0, -gravity),\n boundary_model=solid_boundary_model_2,\n penalty_force=PenaltyForceGanzenmueller(alpha=0.3))\n\n# ==========================================================================================\n# ==== Simulation\nsemi = Semidiscretization(fluid_system, boundary_system, solid_system_1, solid_system_2)\node = semidiscretize(semi, tspan)\n\ninfo_callback = InfoCallback(interval=10)\nsaving_callback = SolutionSavingCallback(dt=0.02, output_directory=\"out\", prefix=\"\",\n write_meta_data=true)\n\ncallbacks = CallbackSet(info_callback, saving_callback)\n\n# Use a Runge-Kutta method with automatic (error based) time step size control.\nsol = solve(ode, RDPK3SpFSAL49(),\n abstol=1e-6, # Default abstol is 1e-6\n reltol=1e-3, # Default reltol is 1e-3\n save_everystep=false, callback=callbacks);\n\n","category":"page"},{"location":"tutorials/tut_beam/","page":"Example file","title":"Example file","text":"EditURL = \"https://github.com/trixi-framework/TrixiParticles.jl/blob/main/docs/src/tutorials_template/tut_beam.md\"","category":"page"},{"location":"tutorials/tut_beam/#Example-file","page":"Example file","title":"Example file","text":"","category":"section"},{"location":"tutorials/tut_beam/","page":"Example file","title":"Example file","text":"using TrixiParticles\nusing OrdinaryDiffEq\n\n# ==========================================================================================\n# ==== Resolution\nn_particles_y = 5\n\n# ==========================================================================================\n# ==== Experiment Setup\ngravity = 2.0\ntspan = (0.0, 5.0)\n\nelastic_beam = (length=0.35, thickness=0.02)\nmaterial = (density=1000.0, E=1.4e6, nu=0.4)\nclamp_radius = 0.05\n\n# The structure starts at the position of the first particle and ends\n# at the position of the last particle.\nparticle_spacing = elastic_beam.thickness / (n_particles_y - 1)\n\n# Add particle_spacing/2 to the clamp_radius to ensure that particles are also placed on the radius\nfixed_particles = SphereShape(particle_spacing, clamp_radius + particle_spacing / 2,\n (0.0, elastic_beam.thickness / 2), material.density,\n cutout_min=(0.0, 0.0),\n cutout_max=(clamp_radius, elastic_beam.thickness),\n tlsph=true)\n\nn_particles_clamp_x = round(Int, clamp_radius / particle_spacing)\n\n# Beam and clamped particles\nn_particles_per_dimension = (round(Int, elastic_beam.length / particle_spacing) +\n n_particles_clamp_x + 1, n_particles_y)\n\n# Note that the `RectangularShape` puts the first particle half a particle spacing away\n# from the boundary, which is correct for fluids, but not for solids.\n# We therefore need to pass `tlsph=true`.\nbeam = RectangularShape(particle_spacing, n_particles_per_dimension,\n (0.0, 0.0), density=material.density, tlsph=true)\n\nsolid = union(beam, fixed_particles)\n\n# ==========================================================================================\n# ==== Solid\n# The kernel in the reference uses a differently scaled smoothing length,\n# so this is equivalent to the smoothing length of `sqrt(2) * particle_spacing` used in the paper.\nsmoothing_length = 2 * sqrt(2) * particle_spacing\nsmoothing_kernel = WendlandC2Kernel{2}()\n\nsolid_system = TotalLagrangianSPHSystem(solid, smoothing_kernel, smoothing_length,\n material.E, material.nu,\n n_fixed_particles=nparticles(fixed_particles),\n acceleration=(0.0, -gravity),\n penalty_force=nothing)\n\n# ==========================================================================================\n# ==== Simulation\nsemi = Semidiscretization(solid_system,\n neighborhood_search=PrecomputedNeighborhoodSearch{2}())\node = semidiscretize(semi, tspan)\n\ninfo_callback = InfoCallback(interval=100)\n\n# Track the position of the particle in the middle of the tip of the beam.\nmiddle_particle_id = Int(n_particles_per_dimension[1] * (n_particles_per_dimension[2] + 1) /\n 2)\nstartposition_x = beam.coordinates[1, middle_particle_id]\nstartposition_y = beam.coordinates[2, middle_particle_id]\n\nfunction deflection_x(v, u, t, system)\n return system.current_coordinates[1, middle_particle_id] - startposition_x\nend\n\nfunction deflection_y(v, u, t, system)\n return system.current_coordinates[2, middle_particle_id] - startposition_y\nend\n\nsaving_callback = SolutionSavingCallback(dt=0.02, prefix=\"\",\n deflection_x=deflection_x,\n deflection_y=deflection_y)\n\ncallbacks = CallbackSet(info_callback, saving_callback)\n\n# Use a Runge-Kutta method with automatic (error based) time step size control\nsol = solve(ode, RDPK3SpFSAL49(), save_everystep=false, callback=callbacks);\n\n","category":"page"},{"location":"general/density_calculators/#density_calculator","page":"Density Calculators","title":"Density Calculators","text":"","category":"section"},{"location":"general/density_calculators/","page":"Density Calculators","title":"Density Calculators","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"general\", \"density_calculators.jl\")]","category":"page"},{"location":"general/density_calculators/#TrixiParticles.ContinuityDensity","page":"Density Calculators","title":"TrixiParticles.ContinuityDensity","text":"ContinuityDensity()\n\nDensity calculator to integrate the density from the continuity equation\n\nfracmathrmdrho_amathrmdt = sum_b m_b v_ab cdot nabla_r_a W(Vert r_a - r_b Vert h)\n\nwhere rho_a denotes the density of particle a and r_ab = r_a - r_b is the difference of the coordinates, v_ab = v_a - v_b of the velocities of particles a and b.\n\n\n\n\n\n","category":"type"},{"location":"general/density_calculators/#TrixiParticles.SummationDensity","page":"Density Calculators","title":"TrixiParticles.SummationDensity","text":"SummationDensity()\n\nDensity calculator to use the summation formula\n\nrho(r) = sum_b m_b W(Vert r - r_b Vert h)\n\nfor the density estimation, where r_b denotes the coordinates and m_b the mass of particle b.\n\n\n\n\n\n","category":"type"},{"location":"general/semidiscretization/#Semidiscretization","page":"Semidiscretization","title":"Semidiscretization","text":"","category":"section"},{"location":"general/semidiscretization/","page":"Semidiscretization","title":"Semidiscretization","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"general\", \"semidiscretization.jl\")]","category":"page"},{"location":"general/semidiscretization/#TrixiParticles.Semidiscretization","page":"Semidiscretization","title":"TrixiParticles.Semidiscretization","text":"Semidiscretization(systems...; neighborhood_search=GridNeighborhoodSearch{NDIMS}())\n\nThe semidiscretization couples the passed systems to one simulation.\n\nArguments\n\nsystems: Systems to be coupled in this semidiscretization\n\nKeywords\n\nneighborhood_search: The neighborhood search to be used in the simulation. By default, the GridNeighborhoodSearch is used. Use nothing to loop over all particles (no neighborhood search). To use other neighborhood search implementations, pass a template of a neighborhood search. See copy_neighborhood_search and the examples below for more details. To use a periodic domain, pass a PeriodicBox to the neighborhood search.\nthreaded_nhs_update=true: Can be used to deactivate thread parallelization in the neighborhood search update. This can be one of the largest sources of variations between simulations with different thread numbers due to particle ordering changes.\n\nExamples\n\nsemi = Semidiscretization(fluid_system, boundary_system)\n\nsemi = Semidiscretization(fluid_system, boundary_system,\n neighborhood_search=GridNeighborhoodSearch{2}(update_strategy=SerialUpdate()))\n\nperiodic_box = PeriodicBox(min_corner = [0.0, 0.0], max_corner = [1.0, 1.0])\nsemi = Semidiscretization(fluid_system, boundary_system,\n neighborhood_search=GridNeighborhoodSearch{2}(; periodic_box))\n\nsemi = Semidiscretization(fluid_system, boundary_system,\n neighborhood_search=PrecomputedNeighborhoodSearch{2}())\n\nsemi = Semidiscretization(fluid_system, boundary_system,\n neighborhood_search=nothing)\n\n\n\n\n\n","category":"type"},{"location":"general/semidiscretization/#TrixiParticles.SourceTermDamping","page":"Semidiscretization","title":"TrixiParticles.SourceTermDamping","text":"SourceTermDamping(; damping_coefficient)\n\nA source term to be used when a damping step is required before running a full simulation. The term -c cdot v_a is added to the acceleration fracmathrmdv_amathrmdt of particle a, where c is the damping coefficient and v_a is the velocity of particle a.\n\nKeywords\n\ndamping_coefficient: The coefficient d above. A higher coefficient means more damping. A coefficient of 1e-4 is a good starting point for damping a fluid at rest.\n\nExamples\n\nsource_terms = SourceTermDamping(; damping_coefficient=1e-4)\n\n\n\n\n\n","category":"type"},{"location":"general/semidiscretization/#TrixiParticles.restart_with!-Tuple{Any, Any}","page":"Semidiscretization","title":"TrixiParticles.restart_with!","text":"restart_with!(semi, sol)\n\nSet the initial coordinates and velocities of all systems in semi to the final values in the solution sol. semidiscretize has to be called again afterwards, or another Semidiscretization can be created with the updated systems.\n\nArguments\n\nsemi: The semidiscretization\nsol: The ODESolution returned by solve of OrdinaryDiffEq\n\n\n\n\n\n","category":"method"},{"location":"general/semidiscretization/#TrixiParticles.semidiscretize-Tuple{Any, Any}","page":"Semidiscretization","title":"TrixiParticles.semidiscretize","text":"semidiscretize(semi, tspan; reset_threads=true)\n\nCreate an ODEProblem from the semidiscretization with the specified tspan.\n\nArguments\n\nsemi: A Semidiscretization holding the systems involved in the simulation.\ntspan: The time span over which the simulation will be run.\n\nKeywords\n\nreset_threads: A boolean flag to reset Polyester.jl threads before the simulation (default: true). After an error within a threaded loop, threading might be disabled. Resetting the threads before the simulation ensures that threading is enabled again for the simulation. See also trixi-framework/Trixi.jl#1583.\n\nReturns\n\nA DynamicalODEProblem (see the OrdinaryDiffEq.jl docs) to be integrated with OrdinaryDiffEq.jl. Note that this is not a true DynamicalODEProblem where the acceleration does not depend on the velocity. Therefore, not all integrators designed for DynamicalODEProblems will work properly. However, all integrators designed for ODEProblems can be used.\n\nExamples\n\nsemi = Semidiscretization(fluid_system, boundary_system)\ntspan = (0.0, 1.0)\node_problem = semidiscretize(semi, tspan)\n\n\n\n\n\n","category":"method"},{"location":"systems/boundary/#Boundary-System","page":"Boundary","title":"Boundary System","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":" BoundarySPHSystem","category":"page"},{"location":"systems/boundary/#TrixiParticles.BoundarySPHSystem","page":"Boundary","title":"TrixiParticles.BoundarySPHSystem","text":"BoundarySPHSystem(initial_condition, boundary_model; movement=nothing, adhesion_coefficient=0.0)\n\nSystem for boundaries modeled by boundary particles. The interaction between fluid and boundary particles is specified by the boundary model.\n\nArguments\n\ninitial_condition: Initial condition (see InitialCondition)\nboundary_model: Boundary model (see Boundary Models)\n\nKeyword Arguments\n\nmovement: For moving boundaries, a BoundaryMovement can be passed.\nadhesion_coefficient: Coefficient specifying the adhesion of a fluid to the surface. Note: currently it is assumed that all fluids have the same adhesion coefficient.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":" BoundaryDEMSystem","category":"page"},{"location":"systems/boundary/#TrixiParticles.BoundaryDEMSystem","page":"Boundary","title":"TrixiParticles.BoundaryDEMSystem","text":"BoundaryDEMSystem(initial_condition, normal_stiffness)\n\nSystem for boundaries modeled by boundary particles. The interaction between fluid and boundary particles is specified by the boundary model.\n\nwarning: Experimental Implementation\nThis is an experimental feature and may change in a future releases.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":" BoundaryMovement","category":"page"},{"location":"systems/boundary/#TrixiParticles.BoundaryMovement","page":"Boundary","title":"TrixiParticles.BoundaryMovement","text":"BoundaryMovement(movement_function, is_moving; moving_particles=nothing)\n\nArguments\n\nmovement_function: Time-dependent function returning an SVector of d dimensions for a d-dimensional problem.\nis_moving: Function to determine in each timestep if the particles are moving or not. Its boolean return value is mandatory to determine if the neighborhood search will be updated.\n\nKeyword Arguments\n\nmoving_particles: Indices of moving particles. Default is each particle in BoundarySPHSystem.\n\nIn the example below, movement describes particles moving in a circle as long as the time is lower than 1.5.\n\nExamples\n\nmovement_function(t) = SVector(cos(2pi*t), sin(2pi*t))\nis_moving(t) = t < 1.5\n\nmovement = BoundaryMovement(movement_function, is_moving)\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/#boundary_models","page":"Boundary","title":"Boundary Models","text":"","category":"section"},{"location":"systems/boundary/#Dummy-Particles","page":"Boundary","title":"Dummy Particles","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Boundaries modeled as dummy particles, which are treated like fluid particles, but their positions and velocities are not evolved in time. Since the force towards the fluid should not change with the material density when used with a TotalLagrangianSPHSystem, the dummy particles need to have a mass corresponding to the fluid's rest density, which we call \"hydrodynamic mass\", as opposed to mass corresponding to the material density of a TotalLagrangianSPHSystem.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Here, initial_density and hydrodynamic_mass are vectors that contains the initial density and the hydrodynamic mass respectively for each boundary particle. Note that when used with SummationDensity (see below), this is only used to determine the element type and the number of boundary particles.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"To establish a relationship between density and pressure, a state_equation has to be passed, which should be the same as for the adjacent fluid systems. To sum over neighboring particles, a smoothing_kernel and smoothing_length needs to be passed. This should be the same as for the adjacent fluid system with the largest smoothing length.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"In the literature, this kind of boundary particles is referred to as \"dummy particles\" (Adami et al., 2012 and Valizadeh & Monaghan, 2015), \"frozen fluid particles\" (Akinci et al., 2012) or \"dynamic boundaries Crespo et al., 2007. The key detail of this boundary condition and the only difference between the boundary models in these references is the way the density and pressure of boundary particles is computed.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Since boundary particles are treated like fluid particles, the force on fluid particle a due to boundary particle b is given by","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"f_ab = m_a m_b left( fracp_arho_a^2 + fracp_brho_b^2 right) nabla_r_a W(Vert r_a - r_b Vert h)","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"The quantities to be defined here are the density rho_b and pressure p_b of the boundary particle b.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":" BoundaryModelDummyParticles","category":"page"},{"location":"systems/boundary/#TrixiParticles.BoundaryModelDummyParticles","page":"Boundary","title":"TrixiParticles.BoundaryModelDummyParticles","text":"BoundaryModelDummyParticles(initial_density, hydrodynamic_mass,\n density_calculator, smoothing_kernel,\n smoothing_length; viscosity=nothing,\n state_equation=nothing, correction=nothing)\n\nBoundary model for BoundarySPHSystem.\n\nArguments\n\ninitial_density: Vector holding the initial density of each boundary particle.\nhydrodynamic_mass: Vector holding the \"hydrodynamic mass\" of each boundary particle. See description above for more information.\ndensity_calculator: Strategy to compute the hydrodynamic density of the boundary particles. See description below for more information.\nsmoothing_kernel: Smoothing kernel should be the same as for the adjacent fluid system.\nsmoothing_length: Smoothing length should be the same as for the adjacent fluid system.\n\nKeywords\n\nstate_equation: This should be the same as for the adjacent fluid system (see e.g. StateEquationCole).\ncorrection: Correction method of the adjacent fluid system (see Corrections).\nviscosity: Slip (default) or no-slip condition. See description below for further information.\n\nExamples\n\n# Free-slip condition\nboundary_model = BoundaryModelDummyParticles(densities, masses, AdamiPressureExtrapolation(),\n smoothing_kernel, smoothing_length)\n\n# No-slip condition\nboundary_model = BoundaryModelDummyParticles(densities, masses, AdamiPressureExtrapolation(),\n smoothing_kernel, smoothing_length,\n viscosity=ViscosityAdami(nu=1e-6))\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/#Hydrodynamic-density-of-dummy-particles","page":"Boundary","title":"Hydrodynamic density of dummy particles","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"We provide six options to compute the boundary density and pressure, determined by the density_calculator:","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"(Recommended) With AdamiPressureExtrapolation, the pressure is extrapolated from the pressure of the fluid according to Adami et al., 2012, and the density is obtained by applying the inverse of the state equation. This option usually yields the best results of the options listed here.\n(Only relevant for FSI) With BernoulliPressureExtrapolation, the pressure is extrapolated from the pressure similar to the AdamiPressureExtrapolation, but a relative velocity-dependent pressure part is calculated between moving solids and fluids, which increases the boundary pressure in areas prone to penetrations.\nWith SummationDensity, the density is calculated by summation over the neighboring particles, and the pressure is computed from the density with the state equation.\nWith ContinuityDensity, the density is integrated from the continuity equation, and the pressure is computed from the density with the state equation. Note that this causes a gap between fluid and boundary where the boundary is initialized without any contact to the fluid. This is due to overestimation of the boundary density as soon as the fluid comes in contact with boundary particles that initially did not have contact to the fluid. Therefore, in dam break simulations, there is a visible \"step\", even though the boundary is supposed to be flat. See also dual.sphysics.org/faq/#Q_13.\nWith PressureZeroing, the density is set to the reference density and the pressure is computed from the density with the state equation. This option is not recommended. The other options yield significantly better results.\nWith PressureMirroring, the density is set to the reference density. The pressure is not used. Instead, the fluid pressure is mirrored as boundary pressure in the momentum equation. This option is not recommended due to stability issues. See PressureMirroring for more details.","category":"page"},{"location":"systems/boundary/#1.-[AdamiPressureExtrapolation](@ref)","page":"Boundary","title":"1. AdamiPressureExtrapolation","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"The pressure of the boundary particles is obtained by extrapolating the pressure of the fluid according to Adami et al., 2012. The pressure of a boundary particle b is given by","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"p_b = fracsum_f (p_f + rho_f (bmg - bma_b) cdot bmr_bf) W(Vert r_bf Vert h)sum_f W(Vert r_bf Vert h)","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"where the sum is over all fluid particles, rho_f and p_f denote the density and pressure of fluid particle f, respectively, r_bf = r_b - r_f denotes the difference of the coordinates of particles b and f, bmg denotes the gravitational acceleration acting on the fluid, and bma_b denotes the acceleration of the boundary particle b.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":" AdamiPressureExtrapolation","category":"page"},{"location":"systems/boundary/#TrixiParticles.AdamiPressureExtrapolation","page":"Boundary","title":"TrixiParticles.AdamiPressureExtrapolation","text":"AdamiPressureExtrapolation(; pressure_offset=0.0)\n\ndensity_calculator for BoundaryModelDummyParticles.\n\nKeywords\n\npressure_offset=0.0: Sometimes it is necessary to artificially increase the boundary pressure to prevent penetration, which is possible by increasing this value.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/#2.-[BernoulliPressureExtrapolation](@ref)","page":"Boundary","title":"2. BernoulliPressureExtrapolation","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Identical to the pressure p_b calculated via AdamiPressureExtrapolation, but it adds the dynamic pressure component of the Bernoulli equation:","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"p_b = fracsum_f (p_f + frac12 rho_textneighbor left( frac (mathbfv_f - mathbfv_textbody) cdot (mathbfx_f - mathbfx_textneighbor) left mathbfx_f - mathbfx_textneighbor right right)^2 times textfactor +rho_f (bmg - bma_b) cdot bmr_bf) W(Vert r_bf Vert h)sum_f W(Vert r_bf Vert h) ","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"where mathbfv_f is the velocity of the fluid and mathbfv_textbody is the velocity of the body. This adjustment provides a higher boundary pressure for solid bodies moving with a relative velocity to the fluid to prevent penetration. This modification is original and not derived from any literature source.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":" BernoulliPressureExtrapolation","category":"page"},{"location":"systems/boundary/#TrixiParticles.BernoulliPressureExtrapolation","page":"Boundary","title":"TrixiParticles.BernoulliPressureExtrapolation","text":"BernoulliPressureExtrapolation(; pressure_offset=0.0, factor=1.0)\n\ndensity_calculator for BoundaryModelDummyParticles.\n\nKeywords\n\npressure_offset=0.0: Sometimes it is necessary to artificially increase the boundary pressure to prevent penetration, which is possible by increasing this value.\nfactor=1.0 : Setting factor allows to just increase the strength of the dynamic pressure part.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/#5.-[PressureZeroing](@ref)","page":"Boundary","title":"5. PressureZeroing","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"This is the simplest way to implement dummy boundary particles. The density of each particle is set to the reference density and the pressure to the reference pressure (the corresponding pressure to the reference density by the state equation).","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":" PressureZeroing","category":"page"},{"location":"systems/boundary/#TrixiParticles.PressureZeroing","page":"Boundary","title":"TrixiParticles.PressureZeroing","text":"PressureZeroing()\n\ndensity_calculator for BoundaryModelDummyParticles.\n\nnote: Note\nThis boundary model produces significantly worse results than all other models and is only included for research purposes.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/#6.-[PressureMirroring](@ref)","page":"Boundary","title":"6. PressureMirroring","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Instead of calculating density and pressure for each boundary particle, we modify the momentum equation,","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"fracmathrmdv_amathrmdt = -sum_b m_b left( fracp_arho_a^2 + fracp_brho_b^2 right) nabla_a W_ab","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"to replace the unknown density rho_b if b is a boundary particle by the reference density and the unknown pressure p_b if b is a boundary particle by the pressure p_a of the interacting fluid particle. The momentum equation therefore becomes","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"fracmathrmdv_amathrmdt = -sum_f m_f left( fracp_arho_a^2 + fracp_frho_f^2 right) nabla_a W_af\n-sum_b m_b left( fracp_arho_a^2 + fracp_arho_0^2 right) nabla_a W_ab","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"where the first sum is over all fluid particles and the second over all boundary particles.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"This approach was first mentioned by Akinci et al. (2012) and written down in this form by Band et al. (2018).","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":" PressureMirroring","category":"page"},{"location":"systems/boundary/#TrixiParticles.PressureMirroring","page":"Boundary","title":"TrixiParticles.PressureMirroring","text":"PressureMirroring()\n\ndensity_calculator for BoundaryModelDummyParticles.\n\nnote: Note\nThis boundary model requires high viscosity for stability with WCSPH. It also produces significantly worse results than AdamiPressureExtrapolation and is not more efficient because smaller time steps are required due to more noise in the pressure. We added this model only for research purposes and for comparison with SPlisHSPlasH.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/#No-slip-conditions","page":"Boundary","title":"No-slip conditions","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"For the interaction of dummy particles and fluid particles, Adami et al. (2012) impose a no-slip boundary condition by assigning a wall velocity v_w to the dummy particle.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"The wall velocity of particle a is calculated from the prescribed boundary particle velocity v_a and the smoothed velocity field","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"v_w = 2 v_a - fracsum_b v_b W_absum_b W_ab","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"where the sum is over all fluid particles.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"By choosing the viscosity model ViscosityAdami for viscosity, a no-slip condition is imposed. It is recommended to choose nu in the order of either the kinematic viscosity parameter of the adjacent fluid or the equivalent from the artificial parameter alpha of the adjacent fluid (nu = fracalpha h c 2d + 4). When omitting the viscous interaction (default viscosity=nothing), a free-slip wall boundary condition is applied.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"warning: Warning\nThe viscosity model ArtificialViscosityMonaghan for BoundaryModelDummyParticles has not been verified yet.","category":"page"},{"location":"systems/boundary/#Repulsive-Particles","page":"Boundary","title":"Repulsive Particles","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Boundaries modeled as boundary particles which exert forces on the fluid particles (Monaghan, Kajtar, 2009). The force on fluid particle a due to boundary particle b is given by","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"f_ab = m_a left(tildef_ab - m_b Pi_ab nabla_r_a W(Vert r_a - r_b Vert h)right)","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"with","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"tildef_ab = fracKbeta^n-1 fracr_abVert r_ab Vert (Vert r_ab Vert - d) Phi(Vert r_ab Vert h)\nfrac2 m_bm_a + m_b","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"where m_a and m_b are the masses of fluid particle a and boundary particle b respectively, r_ab = r_a - r_b is the difference of the coordinates of particles a and b, d denotes the boundary particle spacing and n denotes the number of dimensions (see Monaghan & Kajtar, 2009, Equation (3.1) and Valizadeh & Monaghan, 2015). Note that the repulsive acceleration tildef_ab does not depend on the masses of the boundary particles. Here, Phi denotes the 1D Wendland C4 kernel, normalized to 177 for q=0 (Monaghan & Kajtar, 2009, Section 4), with Phi(r h) = w(rh) and","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"w(q) =\nbegincases\n (17732) (1 + (52)q + 2q^2)(2 - q)^5 textif 0 leq q 2 \n 0 textif q geq 2\nendcases","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"The boundary particles are assumed to have uniform spacing by the factor beta smaller than the expected fluid particle spacing. For example, if the fluid particles have an expected spacing of 03 and the boundary particles have a uniform spacing of 01, then this parameter should be set to beta = 3. According to Monaghan & Kajtar (2009), a value of beta = 3 for the Wendland C4 that we use here is reasonable for most computing purposes.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"The parameter K is used to scale the force exerted by the boundary particles. In Monaghan & Kajtar (2009), a value of gD is used for static tank simulations, where g is the gravitational acceleration and D is the depth of the fluid.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"The viscosity Pi_ab is calculated according to the viscosity used in the simulation, where the density of the boundary particle if needed is assumed to be identical to the density of the fluid particle.","category":"page"},{"location":"systems/boundary/#No-slip-condition","page":"Boundary","title":"No-slip condition","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"By choosing the viscosity model ArtificialViscosityMonaghan for viscosity, a no-slip condition is imposed. When omitting the viscous interaction (default viscosity=nothing), a free-slip wall boundary condition is applied.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"warning: Warning\nThe no-slip conditions for BoundaryModelMonaghanKajtar have not been verified yet.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"boundary\", \"monaghan_kajtar\", \"monaghan_kajtar.jl\")]","category":"page"},{"location":"systems/boundary/#TrixiParticles.BoundaryModelMonaghanKajtar","page":"Boundary","title":"TrixiParticles.BoundaryModelMonaghanKajtar","text":"BoundaryModelMonaghanKajtar(K, beta, boundary_particle_spacing, mass;\n viscosity=nothing)\n\nBoundary model for BoundarySPHSystem.\n\nArguments\n\nK: Scaling factor for repulsive force.\nbeta: Ratio of fluid particle spacing to boundary particle spacing.\nboundary_particle_spacing: Boundary particle spacing.\nmass: Vector holding the mass of each boundary particle.\n\nKeywords\n\nviscosity: Free-slip (default) or no-slip condition. See description above for further information.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/#open_boundary","page":"Boundary","title":"Open Boundaries","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"boundary\", \"open_boundary\", \"system.jl\")]","category":"page"},{"location":"systems/boundary/#TrixiParticles.OpenBoundarySPHSystem","page":"Boundary","title":"TrixiParticles.OpenBoundarySPHSystem","text":"OpenBoundarySPHSystem(boundary_zone::Union{InFlow, OutFlow};\n fluid_system::FluidSystem, buffer_size::Integer,\n boundary_model,\n reference_velocity=nothing,\n reference_pressure=nothing,\n reference_density=nothing)\n\nOpen boundary system for in- and outflow particles.\n\nArguments\n\nboundary_zone: Use InFlow for an inflow and OutFlow for an outflow boundary.\n\nKeywords\n\nfluid_system: The corresponding fluid system\nboundary_model: Boundary model (see Open Boundary Models)\nbuffer_size: Number of buffer particles.\nreference_velocity: Reference velocity is either a function mapping each particle's coordinates and time to its velocity, an array where the i-th column holds the velocity of particle i or, for a constant fluid velocity, a vector holding this velocity.\nreference_pressure: Reference pressure is either a function mapping each particle's coordinates and time to its pressure, a vector holding the pressure of each particle, or a scalar for a constant pressure over all particles.\nreference_density: Reference density is either a function mapping each particle's coordinates and time to its density, a vector holding the density of each particle, or a scalar for a constant density over all particles.\n\nwarning: Experimental Implementation\nThis is an experimental feature and may change in any future releases.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"boundary\", \"open_boundary\", \"boundary_zones.jl\")]","category":"page"},{"location":"systems/boundary/#TrixiParticles.InFlow","page":"Boundary","title":"TrixiParticles.InFlow","text":"InFlow(; plane, flow_direction, density, particle_spacing,\n initial_condition=nothing, extrude_geometry=nothing,\n open_boundary_layers::Integer)\n\nInflow boundary zone for OpenBoundarySPHSystem.\n\nThe specified plane (line in 2D or rectangle in 3D) will be extruded in upstream direction (the direction opposite to flow_direction) to create a box for the boundary zone. There are three ways to specify the actual shape of the inflow:\n\nDon't pass initial_condition or extrude_geometry. The boundary zone box will then be filled with inflow particles (default).\nSpecify extrude_geometry by passing a 1D shape in 2D or a 2D shape in 3D, which is then extruded in upstream direction to create the inflow particles.\nIn 2D, the shape must be either an initial condition with 2D coordinates, which lies on the line specified by plane, or an initial condition with 1D coordinates, which lies on the line specified by plane when a y-coordinate of 0 is added.\nIn 3D, the shape must be either an initial condition with 3D coordinates, which lies in the rectangle specified by plane, or an initial condition with 2D coordinates, which lies in the rectangle specified by plane when a z-coordinate of 0 is added.\nSpecify initial_condition by passing a 2D initial condition in 2D or a 3D initial condition in 3D, which will be used for the inflow particles.\n\nnote: Note\nParticles outside the boundary zone box will be removed.\n\nKeywords\n\nplane: Tuple of points defining a part of the surface of the domain. The points must either span a line in 2D or a rectangle in 3D. This line or rectangle is then extruded in upstream direction to obtain the boundary zone. In 2D, pass two points (A B), so that the interval A B is the inflow surface. In 3D, pass three points (A B C), so that the rectangular inflow surface is spanned by the vectors widehatAB and widehatAC. These two vectors must be orthogonal.\nflow_direction: Vector defining the flow direction.\nopen_boundary_layers: Number of particle layers in upstream direction.\nparticle_spacing: The spacing between the particles (see InitialCondition).\ndensity: Particle density (see InitialCondition).\ninitial_condition=nothing: InitialCondition for the inflow particles. Particles outside the boundary zone will be removed. Do not use together with extrude_geometry.\nextrude_geometry=nothing: 1D shape in 2D or 2D shape in 3D, which lies on the plane and is extruded upstream to obtain the inflow particles. See point 2 above for more details.\n\nExamples\n\n# 2D\nplane_points = ([0.0, 0.0], [0.0, 1.0])\nflow_direction=[1.0, 0.0]\n\ninflow = InFlow(; plane=plane_points, particle_spacing=0.1, flow_direction, density=1.0,\n open_boundary_layers=4)\n\n# 3D\nplane_points = ([0.0, 0.0, 0.0], [1.0, 0.0, 0.0], [0.0, 1.0, 0.0])\nflow_direction=[0.0, 0.0, 1.0]\n\ninflow = InFlow(; plane=plane_points, particle_spacing=0.1, flow_direction, density=1.0,\n open_boundary_layers=4)\n\n# 3D particles sampled as cylinder\ncircle = SphereShape(0.1, 0.5, (0.5, 0.5), 1.0, sphere_type=RoundSphere())\n\ninflow = InFlow(; plane=plane_points, particle_spacing=0.1, flow_direction, density=1.0,\n extrude_geometry=circle, open_boundary_layers=4)\n\nwarning: Experimental Implementation\nThis is an experimental feature and may change in any future releases.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/#TrixiParticles.OutFlow","page":"Boundary","title":"TrixiParticles.OutFlow","text":"OutFlow(; plane, flow_direction, density, particle_spacing,\n initial_condition=nothing, extrude_geometry=nothing,\n open_boundary_layers::Integer)\n\nOutflow boundary zone for OpenBoundarySPHSystem.\n\nThe specified plane (line in 2D or rectangle in 3D) will be extruded in downstream direction (the direction in flow_direction) to create a box for the boundary zone. There are three ways to specify the actual shape of the outflow:\n\nDon't pass initial_condition or extrude_geometry. The boundary zone box will then be filled with outflow particles (default).\nSpecify extrude_geometry by passing a 1D shape in 2D or a 2D shape in 3D, which is then extruded in downstream direction to create the outflow particles.\nIn 2D, the shape must be either an initial condition with 2D coordinates, which lies on the line specified by plane, or an initial condition with 1D coordinates, which lies on the line specified by plane when a y-coordinate of 0 is added.\nIn 3D, the shape must be either an initial condition with 3D coordinates, which lies in the rectangle specified by plane, or an initial condition with 2D coordinates, which lies in the rectangle specified by plane when a z-coordinate of 0 is added.\nSpecify initial_condition by passing a 2D initial condition in 2D or a 3D initial condition in 3D, which will be used for the outflow particles.\n\nnote: Note\nParticles outside the boundary zone box will be removed.\n\nKeywords\n\nplane: Tuple of points defining a part of the surface of the domain. The points must either span a line in 2D or a rectangle in 3D. This line or rectangle is then extruded in downstream direction to obtain the boundary zone. In 2D, pass two points (A B), so that the interval A B is the outflow surface. In 3D, pass three points (A B C), so that the rectangular outflow surface is spanned by the vectors widehatAB and widehatAC. These two vectors must be orthogonal.\nflow_direction: Vector defining the flow direction.\nopen_boundary_layers: Number of particle layers in downstream direction.\nparticle_spacing: The spacing between the particles (see InitialCondition).\ndensity: Particle density (see InitialCondition).\ninitial_condition=nothing: InitialCondition for the outflow particles. Particles outside the boundary zone will be removed. Do not use together with extrude_geometry.\nextrude_geometry=nothing: 1D shape in 2D or 2D shape in 3D, which lies on the plane and is extruded downstream to obtain the outflow particles. See point 2 above for more details.\n\nExamples\n\n# 2D\nplane_points = ([0.0, 0.0], [0.0, 1.0])\nflow_direction = [1.0, 0.0]\n\noutflow = OutFlow(; plane=plane_points, particle_spacing=0.1, flow_direction, density=1.0,\n open_boundary_layers=4)\n\n# 3D\nplane_points = ([0.0, 0.0, 0.0], [1.0, 0.0, 0.0], [0.0, 1.0, 0.0])\nflow_direction = [0.0, 0.0, 1.0]\n\noutflow = OutFlow(; plane=plane_points, particle_spacing=0.1, flow_direction, density=1.0,\n open_boundary_layers=4)\n\n# 3D particles sampled as cylinder\ncircle = SphereShape(0.1, 0.5, (0.5, 0.5), 1.0, sphere_type=RoundSphere())\n\noutflow = OutFlow(; plane=plane_points, particle_spacing=0.1, flow_direction, density=1.0,\n extrude_geometry=circle, open_boundary_layers=4)\n\nwarning: Experimental Implementation\nThis is an experimental feature and may change in any future releases.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/#open_boundary_models","page":"Boundary","title":"Open Boundary Models","text":"","category":"section"},{"location":"systems/boundary/#method_of_characteristics","page":"Boundary","title":"Method of characteristics","text":"","category":"section"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"boundary\", \"open_boundary\", \"method_of_characteristics.jl\")]","category":"page"},{"location":"systems/boundary/#TrixiParticles.BoundaryModelLastiwka","page":"Boundary","title":"TrixiParticles.BoundaryModelLastiwka","text":"BoundaryModelLastiwka()\n\nBoundary model for OpenBoundarySPHSystem. This model uses the characteristic variables to propagate the appropriate values to the outlet or inlet and have been proposed by Lastiwka et al. (2009). For more information about the method see description below.\n\n\n\n\n\n","category":"type"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"The difficulty in non-reflecting boundary conditions, also called open boundaries, is to determine the appropriate boundary values of the exact characteristics of the Euler equations. Assuming the flow near the boundaries is normal to the boundary and free of shock waves and significant viscous effects, it can be shown that three characteristic variables exist:","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"J_1, associated with convection of entropy and propagates at flow velocity,\nJ_2, downstream-running characteristics,\nJ_3, upstream-running characteristics.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Giles (1990) derived those variables based on a linearized set of governing equations:","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"J_1 = -c_s^2 (rho - rho_textref) + (p - p_textref)","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"J_2 = rho c_s (v - v_textref) + (p - p_textref)","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"J_3 = - rho c_s (v - v_textref) + (p - p_textref)","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"where the subscript \"ref\" denotes the reference flow near the boundaries, which can be prescribed.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Specifying the reference variables is not equivalent to prescription of rho, v and p directly, since the perturbation from the reference flow is allowed.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Lastiwka et al. (2009) applied the method of characteristic to SPH and determined the number of variables that should be prescribed at the boundary and the number which should be propagated from the fluid domain to the boundary:","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"For an inflow boundary:\nPrescribe downstream-running characteristics J_1 and J_2\nTransmit J_3 from the fluid domain (allow J_3 to propagate upstream to the boundary).\nFor an outflow boundary:\nPrescribe upstream-running characteristic J_3\nTransmit J_1 and J_2 from the fluid domain.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"Prescribing is done by simply setting the characteristics to zero. To transmit the characteristics from the fluid domain, or in other words, to carry the information of the fluid to the boundaries, Negi et al. (2020) use a Shepard Interpolation","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"f_i = fracsum_j^N f_j W_ijsum_j^N W_ij","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"where the i-th particle is a boundary particle, f is either J_1, J_2 or J_3 and N is the set of neighboring fluid particles.","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"To express pressure p, density rho and velocity v as functions of the characteristic variables, the system of equations from the characteristic variables is inverted and gives","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":" rho - rho_textref = frac1c_s^2 left( -J_1 + frac12 J_2 + frac12 J_3 right)","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"u - u_textref= frac12rho c_s left( J_2 - J_3 right)","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"p - p_textref = frac12 left( J_2 + J_3 right)","category":"page"},{"location":"systems/boundary/","page":"Boundary","title":"Boundary","text":"With J_1, J_2 and J_3 determined, we can easily solve for the actual variables for each particle.","category":"page"},{"location":"tutorials_template/tut_setup/#Setting-up-your-simulation-from-scratch","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"","category":"section"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"In this tutorial, we will guide you through the general structure of simulation files. We will set up a simulation similar to the example simulation examples/fluid/hydrostatic_water_column_2d.jl, which is one of our simplest example simulations. In the second part of this tutorial, we will show how to replace components of TrixiParticles.jl by custom implementations from within a simulation file, without ever cloning the repository.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"For different setups and physics, have a look at our other example files.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"First, we import TrixiParticles.jl and OrdinaryDiffEq.jl, which we will use at the very end for the time integration.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"using TrixiParticles\nusing OrdinaryDiffEq","category":"page"},{"location":"tutorials_template/tut_setup/#Resolution","page":"Setting up your simulation from scratch","title":"Resolution","text":"","category":"section"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"Now, we define the particle spacing, which is our numerical resolution. For a fluid, we usually call the variable fluid_particle_spacing, so that we can easily change the resolution of an example file by overwriting this variable with trixi_include. In 2D, the number of particles will grow quadratically, in 3D cubically with the spacing.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"We also set the number of boundary layers, which need to be sufficiently large, depending on the smoothing kernel and smoothing length, so that the compact support of the smoothing kernel is fully sampled with particles for a fluid particle close to a boundary. In particular, we require the boundary thickness boundary_layers * fluid_particle_spacing to be larger than the compact support of the kernel. The compact support of each kernel can be found in the smoothing kernel overview.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"fluid_particle_spacing = 0.05\nboundary_layers = 3\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/#Experiment-setup","page":"Setting up your simulation from scratch","title":"Experiment setup","text":"","category":"section"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"We want to simulate a water column resting under hydrostatic pressure inside a rectangular tank. (Image: Experiment Setup) First, we define physical parameters like gravitational acceleration, simulation time, initial fluid size, tank size and fluid density.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"gravity = 9.81\ntspan = (0.0, 1.0)\ninitial_fluid_size = (1.0, 0.9)\ntank_size = (1.0, 1.0)\nfluid_density = 1000.0\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"In order to have the initial particle mass and density correspond to the hydrostatic pressure gradient, we need to define a state equation, which relates the fluid density to pressure. Note that we could also skip this part here and define the state equation later when we define the fluid system, but then the fluid would be initialized with constant density, which would cause it to oscillate under gravity.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"sound_speed = 10.0\nstate_equation = StateEquationCole(; sound_speed, reference_density=fluid_density,\n exponent=7)\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"The speed of sound here is numerical and not physical. We artificially lower the speed of sound, since the physical speed of sound in water would lead to prohibitively small time steps. The speed of sound in Weakly Compressible SPH should be chosen as small as possible for numerical efficiency, but large enough to limit density fluctuations to about 1%.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"TrixiParticles.jl requires the initial particle positions and quantities in form of an InitialCondition. Instead of manually defining particle positions, you can work with our pre-defined setups. Among others, we provide setups for rectangular shapes, circles, and spheres. Initial conditions can also be combined with common set operations. See this page for a list of pre-defined setups and details on set operations on initial conditions.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"Here, we use the RectangularTank setup, which generates a rectangular fluid inside a rectangular tank, and supports a hydrostatic pressure gradient by passing a gravitational acceleration and a state equation (see above).","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"tank = RectangularTank(fluid_particle_spacing, initial_fluid_size, tank_size,\n fluid_density, n_layers=boundary_layers,\n acceleration=(0.0, -gravity), state_equation=state_equation)\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/#Fluid-system","page":"Setting up your simulation from scratch","title":"Fluid system","text":"","category":"section"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"To model the water column, we use the Weakly Compressible Smoothed Particle Hydrodynamics (WCSPH) method. This method requires a smoothing kernel and a corresponding smoothing length, which should be chosen in relation to the particle spacing.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"smoothing_length = 1.2 * fluid_particle_spacing\nsmoothing_kernel = SchoenbergCubicSplineKernel{2}()\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"You can find an overview over smoothing kernels and corresponding smoothing lengths here.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"For stability, we need numerical dissipation in form of an artificial viscosity term. Other viscosity models offer a physical approach based on the kinematic viscosity of the fluid.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"viscosity = ArtificialViscosityMonaghan(alpha=0.02, beta=0.0)\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"We choose the parameters as small as possible to avoid non-physical behavior, but as large as possible to stabilize the simulation.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"The WCSPH method can either compute the particle density directly with a kernel summation over all neighboring particles (see SummationDensity) or by making the particle density a variable in the ODE system and integrating its change over time. We choose the latter approach here by using the density calculator ContinuityDensity, which is more efficient and handles free surfaces without the need for additional correction terms.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"fluid_density_calculator = ContinuityDensity()\nfluid_system = WeaklyCompressibleSPHSystem(tank.fluid, fluid_density_calculator,\n state_equation, smoothing_kernel,\n smoothing_length, viscosity=viscosity,\n acceleration=(0.0, -gravity))\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/#Boundary-system","page":"Setting up your simulation from scratch","title":"Boundary system","text":"","category":"section"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"To model the boundary, we use particle-based boundary conditions, in which particles are sampled in the boundary that interact with the fluid particles to avoid penetration. In order to define a boundary system, we first have to choose a boundary model, which defines how the fluid interacts with boundary particles. We will use the BoundaryModelDummyParticles with AdamiPressureExtrapolation. See here for a comprehensive overview over boundary models.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"boundary_model = BoundaryModelDummyParticles(tank.boundary.density, tank.boundary.mass,\n state_equation=state_equation,\n AdamiPressureExtrapolation(),\n smoothing_kernel, smoothing_length)\nboundary_system = BoundarySPHSystem(tank.boundary, boundary_model)\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/#Semidiscretization","page":"Setting up your simulation from scratch","title":"Semidiscretization","text":"","category":"section"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"The key component of every simulation is the Semidiscretization, which couples all systems of the simulation. All simulation methods in TrixiParticles.jl are semidiscretizations, which discretize the equations in time to provide an ordinary differential equation that still has to be solved in time. By providing a simulation time span, we can call semidiscretize, which returns an ODEProblem that can be solved with a time integration method.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"semi = Semidiscretization(fluid_system, boundary_system)\node = semidiscretize(semi, tspan)\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/#Time-integration","page":"Setting up your simulation from scratch","title":"Time integration","text":"","category":"section"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"We use the methods provided by OrdinaryDiffEq.jl, but note that other packages or custom implementations can also be used.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"OrdinaryDiffEq.jl supports callbacks, which are executed during the simulation. For this simulation, we use the InfoCallback, which prints information about the simulation setup at the beginning of the simulation, information about the current simulation time and runtime during the simulation, and a performance summary at the end of the simulation. We also want to save the current solution in regular intervals in terms of simulation time as VTK, so that we can look at the solution in ParaView. The SolutionSavingCallback provides this functionality. To pass the callbacks to OrdinaryDiffEq.jl, we have to bundle them into a CallbackSet.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"info_callback = InfoCallback(interval=50)\nsaving_callback = SolutionSavingCallback(dt=0.02)\n\ncallbacks = CallbackSet(info_callback, saving_callback)\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"Finally, we can start the simulation by solving the ODEProblem. We use the method RDPK3SpFSAL35 of OrdinaryDiffEq.jl, which is a Runge-Kutta method with automatic (error based) time step size control. This method is usually a good choice for prototyping, since we do not have to worry about choosing a stable step size and can just run the simulation. For better performance, it might be beneficial to tweak the tolerances of this method or choose a different method that is more efficient for the respective simulation. You can find both approaches in our example files. Here, we just use the method with the default parameters, and only disable save_everystep to avoid expensive saving of the solution in every time step.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"sol = solve(ode, RDPK3SpFSAL35(), save_everystep=false, callback=callbacks);\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"See Visualization for how to visualize the solution. For the simplest visualization, we can use Plots.jl:","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"using Plots\nplot(sol)\nplot!(dpi=200); savefig(\"tut_setup_plot.png\"); nothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"(Image: plot)","category":"page"},{"location":"tutorials_template/tut_setup/#Replacing-components-with-custom-implementations","page":"Setting up your simulation from scratch","title":"Replacing components with custom implementations","text":"","category":"section"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"If we would like to use an implementation of a component that is not available in TrixiParticles.jl, we can implement it ourselves within the simulation file, without ever cloning the TrixiParticles.jl repository. A good starting point is to check out the available implementations in TrixiParticles.jl, then copy the relevant functions to the simulation file and modify them as needed.","category":"page"},{"location":"tutorials_template/tut_setup/#Custom-smoothing-kernel","page":"Setting up your simulation from scratch","title":"Custom smoothing kernel","text":"","category":"section"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"To implement a custom smoothing kernel, we define a struct extending TrixiParticles.SmoothingKernel. This abstract struct has a type parameter for the number of dimensions, which we set to 2 in this case.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"struct MyGaussianKernel <: TrixiParticles.SmoothingKernel{2} end","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"This kernel is going to be an implementation of the Gaussian kernel with a cutoff for compact support. Note that the same kernel in a more optimized version is already implemented in TrixiParticles.jl as GaussianKernel.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"In order to use our new kernel, we have to define three functions. TrixiParticles.kernel, which is the kernel function itself, TrixiParticles.kernel_deriv, which is the derivative of the kernel function, and TrixiParticles.compact_support, which defines the compact support of the kernel in relation to the smoothing length. The latter is relevant for determining the search radius of the neighborhood search.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"function TrixiParticles.kernel(kernel::MyGaussianKernel, r, h)\n q = r / h\n\n if q < 2\n return 1 / (pi * h^2) * exp(-q^2)\n end\n\n return 0.0\nend\n\nfunction TrixiParticles.kernel_deriv(kernel::MyGaussianKernel, r, h)\n q = r * h\n\n if q < 2\n return 1 / (pi * h^2) * (-2 * q) * exp(-q^2) / h\n end\n\n return 0.0\nend\n\nTrixiParticles.compact_support(::MyGaussianKernel, h) = 3 * h","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"For this kernel, we use a different smoothing length, which yields a similar kernel to the SchoenbergCubicSplineKernel that we used earlier.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"smoothing_length_gauss = 1.0 * fluid_particle_spacing","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"We can compare these kernels in a plot.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"using Plots\nx = range(-0.2, 0.2, length=500)\nplot(x, r -> TrixiParticles.kernel(SchoenbergCubicSplineKernel{2}(), abs(r), smoothing_length),\n label=\"SchoenbergCubicSplineKernel\", xlabel=\"r\")\nplot!(x, r -> TrixiParticles.kernel(MyGaussianKernel(), abs(r), smoothing_length_gauss),\n label=\"MyGaussianKernel\")\nplot!(dpi=200); savefig(\"tut_setup_plot2.png\"); nothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"(Image: plot)","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"This is all we need to use our custom kernel implementation in a simulation. We only need to replace the definition above by","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"smoothing_kernel = MyGaussianKernel()\nnothing # hide","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"and run the simulation file again.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"In order to use our kernel in a pre-defined example file, we can use the function trixi_include to replace the definition of the variable smoothing_kernel. The following will run the example simulation examples/fluid/hydrostatic_water_column_2d.jl with our custom kernel.","category":"page"},{"location":"tutorials_template/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"trixi_include(joinpath(examples_dir(), \"fluid\", \"hydrostatic_water_column_2d.jl\"),\n smoothing_kernel=MyGaussianKernel());\nnothing # hide","category":"page"},{"location":"news/","page":"News","title":"News","text":"EditURL = \"https://github.com/trixi-framework/TrixiParticles.jl/blob/main/NEWS.md\"","category":"page"},{"location":"news/#Changelog","page":"News","title":"Changelog","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"TrixiParticles.jl follows the interpretation of semantic versioning (semver) used in the Julia ecosystem. Notable changes will be documented in this file for human readability.","category":"page"},{"location":"news/#Version-0.2.3","page":"News","title":"Version 0.2.3","text":"","category":"section"},{"location":"news/#Highlights","page":"News","title":"Highlights","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"Transport Velocity Formulation (TVF) based on the work of Ramachandran et al. \"Entropically damped artificial compressibility for SPH\" (2019) was added.","category":"page"},{"location":"news/#Version-0.2.2","page":"News","title":"Version 0.2.2","text":"","category":"section"},{"location":"news/#Highlights-2","page":"News","title":"Highlights","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"Hotfix for threaded sampling of complex geometries.","category":"page"},{"location":"news/#Version-0.2.1","page":"News","title":"Version 0.2.1","text":"","category":"section"},{"location":"news/#Highlights-3","page":"News","title":"Highlights","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"Particle sampling of complex geometries from .stl and .asc files.","category":"page"},{"location":"news/#Version-0.2.0","page":"News","title":"Version 0.2.0","text":"","category":"section"},{"location":"news/#Removed","page":"News","title":"Removed","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"Use of the internal neighborhood search has been removed and replaced with PointNeighbors.jl.","category":"page"},{"location":"news/#Development-Cycle-0.1","page":"News","title":"Development Cycle 0.1","text":"","category":"section"},{"location":"news/#Highlights-4","page":"News","title":"Highlights","text":"","category":"section"},{"location":"news/#Discrete-Element-Method","page":"News","title":"Discrete Element Method","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"A basic implementation of the discrete element method was added.","category":"page"},{"location":"news/#Surface-Tension-and-Adhesion-Model","page":"News","title":"Surface Tension and Adhesion Model","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"A surface tension and adhesion model based on the work by Akinci et al., \"Versatile Surface Tension and Adhesion for SPH Fluids\" (2013) was added to WCSPH.","category":"page"},{"location":"news/#Support-for-Open-Boundaries","page":"News","title":"Support for Open Boundaries","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"Open boundaries using the method of characteristics based on the work of Lastiwka et al., \"Permeable and non-reflecting boundary conditions in SPH\" (2009) were added for WCSPH and EDAC.","category":"page"},{"location":"news/#Pre-Initial-Release-(v0.1.0)","page":"News","title":"Pre Initial Release (v0.1.0)","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"This section summarizes the initial features that TrixiParticles.jl was released with.","category":"page"},{"location":"news/#Highlights-5","page":"News","title":"Highlights","text":"","category":"section"},{"location":"news/#EDAC","page":"News","title":"EDAC","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"An implementation of EDAC (Entropically Damped Artificial Compressibility) was added, which allows for more stable simulations compared to basic WCSPH and reduces spurious pressure oscillations.","category":"page"},{"location":"news/#WCSPH","page":"News","title":"WCSPH","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"An implementation of WCSPH (Weakly Compressible Smoothed Particle Hydrodynamics), which is the classical SPH approach.","category":"page"},{"location":"news/","page":"News","title":"News","text":"Features:","category":"page"},{"location":"news/","page":"News","title":"News","text":"Correction schemes (Shepard (0. Order) ... MixedKernelGradient (1. Order))\nDensity reinitialization\nKernel summation and Continuity equation density formulations\nFlexible boundary conditions e.g. dummy particles with Adami pressure extrapolation, pressure zeroing, pressure mirroring...\nMoving boundaries\nDensity diffusion based on the models by Molteni & Colagrossi (2009), Ferrari et al. (2009) and Antuono et al. (2010).","category":"page"},{"location":"news/#TLSPH","page":"News","title":"TLSPH","text":"","category":"section"},{"location":"news/","page":"News","title":"News","text":"An implementation of TLSPH (Total Lagrangian Smoothed Particle Hydrodynamics) for solid bodies enabling FSI (Fluid Structure Interactions).","category":"page"},{"location":"general/util/#Util","page":"Util","title":"Util","text":"","category":"section"},{"location":"general/util/","page":"Util","title":"Util","text":"Modules = [TrixiParticles]\nPages = [\"util.jl\"]","category":"page"},{"location":"general/util/#TrixiParticles.examples_dir-Tuple{}","page":"Util","title":"TrixiParticles.examples_dir","text":"examples_dir()\n\nReturn the directory where the example files provided with TrixiParticles.jl are located. If TrixiParticles is installed as a regular package (with ]add TrixiParticles), these files are read-only and should not be modified. To find out which files are available, use, e.g., readdir.\n\nCopied from Trixi.jl.\n\nExamples\n\nreaddir(examples_dir())\n\n\n\n\n\n","category":"method"},{"location":"general/util/#TrixiParticles.validation_dir-Tuple{}","page":"Util","title":"TrixiParticles.validation_dir","text":"validation_dir()\n\nReturn the directory where the validation files provided with TrixiParticles.jl are located. If TrixiParticles is installed as a regular package (with ]add TrixiParticles), these files are read-only and should not be modified. To find out which files are available, use, e.g., readdir.\n\nCopied from Trixi.jl.\n\nExamples\n\nreaddir(validation_dir())\n\n\n\n\n\n","category":"method"},{"location":"general/util/#TrixiParticles.@autoinfiltrate","page":"Util","title":"TrixiParticles.@autoinfiltrate","text":"@autoinfiltrate\n@autoinfiltrate condition::Bool\n\nInvoke the @infiltrate macro of the package Infiltrator.jl to create a breakpoint for ad-hoc interactive debugging in the REPL. If the optional argument condition is given, the breakpoint is only enabled if condition evaluates to true.\n\nAs opposed to using Infiltrator.@infiltrate directly, this macro does not require Infiltrator.jl to be added as a dependency to TrixiParticles.jl. As a bonus, the macro will also attempt to load the Infiltrator module if it has not yet been loaded manually.\n\nNote: For this macro to work, the Infiltrator.jl package needs to be installed in your current Julia environment stack.\n\nSee also: Infiltrator.jl\n\nwarning: Internal use only\nPlease note that this macro is intended for internal use only. It is not part of the public API of TrixiParticles.jl, and it thus can altered (or be removed) at any time without it being considered a breaking change.\n\n\n\n\n\n","category":"macro"},{"location":"tutorials_template/tut_falling/#Example-file","page":"Example file","title":"Example file","text":"","category":"section"},{"location":"tutorials_template/tut_falling/","page":"Example file","title":"Example file","text":"!!include:examples/fsi/falling_spheres_2d.jl!!\n","category":"page"},{"location":"code_of_conduct/","page":"Code of Conduct","title":"Code of Conduct","text":"EditURL = \"https://github.com/trixi-framework/TrixiParticles.jl/blob/main/CODE_OF_CONDUCT.md\"","category":"page"},{"location":"code_of_conduct/#Code-of-Conduct","page":"Code of Conduct","title":"Code of Conduct","text":"","category":"section"},{"location":"code_of_conduct/","page":"Code of Conduct","title":"Code of Conduct","text":"Contributor Covenant Code of ConductOur PledgeWe as members, contributors, and leaders pledge to make participation in our community a harassment-free experience for everyone, regardless of age, body size, visible or invisible disability, ethnicity, sex characteristics, gender identity and expression, level of experience, education, socio-economic status, nationality, personal appearance, race, religion, or sexual identity and orientation.We pledge to act and interact in ways that contribute to an open, welcoming, diverse, inclusive, and healthy community.Our StandardsExamples of behavior that contributes to a positive environment for our community include:Demonstrating empathy and kindness toward other people\nBeing respectful of differing opinions, viewpoints, and experiences\nGiving and gracefully accepting constructive feedback\nAccepting responsibility and apologizing to those affected by our mistakes, and learning from the experience\nFocusing on what is best not just for us as individuals, but for the overall communityExamples of unacceptable behavior include:The use of sexualized language or imagery, and sexual attention or advances of any kind\nTrolling, insulting or derogatory comments, and personal or political attacks\nPublic or private harassment\nPublishing others' private information, such as a physical or email address, without their explicit permission\nOther conduct which could reasonably be considered inappropriate in a professional settingEnforcement ResponsibilitiesCommunity leaders are responsible for clarifying and enforcing our standards of acceptable behavior and will take appropriate and fair corrective action in response to any behavior that they deem inappropriate, threatening, offensive, or harmful.Community leaders have the right and responsibility to remove, edit, or reject comments, commits, code, wiki edits, issues, and other contributions that are not aligned to this Code of Conduct, and will communicate reasons for moderation decisions when appropriate.ScopeThis Code of Conduct applies within all community spaces, and also applies when an individual is officially representing the community in public spaces. Examples of representing our community include using an official e-mail address, posting via an official social media account, or acting as an appointed representative at an online or offline event.EnforcementInstances of abusive, harassing, or otherwise unacceptable behavior may be reported to Michael Schlottke-Lakemper, Sven Berger, or any other of the principal developers responsible for enforcement listed in Authors. All complaints will be reviewed and investigated promptly and fairly.All community leaders are obligated to respect the privacy and security of the reporter of any incident.Enforcement GuidelinesCommunity leaders will follow these Community Impact Guidelines in determining the consequences for any action they deem in violation of this Code of Conduct:1. CorrectionCommunity Impact: Use of inappropriate language or other behavior deemed unprofessional or unwelcome in the community.Consequence: A private, written warning from community leaders, providing clarity around the nature of the violation and an explanation of why the behavior was inappropriate. A public apology may be requested.2. WarningCommunity Impact: A violation through a single incident or series of actions.Consequence: A warning with consequences for continued behavior. No interaction with the people involved, including unsolicited interaction with those enforcing the Code of Conduct, for a specified period of time. This includes avoiding interactions in community spaces as well as external channels like social media. Violating these terms may lead to a temporary or permanent ban.3. Temporary BanCommunity Impact: A serious violation of community standards, including sustained inappropriate behavior.Consequence: A temporary ban from any sort of interaction or public communication with the community for a specified period of time. No public or private interaction with the people involved, including unsolicited interaction with those enforcing the Code of Conduct, is allowed during this period. Violating these terms may lead to a permanent ban.4. Permanent BanCommunity Impact: Demonstrating a pattern of violation of community standards, including sustained inappropriate behavior, harassment of an individual, or aggression toward or disparagement of classes of individuals.Consequence: A permanent ban from any sort of public interaction within the community.AttributionThis Code of Conduct is adapted from the [Contributor Covenant][homepage], version 2.0, available at https://www.contributor-covenant.org/version/2/0/codeofconduct.html.Community Impact Guidelines were inspired by Mozilla's code of conduct enforcement ladder.[homepage]: https://www.contributor-covenant.orgFor answers to common questions about this code of conduct, see the FAQ at https://www.contributor-covenant.org/faq. Translations are available at https://www.contributor-covenant.org/translations.","category":"page"},{"location":"systems/entropically_damped_sph/#edac","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility (EDAC) for SPH","text":"","category":"section"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"As opposed to the weakly compressible SPH scheme, which uses an equation of state, this scheme uses a pressure evolution equation to calculate the pressure","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"fracmathrmd p_amathrmdt = - rho c_s^2 nabla cdot v + nu nabla^2 p","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"which is derived by Clausen (2013). This equation is similar to the continuity equation (first term, see ContinuityDensity), but also contains a pressure damping term (second term, similar to density diffusion see DensityDiffusion), which reduces acoustic pressure waves through an entropy-generation mechanism.","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"The pressure evolution is discretized with the SPH method by Ramachandran (2019) as following:","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"The first term is equivalent to the classical artificial compressible methods, which are commonly motivated by assuming the artificial equation of state (StateEquationCole with exponent=1) and is discretized as","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"- rho c_s^2 nabla cdot v = sum_b m_b fracrho_arho_b c_s^2 v_ab cdot nabla_r_a W(Vert r_a - r_b Vert h)","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"where rho_a, rho_b, r_a, r_b, denote the density and coordinates of particles a and b respectively, c_s is the speed of sound and v_ab = v_a - v_b is the difference in the velocity.","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"The second term smooths the pressure through the introduction of entropy and is discretized as","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"nu nabla^2 p = fracV_a^2 + V_b^2m_a tildeeta_ab fracp_abVert r_ab^2 Vert + eta h_ab^2 nabla_r_a\nW(Vert r_a - r_b Vert h) cdot r_ab","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"where V_a, V_b denote the volume of particles a and b respectively and p_ab= p_a -p_b is the difference in the pressure.","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"The viscosity parameter eta_a for a particle a is given as","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"eta_a = rho_a fracalpha h c_s8","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"where it is found in the numerical experiments of Ramachandran (2019) that alpha = 05 is a good choice for a wide range of Reynolds numbers (0.0125 to 10000).","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"note: Note\nThe EDAC formulation keeps the density constant and this eliminates the need for the continuity equation or the use of a summation density to find the pressure. However, in SPH discretizations, mrho is typically used as a proxy for the particle volume. The density of the fluids can therefore be computed using the summation density approach. [19]","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"fluid\", \"entropically_damped_sph\", \"system.jl\")]","category":"page"},{"location":"systems/entropically_damped_sph/#TrixiParticles.EntropicallyDampedSPHSystem","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"TrixiParticles.EntropicallyDampedSPHSystem","text":"EntropicallyDampedSPHSystem(initial_condition, smoothing_kernel,\n smoothing_length, sound_speed;\n pressure_acceleration=inter_particle_averaged_pressure,\n density_calculator=SummationDensity(),\n transport_velocity=nothing,\n alpha=0.5, viscosity=nothing,\n acceleration=ntuple(_ -> 0.0, NDIMS), buffer_size=nothing,\n source_terms=nothing)\n\nSystem for particles of a fluid. As opposed to the weakly compressible SPH scheme, which uses an equation of state, this scheme uses a pressure evolution equation to calculate the pressure. See Entropically Damped Artificial Compressibility for SPH for more details on the method.\n\nArguments\n\ninitial_condition: Initial condition representing the system's particles.\nsound_speed: Speed of sound.\nsmoothing_kernel: Smoothing kernel to be used for this system. See Smoothing Kernels.\nsmoothing_length: Smoothing length to be used for this system. See Smoothing Kernels.\n\nKeyword Arguments\n\nviscosity: Viscosity model for this system (default: no viscosity). Recommended: ViscosityAdami.\nacceleration: Acceleration vector for the system. (default: zero vector)\npressure_acceleration: Pressure acceleration formulation (default: inter-particle averaged pressure). When set to nothing, the pressure acceleration formulation for the corresponding density calculator is chosen.\ndensity_calculator: Density calculator (default: SummationDensity)\ntransport_velocity: Transport Velocity Formulation (TVF). Default is no TVF.\nbuffer_size: Number of buffer particles. This is needed when simulating with OpenBoundarySPHSystem.\nsource_terms: Additional source terms for this system. Has to be either nothing (by default), or a function of (coords, velocity, density, pressure, t) (which are the quantities of a single particle), returning a Tuple or SVector that is to be added to the acceleration of that particle. See, for example, SourceTermDamping. Note that these source terms will not be used in the calculation of the boundary pressure when using a boundary with BoundaryModelDummyParticles and AdamiPressureExtrapolation. The keyword argument acceleration should be used instead for gravity-like source terms.\n\n\n\n\n\n","category":"type"},{"location":"systems/entropically_damped_sph/#transport_velocity_formulation","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Transport Velocity Formulation (TVF)","text":"","category":"section"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"Standard SPH suffers from problems like tensile instability or the creation of void regions in the flow. To address these problems, Adami (2013) modified the advection velocity and added an extra term to the momentum equation. The authors introduced the so-called Transport Velocity Formulation (TVF) for WCSPH. Ramachandran (2019) applied the TVF also for the EDAC scheme.","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"The transport velocity tildev_a of particle a is used to evolve the position of the particle r_a from one time step to the next by","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"fracmathrmd r_amathrmdt = tildev_a","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"and is obtained at every time-step Delta t from","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"tildev_a (t + Delta t) = v_a (t) + Delta t left(fractildemathrmd v_amathrmdt - frac1rho_a nabla p_textbackground right)","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"where rho_a is the density of particle a and p_textbackground is a constant background pressure field. The tilde in the second term of the right hand side indicates that the material derivative has an advection part.","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"The discretized form of the last term is","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":" -frac1rho_a nabla p_textbackground approx -fracp_textbackgroundm_a sum_b left(V_a^2 + V_b^2 right) nabla_a W_ab","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"where V_a, V_b denote the volume of particles a and b respectively. Note that although in the continuous case nabla p_textbackground = 0, the discretization is not 0th-order consistent for non-uniform particle distribution, which means that there is a non-vanishing contribution only when particles are disordered. That also means that p_textbackground occurs as prefactor to correct the trajectory of a particle resulting in uniform pressure distributions. Suggested is a background pressure which is in the order of the reference pressure but can be chosen arbitrarily large when the time-step criterion is adjusted.","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"The inviscid momentum equation with an additional convection term for a particle moving with tildev is","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"fractildemathrmd left( rho v right)mathrmdt = -nabla p + nabla cdot bmA","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"where the tensor bmA = rho vleft(tildev-vright)^T is a consequence of the modified advection velocity and can be interpreted as the convection of momentum with the relative velocity tildev-v.","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"The discretized form of the momentum equation for a particle a reads as","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"fractildemathrmd v_amathrmdt = frac1m_a sum_b left(V_a^2 + V_b^2 right) left -tildep_ab nabla_a W_ab + frac12 left(bmA_a + bmA_b right) cdot nabla_a W_ab right","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"Here, tildep_ab is the density-weighted pressure","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"tildep_ab = fracrho_b p_a + rho_a p_brho_a + rho_b","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"with the density rho_a, rho_b and the pressure p_a, p_b of particles a and b respectively. bmA_a and bmA_b are the convection tensors for particle a and b respectively and are given, e.g. for particle a, as bmA_a = rho v_aleft(tildev_a-v_aright)^T.","category":"page"},{"location":"systems/entropically_damped_sph/","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"Entropically Damped Artificial Compressibility for SPH (Fluid)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"fluid\", \"transport_velocity.jl\")]","category":"page"},{"location":"systems/entropically_damped_sph/#TrixiParticles.TransportVelocityAdami","page":"Entropically Damped Artificial Compressibility for SPH (Fluid)","title":"TrixiParticles.TransportVelocityAdami","text":"TransportVelocityAdami(background_pressure::Real)\n\nTransport Velocity Formulation (TVF) to suppress pairing and tensile instability. See TVF for more details of the method.\n\nArguments\n\nbackground_pressure: Background pressure. Suggested is a background pressure which is on the order of the reference pressure.\n\nnote: Note\nThere is no need for an artificial viscosity to suppress tensile instability when using TransportVelocityAdami. Thus, it is highly recommended to use ViscosityAdami as viscosity model, since ArtificialViscosityMonaghan leads to bad results.\n\n\n\n\n\n","category":"type"},{"location":"authors/","page":"Authors","title":"Authors","text":"EditURL = \"https://github.com/trixi-framework/TrixiParticles.jl/blob/main/AUTHORS.md\"","category":"page"},{"location":"authors/#Authors","page":"Authors","title":"Authors","text":"","category":"section"},{"location":"authors/","page":"Authors","title":"Authors","text":"TrixiParticles.jl's development is coordinated by a group of principal developers, who are also its main contributors and who can be contacted in case of questions about TrixiParticles.jl. In addition, there are contributors who have provided substantial additions or modifications. Together, these two groups form \"The TrixiParticles.jl Authors\" as mentioned under License.","category":"page"},{"location":"authors/#Principal-Developers","page":"Authors","title":"Principal Developers","text":"","category":"section"},{"location":"authors/","page":"Authors","title":"Authors","text":"Erik Faulhaber, University of Cologne, Germany\nNiklas Neher, High-Performance Computing Center Stuttgart (HLRS), Germany\nSven Berger, Helmholtz Center Hereon, Germany","category":"page"},{"location":"authors/#Contributors","page":"Authors","title":"Contributors","text":"","category":"section"},{"location":"authors/","page":"Authors","title":"Authors","text":"The following people contributed major additions or modifications to TrixiParticles.jl and are listed in alphabetical order:","category":"page"},{"location":"authors/","page":"Authors","title":"Authors","text":"Sven Berger\nErik Faulhaber\nGregor Gassner\nNiklas Neher\nHendrik Ranocha\nMichael Schlottke-Lakemper","category":"page"},{"location":"tutorials_template/tut_dam_break/#Example-file","page":"Example file","title":"Example file","text":"","category":"section"},{"location":"tutorials_template/tut_dam_break/","page":"Example file","title":"Example file","text":"!!include:examples/fluid/dam_break_2d.jl!!\n","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"EditURL = \"https://github.com/trixi-framework/TrixiParticles.jl/blob/main/docs/src/tutorials_template/tut_setup.md\"","category":"page"},{"location":"tutorials/tut_setup/#Setting-up-your-simulation-from-scratch","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"","category":"section"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"In this tutorial, we will guide you through the general structure of simulation files. We will set up a simulation similar to the example simulation examples/fluid/hydrostatic_water_column_2d.jl, which is one of our simplest example simulations. In the second part of this tutorial, we will show how to replace components of TrixiParticles.jl by custom implementations from within a simulation file, without ever cloning the repository.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"For different setups and physics, have a look at our other example files.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"First, we import TrixiParticles.jl and OrdinaryDiffEq.jl, which we will use at the very end for the time integration.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"using TrixiParticles\nusing OrdinaryDiffEq","category":"page"},{"location":"tutorials/tut_setup/#Resolution","page":"Setting up your simulation from scratch","title":"Resolution","text":"","category":"section"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"Now, we define the particle spacing, which is our numerical resolution. For a fluid, we usually call the variable fluid_particle_spacing, so that we can easily change the resolution of an example file by overwriting this variable with trixi_include. In 2D, the number of particles will grow quadratically, in 3D cubically with the spacing.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"We also set the number of boundary layers, which need to be sufficiently large, depending on the smoothing kernel and smoothing length, so that the compact support of the smoothing kernel is fully sampled with particles for a fluid particle close to a boundary. In particular, we require the boundary thickness boundary_layers * fluid_particle_spacing to be larger than the compact support of the kernel. The compact support of each kernel can be found in the smoothing kernel overview.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"fluid_particle_spacing = 0.05\nboundary_layers = 3\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/#Experiment-setup","page":"Setting up your simulation from scratch","title":"Experiment setup","text":"","category":"section"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"We want to simulate a water column resting under hydrostatic pressure inside a rectangular tank. (Image: Experiment Setup) First, we define physical parameters like gravitational acceleration, simulation time, initial fluid size, tank size and fluid density.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"gravity = 9.81\ntspan = (0.0, 1.0)\ninitial_fluid_size = (1.0, 0.9)\ntank_size = (1.0, 1.0)\nfluid_density = 1000.0\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"In order to have the initial particle mass and density correspond to the hydrostatic pressure gradient, we need to define a state equation, which relates the fluid density to pressure. Note that we could also skip this part here and define the state equation later when we define the fluid system, but then the fluid would be initialized with constant density, which would cause it to oscillate under gravity.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"sound_speed = 10.0\nstate_equation = StateEquationCole(; sound_speed, reference_density=fluid_density,\n exponent=7)\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"The speed of sound here is numerical and not physical. We artificially lower the speed of sound, since the physical speed of sound in water would lead to prohibitively small time steps. The speed of sound in Weakly Compressible SPH should be chosen as small as possible for numerical efficiency, but large enough to limit density fluctuations to about 1%.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"TrixiParticles.jl requires the initial particle positions and quantities in form of an InitialCondition. Instead of manually defining particle positions, you can work with our pre-defined setups. Among others, we provide setups for rectangular shapes, circles, and spheres. Initial conditions can also be combined with common set operations. See this page for a list of pre-defined setups and details on set operations on initial conditions.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"Here, we use the RectangularTank setup, which generates a rectangular fluid inside a rectangular tank, and supports a hydrostatic pressure gradient by passing a gravitational acceleration and a state equation (see above).","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"tank = RectangularTank(fluid_particle_spacing, initial_fluid_size, tank_size,\n fluid_density, n_layers=boundary_layers,\n acceleration=(0.0, -gravity), state_equation=state_equation)\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/#Fluid-system","page":"Setting up your simulation from scratch","title":"Fluid system","text":"","category":"section"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"To model the water column, we use the Weakly Compressible Smoothed Particle Hydrodynamics (WCSPH) method. This method requires a smoothing kernel and a corresponding smoothing length, which should be chosen in relation to the particle spacing.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"smoothing_length = 1.2 * fluid_particle_spacing\nsmoothing_kernel = SchoenbergCubicSplineKernel{2}()\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"You can find an overview over smoothing kernels and corresponding smoothing lengths here.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"For stability, we need numerical dissipation in form of an artificial viscosity term. Other viscosity models offer a physical approach based on the kinematic viscosity of the fluid.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"viscosity = ArtificialViscosityMonaghan(alpha=0.02, beta=0.0)\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"We choose the parameters as small as possible to avoid non-physical behavior, but as large as possible to stabilize the simulation.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"The WCSPH method can either compute the particle density directly with a kernel summation over all neighboring particles (see SummationDensity) or by making the particle density a variable in the ODE system and integrating its change over time. We choose the latter approach here by using the density calculator ContinuityDensity, which is more efficient and handles free surfaces without the need for additional correction terms.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"fluid_density_calculator = ContinuityDensity()\nfluid_system = WeaklyCompressibleSPHSystem(tank.fluid, fluid_density_calculator,\n state_equation, smoothing_kernel,\n smoothing_length, viscosity=viscosity,\n acceleration=(0.0, -gravity))\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/#Boundary-system","page":"Setting up your simulation from scratch","title":"Boundary system","text":"","category":"section"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"To model the boundary, we use particle-based boundary conditions, in which particles are sampled in the boundary that interact with the fluid particles to avoid penetration. In order to define a boundary system, we first have to choose a boundary model, which defines how the fluid interacts with boundary particles. We will use the BoundaryModelDummyParticles with AdamiPressureExtrapolation. See here for a comprehensive overview over boundary models.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"boundary_model = BoundaryModelDummyParticles(tank.boundary.density, tank.boundary.mass,\n state_equation=state_equation,\n AdamiPressureExtrapolation(),\n smoothing_kernel, smoothing_length)\nboundary_system = BoundarySPHSystem(tank.boundary, boundary_model)\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/#Semidiscretization","page":"Setting up your simulation from scratch","title":"Semidiscretization","text":"","category":"section"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"The key component of every simulation is the Semidiscretization, which couples all systems of the simulation. All simulation methods in TrixiParticles.jl are semidiscretizations, which discretize the equations in time to provide an ordinary differential equation that still has to be solved in time. By providing a simulation time span, we can call semidiscretize, which returns an ODEProblem that can be solved with a time integration method.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"semi = Semidiscretization(fluid_system, boundary_system)\node = semidiscretize(semi, tspan)\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/#Time-integration","page":"Setting up your simulation from scratch","title":"Time integration","text":"","category":"section"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"We use the methods provided by OrdinaryDiffEq.jl, but note that other packages or custom implementations can also be used.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"OrdinaryDiffEq.jl supports callbacks, which are executed during the simulation. For this simulation, we use the InfoCallback, which prints information about the simulation setup at the beginning of the simulation, information about the current simulation time and runtime during the simulation, and a performance summary at the end of the simulation. We also want to save the current solution in regular intervals in terms of simulation time as VTK, so that we can look at the solution in ParaView. The SolutionSavingCallback provides this functionality. To pass the callbacks to OrdinaryDiffEq.jl, we have to bundle them into a CallbackSet.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"info_callback = InfoCallback(interval=50)\nsaving_callback = SolutionSavingCallback(dt=0.02)\n\ncallbacks = CallbackSet(info_callback, saving_callback)\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"Finally, we can start the simulation by solving the ODEProblem. We use the method RDPK3SpFSAL35 of OrdinaryDiffEq.jl, which is a Runge-Kutta method with automatic (error based) time step size control. This method is usually a good choice for prototyping, since we do not have to worry about choosing a stable step size and can just run the simulation. For better performance, it might be beneficial to tweak the tolerances of this method or choose a different method that is more efficient for the respective simulation. You can find both approaches in our example files. Here, we just use the method with the default parameters, and only disable save_everystep to avoid expensive saving of the solution in every time step.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"sol = solve(ode, RDPK3SpFSAL35(), save_everystep=false, callback=callbacks);\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"See Visualization for how to visualize the solution. For the simplest visualization, we can use Plots.jl:","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"using Plots\nplot(sol)\nplot!(dpi=200); savefig(\"tut_setup_plot.png\"); nothing # hide","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"(Image: plot)","category":"page"},{"location":"tutorials/tut_setup/#Replacing-components-with-custom-implementations","page":"Setting up your simulation from scratch","title":"Replacing components with custom implementations","text":"","category":"section"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"If we would like to use an implementation of a component that is not available in TrixiParticles.jl, we can implement it ourselves within the simulation file, without ever cloning the TrixiParticles.jl repository. A good starting point is to check out the available implementations in TrixiParticles.jl, then copy the relevant functions to the simulation file and modify them as needed.","category":"page"},{"location":"tutorials/tut_setup/#Custom-smoothing-kernel","page":"Setting up your simulation from scratch","title":"Custom smoothing kernel","text":"","category":"section"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"To implement a custom smoothing kernel, we define a struct extending TrixiParticles.SmoothingKernel. This abstract struct has a type parameter for the number of dimensions, which we set to 2 in this case.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"struct MyGaussianKernel <: TrixiParticles.SmoothingKernel{2} end","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"This kernel is going to be an implementation of the Gaussian kernel with a cutoff for compact support. Note that the same kernel in a more optimized version is already implemented in TrixiParticles.jl as GaussianKernel.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"In order to use our new kernel, we have to define three functions. TrixiParticles.kernel, which is the kernel function itself, TrixiParticles.kernel_deriv, which is the derivative of the kernel function, and TrixiParticles.compact_support, which defines the compact support of the kernel in relation to the smoothing length. The latter is relevant for determining the search radius of the neighborhood search.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"function TrixiParticles.kernel(kernel::MyGaussianKernel, r, h)\n q = r / h\n\n if q < 2\n return 1 / (pi * h^2) * exp(-q^2)\n end\n\n return 0.0\nend\n\nfunction TrixiParticles.kernel_deriv(kernel::MyGaussianKernel, r, h)\n q = r * h\n\n if q < 2\n return 1 / (pi * h^2) * (-2 * q) * exp(-q^2) / h\n end\n\n return 0.0\nend\n\nTrixiParticles.compact_support(::MyGaussianKernel, h) = 3 * h","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"For this kernel, we use a different smoothing length, which yields a similar kernel to the SchoenbergCubicSplineKernel that we used earlier.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"smoothing_length_gauss = 1.0 * fluid_particle_spacing","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"We can compare these kernels in a plot.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"using Plots\nx = range(-0.2, 0.2, length=500)\nplot(x, r -> TrixiParticles.kernel(SchoenbergCubicSplineKernel{2}(), abs(r), smoothing_length),\n label=\"SchoenbergCubicSplineKernel\", xlabel=\"r\")\nplot!(x, r -> TrixiParticles.kernel(MyGaussianKernel(), abs(r), smoothing_length_gauss),\n label=\"MyGaussianKernel\")\nplot!(dpi=200); savefig(\"tut_setup_plot2.png\"); nothing # hide","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"(Image: plot)","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"This is all we need to use our custom kernel implementation in a simulation. We only need to replace the definition above by","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"smoothing_kernel = MyGaussianKernel()\nnothing # hide","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"and run the simulation file again.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"In order to use our kernel in a pre-defined example file, we can use the function trixi_include to replace the definition of the variable smoothing_kernel. The following will run the example simulation examples/fluid/hydrostatic_water_column_2d.jl with our custom kernel.","category":"page"},{"location":"tutorials/tut_setup/","page":"Setting up your simulation from scratch","title":"Setting up your simulation from scratch","text":"trixi_include(joinpath(examples_dir(), \"fluid\", \"hydrostatic_water_column_2d.jl\"),\n smoothing_kernel=MyGaussianKernel());\nnothing # hide","category":"page"},{"location":"systems/total_lagrangian_sph/#tlsph","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH","text":"","category":"section"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"A Total Lagrangian framework is used wherein the governing equations are formulated such that all relevant quantities and operators are measured with respect to the initial configuration (O’Connor & Rogers, 2021, Belytschko et al., 2000).","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"The governing equations with respect to the initial configuration are given by:","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"fracmathrmDbmvmathrmDt = frac1rho_0 nabla_0 cdot bmP + bmg","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"where the zero subscript denotes a derivative with respect to the initial configuration and bmP is the first Piola-Kirchhoff (PK1) stress tensor.","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"The discretized version of this equation is given by O’Connor & Rogers (2021):","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"fracmathrmdbmv_amathrmdt = sum_b m_0b\n left( fracbmP_a bmL_0arho_0a^2 + fracbmP_b bmL_0brho_0b^2 right)\n nabla_0a W(bmX_ab) + fracbmf_a^PFm_0a + bmg","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"with the correction matrix (see also GradientCorrection)","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"bmL_0a = left( -sum_b fracm_0brho_0b nabla_0a W(bmX_ab) bmX_ab^T right)^-1 in R^d times d","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"The subscripts a and b denote quantities of particle a and b, respectively. The zero subscript on quantities denotes that the quantity is to be measured in the initial configuration. The difference in the initial coordinates is denoted by bmX_ab = bmX_a - bmX_b, the difference in the current coordinates is denoted by bmx_ab = bmx_a - bmx_b.","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"For the computation of the PK1 stress tensor, the deformation gradient bmF is computed per particle as","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"bmF_a = sum_b fracm_0brho_0b bmx_ba (bmL_0anabla_0a W(bmX_ab))^T \n qquad = -left(sum_b fracm_0brho_0b bmx_ab (nabla_0a W(bmX_ab))^T right) bmL_0a^T","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"with 1 leq ij leq d. From the deformation gradient, the Green-Lagrange strain","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"bmE = frac12(bmF^TbmF - bmI)","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"and the second Piola-Kirchhoff stress tensor","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"bmS = lambda operatornametr(bmE) bmI + 2mu bmE","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"are computed to obtain the PK1 stress tensor as","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"bmP = bmFbmS","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"Here,","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"mu = fracE2(1 + nu)","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"and","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"lambda = fracEnu(1 + nu)(1 - 2nu)","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"are the Lamé coefficients, where E is the Young's modulus and nu is the Poisson ratio.","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"The term bmf_a^PF is an optional penalty force. See e.g. PenaltyForceGanzenmueller.","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"solid\", \"total_lagrangian_sph\", \"system.jl\")]","category":"page"},{"location":"systems/total_lagrangian_sph/#TrixiParticles.TotalLagrangianSPHSystem","page":"Total Lagrangian SPH (Elastic Structure)","title":"TrixiParticles.TotalLagrangianSPHSystem","text":"TotalLagrangianSPHSystem(initial_condition,\n smoothing_kernel, smoothing_length,\n young_modulus, poisson_ratio;\n n_fixed_particles=0, boundary_model=nothing,\n acceleration=ntuple(_ -> 0.0, NDIMS),\n penalty_force=nothing, source_terms=nothing)\n\nSystem for particles of an elastic structure.\n\nA Total Lagrangian framework is used wherein the governing equations are formulated such that all relevant quantities and operators are measured with respect to the initial configuration (O’Connor & Rogers 2021, Belytschko et al. 2000). See Total Lagrangian SPH for more details on the method.\n\nArguments\n\ninitial_condition: Initial condition representing the system's particles.\nyoung_modulus: Young's modulus.\npoisson_ratio: Poisson ratio.\nsmoothing_kernel: Smoothing kernel to be used for this system. See Smoothing Kernels.\nsmoothing_length: Smoothing length to be used for this system. See Smoothing Kernels.\n\nKeyword Arguments\n\nn_fixed_particles: Number of fixed particles which are used to clamp the structure particles. Note that the fixed particles must be the last particles in the InitialCondition. See the info box below.\nboundary_model: Boundary model to compute the hydrodynamic density and pressure for fluid-structure interaction (see Boundary Models).\npenalty_force: Penalty force to ensure regular particle position under large deformations (see PenaltyForceGanzenmueller).\nacceleration: Acceleration vector for the system. (default: zero vector)\nsource_terms: Additional source terms for this system. Has to be either nothing (by default), or a function of (coords, velocity, density, pressure) (which are the quantities of a single particle), returning a Tuple or SVector that is to be added to the acceleration of that particle. See, for example, SourceTermDamping.\n\nnote: Note\nThe fixed particles must be the last particles in the InitialCondition. To do so, e.g. use the union function:solid = union(beam, fixed_particles)where beam and fixed_particles are of type InitialCondition.\n\n\n\n\n\n","category":"type"},{"location":"systems/total_lagrangian_sph/#Penalty-Force","page":"Total Lagrangian SPH (Elastic Structure)","title":"Penalty Force","text":"","category":"section"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"In FEM, underintegrated elements can deform without an associated increase of energy. This is caused by the stiffness matrix having zero eigenvalues (so-called hourglass modes). The name \"hourglass modes\" comes from the fact that elements can deform into an hourglass shape.","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"Similar effects can occur in SPH as well. Particles can change positions without changing the SPH approximation of the deformation gradient bmF, thus, without causing an increase of energy. To ensure regular particle positions, we can apply similar correction forces as are used in FEM.","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"Ganzenmüller (2015) introduced a so-called hourglass correction force or penalty force f^PF, which is given by","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"bmf_a^PF = frac12 alpha sum_b fracm_0a m_0b W_0abrho_0arho_0b bmX_ab^2\n left( E delta_ab^a + E delta_ba^b right) fracbmx_abbmx_ab","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"The subscripts a and b denote quantities of particle a and b, respectively. The zero subscript on quantities denotes that the quantity is to be measured in the initial configuration. The difference in the initial coordinates is denoted by bmX_ab = bmX_a - bmX_b, the difference in the current coordinates is denoted by bmx_ab = bmx_a - bmx_b. Note that Ganzenmüller (2015) has a flipped sign here because they define bmx_ab the other way around.","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"This correction force is based on the potential energy density of a Hookean material. Thus, E is the Young's modulus and alpha is a dimensionless coefficient that controls the amplitude of hourglass correction. The separation vector delta_ab^a indicates the change of distance which the particle separation should attain in order to minimize the error and is given by","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":" delta_ab^a = fracbmepsilon_ab^a cdot bmx_abbmx_ab","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"where the error vector is defined as","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":" bmepsilon_ab^a = bmF_a bmX_ab - bmx_ab","category":"page"},{"location":"systems/total_lagrangian_sph/","page":"Total Lagrangian SPH (Elastic Structure)","title":"Total Lagrangian SPH (Elastic Structure)","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"schemes\", \"solid\", \"total_lagrangian_sph\", \"penalty_force.jl\")]","category":"page"},{"location":"systems/total_lagrangian_sph/#TrixiParticles.PenaltyForceGanzenmueller","page":"Total Lagrangian SPH (Elastic Structure)","title":"TrixiParticles.PenaltyForceGanzenmueller","text":"PenaltyForceGanzenmueller(; alpha=0.1)\n\nPenalty force to ensure regular particle positions under large deformations.\n\nKeywords\n\nalpha: Coefficient to control the amplitude of hourglass correction.\n\n\n\n\n\n","category":"type"},{"location":"license/","page":"License","title":"License","text":"EditURL = \"https://github.com/trixi-framework/TrixiParticles.jl/blob/main/LICENSE.md\"","category":"page"},{"location":"license/#License","page":"License","title":"License","text":"","category":"section"},{"location":"license/","page":"License","title":"License","text":"MIT LicenseCopyright (c) 2023-present The TrixiParticles.jl Authors (see Authors) \nCopyright (c) 2023-present Helmholtz-Zentrum hereon GmbH, Institute of Surface Science \n \nPermission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the \"Software\"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.","category":"page"},{"location":"getting_started/#getting_started","page":"Getting started","title":"Getting started","text":"","category":"section"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"If you have not installed TrixiParticles.jl, please follow the instructions given here.","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"In the following sections, we will give a short introduction. For a more thorough discussion, take a look at our Tutorials.","category":"page"},{"location":"getting_started/#Running-an-Example","page":"Getting started","title":"Running an Example","text":"","category":"section"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"The easiest way to run a simulation is to run one of our predefined example files. We will run the file examples/fluid/hydrostatic_water_column_2d.jl, which simulates a fluid resting in a rectangular tank. Since TrixiParticles.jl uses multithreading, you should start Julia with the flag --threads auto (or, e.g. --threads 4 for 4 threads).","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"In the Julia REPL, first load the package TrixiParticles.jl.","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"julia> using TrixiParticles","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"Then start the simulation by executing","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"julia> trixi_include(joinpath(examples_dir(), \"fluid\", \"hydrostatic_water_column_2d.jl\"))","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"The easiest way to quickly visualize the result is to use Plots.jl:","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"julia> using Plots; plot(sol)","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"This will open a new window with a 2D visualization of the final solution: (Image: plot_hydrostatic_water_column)","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"For more information about visualization, see Visualization.","category":"page"},{"location":"getting_started/#Running-other-Examples","page":"Getting started","title":"Running other Examples","text":"","category":"section"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"You can find a list of our other predefined examples under Examples. Execute them as follows from the Julia REPL by replacing subfolder and example_name","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"julia> trixi_include(joinpath(examples_dir(), \"subfolder\", \"example_name.jl\"))","category":"page"},{"location":"getting_started/#Modifying-an-example","page":"Getting started","title":"Modifying an example","text":"","category":"section"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"You can pass keyword arguments to the function trixi_include to overwrite assignments in the file.","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"With trixi_include, we can overwrite variables defined in the example file to run a different simulation without modifying the example file.","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"julia> trixi_include(joinpath(examples_dir(), \"fluid\", \"hydrostatic_water_column_2d.jl\"), initial_fluid_size=(1.0, 0.5))","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"This for example, will change the fluid size from (09 10) to (10 05).","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"To understand why, take a look into the file hydrostatic_water_column_2d.jl in the subfolder fluid inside the examples directory, which is the file that we executed earlier. You can see that the initial size of the fluid is defined in the variable initial_fluid_size, which we could overwrite with the trixi_include call above. Another variable that is worth experimenting with is fluid_particle_spacing, which controls the resolution of the simulation in this case. A lower value will increase the resolution and the runtime.","category":"page"},{"location":"getting_started/#Set-up-you-first-simulation-from-scratch","page":"Getting started","title":"Set up you first simulation from scratch","text":"","category":"section"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"See Set up your first simulation.","category":"page"},{"location":"getting_started/","page":"Getting started","title":"Getting started","text":"Find an overview over the available tutorials under Tutorials.","category":"page"},{"location":"examples/#examples","page":"Examples","title":"Examples","text":"","category":"section"},{"location":"examples/#Fluid","page":"Examples","title":"Fluid","text":"","category":"section"},{"location":"examples/#Accelerated-Tank-2D-(fluid/accelerated_tank_2d.jl)","page":"Examples","title":"Accelerated Tank 2D (fluid/accelerated_tank_2d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Dam-Break-2D-(fluid/dam_break_2d.jl)","page":"Examples","title":"Dam Break 2D (fluid/dam_break_2d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Dam-Break-3D-(fluid/dam_break_3d.jl)","page":"Examples","title":"Dam Break 3D (fluid/dam_break_3d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Falling-Water-Column-(fluid/falling_water_column_2d.jl)","page":"Examples","title":"Falling Water Column (fluid/falling_water_column_2d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Hydrostatic-Water-Column-(fluid/hydrostatic_water_column_*.jl)","page":"Examples","title":"Hydrostatic Water Column (fluid/hydrostatic_water_column_*.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Moving-Wall-(fluid/moving_wall_2d.jl)","page":"Examples","title":"Moving Wall (fluid/moving_wall_2d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Oscillating-Drop-(fluid/oscillating_drop_2d.jl)","page":"Examples","title":"Oscillating Drop (fluid/oscillating_drop_2d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Periodic-Channel-(fluid/periodic_channel_2d.jl)","page":"Examples","title":"Periodic Channel (fluid/periodic_channel_2d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Fluid-Structure-Interaction","page":"Examples","title":"Fluid Structure Interaction","text":"","category":"section"},{"location":"examples/#Dam-Break-with-Elastic-Plate-(fsi/dam_break_plate_2d.jl)","page":"Examples","title":"Dam Break with Elastic Plate (fsi/dam_break_plate_2d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Falling-Sphere-2D-(fsi/falling_sphere_2d.jl)","page":"Examples","title":"Falling Sphere 2D (fsi/falling_sphere_2d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Falling-Spheres-2D-(fsi/falling_spheres_2d.jl)","page":"Examples","title":"Falling Spheres 2D (fsi/falling_spheres_2d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"examples/#Structure-Mechanics","page":"Examples","title":"Structure Mechanics","text":"","category":"section"},{"location":"examples/#Oscillating-Beam-(solid/oscillating_beam_2d.jl)","page":"Examples","title":"Oscillating Beam (solid/oscillating_beam_2d.jl)","text":"","category":"section"},{"location":"examples/","page":"Examples","title":"Examples","text":"","category":"page"},{"location":"overview/#Overview","page":"Overview","title":"Overview","text":"","category":"section"},{"location":"overview/","page":"Overview","title":"Overview","text":"The actual API reference is not listed on a single page, like in most Julia packages, but instead is split into multiple sections that follow a similar structure as the code files themselves. In these sections, API docs are combined with explanations of the theoretical background of these methods.","category":"page"},{"location":"overview/","page":"Overview","title":"Overview","text":"The following page gives a rough overview of important parts of the code.","category":"page"},{"location":"overview/#Program-flow","page":"Overview","title":"Program flow","text":"","category":"section"},{"location":"overview/","page":"Overview","title":"Overview","text":"To initiate a simulation, the goal is to solve an ordinary differential equation, for example, by employing the time integration schemes provided by OrdinaryDiffEq.jl. These schemes are then utilized to integrate mathrmdumathrmdt and mathrmdvmathrmdt, where u represents the particles' positions and v their properties such as velocity and density. During a single time step or an intermediate step of the time integration scheme, the functions drift! and kick! are invoked, followed by the functions depicted in this diagram (with key parts highlighted in orange/yellow).","category":"page"},{"location":"overview/","page":"Overview","title":"Overview","text":"(Image: Main Program Flow)","category":"page"},{"location":"overview/#Structure","page":"Overview","title":"Structure","text":"","category":"section"},{"location":"overview/","page":"Overview","title":"Overview","text":"What we refer to as schemes are various models such as Weakly Compressible Smoothed Particle Hydrodynamics (WCSPH) or Total Lagrangian Smoothed Particle Hydrodynamics (TLSPH). These schemes are categorized based on the applicable physical regimes, namely fluid, solid, gas, and others. Each scheme comprises at least two files: a system.jl file and an rhs.jl file. The system.jl file provides the data structure holding the particles of this scheme and some routines, particularly those for allocation and the main update routines, excluding system interactions. The interactions between particles of this scheme (and with particles of other schemes) are handled in the rhs.jl file.","category":"page"},{"location":"callbacks/#Callbacks","page":"Callbacks","title":"Callbacks","text":"","category":"section"},{"location":"callbacks/","page":"Callbacks","title":"Callbacks","text":"Modules = [TrixiParticles]\nPages = map(file -> joinpath(\"callbacks\", file), readdir(joinpath(\"..\", \"src\", \"callbacks\")))","category":"page"},{"location":"callbacks/#TrixiParticles.DensityReinitializationCallback","page":"Callbacks","title":"TrixiParticles.DensityReinitializationCallback","text":"DensityReinitializationCallback(; interval::Integer=0, dt=0.0)\n\nCallback to reinitialize the density field when using ContinuityDensity [42].\n\nKeywords\n\ninterval=0: Reinitialize the density every interval time steps.\ndt: Reinitialize the density in regular intervals of dt in terms of integration time.\nreinit_initial_solution: Reinitialize the initial solution (default=false)\n\n\n\n\n\n","category":"type"},{"location":"callbacks/#TrixiParticles.InfoCallback-Tuple{}","page":"Callbacks","title":"TrixiParticles.InfoCallback","text":"InfoCallback()\n\nCreate and return a callback that prints a human-readable summary of the simulation setup at the beginning of a simulation and then resets the timer. When the returned callback is executed directly, the current timer values are shown.\n\n\n\n\n\n","category":"method"},{"location":"callbacks/#TrixiParticles.PostprocessCallback","page":"Callbacks","title":"TrixiParticles.PostprocessCallback","text":"PostprocessCallback(; interval::Integer=0, dt=0.0, exclude_boundary=true, filename=\"values\",\n output_directory=\"out\", append_timestamp=false, write_csv=true,\n write_json=true, write_file_interval=1, funcs...)\n\nCreate a callback to post-process simulation data at regular intervals. This callback allows for the execution of a user-defined function func at specified intervals during the simulation. The function is applied to the current state of the simulation, and its results can be saved or used for further analysis. The provided function cannot be anonymous as the function name will be used as part of the name of the value.\n\nThe callback can be triggered either by a fixed number of time steps (interval) or by a fixed interval of simulation time (dt).\n\nKeywords\n\nfuncs...: Functions to be executed at specified intervals during the simulation. Each function must have the arguments (v, u, t, system), and will be called for every system, where v and u are the wrapped solution arrays for the corresponding system and t is the current simulation time. Note that working with these v and u arrays requires undocumented internal functions of TrixiParticles. See Custom Quantities for a list of pre-defined functions that can be used here.\ninterval=0: Specifies the number of time steps between each invocation of the callback. If set to 0, the callback will not be triggered based on time steps. Either interval or dt must be set to something larger than 0.\ndt=0.0: Specifies the simulation time interval between each invocation of the callback. If set to 0.0, the callback will not be triggered based on simulation time. Either interval or dt must be set to something larger than 0.\nexclude_boundary=true: If set to true, boundary particles will be excluded from the post-processing.\nfilename=\"values\": The filename of the postprocessing files to be saved.\noutput_directory=\"out\": The path where the results of the post-processing will be saved.\nwrite_csv=true: If set to true, write a csv file.\nwrite_json=true: If set to true, write a json file.\nappend_timestep=false: If set to true, the current timestamp will be added to the filename.\nwrite_file_interval=1: Files will be written after every write_file_interval number of postprocessing execution steps. A value of 0 indicates that files are only written at the end of the simulation, eliminating I/O overhead.\n\nExamples\n\n# Create a callback that is triggered every 100 time steps\npostprocess_callback = PostprocessCallback(interval=100, example_quantity=kinetic_energy)\n\n# Create a callback that is triggered every 0.1 simulation time units\npostprocess_callback = PostprocessCallback(dt=0.1, example_quantity=kinetic_energy)\n\n\n\n\n\n","category":"type"},{"location":"callbacks/#TrixiParticles.SolutionSavingCallback","page":"Callbacks","title":"TrixiParticles.SolutionSavingCallback","text":"SolutionSavingCallback(; interval::Integer=0, dt=0.0, save_times=Array{Float64, 1}([]),\n save_initial_solution=true, save_final_solution=true,\n output_directory=\"out\", append_timestamp=false, prefix=\"\",\n verbose=false, write_meta_data=true, max_coordinates=2^15,\n custom_quantities...)\n\nCallback to save the current numerical solution in VTK format in regular intervals. Either pass interval to save every interval time steps, or pass dt to save in intervals of dt in terms of integration time by adding additional tstops (note that this may change the solution).\n\nAdditional user-defined quantities can be saved by passing functions as keyword arguments, which map (v, u, t, system) to an Array where the columns represent the particles in the same order as in u. To ignore a custom quantity for a specific system, return nothing.\n\nKeywords\n\ninterval=0: Save the solution every interval time steps.\ndt: Save the solution in regular intervals of dt in terms of integration time by adding additional tstops (note that this may change the solution).\nsave_times=[] List of times at which to save a solution.\nsave_initial_solution=true: Save the initial solution.\nsave_final_solution=true: Save the final solution.\noutput_directory=\"out\": Directory to save the VTK files.\nappend_timestamp=false: Append current timestamp to the output directory.\n'prefix=\"\"': Prefix added to the filename.\ncustom_quantities...: Additional user-defined quantities.\nwrite_meta_data=true: Write meta data.\nverbose=false: Print to standard IO when a file is written.\nmax_coordinates=2^15: The coordinates of particles will be clipped if their absolute values exceed this threshold.\ncustom_quantities...: Additional custom quantities to include in the VTK output. Each custom quantity must be a function of (v, u, t, system), which will be called for every system, where v and u are the wrapped solution arrays for the corresponding system and t is the current simulation time. Note that working with these v and u arrays requires undocumented internal functions of TrixiParticles. See Custom Quantities for a list of pre-defined custom quantities that can be used here.\n\nExamples\n\n# Save every 100 time steps\nsaving_callback = SolutionSavingCallback(interval=100)\n\n# Save in intervals of 0.1 in terms of simulation time\nsaving_callback = SolutionSavingCallback(dt=0.1)\n\n# Additionally store the kinetic energy of each system as \"my_custom_quantity\"\nsaving_callback = SolutionSavingCallback(dt=0.1, my_custom_quantity=kinetic_energy)\n\n\n\n\n\n","category":"type"},{"location":"callbacks/#TrixiParticles.SteadyStateReachedCallback","page":"Callbacks","title":"TrixiParticles.SteadyStateReachedCallback","text":"SteadyStateReachedCallback(; interval::Integer=0, dt=0.0,\n interval_size::Integer=10, abstol=1.0e-8, reltol=1.0e-6)\n\nTerminates the integration when the change of kinetic energy between time steps falls below the threshold specified by abstol + reltol * ekin, where ekin is the total kinetic energy of the simulation.\n\nKeywords\n\ninterval=0: Check steady state condition every interval time steps.\ndt=0.0: Check steady state condition in regular intervals of dt in terms of integration time by adding additional tstops (note that this may change the solution).\ninterval_size: The interval in which the change of the kinetic energy is considered. interval_size is a (integer) multiple of interval or dt.\nabstol: Absolute tolerance.\nreltol: Relative tolerance.\n\n\n\n\n\n","category":"type"},{"location":"callbacks/#TrixiParticles.StepsizeCallback-Tuple{}","page":"Callbacks","title":"TrixiParticles.StepsizeCallback","text":"StepsizeCallback(; cfl::Real)\n\nSet the time step size according to a CFL condition if the time integration method isn't adaptive itself.\n\nThe current implementation is using the simplest form of CFL condition, which chooses a time step size that is constant during the simulation. The step size is therefore only applied once at the beginning of the simulation.\n\nThe step size Delta t is chosen as the minimum\n\n Delta t = min(Delta t_eta Delta t_a Delta t_c)\n\nwhere\n\n Delta t_eta = 0125 h^2 eta quad Delta t_a = 025 sqrth lVert g rVert\n quad Delta t_c = textCFL h c\n\nwith nu = alpha h c (2n + 4), where alpha is the parameter of the viscosity and n is the number of dimensions.\n\nwarning: Experimental implementation\nThis is an experimental feature and may change in future releases.\n\nReferences\n\n[21], [14], [43], [44]\n\n\n\n\n\n","category":"method"},{"location":"callbacks/#TrixiParticles.UpdateCallback-Tuple{}","page":"Callbacks","title":"TrixiParticles.UpdateCallback","text":"UpdateCallback(; interval::Integer, dt=0.0)\n\nCallback to update quantities either at the end of every interval time steps or in intervals of dt in terms of integration time by adding additional tstops (note that this may change the solution).\n\nKeywords\n\ninterval=1: Update quantities at the end of every interval time steps.\ndt: Update quantities in regular intervals of dt in terms of integration time by adding additional tstops (note that this may change the solution).\n\n\n\n\n\n","category":"method"},{"location":"callbacks/#custom_quantities","page":"Callbacks","title":"Custom Quantities","text":"","category":"section"},{"location":"callbacks/","page":"Callbacks","title":"Callbacks","text":"The following pre-defined custom quantities can be used with the SolutionSavingCallback and PostprocessCallback.","category":"page"},{"location":"callbacks/","page":"Callbacks","title":"Callbacks","text":"Modules = [TrixiParticles]\nPages = [\"general/custom_quantities.jl\"]","category":"page"},{"location":"callbacks/#TrixiParticles.avg_density-Tuple{Any, Any, Any, TrixiParticles.FluidSystem}","page":"Callbacks","title":"TrixiParticles.avg_density","text":"avg_density\n\nReturns the average_density over all particles in a system.\n\n\n\n\n\n","category":"method"},{"location":"callbacks/#TrixiParticles.avg_pressure-Tuple{Any, Any, Any, TrixiParticles.FluidSystem}","page":"Callbacks","title":"TrixiParticles.avg_pressure","text":"avg_pressure\n\nReturns the average pressure over all particles in a system.\n\n\n\n\n\n","category":"method"},{"location":"callbacks/#TrixiParticles.kinetic_energy-NTuple{4, Any}","page":"Callbacks","title":"TrixiParticles.kinetic_energy","text":"kinetic_energy\n\nReturns the total kinetic energy of all particles in a system.\n\n\n\n\n\n","category":"method"},{"location":"callbacks/#TrixiParticles.max_density-Tuple{Any, Any, Any, TrixiParticles.FluidSystem}","page":"Callbacks","title":"TrixiParticles.max_density","text":"max_density\n\nReturns the maximum density over all particles in a system.\n\n\n\n\n\n","category":"method"},{"location":"callbacks/#TrixiParticles.max_pressure-Tuple{Any, Any, Any, TrixiParticles.FluidSystem}","page":"Callbacks","title":"TrixiParticles.max_pressure","text":"max_pressure\n\nReturns the maximum pressure over all particles in a system.\n\n\n\n\n\n","category":"method"},{"location":"callbacks/#TrixiParticles.min_density-Tuple{Any, Any, Any, TrixiParticles.FluidSystem}","page":"Callbacks","title":"TrixiParticles.min_density","text":"min_density\n\nReturns the minimum density over all particles in a system.\n\n\n\n\n\n","category":"method"},{"location":"callbacks/#TrixiParticles.min_pressure-Tuple{Any, Any, Any, TrixiParticles.FluidSystem}","page":"Callbacks","title":"TrixiParticles.min_pressure","text":"min_pressure\n\nReturns the minimum pressure over all particles in a system.\n\n\n\n\n\n","category":"method"},{"location":"callbacks/#TrixiParticles.total_mass-NTuple{4, Any}","page":"Callbacks","title":"TrixiParticles.total_mass","text":"total_mass\n\nReturns the total mass of all particles in a system.\n\n\n\n\n\n","category":"method"},{"location":"gpu/#GPU-Support","page":"GPU Support","title":"GPU Support","text":"","category":"section"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"GPU support is still an experimental feature that is actively being worked on. As of now, the WeaklyCompressibleSPHSystem and the BoundarySPHSystem are supported on GPUs. We have tested this on GPUs by Nvidia and AMD.","category":"page"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"To run a simulation on a GPU, we need to use the FullGridCellList as cell list for the GridNeighborhoodSearch. This cell list requires a bounding box for the domain, unlike the default cell list, which uses an unbounded domain. For simulations that are bounded by a closed tank, we can use the boundary of the tank to obtain the bounding box as follows.","category":"page"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"search_radius = TrixiParticles.compact_support(smoothing_kernel, smoothing_length)\nmin_corner = minimum(tank.boundary.coordinates, dims=2) .- search_radius\nmax_corner = maximum(tank.boundary.coordinates, dims=2) .+ search_radius\ncell_list = TrixiParticles.PointNeighbors.FullGridCellList(; min_corner, max_corner)","category":"page"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"We then need to pass this cell list to the neighborhood search and the neighborhood search to the Semidiscretization.","category":"page"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"semi = Semidiscretization(fluid_system, boundary_system,\n neighborhood_search=GridNeighborhoodSearch{2}(; cell_list))","category":"page"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"At this point, we should run the simulation and make sure that it still works and that the bounding box is large enough. For some simulations where particles move outside the initial tank coordinates, for example when the tank is not closed or when the tank is moving, an appropriate bounding box has to be specified.","category":"page"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"Then, we only need to specify the data type that is used for the simulation. On an Nvidia GPU, we specify:","category":"page"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"using CUDA\node = semidiscretize(semi, tspan, data_type=CuArray)","category":"page"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"On an AMD GPU, we use:","category":"page"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"using AMDGPU\node = semidiscretize(semi, tspan, data_type=ROCArray)","category":"page"},{"location":"gpu/","page":"GPU Support","title":"GPU Support","text":"Then, we can run the simulation as usual. All data is transferred to the GPU during initialization and all loops over particles and their neighbors will be executed on the GPU as kernels generated by KernelAbstractions.jl. Data is only copied to the CPU for saving VTK files via the SolutionSavingCallback.","category":"page"},{"location":"general/interpolation/#Interpolation","page":"Interpolation","title":"Interpolation","text":"","category":"section"},{"location":"general/interpolation/","page":"Interpolation","title":"Interpolation","text":"Modules = [TrixiParticles]\nPages = [joinpath(\"general\", \"interpolation.jl\")]","category":"page"},{"location":"general/interpolation/#TrixiParticles.interpolate_line-Tuple{Any, Any, Any, Any, Any, ODESolution}","page":"Interpolation","title":"TrixiParticles.interpolate_line","text":"interpolate_line(start, end_, n_points, semi, ref_system, sol; endpoint=true,\n smoothing_length=ref_system.smoothing_length, cut_off_bnd=true,\n clip_negative_pressure=false)\n\nInterpolates properties along a line in a TrixiParticles simulation. The line interpolation is accomplished by generating a series of evenly spaced points between start and end_. If endpoint is false, the line is interpolated between the start and end points, but does not include these points.\n\nSee also: interpolate_point, interpolate_plane_2d, interpolate_plane_2d_vtk, interpolate_plane_3d.\n\nArguments\n\nstart: The starting point of the line.\nend_: The ending point of the line.\nn_points: The number of points to interpolate along the line.\nsemi: The semidiscretization used for the simulation.\nref_system: The reference system for the interpolation.\nsol: The solution state from which the properties are interpolated.\n\nKeywords\n\nendpoint=true: A boolean to include (true) or exclude (false) the end point in the interpolation.\nsmoothing_length=ref_system.smoothing_length: The smoothing length used in the interpolation.\ncut_off_bnd=true: Boolean to indicate if quantities should be set to NaN when the point is \"closer\" to the boundary than to the fluid in a kernel-weighted sense. Or, in more detail, when the boundary has more influence than the fluid on the density summation in this point, i.e., when the boundary particles add more kernel-weighted mass than the fluid particles.\nclip_negative_pressure=false: One common approach in SPH models is to clip negative pressure values, but this is unphysical. Instead we clip here during interpolation thus only impacting the local interpolated value.\n\nReturns\n\nA NamedTuple of arrays containing interpolated properties at each point along the line.\n\nnote: Note\nThis function is particularly useful for analyzing gradients or creating visualizations along a specified line in the SPH simulation domain.\nThe interpolation accuracy is subject to the density of particles and the chosen smoothing length.\nWith cut_off_bnd, a density-based estimation of the surface is used which is not as accurate as a real surface reconstruction.\n\nExamples\n\n# Interpolating along a line from [1.0, 0.0] to [1.0, 1.0] with 5 points\nresults = interpolate_line([1.0, 0.0], [1.0, 1.0], 5, semi, ref_system, sol)\n\n\n\n\n\n","category":"method"},{"location":"general/interpolation/#TrixiParticles.interpolate_plane_2d-Tuple{Any, Any, Any, Any, Any, ODESolution}","page":"Interpolation","title":"TrixiParticles.interpolate_plane_2d","text":"interpolate_plane_2d(min_corner, max_corner, resolution, semi, ref_system, sol;\n smoothing_length=ref_system.smoothing_length, cut_off_bnd=true,\n clip_negative_pressure=false)\n\nInterpolates properties along a plane in a TrixiParticles simulation. The region for interpolation is defined by its lower left and top right corners, with a specified resolution determining the density of the interpolation points.\n\nThe function generates a grid of points within the defined region, spaced uniformly according to the given resolution.\n\nSee also: interpolate_plane_2d_vtk, interpolate_plane_3d, interpolate_line, interpolate_point.\n\nArguments\n\nmin_corner: The lower left corner of the interpolation region.\nmax_corner: The top right corner of the interpolation region.\nresolution: The distance between adjacent interpolation points in the grid.\nsemi: The semidiscretization used for the simulation.\nref_system: The reference system for the interpolation.\nsol: The solution state from which the properties are interpolated.\n\nKeywords\n\nsmoothing_length=ref_system.smoothing_length: The smoothing length used in the interpolation.\ncut_off_bnd=true: Boolean to indicate if quantities should be set to NaN when the point is \"closer\" to the boundary than to the fluid in a kernel-weighted sense. Or, in more detail, when the boundary has more influence than the fluid on the density summation in this point, i.e., when the boundary particles add more kernel-weighted mass than the fluid particles.\nclip_negative_pressure=false: One common approach in SPH models is to clip negative pressure values, but this is unphysical. Instead we clip here during interpolation thus only impacting the local interpolated value.\n\nReturns\n\nA NamedTuple of arrays containing interpolated properties at each point within the plane.\n\nnote: Note\nThe interpolation accuracy is subject to the density of particles and the chosen smoothing length.\nWith cut_off_bnd, a density-based estimation of the surface is used, which is not as accurate as a real surface reconstruction.\n\nExamples\n\n# Interpolating across a plane from [0.0, 0.0] to [1.0, 1.0] with a resolution of 0.2\nresults = interpolate_plane_2d([0.0, 0.0], [1.0, 1.0], 0.2, semi, ref_system, sol)\n\n\n\n\n\n","category":"method"},{"location":"general/interpolation/#TrixiParticles.interpolate_plane_2d_vtk-Tuple{Any, Any, Any, Any, Any, ODESolution}","page":"Interpolation","title":"TrixiParticles.interpolate_plane_2d_vtk","text":"interpolate_plane_2d_vtk(min_corner, max_corner, resolution, semi, ref_system, sol;\n smoothing_length=ref_system.smoothing_length, cut_off_bnd=true,\n clip_negative_pressure=false, output_directory=\"out\", filename=\"plane\")\n\nInterpolates properties along a plane in a TrixiParticles simulation and exports the result as a VTI file. The region for interpolation is defined by its lower left and top right corners, with a specified resolution determining the density of the interpolation points.\n\nThe function generates a grid of points within the defined region, spaced uniformly according to the given resolution.\n\nSee also: interpolate_plane_2d, interpolate_plane_3d, interpolate_line, interpolate_point.\n\nArguments\n\nmin_corner: The lower left corner of the interpolation region.\nmax_corner: The top right corner of the interpolation region.\nresolution: The distance between adjacent interpolation points in the grid.\nsemi: The semidiscretization used for the simulation.\nref_system: The reference system for the interpolation.\nsol: The solution state from which the properties are interpolated.\n\nKeywords\n\nsmoothing_length=ref_system.smoothing_length: The smoothing length used in the interpolation.\noutput_directory=\"out\": Directory to save the VTI file.\nfilename=\"plane\": Name of the VTI file.\ncut_off_bnd=true: Boolean to indicate if quantities should be set to NaN when the point is \"closer\" to the boundary than to the fluid in a kernel-weighted sense. Or, in more detail, when the boundary has more influence than the fluid on the density summation in this point, i.e., when the boundary particles add more kernel-weighted mass than the fluid particles.\nclip_negative_pressure=false: One common approach in SPH models is to clip negative pressure values, but this is unphysical. Instead we clip here during interpolation thus only impacting the local interpolated value.\n\nnote: Note\nThe interpolation accuracy is subject to the density of particles and the chosen smoothing length.\nWith cut_off_bnd, a density-based estimation of the surface is used, which is not as accurate as a real surface reconstruction.\n\nExamples\n\n# Interpolating across a plane from [0.0, 0.0] to [1.0, 1.0] with a resolution of 0.2\nresults = interpolate_plane_2d([0.0, 0.0], [1.0, 1.0], 0.2, semi, ref_system, sol)\n\n\n\n\n\n","category":"method"},{"location":"general/interpolation/#TrixiParticles.interpolate_plane_3d-Tuple{Any, Any, Any, Any, Any, Any, ODESolution}","page":"Interpolation","title":"TrixiParticles.interpolate_plane_3d","text":"interpolate_plane_3d(point1, point2, point3, resolution, semi, ref_system, sol;\n smoothing_length=ref_system.smoothing_length, cut_off_bnd=true,\n clip_negative_pressure=false)\n\nInterpolates properties along a plane in a 3D space in a TrixiParticles simulation. The plane for interpolation is defined by three points in 3D space, with a specified resolution determining the density of the interpolation points.\n\nThe function generates a grid of points on a parallelogram within the plane defined by the three points, spaced uniformly according to the given resolution.\n\nSee also: interpolate_plane_2d, interpolate_plane_2d_vtk, interpolate_line, interpolate_point.\n\nArguments\n\npoint1: The first point defining the plane.\npoint2: The second point defining the plane.\npoint3: The third point defining the plane. The points must not be collinear.\nresolution: The distance between adjacent interpolation points in the grid.\nsemi: The semidiscretization used for the simulation.\nref_system: The reference system for the interpolation.\nsol: The solution state from which the properties are interpolated.\n\nKeywords\n\nsmoothing_length=ref_system.smoothing_length: The smoothing length used in the interpolation.\ncut_off_bnd=true: Boolean to indicate if quantities should be set to NaN when the point is \"closer\" to the boundary than to the fluid in a kernel-weighted sense. Or, in more detail, when the boundary has more influence than the fluid on the density summation in this point, i.e., when the boundary particles add more kernel-weighted mass than the fluid particles.\nclip_negative_pressure=false: One common approach in SPH models is to clip negative pressure values, but this is unphysical. Instead we clip here during interpolation thus only impacting the local interpolated value.\n\nReturns\n\nA NamedTuple of arrays containing interpolated properties at each point within the plane.\n\nnote: Note\nThe interpolation accuracy is subject to the density of particles and the chosen smoothing length.\nWith cut_off_bnd, a density-based estimation of the surface is used which is not as accurate as a real surface reconstruction.\n\nExamples\n\n# Interpolating across a plane defined by points [0.0, 0.0, 0.0], [1.0, 0.0, 0.0], and [0.0, 1.0, 0.0]\n# with a resolution of 0.1\nresults = interpolate_plane_3d([0.0, 0.0, 0.0], [1.0, 0.0, 0.0], [0.0, 1.0, 0.0], 0.1, semi, ref_system, sol)\n\n\n\n\n\n","category":"method"},{"location":"general/interpolation/#TrixiParticles.interpolate_point-Tuple{Any, Any, Any, ODESolution}","page":"Interpolation","title":"TrixiParticles.interpolate_point","text":"interpolate_point(points_coords::Array{Array{Float64,1},1}, semi, ref_system, sol;\n smoothing_length=ref_system.smoothing_length, cut_off_bnd=true,\n clip_negative_pressure=false)\n\ninterpolate_point(point_coords, semi, ref_system, sol;\n smoothing_length=ref_system.smoothing_length, cut_off_bnd=true,\n clip_negative_pressure=false)\n\nPerforms interpolation of properties at specified points or an array of points in a TrixiParticles simulation.\n\nWhen given an array of points (points_coords), it iterates over each point and applies interpolation individually. For a single point (point_coords), it performs the interpolation at that specific location. The interpolation utilizes the same kernel function of the SPH simulation to weigh contributions from nearby particles.\n\nSee also: interpolate_line, interpolate_plane_2d, interpolate_plane_2d_vtk, interpolate_plane_3d, .\n\nArguments\n\npoints_coords: An array of point coordinates, for which to interpolate properties.\npoint_coords: The coordinates of a single point for interpolation.\nsemi: The semidiscretization used in the SPH simulation.\nref_system: The reference system defining the properties of the SPH particles.\nsol: The current solution state from which properties are interpolated.\n\nKeywords\n\nsmoothing_length=ref_system.smoothing_length: The smoothing length used in the interpolation.\ncut_off_bnd=true: Boolean to indicate if quantities should be set to NaN when the point is \"closer\" to the boundary than to the fluid in a kernel-weighted sense. Or, in more detail, when the boundary has more influence than the fluid on the density summation in this point, i.e., when the boundary particles add more kernel-weighted mass than the fluid particles.\nclip_negative_pressure=false: One common approach in SPH models is to clip negative pressure values, but this is unphysical. Instead we clip here during interpolation thus only impacting the local interpolated value.\n\nReturns\n\nFor multiple points: A NamedTuple of arrays containing interpolated properties at each point.\nFor a single point: A NamedTuple of interpolated properties at the point.\n\nExamples\n\n# For a single point\nresult = interpolate_point([1.0, 0.5], semi, ref_system, sol)\n\n# For multiple points\npoints = [[1.0, 0.5], [1.0, 0.6], [1.0, 0.7]]\nresults = interpolate_point(points, semi, ref_system, sol)\n\nnote: Note\nThis function is particularly useful for analyzing gradients or creating visualizations along a specified line in the SPH simulation domain.\nThe interpolation accuracy is subject to the density of particles and the chosen smoothing length.\nWith cut_off_bnd, a density-based estimation of the surface is used which is not asaccurate as a real surface reconstruction.\n\n\n\n\n\n","category":"method"},{"location":"#TrixiParticles.jl","page":"Home","title":"TrixiParticles.jl","text":"","category":"section"},{"location":"","page":"Home","title":"Home","text":"TrixiParticles.jl is a high-performance particle simulation framework designed to overcome challenges of particle-based numerical methods in multiphysics applications. Existing frameworks often lack user-friendliness, involve complex configuration, and are not easily extensible for development of new methods. In the future we also want to provide seamless scalability from CPU to Exascale-level computing with GPU support. TrixiParticles.jl addresses these limitations with an intuitive interface, straightforward configuration, and an extensible design, facilitating efficient simulation setup and execution.","category":"page"},{"location":"","page":"Home","title":"Home","text":"TrixiParticles.jl focuses on the following use cases:","category":"page"},{"location":"","page":"Home","title":"Home","text":"Development of new particle-based methods and models. By providing an extensible architecture to incorporate additional particle methods easily and not focusing on a single model or numerical method.\nAccurate, reliable and efficient physics-based modelling of complex multiphysics problems by providing a flexible configuration system, tools, high performance and a wide range of validation and test cases.\nEasy setup of accessible simulations for educational purposes, including student projects, coursework, and thesis work through extensive documentation, community engagement and readable configuration files.","category":"page"},{"location":"","page":"Home","title":"Home","text":"Its features include:","category":"page"},{"location":"#Features","page":"Home","title":"Features","text":"","category":"section"},{"location":"","page":"Home","title":"Home","text":"Incompressible Navier-Stokes\nMethods: Weakly Compressible Smoothed Particle Hydrodynamics (WCSPH), Entropically Damped Artificial Compressibility (EDAC)\nModels: Surface Tension\nSolid-body mechanics\nMethods: Total Lagrangian SPH (TLSPH)\nFluid-Structure Interaction\nOutput formats:\nVTK","category":"page"},{"location":"#Examples","page":"Home","title":"Examples","text":"","category":"section"},{"location":"","page":"Home","title":"Home","text":"\n \n \n \n \n \n \n \n \n
\n
2D Dam Break
\n
\n
Moving Wall
\n
\n
Oscillating Beam
\n
\n
Dam Break with Elastic Plate
\n
","category":"page"},{"location":"#Quickstart","page":"Home","title":"Quickstart","text":"","category":"section"},{"location":"","page":"Home","title":"Home","text":"Installation\nGetting started","category":"page"},{"location":"","page":"Home","title":"Home","text":"If you have any questions concerning TrixiParticles.jl you can join our community on Slack or open an issue with your question.","category":"page"},{"location":"#Start-with-development","page":"Home","title":"Start with development","text":"","category":"section"},{"location":"","page":"Home","title":"Home","text":"To get started with development have a look at these pages:","category":"page"},{"location":"","page":"Home","title":"Home","text":"Installation\nDevelopment\nContributing","category":"page"},{"location":"tutorial/#Tutorial","page":"Tutorial","title":"Tutorial","text":"","category":"section"},{"location":"tutorial/#Fluid","page":"Tutorial","title":"Fluid","text":"","category":"section"},{"location":"tutorial/","page":"Tutorial","title":"Tutorial","text":"Setting up your simulation from scratch\nSetting up a dam break simulation","category":"page"},{"location":"tutorial/#Mechanics","page":"Tutorial","title":"Mechanics","text":"","category":"section"},{"location":"tutorial/","page":"Tutorial","title":"Tutorial","text":"Deforming a beam","category":"page"},{"location":"tutorial/#Fluid-Structure-Interaction","page":"Tutorial","title":"Fluid-Structure Interaction","text":"","category":"section"},{"location":"tutorial/","page":"Tutorial","title":"Tutorial","text":"Setting up a falling structure","category":"page"},{"location":"reference-pointneighbors/#pointneighbors","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.jl API","text":"","category":"section"},{"location":"reference-pointneighbors/","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.jl API Reference","text":"CurrentModule = PointNeighbors","category":"page"},{"location":"reference-pointneighbors/","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.jl API Reference","text":"Modules = [PointNeighbors]","category":"page"},{"location":"reference-pointneighbors/#PointNeighbors.DictionaryCellList","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.DictionaryCellList","text":"DictionaryCellList{NDIMS}()\n\nA simple cell list implementation where a cell index (i, j) or (i, j, k) is mapped to a Vector{Int} by a Dict. By using a dictionary, which only stores non-empty cells, the domain is potentially infinite.\n\nThis implementation is very simple, but it neither uses an optimized hash function for integer tuples, nor does it use a contiguous memory layout. Consequently, this cell list is not GPU-compatible.\n\nArguments\n\nNDIMS: Number of dimensions.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.FullGridCellList","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.FullGridCellList","text":"FullGridCellList(; min_corner, max_corner, search_radius = 0.0,\n periodicity = false, backend = DynamicVectorOfVectors{Int32},\n max_points_per_cell = 100)\n\nA simple cell list implementation where each (empty or non-empty) cell of a rectangular (axis-aligned) domain is assigned a list of points. This cell list only works when all points are inside the specified domain at all times.\n\nOnly set min_corner and max_corner and use the default values for the other arguments to create an empty \"template\" cell list that can be used to create an empty \"template\" neighborhood search. See copy_neighborhood_search for more details.\n\nKeywords\n\nmin_corner: Coordinates of the domain corner in negative coordinate directions.\nmax_corner: Coordinates of the domain corner in positive coordinate directions.\nsearch_radius = 0.0: Search radius of the neighborhood search, which will determine the cell size. Use the default of 0.0 to create a template (see above).\nperiodicity = false: Set to true when using a PeriodicBox with the neighborhood search. When using copy_neighborhood_search, this option can be ignored an will be set automatically depending on the periodicity of the neighborhood search.\nbackend = DynamicVectorOfVectors{Int32}: Type of the data structure to store the actual cell lists. Can be\nVector{Vector{Int32}}: Scattered memory, but very memory-efficient.\nDynamicVectorOfVectors{Int32}: Contiguous memory, optimizing cache-hits.\nmax_points_per_cell = 100: Maximum number of points per cell. This will be used to allocate the DynamicVectorOfVectors. It is not used with the Vector{Vector{Int32}} backend.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.GridNeighborhoodSearch","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.GridNeighborhoodSearch","text":"GridNeighborhoodSearch{NDIMS}(; search_radius = 0.0, n_points = 0,\n periodic_box = nothing,\n cell_list = DictionaryCellList{NDIMS}(),\n update_strategy = nothing)\n\nSimple grid-based neighborhood search with uniform search radius. The domain is divided into a regular grid. For each (non-empty) grid cell, a list of points in this cell is stored. Instead of representing a finite domain by an array of cells, a potentially infinite domain is represented by storing cell lists in a hash table (using Julia's Dict data structure), indexed by the cell index tuple\n\nleft( leftlfloor fracxd rightrfloor leftlfloor fracyd rightrfloor right) quad textor quad\nleft( leftlfloor fracxd rightrfloor leftlfloor fracyd rightrfloor leftlfloor fraczd rightrfloor right)\n\nwhere x y z are the space coordinates and d is the search radius.\n\nTo find points within the search radius around a position, only points in the neighboring cells are considered.\n\nSee also (Chalela et al., 2021), (Ihmsen et al. 2011, Section 4.4).\n\nAs opposed to (Ihmsen et al. 2011), we do not sort the points in any way, since not sorting makes our implementation a lot faster (although less parallelizable).\n\nArguments\n\nNDIMS: Number of dimensions.\n\nKeywords\n\nsearch_radius = 0.0: The fixed search radius. The default of 0.0 is useful together with copy_neighborhood_search.\nn_points = 0: Total number of points. The default of 0 is useful together with copy_neighborhood_search.\nperiodic_box = nothing: In order to use a (rectangular) periodic domain, pass a PeriodicBox.\ncell_list: The cell list that maps a cell index to a list of points inside the cell. By default, a DictionaryCellList is used.\nupdate_strategy = nothing: Strategy to parallelize update!. Available options are:\nnothing: Automatically choose the best available option.\nParallelUpdate(): This is not available for all cell list implementations, but is the default when available.\nSemiParallelUpdate(): This is available for all cell list implementations and is the default when ParallelUpdate is not available.\nSerialUpdate()\n\nReferences\n\nM. Chalela, E. Sillero, L. Pereyra, M.A. Garcia, J.B. Cabral, M. Lares, M. Merchán. \"GriSPy: A Python package for fixed-radius nearest neighbors search\". In: Astronomy and Computing 34 (2021). doi: 10.1016/j.ascom.2020.100443\nMarkus Ihmsen, Nadir Akinci, Markus Becker, Matthias Teschner. \"A Parallel SPH Implementation on Multi-Core CPUs\". In: Computer Graphics Forum 30.1 (2011), pages 99–112. doi: 10.1111/J.1467-8659.2010.01832.X\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.ParallelUpdate","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.ParallelUpdate","text":"ParallelUpdate()\n\nFully parallel update by using atomic operations to avoid race conditions when adding points into the same cell. This is not available for all cell list implementations, but is the default when available.\n\nSee GridNeighborhoodSearch for usage information.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.PeriodicBox","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.PeriodicBox","text":"PeriodicBox(; min_corner, max_corner)\n\nDefine a rectangular (axis-aligned) periodic domain.\n\nKeywords\n\nmin_corner: Coordinates of the domain corner in negative coordinate directions.\nmax_corner: Coordinates of the domain corner in positive coordinate directions.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.PolyesterBackend","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.PolyesterBackend","text":"PolyesterBackend()\n\nPass as first argument to the @threaded macro to make the loop multithreaded with Polyester.@batch.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.PrecomputedNeighborhoodSearch","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.PrecomputedNeighborhoodSearch","text":"PrecomputedNeighborhoodSearch{NDIMS}(; search_radius = 0.0, n_points = 0,\n periodic_box = nothing, update_strategy = nothing)\n\nNeighborhood search with precomputed neighbor lists. A list of all neighbors is computed for each point during initialization and update. This neighborhood search maximizes the performance of neighbor loops at the cost of a much slower update!.\n\nA GridNeighborhoodSearch is used internally to compute the neighbor lists during initialization and update.\n\nArguments\n\nNDIMS: Number of dimensions.\n\nKeywords\n\nsearch_radius = 0.0: The fixed search radius. The default of 0.0 is useful together with copy_neighborhood_search.\nn_points = 0: Total number of points. The default of 0 is useful together with copy_neighborhood_search.\nperiodic_box = nothing: In order to use a (rectangular) periodic domain, pass a PeriodicBox.\nupdate_strategy: Strategy to parallelize update! of the internally used GridNeighborhoodSearch. See GridNeighborhoodSearch for available options.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.SemiParallelUpdate","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.SemiParallelUpdate","text":"SemiParallelUpdate()\n\nLoop over all cells in parallel to mark cells with points that now belong to a different cell. Then, move points of affected cells serially to avoid race conditions. This is available for all cell list implementations and is the default when ParallelUpdate is not available.\n\nSee GridNeighborhoodSearch for usage information.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.SerialUpdate","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.SerialUpdate","text":"SerialUpdate()\n\nDeactivate parallelization in the neighborhood search update. Parallel neighborhood search update can be one of the largest sources of error variations between simulations with different thread numbers due to neighbor ordering changes.\n\nSee GridNeighborhoodSearch for usage information.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.ThreadsDynamicBackend","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.ThreadsDynamicBackend","text":"ThreadsDynamicBackend()\n\nPass as first argument to the @threaded macro to make the loop multithreaded with Threads.@threads :dynamic.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.ThreadsStaticBackend","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.ThreadsStaticBackend","text":"ThreadsStaticBackend()\n\nPass as first argument to the @threaded macro to make the loop multithreaded with Threads.@threads :static.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.TrivialNeighborhoodSearch","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.TrivialNeighborhoodSearch","text":"TrivialNeighborhoodSearch{NDIMS}(; search_radius = 0.0, eachpoint = 1:0,\n periodic_box = nothing)\n\nTrivial neighborhood search that simply loops over all points.\n\nArguments\n\nNDIMS: Number of dimensions.\n\nKeywords\n\nsearch_radius = 0.0: The fixed search radius. The default of 0.0 is useful together with copy_neighborhood_search.\neachpoint = 1:0: Iterator for all point indices. Usually just 1:n_points. The default of 1:0 is useful together with copy_neighborhood_search.\nperiodic_box = nothing: In order to use a (rectangular) periodic domain, pass a PeriodicBox.\n\n\n\n\n\n","category":"type"},{"location":"reference-pointneighbors/#PointNeighbors.copy_neighborhood_search-Tuple{PointNeighbors.AbstractNeighborhoodSearch, Any, Any}","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.copy_neighborhood_search","text":"copy_neighborhood_search(search::AbstractNeighborhoodSearch, search_radius, n_points;\n eachpoint = 1:n_points)\n\nCreate a new uninitialized neighborhood search of the same type and with the same configuration options as search, but with a different search radius and number of points.\n\nThe TrivialNeighborhoodSearch also requires an iterator eachpoint, which most of the time will be 1:n_points. If the TrivialNeighborhoodSearch is never going to be used, the keyword argument eachpoint can be ignored.\n\nThis is useful when a simulation code requires multiple neighborhood searches of the same kind. One can then just pass an empty neighborhood search as a template and use this function inside the simulation code to generate similar neighborhood searches with different search radii and different numbers of points.\n\n# Template\nnhs = GridNeighborhoodSearch{2}()\n\n# Inside the simulation code, generate similar neighborhood searches\nnhs1 = copy_neighborhood_search(nhs, 1.0, 100)\n\n# output\nGridNeighborhoodSearch{2, Float64, ...}(...)\n\n\n\n\n\n","category":"method"},{"location":"reference-pointneighbors/#PointNeighbors.foreach_point_neighbor-Union{Tuple{T}, Tuple{T, Any, Any, Any}} where T","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.foreach_point_neighbor","text":"foreach_point_neighbor(f, system_coords, neighbor_coords, neighborhood_search;\n points = axes(system_coords, 2), parallel = true)\n\nLoop for each point in system_coords over all points in neighbor_coords whose distances to that point are smaller than the search radius and execute the function f(i, j, x, y, d), where\n\ni is the column index of the point in system_coords,\nj the column index of the neighbor in neighbor_coords,\nx an SVector of the coordinates of the point (system_coords[:, i]),\ny an SVector of the coordinates of the neighbor (neighbor_coords[:, j]),\nd the distance between x and y.\n\nThe neighborhood_search must have been initialized or updated with system_coords as first coordinate array and neighbor_coords as second coordinate array.\n\nNote that system_coords and neighbor_coords can be identical.\n\nArguments\n\nf: The function explained above.\nsystem_coords: A matrix where the i-th column contains the coordinates of point i.\nneighbor_coords: A matrix where the j-th column contains the coordinates of point j.\nneighborhood_search: A neighborhood search initialized or updated with system_coords as first coordinate array and neighbor_coords as second coordinate array.\n\nKeywords\n\npoints: Loop over these point indices. By default all columns of system_coords.\nparallel=true: Run the outer loop over points thread-parallel.\n\nSee also initialize!, update!.\n\n\n\n\n\n","category":"method"},{"location":"reference-pointneighbors/#PointNeighbors.initialize!-Tuple{PointNeighbors.AbstractNeighborhoodSearch, Any, Any}","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.initialize!","text":"initialize!(search::AbstractNeighborhoodSearch, x, y)\n\nInitialize a neighborhood search with the two coordinate arrays x and y.\n\nIn general, the purpose of a neighborhood search is to find for one point in x all points in y whose distances to that point are smaller than the search radius. x and y are expected to be matrices, where the i-th column contains the coordinates of point i. Note that x and y can be identical.\n\nSee also update!.\n\n\n\n\n\n","category":"method"},{"location":"reference-pointneighbors/#PointNeighbors.update!-Tuple{PointNeighbors.AbstractNeighborhoodSearch, Any, Any}","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.update!","text":"update!(search::AbstractNeighborhoodSearch, x, y; points_moving = (true, true))\n\nUpdate an already initialized neighborhood search with the two coordinate arrays x and y.\n\nLike initialize!, but reusing the existing data structures of the already initialized neighborhood search. When the points only moved a small distance since the last update! or initialize!, this is significantly faster than initialize!.\n\nNot all implementations support incremental updates. If incremental updates are not possible for an implementation, update! will fall back to a regular initialize!.\n\nSome neighborhood searches might not need to update when only x changed since the last update! or initialize! and y did not change. Pass points_moving = (true, false) in this case to avoid unnecessary updates. The first flag in points_moving indicates if points in x are moving. The second flag indicates if points in y are moving.\n\nwarning: Experimental Feature: Backend Specification\nThe keyword argument parallelization_backend allows users to specify the multithreading backend. This feature is currently considered experimental!Possible parallelization backends are:ThreadsDynamicBackend to use Threads.@threads :dynamic\nThreadsStaticBackend to use Threads.@threads :static\nPolyesterBackend to use Polyester.@batch\nKernelAbstractions.Backend to launch a GPU kernel\n\nSee also initialize!.\n\n\n\n\n\n","category":"method"},{"location":"reference-pointneighbors/#PointNeighbors.@threaded-Tuple{Any, Any}","page":"PointNeighbors.jl API Reference","title":"PointNeighbors.@threaded","text":"@threaded x for ... end\n\nRun either a threaded CPU loop or launch a kernel on the GPU, depending on the type of x. Semantically the same as Threads.@threads when iterating over a AbstractUnitRange but without guarantee that the underlying implementation uses Threads.@threads or works for more general for loops.\n\nThe first argument must either be a parallelization backend (see below) or an array from which the backend can be derived to determine if the loop must be run threaded on the CPU or launched as a kernel on the GPU. Passing KernelAbstractions.CPU() will run the GPU kernel on the CPU.\n\nPossible parallelization backends are:\n\nPolyesterBackend to use Polyester.@batch\nThreadsDynamicBackend to use Threads.@threads :dynamic\nThreadsStaticBackend to use Threads.@threads :static\nKernelAbstractions.Backend to execute the loop as a GPU kernel\n\nIn particular, the underlying threading capabilities might be provided by other packages such as Polyester.jl.\n\nwarning: Warning\nThis macro does not necessarily work for general for loops. For example, it does not necessarily support general iterables such as eachline(filename).\n\n\n\n\n\n","category":"macro"}] } diff --git a/previews/PR514/systems/boundary/index.html b/previews/PR514/systems/boundary/index.html index 1beb48c13..3773d9186 100644 --- a/previews/PR514/systems/boundary/index.html +++ b/previews/PR514/systems/boundary/index.html @@ -1,8 +1,8 @@ -Boundary · TrixiParticles.jl

Boundary System

TrixiParticles.BoundarySPHSystemType
BoundarySPHSystem(initial_condition, boundary_model; movement=nothing, adhesion_coefficient=0.0)

System for boundaries modeled by boundary particles. The interaction between fluid and boundary particles is specified by the boundary model.

Arguments

Keyword Arguments

  • movement: For moving boundaries, a BoundaryMovement can be passed.
  • adhesion_coefficient: Coefficient specifying the adhesion of a fluid to the surface. Note: currently it is assumed that all fluids have the same adhesion coefficient.
source
TrixiParticles.BoundaryDEMSystemType
BoundaryDEMSystem(initial_condition, normal_stiffness)

System for boundaries modeled by boundary particles. The interaction between fluid and boundary particles is specified by the boundary model.

Experimental Implementation

This is an experimental feature and may change in a future releases.

source
TrixiParticles.BoundaryMovementType
BoundaryMovement(movement_function, is_moving; moving_particles=nothing)

Arguments

  • movement_function: Time-dependent function returning an SVector of $d$ dimensions for a $d$-dimensional problem.
  • is_moving: Function to determine in each timestep if the particles are moving or not. Its boolean return value is mandatory to determine if the neighborhood search will be updated.

Keyword Arguments

  • moving_particles: Indices of moving particles. Default is each particle in BoundarySPHSystem.

In the example below, movement describes particles moving in a circle as long as the time is lower than 1.5.

Examples

movement_function(t) = SVector(cos(2pi*t), sin(2pi*t))
+Boundary · TrixiParticles.jl

Boundary System

TrixiParticles.BoundarySPHSystemType
BoundarySPHSystem(initial_condition, boundary_model; movement=nothing, adhesion_coefficient=0.0)

System for boundaries modeled by boundary particles. The interaction between fluid and boundary particles is specified by the boundary model.

Arguments

Keyword Arguments

  • movement: For moving boundaries, a BoundaryMovement can be passed.
  • adhesion_coefficient: Coefficient specifying the adhesion of a fluid to the surface. Note: currently it is assumed that all fluids have the same adhesion coefficient.
source
TrixiParticles.BoundaryDEMSystemType
BoundaryDEMSystem(initial_condition, normal_stiffness)

System for boundaries modeled by boundary particles. The interaction between fluid and boundary particles is specified by the boundary model.

Experimental Implementation

This is an experimental feature and may change in a future releases.

source
TrixiParticles.BoundaryMovementType
BoundaryMovement(movement_function, is_moving; moving_particles=nothing)

Arguments

  • movement_function: Time-dependent function returning an SVector of $d$ dimensions for a $d$-dimensional problem.
  • is_moving: Function to determine in each timestep if the particles are moving or not. Its boolean return value is mandatory to determine if the neighborhood search will be updated.

Keyword Arguments

  • moving_particles: Indices of moving particles. Default is each particle in BoundarySPHSystem.

In the example below, movement describes particles moving in a circle as long as the time is lower than 1.5.

Examples

movement_function(t) = SVector(cos(2pi*t), sin(2pi*t))
 is_moving(t) = t < 1.5
 
-movement = BoundaryMovement(movement_function, is_moving)
source

Boundary Models

Dummy Particles

Boundaries modeled as dummy particles, which are treated like fluid particles, but their positions and velocities are not evolved in time. Since the force towards the fluid should not change with the material density when used with a TotalLagrangianSPHSystem, the dummy particles need to have a mass corresponding to the fluid's rest density, which we call "hydrodynamic mass", as opposed to mass corresponding to the material density of a TotalLagrangianSPHSystem.

Here, initial_density and hydrodynamic_mass are vectors that contains the initial density and the hydrodynamic mass respectively for each boundary particle. Note that when used with SummationDensity (see below), this is only used to determine the element type and the number of boundary particles.

To establish a relationship between density and pressure, a state_equation has to be passed, which should be the same as for the adjacent fluid systems. To sum over neighboring particles, a smoothing_kernel and smoothing_length needs to be passed. This should be the same as for the adjacent fluid system with the largest smoothing length.

In the literature, this kind of boundary particles is referred to as "dummy particles" (Adami et al., 2012 and Valizadeh & Monaghan, 2015), "frozen fluid particles" (Akinci et al., 2012) or "dynamic boundaries Crespo et al., 2007. The key detail of this boundary condition and the only difference between the boundary models in these references is the way the density and pressure of boundary particles is computed.

Since boundary particles are treated like fluid particles, the force on fluid particle $a$ due to boundary particle $b$ is given by

\[f_{ab} = m_a m_b \left( \frac{p_a}{\rho_a^2} + \frac{p_b}{\rho_b^2} \right) \nabla_{r_a} W(\Vert r_a - r_b \Vert, h).\]

The quantities to be defined here are the density $\rho_b$ and pressure $p_b$ of the boundary particle $b$.

Boundary Models

Dummy Particles

Boundaries modeled as dummy particles, which are treated like fluid particles, but their positions and velocities are not evolved in time. Since the force towards the fluid should not change with the material density when used with a TotalLagrangianSPHSystem, the dummy particles need to have a mass corresponding to the fluid's rest density, which we call "hydrodynamic mass", as opposed to mass corresponding to the material density of a TotalLagrangianSPHSystem.

Here, initial_density and hydrodynamic_mass are vectors that contains the initial density and the hydrodynamic mass respectively for each boundary particle. Note that when used with SummationDensity (see below), this is only used to determine the element type and the number of boundary particles.

To establish a relationship between density and pressure, a state_equation has to be passed, which should be the same as for the adjacent fluid systems. To sum over neighboring particles, a smoothing_kernel and smoothing_length needs to be passed. This should be the same as for the adjacent fluid system with the largest smoothing length.

In the literature, this kind of boundary particles is referred to as "dummy particles" (Adami et al., 2012 and Valizadeh & Monaghan, 2015), "frozen fluid particles" (Akinci et al., 2012) or "dynamic boundaries Crespo et al., 2007. The key detail of this boundary condition and the only difference between the boundary models in these references is the way the density and pressure of boundary particles is computed.

Since boundary particles are treated like fluid particles, the force on fluid particle $a$ due to boundary particle $b$ is given by

\[f_{ab} = m_a m_b \left( \frac{p_a}{\rho_a^2} + \frac{p_b}{\rho_b^2} \right) \nabla_{r_a} W(\Vert r_a - r_b \Vert, h).\]

The quantities to be defined here are the density $\rho_b$ and pressure $p_b$ of the boundary particle $b$.

TrixiParticles.BoundaryModelDummyParticlesType
BoundaryModelDummyParticles(initial_density, hydrodynamic_mass,
                             density_calculator, smoothing_kernel,
                             smoothing_length; viscosity=nothing,
                             state_equation=nothing, correction=nothing)

Boundary model for BoundarySPHSystem.

Arguments

  • initial_density: Vector holding the initial density of each boundary particle.
  • hydrodynamic_mass: Vector holding the "hydrodynamic mass" of each boundary particle. See description above for more information.
  • density_calculator: Strategy to compute the hydrodynamic density of the boundary particles. See description below for more information.
  • smoothing_kernel: Smoothing kernel should be the same as for the adjacent fluid system.
  • smoothing_length: Smoothing length should be the same as for the adjacent fluid system.

Keywords

  • state_equation: This should be the same as for the adjacent fluid system (see e.g. StateEquationCole).
  • correction: Correction method of the adjacent fluid system (see Corrections).
  • viscosity: Slip (default) or no-slip condition. See description below for further information.

Examples

# Free-slip condition
@@ -12,19 +12,19 @@
 # No-slip condition
 boundary_model = BoundaryModelDummyParticles(densities, masses, AdamiPressureExtrapolation(),
                                              smoothing_kernel, smoothing_length,
-                                             viscosity=ViscosityAdami(nu=1e-6))
source

Hydrodynamic density of dummy particles

We provide six options to compute the boundary density and pressure, determined by the density_calculator:

  1. (Recommended) With AdamiPressureExtrapolation, the pressure is extrapolated from the pressure of the fluid according to Adami et al., 2012, and the density is obtained by applying the inverse of the state equation. This option usually yields the best results of the options listed here.
  2. (Only relevant for FSI) With BernoulliPressureExtrapolation, the pressure is extrapolated from the pressure similar to the AdamiPressureExtrapolation, but a relative velocity-dependent pressure part is calculated between moving solids and fluids, which increases the boundary pressure in areas prone to penetrations.
  3. With SummationDensity, the density is calculated by summation over the neighboring particles, and the pressure is computed from the density with the state equation.
  4. With ContinuityDensity, the density is integrated from the continuity equation, and the pressure is computed from the density with the state equation. Note that this causes a gap between fluid and boundary where the boundary is initialized without any contact to the fluid. This is due to overestimation of the boundary density as soon as the fluid comes in contact with boundary particles that initially did not have contact to the fluid. Therefore, in dam break simulations, there is a visible "step", even though the boundary is supposed to be flat. See also dual.sphysics.org/faq/#Q_13.
  5. With PressureZeroing, the density is set to the reference density and the pressure is computed from the density with the state equation. This option is not recommended. The other options yield significantly better results.
  6. With PressureMirroring, the density is set to the reference density. The pressure is not used. Instead, the fluid pressure is mirrored as boundary pressure in the momentum equation. This option is not recommended due to stability issues. See PressureMirroring for more details.

1. AdamiPressureExtrapolation

The pressure of the boundary particles is obtained by extrapolating the pressure of the fluid according to Adami et al., 2012. The pressure of a boundary particle $b$ is given by

\[p_b = \frac{\sum_f (p_f + \rho_f (\bm{g} - \bm{a}_b) \cdot \bm{r}_{bf}) W(\Vert r_{bf} \Vert, h)}{\sum_f W(\Vert r_{bf} \Vert, h)},\]

where the sum is over all fluid particles, $\rho_f$ and $p_f$ denote the density and pressure of fluid particle $f$, respectively, $r_{bf} = r_b - r_f$ denotes the difference of the coordinates of particles $b$ and $f$, $\bm{g}$ denotes the gravitational acceleration acting on the fluid, and $\bm{a}_b$ denotes the acceleration of the boundary particle $b$.

TrixiParticles.AdamiPressureExtrapolationType
AdamiPressureExtrapolation(; pressure_offset=0.0)

density_calculator for BoundaryModelDummyParticles.

Keywords

  • pressure_offset=0.0: Sometimes it is necessary to artificially increase the boundary pressure to prevent penetration, which is possible by increasing this value.
source

2. BernoulliPressureExtrapolation

Identical to the pressure $p_b$ calculated via AdamiPressureExtrapolation, but it adds the dynamic pressure component of the Bernoulli equation:

\[p_b = \frac{\sum_f (p_f + \frac{1}{2} \, \rho_{\text{neighbor}} \left( \frac{ (\mathbf{v}_f - \mathbf{v}_{\text{body}}) \cdot (\mathbf{x}_f - \mathbf{x}_{\text{neighbor}}) }{ \left\| \mathbf{x}_f - \mathbf{x}_{\text{neighbor}} \right\| } \right)^2 \times \text{factor} +\rho_f (\bm{g} - \bm{a}_b) \cdot \bm{r}_{bf}) W(\Vert r_{bf} \Vert, h)}{\sum_f W(\Vert r_{bf} \Vert, h)} \]

where $\mathbf{v}_f$ is the velocity of the fluid and $\mathbf{v}_{\text{body}}$ is the velocity of the body. This adjustment provides a higher boundary pressure for solid bodies moving with a relative velocity to the fluid to prevent penetration. This modification is original and not derived from any literature source.

TrixiParticles.BernoulliPressureExtrapolationType
BernoulliPressureExtrapolation(; pressure_offset=0.0, factor=1.0)

density_calculator for BoundaryModelDummyParticles.

Keywords

  • pressure_offset=0.0: Sometimes it is necessary to artificially increase the boundary pressure to prevent penetration, which is possible by increasing this value.
  • factor=1.0 : Setting factor allows to just increase the strength of the dynamic pressure part.
source

5. PressureZeroing

This is the simplest way to implement dummy boundary particles. The density of each particle is set to the reference density and the pressure to the reference pressure (the corresponding pressure to the reference density by the state equation).

TrixiParticles.PressureZeroingType
PressureZeroing()

density_calculator for BoundaryModelDummyParticles.

Note

This boundary model produces significantly worse results than all other models and is only included for research purposes.

source

6. PressureMirroring

Instead of calculating density and pressure for each boundary particle, we modify the momentum equation,

\[\frac{\mathrm{d}v_a}{\mathrm{d}t} = -\sum_b m_b \left( \frac{p_a}{\rho_a^2} + \frac{p_b}{\rho_b^2} \right) \nabla_a W_{ab}\]

to replace the unknown density $\rho_b$ if $b$ is a boundary particle by the reference density and the unknown pressure $p_b$ if $b$ is a boundary particle by the pressure $p_a$ of the interacting fluid particle. The momentum equation therefore becomes

\[\frac{\mathrm{d}v_a}{\mathrm{d}t} = -\sum_f m_f \left( \frac{p_a}{\rho_a^2} + \frac{p_f}{\rho_f^2} \right) \nabla_a W_{af} --\sum_b m_b \left( \frac{p_a}{\rho_a^2} + \frac{p_a}{\rho_0^2} \right) \nabla_a W_{ab},\]

where the first sum is over all fluid particles and the second over all boundary particles.

This approach was first mentioned by Akinci et al. (2012) and written down in this form by Band et al. (2018).

TrixiParticles.PressureMirroringType
PressureMirroring()

density_calculator for BoundaryModelDummyParticles.

Note

This boundary model requires high viscosity for stability with WCSPH. It also produces significantly worse results than AdamiPressureExtrapolation and is not more efficient because smaller time steps are required due to more noise in the pressure. We added this model only for research purposes and for comparison with SPlisHSPlasH.

source

No-slip conditions

For the interaction of dummy particles and fluid particles, Adami et al. (2012) impose a no-slip boundary condition by assigning a wall velocity $v_w$ to the dummy particle.

The wall velocity of particle $a$ is calculated from the prescribed boundary particle velocity $v_a$ and the smoothed velocity field

\[v_w = 2 v_a - \frac{\sum_b v_b W_{ab}}{\sum_b W_{ab}},\]

where the sum is over all fluid particles.

By choosing the viscosity model ViscosityAdami for viscosity, a no-slip condition is imposed. It is recommended to choose nu in the order of either the kinematic viscosity parameter of the adjacent fluid or the equivalent from the artificial parameter alpha of the adjacent fluid ($\nu = \frac{\alpha h c }{2d + 4}$). When omitting the viscous interaction (default viscosity=nothing), a free-slip wall boundary condition is applied.

Warning

The viscosity model ArtificialViscosityMonaghan for BoundaryModelDummyParticles has not been verified yet.

Repulsive Particles

Boundaries modeled as boundary particles which exert forces on the fluid particles (Monaghan, Kajtar, 2009). The force on fluid particle $a$ due to boundary particle $b$ is given by

\[f_{ab} = m_a \left(\tilde{f}_{ab} - m_b \Pi_{ab} \nabla_{r_a} W(\Vert r_a - r_b \Vert, h)\right)\]

with

\[\tilde{f}_{ab} = \frac{K}{\beta^{n-1}} \frac{r_{ab}}{\Vert r_{ab} \Vert (\Vert r_{ab} \Vert - d)} \Phi(\Vert r_{ab} \Vert, h) + viscosity=ViscosityAdami(nu=1e-6))

source

Hydrodynamic density of dummy particles

We provide six options to compute the boundary density and pressure, determined by the density_calculator:

  1. (Recommended) With AdamiPressureExtrapolation, the pressure is extrapolated from the pressure of the fluid according to Adami et al., 2012, and the density is obtained by applying the inverse of the state equation. This option usually yields the best results of the options listed here.
  2. (Only relevant for FSI) With BernoulliPressureExtrapolation, the pressure is extrapolated from the pressure similar to the AdamiPressureExtrapolation, but a relative velocity-dependent pressure part is calculated between moving solids and fluids, which increases the boundary pressure in areas prone to penetrations.
  3. With SummationDensity, the density is calculated by summation over the neighboring particles, and the pressure is computed from the density with the state equation.
  4. With ContinuityDensity, the density is integrated from the continuity equation, and the pressure is computed from the density with the state equation. Note that this causes a gap between fluid and boundary where the boundary is initialized without any contact to the fluid. This is due to overestimation of the boundary density as soon as the fluid comes in contact with boundary particles that initially did not have contact to the fluid. Therefore, in dam break simulations, there is a visible "step", even though the boundary is supposed to be flat. See also dual.sphysics.org/faq/#Q_13.
  5. With PressureZeroing, the density is set to the reference density and the pressure is computed from the density with the state equation. This option is not recommended. The other options yield significantly better results.
  6. With PressureMirroring, the density is set to the reference density. The pressure is not used. Instead, the fluid pressure is mirrored as boundary pressure in the momentum equation. This option is not recommended due to stability issues. See PressureMirroring for more details.

1. AdamiPressureExtrapolation

The pressure of the boundary particles is obtained by extrapolating the pressure of the fluid according to Adami et al., 2012. The pressure of a boundary particle $b$ is given by

\[p_b = \frac{\sum_f (p_f + \rho_f (\bm{g} - \bm{a}_b) \cdot \bm{r}_{bf}) W(\Vert r_{bf} \Vert, h)}{\sum_f W(\Vert r_{bf} \Vert, h)},\]

where the sum is over all fluid particles, $\rho_f$ and $p_f$ denote the density and pressure of fluid particle $f$, respectively, $r_{bf} = r_b - r_f$ denotes the difference of the coordinates of particles $b$ and $f$, $\bm{g}$ denotes the gravitational acceleration acting on the fluid, and $\bm{a}_b$ denotes the acceleration of the boundary particle $b$.

TrixiParticles.AdamiPressureExtrapolationType
AdamiPressureExtrapolation(; pressure_offset=0.0)

density_calculator for BoundaryModelDummyParticles.

Keywords

  • pressure_offset=0.0: Sometimes it is necessary to artificially increase the boundary pressure to prevent penetration, which is possible by increasing this value.
source

2. BernoulliPressureExtrapolation

Identical to the pressure $p_b$ calculated via AdamiPressureExtrapolation, but it adds the dynamic pressure component of the Bernoulli equation:

\[p_b = \frac{\sum_f (p_f + \frac{1}{2} \, \rho_{\text{neighbor}} \left( \frac{ (\mathbf{v}_f - \mathbf{v}_{\text{body}}) \cdot (\mathbf{x}_f - \mathbf{x}_{\text{neighbor}}) }{ \left\| \mathbf{x}_f - \mathbf{x}_{\text{neighbor}} \right\| } \right)^2 \times \text{factor} +\rho_f (\bm{g} - \bm{a}_b) \cdot \bm{r}_{bf}) W(\Vert r_{bf} \Vert, h)}{\sum_f W(\Vert r_{bf} \Vert, h)} \]

where $\mathbf{v}_f$ is the velocity of the fluid and $\mathbf{v}_{\text{body}}$ is the velocity of the body. This adjustment provides a higher boundary pressure for solid bodies moving with a relative velocity to the fluid to prevent penetration. This modification is original and not derived from any literature source.

TrixiParticles.BernoulliPressureExtrapolationType
BernoulliPressureExtrapolation(; pressure_offset=0.0, factor=1.0)

density_calculator for BoundaryModelDummyParticles.

Keywords

  • pressure_offset=0.0: Sometimes it is necessary to artificially increase the boundary pressure to prevent penetration, which is possible by increasing this value.
  • factor=1.0 : Setting factor allows to just increase the strength of the dynamic pressure part.
source

5. PressureZeroing

This is the simplest way to implement dummy boundary particles. The density of each particle is set to the reference density and the pressure to the reference pressure (the corresponding pressure to the reference density by the state equation).

TrixiParticles.PressureZeroingType
PressureZeroing()

density_calculator for BoundaryModelDummyParticles.

Note

This boundary model produces significantly worse results than all other models and is only included for research purposes.

source

6. PressureMirroring

Instead of calculating density and pressure for each boundary particle, we modify the momentum equation,

\[\frac{\mathrm{d}v_a}{\mathrm{d}t} = -\sum_b m_b \left( \frac{p_a}{\rho_a^2} + \frac{p_b}{\rho_b^2} \right) \nabla_a W_{ab}\]

to replace the unknown density $\rho_b$ if $b$ is a boundary particle by the reference density and the unknown pressure $p_b$ if $b$ is a boundary particle by the pressure $p_a$ of the interacting fluid particle. The momentum equation therefore becomes

\[\frac{\mathrm{d}v_a}{\mathrm{d}t} = -\sum_f m_f \left( \frac{p_a}{\rho_a^2} + \frac{p_f}{\rho_f^2} \right) \nabla_a W_{af} +-\sum_b m_b \left( \frac{p_a}{\rho_a^2} + \frac{p_a}{\rho_0^2} \right) \nabla_a W_{ab},\]

where the first sum is over all fluid particles and the second over all boundary particles.

This approach was first mentioned by Akinci et al. (2012) and written down in this form by Band et al. (2018).

TrixiParticles.PressureMirroringType
PressureMirroring()

density_calculator for BoundaryModelDummyParticles.

Note

This boundary model requires high viscosity for stability with WCSPH. It also produces significantly worse results than AdamiPressureExtrapolation and is not more efficient because smaller time steps are required due to more noise in the pressure. We added this model only for research purposes and for comparison with SPlisHSPlasH.

source

No-slip conditions

For the interaction of dummy particles and fluid particles, Adami et al. (2012) impose a no-slip boundary condition by assigning a wall velocity $v_w$ to the dummy particle.

The wall velocity of particle $a$ is calculated from the prescribed boundary particle velocity $v_a$ and the smoothed velocity field

\[v_w = 2 v_a - \frac{\sum_b v_b W_{ab}}{\sum_b W_{ab}},\]

where the sum is over all fluid particles.

By choosing the viscosity model ViscosityAdami for viscosity, a no-slip condition is imposed. It is recommended to choose nu in the order of either the kinematic viscosity parameter of the adjacent fluid or the equivalent from the artificial parameter alpha of the adjacent fluid ($\nu = \frac{\alpha h c }{2d + 4}$). When omitting the viscous interaction (default viscosity=nothing), a free-slip wall boundary condition is applied.

Warning

The viscosity model ArtificialViscosityMonaghan for BoundaryModelDummyParticles has not been verified yet.

Repulsive Particles

Boundaries modeled as boundary particles which exert forces on the fluid particles (Monaghan, Kajtar, 2009). The force on fluid particle $a$ due to boundary particle $b$ is given by

\[f_{ab} = m_a \left(\tilde{f}_{ab} - m_b \Pi_{ab} \nabla_{r_a} W(\Vert r_a - r_b \Vert, h)\right)\]

with

\[\tilde{f}_{ab} = \frac{K}{\beta^{n-1}} \frac{r_{ab}}{\Vert r_{ab} \Vert (\Vert r_{ab} \Vert - d)} \Phi(\Vert r_{ab} \Vert, h) \frac{2 m_b}{m_a + m_b},\]

where $m_a$ and $m_b$ are the masses of fluid particle $a$ and boundary particle $b$ respectively, $r_{ab} = r_a - r_b$ is the difference of the coordinates of particles $a$ and $b$, $d$ denotes the boundary particle spacing and $n$ denotes the number of dimensions (see Monaghan & Kajtar, 2009, Equation (3.1) and Valizadeh & Monaghan, 2015). Note that the repulsive acceleration $\tilde{f}_{ab}$ does not depend on the masses of the boundary particles. Here, $\Phi$ denotes the 1D Wendland C4 kernel, normalized to $1.77$ for $q=0$ (Monaghan & Kajtar, 2009, Section 4), with $\Phi(r, h) = w(r/h)$ and

\[w(q) = \begin{cases} (1.77/32) (1 + (5/2)q + 2q^2)(2 - q)^5 & \text{if } 0 \leq q < 2 \\ 0 & \text{if } q \geq 2. \end{cases}\]

The boundary particles are assumed to have uniform spacing by the factor $\beta$ smaller than the expected fluid particle spacing. For example, if the fluid particles have an expected spacing of $0.3$ and the boundary particles have a uniform spacing of $0.1$, then this parameter should be set to $\beta = 3$. According to Monaghan & Kajtar (2009), a value of $\beta = 3$ for the Wendland C4 that we use here is reasonable for most computing purposes.

The parameter $K$ is used to scale the force exerted by the boundary particles. In Monaghan & Kajtar (2009), a value of $gD$ is used for static tank simulations, where $g$ is the gravitational acceleration and $D$ is the depth of the fluid.

The viscosity $\Pi_{ab}$ is calculated according to the viscosity used in the simulation, where the density of the boundary particle if needed is assumed to be identical to the density of the fluid particle.

No-slip condition

By choosing the viscosity model ArtificialViscosityMonaghan for viscosity, a no-slip condition is imposed. When omitting the viscous interaction (default viscosity=nothing), a free-slip wall boundary condition is applied.

Warning

The no-slip conditions for BoundaryModelMonaghanKajtar have not been verified yet.

TrixiParticles.BoundaryModelMonaghanKajtarType
BoundaryModelMonaghanKajtar(K, beta, boundary_particle_spacing, mass;
-                            viscosity=nothing)

Boundary model for BoundarySPHSystem.

Arguments

  • K: Scaling factor for repulsive force.
  • beta: Ratio of fluid particle spacing to boundary particle spacing.
  • boundary_particle_spacing: Boundary particle spacing.
  • mass: Vector holding the mass of each boundary particle.

Keywords

  • viscosity: Free-slip (default) or no-slip condition. See description above for further information.
source

Open Boundaries

TrixiParticles.OpenBoundarySPHSystemType
OpenBoundarySPHSystem(boundary_zone::Union{InFlow, OutFlow};
+                            viscosity=nothing)

Boundary model for BoundarySPHSystem.

Arguments

  • K: Scaling factor for repulsive force.
  • beta: Ratio of fluid particle spacing to boundary particle spacing.
  • boundary_particle_spacing: Boundary particle spacing.
  • mass: Vector holding the mass of each boundary particle.

Keywords

  • viscosity: Free-slip (default) or no-slip condition. See description above for further information.
source

Open Boundaries

TrixiParticles.OpenBoundarySPHSystemType
OpenBoundarySPHSystem(boundary_zone::Union{InFlow, OutFlow};
                       fluid_system::FluidSystem, buffer_size::Integer,
                       boundary_model,
                       reference_velocity=nothing,
                       reference_pressure=nothing,
-                      reference_density=nothing)

Open boundary system for in- and outflow particles.

Arguments

  • boundary_zone: Use InFlow for an inflow and OutFlow for an outflow boundary.

Keywords

  • fluid_system: The corresponding fluid system
  • boundary_model: Boundary model (see Open Boundary Models)
  • buffer_size: Number of buffer particles.
  • reference_velocity: Reference velocity is either a function mapping each particle's coordinates and time to its velocity, an array where the $i$-th column holds the velocity of particle $i$ or, for a constant fluid velocity, a vector holding this velocity.
  • reference_pressure: Reference pressure is either a function mapping each particle's coordinates and time to its pressure, a vector holding the pressure of each particle, or a scalar for a constant pressure over all particles.
  • reference_density: Reference density is either a function mapping each particle's coordinates and time to its density, a vector holding the density of each particle, or a scalar for a constant density over all particles.
Experimental Implementation

This is an experimental feature and may change in any future releases.

source
TrixiParticles.InFlowType
InFlow(; plane, flow_direction, density, particle_spacing,
+                      reference_density=nothing)

Open boundary system for in- and outflow particles.

Arguments

  • boundary_zone: Use InFlow for an inflow and OutFlow for an outflow boundary.

Keywords

  • fluid_system: The corresponding fluid system
  • boundary_model: Boundary model (see Open Boundary Models)
  • buffer_size: Number of buffer particles.
  • reference_velocity: Reference velocity is either a function mapping each particle's coordinates and time to its velocity, an array where the $i$-th column holds the velocity of particle $i$ or, for a constant fluid velocity, a vector holding this velocity.
  • reference_pressure: Reference pressure is either a function mapping each particle's coordinates and time to its pressure, a vector holding the pressure of each particle, or a scalar for a constant pressure over all particles.
  • reference_density: Reference density is either a function mapping each particle's coordinates and time to its density, a vector holding the density of each particle, or a scalar for a constant density over all particles.
Experimental Implementation

This is an experimental feature and may change in any future releases.

source
TrixiParticles.InFlowType
InFlow(; plane, flow_direction, density, particle_spacing,
        initial_condition=nothing, extrude_geometry=nothing,
        open_boundary_layers::Integer)

Inflow boundary zone for OpenBoundarySPHSystem.

The specified plane (line in 2D or rectangle in 3D) will be extruded in upstream direction (the direction opposite to flow_direction) to create a box for the boundary zone. There are three ways to specify the actual shape of the inflow:

  1. Don't pass initial_condition or extrude_geometry. The boundary zone box will then be filled with inflow particles (default).
  2. Specify extrude_geometry by passing a 1D shape in 2D or a 2D shape in 3D, which is then extruded in upstream direction to create the inflow particles.
    • In 2D, the shape must be either an initial condition with 2D coordinates, which lies on the line specified by plane, or an initial condition with 1D coordinates, which lies on the line specified by plane when a y-coordinate of 0 is added.
    • In 3D, the shape must be either an initial condition with 3D coordinates, which lies in the rectangle specified by plane, or an initial condition with 2D coordinates, which lies in the rectangle specified by plane when a z-coordinate of 0 is added.
  3. Specify initial_condition by passing a 2D initial condition in 2D or a 3D initial condition in 3D, which will be used for the inflow particles.
Note

Particles outside the boundary zone box will be removed.

Keywords

  • plane: Tuple of points defining a part of the surface of the domain. The points must either span a line in 2D or a rectangle in 3D. This line or rectangle is then extruded in upstream direction to obtain the boundary zone. In 2D, pass two points $(A, B)$, so that the interval $[A, B]$ is the inflow surface. In 3D, pass three points $(A, B, C)$, so that the rectangular inflow surface is spanned by the vectors $\widehat{AB}$ and $\widehat{AC}$. These two vectors must be orthogonal.
  • flow_direction: Vector defining the flow direction.
  • open_boundary_layers: Number of particle layers in upstream direction.
  • particle_spacing: The spacing between the particles (see InitialCondition).
  • density: Particle density (see InitialCondition).
  • initial_condition=nothing: InitialCondition for the inflow particles. Particles outside the boundary zone will be removed. Do not use together with extrude_geometry.
  • extrude_geometry=nothing: 1D shape in 2D or 2D shape in 3D, which lies on the plane and is extruded upstream to obtain the inflow particles. See point 2 above for more details.

Examples

# 2D
 plane_points = ([0.0, 0.0], [0.0, 1.0])
@@ -44,7 +44,7 @@
 circle = SphereShape(0.1, 0.5, (0.5, 0.5), 1.0, sphere_type=RoundSphere())
 
 inflow = InFlow(; plane=plane_points, particle_spacing=0.1, flow_direction, density=1.0,
-                extrude_geometry=circle, open_boundary_layers=4)
Experimental Implementation

This is an experimental feature and may change in any future releases.

source
TrixiParticles.OutFlowType
OutFlow(; plane, flow_direction, density, particle_spacing,
+                extrude_geometry=circle, open_boundary_layers=4)
Experimental Implementation

This is an experimental feature and may change in any future releases.

source
TrixiParticles.OutFlowType
OutFlow(; plane, flow_direction, density, particle_spacing,
         initial_condition=nothing, extrude_geometry=nothing,
         open_boundary_layers::Integer)

Outflow boundary zone for OpenBoundarySPHSystem.

The specified plane (line in 2D or rectangle in 3D) will be extruded in downstream direction (the direction in flow_direction) to create a box for the boundary zone. There are three ways to specify the actual shape of the outflow:

  1. Don't pass initial_condition or extrude_geometry. The boundary zone box will then be filled with outflow particles (default).
  2. Specify extrude_geometry by passing a 1D shape in 2D or a 2D shape in 3D, which is then extruded in downstream direction to create the outflow particles.
    • In 2D, the shape must be either an initial condition with 2D coordinates, which lies on the line specified by plane, or an initial condition with 1D coordinates, which lies on the line specified by plane when a y-coordinate of 0 is added.
    • In 3D, the shape must be either an initial condition with 3D coordinates, which lies in the rectangle specified by plane, or an initial condition with 2D coordinates, which lies in the rectangle specified by plane when a z-coordinate of 0 is added.
  3. Specify initial_condition by passing a 2D initial condition in 2D or a 3D initial condition in 3D, which will be used for the outflow particles.
Note

Particles outside the boundary zone box will be removed.

Keywords

  • plane: Tuple of points defining a part of the surface of the domain. The points must either span a line in 2D or a rectangle in 3D. This line or rectangle is then extruded in downstream direction to obtain the boundary zone. In 2D, pass two points $(A, B)$, so that the interval $[A, B]$ is the outflow surface. In 3D, pass three points $(A, B, C)$, so that the rectangular outflow surface is spanned by the vectors $\widehat{AB}$ and $\widehat{AC}$. These two vectors must be orthogonal.
  • flow_direction: Vector defining the flow direction.
  • open_boundary_layers: Number of particle layers in downstream direction.
  • particle_spacing: The spacing between the particles (see InitialCondition).
  • density: Particle density (see InitialCondition).
  • initial_condition=nothing: InitialCondition for the outflow particles. Particles outside the boundary zone will be removed. Do not use together with extrude_geometry.
  • extrude_geometry=nothing: 1D shape in 2D or 2D shape in 3D, which lies on the plane and is extruded downstream to obtain the outflow particles. See point 2 above for more details.

Examples

# 2D
 plane_points = ([0.0, 0.0], [0.0, 1.0])
@@ -64,4 +64,4 @@
 circle = SphereShape(0.1, 0.5, (0.5, 0.5), 1.0, sphere_type=RoundSphere())
 
 outflow = OutFlow(; plane=plane_points, particle_spacing=0.1, flow_direction, density=1.0,
-                  extrude_geometry=circle, open_boundary_layers=4)
Experimental Implementation

This is an experimental feature and may change in any future releases.

source

Open Boundary Models

Method of characteristics

TrixiParticles.BoundaryModelLastiwkaType
BoundaryModelLastiwka()

Boundary model for OpenBoundarySPHSystem. This model uses the characteristic variables to propagate the appropriate values to the outlet or inlet and have been proposed by Lastiwka et al. (2009). For more information about the method see description below.

source

The difficulty in non-reflecting boundary conditions, also called open boundaries, is to determine the appropriate boundary values of the exact characteristics of the Euler equations. Assuming the flow near the boundaries is normal to the boundary and free of shock waves and significant viscous effects, it can be shown that three characteristic variables exist:

  • $J_1$, associated with convection of entropy and propagates at flow velocity,
  • $J_2$, downstream-running characteristics,
  • $J_3$, upstream-running characteristics.

Giles (1990) derived those variables based on a linearized set of governing equations:

\[J_1 = -c_s^2 (\rho - \rho_{\text{ref}}) + (p - p_{\text{ref}})\]

\[J_2 = \rho c_s (v - v_{\text{ref}}) + (p - p_{\text{ref}})\]

\[J_3 = - \rho c_s (v - v_{\text{ref}}) + (p - p_{\text{ref}})\]

where the subscript "ref" denotes the reference flow near the boundaries, which can be prescribed.

Specifying the reference variables is not equivalent to prescription of $\rho$, $v$ and $p$ directly, since the perturbation from the reference flow is allowed.

Lastiwka et al. (2009) applied the method of characteristic to SPH and determined the number of variables that should be prescribed at the boundary and the number which should be propagated from the fluid domain to the boundary:

  • For an inflow boundary:

    • Prescribe downstream-running characteristics $J_1$ and $J_2$
    • Transmit $J_3$ from the fluid domain (allow $J_3$ to propagate upstream to the boundary).
  • For an outflow boundary:

    • Prescribe upstream-running characteristic $J_3$
    • Transmit $J_1$ and $J_2$ from the fluid domain.

Prescribing is done by simply setting the characteristics to zero. To transmit the characteristics from the fluid domain, or in other words, to carry the information of the fluid to the boundaries, Negi et al. (2020) use a Shepard Interpolation

\[f_i = \frac{\sum_j^N f_j W_{ij}}{\sum_j^N W_{ij}},\]

where the $i$-th particle is a boundary particle, $f$ is either $J_1$, $J_2$ or $J_3$ and $N$ is the set of neighboring fluid particles.

To express pressure $p$, density $\rho$ and velocity $v$ as functions of the characteristic variables, the system of equations from the characteristic variables is inverted and gives

\[ \rho - \rho_{\text{ref}} = \frac{1}{c_s^2} \left( -J_1 + \frac{1}{2} J_2 + \frac{1}{2} J_3 \right),\]

\[u - u_{\text{ref}}= \frac{1}{2\rho c_s} \left( J_2 - J_3 \right),\]

\[p - p_{\text{ref}} = \frac{1}{2} \left( J_2 + J_3 \right).\]

With $J_1$, $J_2$ and $J_3$ determined, we can easily solve for the actual variables for each particle.

+ extrude_geometry=circle, open_boundary_layers=4)
Experimental Implementation

This is an experimental feature and may change in any future releases.

source

Open Boundary Models

Method of characteristics

TrixiParticles.BoundaryModelLastiwkaType
BoundaryModelLastiwka()

Boundary model for OpenBoundarySPHSystem. This model uses the characteristic variables to propagate the appropriate values to the outlet or inlet and have been proposed by Lastiwka et al. (2009). For more information about the method see description below.

source

The difficulty in non-reflecting boundary conditions, also called open boundaries, is to determine the appropriate boundary values of the exact characteristics of the Euler equations. Assuming the flow near the boundaries is normal to the boundary and free of shock waves and significant viscous effects, it can be shown that three characteristic variables exist:

Giles (1990) derived those variables based on a linearized set of governing equations:

\[J_1 = -c_s^2 (\rho - \rho_{\text{ref}}) + (p - p_{\text{ref}})\]

\[J_2 = \rho c_s (v - v_{\text{ref}}) + (p - p_{\text{ref}})\]

\[J_3 = - \rho c_s (v - v_{\text{ref}}) + (p - p_{\text{ref}})\]

where the subscript "ref" denotes the reference flow near the boundaries, which can be prescribed.

Specifying the reference variables is not equivalent to prescription of $\rho$, $v$ and $p$ directly, since the perturbation from the reference flow is allowed.

Lastiwka et al. (2009) applied the method of characteristic to SPH and determined the number of variables that should be prescribed at the boundary and the number which should be propagated from the fluid domain to the boundary:

Prescribing is done by simply setting the characteristics to zero. To transmit the characteristics from the fluid domain, or in other words, to carry the information of the fluid to the boundaries, Negi et al. (2020) use a Shepard Interpolation

\[f_i = \frac{\sum_j^N f_j W_{ij}}{\sum_j^N W_{ij}},\]

where the $i$-th particle is a boundary particle, $f$ is either $J_1$, $J_2$ or $J_3$ and $N$ is the set of neighboring fluid particles.

To express pressure $p$, density $\rho$ and velocity $v$ as functions of the characteristic variables, the system of equations from the characteristic variables is inverted and gives

\[ \rho - \rho_{\text{ref}} = \frac{1}{c_s^2} \left( -J_1 + \frac{1}{2} J_2 + \frac{1}{2} J_3 \right),\]

\[u - u_{\text{ref}}= \frac{1}{2\rho c_s} \left( J_2 - J_3 \right),\]

\[p - p_{\text{ref}} = \frac{1}{2} \left( J_2 + J_3 \right).\]

With $J_1$, $J_2$ and $J_3$ determined, we can easily solve for the actual variables for each particle.

diff --git a/previews/PR514/systems/dem/index.html b/previews/PR514/systems/dem/index.html index fb5591e81..48a47ac49 100644 --- a/previews/PR514/systems/dem/index.html +++ b/previews/PR514/systems/dem/index.html @@ -1,3 +1,3 @@ Discrete Element Method (Solid) · TrixiParticles.jl

Discrete Element Method

The Discrete Element Method (DEM) is a computational technique widely used in physics, engineering, and applied mathematics for simulating the mechanical behavior of granular materials, such as powders, sand, soil, or rock, as well as other discontinua. Unlike continuum mechanics that treats materials as continuous, DEM considers individual particles or elements and their interactions. This approach provides detailed insights into the micro-mechanical behavior of materials, making it particularly valuable in fields such as geomechanics, material science, and mechanical engineering.

Fundamental Principles

The core idea behind DEM is the discretization of a material system into a finite set of distinct, interacting mass elements (particles). These elements (particles) can vary in shape, size, and properties, and they interact with each other and possibly with their boundaries through contact forces and potential fields. The motion and behavior of each mass element are governed by Newton's laws of motion, accounting for the forces and moments acting upon them.

TrixiParticles.DEMSystemType
DEMSystem(initial_condition, normal_stiffness, elastic_modulus, poissons_ratio;
- damping_coefficient=0.0001, acceleration=ntuple(_ -> 0.0, NDIMS), source_terms=nothing)

Constructs a Discrete Element Method (DEM) system for numerically simulating the dynamics of granular and particulate matter. DEM is employed to simulate and analyze the motion, interactions, and collective behavior of assemblies of discrete, solid particles, typically under mechanical loading. The model accounts for individual particle characteristics and implements interaction laws that govern contact forces (normal and tangential), based on specified material properties and contact mechanics.

Arguments

  • initial_condition: Initial condition of the system, encapsulating the initial positions, velocities, masses, and radii of particles.
  • normal_stiffness: Normal stiffness coefficient for particle-particle and particle-wall contacts.
  • elastic_modulus: Elastic modulus for this particle system.
  • poissons_ratio: Poisson ratio for this particle system.

Keywords

  • acceleration: Global acceleration vector applied to the system, such as gravity. Specified as an SVector of length NDIMS, with a default of zero in each dimension.
  • source_terms: Optional; additional forces or modifications to particle dynamics not captured by standard DEM interactions, such as electromagnetic forces or user-defined perturbations.
  • damping_coefficient=0.0001: Set a damping coefficient for the collision interactions.
Experimental Implementation

This is an experimental feature and may change in a future releases.

References

[9], [10], [11]

source
+ damping_coefficient=0.0001, acceleration=ntuple(_ -> 0.0, NDIMS), source_terms=nothing)

Constructs a Discrete Element Method (DEM) system for numerically simulating the dynamics of granular and particulate matter. DEM is employed to simulate and analyze the motion, interactions, and collective behavior of assemblies of discrete, solid particles, typically under mechanical loading. The model accounts for individual particle characteristics and implements interaction laws that govern contact forces (normal and tangential), based on specified material properties and contact mechanics.

Arguments

Keywords

Experimental Implementation

This is an experimental feature and may change in a future releases.

References

[9], [10], [11]

source diff --git a/previews/PR514/systems/entropically_damped_sph/index.html b/previews/PR514/systems/entropically_damped_sph/index.html index e2e2f97ef..17ced5fe0 100644 --- a/previews/PR514/systems/entropically_damped_sph/index.html +++ b/previews/PR514/systems/entropically_damped_sph/index.html @@ -7,4 +7,4 @@ transport_velocity=nothing, alpha=0.5, viscosity=nothing, acceleration=ntuple(_ -> 0.0, NDIMS), buffer_size=nothing, - source_terms=nothing)

System for particles of a fluid. As opposed to the weakly compressible SPH scheme, which uses an equation of state, this scheme uses a pressure evolution equation to calculate the pressure. See Entropically Damped Artificial Compressibility for SPH for more details on the method.

Arguments

Keyword Arguments

source

Transport Velocity Formulation (TVF)

Standard SPH suffers from problems like tensile instability or the creation of void regions in the flow. To address these problems, Adami (2013) modified the advection velocity and added an extra term to the momentum equation. The authors introduced the so-called Transport Velocity Formulation (TVF) for WCSPH. Ramachandran (2019) applied the TVF also for the EDAC scheme.

The transport velocity $\tilde{v}_a$ of particle $a$ is used to evolve the position of the particle $r_a$ from one time step to the next by

\[\frac{\mathrm{d} r_a}{\mathrm{d}t} = \tilde{v}_a\]

and is obtained at every time-step $\Delta t$ from

\[\tilde{v}_a (t + \Delta t) = v_a (t) + \Delta t \left(\frac{\tilde{\mathrm{d}} v_a}{\mathrm{d}t} - \frac{1}{\rho_a} \nabla p_{\text{background}} \right),\]

where $\rho_a$ is the density of particle $a$ and $p_{\text{background}}$ is a constant background pressure field. The tilde in the second term of the right hand side indicates that the material derivative has an advection part.

The discretized form of the last term is

\[ -\frac{1}{\rho_a} \nabla p_{\text{background}} \approx -\frac{p_{\text{background}}}{m_a} \sum_b \left(V_a^2 + V_b^2 \right) \nabla_a W_{ab},\]

where $V_a$, $V_b$ denote the volume of particles $a$ and $b$ respectively. Note that although in the continuous case $\nabla p_{\text{background}} = 0$, the discretization is not 0th-order consistent for non-uniform particle distribution, which means that there is a non-vanishing contribution only when particles are disordered. That also means that $p_{\text{background}}$ occurs as prefactor to correct the trajectory of a particle resulting in uniform pressure distributions. Suggested is a background pressure which is in the order of the reference pressure but can be chosen arbitrarily large when the time-step criterion is adjusted.

The inviscid momentum equation with an additional convection term for a particle moving with $\tilde{v}$ is

\[\frac{\tilde{\mathrm{d}} \left( \rho v \right)}{\mathrm{d}t} = -\nabla p + \nabla \cdot \bm{A},\]

where the tensor $\bm{A} = \rho v\left(\tilde{v}-v\right)^T$ is a consequence of the modified advection velocity and can be interpreted as the convection of momentum with the relative velocity $\tilde{v}-v$.

The discretized form of the momentum equation for a particle $a$ reads as

\[\frac{\tilde{\mathrm{d}} v_a}{\mathrm{d}t} = \frac{1}{m_a} \sum_b \left(V_a^2 + V_b^2 \right) \left[ -\tilde{p}_{ab} \nabla_a W_{ab} + \frac{1}{2} \left(\bm{A}_a + \bm{A}_b \right) \cdot \nabla_a W_{ab} \right].\]

Here, $\tilde{p}_{ab}$ is the density-weighted pressure

\[\tilde{p}_{ab} = \frac{\rho_b p_a + \rho_a p_b}{\rho_a + \rho_b},\]

with the density $\rho_a$, $\rho_b$ and the pressure $p_a$, $p_b$ of particles $a$ and $b$ respectively. $\bm{A}_a$ and $\bm{A}_b$ are the convection tensors for particle $a$ and $b$ respectively and are given, e.g. for particle $a$, as $\bm{A}_a = \rho v_a\left(\tilde{v}_a-v_a\right)^T$.

TrixiParticles.TransportVelocityAdamiType
TransportVelocityAdami(background_pressure::Real)

Transport Velocity Formulation (TVF) to suppress pairing and tensile instability. See TVF for more details of the method.

Arguments

  • background_pressure: Background pressure. Suggested is a background pressure which is on the order of the reference pressure.
Note

There is no need for an artificial viscosity to suppress tensile instability when using TransportVelocityAdami. Thus, it is highly recommended to use ViscosityAdami as viscosity model, since ArtificialViscosityMonaghan leads to bad results.

source
+ source_terms=nothing)

System for particles of a fluid. As opposed to the weakly compressible SPH scheme, which uses an equation of state, this scheme uses a pressure evolution equation to calculate the pressure. See Entropically Damped Artificial Compressibility for SPH for more details on the method.

Arguments

Keyword Arguments

source

Transport Velocity Formulation (TVF)

Standard SPH suffers from problems like tensile instability or the creation of void regions in the flow. To address these problems, Adami (2013) modified the advection velocity and added an extra term to the momentum equation. The authors introduced the so-called Transport Velocity Formulation (TVF) for WCSPH. Ramachandran (2019) applied the TVF also for the EDAC scheme.

The transport velocity $\tilde{v}_a$ of particle $a$ is used to evolve the position of the particle $r_a$ from one time step to the next by

\[\frac{\mathrm{d} r_a}{\mathrm{d}t} = \tilde{v}_a\]

and is obtained at every time-step $\Delta t$ from

\[\tilde{v}_a (t + \Delta t) = v_a (t) + \Delta t \left(\frac{\tilde{\mathrm{d}} v_a}{\mathrm{d}t} - \frac{1}{\rho_a} \nabla p_{\text{background}} \right),\]

where $\rho_a$ is the density of particle $a$ and $p_{\text{background}}$ is a constant background pressure field. The tilde in the second term of the right hand side indicates that the material derivative has an advection part.

The discretized form of the last term is

\[ -\frac{1}{\rho_a} \nabla p_{\text{background}} \approx -\frac{p_{\text{background}}}{m_a} \sum_b \left(V_a^2 + V_b^2 \right) \nabla_a W_{ab},\]

where $V_a$, $V_b$ denote the volume of particles $a$ and $b$ respectively. Note that although in the continuous case $\nabla p_{\text{background}} = 0$, the discretization is not 0th-order consistent for non-uniform particle distribution, which means that there is a non-vanishing contribution only when particles are disordered. That also means that $p_{\text{background}}$ occurs as prefactor to correct the trajectory of a particle resulting in uniform pressure distributions. Suggested is a background pressure which is in the order of the reference pressure but can be chosen arbitrarily large when the time-step criterion is adjusted.

The inviscid momentum equation with an additional convection term for a particle moving with $\tilde{v}$ is

\[\frac{\tilde{\mathrm{d}} \left( \rho v \right)}{\mathrm{d}t} = -\nabla p + \nabla \cdot \bm{A},\]

where the tensor $\bm{A} = \rho v\left(\tilde{v}-v\right)^T$ is a consequence of the modified advection velocity and can be interpreted as the convection of momentum with the relative velocity $\tilde{v}-v$.

The discretized form of the momentum equation for a particle $a$ reads as

\[\frac{\tilde{\mathrm{d}} v_a}{\mathrm{d}t} = \frac{1}{m_a} \sum_b \left(V_a^2 + V_b^2 \right) \left[ -\tilde{p}_{ab} \nabla_a W_{ab} + \frac{1}{2} \left(\bm{A}_a + \bm{A}_b \right) \cdot \nabla_a W_{ab} \right].\]

Here, $\tilde{p}_{ab}$ is the density-weighted pressure

\[\tilde{p}_{ab} = \frac{\rho_b p_a + \rho_a p_b}{\rho_a + \rho_b},\]

with the density $\rho_a$, $\rho_b$ and the pressure $p_a$, $p_b$ of particles $a$ and $b$ respectively. $\bm{A}_a$ and $\bm{A}_b$ are the convection tensors for particle $a$ and $b$ respectively and are given, e.g. for particle $a$, as $\bm{A}_a = \rho v_a\left(\tilde{v}_a-v_a\right)^T$.

TrixiParticles.TransportVelocityAdamiType
TransportVelocityAdami(background_pressure::Real)

Transport Velocity Formulation (TVF) to suppress pairing and tensile instability. See TVF for more details of the method.

Arguments

  • background_pressure: Background pressure. Suggested is a background pressure which is on the order of the reference pressure.
Note

There is no need for an artificial viscosity to suppress tensile instability when using TransportVelocityAdami. Thus, it is highly recommended to use ViscosityAdami as viscosity model, since ArtificialViscosityMonaghan leads to bad results.

source
diff --git a/previews/PR514/systems/total_lagrangian_sph/index.html b/previews/PR514/systems/total_lagrangian_sph/index.html index 21a212473..ce1680fc6 100644 --- a/previews/PR514/systems/total_lagrangian_sph/index.html +++ b/previews/PR514/systems/total_lagrangian_sph/index.html @@ -7,5 +7,5 @@ young_modulus, poisson_ratio; n_fixed_particles=0, boundary_model=nothing, acceleration=ntuple(_ -> 0.0, NDIMS), - penalty_force=nothing, source_terms=nothing)

System for particles of an elastic structure.

A Total Lagrangian framework is used wherein the governing equations are formulated such that all relevant quantities and operators are measured with respect to the initial configuration (O’Connor & Rogers 2021, Belytschko et al. 2000). See Total Lagrangian SPH for more details on the method.

Arguments

Keyword Arguments

Note

The fixed particles must be the last particles in the InitialCondition. To do so, e.g. use the union function:

solid = union(beam, fixed_particles)

where beam and fixed_particles are of type InitialCondition.

source

Penalty Force

In FEM, underintegrated elements can deform without an associated increase of energy. This is caused by the stiffness matrix having zero eigenvalues (so-called hourglass modes). The name "hourglass modes" comes from the fact that elements can deform into an hourglass shape.

Similar effects can occur in SPH as well. Particles can change positions without changing the SPH approximation of the deformation gradient $\bm{F}$, thus, without causing an increase of energy. To ensure regular particle positions, we can apply similar correction forces as are used in FEM.

Ganzenmüller (2015) introduced a so-called hourglass correction force or penalty force $f^{PF}$, which is given by

\[\bm{f}_a^{PF} = \frac{1}{2} \alpha \sum_b \frac{m_{0a} m_{0b} W_{0ab}}{\rho_{0a}\rho_{0b} |\bm{X}_{ab}|^2} - \left( E \delta_{ab}^a + E \delta_{ba}^b \right) \frac{\bm{x}_{ab}}{|\bm{x}_{ab}|}\]

The subscripts $a$ and $b$ denote quantities of particle $a$ and $b$, respectively. The zero subscript on quantities denotes that the quantity is to be measured in the initial configuration. The difference in the initial coordinates is denoted by $\bm{X}_{ab} = \bm{X}_a - \bm{X}_b$, the difference in the current coordinates is denoted by $\bm{x}_{ab} = \bm{x}_a - \bm{x}_b$. Note that Ganzenmüller (2015) has a flipped sign here because they define $\bm{x}_{ab}$ the other way around.

This correction force is based on the potential energy density of a Hookean material. Thus, $E$ is the Young's modulus and $\alpha$ is a dimensionless coefficient that controls the amplitude of hourglass correction. The separation vector $\delta_{ab}^a$ indicates the change of distance which the particle separation should attain in order to minimize the error and is given by

\[ \delta_{ab}^a = \frac{\bm{\epsilon}_{ab}^a \cdot \bm{x_{ab}}}{|\bm{x}_{ab}|},\]

where the error vector is defined as

\[ \bm{\epsilon}_{ab}^a = \bm{F}_a \bm{X}_{ab} - \bm{x}_{ab}.\]

TrixiParticles.PenaltyForceGanzenmuellerType
PenaltyForceGanzenmueller(; alpha=0.1)

Penalty force to ensure regular particle positions under large deformations.

Keywords

  • alpha: Coefficient to control the amplitude of hourglass correction.
source
+ penalty_force=nothing, source_terms=nothing)

System for particles of an elastic structure.

A Total Lagrangian framework is used wherein the governing equations are formulated such that all relevant quantities and operators are measured with respect to the initial configuration (O’Connor & Rogers 2021, Belytschko et al. 2000). See Total Lagrangian SPH for more details on the method.

Arguments

Keyword Arguments

Note

The fixed particles must be the last particles in the InitialCondition. To do so, e.g. use the union function:

solid = union(beam, fixed_particles)

where beam and fixed_particles are of type InitialCondition.

source

Penalty Force

In FEM, underintegrated elements can deform without an associated increase of energy. This is caused by the stiffness matrix having zero eigenvalues (so-called hourglass modes). The name "hourglass modes" comes from the fact that elements can deform into an hourglass shape.

Similar effects can occur in SPH as well. Particles can change positions without changing the SPH approximation of the deformation gradient $\bm{F}$, thus, without causing an increase of energy. To ensure regular particle positions, we can apply similar correction forces as are used in FEM.

Ganzenmüller (2015) introduced a so-called hourglass correction force or penalty force $f^{PF}$, which is given by

\[\bm{f}_a^{PF} = \frac{1}{2} \alpha \sum_b \frac{m_{0a} m_{0b} W_{0ab}}{\rho_{0a}\rho_{0b} |\bm{X}_{ab}|^2} + \left( E \delta_{ab}^a + E \delta_{ba}^b \right) \frac{\bm{x}_{ab}}{|\bm{x}_{ab}|}\]

The subscripts $a$ and $b$ denote quantities of particle $a$ and $b$, respectively. The zero subscript on quantities denotes that the quantity is to be measured in the initial configuration. The difference in the initial coordinates is denoted by $\bm{X}_{ab} = \bm{X}_a - \bm{X}_b$, the difference in the current coordinates is denoted by $\bm{x}_{ab} = \bm{x}_a - \bm{x}_b$. Note that Ganzenmüller (2015) has a flipped sign here because they define $\bm{x}_{ab}$ the other way around.

This correction force is based on the potential energy density of a Hookean material. Thus, $E$ is the Young's modulus and $\alpha$ is a dimensionless coefficient that controls the amplitude of hourglass correction. The separation vector $\delta_{ab}^a$ indicates the change of distance which the particle separation should attain in order to minimize the error and is given by

\[ \delta_{ab}^a = \frac{\bm{\epsilon}_{ab}^a \cdot \bm{x_{ab}}}{|\bm{x}_{ab}|},\]

where the error vector is defined as

\[ \bm{\epsilon}_{ab}^a = \bm{F}_a \bm{X}_{ab} - \bm{x}_{ab}.\]

TrixiParticles.PenaltyForceGanzenmuellerType
PenaltyForceGanzenmueller(; alpha=0.1)

Penalty force to ensure regular particle positions under large deformations.

Keywords

  • alpha: Coefficient to control the amplitude of hourglass correction.
source
diff --git a/previews/PR514/systems/weakly_compressible_sph/index.html b/previews/PR514/systems/weakly_compressible_sph/index.html index 6ad093eff..d88ab1d63 100644 --- a/previews/PR514/systems/weakly_compressible_sph/index.html +++ b/previews/PR514/systems/weakly_compressible_sph/index.html @@ -5,12 +5,12 @@ viscosity=nothing, density_diffusion=nothing, acceleration=ntuple(_ -> 0.0, NDIMS), buffer_size=nothing, - correction=nothing, source_terms=nothing)

System for particles of a fluid. The weakly compressible SPH (WCSPH) scheme is used, wherein a stiff equation of state generates large pressure changes for small density variations. See Weakly Compressible SPH for more details on the method.

Arguments

Keyword Arguments

source

Equation of State

The equation of state is used to relate fluid density to pressure and thus allow an explicit simulation of the WCSPH system. The equation in the following formulation was introduced by Cole (1948) (pp. 39 and 43). The pressure $p$ is calculated as

\[ p = B \left(\left(\frac{\rho}{\rho_0}\right)^\gamma - 1\right) + p_{\text{background}},\]

where $\rho$ denotes the density, $\rho_0$ the reference density, and $p_{\text{background}}$ the background pressure, which is set to zero when applied to free-surface flows (Adami et al., 2012).

The bulk modulus, $B = \frac{\rho_0 c^2}{\gamma}$, is calculated from the artificial speed of sound $c$ and the isentropic exponent $\gamma$.

An ideal gas equation of state with a linear relationship between pressure and density can be obtained by choosing exponent=1, i.e.

\[ p = B \left( \frac{\rho}{\rho_0} -1 \right) = c^2(\rho - \rho_0).\]

For higher Reynolds numbers, exponent=7 is recommended, whereas at lower Reynolds numbers exponent=1 yields more accurate pressure estimates since pressure and density are proportional (see Morris, 1997).

When using SummationDensity (or DensityReinitializationCallback) and free surfaces, initializing particles with equal spacing will cause underestimated density and therefore strong attractive forces between particles at the free surface. Setting clip_negative_pressure=true can avoid this.

TrixiParticles.StateEquationColeType
StateEquationCole(; sound_speed, reference_density, exponent,
-                  background_pressure=0.0, clip_negative_pressure=false)

Equation of state to describe the relationship between pressure and density of water up to high pressures.

Keywords

  • sound_speed: Artificial speed of sound.
  • reference_density: Reference density of the fluid.
  • exponent: A value of 7 is usually used for most simulations.
  • background_pressure=0.0: Background pressure.
source

Viscosity

TODO: Explain viscosity.

TrixiParticles.ArtificialViscosityMonaghanType
ArtificialViscosityMonaghan(; alpha, beta=0.0, epsilon=0.01)

Artificial viscosity by Monaghan ([16], [17]), given by

\[\Pi_{ab} = + correction=nothing, source_terms=nothing)

System for particles of a fluid. The weakly compressible SPH (WCSPH) scheme is used, wherein a stiff equation of state generates large pressure changes for small density variations. See Weakly Compressible SPH for more details on the method.

Arguments

Keyword Arguments

  • viscosity: Viscosity model for this system (default: no viscosity). See ArtificialViscosityMonaghan or ViscosityAdami.
  • density_diffusion: Density diffusion terms for this system. See DensityDiffusion.
  • acceleration: Acceleration vector for the system. (default: zero vector)
  • buffer_size: Number of buffer particles. This is needed when simulating with OpenBoundarySPHSystem.
  • correction: Correction method used for this system. (default: no correction, see Corrections)
  • source_terms: Additional source terms for this system. Has to be either nothing (by default), or a function of (coords, velocity, density, pressure, t) (which are the quantities of a single particle), returning a Tuple or SVector that is to be added to the acceleration of that particle. See, for example, SourceTermDamping. Note that these source terms will not be used in the calculation of the boundary pressure when using a boundary with BoundaryModelDummyParticles and AdamiPressureExtrapolation. The keyword argument acceleration should be used instead for gravity-like source terms.
  • surface_tension: Surface tension model used for this SPH system. (default: no surface tension)
source

Equation of State

The equation of state is used to relate fluid density to pressure and thus allow an explicit simulation of the WCSPH system. The equation in the following formulation was introduced by Cole (1948) (pp. 39 and 43). The pressure $p$ is calculated as

\[ p = B \left(\left(\frac{\rho}{\rho_0}\right)^\gamma - 1\right) + p_{\text{background}},\]

where $\rho$ denotes the density, $\rho_0$ the reference density, and $p_{\text{background}}$ the background pressure, which is set to zero when applied to free-surface flows (Adami et al., 2012).

The bulk modulus, $B = \frac{\rho_0 c^2}{\gamma}$, is calculated from the artificial speed of sound $c$ and the isentropic exponent $\gamma$.

An ideal gas equation of state with a linear relationship between pressure and density can be obtained by choosing exponent=1, i.e.

\[ p = B \left( \frac{\rho}{\rho_0} -1 \right) = c^2(\rho - \rho_0).\]

For higher Reynolds numbers, exponent=7 is recommended, whereas at lower Reynolds numbers exponent=1 yields more accurate pressure estimates since pressure and density are proportional (see Morris, 1997).

When using SummationDensity (or DensityReinitializationCallback) and free surfaces, initializing particles with equal spacing will cause underestimated density and therefore strong attractive forces between particles at the free surface. Setting clip_negative_pressure=true can avoid this.

TrixiParticles.StateEquationColeType
StateEquationCole(; sound_speed, reference_density, exponent,
+                  background_pressure=0.0, clip_negative_pressure=false)

Equation of state to describe the relationship between pressure and density of water up to high pressures.

Keywords

  • sound_speed: Artificial speed of sound.
  • reference_density: Reference density of the fluid.
  • exponent: A value of 7 is usually used for most simulations.
  • background_pressure=0.0: Background pressure.
source

Viscosity

TODO: Explain viscosity.

TrixiParticles.ArtificialViscosityMonaghanType
ArtificialViscosityMonaghan(; alpha, beta=0.0, epsilon=0.01)

Artificial viscosity by Monaghan ([16], [17]), given by

\[\Pi_{ab} = \begin{cases} -(\alpha c \mu_{ab} + \beta \mu_{ab}^2) / \bar{\rho}_{ab} & \text{if } v_{ab} \cdot r_{ab} < 0, \\ 0 & \text{otherwise} -\end{cases}\]

with

\[\mu_{ab} = \frac{h v_{ab} \cdot r_{ab}}{\Vert r_{ab} \Vert^2 + \epsilon h^2},\]

where $\alpha, \beta, \epsilon$ are parameters, $c$ is the speed of sound, $h$ is the smoothing length, $r_{ab} = r_a - r_b$ is the difference of the coordinates of particles $a$ and $b$, $v_{ab} = v_a - v_b$ is the difference of their velocities, and $\bar{\rho}_{ab}$ is the arithmetic mean of their densities.

Note that $\alpha$ needs to adjusted for different resolutions to maintain a specific Reynolds Number. To do so, Monaghan (2005) defined an equivalent effective physical kinematic viscosity $\nu$ by

\[ \nu = \frac{\alpha h c }{2d + 4},\]

where $d$ is the dimension.

Keywords

  • alpha: A value of 0.02 is usually used for most simulations. For a relation with the kinematic viscosity, see description above.
  • beta=0.0: A value of 0.0 works well for most fluid simulations and simulations with shocks of moderate strength. In simulations where the Mach number can be very high, eg. astrophysical calculation, good results can be obtained by choosing a value of beta=2.0 and alpha=1.0.
  • epsilon=0.01: Parameter to prevent singularities.
source
TrixiParticles.ViscosityAdamiType
ViscosityAdami(; nu, epsilon=0.01)

Viscosity by Adami (2012). The viscous interaction is calculated with the shear force for incompressible flows given by

\[f_{ab} = \sum_w \bar{\eta}_{ab} \left( V_a^2 + V_b^2 \right) \frac{v_{ab}}{||r_{ab}||^2+\epsilon h_{ab}^2} \nabla W_{ab} \cdot r_{ab},\]

where $r_{ab} = r_a - r_b$ is the difference of the coordinates of particles $a$ and $b$, $v_{ab} = v_a - v_b$ is the difference of their velocities, $h$ is the smoothing length and $V$ is the particle volume. The parameter $\epsilon$ prevents singularities (see Ramachandran (2019)). The inter-particle-averaged shear stress is

\[ \bar{\eta}_{ab} =\frac{2 \eta_a \eta_b}{\eta_a + \eta_b},\]

where $\eta_a = \rho_a \nu_a$ with $\nu$ as the kinematic viscosity.

Keywords

  • nu: Kinematic viscosity
  • epsilon=0.01: Parameter to prevent singularities
source
TrixiParticles.ViscosityMorrisType
ViscosityMorris(; nu, epsilon=0.01)

Viscosity by Morris (1997) also used by Fourtakas (2019).

To the force $f_{ab}$ between two particles $a$ and $b$ due to pressure gradients, an additional force term $\tilde{f}_{ab}$ is added with

\[\tilde{f}_{ab} = m_a m_b \frac{(\mu_a + \mu_b) r_{ab} \cdot \nabla W_{ab}}{\rho_a \rho_b (\Vert r_{ab} \Vert^2 + \epsilon h^2)} v_{ab},\]

where $\mu_a = \rho_a \nu$ and $\mu_b = \rho_b \nu$ denote the dynamic viscosity of particle $a$ and $b$ respectively, and $\nu$ is the kinematic viscosity.

Keywords

  • nu: Kinematic viscosity
  • epsilon=0.01: Parameter to prevent singularities
source

Density Diffusion

Density diffusion can be used with ContinuityDensity to remove the noise in the pressure field. It is highly recommended to use density diffusion when using WCSPH.

Formulation

All density diffusion terms extend the continuity equation (see ContinuityDensity) by an additional term

\[\frac{\mathrm{d}\rho_a}{\mathrm{d}t} = \sum_{b} m_b v_{ab} \cdot \nabla_{r_a} W(\Vert r_{ab} \Vert, h) +\end{cases}\]

with

\[\mu_{ab} = \frac{h v_{ab} \cdot r_{ab}}{\Vert r_{ab} \Vert^2 + \epsilon h^2},\]

where $\alpha, \beta, \epsilon$ are parameters, $c$ is the speed of sound, $h$ is the smoothing length, $r_{ab} = r_a - r_b$ is the difference of the coordinates of particles $a$ and $b$, $v_{ab} = v_a - v_b$ is the difference of their velocities, and $\bar{\rho}_{ab}$ is the arithmetic mean of their densities.

Note that $\alpha$ needs to adjusted for different resolutions to maintain a specific Reynolds Number. To do so, Monaghan (2005) defined an equivalent effective physical kinematic viscosity $\nu$ by

\[ \nu = \frac{\alpha h c }{2d + 4},\]

where $d$ is the dimension.

Keywords

source
TrixiParticles.ViscosityAdamiType
ViscosityAdami(; nu, epsilon=0.01)

Viscosity by Adami (2012). The viscous interaction is calculated with the shear force for incompressible flows given by

\[f_{ab} = \sum_w \bar{\eta}_{ab} \left( V_a^2 + V_b^2 \right) \frac{v_{ab}}{||r_{ab}||^2+\epsilon h_{ab}^2} \nabla W_{ab} \cdot r_{ab},\]

where $r_{ab} = r_a - r_b$ is the difference of the coordinates of particles $a$ and $b$, $v_{ab} = v_a - v_b$ is the difference of their velocities, $h$ is the smoothing length and $V$ is the particle volume. The parameter $\epsilon$ prevents singularities (see Ramachandran (2019)). The inter-particle-averaged shear stress is

\[ \bar{\eta}_{ab} =\frac{2 \eta_a \eta_b}{\eta_a + \eta_b},\]

where $\eta_a = \rho_a \nu_a$ with $\nu$ as the kinematic viscosity.

Keywords

  • nu: Kinematic viscosity
  • epsilon=0.01: Parameter to prevent singularities
source
TrixiParticles.ViscosityMorrisType
ViscosityMorris(; nu, epsilon=0.01)

Viscosity by Morris (1997) also used by Fourtakas (2019).

To the force $f_{ab}$ between two particles $a$ and $b$ due to pressure gradients, an additional force term $\tilde{f}_{ab}$ is added with

\[\tilde{f}_{ab} = m_a m_b \frac{(\mu_a + \mu_b) r_{ab} \cdot \nabla W_{ab}}{\rho_a \rho_b (\Vert r_{ab} \Vert^2 + \epsilon h^2)} v_{ab},\]

where $\mu_a = \rho_a \nu$ and $\mu_b = \rho_b \nu$ denote the dynamic viscosity of particle $a$ and $b$ respectively, and $\nu$ is the kinematic viscosity.

Keywords

  • nu: Kinematic viscosity
  • epsilon=0.01: Parameter to prevent singularities
source

Density Diffusion

Density diffusion can be used with ContinuityDensity to remove the noise in the pressure field. It is highly recommended to use density diffusion when using WCSPH.

Formulation

All density diffusion terms extend the continuity equation (see ContinuityDensity) by an additional term

\[\frac{\mathrm{d}\rho_a}{\mathrm{d}t} = \sum_{b} m_b v_{ab} \cdot \nabla_{r_a} W(\Vert r_{ab} \Vert, h) + \delta h c \sum_{b} V_b \psi_{ab} \cdot \nabla_{r_a} W(\Vert r_{ab} \Vert, h),\]

where $V_b = m_b / \rho_b$ is the volume of particle $b$ and $\psi_{ab}$ depends on the density diffusion method (see DensityDiffusion for available terms). Also, $\rho_a$ denotes the density of particle $a$ and $r_{ab} = r_a - r_b$ is the difference of the coordinates, $v_{ab} = v_a - v_b$ of the velocities of particles $a$ and $b$.

Numerical Results

All density diffusion terms remove numerical noise in the pressure field and produce more accurate results than weakly commpressible SPH without density diffusion. This can be demonstrated with dam break examples in 2D and 3D. Here, $δ = 0.1$ has been used for all terms. Note that, due to added stability, the adaptive time integration method that was used here can choose higher time steps in the simulations with density diffusion. For the cheap DensityDiffusionMolteniColagrossi, this results in reduced runtime.

density_diffusion_2d
Dam break in 2D with different density diffusion terms
@@ -20,9 +20,9 @@

The simpler terms DensityDiffusionMolteniColagrossi and DensityDiffusionFerrari do not solve the hydrostatic problem and lead to incorrect solutions in long-running steady-state hydrostatic simulations with free surfaces (Antuono et al., 2012). This can be seen when running the simple rectangular tank example until $t = 40$ (again using $δ = 0.1$):

density_diffusion_tank
Tank in rest under gravity in 3D with different density diffusion terms
-

DensityDiffusionAntuono adds a correction term to solve this problem, but this term is very expensive and adds about 40–50% of computational cost.

API

TrixiParticles.DensityDiffusionType
DensityDiffusion

An abstract supertype of all density diffusion formulations.

Currently, the following formulations are available:

FormulationSuitable for Steady-State SimulationsLow Computational Cost
DensityDiffusionMolteniColagrossi
DensityDiffusionFerrari
DensityDiffusionAntuono

See Density Diffusion for a comparison and more details.

source
TrixiParticles.DensityDiffusionAntuonoType
DensityDiffusionAntuono(initial_condition; delta)

The commonly used density diffusion terms by Antuono (2010), also referred to as δ-SPH. The density diffusion term by Molteni (2009) is extended by a second term, which is nicely written down by Antuono (2012).

The term $\psi_{ab}$ in the continuity equation in DensityDiffusion is defined by

\[\psi_{ab} = 2\left(\rho_a - \rho_b - \frac{1}{2}\big(\nabla\rho^L_a + \nabla\rho^L_b\big) \cdot r_{ab}\right) - \frac{r_{ab}}{\Vert r_{ab} \Vert^2},\]

where $\rho_a$ and $\rho_b$ denote the densities of particles $a$ and $b$ respectively and $r_{ab} = r_a - r_b$ is the difference of the coordinates of particles $a$ and $b$. The symbol $\nabla\rho^L_a$ denotes the renormalized density gradient defined as

\[\nabla\rho^L_a = -\sum_b (\rho_a - \rho_b) V_b L_a \nabla_{r_a} W(\Vert r_{ab} \Vert, h)\]

with

\[L_a := \left( -\sum_{b} V_b r_{ab} \otimes \nabla_{r_a} W(\Vert r_{ab} \Vert, h) \right)^{-1} \in \R^{d \times d},\]

where $d$ is the number of dimensions.

See DensityDiffusion for an overview and comparison of implemented density diffusion terms.

source
TrixiParticles.DensityDiffusionFerrariType
DensityDiffusionFerrari()

A density diffusion term by Ferrari (2009).

The term $\psi_{ab}$ in the continuity equation in DensityDiffusion is defined by

\[\psi_{ab} = \frac{\rho_a - \rho_b}{2h} \frac{r_{ab}}{\Vert r_{ab} \Vert},\]

where $\rho_a$ and $\rho_b$ denote the densities of particles $a$ and $b$ respectively, $r_{ab} = r_a - r_b$ is the difference of the coordinates of particles $a$ and $b$ and $h$ is the smoothing length.

See DensityDiffusion for an overview and comparison of implemented density diffusion terms.

source
TrixiParticles.DensityDiffusionMolteniColagrossiType
DensityDiffusionMolteniColagrossi(; delta)

The commonly used density diffusion term by Molteni (2009).

The term $\psi_{ab}$ in the continuity equation in DensityDiffusion is defined by

\[\psi_{ab} = 2(\rho_a - \rho_b) \frac{r_{ab}}{\Vert r_{ab} \Vert^2},\]

where $\rho_a$ and $\rho_b$ denote the densities of particles $a$ and $b$ respectively and $r_{ab} = r_a - r_b$ is the difference of the coordinates of particles $a$ and $b$.

See DensityDiffusion for an overview and comparison of implemented density diffusion terms.

source

Corrections

TrixiParticles.AkinciFreeSurfaceCorrectionType
AkinciFreeSurfaceCorrection(rho0)

Free surface correction according to Akinci et al. (2013). At a free surface, the mean density is typically lower than the reference density, resulting in reduced surface tension and viscosity forces. The free surface correction adjusts the viscosity, pressure, and surface tension forces near free surfaces to counter this effect. It's important to note that this correlation is unphysical and serves as an approximation. The computation time added by this method is about 2–3%.

Mathematically the idea is quite simple. If we have an SPH particle in the middle of a volume at rest, its density will be identical to the rest density $\rho_0$. If we now consider an SPH particle at a free surface at rest, it will have neighbors missing in the direction normal to the surface, which will result in a lower density. If we calculate the correction factor

\[k = \rho_0/\rho_\text{mean},\]

this value will be about ~1.5 for particles at the free surface and can then be used to increase the pressure and viscosity accordingly.

Arguments

  • rho0: Rest density.
source
TrixiParticles.BlendedGradientCorrectionType
BlendedGradientCorrection()

Calculate a blended gradient to reduce the stability issues of the GradientCorrection as explained by Bonet (1999).

This calculates the following,

\[\tilde\nabla A_i = (1-\lambda) \nabla A_i + \lambda L_i \nabla A_i\]

with $0 \leq \lambda \leq 1$ being the blending factor.

Arguments

  • blending_factor: Blending factor between corrected and regular SPH gradient.
source
TrixiParticles.GradientCorrectionType
GradientCorrection()

Compute the corrected gradient of particle interactions based on their relative positions (see Bonet, 1999).

Mathematical Details

Given the standard SPH representation, the gradient of a field $A$ at particle $a$ is given by

\[\nabla A_a = \sum_b m_b \frac{A_b - A_a}{\rho_b} \nabla_{r_a} W(\Vert r_a - r_b \Vert, h),\]

where $m_b$ is the mass of particle $b$ and $\rho_b$ is the density of particle $b$.

The gradient correction, as commonly proposed, involves multiplying this gradient with a correction matrix $L$:

\[\tilde{\nabla} A_a = \bm{L}_a \nabla A_a\]

The correction matrix $\bm{L}_a$ is computed based on the provided particle configuration, aiming to make the corrected gradient more accurate, especially near domain boundaries.

To satisfy

\[\sum_b V_b r_{ba} \otimes \tilde{\nabla}W_b(r_a) = \left( \sum_b V_b r_{ba} \otimes \nabla W_b(r_a) \right) \bm{L}_a^T = \bm{I}\]

the correction matrix $\bm{L}_a$ is evaluated explicitly as

\[\bm{L}_a = \left( \sum_b V_b \nabla W_b(r_{a}) \otimes r_{ba} \right)^{-1}.\]

Note
  • Stability issues arise, especially when particles separate into small clusters.
  • Doubles the computational effort.
  • Better stability with smoother smoothing Kernels with larger support, e.g. SchoenbergQuinticSplineKernel or WendlandC6Kernel.
  • Set dt_max =< 1e-3 for stability.
source
TrixiParticles.KernelCorrectionType
KernelCorrection()

Kernel correction, as explained by Bonet (1999), uses Shepard interpolation to obtain a 0-th order accurate result, which was first proposed by Li et al. This can be further extended to obtain a kernel corrected gradient as shown by Basa et al. (2008).

The kernel correction coefficient is determined by

\[c(x) = \sum_{b=1} V_b W_b(x)\]

The gradient of corrected kernel is determined by

\[\nabla \tilde{W}_{b}(r) =\frac{\nabla W_{b}(r) - W_b(r) \gamma(r)}{\sum_{b=1} V_b W_b(r)} , \quad \text{where} \quad -\gamma(r) = \frac{\sum_{b=1} V_b \nabla W_b(r)}{\sum_{b=1} V_b W_b(r)}.\]

This correction can be applied with SummationDensity and ContinuityDensity, which leads to an improvement, especially at free surfaces.

Note
  • This only works when the boundary model uses SummationDensity (yet).
  • It is also referred to as "0th order correction".
  • In 2D, we can expect an increase of about 10–15% in computation time.
source
TrixiParticles.MixedKernelGradientCorrectionType
MixedKernelGradientCorrection()

Combines GradientCorrection and KernelCorrection, which results in a 1st-order-accurate SPH method (see Bonet, 1999).

Notes:

  • Stability issues, especially when particles separate into small clusters.
  • Doubles the computational effort.
source
TrixiParticles.ShepardKernelCorrectionType
ShepardKernelCorrection()

Kernel correction, as explained by Bonet (1999), uses Shepard interpolation to obtain a 0-th order accurate result, which was first proposed by Li et al. (1996).

The kernel correction coefficient is determined by

\[c(x) = \sum_{b=1} V_b W_b(x),\]

where $V_b = m_b / \rho_b$ is the volume of particle $b$.

This correction is applied with SummationDensity to correct the density and leads to an improvement, especially at free surfaces.

Note
  • It is also referred to as "0th order correction".
  • In 2D, we can expect an increase of about 5–6% in computation time.
source

Surface Tension

Akinci-based intra-particle force surface tension and wall adhesion model

The work by Akinci proposes three forces:

The classical model is composed of the curvature minimization and cohesion force.

Cohesion force

The model calculates the cohesion force based on the distance between particles and the support radius $h_c$. This force is determined using two distinct regimes within the support radius:

The cohesion force, $F_{\text{cohesion}}$, for a pair of particles is given by:

\[F_{\text{cohesion}} = -\sigma m_b C(r) \frac{r}{\Vert r \Vert},\]

where:

The cohesion kernel $C$ is defined as

\[C(r)=\frac{32}{\pi h_c^9} +

DensityDiffusionAntuono adds a correction term to solve this problem, but this term is very expensive and adds about 40–50% of computational cost.

API

TrixiParticles.DensityDiffusionType
DensityDiffusion

An abstract supertype of all density diffusion formulations.

Currently, the following formulations are available:

FormulationSuitable for Steady-State SimulationsLow Computational Cost
DensityDiffusionMolteniColagrossi
DensityDiffusionFerrari
DensityDiffusionAntuono

See Density Diffusion for a comparison and more details.

source
TrixiParticles.DensityDiffusionAntuonoType
DensityDiffusionAntuono(initial_condition; delta)

The commonly used density diffusion terms by Antuono (2010), also referred to as δ-SPH. The density diffusion term by Molteni (2009) is extended by a second term, which is nicely written down by Antuono (2012).

The term $\psi_{ab}$ in the continuity equation in DensityDiffusion is defined by

\[\psi_{ab} = 2\left(\rho_a - \rho_b - \frac{1}{2}\big(\nabla\rho^L_a + \nabla\rho^L_b\big) \cdot r_{ab}\right) + \frac{r_{ab}}{\Vert r_{ab} \Vert^2},\]

where $\rho_a$ and $\rho_b$ denote the densities of particles $a$ and $b$ respectively and $r_{ab} = r_a - r_b$ is the difference of the coordinates of particles $a$ and $b$. The symbol $\nabla\rho^L_a$ denotes the renormalized density gradient defined as

\[\nabla\rho^L_a = -\sum_b (\rho_a - \rho_b) V_b L_a \nabla_{r_a} W(\Vert r_{ab} \Vert, h)\]

with

\[L_a := \left( -\sum_{b} V_b r_{ab} \otimes \nabla_{r_a} W(\Vert r_{ab} \Vert, h) \right)^{-1} \in \R^{d \times d},\]

where $d$ is the number of dimensions.

See DensityDiffusion for an overview and comparison of implemented density diffusion terms.

source
TrixiParticles.DensityDiffusionFerrariType
DensityDiffusionFerrari()

A density diffusion term by Ferrari (2009).

The term $\psi_{ab}$ in the continuity equation in DensityDiffusion is defined by

\[\psi_{ab} = \frac{\rho_a - \rho_b}{2h} \frac{r_{ab}}{\Vert r_{ab} \Vert},\]

where $\rho_a$ and $\rho_b$ denote the densities of particles $a$ and $b$ respectively, $r_{ab} = r_a - r_b$ is the difference of the coordinates of particles $a$ and $b$ and $h$ is the smoothing length.

See DensityDiffusion for an overview and comparison of implemented density diffusion terms.

source
TrixiParticles.DensityDiffusionMolteniColagrossiType
DensityDiffusionMolteniColagrossi(; delta)

The commonly used density diffusion term by Molteni (2009).

The term $\psi_{ab}$ in the continuity equation in DensityDiffusion is defined by

\[\psi_{ab} = 2(\rho_a - \rho_b) \frac{r_{ab}}{\Vert r_{ab} \Vert^2},\]

where $\rho_a$ and $\rho_b$ denote the densities of particles $a$ and $b$ respectively and $r_{ab} = r_a - r_b$ is the difference of the coordinates of particles $a$ and $b$.

See DensityDiffusion for an overview and comparison of implemented density diffusion terms.

source

Corrections

TrixiParticles.AkinciFreeSurfaceCorrectionType
AkinciFreeSurfaceCorrection(rho0)

Free surface correction according to Akinci et al. (2013). At a free surface, the mean density is typically lower than the reference density, resulting in reduced surface tension and viscosity forces. The free surface correction adjusts the viscosity, pressure, and surface tension forces near free surfaces to counter this effect. It's important to note that this correlation is unphysical and serves as an approximation. The computation time added by this method is about 2–3%.

Mathematically the idea is quite simple. If we have an SPH particle in the middle of a volume at rest, its density will be identical to the rest density $\rho_0$. If we now consider an SPH particle at a free surface at rest, it will have neighbors missing in the direction normal to the surface, which will result in a lower density. If we calculate the correction factor

\[k = \rho_0/\rho_\text{mean},\]

this value will be about ~1.5 for particles at the free surface and can then be used to increase the pressure and viscosity accordingly.

Arguments

  • rho0: Rest density.
source
TrixiParticles.BlendedGradientCorrectionType
BlendedGradientCorrection()

Calculate a blended gradient to reduce the stability issues of the GradientCorrection as explained by Bonet (1999).

This calculates the following,

\[\tilde\nabla A_i = (1-\lambda) \nabla A_i + \lambda L_i \nabla A_i\]

with $0 \leq \lambda \leq 1$ being the blending factor.

Arguments

  • blending_factor: Blending factor between corrected and regular SPH gradient.
source
TrixiParticles.GradientCorrectionType
GradientCorrection()

Compute the corrected gradient of particle interactions based on their relative positions (see Bonet, 1999).

Mathematical Details

Given the standard SPH representation, the gradient of a field $A$ at particle $a$ is given by

\[\nabla A_a = \sum_b m_b \frac{A_b - A_a}{\rho_b} \nabla_{r_a} W(\Vert r_a - r_b \Vert, h),\]

where $m_b$ is the mass of particle $b$ and $\rho_b$ is the density of particle $b$.

The gradient correction, as commonly proposed, involves multiplying this gradient with a correction matrix $L$:

\[\tilde{\nabla} A_a = \bm{L}_a \nabla A_a\]

The correction matrix $\bm{L}_a$ is computed based on the provided particle configuration, aiming to make the corrected gradient more accurate, especially near domain boundaries.

To satisfy

\[\sum_b V_b r_{ba} \otimes \tilde{\nabla}W_b(r_a) = \left( \sum_b V_b r_{ba} \otimes \nabla W_b(r_a) \right) \bm{L}_a^T = \bm{I}\]

the correction matrix $\bm{L}_a$ is evaluated explicitly as

\[\bm{L}_a = \left( \sum_b V_b \nabla W_b(r_{a}) \otimes r_{ba} \right)^{-1}.\]

Note
  • Stability issues arise, especially when particles separate into small clusters.
  • Doubles the computational effort.
  • Better stability with smoother smoothing Kernels with larger support, e.g. SchoenbergQuinticSplineKernel or WendlandC6Kernel.
  • Set dt_max =< 1e-3 for stability.
source
TrixiParticles.KernelCorrectionType
KernelCorrection()

Kernel correction, as explained by Bonet (1999), uses Shepard interpolation to obtain a 0-th order accurate result, which was first proposed by Li et al. This can be further extended to obtain a kernel corrected gradient as shown by Basa et al. (2008).

The kernel correction coefficient is determined by

\[c(x) = \sum_{b=1} V_b W_b(x)\]

The gradient of corrected kernel is determined by

\[\nabla \tilde{W}_{b}(r) =\frac{\nabla W_{b}(r) - W_b(r) \gamma(r)}{\sum_{b=1} V_b W_b(r)} , \quad \text{where} \quad +\gamma(r) = \frac{\sum_{b=1} V_b \nabla W_b(r)}{\sum_{b=1} V_b W_b(r)}.\]

This correction can be applied with SummationDensity and ContinuityDensity, which leads to an improvement, especially at free surfaces.

Note
  • This only works when the boundary model uses SummationDensity (yet).
  • It is also referred to as "0th order correction".
  • In 2D, we can expect an increase of about 10–15% in computation time.
source
TrixiParticles.MixedKernelGradientCorrectionType
MixedKernelGradientCorrection()

Combines GradientCorrection and KernelCorrection, which results in a 1st-order-accurate SPH method (see Bonet, 1999).

Notes:

  • Stability issues, especially when particles separate into small clusters.
  • Doubles the computational effort.
source
TrixiParticles.ShepardKernelCorrectionType
ShepardKernelCorrection()

Kernel correction, as explained by Bonet (1999), uses Shepard interpolation to obtain a 0-th order accurate result, which was first proposed by Li et al. (1996).

The kernel correction coefficient is determined by

\[c(x) = \sum_{b=1} V_b W_b(x),\]

where $V_b = m_b / \rho_b$ is the volume of particle $b$.

This correction is applied with SummationDensity to correct the density and leads to an improvement, especially at free surfaces.

Note
  • It is also referred to as "0th order correction".
  • In 2D, we can expect an increase of about 5–6% in computation time.
source

Surface Tension

Akinci-based intra-particle force surface tension and wall adhesion model

The work by Akinci proposes three forces:

The classical model is composed of the curvature minimization and cohesion force.

Cohesion force

The model calculates the cohesion force based on the distance between particles and the support radius $h_c$. This force is determined using two distinct regimes within the support radius:

The cohesion force, $F_{\text{cohesion}}$, for a pair of particles is given by:

\[F_{\text{cohesion}} = -\sigma m_b C(r) \frac{r}{\Vert r \Vert},\]

where:

The cohesion kernel $C$ is defined as

\[C(r)=\frac{32}{\pi h_c^9} \begin{cases} (h_c-r)^3 r^3, & \text{if } 2r > h_c \\ 2(h_c-r)^3 r^3 - \frac{h^6}{64}, & \text{if } r > 0 \text{ and } 2r \leq h_c \\ @@ -31,4 +31,4 @@ \begin{cases} \sqrt[4]{- \frac{4r^2}{h_c} + 6r - 2h_c}, & \text{if } 2r > h_c \text{ and } r \leq h_c \\ 0, & \text{otherwise.} -\end{cases}\]

TrixiParticles.CohesionForceAkinciType
CohesionForceAkinci(surface_tension_coefficient=1.0)

This model only implements the cohesion force of the [25] surface tension model.

Keywords

  • surface_tension_coefficient=1.0: Modifies the intensity of the surface tension-induced force, enabling the tuning of the fluid's surface tension properties within the simulation.
source
TrixiParticles.SurfaceTensionAkinciType
SurfaceTensionAkinci(surface_tension_coefficient=1.0)

Implements a model for surface tension and adhesion effects drawing upon the principles outlined by [25]. This model is instrumental in capturing the nuanced behaviors of fluid surfaces, such as droplet formation and the dynamics of merging or separation, by utilizing intra-particle forces.

Keywords

  • surface_tension_coefficient=1.0: A parameter to adjust the magnitude of surface tension forces, facilitating the fine-tuning of how surface tension phenomena are represented in the simulation.
source
+\end{cases}\]

TrixiParticles.CohesionForceAkinciType
CohesionForceAkinci(surface_tension_coefficient=1.0)

This model only implements the cohesion force of the [25] surface tension model.

Keywords

  • surface_tension_coefficient=1.0: Modifies the intensity of the surface tension-induced force, enabling the tuning of the fluid's surface tension properties within the simulation.
source
TrixiParticles.SurfaceTensionAkinciType
SurfaceTensionAkinci(surface_tension_coefficient=1.0)

Implements a model for surface tension and adhesion effects drawing upon the principles outlined by [25]. This model is instrumental in capturing the nuanced behaviors of fluid surfaces, such as droplet formation and the dynamics of merging or separation, by utilizing intra-particle forces.

Keywords

  • surface_tension_coefficient=1.0: A parameter to adjust the magnitude of surface tension forces, facilitating the fine-tuning of how surface tension phenomena are represented in the simulation.
source
diff --git a/previews/PR514/time_integration/index.html b/previews/PR514/time_integration/index.html index eb4e5e30c..7b36a74c5 100644 --- a/previews/PR514/time_integration/index.html +++ b/previews/PR514/time_integration/index.html @@ -1,2 +1,2 @@ -Time Integration · TrixiParticles.jl
+Time Integration · TrixiParticles.jl
diff --git a/previews/PR514/tutorial/index.html b/previews/PR514/tutorial/index.html index 603b61ec8..330cce425 100644 --- a/previews/PR514/tutorial/index.html +++ b/previews/PR514/tutorial/index.html @@ -1,2 +1,2 @@ -Tutorial · TrixiParticles.jl
+Tutorial · TrixiParticles.jl
diff --git a/previews/PR514/tutorials/out/boundary_1_0.vtu b/previews/PR514/tutorials/out/boundary_1_0.vtu index 1ddde76be..5a8dc2524 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_0.vtu and b/previews/PR514/tutorials/out/boundary_1_0.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_1.vtu b/previews/PR514/tutorials/out/boundary_1_1.vtu index 48250dc62..fb357e618 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_1.vtu and b/previews/PR514/tutorials/out/boundary_1_1.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_10.vtu b/previews/PR514/tutorials/out/boundary_1_10.vtu index 99c9e2fa9..a662a3e31 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_10.vtu and b/previews/PR514/tutorials/out/boundary_1_10.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_11.vtu b/previews/PR514/tutorials/out/boundary_1_11.vtu index 9202c0890..4fa09c873 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_11.vtu and b/previews/PR514/tutorials/out/boundary_1_11.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_12.vtu b/previews/PR514/tutorials/out/boundary_1_12.vtu index 8dfeaf324..197ff4ef6 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_12.vtu and b/previews/PR514/tutorials/out/boundary_1_12.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_13.vtu b/previews/PR514/tutorials/out/boundary_1_13.vtu index 813990aff..dff2eb4fb 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_13.vtu and b/previews/PR514/tutorials/out/boundary_1_13.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_14.vtu b/previews/PR514/tutorials/out/boundary_1_14.vtu index 320791526..ea325df3c 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_14.vtu and b/previews/PR514/tutorials/out/boundary_1_14.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_15.vtu b/previews/PR514/tutorials/out/boundary_1_15.vtu index 51d20761a..e9ccf05ed 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_15.vtu and b/previews/PR514/tutorials/out/boundary_1_15.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_16.vtu b/previews/PR514/tutorials/out/boundary_1_16.vtu index 7cfe266bd..87d22887b 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_16.vtu and b/previews/PR514/tutorials/out/boundary_1_16.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_17.vtu b/previews/PR514/tutorials/out/boundary_1_17.vtu index 7b84e680c..588e7bfa1 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_17.vtu and b/previews/PR514/tutorials/out/boundary_1_17.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_18.vtu b/previews/PR514/tutorials/out/boundary_1_18.vtu index b715321f1..5b4b39148 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_18.vtu and b/previews/PR514/tutorials/out/boundary_1_18.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_19.vtu b/previews/PR514/tutorials/out/boundary_1_19.vtu index 1fd494e1c..df322517e 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_19.vtu and b/previews/PR514/tutorials/out/boundary_1_19.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_2.vtu b/previews/PR514/tutorials/out/boundary_1_2.vtu index cb7b5c88b..87bedcbe5 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_2.vtu and b/previews/PR514/tutorials/out/boundary_1_2.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_20.vtu b/previews/PR514/tutorials/out/boundary_1_20.vtu index 2f2e2e514..0bfeb0d21 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_20.vtu and b/previews/PR514/tutorials/out/boundary_1_20.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_21.vtu b/previews/PR514/tutorials/out/boundary_1_21.vtu index 54a1d4ccb..8fb8b2886 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_21.vtu and b/previews/PR514/tutorials/out/boundary_1_21.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_22.vtu b/previews/PR514/tutorials/out/boundary_1_22.vtu index 2d19355dd..9b779aa33 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_22.vtu and b/previews/PR514/tutorials/out/boundary_1_22.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_23.vtu b/previews/PR514/tutorials/out/boundary_1_23.vtu index 745c9f360..e421e3686 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_23.vtu and b/previews/PR514/tutorials/out/boundary_1_23.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_24.vtu b/previews/PR514/tutorials/out/boundary_1_24.vtu index f329f7290..e6147e8d6 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_24.vtu and b/previews/PR514/tutorials/out/boundary_1_24.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_25.vtu b/previews/PR514/tutorials/out/boundary_1_25.vtu index e40cb92f6..325e85da8 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_25.vtu and b/previews/PR514/tutorials/out/boundary_1_25.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_26.vtu b/previews/PR514/tutorials/out/boundary_1_26.vtu index 19d56a6bb..1c71de072 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_26.vtu and b/previews/PR514/tutorials/out/boundary_1_26.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_27.vtu b/previews/PR514/tutorials/out/boundary_1_27.vtu index 03c9c7450..e99b0537e 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_27.vtu and b/previews/PR514/tutorials/out/boundary_1_27.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_28.vtu b/previews/PR514/tutorials/out/boundary_1_28.vtu index 032d6c50c..edcb3960d 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_28.vtu and b/previews/PR514/tutorials/out/boundary_1_28.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_29.vtu b/previews/PR514/tutorials/out/boundary_1_29.vtu index a22930d46..e150c21d2 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_29.vtu and b/previews/PR514/tutorials/out/boundary_1_29.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_3.vtu b/previews/PR514/tutorials/out/boundary_1_3.vtu index f3498dd20..e3d2ff30e 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_3.vtu and b/previews/PR514/tutorials/out/boundary_1_3.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_30.vtu b/previews/PR514/tutorials/out/boundary_1_30.vtu index d963d3c92..2607732b3 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_30.vtu and b/previews/PR514/tutorials/out/boundary_1_30.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_31.vtu b/previews/PR514/tutorials/out/boundary_1_31.vtu index 2878bcf52..02a2322d4 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_31.vtu and b/previews/PR514/tutorials/out/boundary_1_31.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_32.vtu b/previews/PR514/tutorials/out/boundary_1_32.vtu index 5e176ce09..aed049666 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_32.vtu and b/previews/PR514/tutorials/out/boundary_1_32.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_33.vtu b/previews/PR514/tutorials/out/boundary_1_33.vtu index 10cc48479..1a7e7f2b7 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_33.vtu and b/previews/PR514/tutorials/out/boundary_1_33.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_34.vtu b/previews/PR514/tutorials/out/boundary_1_34.vtu index bfee37707..478556350 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_34.vtu and b/previews/PR514/tutorials/out/boundary_1_34.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_35.vtu b/previews/PR514/tutorials/out/boundary_1_35.vtu index 02e54dec9..a9a87a797 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_35.vtu and b/previews/PR514/tutorials/out/boundary_1_35.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_36.vtu b/previews/PR514/tutorials/out/boundary_1_36.vtu index c1957b3dc..6390cb4b7 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_36.vtu and b/previews/PR514/tutorials/out/boundary_1_36.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_37.vtu b/previews/PR514/tutorials/out/boundary_1_37.vtu index 34f552693..544037ad9 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_37.vtu and b/previews/PR514/tutorials/out/boundary_1_37.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_38.vtu b/previews/PR514/tutorials/out/boundary_1_38.vtu index f7d03f859..e4529321a 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_38.vtu and b/previews/PR514/tutorials/out/boundary_1_38.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_39.vtu b/previews/PR514/tutorials/out/boundary_1_39.vtu index 494e2314e..264d79fb9 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_39.vtu and b/previews/PR514/tutorials/out/boundary_1_39.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_4.vtu b/previews/PR514/tutorials/out/boundary_1_4.vtu index 64367083d..4d1a7baac 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_4.vtu and b/previews/PR514/tutorials/out/boundary_1_4.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_40.vtu b/previews/PR514/tutorials/out/boundary_1_40.vtu index 668e97ed1..362afe77f 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_40.vtu and b/previews/PR514/tutorials/out/boundary_1_40.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_41.vtu b/previews/PR514/tutorials/out/boundary_1_41.vtu index b5f78dbc6..6a5740144 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_41.vtu and b/previews/PR514/tutorials/out/boundary_1_41.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_42.vtu b/previews/PR514/tutorials/out/boundary_1_42.vtu index bb8cb4dfe..00fa81ab1 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_42.vtu and b/previews/PR514/tutorials/out/boundary_1_42.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_43.vtu b/previews/PR514/tutorials/out/boundary_1_43.vtu index d6b086850..fa3a0e474 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_43.vtu and b/previews/PR514/tutorials/out/boundary_1_43.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_44.vtu b/previews/PR514/tutorials/out/boundary_1_44.vtu index c595993af..7e8e611d7 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_44.vtu and b/previews/PR514/tutorials/out/boundary_1_44.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_45.vtu b/previews/PR514/tutorials/out/boundary_1_45.vtu index 2816e48c0..ec47dffe1 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_45.vtu and b/previews/PR514/tutorials/out/boundary_1_45.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_46.vtu b/previews/PR514/tutorials/out/boundary_1_46.vtu index b030022f1..ae0217772 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_46.vtu and b/previews/PR514/tutorials/out/boundary_1_46.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_47.vtu b/previews/PR514/tutorials/out/boundary_1_47.vtu index 3b5b50e5b..37ea860b4 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_47.vtu and b/previews/PR514/tutorials/out/boundary_1_47.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_48.vtu b/previews/PR514/tutorials/out/boundary_1_48.vtu index 30a5038b5..b9caedef7 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_48.vtu and b/previews/PR514/tutorials/out/boundary_1_48.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_49.vtu b/previews/PR514/tutorials/out/boundary_1_49.vtu index ddf1b7362..1126fc34d 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_49.vtu and b/previews/PR514/tutorials/out/boundary_1_49.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_5.vtu b/previews/PR514/tutorials/out/boundary_1_5.vtu index 5068ae0a7..04b6b610a 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_5.vtu and b/previews/PR514/tutorials/out/boundary_1_5.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_50.vtu b/previews/PR514/tutorials/out/boundary_1_50.vtu index 9b8936fb5..7c4f02315 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_50.vtu and b/previews/PR514/tutorials/out/boundary_1_50.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_6.vtu b/previews/PR514/tutorials/out/boundary_1_6.vtu index 9f3f2dbab..f2e4836e2 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_6.vtu and b/previews/PR514/tutorials/out/boundary_1_6.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_7.vtu b/previews/PR514/tutorials/out/boundary_1_7.vtu index 02987fc25..26a568c22 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_7.vtu and b/previews/PR514/tutorials/out/boundary_1_7.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_8.vtu b/previews/PR514/tutorials/out/boundary_1_8.vtu index 6c1594101..2a0c861f9 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_8.vtu and b/previews/PR514/tutorials/out/boundary_1_8.vtu differ diff --git a/previews/PR514/tutorials/out/boundary_1_9.vtu b/previews/PR514/tutorials/out/boundary_1_9.vtu index f9bbb4864..a2b6b1685 100644 Binary files a/previews/PR514/tutorials/out/boundary_1_9.vtu and b/previews/PR514/tutorials/out/boundary_1_9.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_0.vtu b/previews/PR514/tutorials/out/fluid_1_0.vtu index 352c9b5fa..39d2ccf26 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_0.vtu and b/previews/PR514/tutorials/out/fluid_1_0.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_1.vtu b/previews/PR514/tutorials/out/fluid_1_1.vtu index 5be5d6b6e..51064283a 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_1.vtu and b/previews/PR514/tutorials/out/fluid_1_1.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_10.vtu b/previews/PR514/tutorials/out/fluid_1_10.vtu index e0466e4c9..7989af8f8 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_10.vtu and b/previews/PR514/tutorials/out/fluid_1_10.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_11.vtu b/previews/PR514/tutorials/out/fluid_1_11.vtu index 3cc68de89..4c7c77daf 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_11.vtu and b/previews/PR514/tutorials/out/fluid_1_11.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_12.vtu b/previews/PR514/tutorials/out/fluid_1_12.vtu index 5e71b5282..6758e67fc 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_12.vtu and b/previews/PR514/tutorials/out/fluid_1_12.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_13.vtu b/previews/PR514/tutorials/out/fluid_1_13.vtu index 41f88e425..cb59f65b9 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_13.vtu and b/previews/PR514/tutorials/out/fluid_1_13.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_14.vtu b/previews/PR514/tutorials/out/fluid_1_14.vtu index 43b01e21e..3441df0df 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_14.vtu and b/previews/PR514/tutorials/out/fluid_1_14.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_15.vtu b/previews/PR514/tutorials/out/fluid_1_15.vtu index ce1ae8bf7..555b377c0 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_15.vtu and b/previews/PR514/tutorials/out/fluid_1_15.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_16.vtu b/previews/PR514/tutorials/out/fluid_1_16.vtu index 9373ed0ca..d3f81c80d 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_16.vtu and b/previews/PR514/tutorials/out/fluid_1_16.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_17.vtu b/previews/PR514/tutorials/out/fluid_1_17.vtu index 1f461bcce..d8f33a80f 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_17.vtu and b/previews/PR514/tutorials/out/fluid_1_17.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_18.vtu b/previews/PR514/tutorials/out/fluid_1_18.vtu index 600542b58..d2f0c7fe0 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_18.vtu and b/previews/PR514/tutorials/out/fluid_1_18.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_19.vtu b/previews/PR514/tutorials/out/fluid_1_19.vtu index 5f8d6656e..d0c2a9706 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_19.vtu and b/previews/PR514/tutorials/out/fluid_1_19.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_2.vtu b/previews/PR514/tutorials/out/fluid_1_2.vtu index 6cf55b7f2..5c9f70d2d 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_2.vtu and b/previews/PR514/tutorials/out/fluid_1_2.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_20.vtu b/previews/PR514/tutorials/out/fluid_1_20.vtu index 4b802181a..80e49c902 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_20.vtu and b/previews/PR514/tutorials/out/fluid_1_20.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_21.vtu b/previews/PR514/tutorials/out/fluid_1_21.vtu index 1870c80be..a98c5dec3 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_21.vtu and b/previews/PR514/tutorials/out/fluid_1_21.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_22.vtu b/previews/PR514/tutorials/out/fluid_1_22.vtu index 8c90807f3..88e06edb8 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_22.vtu and b/previews/PR514/tutorials/out/fluid_1_22.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_23.vtu b/previews/PR514/tutorials/out/fluid_1_23.vtu index 26673d83c..3ae629adf 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_23.vtu and b/previews/PR514/tutorials/out/fluid_1_23.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_24.vtu b/previews/PR514/tutorials/out/fluid_1_24.vtu index 1c06134aa..a149829b4 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_24.vtu and b/previews/PR514/tutorials/out/fluid_1_24.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_25.vtu b/previews/PR514/tutorials/out/fluid_1_25.vtu index 405b1301f..da2e2006f 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_25.vtu and b/previews/PR514/tutorials/out/fluid_1_25.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_26.vtu b/previews/PR514/tutorials/out/fluid_1_26.vtu index 51e8e40de..d3faf24a7 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_26.vtu and b/previews/PR514/tutorials/out/fluid_1_26.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_27.vtu b/previews/PR514/tutorials/out/fluid_1_27.vtu index d76266705..5d7ac4050 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_27.vtu and b/previews/PR514/tutorials/out/fluid_1_27.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_28.vtu b/previews/PR514/tutorials/out/fluid_1_28.vtu index f584ab67f..61ec50504 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_28.vtu and b/previews/PR514/tutorials/out/fluid_1_28.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_29.vtu b/previews/PR514/tutorials/out/fluid_1_29.vtu index 112eff0a0..9cbdbf2c8 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_29.vtu and b/previews/PR514/tutorials/out/fluid_1_29.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_3.vtu b/previews/PR514/tutorials/out/fluid_1_3.vtu index fd9b3abec..c63f565bb 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_3.vtu and b/previews/PR514/tutorials/out/fluid_1_3.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_30.vtu b/previews/PR514/tutorials/out/fluid_1_30.vtu index 8625be987..43ef34c0a 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_30.vtu and b/previews/PR514/tutorials/out/fluid_1_30.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_31.vtu b/previews/PR514/tutorials/out/fluid_1_31.vtu index 311847e6d..524f97ca3 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_31.vtu and b/previews/PR514/tutorials/out/fluid_1_31.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_32.vtu b/previews/PR514/tutorials/out/fluid_1_32.vtu index 4df9e5e76..ec13d42a4 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_32.vtu and b/previews/PR514/tutorials/out/fluid_1_32.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_33.vtu b/previews/PR514/tutorials/out/fluid_1_33.vtu index 5baee3122..89a1ef58a 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_33.vtu and b/previews/PR514/tutorials/out/fluid_1_33.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_34.vtu b/previews/PR514/tutorials/out/fluid_1_34.vtu index 0f79df2d9..30cdd731e 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_34.vtu and b/previews/PR514/tutorials/out/fluid_1_34.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_35.vtu b/previews/PR514/tutorials/out/fluid_1_35.vtu index 4637cfa75..95adde387 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_35.vtu and b/previews/PR514/tutorials/out/fluid_1_35.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_36.vtu b/previews/PR514/tutorials/out/fluid_1_36.vtu index 00e2c5fe1..69d814de8 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_36.vtu and b/previews/PR514/tutorials/out/fluid_1_36.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_37.vtu b/previews/PR514/tutorials/out/fluid_1_37.vtu index 9c83b3399..8b4da2659 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_37.vtu and b/previews/PR514/tutorials/out/fluid_1_37.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_38.vtu b/previews/PR514/tutorials/out/fluid_1_38.vtu index 9f681898e..57763b28e 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_38.vtu and b/previews/PR514/tutorials/out/fluid_1_38.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_39.vtu b/previews/PR514/tutorials/out/fluid_1_39.vtu index ca604e7d6..9db539507 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_39.vtu and b/previews/PR514/tutorials/out/fluid_1_39.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_4.vtu b/previews/PR514/tutorials/out/fluid_1_4.vtu index fc3871143..a18720571 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_4.vtu and b/previews/PR514/tutorials/out/fluid_1_4.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_40.vtu b/previews/PR514/tutorials/out/fluid_1_40.vtu index 56d3fa228..8e09fd286 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_40.vtu and b/previews/PR514/tutorials/out/fluid_1_40.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_41.vtu b/previews/PR514/tutorials/out/fluid_1_41.vtu index 2fdff4be9..b2d0abbc7 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_41.vtu and b/previews/PR514/tutorials/out/fluid_1_41.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_42.vtu b/previews/PR514/tutorials/out/fluid_1_42.vtu index da424ca8d..32632d2f4 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_42.vtu and b/previews/PR514/tutorials/out/fluid_1_42.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_43.vtu b/previews/PR514/tutorials/out/fluid_1_43.vtu index 2d424e1dc..5ffbaf59b 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_43.vtu and b/previews/PR514/tutorials/out/fluid_1_43.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_44.vtu b/previews/PR514/tutorials/out/fluid_1_44.vtu index eeb5c4182..3994ac26c 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_44.vtu and b/previews/PR514/tutorials/out/fluid_1_44.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_45.vtu b/previews/PR514/tutorials/out/fluid_1_45.vtu index 4b5a3440b..c92a24098 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_45.vtu and b/previews/PR514/tutorials/out/fluid_1_45.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_46.vtu b/previews/PR514/tutorials/out/fluid_1_46.vtu index c80864d19..6100daa8e 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_46.vtu and b/previews/PR514/tutorials/out/fluid_1_46.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_47.vtu b/previews/PR514/tutorials/out/fluid_1_47.vtu index 99c96bcef..dfda15f6e 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_47.vtu and b/previews/PR514/tutorials/out/fluid_1_47.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_48.vtu b/previews/PR514/tutorials/out/fluid_1_48.vtu index a586cc3c6..7fcb3a29c 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_48.vtu and b/previews/PR514/tutorials/out/fluid_1_48.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_49.vtu b/previews/PR514/tutorials/out/fluid_1_49.vtu index 86ee8d5e1..42acb90ea 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_49.vtu and b/previews/PR514/tutorials/out/fluid_1_49.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_5.vtu b/previews/PR514/tutorials/out/fluid_1_5.vtu index 1d3fad584..0e31aa868 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_5.vtu and b/previews/PR514/tutorials/out/fluid_1_5.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_50.vtu b/previews/PR514/tutorials/out/fluid_1_50.vtu index f97f7d3bd..330047a5b 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_50.vtu and b/previews/PR514/tutorials/out/fluid_1_50.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_6.vtu b/previews/PR514/tutorials/out/fluid_1_6.vtu index c47305385..d7f0b8ca0 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_6.vtu and b/previews/PR514/tutorials/out/fluid_1_6.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_7.vtu b/previews/PR514/tutorials/out/fluid_1_7.vtu index 64f015ca0..90862d00e 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_7.vtu and b/previews/PR514/tutorials/out/fluid_1_7.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_8.vtu b/previews/PR514/tutorials/out/fluid_1_8.vtu index f3c013b50..f88a83254 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_8.vtu and b/previews/PR514/tutorials/out/fluid_1_8.vtu differ diff --git a/previews/PR514/tutorials/out/fluid_1_9.vtu b/previews/PR514/tutorials/out/fluid_1_9.vtu index f676cf7ed..9ca82a4f3 100644 Binary files a/previews/PR514/tutorials/out/fluid_1_9.vtu and b/previews/PR514/tutorials/out/fluid_1_9.vtu differ diff --git a/previews/PR514/tutorials/tut_beam/index.html b/previews/PR514/tutorials/tut_beam/index.html index 408fff344..05bf13e33 100644 --- a/previews/PR514/tutorials/tut_beam/index.html +++ b/previews/PR514/tutorials/tut_beam/index.html @@ -84,4 +84,4 @@ # Use a Runge-Kutta method with automatic (error based) time step size control sol = solve(ode, RDPK3SpFSAL49(), save_everystep=false, callback=callbacks); - + diff --git a/previews/PR514/tutorials/tut_dam_break/index.html b/previews/PR514/tutorials/tut_dam_break/index.html index 153131db8..fb312b339 100644 --- a/previews/PR514/tutorials/tut_dam_break/index.html +++ b/previews/PR514/tutorials/tut_dam_break/index.html @@ -111,4 +111,4 @@ dt=1.0, # This is overwritten by the stepsize callback save_everystep=false, callback=callbacks); - + diff --git a/previews/PR514/tutorials/tut_falling/index.html b/previews/PR514/tutorials/tut_falling/index.html index 114c16746..2650c40f9 100644 --- a/previews/PR514/tutorials/tut_falling/index.html +++ b/previews/PR514/tutorials/tut_falling/index.html @@ -129,4 +129,4 @@ reltol=1e-3, # Default reltol is 1e-3 save_everystep=false, callback=callbacks); - + diff --git a/previews/PR514/tutorials/tut_setup/index.html b/previews/PR514/tutorials/tut_setup/index.html index a161361ce..449dfd701 100644 --- a/previews/PR514/tutorials/tut_setup/index.html +++ b/previews/PR514/tutorials/tut_setup/index.html @@ -1,7 +1,7 @@ Setting up your simulation from scratch · TrixiParticles.jl

Setting up your simulation from scratch

In this tutorial, we will guide you through the general structure of simulation files. We will set up a simulation similar to the example simulation examples/fluid/hydrostatic_water_column_2d.jl, which is one of our simplest example simulations. In the second part of this tutorial, we will show how to replace components of TrixiParticles.jl by custom implementations from within a simulation file, without ever cloning the repository.

For different setups and physics, have a look at our other example files.

First, we import TrixiParticles.jl and OrdinaryDiffEq.jl, which we will use at the very end for the time integration.

using TrixiParticles
-using OrdinaryDiffEq

Resolution

Now, we define the particle spacing, which is our numerical resolution. We usually call the variable fluid_particle_spacing, so that we can easily change the resolution of an example file by overwriting this variable with trixi_include. In 2D, the number of particles will grow quadratically, in 3D cubically with the spacing.

We also set the number of boundary layers, which need to be sufficiently large, depending on the smoothing kernel and smoothing length, so that the compact support of the smoothing kernel is fully sampled with particles for a fluid particle close to a boundary. In particular, we require boundary_layers >= compact_support. The value for the compact support for each kernel can be found in the smoothing kernel overview.

fluid_particle_spacing = 0.05
-boundary_layers = 3

Experiment setup

We want to simulate a water column resting under hydrostatic pressure inside a rectangular tank. Experiment Setup First, we define the physical parameters gravitational acceleration, simulation time, initial fluid size, tank size and fluid density.

gravity = 9.81
+using OrdinaryDiffEq

Resolution

Now, we define the particle spacing, which is our numerical resolution. For a fluid, we usually call the variable fluid_particle_spacing, so that we can easily change the resolution of an example file by overwriting this variable with trixi_include. In 2D, the number of particles will grow quadratically, in 3D cubically with the spacing.

We also set the number of boundary layers, which need to be sufficiently large, depending on the smoothing kernel and smoothing length, so that the compact support of the smoothing kernel is fully sampled with particles for a fluid particle close to a boundary. In particular, we require the boundary thickness boundary_layers * fluid_particle_spacing to be larger than the compact support of the kernel. The compact support of each kernel can be found in the smoothing kernel overview.

fluid_particle_spacing = 0.05
+boundary_layers = 3

Experiment setup

We want to simulate a water column resting under hydrostatic pressure inside a rectangular tank. Experiment Setup First, we define physical parameters like gravitational acceleration, simulation time, initial fluid size, tank size and fluid density.

gravity = 9.81
 tspan = (0.0, 1.0)
 initial_fluid_size = (1.0, 0.9)
 tank_size = (1.0, 1.0)
@@ -10,7 +10,7 @@
                                    exponent=7)

The speed of sound here is numerical and not physical. We artificially lower the speed of sound, since the physical speed of sound in water would lead to prohibitively small time steps. The speed of sound in Weakly Compressible SPH should be chosen as small as possible for numerical efficiency, but large enough to limit density fluctuations to about 1%.

TrixiParticles.jl requires the initial particle positions and quantities in form of an InitialCondition. Instead of manually defining particle positions, you can work with our pre-defined setups. Among others, we provide setups for rectangular shapes, circles, and spheres. Initial conditions can also be combined with common set operations. See this page for a list of pre-defined setups and details on set operations on initial conditions.

Here, we use the RectangularTank setup, which generates a rectangular fluid inside a rectangular tank, and supports a hydrostatic pressure gradient by passing a gravitational acceleration and a state equation (see above).

tank = RectangularTank(fluid_particle_spacing, initial_fluid_size, tank_size,
                        fluid_density, n_layers=boundary_layers,
                        acceleration=(0.0, -gravity), state_equation=state_equation)

Fluid system

To model the water column, we use the Weakly Compressible Smoothed Particle Hydrodynamics (WCSPH) method. This method requires a smoothing kernel and a corresponding smoothing length, which should be chosen in relation to the particle spacing.

smoothing_length = 1.2 * fluid_particle_spacing
-smoothing_kernel = SchoenbergCubicSplineKernel{2}()

You can find an overview over smoothing kernels and corresponding smoothing lengths here.

For stability, we need numerical dissipation in form of an artificial viscosity term. Other viscosity models offer a physical approach based on the kinematic viscosity of the fluid.

viscosity = ArtificialViscosityMonaghan(alpha=0.02, beta=0.0)

We choose the parameters as small as possible to avoid non-physical behavior, but as large as possible to stabilize the simulation.

The WCSPH method can either compute the particle density directly with a kernel summation over all neighboring particles (see SummationDensity) or by making the particle density a variable in the ODE system and integrating its change over time. We choose the latter approach here by using the density calculator ContinuityDensity, which is more efficient and handles free surfaces without the need for additional correction terms.

fluid_density_calculator = ContinuityDensity()
+smoothing_kernel = SchoenbergCubicSplineKernel{2}()

You can find an overview over smoothing kernels and corresponding smoothing lengths here.

For stability, we need numerical dissipation in form of an artificial viscosity term. Other viscosity models offer a physical approach based on the kinematic viscosity of the fluid.

viscosity = ArtificialViscosityMonaghan(alpha=0.02, beta=0.0)

We choose the parameters as small as possible to avoid non-physical behavior, but as large as possible to stabilize the simulation.

The WCSPH method can either compute the particle density directly with a kernel summation over all neighboring particles (see SummationDensity) or by making the particle density a variable in the ODE system and integrating its change over time. We choose the latter approach here by using the density calculator ContinuityDensity, which is more efficient and handles free surfaces without the need for additional correction terms.

fluid_density_calculator = ContinuityDensity()
 fluid_system = WeaklyCompressibleSPHSystem(tank.fluid, fluid_density_calculator,
                                            state_equation, smoothing_kernel,
                                            smoothing_length, viscosity=viscosity,
@@ -18,7 +18,7 @@
                                              state_equation=state_equation,
                                              AdamiPressureExtrapolation(),
                                              smoothing_kernel, smoothing_length)
-boundary_system = BoundarySPHSystem(tank.boundary, boundary_model)

Semidiscretization

The key component of every simulation is the Semidiscretization, which couples all systems of the simulation. All methods in TrixiParticles.jl are semidiscretizations, which discretize the equations in time to provide an ordinary differential equation that still has to be solved in time. By providing a simulation time span, we can call semidiscretize, which returns an ODEProblem that can be solved with a time integration method.

semi = Semidiscretization(fluid_system, boundary_system)
+boundary_system = BoundarySPHSystem(tank.boundary, boundary_model)

Semidiscretization

The key component of every simulation is the Semidiscretization, which couples all systems of the simulation. All simulation methods in TrixiParticles.jl are semidiscretizations, which discretize the equations in time to provide an ordinary differential equation that still has to be solved in time. By providing a simulation time span, we can call semidiscretize, which returns an ODEProblem that can be solved with a time integration method.

semi = Semidiscretization(fluid_system, boundary_system)
 ode = semidiscretize(semi, tspan)

Time integration

We use the methods provided by OrdinaryDiffEq.jl, but note that other packages or custom implementations can also be used.

OrdinaryDiffEq.jl supports callbacks, which are executed during the simulation. For this simulation, we use the InfoCallback, which prints information about the simulation setup at the beginning of the simulation, information about the current simulation time and runtime during the simulation, and a performance summary at the end of the simulation. We also want to save the current solution in regular intervals in terms of simulation time as VTK, so that we can look at the solution in ParaView. The SolutionSavingCallback provides this functionality. To pass the callbacks to OrdinaryDiffEq.jl, we have to bundle them into a CallbackSet.

info_callback = InfoCallback(interval=50)
 saving_callback = SolutionSavingCallback(dt=0.02)
 
@@ -92,10 +92,10 @@
 │ #threads: ……………………………………………………… 1                                                                │
 └──────────────────────────────────────────────────────────────────────────────────────────────────┘
 
-#timesteps:     50 │ Δt: 1.4965e-03 │ sim. time: 1.8236e-01 (18.236%)  │ run time: 1.1508e-01 s
-#timesteps:    100 │ Δt: 5.3979e-03 │ sim. time: 4.2540e-01 (42.540%)  │ run time: 2.5941e-01 s
-#timesteps:    150 │ Δt: 5.7273e-03 │ sim. time: 6.7662e-01 (67.662%)  │ run time: 3.8571e-01 s
-#timesteps:    200 │ Δt: 4.8634e-03 │ sim. time: 9.1219e-01 (91.219%)  │ run time: 5.2338e-01 s
+#timesteps:     50 │ Δt: 1.4965e-03 │ sim. time: 1.8236e-01 (18.236%)  │ run time: 1.1851e-01 s
+#timesteps:    100 │ Δt: 5.3979e-03 │ sim. time: 4.2540e-01 (42.540%)  │ run time: 2.6537e-01 s
+#timesteps:    150 │ Δt: 5.7273e-03 │ sim. time: 6.7662e-01 (67.662%)  │ run time: 3.9621e-01 s
+#timesteps:    200 │ Δt: 4.8634e-03 │ sim. time: 9.1219e-01 (91.219%)  │ run time: 5.3892e-01 s
 ────────────────────────────────────────────────────────────────────────────────────────────────────
 Trixi simulation finished.  Final time: 1.0  Time steps: 222 (accepted), 222 (total)
 ────────────────────────────────────────────────────────────────────────────────────────────────────
@@ -103,43 +103,43 @@
 ────────────────────────────────────────────────────────────────────────────────
       TrixiParticles.jl                Time                    Allocations
                               ───────────────────────   ────────────────────────
-      Tot / % measured:            574ms /  98.0%           95.4MiB /  98.4%
+      Tot / % measured:            591ms /  97.9%           95.4MiB /  98.4%
 
 Section               ncalls     time    %tot     avg     alloc    %tot      avg
 ────────────────────────────────────────────────────────────────────────────────
-kick!                  1.11k    350ms   62.2%   315μs   3.77MiB    4.0%  3.47KiB
-  system interaction   1.11k    279ms   49.5%   250μs   1.05MiB    1.1%     985B
-    fluid1-fluid1      1.11k    221ms   39.2%   198μs     0.00B    0.0%    0.00B
-    fluid1-boundary2   1.11k   54.8ms    9.7%  49.3μs     0.00B    0.0%    0.00B
-    ~system intera...  1.11k   3.06ms    0.5%  2.75μs   1.05MiB    1.1%     985B
-    boundary2-fluid1   1.11k   32.7μs    0.0%  29.4ns     0.00B    0.0%    0.00B
-    boundary2-boun...  1.11k   32.2μs    0.0%  29.0ns     0.00B    0.0%    0.00B
-  update systems a...  1.11k   70.4ms   12.5%  63.3μs   2.72MiB    2.9%  2.50KiB
-    compute bounda...  1.11k   48.5ms    8.6%  43.6μs     0.00B    0.0%    0.00B
-    update nhs         1.11k   11.2ms    2.0%  10.0μs   2.72MiB    2.9%  2.50KiB
-    inverse state ...  1.11k   6.00ms    1.1%  5.39μs     0.00B    0.0%    0.00B
-    ~update system...  1.11k   4.78ms    0.8%  4.29μs   1.55KiB    0.0%    1.42B
-    update density...  1.11k   32.4μs    0.0%  29.1ns     0.00B    0.0%    0.00B
-  source terms         1.11k    423μs    0.1%   380ns     0.00B    0.0%    0.00B
-  ~kick!~              1.11k    386μs    0.1%   346ns   1.55KiB    0.0%    1.42B
-  reset ∂v/∂t          1.11k    149μs    0.0%   134ns     0.00B    0.0%    0.00B
-save solution             50    212ms   37.6%  4.23ms   90.0MiB   96.0%  1.80MiB
-  write to vtk           100    185ms   32.9%  1.85ms   86.7MiB   92.5%   888KiB
-  ~save solution~         50   23.3ms    4.1%   466μs   3.15MiB    3.4%  64.6KiB
-  update systems          50   3.05ms    0.5%  61.1μs    127KiB    0.1%  2.53KiB
-    compute bounda...     50   2.07ms    0.4%  41.5μs     0.00B    0.0%    0.00B
+kick!                  1.11k    349ms   60.3%   313μs   3.77MiB    4.0%  3.47KiB
+  system interaction   1.11k    279ms   48.2%   250μs   1.05MiB    1.1%     985B
+    fluid1-fluid1      1.11k    220ms   38.0%   198μs     0.00B    0.0%    0.00B
+    fluid1-boundary2   1.11k   55.4ms    9.6%  49.8μs     0.00B    0.0%    0.00B
+    ~system intera...  1.11k   3.16ms    0.5%  2.84μs   1.05MiB    1.1%     985B
+    boundary2-fluid1   1.11k   32.8μs    0.0%  29.5ns     0.00B    0.0%    0.00B
+    boundary2-boun...  1.11k   32.4μs    0.0%  29.1ns     0.00B    0.0%    0.00B
+  update systems a...  1.11k   69.3ms   12.0%  62.3μs   2.72MiB    2.9%  2.50KiB
+    compute bounda...  1.11k   47.0ms    8.1%  42.2μs     0.00B    0.0%    0.00B
+    update nhs         1.11k   11.2ms    1.9%  10.0μs   2.72MiB    2.9%  2.50KiB
+    inverse state ...  1.11k   6.10ms    1.1%  5.48μs     0.00B    0.0%    0.00B
+    ~update system...  1.11k   4.96ms    0.9%  4.45μs   1.55KiB    0.0%    1.42B
+    update density...  1.11k   32.8μs    0.0%  29.5ns     0.00B    0.0%    0.00B
+  source terms         1.11k    408μs    0.1%   366ns     0.00B    0.0%    0.00B
+  ~kick!~              1.11k    403μs    0.1%   362ns   1.55KiB    0.0%    1.42B
+  reset ∂v/∂t          1.11k    180μs    0.0%   162ns     0.00B    0.0%    0.00B
+save solution             50    229ms   39.5%  4.57ms   90.0MiB   96.0%  1.80MiB
+  write to vtk           100    195ms   33.6%  1.95ms   86.7MiB   92.5%   888KiB
+  ~save solution~         50   31.0ms    5.4%   619μs   3.15MiB    3.4%  64.6KiB
+  update systems          50   3.02ms    0.5%  60.3μs    127KiB    0.1%  2.53KiB
+    compute bounda...     50   2.01ms    0.3%  40.2μs     0.00B    0.0%    0.00B
     update nhs            50    494μs    0.1%  9.88μs    125KiB    0.1%  2.50KiB
-    inverse state ...     50    271μs    0.0%  5.41μs     0.00B    0.0%    0.00B
-    ~update systems~      50    216μs    0.0%  4.32μs   1.55KiB    0.0%    31.7B
-    update density...     50   1.48μs    0.0%  29.6ns     0.00B    0.0%    0.00B
-drift!                 1.11k    862μs    0.2%   775ns      976B    0.0%    0.88B
-  velocity             1.11k    515μs    0.1%   463ns     0.00B    0.0%    0.00B
-  ~drift!~             1.11k    203μs    0.0%   182ns      976B    0.0%    0.88B
-  reset ∂u/∂t          1.11k    145μs    0.0%   130ns     0.00B    0.0%    0.00B
-compute boundary p...      1   68.4μs    0.0%  68.4μs     0.00B    0.0%    0.00B
-update nhs                 1   44.0μs    0.0%  44.0μs   2.50KiB    0.0%  2.50KiB
-inverse state equa...      1   9.27μs    0.0%  9.27μs     0.00B    0.0%    0.00B
-update density dif...      1   40.0ns    0.0%  40.0ns     0.00B    0.0%    0.00B
+    inverse state ...     50    268μs    0.0%  5.35μs     0.00B    0.0%    0.00B
+    ~update systems~      50    237μs    0.0%  4.75μs   1.55KiB    0.0%    31.7B
+    update density...     50   6.35μs    0.0%   127ns     0.00B    0.0%    0.00B
+drift!                 1.11k    869μs    0.2%   781ns      976B    0.0%    0.88B
+  velocity             1.11k    528μs    0.1%   475ns     0.00B    0.0%    0.00B
+  ~drift!~             1.11k    212μs    0.0%   190ns      976B    0.0%    0.88B
+  reset ∂u/∂t          1.11k    129μs    0.0%   116ns     0.00B    0.0%    0.00B
+compute boundary p...      1   55.2μs    0.0%  55.2μs     0.00B    0.0%    0.00B
+update nhs                 1   48.6μs    0.0%  48.6μs   2.50KiB    0.0%  2.50KiB
+inverse state equa...      1   7.49μs    0.0%  7.49μs     0.00B    0.0%    0.00B
+update density dif...      1   31.0ns    0.0%  31.0ns     0.00B    0.0%    0.00B
 ────────────────────────────────────────────────────────────────────────────────

See Visualization for how to visualize the solution. For the simplest visualization, we can use Plots.jl:

using Plots
 plot(sol)

plot

Replacing components with custom implementations

If we would like to use an implementation of a component that is not available in TrixiParticles.jl, we can implement it ourselves within the simulation file, without ever cloning the TrixiParticles.jl repository. A good starting point is to check out the available implementations in TrixiParticles.jl, then copy the relevant functions to the simulation file and modify them as needed.

Custom smoothing kernel

To implement a custom smoothing kernel, we define a struct extending TrixiParticles.SmoothingKernel. This abstract struct has a type parameter for the number of dimensions, which we set to 2 in this case.

struct MyGaussianKernel <: TrixiParticles.SmoothingKernel{2} end

This kernel is going to be an implementation of the Gaussian kernel with a cutoff for compact support. Note that the same kernel in a more optimized version is already implemented in TrixiParticles.jl as GaussianKernel.

In order to use our new kernel, we have to define three functions. TrixiParticles.kernel, which is the kernel function itself, TrixiParticles.kernel_deriv, which is the derivative of the kernel function, and TrixiParticles.compact_support, which defines the compact support of the kernel in relation to the smoothing length. The latter is relevant for determining the search radius of the neighborhood search.

function TrixiParticles.kernel(kernel::MyGaussianKernel, r, h)
     q = r / h
@@ -238,26 +238,26 @@
 │ #threads: ……………………………………………………… 1                                                                │
 └──────────────────────────────────────────────────────────────────────────────────────────────────┘
 
-#timesteps:     50 │ Δt: 2.1194e-06 │ sim. time: 2.6343e-02 (2.634%)   │ run time: 2.8393e+00 s
-#timesteps:    100 │ Δt: 3.6867e-07 │ sim. time: 2.6368e-02 (2.637%)   │ run time: 3.2136e+00 s
-#timesteps:    150 │ Δt: 3.8641e-06 │ sim. time: 2.7157e-02 (2.716%)   │ run time: 3.5826e+00 s
-#timesteps:    200 │ Δt: 2.7500e-05 │ sim. time: 2.9021e-02 (2.902%)   │ run time: 3.9436e+00 s
-#timesteps:    250 │ Δt: 3.0345e-08 │ sim. time: 2.9289e-02 (2.929%)   │ run time: 4.3407e+00 s
-#timesteps:    300 │ Δt: 5.4341e-07 │ sim. time: 2.9308e-02 (2.931%)   │ run time: 4.7031e+00 s
-#timesteps:    350 │ Δt: 1.1375e-08 │ sim. time: 2.9930e-02 (2.993%)   │ run time: 5.1028e+00 s
-#timesteps:    400 │ Δt: 3.7655e-08 │ sim. time: 3.0011e-02 (3.001%)   │ run time: 5.4507e+00 s
-#timesteps:    450 │ Δt: 2.5171e-06 │ sim. time: 3.0049e-02 (3.005%)   │ run time: 5.7929e+00 s
-#timesteps:    500 │ Δt: 4.6396e-05 │ sim. time: 3.1465e-02 (3.146%)   │ run time: 6.1538e+00 s
-#timesteps:    550 │ Δt: 2.8950e-04 │ sim. time: 3.4385e-02 (3.438%)   │ run time: 6.5332e+00 s
-#timesteps:    600 │ Δt: 4.7243e-04 │ sim. time: 4.0000e-02 (4.000%)   │ run time: 6.8625e+00 s
-#timesteps:    650 │ Δt: 9.2229e-04 │ sim. time: 7.3763e-02 (7.376%)   │ run time: 7.0718e+00 s
-#timesteps:    700 │ Δt: 1.5410e-03 │ sim. time: 1.2753e-01 (12.753%)  │ run time: 7.2501e+00 s
-#timesteps:    750 │ Δt: 7.9337e-04 │ sim. time: 2.1356e-01 (21.356%)  │ run time: 7.4886e+00 s
-#timesteps:    800 │ Δt: 1.5629e-03 │ sim. time: 2.9162e-01 (29.162%)  │ run time: 7.6916e+00 s
-#timesteps:    850 │ Δt: 2.4390e-03 │ sim. time: 3.6859e-01 (36.859%)  │ run time: 7.9324e+00 s
-#timesteps:    900 │ Δt: 2.6538e-03 │ sim. time: 4.5664e-01 (45.664%)  │ run time: 8.1282e+00 s
-#timesteps:    950 │ Δt: 7.2062e-03 │ sim. time: 6.5246e-01 (65.246%)  │ run time: 8.2757e+00 s
-#timesteps:   1000 │ Δt: 6.7704e-03 │ sim. time: 9.7138e-01 (97.138%)  │ run time: 8.7225e+00 s
+#timesteps:     50 │ Δt: 2.1194e-06 │ sim. time: 2.6343e-02 (2.634%)   │ run time: 2.9891e+00 s
+#timesteps:    100 │ Δt: 3.6867e-07 │ sim. time: 2.6368e-02 (2.637%)   │ run time: 3.3637e+00 s
+#timesteps:    150 │ Δt: 3.8641e-06 │ sim. time: 2.7157e-02 (2.716%)   │ run time: 3.7325e+00 s
+#timesteps:    200 │ Δt: 2.7500e-05 │ sim. time: 2.9021e-02 (2.902%)   │ run time: 4.0928e+00 s
+#timesteps:    250 │ Δt: 3.0345e-08 │ sim. time: 2.9289e-02 (2.929%)   │ run time: 4.4976e+00 s
+#timesteps:    300 │ Δt: 5.4341e-07 │ sim. time: 2.9308e-02 (2.931%)   │ run time: 4.8600e+00 s
+#timesteps:    350 │ Δt: 1.1375e-08 │ sim. time: 2.9930e-02 (2.993%)   │ run time: 5.2596e+00 s
+#timesteps:    400 │ Δt: 3.7655e-08 │ sim. time: 3.0011e-02 (3.001%)   │ run time: 5.6081e+00 s
+#timesteps:    450 │ Δt: 2.5171e-06 │ sim. time: 3.0049e-02 (3.005%)   │ run time: 5.9497e+00 s
+#timesteps:    500 │ Δt: 4.6396e-05 │ sim. time: 3.1465e-02 (3.146%)   │ run time: 6.3099e+00 s
+#timesteps:    550 │ Δt: 2.8950e-04 │ sim. time: 3.4385e-02 (3.438%)   │ run time: 6.6882e+00 s
+#timesteps:    600 │ Δt: 4.7243e-04 │ sim. time: 4.0000e-02 (4.000%)   │ run time: 7.0156e+00 s
+#timesteps:    650 │ Δt: 9.2229e-04 │ sim. time: 7.3763e-02 (7.376%)   │ run time: 7.2248e+00 s
+#timesteps:    700 │ Δt: 1.5410e-03 │ sim. time: 1.2753e-01 (12.753%)  │ run time: 7.4031e+00 s
+#timesteps:    750 │ Δt: 7.9337e-04 │ sim. time: 2.1356e-01 (21.356%)  │ run time: 7.6141e+00 s
+#timesteps:    800 │ Δt: 1.5629e-03 │ sim. time: 2.9162e-01 (29.162%)  │ run time: 7.8193e+00 s
+#timesteps:    850 │ Δt: 2.4390e-03 │ sim. time: 3.6859e-01 (36.859%)  │ run time: 8.0144e+00 s
+#timesteps:    900 │ Δt: 2.6538e-03 │ sim. time: 4.5664e-01 (45.664%)  │ run time: 8.2197e+00 s
+#timesteps:    950 │ Δt: 7.2062e-03 │ sim. time: 6.5246e-01 (65.246%)  │ run time: 8.3753e+00 s
+#timesteps:   1000 │ Δt: 6.7704e-03 │ sim. time: 9.7138e-01 (97.138%)  │ run time: 8.5185e+00 s
 ────────────────────────────────────────────────────────────────────────────────────────────────────
 Trixi simulation finished.  Final time: 1.0  Time steps: 1003 (accepted), 1891 (total)
 ────────────────────────────────────────────────────────────────────────────────────────────────────
@@ -265,41 +265,41 @@
 ────────────────────────────────────────────────────────────────────────────────
       TrixiParticles.jl                Time                    Allocations
                               ───────────────────────   ────────────────────────
-      Tot / % measured:            8.73s /  74.7%            228MiB /  53.4%
+      Tot / % measured:            8.53s /  72.6%            228MiB /  53.4%
 
 Section               ncalls     time    %tot     avg     alloc    %tot      avg
 ────────────────────────────────────────────────────────────────────────────────
-kick!                  9.46k    5.64s   86.4%   596μs   22.6MiB   18.6%  2.45KiB
-  system interaction   9.46k    4.76s   73.0%   504μs   8.88MiB    7.3%     984B
-    fluid1-fluid1      9.46k    3.77s   57.8%   398μs     0.00B    0.0%    0.00B
-    fluid1-boundary2   9.46k    969ms   14.9%   102μs     0.00B    0.0%    0.00B
-    ~system intera...  9.46k   24.7ms    0.4%  2.61μs   8.88MiB    7.3%     984B
-    boundary2-fluid1   9.46k    290μs    0.0%  30.7ns     0.00B    0.0%    0.00B
-    boundary2-boun...  9.46k    275μs    0.0%  29.1ns     0.00B    0.0%    0.00B
-  update systems a...  9.46k    868ms   13.3%  91.7μs   13.7MiB   11.3%  1.49KiB
-    compute bounda...  9.46k    702ms   10.8%  74.2μs     0.00B    0.0%    0.00B
-    update nhs         9.46k   74.1ms    1.1%  7.84μs   13.7MiB   11.3%  1.49KiB
-    inverse state ...  9.46k   50.4ms    0.8%  5.33μs     0.00B    0.0%    0.00B
-    ~update system...  9.46k   40.6ms    0.6%  4.29μs   1.55KiB    0.0%    0.17B
-    update density...  9.46k    275μs    0.0%  29.1ns     0.00B    0.0%    0.00B
-  source terms         9.46k   3.56ms    0.1%   376ns     0.00B    0.0%    0.00B
-  ~kick!~              9.46k   3.13ms    0.0%   331ns   1.55KiB    0.0%    0.17B
-  reset ∂v/∂t          9.46k   1.24ms    0.0%   131ns     0.00B    0.0%    0.00B
-save solution             50    877ms   13.4%  17.5ms   99.2MiB   81.4%  1.98MiB
-  write to vtk           100    634ms    9.7%  6.34ms   86.9MiB   71.4%   890KiB
-  ~save solution~         50    239ms    3.7%  4.77ms   12.1MiB   10.0%   248KiB
-  update systems          50   4.50ms    0.1%  90.0μs    161KiB    0.1%  3.23KiB
-    compute bounda...     50   3.39ms    0.1%  67.7μs     0.00B    0.0%    0.00B
-    update nhs            50    645μs    0.0%  12.9μs    160KiB    0.1%  3.19KiB
-    inverse state ...     50    237μs    0.0%  4.74μs     0.00B    0.0%    0.00B
-    ~update systems~      50    230μs    0.0%  4.60μs   1.55KiB    0.0%    31.7B
-    update density...     50   2.19μs    0.0%  43.7ns     0.00B    0.0%    0.00B
-drift!                 9.46k   6.99ms    0.1%   739ns      976B    0.0%    0.10B
-  velocity             9.46k   4.22ms    0.1%   446ns     0.00B    0.0%    0.00B
-  ~drift!~             9.46k   1.67ms    0.0%   177ns      976B    0.0%    0.10B
-  reset ∂u/∂t          9.46k   1.10ms    0.0%   117ns     0.00B    0.0%    0.00B
-compute boundary p...      1    104μs    0.0%   104μs     0.00B    0.0%    0.00B
-update nhs                 1   36.5μs    0.0%  36.5μs   1.12KiB    0.0%  1.12KiB
-inverse state equa...      1   7.71μs    0.0%  7.71μs     0.00B    0.0%    0.00B
+kick!                  9.46k    5.62s   90.7%   594μs   22.6MiB   18.6%  2.45KiB
+  system interaction   9.46k    4.75s   76.7%   502μs   8.88MiB    7.3%     984B
+    fluid1-fluid1      9.46k    3.75s   60.5%   396μs     0.00B    0.0%    0.00B
+    fluid1-boundary2   9.46k    975ms   15.7%   103μs     0.00B    0.0%    0.00B
+    ~system intera...  9.46k   24.9ms    0.4%  2.63μs   8.88MiB    7.3%     984B
+    boundary2-fluid1   9.46k    293μs    0.0%  31.0ns     0.00B    0.0%    0.00B
+    boundary2-boun...  9.46k    276μs    0.0%  29.1ns     0.00B    0.0%    0.00B
+  update systems a...  9.46k    857ms   13.8%  90.6μs   13.7MiB   11.3%  1.49KiB
+    compute bounda...  9.46k    692ms   11.2%  73.2μs     0.00B    0.0%    0.00B
+    update nhs         9.46k   74.0ms    1.2%  7.82μs   13.7MiB   11.3%  1.49KiB
+    inverse state ...  9.46k   50.7ms    0.8%  5.36μs     0.00B    0.0%    0.00B
+    ~update system...  9.46k   40.0ms    0.6%  4.23μs   1.55KiB    0.0%    0.17B
+    update density...  9.46k    275μs    0.0%  29.0ns     0.00B    0.0%    0.00B
+  source terms         9.46k   4.11ms    0.1%   434ns     0.00B    0.0%    0.00B
+  ~kick!~              9.46k   3.12ms    0.1%   329ns   1.55KiB    0.0%    0.17B
+  reset ∂v/∂t          9.46k   1.30ms    0.0%   137ns     0.00B    0.0%    0.00B
+save solution             50    571ms    9.2%  11.4ms   99.2MiB   81.4%  1.98MiB
+  write to vtk           100    311ms    5.0%  3.11ms   86.9MiB   71.4%   890KiB
+  ~save solution~         50    256ms    4.1%  5.12ms   12.1MiB   10.0%   248KiB
+  update systems          50   4.47ms    0.1%  89.5μs    161KiB    0.1%  3.23KiB
+    compute bounda...     50   3.36ms    0.1%  67.2μs     0.00B    0.0%    0.00B
+    update nhs            50    650μs    0.0%  13.0μs    160KiB    0.1%  3.19KiB
+    ~update systems~      50    235μs    0.0%  4.70μs   1.55KiB    0.0%    31.7B
+    inverse state ...     50    224μs    0.0%  4.48μs     0.00B    0.0%    0.00B
+    update density...     50   1.52μs    0.0%  30.4ns     0.00B    0.0%    0.00B
+drift!                 9.46k   7.20ms    0.1%   761ns      976B    0.0%    0.10B
+  velocity             9.46k   4.41ms    0.1%   466ns     0.00B    0.0%    0.00B
+  ~drift!~             9.46k   1.65ms    0.0%   174ns      976B    0.0%    0.10B
+  reset ∂u/∂t          9.46k   1.15ms    0.0%   121ns     0.00B    0.0%    0.00B
+compute boundary p...      1   89.3μs    0.0%  89.3μs     0.00B    0.0%    0.00B
+update nhs                 1   30.0μs    0.0%  30.0μs   1.12KiB    0.0%  1.12KiB
+inverse state equa...      1   8.11μs    0.0%  8.11μs     0.00B    0.0%    0.00B
 update density dif...      1   40.0ns    0.0%  40.0ns     0.00B    0.0%    0.00B
-────────────────────────────────────────────────────────────────────────────────
+──────────────────────────────────────────────────────────────────────────────── diff --git a/previews/PR514/tutorials_template/out/boundary_1_0.vtu b/previews/PR514/tutorials_template/out/boundary_1_0.vtu index 1ddde76be..5a8dc2524 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_0.vtu and b/previews/PR514/tutorials_template/out/boundary_1_0.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_1.vtu b/previews/PR514/tutorials_template/out/boundary_1_1.vtu index 48250dc62..fb357e618 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_1.vtu and b/previews/PR514/tutorials_template/out/boundary_1_1.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_10.vtu b/previews/PR514/tutorials_template/out/boundary_1_10.vtu index 99c9e2fa9..a662a3e31 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_10.vtu and b/previews/PR514/tutorials_template/out/boundary_1_10.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_11.vtu b/previews/PR514/tutorials_template/out/boundary_1_11.vtu index 9202c0890..4fa09c873 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_11.vtu and b/previews/PR514/tutorials_template/out/boundary_1_11.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_12.vtu b/previews/PR514/tutorials_template/out/boundary_1_12.vtu index 8dfeaf324..197ff4ef6 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_12.vtu and b/previews/PR514/tutorials_template/out/boundary_1_12.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_13.vtu b/previews/PR514/tutorials_template/out/boundary_1_13.vtu index 813990aff..dff2eb4fb 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_13.vtu and b/previews/PR514/tutorials_template/out/boundary_1_13.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_14.vtu b/previews/PR514/tutorials_template/out/boundary_1_14.vtu index 320791526..ea325df3c 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_14.vtu and b/previews/PR514/tutorials_template/out/boundary_1_14.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_15.vtu b/previews/PR514/tutorials_template/out/boundary_1_15.vtu index 51d20761a..e9ccf05ed 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_15.vtu and b/previews/PR514/tutorials_template/out/boundary_1_15.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_16.vtu b/previews/PR514/tutorials_template/out/boundary_1_16.vtu index 7cfe266bd..87d22887b 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_16.vtu and b/previews/PR514/tutorials_template/out/boundary_1_16.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_17.vtu b/previews/PR514/tutorials_template/out/boundary_1_17.vtu index 7b84e680c..588e7bfa1 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_17.vtu and b/previews/PR514/tutorials_template/out/boundary_1_17.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_18.vtu b/previews/PR514/tutorials_template/out/boundary_1_18.vtu index b715321f1..5b4b39148 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_18.vtu and b/previews/PR514/tutorials_template/out/boundary_1_18.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_19.vtu b/previews/PR514/tutorials_template/out/boundary_1_19.vtu index 1fd494e1c..df322517e 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_19.vtu and b/previews/PR514/tutorials_template/out/boundary_1_19.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_2.vtu b/previews/PR514/tutorials_template/out/boundary_1_2.vtu index cb7b5c88b..87bedcbe5 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_2.vtu and b/previews/PR514/tutorials_template/out/boundary_1_2.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_20.vtu b/previews/PR514/tutorials_template/out/boundary_1_20.vtu index 2f2e2e514..0bfeb0d21 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_20.vtu and b/previews/PR514/tutorials_template/out/boundary_1_20.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_21.vtu b/previews/PR514/tutorials_template/out/boundary_1_21.vtu index 54a1d4ccb..8fb8b2886 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_21.vtu and b/previews/PR514/tutorials_template/out/boundary_1_21.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_22.vtu b/previews/PR514/tutorials_template/out/boundary_1_22.vtu index 2d19355dd..9b779aa33 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_22.vtu and b/previews/PR514/tutorials_template/out/boundary_1_22.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_23.vtu b/previews/PR514/tutorials_template/out/boundary_1_23.vtu index 745c9f360..e421e3686 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_23.vtu and b/previews/PR514/tutorials_template/out/boundary_1_23.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_24.vtu b/previews/PR514/tutorials_template/out/boundary_1_24.vtu index f329f7290..e6147e8d6 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_24.vtu and b/previews/PR514/tutorials_template/out/boundary_1_24.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_25.vtu b/previews/PR514/tutorials_template/out/boundary_1_25.vtu index e40cb92f6..325e85da8 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_25.vtu and b/previews/PR514/tutorials_template/out/boundary_1_25.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_26.vtu b/previews/PR514/tutorials_template/out/boundary_1_26.vtu index 19d56a6bb..1c71de072 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_26.vtu and b/previews/PR514/tutorials_template/out/boundary_1_26.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_27.vtu b/previews/PR514/tutorials_template/out/boundary_1_27.vtu index 03c9c7450..e99b0537e 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_27.vtu and b/previews/PR514/tutorials_template/out/boundary_1_27.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_28.vtu b/previews/PR514/tutorials_template/out/boundary_1_28.vtu index 032d6c50c..edcb3960d 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_28.vtu and b/previews/PR514/tutorials_template/out/boundary_1_28.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_29.vtu b/previews/PR514/tutorials_template/out/boundary_1_29.vtu index a22930d46..e150c21d2 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_29.vtu and b/previews/PR514/tutorials_template/out/boundary_1_29.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_3.vtu b/previews/PR514/tutorials_template/out/boundary_1_3.vtu index f3498dd20..e3d2ff30e 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_3.vtu and b/previews/PR514/tutorials_template/out/boundary_1_3.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_30.vtu b/previews/PR514/tutorials_template/out/boundary_1_30.vtu index d963d3c92..2607732b3 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_30.vtu and b/previews/PR514/tutorials_template/out/boundary_1_30.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_31.vtu b/previews/PR514/tutorials_template/out/boundary_1_31.vtu index 2878bcf52..02a2322d4 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_31.vtu and b/previews/PR514/tutorials_template/out/boundary_1_31.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_32.vtu b/previews/PR514/tutorials_template/out/boundary_1_32.vtu index 5e176ce09..aed049666 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_32.vtu and b/previews/PR514/tutorials_template/out/boundary_1_32.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_33.vtu b/previews/PR514/tutorials_template/out/boundary_1_33.vtu index 10cc48479..1a7e7f2b7 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_33.vtu and b/previews/PR514/tutorials_template/out/boundary_1_33.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_34.vtu b/previews/PR514/tutorials_template/out/boundary_1_34.vtu index bfee37707..478556350 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_34.vtu and b/previews/PR514/tutorials_template/out/boundary_1_34.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_35.vtu b/previews/PR514/tutorials_template/out/boundary_1_35.vtu index 02e54dec9..a9a87a797 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_35.vtu and b/previews/PR514/tutorials_template/out/boundary_1_35.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_36.vtu b/previews/PR514/tutorials_template/out/boundary_1_36.vtu index c1957b3dc..6390cb4b7 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_36.vtu and b/previews/PR514/tutorials_template/out/boundary_1_36.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_37.vtu b/previews/PR514/tutorials_template/out/boundary_1_37.vtu index 34f552693..544037ad9 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_37.vtu and b/previews/PR514/tutorials_template/out/boundary_1_37.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_38.vtu b/previews/PR514/tutorials_template/out/boundary_1_38.vtu index f7d03f859..e4529321a 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_38.vtu and b/previews/PR514/tutorials_template/out/boundary_1_38.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_39.vtu b/previews/PR514/tutorials_template/out/boundary_1_39.vtu index 494e2314e..264d79fb9 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_39.vtu and b/previews/PR514/tutorials_template/out/boundary_1_39.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_4.vtu b/previews/PR514/tutorials_template/out/boundary_1_4.vtu index 64367083d..4d1a7baac 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_4.vtu and b/previews/PR514/tutorials_template/out/boundary_1_4.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_40.vtu b/previews/PR514/tutorials_template/out/boundary_1_40.vtu index 668e97ed1..362afe77f 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_40.vtu and b/previews/PR514/tutorials_template/out/boundary_1_40.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_41.vtu b/previews/PR514/tutorials_template/out/boundary_1_41.vtu index b5f78dbc6..6a5740144 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_41.vtu and b/previews/PR514/tutorials_template/out/boundary_1_41.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_42.vtu b/previews/PR514/tutorials_template/out/boundary_1_42.vtu index bb8cb4dfe..00fa81ab1 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_42.vtu and b/previews/PR514/tutorials_template/out/boundary_1_42.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_43.vtu b/previews/PR514/tutorials_template/out/boundary_1_43.vtu index d6b086850..fa3a0e474 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_43.vtu and b/previews/PR514/tutorials_template/out/boundary_1_43.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_44.vtu b/previews/PR514/tutorials_template/out/boundary_1_44.vtu index c595993af..7e8e611d7 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_44.vtu and b/previews/PR514/tutorials_template/out/boundary_1_44.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_45.vtu b/previews/PR514/tutorials_template/out/boundary_1_45.vtu index 2816e48c0..ec47dffe1 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_45.vtu and b/previews/PR514/tutorials_template/out/boundary_1_45.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_46.vtu b/previews/PR514/tutorials_template/out/boundary_1_46.vtu index b030022f1..ae0217772 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_46.vtu and b/previews/PR514/tutorials_template/out/boundary_1_46.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_47.vtu b/previews/PR514/tutorials_template/out/boundary_1_47.vtu index 3b5b50e5b..37ea860b4 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_47.vtu and b/previews/PR514/tutorials_template/out/boundary_1_47.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_48.vtu b/previews/PR514/tutorials_template/out/boundary_1_48.vtu index 30a5038b5..b9caedef7 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_48.vtu and b/previews/PR514/tutorials_template/out/boundary_1_48.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_49.vtu b/previews/PR514/tutorials_template/out/boundary_1_49.vtu index ddf1b7362..1126fc34d 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_49.vtu and b/previews/PR514/tutorials_template/out/boundary_1_49.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_5.vtu b/previews/PR514/tutorials_template/out/boundary_1_5.vtu index 5068ae0a7..04b6b610a 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_5.vtu and b/previews/PR514/tutorials_template/out/boundary_1_5.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_50.vtu b/previews/PR514/tutorials_template/out/boundary_1_50.vtu index 9b8936fb5..7c4f02315 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_50.vtu and b/previews/PR514/tutorials_template/out/boundary_1_50.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_6.vtu b/previews/PR514/tutorials_template/out/boundary_1_6.vtu index 9f3f2dbab..f2e4836e2 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_6.vtu and b/previews/PR514/tutorials_template/out/boundary_1_6.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_7.vtu b/previews/PR514/tutorials_template/out/boundary_1_7.vtu index 02987fc25..26a568c22 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_7.vtu and b/previews/PR514/tutorials_template/out/boundary_1_7.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_8.vtu b/previews/PR514/tutorials_template/out/boundary_1_8.vtu index 6c1594101..2a0c861f9 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_8.vtu and b/previews/PR514/tutorials_template/out/boundary_1_8.vtu differ diff --git a/previews/PR514/tutorials_template/out/boundary_1_9.vtu b/previews/PR514/tutorials_template/out/boundary_1_9.vtu index f9bbb4864..a2b6b1685 100644 Binary files a/previews/PR514/tutorials_template/out/boundary_1_9.vtu and b/previews/PR514/tutorials_template/out/boundary_1_9.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_0.vtu b/previews/PR514/tutorials_template/out/fluid_1_0.vtu index 352c9b5fa..39d2ccf26 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_0.vtu and b/previews/PR514/tutorials_template/out/fluid_1_0.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_1.vtu b/previews/PR514/tutorials_template/out/fluid_1_1.vtu index 5be5d6b6e..51064283a 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_1.vtu and b/previews/PR514/tutorials_template/out/fluid_1_1.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_10.vtu b/previews/PR514/tutorials_template/out/fluid_1_10.vtu index e0466e4c9..7989af8f8 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_10.vtu and b/previews/PR514/tutorials_template/out/fluid_1_10.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_11.vtu b/previews/PR514/tutorials_template/out/fluid_1_11.vtu index 3cc68de89..4c7c77daf 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_11.vtu and b/previews/PR514/tutorials_template/out/fluid_1_11.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_12.vtu b/previews/PR514/tutorials_template/out/fluid_1_12.vtu index 5e71b5282..6758e67fc 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_12.vtu and b/previews/PR514/tutorials_template/out/fluid_1_12.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_13.vtu b/previews/PR514/tutorials_template/out/fluid_1_13.vtu index 41f88e425..cb59f65b9 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_13.vtu and b/previews/PR514/tutorials_template/out/fluid_1_13.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_14.vtu b/previews/PR514/tutorials_template/out/fluid_1_14.vtu index 43b01e21e..3441df0df 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_14.vtu and b/previews/PR514/tutorials_template/out/fluid_1_14.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_15.vtu b/previews/PR514/tutorials_template/out/fluid_1_15.vtu index ce1ae8bf7..555b377c0 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_15.vtu and b/previews/PR514/tutorials_template/out/fluid_1_15.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_16.vtu b/previews/PR514/tutorials_template/out/fluid_1_16.vtu index 9373ed0ca..d3f81c80d 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_16.vtu and b/previews/PR514/tutorials_template/out/fluid_1_16.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_17.vtu b/previews/PR514/tutorials_template/out/fluid_1_17.vtu index 1f461bcce..d8f33a80f 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_17.vtu and b/previews/PR514/tutorials_template/out/fluid_1_17.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_18.vtu b/previews/PR514/tutorials_template/out/fluid_1_18.vtu index 600542b58..d2f0c7fe0 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_18.vtu and b/previews/PR514/tutorials_template/out/fluid_1_18.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_19.vtu b/previews/PR514/tutorials_template/out/fluid_1_19.vtu index 5f8d6656e..d0c2a9706 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_19.vtu and b/previews/PR514/tutorials_template/out/fluid_1_19.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_2.vtu b/previews/PR514/tutorials_template/out/fluid_1_2.vtu index 6cf55b7f2..5c9f70d2d 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_2.vtu and b/previews/PR514/tutorials_template/out/fluid_1_2.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_20.vtu b/previews/PR514/tutorials_template/out/fluid_1_20.vtu index 4b802181a..80e49c902 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_20.vtu and b/previews/PR514/tutorials_template/out/fluid_1_20.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_21.vtu b/previews/PR514/tutorials_template/out/fluid_1_21.vtu index 1870c80be..a98c5dec3 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_21.vtu and b/previews/PR514/tutorials_template/out/fluid_1_21.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_22.vtu b/previews/PR514/tutorials_template/out/fluid_1_22.vtu index 8c90807f3..88e06edb8 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_22.vtu and b/previews/PR514/tutorials_template/out/fluid_1_22.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_23.vtu b/previews/PR514/tutorials_template/out/fluid_1_23.vtu index 26673d83c..3ae629adf 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_23.vtu and b/previews/PR514/tutorials_template/out/fluid_1_23.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_24.vtu b/previews/PR514/tutorials_template/out/fluid_1_24.vtu index 1c06134aa..a149829b4 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_24.vtu and b/previews/PR514/tutorials_template/out/fluid_1_24.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_25.vtu b/previews/PR514/tutorials_template/out/fluid_1_25.vtu index 405b1301f..da2e2006f 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_25.vtu and b/previews/PR514/tutorials_template/out/fluid_1_25.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_26.vtu b/previews/PR514/tutorials_template/out/fluid_1_26.vtu index 51e8e40de..d3faf24a7 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_26.vtu and b/previews/PR514/tutorials_template/out/fluid_1_26.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_27.vtu b/previews/PR514/tutorials_template/out/fluid_1_27.vtu index d76266705..5d7ac4050 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_27.vtu and b/previews/PR514/tutorials_template/out/fluid_1_27.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_28.vtu b/previews/PR514/tutorials_template/out/fluid_1_28.vtu index f584ab67f..61ec50504 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_28.vtu and b/previews/PR514/tutorials_template/out/fluid_1_28.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_29.vtu b/previews/PR514/tutorials_template/out/fluid_1_29.vtu index 112eff0a0..9cbdbf2c8 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_29.vtu and b/previews/PR514/tutorials_template/out/fluid_1_29.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_3.vtu b/previews/PR514/tutorials_template/out/fluid_1_3.vtu index fd9b3abec..c63f565bb 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_3.vtu and b/previews/PR514/tutorials_template/out/fluid_1_3.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_30.vtu b/previews/PR514/tutorials_template/out/fluid_1_30.vtu index 8625be987..43ef34c0a 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_30.vtu and b/previews/PR514/tutorials_template/out/fluid_1_30.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_31.vtu b/previews/PR514/tutorials_template/out/fluid_1_31.vtu index 311847e6d..524f97ca3 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_31.vtu and b/previews/PR514/tutorials_template/out/fluid_1_31.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_32.vtu b/previews/PR514/tutorials_template/out/fluid_1_32.vtu index 4df9e5e76..ec13d42a4 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_32.vtu and b/previews/PR514/tutorials_template/out/fluid_1_32.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_33.vtu b/previews/PR514/tutorials_template/out/fluid_1_33.vtu index 5baee3122..89a1ef58a 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_33.vtu and b/previews/PR514/tutorials_template/out/fluid_1_33.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_34.vtu b/previews/PR514/tutorials_template/out/fluid_1_34.vtu index 0f79df2d9..30cdd731e 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_34.vtu and b/previews/PR514/tutorials_template/out/fluid_1_34.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_35.vtu b/previews/PR514/tutorials_template/out/fluid_1_35.vtu index 4637cfa75..95adde387 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_35.vtu and b/previews/PR514/tutorials_template/out/fluid_1_35.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_36.vtu b/previews/PR514/tutorials_template/out/fluid_1_36.vtu index 00e2c5fe1..69d814de8 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_36.vtu and b/previews/PR514/tutorials_template/out/fluid_1_36.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_37.vtu b/previews/PR514/tutorials_template/out/fluid_1_37.vtu index 9c83b3399..8b4da2659 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_37.vtu and b/previews/PR514/tutorials_template/out/fluid_1_37.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_38.vtu b/previews/PR514/tutorials_template/out/fluid_1_38.vtu index 9f681898e..57763b28e 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_38.vtu and b/previews/PR514/tutorials_template/out/fluid_1_38.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_39.vtu b/previews/PR514/tutorials_template/out/fluid_1_39.vtu index ca604e7d6..9db539507 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_39.vtu and b/previews/PR514/tutorials_template/out/fluid_1_39.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_4.vtu b/previews/PR514/tutorials_template/out/fluid_1_4.vtu index fc3871143..a18720571 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_4.vtu and b/previews/PR514/tutorials_template/out/fluid_1_4.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_40.vtu b/previews/PR514/tutorials_template/out/fluid_1_40.vtu index 56d3fa228..8e09fd286 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_40.vtu and b/previews/PR514/tutorials_template/out/fluid_1_40.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_41.vtu b/previews/PR514/tutorials_template/out/fluid_1_41.vtu index 2fdff4be9..b2d0abbc7 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_41.vtu and b/previews/PR514/tutorials_template/out/fluid_1_41.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_42.vtu b/previews/PR514/tutorials_template/out/fluid_1_42.vtu index da424ca8d..32632d2f4 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_42.vtu and b/previews/PR514/tutorials_template/out/fluid_1_42.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_43.vtu b/previews/PR514/tutorials_template/out/fluid_1_43.vtu index 2d424e1dc..5ffbaf59b 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_43.vtu and b/previews/PR514/tutorials_template/out/fluid_1_43.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_44.vtu b/previews/PR514/tutorials_template/out/fluid_1_44.vtu index eeb5c4182..3994ac26c 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_44.vtu and b/previews/PR514/tutorials_template/out/fluid_1_44.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_45.vtu b/previews/PR514/tutorials_template/out/fluid_1_45.vtu index 4b5a3440b..c92a24098 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_45.vtu and b/previews/PR514/tutorials_template/out/fluid_1_45.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_46.vtu b/previews/PR514/tutorials_template/out/fluid_1_46.vtu index c80864d19..6100daa8e 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_46.vtu and b/previews/PR514/tutorials_template/out/fluid_1_46.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_47.vtu b/previews/PR514/tutorials_template/out/fluid_1_47.vtu index 99c96bcef..dfda15f6e 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_47.vtu and b/previews/PR514/tutorials_template/out/fluid_1_47.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_48.vtu b/previews/PR514/tutorials_template/out/fluid_1_48.vtu index a586cc3c6..7fcb3a29c 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_48.vtu and b/previews/PR514/tutorials_template/out/fluid_1_48.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_49.vtu b/previews/PR514/tutorials_template/out/fluid_1_49.vtu index 86ee8d5e1..42acb90ea 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_49.vtu and b/previews/PR514/tutorials_template/out/fluid_1_49.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_5.vtu b/previews/PR514/tutorials_template/out/fluid_1_5.vtu index 1d3fad584..0e31aa868 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_5.vtu and b/previews/PR514/tutorials_template/out/fluid_1_5.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_50.vtu b/previews/PR514/tutorials_template/out/fluid_1_50.vtu index f97f7d3bd..330047a5b 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_50.vtu and b/previews/PR514/tutorials_template/out/fluid_1_50.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_6.vtu b/previews/PR514/tutorials_template/out/fluid_1_6.vtu index c47305385..d7f0b8ca0 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_6.vtu and b/previews/PR514/tutorials_template/out/fluid_1_6.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_7.vtu b/previews/PR514/tutorials_template/out/fluid_1_7.vtu index 64f015ca0..90862d00e 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_7.vtu and b/previews/PR514/tutorials_template/out/fluid_1_7.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_8.vtu b/previews/PR514/tutorials_template/out/fluid_1_8.vtu index f3c013b50..f88a83254 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_8.vtu and b/previews/PR514/tutorials_template/out/fluid_1_8.vtu differ diff --git a/previews/PR514/tutorials_template/out/fluid_1_9.vtu b/previews/PR514/tutorials_template/out/fluid_1_9.vtu index f676cf7ed..9ca82a4f3 100644 Binary files a/previews/PR514/tutorials_template/out/fluid_1_9.vtu and b/previews/PR514/tutorials_template/out/fluid_1_9.vtu differ diff --git a/previews/PR514/tutorials_template/tut_beam/index.html b/previews/PR514/tutorials_template/tut_beam/index.html index e7ce64fdf..d443b295e 100644 --- a/previews/PR514/tutorials_template/tut_beam/index.html +++ b/previews/PR514/tutorials_template/tut_beam/index.html @@ -1,3 +1,3 @@ Example file · TrixiParticles.jl

Example file

!!include:examples/solid/oscillating_beam_2d.jl!!
-
+ diff --git a/previews/PR514/tutorials_template/tut_dam_break/index.html b/previews/PR514/tutorials_template/tut_dam_break/index.html index c4810d980..c2c881fad 100644 --- a/previews/PR514/tutorials_template/tut_dam_break/index.html +++ b/previews/PR514/tutorials_template/tut_dam_break/index.html @@ -1,3 +1,3 @@ Example file · TrixiParticles.jl
+ diff --git a/previews/PR514/tutorials_template/tut_falling/index.html b/previews/PR514/tutorials_template/tut_falling/index.html index d7e5635ee..734e53949 100644 --- a/previews/PR514/tutorials_template/tut_falling/index.html +++ b/previews/PR514/tutorials_template/tut_falling/index.html @@ -1,3 +1,3 @@ Example file · TrixiParticles.jl

Example file

!!include:examples/fsi/falling_spheres_2d.jl!!
-
+ diff --git a/previews/PR514/tutorials_template/tut_setup/index.html b/previews/PR514/tutorials_template/tut_setup/index.html index 782df5b6b..21a588c4e 100644 --- a/previews/PR514/tutorials_template/tut_setup/index.html +++ b/previews/PR514/tutorials_template/tut_setup/index.html @@ -1,7 +1,7 @@ Setting up your simulation from scratch · TrixiParticles.jl

Setting up your simulation from scratch

In this tutorial, we will guide you through the general structure of simulation files. We will set up a simulation similar to the example simulation examples/fluid/hydrostatic_water_column_2d.jl, which is one of our simplest example simulations. In the second part of this tutorial, we will show how to replace components of TrixiParticles.jl by custom implementations from within a simulation file, without ever cloning the repository.

For different setups and physics, have a look at our other example files.

First, we import TrixiParticles.jl and OrdinaryDiffEq.jl, which we will use at the very end for the time integration.

using TrixiParticles
-using OrdinaryDiffEq

Resolution

Now, we define the particle spacing, which is our numerical resolution. We usually call the variable fluid_particle_spacing, so that we can easily change the resolution of an example file by overwriting this variable with trixi_include. In 2D, the number of particles will grow quadratically, in 3D cubically with the spacing.

We also set the number of boundary layers, which need to be sufficiently large, depending on the smoothing kernel and smoothing length, so that the compact support of the smoothing kernel is fully sampled with particles for a fluid particle close to a boundary. In particular, we require boundary_layers >= compact_support. The value for the compact support for each kernel can be found in the smoothing kernel overview.

fluid_particle_spacing = 0.05
-boundary_layers = 3

Experiment setup

We want to simulate a water column resting under hydrostatic pressure inside a rectangular tank. Experiment Setup First, we define the physical parameters gravitational acceleration, simulation time, initial fluid size, tank size and fluid density.

gravity = 9.81
+using OrdinaryDiffEq

Resolution

Now, we define the particle spacing, which is our numerical resolution. For a fluid, we usually call the variable fluid_particle_spacing, so that we can easily change the resolution of an example file by overwriting this variable with trixi_include. In 2D, the number of particles will grow quadratically, in 3D cubically with the spacing.

We also set the number of boundary layers, which need to be sufficiently large, depending on the smoothing kernel and smoothing length, so that the compact support of the smoothing kernel is fully sampled with particles for a fluid particle close to a boundary. In particular, we require the boundary thickness boundary_layers * fluid_particle_spacing to be larger than the compact support of the kernel. The compact support of each kernel can be found in the smoothing kernel overview.

fluid_particle_spacing = 0.05
+boundary_layers = 3

Experiment setup

We want to simulate a water column resting under hydrostatic pressure inside a rectangular tank. Experiment Setup First, we define physical parameters like gravitational acceleration, simulation time, initial fluid size, tank size and fluid density.

gravity = 9.81
 tspan = (0.0, 1.0)
 initial_fluid_size = (1.0, 0.9)
 tank_size = (1.0, 1.0)
@@ -10,7 +10,7 @@
                                    exponent=7)

The speed of sound here is numerical and not physical. We artificially lower the speed of sound, since the physical speed of sound in water would lead to prohibitively small time steps. The speed of sound in Weakly Compressible SPH should be chosen as small as possible for numerical efficiency, but large enough to limit density fluctuations to about 1%.

TrixiParticles.jl requires the initial particle positions and quantities in form of an InitialCondition. Instead of manually defining particle positions, you can work with our pre-defined setups. Among others, we provide setups for rectangular shapes, circles, and spheres. Initial conditions can also be combined with common set operations. See this page for a list of pre-defined setups and details on set operations on initial conditions.

Here, we use the RectangularTank setup, which generates a rectangular fluid inside a rectangular tank, and supports a hydrostatic pressure gradient by passing a gravitational acceleration and a state equation (see above).

tank = RectangularTank(fluid_particle_spacing, initial_fluid_size, tank_size,
                        fluid_density, n_layers=boundary_layers,
                        acceleration=(0.0, -gravity), state_equation=state_equation)

Fluid system

To model the water column, we use the Weakly Compressible Smoothed Particle Hydrodynamics (WCSPH) method. This method requires a smoothing kernel and a corresponding smoothing length, which should be chosen in relation to the particle spacing.

smoothing_length = 1.2 * fluid_particle_spacing
-smoothing_kernel = SchoenbergCubicSplineKernel{2}()

You can find an overview over smoothing kernels and corresponding smoothing lengths here.

For stability, we need numerical dissipation in form of an artificial viscosity term. Other viscosity models offer a physical approach based on the kinematic viscosity of the fluid.

viscosity = ArtificialViscosityMonaghan(alpha=0.02, beta=0.0)

We choose the parameters as small as possible to avoid non-physical behavior, but as large as possible to stabilize the simulation.

The WCSPH method can either compute the particle density directly with a kernel summation over all neighboring particles (see SummationDensity) or by making the particle density a variable in the ODE system and integrating its change over time. We choose the latter approach here by using the density calculator ContinuityDensity, which is more efficient and handles free surfaces without the need for additional correction terms.

fluid_density_calculator = ContinuityDensity()
+smoothing_kernel = SchoenbergCubicSplineKernel{2}()

You can find an overview over smoothing kernels and corresponding smoothing lengths here.

For stability, we need numerical dissipation in form of an artificial viscosity term. Other viscosity models offer a physical approach based on the kinematic viscosity of the fluid.

viscosity = ArtificialViscosityMonaghan(alpha=0.02, beta=0.0)

We choose the parameters as small as possible to avoid non-physical behavior, but as large as possible to stabilize the simulation.

The WCSPH method can either compute the particle density directly with a kernel summation over all neighboring particles (see SummationDensity) or by making the particle density a variable in the ODE system and integrating its change over time. We choose the latter approach here by using the density calculator ContinuityDensity, which is more efficient and handles free surfaces without the need for additional correction terms.

fluid_density_calculator = ContinuityDensity()
 fluid_system = WeaklyCompressibleSPHSystem(tank.fluid, fluid_density_calculator,
                                            state_equation, smoothing_kernel,
                                            smoothing_length, viscosity=viscosity,
@@ -18,7 +18,7 @@
                                              state_equation=state_equation,
                                              AdamiPressureExtrapolation(),
                                              smoothing_kernel, smoothing_length)
-boundary_system = BoundarySPHSystem(tank.boundary, boundary_model)

Semidiscretization

The key component of every simulation is the Semidiscretization, which couples all systems of the simulation. All methods in TrixiParticles.jl are semidiscretizations, which discretize the equations in time to provide an ordinary differential equation that still has to be solved in time. By providing a simulation time span, we can call semidiscretize, which returns an ODEProblem that can be solved with a time integration method.

semi = Semidiscretization(fluid_system, boundary_system)
+boundary_system = BoundarySPHSystem(tank.boundary, boundary_model)

Semidiscretization

The key component of every simulation is the Semidiscretization, which couples all systems of the simulation. All simulation methods in TrixiParticles.jl are semidiscretizations, which discretize the equations in time to provide an ordinary differential equation that still has to be solved in time. By providing a simulation time span, we can call semidiscretize, which returns an ODEProblem that can be solved with a time integration method.

semi = Semidiscretization(fluid_system, boundary_system)
 ode = semidiscretize(semi, tspan)

Time integration

We use the methods provided by OrdinaryDiffEq.jl, but note that other packages or custom implementations can also be used.

OrdinaryDiffEq.jl supports callbacks, which are executed during the simulation. For this simulation, we use the InfoCallback, which prints information about the simulation setup at the beginning of the simulation, information about the current simulation time and runtime during the simulation, and a performance summary at the end of the simulation. We also want to save the current solution in regular intervals in terms of simulation time as VTK, so that we can look at the solution in ParaView. The SolutionSavingCallback provides this functionality. To pass the callbacks to OrdinaryDiffEq.jl, we have to bundle them into a CallbackSet.

info_callback = InfoCallback(interval=50)
 saving_callback = SolutionSavingCallback(dt=0.02)
 
@@ -92,10 +92,10 @@
 │ #threads: ……………………………………………………… 1                                                                │
 └──────────────────────────────────────────────────────────────────────────────────────────────────┘
 
-#timesteps:     50 │ Δt: 1.4965e-03 │ sim. time: 1.8236e-01 (18.236%)  │ run time: 1.2545e-01 s
-#timesteps:    100 │ Δt: 5.3979e-03 │ sim. time: 4.2540e-01 (42.540%)  │ run time: 2.5747e-01 s
-#timesteps:    150 │ Δt: 5.7273e-03 │ sim. time: 6.7662e-01 (67.662%)  │ run time: 3.8213e-01 s
-#timesteps:    200 │ Δt: 4.8634e-03 │ sim. time: 9.1219e-01 (91.219%)  │ run time: 5.0726e-01 s
+#timesteps:     50 │ Δt: 1.4965e-03 │ sim. time: 1.8236e-01 (18.236%)  │ run time: 1.1429e-01 s
+#timesteps:    100 │ Δt: 5.3979e-03 │ sim. time: 4.2540e-01 (42.540%)  │ run time: 2.3899e-01 s
+#timesteps:    150 │ Δt: 5.7273e-03 │ sim. time: 6.7662e-01 (67.662%)  │ run time: 3.7002e-01 s
+#timesteps:    200 │ Δt: 4.8634e-03 │ sim. time: 9.1219e-01 (91.219%)  │ run time: 5.0611e-01 s
 ────────────────────────────────────────────────────────────────────────────────────────────────────
 Trixi simulation finished.  Final time: 1.0  Time steps: 222 (accepted), 222 (total)
 ────────────────────────────────────────────────────────────────────────────────────────────────────
@@ -103,42 +103,42 @@
 ────────────────────────────────────────────────────────────────────────────────
       TrixiParticles.jl                Time                    Allocations
                               ───────────────────────   ────────────────────────
-      Tot / % measured:            559ms /  98.0%           95.4MiB /  98.4%
+      Tot / % measured:            559ms /  97.9%           95.4MiB /  98.4%
 
 Section               ncalls     time    %tot     avg     alloc    %tot      avg
 ────────────────────────────────────────────────────────────────────────────────
-kick!                  1.11k    350ms   64.0%   315μs   3.77MiB    4.0%  3.47KiB
-  system interaction   1.11k    280ms   51.1%   251μs   1.05MiB    1.1%     985B
-    fluid1-fluid1      1.11k    221ms   40.3%   198μs     0.00B    0.0%    0.00B
-    fluid1-boundary2   1.11k   56.0ms   10.2%  50.3μs     0.00B    0.0%    0.00B
-    ~system intera...  1.11k   3.01ms    0.6%  2.71μs   1.05MiB    1.1%     985B
-    boundary2-boun...  1.11k   32.4μs    0.0%  29.1ns     0.00B    0.0%    0.00B
-    boundary2-fluid1   1.11k   32.3μs    0.0%  29.0ns     0.00B    0.0%    0.00B
-  update systems a...  1.11k   69.3ms   12.7%  62.3μs   2.72MiB    2.9%  2.50KiB
-    compute bounda...  1.11k   47.8ms    8.7%  42.9μs     0.00B    0.0%    0.00B
-    update nhs         1.11k   10.7ms    2.0%  9.63μs   2.72MiB    2.9%  2.50KiB
-    inverse state ...  1.11k   6.01ms    1.1%  5.40μs     0.00B    0.0%    0.00B
-    ~update system...  1.11k   4.72ms    0.9%  4.24μs   1.55KiB    0.0%    1.42B
-    update density...  1.11k   32.2μs    0.0%  28.9ns     0.00B    0.0%    0.00B
-  source terms         1.11k    415μs    0.1%   373ns     0.00B    0.0%    0.00B
-  ~kick!~              1.11k    393μs    0.1%   353ns   1.55KiB    0.0%    1.42B
-  reset ∂v/∂t          1.11k    148μs    0.0%   133ns     0.00B    0.0%    0.00B
-save solution             50    196ms   35.9%  3.93ms   90.0MiB   96.0%  1.80MiB
-  write to vtk           100    174ms   31.8%  1.74ms   86.8MiB   92.5%   888KiB
-  ~save solution~         50   19.1ms    3.5%   382μs   3.15MiB    3.4%  64.6KiB
-  update systems          50   3.01ms    0.6%  60.3μs    127KiB    0.1%  2.53KiB
-    compute bounda...     50   2.05ms    0.4%  41.1μs     0.00B    0.0%    0.00B
-    update nhs            50    479μs    0.1%  9.57μs    125KiB    0.1%  2.50KiB
-    inverse state ...     50    266μs    0.0%  5.32μs     0.00B    0.0%    0.00B
-    ~update systems~      50    214μs    0.0%  4.27μs   1.55KiB    0.0%    31.7B
-    update density...     50   1.46μs    0.0%  29.2ns     0.00B    0.0%    0.00B
-drift!                 1.11k    828μs    0.2%   744ns      976B    0.0%    0.88B
-  velocity             1.11k    496μs    0.1%   445ns     0.00B    0.0%    0.00B
-  ~drift!~             1.11k    202μs    0.0%   182ns      976B    0.0%    0.88B
-  reset ∂u/∂t          1.11k    130μs    0.0%   117ns     0.00B    0.0%    0.00B
-compute boundary p...      1   69.9μs    0.0%  69.9μs     0.00B    0.0%    0.00B
-update nhs                 1   21.6μs    0.0%  21.6μs   2.50KiB    0.0%  2.50KiB
-inverse state equa...      1   7.84μs    0.0%  7.84μs     0.00B    0.0%    0.00B
+kick!                  1.11k    350ms   63.9%   315μs   3.77MiB    4.0%  3.47KiB
+  system interaction   1.11k    280ms   51.1%   252μs   1.05MiB    1.1%     985B
+    fluid1-fluid1      1.11k    222ms   40.4%   199μs     0.00B    0.0%    0.00B
+    fluid1-boundary2   1.11k   55.4ms   10.1%  49.7μs     0.00B    0.0%    0.00B
+    ~system intera...  1.11k   3.04ms    0.6%  2.73μs   1.05MiB    1.1%     985B
+    boundary2-fluid1   1.11k   32.4μs    0.0%  29.1ns     0.00B    0.0%    0.00B
+    boundary2-boun...  1.11k   32.1μs    0.0%  28.8ns     0.00B    0.0%    0.00B
+  update systems a...  1.11k   69.1ms   12.6%  62.1μs   2.72MiB    2.9%  2.50KiB
+    compute bounda...  1.11k   47.2ms    8.6%  42.4μs     0.00B    0.0%    0.00B
+    update nhs         1.11k   11.0ms    2.0%  9.86μs   2.72MiB    2.9%  2.50KiB
+    inverse state ...  1.11k   6.07ms    1.1%  5.46μs     0.00B    0.0%    0.00B
+    ~update system...  1.11k   4.86ms    0.9%  4.37μs   1.55KiB    0.0%    1.42B
+    update density...  1.11k   35.1μs    0.0%  31.6ns     0.00B    0.0%    0.00B
+  source terms         1.11k    436μs    0.1%   392ns     0.00B    0.0%    0.00B
+  ~kick!~              1.11k    401μs    0.1%   360ns   1.55KiB    0.0%    1.42B
+  reset ∂v/∂t          1.11k    152μs    0.0%   137ns     0.00B    0.0%    0.00B
+save solution             50    197ms   35.9%  3.93ms   90.0MiB   96.0%  1.80MiB
+  write to vtk           100    174ms   31.7%  1.74ms   86.8MiB   92.5%   888KiB
+  ~save solution~         50   19.9ms    3.6%   398μs   3.15MiB    3.4%  64.6KiB
+  update systems          50   2.99ms    0.5%  59.8μs    127KiB    0.1%  2.53KiB
+    compute bounda...     50   2.02ms    0.4%  40.5μs     0.00B    0.0%    0.00B
+    update nhs            50    477μs    0.1%  9.54μs    125KiB    0.1%  2.50KiB
+    inverse state ...     50    270μs    0.0%  5.40μs     0.00B    0.0%    0.00B
+    ~update systems~      50    219μs    0.0%  4.39μs   1.55KiB    0.0%    31.7B
+    update density...     50   1.43μs    0.0%  28.6ns     0.00B    0.0%    0.00B
+drift!                 1.11k    858μs    0.2%   771ns      976B    0.0%    0.88B
+  velocity             1.11k    515μs    0.1%   463ns     0.00B    0.0%    0.00B
+  ~drift!~             1.11k    208μs    0.0%   187ns      976B    0.0%    0.88B
+  reset ∂u/∂t          1.11k    134μs    0.0%   121ns     0.00B    0.0%    0.00B
+compute boundary p...      1   67.6μs    0.0%  67.6μs     0.00B    0.0%    0.00B
+update nhs                 1   41.6μs    0.0%  41.6μs   2.50KiB    0.0%  2.50KiB
+inverse state equa...      1   41.4μs    0.0%  41.4μs     0.00B    0.0%    0.00B
 update density dif...      1   40.0ns    0.0%  40.0ns     0.00B    0.0%    0.00B
 ────────────────────────────────────────────────────────────────────────────────

See Visualization for how to visualize the solution. For the simplest visualization, we can use Plots.jl:

using Plots
 plot(sol)

plot

Replacing components with custom implementations

If we would like to use an implementation of a component that is not available in TrixiParticles.jl, we can implement it ourselves within the simulation file, without ever cloning the TrixiParticles.jl repository. A good starting point is to check out the available implementations in TrixiParticles.jl, then copy the relevant functions to the simulation file and modify them as needed.

Custom smoothing kernel

To implement a custom smoothing kernel, we define a struct extending TrixiParticles.SmoothingKernel. This abstract struct has a type parameter for the number of dimensions, which we set to 2 in this case.

struct MyGaussianKernel <: TrixiParticles.SmoothingKernel{2} end

This kernel is going to be an implementation of the Gaussian kernel with a cutoff for compact support. Note that the same kernel in a more optimized version is already implemented in TrixiParticles.jl as GaussianKernel.

In order to use our new kernel, we have to define three functions. TrixiParticles.kernel, which is the kernel function itself, TrixiParticles.kernel_deriv, which is the derivative of the kernel function, and TrixiParticles.compact_support, which defines the compact support of the kernel in relation to the smoothing length. The latter is relevant for determining the search radius of the neighborhood search.

function TrixiParticles.kernel(kernel::MyGaussianKernel, r, h)
@@ -238,26 +238,26 @@
 │ #threads: ……………………………………………………… 1                                                                │
 └──────────────────────────────────────────────────────────────────────────────────────────────────┘
 
-#timesteps:     50 │ Δt: 2.1194e-06 │ sim. time: 2.6343e-02 (2.634%)   │ run time: 3.0414e+00 s
-#timesteps:    100 │ Δt: 3.6867e-07 │ sim. time: 2.6368e-02 (2.637%)   │ run time: 3.4194e+00 s
-#timesteps:    150 │ Δt: 3.8641e-06 │ sim. time: 2.7157e-02 (2.716%)   │ run time: 3.7922e+00 s
-#timesteps:    200 │ Δt: 2.7500e-05 │ sim. time: 2.9021e-02 (2.902%)   │ run time: 4.1568e+00 s
-#timesteps:    250 │ Δt: 3.0345e-08 │ sim. time: 2.9289e-02 (2.929%)   │ run time: 4.5589e+00 s
-#timesteps:    300 │ Δt: 5.4341e-07 │ sim. time: 2.9308e-02 (2.931%)   │ run time: 4.9262e+00 s
-#timesteps:    350 │ Δt: 1.1375e-08 │ sim. time: 2.9930e-02 (2.993%)   │ run time: 5.3300e+00 s
-#timesteps:    400 │ Δt: 3.7655e-08 │ sim. time: 3.0011e-02 (3.001%)   │ run time: 5.6816e+00 s
-#timesteps:    450 │ Δt: 2.5171e-06 │ sim. time: 3.0049e-02 (3.005%)   │ run time: 6.0268e+00 s
-#timesteps:    500 │ Δt: 4.6396e-05 │ sim. time: 3.1465e-02 (3.146%)   │ run time: 6.3917e+00 s
-#timesteps:    550 │ Δt: 2.8950e-04 │ sim. time: 3.4385e-02 (3.438%)   │ run time: 6.7744e+00 s
-#timesteps:    600 │ Δt: 4.7243e-04 │ sim. time: 4.0000e-02 (4.000%)   │ run time: 7.1063e+00 s
-#timesteps:    650 │ Δt: 9.2229e-04 │ sim. time: 7.3763e-02 (7.376%)   │ run time: 7.3176e+00 s
-#timesteps:    700 │ Δt: 1.5410e-03 │ sim. time: 1.2753e-01 (12.753%)  │ run time: 7.4979e+00 s
-#timesteps:    750 │ Δt: 7.9337e-04 │ sim. time: 2.1356e-01 (21.356%)  │ run time: 7.7110e+00 s
-#timesteps:    800 │ Δt: 1.5629e-03 │ sim. time: 2.9162e-01 (29.162%)  │ run time: 7.9160e+00 s
-#timesteps:    850 │ Δt: 2.4390e-03 │ sim. time: 3.6859e-01 (36.859%)  │ run time: 8.1193e+00 s
-#timesteps:    900 │ Δt: 2.6538e-03 │ sim. time: 4.5664e-01 (45.664%)  │ run time: 8.3157e+00 s
-#timesteps:    950 │ Δt: 7.2062e-03 │ sim. time: 6.5246e-01 (65.246%)  │ run time: 8.4640e+00 s
-#timesteps:   1000 │ Δt: 6.7704e-03 │ sim. time: 9.7138e-01 (97.138%)  │ run time: 8.6036e+00 s
+#timesteps:     50 │ Δt: 2.1194e-06 │ sim. time: 2.6343e-02 (2.634%)   │ run time: 3.1610e+00 s
+#timesteps:    100 │ Δt: 3.6867e-07 │ sim. time: 2.6368e-02 (2.637%)   │ run time: 3.5367e+00 s
+#timesteps:    150 │ Δt: 3.8641e-06 │ sim. time: 2.7157e-02 (2.716%)   │ run time: 3.9070e+00 s
+#timesteps:    200 │ Δt: 2.7500e-05 │ sim. time: 2.9021e-02 (2.902%)   │ run time: 4.2688e+00 s
+#timesteps:    250 │ Δt: 3.0345e-08 │ sim. time: 2.9289e-02 (2.929%)   │ run time: 4.6677e+00 s
+#timesteps:    300 │ Δt: 5.4341e-07 │ sim. time: 2.9308e-02 (2.931%)   │ run time: 5.0316e+00 s
+#timesteps:    350 │ Δt: 1.1375e-08 │ sim. time: 2.9930e-02 (2.993%)   │ run time: 5.4327e+00 s
+#timesteps:    400 │ Δt: 3.7655e-08 │ sim. time: 3.0011e-02 (3.001%)   │ run time: 5.7824e+00 s
+#timesteps:    450 │ Δt: 2.5171e-06 │ sim. time: 3.0049e-02 (3.005%)   │ run time: 6.1259e+00 s
+#timesteps:    500 │ Δt: 4.6396e-05 │ sim. time: 3.1465e-02 (3.146%)   │ run time: 6.4877e+00 s
+#timesteps:    550 │ Δt: 2.8950e-04 │ sim. time: 3.4385e-02 (3.438%)   │ run time: 6.8678e+00 s
+#timesteps:    600 │ Δt: 4.7243e-04 │ sim. time: 4.0000e-02 (4.000%)   │ run time: 7.2051e+00 s
+#timesteps:    650 │ Δt: 9.2229e-04 │ sim. time: 7.3763e-02 (7.376%)   │ run time: 7.4157e+00 s
+#timesteps:    700 │ Δt: 1.5410e-03 │ sim. time: 1.2753e-01 (12.753%)  │ run time: 7.5960e+00 s
+#timesteps:    750 │ Δt: 7.9337e-04 │ sim. time: 2.1356e-01 (21.356%)  │ run time: 7.8101e+00 s
+#timesteps:    800 │ Δt: 1.5629e-03 │ sim. time: 2.9162e-01 (29.162%)  │ run time: 8.0303e+00 s
+#timesteps:    850 │ Δt: 2.4390e-03 │ sim. time: 3.6859e-01 (36.859%)  │ run time: 8.2275e+00 s
+#timesteps:    900 │ Δt: 2.6538e-03 │ sim. time: 4.5664e-01 (45.664%)  │ run time: 8.4261e+00 s
+#timesteps:    950 │ Δt: 7.2062e-03 │ sim. time: 6.5246e-01 (65.246%)  │ run time: 8.5792e+00 s
+#timesteps:   1000 │ Δt: 6.7704e-03 │ sim. time: 9.7138e-01 (97.138%)  │ run time: 8.7279e+00 s
 ────────────────────────────────────────────────────────────────────────────────────────────────────
 Trixi simulation finished.  Final time: 1.0  Time steps: 1003 (accepted), 1891 (total)
 ────────────────────────────────────────────────────────────────────────────────────────────────────
@@ -265,41 +265,41 @@
 ────────────────────────────────────────────────────────────────────────────────
       TrixiParticles.jl                Time                    Allocations
                               ───────────────────────   ────────────────────────
-      Tot / % measured:            8.61s /  72.5%            228MiB /  53.4%
+      Tot / % measured:            8.74s /  71.2%            228MiB /  53.4%
 
 Section               ncalls     time    %tot     avg     alloc    %tot      avg
 ────────────────────────────────────────────────────────────────────────────────
-kick!                  9.46k    5.66s   90.8%   599μs   22.6MiB   18.6%  2.45KiB
-  system interaction   9.46k    4.77s   76.4%   504μs   8.88MiB    7.3%     984B
-    fluid1-fluid1      9.46k    3.77s   60.4%   398μs     0.00B    0.0%    0.00B
-    fluid1-boundary2   9.46k    979ms   15.7%   103μs     0.00B    0.0%    0.00B
-    ~system intera...  9.46k   24.0ms    0.4%  2.53μs   8.88MiB    7.3%     984B
-    boundary2-fluid1   9.46k    278μs    0.0%  29.3ns     0.00B    0.0%    0.00B
-    boundary2-boun...  9.46k    275μs    0.0%  29.1ns     0.00B    0.0%    0.00B
-  update systems a...  9.46k    886ms   14.2%  93.7μs   13.7MiB   11.3%  1.49KiB
-    compute bounda...  9.46k    719ms   11.5%  76.1μs     0.00B    0.0%    0.00B
-    update nhs         9.46k   74.0ms    1.2%  7.83μs   13.7MiB   11.3%  1.49KiB
-    inverse state ...  9.46k   52.6ms    0.8%  5.56μs     0.00B    0.0%    0.00B
-    ~update system...  9.46k   39.6ms    0.6%  4.19μs   1.55KiB    0.0%    0.17B
+kick!                  9.46k    5.63s   90.5%   596μs   22.6MiB   18.6%  2.45KiB
+  system interaction   9.46k    4.76s   76.5%   503μs   8.88MiB    7.3%     984B
+    fluid1-fluid1      9.46k    3.76s   60.5%   398μs     0.00B    0.0%    0.00B
+    fluid1-boundary2   9.46k    971ms   15.6%   103μs     0.00B    0.0%    0.00B
+    ~system intera...  9.46k   24.7ms    0.4%  2.61μs   8.88MiB    7.3%     984B
+    boundary2-fluid1   9.46k    276μs    0.0%  29.2ns     0.00B    0.0%    0.00B
+    boundary2-boun...  9.46k    276μs    0.0%  29.1ns     0.00B    0.0%    0.00B
+  update systems a...  9.46k    868ms   13.9%  91.7μs   13.7MiB   11.3%  1.49KiB
+    compute bounda...  9.46k    699ms   11.2%  73.9μs     0.00B    0.0%    0.00B
+    update nhs         9.46k   77.3ms    1.2%  8.18μs   13.7MiB   11.3%  1.49KiB
+    inverse state ...  9.46k   50.7ms    0.8%  5.36μs     0.00B    0.0%    0.00B
+    ~update system...  9.46k   40.1ms    0.6%  4.24μs   1.55KiB    0.0%    0.17B
     update density...  9.46k    275μs    0.0%  29.1ns     0.00B    0.0%    0.00B
-  source terms         9.46k   4.19ms    0.1%   443ns     0.00B    0.0%    0.00B
-  ~kick!~              9.46k   3.18ms    0.1%   336ns   1.55KiB    0.0%    0.17B
+  source terms         9.46k   3.50ms    0.1%   370ns     0.00B    0.0%    0.00B
+  ~kick!~              9.46k   3.22ms    0.1%   341ns   1.55KiB    0.0%    0.17B
   reset ∂v/∂t          9.46k   1.22ms    0.0%   129ns     0.00B    0.0%    0.00B
-save solution             50    569ms    9.1%  11.4ms   99.2MiB   81.4%  1.98MiB
-  write to vtk           100    323ms    5.2%  3.23ms   86.9MiB   71.4%   890KiB
-  ~save solution~         50    242ms    3.9%  4.83ms   12.1MiB   10.0%   248KiB
-  update systems          50   4.48ms    0.1%  89.6μs    161KiB    0.1%  3.23KiB
-    compute bounda...     50   3.38ms    0.1%  67.6μs     0.00B    0.0%    0.00B
-    update nhs            50    631μs    0.0%  12.6μs    160KiB    0.1%  3.19KiB
-    ~update systems~      50    243μs    0.0%  4.86μs   1.55KiB    0.0%    31.7B
-    inverse state ...     50    228μs    0.0%  4.55μs     0.00B    0.0%    0.00B
-    update density...     50   1.97μs    0.0%  39.4ns     0.00B    0.0%    0.00B
-drift!                 9.46k   7.20ms    0.1%   761ns      976B    0.0%    0.10B
-  velocity             9.46k   4.19ms    0.1%   443ns     0.00B    0.0%    0.00B
-  ~drift!~             9.46k   1.75ms    0.0%   185ns      976B    0.0%    0.10B
-  reset ∂u/∂t          9.46k   1.26ms    0.0%   133ns     0.00B    0.0%    0.00B
-compute boundary p...      1    106μs    0.0%   106μs     0.00B    0.0%    0.00B
-update nhs                 1   26.4μs    0.0%  26.4μs   1.12KiB    0.0%  1.12KiB
-inverse state equa...      1   8.71μs    0.0%  8.71μs     0.00B    0.0%    0.00B
+save solution             50    581ms    9.3%  11.6ms   99.2MiB   81.4%  1.98MiB
+  write to vtk           100    320ms    5.1%  3.20ms   86.9MiB   71.3%   890KiB
+  ~save solution~         50    257ms    4.1%  5.14ms   12.1MiB   10.0%   249KiB
+  update systems          50   4.46ms    0.1%  89.1μs    161KiB    0.1%  3.23KiB
+    compute bounda...     50   3.33ms    0.1%  66.6μs     0.00B    0.0%    0.00B
+    update nhs            50    655μs    0.0%  13.1μs    160KiB    0.1%  3.19KiB
+    ~update systems~      50    248μs    0.0%  4.96μs   1.55KiB    0.0%    31.7B
+    inverse state ...     50    221μs    0.0%  4.42μs     0.00B    0.0%    0.00B
+    update density...     50   1.42μs    0.0%  28.4ns     0.00B    0.0%    0.00B
+drift!                 9.46k   6.98ms    0.1%   738ns      976B    0.0%    0.10B
+  velocity             9.46k   4.34ms    0.1%   459ns     0.00B    0.0%    0.00B
+  ~drift!~             9.46k   1.62ms    0.0%   171ns      976B    0.0%    0.10B
+  reset ∂u/∂t          9.46k   1.02ms    0.0%   108ns     0.00B    0.0%    0.00B
+compute boundary p...      1    105μs    0.0%   105μs     0.00B    0.0%    0.00B
+update nhs                 1   31.0μs    0.0%  31.0μs   1.12KiB    0.0%  1.12KiB
+inverse state equa...      1   9.31μs    0.0%  9.31μs     0.00B    0.0%    0.00B
 update density dif...      1   40.0ns    0.0%  40.0ns     0.00B    0.0%    0.00B
-────────────────────────────────────────────────────────────────────────────────
+──────────────────────────────────────────────────────────────────────────────── diff --git a/previews/PR514/visualization/index.html b/previews/PR514/visualization/index.html index 22224e685..5bdfb0655 100644 --- a/previews/PR514/visualization/index.html +++ b/previews/PR514/visualization/index.html @@ -3,6 +3,6 @@ write_meta_data=true, max_coordinates=Inf, custom_quantities...)

Convert Trixi simulation data to VTK format.

Arguments

Keywords

Example

trixi2vtk(sol.u[end], semi, 0.0, iter=1, output_directory="output", prefix="solution")
 
 # Additionally store the kinetic energy of each system as "my_custom_quantity"
-trixi2vtk(sol.u[end], semi, 0.0, iter=1, my_custom_quantity=kinetic_energy)
source
TrixiParticles.trixi2vtkMethod
trixi2vtk(coordinates; output_directory="out", prefix="", filename="coordinates",
-          custom_quantities...)

Convert coordinate data to VTK format.

Arguments

  • coordinates: Coordinates to be saved.

Keywords

  • output_directory="out": Output directory path.
  • prefix="": Prefix for the output file.
  • filename="coordinates": Name of the output file.
  • custom_quantities...: Additional custom quantities to include in the VTK output.

Returns

  • file::AbstractString: Path to the generated VTK file.
source
TrixiParticles.trixi2vtkMethod
trixi2vtk(initial_condition::InitialCondition; output_directory="out",
-          prefix="", filename="initial_condition", custom_quantities...)

Convert InitialCondition data to VTK format.

Arguments

Keywords

  • output_directory="out": Output directory path.
  • prefix="": Prefix for the output file.
  • filename="coordinates": Name of the output file.
  • custom_quantities...: Additional custom quantities to include in the VTK output.

Returns

  • file::AbstractString: Path to the generated VTK file.
source
+trixi2vtk(sol.u[end], semi, 0.0, iter=1, my_custom_quantity=kinetic_energy)source
TrixiParticles.trixi2vtkMethod
trixi2vtk(coordinates; output_directory="out", prefix="", filename="coordinates",
+          custom_quantities...)

Convert coordinate data to VTK format.

Arguments

  • coordinates: Coordinates to be saved.

Keywords

  • output_directory="out": Output directory path.
  • prefix="": Prefix for the output file.
  • filename="coordinates": Name of the output file.
  • custom_quantities...: Additional custom quantities to include in the VTK output.

Returns

  • file::AbstractString: Path to the generated VTK file.
source
TrixiParticles.trixi2vtkMethod
trixi2vtk(initial_condition::InitialCondition; output_directory="out",
+          prefix="", filename="initial_condition", custom_quantities...)

Convert InitialCondition data to VTK format.

Arguments

Keywords

  • output_directory="out": Output directory path.
  • prefix="": Prefix for the output file.
  • filename="coordinates": Name of the output file.
  • custom_quantities...: Additional custom quantities to include in the VTK output.

Returns

  • file::AbstractString: Path to the generated VTK file.
source