The sensitivity is calculated in routines results_se.c, mafillsmmain_se.c and objectivemain_se.c.
In routine results_se.c
is
determined. For geometrically nonlinear calculations (parameter NLGEOM on the
*STEP card) the unperturbed displacements leading to the internal force
correspond to the displacements at the end of the previous static step, if
any, augmented by the actual prescribed displacements. For linear geometric
calculations the unperturbed displacements correspond to zero augmented by the
actual prescribed displacements. Indeed, nonzero prescribed displacements lead
to internal forces in linear calculations. Therefore, the term
in Equation (809) can be replaced by
for linear calculations, noting that only nonzero initial displacement
boundary conditions lead to internal forces (and not any previous
displacements). The second reason why results_se.c has to be called for
linear calculations too is that the material tangent
at each integration point, which is needed in
mafillsmmain_se.c for the set up of the stiffness matrix, is also determined in
results_se.c.
Routine mafillsmmain_se.c calculates the derivative of the external forces and of the stiffness matrix (and similar matrices):
![]() |
(934) |
is calculated.
![]() |
(935) |
is determined.
![]() |
(936) |
is calculated, where is an appropriately defined scalar.
Out of computational efficiency the latter terms are calculated at the element level and assembled into a global matrix thereupon.
The last major routine, objectivemain_se.c assembles the previous information to obtain the final sensitivity. For the orientation as design variable these sensitivities are immediately stored in the .dat or the .frd file. The sensitivity for the geometry (normal directions of nodes on the external surface) as design variable, however, is kept for further postprocessing in sens_coor.c.
For the objective the total sensitivity
is written as
. So the objective function is used in the terms
and
. The routine
objectivemain_se.c is split according to the
objective function:
In general, the objective function does not have to apply to the total structure, e.g. one can define the mass of part of the structure as design variable. In that case all other elements are deactivated. This is done in routine actideacti.f. This applies to all objective functions, for which only part of the structure is included.
![]() |
(937) |
The first term on the right hand side is calculated in a similar way as for the MASS in routine objective_shapeener_dx.f. The term in brackets on the right hand side was already determined in results_se.f and mafillsmmain_se.f. Premultiplying it with the displacements from the previous static step and adding the first term yields the sensitivities (objective_shapeener_tot.f).
For a nonlinear geometric calculation Equation (810) reduces to:
![]() |
(938) |
Now,
is calculated by solving the set of
equations
. The remaining operations are similar to the
linear case.
![]() |
(939) |
The part starting with the brackets on the right hand side has been determined in mafillsmmain_se.f. Consequently,the sensitivity of the eigenfrequencies only requires the premultiplication with the eigenmodes. This is done in objective_freq.f and objective_freq_cs.f (cyclic symmetry).
For the sensitivity of the eigenmodes (only calculated for the orientation as design variable) the relevant equation is Equation (814), which can also be written as:
Assuming the sensitivity to be a linear combination of the eigenmodes:
leads to the following expressions for :
and . The latter equation results from the differentiation of the mass
normalization condition
. Indeed, one obtains:
since is symmetric and not dependent on the orientation of the
material. This means that
is orthogonal to
. Mathematically, this can also be interpreted as follows: the solution of Equation (940) can be seen
as a particular solution complemented by an arbitrary multiple of the
homogeoneous solution. The homogeneous solution is
, a particular
solution is obtained by a linear combination of the other eigenmodes, which
are orthogonal to
(since K and M are symmetric). Now, Equation
(943) expresses that the solution is orthogonal to
, so only
the particular solution with coefficients from Equation (942) remains.
The determination of
![]() |
(944) |
is straightforward and is based on the expression calculated in mafillsmmain_se.f and the sensitivity of the eigenfrequencies.
![]() |
(945) |
and requires the solution of a system of equations for each design variable. The system matrix, however, does not change, so the LU-decomposition of the matrix has only to be done once.
For the orientation as design variable the frequency sensitivities are stored in the .dat file, whereas the sensitivities of the eigenmodes and/or Green functions are stored in the .frd file (frd_sen.c, called from objectivemain_se.c). For the geometry as design variable only the frequency sensitivities are determined. They are not stored in objectivemain_se.c since they may need further postprocessing in sensi_coor.c.
![]() |
(946) |
for geometrically nonlinear calculations and
![]() |
(947) |
for geometrically linear calculations. In both cases the term in brackets on
the right hand side (let us call it ) has been calculated
before. Therefore, the complete right hand side is determined by solving
for each design variable. Since the matrix of the system does not depend
on the design variable, it has only to be LU-decomposed once.
For orientation design variables the result is transferred from the degrees of freedom to the (node,direction) representation in resultsnoddir.f and stored in the .frd file in frd_sen.c. For geometrical design variables the result is processed in objective_disp_dx.f. This is due to the fact that the displacement geometric function for geometrical design variables is defined as the square root of the sum of the square of the displacements in all design nodes. After leaving objective_disp_dx the result is kept for further postprocessing.
![]() |
(948) |
i.e. the stress S is in general a direct function of the design variables and an indirect function through the displacements. Indeed, the stress is the result of the “multiplication” of the material contants with the derivative of the displacements with respect to the geometry.
For geometrical design variables both terms
and
have to be evaluated, i.e. keeping the displacements
at the nodes constant while changing the geometry will lead to stress changes,
as well as changing the displacements while keeping the geometry constant.
For orientation design variables
since the stress
change only comes through the material law.
For geometrical design variables the stress in the unperturbed state is
calculated in resultsstr.c, while the derivative w.r.t. s is done in
stress_sen_dx.f and the derivative w.r.t. U in stress_sen_dv.f. The
calculation of Equation (810) is done exactly as explained in
Section 6.9.23 from left to right, i.e. first calculating
before continuing with the expression in
brackets.
For orientation design variables the sensivity can be written as
![]() |
(949) |
where
is approximated by
![]() |
(950) |
So the stress sensitivities require the knowledge of the displacement sensitivities. This was treated in the previous item. For orientation design variables the above operations require the routines resultsnoddir.f and resultsstr.c (and their subroutines). The results (i.e. the sensitivity of the von Mises stress at all nodes w.r.t. a change in an anisotropic orientation) are stored in the frd-file.
![]() |
(951) |
and now satisfies:
![]() |
(952) |
instead of being zero. Defining
![]() |
(953) |
one obtains for the stress sensitivity for mode :
![]() |
(954) |