40 static void ga_init_scalar(bgeot::multi_index &mi) { mi.resize(0); }
41 static void ga_init_vector(bgeot::multi_index &mi,
size_type N)
42 { mi.resize(1); mi[0] = N; }
44 { mi.resize(2); mi[0] = M; mi[1] = N; }
45 static void ga_init_square_matrix(bgeot::multi_index &mi,
size_type N)
46 { mi.resize(2); mi[0] = mi[1] = N; }
49 bool expm(
const base_matrix &a_, base_matrix &aexp, scalar_type tol=1e-15) {
55 frexp(gmm::mat_norminf(a), &e);
56 e = std::max(0, std::min(1023, e));
57 gmm::scale(a, pow(scalar_type(2),-scalar_type(e)));
59 base_matrix atmp(a), an(a);
61 gmm::add(gmm::identity_matrix(), aexp);
65 factn /= scalar_type(n);
68 gmm::scale(atmp, factn);
70 if (gmm::mat_euclidean_norm(atmp) < tol) {
76 for (
int i=0; i < e; ++i) {
83 bool expm_deriv(
const base_matrix &a_, base_tensor &daexp,
84 base_matrix *paexp=NULL, scalar_type tol=1e-15) {
93 frexp(gmm::mat_norminf(a), &e);
94 e = std::max(0, std::min(1023, e));
95 scalar_type scale = pow(scalar_type(2),-scalar_type(e));
98 base_vector factnn(itmax);
99 base_matrix atmp(a), an(a), aexp(a);
100 base_tensor ann(bgeot::multi_index(N,N,itmax));
101 gmm::add(gmm::identity_matrix(), aexp);
103 std::copy(atmp.begin(), atmp.end(), ann.begin());
105 std::copy(a.begin(), a.end(), ann.begin()+N2);
108 for (n=2; n < itmax; ++n) {
109 factnn[n] = factnn[n-1]/scalar_type(n);
112 std::copy(an.begin(), an.end(), ann.begin()+n*N2);
113 gmm::scale(atmp, factnn[n]);
115 if (gmm::mat_euclidean_norm(atmp) < tol) {
125 gmm::scale(factnn, scale);
126 for (--n; n >= 1; --n) {
127 scalar_type factn = factnn[n];
133 daexp(i,j,k,l) += factn*ann(i,k,m-1)*ann(l,j,n-m);
137 base_matrix atmp1(a), atmp2(a);
138 for (
int i=0; i < e; ++i) {
141 std::copy(&daexp(0,0,k,l), &daexp(0,0,k,l)+N*N, atmp.begin());
145 std::copy(atmp.begin(), atmp.end(), &daexp(0,0,k,l));
158 bool logm_deriv(
const base_matrix &a, base_tensor &dalog,
159 base_matrix *palog=NULL) {
162 base_matrix a1(a), alog1(a), alog(a);
164 scalar_type eps(1e-8);
172 dalog(i,j,k,l) = (alog1(i,j) - alog(i,j))/eps;
180 struct matrix_exponential_operator :
public ga_nonlinear_operator {
181 bool result_size(
const arg_list &args, bgeot::multi_index &sizes)
const {
182 if (args.size() != 1 || args[0]->sizes().size() != 2
183 || args[0]->sizes()[0] != args[0]->sizes()[1])
return false;
184 ga_init_matrix(sizes, args[0]->sizes()[0], args[0]->sizes()[1]);
189 void value(
const arg_list &args, base_tensor &result)
const {
191 base_matrix inpmat(N,N), outmat(N,N);
192 gmm::copy(args[0]->as_vector(), inpmat.as_vector());
193 bool info = expm(inpmat, outmat);
194 GMM_ASSERT1(info,
"Matrix exponential calculation "
195 "failed to converge");
196 gmm::copy(outmat.as_vector(), result.as_vector());
200 void derivative(
const arg_list &args,
size_type ,
201 base_tensor &result)
const {
203 base_matrix inpmat(N,N);
204 gmm::copy(args[0]->as_vector(), inpmat.as_vector());
205 bool info = expm_deriv(inpmat, result);
206 GMM_ASSERT1(info,
"Matrix exponential derivative calculation "
207 "failed to converge");
212 base_tensor &)
const {
213 GMM_ASSERT1(
false,
"Sorry, second derivative not implemented");
219 struct matrix_logarithm_operator :
public ga_nonlinear_operator {
220 bool result_size(
const arg_list &args, bgeot::multi_index &sizes)
const {
221 if (args.size() != 1 || args[0]->sizes().size() != 2
222 || args[0]->sizes()[0] != args[0]->sizes()[1])
return false;
223 ga_init_matrix(sizes, args[0]->sizes()[0], args[0]->sizes()[1]);
228 void value(
const arg_list &args, base_tensor &result)
const {
230 base_matrix inpmat(N,N), outmat(N,N);
231 gmm::copy(args[0]->as_vector(), inpmat.as_vector());
233 gmm::copy(outmat.as_vector(), result.as_vector());
237 void derivative(
const arg_list &args,
size_type ,
238 base_tensor &result)
const {
240 base_matrix inpmat(N,N), outmat(N,N), tmpmat(N*N,N*N);
241 gmm::copy(args[0]->as_vector(), inpmat.as_vector());
243 bool info = expm_deriv(outmat, result);
245 gmm::copy(result.as_vector(), tmpmat.as_vector());
247 if (det <= 0)
gmm::copy(gmm::identity_matrix(), tmpmat);
248 gmm::copy(tmpmat.as_vector(), result.as_vector());
250 GMM_ASSERT1(info,
"Matrix logarithm derivative calculation "
251 "failed to converge");
256 base_tensor &)
const {
257 GMM_ASSERT1(
false,
"Sorry, second derivative not implemented");
263 struct normalized_operator :
public ga_nonlinear_operator {
264 bool result_size(
const arg_list &args, bgeot::multi_index &sizes)
const {
265 if (args.size() != 1 || args[0]->sizes().size() > 2
266 || args[0]->sizes().size() < 1)
return false;
267 if (args[0]->sizes().size() == 1)
268 ga_init_vector(sizes, args[0]->sizes()[0]);
270 ga_init_matrix(sizes, args[0]->sizes()[0], args[0]->sizes()[1]);
275 void value(
const arg_list &args, base_tensor &result)
const {
277 const base_tensor &t = *args[0];
278 scalar_type eps = 1E-25;
279 scalar_type no = ::sqrt(gmm::vect_norm2_sqr(t.as_vector())+gmm::sqr(eps));
280 gmm::copy(gmm::scaled(t.as_vector(), scalar_type(1)/no),
287 gmm::copy(gmm::scaled(args[0]->as_vector(), scalar_type(1)/no),
293 void derivative(
const arg_list &args,
size_type,
294 base_tensor &result)
const {
295 const base_tensor &t = *args[0];
298 scalar_type eps = 1E-25;
299 scalar_type no = ::sqrt(gmm::vect_norm2_sqr(t.as_vector())+gmm::sqr(eps));
300 scalar_type no3 = no*no*no;
304 result[i*N+i] += scalar_type(1)/no;
306 result[j*N+i] -= t[i]*t[j] / no3;
313 scalar_type no3 = no*no*no;
315 result[i*N+i] += scalar_type(1)/no;
317 result[j*N+i] -= t[i]*t[j] / no3;
325 base_tensor &)
const {
326 GMM_ASSERT1(
false,
"Sorry, second derivative not implemented");
332 struct Ball_projection_operator :
public ga_nonlinear_operator {
333 bool result_size(
const arg_list &args, bgeot::multi_index &sizes)
const {
334 if (args.size() != 2 || args[0]->sizes().size() > 2
335 || (args[0]->sizes().size() < 1 && args[0]->size() != 1)
336 || args[1]->size() != 1)
return false;
337 if (args[0]->sizes().size() < 1)
338 ga_init_scalar(sizes);
339 else if (args[0]->sizes().size() == 1)
340 ga_init_vector(sizes, args[0]->sizes()[0]);
342 ga_init_matrix(sizes, args[0]->sizes()[0], args[0]->sizes()[1]);
347 void value(
const arg_list &args, base_tensor &result)
const {
348 const base_tensor &t = *args[0];
349 scalar_type r = (*args[1])[0];
352 gmm::copy(gmm::scaled(t.as_vector(), r/no), result.as_vector());
354 gmm::copy(t.as_vector(), result.as_vector());
358 void derivative(
const arg_list &args,
size_type n,
359 base_tensor &result)
const {
360 const base_tensor &t = *args[0];
362 scalar_type r = (*args[1])[0];
373 result[i*N+i] += scalar_type(1);
376 result[i*N+i] += r/no;
378 result[j*N+i] -= t[i]*t[j]*rno3;
384 if (r > 0 && no > r) {
389 default : GMM_ASSERT1(
false,
"Wrong derivative number");
395 base_tensor &)
const {
396 GMM_ASSERT1(
false,
"Sorry, second derivative not implemented");
402 struct normalized_reg_operator :
public ga_nonlinear_operator {
403 bool result_size(
const arg_list &args, bgeot::multi_index &sizes)
const {
404 if (args.size() != 2 || args[0]->sizes().size() > 2
405 || args[0]->sizes().size() < 1 || args[1]->size() != 1)
return false;
406 if (args[0]->sizes().size() == 1)
407 ga_init_vector(sizes, args[0]->sizes()[0]);
409 ga_init_matrix(sizes, args[0]->sizes()[0], args[0]->sizes()[1]);
414 void value(
const arg_list &args, base_tensor &result)
const {
415 const base_tensor &t = *args[0];
416 scalar_type eps = (*args[1])[0];
417 scalar_type no = ::sqrt(gmm::vect_norm2_sqr(t.as_vector())+gmm::sqr(eps));
418 gmm::copy(gmm::scaled(t.as_vector(), scalar_type(1)/no),
424 void derivative(
const arg_list &args,
size_type nder,
425 base_tensor &result)
const {
426 const base_tensor &t = *args[0];
427 scalar_type eps = (*args[1])[0];
430 scalar_type no = ::sqrt(gmm::vect_norm2_sqr(t.as_vector())+gmm::sqr(eps));
431 scalar_type no3 = no*no*no;
437 result[i*N+i] += scalar_type(1)/no;
439 result[j*N+i] -= t[i]*t[j] / no3;
444 gmm::copy(gmm::scaled(t.as_vector(), -scalar_type(eps)/no3),
452 base_tensor &)
const {
453 GMM_ASSERT1(
false,
"Sorry, second derivative not implemented");
463 struct Von_Mises_projection_operator :
public ga_nonlinear_operator {
464 bool result_size(
const arg_list &args, bgeot::multi_index &sizes)
const {
465 if (args.size() != 2 || args[0]->sizes().size() > 2
466 || args[1]->size() != 1)
return false;
467 size_type N = (args[0]->sizes().size() == 2) ? args[0]->sizes()[0] : 1;
468 if (args[0]->sizes().size() == 2 && args[0]->sizes()[1] != N)
return false;
469 if (args[0]->sizes().size() != 2 && args[0]->size() != 1)
return false;
470 if (N > 1) ga_init_square_matrix(sizes, N);
else ga_init_scalar(sizes);
475 void value(
const arg_list &args, base_tensor &result)
const {
476 size_type N = (args[0]->sizes().size() == 2) ? args[0]->sizes()[0] : 1;
477 base_matrix tau(N, N), tau_D(N, N);
478 gmm::copy(args[0]->as_vector(), tau.as_vector());
479 scalar_type s = (*(args[1]))[0];
483 for (
size_type i = 0; i < N; ++i) tau_D(i,i) -= tau_m;
487 if (norm_tau_D > s) gmm::scale(tau_D, s / norm_tau_D);
489 for (
size_type i = 0; i < N; ++i) tau_D(i,i) += tau_m;
491 gmm::copy(tau_D.as_vector(), result.as_vector());
495 void derivative(
const arg_list &args,
size_type nder,
496 base_tensor &result)
const {
497 size_type N = (args[0]->sizes().size() == 2) ? args[0]->sizes()[0] : 1;
498 base_matrix tau(N, N), tau_D(N, N);
499 gmm::copy(args[0]->as_vector(), tau.as_vector());
500 scalar_type s = (*(args[1]))[0];
503 for (
size_type i = 0; i < N; ++i) tau_D(i,i) -= tau_m;
506 if (norm_tau_D != scalar_type(0))
507 gmm::scale(tau_D, scalar_type(1)/norm_tau_D);
511 if (norm_tau_D <= s) {
515 result(i,j,i,j) = scalar_type(1);
522 = s * (-tau_D(i,j) * tau_D(m,n)
523 + ((i == m && j == n) ? scalar_type(1) : scalar_type(0))
524 - ((i == j && m == n) ? scalar_type(1)/scalar_type(N)
525 : scalar_type(0))) / norm_tau_D;
528 result(i,i,j,j) += scalar_type(1)/scalar_type(N);
535 gmm::copy(tau_D.as_vector(), result.as_vector());
542 base_tensor &)
const {
543 GMM_ASSERT1(
false,
"Sorry, second derivative not implemented");
547 static bool init_predef_operators(
void) {
549 ga_predef_operator_tab &PREDEF_OPERATORS
552 PREDEF_OPERATORS.add_method(
"Expm",
553 std::make_shared<matrix_exponential_operator>());
554 PREDEF_OPERATORS.add_method(
"Logm",
555 std::make_shared<matrix_logarithm_operator>());
556 PREDEF_OPERATORS.add_method(
"Normalized",
557 std::make_shared<normalized_operator>());
558 PREDEF_OPERATORS.add_method(
"Normalized_reg",
559 std::make_shared<normalized_reg_operator>());
560 PREDEF_OPERATORS.add_method(
"Ball_projection",
561 std::make_shared<Ball_projection_operator>());
562 PREDEF_OPERATORS.add_method(
"Von_Mises_projection",
563 std::make_shared<Von_Mises_projection_operator>());
568 bool predef_operators_plasticity_initialized
569 = init_predef_operators();
581 void build_isotropic_perfect_elastoplasticity_expressions_mult
582 (model &md,
const std::string &dispname,
const std::string &xi,
583 const std::string &Previous_Ep,
const std::string &lambda,
584 const std::string &mu,
const std::string &sigma_y,
585 const std::string &theta,
const std::string &dt,
586 std::string &sigma_np1, std::string &Epnp1, std::string &compcond,
587 std::string &sigma_after, std::string &von_mises) {
589 const mesh_fem *mfu = md.pmesh_fem_of_variable(dispname);
591 GMM_ASSERT1(mfu && mfu->get_qdim() == N,
"The small strain "
592 "elastoplasticity brick can only be applied on a fem "
593 "variable of the same dimension as the mesh");
595 GMM_ASSERT1(!(md.is_data(xi)) && md.pmesh_fem_of_variable(xi),
596 "The provided name '" << xi <<
"' for the plastic multiplier, "
597 "should be defined as a fem variable");
599 GMM_ASSERT1(md.is_data(Previous_Ep) &&
600 (md.pim_data_of_variable(Previous_Ep) ||
601 md.pmesh_fem_of_variable(Previous_Ep)),
602 "The provided name '" << Previous_Ep <<
"' for the plastic "
603 "strain tensor at the previous timestep, should be defined "
604 "either as fem or as im data");
606 bgeot::multi_index Ep_size(N, N);
607 GMM_ASSERT1((md.pim_data_of_variable(Previous_Ep) &&
608 md.pim_data_of_variable(Previous_Ep)->tensor_size() == Ep_size)
610 (md.pmesh_fem_of_variable(Previous_Ep) &&
611 md.pmesh_fem_of_variable(Previous_Ep)->get_qdims() == Ep_size),
612 "Wrong size of " << Previous_Ep);
614 std::map<std::string, std::string> dict;
615 dict[
"Grad_u"] =
"Grad_"+dispname; dict[
"xi"] = xi;
616 dict[
"Previous_xi"] =
"Previous_"+xi;
617 dict[
"Grad_Previous_u"] =
"Grad_Previous_"+dispname;
618 dict[
"theta"] = theta; dict[
"dt"] = dt; dict[
"Epn"] = Previous_Ep;
619 dict[
"lambda"] = lambda; dict[
"mu"] = mu; dict[
"sigma_y"] = sigma_y;
621 dict[
"Enp1"] = ga_substitute(
"Sym(Grad_u)", dict);
622 dict[
"En"] = ga_substitute(
"Sym(Grad_Previous_u)", dict);
623 dict[
"zetan"] = ga_substitute
624 (
"((Epn)+(1-(theta))*(2*(mu)*(dt)*(Previous_xi))*(Deviator(En)-(Epn)))",
626 Epnp1 = ga_substitute
627 (
"((zetan)+(1-1/(1+(theta)*2*(mu)*(dt)*(xi)))*(Deviator(Enp1)-(zetan)))",
629 dict[
"Epnp1"] = Epnp1;
630 sigma_np1 = ga_substitute
631 (
"((lambda)*Trace(Enp1)*Id(meshdim)+2*(mu)*((Enp1)-(Epnp1)))", dict);
632 dict[
"fbound"] = ga_substitute
633 (
"(2*(mu)*Norm(Deviator(Enp1)-(Epnp1))-sqrt(2/3)*(sigma_y))", dict);
634 dict[
"sigma_after"] = sigma_after = ga_substitute
635 (
"((lambda)*Trace(Enp1)*Id(meshdim)+2*(mu)*((Enp1)-(Epn)))", dict);
636 compcond = ga_substitute
637 (
"((mu)*xi-pos_part((mu)*xi+100*(fbound)/(mu)))", dict);
638 von_mises = ga_substitute
639 (
"sqrt(3/2)*Norm(Deviator(sigma_after))", dict);
645 void build_isotropic_perfect_elastoplasticity_expressions_no_mult
646 (model &md,
const std::string &dispname,
const std::string &xi,
647 const std::string &Previous_Ep,
const std::string &lambda,
648 const std::string &mu,
const std::string &sigma_y,
649 const std::string &theta,
const std::string &dt,
650 std::string &sigma_np1, std::string &Epnp1, std::string &xi_np1,
651 std::string &sigma_after, std::string &von_mises) {
653 const mesh_fem *mfu = md.pmesh_fem_of_variable(dispname);
655 GMM_ASSERT1(mfu && mfu->get_qdim() == N,
"The small strain "
656 "elastoplasticity brick can only be applied on a fem "
657 "variable of the same dimension as the mesh");
659 GMM_ASSERT1(md.is_data(xi) &&
660 (md.pim_data_of_variable(xi) ||
661 md.pmesh_fem_of_variable(xi)),
662 "The provided name '" << xi <<
"' for the plastic multiplier, "
663 "should be defined either as fem data or as im data");
665 GMM_ASSERT1(md.is_data(Previous_Ep) &&
666 (md.pim_data_of_variable(Previous_Ep) ||
667 md.pmesh_fem_of_variable(Previous_Ep)),
668 "The provided name '" << Previous_Ep <<
"' for the plastic "
669 "strain tensor at the previous timestep, should be defined "
670 "either as fem or as im data");
672 bgeot::multi_index Ep_size(N, N);
673 GMM_ASSERT1((md.pim_data_of_variable(Previous_Ep) &&
674 md.pim_data_of_variable(Previous_Ep)->tensor_size() == Ep_size)
676 (md.pmesh_fem_of_variable(Previous_Ep) &&
677 md.pmesh_fem_of_variable(Previous_Ep)->get_qdims() == Ep_size),
678 "Wrong size of " << Previous_Ep);
680 std::map<std::string, std::string> dict;
681 dict[
"Grad_u"] =
"Grad_"+dispname; dict[
"xi"] = xi;
682 dict[
"Previous_xi"] =
"Previous_"+xi;
683 dict[
"Grad_Previous_u"] =
"Grad_Previous_"+dispname;
684 dict[
"theta"] = theta; dict[
"dt"] = dt; dict[
"Epn"] = Previous_Ep;
685 dict[
"lambda"] = lambda; dict[
"mu"] = mu; dict[
"sigma_y"] = sigma_y;
687 dict[
"Enp1"] = ga_substitute(
"Sym(Grad_u)", dict);
688 dict[
"En"] = ga_substitute(
"Sym(Grad_Previous_u)", dict) ;
691 dict[
"zetan"] = ga_substitute
692 (
"(Epn)+(1-(theta))*(2*(mu)*(dt)*(Previous_xi))*(Deviator(En)-(Epn))",
694 dict[
"B"] = ga_substitute(
"Deviator(Enp1)-(zetan)", dict);
695 Epnp1 = ga_substitute
696 (
"(zetan)+pos_part(1-sqrt(2/3)*(sigma_y)/(2*(mu)*Norm(B)+1e-40))*(B)",
698 dict[
"Epnp1"] = Epnp1;
700 sigma_np1 = ga_substitute
701 (
"(lambda)*Trace(Enp1)*Id(meshdim)+2*(mu)*((Enp1)-(Epnp1))", dict);
702 dict[
"sigma_after"] = sigma_after = ga_substitute
703 (
"(lambda)*Trace(Enp1)*Id(meshdim)+2*(mu)*((Enp1)-(Epn))", dict);
704 xi_np1 = ga_substitute
705 (
"pos_part(sqrt(3/2)*Norm(B)/(sigma_y)-1/(2*(mu)))/((theta)*(dt))", dict);
706 von_mises = ga_substitute
707 (
"sqrt(3/2)*Norm(Deviator(sigma_after))", dict);
713 void build_isotropic_perfect_elastoplasticity_expressions_mult_ps
714 (model &md,
const std::string &dispname,
const std::string &xi,
715 const std::string &Previous_Ep,
const std::string &lambda,
716 const std::string &mu,
const std::string &sigma_y,
717 const std::string &theta,
const std::string &dt,
718 std::string &sigma_np1, std::string &Epnp1, std::string &compcond,
719 std::string &sigma_after, std::string &von_mises) {
721 const mesh_fem *mfu = md.pmesh_fem_of_variable(dispname);
723 GMM_ASSERT1(N == 2,
"This plastic law is restricted to 2D");
725 GMM_ASSERT1(mfu && mfu->get_qdim() == N,
"The small strain "
726 "elastoplasticity brick can only be applied on a fem "
727 "variable of the same dimension as the mesh");
729 GMM_ASSERT1(!(md.is_data(xi)) && md.pmesh_fem_of_variable(xi),
730 "The provided name '" << xi <<
"' for the plastic multiplier, "
731 "should be defined as a fem variable");
733 GMM_ASSERT1(md.is_data(Previous_Ep) &&
734 (md.pim_data_of_variable(Previous_Ep) ||
735 md.pmesh_fem_of_variable(Previous_Ep)),
736 "The provided name '" << Previous_Ep <<
"' for the plastic "
737 "strain tensor at the previous timestep, should be defined "
738 "either as fem or as im data");
740 bgeot::multi_index Ep_size(N, N);
741 GMM_ASSERT1((md.pim_data_of_variable(Previous_Ep) &&
742 md.pim_data_of_variable(Previous_Ep)->tensor_size() == Ep_size)
744 (md.pmesh_fem_of_variable(Previous_Ep) &&
745 md.pmesh_fem_of_variable(Previous_Ep)->get_qdims() == Ep_size),
746 "Wrong size of " << Previous_Ep);
748 std::map<std::string, std::string> dict;
749 dict[
"Grad_u"] =
"Grad_"+dispname; dict[
"xi"] = xi;
750 dict[
"Previous_xi"] =
"Previous_"+xi;
751 dict[
"Grad_Previous_u"] =
"Grad_Previous_"+dispname;
752 dict[
"theta"] = theta; dict[
"dt"] = dt; dict[
"Epn"] = Previous_Ep;
753 dict[
"lambda"] = lambda; dict[
"mu"] = mu; dict[
"sigma_y"] = sigma_y;
755 dict[
"Enp1"] = ga_substitute(
"Sym(Grad_u)", dict);
756 dict[
"En"] = ga_substitute(
"Sym(Grad_Previous_u)", dict);
757 dict[
"Dev_En"]= ga_substitute(
"(En-(Trace(En)/3)*Id(meshdim))", dict);
758 dict[
"Dev_Enp1"]= ga_substitute(
"(Enp1-(Trace(Enp1)/3)*Id(meshdim))", dict);
759 dict[
"zetan"] = ga_substitute
760 (
"((Epn)+(1-(theta))*(2*(mu)*(dt)*(Previous_xi))*((Dev_En)-(Epn)))",
762 Epnp1 = ga_substitute
763 (
"((zetan)+(1-1/(1+(theta)*2*(mu)*(dt)*(xi)))*((Dev_Enp1)-(zetan)))",
765 dict[
"Epnp1"] = Epnp1;
766 sigma_np1 = ga_substitute
767 (
"((lambda)*Trace(Enp1)*Id(meshdim)+2*(mu)*((Enp1)-(Epnp1)))", dict);
768 dict[
"fbound"] = ga_substitute
769 (
"(2*(mu)*sqrt(Norm_sqr(Dev_Enp1-(Epnp1))"
770 "+sqr(Trace(Enp1)/3-Trace(Epnp1)))-sqrt(2/3)*(sigma_y))", dict);
772 sigma_after = ga_substitute
773 (
"((lambda)*Trace(Enp1)*Id(meshdim)+2*(mu)*((Enp1)-(Epn)))", dict);
774 compcond = ga_substitute
775 (
"((mu)*xi-pos_part((mu)*xi+100*(fbound)/(mu)))", dict);
776 von_mises = ga_substitute
777 (
"sqrt(3/2)*sqrt(Norm_sqr((2*(mu))*(Dev_En)-(2*(mu))*(Epn))"
778 "+sqr(2*(mu)*Trace(En)/3-(2*(mu))*Trace(Epn)))", dict);
785 void build_isotropic_perfect_elastoplasticity_expressions_no_mult_ps
786 (model &md,
const std::string &dispname,
const std::string &xi,
787 const std::string &Previous_Ep,
const std::string &lambda,
788 const std::string &mu,
const std::string &sigma_y,
789 const std::string &theta,
const std::string &dt,
790 std::string &sigma_np1, std::string &Epnp1,
791 std::string &xi_np1, std::string &sigma_after, std::string &von_mises) {
793 const mesh_fem *mfu = md.pmesh_fem_of_variable(dispname);
795 GMM_ASSERT1(N == 2,
"This plastic law is restricted to 2D");
797 GMM_ASSERT1(mfu && mfu->get_qdim() == N,
"The small strain "
798 "elastoplasticity brick can only be applied on a fem "
799 "variable of the same dimension as the mesh");
801 GMM_ASSERT1(md.is_data(xi) &&
802 (md.pim_data_of_variable(xi) ||
803 md.pmesh_fem_of_variable(xi)),
804 "The provided name '" << xi <<
"' for the plastic multiplier, "
805 "should be defined either as fem data or as im data");
807 GMM_ASSERT1(md.is_data(Previous_Ep) &&
808 (md.pim_data_of_variable(Previous_Ep) ||
809 md.pmesh_fem_of_variable(Previous_Ep)),
810 "The provided name '" << Previous_Ep <<
"' for the plastic "
811 "strain tensor at the previous timestep, should be defined "
812 "either as fem or as im data");
814 bgeot::multi_index Ep_size(N, N);
815 GMM_ASSERT1((md.pim_data_of_variable(Previous_Ep) &&
816 md.pim_data_of_variable(Previous_Ep)->tensor_size() == Ep_size)
818 (md.pmesh_fem_of_variable(Previous_Ep) &&
819 md.pmesh_fem_of_variable(Previous_Ep)->get_qdims() == Ep_size),
820 "Wrong size of " << Previous_Ep);
822 std::map<std::string, std::string> dict;
823 dict[
"Grad_u"] =
"Grad_"+dispname; dict[
"xi"] = xi;
824 dict[
"Previous_xi"] =
"Previous_"+xi;
825 dict[
"Grad_Previous_u"] =
"Grad_Previous_"+dispname;
826 dict[
"theta"] = theta; dict[
"dt"] = dt; dict[
"Epn"] = Previous_Ep;
827 dict[
"lambda"] = lambda; dict[
"mu"] = mu; dict[
"sigma_y"] = sigma_y;
829 dict[
"Enp1"] = ga_substitute(
"Sym(Grad_u)", dict);
830 dict[
"En"] = ga_substitute(
"Sym(Grad_Previous_u)", dict) ;
831 dict[
"Dev_En"]= ga_substitute(
"(En-(Trace(En)/3)*Id(meshdim))", dict);
832 dict[
"Dev_Enp1"]= ga_substitute(
"(Enp1-(Trace(Enp1)/3)*Id(meshdim))", dict);
833 dict[
"zetan"] = ga_substitute
834 (
"((Epn)+(1-(theta))*(2*(mu)*(dt)*(Previous_xi))*(Dev_En-(Epn)))",
836 dict[
"B"] = ga_substitute(
"(Dev_Enp1)-(zetan)", dict);
837 Epnp1 = ga_substitute
838 (
"(zetan)+pos_part(1-sqrt(2/3)*(sigma_y)/(2*(mu)*(sqrt(Norm_sqr(B)+"
839 "sqr(Trace(Enp1)/3-Trace(zetan))))+1e-25))*(B)", dict);
840 dict[
"Epnp1"] = Epnp1;
842 sigma_np1 = ga_substitute
843 (
"(lambda)*Trace(Enp1)*Id(meshdim)+2*(mu)*((Enp1)-(Epnp1))", dict);
844 sigma_after = ga_substitute
845 (
"(lambda)*Trace(Enp1)*Id(meshdim)+2*(mu)*((Enp1)-(Epn))", dict);
846 xi_np1 = ga_substitute
847 (
"pos_part(sqrt(3/2)*Norm(B)/(sigma_y)-1/(2*(mu)))/((theta)*(dt))", dict);
848 von_mises = ga_substitute
849 (
"sqrt(3/2)*sqrt(Norm_sqr((2*(mu))*(Dev_En)-(2*(mu))*(Epn))"
850 "+sqr(2*(mu)*Trace(En)/3-(2*(mu))*Trace(Epn)))", dict);
856 void build_isotropic_perfect_elastoplasticity_expressions_hard_mult
857 (model &md,
const std::string &dispname,
const std::string &xi,
858 const std::string &Previous_Ep,
const std::string &alpha,
859 const std::string &lambda,
const std::string &mu,
860 const std::string &sigma_y,
const std::string &Hk,
const std::string &Hi,
861 const std::string &theta,
const std::string &dt,
862 std::string &sigma_np1, std::string &Epnp1, std::string &compcond,
863 std::string &sigma_after, std::string &von_mises, std::string &alphanp1) {
865 const mesh_fem *mfu = md.pmesh_fem_of_variable(dispname);
867 GMM_ASSERT1(mfu && mfu->get_qdim() == N,
"The small strain "
868 "elastoplasticity brick can only be applied on a fem "
869 "variable of the same dimension as the mesh");
871 GMM_ASSERT1(!(md.is_data(xi)) && md.pmesh_fem_of_variable(xi),
872 "The provided name '" << xi <<
"' for the plastic multiplier, "
873 "should be defined as a fem variable");
875 GMM_ASSERT1(md.is_data(Previous_Ep) &&
876 (md.pim_data_of_variable(Previous_Ep) ||
877 md.pmesh_fem_of_variable(Previous_Ep)),
878 "The provided name '" << Previous_Ep <<
"' for the plastic "
879 "strain tensor at the previous timestep, should be defined "
880 "either as fem or as im data");
882 bgeot::multi_index Ep_size(N, N);
883 GMM_ASSERT1((md.pim_data_of_variable(Previous_Ep) &&
884 md.pim_data_of_variable(Previous_Ep)->tensor_size() == Ep_size)
886 (md.pmesh_fem_of_variable(Previous_Ep) &&
887 md.pmesh_fem_of_variable(Previous_Ep)->get_qdims() == Ep_size),
888 "Wrong size of " << Previous_Ep);
890 std::map<std::string, std::string> dict;
891 dict[
"Hk"] = Hk; dict[
"Hi"] = Hi; dict[
"alphan"] =
alpha;
892 dict[
"Grad_u"] =
"Grad_"+dispname; dict[
"xi"] = xi;
893 dict[
"Previous_xi"] =
"Previous_"+xi;
894 dict[
"Grad_Previous_u"] =
"Grad_Previous_"+dispname;
895 dict[
"theta"] = theta; dict[
"dt"] = dt; dict[
"Epn"] = Previous_Ep;
896 dict[
"lambda"] = lambda; dict[
"mu"] = mu; dict[
"sigma_y"] = sigma_y;
898 dict[
"Enp1"] = ga_substitute(
"Sym(Grad_u)", dict);
899 dict[
"En"] = ga_substitute(
"Sym(Grad_Previous_u)", dict);
900 dict[
"zetan"] = ga_substitute
901 (
"((Epn)+(1-(theta))*((dt)*(Previous_xi))*((2*(mu))*Deviator(En)"
902 "-(2*(mu)+2/3*(Hk))*(Epn)))", dict);
903 dict[
"etan"] = ga_substitute
904 (
"((alphan)+sqrt(2/3)*(1-(theta))*((dt)*(Previous_xi))*"
905 "Norm((2*(mu))*Deviator(En)-(2*(mu)+2/3*(Hk))*(Epn)))", dict);
906 dict[
"B"] = ga_substitute
907 (
"((2*(mu))*Deviator(Enp1)-(2*(mu)+2/3*(Hk))*(zetan))", dict);
908 dict[
"beta"] = ga_substitute
909 (
"((theta)*(dt)*(xi)/(1+(2*(mu)+2/3*(Hk))*(theta)*(dt)*(xi)))", dict);
910 dict[
"Epnp1"] = Epnp1 = ga_substitute(
"((zetan)+(beta)*(B))", dict);
911 alphanp1 = ga_substitute(
"((etan)+sqrt(2/3)*(beta)*Norm(B))", dict);
912 dict[
"alphanp1"] = alphanp1;
913 sigma_np1 = ga_substitute
914 (
"((lambda)*Trace(Enp1)*Id(meshdim)+2*(mu)*((Enp1)-(Epnp1)))", dict);
915 dict[
"fbound"] = ga_substitute
916 (
"(Norm((2*(mu))*Deviator(Enp1)-(2*(mu)+2/3*(Hk))*(Epnp1))"
917 "-sqrt(2/3)*(sigma_y+(Hi)*(alphanp1)))", dict);
918 dict[
"sigma_after"] = sigma_after = ga_substitute
919 (
"((lambda)*Trace(Enp1)*Id(meshdim)+2*(mu)*((Enp1)-(Epn)))", dict);
920 compcond = ga_substitute
921 (
"((mu)*xi-pos_part((mu)*xi+100*(fbound)/(mu)))", dict);
922 von_mises = ga_substitute
923 (
"sqrt(3/2)*Norm(Deviator(sigma_after))", dict);
929 void build_isotropic_perfect_elastoplasticity_expressions_hard_no_mult
930 (model &md,
const std::string &dispname,
const std::string &xi,
931 const std::string &Previous_Ep,
const std::string &alpha,
932 const std::string &lambda,
const std::string &mu,
933 const std::string &sigma_y,
const std::string &Hk,
const std::string &Hi,
934 const std::string &theta,
const std::string &dt,
935 std::string &sigma_np1, std::string &Epnp1, std::string &xi_np1,
936 std::string &sigma_after, std::string &von_mises, std::string &alphanp1) {
938 const mesh_fem *mfu = md.pmesh_fem_of_variable(dispname);
940 GMM_ASSERT1(mfu && mfu->get_qdim() == N,
"The small strain "
941 "elastoplasticity brick can only be applied on a fem "
942 "variable of the same dimension as the mesh");
944 GMM_ASSERT1(md.is_data(xi) &&
945 (md.pim_data_of_variable(xi) ||
946 md.pmesh_fem_of_variable(xi)),
947 "The provided name '" << xi <<
"' for the plastic multiplier, "
948 "should be defined either as fem data or as im data");
950 GMM_ASSERT1(md.is_data(Previous_Ep) &&
951 (md.pim_data_of_variable(Previous_Ep) ||
952 md.pmesh_fem_of_variable(Previous_Ep)),
953 "The provided name '" << Previous_Ep <<
"' for the plastic "
954 "strain tensor at the previous timestep, should be defined "
955 "either as fem or as im data");
957 bgeot::multi_index Ep_size(N, N);
958 GMM_ASSERT1((md.pim_data_of_variable(Previous_Ep) &&
959 md.pim_data_of_variable(Previous_Ep)->tensor_size() == Ep_size)
961 (md.pmesh_fem_of_variable(Previous_Ep) &&
962 md.pmesh_fem_of_variable(Previous_Ep)->get_qdims() == Ep_size),
963 "Wrong size of " << Previous_Ep);
965 std::map<std::string, std::string> dict;
966 dict[
"Hk"] = Hk; dict[
"Hi"] = Hi; dict[
"alphan"] =
alpha;
967 dict[
"Grad_u"] =
"Grad_"+dispname; dict[
"xi"] = xi;
968 dict[
"Previous_xi"] =
"Previous_"+xi;
969 dict[
"Grad_Previous_u"] =
"Grad_Previous_"+dispname;
970 dict[
"theta"] = theta; dict[
"dt"] = dt; dict[
"Epn"] = Previous_Ep;
971 dict[
"lambda"] = lambda; dict[
"mu"] = mu; dict[
"sigma_y"] = sigma_y;
973 dict[
"Enp1"] = ga_substitute(
"Sym(Grad_u)", dict);
974 dict[
"En"] = ga_substitute(
"Sym(Grad_Previous_u)", dict);
975 dict[
"zetan"] = ga_substitute
976 (
"((Epn)+(1-(theta))*((dt)*(Previous_xi))*((2*(mu))*Deviator(En)"
977 "-(2*(mu)+2/3*(Hk))*(Epn)))", dict);
978 dict[
"etan"] = ga_substitute
979 (
"((alphan)+sqrt(2/3)*(1-(theta))*((dt)*(Previous_xi))*"
980 "Norm((2*(mu))*Deviator(En)-(2*(mu)+2/3*(Hk))*(Epn)))", dict);
982 dict[
"B"] = ga_substitute
983 (
"((2*(mu))*Deviator(Enp1)-(2*(mu)+2/3*(Hk))*(zetan))", dict);
985 ga_substitute(
"(1/((Norm(B)+1e-40)*(2*(mu)+2/3*(Hk)+(2/3)*(Hi))))"
986 "*pos_part(Norm(B)-sqrt(2/3)*((sigma_y)+(Hi)*(etan)))", dict);
987 dict[
"Epnp1"] = Epnp1 = ga_substitute(
"((zetan)+(beta)*(B))", dict);
988 alphanp1 = ga_substitute(
"((etan)+sqrt(2/3)*(beta)*Norm(B))", dict);
989 dict[
"alphanp1"] = alphanp1;
991 sigma_np1 = ga_substitute
992 (
"(lambda)*Trace(Enp1)*Id(meshdim)+2*(mu)*((Enp1)-(Epnp1))", dict);
993 dict[
"sigma_after"] = sigma_after = ga_substitute
994 (
"(lambda)*Trace(Enp1)*Id(meshdim)+2*(mu)*((Enp1)-(Epn))", dict);
995 xi_np1 = ga_substitute
996 (
"(((beta)/(1-(2*(mu)+2/3*(Hk))*(beta)))/((theta)*(dt)))", dict);
997 von_mises = ga_substitute
998 (
"sqrt(3/2)*Norm(Deviator(sigma_after))", dict);
1004 void build_isotropic_perfect_elastoplasticity_expressions_hard_mult_ps
1005 (model &md,
const std::string &dispname,
const std::string &xi,
1006 const std::string &Previous_Ep,
const std::string &alpha,
1007 const std::string &lambda,
const std::string &mu,
1008 const std::string &sigma_y,
const std::string &Hk,
const std::string &Hi,
1009 const std::string &theta,
const std::string &dt,
1010 std::string &sigma_np1, std::string &Epnp1, std::string &compcond,
1011 std::string &sigma_after, std::string &von_mises, std::string &alphanp1) {
1013 const mesh_fem *mfu = md.pmesh_fem_of_variable(dispname);
1015 GMM_ASSERT1(N == 2,
"This plastic law is restricted to 2D");
1017 GMM_ASSERT1(mfu && mfu->get_qdim() == N,
"The small strain "
1018 "elastoplasticity brick can only be applied on a fem "
1019 "variable of the same dimension as the mesh");
1021 GMM_ASSERT1(!(md.is_data(xi)) && md.pmesh_fem_of_variable(xi),
1022 "The provided name '" << xi <<
"' for the plastic multiplier, "
1023 "should be defined as a fem variable");
1025 GMM_ASSERT1(md.is_data(Previous_Ep) &&
1026 (md.pim_data_of_variable(Previous_Ep) ||
1027 md.pmesh_fem_of_variable(Previous_Ep)),
1028 "The provided name '" << Previous_Ep <<
"' for the plastic "
1029 "strain tensor at the previous timestep, should be defined "
1030 "either as fem or as im data");
1032 bgeot::multi_index Ep_size(N, N);
1033 GMM_ASSERT1((md.pim_data_of_variable(Previous_Ep) &&
1034 md.pim_data_of_variable(Previous_Ep)->tensor_size() == Ep_size)
1036 (md.pmesh_fem_of_variable(Previous_Ep) &&
1037 md.pmesh_fem_of_variable(Previous_Ep)->get_qdims() == Ep_size),
1038 "Wrong size of " << Previous_Ep);
1040 std::map<std::string, std::string> dict;
1041 dict[
"Hk"] = Hk; dict[
"Hi"] = Hi; dict[
"alphan"] =
alpha;
1042 dict[
"Grad_u"] =
"Grad_"+dispname; dict[
"xi"] = xi;
1043 dict[
"Previous_xi"] =
"Previous_"+xi;
1044 dict[
"Grad_Previous_u"] =
"Grad_Previous_"+dispname;
1045 dict[
"theta"] = theta; dict[
"dt"] = dt; dict[
"Epn"] = Previous_Ep;
1046 dict[
"lambda"] = lambda; dict[
"mu"] = mu; dict[
"sigma_y"] = sigma_y;
1048 dict[
"Enp1"] = ga_substitute(
"Sym(Grad_u)", dict);
1049 dict[
"En"] = ga_substitute(
"Sym(Grad_Previous_u)", dict);
1050 dict[
"Dev_En"]= ga_substitute(
"(En-(Trace(En)/3)*Id(meshdim))", dict);
1051 dict[
"Dev_Enp1"]= ga_substitute(
"(Enp1-(Trace(Enp1)/3)*Id(meshdim))", dict);
1052 dict[
"zetan"] = ga_substitute
1053 (
"((Epn)+(1-(theta))*((dt)*(Previous_xi))*((2*(mu))*(Dev_En)"
1054 "-(2*(mu)+2/3*(Hk))*(Epn)))", dict);
1055 dict[
"etan"] = ga_substitute
1056 (
"((alphan)+sqrt(2/3)*(1-(theta))*((dt)*(Previous_xi))*"
1057 "sqrt(Norm_sqr((2*(mu))*(Dev_En)-(2*(mu)+2/3*(Hk))*(Epn))"
1058 "+sqr(2*(mu)*Trace(En)/3-(2*(mu)+2/3*(Hk))*Trace(Epn))))", dict);
1059 dict[
"B"] = ga_substitute(
"((2*(mu))*(Dev_Enp1)-(2*(mu)+2/3*(Hk))*(zetan))",
1061 dict[
"Norm_B"] = ga_substitute(
"sqrt(Norm_sqr(B)+sqr(2*(mu)*Trace(Enp1)/3"
1062 "-(2*(mu)+2/3*(Hk))*Trace(zetan)))", dict);
1064 dict[
"beta"] = ga_substitute
1065 (
"((theta)*(dt)*(xi)/(1+(2*(mu)+2/3*(Hk))*(theta)*(dt)*(xi)))", dict);
1066 dict[
"Epnp1"] = Epnp1 = ga_substitute(
"((zetan)+(beta)*(B))", dict);
1067 alphanp1 = ga_substitute(
"((etan)+sqrt(2/3)*(beta)*(Norm_B))", dict);
1068 dict[
"alphanp1"] = alphanp1;
1069 sigma_np1 = ga_substitute
1070 (
"((lambda)*Trace(Enp1)*Id(meshdim)+2*(mu)*((Enp1)-(Epnp1)))", dict);
1071 dict[
"fbound"] = ga_substitute
1072 (
"(sqrt(Norm_sqr((2*(mu))*(Dev_Enp1)-(2*(mu)+2/3*(Hk))*(Epnp1))"
1073 "+sqr(2*(mu)*Trace(Enp1)/3-(2*(mu)+2/3*(Hk))*Trace(Epnp1)))"
1074 "-sqrt(2/3)*(sigma_y+(Hi)*(alphanp1)))", dict);
1075 sigma_after = ga_substitute
1076 (
"((lambda)*Trace(Enp1)*Id(meshdim)+2*(mu)*((Enp1)-(Epn)))", dict);
1077 compcond = ga_substitute
1078 (
"((mu)*xi-pos_part((mu)*xi+100*(fbound)/(mu)))", dict);
1079 von_mises = ga_substitute
1080 (
"sqrt(3/2)*sqrt(Norm_sqr((2*(mu))*(Dev_En)-(2*(mu)+2/3*(Hk))*(Epn))"
1081 "+sqr(2*(mu)*Trace(En)/3-(2*(mu)+2/3*(Hk))*Trace(Epn)))", dict);
1088 void build_isotropic_perfect_elastoplasticity_expressions_hard_no_mult_ps
1089 (model &md,
const std::string &dispname,
const std::string &xi,
1090 const std::string &Previous_Ep,
const std::string &alpha,
1091 const std::string &lambda,
const std::string &mu,
1092 const std::string &sigma_y,
const std::string &Hk,
const std::string &Hi,
1093 const std::string &theta,
const std::string &dt,
1094 std::string &sigma_np1, std::string &Epnp1, std::string &xi_np1,
1095 std::string &sigma_after, std::string &von_mises, std::string &alphanp1) {
1097 const mesh_fem *mfu = md.pmesh_fem_of_variable(dispname);
1099 GMM_ASSERT1(N == 2,
"This plastic law is restricted to 2D");
1101 GMM_ASSERT1(mfu && mfu->get_qdim() == N,
"The small strain "
1102 "elastoplasticity brick can only be applied on a fem "
1103 "variable of the same dimension as the mesh");
1105 GMM_ASSERT1(md.is_data(xi) &&
1106 (md.pim_data_of_variable(xi) ||
1107 md.pmesh_fem_of_variable(xi)),
1108 "The provided name '" << xi <<
"' for the plastic multiplier, "
1109 "should be defined either as fem data or as im data");
1111 GMM_ASSERT1(md.is_data(Previous_Ep) &&
1112 (md.pim_data_of_variable(Previous_Ep) ||
1113 md.pmesh_fem_of_variable(Previous_Ep)),
1114 "The provided name '" << Previous_Ep <<
"' for the plastic "
1115 "strain tensor at the previous timestep, should be defined "
1116 "either as fem or as im data");
1118 bgeot::multi_index Ep_size(N, N);
1119 GMM_ASSERT1((md.pim_data_of_variable(Previous_Ep) &&
1120 md.pim_data_of_variable(Previous_Ep)->tensor_size() == Ep_size)
1122 (md.pmesh_fem_of_variable(Previous_Ep) &&
1123 md.pmesh_fem_of_variable(Previous_Ep)->get_qdims() == Ep_size),
1124 "Wrong size of " << Previous_Ep);
1126 std::map<std::string, std::string> dict;
1127 dict[
"Hk"] = Hk; dict[
"Hi"] = Hi; dict[
"alphan"] =
alpha;
1128 dict[
"Grad_u"] =
"Grad_"+dispname; dict[
"xi"] = xi;
1129 dict[
"Previous_xi"] =
"Previous_"+xi;
1130 dict[
"Grad_Previous_u"] =
"Grad_Previous_"+dispname;
1131 dict[
"theta"] = theta; dict[
"dt"] = dt; dict[
"Epn"] = Previous_Ep;
1132 dict[
"lambda"] = lambda; dict[
"mu"] = mu; dict[
"sigma_y"] = sigma_y;
1134 dict[
"Enp1"] = ga_substitute(
"Sym(Grad_u)", dict);
1135 dict[
"En"] = ga_substitute(
"Sym(Grad_Previous_u)", dict);
1136 dict[
"Dev_En"]= ga_substitute(
"(En-(Trace(En)/3)*Id(meshdim))", dict);
1137 dict[
"Dev_Enp1"]= ga_substitute(
"(Enp1-(Trace(Enp1)/3)*Id(meshdim))", dict);
1138 dict[
"zetan"] = ga_substitute
1139 (
"((Epn)+(1-(theta))*((dt)*(Previous_xi))*((2*(mu))*(Dev_En)"
1140 "-(2*(mu)+2/3*(Hk))*(Epn)))", dict);
1141 dict[
"etan"] = ga_substitute
1142 (
"((alphan)+sqrt(2/3)*(1-(theta))*((dt)*(Previous_xi))*"
1143 "sqrt(Norm_sqr((2*(mu))*(Dev_En)-(2*(mu)+2/3*(Hk))*(Epn))"
1144 "+sqr(2*(mu)*Trace(En)/3-(2*(mu)+2/3*(Hk))*Trace(Epn))))", dict);
1146 dict[
"B"] = ga_substitute
1147 (
"((2*(mu))*(Dev_Enp1)-(2*(mu)+2/3*(Hk))*(zetan))", dict);
1148 dict[
"Norm_B"] = ga_substitute(
"sqrt(Norm_sqr(B)+sqr(2*(mu)*Trace(Enp1)/3"
1149 "-(2*(mu)+2/3*(Hk))*Trace(zetan)))", dict);
1151 ga_substitute(
"(1/(((Norm_B)+1e-40)*(2*(mu)+2/3*(Hk)+(2/3)*(Hi))))"
1152 "*pos_part((Norm_B)-sqrt(2/3)*((sigma_y)+(Hi)*(etan)))", dict);
1153 dict[
"Epnp1"] = Epnp1 = ga_substitute(
"((zetan)+(beta)*(B))", dict);
1154 alphanp1 = ga_substitute(
"((etan)+sqrt(2/3)*(beta)*(Norm_B))", dict);
1155 dict[
"alphanp1"] = alphanp1;
1157 sigma_np1 = ga_substitute
1158 (
"(lambda)*Trace(Enp1)*Id(meshdim)+2*(mu)*((Enp1)-(Epnp1))", dict);
1159 sigma_after = ga_substitute
1160 (
"(lambda)*Trace(Enp1)*Id(meshdim)+2*(mu)*((Enp1)-(Epn))", dict);
1161 xi_np1 = ga_substitute
1162 (
"(((beta)/(1-(2*(mu)+2/3*(Hk))*(beta)))/((theta)*(dt)))", dict);
1163 von_mises = ga_substitute
1164 (
"sqrt(3/2)*sqrt(Norm_sqr((2*(mu))*(Dev_En)-(2*(mu)+2/3*(Hk))*(Epn))"
1165 "+sqr(2*(mu)*Trace(En)/3-(2*(mu)+2/3*(Hk))*Trace(Epn)))", dict);
1168 void build_isotropic_perfect_elastoplasticity_expressions_generic
1169 (model &md,
const std::string &lawname,
1170 plasticity_unknowns_type unknowns_type,
1171 const std::vector<std::string> &varnames,
1172 const std::vector<std::string> ¶ms,
1173 std::string &sigma_np1, std::string &Epnp1,
1174 std::string &compcond, std::string &xi_np1,
1175 std::string &sigma_after, std::string &von_mises,
1176 std::string &alphanp1) {
1178 GMM_ASSERT1(unknowns_type == DISPLACEMENT_ONLY ||
1179 unknowns_type == DISPLACEMENT_AND_PLASTIC_MULTIPLIER,
1180 "Not supported type of unknowns");
1181 bool plastic_multiplier_is_var
1182 = (unknowns_type == DISPLACEMENT_AND_PLASTIC_MULTIPLIER);
1184 bool hardening = (lawname.find(
"_hardening") != std::string::npos);
1187 GMM_ASSERT1(varnames.size() == (hardening ? 4 : 3),
1188 "Incorrect number of variables: " << varnames.size());
1189 GMM_ASSERT1(params.size() >= 3+nhard &&
1190 params.size() <= 5+nhard,
1191 "Incorrect number of parameters: " << params.size());
1192 const std::string &dispname = sup_previous_and_dot_to_varname(varnames[0]);
1193 const std::string &xi = sup_previous_and_dot_to_varname(varnames[1]);
1194 const std::string &Previous_Ep = varnames[2];
1195 const std::string &
alpha = hardening ? varnames[3] :
"";
1196 const std::string &lambda = params[0];
1197 const std::string &mu = params[1];
1198 const std::string &sigma_y = params[2];
1199 const std::string &Hk = hardening ? params[3] :
"";
1200 const std::string &Hi = hardening ? params[4] :
"";
1201 const std::string &theta = (params.size() >= 4+nhard)
1202 ? params[3+nhard] :
"1";
1203 const std::string &dt = (params.size() >= 5+nhard)
1204 ? params[4+nhard] :
"timestep";
1206 sigma_np1 = Epnp1 = compcond = xi_np1 =
"";;
1207 sigma_after = von_mises = alphanp1 =
"";
1209 if (lawname.compare(
"isotropic_perfect_plasticity") == 0 ||
1210 lawname.compare(
"prandtl_reuss") == 0) {
1211 if (plastic_multiplier_is_var) {
1212 build_isotropic_perfect_elastoplasticity_expressions_mult
1213 (md, dispname, xi, Previous_Ep, lambda, mu, sigma_y, theta, dt,
1214 sigma_np1, Epnp1, compcond, sigma_after, von_mises);
1216 build_isotropic_perfect_elastoplasticity_expressions_no_mult
1217 (md, dispname, xi, Previous_Ep, lambda, mu, sigma_y, theta, dt,
1218 sigma_np1, Epnp1, xi_np1, sigma_after, von_mises);
1220 }
else if (lawname.compare(
"plane_strain_isotropic_perfect_plasticity")
1222 lawname.compare(
"plane_strain_prandtl_reuss") == 0) {
1223 if (plastic_multiplier_is_var) {
1224 build_isotropic_perfect_elastoplasticity_expressions_mult_ps
1225 (md, dispname, xi, Previous_Ep, lambda, mu, sigma_y, theta, dt,
1226 sigma_np1, Epnp1, compcond, sigma_after, von_mises);
1228 build_isotropic_perfect_elastoplasticity_expressions_no_mult_ps
1229 (md, dispname, xi, Previous_Ep, lambda, mu, sigma_y, theta, dt,
1230 sigma_np1, Epnp1, xi_np1, sigma_after, von_mises);
1232 }
else if (lawname.compare(
"isotropic_plasticity_linear_hardening") == 0 ||
1233 lawname.compare(
"prandtl_reuss_linear_hardening") == 0) {
1234 if (plastic_multiplier_is_var) {
1235 build_isotropic_perfect_elastoplasticity_expressions_hard_mult
1236 (md, dispname, xi, Previous_Ep, alpha,
1237 lambda, mu, sigma_y, Hk, Hi, theta, dt,
1238 sigma_np1, Epnp1, compcond, sigma_after, von_mises, alphanp1);
1240 build_isotropic_perfect_elastoplasticity_expressions_hard_no_mult
1241 (md, dispname, xi, Previous_Ep, alpha,
1242 lambda, mu, sigma_y, Hk, Hi, theta, dt,
1243 sigma_np1, Epnp1, xi_np1, sigma_after, von_mises, alphanp1);
1246 (lawname.compare(
"plane_strain_isotropic_plasticity_linear_hardening")
1248 lawname.compare(
"plane_strain_prandtl_reuss_linear_hardening") == 0) {
1249 if (plastic_multiplier_is_var) {
1250 build_isotropic_perfect_elastoplasticity_expressions_hard_mult_ps
1251 (md, dispname, xi, Previous_Ep, alpha,
1252 lambda, mu, sigma_y, Hk, Hi, theta, dt,
1253 sigma_np1, Epnp1, compcond, sigma_after, von_mises, alphanp1);
1255 build_isotropic_perfect_elastoplasticity_expressions_hard_no_mult_ps
1256 (md, dispname, xi, Previous_Ep, alpha,
1257 lambda, mu, sigma_y, Hk, Hi, theta, dt,
1258 sigma_np1, Epnp1, xi_np1, sigma_after, von_mises, alphanp1);
1261 GMM_ASSERT1(
false, lawname <<
" is not an implemented elastoplastic law");
1264 static void filter_lawname(std::string &lawname) {
1265 for (
auto &c : lawname)
1266 {
if (c ==
' ') c =
'_';
if (c >=
'A' && c <=
'Z') c = char(c+
'a'-
'A'); }
1271 std::string lawname, plasticity_unknowns_type unknowns_type,
1272 const std::vector<std::string> &varnames,
1273 const std::vector<std::string> ¶ms,
size_type region) {
1275 filter_lawname(lawname);
1276 std::string sigma_np1, compcond;
1278 std::string dum2, dum4, dum5, dum6, dum7;
1279 build_isotropic_perfect_elastoplasticity_expressions_generic
1280 (md, lawname, unknowns_type, varnames, params,
1281 sigma_np1, dum2, compcond, dum4, dum5, dum6, dum7);
1284 const std::string dispname=sup_previous_and_dot_to_varname(varnames[0]);
1285 const std::string xi =sup_previous_and_dot_to_varname(varnames[1]);
1287 if (unknowns_type == DISPLACEMENT_AND_PLASTIC_MULTIPLIER) {
1288 std::string expr = (
"("+sigma_np1+
"):Grad_Test_"+dispname
1289 +
"+("+compcond+
")*Test_"+xi);
1291 (md, mim, expr, region,
false,
false,
1292 "Small strain isotropic perfect elastoplasticity brick");
1295 (md, mim,
"("+sigma_np1+
"):Grad_Test_"+dispname, region,
true,
false,
1296 "Small strain isotropic perfect elastoplasticity brick");
1302 std::string lawname, plasticity_unknowns_type unknowns_type,
1303 const std::vector<std::string> &varnames,
1304 const std::vector<std::string> ¶ms,
size_type region) {
1306 filter_lawname(lawname);
1307 std::string Epnp1, xi_np1, alphanp1;
1309 std::string dum1, dum3, dum5, dum6;
1310 build_isotropic_perfect_elastoplasticity_expressions_generic
1311 (md, lawname, unknowns_type, varnames, params,
1312 dum1, Epnp1, dum3, xi_np1, dum5, dum6, alphanp1);
1315 std::string disp = sup_previous_and_dot_to_varname(varnames[0]);
1316 std::string xi = sup_previous_and_dot_to_varname(varnames[1]);
1317 std::string Previous_Ep = varnames[2];
1319 std::string Previous_alpha;
1320 base_vector tmpv_alpha;
1321 if (alphanp1.size()) {
1322 Previous_alpha = varnames[3];
1323 tmpv_alpha.resize(gmm::vect_size(md.
real_variable(Previous_alpha)));
1324 const im_data *pimd = md.pim_data_of_variable(Previous_alpha);
1326 ga_interpolation_im_data(md, alphanp1, *pimd, tmpv_alpha, region);
1329 GMM_ASSERT1(pmf,
"Provided data " << Previous_alpha
1330 <<
" should be defined on a im_data or a mesh_fem object");
1335 base_vector tmpv_xi;
1336 if (xi_np1.size()) {
1340 const im_data *pimd = md.pim_data_of_variable(xi);
1342 ga_interpolation_im_data(md, xi_np1, *pimd, tmpv_xi, region);
1345 GMM_ASSERT1(pmf,
"Provided data " << xi
1346 <<
" should be defined on a im_data or a mesh_fem object");
1351 base_vector tmpv_ep(gmm::vect_size(md.
real_variable(Previous_Ep)));
1352 const im_data *pimd = md.pim_data_of_variable(Previous_Ep);
1354 ga_interpolation_im_data(md, Epnp1, *pimd, tmpv_ep, region);
1357 GMM_ASSERT1(pmf,
"Provided data " << Previous_Ep
1358 <<
" should be defined on a im_data or a mesh_fem object");
1364 if (alphanp1.size())
1374 std::string lawname, plasticity_unknowns_type unknowns_type,
1375 const std::vector<std::string> &varnames,
1376 const std::vector<std::string> ¶ms,
1380 "Von mises stress can only be approximated on a scalar fem");
1381 VM.resize(mf_vm.
nb_dof());
1383 filter_lawname(lawname);
1385 std::string sigma_after, von_mises;
1387 std::string dum1, dum2, dum3, dum4, dum7;
1388 build_isotropic_perfect_elastoplasticity_expressions_generic
1389 (md, lawname, unknowns_type, varnames, params,
1390 dum1, dum2, dum3, dum4, sigma_after, von_mises, dum7);
1395 const im_data *pimd = md.pim_data_of_variable(varnames[n_ep]);
1401 GMM_ASSERT1(pmf,
"Provided data " << varnames[n_ep]
1402 <<
" should be defined on a im_data or a mesh_fem object");
1403 ga_interpolation_Lagrange_fem(md, von_mises, mf_vm, VM, region);
1415 const std::string _TWOTHIRD_(
"0.6666666666666666667");
1416 const std::string _FIVETHIRD_(
"1.6666666666666666667");
1417 const std::string _SQRTTHREEHALF_(
"1.2247448713915889407");
1420 (
const std::string &name, scalar_type sigma_y0, scalar_type H,
bool frobenius)
1423 sigma_y0 *= sqrt(2./3.);
1426 std::stringstream expr, der;
1427 expr << std::setprecision(17) << sigma_y0 <<
"+" << H <<
"*t";
1428 der << std::setprecision(17) << H;
1429 ga_define_function(name, 1, expr.str(), der.str());
1433 (
const std::string &name,
1434 scalar_type sigma_ref, scalar_type eps_ref, scalar_type n,
bool frobenius)
1436 scalar_type coef = sigma_ref / pow(eps_ref, 1./n);
1438 coef *= pow(2./3., 0.5 + 0.5/n);
1440 std::stringstream expr, der;
1441 expr << std::setprecision(17) << coef <<
"*pow(t+1e-12," << 1./n <<
")";
1442 der << std::setprecision(17) << coef/n <<
"*pow(t+1e-12," << 1./n-1 <<
")";
1443 ga_define_function(name, 1, expr.str(), der.str());
1447 void build_Simo_Miehe_elastoplasticity_expressions
1448 (model &md, plasticity_unknowns_type unknowns_type,
1449 const std::vector<std::string> &varnames,
1450 const std::vector<std::string> ¶ms,
1451 std::string &expr, std::string &plaststrain, std::string &invCp, std::string &vm)
1453 GMM_ASSERT1(unknowns_type == DISPLACEMENT_AND_PLASTIC_MULTIPLIER ||
1454 unknowns_type == DISPLACEMENT_AND_PLASTIC_MULTIPLIER_AND_PRESSURE,
1455 "Not supported type of unknowns for this type of plasticity law");
1456 bool has_pressure_var(unknowns_type ==
1457 DISPLACEMENT_AND_PLASTIC_MULTIPLIER_AND_PRESSURE);
1458 GMM_ASSERT1(varnames.size() == (has_pressure_var ? 5 : 4),
1459 "Wrong number of variables.");
1460 GMM_ASSERT1(params.size() == 3,
"Wrong number of parameters, "
1461 << params.size() <<
" != 3.");
1462 const std::string &dispname = sup_previous_and_dot_to_varname(varnames[0]);
1463 const std::string &multname = sup_previous_and_dot_to_varname(varnames[1]);
1464 const std::string &pressname = has_pressure_var ? varnames[2] :
"";
1465 const std::string &plaststrain0 = varnames[has_pressure_var ? 3 : 2];
1466 const std::string &invCp0 = varnames[has_pressure_var ? 4 : 3];
1467 const std::string &K = params[0];
1468 const std::string &G = params[1];
1469 const std::string &sigma_y = params[2];
1471 const mesh_fem *mfu = md.pmesh_fem_of_variable(dispname);
1472 GMM_ASSERT1(mfu,
"The provided displacement variable " << dispname <<
1473 " has to be defined on a mesh_fem");
1475 GMM_ASSERT1(N >= 2 && N <= 3,
1476 "Finite strain elastoplasticity brick works only in 2D or 3D");
1477 GMM_ASSERT1(mfu && mfu->get_qdim() == N,
"The finite strain "
1478 "elastoplasticity brick can only be applied on a fem "
1479 "variable of the same dimension as the mesh");
1480 const mesh_fem *mfmult = md.pmesh_fem_of_variable(multname);
1481 GMM_ASSERT1(mfmult && mfmult->get_qdim() == 1,
"The plastic multiplier "
1482 "for the finite strain elastoplasticity brick has to be a "
1483 "scalar fem variable");
1484 bool mixed(!pressname.empty());
1485 const mesh_fem *mfpress = mixed ? md.pmesh_fem_of_variable(pressname) : 0;
1486 GMM_ASSERT1(!mixed || (mfpress && mfpress->get_qdim() == 1),
1487 "The hydrostatic pressure multiplier for the finite strain "
1488 "elastoplasticity brick has to be a scalar fem variable");
1490 GMM_ASSERT1(ga_function_exists(sigma_y),
"The provided isotropic "
1491 "hardening function name '" << sigma_y <<
"' is not defined");
1493 GMM_ASSERT1(md.is_data(plaststrain0) &&
1494 (md.pim_data_of_variable(plaststrain0) ||
1495 md.pmesh_fem_of_variable(plaststrain0)),
1496 "The provided name '" << plaststrain0 <<
"' for the plastic "
1497 "strain field at the previous timestep, should be defined "
1498 "either as fem or as im data");
1499 GMM_ASSERT1((md.pim_data_of_variable(plaststrain0) &&
1500 md.pim_data_of_variable(plaststrain0)->nb_tensor_elem() == 1) ||
1501 (md.pmesh_fem_of_variable(plaststrain0) &&
1502 md.pmesh_fem_of_variable(plaststrain0)->get_qdim() == 1),
1503 "Wrong size of " << plaststrain0);
1504 GMM_ASSERT1(md.is_data(invCp0) &&
1505 (md.pim_data_of_variable(invCp0) ||
1506 md.pmesh_fem_of_variable(invCp0)),
1507 "The provided name '" << invCp0 <<
"' for the inverse of the "
1508 "plastic right Cauchy-Green tensor field at the previous "
1509 "timestep, should be defined either as fem or as im data");
1510 bgeot::multi_index Cp_size(1);
1511 Cp_size[0] = 4 + (N==3)*2;
1512 GMM_ASSERT1((md.pim_data_of_variable(invCp0) &&
1513 md.pim_data_of_variable(invCp0)->tensor_size() == Cp_size) ||
1514 (md.pmesh_fem_of_variable(invCp0) &&
1515 md.pmesh_fem_of_variable(invCp0)->get_qdims() == Cp_size),
1516 "Wrong size of " << invCp0);
1518 const std::string _U_ = sup_previous_and_dot_to_varname(dispname);
1519 const std::string _KSI_ = sup_previous_and_dot_to_varname(multname);
1520 const std::string _I_(N == 2 ?
"Id(2)" :
"Id(3)");
1521 const std::string _F_(
"("+_I_+
"+Grad_"+_U_+
")");
1522 const std::string _J_(
"Det"+_F_);
1526 _P_ =
"-"+pressname+
"*"+_J_;
1528 _P_ =
"("+K+
")*log("+_J_+
")";
1530 std::string _INVCP0_, _F3d_, _DEVLOGBETR_, _DEVLOGBETR_3D_;
1532 _INVCP0_ =
"([[[1,0,0],[0,0,0],[0,0,0]],"
1533 "[[0,0,0],[0,1,0],[0,0,0]],"
1534 "[[0,0,0],[0,0,0],[0,0,1]],"
1535 "[[0,1,0],[1,0,0],[0,0,0]]]."+invCp0+
")";
1536 _F3d_ =
"(Id(3)+[[1,0,0],[0,1,0]]*Grad_"+_U_+
"*[[1,0,0],[0,1,0]]')";
1537 _DEVLOGBETR_3D_ =
"(Deviator(Logm("+_F3d_+
"*"+_INVCP0_+
"*"+_F3d_+
"')))";
1538 _DEVLOGBETR_ =
"([[[[1,0],[0,0]],[[0,1],[0,0]],[[0,0],[0,0]]],"
1539 "[[[0,0],[1,0]],[[0,0],[0,1]],[[0,0],[0,0]]],"
1540 "[[[0,0],[0,0]],[[0,0],[0,0]],[[0,0],[0,0]]]]"
1541 ":"+_DEVLOGBETR_3D_+
")";
1543 _INVCP0_ =
"([[[1,0,0],[0,0,0],[0,0,0]],"
1544 "[[0,0,0],[0,1,0],[0,0,0]],"
1545 "[[0,0,0],[0,0,0],[0,0,1]],"
1546 "[[0,1,0],[1,0,0],[0,0,0]],"
1547 "[[0,0,1],[0,0,0],[1,0,0]],"
1548 "[[0,0,0],[0,0,1],[0,1,0]]]."+invCp0+
")";
1551 _DEVLOGBETR_ =
"(Deviator(Logm("+_F_+
"*"+_INVCP0_+
"*"+_F_+
"')))";
1553 const std::string _DEVTAUTR_(
"("+G+
"*"+_DEVLOGBETR_+
")");
1554 const std::string _DEVTAUTR_3D_(
"("+G+
"*"+_DEVLOGBETR_3D_+
")");
1555 const std::string _DEVTAU_(
"((1-2*"+_KSI_+
")*"+_DEVTAUTR_+
")");
1556 const std::string _DEVTAU_3D_(
"((1-2*"+_KSI_+
")*"+_DEVTAUTR_3D_+
")");
1557 const std::string _TAU_(
"("+_P_+
"*"+_I_+
"+"+_DEVTAU_+
")");
1559 const std::string _PLASTSTRAIN_(
"("+plaststrain0+
"+"+_KSI_+
"*Norm"+_DEVLOGBETR_3D_+
")");
1560 const std::string _SIGMA_Y_(sigma_y+
"("+_PLASTSTRAIN_+
")");
1563 expr = _TAU_+
":(Grad_Test_"+_U_+
"*Inv"+_F_+
")";
1565 expr +=
"+("+pressname+
"/("+K+
")+log("+_J_+
")/"+_J_+
")*Test_"+pressname;
1566 expr +=
"+(Norm"+_DEVTAU_+
1567 "-min("+_SIGMA_Y_+
",Norm"+_DEVTAUTR_+
"+1e-12*"+_KSI_+
"))*Test_"+_KSI_;
1569 plaststrain = _PLASTSTRAIN_;
1571 if (N==2) invCp =
"[[[1,0,0,0.0],[0,0,0,0.5],[0,0,0,0]],"
1572 "[[0,0,0,0.5],[0,1,0,0.0],[0,0,0,0]],"
1573 "[[0,0,0,0.0],[0,0,0,0.0],[0,0,1,0]]]";
1574 else invCp =
"[[[1.0,0,0,0,0,0],[0,0,0,0.5,0,0],[0,0,0,0,0.5,0]],"
1575 "[[0,0,0,0.5,0,0],[0,1.0,0,0,0,0],[0,0,0,0,0,0.5]],"
1576 "[[0,0,0,0,0.5,0],[0,0,0,0,0,0.5],[0,0,1.0,0,0,0]]]";
1577 invCp +=
":((Inv"+_F3d_+
"*Expm(-"+_KSI_+
"*"+_DEVLOGBETR_3D_+
")*"+_F3d_+
")*"+_INVCP0_+
1578 "*(Inv"+_F3d_+
"*Expm(-"+_KSI_+
"*"+_DEVLOGBETR_3D_+
")*"+_F3d_+
")')";
1580 vm = _SQRTTHREEHALF_+
"*Norm("+_DEVTAU_+
")/"+_J_;
1584 (model &md,
const mesh_im &mim,
1585 std::string lawname, plasticity_unknowns_type unknowns_type,
1586 const std::vector<std::string> &varnames,
1587 const std::vector<std::string> ¶ms,
size_type region)
1589 filter_lawname(lawname);
1590 if (lawname.compare(
"simo_miehe") == 0 ||
1591 lawname.compare(
"eterovic_bathe") == 0) {
1592 std::string expr, dummy1, dummy2, dummy3;
1593 build_Simo_Miehe_elastoplasticity_expressions
1594 (md, unknowns_type, varnames, params, expr, dummy1, dummy2, dummy3);
1596 (md, mim, expr, region,
true,
false,
"Simo Miehe elastoplasticity brick");
1598 GMM_ASSERT1(
false, lawname <<
" is not a known elastoplastic law");
1603 (model &md,
const mesh_im &mim,
1604 std::string lawname, plasticity_unknowns_type unknowns_type,
1605 const std::vector<std::string> &varnames,
1606 const std::vector<std::string> ¶ms,
size_type region) {
1608 filter_lawname(lawname);
1609 if (lawname.compare(
"simo_miehe") == 0 ||
1610 lawname.compare(
"eterovic_bathe") == 0) {
1611 std::string dummy1, dummy2, plaststrain, invCp;
1612 build_Simo_Miehe_elastoplasticity_expressions
1613 (md, unknowns_type, varnames, params, dummy1, plaststrain, invCp, dummy2);
1615 bool has_pressure_var(unknowns_type ==
1616 DISPLACEMENT_AND_PLASTIC_MULTIPLIER_AND_PRESSURE);
1617 const std::string &multname = sup_previous_and_dot_to_varname(varnames[1]);
1618 const std::string &plaststrain0 = varnames[has_pressure_var ? 3 : 2];
1619 const std::string &invCp0 = varnames[has_pressure_var ? 4 : 3];
1621 model_real_plain_vector tmpvec(gmm::vect_size(md.real_variable(plaststrain0)));
1622 const im_data *pimd = md.pim_data_of_variable(plaststrain0);
1624 ga_interpolation_im_data(md, plaststrain, *pimd, tmpvec, region);
1626 const mesh_fem *pmf = md.pmesh_fem_of_variable(plaststrain0);
1627 GMM_ASSERT1(pmf,
"Provided data " << plaststrain0 <<
" should be defined "
1628 "either on a im_data or a mesh_fem object");
1632 gmm::copy(tmpvec, md.set_real_variable(plaststrain0));
1636 model_real_plain_vector tmpvec(gmm::vect_size(md.real_variable(invCp0)));
1637 const im_data *pimd = md.pim_data_of_variable(invCp0);
1639 ga_interpolation_im_data(md, invCp, *pimd, tmpvec, region);
1641 const mesh_fem *pmf = md.pmesh_fem_of_variable(invCp0);
1642 GMM_ASSERT1(pmf,
"Provided data " << invCp0 <<
" should be defined "
1643 "either on a im_data or a mesh_fem object");
1647 gmm::copy(tmpvec, md.set_real_variable(invCp0));
1653 GMM_ASSERT1(
false, lawname <<
" is not a known elastoplastic law");
1657 (model &md,
const mesh_im &mim,
1658 std::string lawname, plasticity_unknowns_type unknowns_type,
1659 const std::vector<std::string> &varnames,
1660 const std::vector<std::string> ¶ms,
1661 const mesh_fem &mf_vm, model_real_plain_vector &VM,
1664 filter_lawname(lawname);
1665 if (lawname.compare(
"simo_miehe") == 0 ||
1666 lawname.compare(
"eterovic_bathe") == 0) {
1667 std::string dummy1, dummy2, dummy3, von_mises;
1668 build_Simo_Miehe_elastoplasticity_expressions
1669 (md, unknowns_type, varnames, params, dummy1, dummy2, dummy3, von_mises);
1670 VM.resize(mf_vm.nb_dof());
1672 bool has_pressure_var(unknowns_type ==
1673 DISPLACEMENT_AND_PLASTIC_MULTIPLIER_AND_PRESSURE);
1674 const std::string &plaststrain0 = varnames[has_pressure_var ? 3 : 2];
1675 const std::string &invCp0 = varnames[has_pressure_var ? 4 : 3];
1676 bool im_data1 = md.pim_data_of_variable(plaststrain0) != 0;
1677 bool im_data2 = md.pim_data_of_variable(invCp0) != 0;
1678 bool fem_data1 = md.pmesh_fem_of_variable(plaststrain0) != 0;
1679 bool fem_data2 = md.pmesh_fem_of_variable(invCp0) != 0;
1680 GMM_ASSERT1(im_data1 || fem_data1,
1681 "Provided data " << plaststrain0 <<
1682 " should be defined on a im_data or a mesh_fem object");
1683 GMM_ASSERT1(im_data2 || fem_data2,
1684 "Provided data " << invCp0 <<
1685 " should be defined on a im_data or a mesh_fem object");
1686 if (im_data1 || im_data2) {
1689 ga_interpolation_Lagrange_fem(md, von_mises, mf_vm, VM, region);
1693 GMM_ASSERT1(
false, lawname <<
" is not a known elastoplastic law");
1722 enum elastoplasticity_nonlinear_term_version { PROJ,
1729 class elastoplasticity_nonlinear_term :
public nonlinear_elem_term {
1733 const mesh_fem &mf_u;
1734 const mesh_fem &mf_sigma;
1735 const mesh_fem *pmf_data;
1736 model_real_plain_vector U_n, U_np1;
1737 model_real_plain_vector Sigma_n;
1738 model_real_plain_vector threshold, lambda, mu;
1739 const abstract_constraints_projection &t_proj;
1742 const bool store_sigma;
1744 bgeot::multi_index sizes_;
1751 model_real_plain_vector convex_coeffs, interpolated_val;
1754 model_real_plain_vector cumulated_sigma;
1756 model_real_plain_vector cumulated_count;
1758 fem_precomp_pool fppool;
1762 void compute_convex_coeffs(
size_type cv) {
1766 pfem pf_sigma = mf_sigma.fem_of_element(cv);
1767 size_type nbd_sigma = pf_sigma->nb_dof(cv);
1768 size_type qdim_sigma = mf_sigma.get_qdim();
1773 bgeot::vectors_to_base_matrix
1774 (G, mf_u.linked_mesh().points_of_convex(cv));
1776 mf_u.linked_mesh().trans_of_convex(cv);
1779 base_vector coeff_data;
1781 fem_interpolation_context ctx_data;
1783 pf_data = pmf_data->fem_of_element(cv);
1784 size_type nbd_data = pf_data->nb_dof(cv);
1785 coeff_data.resize(nbd_data*3);
1788 mesh_fem::ind_dof_ct::const_iterator itdof
1789 = pmf_data->ind_basic_dof_of_element(cv).begin();
1790 for (
size_type k = 0; k < nbd_data; ++k, ++itdof) {
1791 coeff_data[k*3] = lambda[*itdof];
1792 coeff_data[k*3+1] = mu[*itdof];
1793 coeff_data[k*3+2] = threshold[*itdof];
1795 GMM_ASSERT1(pf_data->target_dim() == 1,
1796 "won't interpolate on a vector FEM... ");
1798 pfem_precomp pfp_data = fppool(pf_data, pf_sigma->node_tab(cv));
1799 ctx_data = fem_interpolation_context
1804 size_type cvnbdof_u = mf_u.nb_basic_dof_of_element(cv);
1805 model_real_plain_vector coeff_du(cvnbdof_u);
1806 model_real_plain_vector coeff_u_np1(cvnbdof_u);
1807 mesh_fem::ind_dof_ct::const_iterator itdof
1808 = mf_u.ind_basic_dof_of_element(cv).begin();
1809 for (
size_type k = 0; k < cvnbdof_u; ++k, ++itdof) {
1810 coeff_du[k] = U_np1[*itdof] - U_n[*itdof];
1811 coeff_u_np1[k] = U_np1[*itdof];
1814 pfem pf_u = mf_u.fem_of_element(cv);
1815 pfem_precomp pfp_u = fppool(pf_u, pf_sigma->node_tab(cv));
1816 fem_interpolation_context
1820 base_matrix G_du(qdim, qdim), G_u_np1(qdim, qdim);
1822 for (
size_type ii = 0; ii < nbd_sigma; ++ii) {
1826 ctx_data.set_ii(ii);
1827 pf_data->interpolation(ctx_data, coeff_data, params, 3);
1832 pf_u->interpolation_grad(ctx_u, coeff_du, G_du, dim_type(qdim));
1833 if (option == PLAST)
1834 pf_u->interpolation_grad(ctx_u, coeff_u_np1, G_u_np1, dim_type(qdim));
1838 scalar_type ltrace_eps_np1 = (option == PLAST) ?
1839 params[0]*gmm::mat_trace(G_u_np1) : 0.;
1843 base_matrix sigma_hat(qdim, qdim);
1844 size_type sigma_dof = mf_sigma.ind_basic_dof_of_element(cv)[ii*qdim_sigma];
1845 for (dim_type j = 0; j < qdim; ++j) {
1846 for (dim_type i = 0; i < qdim; ++i)
1847 sigma_hat(i,j) = Sigma_n[sigma_dof++]
1848 + params[1]*(G_du(i,j) + G_du(j,i));
1849 sigma_hat(j,j) += ltrace_deps;
1854 t_proj.do_projection(sigma_hat, params[2], proj, flag_proj);
1857 if (option == PLAST)
1858 for (dim_type i = 0; i < qdim; ++i) {
1859 for (dim_type j = 0; j < qdim; ++j)
1860 proj(i,j) -= params[1]*(G_u_np1(i,j) + G_u_np1(j,i));
1861 proj(i,i) -= ltrace_eps_np1;
1865 std::copy(proj.begin(), proj.end(),
1866 convex_coeffs.begin() + proj.size() * ii);
1870 sigma_dof = mf_sigma.ind_basic_dof_of_element(cv)[ii*qdim_sigma];
1871 for (dim_type j = 0; j < qdim; ++j) {
1872 for (dim_type i = 0; i < qdim; ++i) {
1873 cumulated_count[sigma_dof] += 1;
1874 cumulated_sigma[sigma_dof++] += proj(i,j);
1886 elastoplasticity_nonlinear_term
1887 (
const mesh_im &mim_,
1888 const mesh_fem &mf_u_,
1889 const mesh_fem &mf_sigma_,
1890 const mesh_fem *pmf_data_,
1891 const model_real_plain_vector &U_n_,
1892 const model_real_plain_vector &U_np1_,
1893 const model_real_plain_vector &Sigma_n_,
1894 const model_real_plain_vector &threshold_,
1895 const model_real_plain_vector &lambda_,
1896 const model_real_plain_vector &mu_,
1897 const abstract_constraints_projection &t_proj_,
1899 bool store_sigma_) :
1900 mim(mim_), mf_u(mf_u_), mf_sigma(mf_sigma_), pmf_data(pmf_data_),
1901 Sigma_n(Sigma_n_), t_proj(t_proj_), option(option_),
1902 flag_proj(option == GRADPROJ ? 1 : 0),
1903 store_sigma(option == GRADPROJ ? false : store_sigma_) {
1906 N = mf_u.linked_mesh().dim();
1908 sizes_ = (flag_proj == 0 ? bgeot::multi_index(N,N)
1909 :
bgeot::multi_index(N,N,N,N));
1913 size_proj = (flag_proj == 0 ? N*N : N*N*N*N);
1918 mf_u.extend_vector(gmm::sub_vector(U_n_,
1919 gmm::sub_interval(0,mf_u.nb_dof())),
1921 mf_u.extend_vector(gmm::sub_vector(U_np1_,
1922 gmm::sub_interval(0,mf_u.nb_dof())),
1924 mf_sigma.extend_vector(gmm::sub_vector(Sigma_n_,
1925 gmm::sub_interval(0,mf_sigma.nb_dof())),
1932 pmf_data->extend_vector(threshold_, threshold);
1933 pmf_data->extend_vector(lambda_, lambda);
1934 pmf_data->extend_vector(mu_, mu);
1938 gmm::resize(threshold, 1); threshold[0] = threshold_[0];
1939 params[0] = lambda[0];
1941 params[2] = threshold[0];
1943 GMM_ASSERT1(mf_u.get_qdim() == N,
1944 "wrong qdim for the mesh_fem");
1949 cumulated_sigma.resize(mf_sigma.nb_dof());
1950 cumulated_count.resize(mf_sigma.nb_dof());
1961 const bgeot::multi_index &sizes(
size_type)
const {
return sizes_; }
1965 virtual void compute(fem_interpolation_context& ctx,
1966 bgeot::base_tensor &t) {
1968 pfem pf_sigma = ctx.pf();
1969 GMM_ASSERT1(pf_sigma->is_lagrange(),
1970 "Sorry, works only for Lagrange fems");
1973 if (cv != current_cv)
1974 compute_convex_coeffs(cv);
1977 pf_sigma->interpolation(ctx, convex_coeffs, interpolated_val, dim_type(size_proj));
1980 t.adjust_sizes(sizes_);
1981 std::copy(interpolated_val.begin(), interpolated_val.end(), t.begin());
1986 void get_averaged_sigmas(model_real_plain_vector &sigma) {
1987 model_real_plain_vector glob_cumulated_count(mf_sigma.nb_dof());
1988 MPI_SUM_VECTOR(cumulated_sigma, sigma);
1989 MPI_SUM_VECTOR(cumulated_count, glob_cumulated_count);
1992 sigma[i] /= glob_cumulated_count[i];
2003 void asm_elastoplasticity_rhs
2004 (model_real_plain_vector &V,
2005 model_real_plain_vector *saved_sigma,
2007 const mesh_fem &mf_u,
2008 const mesh_fem &mf_sigma,
2009 const mesh_fem &mf_data,
2010 const model_real_plain_vector &u_n,
2011 const model_real_plain_vector &u_np1,
2012 const model_real_plain_vector &sigma_n,
2013 const model_real_plain_vector &lambda,
2014 const model_real_plain_vector &mu,
2015 const model_real_plain_vector &threshold,
2016 const abstract_constraints_projection &t_proj,
2020 GMM_ASSERT1(mf_u.get_qdim() == mf_u.linked_mesh().dim(),
2021 "wrong qdim for the mesh_fem");
2022 GMM_ASSERT1(option_sigma == PROJ || option_sigma == PLAST,
2023 "wrong option parameter");
2025 elastoplasticity_nonlinear_term plast(mim, mf_u, mf_sigma, &mf_data,
2026 u_n, u_np1, sigma_n,
2027 threshold, lambda, mu,
2028 t_proj, option_sigma, (saved_sigma != NULL));
2030 generic_assembly assem(
"V(#1) + =comp(NonLin(#2).vGrad(#1))(i,j,:,i,j);");
2033 assem.push_mf(mf_u);
2034 assem.push_mf(mf_sigma);
2035 assem.push_nonlinear_term(&plast);
2040 plast.get_averaged_sigmas(*saved_sigma);
2048 void asm_elastoplasticity_tangent_matrix
2049 (model_real_sparse_matrix &H,
2051 const mesh_fem &mf_u,
2052 const mesh_fem &mf_sigma,
2053 const mesh_fem *pmf_data,
2054 const model_real_plain_vector &u_n,
2055 const model_real_plain_vector &u_np1,
2056 const model_real_plain_vector &sigma_n,
2057 const model_real_plain_vector &lambda,
2058 const model_real_plain_vector &mu,
2059 const model_real_plain_vector &threshold,
2060 const abstract_constraints_projection &t_proj,
2063 GMM_ASSERT1(mf_u.get_qdim() == mf_u.linked_mesh().dim(),
2064 "wrong qdim for the mesh_fem");
2066 elastoplasticity_nonlinear_term gradplast(mim, mf_u, mf_sigma, pmf_data,
2067 u_n, u_np1, sigma_n,
2068 threshold, lambda, mu,
2069 t_proj, GRADPROJ,
false);
2071 generic_assembly assem;
2074 assem.set(
"lambda=data$1(#3); mu=data$2(#3);"
2075 "t=comp(NonLin(#2).vGrad(#1).vGrad(#1).Base(#3))(i,j,:,:,:,:,:,:,i,j,:);"
2076 "M(#1,#1)+= sym(t(k,l,:,l,k,:,m).mu(m)+t(k,l,:,k,l,:,m).mu(m)+t(k,k,:,l,l,:,m).lambda(m))");
2078 assem.set(
"lambda=data$1(1); mu=data$2(1);"
2079 "t=comp(NonLin(#2).vGrad(#1).vGrad(#1))(i,j,:,:,:,:,:,:,i,j);"
2080 "M(#1,#1)+= sym(t(k,l,:,l,k,:).mu(1)+t(k,l,:,k,l,:).mu(1)+t(k,k,:,l,l,:).lambda(1))");
2083 assem.push_mf(mf_u);
2084 assem.push_mf(mf_sigma);
2086 assem.push_mf(*pmf_data);
2087 assem.push_data(lambda);
2088 assem.push_data(mu);
2089 assem.push_nonlinear_term(&gradplast);
2101 struct elastoplasticity_brick :
public virtual_brick {
2103 pconstraints_projection t_proj;
2105 virtual void asm_real_tangent_terms(
const model &md,
2107 const model::varnamelist &vl,
2108 const model::varnamelist &dl,
2109 const model::mimlist &mims,
2110 model::real_matlist &matl,
2111 model::real_veclist &vecl,
2112 model::real_veclist &,
2114 build_version version)
const {
2116 GMM_ASSERT1(mims.size() == 1,
2117 "Elastoplasticity brick need a single mesh_im");
2118 GMM_ASSERT1(vl.size() == 1,
2119 "Elastoplasticity brick need one variable");
2122 GMM_ASSERT1(dl.size() == 5,
2123 "Wrong number of data for elastoplasticity brick, "
2124 << dl.size() <<
" should be 4.");
2125 GMM_ASSERT1(matl.size() == 1,
"Wrong number of terms for "
2126 "elastoplasticity brick");
2128 const model_real_plain_vector &u_np1 = md.real_variable(vl[0]);
2129 const model_real_plain_vector &u_n = md.real_variable(dl[4]);
2130 const mesh_fem &mf_u = *(md.pmesh_fem_of_variable(vl[0]));
2131 GMM_ASSERT1(&mf_u == md.pmesh_fem_of_variable(dl[4]),
2132 "The previous displacement data have to be defined on the "
2133 "same mesh_fem as the displacement variable");
2135 const model_real_plain_vector &lambda = md.real_variable(dl[0]);
2136 const model_real_plain_vector &mu = md.real_variable(dl[1]);
2137 const model_real_plain_vector &threshold = md.real_variable(dl[2]);
2138 const mesh_fem *mf_data = md.pmesh_fem_of_variable(dl[0]);
2140 const model_real_plain_vector &sigma_n = md.real_variable(dl[3]);
2141 const mesh_fem &mf_sigma = *(md.pmesh_fem_of_variable(dl[3]));
2142 GMM_ASSERT1(!(mf_sigma.is_reduced()),
2143 "Works only for pure Lagrange fems");
2145 const mesh_im &mim = *mims[0];
2146 mesh_region rg(region);
2147 mim.linked_mesh().intersect_with_mpi_region(rg);
2149 if (version & model::BUILD_MATRIX) {
2151 asm_elastoplasticity_tangent_matrix
2152 (matl[0], mim, mf_u, mf_sigma, mf_data, u_n,
2153 u_np1, sigma_n, lambda, mu, threshold, *t_proj, rg);
2156 if (version & model::BUILD_RHS) {
2157 model_real_plain_vector *dummy = 0;
2158 asm_elastoplasticity_rhs(vecl[0], dummy,
2159 mim, mf_u, mf_sigma, *mf_data,
2160 u_n, u_np1, sigma_n,
2161 lambda, mu, threshold, *t_proj, PROJ, rg);
2162 gmm::scale(vecl[0], scalar_type(-1));
2168 elastoplasticity_brick(
const pconstraints_projection &t_proj_)
2170 set_flags(
"Elastoplasticity brick",
false ,
2185 const pconstraints_projection &ACP,
2186 const std::string &varname,
2187 const std::string &data_previous_disp,
2188 const std::string &datalambda,
2189 const std::string &datamu,
2190 const std::string &datathreshold,
2191 const std::string &datasigma,
2194 pbrick pbr = std::make_shared<elastoplasticity_brick>(ACP);
2196 model::termlist tl{model::term_description(varname, varname,
true)};
2198 dl{datalambda, datamu, datathreshold, datasigma, data_previous_disp},
2201 return md.add_brick(pbr, vl, dl, tl,
2202 model::mimlist(1,&mim), region);
2214 const std::string &varname,
2215 const std::string &data_previous_disp,
2216 const pconstraints_projection &ACP,
2217 const std::string &datalambda,
2218 const std::string &datamu,
2219 const std::string &datathreshold,
2220 const std::string &datasigma) {
2222 const model_real_plain_vector &u_np1 = md.real_variable(varname);
2223 model_real_plain_vector &u_n = md.set_real_variable(data_previous_disp);
2224 const mesh_fem &mf_u = *(md.pmesh_fem_of_variable(varname));
2226 const model_real_plain_vector &lambda = md.real_variable(datalambda);
2227 const model_real_plain_vector &mu = md.real_variable(datamu);
2228 const model_real_plain_vector &threshold = md.real_variable(datathreshold);
2229 const mesh_fem *mf_data = md.pmesh_fem_of_variable(datalambda);
2231 const model_real_plain_vector &sigma_n = md.real_variable(datasigma);
2232 const mesh_fem &mf_sigma = *(md.pmesh_fem_of_variable(datasigma));
2236 mesh_region rg = mim.linked_mesh().get_mpi_region();
2238 model_real_plain_vector sigma_np1(mf_sigma.nb_dof());
2239 model_real_plain_vector dummyV(mf_u.nb_dof());
2240 asm_elastoplasticity_rhs(dummyV, &sigma_np1,
2241 mim, mf_u, mf_sigma, *mf_data,
2242 u_n, u_np1, sigma_n,
2243 lambda, mu, threshold, *ACP, PROJ, rg);
2248 gmm::copy(sigma_np1, md.set_real_variable(datasigma));
2261 const std::string &datasigma,
2262 const mesh_fem &mf_vm,
2263 model_real_plain_vector &VM,
2266 GMM_ASSERT1(gmm::vect_size(VM) == mf_vm.nb_dof(),
2267 "The vector has not the right size");
2269 const model_real_plain_vector &sigma_np1 = md.real_variable(datasigma);
2270 const mesh_fem &mf_sigma = md.mesh_fem_of_variable(datasigma);
2273 dim_type N = mf_sigma.linked_mesh().dim();
2275 GMM_ASSERT1(mf_vm.get_qdim() == 1,
2276 "Target dimension of mf_vm should be 1");
2278 base_matrix sigma(N, N), Id(N, N);
2280 base_vector sigma_vm(mf_vm.nb_dof()*N*N);
2287 for (
size_type ii = 0; ii < mf_vm.nb_dof(); ++ii) {
2290 std::copy(sigma_vm.begin()+ii*N*N, sigma_vm.begin()+(ii+1)*N*N,
2295 gmm::add(gmm::scaled(Id, -gmm::mat_trace(sigma) / N), sigma);
2301 gmm::symmetric_qr_algorithm(sigma, eig);
2302 std::sort(eig.begin(), eig.end());
2303 VM[ii] = eig.back() - eig.front();
2316 const mesh_fem &mf_pl,
2317 const std::string &varname,
2318 const std::string &data_previous_disp,
2319 const pconstraints_projection &ACP,
2320 const std::string &datalambda,
2321 const std::string &datamu,
2322 const std::string &datathreshold,
2323 const std::string &datasigma,
2324 model_real_plain_vector &plast) {
2326 const model_real_plain_vector &u_np1 = md.real_variable(varname);
2327 const model_real_plain_vector &u_n = md.real_variable(data_previous_disp);
2328 const mesh_fem &mf_u = *(md.pmesh_fem_of_variable(varname));
2330 const model_real_plain_vector &lambda = md.real_variable(datalambda);
2331 const model_real_plain_vector &mu = md.real_variable(datamu);
2332 const model_real_plain_vector &threshold = md.real_variable(datathreshold);
2333 const mesh_fem *pmf_data = md.pmesh_fem_of_variable(datalambda);
2335 const model_real_plain_vector &sigma_n = md.real_variable(datasigma);
2336 const mesh_fem &mf_sigma = *(md.pmesh_fem_of_variable(datasigma));
2338 dim_type N = mf_sigma.linked_mesh().dim();
2340 mesh_region rg = mim.linked_mesh().get_mpi_region();
2342 model_real_plain_vector dummyV(mf_u.nb_dof());
2343 model_real_plain_vector saved_plast(mf_sigma.nb_dof());
2344 asm_elastoplasticity_rhs(dummyV, &saved_plast,
2345 mim, mf_u, mf_sigma, *pmf_data,
2346 u_n, u_np1, sigma_n,
2347 lambda, mu, threshold, *ACP, PLAST, rg);
2350 GMM_ASSERT1(gmm::vect_size(plast) == mf_pl.nb_dof(),
2351 "The vector has not the right size");
2352 GMM_ASSERT1(mf_pl.get_qdim() == 1,
2353 "Target dimension of mf_pl should be 1");
2355 base_vector saved_pl(mf_pl.nb_dof()*N*N);
2359 base_matrix plast_tmp(N, N);
2360 for (
size_type ii = 0; ii < mf_pl.nb_dof(); ++ii) {
2363 std::copy(saved_pl.begin()+ii*N*N, saved_pl.begin()+(ii+1)*N*N,
static T & instance()
Instance from the current thread.
im_data provides indexing to the integration points of a mesh im object.
Describe a finite element method linked to a mesh.
virtual dim_type get_qdim() const
Return the Q dimension.
virtual size_type nb_dof() const
Return the total number of degrees of freedom.
Describe an integration method linked to a mesh.
static mesh_region all_convexes()
provide a default value for the mesh_region parameters of assembly procedures etc.
`‘Model’' variables store the variables, the data and the description of a model.
const mesh_fem * pmesh_fem_of_variable(const std::string &name) const
Gives a pointer to the mesh_fem of a variable if any.
model_real_plain_vector & set_real_variable(const std::string &name, size_type niter) const
Gives the write access to the vector value of a variable.
const model_real_plain_vector & real_variable(const std::string &name, size_type niter) const
Gives the access to the vector value of a variable.
A language for generic assembly of pde boundary value problems.
Interpolation of fields from a mesh_fem onto another.
Model representation in Getfem.
void copy(const L1 &l1, L2 &l2)
*/
number_traits< typename linalg_traits< V >::value_type >::magnitude_type vect_norm2(const V &v)
Euclidean norm of a vector.
linalg_traits< M >::value_type mat_trace(const M &m)
Trace of a matrix.
number_traits< typename linalg_traits< M >::value_type >::magnitude_type mat_euclidean_norm(const M &m)
Euclidean norm of a matrix.
void clear(L &l)
clear (fill with zeros) a vector or matrix.
void resize(V &v, size_type n)
*/
void mult(const L1 &l1, const L2 &l2, L3 &l3)
*/
void add(const L1 &l1, L2 &l2)
*/
void lu_inverse(const DenseMatrixLU &LU, const Pvector &pvector, const DenseMatrix &AInv_)
Given an LU factored matrix, build the inverse of the matrix.
Common matrix functions for dense matrices.
void logm(const dense_matrix< T > &A, dense_matrix< T > &LOGMA)
Matrix logarithm (from GNU/Octave)
std::shared_ptr< const getfem::virtual_fem > pfem
type of pointer on a fem description
gmm::uint16_type short_type
used as the common short type integer in the library
size_t size_type
used as the common size type in the library
std::shared_ptr< const bgeot::geometric_trans > pgeometric_trans
pointer type for a geometric transformation
size_type alpha(short_type n, short_type d)
Return the value of which is the number of monomials of a polynomial of variables and degree .
GEneric Tool for Finite Element Methods.
size_type APIDECL add_nonlinear_term(model &md, const mesh_im &mim, const std::string &expr, size_type region=size_type(-1), bool is_sym=false, bool is_coercive=false, const std::string &brickname="")
Add a nonlinear term given by the weak form language expression expr which will be assembled in regio...
void ga_define_linear_hardening_function(const std::string &name, scalar_type sigma_y0, scalar_type H, bool frobenius=true)
Add a linear function with the name specified by name to represent linear isotropoc hardening in plas...
void compute_plastic_part(model &md, const mesh_im &mim, const mesh_fem &mf_pl, const std::string &varname, const std::string &previous_dep_name, const pconstraints_projection &ACP, const std::string &datalambda, const std::string &datamu, const std::string &datathreshold, const std::string &datasigma, model_real_plain_vector &plast)
This function computes on mf_pl the plastic part, that could appear after a load and an unload,...
void compute_small_strain_elastoplasticity_Von_Mises(model &md, const mesh_im &mim, std::string lawname, plasticity_unknowns_type unknowns_type, const std::vector< std::string > &varnames, const std::vector< std::string > ¶ms, const mesh_fem &mf_vm, model_real_plain_vector &VM, size_type region=size_type(-1))
This function computes the Von Mises stress field with respect to a small strain elastoplasticity ter...
void finite_strain_elastoplasticity_next_iter(model &md, const mesh_im &mim, std::string lawname, plasticity_unknowns_type unknowns_type, const std::vector< std::string > &varnames, const std::vector< std::string > ¶ms, size_type region=size_type(-1))
This function permits to update the state variables for a finite strain elastoplasticity brick,...
size_type add_finite_strain_elastoplasticity_brick(model &md, const mesh_im &mim, std::string lawname, plasticity_unknowns_type unknowns_type, const std::vector< std::string > &varnames, const std::vector< std::string > ¶ms, size_type region=size_type(-1))
Add a finite strain elastoplasticity brick to the model.
void elastoplasticity_next_iter(model &md, const mesh_im &mim, const std::string &varname, const std::string &previous_dep_name, const pconstraints_projection &ACP, const std::string &datalambda, const std::string &datamu, const std::string &datathreshold, const std::string &datasigma)
This function permits to compute the new stress constraints values supported by the material after a ...
void interpolation(const mesh_fem &mf_source, const mesh_fem &mf_target, const VECTU &U, VECTV &V, int extrapolation=0, double EPS=1E-10, mesh_region rg_source=mesh_region::all_convexes(), mesh_region rg_target=mesh_region::all_convexes())
interpolation/extrapolation of (mf_source, U) on mf_target.
void compute_elastoplasticity_Von_Mises_or_Tresca(model &md, const std::string &datasigma, const mesh_fem &mf_vm, model_real_plain_vector &VM, bool tresca)
This function computes on mf_vm the Von Mises or Tresca stress of a field for elastoplasticity and re...
std::shared_ptr< const virtual_brick > pbrick
type of pointer on a brick
void ga_define_Ramberg_Osgood_hardening_function(const std::string &name, scalar_type sigma_ref, scalar_type eps_ref, scalar_type n, bool frobenius=false)
Add a Ramberg-Osgood hardening function with the name specified by name, for reference stress and str...
size_type add_small_strain_elastoplasticity_brick(model &md, const mesh_im &mim, std::string lawname, plasticity_unknowns_type unknowns_type, const std::vector< std::string > &varnames, const std::vector< std::string > ¶ms, size_type region=size_type(-1))
Adds a small strain plasticity term to the model md.
void ga_local_projection(const getfem::model &md, const mesh_im &mim, const std::string &expr, const mesh_fem &mf, base_vector &result, const mesh_region &rg=mesh_region::all_convexes())
Make an elementwise L2 projection of an expression with respect to the mesh_fem mf.
void compute_finite_strain_elastoplasticity_Von_Mises(model &md, const mesh_im &mim, std::string lawname, plasticity_unknowns_type unknowns_type, const std::vector< std::string > &varnames, const std::vector< std::string > ¶ms, const mesh_fem &mf_vm, model_real_plain_vector &VM, size_type region=size_type(-1))
This function computes the Von Mises stress field with respect to a finite strain elastoplasticity te...
void small_strain_elastoplasticity_next_iter(model &md, const mesh_im &mim, std::string lawname, plasticity_unknowns_type unknowns_type, const std::vector< std::string > &varnames, const std::vector< std::string > ¶ms, size_type region=size_type(-1))
Function that allows to pass from a time step to another for the small strain plastic brick.
size_type add_elastoplasticity_brick(model &md, const mesh_im &mim, const pconstraints_projection &ACP, const std::string &varname, const std::string &previous_dep_name, const std::string &datalambda, const std::string &datamu, const std::string &datathreshold, const std::string &datasigma, size_type region=size_type(-1))
Add a nonlinear elastoplasticity term to the model for small deformations and an isotropic material,...