MFEM  v4.4.0
Finite element discretization library
 All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Pages
tmop.cpp
Go to the documentation of this file.
1 // Copyright (c) 2010-2022, Lawrence Livermore National Security, LLC. Produced
2 // at the Lawrence Livermore National Laboratory. All Rights reserved. See files
3 // LICENSE and NOTICE for details. LLNL-CODE-806117.
4 //
5 // This file is part of the MFEM library. For more information and source code
6 // availability visit https://mfem.org.
7 //
8 // MFEM is free software; you can redistribute it and/or modify it under the
9 // terms of the BSD-3 license. We welcome feedback and contributions, see file
10 // CONTRIBUTING.md for details.
11 
12 #include "tmop.hpp"
13 #include "linearform.hpp"
14 #include "pgridfunc.hpp"
15 #include "tmop_tools.hpp"
16 #include "../general/forall.hpp"
17 
18 namespace mfem
19 {
20 
21 // Target-matrix optimization paradigm (TMOP) mesh quality metrics.
22 
24 {
25  double metric = 0.;
26  for (int i = 0; i < tmop_q_arr.Size(); i++)
27  {
28  metric += wt_arr[i]*tmop_q_arr[i]->EvalW(Jpt);
29  }
30  return metric;
31 }
32 
34  DenseMatrix &P) const
35 {
36  DenseMatrix Pt(P.Size());
37  P = 0.0;
38  for (int i = 0; i < tmop_q_arr.Size(); i++)
39  {
40  tmop_q_arr[i]->EvalP(Jpt, Pt);
41  Pt *= wt_arr[i];
42  P += Pt;
43  }
44 }
45 
47  const DenseMatrix &DS,
48  const double weight,
49  DenseMatrix &A) const
50 {
51  DenseMatrix At(A.Size());
52  for (int i = 0; i < tmop_q_arr.Size(); i++)
53  {
54  At = 0.0;
55  tmop_q_arr[i]->AssembleH(Jpt, DS, weight, At);
56  At *= wt_arr[i];
57  A += At;
58  }
59 }
60 
61 double TMOP_Metric_001::EvalW(const DenseMatrix &Jpt) const
62 {
63  ie.SetJacobian(Jpt.GetData());
64  return ie.Get_I1();
65 }
66 
68 {
69  ie.SetJacobian(Jpt.GetData());
70  P = ie.Get_dI1();
71 }
72 
74  const DenseMatrix &DS,
75  const double weight,
76  DenseMatrix &A) const
77 {
78  ie.SetJacobian(Jpt.GetData());
80  ie.Assemble_ddI1(weight, A.GetData());
81 }
82 
83 double TMOP_Metric_skew2D::EvalW(const DenseMatrix &Jpt) const
84 {
85  MFEM_VERIFY(Jtr != NULL,
86  "Requires a target Jacobian, use SetTargetJacobian().");
87 
88  DenseMatrix Jpr(2, 2);
89  Mult(Jpt, *Jtr, Jpr);
90 
91  Vector col1, col2;
92  Jpr.GetColumn(0, col1);
93  Jpr.GetColumn(1, col2);
94  double norm_prod = col1.Norml2() * col2.Norml2();
95  const double cos_Jpr = (col1 * col2) / norm_prod,
96  sin_Jpr = fabs(Jpr.Det()) / norm_prod;
97 
98  Jtr->GetColumn(0, col1);
99  Jtr->GetColumn(1, col2);
100  norm_prod = col1.Norml2() * col2.Norml2();
101  const double cos_Jtr = (col1 * col2) / norm_prod,
102  sin_Jtr = fabs(Jtr->Det()) / norm_prod;
103 
104  return 0.5 * (1.0 - cos_Jpr * cos_Jtr - sin_Jpr * sin_Jtr);
105 }
106 
107 double TMOP_Metric_skew3D::EvalW(const DenseMatrix &Jpt) const
108 {
109  MFEM_VERIFY(Jtr != NULL,
110  "Requires a target Jacobian, use SetTargetJacobian().");
111 
112  DenseMatrix Jpr(3, 3);
113  Mult(Jpt, *Jtr, Jpr);
114 
115  Vector col1, col2, col3;
116  Jpr.GetColumn(0, col1);
117  Jpr.GetColumn(1, col2);
118  Jpr.GetColumn(2, col3);
119  double norm_c1 = col1.Norml2(),
120  norm_c2 = col2.Norml2(),
121  norm_c3 = col3.Norml2();
122  double cos_Jpr_12 = (col1 * col2) / (norm_c1 * norm_c2),
123  cos_Jpr_13 = (col1 * col3) / (norm_c1 * norm_c3),
124  cos_Jpr_23 = (col2 * col3) / (norm_c2 * norm_c3);
125  double sin_Jpr_12 = std::sqrt(1.0 - cos_Jpr_12 * cos_Jpr_12),
126  sin_Jpr_13 = std::sqrt(1.0 - cos_Jpr_13 * cos_Jpr_13),
127  sin_Jpr_23 = std::sqrt(1.0 - cos_Jpr_23 * cos_Jpr_23);
128 
129  Jtr->GetColumn(0, col1);
130  Jtr->GetColumn(1, col2);
131  Jtr->GetColumn(2, col3);
132  norm_c1 = col1.Norml2();
133  norm_c2 = col2.Norml2(),
134  norm_c3 = col3.Norml2();
135  double cos_Jtr_12 = (col1 * col2) / (norm_c1 * norm_c2),
136  cos_Jtr_13 = (col1 * col3) / (norm_c1 * norm_c3),
137  cos_Jtr_23 = (col2 * col3) / (norm_c2 * norm_c3);
138  double sin_Jtr_12 = std::sqrt(1.0 - cos_Jtr_12 * cos_Jtr_12),
139  sin_Jtr_13 = std::sqrt(1.0 - cos_Jtr_13 * cos_Jtr_13),
140  sin_Jtr_23 = std::sqrt(1.0 - cos_Jtr_23 * cos_Jtr_23);
141 
142  return (3.0 - cos_Jpr_12 * cos_Jtr_12 - sin_Jpr_12 * sin_Jtr_12
143  - cos_Jpr_13 * cos_Jtr_13 - sin_Jpr_13 * sin_Jtr_13
144  - cos_Jpr_23 * cos_Jtr_23 - sin_Jpr_23 * sin_Jtr_23) / 6.0;
145 }
146 
148 {
149  MFEM_VERIFY(Jtr != NULL,
150  "Requires a target Jacobian, use SetTargetJacobian().");
151 
152  DenseMatrix Jpr(2, 2);
153  Mult(Jpt, *Jtr, Jpr);
154 
155  Vector col1, col2;
156  Jpr.GetColumn(0, col1);
157  Jpr.GetColumn(1, col2);
158  const double ratio_Jpr = col2.Norml2() / col1.Norml2();
159 
160  Jtr->GetColumn(0, col1);
161  Jtr->GetColumn(1, col2);
162  const double ratio_Jtr = col2.Norml2() / col1.Norml2();
163 
164  return 0.5 * (ratio_Jpr / ratio_Jtr + ratio_Jtr / ratio_Jpr) - 1.0;
165 }
166 
168 {
169  MFEM_VERIFY(Jtr != NULL,
170  "Requires a target Jacobian, use SetTargetJacobian().");
171 
172  DenseMatrix Jpr(3, 3);
173  Mult(Jpt, *Jtr, Jpr);
174 
175  Vector col1, col2, col3;
176  Jpr.GetColumn(0, col1);
177  Jpr.GetColumn(1, col2);
178  Jpr.GetColumn(2, col3);
179  double norm_c1 = col1.Norml2(),
180  norm_c2 = col2.Norml2(),
181  norm_c3 = col3.Norml2();
182  double ratio_Jpr_1 = norm_c1 / std::sqrt(norm_c2 * norm_c3),
183  ratio_Jpr_2 = norm_c2 / std::sqrt(norm_c1 * norm_c3),
184  ratio_Jpr_3 = norm_c3 / std::sqrt(norm_c1 * norm_c2);
185 
186  Jtr->GetColumn(0, col1);
187  Jtr->GetColumn(1, col2);
188  Jtr->GetColumn(2, col3);
189  norm_c1 = col1.Norml2();
190  norm_c2 = col2.Norml2();
191  norm_c3 = col3.Norml2();
192  double ratio_Jtr_1 = norm_c1 / std::sqrt(norm_c2 * norm_c3),
193  ratio_Jtr_2 = norm_c2 / std::sqrt(norm_c1 * norm_c3),
194  ratio_Jtr_3 = norm_c3 / std::sqrt(norm_c1 * norm_c2);
195 
196  return ( 0.5 * (ratio_Jpr_1 / ratio_Jtr_1 + ratio_Jtr_1 / ratio_Jpr_1) +
197  0.5 * (ratio_Jpr_2 / ratio_Jtr_2 + ratio_Jtr_2 / ratio_Jpr_2) +
198  0.5 * (ratio_Jpr_3 / ratio_Jtr_3 + ratio_Jtr_3 / ratio_Jpr_3) - 3.0
199  ) / 3.0;
200 }
201 
202 double TMOP_Metric_002::EvalW(const DenseMatrix &Jpt) const
203 {
204  ie.SetJacobian(Jpt.GetData());
205  return 0.5 * ie.Get_I1b() - 1.0;
206 }
207 
209 {
210  ie.SetJacobian(Jpt.GetData());
211  P.Set(0.5, ie.Get_dI1b());
212 }
213 
215  const DenseMatrix &DS,
216  const double weight,
217  DenseMatrix &A) const
218 {
219  ie.SetJacobian(Jpt.GetData());
220  ie.SetDerivativeMatrix(DS.Height(), DS.GetData());
221  ie.Assemble_ddI1b(0.5*weight, A.GetData());
222 }
223 
224 double TMOP_Metric_007::EvalW(const DenseMatrix &Jpt) const
225 {
226  // mu_7 = |J-J^{-t}|^2 = |J|^2 + |J^{-1}|^2 - 4
227  ie.SetJacobian(Jpt.GetData());
228  return ie.Get_I1()*(1. + 1./ie.Get_I2()) - 4.0;
229 }
230 
232 {
233  // P = d(I1*(1 + 1/I2)) = (1 + 1/I2) dI1 - I1/I2^2 dI2
234  ie.SetJacobian(Jpt.GetData());
235  const double I2 = ie.Get_I2();
236  Add(1. + 1./I2, ie.Get_dI1(), -ie.Get_I1()/(I2*I2), ie.Get_dI2(), P);
237 }
238 
240  const DenseMatrix &DS,
241  const double weight,
242  DenseMatrix &A) const
243 {
244  // P = d(I1*(1 + 1/I2))
245  // = (1 + 1/I2) dI1 - I1/I2^2 dI2
246  //
247  // dP = (-1/I2^2) (dI1 x dI2) + (1 + 1/I2) ddI1 -
248  // (dI2 x d(I1/I2^2)) - I1/I2^2 ddI2
249  // = (-1/I2^2) (dI1 x dI2) + (1 + 1/I2) ddI1 +
250  // (-1/I2^2) (dI2 x [dI1 - 2 I1/I2 dI2]) - I1/I2^2 ddI2
251  // = (-1/I2^2) (dI1 x dI2 + dI2 x dI1) + (1 + 1/I2) ddI1 +
252  // (2 I1/I2^3) (dI2 x dI2) - I1/I2^2 ddI2
253  ie.SetJacobian(Jpt.GetData());
254  ie.SetDerivativeMatrix(DS.Height(), DS.GetData());
255  const double c1 = 1./ie.Get_I2();
256  const double c2 = weight*c1*c1;
257  const double c3 = ie.Get_I1()*c2;
258  ie.Assemble_ddI1(weight*(1. + c1), A.GetData());
259  ie.Assemble_ddI2(-c3, A.GetData());
260  ie.Assemble_TProd(-c2, ie.Get_dI1(), ie.Get_dI2(), A.GetData());
261  ie.Assemble_TProd(2*c1*c3, ie.Get_dI2(), A.GetData());
262 }
263 
264 double TMOP_Metric_009::EvalW(const DenseMatrix &Jpt) const
265 {
266  // mu_9 = det(J)*|J-J^{-t}|^2 = I1b * (I2b^2 + 1) - 4 * I2b
267  // = (I1 - 4)*I2b + I1b
268  ie.SetJacobian(Jpt.GetData());
269  return (ie.Get_I1() - 4.0)*ie.Get_I2b() + ie.Get_I1b();
270 }
271 
273 {
274  // mu_9 = (I1 - 4)*I2b + I1b
275  // P = (I1 - 4)*dI2b + I2b*dI1 + dI1b
276  ie.SetJacobian(Jpt.GetData());
277  Add(ie.Get_I1() - 4.0, ie.Get_dI2b(), ie.Get_I2b(), ie.Get_dI1(), P);
278  P += ie.Get_dI1b();
279 }
280 
282  const DenseMatrix &DS,
283  const double weight,
284  DenseMatrix &A) const
285 {
286  // P = (I1 - 4)*dI2b + I2b*dI1 + dI1b
287  // dP = dI2b x dI1 + (I1-4)*ddI2b + dI1 x dI2b + I2b*ddI1 + ddI1b
288  // = (dI1 x dI2b + dI2b x dI1) + (I1-4)*ddI2b + I2b*ddI1 + ddI1b
289  ie.SetJacobian(Jpt.GetData());
290  ie.SetDerivativeMatrix(DS.Height(), DS.GetData());
291  ie.Assemble_TProd(weight, ie.Get_dI1(), ie.Get_dI2b(), A.GetData());
292  ie.Assemble_ddI2b(weight*(ie.Get_I1()-4.0), A.GetData());
293  ie.Assemble_ddI1(weight*ie.Get_I2b(), A.GetData());
294  ie.Assemble_ddI1b(weight, A.GetData());
295 }
296 
297 // mu_14 = |T-I|^2
298 double TMOP_Metric_014::EvalW(const DenseMatrix &Jpt) const
299 {
300  MFEM_VERIFY(Jtr != NULL,
301  "Requires a target Jacobian, use SetTargetJacobian().");
302 
303  DenseMatrix Id(2,2);
304 
305  Id(0,0) = 1; Id(0,1) = 0;
306  Id(1,0) = 0; Id(1,1) = 1;
307 
308  DenseMatrix Mat(2,2);
309  Mat = Jpt;
310  Mat.Add(-1,Id);
311  return Mat.FNorm2();
312 }
313 
314 double TMOP_Metric_022::EvalW(const DenseMatrix &Jpt) const
315 {
316  // mu_22 = (0.5*|J|^2 - det(J)) / (det(J) - tau0)
317  // = (0.5*I1 - I2b) / (I2b - tau0)
318  ie.SetJacobian(Jpt.GetData());
319  const double I2b = ie.Get_I2b();
320 
321  double d = I2b - min_detT;
322  if (d < 0.0 && min_detT == 0.0)
323  {
324  // The mesh has been untangled, but it's still possible to get negative
325  // detJ in FD calculations, as they move the nodes around with some small
326  // increments and can produce negative determinants. Thus we put a small
327  // value in the denominator. Note that here I2b < 0.
328  d = - I2b * 0.1;
329  }
330 
331  return (0.5*ie.Get_I1() - I2b) / d;
332 }
333 
335 {
336  // mu_22 = (0.5*I1 - I2b) / (I2b - tau0)
337  // P = 1/(I2b - tau0)*(0.5*dI1 - dI2b) - (0.5*I1 - I2b)/(I2b - tau0)^2*dI2b
338  // = 0.5/(I2b - tau0)*dI1 + (tau0 - 0.5*I1)/(I2b - tau0)^2*dI2b
339  ie.SetJacobian(Jpt.GetData());
340  const double c1 = 1.0/(ie.Get_I2b() - min_detT);
341  Add(c1/2, ie.Get_dI1(), (min_detT - ie.Get_I1()/2)*c1*c1, ie.Get_dI2b(), P);
342 }
343 
345  const DenseMatrix &DS,
346  const double weight,
347  DenseMatrix &A) const
348 {
349  // P = 0.5/(I2b - tau0)*dI1 + (tau0 - 0.5*I1)/(I2b - tau0)^2*dI2b
350  // dP = -0.5/(I2b - tau0)^2*(dI1 x dI2b) + 0.5/(I2b - tau0)*ddI1
351  // + (dI2b x dz) + z*ddI2b
352  //
353  // z = (tau0 - 0.5*I1)/(I2b - tau0)^2
354  // dz = -0.5/(I2b - tau0)^2*dI1 - 2*(tau0 - 0.5*I1)/(I2b - tau0)^3*dI2b
355  //
356  // dP = -0.5/(I2b - tau0)^2*(dI1 x dI2b + dI2b x dI1)
357  // -2*z/(I2b - tau0)*(dI2b x dI2b)
358  // +0.5/(I2b - tau0)*ddI1 + z*ddI2b
359  ie.SetJacobian(Jpt.GetData());
360  ie.SetDerivativeMatrix(DS.Height(), DS.GetData());
361  const double c1 = 1.0/(ie.Get_I2b() - min_detT);
362  const double c2 = weight*c1/2;
363  const double c3 = c1*c2;
364  const double c4 = (2*min_detT - ie.Get_I1())*c3; // weight*z
365  ie.Assemble_TProd(-c3, ie.Get_dI1(), ie.Get_dI2b(), A.GetData());
366  ie.Assemble_TProd(-2*c1*c4, ie.Get_dI2b(), A.GetData());
367  ie.Assemble_ddI1(c2, A.GetData());
368  ie.Assemble_ddI2b(c4, A.GetData());
369 }
370 
371 double TMOP_Metric_050::EvalW(const DenseMatrix &Jpt) const
372 {
373  // mu_50 = 0.5*|J^t J|^2/det(J)^2 - 1
374  // = 0.5*(l1^4 + l2^4)/(l1*l2)^2 - 1
375  // = 0.5*((l1/l2)^2 + (l2/l1)^2) - 1 = 0.5*(l1/l2 - l2/l1)^2
376  // = 0.5*(l1/l2 + l2/l1)^2 - 2 = 0.5*I1b^2 - 2
377  ie.SetJacobian(Jpt.GetData());
378  const double I1b = ie.Get_I1b();
379  return 0.5*I1b*I1b - 2.0;
380 }
381 
383 {
384  // mu_50 = 0.5*I1b^2 - 2
385  // P = I1b*dI1b
386  ie.SetJacobian(Jpt.GetData());
387  P.Set(ie.Get_I1b(), ie.Get_dI1b());
388 }
389 
391  const DenseMatrix &DS,
392  const double weight,
393  DenseMatrix &A) const
394 {
395  // P = I1b*dI1b
396  // dP = dI1b x dI1b + I1b*ddI1b
397  ie.SetJacobian(Jpt.GetData());
398  ie.SetDerivativeMatrix(DS.Height(), DS.GetData());
399  ie.Assemble_TProd(weight, ie.Get_dI1b(), A.GetData());
400  ie.Assemble_ddI1b(weight*ie.Get_I1b(), A.GetData());
401 }
402 
403 double TMOP_Metric_055::EvalW(const DenseMatrix &Jpt) const
404 {
405  // mu_55 = (det(J) - 1)^2 = (I2b - 1)^2
406  ie.SetJacobian(Jpt.GetData());
407  const double c1 = ie.Get_I2b() - 1.0;
408  return c1*c1;
409 }
410 
412 {
413  // mu_55 = (I2b - 1)^2
414  // P = 2*(I2b - 1)*dI2b
415  ie.SetJacobian(Jpt.GetData());
416  P.Set(2*(ie.Get_I2b() - 1.0), ie.Get_dI2b());
417 }
418 
420  const DenseMatrix &DS,
421  const double weight,
422  DenseMatrix &A) const
423 {
424  // P = 2*(I2b - 1)*dI2b
425  // dP = 2*(dI2b x dI2b) + 2*(I2b - 1)*ddI2b
426  ie.SetJacobian(Jpt.GetData());
427  ie.SetDerivativeMatrix(DS.Height(), DS.GetData());
428  ie.Assemble_TProd(2*weight, ie.Get_dI2b(), A.GetData());
429  ie.Assemble_ddI2b(2*weight*(ie.Get_I2b() - 1.0), A.GetData());
430 }
431 
432 double TMOP_Metric_056::EvalW(const DenseMatrix &Jpt) const
433 {
434  // mu_56 = 0.5*(I2b + 1/I2b) - 1
435  ie.SetJacobian(Jpt.GetData());
436  const double I2b = ie.Get_I2b();
437  return 0.5*(I2b + 1.0/I2b) - 1.0;
438 }
439 
441 {
442  // mu_56 = 0.5*(I2b + 1/I2b) - 1
443  // P = 0.5*(1 - 1/I2b^2)*dI2b
444  ie.SetJacobian(Jpt.GetData());
445  P.Set(0.5 - 0.5/ie.Get_I2(), ie.Get_dI2b());
446 }
447 
449  const DenseMatrix &DS,
450  const double weight,
451  DenseMatrix &A) const
452 {
453  // P = 0.5*(1 - 1/I2b^2)*dI2b
454  // dP = (1/I2b^3)*(dI2b x dI2b) + (0.5 - 0.5/I2)*ddI2b
455  ie.SetJacobian(Jpt.GetData());
456  ie.SetDerivativeMatrix(DS.Height(), DS.GetData());
457  ie.Assemble_TProd(weight/(ie.Get_I2()*ie.Get_I2b()),
458  ie.Get_dI2b(), A.GetData());
459  ie.Assemble_ddI2b(weight*(0.5 - 0.5/ie.Get_I2()), A.GetData());
460 }
461 
462 double TMOP_Metric_058::EvalW(const DenseMatrix &Jpt) const
463 {
464  // mu_58 = I1b*(I1b - 2)
465  ie.SetJacobian(Jpt.GetData());
466  const double I1b = ie.Get_I1b();
467  return I1b*(I1b - 1.0);
468 }
469 
471 {
472  // mu_58 = I1b*(I1b - 2)
473  // P = (2*I1b - 2)*dI1b
474  ie.SetJacobian(Jpt.GetData());
475  P.Set(2*ie.Get_I1b() - 2.0, ie.Get_dI1b());
476 }
477 
479  const DenseMatrix &DS,
480  const double weight,
481  DenseMatrix &A) const
482 {
483  // P = (2*I1b - 2)*dI1b
484  // dP = 2*(dI1b x dI1b) + (2*I1b - 2)*ddI1b
485  ie.SetJacobian(Jpt.GetData());
486  ie.SetDerivativeMatrix(DS.Height(), DS.GetData());
487  ie.Assemble_TProd(2*weight, ie.Get_dI1b(), A.GetData());
488  ie.Assemble_ddI1b(weight*(2*ie.Get_I1b() - 2.0), A.GetData());
489 }
490 
491 double TMOP_Metric_077::EvalW(const DenseMatrix &Jpt) const
492 {
493  ie.SetJacobian(Jpt.GetData());
494  const double I2b = ie.Get_I2b();
495  return 0.5*(I2b*I2b + 1./(I2b*I2b) - 2.);
496 }
497 
499 {
500  // Using I2b^2 = I2.
501  // dmu77_dJ = 1/2 (1 - 1/I2^2) dI2_dJ.
502  ie.SetJacobian(Jpt.GetData());
503  const double I2 = ie.Get_I2();
504  P.Set(0.5 * (1.0 - 1.0 / (I2 * I2)), ie.Get_dI2());
505 }
506 
508  const DenseMatrix &DS,
509  const double weight,
510  DenseMatrix &A) const
511 {
512  ie.SetJacobian(Jpt.GetData());
513  ie.SetDerivativeMatrix(DS.Height(), DS.GetData());
514  const double I2 = ie.Get_I2(), I2inv_sq = 1.0 / (I2 * I2);
515  ie.Assemble_ddI2(weight*0.5*(1.0 - I2inv_sq), A.GetData());
516  ie.Assemble_TProd(weight * I2inv_sq / I2, ie.Get_dI2(), A.GetData());
517 }
518 
519 // mu_85 = |T-T'|^2, where T'= |T|*I/sqrt(2)
520 double TMOP_Metric_085::EvalW(const DenseMatrix &Jpt) const
521 {
522  MFEM_VERIFY(Jtr != NULL,
523  "Requires a target Jacobian, use SetTargetJacobian().");
524 
525  DenseMatrix Id(2,2);
526  DenseMatrix Mat(2,2);
527  Mat = Jpt;
528 
529  Id(0,0) = 1; Id(0,1) = 0;
530  Id(1,0) = 0; Id(1,1) = 1;
531  Id *= Mat.FNorm()/pow(2,0.5);
532 
533  Mat.Add(-1.,Id);
534  return Mat.FNorm2();
535 }
536 
537 // mu_98 = 1/(tau)|T-I|^2
538 double TMOP_Metric_098::EvalW(const DenseMatrix &Jpt) const
539 {
540  MFEM_VERIFY(Jtr != NULL,
541  "Requires a target Jacobian, use SetTargetJacobian().");
542 
543  DenseMatrix Id(2,2);
544 
545  Id(0,0) = 1; Id(0,1) = 0;
546  Id(1,0) = 0; Id(1,1) = 1;
547 
548  DenseMatrix Mat(2,2);
549  Mat = Jpt;
550  Mat.Add(-1,Id);
551  return Mat.FNorm2()/Jtr->Det();
552 }
553 
554 double TMOP_Metric_211::EvalW(const DenseMatrix &Jpt) const
555 {
556  // mu_211 = (det(J) - 1)^2 - det(J) + (det(J)^2 + eps)^{1/2}
557  // = (I2b - 1)^2 - I2b + sqrt(I2b^2 + eps)
558  ie.SetJacobian(Jpt.GetData());
559  const double I2b = ie.Get_I2b();
560  return (I2b - 1.0)*(I2b - 1.0) - I2b + std::sqrt(I2b*I2b + eps);
561 }
562 
564 {
565  MFEM_ABORT("Metric not implemented yet. Use metric mu_55 instead.");
566 }
567 
569  const DenseMatrix &DS,
570  const double weight,
571  DenseMatrix &A) const
572 {
573  MFEM_ABORT("Metric not implemented yet. Use metric mu_55 instead.");
574 }
575 
576 double TMOP_Metric_252::EvalW(const DenseMatrix &Jpt) const
577 {
578  // mu_252 = 0.5*(det(J) - 1)^2 / (det(J) - tau0).
579  ie.SetJacobian(Jpt.GetData());
580  const double I2b = ie.Get_I2b();
581  return 0.5*(I2b - 1.0)*(I2b - 1.0)/(I2b - tau0);
582 }
583 
585 {
586  // mu_252 = 0.5*(det(J) - 1)^2 / (det(J) - tau0)
587  // P = (c - 0.5*c*c ) * dI2b
588  //
589  // c = (I2b - 1)/(I2b - tau0), see TMOP_Metric_352 for details
590  ie.SetJacobian(Jpt.GetData());
591  const double I2b = ie.Get_I2b();
592  const double c = (I2b - 1.0)/(I2b - tau0);
593  P.Set(c - 0.5*c*c, ie.Get_dI2b());
594 }
595 
597  const DenseMatrix &DS,
598  const double weight,
599  DenseMatrix &A) const
600 {
601  // c = (I2b - 1)/(I2b - tau0), see TMOP_Metric_352 for details
602  //
603  // P = (c - 0.5*c*c ) * dI2b
604  // dP = (1 - c)^2/(I2b - tau0)*(dI2b x dI2b) + (c - 0.5*c*c)*ddI2b
605  ie.SetJacobian(Jpt.GetData());
606  ie.SetDerivativeMatrix(DS.Height(), DS.GetData());
607  const double I2b = ie.Get_I2b();
608  const double c0 = 1.0/(I2b - tau0);
609  const double c = c0*(I2b - 1.0);
610  ie.Assemble_TProd(weight*c0*(1.0 - c)*(1.0 - c), ie.Get_dI2b(), A.GetData());
611  ie.Assemble_ddI2b(weight*(c - 0.5*c*c), A.GetData());
612 }
613 
614 double TMOP_Metric_301::EvalW(const DenseMatrix &Jpt) const
615 {
616  ie.SetJacobian(Jpt.GetData());
617  return std::sqrt(ie.Get_I1b()*ie.Get_I2b())/3. - 1.;
618 }
619 
621 {
622  // W = (1/3)*sqrt(I1b*I2b) - 1
623  // dW = (1/6)/sqrt(I1b*I2b)*[I2b*dI1b + I1b*dI2b]
624  ie.SetJacobian(Jpt.GetData());
625  const double a = 1./(6.*std::sqrt(ie.Get_I1b()*ie.Get_I2b()));
626  Add(a*ie.Get_I2b(), ie.Get_dI1b(), a*ie.Get_I1b(), ie.Get_dI2b(), P);
627 }
628 
630  const DenseMatrix &DS,
631  const double weight,
632  DenseMatrix &A) const
633 {
634  // dW = (1/6)/sqrt(I1b*I2b)*[I2b*dI1b + I1b*dI2b]
635  // dW = (1/6)*[z2*dI1b + z1*dI2b], z1 = sqrt(I1b/I2b), z2 = sqrt(I2b/I1b)
636  // ddW = (1/6)*[dI1b x dz2 + z2*ddI1b + dI2b x dz1 + z1*ddI2b]
637  //
638  // dz1 = (1/2)*sqrt(I2b/I1b) [ (1/I2b)*dI1b + (I1b/(I2b*I2b))*dI2b ]
639  // = (1/2)/sqrt(I1b*I2b) [ dI1b + (I1b/I2b)*dI2b ]
640  // dz2 = (1/2)/sqrt(I1b*I2b) [ (I2b/I1b)*dI1b + dI2b ]
641  //
642  // dI1b x dz2 + dI2b x dz1 =
643  // (1/2)/sqrt(I1b*I2b) dI1b x [ (I2b/I1b)*dI1b + dI2b ] +
644  // (1/2)/sqrt(I1b*I2b) dI2b x [ dI1b + (I1b/I2b)*dI2b ] =
645  // (1/2)/sqrt(I1b*I2b) [sqrt(I2b/I1b)*dI1b + sqrt(I1b/I2b)*dI2b] x
646  // [sqrt(I2b/I1b)*dI1b + sqrt(I1b/I2b)*dI2b] =
647  // (1/2)/sqrt(I1b*I2b) [ 6*dW x 6*dW ] =
648  // (1/2)*(I1b*I2b)^{-3/2} (I2b*dI1b + I1b*dI2b) x (I2b*dI1b + I1b*dI2b)
649  //
650  // z1 = I1b/sqrt(I1b*I2b), z2 = I2b/sqrt(I1b*I2b)
651 
652  ie.SetJacobian(Jpt.GetData());
653  ie.SetDerivativeMatrix(DS.Height(), DS.GetData());
654  double d_I1b_I2b_data[9];
655  DenseMatrix d_I1b_I2b(d_I1b_I2b_data, 3, 3);
656  Add(ie.Get_I2b(), ie.Get_dI1b(), ie.Get_I1b(), ie.Get_dI2b(), d_I1b_I2b);
657  const double I1b_I2b = ie.Get_I1b()*ie.Get_I2b();
658  const double a = weight/(6*std::sqrt(I1b_I2b));
659  ie.Assemble_ddI1b(a*ie.Get_I2b(), A.GetData());
660  ie.Assemble_ddI2b(a*ie.Get_I1b(), A.GetData());
661  ie.Assemble_TProd(a/(2*I1b_I2b), d_I1b_I2b_data, A.GetData());
662 }
663 
664 double TMOP_Metric_302::EvalW(const DenseMatrix &Jpt) const
665 {
666  // mu_2 = |J|^2 |J^{-1}|^2 / 9 - 1
667  // = (l1^2 + l2^2 + l3^3)*(l1^{-2} + l2^{-2} + l3^{-2}) / 9 - 1
668  // = I1*(l2^2*l3^2 + l1^2*l3^2 + l1^2*l2^2)/l1^2/l2^2/l3^2/9 - 1
669  // = I1*I2/det(J)^2/9 - 1 = I1b*I2b/9-1
670  ie.SetJacobian(Jpt.GetData());
671  return ie.Get_I1b()*ie.Get_I2b()/9. - 1.;
672 }
673 
675 {
676  // mu_2 = I1b*I2b/9-1
677  // P = (I1b/9)*dI2b + (I2b/9)*dI1b
678  ie.SetJacobian(Jpt.GetData());
679  Add(ie.Get_I1b()/9, ie.Get_dI2b(), ie.Get_I2b()/9, ie.Get_dI1b(), P);
680 }
681 
683  const DenseMatrix &DS,
684  const double weight,
685  DenseMatrix &A) const
686 {
687  // P = (I1b/9)*dI2b + (I2b/9)*dI1b
688  // dP = (dI2b x dI1b)/9 + (I1b/9)*ddI2b + (dI1b x dI2b)/9 + (I2b/9)*ddI1b
689  // = (dI2b x dI1b + dI1b x dI2b)/9 + (I1b/9)*ddI2b + (I2b/9)*ddI1b
690  ie.SetJacobian(Jpt.GetData());
691  ie.SetDerivativeMatrix(DS.Height(), DS.GetData());
692  const double c1 = weight/9;
693  ie.Assemble_TProd(c1, ie.Get_dI1b(), ie.Get_dI2b(), A.GetData());
694  ie.Assemble_ddI2b(c1*ie.Get_I1b(), A.GetData());
695  ie.Assemble_ddI1b(c1*ie.Get_I2b(), A.GetData());
696 }
697 
698 double TMOP_Metric_303::EvalW(const DenseMatrix &Jpt) const
699 {
700  ie.SetJacobian(Jpt.GetData());
701  return ie.Get_I1b()/3.0 - 1.0;
702 }
703 
705 {
706  ie.SetJacobian(Jpt.GetData());
707  P.Set(1./3., ie.Get_dI1b());
708 }
709 
711  const DenseMatrix &DS,
712  const double weight,
713  DenseMatrix &A) const
714 {
715  ie.SetJacobian(Jpt.GetData());
716  ie.SetDerivativeMatrix(DS.Height(), DS.GetData());
717  ie.Assemble_ddI1b(weight/3., A.GetData());
718 }
719 
720 double TMOP_Metric_311::EvalW(const DenseMatrix &Jpt) const
721 {
722  // mu_311 = (det(J) - 1)^2 - det(J) + (det(J)^2 + eps)^{1/2}
723  // = (I3b - 1)^2 - I3b + sqrt(I3b^2 + eps)
724  ie.SetJacobian(Jpt.GetData());
725  const double I3b = ie.Get_I3b();
726  return (I3b - 1.0)*(I3b - 1.0) - I3b + std::sqrt(I3b*I3b + eps);
727 }
728 
730 {
731  ie.SetJacobian(Jpt.GetData());
732  const double I3b = ie.Get_I3b();
733  const double c = 2*I3b-3+(I3b)/(std::pow((I3b*I3b+eps),0.5));
734  P.Set(c, ie.Get_dI3b());
735 }
736 
738  const DenseMatrix &DS,
739  const double weight,
740  DenseMatrix &A) const
741 {
742  ie.SetJacobian(Jpt.GetData());
743  ie.SetDerivativeMatrix(DS.Height(), DS.GetData());
744  const double I3b = ie.Get_I3b();
745  const double c0 = I3b*I3b+eps;
746  const double c1 = 2 + 1/(pow(c0,0.5)) - I3b*I3b/(pow(c0,1.5));
747  const double c2 = 2*I3b - 3 + I3b/(pow(c0,0.5));
748  ie.Assemble_TProd(weight*c1, ie.Get_dI3b(), A.GetData());
749  ie.Assemble_ddI3b(c2*weight, A.GetData());
750 }
751 
752 double TMOP_Metric_313::EvalW(const DenseMatrix &Jpt) const
753 {
754  ie.SetJacobian(Jpt.GetData());
755 
756  const double I3b = ie.Get_I3b();
757  double d = I3b - min_detT;
758  if (d < 0.0 && min_detT == 0.0)
759  {
760  // The mesh has been untangled, but it's still possible to get negative
761  // detJ in FD calculations, as they move the nodes around with some small
762  // increments and can produce negative determinants. Thus we put a small
763  // value in the denominator. Note that here I3b < 0.
764  d = - I3b * 0.1;
765  }
766 
767  const double c = std::pow(d, -2.0/3.0);
768 
769  return ie.Get_I1() * c / 3.0;
770 }
771 
773 {
774  MFEM_ABORT("Metric not implemented yet.");
775 }
776 
778  const DenseMatrix &DS,
779  const double weight,
780  DenseMatrix &A) const
781 {
782  MFEM_ABORT("Metric not implemented yet.");
783 }
784 
785 double TMOP_Metric_315::EvalW(const DenseMatrix &Jpt) const
786 {
787  // mu_315 = mu_15_3D = (det(J) - 1)^2
788  ie.SetJacobian(Jpt.GetData());
789  const double c1 = ie.Get_I3b() - 1.0;
790  return c1*c1;
791 }
792 
794 {
795  // mu_315 = (I3b - 1)^2
796  // P = 2*(I3b - 1)*dI3b
797  ie.SetJacobian(Jpt.GetData());
798  P.Set(2*(ie.Get_I3b() - 1.0), ie.Get_dI3b());
799 }
800 
802  const DenseMatrix &DS,
803  const double weight,
804  DenseMatrix &A) const
805 {
806  // P = 2*(I3b - 1)*dI3b
807  // dP = 2*(dI3b x dI3b) + 2*(I3b - 1)*ddI3b
808  ie.SetJacobian(Jpt.GetData());
809  ie.SetDerivativeMatrix(DS.Height(), DS.GetData());
810  ie.Assemble_TProd(2*weight, ie.Get_dI3b(), A.GetData());
811  ie.Assemble_ddI3b(2*weight*(ie.Get_I3b() - 1.0), A.GetData());
812 }
813 
814 double TMOP_Metric_316::EvalW(const DenseMatrix &Jpt) const
815 {
816  // mu_316 = mu_16_3D = 0.5*(I3b + 1/I3b) - 1
817  ie.SetJacobian(Jpt.GetData());
818  const double I3b = ie.Get_I3b();
819  return 0.5*(I3b + 1.0/I3b) - 1.0;
820 }
821 
823 {
824  // mu_316 = mu_16_3D = 0.5*(I3b + 1/I3b) - 1
825  // P = 0.5*(1 - 1/I3b^2)*dI3b = (0.5 - 0.5/I3)*dI3b
826  ie.SetJacobian(Jpt.GetData());
827  P.Set(0.5 - 0.5/ie.Get_I3(), ie.Get_dI3b());
828 }
829 
831  const DenseMatrix &DS,
832  const double weight,
833  DenseMatrix &A) const
834 {
835  // P = 0.5*(1 - 1/I3b^2)*dI3b = (0.5 - 0.5/I3)*dI3b
836  // dP = (1/I3b^3)*(dI3b x dI3b) + (0.5 - 0.5/I3)*ddI3b
837  ie.SetJacobian(Jpt.GetData());
838  ie.SetDerivativeMatrix(DS.Height(), DS.GetData());
839  ie.Assemble_TProd(weight/(ie.Get_I3()*ie.Get_I3b()),
840  ie.Get_dI3b(), A.GetData());
841  ie.Assemble_ddI3b(weight*(0.5 - 0.5/ie.Get_I3()), A.GetData());
842 }
843 
844 double TMOP_Metric_321::EvalW(const DenseMatrix &Jpt) const
845 {
846  // mu_321 = mu_21_3D = |J - J^{-t}|^2
847  // = |J|^2 + |J^{-1}|^2 - 6
848  // = |J|^2 + (l1^{-2} + l2^{-2} + l3^{-2}) - 6
849  // = |J|^2 + (l2^2*l3^2 + l1^2*l3^2 + l1^2*l2^2)/det(J)^2 - 6
850  // = I1 + I2/I3b^2 - 6 = I1 + I2/I3 - 6
851  ie.SetJacobian(Jpt.GetData());
852  return ie.Get_I1() + ie.Get_I2()/ie.Get_I3() - 6.0;
853 }
854 
856 {
857  // mu_321 = I1 + I2/I3b^2 - 6 = I1 + I2/I3 - 6
858  // P = dI1 + (1/I3)*dI2 - (2*I2/I3b^3)*dI3b
859  ie.SetJacobian(Jpt.GetData());
860  const double I3 = ie.Get_I3();
861  Add(1.0/I3, ie.Get_dI2(),
862  -2*ie.Get_I2()/(I3*ie.Get_I3b()), ie.Get_dI3b(), P);
863  P += ie.Get_dI1();
864 }
865 
867  const DenseMatrix &DS,
868  const double weight,
869  DenseMatrix &A) const
870 {
871  // P = dI1 + (1/I3)*dI2 - (2*I2/I3b^3)*dI3b
872  // dP = ddI1 + (-2/I3b^3)*(dI2 x dI3b) + (1/I3)*ddI2 + (dI3b x dz) + z*ddI3b
873  //
874  // z = -2*I2/I3b^3
875  // dz = (-2/I3b^3)*dI2 + (2*I2)*(3/I3b^4)*dI3b
876  //
877  // dP = ddI1 + (-2/I3b^3)*(dI2 x dI3b + dI3b x dI2) + (1/I3)*ddI2
878  // + (6*I2/I3b^4)*(dI3b x dI3b) + (-2*I2/I3b^3)*ddI3b
879  ie.SetJacobian(Jpt.GetData());
880  ie.SetDerivativeMatrix(DS.Height(), DS.GetData());
881  const double c0 = 1.0/ie.Get_I3b();
882  const double c1 = weight*c0*c0;
883  const double c2 = -2*c0*c1;
884  const double c3 = c2*ie.Get_I2();
885  ie.Assemble_ddI1(weight, A.GetData());
886  ie.Assemble_ddI2(c1, A.GetData());
887  ie.Assemble_ddI3b(c3, A.GetData());
888  ie.Assemble_TProd(c2, ie.Get_dI2(), ie.Get_dI3b(), A.GetData());
889  ie.Assemble_TProd(-3*c0*c3, ie.Get_dI3b(), A.GetData());
890 }
891 
892 double TMOP_Metric_352::EvalW(const DenseMatrix &Jpt) const
893 {
894  // mu_352 = 0.5*(det(J) - 1)^2 / (det(J) - tau0)
895  ie.SetJacobian(Jpt.GetData());
896  const double I3b = ie.Get_I3b();
897  return 0.5*(I3b - 1.0)*(I3b - 1.0)/(I3b - tau0);
898 }
899 
901 {
902  // mu_352 = 0.5*(det(J) - 1)^2 / (det(J) - tau0)
903  // P = (I3b - 1)/(I3b - tau0)*dI3b + 0.5*(I3b - 1)^2*(-1/(I3b - tau0)^2)*dI3b
904  // = [ (I3b - 1)/(I3b - tau0) - 0.5*(I3b - 1)^2/(I3b - tau0)^2 ] * dI3b
905  // = (c - 0.5*c*c) * dI3b
906  ie.SetJacobian(Jpt.GetData());
907  const double I3b = ie.Get_I3b();
908  const double c = (I3b - 1.0)/(I3b - tau0);
909  P.Set(c - 0.5*c*c, ie.Get_dI3b());
910 }
911 
913  const DenseMatrix &DS,
914  const double weight,
915  DenseMatrix &A) const
916 {
917  // c = (I3b - 1)/(I3b - tau0)
918  //
919  // P = (c - 0.5*c*c) * dI3b
920  // dP = (1 - c)*(dI3b x dc) + (c - 0.5*c*c)*ddI3b
921  //
922  // dc = 1/(I3b - tau0)*dI3b - (I3b - 1)/(I3b - tau)^2*dI3b =
923  // = (1 - c)/(I3b - tau0)*dI3b
924  //
925  // dP = (1 - c)^2/(I3b - tau0)*(dI3b x dI3b) + (c - 0.5*c*c)*ddI3b
926  ie.SetJacobian(Jpt.GetData());
927  ie.SetDerivativeMatrix(DS.Height(), DS.GetData());
928  const double I3b = ie.Get_I3b();
929  const double c0 = 1.0/(I3b - tau0);
930  const double c = c0*(I3b - 1.0);
931  ie.Assemble_TProd(weight*c0*(1.0 - c)*(1.0 - c), ie.Get_dI3b(), A.GetData());
932  ie.Assemble_ddI3b(weight*(c - 0.5*c*c), A.GetData());
933 }
934 
935 double TMOP_AMetric_011::EvalW(const DenseMatrix &Jpt) const
936 {
937  MFEM_VERIFY(Jtr != NULL,
938  "Requires a target Jacobian, use SetTargetJacobian().");
939 
940  int dim = Jpt.Size();
941 
942  DenseMatrix Jpr(dim, dim);
943  Mult(Jpt, *Jtr, Jpr);
944 
945  double alpha = Jpr.Det(),
946  omega = Jtr->Det();
947 
948  DenseMatrix AdjAt(dim), WtW(dim), WRK(dim), Jtrt(dim);
949  CalcAdjugateTranspose(Jpr, AdjAt);
950  Jtrt.Transpose(*Jtr);
951  MultAAt(Jtrt, WtW);
952  WtW *= 1./omega;
953  Mult(AdjAt, WtW, WRK);
954 
955  WRK -= Jpr;
956  WRK *= -1.;
957 
958  return (0.25/alpha)*WRK.FNorm2();
959 }
960 
961 double TMOP_AMetric_014a::EvalW(const DenseMatrix &Jpt) const
962 {
963  MFEM_VERIFY(Jtr != NULL,
964  "Requires a target Jacobian, use SetTargetJacobian().");
965 
966  int dim = Jpt.Size();
967 
968  DenseMatrix Jpr(dim, dim);
969  Mult(Jpt, *Jtr, Jpr);
970 
971  double sqalpha = pow(Jpr.Det(), 0.5),
972  sqomega = pow(Jtr->Det(), 0.5);
973 
974  return 0.5*pow(sqalpha/sqomega - sqomega/sqalpha, 2.);
975 }
976 
977 double TMOP_AMetric_036::EvalW(const DenseMatrix &Jpt) const
978 {
979  MFEM_VERIFY(Jtr != NULL,
980  "Requires a target Jacobian, use SetTargetJacobian().");
981 
982  int dim = Jpt.Size();
983 
984  DenseMatrix Jpr(dim, dim);
985  Mult(Jpt, *Jtr, Jpr); // T*W = A
986 
987  double alpha = Jpr.Det(); // det(A)
988  Jpr -= *Jtr; // A-W
989 
990  return (1./alpha)*(Jpr.FNorm2()); //(1/alpha)*(|A-W|^2)
991 }
992 
993 double TMOP_AMetric_107a::EvalW(const DenseMatrix &Jpt) const
994 {
995  MFEM_VERIFY(Jtr != NULL,
996  "Requires a target Jacobian, use SetTargetJacobian().");
997 
998  int dim = Jpt.Size();
999 
1000  DenseMatrix Jpr(dim, dim);
1001  Mult(Jpt, *Jtr, Jpr);
1002 
1003  double alpha = Jpr.Det(),
1004  aw = Jpr.FNorm()/Jtr->FNorm();
1005 
1006  DenseMatrix W = *Jtr;
1007  W *= aw;
1008  Jpr -= W;
1009 
1010  return (0.5/alpha)*Jpr.FNorm2();
1011 }
1012 
1013 
1015 {
1016  MFEM_VERIFY(nodes, "Nodes are not given!");
1017  MFEM_ASSERT(avg_volume == 0.0, "The average volume is already computed!");
1018 
1019  Mesh *mesh = nodes->FESpace()->GetMesh();
1020  const int NE = mesh->GetNE();
1022  double volume = 0.0;
1023 
1024  for (int i = 0; i < NE; i++)
1025  {
1026  mesh->GetElementTransformation(i, *nodes, &Tr);
1027  const IntegrationRule &ir =
1028  IntRules.Get(mesh->GetElementBaseGeometry(i), Tr.OrderJ());
1029  for (int j = 0; j < ir.GetNPoints(); j++)
1030  {
1031  const IntegrationPoint &ip = ir.IntPoint(j);
1032  Tr.SetIntPoint(&ip);
1033  volume += ip.weight * Tr.Weight();
1034  }
1035  }
1036 
1037  NCMesh *ncmesh = mesh->ncmesh;
1038  if (Parallel() == false)
1039  {
1040  avg_volume = (ncmesh == NULL) ?
1041  volume / NE : volume / ncmesh->GetNumRootElements();
1042 
1043  }
1044 #ifdef MFEM_USE_MPI
1045  else
1046  {
1047  double area_NE[4];
1048  area_NE[0] = volume; area_NE[1] = NE;
1049  MPI_Allreduce(area_NE, area_NE + 2, 2, MPI_DOUBLE, MPI_SUM, comm);
1050  avg_volume = (ncmesh == NULL) ?
1051  area_NE[2] / area_NE[3] : area_NE[2] / ncmesh->GetNumRootElements();
1052  }
1053 #endif
1054 }
1055 
1057  const FiniteElementSpace &fes,
1058  const IntegrationRule &ir,
1059  const Vector &xe,
1060  DenseTensor &Jtr) const
1061 {
1062  // Fallback to the 1-element method, ComputeElementTargets()
1063 
1064  // When UsesPhysicalCoordinates() == true, we assume 'xe' uses
1065  // ElementDofOrdering::LEXICOGRAPHIC iff 'fe' is a TensorFiniteElement.
1066 
1067  const Mesh *mesh = fes.GetMesh();
1068  const int NE = mesh->GetNE();
1069  // Quick return for empty processors:
1070  if (NE == 0) { return; }
1071  const int dim = mesh->Dimension();
1072  MFEM_VERIFY(mesh->GetNumGeometries(dim) <= 1,
1073  "mixed meshes are not supported");
1074  MFEM_VERIFY(!fes.IsVariableOrder(), "variable orders are not supported");
1075  const FiniteElement &fe = *fes.GetFE(0);
1076  const int sdim = fes.GetVDim();
1077  const int nvdofs = sdim*fe.GetDof();
1078  MFEM_VERIFY(!UsesPhysicalCoordinates() ||
1079  xe.Size() == NE*nvdofs, "invalid input Vector 'xe'!");
1080  const int NQ = ir.GetNPoints();
1081  const Array<int> *dof_map = nullptr;
1083  {
1084  const TensorBasisElement *tfe =
1085  dynamic_cast<const TensorBasisElement *>(&fe);
1086  if (tfe)
1087  {
1088  dof_map = &tfe->GetDofMap();
1089  if (dof_map->Size() == 0) { dof_map = nullptr; }
1090  }
1091  }
1092 
1093  Vector elfun_lex, elfun_nat;
1094  DenseTensor J;
1095  xe.HostRead();
1096  Jtr.HostWrite();
1097  if (UsesPhysicalCoordinates() && dof_map != nullptr)
1098  {
1099  elfun_nat.SetSize(nvdofs);
1100  }
1101  for (int e = 0; e < NE; e++)
1102  {
1104  {
1105  if (!dof_map)
1106  {
1107  elfun_nat.SetDataAndSize(xe.GetData()+e*nvdofs, nvdofs);
1108  }
1109  else
1110  {
1111  elfun_lex.SetDataAndSize(xe.GetData()+e*nvdofs, nvdofs);
1112  const int ndofs = fe.GetDof();
1113  for (int d = 0; d < sdim; d++)
1114  {
1115  for (int i_lex = 0; i_lex < ndofs; i_lex++)
1116  {
1117  elfun_nat[(*dof_map)[i_lex]+d*ndofs] =
1118  elfun_lex[i_lex+d*ndofs];
1119  }
1120  }
1121  }
1122  }
1123  J.UseExternalData(Jtr(e*NQ).Data(), sdim, dim, NQ);
1124  ComputeElementTargets(e, fe, ir, elfun_nat, J);
1125  }
1126 }
1127 
1129 {
1130  switch (target_type)
1131  {
1132  case IDEAL_SHAPE_UNIT_SIZE: return false;
1135  case GIVEN_SHAPE_AND_SIZE:
1136  case GIVEN_FULL: return true;
1137  default: MFEM_ABORT("TargetType not added to ContainsVolumeInfo.");
1138  }
1139  return false;
1140 }
1141 
1143  const IntegrationRule &ir,
1144  const Vector &elfun,
1145  DenseTensor &Jtr) const
1146 {
1147  MFEM_CONTRACT_VAR(elfun);
1148  MFEM_ASSERT(target_type == IDEAL_SHAPE_UNIT_SIZE || nodes != NULL, "");
1149 
1150  const FiniteElement *nfe = (target_type != IDEAL_SHAPE_UNIT_SIZE) ?
1151  nodes->FESpace()->GetFE(e_id) : NULL;
1152  const DenseMatrix &Wideal =
1154  MFEM_ASSERT(Wideal.Height() == Jtr.SizeI(), "");
1155  MFEM_ASSERT(Wideal.Width() == Jtr.SizeJ(), "");
1156 
1157  switch (target_type)
1158  {
1159  case IDEAL_SHAPE_UNIT_SIZE:
1160  {
1161  for (int i = 0; i < ir.GetNPoints(); i++) { Jtr(i) = Wideal; }
1162  break;
1163  }
1165  {
1166  if (avg_volume == 0.0) { ComputeAvgVolume(); }
1167  DenseMatrix W(Wideal.Height());
1168 
1169  NCMesh *ncmesh = nodes->FESpace()->GetMesh()->ncmesh;
1170  double el_volume = avg_volume;
1171  if (ncmesh)
1172  {
1173  el_volume = avg_volume / ncmesh->GetElementSizeReduction(e_id);
1174  }
1175 
1176  W.Set(std::pow(volume_scale * el_volume / Wideal.Det(),
1177  1./W.Height()), Wideal);
1178  for (int i = 0; i < ir.GetNPoints(); i++) { Jtr(i) = W; }
1179  break;
1180  }
1182  case GIVEN_SHAPE_AND_SIZE:
1183  {
1184  const int dim = nfe->GetDim(), dof = nfe->GetDof();
1185  MFEM_ASSERT(dim == nodes->FESpace()->GetVDim(), "");
1186  DenseMatrix dshape(dof, dim), pos(dof, dim);
1187  Array<int> xdofs(dof * dim);
1188  Vector posV(pos.Data(), dof * dim);
1189  double detW;
1190 
1191  // always initialize detW to suppress a warning:
1192  detW = (target_type == IDEAL_SHAPE_GIVEN_SIZE) ? Wideal.Det() : 0.0;
1193  nodes->FESpace()->GetElementVDofs(e_id, xdofs);
1194  nodes->GetSubVector(xdofs, posV);
1195  for (int i = 0; i < ir.GetNPoints(); i++)
1196  {
1197  nfe->CalcDShape(ir.IntPoint(i), dshape);
1198  MultAtB(pos, dshape, Jtr(i));
1200  {
1201  const double det = Jtr(i).Det();
1202  MFEM_VERIFY(det > 0.0, "The given mesh is inverted!");
1203  Jtr(i).Set(std::pow(det / detW, 1./dim), Wideal);
1204  }
1205  }
1206  break;
1207  }
1208  default:
1209  MFEM_ABORT("invalid target type!");
1210  }
1211 }
1212 
1214  const IntegrationRule &ir,
1215  const Vector &elfun,
1217  DenseTensor &dJtr) const
1218 {
1219  MFEM_CONTRACT_VAR(elfun);
1220  MFEM_ASSERT(target_type == IDEAL_SHAPE_UNIT_SIZE || nodes != NULL, "");
1221 
1222  // TODO: Compute derivative for targets with GIVEN_SHAPE or/and GIVEN_SIZE
1223  for (int i = 0; i < Tpr.GetFE()->GetDim()*ir.GetNPoints(); i++)
1224  { dJtr(i) = 0.; }
1225 }
1226 
1228  VectorCoefficient *vspec,
1229  TMOPMatrixCoefficient *mspec)
1230 {
1231  scalar_tspec = sspec;
1232  vector_tspec = vspec;
1233  matrix_tspec = mspec;
1234 }
1235 
1237  const IntegrationRule &ir,
1238  const Vector &elfun,
1239  DenseTensor &Jtr) const
1240 {
1241  DenseMatrix point_mat;
1242  point_mat.UseExternalData(elfun.GetData(), fe.GetDof(), fe.GetDim());
1243 
1244  switch (target_type)
1245  {
1246  case GIVEN_FULL:
1247  {
1248  MFEM_VERIFY(matrix_tspec != NULL,
1249  "Target type GIVEN_FULL requires a MatrixCoefficient.");
1250 
1252  Tpr.SetFE(&fe);
1253  Tpr.ElementNo = e_id;
1255  Tpr.GetPointMat().Transpose(point_mat);
1256 
1257  for (int i = 0; i < ir.GetNPoints(); i++)
1258  {
1259  const IntegrationPoint &ip = ir.IntPoint(i);
1260  Tpr.SetIntPoint(&ip);
1261  matrix_tspec->Eval(Jtr(i), Tpr, ip);
1262  }
1263  break;
1264  }
1265  default:
1266  MFEM_ABORT("Incompatible target type for analytic adaptation!");
1267  }
1268 }
1269 
1271  const Vector &elfun,
1273  DenseTensor &dJtr) const
1274 {
1275  const FiniteElement *fe = Tpr.GetFE();
1276  DenseMatrix point_mat;
1277  point_mat.UseExternalData(elfun.GetData(), fe->GetDof(), fe->GetDim());
1278 
1279  switch (target_type)
1280  {
1281  case GIVEN_FULL:
1282  {
1283  MFEM_VERIFY(matrix_tspec != NULL,
1284  "Target type GIVEN_FULL requires a TMOPMatrixCoefficient.");
1285 
1286  for (int d = 0; d < fe->GetDim(); d++)
1287  {
1288  for (int i = 0; i < ir.GetNPoints(); i++)
1289  {
1290  const IntegrationPoint &ip = ir.IntPoint(i);
1291  Tpr.SetIntPoint(&ip);
1292  DenseMatrix &dJtr_i = dJtr(i + d*ir.GetNPoints());
1293  matrix_tspec->EvalGrad(dJtr_i, Tpr, ip, d);
1294  }
1295  }
1296  break;
1297  }
1298  default:
1299  MFEM_ABORT("Incompatible target type for analytic adaptation!");
1300  }
1301 }
1302 
1303 namespace internal
1304 {
1305 
1306 // MFEM_FORALL-based copy kernel -- used by protected methods below.
1307 // Needed as a workaround for the nvcc restriction that methods with MFEM_FORALL
1308 // in them must to be public.
1309 static inline void device_copy(double *d_dest, const double *d_src, int size)
1310 {
1311  MFEM_FORALL(i, size, d_dest[i] = d_src[i];);
1312 }
1313 
1314 } // namespace internal
1315 
1316 #ifdef MFEM_USE_MPI
1318 {
1319  MFEM_VERIFY(adapt_eval, "SetAdaptivityEvaluator() has not been called!")
1320  MFEM_VERIFY(ncomp > 0, "No target specifications have been set!");
1321 
1322  ParFiniteElementSpace *ptspec_fes = t.ParFESpace();
1323 
1324  adapt_eval->SetParMetaInfo(*ptspec_fes->GetParMesh(),
1325  *ptspec_fes->FEColl(), ncomp);
1326  adapt_eval->SetInitialField(*ptspec_fes->GetMesh()->GetNodes(), tspec);
1327 
1328  tspec_sav = tspec;
1329 
1330  delete tspec_fesv;
1331  tspec_fesv = new FiniteElementSpace(ptspec_fes->GetMesh(),
1332  ptspec_fes->FEColl(), ncomp);
1333 
1334  delete ptspec_fesv;
1335  ptspec_fesv = new ParFiniteElementSpace(ptspec_fes->GetParMesh(),
1336  ptspec_fes->FEColl(), ncomp);
1337 
1338  delete tspec_pgf;
1339  tspec_pgf = new ParGridFunction(ptspec_fesv, tspec);
1340  tspec_gf = tspec_pgf;
1341 }
1342 
1344 {
1345  ptspec_fesv->Update();
1346  if (tspec_fesv)
1347  {
1348  delete tspec_fesv;
1350  ptspec_fesv->FEColl(), ncomp);
1351  }
1352  tspec_pgf->Update();
1353  tspec_gf = tspec_pgf;
1355  tspec_sav = tspec;
1356 
1358  *ptspec_fesv->FEColl(), ncomp);
1360 }
1361 
1363 {
1364  const int vdim = tspec_.FESpace()->GetVDim(),
1365  ndof = tspec_.FESpace()->GetNDofs();
1366  MFEM_VERIFY(ndof == tspec.Size()/ncomp, "Inconsistency in SetTspecAtIndex.");
1367 
1368  const auto tspec__d = tspec_.Read();
1369  auto tspec_d = tspec.ReadWrite();
1370  const int offset = idx*ndof;
1371  internal::device_copy(tspec_d + offset, tspec__d, ndof*vdim);
1373 }
1374 
1376 {
1377  if (sizeidx > -1) { SetTspecAtIndex(sizeidx, tspec_); return; }
1378  sizeidx = ncomp;
1379  SetDiscreteTargetBase(tspec_);
1381 }
1382 
1384 {
1385  if (skewidx > -1) { SetTspecAtIndex(skewidx, tspec_); return; }
1386  skewidx = ncomp;
1387  SetDiscreteTargetBase(tspec_);
1389 }
1390 
1392 {
1393  if (aspectratioidx > -1) { SetTspecAtIndex(aspectratioidx, ar); return; }
1397 }
1398 
1400 {
1401  if (orientationidx > -1) { SetTspecAtIndex(orientationidx, o); return; }
1405 }
1406 
1408 {
1409  SetParDiscreteTargetSize(tspec_);
1410 }
1411 #endif // MFEM_USE_MPI
1412 
1414 {
1415  const int vdim = tspec_.FESpace()->GetVDim(),
1416  ndof = tspec_.FESpace()->GetNDofs();
1417 
1418  ncomp += vdim;
1419 
1420  // need to append data to tspec
1421  // make a copy of tspec->tspec_temp, increase its size, and
1422  // copy data from tspec_temp -> tspec, then add new entries
1423  Vector tspec_temp = tspec;
1424  tspec.UseDevice(true);
1425  tspec_sav.UseDevice(true);
1426  tspec.SetSize(ncomp*ndof);
1427 
1428  const auto tspec_temp_d = tspec_temp.Read();
1429  auto tspec_d = tspec.ReadWrite();
1430  internal::device_copy(tspec_d, tspec_temp_d, tspec_temp.Size());
1431 
1432  const auto tspec__d = tspec_.Read();
1433  const int offset = (ncomp-vdim)*ndof;
1434  internal::device_copy(tspec_d + offset, tspec__d, ndof*vdim);
1435 }
1436 
1438 {
1439  const int vdim = tspec_.FESpace()->GetVDim(),
1440  ndof = tspec_.FESpace()->GetNDofs();
1441  MFEM_VERIFY(ndof == tspec.Size()/ncomp, "Inconsistency in SetTargetSpec.");
1442 
1443  const auto tspec__d = tspec_.Read();
1444  auto tspec_d = tspec.ReadWrite();
1445  const int offset = idx*ndof;
1446  internal::device_copy(tspec_d + offset, tspec__d, ndof*vdim);
1448 }
1449 
1451 {
1452  if (sizeidx > -1) { SetTspecAtIndex(sizeidx, tspec_); return; }
1453  sizeidx = ncomp;
1454  SetDiscreteTargetBase(tspec_);
1456 }
1457 
1459 {
1460  if (skewidx > -1) { SetTspecAtIndex(skewidx, tspec_); return; }
1461  skewidx = ncomp;
1462  SetDiscreteTargetBase(tspec_);
1464 }
1465 
1467 {
1468  if (aspectratioidx > -1) { SetTspecAtIndex(aspectratioidx, ar); return; }
1472 }
1473 
1475 {
1476  if (orientationidx > -1) { SetTspecAtIndex(orientationidx, o); return; }
1480 }
1481 
1483 {
1484  MFEM_VERIFY(adapt_eval, "SetAdaptivityEvaluator() has not been called!")
1485  MFEM_VERIFY(ncomp > 0, "No target specifications have been set!");
1486 
1487  const FiniteElementSpace *tspec_fes = t.FESpace();
1488  adapt_eval->SetSerialMetaInfo(*tspec_fes->GetMesh(),
1489  *tspec_fes->FEColl(), ncomp);
1490  adapt_eval->SetInitialField(*tspec_fes->GetMesh()->GetNodes(), tspec);
1491 
1492  tspec_sav = tspec;
1493 
1494  delete tspec_fesv;
1495  tspec_fesv = new FiniteElementSpace(tspec_fes->GetMesh(),
1496  tspec_fes->FEColl(), ncomp);
1497 
1498  delete tspec_gf;
1499  tspec_gf = new GridFunction(tspec_fesv, tspec);
1500 }
1501 
1503 {
1504  if (idx < 0) { return; }
1505  const int ndof = tspec_.FESpace()->GetNDofs(),
1506  vdim = tspec_.FESpace()->GetVDim();
1507  MFEM_VERIFY(ndof == tspec.Size()/ncomp,
1508  "Inconsistency in GetSerialDiscreteTargetSpec.");
1509 
1510  for (int i = 0; i < ndof*vdim; i++)
1511  {
1512  tspec_(i) = tspec(i + idx*ndof);
1513  }
1514 }
1515 
1517 {
1518  tspec_fesv->Update();
1519  tspec_gf->Update();
1521  tspec_sav = tspec;
1522 
1524  *tspec_fesv->FEColl(), ncomp);
1526 }
1527 
1529 {
1531 }
1532 
1533 
1535  bool use_flag)
1536 {
1537  if (use_flag && good_tspec) { return; }
1538 
1539  MFEM_VERIFY(tspec.Size() > 0, "Target specification is not set!");
1541  tspec_sav = tspec;
1542 
1543  good_tspec = use_flag;
1544 }
1545 
1547  Vector &IntData)
1548 {
1549  adapt_eval->ComputeAtNewPosition(new_x, IntData);
1550 }
1551 
1554  int dofidx, int dir,
1555  const Vector &IntData)
1556 {
1557  MFEM_VERIFY(tspec.Size() > 0, "Target specification is not set!");
1558 
1559  Array<int> dofs;
1561  const int cnt = tspec.Size()/ncomp; // dofs per scalar-field
1562 
1563  for (int i = 0; i < ncomp; i++)
1564  {
1565  tspec(dofs[dofidx]+i*cnt) = IntData(dofs[dofidx] + i*cnt + dir*cnt*ncomp);
1566  }
1567 }
1568 
1570  int dofidx)
1571 {
1572  MFEM_VERIFY(tspec.Size() > 0, "Target specification is not set!");
1573 
1574  Array<int> dofs;
1576  const int cnt = tspec.Size()/ncomp;
1577  for (int i = 0; i < ncomp; i++)
1578  {
1579  tspec(dofs[dofidx] + i*cnt) = tspec_sav(dofs[dofidx] + i*cnt);
1580  }
1581 }
1582 
1584  const IntegrationRule &intrule)
1585 {
1586  switch (target_type)
1587  {
1589  case GIVEN_SHAPE_AND_SIZE:
1590  {
1591  const int ndofs = tspec_fesv->GetFE(e_id)->GetDof(),
1592  ntspec_dofs = ndofs*ncomp;
1593 
1594  Vector tspec_vals(ntspec_dofs);
1595 
1596  Array<int> dofs;
1597  tspec_fesv->GetElementVDofs(e_id, dofs);
1598  tspec.GetSubVector(dofs, tspec_vals);
1599  DenseMatrix tr;
1600  tspec_gf->GetVectorValues(e_id, intrule, tspec_refine, tr);
1602  break;
1603  }
1604  default:
1605  MFEM_ABORT("Incompatible target type for discrete adaptation!");
1606  }
1607 }
1608 
1610 {
1611  coarse_tspec_fesv = fes;
1612  const Operator *c_op = fes->GetUpdateOperator();
1613  tspec_derefine.SetSize(c_op->Height());
1614  c_op->Mult(tspec, tspec_derefine);
1615 }
1616 
1618  const IntegrationRule &ir,
1619  const Vector &elfun,
1620  DenseTensor &Jtr) const
1621 {
1622  MFEM_VERIFY(tspec_fesv, "No target specifications have been set.");
1623  const int dim = fe.GetDim(),
1624  nqp = ir.GetNPoints();
1625  Jtrcomp.SetSize(dim, dim, 4*nqp);
1626 
1627  FiniteElementSpace *src_fes = tspec_fesv;
1628 
1629  switch (target_type)
1630  {
1632  case GIVEN_SHAPE_AND_SIZE:
1633  {
1634  const DenseMatrix &Wideal =
1636  const int ndofs = tspec_fesv->GetFE(e_id)->GetDof(),
1637  ntspec_dofs = ndofs*ncomp;
1638 
1639  Vector shape(ndofs), tspec_vals(ntspec_dofs), par_vals,
1640  par_vals_c1, par_vals_c2, par_vals_c3;
1641 
1642  Array<int> dofs;
1643  DenseMatrix D_rho(dim), Q_phi(dim), R_theta(dim);
1644  tspec_fesv->GetElementVDofs(e_id, dofs);
1645  tspec.UseDevice(true);
1646  tspec.GetSubVector(dofs, tspec_vals);
1647  if (tspec_refine.NumCols() > 0) // Refinement
1648  {
1649  MFEM_VERIFY(amr_el >= 0, " Target being constructed for an AMR element.");
1650  for (int i = 0; i < ncomp; i++)
1651  {
1652  for (int j = 0; j < ndofs; j++)
1653  {
1654  tspec_vals(j + i*ndofs) = tspec_refine(j + amr_el*ndofs, i);
1655  }
1656  }
1657  }
1658  else if (tspec_derefine.Size() > 0) // Derefinement
1659  {
1660  dofs.SetSize(0);
1661  coarse_tspec_fesv->GetElementVDofs(e_id, dofs);
1662  tspec_derefine.GetSubVector(dofs, tspec_vals);
1663  src_fes = coarse_tspec_fesv;
1664  }
1665 
1666  for (int q = 0; q < nqp; q++)
1667  {
1668  const IntegrationPoint &ip = ir.IntPoint(q);
1669  src_fes->GetFE(e_id)->CalcShape(ip, shape);
1670  Jtr(q) = Wideal; // Initialize to identity
1671  for (int d = 0; d < 4; d++)
1672  {
1673  DenseMatrix Jtrcomp_q(Jtrcomp.GetData(d + 4*q), dim, dim);
1674  Jtrcomp_q = Wideal; // Initialize to identity
1675  }
1676 
1677  if (sizeidx != -1) // Set size
1678  {
1679  par_vals.SetDataAndSize(tspec_vals.GetData()+sizeidx*ndofs, ndofs);
1680  double min_size = par_vals.Min();//0.001; //
1681  if (lim_min_size > 0.)
1682  {
1683  min_size = lim_min_size;
1684  }
1685  else
1686  {
1687  MFEM_VERIFY(min_size > 0.0,
1688  "Non-positive size propagated in the target definition.");
1689  }
1690  const double size = std::max(shape * par_vals, min_size);
1691  Jtr(q).Set(std::pow(size, 1.0/dim), Jtr(q));
1692  DenseMatrix Jtrcomp_q(Jtrcomp.GetData(0 + 4*q), dim, dim);
1693  Jtrcomp_q = Jtr(q);
1694  } // Done size
1695 
1696  if (target_type == IDEAL_SHAPE_GIVEN_SIZE) { continue; }
1697 
1698  if (aspectratioidx != -1) // Set aspect ratio
1699  {
1700  if (dim == 2)
1701  {
1702  par_vals.SetDataAndSize(tspec_vals.GetData()+
1703  aspectratioidx*ndofs, ndofs);
1704  const double min_size = par_vals.Min();
1705  MFEM_VERIFY(min_size > 0.0,
1706  "Non-positive aspect-ratio propagated in the target definition.");
1707 
1708  const double aspectratio = shape * par_vals;
1709  D_rho = 0.;
1710  D_rho(0,0) = 1./pow(aspectratio,0.5);
1711  D_rho(1,1) = pow(aspectratio,0.5);
1712  }
1713  else
1714  {
1715  par_vals.SetDataAndSize(tspec_vals.GetData()+
1716  aspectratioidx*ndofs, ndofs*3);
1717  par_vals_c1.SetDataAndSize(par_vals.GetData(), ndofs);
1718  par_vals_c2.SetDataAndSize(par_vals.GetData()+ndofs, ndofs);
1719  par_vals_c3.SetDataAndSize(par_vals.GetData()+2*ndofs, ndofs);
1720 
1721  const double rho1 = shape * par_vals_c1;
1722  const double rho2 = shape * par_vals_c2;
1723  const double rho3 = shape * par_vals_c3;
1724  D_rho = 0.;
1725  D_rho(0,0) = pow(rho1,2./3.);
1726  D_rho(1,1) = pow(rho2,2./3.);
1727  D_rho(2,2) = pow(rho3,2./3.);
1728  }
1729  DenseMatrix Jtrcomp_q(Jtrcomp.GetData(1 + 4*q), dim, dim);
1730  Jtrcomp_q = D_rho;
1731  DenseMatrix Temp = Jtr(q);
1732  Mult(D_rho, Temp, Jtr(q));
1733  } // Done aspect ratio
1734 
1735  if (skewidx != -1) // Set skew
1736  {
1737  if (dim == 2)
1738  {
1739  par_vals.SetDataAndSize(tspec_vals.GetData()+
1740  skewidx*ndofs, ndofs);
1741 
1742  const double skew = shape * par_vals;
1743 
1744  Q_phi = 0.;
1745  Q_phi(0,0) = 1.;
1746  Q_phi(0,1) = cos(skew);
1747  Q_phi(1,1) = sin(skew);
1748  }
1749  else
1750  {
1751  par_vals.SetDataAndSize(tspec_vals.GetData()+
1752  skewidx*ndofs, ndofs*3);
1753  par_vals_c1.SetDataAndSize(par_vals.GetData(), ndofs);
1754  par_vals_c2.SetDataAndSize(par_vals.GetData()+ndofs, ndofs);
1755  par_vals_c3.SetDataAndSize(par_vals.GetData()+2*ndofs, ndofs);
1756 
1757  const double phi12 = shape * par_vals_c1;
1758  const double phi13 = shape * par_vals_c2;
1759  const double chi = shape * par_vals_c3;
1760 
1761  Q_phi = 0.;
1762  Q_phi(0,0) = 1.;
1763  Q_phi(0,1) = cos(phi12);
1764  Q_phi(0,2) = cos(phi13);
1765 
1766  Q_phi(1,1) = sin(phi12);
1767  Q_phi(1,2) = sin(phi13)*cos(chi);
1768 
1769  Q_phi(2,2) = sin(phi13)*sin(chi);
1770  }
1771  DenseMatrix Jtrcomp_q(Jtrcomp.GetData(2 + 4*q), dim, dim);
1772  Jtrcomp_q = Q_phi;
1773  DenseMatrix Temp = Jtr(q);
1774  Mult(Q_phi, Temp, Jtr(q));
1775  } // Done skew
1776 
1777  if (orientationidx != -1) // Set orientation
1778  {
1779  if (dim == 2)
1780  {
1781  par_vals.SetDataAndSize(tspec_vals.GetData()+
1782  orientationidx*ndofs, ndofs);
1783 
1784  const double theta = shape * par_vals;
1785  R_theta(0,0) = cos(theta);
1786  R_theta(0,1) = -sin(theta);
1787  R_theta(1,0) = sin(theta);
1788  R_theta(1,1) = cos(theta);
1789  }
1790  else
1791  {
1792  par_vals.SetDataAndSize(tspec_vals.GetData()+
1793  orientationidx*ndofs, ndofs*3);
1794  par_vals_c1.SetDataAndSize(par_vals.GetData(), ndofs);
1795  par_vals_c2.SetDataAndSize(par_vals.GetData()+ndofs, ndofs);
1796  par_vals_c3.SetDataAndSize(par_vals.GetData()+2*ndofs, ndofs);
1797 
1798  const double theta = shape * par_vals_c1;
1799  const double psi = shape * par_vals_c2;
1800  const double beta = shape * par_vals_c3;
1801 
1802  double ct = cos(theta), st = sin(theta),
1803  cp = cos(psi), sp = sin(psi),
1804  cb = cos(beta), sb = sin(beta);
1805 
1806  R_theta = 0.;
1807  R_theta(0,0) = ct*sp;
1808  R_theta(1,0) = st*sp;
1809  R_theta(2,0) = cp;
1810 
1811  R_theta(0,1) = -st*cb + ct*cp*sb;
1812  R_theta(1,1) = ct*cb + st*cp*sb;
1813  R_theta(2,1) = -sp*sb;
1814 
1815  R_theta(0,0) = -st*sb - ct*cp*cb;
1816  R_theta(1,0) = ct*sb - st*cp*cb;
1817  R_theta(2,0) = sp*cb;
1818  }
1819  DenseMatrix Jtrcomp_q(Jtrcomp.GetData(3 + 4*q), dim, dim);
1820  Jtrcomp_q = R_theta;
1821  DenseMatrix Temp = Jtr(q);
1822  Mult(R_theta, Temp, Jtr(q));
1823  } // Done orientation
1824  }
1825  break;
1826  }
1827  default:
1828  MFEM_ABORT("Incompatible target type for discrete adaptation!");
1829  }
1830 }
1831 
1833  const Vector &elfun,
1835  DenseTensor &dJtr) const
1836 {
1837  MFEM_ASSERT(target_type == IDEAL_SHAPE_UNIT_SIZE || nodes != NULL, "");
1838 
1839  MFEM_VERIFY(tspec_fesv, "No target specifications have been set.");
1840 
1841  dJtr = 0.;
1842  const int e_id = Tpr.ElementNo;
1843  const FiniteElement *fe = Tpr.GetFE();
1844 
1845  switch (target_type)
1846  {
1848  case GIVEN_SHAPE_AND_SIZE:
1849  {
1850  const DenseMatrix &Wideal =
1852  const int dim = Wideal.Height(),
1853  ndofs = fe->GetDof(),
1854  ntspec_dofs = ndofs*ncomp;
1855 
1856  Vector shape(ndofs), tspec_vals(ntspec_dofs), par_vals,
1857  par_vals_c1(ndofs), par_vals_c2(ndofs), par_vals_c3(ndofs);
1858 
1859  Array<int> dofs;
1860  DenseMatrix dD_rho(dim), dQ_phi(dim), dR_theta(dim);
1861  DenseMatrix dQ_phi13(dim), dQ_phichi(dim); // dQ_phi is used for dQ/dphi12 in 3D
1862  DenseMatrix dR_psi(dim), dR_beta(dim);
1863  tspec_fesv->GetElementVDofs(e_id, dofs);
1864  tspec.GetSubVector(dofs, tspec_vals);
1865 
1866  DenseMatrix grad_e_c1(ndofs, dim),
1867  grad_e_c2(ndofs, dim),
1868  grad_e_c3(ndofs, dim);
1869  Vector grad_ptr_c1(grad_e_c1.GetData(), ndofs*dim),
1870  grad_ptr_c2(grad_e_c2.GetData(), ndofs*dim),
1871  grad_ptr_c3(grad_e_c3.GetData(), ndofs*dim);
1872 
1873  DenseMatrix grad_phys; // This will be (dof x dim, dof).
1874  fe->ProjectGrad(*fe, Tpr, grad_phys);
1875 
1876  for (int i = 0; i < ir.GetNPoints(); i++)
1877  {
1878  const IntegrationPoint &ip = ir.IntPoint(i);
1879  DenseMatrix Jtrcomp_s(Jtrcomp.GetData(0 + 4*i), dim, dim); // size
1880  DenseMatrix Jtrcomp_d(Jtrcomp.GetData(1 + 4*i), dim, dim); // aspect-ratio
1881  DenseMatrix Jtrcomp_q(Jtrcomp.GetData(2 + 4*i), dim, dim); // skew
1882  DenseMatrix Jtrcomp_r(Jtrcomp.GetData(3 + 4*i), dim, dim); // orientation
1883  DenseMatrix work1(dim), work2(dim), work3(dim);
1884 
1885  if (sizeidx != -1) // Set size
1886  {
1887  par_vals.SetDataAndSize(tspec_vals.GetData()+sizeidx*ndofs, ndofs);
1888 
1889  grad_phys.Mult(par_vals, grad_ptr_c1);
1890  Vector grad_q(dim);
1891  tspec_fesv->GetFE(e_id)->CalcShape(ip, shape);
1892  grad_e_c1.MultTranspose(shape, grad_q);
1893 
1894  const double min_size = par_vals.Min();
1895  MFEM_VERIFY(min_size > 0.0,
1896  "Non-positive size propagated in the target definition.");
1897  const double size = std::max(shape * par_vals, min_size);
1898  double dz_dsize = (1./dim)*pow(size, 1./dim - 1.);
1899 
1900  Mult(Jtrcomp_q, Jtrcomp_d, work1); // Q*D
1901  Mult(Jtrcomp_r, work1, work2); // R*Q*D
1902 
1903  for (int d = 0; d < dim; d++)
1904  {
1905  DenseMatrix &dJtr_i = dJtr(i + d*ir.GetNPoints());
1906  work1 = Wideal;
1907  work1.Set(dz_dsize, work1); // dz/dsize
1908  work1 *= grad_q(d); // dz/dsize*dsize/dx
1909  AddMult(work1, work2, dJtr_i); // dz/dx*R*Q*D
1910  }
1911  } // Done size
1912 
1913  if (target_type == IDEAL_SHAPE_GIVEN_SIZE) { continue; }
1914 
1915  if (aspectratioidx != -1) // Set aspect ratio
1916  {
1917  if (dim == 2)
1918  {
1919  par_vals.SetDataAndSize(tspec_vals.GetData()+
1920  aspectratioidx*ndofs, ndofs);
1921 
1922  grad_phys.Mult(par_vals, grad_ptr_c1);
1923  Vector grad_q(dim);
1924  tspec_fesv->GetFE(e_id)->CalcShape(ip, shape);
1925  grad_e_c1.MultTranspose(shape, grad_q);
1926 
1927  const double aspectratio = shape * par_vals;
1928  dD_rho = 0.;
1929  dD_rho(0,0) = -0.5*pow(aspectratio,-1.5);
1930  dD_rho(1,1) = 0.5*pow(aspectratio,-0.5);
1931 
1932  Mult(Jtrcomp_s, Jtrcomp_r, work1); // z*R
1933  Mult(work1, Jtrcomp_q, work2); // z*R*Q
1934 
1935  for (int d = 0; d < dim; d++)
1936  {
1937  DenseMatrix &dJtr_i = dJtr(i + d*ir.GetNPoints());
1938  work1 = dD_rho;
1939  work1 *= grad_q(d); // work1 = dD/drho*drho/dx
1940  AddMult(work2, work1, dJtr_i); // z*R*Q*dD/dx
1941  }
1942  }
1943  else // 3D
1944  {
1945  par_vals.SetDataAndSize(tspec_vals.GetData()+
1946  aspectratioidx*ndofs, ndofs*3);
1947  par_vals_c1.SetData(par_vals.GetData());
1948  par_vals_c2.SetData(par_vals.GetData()+ndofs);
1949  par_vals_c3.SetData(par_vals.GetData()+2*ndofs);
1950 
1951  grad_phys.Mult(par_vals_c1, grad_ptr_c1);
1952  grad_phys.Mult(par_vals_c2, grad_ptr_c2);
1953  grad_phys.Mult(par_vals_c3, grad_ptr_c3);
1954  Vector grad_q1(dim), grad_q2(dim), grad_q3(dim);
1955  tspec_fesv->GetFE(e_id)->CalcShape(ip, shape);
1956  grad_e_c1.MultTranspose(shape, grad_q1);
1957  grad_e_c2.MultTranspose(shape, grad_q2);
1958  grad_e_c3.MultTranspose(shape, grad_q3);
1959 
1960  const double rho1 = shape * par_vals_c1;
1961  const double rho2 = shape * par_vals_c2;
1962  const double rho3 = shape * par_vals_c3;
1963  dD_rho = 0.;
1964  dD_rho(0,0) = (2./3.)*pow(rho1,-1./3.);
1965  dD_rho(1,1) = (2./3.)*pow(rho2,-1./3.);
1966  dD_rho(2,2) = (2./3.)*pow(rho3,-1./3.);
1967 
1968  Mult(Jtrcomp_s, Jtrcomp_r, work1); // z*R
1969  Mult(work1, Jtrcomp_q, work2); // z*R*Q
1970 
1971 
1972  for (int d = 0; d < dim; d++)
1973  {
1974  DenseMatrix &dJtr_i = dJtr(i + d*ir.GetNPoints());
1975  work1 = dD_rho;
1976  work1(0,0) *= grad_q1(d);
1977  work1(1,2) *= grad_q2(d);
1978  work1(2,2) *= grad_q3(d);
1979  // work1 = dD/dx = dD/drho1*drho1/dx + dD/drho2*drho2/dx
1980  AddMult(work2, work1, dJtr_i); // z*R*Q*dD/dx
1981  }
1982  }
1983  } // Done aspect ratio
1984 
1985  if (skewidx != -1) // Set skew
1986  {
1987  if (dim == 2)
1988  {
1989  par_vals.SetDataAndSize(tspec_vals.GetData()+
1990  skewidx*ndofs, ndofs);
1991 
1992  grad_phys.Mult(par_vals, grad_ptr_c1);
1993  Vector grad_q(dim);
1994  tspec_fesv->GetFE(e_id)->CalcShape(ip, shape);
1995  grad_e_c1.MultTranspose(shape, grad_q);
1996 
1997  const double skew = shape * par_vals;
1998 
1999  dQ_phi = 0.;
2000  dQ_phi(0,0) = 1.;
2001  dQ_phi(0,1) = -sin(skew);
2002  dQ_phi(1,1) = cos(skew);
2003 
2004  Mult(Jtrcomp_s, Jtrcomp_r, work2); // z*R
2005 
2006  for (int d = 0; d < dim; d++)
2007  {
2008  DenseMatrix &dJtr_i = dJtr(i + d*ir.GetNPoints());
2009  work1 = dQ_phi;
2010  work1 *= grad_q(d); // work1 = dQ/dphi*dphi/dx
2011  Mult(work1, Jtrcomp_d, work3); // dQ/dx*D
2012  AddMult(work2, work3, dJtr_i); // z*R*dQ/dx*D
2013  }
2014  }
2015  else
2016  {
2017  par_vals.SetDataAndSize(tspec_vals.GetData()+
2018  skewidx*ndofs, ndofs*3);
2019  par_vals_c1.SetData(par_vals.GetData());
2020  par_vals_c2.SetData(par_vals.GetData()+ndofs);
2021  par_vals_c3.SetData(par_vals.GetData()+2*ndofs);
2022 
2023  grad_phys.Mult(par_vals_c1, grad_ptr_c1);
2024  grad_phys.Mult(par_vals_c2, grad_ptr_c2);
2025  grad_phys.Mult(par_vals_c3, grad_ptr_c3);
2026  Vector grad_q1(dim), grad_q2(dim), grad_q3(dim);
2027  tspec_fesv->GetFE(e_id)->CalcShape(ip, shape);
2028  grad_e_c1.MultTranspose(shape, grad_q1);
2029  grad_e_c2.MultTranspose(shape, grad_q2);
2030  grad_e_c3.MultTranspose(shape, grad_q3);
2031 
2032  const double phi12 = shape * par_vals_c1;
2033  const double phi13 = shape * par_vals_c2;
2034  const double chi = shape * par_vals_c3;
2035 
2036  dQ_phi = 0.;
2037  dQ_phi(0,0) = 1.;
2038  dQ_phi(0,1) = -sin(phi12);
2039  dQ_phi(1,1) = cos(phi12);
2040 
2041  dQ_phi13 = 0.;
2042  dQ_phi13(0,2) = -sin(phi13);
2043  dQ_phi13(1,2) = cos(phi13)*cos(chi);
2044  dQ_phi13(2,2) = cos(phi13)*sin(chi);
2045 
2046  dQ_phichi = 0.;
2047  dQ_phichi(1,2) = -sin(phi13)*sin(chi);
2048  dQ_phichi(2,2) = sin(phi13)*cos(chi);
2049 
2050  Mult(Jtrcomp_s, Jtrcomp_r, work2); // z*R
2051 
2052  for (int d = 0; d < dim; d++)
2053  {
2054  DenseMatrix &dJtr_i = dJtr(i + d*ir.GetNPoints());
2055  work1 = dQ_phi;
2056  work1 *= grad_q1(d); // work1 = dQ/dphi12*dphi12/dx
2057  work1.Add(grad_q2(d), dQ_phi13); // + dQ/dphi13*dphi13/dx
2058  work1.Add(grad_q3(d), dQ_phichi); // + dQ/dchi*dchi/dx
2059  Mult(work1, Jtrcomp_d, work3); // dQ/dx*D
2060  AddMult(work2, work3, dJtr_i); // z*R*dQ/dx*D
2061  }
2062  }
2063  } // Done skew
2064 
2065  if (orientationidx != -1) // Set orientation
2066  {
2067  if (dim == 2)
2068  {
2069  par_vals.SetDataAndSize(tspec_vals.GetData()+
2070  orientationidx*ndofs, ndofs);
2071 
2072  grad_phys.Mult(par_vals, grad_ptr_c1);
2073  Vector grad_q(dim);
2074  tspec_fesv->GetFE(e_id)->CalcShape(ip, shape);
2075  grad_e_c1.MultTranspose(shape, grad_q);
2076 
2077  const double theta = shape * par_vals;
2078  dR_theta(0,0) = -sin(theta);
2079  dR_theta(0,1) = -cos(theta);
2080  dR_theta(1,0) = cos(theta);
2081  dR_theta(1,1) = -sin(theta);
2082 
2083  Mult(Jtrcomp_q, Jtrcomp_d, work1); // Q*D
2084  Mult(Jtrcomp_s, work1, work2); // z*Q*D
2085  for (int d = 0; d < dim; d++)
2086  {
2087  DenseMatrix &dJtr_i = dJtr(i + d*ir.GetNPoints());
2088  work1 = dR_theta;
2089  work1 *= grad_q(d); // work1 = dR/dtheta*dtheta/dx
2090  AddMult(work1, work2, dJtr_i); // z*dR/dx*Q*D
2091  }
2092  }
2093  else
2094  {
2095  par_vals.SetDataAndSize(tspec_vals.GetData()+
2096  orientationidx*ndofs, ndofs*3);
2097  par_vals_c1.SetData(par_vals.GetData());
2098  par_vals_c2.SetData(par_vals.GetData()+ndofs);
2099  par_vals_c3.SetData(par_vals.GetData()+2*ndofs);
2100 
2101  grad_phys.Mult(par_vals_c1, grad_ptr_c1);
2102  grad_phys.Mult(par_vals_c2, grad_ptr_c2);
2103  grad_phys.Mult(par_vals_c3, grad_ptr_c3);
2104  Vector grad_q1(dim), grad_q2(dim), grad_q3(dim);
2105  tspec_fesv->GetFE(e_id)->CalcShape(ip, shape);
2106  grad_e_c1.MultTranspose(shape, grad_q1);
2107  grad_e_c2.MultTranspose(shape, grad_q2);
2108  grad_e_c3.MultTranspose(shape, grad_q3);
2109 
2110  const double theta = shape * par_vals_c1;
2111  const double psi = shape * par_vals_c2;
2112  const double beta = shape * par_vals_c3;
2113 
2114  const double ct = cos(theta), st = sin(theta),
2115  cp = cos(psi), sp = sin(psi),
2116  cb = cos(beta), sb = sin(beta);
2117 
2118  dR_theta = 0.;
2119  dR_theta(0,0) = -st*sp;
2120  dR_theta(1,0) = ct*sp;
2121  dR_theta(2,0) = 0;
2122 
2123  dR_theta(0,1) = -ct*cb - st*cp*sb;
2124  dR_theta(1,1) = -st*cb + ct*cp*sb;
2125  dR_theta(2,1) = 0.;
2126 
2127  dR_theta(0,0) = -ct*sb + st*cp*cb;
2128  dR_theta(1,0) = -st*sb - ct*cp*cb;
2129  dR_theta(2,0) = 0.;
2130 
2131  dR_beta = 0.;
2132  dR_beta(0,0) = 0.;
2133  dR_beta(1,0) = 0.;
2134  dR_beta(2,0) = 0.;
2135 
2136  dR_beta(0,1) = st*sb + ct*cp*cb;
2137  dR_beta(1,1) = -ct*sb + st*cp*cb;
2138  dR_beta(2,1) = -sp*cb;
2139 
2140  dR_beta(0,0) = -st*cb + ct*cp*sb;
2141  dR_beta(1,0) = ct*cb + st*cp*sb;
2142  dR_beta(2,0) = 0.;
2143 
2144  dR_psi = 0.;
2145  dR_psi(0,0) = ct*cp;
2146  dR_psi(1,0) = st*cp;
2147  dR_psi(2,0) = -sp;
2148 
2149  dR_psi(0,1) = 0. - ct*sp*sb;
2150  dR_psi(1,1) = 0. + st*sp*sb;
2151  dR_psi(2,1) = -cp*sb;
2152 
2153  dR_psi(0,0) = 0. + ct*sp*cb;
2154  dR_psi(1,0) = 0. + st*sp*cb;
2155  dR_psi(2,0) = cp*cb;
2156 
2157  Mult(Jtrcomp_q, Jtrcomp_d, work1); // Q*D
2158  Mult(Jtrcomp_s, work1, work2); // z*Q*D
2159  for (int d = 0; d < dim; d++)
2160  {
2161  DenseMatrix &dJtr_i = dJtr(i + d*ir.GetNPoints());
2162  work1 = dR_theta;
2163  work1 *= grad_q1(d); // work1 = dR/dtheta*dtheta/dx
2164  work1.Add(grad_q2(d), dR_psi); // +dR/dpsi*dpsi/dx
2165  work1.Add(grad_q3(d), dR_beta); // +dR/dbeta*dbeta/dx
2166  AddMult(work1, work2, dJtr_i); // z*dR/dx*Q*D
2167  }
2168  }
2169  } // Done orientation
2170  }
2171  break;
2172  }
2173  default:
2174  MFEM_ABORT("Incompatible target type for discrete adaptation!");
2175  }
2176  Jtrcomp.Clear();
2177 }
2178 
2180  const double dx,
2181  bool use_flag)
2182 {
2183  if (use_flag && good_tspec_grad) { return; }
2184 
2185  const int dim = tspec_fesv->GetFE(0)->GetDim(),
2186  cnt = x.Size()/dim;
2187 
2189 
2190  Vector TSpecTemp;
2191  Vector xtemp = x;
2192  for (int j = 0; j < dim; j++)
2193  {
2194  for (int i = 0; i < cnt; i++) { xtemp(j*cnt+i) += dx; }
2195 
2196  TSpecTemp.NewDataAndSize(tspec_pert1h.GetData() + j*cnt*ncomp, cnt*ncomp);
2197  UpdateTargetSpecification(xtemp, TSpecTemp);
2198 
2199  for (int i = 0; i < cnt; i++) { xtemp(j*cnt+i) -= dx; }
2200  }
2201 
2202  good_tspec_grad = use_flag;
2203 }
2204 
2206  double dx, bool use_flag)
2207 {
2208 
2209  if (use_flag && good_tspec_hess) { return; }
2210 
2211  const int dim = tspec_fesv->GetFE(0)->GetDim(),
2212  cnt = x.Size()/dim,
2213  totmix = 1+2*(dim-2);
2214 
2215  tspec_pert2h.SetSize(cnt*dim*ncomp);
2216  tspec_pertmix.SetSize(cnt*totmix*ncomp);
2217 
2218  Vector TSpecTemp;
2219  Vector xtemp = x;
2220 
2221  // T(x+2h)
2222  for (int j = 0; j < dim; j++)
2223  {
2224  for (int i = 0; i < cnt; i++) { xtemp(j*cnt+i) += 2*dx; }
2225 
2226  TSpecTemp.NewDataAndSize(tspec_pert2h.GetData() + j*cnt*ncomp, cnt*ncomp);
2227  UpdateTargetSpecification(xtemp, TSpecTemp);
2228 
2229  for (int i = 0; i < cnt; i++) { xtemp(j*cnt+i) -= 2*dx; }
2230  }
2231 
2232  // T(x+h,y+h)
2233  int j = 0;
2234  for (int k1 = 0; k1 < dim; k1++)
2235  {
2236  for (int k2 = 0; (k1 != k2) && (k2 < dim); k2++)
2237  {
2238  for (int i = 0; i < cnt; i++)
2239  {
2240  xtemp(k1*cnt+i) += dx;
2241  xtemp(k2*cnt+i) += dx;
2242  }
2243 
2244  TSpecTemp.NewDataAndSize(tspec_pertmix.GetData() + j*cnt*ncomp, cnt*ncomp);
2245  UpdateTargetSpecification(xtemp, TSpecTemp);
2246 
2247  for (int i = 0; i < cnt; i++)
2248  {
2249  xtemp(k1*cnt+i) -= dx;
2250  xtemp(k2*cnt+i) -= dx;
2251  }
2252  j++;
2253  }
2254  }
2255 
2256  good_tspec_hess = use_flag;
2257 }
2258 
2260 {
2261  delete tspec_gf;
2262  delete adapt_eval;
2263  delete tspec_fesv;
2264 #ifdef MFEM_USE_MPI
2265  delete ptspec_fesv;
2266 #endif
2267 }
2268 
2270  const FiniteElementCollection &fec,
2271  int num_comp)
2272 {
2273  delete fes;
2274  delete mesh;
2275  mesh = new Mesh(m, true);
2276  fes = new FiniteElementSpace(mesh, &fec, num_comp);
2277  dim = fes->GetFE(0)->GetDim();
2278  ncomp = num_comp;
2279 }
2280 
2281 #ifdef MFEM_USE_MPI
2283  const FiniteElementCollection &fec,
2284  int num_comp)
2285 {
2286  delete pfes;
2287  delete pmesh;
2288  pmesh = new ParMesh(m, true);
2289  pfes = new ParFiniteElementSpace(pmesh, &fec, num_comp);
2290  dim = pfes->GetFE(0)->GetDim();
2291  ncomp = num_comp;
2292 }
2293 #endif
2294 
2296 {
2297 #ifdef MFEM_USE_MPI
2298  if (pmesh) { pmesh->DeleteGeometricFactors(); }
2299 #else
2300  if (mesh) { mesh->DeleteGeometricFactors(); }
2301 #endif
2302 }
2303 
2305 {
2306  delete fes;
2307  delete mesh;
2308 #ifdef MFEM_USE_MPI
2309  delete pfes;
2310  delete pmesh;
2311 #endif
2312 }
2313 
2315 {
2316  if (PA.enabled)
2317  {
2318  PA.H.GetMemory().DeleteDevice(copy_to_host);
2319  PA.H0.GetMemory().DeleteDevice(copy_to_host);
2320  if (!copy_to_host && !PA.Jtr.GetMemory().HostIsValid())
2321  {
2322  PA.Jtr_needs_update = true;
2323  }
2324  PA.Jtr.GetMemory().DeleteDevice(copy_to_host);
2325  }
2326 }
2327 
2329 {
2330  delete lim_func;
2331  delete adapt_lim_gf;
2332  delete surf_fit_gf;
2333  for (int i = 0; i < ElemDer.Size(); i++)
2334  {
2335  delete ElemDer[i];
2336  delete ElemPertEnergy[i];
2337  }
2338 }
2339 
2341  const GridFunction &dist, Coefficient &w0,
2342  TMOP_LimiterFunction *lfunc)
2343 {
2344  lim_nodes0 = &n0;
2345  lim_coeff = &w0;
2346  lim_dist = &dist;
2347  MFEM_VERIFY(lim_dist->FESpace()->GetVDim() == 1,
2348  "'dist' must be a scalar GridFunction!");
2349 
2350  delete lim_func;
2351  lim_func = (lfunc) ? lfunc : new TMOP_QuadraticLimiter;
2352 }
2353 
2355  TMOP_LimiterFunction *lfunc)
2356 {
2357  lim_nodes0 = &n0;
2358  lim_coeff = &w0;
2359  lim_dist = NULL;
2360 
2361  delete lim_func;
2362  lim_func = (lfunc) ? lfunc : new TMOP_QuadraticLimiter;
2363 }
2364 
2366  Coefficient &coeff,
2367  AdaptivityEvaluator &ae)
2368 {
2369  adapt_lim_gf0 = &z0;
2370  delete adapt_lim_gf;
2371  adapt_lim_gf = new GridFunction(z0);
2372  adapt_lim_coeff = &coeff;
2373  adapt_lim_eval = &ae;
2374 
2376  *adapt_lim_gf->FESpace()->FEColl(), 1);
2379 }
2380 
2381 #ifdef MFEM_USE_MPI
2383  Coefficient &coeff,
2384  AdaptivityEvaluator &ae)
2385 {
2386  adapt_lim_gf0 = &z0;
2387  adapt_lim_pgf0 = &z0;
2388  delete adapt_lim_gf;
2389  adapt_lim_gf = new GridFunction(z0);
2390  adapt_lim_coeff = &coeff;
2391  adapt_lim_eval = &ae;
2392 
2394  *z0.ParFESpace()->FEColl(), 1);
2397 }
2398 #endif
2399 
2401  const Array<bool> &smarker,
2402  Coefficient &coeff,
2403  AdaptivityEvaluator &ae)
2404 {
2405  delete surf_fit_gf;
2406  surf_fit_gf = new GridFunction(s0);
2407  surf_fit_marker = &smarker;
2408  surf_fit_coeff = &coeff;
2409  surf_fit_eval = &ae;
2410 
2412  *s0.FESpace()->FEColl(), 1);
2415 }
2416 
2417 #ifdef MFEM_USE_MPI
2419  const Array<bool> &smarker,
2420  Coefficient &coeff,
2421  AdaptivityEvaluator &ae)
2422 {
2423  delete surf_fit_gf;
2424  surf_fit_gf = new GridFunction(s0);
2425  surf_fit_marker = &smarker;
2426  surf_fit_coeff = &coeff;
2427  surf_fit_eval = &ae;
2428 
2430  *s0.ParFESpace()->FEColl(), 1);
2433 }
2434 #endif
2435 
2436 void TMOP_Integrator::GetSurfaceFittingErrors(double &err_avg, double &err_max)
2437 {
2438  MFEM_VERIFY(surf_fit_gf, "Surface fitting has not been enabled.");
2439 
2440  int loc_cnt = 0;
2441  double loc_max = 0.0, loc_sum = 0.0;
2442  for (int i = 0; i < surf_fit_marker->Size(); i++)
2443  {
2444  if ((*surf_fit_marker)[i] == true)
2445  {
2446  loc_cnt++;
2447  loc_max = std::max(loc_max, std::abs((*surf_fit_gf)(i)));
2448  loc_sum += std::abs((*surf_fit_gf)(i));
2449  }
2450  }
2451  err_avg = loc_sum / loc_cnt;
2452  err_max = loc_max;
2453 
2454 #ifdef MFEM_USE_MPI
2455  if (targetC->Parallel() == false) { return; }
2456  int glob_cnt;
2457  MPI_Comm comm = targetC->GetComm();
2458  MPI_Allreduce(&loc_max, &err_max, 1, MPI_DOUBLE, MPI_MAX, comm);
2459  MPI_Allreduce(&loc_cnt, &glob_cnt, 1, MPI_INT, MPI_SUM, comm);
2460  MPI_Allreduce(&loc_sum, &err_avg, 1, MPI_DOUBLE, MPI_SUM, comm);
2461  err_avg = err_avg / glob_cnt;
2462 #endif
2463 }
2464 
2466 {
2467  if (adapt_lim_gf)
2468  {
2469  adapt_lim_gf->Update();
2471  *adapt_lim_gf->FESpace()->FEColl(), 1);
2474  }
2475 }
2476 
2477 #ifdef MFEM_USE_MPI
2479 {
2480  if (adapt_lim_gf)
2481  {
2482  adapt_lim_gf->Update();
2484  *adapt_lim_pgf0->ParFESpace()->FEColl(), 1);
2487  }
2488 }
2489 #endif
2490 
2493  const Vector &elfun)
2494 {
2495  const int dof = el.GetDof(), dim = el.GetDim();
2496  const int el_id = T.ElementNo;
2497  double energy;
2498 
2499  // No adaptive limiting / surface fitting terms if the function is called
2500  // as part of a FD derivative computation (because we include the exact
2501  // derivatives of these terms in FD computations).
2502  const bool adaptive_limiting = (adapt_lim_gf && fd_call_flag == false);
2503  const bool surface_fit = (surf_fit_gf && fd_call_flag == false);
2504 
2505  DSh.SetSize(dof, dim);
2506  Jrt.SetSize(dim);
2507  Jpr.SetSize(dim);
2508  Jpt.SetSize(dim);
2509  PMatI.UseExternalData(elfun.GetData(), dof, dim);
2510 
2512 
2513  energy = 0.0;
2514  DenseTensor Jtr(dim, dim, ir.GetNPoints());
2515  targetC->ComputeElementTargets(el_id, el, ir, elfun, Jtr);
2516 
2517  // Limited case.
2518  Vector shape, p, p0, d_vals;
2519  DenseMatrix pos0;
2520  if (lim_coeff)
2521  {
2522  shape.SetSize(dof);
2523  p.SetSize(dim);
2524  p0.SetSize(dim);
2525  pos0.SetSize(dof, dim);
2526  Vector pos0V(pos0.Data(), dof * dim);
2527  Array<int> pos_dofs;
2528  lim_nodes0->FESpace()->GetElementVDofs(el_id, pos_dofs);
2529  lim_nodes0->GetSubVector(pos_dofs, pos0V);
2530  if (lim_dist)
2531  {
2532  lim_dist->GetValues(el_id, ir, d_vals);
2533  }
2534  else
2535  {
2536  d_vals.SetSize(ir.GetNPoints()); d_vals = 1.0;
2537  }
2538  }
2539 
2540  // Define ref->physical transformation, when a Coefficient is specified.
2541  IsoparametricTransformation *Tpr = NULL;
2542  if (metric_coeff || lim_coeff || adaptive_limiting || surface_fit)
2543  {
2544  Tpr = new IsoparametricTransformation;
2545  Tpr->SetFE(&el);
2546  Tpr->ElementNo = el_id;
2548  Tpr->Attribute = T.Attribute;
2549  Tpr->mesh = T.mesh;
2550  Tpr->GetPointMat().Transpose(PMatI); // PointMat = PMatI^T
2551  }
2552  // TODO: computing the coefficients 'metric_coeff' and 'lim_coeff' in physical
2553  // coordinates means that, generally, the gradient and Hessian of the
2554  // TMOP_Integrator will depend on the derivatives of the coefficients.
2555  //
2556  // In some cases the coefficients are independent of any movement of
2557  // the physical coordinates (i.e. changes in 'elfun'), e.g. when the
2558  // coefficient is a ConstantCoefficient or a GridFunctionCoefficient.
2559 
2560  Vector adapt_lim_gf_q, adapt_lim_gf0_q;
2561  if (adaptive_limiting)
2562  {
2563  adapt_lim_gf->GetValues(el_id, ir, adapt_lim_gf_q);
2564  adapt_lim_gf0->GetValues(el_id, ir, adapt_lim_gf0_q);
2565  }
2566 
2567  for (int i = 0; i < ir.GetNPoints(); i++)
2568  {
2569  const IntegrationPoint &ip = ir.IntPoint(i);
2570 
2571  const DenseMatrix &Jtr_i = Jtr(i);
2572  metric->SetTargetJacobian(Jtr_i);
2573  CalcInverse(Jtr_i, Jrt);
2574  const double weight = ip.weight * Jtr_i.Det();
2575 
2576  el.CalcDShape(ip, DSh);
2577  MultAtB(PMatI, DSh, Jpr);
2578  Mult(Jpr, Jrt, Jpt);
2579 
2580  double val = metric_normal * metric->EvalW(Jpt);
2581  if (metric_coeff) { val *= metric_coeff->Eval(*Tpr, ip); }
2582 
2583  if (lim_coeff)
2584  {
2585  el.CalcShape(ip, shape);
2586  PMatI.MultTranspose(shape, p);
2587  pos0.MultTranspose(shape, p0);
2588  val += lim_normal *
2589  lim_func->Eval(p, p0, d_vals(i)) *
2590  lim_coeff->Eval(*Tpr, ip);
2591  }
2592 
2593  // Contribution from the adaptive limiting term.
2594  if (adaptive_limiting)
2595  {
2596  const double diff = adapt_lim_gf_q(i) - adapt_lim_gf0_q(i);
2597  val += adapt_lim_coeff->Eval(*Tpr, ip) * lim_normal * diff * diff;
2598  }
2599 
2600  energy += weight * val;
2601  }
2602 
2603  // Contribution from the surface fitting term.
2604  if (surface_fit)
2605  {
2606  const IntegrationRule &ir_s =
2607  surf_fit_gf->FESpace()->GetFE(el_id)->GetNodes();
2608  Array<int> dofs;
2609  Vector sigma_e;
2610  surf_fit_gf->FESpace()->GetElementDofs(el_id, dofs);
2611  surf_fit_gf->GetSubVector(dofs, sigma_e);
2612  for (int s = 0; s < dofs.Size(); s++)
2613  {
2614  if ((*surf_fit_marker)[dofs[s]] == true)
2615  {
2616  const IntegrationPoint &ip_s = ir_s.IntPoint(s);
2617  Tpr->SetIntPoint(&ip_s);
2618  energy += surf_fit_coeff->Eval(*Tpr, ip_s) * surf_fit_normal *
2619  sigma_e(s) * sigma_e(s);
2620  }
2621  }
2622  }
2623 
2624  delete Tpr;
2625  return energy;
2626 }
2627 
2630  const Vector &elfun,
2631  const IntegrationRule &irule)
2632 {
2633  int dof = el.GetDof(), dim = el.GetDim(),
2634  NEsplit = elfun.Size() / (dof*dim), el_id = T.ElementNo;
2635  double energy = 0.;
2636 
2637  TargetConstructor *tc = const_cast<TargetConstructor *>(targetC);
2638  DiscreteAdaptTC *dtc = dynamic_cast<DiscreteAdaptTC *>(tc);
2639  // For DiscreteAdaptTC the GridFunctions used to set the targets must be
2640  // mapped onto the fine elements.
2641  if (dtc) { dtc->SetTspecFromIntRule(el_id, irule); }
2642 
2643  for (int e = 0; e < NEsplit; e++)
2644  {
2645  DSh.SetSize(dof, dim);
2646  Jrt.SetSize(dim);
2647  Jpr.SetSize(dim);
2648  Jpt.SetSize(dim);
2649  Vector elfun_child(dof*dim);
2650  for (int i = 0; i < dof; i++)
2651  {
2652  for (int d = 0; d < dim; d++)
2653  {
2654  // elfun is (xe1,xe2,...xen,ye1,ye2...yen) and has nodal coordinates
2655  // for all the children element of the parent element being considered.
2656  // So we must index and get (xek, yek) i.e. nodal coordinates for
2657  // the fine element being considered.
2658  elfun_child(i + d*dof) = elfun(i + e*dof + d*dof*NEsplit);
2659  }
2660  }
2661  PMatI.UseExternalData(elfun_child.GetData(), dof, dim);
2662 
2664 
2665  double el_energy = 0;
2666  DenseTensor Jtr(dim, dim, ir.GetNPoints());
2667  if (dtc)
2668  {
2669  // This is used to index into the tspec vector inside DiscreteAdaptTC.
2670  dtc->SetRefinementSubElement(e);
2671  }
2672  targetC->ComputeElementTargets(el_id, el, ir, elfun_child, Jtr);
2673 
2674  // Define ref->physical transformation, wn a Coefficient is specified.
2675  IsoparametricTransformation *Tpr = NULL;
2676  if (metric_coeff || lim_coeff)
2677  {
2678  Tpr = new IsoparametricTransformation;
2679  Tpr->SetFE(&el);
2680  Tpr->ElementNo = T.ElementNo;
2682  Tpr->Attribute = T.Attribute;
2683  Tpr->mesh = T.mesh;
2684  Tpr->GetPointMat().Transpose(PMatI); // PointMat = PMatI^T
2685  }
2686 
2687  for (int i = 0; i < ir.GetNPoints(); i++)
2688  {
2689  const IntegrationPoint &ip = ir.IntPoint(i);
2690  const DenseMatrix &Jtr_i = Jtr(i);
2691  h_metric->SetTargetJacobian(Jtr_i);
2692  CalcInverse(Jtr_i, Jrt);
2693  const double weight = ip.weight * Jtr_i.Det();
2694 
2695  el.CalcDShape(ip, DSh);
2696  MultAtB(PMatI, DSh, Jpr);
2697  Mult(Jpr, Jrt, Jpt);
2698 
2699  double val = metric_normal * h_metric->EvalW(Jpt);
2700  if (metric_coeff) { val *= metric_coeff->Eval(*Tpr, ip); }
2701 
2702  el_energy += weight * val;
2703  delete Tpr;
2704  }
2705  energy += el_energy;
2706  }
2707  energy /= NEsplit;
2708 
2709  if (dtc) { dtc->ResetRefinementTspecData(); }
2710 
2711  return energy;
2712 }
2713 
2716  const Vector &elfun)
2717 {
2718  int dof = el.GetDof(), dim = el.GetDim();
2719  double energy = 0.;
2720 
2721  DSh.SetSize(dof, dim);
2722  Jrt.SetSize(dim);
2723  Jpr.SetSize(dim);
2724  Jpt.SetSize(dim);
2725  PMatI.UseExternalData(elfun.GetData(), dof, dim);
2726 
2728 
2729  energy = 0.0;
2730  DenseTensor Jtr(dim, dim, ir.GetNPoints());
2731  targetC->ComputeElementTargets(T.ElementNo, el, ir, elfun, Jtr);
2732 
2733  // Define ref->physical transformation, wn a Coefficient is specified.
2734  IsoparametricTransformation *Tpr = NULL;
2735  if (metric_coeff)
2736  {
2737  Tpr = new IsoparametricTransformation;
2738  Tpr->SetFE(&el);
2739  Tpr->ElementNo = T.ElementNo;
2741  Tpr->Attribute = T.Attribute;
2742  Tpr->mesh = T.mesh;
2743  Tpr->GetPointMat().Transpose(PMatI); // PointMat = PMatI^T
2744  }
2745 
2746  for (int i = 0; i < ir.GetNPoints(); i++)
2747  {
2748  const IntegrationPoint &ip = ir.IntPoint(i);
2749  const DenseMatrix &Jtr_i = Jtr(i);
2750  h_metric->SetTargetJacobian(Jtr_i);
2751  CalcInverse(Jtr_i, Jrt);
2752  const double weight = ip.weight * Jtr_i.Det();
2753 
2754  el.CalcDShape(ip, DSh);
2755  MultAtB(PMatI, DSh, Jpr);
2756  Mult(Jpr, Jrt, Jpt);
2757 
2758  double val = metric_normal * h_metric->EvalW(Jpt);
2759  if (metric_coeff) { val *= metric_coeff->Eval(*Tpr, ip); }
2760 
2761  energy += weight * val;
2762  }
2763 
2764  delete Tpr;
2765  return energy;
2766 }
2767 
2770  const Vector &elfun, Vector &elvect)
2771 {
2772  if (!fdflag)
2773  {
2774  AssembleElementVectorExact(el, T, elfun, elvect);
2775  }
2776  else
2777  {
2778  AssembleElementVectorFD(el, T, elfun, elvect);
2779  }
2780 }
2781 
2784  const Vector &elfun,
2785  DenseMatrix &elmat)
2786 {
2787  if (!fdflag)
2788  {
2789  AssembleElementGradExact(el, T, elfun, elmat);
2790  }
2791  else
2792  {
2793  AssembleElementGradFD(el, T, elfun, elmat);
2794  }
2795 }
2796 
2799  const Vector &elfun,
2800  Vector &elvect)
2801 {
2802  const int dof = el.GetDof(), dim = el.GetDim();
2803 
2804  DenseMatrix Amat(dim), work1(dim), work2(dim);
2805  DSh.SetSize(dof, dim);
2806  DS.SetSize(dof, dim);
2807  Jrt.SetSize(dim);
2808  Jpt.SetSize(dim);
2809  P.SetSize(dim);
2810  PMatI.UseExternalData(elfun.GetData(), dof, dim);
2811  elvect.SetSize(dof*dim);
2812  PMatO.UseExternalData(elvect.GetData(), dof, dim);
2813 
2815  const int nqp = ir.GetNPoints();
2816 
2817  elvect = 0.0;
2818  Vector weights(nqp);
2819  DenseTensor Jtr(dim, dim, nqp);
2820  DenseTensor dJtr(dim, dim, dim*nqp);
2821  targetC->ComputeElementTargets(T.ElementNo, el, ir, elfun, Jtr);
2822 
2823  // Limited case.
2824  DenseMatrix pos0;
2825  Vector shape, p, p0, d_vals, grad;
2826  shape.SetSize(dof);
2827  if (lim_coeff)
2828  {
2829  p.SetSize(dim);
2830  p0.SetSize(dim);
2831  pos0.SetSize(dof, dim);
2832  Vector pos0V(pos0.Data(), dof * dim);
2833  Array<int> pos_dofs;
2834  lim_nodes0->FESpace()->GetElementVDofs(T.ElementNo, pos_dofs);
2835  lim_nodes0->GetSubVector(pos_dofs, pos0V);
2836  if (lim_dist)
2837  {
2838  lim_dist->GetValues(T.ElementNo, ir, d_vals);
2839  }
2840  else
2841  {
2842  d_vals.SetSize(nqp); d_vals = 1.0;
2843  }
2844  }
2845 
2846  // Define ref->physical transformation, when a Coefficient is specified.
2847  IsoparametricTransformation *Tpr = NULL;
2849  {
2850  Tpr = new IsoparametricTransformation;
2851  Tpr->SetFE(&el);
2852  Tpr->ElementNo = T.ElementNo;
2854  Tpr->Attribute = T.Attribute;
2855  Tpr->mesh = T.mesh;
2856  Tpr->GetPointMat().Transpose(PMatI); // PointMat = PMatI^T
2857  if (exact_action)
2858  {
2859  targetC->ComputeElementTargetsGradient(ir, elfun, *Tpr, dJtr);
2860  }
2861  }
2862 
2863 
2864  Vector d_detW_dx(dim);
2865  Vector d_Winv_dx(dim);
2866 
2867  for (int q = 0; q < nqp; q++)
2868  {
2869  const IntegrationPoint &ip = ir.IntPoint(q);
2870  const DenseMatrix &Jtr_q = Jtr(q);
2871  metric->SetTargetJacobian(Jtr_q);
2872  CalcInverse(Jtr_q, Jrt);
2873  weights(q) = ip.weight * Jtr_q.Det();
2874  double weight_m = weights(q) * metric_normal;
2875 
2876  el.CalcDShape(ip, DSh);
2877  Mult(DSh, Jrt, DS);
2878  MultAtB(PMatI, DS, Jpt);
2879 
2880  metric->EvalP(Jpt, P);
2881 
2882  if (metric_coeff) { weight_m *= metric_coeff->Eval(*Tpr, ip); }
2883 
2884  P *= weight_m;
2885  AddMultABt(DS, P, PMatO); // w_q det(W) dmu/dx : dA/dx Winv
2886 
2887  if (exact_action)
2888  {
2889  el.CalcShape(ip, shape);
2890  // Derivatives of adaptivity-based targets.
2891  // First term: w_q d*(Det W)/dx * mu(T)
2892  // d(Det W)/dx = det(W)*Tr[Winv*dW/dx]
2893  DenseMatrix dwdx(dim);
2894  for (int d = 0; d < dim; d++)
2895  {
2896  const DenseMatrix &dJtr_q = dJtr(q + d * nqp);
2897  Mult(Jrt, dJtr_q, dwdx );
2898  d_detW_dx(d) = dwdx.Trace();
2899  }
2900  d_detW_dx *= weight_m*metric->EvalW(Jpt); // *[w_q*det(W)]*mu(T)
2901 
2902  // Second term: w_q det(W) dmu/dx : AdWinv/dx
2903  // dWinv/dx = -Winv*dW/dx*Winv
2904  MultAtB(PMatI, DSh, Amat);
2905  for (int d = 0; d < dim; d++)
2906  {
2907  const DenseMatrix &dJtr_q = dJtr(q + d*nqp);
2908  Mult(Jrt, dJtr_q, work1); // Winv*dw/dx
2909  Mult(work1, Jrt, work2); // Winv*dw/dx*Winv
2910  Mult(Amat, work2, work1); // A*Winv*dw/dx*Winv
2911  MultAtB(P, work1, work2); // dmu/dT^T*A*Winv*dw/dx*Winv
2912  d_Winv_dx(d) = work2.Trace(); // Tr[dmu/dT : AWinv*dw/dx*Winv]
2913  }
2914  d_Winv_dx *= -weight_m; // Include (-) factor as well
2915 
2916  d_detW_dx += d_Winv_dx;
2917  AddMultVWt(shape, d_detW_dx, PMatO);
2918  }
2919 
2920  if (lim_coeff)
2921  {
2922  if (!exact_action) { el.CalcShape(ip, shape); }
2923  PMatI.MultTranspose(shape, p);
2924  pos0.MultTranspose(shape, p0);
2925  lim_func->Eval_d1(p, p0, d_vals(q), grad);
2926  grad *= weights(q) * lim_normal * lim_coeff->Eval(*Tpr, ip);
2927  AddMultVWt(shape, grad, PMatO);
2928  }
2929  }
2930 
2931  if (adapt_lim_gf) { AssembleElemVecAdaptLim(el, *Tpr, ir, weights, PMatO); }
2932  if (surf_fit_gf) { AssembleElemVecSurfFit(el, *Tpr, PMatO); }
2933 
2934  delete Tpr;
2935 }
2936 
2939  const Vector &elfun,
2940  DenseMatrix &elmat)
2941 {
2942  const int dof = el.GetDof(), dim = el.GetDim();
2943 
2944  DSh.SetSize(dof, dim);
2945  DS.SetSize(dof, dim);
2946  Jrt.SetSize(dim);
2947  Jpt.SetSize(dim);
2948  PMatI.UseExternalData(elfun.GetData(), dof, dim);
2949  elmat.SetSize(dof*dim);
2950 
2952  const int nqp = ir.GetNPoints();
2953 
2954  elmat = 0.0;
2955  Vector weights(nqp);
2956  DenseTensor Jtr(dim, dim, nqp);
2957  targetC->ComputeElementTargets(T.ElementNo, el, ir, elfun, Jtr);
2958 
2959  // Limited case.
2960  DenseMatrix pos0, hess;
2961  Vector shape, p, p0, d_vals;
2962  if (lim_coeff)
2963  {
2964  shape.SetSize(dof);
2965  p.SetSize(dim);
2966  p0.SetSize(dim);
2967  pos0.SetSize(dof, dim);
2968  Vector pos0V(pos0.Data(), dof * dim);
2969  Array<int> pos_dofs;
2970  lim_nodes0->FESpace()->GetElementVDofs(T.ElementNo, pos_dofs);
2971  lim_nodes0->GetSubVector(pos_dofs, pos0V);
2972  if (lim_dist)
2973  {
2974  lim_dist->GetValues(T.ElementNo, ir, d_vals);
2975  }
2976  else
2977  {
2978  d_vals.SetSize(nqp); d_vals = 1.0;
2979  }
2980  }
2981 
2982  // Define ref->physical transformation, when a Coefficient is specified.
2983  IsoparametricTransformation *Tpr = NULL;
2985  {
2986  Tpr = new IsoparametricTransformation;
2987  Tpr->SetFE(&el);
2988  Tpr->ElementNo = T.ElementNo;
2990  Tpr->Attribute = T.Attribute;
2991  Tpr->mesh = T.mesh;
2992  Tpr->GetPointMat().Transpose(PMatI);
2993  }
2994 
2995  for (int q = 0; q < nqp; q++)
2996  {
2997  const IntegrationPoint &ip = ir.IntPoint(q);
2998  const DenseMatrix &Jtr_q = Jtr(q);
2999  metric->SetTargetJacobian(Jtr_q);
3000  CalcInverse(Jtr_q, Jrt);
3001  weights(q) = ip.weight * Jtr_q.Det();
3002  double weight_m = weights(q) * metric_normal;
3003 
3004  el.CalcDShape(ip, DSh);
3005  Mult(DSh, Jrt, DS);
3006  MultAtB(PMatI, DS, Jpt);
3007 
3008  if (metric_coeff) { weight_m *= metric_coeff->Eval(*Tpr, ip); }
3009 
3010  metric->AssembleH(Jpt, DS, weight_m, elmat);
3011 
3012  // TODO: derivatives of adaptivity-based targets.
3013 
3014  if (lim_coeff)
3015  {
3016  el.CalcShape(ip, shape);
3017  PMatI.MultTranspose(shape, p);
3018  pos0.MultTranspose(shape, p0);
3019  weight_m = weights(q) * lim_normal * lim_coeff->Eval(*Tpr, ip);
3020  lim_func->Eval_d2(p, p0, d_vals(q), hess);
3021  for (int i = 0; i < dof; i++)
3022  {
3023  const double w_shape_i = weight_m * shape(i);
3024  for (int j = 0; j < dof; j++)
3025  {
3026  const double w = w_shape_i * shape(j);
3027  for (int d1 = 0; d1 < dim; d1++)
3028  {
3029  for (int d2 = 0; d2 < dim; d2++)
3030  {
3031  elmat(d1*dof + i, d2*dof + j) += w * hess(d1, d2);
3032  }
3033  }
3034  }
3035  }
3036  }
3037  }
3038 
3039  if (adapt_lim_gf) { AssembleElemGradAdaptLim(el, *Tpr, ir, weights, elmat); }
3040  if (surf_fit_gf) { AssembleElemGradSurfFit(el, *Tpr, elmat); }
3041 
3042  delete Tpr;
3043 }
3044 
3047  const IntegrationRule &ir,
3048  const Vector &weights,
3049  DenseMatrix &mat)
3050 {
3051  const int dof = el.GetDof(), dim = el.GetDim(), nqp = weights.Size();
3052  Vector shape(dof), adapt_lim_gf_e, adapt_lim_gf_q, adapt_lim_gf0_q(nqp);
3053 
3054  Array<int> dofs;
3056  adapt_lim_gf->GetSubVector(dofs, adapt_lim_gf_e);
3057  adapt_lim_gf->GetValues(Tpr.ElementNo, ir, adapt_lim_gf_q);
3058  adapt_lim_gf0->GetValues(Tpr.ElementNo, ir, adapt_lim_gf0_q);
3059 
3060  // Project the gradient of adapt_lim_gf in the same space.
3061  // The FE coefficients of the gradient go in adapt_lim_gf_grad_e.
3062  DenseMatrix adapt_lim_gf_grad_e(dof, dim);
3063  DenseMatrix grad_phys; // This will be (dof x dim, dof).
3064  el.ProjectGrad(el, Tpr, grad_phys);
3065  Vector grad_ptr(adapt_lim_gf_grad_e.GetData(), dof*dim);
3066  grad_phys.Mult(adapt_lim_gf_e, grad_ptr);
3067 
3068  Vector adapt_lim_gf_grad_q(dim);
3069 
3070  for (int q = 0; q < nqp; q++)
3071  {
3072  const IntegrationPoint &ip = ir.IntPoint(q);
3073  el.CalcShape(ip, shape);
3074  adapt_lim_gf_grad_e.MultTranspose(shape, adapt_lim_gf_grad_q);
3075  adapt_lim_gf_grad_q *= 2.0 * (adapt_lim_gf_q(q) - adapt_lim_gf0_q(q));
3076  adapt_lim_gf_grad_q *= weights(q) * lim_normal * adapt_lim_coeff->Eval(Tpr, ip);
3077  AddMultVWt(shape, adapt_lim_gf_grad_q, mat);
3078  }
3079 }
3080 
3083  const IntegrationRule &ir,
3084  const Vector &weights,
3085  DenseMatrix &mat)
3086 {
3087  const int dof = el.GetDof(), dim = el.GetDim(), nqp = weights.Size();
3088  Vector shape(dof), adapt_lim_gf_e, adapt_lim_gf_q, adapt_lim_gf0_q(nqp);
3089 
3090  Array<int> dofs;
3092  adapt_lim_gf->GetSubVector(dofs, adapt_lim_gf_e);
3093  adapt_lim_gf->GetValues(Tpr.ElementNo, ir, adapt_lim_gf_q);
3094  adapt_lim_gf0->GetValues(Tpr.ElementNo, ir, adapt_lim_gf0_q);
3095 
3096  // Project the gradient of adapt_lim_gf in the same space.
3097  // The FE coefficients of the gradient go in adapt_lim_gf_grad_e.
3098  DenseMatrix adapt_lim_gf_grad_e(dof, dim);
3099  DenseMatrix grad_phys; // This will be (dof x dim, dof).
3100  el.ProjectGrad(el, Tpr, grad_phys);
3101  Vector grad_ptr(adapt_lim_gf_grad_e.GetData(), dof*dim);
3102  grad_phys.Mult(adapt_lim_gf_e, grad_ptr);
3103 
3104  // Project the gradient of each gradient of adapt_lim_gf in the same space.
3105  // The FE coefficients of the second derivatives go in adapt_lim_gf_hess_e.
3106  DenseMatrix adapt_lim_gf_hess_e(dof*dim, dim);
3107  Mult(grad_phys, adapt_lim_gf_grad_e, adapt_lim_gf_hess_e);
3108  // Reshape to be more convenient later (no change in the data).
3109  adapt_lim_gf_hess_e.SetSize(dof, dim*dim);
3110 
3111  Vector adapt_lim_gf_grad_q(dim);
3112  DenseMatrix adapt_lim_gf_hess_q(dim, dim);
3113 
3114  for (int q = 0; q < nqp; q++)
3115  {
3116  const IntegrationPoint &ip = ir.IntPoint(q);
3117  el.CalcShape(ip, shape);
3118 
3119  adapt_lim_gf_grad_e.MultTranspose(shape, adapt_lim_gf_grad_q);
3120  Vector gg_ptr(adapt_lim_gf_hess_q.GetData(), dim*dim);
3121  adapt_lim_gf_hess_e.MultTranspose(shape, gg_ptr);
3122 
3123  const double w = weights(q) * lim_normal * adapt_lim_coeff->Eval(Tpr, ip);
3124  for (int i = 0; i < dof * dim; i++)
3125  {
3126  const int idof = i % dof, idim = i / dof;
3127  for (int j = 0; j <= i; j++)
3128  {
3129  const int jdof = j % dof, jdim = j / dof;
3130  const double entry =
3131  w * ( 2.0 * adapt_lim_gf_grad_q(idim) * shape(idof) *
3132  /* */ adapt_lim_gf_grad_q(jdim) * shape(jdof) +
3133  2.0 * (adapt_lim_gf_q(q) - adapt_lim_gf0_q(q)) *
3134  adapt_lim_gf_hess_q(idim, jdim) * shape(idof) * shape(jdof));
3135  mat(i, j) += entry;
3136  if (i != j) { mat(j, i) += entry; }
3137  }
3138  }
3139  }
3140 }
3141 
3144  DenseMatrix &mat)
3145 {
3146  const int el_id = Tpr.ElementNo;
3147  // Check if the element has any DOFs marked for surface fitting.
3148  Array<int> dofs;
3149  surf_fit_gf->FESpace()->GetElementDofs(el_id, dofs);
3150  int count = 0;
3151  for (int s = 0; s < dofs.Size(); s++)
3152  {
3153  count += ((*surf_fit_marker)[dofs[s]]) ? 1 : 0;
3154  }
3155  if (count == 0) { return; }
3156 
3157  const FiniteElement &el_s = *surf_fit_gf->FESpace()->GetFE(el_id);
3158 
3159  const int dof_x = el_x.GetDof(), dim = el_x.GetDim(),
3160  dof_s = el_s.GetDof();
3161 
3162  Vector sigma_e;
3163  surf_fit_gf->GetSubVector(dofs, sigma_e);
3164 
3165  // Project the gradient of sigma in the same space.
3166  // The FE coefficients of the gradient go in surf_fit_grad_e.
3167  DenseMatrix surf_fit_grad_e(dof_s, dim);
3168  Vector grad_ptr(surf_fit_grad_e.GetData(), dof_s * dim);
3169  DenseMatrix grad_phys; // This will be (dof x dim, dof).
3170  el_s.ProjectGrad(el_s, Tpr, grad_phys);
3171  grad_phys.Mult(sigma_e, grad_ptr);
3172 
3173  Vector shape_x(dof_x), shape_s(dof_s);
3174  const IntegrationRule &ir = el_s.GetNodes();
3175  Vector surf_fit_grad_s(dim);
3176  surf_fit_grad_s = 0.0;
3177 
3178  for (int s = 0; s < dof_s; s++)
3179  {
3180  if ((*surf_fit_marker)[dofs[s]] == false) { continue; }
3181 
3182  const IntegrationPoint &ip = ir.IntPoint(s);
3183  Tpr.SetIntPoint(&ip);
3184  el_x.CalcShape(ip, shape_x);
3185  el_s.CalcShape(ip, shape_s);
3186 
3187  // Note that this gradient is already in physical space.
3188  surf_fit_grad_e.MultTranspose(shape_s, surf_fit_grad_s);
3189  surf_fit_grad_s *= 2.0 * surf_fit_normal *
3190  surf_fit_coeff->Eval(Tpr, ip) * sigma_e(s);
3191 
3192  AddMultVWt(shape_x, surf_fit_grad_s, mat);
3193  }
3194 }
3195 
3198  DenseMatrix &mat)
3199 {
3200  const int el_id = Tpr.ElementNo;
3201  // Check if the element has any DOFs marked for surface fitting.
3202  Array<int> dofs;
3203  surf_fit_gf->FESpace()->GetElementDofs(el_id, dofs);
3204  int count = 0;
3205  for (int s = 0; s < dofs.Size(); s++)
3206  {
3207  count += ((*surf_fit_marker)[dofs[s]]) ? 1 : 0;
3208  }
3209  if (count == 0) { return; }
3210 
3211  const FiniteElement &el_s = *surf_fit_gf->FESpace()->GetFE(el_id);
3212 
3213  const int dof_x = el_x.GetDof(), dim = el_x.GetDim(),
3214  dof_s = el_s.GetDof();
3215 
3216  Vector sigma_e;
3217  surf_fit_gf->GetSubVector(dofs, sigma_e);
3218 
3219  DenseMatrix surf_fit_grad_e(dof_s, dim);
3220  Vector grad_ptr(surf_fit_grad_e.GetData(), dof_s * dim);
3221  DenseMatrix grad_phys;
3222  el_s.ProjectGrad(el_s, Tpr, grad_phys);
3223  grad_phys.Mult(sigma_e, grad_ptr);
3224 
3225  DenseMatrix surf_fit_hess_e(dof_s, dim*dim);
3226  Vector hess_ptr(surf_fit_hess_e.GetData(), dof_s*dim*dim);
3227  surf_fit_hess_e.SetSize(dof_s*dim, dim);
3228  Mult(grad_phys, surf_fit_grad_e, surf_fit_hess_e);
3229  surf_fit_hess_e.SetSize(dof_s, dim * dim);
3230 
3231  const IntegrationRule &ir = el_s.GetNodes();
3232  Vector shape_x(dof_x), shape_s(dof_s);
3233 
3234  Vector surf_fit_grad_s(dim);
3235  DenseMatrix surf_fit_hess_s(dim, dim);
3236 
3237 
3238  for (int s = 0; s < dof_s; s++)
3239  {
3240  if ((*surf_fit_marker)[dofs[s]] == false) { continue; }
3241 
3242  const IntegrationPoint &ip = ir.IntPoint(s);
3243  Tpr.SetIntPoint(&ip);
3244  el_x.CalcShape(ip, shape_x);
3245  el_s.CalcShape(ip, shape_s);
3246 
3247  // These are the sums over k at the dof s (looking at the notes).
3248  surf_fit_grad_e.MultTranspose(shape_s, surf_fit_grad_s);
3249  Vector gg_ptr(surf_fit_hess_s.GetData(), dim * dim);
3250  surf_fit_hess_e.MultTranspose(shape_s, gg_ptr);
3251 
3252  // Loops over the local matrix.
3253  const double w = surf_fit_normal * surf_fit_coeff->Eval(Tpr, ip);
3254  for (int i = 0; i < dof_x * dim; i++)
3255  {
3256  const int idof = i % dof_x, idim = i / dof_x;
3257  for (int j = 0; j <= i; j++)
3258  {
3259  const int jdof = j % dof_x, jdim = j / dof_x;
3260  const double entry =
3261  w * ( 2.0 * surf_fit_grad_s(idim) * shape_x(idof) *
3262  /* */ surf_fit_grad_s(jdim) * shape_x(jdof) +
3263  2.0 * sigma_e(s) * surf_fit_hess_s(idim, jdim) *
3264  /* */ shape_x(idof) * shape_x(jdof));
3265  mat(i, j) += entry;
3266  if (i != j) { mat(j, i) += entry; }
3267  }
3268  }
3269  }
3270 }
3271 
3274  Vector &elfun, const int dofidx,
3275  const int dir, const double e_fx,
3276  bool update_stored)
3277 {
3278  int dof = el.GetDof();
3279  int idx = dir*dof+dofidx;
3280  elfun[idx] += dx;
3281  double e_fxph = GetElementEnergy(el, T, elfun);
3282  elfun[idx] -= dx;
3283  double dfdx = (e_fxph-e_fx)/dx;
3284 
3285  if (update_stored)
3286  {
3287  (*(ElemPertEnergy[T.ElementNo]))(idx) = e_fxph;
3288  (*(ElemDer[T.ElementNo]))(idx) = dfdx;
3289  }
3290 
3291  return dfdx;
3292 }
3293 
3296  const Vector &elfun,
3297  Vector &elvect)
3298 {
3299  const int dof = el.GetDof(), dim = el.GetDim(), elnum = T.ElementNo;
3300  if (elnum>=ElemDer.Size())
3301  {
3302  ElemDer.Append(new Vector);
3303  ElemPertEnergy.Append(new Vector);
3304  ElemDer[elnum]->SetSize(dof*dim);
3305  ElemPertEnergy[elnum]->SetSize(dof*dim);
3306  }
3307 
3308  elvect.SetSize(dof*dim);
3309  Vector elfunmod(elfun);
3310 
3311  // In GetElementEnergy(), skip terms that have exact derivative calculations.
3312  fd_call_flag = true;
3313 
3314  // Energy for unperturbed configuration.
3315  const double e_fx = GetElementEnergy(el, T, elfun);
3316 
3317  for (int j = 0; j < dim; j++)
3318  {
3319  for (int i = 0; i < dof; i++)
3320  {
3321  if (discr_tc)
3322  {
3324  el, T, i, j, discr_tc->GetTspecPert1H());
3325  }
3326  elvect(j*dof+i) = GetFDDerivative(el, T, elfunmod, i, j, e_fx, true);
3328  }
3329  }
3330  fd_call_flag = false;
3331 
3332  // Contributions from adaptive limiting, surface fitting (exact derivatives).
3333  if (adapt_lim_gf || surf_fit_gf)
3334  {
3336  const int nqp = ir.GetNPoints();
3337  DenseTensor Jtr(dim, dim, nqp);
3338  targetC->ComputeElementTargets(T.ElementNo, el, ir, elfun, Jtr);
3339 
3341  Tpr.SetFE(&el);
3342  Tpr.ElementNo = T.ElementNo;
3343  Tpr.Attribute = T.Attribute;
3344  Tpr.mesh = T.mesh;
3345  PMatI.UseExternalData(elfun.GetData(), dof, dim);
3346  Tpr.GetPointMat().Transpose(PMatI); // PointMat = PMatI^T
3347 
3348  Vector weights(nqp);
3349  for (int q = 0; q < nqp; q++)
3350  {
3351  weights(q) = ir.IntPoint(q).weight * Jtr(q).Det();
3352  }
3353 
3354  PMatO.UseExternalData(elvect.GetData(), dof, dim);
3355  if (adapt_lim_gf) { AssembleElemVecAdaptLim(el, Tpr, ir, weights, PMatO); }
3356  if (surf_fit_gf) { AssembleElemVecSurfFit(el, Tpr, PMatO); }
3357  }
3358 }
3359 
3362  const Vector &elfun,
3363  DenseMatrix &elmat)
3364 {
3365  const int dof = el.GetDof(), dim = el.GetDim();
3366 
3367  elmat.SetSize(dof*dim);
3368  Vector elfunmod(elfun);
3369 
3370  const Vector &ElemDerLoc = *(ElemDer[T.ElementNo]);
3371  const Vector &ElemPertLoc = *(ElemPertEnergy[T.ElementNo]);
3372 
3373  // In GetElementEnergy(), skip terms that have exact derivative calculations.
3374  fd_call_flag = true;
3375  for (int i = 0; i < dof; i++)
3376  {
3377  for (int j = 0; j < i+1; j++)
3378  {
3379  for (int k1 = 0; k1 < dim; k1++)
3380  {
3381  for (int k2 = 0; k2 < dim; k2++)
3382  {
3383  elfunmod(k2*dof+j) += dx;
3384 
3385  if (discr_tc)
3386  {
3388  el, T, j, k2, discr_tc->GetTspecPert1H());
3389  if (j != i)
3390  {
3392  el, T, i, k1, discr_tc->GetTspecPert1H());
3393  }
3394  else // j==i
3395  {
3396  if (k1 != k2)
3397  {
3398  int idx = k1+k2-1;
3400  el, T, i, idx, discr_tc->GetTspecPertMixH());
3401  }
3402  else // j==i && k1==k2
3403  {
3405  el, T, i, k1, discr_tc->GetTspecPert2H());
3406  }
3407  }
3408  }
3409 
3410  double e_fx = ElemPertLoc(k2*dof+j);
3411  double e_fpxph = GetFDDerivative(el, T, elfunmod, i, k1, e_fx,
3412  false);
3413  elfunmod(k2*dof+j) -= dx;
3414  double e_fpx = ElemDerLoc(k1*dof+i);
3415 
3416  elmat(k1*dof+i, k2*dof+j) = (e_fpxph - e_fpx) / dx;
3417  elmat(k2*dof+j, k1*dof+i) = (e_fpxph - e_fpx) / dx;
3418 
3419  if (discr_tc)
3420  {
3423  }
3424  }
3425  }
3426  }
3427  }
3428  fd_call_flag = false;
3429 
3430  // Contributions from adaptive limiting.
3431  if (adapt_lim_gf || surf_fit_gf)
3432  {
3434  const int nqp = ir.GetNPoints();
3435  DenseTensor Jtr(dim, dim, nqp);
3436  targetC->ComputeElementTargets(T.ElementNo, el, ir, elfun, Jtr);
3437 
3439  Tpr.SetFE(&el);
3440  Tpr.ElementNo = T.ElementNo;
3441  Tpr.Attribute = T.Attribute;
3442  Tpr.mesh = T.mesh;
3443  PMatI.UseExternalData(elfun.GetData(), dof, dim);
3444  Tpr.GetPointMat().Transpose(PMatI); // PointMat = PMatI^T
3445 
3446  Vector weights(nqp);
3447  for (int q = 0; q < nqp; q++)
3448  {
3449  weights(q) = ir.IntPoint(q).weight * Jtr(q).Det();
3450  }
3451 
3452  if (adapt_lim_gf) { AssembleElemGradAdaptLim(el, Tpr, ir, weights, elmat); }
3453  if (surf_fit_gf) { AssembleElemGradSurfFit(el, Tpr, elmat); }
3454  }
3455 }
3456 
3458 {
3459  if (!surf_fit_coeff) { return; }
3460 
3461  if (surf_fit_coeff)
3462  {
3463  auto cf = dynamic_cast<ConstantCoefficient *>(surf_fit_coeff);
3464  MFEM_VERIFY(cf, "Dynamic weight works only with a ConstantCoefficient.");
3465  cf->constant *= factor;
3466  }
3467 }
3468 
3470 {
3471  if (surf_fit_coeff)
3472  {
3473  auto cf = dynamic_cast<ConstantCoefficient *>(surf_fit_coeff);
3474  MFEM_VERIFY(cf, "Dynamic weight works only with a ConstantCoefficient.");
3475  return cf->constant;
3476  }
3477  return 0.0;
3478 }
3479 
3481 {
3483  metric_normal = 1.0 / metric_normal;
3484  lim_normal = 1.0 / lim_normal;
3485  //if (surf_fit_gf) { surf_fit_normal = 1.0 / surf_fit_normal; }
3487 }
3488 
3489 #ifdef MFEM_USE_MPI
3491 {
3492  double loc[3];
3493  ComputeNormalizationEnergies(x, loc[0], loc[1], loc[2]);
3494  double rdc[3];
3495  MPI_Allreduce(loc, rdc, 3, MPI_DOUBLE, MPI_SUM, x.ParFESpace()->GetComm());
3496  metric_normal = 1.0 / rdc[0];
3497  lim_normal = 1.0 / rdc[1];
3498  // if (surf_fit_gf) { surf_fit_normal = 1.0 / rdc[2]; }
3500 }
3501 #endif
3502 
3504  double &metric_energy,
3505  double &lim_energy,
3506  double &surf_fit_gf_energy)
3507 {
3508  Array<int> vdofs;
3509  Vector x_vals;
3510  const FiniteElementSpace* const fes = x.FESpace();
3511 
3512  const int dim = fes->GetMesh()->Dimension();
3513  Jrt.SetSize(dim);
3514  Jpr.SetSize(dim);
3515  Jpt.SetSize(dim);
3516 
3517  metric_energy = 0.0;
3518  lim_energy = 0.0;
3519  surf_fit_gf_energy = 0.0;
3520  for (int i = 0; i < fes->GetNE(); i++)
3521  {
3522  const FiniteElement *fe = fes->GetFE(i);
3524  const int nqp = ir.GetNPoints();
3525  DenseTensor Jtr(dim, dim, nqp);
3526  const int dof = fe->GetDof();
3527  DSh.SetSize(dof, dim);
3528 
3529  fes->GetElementVDofs(i, vdofs);
3530  x.GetSubVector(vdofs, x_vals);
3531  PMatI.UseExternalData(x_vals.GetData(), dof, dim);
3532 
3533  targetC->ComputeElementTargets(i, *fe, ir, x_vals, Jtr);
3534 
3535  for (int q = 0; q < nqp; q++)
3536  {
3537  const IntegrationPoint &ip = ir.IntPoint(q);
3539  CalcInverse(Jtr(q), Jrt);
3540  const double weight = ip.weight * Jtr(q).Det();
3541 
3542  fe->CalcDShape(ip, DSh);
3543  MultAtB(PMatI, DSh, Jpr);
3544  Mult(Jpr, Jrt, Jpt);
3545 
3546  metric_energy += weight * metric->EvalW(Jpt);
3547  lim_energy += weight;
3548  }
3549 
3550  // Normalization of the surface fitting term.
3551  if (surf_fit_gf)
3552  {
3553  Array<int> dofs;
3554  Vector sigma_e;
3555  surf_fit_gf->FESpace()->GetElementDofs(i, dofs);
3556  surf_fit_gf->GetSubVector(dofs, sigma_e);
3557  for (int s = 0; s < dofs.Size(); s++)
3558  {
3559  if ((*surf_fit_marker)[dofs[s]] == true)
3560  {
3561  surf_fit_gf_energy += sigma_e(s) * sigma_e(s);
3562  }
3563  }
3564  }
3565  }
3566 
3567  if (targetC->ContainsVolumeInfo() == false)
3568  {
3569  // Special case when the targets don't contain volumetric information.
3570  lim_energy = fes->GetNE();
3571  }
3572 }
3573 
3575  const FiniteElementSpace &fes)
3576 {
3577  const FiniteElement *fe = fes.GetFE(0);
3579  const int NE = fes.GetMesh()->GetNE(), dim = fe->GetDim(),
3580  dof = fe->GetDof(), nsp = ir.GetNPoints();
3581 
3582  Array<int> xdofs(dof * dim);
3583  DenseMatrix dshape(dof, dim), pos(dof, dim);
3584  Vector posV(pos.Data(), dof * dim);
3585  Jpr.SetSize(dim);
3586 
3587  dx = std::numeric_limits<float>::max();
3588 
3589  double detv_sum;
3590  double detv_avg_min = std::numeric_limits<float>::max();
3591  for (int i = 0; i < NE; i++)
3592  {
3593  fes.GetElementVDofs(i, xdofs);
3594  x.GetSubVector(xdofs, posV);
3595  detv_sum = 0.;
3596  for (int j = 0; j < nsp; j++)
3597  {
3598  fes.GetFE(i)->CalcDShape(ir.IntPoint(j), dshape);
3599  MultAtB(pos, dshape, Jpr);
3600  detv_sum += std::fabs(Jpr.Det());
3601  }
3602  double detv_avg = pow(detv_sum/nsp, 1./dim);
3603  detv_avg_min = std::min(detv_avg, detv_avg_min);
3604  }
3605  dx = detv_avg_min / dxscale;
3606 }
3607 
3609 {
3610  if (discr_tc)
3611  {
3612  PA.Jtr_needs_update = true;
3613  }
3614  // Update adapt_lim_gf if adaptive limiting is enabled.
3616 
3617  // Update surf_fit_gf if surface fitting is enabled.
3618  if (surf_fit_gf)
3619  {
3621  }
3622 }
3623 
3625 {
3626  if (!fdflag) { return; }
3627  ComputeMinJac(x, fes);
3628 }
3629 
3630 #ifdef MFEM_USE_MPI
3632  const ParFiniteElementSpace &pfes)
3633 {
3634  if (!fdflag) { return; }
3635  ComputeMinJac(x, pfes);
3636  double min_jac_all;
3637  MPI_Allreduce(&dx, &min_jac_all, 1, MPI_DOUBLE, MPI_MIN, pfes.GetComm());
3638  dx = min_jac_all;
3639 }
3640 #endif
3641 
3643 {
3644  fdflag = true;
3645  const FiniteElementSpace *fes = x.FESpace();
3646  ComputeFDh(x,*fes);
3647  if (discr_tc)
3648  {
3652  }
3653 }
3654 
3655 #ifdef MFEM_USE_MPI
3657 {
3658  fdflag = true;
3659  const ParFiniteElementSpace *pfes = x.ParFESpace();
3660  ComputeFDh(x,*pfes);
3661  if (discr_tc)
3662  {
3666  }
3667 }
3668 #endif
3669 
3671  const GridFunction &dist,
3672  Coefficient &w0,
3673  TMOP_LimiterFunction *lfunc)
3674 {
3675  MFEM_VERIFY(tmopi.Size() > 0, "No TMOP_Integrators were added.");
3676 
3677  tmopi[0]->EnableLimiting(n0, dist, w0, lfunc);
3678  for (int i = 1; i < tmopi.Size(); i++) { tmopi[i]->DisableLimiting(); }
3679 }
3680 
3682  Coefficient &w0,
3683  TMOP_LimiterFunction *lfunc)
3684 {
3685  MFEM_VERIFY(tmopi.Size() > 0, "No TMOP_Integrators were added.");
3686 
3687  tmopi[0]->EnableLimiting(n0, w0, lfunc);
3688  for (int i = 1; i < tmopi.Size(); i++) { tmopi[i]->DisableLimiting(); }
3689 }
3690 
3692 {
3693  MFEM_VERIFY(tmopi.Size() > 0, "No TMOP_Integrators were added.");
3694 
3695  tmopi[0]->SetLimitingNodes(n0);
3696  for (int i = 1; i < tmopi.Size(); i++) { tmopi[i]->DisableLimiting(); }
3697 }
3698 
3701  const Vector &elfun)
3702 {
3703  double energy= 0.0;
3704  for (int i = 0; i < tmopi.Size(); i++)
3705  {
3706  energy += tmopi[i]->GetElementEnergy(el, T, elfun);
3707  }
3708  return energy;
3709 }
3710 
3713  const Vector &elfun,
3714  Vector &elvect)
3715 {
3716  MFEM_VERIFY(tmopi.Size() > 0, "No TMOP_Integrators were added.");
3717 
3718  tmopi[0]->AssembleElementVector(el, T, elfun, elvect);
3719  for (int i = 1; i < tmopi.Size(); i++)
3720  {
3721  Vector elvect_i;
3722  tmopi[i]->AssembleElementVector(el, T, elfun, elvect_i);
3723  elvect += elvect_i;
3724  }
3725 }
3726 
3729  const Vector &elfun,
3730  DenseMatrix &elmat)
3731 {
3732  MFEM_VERIFY(tmopi.Size() > 0, "No TMOP_Integrators were added.");
3733 
3734  tmopi[0]->AssembleElementGrad(el, T, elfun, elmat);
3735  for (int i = 1; i < tmopi.Size(); i++)
3736  {
3737  DenseMatrix elmat_i;
3738  tmopi[i]->AssembleElementGrad(el, T, elfun, elmat_i);
3739  elmat += elmat_i;
3740  }
3741 }
3742 
3745  const Vector &elfun,
3746  const IntegrationRule &irule)
3747 {
3748  double energy= 0.0;
3749  for (int i = 0; i < tmopi.Size(); i++)
3750  {
3751  energy += tmopi[i]->GetRefinementElementEnergy(el, T, elfun, irule);
3752  }
3753  return energy;
3754 }
3755 
3757  const FiniteElement &el,
3759  const Vector &elfun)
3760 {
3761  double energy= 0.0;
3762  for (int i = 0; i < tmopi.Size(); i++)
3763  {
3764  energy += tmopi[i]->GetDerefinementElementEnergy(el, T, elfun);
3765  }
3766  return energy;
3767 }
3768 
3770 {
3771  const int cnt = tmopi.Size();
3772  double total_integral = 0.0;
3773  for (int i = 0; i < cnt; i++)
3774  {
3775  tmopi[i]->EnableNormalization(x);
3776  total_integral += 1.0 / tmopi[i]->metric_normal;
3777  }
3778  for (int i = 0; i < cnt; i++)
3779  {
3780  tmopi[i]->metric_normal = 1.0 / total_integral;
3781  }
3782 }
3783 
3784 #ifdef MFEM_USE_MPI
3786 {
3787  const int cnt = tmopi.Size();
3788  double total_integral = 0.0;
3789  for (int i = 0; i < cnt; i++)
3790  {
3791  tmopi[i]->ParEnableNormalization(x);
3792  total_integral += 1.0 / tmopi[i]->metric_normal;
3793  }
3794  for (int i = 0; i < cnt; i++)
3795  {
3796  tmopi[i]->metric_normal = 1.0 / total_integral;
3797  }
3798 }
3799 #endif
3800 
3802 {
3803  for (int i = 0; i < tmopi.Size(); i++)
3804  {
3805  tmopi[i]->AssemblePA(fes);
3806  }
3807 }
3808 
3810  const FiniteElementSpace &fes)
3811 {
3812  for (int i = 0; i < tmopi.Size(); i++)
3813  {
3814  tmopi[i]->AssembleGradPA(xe,fes);
3815  }
3816 }
3817 
3819 {
3820  for (int i = 0; i < tmopi.Size(); i++)
3821  {
3822  tmopi[i]->AssembleGradDiagonalPA(de);
3823  }
3824 }
3825 
3827 {
3828  for (int i = 0; i < tmopi.Size(); i++)
3829  {
3830  tmopi[i]->AddMultPA(xe, ye);
3831  }
3832 }
3833 
3835 {
3836  for (int i = 0; i < tmopi.Size(); i++)
3837  {
3838  tmopi[i]->AddMultGradPA(re, ce);
3839  }
3840 }
3841 
3843 {
3844  double energy = 0.0;
3845  for (int i = 0; i < tmopi.Size(); i++)
3846  {
3847  energy += tmopi[i]->GetLocalStateEnergyPA(xe);
3848  }
3849  return energy;
3850 }
3851 
3853  const TargetConstructor &tc,
3854  const Mesh &mesh, GridFunction &metric_gf)
3855 {
3856  const int NE = mesh.GetNE();
3857  const GridFunction &nodes = *mesh.GetNodes();
3858  const int dim = mesh.Dimension();
3859  DenseMatrix Winv(dim), T(dim), A(dim), dshape, pos;
3860  Array<int> pos_dofs, gf_dofs;
3861  DenseTensor W;
3862  Vector posV;
3863 
3864  for (int i = 0; i < NE; i++)
3865  {
3866  const FiniteElement &fe_pos = *nodes.FESpace()->GetFE(i);
3867  const IntegrationRule &ir = metric_gf.FESpace()->GetFE(i)->GetNodes();
3868  const int nsp = ir.GetNPoints(), dof = fe_pos.GetDof();
3869 
3870  dshape.SetSize(dof, dim);
3871  pos.SetSize(dof, dim);
3872  posV.SetDataAndSize(pos.Data(), dof * dim);
3873 
3874  metric_gf.FESpace()->GetElementDofs(i, gf_dofs);
3875  nodes.FESpace()->GetElementVDofs(i, pos_dofs);
3876  nodes.GetSubVector(pos_dofs, posV);
3877 
3878  W.SetSize(dim, dim, nsp);
3879  tc.ComputeElementTargets(i, fe_pos, ir, posV, W);
3880 
3881  for (int j = 0; j < nsp; j++)
3882  {
3883  const DenseMatrix &Wj = W(j);
3884  metric.SetTargetJacobian(Wj);
3885  CalcInverse(Wj, Winv);
3886 
3887  const IntegrationPoint &ip = ir.IntPoint(j);
3888  fe_pos.CalcDShape(ip, dshape);
3889  MultAtB(pos, dshape, A);
3890  Mult(A, Winv, T);
3891 
3892  metric_gf(gf_dofs[j]) = metric.EvalW(T);
3893  }
3894  }
3895 }
3896 
3897 } // namespace mfem
int GetNPoints() const
Returns the number of the points in the integration rule.
Definition: intrules.hpp:247
Abstract class for all finite elements.
Definition: fe_base.hpp:235
InvariantsEvaluator2D< double > ie
Definition: tmop.hpp:343
int Size() const
For backward compatibility define Size to be synonym of Width()
Definition: densemat.hpp:102
virtual double GetDerefinementElementEnergy(const FiniteElement &el, ElementTransformation &T, const Vector &elfun)
Definition: tmop.cpp:2714
virtual void AssembleH(const DenseMatrix &Jpt, const DenseMatrix &DS, const double weight, DenseMatrix &A) const
Evaluate the derivative of the 1st Piola-Kirchhoff stress tensor and assemble its contribution to the...
Definition: tmop.cpp:344
virtual int Id() const
Return the metric ID.
Definition: tmop.hpp:74
virtual void SetParDiscreteTargetSize(const ParGridFunction &tspec_)
Definition: tmop.cpp:1375
virtual void AssembleH(const DenseMatrix &Jpt, const DenseMatrix &DS, const double weight, DenseMatrix &A) const
Evaluate the derivative of the 1st Piola-Kirchhoff stress tensor and assemble its contribution to the...
Definition: tmop.cpp:73
int Size() const
Return the logical size of the array.
Definition: array.hpp:138
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:961
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:314
InvariantsEvaluator3D< double > ie
Definition: tmop.hpp:506
virtual void SetInitialField(const Vector &init_nodes, const Vector &init_field)=0
int GetDim() const
Returns the reference space dimension for the finite element.
Definition: fe_base.hpp:311
virtual void AssembleH(const DenseMatrix &Jpt, const DenseMatrix &DS, const double weight, DenseMatrix &A) const =0
Evaluate the derivative of the 1st Piola-Kirchhoff stress tensor and assemble its contribution to the...
void AddMultVWt(const Vector &v, const Vector &w, DenseMatrix &VWt)
VWt += v w^t.
Definition: densemat.cpp:2761
int GetNDofs() const
Returns number of degrees of freedom.
Definition: fespace.hpp:560
void AssembleElemVecSurfFit(const FiniteElement &el_x, IsoparametricTransformation &Tpr, DenseMatrix &mat)
Definition: tmop.cpp:3142
void SetSerialMetaInfo(const Mesh &m, const FiniteElementCollection &fec, int num_comp)
Definition: tmop.cpp:2269
Class for an integration rule - an Array of IntegrationPoint.
Definition: intrules.hpp:90
VectorCoefficient * vector_tspec
Definition: tmop.hpp:1053
void GetVectorValues(int i, const IntegrationRule &ir, DenseMatrix &vals, DenseMatrix &tr) const
Definition: gridfunc.cpp:703
void EnableFiniteDifferences(const GridFunction &x)
Enables FD-based approximation and computes dx.
Definition: tmop.cpp:3642
void NewDataAndSize(double *d, int s)
Set the Vector data and size, deleting the old data, if owned.
Definition: vector.hpp:162
const TargetType target_type
Definition: tmop.hpp:935
Class for grid function - Vector with associated FE space.
Definition: gridfunc.hpp:30
DenseMatrix PMatO
Definition: tmop.hpp:1376
virtual void AddMultPA(const Vector &, Vector &) const
Method for partially assembled action.
Definition: tmop.cpp:3826
DenseTensor Jtr
Definition: tmop.hpp:1406
void UseExternalData(double *ext_data, int i, int j, int k)
Definition: densemat.hpp:847
double & min_detT
Definition: tmop.hpp:256
virtual void Update(bool want_transform=true)
Reflect changes in the mesh: update number of DOFs, etc. Also, calculate GridFunction transformation ...
Definition: fespace.cpp:3336
InvariantsEvaluator3D< double > ie
Definition: tmop.hpp:525
const DenseMatrix * Jtr
Definition: tmop.hpp:26
InvariantsEvaluator3D< double > ie
Definition: tmop.hpp:581
bool IsVariableOrder() const
Returns true if the space contains elements of varying polynomial orders.
Definition: fespace.hpp:455
void Assemble_ddI2b(scalar_t w, scalar_t *A)
Definition: invariants.hpp:384
const IntegrationRule & Get(int GeomType, int Order)
Returns an integration rule for given GeomType and Order.
Definition: intrules.cpp:923
virtual void EvalP(const DenseMatrix &Jpt, DenseMatrix &P) const
Evaluate the 1st Piola-Kirchhoff stress tensor, P = P(Jpt).
Definition: tmop.cpp:674
double GetSurfaceFittingWeight()
Get the surface fitting weight.
Definition: tmop.cpp:3469
ParMesh * GetParMesh() const
Definition: pfespace.hpp:277
Base class for vector Coefficients that optionally depend on time and space.
A coefficient that is constant across space and time.
Definition: coefficient.hpp:78
void Assemble_ddI2b(scalar_t w, scalar_t *A)
DenseMatrix Jrt
Definition: tmop.hpp:1376
void SetDerivativeMatrix(int height, const scalar_t *Deriv)
The Deriv matrix is dof x 2, using column-major storage.
Definition: invariants.hpp:184
const GridFunction * lim_dist
Definition: tmop.hpp:1328
virtual void SetSerialDiscreteTargetOrientation(const GridFunction &tspec_)
Definition: tmop.cpp:1474
virtual double GetElementEnergy(const FiniteElement &el, ElementTransformation &T, const Vector &elfun)
Compute the local energy.
Definition: tmop.cpp:3699
virtual void EvalGrad(DenseMatrix &K, ElementTransformation &T, const IntegrationPoint &ip, int comp)=0
Evaluate the derivative of the matrix coefficient with respect to comp in the element described by T ...
int GetNumGeometries(int dim) const
Return the number of geometries of the given dimension present in the mesh.
Definition: mesh.cpp:5877
InvariantsEvaluator2D< double > ie
Definition: tmop.hpp:291
const double eps
Definition: tmop.hpp:505
void EnableNormalization(const GridFunction &x)
Computes the normalization factors of the metric and limiting integrals using the mesh position given...
Definition: tmop.cpp:3480
const scalar_t * Get_dI3b()
Definition: invariants.hpp:780
void SetSize(int s)
Resize the vector to size s.
Definition: vector.hpp:521
void InterpolateTMOP_QualityMetric(TMOP_QualityMetric &metric, const TargetConstructor &tc, const Mesh &mesh, GridFunction &metric_gf)
Interpolates the metric&#39;s values at the nodes of metric_gf.
Definition: tmop.cpp:3852
TMOP_QualityMetric * metric
Definition: tmop.hpp:1310
InvariantsEvaluator2D< double > ie
Definition: tmop.hpp:308
double Det() const
Definition: densemat.cpp:436
Base class for limiting functions to be used in class TMOP_Integrator.
Definition: tmop.hpp:798
virtual double EvalW(const DenseMatrix &Jpt) const =0
Evaluate the strain energy density function, W = W(Jpt).
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:935
virtual void Eval_d2(const Vector &x, const Vector &x0, double dist, DenseMatrix &d2) const =0
Returns the Hessian of the limiting function f(x, x0, d) with respect to x.
virtual void ComputeElementTargetsGradient(const IntegrationRule &ir, const Vector &elfun, IsoparametricTransformation &Tpr, DenseTensor &dJtr) const
Definition: tmop.cpp:1832
void Mult(const Table &A, const Table &B, Table &C)
C = A * B (as boolean matrices)
Definition: table.cpp:472
virtual void AssembleH(const DenseMatrix &Jpt, const DenseMatrix &DS, const double weight, DenseMatrix &A) const
Evaluate the derivative of the 1st Piola-Kirchhoff stress tensor and assemble its contribution to the...
Definition: tmop.cpp:866
int Width() const
Get the width (size of input) of the Operator. Synonym with NumCols().
Definition: operator.hpp:72
virtual void EvalP(const DenseMatrix &Jpt, DenseMatrix &P) const
Evaluate the 1st Piola-Kirchhoff stress tensor, P = P(Jpt).
Definition: tmop.cpp:793
virtual void Update(bool want_transform=true)
Definition: pfespace.cpp:3284
double Norml2() const
Returns the l2 norm of the vector.
Definition: vector.cpp:792
void ParUpdateAfterMeshTopologyChange()
Definition: tmop.cpp:1343
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:202
DenseMatrix Jpr
Definition: tmop.hpp:1376
virtual void Eval(DenseMatrix &K, ElementTransformation &T, const IntegrationPoint &ip)=0
Evaluate the matrix coefficient in the element described by T at the point ip, storing the result in ...
virtual void EvalP(const DenseMatrix &Jpt, DenseMatrix &P) const
Evaluate the 1st Piola-Kirchhoff stress tensor, P = P(Jpt).
Definition: tmop.cpp:584
void SetIntPoint(const IntegrationPoint *ip)
Set the integration point ip that weights and Jacobians will be evaluated at.
Definition: eltrans.hpp:93
void GetSubVector(const Array< int > &dofs, Vector &elemvect) const
Extract entries listed in dofs to the output Vector elemvect.
Definition: vector.cpp:534
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:993
virtual void EvalP(const DenseMatrix &Jpt, DenseMatrix &P) const =0
Evaluate the 1st Piola-Kirchhoff stress tensor, P = P(Jpt).
InvariantsEvaluator2D< double > ie
Definition: tmop.hpp:111
virtual void ComputeElementTargetsGradient(const IntegrationRule &ir, const Vector &elfun, IsoparametricTransformation &Tpr, DenseTensor &dJtr) const
Definition: tmop.cpp:1270
ParFiniteElementSpace * pfes
Definition: tmop.hpp:863
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:664
Data type dense matrix using column-major storage.
Definition: densemat.hpp:23
void GetSurfaceFittingErrors(double &err_avg, double &err_max)
Definition: tmop.cpp:2436
int Size() const
Returns the size of the vector.
Definition: vector.hpp:199
struct::_p_Mat * Mat
Definition: petsc.hpp:70
void EnableAdaptiveLimiting(const GridFunction &z0, Coefficient &coeff, AdaptivityEvaluator &ae)
Restriction of the node positions to certain regions.
Definition: tmop.cpp:2365
virtual void AssembleH(const DenseMatrix &Jpt, const DenseMatrix &DS, const double weight, DenseMatrix &A) const
Evaluate the derivative of the 1st Piola-Kirchhoff stress tensor and assemble its contribution to the...
Definition: tmop.cpp:737
int GetNE() const
Returns number of elements.
Definition: mesh.hpp:928
void EnableNormalization(const GridFunction &x)
Normalization factor that considers all integrators in the combination.
Definition: tmop.cpp:3769
virtual void AssembleH(const DenseMatrix &Jpt, const DenseMatrix &DS, const double weight, DenseMatrix &A) const
Evaluate the derivative of the 1st Piola-Kirchhoff stress tensor and assemble its contribution to the...
Definition: tmop.cpp:596
double & min_detT
Definition: tmop.hpp:524
virtual void Mult(const Vector &x, Vector &y) const =0
Operator application: y=A(x).
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:520
Abstract parallel finite element space.
Definition: pfespace.hpp:28
virtual void AssemblePA(const FiniteElementSpace &)
Method defining partial assembly.
Definition: tmop.cpp:3801
const FiniteElement * GetFE() const
Get the current element used to compute the transformations.
Definition: eltrans.hpp:389
InvariantsEvaluator2D< double > ie
Definition: tmop.hpp:225
const IntegrationRule * ir
Definition: tmop.hpp:1414
InvariantsEvaluator2D< double > ie
Definition: tmop.hpp:326
void AssembleElementVectorFD(const FiniteElement &el, ElementTransformation &T, const Vector &elfun, Vector &elvect)
Definition: tmop.cpp:3294
const GridFunction * nodes
Definition: tmop.hpp:932
virtual void SetSerialDiscreteTargetSkew(const GridFunction &tspec_)
Definition: tmop.cpp:1458
void UpdateSurfaceFittingWeight(double factor)
Update the surface fitting weight as surf_fit_coeff *= factor;.
Definition: tmop.cpp:3457
struct mfem::TMOP_Integrator::@13 PA
void UpdateGradientTargetSpecification(const Vector &x, double dx, bool use_flag=false)
Definition: tmop.cpp:2179
void SetRefinementSubElement(int amr_el_)
Definition: tmop.hpp:1291
void Assemble_TProd(scalar_t w, const scalar_t *X, const scalar_t *Y, scalar_t *A)
Definition: invariants.hpp:432
void Assemble_ddI1(scalar_t w, scalar_t *A)
Definition: invariants.hpp:226
double * GetData() const
Returns the matrix data array.
Definition: densemat.hpp:115
virtual int OrderJ() const
Return the order of the elements of the Jacobian of the transformation.
Definition: eltrans.cpp:443
virtual void ProjectGrad(const FiniteElement &fe, ElementTransformation &Trans, DenseMatrix &grad) const
Compute the discrete gradient matrix from the given FiniteElement onto &#39;this&#39; FiniteElement. The ElementTransformation is included to support cases when the matrix depends on it.
Definition: fe_base.cpp:160
double * GetData() const
Return a pointer to the beginning of the Vector data.
Definition: vector.hpp:208
virtual void EvalP(const DenseMatrix &Jpt, DenseMatrix &P) const
Evaluate the 1st Piola-Kirchhoff stress tensor, P = P(Jpt).
Definition: tmop.cpp:855
Default limiter function in TMOP_Integrator.
Definition: tmop.hpp:819
Geometry::Type GetElementBaseGeometry(int i) const
Definition: mesh.hpp:1062
virtual void UseDevice(bool use_dev) const
Enable execution of Vector operations using the mfem::Device.
Definition: vector.hpp:117
void Set(double alpha, const double *A)
Set the matrix to alpha * A, assuming that A has the same dimensions as the matrix and uses column-ma...
Definition: densemat.cpp:520
virtual void ComputeElementTargets(int e_id, const FiniteElement &fe, const IntegrationRule &ir, const Vector &elfun, DenseTensor &Jtr) const
Given an element and quadrature rule, computes ref-&gt;target transformation Jacobians for each quadratur...
Definition: tmop.cpp:1617
void AssembleElementGradExact(const FiniteElement &el, ElementTransformation &T, const Vector &elfun, DenseMatrix &elmat)
Definition: tmop.cpp:2937
Array< Vector * > ElemDer
Definition: tmop.hpp:1363
const Vector & GetTspecPertMixH()
Definition: tmop.hpp:1241
void EnableLimiting(const GridFunction &n0, const GridFunction &dist, Coefficient &w0, TMOP_LimiterFunction *lfunc=NULL)
Adds the limiting term to the first integrator. Disables it for the rest.
Definition: tmop.cpp:3670
void Assemble_ddI2(scalar_t w, scalar_t *A)
Definition: invariants.hpp:335
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:432
virtual void ComputeElementTargets(int e_id, const FiniteElement &fe, const IntegrationRule &ir, const Vector &elfun, DenseTensor &Jtr) const
Given an element and quadrature rule, computes ref-&gt;target transformation Jacobians for each quadratur...
Definition: tmop.cpp:1236
Coefficient * scalar_tspec
Definition: tmop.hpp:1052
virtual void EvalP(const DenseMatrix &Jpt, DenseMatrix &P) const
Evaluate the 1st Piola-Kirchhoff stress tensor, P = P(Jpt).
Definition: tmop.cpp:620
virtual void SetSerialDiscreteTargetAspectRatio(const GridFunction &tspec_)
Definition: tmop.cpp:1466
const DenseMatrix & GetGeomToPerfGeomJac(int GeomType) const
Definition: geom.hpp:98
void UpdateTargetSpecification(const Vector &new_x, bool use_flag=false)
Definition: tmop.cpp:1534
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:720
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:107
virtual void AssembleH(const DenseMatrix &Jpt, const DenseMatrix &DS, const double weight, DenseMatrix &A) const
Evaluate the derivative of the 1st Piola-Kirchhoff stress tensor and assemble its contribution to the...
Definition: tmop.cpp:710
virtual void AssembleH(const DenseMatrix &Jpt, const DenseMatrix &DS, const double weight, DenseMatrix &A) const
Evaluate the derivative of the 1st Piola-Kirchhoff stress tensor and assemble its contribution to the...
Definition: tmop.cpp:448
virtual void EvalP(const DenseMatrix &Jpt, DenseMatrix &P) const
Evaluate the 1st Piola-Kirchhoff stress tensor, P = P(Jpt).
Definition: tmop.cpp:33
virtual void AddMultGradPA(const Vector &, Vector &) const
Method for partially assembled gradient action.
Definition: tmop.cpp:3834
virtual void ComputeElementTargetsGradient(const IntegrationRule &ir, const Vector &elfun, IsoparametricTransformation &Tpr, DenseTensor &dJtr) const
Definition: tmop.cpp:1213
Geometry::Type GetGeomType() const
Returns the Geometry::Type of the reference element.
Definition: fe_base.hpp:320
void Add(const DenseMatrix &A, const DenseMatrix &B, double alpha, DenseMatrix &C)
C = A + alpha*B.
Definition: densemat.cpp:1916
int GetNE() const
Returns number of elements in the mesh.
Definition: fespace.hpp:590
Abstract class for local mesh quality metrics in the target-matrix optimization paradigm (TMOP) by P...
Definition: tmop.hpp:23
void FinalizeSerialDiscreteTargetSpec(const GridFunction &tspec_)
Definition: tmop.cpp:1482
Geometry Geometries
Definition: fe.cpp:49
IntegrationPoint & IntPoint(int i)
Returns a reference to the i-th integration point.
Definition: intrules.hpp:250
Coefficient * adapt_lim_coeff
Definition: tmop.hpp:1340
void MultTranspose(const double *x, double *y) const
Multiply a vector with the transpose matrix.
Definition: densemat.cpp:188
virtual ~AdaptivityEvaluator()
Definition: tmop.cpp:2304
void SetJacobian(const scalar_t *Jac)
The Jacobian should use column-major storage.
Definition: invariants.hpp:181
void CalcAdjugateTranspose(const DenseMatrix &a, DenseMatrix &adjat)
Calculate the transposed adjugate of a matrix (for NxN matrices, N=1,2,3)
Definition: densemat.cpp:2146
void UpdateAfterMeshPositionChange(const Vector &new_x)
Definition: tmop.cpp:3608
void Assemble_ddI3b(scalar_t w, scalar_t *A)
InvariantsEvaluator3D< double > ie
Definition: tmop.hpp:545
DofTransformation * GetElementVDofs(int i, Array< int > &vdofs) const
Returns indexes of degrees of freedom in array dofs for i&#39;th element.
Definition: fespace.cpp:281
void SetLimitingNodes(const GridFunction &n0)
Update the original/reference nodes used for limiting.
Definition: tmop.cpp:3691
void SetDiscreteTargetBase(const GridFunction &tspec_)
Definition: tmop.cpp:1413
AdaptivityEvaluator * adapt_lim_eval
Definition: tmop.hpp:1341
MPI_Comm GetComm() const
Definition: pfespace.hpp:273
virtual const FiniteElement * GetFE(int i) const
Definition: pfespace.cpp:535
virtual void EvalP(const DenseMatrix &Jpt, DenseMatrix &P) const
Evaluate the 1st Piola-Kirchhoff stress tensor, P = P(Jpt).
Definition: tmop.cpp:822
void AddMult(const DenseMatrix &b, const DenseMatrix &c, DenseMatrix &a)
Matrix matrix multiplication. A += B * C.
Definition: densemat.cpp:2042
DiscreteAdaptTC * discr_tc
Definition: tmop.hpp:1350
void EnableLimiting(const GridFunction &n0, const GridFunction &dist, Coefficient &w0, TMOP_LimiterFunction *lfunc=NULL)
Limiting of the mesh displacements (general version).
Definition: tmop.cpp:2340
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:554
void UpdateTargetSpecificationAtNode(const FiniteElement &el, ElementTransformation &T, int nodenum, int idir, const Vector &IntData)
Definition: tmop.cpp:1552
int Height() const
Get the height (size of output) of the Operator. Synonym with NumRows().
Definition: operator.hpp:66
void SetTspecFromIntRule(int e_id, const IntegrationRule &intrule)
Definition: tmop.cpp:1583
virtual void AssembleH(const DenseMatrix &Jpt, const DenseMatrix &DS, const double weight, DenseMatrix &A) const
Evaluate the derivative of the 1st Piola-Kirchhoff stress tensor and assemble its contribution to the...
Definition: tmop.cpp:390
virtual void Update()
Transform by the Space UpdateMatrix (e.g., on Mesh change).
Definition: pgridfunc.cpp:81
virtual void AssembleH(const DenseMatrix &Jpt, const DenseMatrix &DS, const double weight, DenseMatrix &A) const
Evaluate the derivative of the 1st Piola-Kirchhoff stress tensor and assemble its contribution to the...
Definition: tmop.cpp:46
Mesh * GetMesh() const
Returns the mesh.
Definition: fespace.hpp:433
virtual void AssembleH(const DenseMatrix &Jpt, const DenseMatrix &DS, const double weight, DenseMatrix &A) const
Evaluate the derivative of the 1st Piola-Kirchhoff stress tensor and assemble its contribution to the...
Definition: tmop.cpp:419
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:264
void Add(const double c, const DenseMatrix &A)
Adds the matrix A multiplied by the number c to the matrix.
Definition: densemat.cpp:529
void ResetRefinementTspecData()
Definition: tmop.hpp:1276
InvariantsEvaluator2D< double > ie
Definition: tmop.hpp:415
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:698
void AssembleElementVectorExact(const FiniteElement &el, ElementTransformation &T, const Vector &elfun, Vector &elvect)
Definition: tmop.cpp:2797
virtual void AssembleGradDiagonalPA(Vector &) const
Method for computing the diagonal of the gradient with partial assembly.
Definition: tmop.cpp:3818
const TargetConstructor * targetC
Definition: tmop.hpp:1311
Array< Vector * > ElemPertEnergy
Definition: tmop.hpp:1364
const GridFunction * lim_nodes0
Definition: tmop.hpp:1325
virtual void EvalP(const DenseMatrix &Jpt, DenseMatrix &P) const
Evaluate the 1st Piola-Kirchhoff stress tensor, P = P(Jpt).
Definition: tmop.cpp:563
virtual void CalcShape(const IntegrationPoint &ip, Vector &shape) const =0
Evaluate the values of all shape functions of a scalar finite element in reference space at the given...
const ParGridFunction * adapt_lim_pgf0
Definition: tmop.hpp:1337
const scalar_t * Get_dI2b()
Definition: invariants.hpp:216
virtual void AssembleH(const DenseMatrix &Jpt, const DenseMatrix &DS, const double weight, DenseMatrix &A) const
Evaluate the derivative of the 1st Piola-Kirchhoff stress tensor and assemble its contribution to the...
Definition: tmop.cpp:478
InvariantsEvaluator2D< double > ie
Definition: tmop.hpp:257
const IntegrationRule & GetNodes() const
Get a const reference to the nodes of the element.
Definition: fe_base.hpp:390
void SetDerivativeMatrix(int height, const scalar_t *Deriv)
The Deriv matrix is dof x 3, using column-major storage.
Definition: invariants.hpp:737
virtual void AssembleGradPA(const Vector &, const FiniteElementSpace &)
Prepare the integrator for partial assembly (PA) gradient evaluations on the given FE space fes at th...
Definition: tmop.cpp:3809
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:147
virtual void SetParDiscreteTargetAspectRatio(const ParGridFunction &tspec_)
Definition: tmop.cpp:1391
double * GetData(int k)
Definition: densemat.hpp:886
double Weight()
Return the weight of the Jacobian matrix of the transformation at the currently set IntegrationPoint...
Definition: eltrans.hpp:131
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:491
void SetSize(int i, int j, int k, MemoryType mt_=MemoryType::PRESERVE)
Definition: densemat.hpp:838
void ParEnableNormalization(const ParGridFunction &x)
Definition: tmop.cpp:3785
InvariantsEvaluator2D< double > ie
Definition: tmop.hpp:434
int GetNumRootElements()
Return the number of root elements.
Definition: ncmesh.hpp:366
void GetValues(int i, const IntegrationRule &ir, Vector &vals, int vdim=1) const
Definition: gridfunc.cpp:510
virtual void AssembleElementVector(const FiniteElement &el, ElementTransformation &T, const Vector &elfun, Vector &elvect)
Perform the local action of the NonlinearFormIntegrator.
Definition: tmop.cpp:2768
virtual double GetLocalStateEnergyPA(const Vector &) const
Compute the local (to the MPI rank) energy with partial assembly.
Definition: tmop.cpp:3842
virtual void AssembleH(const DenseMatrix &Jpt, const DenseMatrix &DS, const double weight, DenseMatrix &A) const
Evaluate the derivative of the 1st Piola-Kirchhoff stress tensor and assemble its contribution to the...
Definition: tmop.cpp:777
FiniteElementSpace * tspec_fesv
Definition: tmop.hpp:1117
int GetVDim() const
Returns vector dimension.
Definition: fespace.hpp:557
virtual void AssembleH(const DenseMatrix &Jpt, const DenseMatrix &DS, const double weight, DenseMatrix &A) const
Evaluate the derivative of the 1st Piola-Kirchhoff stress tensor and assemble its contribution to the...
Definition: tmop.cpp:239
A class for non-conforming AMR. The class is not used directly by the user, rather it is an extension...
Definition: ncmesh.hpp:121
Coefficient * metric_coeff
Definition: tmop.hpp:1318
virtual void EvalP(const DenseMatrix &Jpt, DenseMatrix &P) const
Evaluate the 1st Piola-Kirchhoff stress tensor, P = P(Jpt).
Definition: tmop.cpp:67
void AssembleElemGradSurfFit(const FiniteElement &el_x, IsoparametricTransformation &Tpr, DenseMatrix &mat)
Definition: tmop.cpp:3196
void SetData(double *d)
Definition: vector.hpp:149
FiniteElementSpace * FESpace()
Definition: gridfunc.hpp:652
virtual bool ContainsVolumeInfo() const
Checks if the target matrices contain non-trivial size specification.
Definition: tmop.cpp:1128
TMOP_QualityMetric * h_metric
Definition: tmop.hpp:1309
Array< TMOP_QualityMetric * > tmop_q_arr
Definition: tmop.hpp:81
int Dimension() const
Definition: mesh.hpp:999
void GetColumn(int c, Vector &col) const
Definition: densemat.cpp:1270
void Assemble_ddI1b(scalar_t w, scalar_t *A)
Definition: invariants.hpp:257
virtual const double * HostRead() const
Shortcut for mfem::Read(vec.GetMemory(), vec.Size(), false).
Definition: vector.hpp:442
void RestoreTargetSpecificationAtNode(ElementTransformation &T, int nodenum)
Definition: tmop.cpp:1569
virtual void EvalP(const DenseMatrix &Jpt, DenseMatrix &P) const
Evaluate the 1st Piola-Kirchhoff stress tensor, P = P(Jpt).
Definition: tmop.cpp:231
virtual DofTransformation * GetElementDofs(int elem, Array< int > &dofs) const
Returns indices of degrees of freedom of element &#39;elem&#39;.
Definition: fespace.cpp:2680
void SetTspecAtIndex(int idx, const GridFunction &tspec_)
Definition: tmop.cpp:1437
virtual double GetRefinementElementEnergy(const FiniteElement &el, ElementTransformation &T, const Vector &elfun, const IntegrationRule &irule)
Computes the mean of the energies of the given element&#39;s children.
Definition: tmop.cpp:2628
void ComputeAvgVolume() const
Definition: tmop.cpp:1014
FDualNumber< tbase > cos(const FDualNumber< tbase > &f)
cos([dual number])
Definition: fdual.hpp:471
int SizeI() const
Definition: densemat.hpp:832
void CalcInverse(const DenseMatrix &a, DenseMatrix &inva)
Definition: densemat.cpp:2182
FDualNumber< tbase > pow(const FDualNumber< tbase > &a, const FDualNumber< tbase > &b)
pow([dual number],[dual number])
Definition: fdual.hpp:543
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:844
double omega
Definition: ex25.cpp:141
Coefficient * lim_coeff
Definition: tmop.hpp:1326
double Min() const
Returns the minimal element of the vector.
Definition: vector.cpp:1182
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:23
AdaptivityEvaluator * surf_fit_eval
Definition: tmop.hpp:1347
double * Data() const
Returns the matrix data array.
Definition: densemat.hpp:111
virtual void SetTargetJacobian(const DenseMatrix &Jtr_)
Specify the reference-element -&gt; target-element Jacobian matrix for the point of interest.
Definition: tmop.hpp:43
FDualNumber< tbase > sin(const FDualNumber< tbase > &f)
sin([dual number])
Definition: fdual.hpp:572
void UpdateHessianTargetSpecification(const Vector &x, double dx, bool use_flag=false)
Definition: tmop.cpp:2205
void Transpose()
(*this) = (*this)^t
Definition: densemat.cpp:1359
void AssembleElemGradAdaptLim(const FiniteElement &el, IsoparametricTransformation &Tpr, const IntegrationRule &ir, const Vector &weights, DenseMatrix &m)
Definition: tmop.cpp:3081
double Trace() const
Trace of a square matrix.
Definition: densemat.cpp:412
void AddMultABt(const DenseMatrix &A, const DenseMatrix &B, DenseMatrix &ABt)
ABt += A * B^t.
Definition: densemat.cpp:2481
virtual void AssembleElementGrad(const FiniteElement &el, ElementTransformation &T, const Vector &elfun, DenseMatrix &elmat)
Assemble the local gradient matrix.
Definition: tmop.cpp:2782
Class FiniteElementSpace - responsible for providing FEM view of the mesh, mainly managing the set of...
Definition: fespace.hpp:88
int GetDof() const
Returns the number of degrees of freedom in the finite element.
Definition: fe_base.hpp:323
void Assemble_ddI2(scalar_t w, scalar_t *A)
Definition: invariants.hpp:948
const double eps
Definition: tmop.hpp:414
DenseMatrix tspec_refine
Definition: tmop.hpp:1106
const scalar_t * Get_dI1()
Definition: invariants.hpp:760
virtual void EvalP(const DenseMatrix &Jpt, DenseMatrix &P) const
Evaluate the 1st Piola-Kirchhoff stress tensor, P = P(Jpt).
Definition: tmop.cpp:498
Base class Coefficients that optionally depend on space and time. These are used by the BilinearFormI...
Definition: coefficient.hpp:39
InvariantsEvaluator3D< double > ie
Definition: tmop.hpp:686
Collection of finite elements from the same family in multiple dimensions. This class is used to matc...
Definition: fe_coll.hpp:26
TMOPMatrixCoefficient * matrix_tspec
Definition: tmop.hpp:1054
const scalar_t * Get_dI1()
Definition: invariants.hpp:204
virtual void ComputeElementTargets(int e_id, const FiniteElement &fe, const IntegrationRule &ir, const Vector &elfun, DenseTensor &Jtr) const
Given an element and quadrature rule, computes ref-&gt;target transformation Jacobians for each quadratur...
Definition: tmop.cpp:1142
const IntegrationRule & ActionIntegrationRule(const FiniteElement &el) const
Definition: tmop.hpp:1486
void SetSize(int nsize)
Change the logical size of the array, keep existing entries.
Definition: array.hpp:679
InvariantsEvaluator2D< double > ie
Definition: tmop.hpp:275
void MultAAt(const DenseMatrix &a, DenseMatrix &aat)
Calculate the matrix A.At.
Definition: densemat.cpp:2314
int NumCols() const
Get the number of columns (size of input) of the Operator. Synonym with Width().
Definition: operator.hpp:75
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:83
void ReleasePADeviceMemory(bool copy_to_host=true)
Definition: tmop.cpp:2314
bool UsesPhysicalCoordinates() const
Return true if the methods ComputeElementTargets(), ComputeAllElementTargets(), and ComputeElementTar...
Definition: tmop.hpp:1001
virtual void AssembleH(const DenseMatrix &Jpt, const DenseMatrix &DS, const double weight, DenseMatrix &A) const
Evaluate the derivative of the 1st Piola-Kirchhoff stress tensor and assemble its contribution to the...
Definition: tmop.cpp:214
double * HostWrite()
Shortcut for mfem::Write(GetMemory(), TotalSize(), false).
Definition: densemat.hpp:921
virtual void AssembleElementGrad(const FiniteElement &el, ElementTransformation &T, const Vector &elfun, DenseMatrix &elmat)
Assemble the local gradient matrix.
Definition: tmop.cpp:3727
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:371
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:403
const scalar_t * Get_dI1b()
Definition: invariants.hpp:208
DenseTensor Jtrcomp
Definition: tmop.hpp:1113
void SetDataAndSize(double *d, int s)
Set the Vector data and size.
Definition: vector.hpp:156
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:576
virtual void Update()
Transform by the Space UpdateMatrix (e.g., on Mesh change).
Definition: gridfunc.cpp:164
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:462
GridFunction * tspec_gf
Definition: tmop.hpp:1119
const scalar_t * Get_dI1b()
Definition: invariants.hpp:764
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:752
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:298
const Array< int > & GetDofMap() const
Get an Array&lt;int&gt; that maps lexicographically ordered indices to the indices of the respective nodes/...
Definition: fe_base.hpp:1181
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:224
virtual void AssembleH(const DenseMatrix &Jpt, const DenseMatrix &DS, const double weight, DenseMatrix &A) const
Evaluate the derivative of the 1st Piola-Kirchhoff stress tensor and assemble its contribution to the...
Definition: tmop.cpp:281
virtual void AssembleH(const DenseMatrix &Jpt, const DenseMatrix &DS, const double weight, DenseMatrix &A) const
Evaluate the derivative of the 1st Piola-Kirchhoff stress tensor and assemble its contribution to the...
Definition: tmop.cpp:801
virtual void EvalP(const DenseMatrix &Jpt, DenseMatrix &P) const
Evaluate the 1st Piola-Kirchhoff stress tensor, P = P(Jpt).
Definition: tmop.cpp:470
void UpdateAfterMeshTopologyChange()
Definition: tmop.cpp:2465
DenseMatrix PMatI
Definition: tmop.hpp:1376
InvariantsEvaluator3D< double > ie
Definition: tmop.hpp:563
virtual void EvalP(const DenseMatrix &Jpt, DenseMatrix &P) const
Evaluate the 1st Piola-Kirchhoff stress tensor, P = P(Jpt).
Definition: tmop.cpp:272
int SizeJ() const
Definition: densemat.hpp:833
double a
Definition: lissajous.cpp:41
void ComputeFDh(const Vector &x, const FiniteElementSpace &fes)
Determines the perturbation, h, for FD-based approximation.
Definition: tmop.cpp:3624
const scalar_t * Get_dI2b()
Definition: invariants.hpp:772
virtual double * ReadWrite(bool on_dev=true)
Shortcut for mfem::ReadWrite(vec.GetMemory(), vec.Size(), on_dev).
Definition: vector.hpp:454
void ComputeNormalizationEnergies(const GridFunction &x, double &metric_energy, double &lim_energy, double &surf_fit_gf_energy)
Definition: tmop.cpp:3503
virtual double Eval(const Vector &x, const Vector &x0, double d) const =0
Returns the limiting function, f(x, x0, d).
Coefficient * surf_fit_coeff
Definition: tmop.hpp:1346
Class for integration point with weight.
Definition: intrules.hpp:25
AdaptivityEvaluator * adapt_eval
Definition: tmop.hpp:1136
ParFiniteElementSpace * ptspec_fesv
Definition: tmop.hpp:1122
InvariantsEvaluator3D< double > ie
Definition: tmop.hpp:453
void Assemble_TProd(scalar_t w, const scalar_t *X, const scalar_t *Y, scalar_t *A)
const FiniteElementSpace * fes
Definition: tmop.hpp:1413
Array< TMOP_Integrator * > tmopi
Definition: tmop.hpp:1724
A standard isoparametric element transformation.
Definition: eltrans.hpp:361
const scalar_t * Get_dI2()
Definition: invariants.hpp:768
void ComputeMinJac(const Vector &x, const FiniteElementSpace &fes)
Definition: tmop.cpp:3574
virtual void AssembleH(const DenseMatrix &Jpt, const DenseMatrix &DS, const double weight, DenseMatrix &A) const
Evaluate the derivative of the 1st Piola-Kirchhoff stress tensor and assemble its contribution to the...
Definition: tmop.cpp:682
void GetElementTransformation(int i, IsoparametricTransformation *ElTr)
Definition: mesh.cpp:348
const IntegrationRule & EnergyIntegrationRule(const FiniteElement &el) const
Definition: tmop.hpp:1477
const IntegrationRule & GradientIntegrationRule(const FiniteElement &el) const
Definition: tmop.hpp:1491
double GetFDDerivative(const FiniteElement &el, ElementTransformation &T, Vector &elfun, const int nodenum, const int idir, const double baseenergy, bool update_stored)
Definition: tmop.cpp:3272
FDualNumber< tbase > sqrt(const FDualNumber< tbase > &f)
sqrt([dual number])
Definition: fdual.hpp:600
void SetParMetaInfo(const ParMesh &m, const FiniteElementCollection &fec, int num_comp)
Parallel version of SetSerialMetaInfo.
Definition: tmop.cpp:2282
virtual void EvalP(const DenseMatrix &Jpt, DenseMatrix &P) const
Evaluate the 1st Piola-Kirchhoff stress tensor, P = P(Jpt).
Definition: tmop.cpp:440
double surf_fit_normal
Definition: tmop.hpp:1348
int dim
Definition: ex24.cpp:53
virtual double GetElementEnergy(const FiniteElement &el, ElementTransformation &T, const Vector &elfun)
Computes the integral of W(Jacobian(Trt)) over a target zone.
Definition: tmop.cpp:2491
void AssembleElemVecAdaptLim(const FiniteElement &el, IsoparametricTransformation &Tpr, const IntegrationRule &ir, const Vector &weights, DenseMatrix &mat)
Definition: tmop.cpp:3045
virtual double EvalW(const DenseMatrix &Jpt) const
Evaluate the strain energy density function, W = W(Jpt).
Definition: tmop.cpp:61
void SetFE(const FiniteElement *FE)
Set the element that will be used to compute the transformations.
Definition: eltrans.hpp:381
void ParUpdateAfterMeshTopologyChange()
Definition: tmop.cpp:2478
virtual double GetDerefinementElementEnergy(const FiniteElement &el, ElementTransformation &T, const Vector &elfun)
Definition: tmop.cpp:3756
void Assemble_ddI1(scalar_t w, scalar_t *A)
Definition: invariants.hpp:790
virtual void Eval_d1(const Vector &x, const Vector &x0, double dist, Vector &d1) const =0
Returns the gradient of the limiting function f(x, x0, d) with respect to x.
virtual const FiniteElement * GetFE(int i) const
Returns pointer to the FiniteElement in the FiniteElementCollection associated with i&#39;th element in t...
Definition: fespace.cpp:2783
const Operator * GetUpdateOperator()
Get the GridFunction update operator.
Definition: fespace.hpp:873
void ComputeAllElementTargets_Fallback(const FiniteElementSpace &fes, const IntegrationRule &ir, const Vector &xe, DenseTensor &Jtr) const
Definition: tmop.cpp:1056
const Vector & GetTspecPert2H()
Definition: tmop.hpp:1240