Advanced Option Selection
DuctAPE has been written in an attempt to make as many of the available options exposed to the user as possible. This means that there are quite a few options to select from if not using the option convenience functions. To help the user, the majority of overarching option types are defined using the @kwdef
macro and have default values that should be reasonable in most cases. We will introduce some of the available options here that may be of common interest.
General Option Selection
In general, options are all accessed through the options
argument of the analysis function being called. Said options are passed via an Options
struct.
DuctAPE.Options
— Typestruct Options
Type containing (nearly) all the available user options.
Fields
General Options
verbose::Bool = false
: flag to print verbose statementssilence_warnings::Bool = true
: flag to silence warningsmultipoint_index::Int = [1]
: holds current index of multi-point solver (no need for user to change this usually)
Pre-processing Options
Geometry interpolation and generation options :
finterp::Interplation Method = FLOWMath.akima
: interpolation method used for re-paneling bodiesautoshiftduct::Bool = true
: flag as to whether duct geometry should be shifted based on rotor tip locationlu_decomp_flag::Bool = false
: flag indicating if panel method LHS matrix factorization was successful
paneling options
itcpshift::Float = 0.05
: factor for internal trailing edge psuedo-panel placement (default is DFDC hard-coded value)axistol::Float = 1e-15
: tolerance for how close the the axis of rotation should be considered on the axistegaptol::Float = 1e1 * eps()
: tolerance for how large of a trailing edge gap should be considered a gap
Integration Options
integration_options::IntegrationOptions type = IntegrationOptions()
: integration options
Post-processing Options
boundary_layer_options::BoundaryLayerOptions
: BoundaryLayerOptions objectwrite_outputs::AbstractArray{Bool} = [false]
: Bool for whether to write the outputs of the analysis to an external file (slow)outfile::AbstractArray{String} = ["outputs.jl"]
: External output file name (including path information) for files to writecheckoutfileexists::Bool = false
: Flag for whether to check if file exists before overwritingoutput_tuple_name::AbstractArray{String} = ["outs"]
: variable name for named tuple written to out file
Solving Options
grid_solver_options::GridSolverOptionsType = GridSolverOptions()
: elliptic grid solver optionssolver_options::SolverOptionsType = ChainSolverOptions()
: solver options
Failure Options
hard_fail::Bool = true
: flag as to whether DuctAPE should return nothing immediately after a failed initialization of the elliptic grid or a failed decomposition of the body influence matrix. If set to false, DuctAPE will attempt to return objects of the correct size, but with initialized values only.
Options are selected through the set_options
function
DuctAPE.set_options
— Functionset_options(; kwargs...)
set_options(multipoint; kwargs...)
Set the options for DuctAPE to use.
Note that the vast majority of the available options are defined through keyword arguments. See the documentation for the various option types for more information.
Arguments
multipoint::AbstractArray{OperatingPoint}
: a vector of operating points to use if running a multi-point analysis.
There are three main sub-option objects for quadrature, wake geometry solver, and aerodyanmic solver; these are explained in more detail below. In addition, there are various options for pre- and post-processing as well as miscellaneous options for things such as supressing warnings and printing verbose statements throughout the analysis, which can be seen in the docstring above.
Quadrature
There are several implementations for different quadrature approaches depending on user desires; they include:
- Gauss-Legendre quadature (default),
- Gauss-Kronrod Quadrature, and
- Romberg Quadrature methods.
The default method is Gauss-Legendre quadrature using 8 sample points for both the nominal and singular integrals. To modify the quadrature methods and settings, an IntegrationOptions
struct needs to be passed to the set_options
method.
DuctAPE.IntegrationOptions
— Typestruct IntegrationOptions
A struct used to hold the integration options for both the nominal and singular cases.
Fields
nominal::IntegrationMethod=GaussLegendre(8)
: the integration options to use for the nominal case.singular::IntegrationMethod=GaussLegendre(8)
: the integration options to use for the self-induced case.
The IntegraionOptions
type takes in two objects of type IntegrationMethod
, one for the nominal integrals, and one for the singular integrals. These methods can be mixed and matched between quadrature methods as well as settings.
For example, if one wanted to use a 10-point Gauss-Legendre method for the nominal integrals, and a order 7 Gauss-Kronrod method with an absolute tolerance of 2e-16 the following would need to be included in the set_options
call:
# set nominal options using a GaussLegendre object (which is an InterationMethod type)
# note that a convenience method is used here that takes in the number of points and
#calculates the appropriate sample locations and weights.
nominal_integration_method = DuctAPE.GaussLegendre(10)
# set singular options using a GaussKronrod object (which is an InterationMethod type)
# note that like most option structs, these are defined using @kwdef allowing the fields
#to be treated as keyword arguments.
# also note that we haven't changed the evaluation limit (default 10^7)
singular_integration_method = DuctAPE.GaussKronrod(; order=7, atol=2e-16)
# put the quadrature options together
integration_options = DuctAPE.IntegrationOptions(;
nominal=nominal_integration_method, singular=singular_integration_method
)
# example of calling the set_options function
options = DuctAPE.set_options(; integration_options=integration_options)
Elliptic Grid Solvers
As part of the pre-process, an elliptic grid defining the wake geometry is solved with a system of Poisson equations. For this solve there currently two options:
The SLOR (successive line over relaxation) is the method employed by DFDC.
Selection of solver and solver settings follows the same pattern as with the quadrature settings, in that the user must pass the appropriate GridSolverOptionsType
into the set_options
call.
For the SLOR method alone, the type is
DuctAPE.SLORGridSolverOptions
— Typestruct SLORGridSolverOptions <: GridSolverOptionsType
Options for SLOR (successive line over relaxation) elliptic grid solver.
Fields
iteration_limit::Int = 100
: maximum number of iterationsatol::Float = 1e-9
: absolute convergence toleranceconverged::AbstractArray{Bool}
= [false]iterations::AbstractArray{Int} = [0]
: iteration counter
And for the default method compatible with ImplicitAD, the type is
DuctAPE.GridSolverOptions
— Typestruct GridSolverOptions <: GridSolverOptionsType
Options for Newton elliptic grid solver.
Fields
iteration_limit::Int = 20
: maximum number of iterationsatol::Float = 3e-10
: absolute convergence tolerancealgorithm::Symbol = :newton
: algorithm to use in NLsolve.jlautodiff::Symbol = :forward
: differentiation method to use in NLsolve.jlprecondition = false
: flag to precondition with SLORprecondition_max_iterations = 3
: number of precondition iterationsconverged::AbstractArray{Bool}
= [false]iterations::AbstractArray{Int} = [0]
: iteration counterresidual_value::AbstractArray{Int} = [0]
: residual value
As an example, this is the input that would be required to use the default method with an absolute convergence tolerance of 1e-10, and also including the quadrature settings from above:
# define wake grid solver settings
wake_solve_options = DuctAPE.GridSolverOptions(; atol=1e-10)
# set all options
options = DuctAPE.set_options(;
integration_options=integration_options, grid_solver_options=wake_solve_options
)
The convergence flags default to false, and in general should be left alone as they are modified in-place in the various solves by the analysis.
Aerodynamics Solvers
There are two general types of solvers available in DuctAPE, the first is very similar to the solver in DFDC and converges a residual very similar to DFDC's. The other type is for external solvers that converge an alternate residual that is default in DuctAPE. The various solver options include:
- CSOR: the DFDC solver
- ModCSOR: modified DFDC solver for ImplicitAD compatibility
- FixedPoint.jl
- SpeedMapping.jl
- MINPACK.jl
- SIAMFANLEquations.jl
- NLsolve.jl
- SimpleNonlinearSolve.jl
Note that the CSOR, ModCSOR, FixedPoint.jl, and SpeedMapping.jl are all different fixed-point iteration solvers, MINPACK.jl and SIAMFANLEquations.jl are primarily quasi-newton solvers, and NLsolve.jl and SimpleNonlinearSolve.jl have various solver options.
DuctAPE also has some poly-algorithm solvers that employ more than one solver. The Chain Solver option is the default which starts with a fixed-point iteration, and if it doesn't converge, moves on to a quasi-, then full Newton solver until either convergence is reached, or no convergence is found. The other poly-algorithm that is available, but is less robust is the Composite Solver which partially converges with one solver, and finishes with another.
Each of the solve methods have a variety of different settings associated with them, detailed in their respective docstrings. The following example should contain all the principles required to be able to adapt to the most complex use cases.
# Define settings for NLsolve's newton method
aero_solver_options = DuctAPE.NLsolveOptions(;
algorithm=:newton,
atol=1e-10,
iteration_limit=30,
linesearch_method=LineSearches.BackTracking, #don't include parentheses on method handle
linesearch_kwargs=(; order=3, maxstep=1e6),
additional_kwargs=(; autoscale=false),
)
# set all the options
DuctAPE.set_options(;
integration_options=integration_options,
grid_solver_options=wake_solve_options,
solver_options=aero_solver_options,
)
The iterations
field (not to be confused with the iterations_limit
field) in the solver options should generally not be changed. They automatically save (in-place) the number of iterations the solver performs and can be accessed after the analysis is run.
Boundary Layer Solvers
If desired, a one-way turbulent boundary layer can be modeled, from which an approximate viscous drag can be determined. Currently, only the Head's method for turbulent boundary layer computation is working. In the case of separation, a separation drag penalty is applied based on values selected in the options.
DuctAPE.HeadsBoundaryLayerOptions
— Typestruct HeadsBoundaryLayerOptions
Fields:
model_drag::Tb=false
: flag to turn on viscous drag approximationn_steps::Int = Int(5e2)
: number of steps to use in boundary layer integrationfirst_step_size::Float = 1e-6
: size of first step in boundary layer integrationoffset::Float = 1e-3
: size of offset for (where to initialize) boundary layer integrationsolver_type::Type = DiffEq()
: type of ODE solver (RK() or DiffEq())ode::Function = RadauIIA5
: solver to use for boundary layer integration (RadauIIA5, RK4, or RK2 available)separation_criteria::Float=3.0
: value of H12 after which separation should happen.separation_allowance_upper::Int=10
: upper side allowance for how many steps ahead of the trailing edge we'll allow separation without penaltyseparation_allowance_lower::Int=10
: lower side allowance for how many steps ahead of the trailing edge we'll allow separation without penaltyseparation_penalty_upper::Float=0.2
: upper side maximum penalty value for separation (at leading edge)separation_penalty_lower::Float=0.2
: lower side maximum penalty value for separation (at leading edge)
Here is an example of a possible boundary layer option setting:
# Define Boundary Layer Settings
boundary_layer_options = DuctAPE.HeadsBoundaryLayerOptions(;
model_drag=true,
separation_penalty_upper=0.1,
separation_penalty_lower=0.1,
separation_allowance_upper=3,
separation_allowance_lower=25,
)
# set all the options
DuctAPE.set_options(;
integration_options=integration_options,
grid_solver_options=wake_solve_options,
solver_options=aero_solver_options,
boundary_layer_options=boundary_layer_options,
)
Advanced Options for Multi-point analyses
For using advanced options in multi-point analyses, there are various changes that need to be made to avoid run-time errors. Here is an example for setting options with the CSOR solver.
# number of operating points to analyze
nop = 3
options = DuctAPE.set_options(;
solver_options=DuctAPE.ModCSORSolverOptions(;
converged=fill(false, (1, nop)), # need a convergence flag for each operating point
iterations=zeros(Int, (1, nop)), # need a iteration count for each operating point
),
write_outputs=fill(false, nop), # we need to know which of the operating point outputs to write
outfile=fill("", nop), # we need to include names, even if they won't be used.
output_tuple_name=fill("outs", nop), # we need to include names, even if they won't be used.
)
If using a poly-algorithm with a multi-point solve, then each of the solvers needs to have the multiple converged
and iterations
fields for each operating point, and the overall solve type needs to have a converged
and iterations
field for each solver and each operating point.
options = DuctAPE.set_options(;
solver_options=DuctAPE.ChainSolverOptions(;
solvers=[ # vector of solvers to use in poly-algorithm
DuctAPE.NLsolveOptions(;
algorithm=:anderson,
atol=1e-12,
iteration_limit=200,
converged=fill(false, (1, nop)), # flags for each operating point
iterations=zeros(Int, (1, nop)), # counters for each operating point
),
DuctAPE.MinpackOptions(;
atol=1e-12,
iteration_limit=100,
converged=fill(false, (1, nop)),
iterations=zeros(Int, (1, nop)),
),
],
converged=fill(false, (2, nop)), # flags for each solver and each operating point
iterations=zeros(Int, (2, nop)), # counts for each solver and each operating point
),
)