Herezh_dev/Elements/Mecanique/ElemMeca.cc
Gérard Rio ea11c75ff8 - corr bug affichage intempestif contact
- fct nD: amélioration message d'erreur (arguments), et en fct du niveau géré par une fct nD
- première validation du fct. en // en implicite statique et en RD,
  nombreuses modifications et amélioration !

Signed-off-by: Gérard Rio <gerardrio56@free.fr>
2024-03-24 11:43:58 +01:00

3608 lines
182 KiB
C++

// This file is part of the Herezh++ application.
//
// The finite element software Herezh++ is dedicated to the field
// of mechanics for large transformations of solid structures.
// It is developed by Gérard Rio (APP: IDDN.FR.010.0106078.000.R.P.2006.035.20600)
// INSTITUT DE RECHERCHE DUPUY DE LÔME (IRDL) <https://www.irdl.fr/>.
//
// Herezh++ is distributed under GPL 3 license ou ultérieure.
//
// Copyright (C) 1997-2022 Université Bretagne Sud (France)
// AUTHOR : Gérard Rio
// E-MAIL : gerardrio56@free.fr
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License,
// or (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty
// of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
// See the GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
//
// For more information, please consult: <https://herezh.irdl.fr/>.
//#include "Debug.h"
#include <cmath>
#include "ElemMeca.h"
#include <iomanip>
#include "ConstMath.h"
#include "Util.h"
#include "Coordonnee2.h"
#include "Coordonnee3.h"
#include "ParaAlgoControle.h"
#include "ExceptionsElemMeca.h"
#include "Loi_Umat.h"
#include "ExceptionsLoiComp.h"
#include "CharUtil.h"
// =========================== variables communes ===============================
// a priori les pointeurs de fonction pointent sur rien au debut
const Coordonnee & (Met_abstraite::*(ElemMeca::PointM))
(const Tableau<Noeud *>& tab_noeud,const Vecteur& Phi) ;
void (Met_abstraite::*(ElemMeca::BaseND))
(const Tableau<Noeud *>& tab_noeud,const Mat_pleine& dphi,
const Vecteur& phi,BaseB& bB,BaseH& bH);
int ElemMeca::bulk_viscosity = 0; // indique si oui ou non on utilise le bulk viscosity
double ElemMeca::c_traceBulk = 0.; // coeff de la trace de D pour le bulk
double ElemMeca::c_carreBulk = 0.; // coeff du carré de la trace de D pour le bulk
TenseurHH* ElemMeca::sig_bulkHH = NULL; // variable de travail pour le bulk
// ---------- pour le calcul de la masse pour la méthode de relaxation dynamique
Ddl_enum_etendu ElemMeca::masse_relax_dyn; // au début rien, def à la première utilisation
// pour la gestion d'hourglass: dimensionnement dans ElemMeca::Init_hourglass_comp
Tableau< Vecteur> ElemMeca::vec_hourglass; // tableau de vecteurs de travail, éventuellement défini
// pour la stabilisation de membrane: dimensionnement lors de l'utilisation
//Tableau< Vecteur> ElemMeca::vec_StabMembBiel; // tableau de vecteurs de travail, éventuellement défini
// =========================== fin variables communes ===========================
// -------------------- stabilisation transversale éventuelle de membrane ou de biel -------------------
// utilisée et alimentées par les classes dérivées: ElemMeca sert de conteneur ce qui permet
// d'éviter de redéfinir des conteneurs locaux aux éléments membranes et biel
// par défaut
ElemMeca::StabMembBiel::StabMembBiel() :
type_stabMembrane(STABMEMBRANE_BIEL_NON_DEFINIE)
,gamma(0.),pt_fct_gamma(NULL),stab_ref(1.),activite_en_explicite(true)
,F_StabMembBiel(),F_StabMembBiel_t()
,E_StabMembBiel(),E_StabMembBiel_t()
,nom_fctnD(NULL),nom_fctnD_2(NULL)
,affichage_stabilisation(0),gestion_maxi_mini(false)
,beta(1.) // valeur arbitraire pour l'instant
,f_mini(ConstMath::petit) // valeur arbitraire pour l'instant
,d_maxi(1000.) // déplacement arbitraire de 1000
,pt_fct_ponder_Ft(NULL)
{};
// fonction du nombre de noeud ou de point d'interpolation
ElemMeca::StabMembBiel::StabMembBiel(int nbnoe) :
type_stabMembrane(STABMEMBRANE_BIEL_NON_DEFINIE)
,gamma(0.),pt_fct_gamma(NULL),stab_ref(1.),activite_en_explicite(true)
,F_StabMembBiel(nbnoe),F_StabMembBiel_t(nbnoe)
,E_StabMembBiel(nbnoe),E_StabMembBiel_t(nbnoe)
,nom_fctnD(NULL),nom_fctnD_2(NULL)
,affichage_stabilisation(0),gestion_maxi_mini(false)
,beta(1.) // valeur arbitraire pour l'instant
,f_mini(ConstMath::petit) // valeur arbitraire pour l'instant
,d_maxi(1000.) // déplacement arbitraire de 100
,pt_fct_ponder_Ft(NULL)
{};
// fonction d'une valeur numérique et des noms éventuels de fonction
ElemMeca::StabMembBiel::StabMembBiel(double v,string * sgamma,string* sponder)
: type_stabMembrane(STABMEMBRANE_BIEL_NON_DEFINIE)
,gamma(v),pt_fct_gamma(NULL),F_StabMembBiel()
,E_StabMembBiel(),E_StabMembBiel_t()
,nom_fctnD(NULL),nom_fctnD_2(NULL)
,stab_ref(1.),activite_en_explicite(true)
,affichage_stabilisation(0),gestion_maxi_mini(false)
,beta(1.) // valeur arbitraire pour l'instant
,f_mini(ConstMath::petit) // valeur arbitraire pour l'instant
,d_maxi(1000.) // déplacement arbitraire de 100
,pt_fct_ponder_Ft(NULL)
{if (sgamma != NULL) nom_fctnD= new string (*sgamma);
if (sponder != NULL) nom_fctnD_2= new string (*sponder);
};
// de copie
ElemMeca::StabMembBiel::StabMembBiel(const StabMembBiel& a):
type_stabMembrane(a.type_stabMembrane)
,gamma(a.gamma),pt_fct_gamma(a.pt_fct_gamma),stab_ref(a.stab_ref)
,activite_en_explicite(a.activite_en_explicite)
,F_StabMembBiel(a.F_StabMembBiel),F_StabMembBiel_t(a.F_StabMembBiel_t)
,E_StabMembBiel(a.E_StabMembBiel),E_StabMembBiel_t(a.E_StabMembBiel_t)
,nom_fctnD(NULL),nom_fctnD_2(NULL)
,affichage_stabilisation(a.affichage_stabilisation)
,gestion_maxi_mini(a.gestion_maxi_mini)
,beta(a.beta)
,f_mini(a.f_mini)
,d_maxi(a.d_maxi),pt_fct_ponder_Ft(a.pt_fct_ponder_Ft)
{if (a.nom_fctnD != NULL) nom_fctnD = new string (*(a.nom_fctnD));
if (a.nom_fctnD_2 != NULL) nom_fctnD_2 = new string (*(a.nom_fctnD_2));
};
ElemMeca::StabMembBiel::~StabMembBiel()
{pt_fct_gamma=NULL; // la fonction est forcément globale, donc on ne la détruie pas
if (nom_fctnD != NULL) delete nom_fctnD;
// idem pour la pondération
if (nom_fctnD_2 != NULL) delete nom_fctnD_2;
};
ElemMeca::StabMembBiel& ElemMeca::StabMembBiel::operator= (const ElemMeca::StabMembBiel& a)
{type_stabMembrane = a.type_stabMembrane;
gamma=a.gamma;pt_fct_gamma=a.pt_fct_gamma;
stab_ref=a.stab_ref;activite_en_explicite = a.activite_en_explicite;
F_StabMembBiel=a.F_StabMembBiel;F_StabMembBiel_t=a.F_StabMembBiel_t;
E_StabMembBiel=a.E_StabMembBiel;E_StabMembBiel_t = a.E_StabMembBiel_t;
nom_fctnD=NULL;pt_fct_ponder_Ft=a.pt_fct_ponder_Ft;
affichage_stabilisation = a.affichage_stabilisation;
gestion_maxi_mini=a.gestion_maxi_mini;
beta=a.beta;f_mini=a.f_mini;d_maxi=a.d_maxi;
if (a.nom_fctnD != NULL) nom_fctnD = new string (*(a.nom_fctnD));
if (a.nom_fctnD_2 != NULL) nom_fctnD_2 = new string (*(a.nom_fctnD_2));
return (*this);
};
// énergie totale à t+dt développée sur l'élément pour la stabilisation
double ElemMeca::StabMembBiel::EE_total_StabMembBiel() const
{ double retour = 0; // init
int taille = E_StabMembBiel.Taille();
for (int i=1; i<=taille;i++)
retour += E_StabMembBiel(i);
return retour;
};
// énergie totale à t développée sur l'élément pour la stabilisation
double ElemMeca::StabMembBiel::EE_total_StabMembBiel_t() const
{ double retour = 0; // init
int taille = E_StabMembBiel_t.Taille();
for (int i=1; i<=taille;i++)
retour += E_StabMembBiel_t(i);
return retour;
};
// actualisation de la force de stabilisation de t+dt vers t
void ElemMeca::StabMembBiel::TdtversT()
{F_StabMembBiel_t = F_StabMembBiel;
E_StabMembBiel_t = E_StabMembBiel;
};
// actualisation de la force de stabilisation de t vers tdt
void ElemMeca::StabMembBiel::TversTdt()
{F_StabMembBiel = F_StabMembBiel_t;
E_StabMembBiel = E_StabMembBiel_t;
};
// cas donne le niveau de la récupération
// = 1 : on récupère tout
// = 2 : on récupère uniquement les données variables (supposées comme telles)
void ElemMeca::StabMembBiel::Lecture_bas_inf(ifstream& ent,const int cas)
{string toto,nom;
switch (cas)
{ case 1 : // ------- on récupère tout -------------------------
{ // récup du type de stabilisation
ent >> toto;
type_stabMembrane = Id_Enum_StabMembraneBiel(toto);
// récup pour alpha
ent >> toto;
if (toto == "par_fonction_nD")
{ ent >> nom;
if (nom_fctnD == NULL)
nom_fctnD = new string(nom);
else *nom_fctnD = nom;
gamma = 0.; // init au cas ou
pt_fct_gamma = NULL; // même si diff de NULL
// il n'y a pas delete à faire car il s'agit de fonction globale
// ensuite il faudra définir la fonction de stabilisation
// au moment de compléter l'élément
}
else if (toto == "par_valeur")
{ent >> gamma;}
else
{ cout << "\n *** erreur en lecture, on attendait"
<< " par_fonction_nD ou par_valeur et on a lue "
<< toto ;
Sortie(1);
};
// activite_en_explicite
ent >> toto >> activite_en_explicite;
// pondération éventuelle
ent >> toto;
if (toto == "ponderation_F_t")
{ ent >> nom;
if (nom_fctnD_2 == NULL)
nom_fctnD_2 = new string(nom);
else *nom_fctnD_2 = nom;
pt_fct_ponder_Ft = NULL;// même si diff de NULL
// il n'y a pas delete à faire car il s'agit de fonction globale
// ensuite il faudra définir la fonction de pondération
// au moment de compléter l'élément
};
// affichage et les limitations
ent >> toto >> affichage_stabilisation
>> toto >> gestion_maxi_mini
>> toto >> beta >> toto >> f_mini >> toto >> d_maxi;
break;
}
case 2 : // ----------- lecture uniquement de se qui varie --------------------
{ ent >> toto >> F_StabMembBiel;
ent >> toto >> E_StabMembBiel;
break;
}
default :
{ cout << "\nErreur : valeur incorrecte du type de lecture !\n";
cout << "ElemMeca::StabMembBiel::Lecture_bas_inf(ifstream& ent,const int cas)"
<< " cas= " << cas << endl;
Sortie(1);
};
};
};
// cas donne le niveau de sauvegarde
// = 1 : on sauvegarde tout
// = 2 : on sauvegarde uniquement les données variables (supposées comme telles)
void ElemMeca::StabMembBiel::Ecriture_bas_inf(ofstream& sort,const int cas)
{switch (cas)
{ case 1 : // ------- on sauvegarde tout -------------------------
{ sort << " "<<Nom_StabMembraneBiel(type_stabMembrane);
if (pt_fct_gamma != NULL)
{sort << " par_fonction_nD " << pt_fct_gamma->NomFonction()<<" ";}
else {sort << " par_valeur " << gamma<< " ";};
sort << " activite_en_explicite "<< activite_en_explicite;
if (pt_fct_ponder_Ft != NULL)
{sort << " ponderation_F_t " << pt_fct_ponder_Ft->NomFonction()<<" ";}
else {sort << " pas_de_ponderation_F_t " ;};
sort << "\n affichage_stabilisation= "<<affichage_stabilisation
<< " gestion_maxi_mini= "<< gestion_maxi_mini
<< " beta= "<< beta << " f_mini= "
<< f_mini << " d_maxi= " << d_maxi << " ";
break;
}
case 2 : // ----------- sauvegarde uniquement de se qui varie --------------------
{ sort << " F_StabMembBiel "<< F_StabMembBiel;
sort << " E_StabMembBiel "<< E_StabMembBiel
<< " ";
break;
}
default :
{ cout << "\nErreur : valeur incorrecte du type d'écriture !\n";
cout << "ElemMeca::StabMembBiel::Ecriture_bas_inf(ofstream& sort,const int cas)"
<< " cas= " << cas << endl;
Sortie(1);
};
};
};
// complète en attribuant les fct nD si besoin, ceci après une lecture
// initiale base_info
void ElemMeca::StabMembBiel::Complete_StabMembBiel(LesFonctions_nD* lesFonctionsnD)
{ // on complète si les pointeurs de fonction sont nulle et qu'il y a des noms de fonction
if (nom_fctnD != NULL)
if (pt_fct_gamma == NULL)
pt_fct_gamma = lesFonctionsnD->Trouve(*nom_fctnD);
if (nom_fctnD_2 != NULL)
if (pt_fct_ponder_Ft == NULL)
pt_fct_ponder_Ft = lesFonctionsnD->Trouve(*nom_fctnD_2);
};
// changement du nombre de noeuds ou de pti
void ElemMeca::StabMembBiel::Change_nb_pti(int nbnoe)
{F_StabMembBiel.Change_taille(nbnoe);
F_StabMembBiel_t.Change_taille(nbnoe);
E_StabMembBiel.Change_taille(nbnoe);
E_StabMembBiel_t.Change_taille(nbnoe);
};
// -------------------- fin conteneur stabilisation transversale éventuelle de membrane ou de biel -------------------
// Constructeur par defaut
ElemMeca::ElemMeca () :
Element(),tabSaveDon(),tabSaveTP(),tabSaveDefDon(),defSurf(),defArete()
,masse_volumique(masse_volumique_defaut),dilatation(0),tab_energ(),tab_energ_t()
,energie_totale(),energie_totale_t()
,E_elem_bulk_t(0.),E_elem_bulk_tdt(0.),P_elem_bulk(0.),E_Hourglass(0.)
,premier_calcul_meca_impli_expli(true),lesPtIntegMecaInterne(NULL)
,lesChargeExtSurEle(NULL)
// hourglass
,type_stabHourglass(STABHOURGLASS_NON_DEFINIE),tab_elHourglass()
,coefStabHourglass(0.),raid_hourglass_transitoire(NULL)
// stab transversale
,pt_StabMembBiel(NULL),matD(NULL),resD(NULL),maxi_F_t(0.)
{ loiComp = NULL;loiTP=NULL;loiFrot=NULL;
met = NULL;
def = NULL;
defEr = NULL;
defMas = NULL;
sigErreur = NULL;
sigErreur_relative = NULL;
};
// Constructeur utile quand le numero d'identification de l'element est connu
ElemMeca::ElemMeca (int num_mail,int num_id) :
Element(num_mail,num_id),tabSaveDon(),tabSaveTP(),tabSaveDefDon(),defSurf(),defArete()
,masse_volumique(masse_volumique_defaut),dilatation(0),tab_energ(),tab_energ_t()
,energie_totale(),energie_totale_t()
,E_elem_bulk_t(0.),E_elem_bulk_tdt(0.),P_elem_bulk(0.),E_Hourglass(0.)
,premier_calcul_meca_impli_expli(true)
,lesPtIntegMecaInterne(NULL)
,lesChargeExtSurEle(NULL)
// hourglass
,type_stabHourglass(STABHOURGLASS_NON_DEFINIE),tab_elHourglass()
,coefStabHourglass(0.),raid_hourglass_transitoire(NULL)
// stab transversale
,pt_StabMembBiel(NULL),matD(NULL),resD(NULL),maxi_F_t(0.)
{ loiComp = NULL;loiTP=NULL;loiFrot=NULL;
met = NULL;
def = NULL;
defEr = NULL;
defMas = NULL;
sigErreur = NULL; sigErreur_relative= NULL;
};
// Constructeur utile quand le numero et le tableau des noeuds
// de l'element sont connu
ElemMeca::ElemMeca (int num_mail,int num_id,const Tableau<Noeud *>& tab):
Element(num_mail,num_id,tab)
,tabSaveDon(),tabSaveTP(),tabSaveDefDon(),defSurf(),defArete()
,masse_volumique(masse_volumique_defaut),dilatation(0),tab_energ(),tab_energ_t()
,energie_totale(),energie_totale_t()
,E_elem_bulk_t(0.),E_elem_bulk_tdt(0.),P_elem_bulk(0.),E_Hourglass(0.)
,premier_calcul_meca_impli_expli(true)
,lesPtIntegMecaInterne(NULL)
,lesChargeExtSurEle(NULL)
// hourglass
,type_stabHourglass(STABHOURGLASS_NON_DEFINIE),tab_elHourglass()
,coefStabHourglass(0.),raid_hourglass_transitoire(NULL)
// stab transversale
,pt_StabMembBiel(NULL),matD(NULL),resD(NULL),maxi_F_t(0.)
{ loiComp = NULL;loiTP=NULL;loiFrot=NULL;
met = NULL;
def = NULL;
defEr = NULL;
defMas = NULL;
sigErreur = NULL; sigErreur_relative = NULL;
};
// Constructeur utile quand le numero d'identification est connu,
// ainsi que la geometrie et le type d'interpolation de l'element
ElemMeca::ElemMeca (int num_mail,int num_id,Enum_interpol id_interp_elt,Enum_geom id_geom_elt,string info):
Element (num_mail,num_id,id_interp_elt,id_geom_elt,MECA_SOLIDE_DEFORMABLE,info)
,tabSaveDon(),tabSaveTP(),tabSaveDefDon()
,defSurf(),defArete(),masse_volumique(masse_volumique_defaut),dilatation(0)
,tab_energ(),tab_energ_t(),energie_totale(),energie_totale_t()
,E_elem_bulk_t(0.),E_elem_bulk_tdt(0.),P_elem_bulk(0.),E_Hourglass(0.)
,premier_calcul_meca_impli_expli(true)
,lesPtIntegMecaInterne(NULL)
,lesChargeExtSurEle(NULL)
,type_stabHourglass(STABHOURGLASS_NON_DEFINIE),tab_elHourglass()
,coefStabHourglass(0.),raid_hourglass_transitoire(NULL)
//--- stab transversale
,pt_StabMembBiel(NULL),matD(NULL),resD(NULL),maxi_F_t(0.)
{ loiComp = NULL;loiTP=NULL;loiFrot=NULL;
met = NULL;
def = NULL;
defEr = NULL;
defMas = NULL;
sigErreur = NULL; sigErreur_relative = NULL;
};
ElemMeca::ElemMeca (int num_mail,int num_id,char* nom_interpol,char* nom_geom,string info):
// Constructeur utile quand le numero d'identification est connu,
// ainsi que la geometrie et le type d'interpolation de l'element
Element(num_mail,num_id,nom_interpol,nom_geom,NomElemTypeProblem(MECA_SOLIDE_DEFORMABLE),info)
,tabSaveDon()
,tabSaveTP(),tabSaveDefDon(),defSurf(),defArete(),masse_volumique(masse_volumique_defaut),dilatation(0)
,tab_energ(),tab_energ_t(),energie_totale(),energie_totale_t(),premier_calcul_meca_impli_expli(true)
,E_elem_bulk_t(0.),E_elem_bulk_tdt(0.),P_elem_bulk(0.),E_Hourglass(0.)
,lesPtIntegMecaInterne(NULL)
,lesChargeExtSurEle(NULL)
,type_stabHourglass(STABHOURGLASS_NON_DEFINIE),tab_elHourglass()
,coefStabHourglass(0.),raid_hourglass_transitoire(NULL)
//--- stab transversale
,pt_StabMembBiel(NULL),matD(NULL),resD(NULL),maxi_F_t(0.)
{ loiComp = NULL;loiTP=NULL;loiFrot=NULL;
met = NULL;
def = NULL;
defEr = NULL;
defMas = NULL;
sigErreur = NULL; sigErreur_relative = NULL;
};
ElemMeca::ElemMeca (int num_mail,int num_id,const Tableau<Noeud *>& tab,Enum_interpol id_interp_elt,
Enum_geom id_geom_elt,string info):
// Constructeur utile quand toutes les donnees sont connues
Element (num_mail,num_id,tab,id_interp_elt,id_geom_elt,MECA_SOLIDE_DEFORMABLE,info)
,tabSaveDon(),tabSaveTP()
,tabSaveDefDon(),defSurf(),defArete(),masse_volumique(masse_volumique_defaut),dilatation(0)
,tab_energ(),tab_energ_t(),energie_totale(),energie_totale_t(),premier_calcul_meca_impli_expli(true)
,E_elem_bulk_t(0.),E_elem_bulk_tdt(0.),P_elem_bulk(0.),E_Hourglass(0.)
,lesPtIntegMecaInterne(NULL)
,lesChargeExtSurEle(NULL)
,type_stabHourglass(STABHOURGLASS_NON_DEFINIE),tab_elHourglass()
,coefStabHourglass(0.),raid_hourglass_transitoire(NULL)
//--- stab transversale
,pt_StabMembBiel(NULL),matD(NULL),resD(NULL),maxi_F_t(0.)
{ loiComp = NULL;loiTP=NULL;loiFrot=NULL;
met = NULL;
def = NULL;
defEr = NULL;
defMas = NULL;
sigErreur = NULL; sigErreur_relative = NULL;
};
// Constructeur utile quand toutes les donnees sont connues
ElemMeca::ElemMeca (int num_mail,int num_id,const Tableau<Noeud *>& tab,char* nom_interpol,
char* nom_geom,string info):
Element (num_mail,num_id,tab,nom_interpol,nom_geom,NomElemTypeProblem(MECA_SOLIDE_DEFORMABLE),info)
,tabSaveDon()
,tabSaveTP(),tabSaveDefDon(),defSurf(),defArete(),masse_volumique(masse_volumique_defaut),dilatation(0)
,tab_energ(),tab_energ_t(),energie_totale(),energie_totale_t(),premier_calcul_meca_impli_expli(true)
,E_elem_bulk_t(0.),E_elem_bulk_tdt(0.),P_elem_bulk(0.),E_Hourglass(0.)
,lesPtIntegMecaInterne(NULL)
,lesChargeExtSurEle(NULL)
,type_stabHourglass(STABHOURGLASS_NON_DEFINIE),tab_elHourglass()
,coefStabHourglass(0.),raid_hourglass_transitoire(NULL)
//--- stab transversale
,pt_StabMembBiel(NULL),matD(NULL),resD(NULL),maxi_F_t(0.)
{ loiComp = NULL;loiTP=NULL;loiFrot=NULL;
met = NULL;
def = NULL;
defEr = NULL;
defMas = NULL;
sigErreur = NULL; sigErreur_relative = NULL;
};
// Constructeur de copie
ElemMeca::ElemMeca (const ElemMeca& elt):
Element (elt),defSurf(elt.defSurf),defArete(elt.defArete)
,masse_volumique(elt.masse_volumique),dilatation(elt.dilatation)
,tab_energ(elt.tab_energ),tab_energ_t(elt.tab_energ_t)
,energie_totale(elt.energie_totale),energie_totale_t(elt.energie_totale_t)
,E_elem_bulk_t(elt.E_elem_bulk_t),E_elem_bulk_tdt(elt.E_elem_bulk_tdt),P_elem_bulk(elt.P_elem_bulk)
//--- hourglass
,E_Hourglass(elt.E_Hourglass)
//--- stab transversale
,pt_StabMembBiel(NULL),matD(elt.matD),resD(elt.resD),maxi_F_t(0.)
,premier_calcul_meca_impli_expli(true)
,lesPtIntegMecaInterne(NULL)
,lesChargeExtSurEle(NULL)
,type_stabHourglass(elt.type_stabHourglass),tab_elHourglass()
,coefStabHourglass(elt.coefStabHourglass),raid_hourglass_transitoire(NULL)
{ // pour la loi on pointe sur la même que l'élément de référence, car de toute
// la loi ne contient pas les données spécifique
loiComp = (elt.loiComp);loiTP = (elt.loiTP);loiFrot = (elt.loiFrot);
// idem pour la métrique qui est a priori une métrique générique
met = (elt.met);
// la déformation est spécifique à l'élément
if (elt.def!=NULL) def = elt.def->Nevez_deformation(tab_noeud); // *def = *(elt.def);
if (elt.defEr!=NULL) defEr = elt.defEr->Nevez_deformation(tab_noeud); //*defEr = *(elt.defEr);
if (elt.defMas!=NULL) defMas = elt.defMas->Nevez_deformation(tab_noeud); //*defMas = *(elt.defMas);
// les données spécifiques à la déformation mécanique
int idefdtr = (elt.tabSaveDefDon).Taille();
tabSaveDefDon.Change_taille(idefdtr);
for (int i=1;i<= idefdtr; i++)
{ tabSaveDefDon(i) = def->New_et_Initialise();
if (tabSaveDefDon(i)!= NULL) *tabSaveDefDon(i) = *(elt.tabSaveDefDon)(i);
};
// def: cas des arrêtes et des faces si besoin
int tailledefSurf = defSurf.Taille();
int itab = 1; // indice pour parcourir tabb
for (int idsu =1;idsu<=tailledefSurf;idsu++,itab++)
{if ((defSurf(idsu) != NULL) && (tabb.Taille() != 0)) // dans le cas des coques
// on peut avoir defSurf qui est défini, mais pas la frontière donc dans ce cas
// on ne peut pas utiliser cette copie
{ defSurf(idsu) = elt.defSurf(idsu)->Nevez_deformation(tabb(itab)->TabNoeud());}
else // sinon on supprime, il sera reconstruit si besoin est
{ // dans le cas où on a qu'une seule surface a priori et que tabb n'est pas définit
// on considère que le tableau de noeud associé est celui de l'élément
delete(defSurf(idsu));defSurf(idsu)=NULL;
};
};
int tailledefArete = defArete.Taille();
for (int idA =1;idA<=tailledefArete;idA++)
{if ((defArete(idA) != NULL) && (tabb.Taille() != 0))
defArete(idA) = elt.defArete(idA)->Nevez_deformation(tabb(itab)->TabNoeud());
else {delete(defArete(idA));defArete(idA)=NULL; }
};
// les donnees de la loi de comportement mécanique
int dtr = (elt.tabSaveDon).Taille();
tabSaveDon.Change_taille(dtr);
for (int i=1;i<= dtr; i++)
{ tabSaveDon(i) = loiComp->New_et_Initialise();
if (tabSaveDon(i)!= NULL) *tabSaveDon(i) = *(elt.tabSaveDon)(i);
};
// les donnees de la loi de comportement thermo physique
int dtrTP = (elt.tabSaveTP).Taille();
tabSaveTP.Change_taille(dtrTP);
for (int i=1;i<= dtrTP; i++)
{ tabSaveTP(i) = NULL; // init
if (loiTP != NULL ) tabSaveTP(i) = loiTP->New_et_Initialise();
if (tabSaveTP(i)!= NULL) *tabSaveTP(i) = *(elt.tabSaveTP)(i);
};
// et autre
if (elt.sigErreur != NULL)
{ sigErreur = new double;
(*sigErreur) = *(elt.sigErreur);
sigErreur_relative = new double;
*sigErreur_relative = *(elt.sigErreur_relative);
}
else
{sigErreur = sigErreur_relative = NULL;};
//---- stabilisation transversale éventuelle
if (elt.pt_StabMembBiel != NULL)
pt_StabMembBiel = new StabMembBiel(*(elt.pt_StabMembBiel));
};
// Destructeur
ElemMeca::~ElemMeca ()
{ // les donnees pour la loi de comportement mécanique
int imaxi = tabSaveDon.Taille();
for (int i=1;i<=imaxi; i++)
if (tabSaveDon(i) != NULL) {delete tabSaveDon(i) ; tabSaveDon(i)=NULL;}
// les donnees pour la loi de comportement thermo physique
int imaxTP = tabSaveTP.Taille();
for (int i=1;i<=imaxTP; i++)
if (tabSaveTP(i) != NULL) {delete tabSaveTP(i) ; tabSaveTP(i)=NULL;}
// les donnees à la déformation mécanique
int idefmaxi = tabSaveDefDon.Taille();
for (int i=1;i<=idefmaxi; i++)
if (tabSaveDefDon(i) != NULL) {delete tabSaveDefDon(i) ; tabSaveDefDon(i)=NULL;}
// les déformations sont spécifiquent aux elements, on les detruits si necessaire
if (def != NULL) { delete def; def = NULL;}
if (defEr != NULL) {delete defEr; defEr = NULL;}
if (defMas != NULL) {delete defMas; defMas = NULL;}
int nbdefSurf = defSurf.Taille();
for (int i = 1; i<= nbdefSurf; i++)
if (defSurf(i) != NULL) {delete defSurf(i); defSurf(i) = NULL;}
int nbdefArete = defArete.Taille();
for (int i = 1; i<= nbdefArete; i++)
if (defArete(i) != NULL) {delete defArete(i);defArete(i) = NULL;}
// par contre la metrique et la loi de comportement sont communes a une classe d'element
// donc ici met et loicomp ne sont que des pointeurs dont il ne faut pas supprimer les objets
// pointes
if (sigErreur!= NULL) {delete sigErreur; sigErreur = NULL;
delete sigErreur_relative; sigErreur_relative=NULL;
};
// --- cas des éléments de stabilisation d'hourglass ---
if (tab_elHourglass.Taille() != 0)
{ int taillhour = tab_elHourglass.Taille();
for (int i=1;i<= taillhour;i++)
{ if (tab_elHourglass(i) != NULL) delete tab_elHourglass(i);};
};
if (raid_hourglass_transitoire != NULL) delete raid_hourglass_transitoire;
// --- cas de la stabilisation transversale ---
if (pt_StabMembBiel != NULL)
delete pt_StabMembBiel;
//---- suppression du dimensionnement des charges externes éventuellement ----
if (lesChargeExtSurEle != NULL) delete lesChargeExtSurEle;
};
// =========================== methodes publiques ===============================
// calcul si un point est a l'interieur de l'element ou non
// il faut que M est la dimension globale
// les trois fonctions sont pour l'etude a t=0, t et tdt
// retour : =0 le point est externe, =1 le point est interne ,
// = 2 le point est sur la frontière à la précision près
// coor_locales : s'il est différent de NULL, est affecté des coordonnées locales calculées,
// uniquement précises si le point est interne
// dans le cas où l'on veut un fonctionnement différent, c'est surchargé dans les
// éléments fils
int ElemMeca::Interne_0(const Coordonnee& M,Coordonnee* coor_locales)
{ PointM = &Met_abstraite::PointM_0;
BaseND = &Met_abstraite::BaseND_0;
return Interne(TEMPS_0,M,coor_locales);
};
int ElemMeca::Interne_t(const Coordonnee& M,Coordonnee* coor_locales)
{ PointM = &Met_abstraite::PointM_t;
BaseND = &Met_abstraite::BaseND_t;
return Interne(TEMPS_t,M,coor_locales);
};
int ElemMeca::Interne_tdt(const Coordonnee& M,Coordonnee* coor_locales)
{ PointM = &Met_abstraite::PointM_tdt;
BaseND = &Met_abstraite::BaseND_tdt;
return Interne(TEMPS_tdt,M,coor_locales);
};
// test si l'element est complet
int ElemMeca::TestComplet()
{ int ret = Element::TestComplet(); // teste de la class mere
if (loiComp == NULL)
{ cout << " \n la loi n'est pas defini dans l element " << num_elt << '\n';
ret = 0;
}
else if (loiComp->TestComplet() != 1)
{ cout << " \n la loi n'est pas complete dans l element " << num_elt << '\n';
ret = 0;
}
// dans le cas d'une loi thermo physique on vérifie qu'elle est complète
if (loiTP != NULL)
{ if (loiTP->TestComplet() != 1)
{ cout << " \n la loi thermo physique n'est pas complete dans l element " << num_elt << '\n';
ret = 0;
}
};
// dans le cas où l'on prend en compte de la dilatation, on vérifie qu'une loi thermo plysique
// a été défini
if (dilatation)
{if (loiTP == NULL)
{cout << " \n L element " << num_elt
<< " prend en compte la dilatation, alors qu'aucune loi thermo physique n'est defini, "
<< " il faut donc en definir une ! \n ";
ret = 0;
}
}
// dans le cas d'une loi de frottement on vérifie qu'elle est complète
if (loiFrot != NULL)
{ if (loiFrot->TestComplet() != 1)
{ cout << " \n la loi de frottement n'est pas complete dans l element " << num_elt << '\n';
ret = 0;
}
};
// suite des datas
if ( masse_volumique == masse_volumique_defaut)
{ cout << "\n la masse volumique n'est pas defini dans l element " << num_elt << '\n';
ret = 0;
}
if (met == NULL)
{ cout << " \n la metrique n'est pas defini dans l element " << num_elt << '\n';
ret = 0;
}
if (def == NULL)
{ cout << " \n le type de deformation n'est pas defini dans l element "
<< num_elt << '\n';
ret = 0;
}
return ret;
};
// test pour savoir si le calcul de contrainte en absolu est possible
bool ElemMeca::ContrainteAbsoluePossible()
{ // il faut qu'il soit complet et il faut que les coordonnées a t =0 et t existe
bool retour = true;
if (TestComplet() !=0)
{// on regarde la définition des coordonnées
int nbmaxnoeud=tab_noeud.Taille();
for (int i=1;i<=nbmaxnoeud;i++)
if((!tab_noeud(i)->ExisteCoord0())||(!tab_noeud(i)->ExisteCoord1()))
{ retour = false; break;}
}
else
retour = false;
return retour;
};
// --------- calculs utils dans le cadre de la recherche du flambement linéaire
// dans un premier temps uniquement virtuelles, ensuite se sera virtuelle pure pour éviter
// les oubli de définition ----> AMODIFIER !!!!!!!!
// Calcul de la matrice géométrique et initiale
ElemMeca::MatGeomInit ElemMeca::MatricesGeometrique_Et_Initiale (const ParaAlgoControle & )
{ cout << "\n attention le calcul de la matrice g\'eom\'etrique et de la matrice "
<< " initiale ne sont pas implant\'ees \n" ;
this->Affiche(1);
cout << "\nvoid ElemMeca::MatricesGeometrique_Et_Initiale (Mat_pleine &,Mat_pleine &)"
<< endl;
Sortie(1);
return MatGeomInit(NULL,NULL); // pour supprimer le warning
};
// calcul de l'erreur sur l'élément. Ce calcul n'est disponible
// qu'une fois la remontée aux contraintes effectuées sinon aucune
// action. En retour la valeur de l'erreur sur l'élément
// = 1 : erreur = (int (delta sigma):(delta sigma) dv)/(int sigma:sigma dv)
// le numerateur et le denominateur sont tel que :
// errElemRelative = numerateur / denominateur , si denominateur different de 0
// sinon denominateur = numerateur si numerateur est different de 0, sinon
// tous sont nuls mais on n'effectue pas la division
//!!!!!!!!!!!!! pour l'instant en virtuelle il faudra après en
// virtuelle pure !!!!!!!!!!!!!!!!!!!!!!!!!!! donc le code est à virer
void ElemMeca::ErreurElement(int ,double& ,double& , double& )
{ cout << "\n attention le calcul de l'erreur sur l'element n'est pas implant\'e "
<< " pour cet element \n" ;
this->Affiche(1);
cout << "\nElemMeca::ErreurElement(int....."
<< endl;
Sortie(1);
};
// ajout des ddl d'erreur pour les noeuds de l'élément
void ElemMeca::Plus_ddl_Erreur()
{int nbne = tab_noeud.Taille();
for (int ne=1; ne<= nbne; ne++) // pour chaque noeud
{ Ddl ddl(ERREUR,0.,LIBRE);
tab_noeud(ne)->PlusDdl(ddl);
}
};
// inactive les ddl d'erreur
void ElemMeca::Inactive_ddl_Erreur()
{ int nbne = tab_noeud.Taille();
for (int ne=1; ne<= nbne; ne++)
tab_noeud(ne)->Met_hors_service(ERREUR);
};
// active les ddl d'erreur
void ElemMeca::Active_ddl_Erreur()
{ int nbne = tab_noeud.Taille();
for (int ne=1; ne<= nbne; ne++)
tab_noeud(ne)->Met_en_service(ERREUR);
};
// retourne un tableau de ddl element, correspondant à la
// composante d'erreur -> ERREUR, pour chaque noeud
// -> utilisé pour l'assemblage de la raideur d'erreur
//dans cette version tous les noeuds sont supposés avoi un ddl erreur
// dans le cas contraire il faut redéfinir la fonction dans l'élément terminal
DdlElement ElemMeca::Tableau_de_ERREUR() const
{ return DdlElement(tab_noeud.Taille(), DdlNoeudElement(ERREUR) );
};
// =========================== methodes protegees ===============================
// calcul si un point est a l'interieur de l'element ou non
// il faut que M est la dimension globale
// retour : = 0 le point est externe, = 1 le point est interne ,
// = 2 le point est sur la frontière à la précision près
// coor_locales : s'il est différent de NULL, est affecté des coordonnées locales calculées,
int ElemMeca::Interne(Enum_dure temps,const Coordonnee& M,Coordonnee* coor_locales)
{ int dim_glob = M.Dimension(); // dim de M = dim de l'espace géométrique
ElemGeomC0& elgeom = this->ElementGeometrique(); // la geometrie
int dim_loc = elgeom.Dimension(); // dim de l'élément de référence
#ifdef MISE_AU_POINT
if (ParaGlob::Dimension() != dim_glob)
{ cout << "\n *** erreur, la dimension du point = (" << dim_glob
<< ") et de l'espace geometrique (" << ParaGlob::Dimension() << ") sont differentes! ";
cout << "\nElemMeca::Interne(Coordonnee& M) " << endl;
this->Affiche();
Sortie(1);
};
#endif
////----- debug
//if ((Num_elt() == 3) && (Num_maillage() == 2))
// {cout << "\n debug ElemMeca::Interne(...";
// Tableau<Noeud *>& tan = Tab_noeud();
//
// for (int i = 1;i<= tan.Taille();i++)
// {cout << " noeud ("<<i<<")";
// tan(i)->Coord2().Affiche();
// };
// };
////---- fin debug
// la methode consiste tout d'abord a calculer les coordonnees locales correspondants
// a M. Pour cela on utilise un processus iteratif
// ---- 1 :init de la recherche
const Coordonnee refL(dim_loc) ; // coordonnees locales du point de reference, initialisees a zero
// - construction du point local en fonction de l'existance ou non de coor_locales
Coordonnee* pt_ret=NULL; Coordonnee theta_local(refL);
if (coor_locales != NULL) {pt_ret = coor_locales;} else {pt_ret = &theta_local;};
Coordonnee& theta = *pt_ret; // init de AL qui representera les coordonnes locales de M
// - fin construction du point local
Vecteur ph_i = elgeom.Phi_point(refL); // fonctions d'interpolation en refL
Mat_pleine dph_i = elgeom.Dphi_point(refL); // derivees des fonctions d'interpolation en refL
// l'image de theta par l'interpolation, de dimension : espace géométrique
Coordonnee A = (met->*PointM) (tab_noeud,ph_i); //
// calcul de la base naturelle et duale en refL, en fonction des coord a t
BaseB bB(dim_glob,dim_loc);BaseH bH(dim_glob,dim_loc);
(met->*BaseND)(tab_noeud,dph_i,ph_i,bB,bH);
// calcul des coordonnees locales du point M dans le repere en refl
Coordonnee AM = M - A; // en global
for (int i=1;i<=dim_loc;i++)
theta(i) = AM * bH(i).Coor(); // ok même si dim_loc < dim_glob -> projection sur le plan tangent
Coordonnee theta_repere(theta); // le point où l'on calcul le repère
Coordonnee theta_initial(theta); Coordonnee delta_theta(theta);
// dans le cas où le point est externe à l'élément, on limite le repère de calcul au point externe
// de l'élément dans la direction de theta
if (!(elgeom.Interieur(theta)))
{ // calcul d'un point extreme de l'élément dans le sens de M
theta_repere = elgeom.Maxi_Coor_dans_directionGM(theta);
};
// calcul du point correspondant à theta dans l'element, qui remplace A
ph_i = elgeom.Phi_point(theta_repere); // fonctions d'interpolation en A
Coordonnee A_sauve= A;
A = (met->*PointM) (tab_noeud,ph_i);
AM = M-A; // nouveau gap
// ---- 2 :iteration
// on boucle avec un test de stabilité des coordonnées locales
int compteur = 1; // contrôle du maxi d'itération
int nb_exterieur=0; // contrôle du maxi de fois que l'on trouve consécutivement le point à l'extérieur
// - récup des paramètres de controle
double delta_thetai_maxi = ParaGlob::param->ParaAlgoControleActifs().PointInterneDeltaThetaiMaxi();
int nb_boucle_sur_delta_thetai= ParaGlob::param->ParaAlgoControleActifs().PointInterneNbBoucleSurDeltaThetai();
int nb_max_a_l_exterieur= ParaGlob::param->ParaAlgoControleActifs().PointInterneNbExterne();
double prec_tehetai_interne= ParaGlob::param->ParaAlgoControleActifs().PointInternePrecThetaiInterne();
// - boucle de recherche des thetai
while (delta_theta.Norme() >= delta_thetai_maxi)
{dph_i = elgeom.Dphi_point(theta_repere); // derivees des fonctions d'interpolation en theta_repere
ph_i = elgeom.Phi_point(theta_repere); // fonctions d'interpolation en theta_repere (déjà calculé la première fois)
// calcul de la base naturelle et duale en fonction des coord a t
(met->*BaseND)(tab_noeud,dph_i,ph_i,bB,bH);
// amelioration des coordonnees locales du point M dans le repere en theta
for (int i=1;i<=dim_loc;i++)
delta_theta(i)=AM * bH(i).Coor();
// theta += delta_theta; erreur je pense: le 11 juin 2007
// le nouveau theta = la position du repère + le deltat theta
theta = theta_repere + delta_theta;
theta_repere=theta;
if (!(elgeom.Interieur(theta))) // si le point est à l'extérieur
{ // calcul d'un point extreme de l'élément dans le sens de M
theta_repere = elgeom.Maxi_Coor_dans_directionGM(theta);
nb_exterieur++;
}
else // si nb_exterieur est diff de zero, on le remet à zéro, car on est de nouveau à l'intérieur
{ if (nb_exterieur != 0) nb_exterieur=0; };
// si cela fait nb_max_a_l_exterieur fois que l'on est à l'extérieur on arrête
if ( nb_exterieur >= nb_max_a_l_exterieur) break;
// calcul du point correspondant dans l'element, qui remplace theta
ph_i = elgeom.Phi_point(theta_repere); // fonctions d'interpolation en theta
A = (met->*PointM) (tab_noeud,ph_i);
AM = M-A; // nouveau gap
compteur++;
if (compteur > nb_boucle_sur_delta_thetai)
{ // cela signifie que l'on n'arrive pas à converger, petit message si besoin
// et on choisit le point initial pour le teste
if (ParaGlob::NiveauImpression() >= 4)
cout << " \n ** warning, l'algorithme de recherche d'un point a l'interieur d'un "
<< "element ne converge pas, utilisation du repere a l'origine"
<< "\nElemMeca::Interne(Coordonnee& M)" << endl;
if (elgeom.Interieur(theta_initial)) {theta=theta_initial; }
else {theta=theta_initial; };
break;
};
};
// // à la sortie de la boucle on calcul les thétas correspondants à M final
// dph_i = elgeom.Dphi(theta_repere); // derivees des fonctions d'interpolation en theta
// ph_i = elgeom.Phi(theta_repere); // fonctions d'interpolation en theta
// // calcul de la base naturelle et duale en fonction des coord a t
// (met->*BaseND)(tab_noeud,dph_i,ph_i,bB,bH);
// // amelioration des coordonnees locales du point M dans le repere en theta
// for (int i=1;i<=dim_glob;i++)
// theta(i) += AM * bH(i).Vect();
//
// -- maintenant on regarde si le point est interne ou pas a l'element
int retour=0;
if (elgeom.Interieur(theta))
{ // on regarde si le point est intérieur d'une toute petite précision
// calcul d'un point extreme de l'élément dans le sens de M
theta_repere = elgeom.Maxi_Coor_dans_directionGM(theta);
double diff =(theta_repere-theta).Norme();
if (diff <= prec_tehetai_interne)
{retour = 2;} // !-! à l'erreur près, le point est sur la frontière
else {retour = 1;}; // !-! point à l'intérieur
}
else {retour = 0;}; // !-! point à l'extérieur -> retour directe
if (!retour)
{// --- cas particuliers ---
// -- dans le cas où "retour" est différent de 0, on regarde les cas particuliers
switch (dim_glob)
{case 3:
{switch (dim_loc)
{ case 3: break; // cas pas de pb
case 2: // cas des "coques ou plaques" on regarde si le point est réellement à l'intérieur (+- 1/2 épaisseur)
{ double epaisseur = Epaisseurs(temps,theta);
A = (met->*PointM) (tab_noeud,ph_i); // le point projeté sur la surface de l'élément
AM = M-A; double delta_hauteur = 0.5 * epaisseur - AM.Norme();
if (Dabs(delta_hauteur) <= prec_tehetai_interne)
{retour = 2;} // !-! à l'erreur près, le point est sur la frontière
else if (delta_hauteur < 0.)
{retour = 0;} // !-! point externe
else {retour = 1;}; // !-! point à l'intérieur
break;
}
default:
cout << " \n *** erreur : cas non implemente pour l'instant "
<< " dim_glob= " << dim_glob << " dim_loc= " << dim_loc
<< "\n ElemMeca::Interne(... ";
Sortie(1);
};// -- fin du switch dim_loc pour dim_glob=3
break;
} // fin dim_glob=3
case 2:
{switch (dim_loc)
{case 2: break; // pas de pb
case 1: case 3: // cas des poutres et volume (pb non consistance) pour l'instant non traité
// donc on laisse aller à la valeur par défaut
default:
cout << " \n *** erreur : cas non implemente pour l'instant "
<< " dim_glob= " << dim_glob << " dim_loc= " << dim_loc
<< "\n ElemMeca::Interne(... ";
Sortie(1);
};// -- fin du switch dim_loc pour dim_glob=3
break;
} // fin dim_glob=2
case 1: // la dimension locale est forcément 1, et pas de pb
{break;
}
default: // default pour le switch sur di_glob
cout << " \n *** erreur : cas non implemente pour l'instant "
<< " dim_glob= " << dim_glob << " dim_loc= " << dim_loc
<< "\n ElemMeca::Interne(... ";
Sortie(1);
break;
}; // -- fin du switch sur dim_glob
};
// //----- debug
// if ((Num_elt() == 3) && (Num_maillage() == 2))
// {cout << "\n debug ElemMeca::Interne(...)";
// cout << " retour = "<<retour<< flush;
// };
// //---- fin debug
// retour
return retour;
};
// Calcul du residu local et de la raideur locale,
// pour le schema implicite
// cald_Dvirtuelle = indique si l'on doit calculer la dérivée de la vitesse de déformation virtuelle
void ElemMeca::Cal_implicit (DdlElement & tab_ddl
,Tableau <TenseurBB *> & d_epsBB,Tableau < Tableau2 <TenseurBB *> > d2_epsBB_
,Tableau <TenseurHH *>& d_sigHH,int nbint
,const Vecteur& poids,const ParaAlgoControle & pa,bool cald_Dvirtuelle)
{
int nbddl = tab_ddl.NbDdl();
double jacobien; // def du jacobien
volume = 0. ; // init
Vecteur d_jacobien_tdt;
energie_totale.Initialisation_differenciee(energie_totale_t); // init de l'énergie totale sur l'élément
E_elem_bulk_tdt = E_elem_bulk_t; P_elem_bulk = 0.; // init pour l'énergie et la puissance associées au bulk
// init éventuel de la contrainte de bulk viscosity
TenseurHH* sigHH_t_1 = (*lesPtIntegMecaInterne)(1).SigHH_t(); // simplification d'écriture
if (bulk_viscosity)
{ if (sig_bulkHH == NULL) {sig_bulkHH = NevezTenseurHH(*sigHH_t_1);}
else if (sig_bulkHH->Dimension() != sigHH_t_1->Dimension())
{delete sig_bulkHH; sig_bulkHH = NevezTenseurHH(*sigHH_t_1);};
};
// init éventuel des intégrales de volume et temps
Init_Integ_vol_et_temps();
/* 19 nov: en fait il ne faut pas prendre en compte l'épaisseur méca: cf. doc théorique
// --- init éventuel pour les épaisseurs et largeurs
double epaisseur_0 = EpaisseurMoyenne(TEMPS_0 );
double epaisseur = epaisseur_0;double epaisseur_moyenne = 0.; // init
*/
// --- init pour l'init de la stabilisation de membrane-biel éventuelle
const Met_abstraite::Impli* ex_final=NULL; // contiendra après les boucles la dernière métrique
Mise_a_jour_A_calculer_force_stab(); // permettra ensuite de savoir où le calcul doit être fait
for (int ni =1; ni<= nbint; ni++) // boucle sur les pt d'integ
{def->ChangeNumInteg(ni);def->Mise_a_jour_data_specif(tabSaveDefDon(ni));
PtIntegMecaInterne & ptIntegMeca = (*lesPtIntegMecaInterne)(ni);
TenseurHH & sigHH = *(ptIntegMeca.SigHH());
TenseurBB & DepsBB_tdt = *(ptIntegMeca.DepsBB());
EnergieMeca& energ = tab_energ(ni);
EnergieMeca& energ_t = tab_energ_t(ni);
CompThermoPhysiqueAbstraite::SaveResul* sTP=NULL; // les données spécifique thermo physiques
if (loiTP != NULL) {sTP = tabSaveTP(ni);}; // au pt d'integ si elles existes
if ((loiComp->Id_comport()==LOI_VIA_UMAT) // cas de l'utilisation d'une loi umat
|| (loiComp->Id_comport()==LOI_VIA_UMAT_CP))
((Loi_Umat*) loiComp)->Mise_a_jour_nbe_nbptinteg(this->Num_elt(),ni);
// -- on gère les exceptions éventuelles en mettant le bloc sous surveillance
const Met_abstraite::Impli* ex_pointe = NULL; // c'est pour construire ex
try // ce qui permet de donner les numéros d'éléments et de pti
{ ex_pointe = &loiComp->Cal_implicit(tabSaveDon(ni)
, *def,tab_ddl,ptIntegMeca,d_epsBB
,jacobien,d_jacobien_tdt,d_sigHH
,pa,sTP,loiTP,dilatation,energ,energ_t
,premier_calcul_meca_impli_expli);
// //---debug
// {double maxsig = sigHH.MaxiComposante();
// if ((!isfinite(maxsig)) || (isnan(maxsig)) )
// cout << "\n debug ElemMeca::Cal_implicit "
// << " maxsig= "
// << (maxsig)
// << flush;
//
//
// }
// //--- fin debug
}
catch (ErrNonConvergence_loiDeComportement excep)
// cas d'une d'une erreur survenue à cause d'une non convergence pour la résolution
// d'une loi de comportement incrémentale
{ cout << "\n erreur de loi de comportement element= " << this->Num_elt()
<< " point d'integration= "<<ni << endl;
throw excep;
}
catch (ErrSortieFinale)
// cas d'une direction voulue vers la sortie
// on relance l'interuption pour le niveau supérieur
{ ErrSortieFinale toto;
throw (toto);
}
catch ( ... )
{ cout << "\n erreur inconnue de loi de comportement, element= " << this->Num_elt()
<< " point d'integration= "<<ni << endl;
throw (Err_inconnue_ElemMeca());
};
const Met_abstraite::Impli& ex=(*ex_pointe);
ex_final = ex_pointe; // stockage pour la stabilisation éventuelle de membraneBiel
// on calcul éventuellement la dérivée de la vitesse de déformation virtuelle
Tableau2 <TenseurBB *>& d2_epsBB = d2_epsBB_(ni); // pour simplifier
if (cald_Dvirtuelle)
def->Cal_var_def_virtuelle(false,d2_epsBB);
// examen du cas où on désire utiliser la méthode d'atténuation des hautes fréquences avec le bulk viscosity
DeuxDoubles delta_ener_bulk_vol; // init
if (bulk_viscosity)
delta_ener_bulk_vol=ModifContrainteBulk(TEMPS_tdt,*ex.gijHH_tdt,sigHH,DepsBB_tdt,(*sig_bulkHH) );
double rap=1.; // pour le calcul éventuel du rapport de jacobien actuel
if (pa.MaxiVarJacobien() > 1.) // cas de la demande du calcul du rapport de jacobien
{if (Dabs(jacobien) > (*(ex.jacobien_0))) {rap = Dabs(jacobien) / (*(ex.jacobien_0));}
else {rap = (*(ex.jacobien_0)) / Dabs(jacobien) ;}
};
//// -- pour le debug
// if ((pa.Variables_de_temps().TempsCourant()==0.021)&&(Num_elt()==55)&&(ni==7))
// {// on affiche les infos des noeuds
// for (int i=1;i<= tab_noeud.Taille();i++) tab_noeud(i)->Affiche();
// // on affiche les vecteurs de bases à 0 et t
// cout << "\n gi_0: " << *(ex.giB_0)<< "\n gi_tdt: " << *(ex.giB_tdt);
// cout << "\n debug : ElemMeca::Cal_implicit ";
// };
////--fin debug
if ((jacobien <= 0.)|| (std::isinf(jacobien))||( std::isnan(jacobien))) // vérif du jacobien
{ if ((std::isinf(jacobien))||( std::isnan(jacobien)))
// on met un message quelque soit le niveau d'impression
{ cout << "\n ********** attention on a trouve un jacobien infini ou nan !! = ("<<jacobien<<")********* " << endl;
// là on ne peut plus rien faire donc on génère une exception
switch (pa.JabobienNegatif())
{ case 1:
{ cout << "\n on annulle sa contribution. Element nb: " << Num_elt() << " nb d'integ : " << ni;
break;
}
case 2:
{ // on génère une exception
throw ErrJacobienNegatif_ElemMeca();break;
}
// dans les autres cas on ne fait rien
};
};
if (ParaGlob::NiveauImpression() >= 1)
{ cout << "\n ********** attention on a trouve un jacobien negatif!! ("<<jacobien<<")********* " << endl; };
};
if ((jacobien <= 0.) && (pa.JabobienNegatif()!= 0)) // vérif du jacobien et traitement adéquate si besoin
{ switch (pa.JabobienNegatif())
{ case 1:
{ cout << "\n on annulle sa contribution. Element nb: " << Num_elt() << " nb d'integ : " << ni;
break;
}
case 2:
{ // on génère une exception
throw ErrJacobienNegatif_ElemMeca();break;
}
// dans les autres cas on ne fait rien
};
}
else if ((pa.MaxiVarJacobien() > 1.) && ( rap > pa.MaxiVarJacobien()))
{ if (ParaGlob::NiveauImpression() >= 6)
{// on affiche les infos des noeuds
for (int i=1;i<= tab_noeud.Taille();i++) tab_noeud(i)->Affiche();
// on affiche les vecteurs de bases à 0 et t
cout << "\n gi_0: " << *(ex.giB_0)<< "\n gi_tdt: " << *(ex.giB_tdt);
};
if (ParaGlob::NiveauImpression() >= 3)
{ cout << "\n *** attention la variation maximal du jacobien est atteinte !! *** "
<< "\n ele= "<< Num_elt() << " nbi= " << ni << " jacobien= " << jacobien << " jacobien_0= "
<< (*(ex.jacobien_0)) << endl; };
// on génère une exception
throw ErrVarJacobienMini_ElemMeca();break;
}
else
{ // on prend en compte toutes les variations
double poid_jacobien= poids(ni) * jacobien;
// calcul des intégrales de volume et de volume + temps
// cas = 1 : pour un appel après un calcul implicit
// cas = 2 : pour un appel après un calcul explicit
Calcul_Integ_vol_et_temps(1,poid_jacobien,ni);
/* 19 nov: en fait il ne faut pas prendre en compte l'épaisseur méca: cf. doc théorique
//---cas éventuel de la prise en compte de la variation d'une épaisseur, ou section
switch (loiComp->Comportement_3D_CP_DP_1D())
{ case COMP_CONTRAINTES_PLANES :
// en contrainte plane on recalcule l'épaisseur
{epaisseur = epaisseur_0 * loiComp->HsurH0(tabSaveDon(ni));
epaisseur_moyenne += epaisseur;
poid_jacobien *= epaisseur;
break;
}
case COMP_DEFORMATIONS_PLANES :
// en deformation plane, l'épaisseur ne change pas
{epaisseur = epaisseur_0 ;
epaisseur_moyenne += epaisseur;
poid_jacobien *= epaisseur;
break;
}
default : // on n'a rien à faire
break;
};
*/
volume += poid_jacobien;ptIntegMeca.Volume_pti()=poid_jacobien;
// on continue que s'il s'agit d'une loi différente de rien
if (!Loi_rien(loiComp->Id_comport()))
{// il ne faut pas interpoler l'énergie à t, car elle était associée à un volume qui a changé
// donc il ne faut considérer que l'accroissement d'énergie
energie_totale.Ajout_differenciee(energ,energ_t,poid_jacobien);
// energie_totale += (energ-energ_t) * poid_jacobien;
E_elem_bulk_tdt += delta_ener_bulk_vol.un * poid_jacobien;
P_elem_bulk += delta_ener_bulk_vol.deux * poid_jacobien;
// calcul du résidu et variation
for (int j =1; j<= nbddl; j++) // 1ere boucle sur les ddl
{(*residu)(j) -= ((*(d_epsBB(j))) && sigHH ) * poid_jacobien;
for (int i =1; i<= nbddl; i++) // 2ere boucle sur les ddl
(*raideur)(j,i) += ( ((*d_sigHH(i)) && (*(d_epsBB(j)))) * jacobien
+ (sigHH && (*(d2_epsBB(j,i)))) * jacobien
+ (sigHH && (*(d_epsBB(j)))) * d_jacobien_tdt(i)
) * poids(ni);
};
};
// --- cas éventuel d'une stabilisation membrane-biel ---
// ici il s'agit de la contribution précise à chaque pti
if (pt_StabMembBiel != NULL)
if (!(pt_StabMembBiel->Aa_calculer()))
Cal_implicit_StabMembBiel(ni,*ex_final,nbint,poids(ni),NULL);
////debug
//cout << "\n ElemMeca::Cal_implicit debug : ";
//raideur->Affiche();
//// fin debug
};// fin du if sur la valeur non négative du jacobien
if (bulk_viscosity) // dans le cas du bulk on retire la contrainte du bulk, qui est non physique
{ sigHH -= (*sig_bulkHH); } // ceci pour la sauvegarde ou autre utilisation
}; // fin boucle sur les points d'intégration
#ifdef MISE_AU_POINT
#ifdef UTILISATION_MPI
if (ParaGlob::Monde()->rank() != 0)
#endif
if (ParaGlob::NiveauImpression() > 9)
{ cout << "\n Raideur et second membre locaux: ? (o/n) "; string rep(" ");
// procédure de lecture avec prise en charge d'un retour chariot
rep = lect_return_defaut(false,"n");
if ((rep == "0")||(rep == "o"))
{cout << "\n Raideur et second membre locaux: elem= " << Num_elt_const()
<< ", maillage= " << Num_maillage();
cout << "\n raideur: ";
raideur->Affiche();
cout << "\n second membre: " << (*residu);
};
};
#endif
// --- intervention dans le cas d'une stabilisation d'hourglass
// stabilisation pour un calcul implicit
if (type_stabHourglass)
Cal_implicit_hourglass();
// --- cas éventuel d'une stabilisation membrane-biel ---
// ici il s'agit soit du calcul approché d'initialisation, soit de la fin du calcul après la boucle
// modif: 10 janvier 2019 non c'est le calcul correct une fois la raideur calculée
if (pt_StabMembBiel != NULL)
{if (pt_StabMembBiel->Aa_calculer())
{Cal_implicit_StabMembBiel(0,*ex_final, nbint,volume,NULL);}
else {Cal_implicit_StabMembBiel(-1,*ex_final, nbint,volume,NULL);} ;
};
#ifdef MISE_AU_POINT
#ifdef UTILISATION_MPI
if (ParaGlob::Monde()->rank() != 0)
#endif
if (ParaGlob::NiveauImpression() > 9)
{ if ((type_stabHourglass)||(pt_StabMembBiel != NULL))
{cout << "\n apres stabilisation: Raideur et second membre locaux: ? (o/n) "; string rep(" ");
// procédure de lecture avec prise en charge d'un retour chariot
rep = lect_return_defaut(false,"n");
if ((rep == "0")||(rep == "o"))
{cout << "\n Raideur et second membre locaux: elem= " << Num_elt_const()
<< ", maillage= " << Num_maillage();
cout << "\n raideur: ";
raideur->Affiche();
cout << "\n second membre: " << (*residu);
};
};
};
#endif
////---- debug ---
//if ((num_elt == 3)&&((integ_vol_typeQuel != NULL) || (integ_vol_t_typeQuel != NULL)))
// {cout << "\n debug ElemMeca::Cal_implicit ( ";
// if (integ_vol_typeQuel != NULL)
// {int taille = integ_vol_typeQuel->Taille();
// Tableau <TypeQuelconque> & tab_integ = *integ_vol_typeQuel; // pour simplifier
// for (int il =1;il <= taille ;il++)
// cout << "\n tab_integ("<<il<<")= "<< *(tab_integ(il).Grandeur_pointee());
// };
// // 2) intégration de volume et en temps
// if (integ_vol_t_typeQuel != NULL)
// {int taille = integ_vol_t_typeQuel->Taille();
// Tableau <TypeQuelconque> & tab_integ = *integ_vol_t_typeQuel; // pour simplifier
// Tableau <TypeQuelconque> & tab_integ_t = *integ_vol_t_typeQuel_t; // ""
// for (int il =1;il <= taille ;il++)
// cout << "\n tab_integ_et_temps("<<il<<")= "<< *(tab_integ(il).Grandeur_pointee());
// };
// cout << endl;
// };
//
//
////---- fin debug ----
// liberation des tenseurs intermediaires
LibereTenseur();
};
// Calcul uniquement du residu local a l'instant t ou pas suivant la variable atdt
// pour le schema explicit par exemple
void ElemMeca::Cal_explicit (DdlElement & tab_ddl,Tableau <TenseurBB *> & d_epsBB
,int nbint,const Vecteur& poids,const ParaAlgoControle & pa,bool atdt)
{ double jacobien; // def du jacobien
int nbddl = tab_ddl.NbDdl();
energie_totale.Initialisation_differenciee(energie_totale_t); // init de l'énergie totale sur l'élément
volume = 0. ; // init
E_elem_bulk_tdt = E_elem_bulk_t; P_elem_bulk = 0.; // init pour l'énergie et la puissance associées au bulk
// init éventuel de la contrainte de bulk viscosity
TenseurHH* sigHH_t_1 = (*lesPtIntegMecaInterne)(1).SigHH_t(); // simplification d'écriture
if (bulk_viscosity)
{ if (sig_bulkHH == NULL) {sig_bulkHH = NevezTenseurHH(*sigHH_t_1);}
else if (sig_bulkHH->Dimension() != sigHH_t_1->Dimension())
{delete sig_bulkHH; sig_bulkHH = NevezTenseurHH(*sigHH_t_1);};
};
// --- init éventuel des intégrales de volume et temps
Init_Integ_vol_et_temps();
/* 19 nov: en fait il ne faut pas prendre en compte l'épaisseur méca: cf. doc théorique
// --- init éventuel pour les épaisseurs et largeurs
double epaisseur_0 = EpaisseurMoyenne(TEMPS_0 );
double epaisseur = epaisseur_0;double epaisseur_moyenne = 0.; // init
*/
// --- init pour l'init de la stabilisation de membrane-biel éventuelle
const Met_abstraite::Expli_t_tdt* ex_final=NULL; // contiendra après les boucles la dernière métrique
Mise_a_jour_A_calculer_force_stab(); // permettra ensuite de savoir où le calcul doit être fait
for (int ni =1; ni<= nbint; ni++) // boucle sur les pt d'integ
{ def->ChangeNumInteg(ni);def->Mise_a_jour_data_specif(tabSaveDefDon(ni));
PtIntegMecaInterne & ptIntegMeca = (*lesPtIntegMecaInterne)(ni);
TenseurHH & sigHH_t = *(ptIntegMeca.SigHH_t());
TenseurHH & sigHH = *(ptIntegMeca.SigHH());
TenseurBB & epsBB = *(ptIntegMeca.EpsBB());
TenseurBB & DepsBB_ = *(ptIntegMeca.DepsBB());
EnergieMeca& energ = tab_energ(ni);
EnergieMeca& energ_t = tab_energ_t(ni);
CompThermoPhysiqueAbstraite::SaveResul* sTP=NULL; // les données spécifique thermo physiques
if (loiTP != NULL) {sTP = tabSaveTP(ni);}; // au pt d'integ si elles existes
if ((loiComp->Id_comport()==LOI_VIA_UMAT) // cas de l'utilisation d'une loi umat
|| (loiComp->Id_comport()==LOI_VIA_UMAT_CP))
((Loi_Umat*) loiComp)->Mise_a_jour_nbe_nbptinteg(this->Num_elt(),ni);
DeuxDoubles delta_ener_bulk_vol; // init
Met_abstraite::Expli_t_tdt ex_inter; // une grandeur intermédiaire
// -- on gère les exceptions éventuelles en mettant le bloc sous surveillance
try // ce qui permet de donner les numéros d'éléments et de pti
{if (atdt)
{ const Met_abstraite::Expli_t_tdt& ex = loiComp->Cal_explicit_tdt
(tabSaveDon(ni), *def,tab_ddl,ptIntegMeca,d_epsBB,jacobien
,sTP,loiTP,dilatation,energ,energ_t,premier_calcul_meca_impli_expli);
ex_inter= ex;
ex_final = &ex_inter; // stockage pour la stabilisation éventuelle de membraneBiel
// examen du cas où on désire utiliser la méthode d'atténuation des hautes fréquences avec le bulk viscosity
if (bulk_viscosity)
delta_ener_bulk_vol=ModifContrainteBulk(TEMPS_tdt,*ex.gijHH_tdt,sigHH,DepsBB_,(*sig_bulkHH) );
}
else
{ const Met_abstraite::Expli& ex = loiComp->Cal_explicit_t
(tabSaveDon(ni), *def,tab_ddl,ptIntegMeca,d_epsBB,jacobien
,sTP,loiTP,dilatation,energ,energ_t,premier_calcul_meca_impli_expli);
ex_inter = ex.T_dans_tdt();
ex_final = &ex_inter; // stockage pour la stabilisation éventuelle de membraneBiel
// examen du cas où on désire utiliser la méthode d'atténuation des hautes fréquences avec le bulk viscosity
if (bulk_viscosity)
delta_ener_bulk_vol=ModifContrainteBulk(TEMPS_t,*ex.gijHH_t,sigHH,DepsBB_,(*sig_bulkHH) );
};
}
catch (ErrNonConvergence_loiDeComportement excep)
// cas d'une d'une erreur survenue à cause d'une non convergence pour la résolution
// d'une loi de comportement incrémentale
{ cout << "\n erreur de loi de comportement element= " << this->Num_elt()
<< " point d'integration= "<<ni << endl;
throw excep;
}
catch (ErrSortieFinale)
// cas d'une direction voulue vers la sortie
// on relance l'interuption pour le niveau supérieur
{ ErrSortieFinale toto;
throw (toto);
}
catch ( ... )
{ cout << "\n erreur inconnue de loi de comportement, element= " << this->Num_elt()
<< " point d'integration= "<<ni << endl;
throw (Err_inconnue_ElemMeca());
};
double rap=1.; // pour le calcul éventuel du rapport de jacobien actuel
if (pa.MaxiVarJacobien() > 1.) // cas de la demande du calcul du rapport de jacobien
{if (Dabs(jacobien) > (*(ex_inter.jacobien_0))) {rap = Dabs(jacobien) / (*(ex_inter.jacobien_0));}
else {rap = (*(ex_inter.jacobien_0)) / Dabs(jacobien) ;}
};
if ((jacobien <= 0.)|| (std::isinf(jacobien))||( std::isnan(jacobien))) // vérif du jacobien
{ if ((std::isinf(jacobien))||( std::isnan(jacobien)))
// on met un message quelque soit le niveau d'impression
{ cout << "\n ********** attention on a trouve un jacobien infini ou nan !! = ("<<jacobien<<")********* " << endl;
// là on ne peut plus rien faire donc on génère une exception
switch (pa.JabobienNegatif())
{ case 1:
{ cout << "\n on annulle sa contribution. Element nb: " << Num_elt() << " nb d'integ : " << ni;
break;
}
case 2:
{ // on génère une exception
throw ErrJacobienNegatif_ElemMeca();break;
}
// dans les autres cas on ne fait rien
};
};
if (ParaGlob::NiveauImpression() >= 1)
{ cout << "\n ********** attention on a trouve un jacobien negatif!! ("<<jacobien<<")********* " << endl; };
};
if ((jacobien <= 0.) && (pa.JabobienNegatif() != 0)) // vérif du jacobien et traitement adéquate si besoin
{ switch (pa.JabobienNegatif())
{ case 1:
{ cout << "\n on annulle sa contribution. Element nb: " << Num_elt() << " nb d'integ : " << ni;
break;
}
case 2:
{ // on génère une exception
throw ErrJacobienNegatif_ElemMeca();break;
}
// dans les autres cas on ne fait rien
};
}
else if ((pa.MaxiVarJacobien() > 1.) && ( rap > pa.MaxiVarJacobien()))
{ if (ParaGlob::NiveauImpression() >= 6)
{// on affiche les infos des noeuds
for (int i=1;i<= tab_noeud.Taille();i++) tab_noeud(i)->Affiche();
// on affiche les vecteurs de bases à 0 et t
cout << "\n gi_0: " << *(ex_inter.giB_0)<< "\n gi_tdt: " << *(ex_inter.giB_tdt);
};
if (ParaGlob::NiveauImpression() >= 3)
{ cout << "\n *** attention la variation maximal du jacobien est atteinte !! *** "
<< "\n ele= "<< Num_elt() << " nbi= " << ni << " jacobien= " << jacobien << " jacobien_0= "
<< (*(ex_inter.jacobien_0)) << endl; };
// on génère une exception
throw ErrVarJacobienMini_ElemMeca();break;
}
else
{ double poid_jacobien= poids(ni) * jacobien;
// calcul des intégrales de volume et de volume + temps
// cas = 1 : pour un appel après un calcul implicit
// cas = 2 : pour un appel après un calcul explicit
Calcul_Integ_vol_et_temps(2,poid_jacobien,ni);
/* 19 nov: en fait il ne faut pas prendre en compte l'épaisseur méca: cf. doc théorique
//---cas éventuel de la prise en compte de la variation d'une épaisseur, ou section
switch (loiComp->Comportement_3D_CP_DP_1D())
{ case COMP_CONTRAINTES_PLANES :
// en contrainte plane on recalcule l'épaisseur
{epaisseur = epaisseur_0 * loiComp->HsurH0(tabSaveDon(ni));
epaisseur_moyenne += epaisseur;
poid_jacobien *= epaisseur;
break;
}
case COMP_DEFORMATIONS_PLANES :
// en deformation plane, l'épaisseur ne change pas
{epaisseur = epaisseur_0 ;
epaisseur_moyenne += epaisseur;
poid_jacobien *= epaisseur;
break;
}
default : // on n'a rien à faire
break;
};
*/
// il ne faut pas interpoler l'énergie à t, car elle était associée à un volume qui a changé
// donc il ne faut considérer que l'accroissement d'énergie
energie_totale.Ajout_differenciee(energ,energ_t,poid_jacobien);
// energie_totale += (energ-energ_t) * poid_jacobien;
// energie_totale += energ * poid_jacobien;
////debug
//cout << "\n debug: ElemMeca::Cal_explicit "
// << " energ visqueux = "<< energie_totale.DissipationVisqueuse() << " brut " << energ.DissipationVisqueuse()
// << " poid_jacobien = "<< poid_jacobien << endl;
//
////fin debug
volume += poid_jacobien;
ptIntegMeca.Volume_pti()=poid_jacobien;
// calcul du résidu si ce n'est pas une loi rien
if (!Loi_rien(loiComp->Id_comport()))
{E_elem_bulk_tdt += delta_ener_bulk_vol.un * poid_jacobien;
P_elem_bulk += delta_ener_bulk_vol.deux * poid_jacobien;
for (int j =1; j<= nbddl; j++)
{(*residu)(j) -= (sigHH && (*(d_epsBB(j)))) * poid_jacobien;
};
// --- cas éventuel d'une stabilisation membrane-biel ---
// ici il s'agit de la contribution précise à chaque pti
// pour une stabilisation via les normales aux pti
if (pt_StabMembBiel != NULL)
if (!(pt_StabMembBiel->Aa_calculer()))
Cal_explicit_StabMembBiel(ni,*ex_final,nbint,poids(ni),NULL);
////---debug
// #ifdef MISE_AU_POINT
// if (ParaGlob::NiveauImpression() > 2)
// if ((std::isinf((*residu)(j))||( std::isnan((*residu)(j)))))
// { cout << "\n ********** attention on a trouve une composante ("<<j<<") de residu infini ou nan !! = ("
// <<(*residu)(j)<<")********* " << endl;
// Sortie(2);
// };
// #endif
////--- fin debug
};
// --- cas éventuel d'une stabilisation membrane-biel ---
// modif: 10 janvier 2019 : je ne vois pas l'intérêt de faire le calcul à chaque pti
// car en fait le facteur de stabilisation est calculé à partir du résidu, or celui-ci
// n'est pas fini, et il commule à chaque nouveau pti, du coup le facteur est calculé à chaque pti
// via un résidu qui est transitoire et surtout qui ne contient pas que ce qui est calculé au pti !!
// donc ce n'est pas cohérent
// // ici il s'agit d'un calcul précis à chaque pti
// if (pt_StabMembBiel != NULL)
// if (pt_StabMembBiel->Aa_calculer())
// Cal_explicit_StabMembBiel(*ex_final);
};
if (bulk_viscosity) // dans le cas du bulk on retire la contrainte du bulk, qui est non physique
{ sigHH -= (*sig_bulkHH); } // ceci pour la sauvegarde ou autre utilisation
} // // fin boucle sur les points d'intégration
#ifdef MISE_AU_POINT
if (ParaGlob::NiveauImpression() > 9)
{ cout << "\n Second membre local: ? (o/n) "; string rep(" ");
// procédure de lecture avec prise en charge d'un retour chariot
rep = lect_return_defaut(false,"n");
if ((rep == "0")||(rep == "o"))
{cout << "\n Second membre local: elem= " << Num_elt_const()
<< ", maillage= " << Num_maillage();
cout << "\n second membre: " << (*residu);
};
};
#endif
/* 19 nov: en fait il ne faut pas prendre en compte l'épaisseur méca: cf. doc théorique
//---cas éventuel de la prise en compte de la variation d'une épaisseur, ou section
switch (loiComp->Comportement_3D_CP_DP_1D())
{ case COMP_CONTRAINTES_PLANES :
// en contrainte plane on recalcule l'épaisseur
{epaisseur_moyenne /= nbint;
Modifie_epaisseur_moyenne_tdt(epaisseur);
break;
}
default : // on n'a rien à faire
break;
};
*/
// --- intervention dans le cas d'une stabilisation d'hourglass
// stabilisation pour un calcul implicit
if (type_stabHourglass)
Cal_explicit_hourglass(atdt);
// --- cas éventuel d'une stabilisation membrane-biel ---
// ici il s'agit soit du calcul approché d'initialisation, soit de la fin du calcul après la boucle
// modif: 10 janvier 2019 non c'est le calcul correct une fois la raideur calculée
if (pt_StabMembBiel != NULL)
{Cal_explicit_StabMembBiel(-1,*ex_final, nbint,volume,NULL);
};
#ifdef MISE_AU_POINT
if (ParaGlob::NiveauImpression() > 9)
{ if ((type_stabHourglass)||(pt_StabMembBiel != NULL))
{cout << "\n apres stabilisation: Second membre local: ? (o/n) "; string rep(" ");
// procédure de lecture avec prise en charge d'un retour chariot
rep = lect_return_defaut(false,"n");
if ((rep == "0")||(rep == "o"))
{cout << "\n Second membre local: elem= " << Num_elt_const()
<< ", maillage= " << Num_maillage();
cout << "\n second membre: " << (*residu);
};
};
};
#endif
// liberation des tenseurs intermediaires
LibereTenseur();
};
// Calcul de la matrice géométrique et de la matrice initiale
// cette fonction est éventuellement appelée par les classes dérivées
// ddl represente les degres de liberte specifiques a l'element
// epsBB = deformation, sigHH = contrainte, d_epsbb = variation des def
// nbint = nb de pt d'integration , poids = poids d'integration
// cald_Dvirtuelle = indique si l'on doit calculer la dérivée de la vitesse de déformation virtuelle
void ElemMeca::Cal_matGeom_Init (Mat_pleine & matGeom, Mat_pleine & matInit,DdlElement & tab_ddl
,Tableau <TenseurBB *>& d_epsBB, Tableau < Tableau2 <TenseurBB *> > d2_epsBB_
,Tableau <TenseurHH *>& d_sigHH,int nbint,const Vecteur& poids
,const ParaAlgoControle & pa,bool cald_Dvirtuelle)
{ // le début du programme est semblable au calcul implicit
// seule la constitution des matrices finales est différente
int nbddl = tab_ddl.NbDdl();
double jacobien; // def du jacobien
Vecteur d_jacobien_tdt;
volume = 0. ; // init
for (int ni =1; ni<= nbint; ni++) // boucle sur les pt d'integ
{def->ChangeNumInteg(ni);def->Mise_a_jour_data_specif(tabSaveDefDon(ni));
PtIntegMecaInterne & ptIntegMeca = (*lesPtIntegMecaInterne)(ni);
TenseurHH & sigHH = *(ptIntegMeca.SigHH());
TenseurBB & DepsBB_tdt = *(ptIntegMeca.DepsBB());
EnergieMeca& energ = tab_energ(ni);
EnergieMeca& energ_t = tab_energ_t(ni);
CompThermoPhysiqueAbstraite::SaveResul* sTP=NULL; // les données spécifique thermo physiques
if (loiTP != NULL) {sTP = tabSaveTP(ni);}; // au pt d'integ si elles existes
if ((loiComp->Id_comport()==LOI_VIA_UMAT) // cas de l'utilisation d'une loi umat
|| (loiComp->Id_comport()==LOI_VIA_UMAT_CP))
((Loi_Umat*) loiComp)->Mise_a_jour_nbe_nbptinteg(this->Num_elt(),ni);
// -- on gère les exceptions éventuelles en mettant le bloc sous surveillance
try // ce qui permet de donner les numéros d'éléments et de pti
{loiComp->Cal_flamb_lin(tabSaveDon(ni), *def,tab_ddl,ptIntegMeca
,d_epsBB,jacobien,d_jacobien_tdt,d_sigHH
,pa,sTP,loiTP,dilatation,energ,energ_t
,premier_calcul_meca_impli_expli);
}
catch (ErrNonConvergence_loiDeComportement excep)
// cas d'une d'une erreur survenue à cause d'une non convergence pour la résolution
// d'une loi de comportement incrémentale
{ cout << "\n erreur de loi de comportement element= " << this->Num_elt()
<< " point d'integration= "<<ni << endl;
throw excep;
}
catch (ErrSortieFinale)
// cas d'une direction voulue vers la sortie
// on relance l'interuption pour le niveau supérieur
{ ErrSortieFinale toto;
throw (toto);
}
catch ( ... )
{ cout << "\n erreur inconnue de loi de comportement, element= " << this->Num_elt()
<< " point d'integration= "<<ni << endl;
throw (Err_inconnue_ElemMeca());
};
// on calcul éventuellement la dérivée de la vitesse de déformation virtuelle
Tableau2 <TenseurBB *>& d2_epsBB = d2_epsBB_(ni); // pour simplifier
if (cald_Dvirtuelle)
def->Cal_var_def_virtuelle(false,d2_epsBB);
// calcul du volume
double poid_jacobien= poids(ni) * jacobien;
volume += poid_jacobien;ptIntegMeca.Volume_pti()=poid_jacobien;
for (int j =1; j<= nbddl; j++) // 1ere boucle sur les ddl
for (int i =1; i<= nbddl; i++) // 2ere boucle sur les ddl
{matInit(j,i) += ((*d_sigHH(i)) && (*(d_epsBB(j)))) * jacobien * poids(ni);
matGeom(j,i) += (sigHH && (*(d2_epsBB(j,i)))) * poid_jacobien;
}
}
// liberation des tenseurs intermediaires
LibereTenseur();
};
// Calcul de la matrice masse selon différent choix donné par type_matrice_masse,
void ElemMeca::Cal_Mat_masse (DdlElement & tab_ddl,Enum_calcul_masse type_matrice_masse,
int nbint,const Tableau <Vecteur>& taphi,int nbne
,const Vecteur& poids)
{
int nbddl = tab_ddl.NbDdl();
double jacobien_0; // def du jacobien initial
// initialisation de la matrice masse
mat_masse->Initialise ();
// le calcul est fonction du type de matrice masse demandé
switch (type_matrice_masse)
{case MASSE_DIAG_COEF_EGAUX :
// même masse sur chaque noeud
{ // calcul du volume totale
double vol_totale =0.;
for (int ni =1; ni<= nbint; ni++) // boucle sur les pt d'integ
{ defMas->ChangeNumInteg(ni);
const Met_abstraite::Dynamiq& ex = defMas->Cal_matMass();
jacobien_0 = *ex.jacobien_0;
vol_totale += jacobien_0 * poids(ni);
}
// la masse élémentaire pour chaque noeud
double masse_elementaire = vol_totale * masse_volumique / tab_noeud.Taille();
// on initialise toutes les valeurs de la matrice masse à la valeur moyenne
mat_masse->Initialise (masse_elementaire );
break;
}
case MASSE_DIAG_COEF_VAR :
// masse diagonale répartie au prorata des fonctions d'interpolation
// (cf page 304, tome 2 Batoz)
{ // définition d'un vecteur qui contiendra l'intégral de chaque
// fonction d'interpolation (ici la masse est considérée constante)
Vecteur fonc2(nbne);
// calcul de fonc2 et du volume totale
double vol_totale =0.;
for (int ni =1; ni<= nbint; ni++) // boucle sur les pt d'integ
{ defMas->ChangeNumInteg(ni);
const Met_abstraite::Dynamiq& ex = defMas->Cal_matMass();
jacobien_0 = *ex.jacobien_0;
for (int ne =1;ne<=nbne;ne++)
fonc2(ne) += taphi(ni)(ne) * taphi(ni)(ne) * jacobien_0 * poids(ni);
vol_totale += jacobien_0 * poids(ni);
};
// calcul de la somme des composantes de fonc2
double somme_fonc2 = 0.;
for (int ne = 1;ne<=nbne;ne++) somme_fonc2 += fonc2(ne);
// calcul des termes diagonaux de la matrice de masse
int dimension = ParaGlob::Dimension();
if(ParaGlob::AxiSymetrie())
// cas d'élément axisymétrique, dans ce cas on ne prend en compte que les
// dimension-1 coordonnées donc on décrémente
dimension--;
int ix=1;
for (int ne =1; ne<= nbne; ne++)
{ double inter_toto = vol_totale * masse_volumique * fonc2(ne) / somme_fonc2;
for (int i=1;i<= dimension;i++,ix++)
(*mat_masse)(1,ix) = inter_toto;
};
break;
}
case MASSE_CONSISTANTE :
// matrice masse complète avec les mêmes fonctions d'interpolation
// que pour la raideur.
{
int dimension = ParaGlob::Dimension();
if(ParaGlob::AxiSymetrie())
// cas d'élément axisymétrique, dans ce cas on ne prend en compte que les
// dimension-1 coordonnées donc on décrémente
dimension--;
for (int ni =1; ni<= nbint; ni++) // boucle sur les pt d'integ
{defMas->ChangeNumInteg(ni); // def du pt d'intégration
// récup des fonctions d'interpolation et du jacobien
const Met_abstraite::Dynamiq& ex = defMas->Cal_matMass();
double masse_i = (*ex.jacobien_0) * masse_volumique;
// calcul de la matrice
int ix=1;
for (int ne =1; ne<= nbne; ne++)
for (int a=1;a<= dimension;a++,ix++)
{ int jy=1;
for (int me =1; me<= nbne; me++)
for (int b=1;b<= dimension;b++,jy++)
if (a==b) // seule des termes de même indice sont non nulles
(*mat_masse)(ix,jy) += taphi(ni)(ne)* taphi(ni)(me) * masse_i * poids(ni);
};
};
break;
}
default :
cout << "\nErreur : valeur incorrecte du type Enum_calcul_masse !\n";
cout << "ElemMeca::Cal_Mat_masse (... \n";
Sortie(1);
};
// liberation des tenseurs intermediaires
LibereTenseur();
};
// ------------ calcul de second membre ou de l'action des efforts extérieures -----------
// initialisation éventuelle, nécessaire avant d'appliquer l'ensemble des charges
// par exemple des stockages intermédiaires
void ElemMeca::Initialisation_avant_chargement()
{ // init à 0 des efforts extérieurs éventuels
if (lesChargeExtSurEle != NULL)
lesChargeExtSurEle->Zero();
};
// calcul des seconds membres suivant les chargements
// cas d'un chargement surfacique de direction constante, sur les frontières des éléments
// force indique la force surfacique appliquée
// retourne le second membre résultant
// nSurf : le numéro de la surface externe
// calcul à tdt ou t suivant la variable atdt
Vecteur& ElemMeca::SM_charge_surf_E (DdlElement & ,int nSurf
,const Tableau <Vecteur>& taphi,int nbne
,const Vecteur& poids,const Coordonnee& force,Fonction_nD* pt_fonct
,const ParaAlgoControle & ,bool atdt)
{ // définition du vecteur de retour
Vecteur& SM = *((*res_extS)(nSurf));
Deformation & defS = *defSurf(nSurf); // pour simplifier l'écriture
// éventuellement dimensionnement de la sauvegarde
if (lesChargeExtSurEle == NULL)
lesChargeExtSurEle = new LesChargeExtSurElement;
if (lesChargeExtSurEle->LesEffortsDirFixe() == NULL)
{if(ParaGlob::AxiSymetrie())
{lesChargeExtSurEle->LesEffortsDirFixe_Change_taille(ElementGeometrique().NbSe());}
else
{lesChargeExtSurEle->LesEffortsDirFixe_Change_taille(ElementGeometrique().NbFe());}
};
Tableau <Coordonnee>& tab_DirFixe= (*lesChargeExtSurEle->LesEffortsDirFixe())(nSurf); // pour simplifier
if (tab_DirFixe.Taille() != defS.Phi1_Taille())
tab_DirFixe.Change_taille(defS.Phi1_Taille(),Coordonnee(ParaGlob::Dimension()));
int ni; // compteur globale de point d'integration
for (defS.PremierPtInteg(), ni = 1;defS.DernierPtInteg();defS.NevezPtInteg(),ni++)
{// calcul des éléments de la métrique, entre autre le jacobien, on utilise le même calcul
// que pour un calcul primaire en explicit mais pour un calcul autre que mécanique
double jacobien;
bool premier_calcul=true; // contrairement à la déformation, pas de sauvegarde
// donc il faut calculer tous les éléments de la métrique
const Met_abstraite::Impli* ex_impli = NULL;
const Met_abstraite::Expli_t_tdt* ex_expli_tdt = NULL;
const Met_abstraite::Expli* ex_expli = NULL;
if (atdt)
{ const Met_abstraite::Expli_t_tdt& ex = defS.Cal_explicit_tdt(premier_calcul);
jacobien = (*ex.jacobien_tdt);
ex_expli_tdt = &ex;
}
else // normalement le cas à t n'est pas courant, donc on tolère une création
{ const Met_abstraite::Expli& ex = defS.Cal_explicit_t(premier_calcul);
jacobien = (*ex.jacobien_t);
ex_expli = &ex;
}
int ix=1; int dimf = force.Dimension();
if(ParaGlob::AxiSymetrie())
// cas d'un chargement axisymétrique, dans ce cas on ne prend en compte que les
// dimf-1 coordonnées donc on décrémente
dimf--;
// prise en compte d'une dépendance à une fonction nD
Coordonnee force_reelle(force); // init par défaut
if (pt_fonct != NULL)
{ // ici on utilise les variables connues aux noeuds, ou calculées à partir de
// on commence par récupérer les conteneurs des grandeurs à fournir
List_io <Ddl_enum_etendu>& li_enu_scal = pt_fonct->Li_enu_etendu_scalaire();
List_io <TypeQuelconque >& li_quelc = pt_fonct->Li_equi_Quel_evolue();
bool absolue = true; // on se place systématiquement en absolu
// on va utiliser la méhode Valeur_multi_interpoler_ou_calculer
// pour les grandeurs strictement scalaire
Tableau <double> val_ddl_enum(Valeur_multi_interpoler_ou_calculer
(absolue,TEMPS_tdt,li_enu_scal,defS,ex_impli,ex_expli_tdt,ex_expli)
);
// on utilise la méthode Valeurs_Tensorielles_interpoler_ou_calculer
// pour les Coordonnees et Tenseur
Valeurs_Tensorielles_interpoler_ou_calculer
(absolue,TEMPS_tdt,li_quelc,defS,ex_impli,ex_expli_tdt,ex_expli);
// calcul de la valeur et retour dans tab_ret
Tableau <double> & tab_val = pt_fonct->Valeur_FnD_Evoluee(&val_ddl_enum,&li_enu_scal,&li_quelc,NULL,NULL);
// if (pt_fonct->Depend_M())
// {// 1) on récupère la position dans I_a du point d'intégration
// const Coordonnee & M = defS.Position_tdt();
// // 2) on appel la fonction, qui ne doit dépendre que de M en var locales
// Tableau <Coordonnee> tab(1); tab(1)=M;
// Tableau <double> tab_val = pt_fonct->Val_FnD_Evoluee(NULL,&tab,NULL);
// deux cas: soit une fonction scalaire soit dimf composantes
if (tab_val.Taille() == 1)
{force_reelle *= tab_val(1);}
else if (tab_val.Taille() == dimf)
{switch (dimf)
{case 3:force_reelle(3) *= tab_val(3);
case 2:force_reelle(2) *= tab_val(2);
case 1:force_reelle(1) *= tab_val(1);
break;
default: cout << "\n *** erreur ElemMeca::SM_charge_surf_E(.. ";
Sortie(1);
};
}
else
{cout << "\n *** erreur dans l'utilisation d'une fct nD pour un chargement"
<< " surfacique: la dimension de la fct nD "<<tab_val.Taille()
<< " devrait etre soit 1, soit "<<dimf << endl;
Sortie(1);
};
// }
// else
// {cout << "\n *** erreur dans l'utilisation d'une fct nD pour un chargement"
// << " surfacique: dependance autre que du point "<< endl;
// Sortie(1);
// };
};
tab_DirFixe(ni)=force_reelle; // sauvegarde de l'effort
for (int ne =1; ne<= nbne; ne++)
for (int i=1;i<= dimf;i++,ix++)
SM(ix) += taphi(ni)(ne)* force_reelle(i) * (poids(ni) * jacobien);
}; // fin de la boucle sur les points d'intégraion
// retour du second membre
return SM;
};
// calcul des seconds membres suivant les chargements
// cas d'un chargement surfacique de direction constante, sur les frontières des éléments
// force indique la force surfacique appliquée
// retourne le second membre résultant
// nSurf : le numéro de la surface externe
// -> implicite,
// pa : permet de déterminer si oui ou non on calcul la contribution à la raideur
// retourne le second membre et la matrice de raideur correspondant
Element::ResRaid ElemMeca::SMR_charge_surf_I (DdlElement & ddls,int nSurf
,const Tableau <Vecteur>& taphi,int nbne
,const Vecteur& poids,const Coordonnee& force,Fonction_nD* pt_fonct
,const ParaAlgoControle & pa)
{ bool avec_raid = pa.Var_charge_externe(); // récup indicateur
bool premier_calcul=true; // contrairement à la déformation, pas de sauvegarde
// donc il faut calculer tous les éléments de la métrique
// // dans le cas où la variation sur la raideur est demandé on s'assure que la variation
// // du jacobien est bien effective sinon on l'impose
// ParaAlgoControle pa_nevez(pa);
// if (!pa_nevez.Var_jacobien()) pa_nevez.Modif_Var_jacobien(true);
int ni; // compteur globale de point d'integration
Deformation & defS = *defSurf(nSurf); // pour simplifier l'écriture
// éventuellement dimensionnement de la sauvegarde
if (lesChargeExtSurEle == NULL)
lesChargeExtSurEle = new LesChargeExtSurElement;
if (lesChargeExtSurEle->LesEffortsDirFixe() == NULL)
{if(ParaGlob::AxiSymetrie())
{lesChargeExtSurEle->LesEffortsDirFixe_Change_taille(ElementGeometrique().NbSe());}
else
{lesChargeExtSurEle->LesEffortsDirFixe_Change_taille(ElementGeometrique().NbFe());}
};
Tableau <Coordonnee>& tab_DirFixe= (*lesChargeExtSurEle->LesEffortsDirFixe())(nSurf); // pour simplifier
if (tab_DirFixe.Taille() != defS.Phi1_Taille())
tab_DirFixe.Change_taille(defS.Phi1_Taille(),Coordonnee(ParaGlob::Dimension()));
int nbddl = ddls.NbDdl();
Vecteur& SM = (*((*res_extS)(nSurf))); // " " "
Mat_pleine & KM = (*((*raid_extS)(nSurf))); // " " "
for (defS.PremierPtInteg(), ni = 1;defS.DernierPtInteg();defS.NevezPtInteg(),ni++)
{// calcul des éléments de la métrique, entre autre le jacobien, on utilise le même calcul
// que pour un calcul primaire en implicit
const Met_abstraite::Impli& ex = defS.Cal_implicit(premier_calcul);
const Met_abstraite::Impli* ex_impli = &ex;
const Met_abstraite::Expli_t_tdt* ex_expli_tdt = NULL;
const Met_abstraite::Expli* ex_expli = NULL;
int ix=1; int dimf = force.Dimension();
if(ParaGlob::AxiSymetrie())
// cas d'un chargement axisymétrique, dans ce cas on ne prend en compte que les
// dimf-1 coordonnées donc on décrémente
dimf--;
// prise en compte d'une dépendance à une fonction nD
// on ne tient pas compte de la variation du point dans la fct nD
Coordonnee force_reelle(force); // init par défaut
if (pt_fonct != NULL)
{ // ici on utilise les variables connues aux noeuds, ou calculées à partir de
// on commence par récupérer les conteneurs des grandeurs à fournir
List_io <Ddl_enum_etendu>& li_enu_scal = pt_fonct->Li_enu_etendu_scalaire();
List_io <TypeQuelconque >& li_quelc = pt_fonct->Li_equi_Quel_evolue();
bool absolue = true; // on se place systématiquement en absolu
// on va utiliser la méhode Valeur_multi_interpoler_ou_calculer
// pour les grandeurs strictement scalaire
Tableau <double> val_ddl_enum(Valeur_multi_interpoler_ou_calculer
(absolue,TEMPS_tdt,li_enu_scal,defS,ex_impli,ex_expli_tdt,ex_expli)
);
// on utilise la méthode Valeurs_Tensorielles_interpoler_ou_calculer
// pour les Coordonnees et Tenseur
Valeurs_Tensorielles_interpoler_ou_calculer
(absolue,TEMPS_tdt,li_quelc,defS,ex_impli,ex_expli_tdt,ex_expli);
// calcul de la valeur et retour dans tab_ret
Tableau <double> & tab_val = pt_fonct->Valeur_FnD_Evoluee(&val_ddl_enum,&li_enu_scal,&li_quelc,NULL,NULL);
// if (pt_fonct->Depend_M())
// {// 1) on récupère la position dans I_a du point d'intégration
// const Coordonnee & M = defS.Position_tdt();
// // 2) on appel la fonction, qui ne doit dépendre que de M en var locales
// Tableau <Coordonnee> tab(1); tab(1)=M;
// Tableau <double> tab_val = pt_fonct->Val_FnD_Evoluee(NULL,&tab,NULL);
// deux cas: soit une fonction scalaire soit dimf composantes
if (tab_val.Taille() == 1)
{force_reelle *= tab_val(1);}
else if (tab_val.Taille() == dimf)
{switch (dimf)
{case 3:force_reelle(3) *= tab_val(3);
case 2:force_reelle(2) *= tab_val(2);
case 1:force_reelle(1) *= tab_val(1);
break;
default: cout << "\n *** erreur ElemMeca::SM_charge_surf_E(.. ";
Sortie(1);
};
}
else
{cout << "\n *** erreur dans l'utilisation d'une fct nD pour un chargement"
<< " surfacique: la dimension de la fct nD "<<tab_val.Taille()
<< " devrait etre soit 1, soit "<<dimf << endl;
Sortie(1);
};
// }
// else
// {cout << "\n *** erreur dans l'utilisation d'une fct nD pour un chargement"
// << " surfacique: dependance autre que du point "<< endl;
// Sortie(1);
// };
};
tab_DirFixe(ni)=force_reelle; // sauvegarde de l'effort
for (int ne =1; ne<= nbne; ne++)
for (int i=1;i<= dimf;i++,ix++)
{ SM(ix) += taphi(ni)(ne)* force_reelle(i) * (poids(ni) * (*ex.jacobien_tdt));
// dans le cas avec_raideur on calcul la contribution à la raideur
if (avec_raid)
for (int j =1; j<= nbddl; j++)
KM(ix,j) +=
taphi(ni)(ne)* force_reelle(i) * (poids(ni) * (*ex.d_jacobien_tdt)(j));
};
}
// liberation des tenseurs intermediaires
LibereTenseur();
Element::ResRaid el;
el.res = (*res_extS)(nSurf);
el.raid = (*raid_extS)(nSurf);
return el;
};
// calcul des seconds membres suivant les chargements
// cas d'un chargement pression, sur les frontières des éléments
// pression indique la pression appliquée
// retourne le second membre résultant
// nSurf : le numéro de la surface externe
// calcul à l'instant tdt ou t en fonction de la variable atdt
Vecteur& ElemMeca::SM_charge_pres_E (DdlElement & ,int nSurf
,const Tableau <Vecteur>& taphi,int nbne
,const Vecteur& poids,double pression,Fonction_nD* pt_fonct
,const ParaAlgoControle & ,bool atdt)
{
Deformation* definter=NULL; // variable de travail transitoire
Vecteur* SM_P = NULL;
if (!(ParaGlob::AxiSymetrie()))
{ definter = defSurf(nSurf); // le cas normal est non axisymétrique
SM_P = ((*res_extS)(nSurf));
}
else { definter = defArete(nSurf); // en axisymétrie, c'est une def d'arête
SM_P = ((*res_extA)(nSurf));
};
Deformation & defS = *definter; // pour simplifier l'écriture
Vecteur& SM = *SM_P; // " " "
// éventuellement dimensionnement de la sauvegarde
if (lesChargeExtSurEle == NULL)
lesChargeExtSurEle = new LesChargeExtSurElement;
if (lesChargeExtSurEle->LesPressionsExternes() == NULL)
{if(ParaGlob::AxiSymetrie())
{lesChargeExtSurEle->LesPressionsExternes_Change_taille(ElementGeometrique().NbSe());}
else
{lesChargeExtSurEle->LesPressionsExternes_Change_taille(ElementGeometrique().NbFe());}
};
Tableau <Pression_appliquee>& tab_Press= (*lesChargeExtSurEle->LesPressionsExternes())(nSurf); // pour simplifier
if (tab_Press.Taille() != defS.Phi1_Taille())
tab_Press.Change_taille(defS.Phi1_Taille());
// définition du vecteur de retour
int ni; // compteur globale de point d'integration
bool premier_calcul=true; // contrairement à la déformation, pas de sauvegarde
// donc il faut calculer tous les éléments de la métrique
for (defS.PremierPtInteg(), ni = 1;defS.DernierPtInteg();defS.NevezPtInteg(),ni++)
{
// calcul des éléments de la métrique, entre autre le jacobien, on utilise le même calcul
// que pour un calcul primaire en explicit mais pour un calcul autre que mécanique
Met_abstraite::Expli ex;
if (atdt)
ex = ((defS.Cal_explicit_tdt(premier_calcul)).Tdt_dans_t());
else
ex.operator=( defS.Cal_explicit_t(premier_calcul));
const Met_abstraite::Impli* ex_impli = NULL;
const Met_abstraite::Expli_t_tdt* ex_expli_tdt = NULL;
const Met_abstraite::Expli* ex_expli = &ex;
// détermination de la normale à la surface
#ifdef MISE_AU_POINT
// test pour savoir si l'on a réellement affaire à une surface
if ((ex.giB_t->NbVecteur() != 2) || ( ParaGlob::Dimension() != 3))
{ cout << "\n erreur, il doit y avoir deux vecteurs pour la surface de dimension 3"
<< "\n ici on a nbvec= " << ex.giB_t->NbVecteur() <<" vecteur(s) et on est en dimension "
<< " dim = " << ParaGlob::Dimension();
if (ParaGlob::NiveauImpression() > 2)
cout << "\n ElemMeca::SM_charge_pres (DdlElement & ...nbvec= ";
Sortie(1);
}
#endif
// la normale vaut le produit vectoriel des 2 premiers vecteurs
Coordonnee normale = Util::ProdVec_coorBN( (*ex.giB_t)(1), (*ex.giB_t)(2));
// dans le cas où il y a une fonction nD, et qu'un de ces arguments
// est des coordonnées Xi on l'utilise
double pression_applique = pression;
if (pt_fonct != NULL)
{ // ici on utilise les variables connues aux noeuds, ou calculées à partir de
// on commence par récupérer les conteneurs des grandeurs à fournir
List_io <Ddl_enum_etendu>& li_enu_scal = pt_fonct->Li_enu_etendu_scalaire();
List_io <TypeQuelconque >& li_quelc = pt_fonct->Li_equi_Quel_evolue();
bool absolue = true; // on se place systématiquement en absolu
// on va utiliser la méhode Valeur_multi_interpoler_ou_calculer
// pour les grandeurs strictement scalaire
Tableau <double> val_ddl_enum(Valeur_multi_interpoler_ou_calculer
(absolue,TEMPS_tdt,li_enu_scal,defS,ex_impli,ex_expli_tdt,ex_expli)
);
// on utilise la méthode Valeurs_Tensorielles_interpoler_ou_calculer
// pour les Coordonnees et Tenseur
Valeurs_Tensorielles_interpoler_ou_calculer
(absolue,TEMPS_tdt,li_quelc,defS,ex_impli,ex_expli_tdt,ex_expli);
// calcul de la valeur et retour dans tab_ret
Tableau <double> & tab_val = pt_fonct->Valeur_FnD_Evoluee(&val_ddl_enum,&li_enu_scal,&li_quelc,NULL,NULL);
// if (pt_fonct->Depend_M())
// {// calcul de la pression au niveau de la position du point d'intégration
// // 1) on récupère la position dans I_a du point d'intégration
// const Coordonnee & M = defS.Position_tdt();
// // 2) on appel la fonction, qui ne doit dépendre que de M en var locales
// Tableau <Coordonnee> tab(1); tab(1)=M;
// Tableau <double> tab_val = pt_fonct->Val_FnD_Evoluee(NULL,&tab,NULL);
// seule la première valeur est ok
pression_applique *= tab_val(1);
// }
// else
// {cout << "\n *** erreur dans l'utilisation d'une fct nD pour un chargement"
// << " surfacique: dependance autre que du point "<< endl;
// Sortie(1);
// };
};
// calcul de la normale normée et pondérée de la pression
// la pression est positive qu'en elle appuis (diminution de volume)
normale.Normer();normale *= -pression_applique;
tab_Press(ni).P += normale;
int ix=1; int dimf = 3; // ne fonctionne qu'en dim 3
if(ParaGlob::AxiSymetrie())
// cas d'un chargement axisymétrique, dans ce cas on ne prend en compte que les
// dimf-1 coordonnées donc on décrémente
dimf--;
for (int ne =1; ne<= nbne; ne++)
for (int i=1;i<= dimf;i++,ix++)
SM(ix) += taphi(ni)(ne)* normale(i) * (poids(ni) * (*ex.jacobien_t));
// sauvegarde de la pression appliquée
tab_Press(ni).press += pression_applique;
//debug
//if (num_elt == 33839)
// cout << "\n debug: ElemMeca::SM_charge_pres_E "
// << " pression= " << pression << endl;
//if (tab_press_appliquee(ni).press_tdt > 120)
// cout << "\n debug: ElemMeca::SM_charge_pres_E "<< " on s'arrete ";
//fin debug
};
// retour du second membre
return SM;
};
// calcul des seconds membres suivant les chargements
// cas d'un chargement pression, sur les frontières des éléments
// pression indique la pression appliquée
// retourne le second membre résultant
// nSurf : le numéro de la surface externe -> implicite,
// pa: permet de déterminer si oui ou non on calcul la contribution à la raideur
Element::ResRaid ElemMeca::SMR_charge_pres_I(DdlElement & ddlS,int nSurf
,const Tableau <Vecteur>& taphi,int nbne
,const Vecteur& poids,double pression,Fonction_nD* pt_fonct
,const ParaAlgoControle & pa)
{
int ni; // compteur globale de point d'integration
Deformation* definter=NULL; // variable de travail transitoire
Vecteur* SM_P = NULL; Mat_pleine* KM_P = NULL;
if (!(ParaGlob::AxiSymetrie()))
{ definter = defSurf(nSurf); // le cas normal est non axisymétrique
SM_P = ((*res_extS)(nSurf));
KM_P = ((*raid_extS)(nSurf));
}
else { definter = defArete(nSurf); // en axisymétrie, c'est une def d'arête
SM_P = ((*res_extA)(nSurf));
KM_P = ((*raid_extA)(nSurf));
};
Deformation & defS = *definter; // pour simplifier l'écriture
Vecteur& SM = *SM_P; // " " "
Mat_pleine & KM = *KM_P; // " " "
// éventuellement dimensionnement de la sauvegarde
if (lesChargeExtSurEle == NULL)
lesChargeExtSurEle = new LesChargeExtSurElement;
if (lesChargeExtSurEle->LesPressionsExternes() == NULL)
{if(ParaGlob::AxiSymetrie())
{lesChargeExtSurEle->LesPressionsExternes_Change_taille(ElementGeometrique().NbSe());}
else
{lesChargeExtSurEle->LesPressionsExternes_Change_taille(ElementGeometrique().NbFe());}
};
Tableau <Pression_appliquee>& tab_Press= (*lesChargeExtSurEle->LesPressionsExternes())(nSurf); // pour simplifier
if (tab_Press.Taille() != defS.Phi1_Taille())
tab_Press.Change_taille(defS.Phi1_Taille());
int nbddl = ddlS.NbDdl();
// controle
bool avec_raid = pa.Var_charge_externe();
// // dans le cas où la variation sur la raideur est demandé on s'assure que la variation
// // du jacobien est bien effective sinon on l'impose
// ParaAlgoControle pa_nevez(pa);
// if (!pa_nevez.Var_jacobien()) pa_nevez.Modif_Var_jacobien(true);
bool premier_calcul=true; // contrairement à la déformation, pas de sauvegarde
// donc il faut calculer tous les éléments de la métrique
for (defS.PremierPtInteg(), ni = 1;defS.DernierPtInteg();defS.NevezPtInteg(),ni++)
{
// calcul des éléments de la métrique, entre autre le jacobien, on utilise le même calcul
// que pour un calcul primaire en explicit mais pour un calcul autre que mécanique
const Met_abstraite::Impli& ex = defS.Cal_implicit(premier_calcul);
const Met_abstraite::Impli* ex_impli = &ex;
const Met_abstraite::Expli_t_tdt* ex_expli_tdt = NULL;
const Met_abstraite::Expli* ex_expli = NULL;
// détermination de la normale à la surface
#ifdef MISE_AU_POINT
// test pour savoir si l'on a réellement affaire à une surface
if ((ex.giB_tdt->NbVecteur() != 2) || ( ParaGlob::Dimension() != 3))
{ cout << "\n erreur, il doit y avoir deux vecteurs pour la surface de dimension 3"
<< " ElemMeca::SM_charge_pres (DdlElement & ...nbvec= " << ex.giB_tdt->NbVecteur()
<< " dim = " << ParaGlob::Dimension();
Sortie(1);
}
#endif
// la normale vaut le produit vectoriel des 2 premiers vecteurs
CoordonneeB normale = Util::ProdVec_coorB( (*ex.giB_tdt)(1), (*ex.giB_tdt)(2));
// calcul de la variation de la normale
// 1) variation du produit vectoriel
Tableau <CoordonneeB > D_pasnormale =
Util::VarProdVect_coorB( (*ex.giB_tdt)(1),(*ex.giB_tdt)(2),(*ex.d_giB_tdt));
// 2) de la normale
Tableau <CoordonneeB> D_normale = Util::VarUnVect_coorB(normale,D_pasnormale,normale.Norme());
// dans le cas où il y a une fonction nD, et qu'un de ces arguments
// est des coordonnées Xi on l'utilise
double pression_applique = pression;
if (pt_fonct != NULL)
{ // ici on utilise les variables connues aux noeuds, ou calculées à partir de
// on commence par récupérer les conteneurs des grandeurs à fournir
List_io <Ddl_enum_etendu>& li_enu_scal = pt_fonct->Li_enu_etendu_scalaire();
List_io <TypeQuelconque >& li_quelc = pt_fonct->Li_equi_Quel_evolue();
bool absolue = true; // on se place systématiquement en absolu
// on va utiliser la méhode Valeur_multi_interpoler_ou_calculer
// pour les grandeurs strictement scalaire
Tableau <double> val_ddl_enum(Valeur_multi_interpoler_ou_calculer
(absolue,TEMPS_tdt,li_enu_scal,defS,ex_impli,ex_expli_tdt,ex_expli)
);
// on utilise la méthode Valeurs_Tensorielles_interpoler_ou_calculer
// pour les Coordonnees et Tenseur
Valeurs_Tensorielles_interpoler_ou_calculer
(absolue,TEMPS_tdt,li_quelc,defS,ex_impli,ex_expli_tdt,ex_expli);
// //---debug
// cout << "\n debug ElemMeca::SMR_charge_pres_I ";
// cout << li_quelc ;
// //-- fin debug
//
// calcul de la valeur et retour dans tab_ret
Tableau <double> & tab_val = pt_fonct->Valeur_FnD_Evoluee(&val_ddl_enum,&li_enu_scal,&li_quelc,NULL,NULL);
// if (pt_fonct->Depend_M())
// {// calcul de la pression au niveau de la position du point d'intégration
// // 1) on récupère la position dans I_a du point d'intégration
// const Coordonnee & M = defS.Position_tdt();
// // 2) on appelle la fonction, qui ne doit dépendre que de M en var locales
// Tableau <Coordonnee> tab(1); tab(1)=M;
// Tableau <double> tab_val = pt_fonct->Val_FnD_Evoluee(NULL,&tab,NULL);
// seule la première valeur est ok
pression_applique *= tab_val(1);
// }
// else
// {cout << "\n *** erreur dans l'utilisation d'une fct nD pour un chargement"
// << " surfacique: dependance autre que du point "<< endl;
// Sortie(1);
// };
};
//*** important: pour l'instant on ne calcule pas la variation de la
// pression causée par la variation du point ==> ce qui est différent
// du calcul hydrostatique par exemple. Du coup, la convergence est
// sans doute un peu moins bonne pour des grands pas de temps ou des
// grandes variation de positions
// calcul de la normale normée et pondérée de la pression
// la pression est positive qu'en elle appuis (diminution de volume)
normale.Normer();normale *= -pression_applique;
tab_Press(ni).P += normale.Coor_const();
// également pondération de la variation de la
for (int ihi =1;ihi<= nbddl;ihi++)
D_normale(ihi) *= -pression_applique;
// sauvegarde de la pression appliquée
tab_Press(ni).press += pression_applique;
int ix=1; int dimf = 3; // ne fonctionne qu'en dim 3
if(ParaGlob::AxiSymetrie())
// cas d'un chargement axisymétrique, dans ce cas on ne prend en compte que les
// dimf-1 coordonnées donc on décrémente
dimf--;
for (int ne =1; ne<= nbne; ne++)
for (int i=1;i<= dimf;i++,ix++)
{ SM(ix) += taphi(ni)(ne)* normale(i) * (poids(ni) * (*ex.jacobien_tdt));
// dans le cas avec_raideur on calcul la contribution à la raideur
if (avec_raid)
for (int j =1; j<= nbddl; j++)
KM(ix,j) -=
taphi(ni)(ne)* normale(i) * (poids(ni) * (*ex.d_jacobien_tdt)(j))
+ taphi(ni)(ne)* D_normale(j)(i) * (poids(ni) * (*ex.jacobien_tdt)) ;
}
}
////-------- debug
//{cout << "\n debug: ElemMeca::SMR_charge_pres_I(..";
// cout << "\n effort externe de pression: Raideur et second membre : elem= " << Num_elt_const()
// << ", maillage= " << Num_maillage();
// cout << "\n raideur: ";
// KM.Affiche();
// cout << "\n second membre: " << SM;
//};
//
//
////------- fin debug
// liberation des tenseurs intermediaires
LibereTenseur();
Element::ResRaid el;
el.res = SM_P;
el.raid = KM_P;
return el;
};
// calcul des seconds membres suivant les chargements
// cas d'un chargement lineique, sur les aretes frontières des éléments
// force indique la force lineique appliquée
// retourne le second membre résultant
// nArete : le numéro de la surface externe
// calcul à l'instant tdt ou t en fonction de la variable atdt
Vecteur& ElemMeca::SM_charge_line_E (DdlElement & ,int nArete
,const Tableau <Vecteur>& taphi,int nbne
,const Vecteur& poids,const Coordonnee& force,Fonction_nD* pt_fonct
,const ParaAlgoControle & ,bool atdt)
{ // définition du vecteur de retour
Vecteur& SM = *((*res_extA)(nArete));
int ni; // compteur globale de point d'integration
Deformation & defA = *defArete(nArete); // pour simplifier l'écriture
// éventuellement dimensionnement de la sauvegarde
if (lesChargeExtSurEle == NULL)
lesChargeExtSurEle = new LesChargeExtSurElement;
if (lesChargeExtSurEle->LesLineique() == NULL)
{if(ParaGlob::AxiSymetrie())
{lesChargeExtSurEle->LesLineique_Change_taille(ElementGeometrique().Nbne());}
else
{lesChargeExtSurEle->LesLineique_Change_taille(ElementGeometrique().NbSe());}
};
Tableau <Coordonnee>& tab_line= (*lesChargeExtSurEle->LesLineique())(nArete); // pour simplifier
if (tab_line.Taille() != defA.Phi1_Taille())
tab_line.Change_taille(defA.Phi1_Taille(),Coordonnee(ParaGlob::Dimension()));
for (defA.PremierPtInteg(), ni = 1;defA.DernierPtInteg();defA.NevezPtInteg(),ni++)
{
// calcul des éléments de la métrique, entre autre le jacobien, on utilise le même calcul
// que pour un calcul primaire en explicit mais pour un calcul autre que mécanique
Met_abstraite::Expli ex;
bool premier_calcul=true; // contrairement à la déformation, pas de sauvegarde
// donc il faut calculer tous les éléments de la métrique
if (atdt)
ex = (defA.Cal_explicit_tdt(premier_calcul)).Tdt_dans_t();
else
ex = defA.Cal_explicit_t(premier_calcul);
const Met_abstraite::Impli* ex_impli = NULL;
const Met_abstraite::Expli_t_tdt* ex_expli_tdt = NULL;
const Met_abstraite::Expli* ex_expli = &ex;
int ix=1; int dimf = force.Dimension();
if(ParaGlob::AxiSymetrie())
// cas d'un chargement axisymétrique, dans ce cas on ne prend en compte que les
// dimf-1 coordonnées donc on décrémente
dimf--;
// prise en compte d'une dépendance à une fonction nD
Coordonnee force_reelle(force); // init par défaut
if (pt_fonct != NULL)
{ // ici on utilise les variables connues aux noeuds, ou calculées à partir de
// on commence par récupérer les conteneurs des grandeurs à fournir
List_io <Ddl_enum_etendu>& li_enu_scal = pt_fonct->Li_enu_etendu_scalaire();
List_io <TypeQuelconque >& li_quelc = pt_fonct->Li_equi_Quel_evolue();
bool absolue = true; // on se place systématiquement en absolu
// on va utiliser la méhode Valeur_multi_interpoler_ou_calculer
// pour les grandeurs strictement scalaire
Tableau <double> val_ddl_enum(Valeur_multi_interpoler_ou_calculer
(absolue,TEMPS_tdt,li_enu_scal,defA,ex_impli,ex_expli_tdt,ex_expli)
);
// on utilise la méthode Valeurs_Tensorielles_interpoler_ou_calculer
// pour les Coordonnees et Tenseur
Valeurs_Tensorielles_interpoler_ou_calculer
(absolue,TEMPS_tdt,li_quelc,defA,ex_impli,ex_expli_tdt,ex_expli);
// calcul de la valeur et retour dans tab_ret
try
{ Tableau <double> & tab_val =
pt_fonct->Valeur_FnD_Evoluee(&val_ddl_enum,&li_enu_scal,&li_quelc,NULL,NULL);
// deux cas: soit une fonction scalaire soit dimf composantes
if (tab_val.Taille() == 1)
{force_reelle *= tab_val(1);}
else if (tab_val.Taille() == dimf)
{switch (dimf)
{case 3:force_reelle(3) *= tab_val(3);
case 2:force_reelle(2) *= tab_val(2);
case 1:force_reelle(1) *= tab_val(1);
break;
default: cout << "\n *** erreur ElemMeca::SM_charge_line_E(.. ";
ErrCalculFct_nD toto; throw (toto);
Sortie(1);
};
}
else
{cout << "\n *** erreur dans l'utilisation d'une fct nD pour un chargement"
<< " lineique: la dimension de la fct nD "<<tab_val.Taille()
<< " devrait etre soit 1, soit "<<dimf << endl;
ErrCalculFct_nD toto; throw (toto);
Sortie(1);
};
}
catch (ErrCalculFct_nD excep)
{ if (ParaGlob::NiveauImpression() > 0)
cout << "\n *** erreur ElemMeca::SM_charge_line_E(.. "
<< " pb dans le calcul de la fonction nD associee ";
if (ParaGlob::param->ParaAlgoControleActifs().Cas_fctnD_charge() == 0)
{ErrCalculFct_nD toto; throw (toto);}
else // on neutralise la force
force_reelle *= 0.;
};
};
tab_line(ni)=force_reelle; // sauvegarde
for (int ne =1; ne<= nbne; ne++)
for (int i=1;i<= dimf;i++,ix++)
SM(ix) += taphi(ni)(ne)* force_reelle(i) * (poids(ni) * (*ex.jacobien_t));
}
// retour du second membre
return SM;
};
// cas d'un chargement lineique, sur les arêtes frontières des éléments
// force indique la force lineique appliquée
// retourne le second membre résultant
// nArete : le numéro de l'arête externe -> implicite,
// pa: permet de déterminer si oui ou non on calcul la contribution à la raideur
Element::ResRaid ElemMeca::SMR_charge_line_I(DdlElement & ddlA,int nArete
,const Tableau <Vecteur>& taphi,int nbne
,const Vecteur& poids,const Coordonnee& force,Fonction_nD* pt_fonct
,const ParaAlgoControle & pa)
{ int ni; // compteur globale de point d'integration
Deformation & defA = *defArete(nArete); // pour simplifier l'écriture
int nbddl = ddlA.NbDdl();
Vecteur& SM = (*((*res_extA)(nArete))); // " " "
Mat_pleine & KM = (*((*raid_extA)(nArete))); // " " "
// controle
bool avec_raid = pa.Var_charge_externe();
// éventuellement dimensionnement de la sauvegarde
if (lesChargeExtSurEle == NULL)
lesChargeExtSurEle = new LesChargeExtSurElement;
if (lesChargeExtSurEle->LesLineique() == NULL)
{if(ParaGlob::AxiSymetrie())
{lesChargeExtSurEle->LesLineique_Change_taille(ElementGeometrique().Nbne());}
else
{lesChargeExtSurEle->LesLineique_Change_taille(ElementGeometrique().NbSe());}
};
Tableau <Coordonnee>& tab_line= (*lesChargeExtSurEle->LesLineique())(nArete); // pour simplifier
if (tab_line.Taille() != defA.Phi1_Taille())
tab_line.Change_taille(defA.Phi1_Taille(),Coordonnee(ParaGlob::Dimension()));
// // dans le cas où la variation sur la raideur est demandé on s'assure que la variation
// // du jacobien est bien effective sinon on l'impose
// ParaAlgoControle pa_nevez(pa);
// if (!pa_nevez.Var_jacobien()) pa_nevez.Modif_Var_jacobien(true);
bool premier_calcul=true; // contrairement à la déformation, pas de sauvegarde
// donc il faut calculer tous les éléments de la métrique
for (defA.PremierPtInteg(), ni = 1;defA.DernierPtInteg();defA.NevezPtInteg(),ni++)
{
// calcul des éléments de la métrique, entre autre le jacobien, on utilise le même calcul
// que pour un calcul primaire en implicit
const Met_abstraite::Impli& ex = defA.Cal_implicit(premier_calcul);
const Met_abstraite::Impli* ex_impli = &ex;
const Met_abstraite::Expli_t_tdt* ex_expli_tdt = NULL;
const Met_abstraite::Expli* ex_expli = NULL;
int ix=1; int dimf = force.Dimension();
if(ParaGlob::AxiSymetrie())
// cas d'un chargement axisymétrique, dans ce cas on ne prend en compte que les
// dimf-1 coordonnées donc on décrémente
dimf--;
// prise en compte d'une dépendance à une fonction nD
// on ne tient pas compte de la variation du point dans la fct nD
Coordonnee force_reelle(force); // init par défaut
if (pt_fonct != NULL)
{ // ici on utilise les variables connues aux noeuds, ou calculées à partir de
// on commence par récupérer les conteneurs des grandeurs à fournir
List_io <Ddl_enum_etendu>& li_enu_scal = pt_fonct->Li_enu_etendu_scalaire();
List_io <TypeQuelconque >& li_quelc = pt_fonct->Li_equi_Quel_evolue();
bool absolue = true; // on se place systématiquement en absolu
// on va utiliser la méhode Valeur_multi_interpoler_ou_calculer
// pour les grandeurs strictement scalaire
Tableau <double> val_ddl_enum(Valeur_multi_interpoler_ou_calculer
(absolue,TEMPS_tdt,li_enu_scal,defA,ex_impli,ex_expli_tdt,ex_expli)
);
// on utilise la méthode Valeurs_Tensorielles_interpoler_ou_calculer
// pour les Coordonnees et Tenseur
Valeurs_Tensorielles_interpoler_ou_calculer
(absolue,TEMPS_tdt,li_quelc,defA,ex_impli,ex_expli_tdt,ex_expli);
// calcul de la valeur et retour dans tab_ret
Tableau <double> & tab_val = pt_fonct->Valeur_FnD_Evoluee(&val_ddl_enum,&li_enu_scal,&li_quelc,NULL,NULL);
// {if (pt_fonct->Depend_M())
// {// 1) on récupère la position dans I_a du point d'intégration
// const Coordonnee & M = defA.Position_tdt();
// // 2) on appel la fonction, qui ne doit dépendre que de M en var locales
// Tableau <Coordonnee> tab(1); tab(1)=M;
// Tableau <double> tab_val = pt_fonct->Val_FnD_Evoluee(NULL,&tab,NULL);
// deux cas: soit une fonction scalaire soit dimf composantes
if (tab_val.Taille() == 1)
{force_reelle *= tab_val(1);}
else if (tab_val.Taille() == dimf)
{switch (dimf)
{case 3:force_reelle(3) *= tab_val(3);
case 2:force_reelle(2) *= tab_val(2);
case 1:force_reelle(1) *= tab_val(1);
break;
default: cout << "\n *** erreur ElemMeca::SM_charge_line_I(.. ";
Sortie(1);
};
}
else
{cout << "\n *** erreur dans l'utilisation d'une fct nD pour un chargement"
<< " lineique: la dimension de la fct nD "<<tab_val.Taille()
<< " devrait etre soit 1, soit "<<dimf << endl;
Sortie(1);
};
// }
// else
// {cout << "\n *** erreur dans l'utilisation d'une fct nD pour un chargement"
// << " lineique: dependance autre que du point "<< endl;
// Sortie(1);
// };
};
tab_line(ni)=force_reelle; // sauvegarde
for (int ne =1; ne<= nbne; ne++)
for (int i=1;i<= dimf;i++,ix++)
{ SM(ix) +=
taphi(ni)(ne)* force_reelle(i) * (poids(ni) * (*ex.jacobien_tdt));
// dans le cas avec_raideur on calcul la contribution à la raideur
if (avec_raid)
for (int j =1; j<= nbddl; j++)
KM(ix,j) -=
taphi(ni)(ne)* force_reelle(i) * (poids(ni) * (*ex.d_jacobien_tdt)(j));
}
}
// liberation des tenseurs intermediaires
LibereTenseur();
Element::ResRaid el;
el.res = (*res_extA)(nArete);
el.raid = (*raid_extA)(nArete);
return el;
};
// cas d'un chargement lineique suiveur, sur les aretes frontières des éléments
// force indique la force lineique appliquée
// retourne le second membre résultant
// nArete : le numéro de la surface externe
// calcul à l'instant tdt ou t en fonction de la variable atdt
Vecteur& ElemMeca::SM_charge_line_Suiv_E (DdlElement & ,int nArete
,const Tableau <Vecteur>& taphi,int nbne
,const Vecteur& poids,const Coordonnee& force,Fonction_nD* pt_fonct
,const ParaAlgoControle & ,bool atdt)
{ // définition du vecteur de retour
Vecteur& SM = *((*res_extA)(nArete));
int ni; // compteur globale de point d'integration
Deformation & defA = *defArete(nArete); // pour simplifier l'écriture
// éventuellement dimensionnement de la sauvegarde
if (lesChargeExtSurEle == NULL)
lesChargeExtSurEle = new LesChargeExtSurElement;
if (lesChargeExtSurEle->LesLineiqueSuiveuse() == NULL)
{if(ParaGlob::AxiSymetrie())
{lesChargeExtSurEle->LesLineiqueSuiveuse_Change_taille(ElementGeometrique().Nbne());}
else
{lesChargeExtSurEle->LesLineiqueSuiveuse_Change_taille(ElementGeometrique().NbSe());}
};
Tableau <Coordonnee>& tab_line= (*lesChargeExtSurEle->LesLineiqueSuiveuse())(nArete); // pour simplifier
if (tab_line.Taille() != defA.Phi1_Taille())
tab_line.Change_taille(defA.Phi1_Taille(),Coordonnee(ParaGlob::Dimension()));
bool premier_calcul=true; // contrairement à la déformation, pas de sauvegarde
// donc il faut calculer tous les éléments de la métrique
for (defA.PremierPtInteg(), ni = 1;defA.DernierPtInteg();defA.NevezPtInteg(),ni++)
{
// calcul des éléments de la métrique, entre autre le jacobien, on utilise le même calcul
// que pour un calcul primaire en explicit mais pour un calcul autre que mécanique
Met_abstraite::Expli ex;
if (atdt)
ex = (defA.Cal_explicit_tdt(premier_calcul)).Tdt_dans_t();
else
ex = defA.Cal_explicit_t(premier_calcul);
const Met_abstraite::Impli* ex_impli = NULL;
const Met_abstraite::Expli_t_tdt* ex_expli_tdt = NULL;
const Met_abstraite::Expli* ex_expli = &ex;
#ifdef MISE_AU_POINT
// test pour savoir si l'on est en dimension 2 ou en axy sinon pas possible
if ((ParaGlob::Dimension() != 2) & (!(ParaGlob::AxiSymetrie())))
{ cout << "\n erreur, pour l'instant seul la dimension 2 est permise avec les forces suiveuses"
<< "\n si pb contacter gerard Rio ! "
<< " ElemMeca::SM_charge_line_Suiv_E ( ..."
<< " \n dim = " << ParaGlob::Dimension();
Sortie(1);
}
#endif
// calcul du repère locale initiale orthonormée
// Coordonnee2 v1((*ex.giB_0)(1).Vect()); v1.Normer();
// on récupère que les deux premières coordonnées mais on travaille en 3D pour intégrer le cas axi
Coordonnee v1((*ex.giB_0).Coordo(1)); v1.Normer();
int dim = v1.Dimension();
Coordonnee v2(dim); v2(1)=-v1(2); v2(2)=v1(1);
// composante du vecteur force linéique dans le repère initiale
Coordonnee force_0(v1*force,v2*force);
//repère locale actuel orthonormée
Coordonnee giB_t = (*ex.giB_t)(1).Coor();
Coordonnee vf1(giB_t);
Coordonnee vf2(dim);vf2(1)=-vf1(2);vf2(2)=vf1(1);
// composantes actuelles de la force
Coordonnee force_t = force_0(1)*vf1 + force_0(2)*vf2;
// calcul du résidu
int ix=1; int dimf = 2 ; // ici uniquement en dim 2 sinon pb
// if(ParaGlob::AxiSymetrie())
// // cas d'un chargement axisymétrique, dans ce cas on ne prend en compte que les
// // dimf-1 coordonnées donc on décrémente
// dimf--;
// prise en compte d'une dépendance à une fonction nD
if (pt_fonct != NULL)
{ // ici on utilise les variables connues aux noeuds, ou calculées à partir de
// on commence par récupérer les conteneurs des grandeurs à fournir
List_io <Ddl_enum_etendu>& li_enu_scal = pt_fonct->Li_enu_etendu_scalaire();
List_io <TypeQuelconque >& li_quelc = pt_fonct->Li_equi_Quel_evolue();
bool absolue = true; // on se place systématiquement en absolu
// on va utiliser la méhode Valeur_multi_interpoler_ou_calculer
// pour les grandeurs strictement scalaire
Tableau <double> val_ddl_enum(Valeur_multi_interpoler_ou_calculer
(absolue,TEMPS_tdt,li_enu_scal,defA,ex_impli,ex_expli_tdt,ex_expli)
);
// on utilise la méthode Valeurs_Tensorielles_interpoler_ou_calculer
// pour les Coordonnees et Tenseur
Valeurs_Tensorielles_interpoler_ou_calculer
(absolue,TEMPS_tdt,li_quelc,defA,ex_impli,ex_expli_tdt,ex_expli);
// calcul de la valeur et retour dans tab_ret
Tableau <double> & tab_val = pt_fonct->Valeur_FnD_Evoluee(&val_ddl_enum,&li_enu_scal,&li_quelc,NULL,NULL);
// {if (pt_fonct->Depend_M())
// {// 1) on récupère la position dans I_a du point d'intégration
// const Coordonnee & M = defA.Position_tdt();
// // 2) on appel la fonction, qui ne doit dépendre que de M en var locales
// Tableau <Coordonnee> tab(1); tab(1)=M;
// Tableau <double> tab_val = pt_fonct->Val_FnD_Evoluee(NULL,&tab,NULL);
// deux cas: soit une fonction scalaire soit dimf composantes
if (tab_val.Taille() == 1)
{force_t *= tab_val(1);}
else if (tab_val.Taille() == dimf)
{switch (dimf)
{case 3:force_t(3) *= tab_val(3);
case 2:force_t(2) *= tab_val(2);
case 1:force_t(1) *= tab_val(1);
break;
default: cout << "\n *** erreur ElemMeca::SM_charge_line_E(.. ";
Sortie(1);
};
}
else
{cout << "\n *** erreur dans l'utilisation d'une fct nD pour un chargement"
<< " lineique suiveuse: la dimension de la fct nD "<<tab_val.Taille()
<< " devrait etre soit 1, soit "<<dimf << endl;
Sortie(1);
};
// }
// else
// {cout << "\n *** erreur dans l'utilisation d'une fct nD pour un chargement"
// << " lineique: dependance autre que du point "<< endl;
// Sortie(1);
// };
};
tab_line(ni)=force_t; // sauvegarde
for (int ne =1; ne<= nbne; ne++)
for (int i=1;i<= dimf;i++,ix++)
SM(ix) += taphi(ni)(ne)* force_t(i) * (poids(ni) * (*ex.jacobien_t));
}
// retour du second membre
return SM;
};
// cas d'un chargement lineique suiveur, sur les arêtes frontières des éléments
// force indique la force lineique appliquée
// retourne le second membre résultant
// nArete : le numéro de l'arête externe -> implicite,
// pa: permet de déterminer si oui ou non on calcul la contribution à la raideur
Element::ResRaid ElemMeca::SMR_charge_line_Suiv_I(DdlElement & ddlA,int nArete
,const Tableau <Vecteur>& taphi,int nbne
,const Vecteur& poids,const Coordonnee& force,Fonction_nD* pt_fonct
,const ParaAlgoControle & pa)
{ int ni; // compteur globale de point d'integration
Deformation & defA = *defArete(nArete); // pour simplifier l'écriture
int nbddl = ddlA.NbDdl();
Vecteur& SM = (*((*res_extA)(nArete))); // " " "
Mat_pleine & KM = (*((*raid_extA)(nArete))); // " " "
// controle
bool avec_raid = pa.Var_charge_externe();
// éventuellement dimensionnement de la sauvegarde
if (lesChargeExtSurEle == NULL)
lesChargeExtSurEle = new LesChargeExtSurElement;
if (lesChargeExtSurEle->LesLineiqueSuiveuse() == NULL)
{if(ParaGlob::AxiSymetrie())
{lesChargeExtSurEle->LesLineiqueSuiveuse_Change_taille(ElementGeometrique().Nbne());}
else
{lesChargeExtSurEle->LesLineiqueSuiveuse_Change_taille(ElementGeometrique().NbSe());}
};
Tableau <Coordonnee>& tab_line= (*lesChargeExtSurEle->LesLineiqueSuiveuse())(nArete); // pour simplifier
if (tab_line.Taille() != defA.Phi1_Taille())
tab_line.Change_taille(defA.Phi1_Taille(),Coordonnee(ParaGlob::Dimension()));
// // dans le cas où la variation sur la raideur est demandé on s'assure que la variation
// // du jacobien est bien effective sinon on l'impose
// ParaAlgoControle pa_nevez(pa);
// if (!pa_nevez.Var_jacobien()) pa_nevez.Modif_Var_jacobien(true);
bool premier_calcul=true; // contrairement à la déformation, pas de sauvegarde
// donc il faut calculer tous les éléments de la métrique
for (defA.PremierPtInteg(), ni = 1;defA.DernierPtInteg();defA.NevezPtInteg(),ni++)
{
// calcul des éléments de la métrique, entre autre le jacobien, on utilise le même calcul
// que pour un calcul primaire en implicit
const Met_abstraite::Impli& ex = defA.Cal_implicit(premier_calcul);
const Met_abstraite::Impli* ex_impli = &ex;
const Met_abstraite::Expli_t_tdt* ex_expli_tdt = NULL;
const Met_abstraite::Expli* ex_expli = NULL;
#ifdef MISE_AU_POINT
// test pour savoir si l'on est en dimension 2 ou en axy sinon pas possible
if ((ParaGlob::Dimension() != 2) & (!(ParaGlob::AxiSymetrie())))
{ cout << "\n erreur, pour l'instant seul la dimension 2 est permise avec les forces suiveuses"
<< "\n si pb contacter gerard Rio ! "
<< " ElemMeca::SM_charge_line_Suiv_E ( ..."
<< " \n dim = " << ParaGlob::Dimension();
Sortie(1);
}
#endif
// calcul du repère locale initiale orthonormée
// Coordonnee2 v1((*ex.giB_0)(1).Vect()); v1.Normer();
// on récupère que les deux premières coordonnées mais on travaille en 3D pour intégrer le cas axi
Coordonnee v1((*ex.giB_0).Coordo(1)); v1.Normer();
int dim = v1.Dimension();
Coordonnee v2(dim); v2(1)=-v1(2); v2(2)=v1(1);
// composante du vecteur force linéique dans le repère initiale
Coordonnee force_0(v1*force,v2*force);
//repère locale actuel orthonormée
Coordonnee giB_tdt = (*ex.giB_tdt)(1).Coor();
Coordonnee vf1(giB_tdt);
double vf1Norme = vf1.Norme();vf1 /= vf1Norme;
Coordonnee vf2(dim);vf2(1)=-vf1(2);vf2(2)=vf1(1);
// composantes actuelles de la force
Coordonnee force_t = force_0(1)*vf1 + force_0(2)*vf2;
// calcul du résidu
int ix=1; int dimf = 2 ; // ici uniquement en dim 2 sinon pb
// prise en compte d'une dépendance à une fonction nD
// on ne tient pas compte de la variation du point dans la fct nD
if (pt_fonct != NULL)
{ // ici on utilise les variables connues aux noeuds, ou calculées à partir de
// on commence par récupérer les conteneurs des grandeurs à fournir
List_io <Ddl_enum_etendu>& li_enu_scal = pt_fonct->Li_enu_etendu_scalaire();
List_io <TypeQuelconque >& li_quelc = pt_fonct->Li_equi_Quel_evolue();
bool absolue = true; // on se place systématiquement en absolu
// on va utiliser la méhode Valeur_multi_interpoler_ou_calculer
// pour les grandeurs strictement scalaire
Tableau <double> val_ddl_enum(Valeur_multi_interpoler_ou_calculer
(absolue,TEMPS_tdt,li_enu_scal,defA,ex_impli,ex_expli_tdt,ex_expli)
);
// on utilise la méthode Valeurs_Tensorielles_interpoler_ou_calculer
// pour les Coordonnees et Tenseur
Valeurs_Tensorielles_interpoler_ou_calculer
(absolue,TEMPS_tdt,li_quelc,defA,ex_impli,ex_expli_tdt,ex_expli);
// calcul de la valeur et retour dans tab_ret
Tableau <double> & tab_val = pt_fonct->Valeur_FnD_Evoluee(&val_ddl_enum,&li_enu_scal,&li_quelc,NULL,NULL);
// {if (pt_fonct->Depend_M())
// {// 1) on récupère la position dans I_a du point d'intégration
// const Coordonnee & M = defA.Position_tdt();
// // 2) on appel la fonction, qui ne doit dépendre que de M en var locales
// Tableau <Coordonnee> tab(1); tab(1)=M;
// Tableau <double> tab_val = pt_fonct->Val_FnD_Evoluee(NULL,&tab,NULL);
// deux cas: soit une fonction scalaire soit dimf composantes
if (tab_val.Taille() == 1)
{force_t *= tab_val(1);}
else if (tab_val.Taille() == dimf)
{switch (dimf)
{case 3:force_t(3) *= tab_val(3);
case 2:force_t(2) *= tab_val(2);
case 1:force_t(1) *= tab_val(1);
break;
default: cout << "\n *** erreur ElemMeca::SM_charge_line_E(.. ";
Sortie(1);
};
}
else
{cout << "\n *** erreur dans l'utilisation d'une fct nD pour un chargement"
<< " lineique suiveuse: la dimension de la fct nD "<<tab_val.Taille()
<< " devrait etre soit 1, soit "<<dimf << endl;
Sortie(1);
};
// }
// else
// {cout << "\n *** erreur dans l'utilisation d'une fct nD pour un chargement"
// << " lineique: dependance autre que du point "<< endl;
// Sortie(1);
// };
};
tab_line(ni)=force_t; // sauvegarde
Coordonnee dvf1,dvf2,d_giB_tdt; // def de vecteurs intermediaires
for (int ne =1; ne<= nbne; ne++)
for (int i=1;i<= dimf;i++,ix++)
{ SM(ix) += taphi(ni)(ne)* force_t(i) * (poids(ni) * (*ex.jacobien_tdt));
// dans le cas avec_raideur on calcul la contribution à la raideur
if (avec_raid)
for (int j =1; j<= nbddl; j++)
{ // calcul des variations de la force
d_giB_tdt = ((((*ex.d_giB_tdt)(j))(1)).Coor());
dvf1 = Util::VarUnVect_coor // variation de vf1 par rapport au ddl j
(giB_tdt,d_giB_tdt,vf1Norme);
dvf2 = -(dvf1 * vf2) * vf1;
KM(ix,j) -= taphi(ni)(ne) * poids(ni)
* (force_t(i) *(*ex.d_jacobien_tdt)(j)
+ (*ex.jacobien_tdt) * ((force_0(1) * dvf1 + force_0(2) * dvf2))(i)
);
}
}
}
// liberation des tenseurs intermediaires
LibereTenseur();
Element::ResRaid el;
el.res = (*res_extA)(nArete);
el.raid = (*raid_extA)(nArete);
return el;
};
// cas d'un chargement surfacique suiveur, sur les surfaces de l'élément
// la direction varie selon le système suivant: on définit les coordonnées matérielles
// de la direction, ce qui sert ensuite à calculer les nouvelles directions. L'intensité
// elle est constante.
// force indique la force surfacique appliquée
// retourne le second membre résultant
// nSurf : le numéro de la surface externe
// calcul à l'instant tdt ou t en fonction de la variable atdt
Vecteur& ElemMeca::SM_charge_surf_Suiv_E (DdlElement & ,int nSurf
,const Tableau <Vecteur>& taphi,int nbne
,const Vecteur& poids,const Coordonnee& forc,Fonction_nD* pt_fonct
,const ParaAlgoControle & ,bool atdt)
{ // *** dans l'ensemble du sp on utilise des vecteurs et on gére directement la variance
// *** sinon c'est intractable et long
// définition du vecteur de retour
Vecteur& SM = *((*res_extS)(nSurf));
double module_f = forc.Norme();
// dans le cas ou la force est de module nul, on retourne sans calcul
if (module_f < ConstMath::trespetit)
return SM;
#ifdef MISE_AU_POINT
// test : a priori non valide pour les éléments axi et pour un pb autre que 3D
if (( ParaGlob::Dimension() != 3) || ((ParaGlob::Dimension() == 3)&&(ParaGlob::AxiSymetrie())))
{ cout << "\n erreur, ce type de chargement n'est valide que pour la dimension 3 !! et non axi"
<< " ElemMeca::SM_charge_line_Suiv_E ( ..."
<< " \n dim = " << ParaGlob::Dimension();
Sortie(1);
};
#endif
int ni; // compteur globale de point d'integration
Deformation & defS = *defSurf(nSurf); // pour simplifier l'écriture
// éventuellement dimensionnement de la sauvegarde
if (lesChargeExtSurEle == NULL)
lesChargeExtSurEle = new LesChargeExtSurElement;
if (lesChargeExtSurEle->LesPressDir() == NULL)
{if(ParaGlob::AxiSymetrie())
{lesChargeExtSurEle->LesPressDir_Change_taille(ElementGeometrique().NbSe());}
else
{lesChargeExtSurEle->LesPressDir_Change_taille(ElementGeometrique().NbFe());}
};
Tableau <Coordonnee>& tab_Press= (*lesChargeExtSurEle->LesPressDir())(nSurf); // pour simplifier
if (tab_Press.Taille() != defS.Phi1_Taille())
tab_Press.Change_taille(defS.Phi1_Taille(),Coordonnee(ParaGlob::Dimension()));
bool premier_calcul=true; // contrairement à la déformation, pas de sauvegarde
// donc il faut calculer tous les éléments de la métrique
for (defS.PremierPtInteg(), ni = 1;defS.DernierPtInteg();defS.NevezPtInteg(),ni++)
{
// calcul des éléments de la métrique, entre autre le jacobien, on utilise le même calcul
// que pour un calcul primaire en explicit mais pour un calcul autre que mécanique
Met_abstraite::Expli ex;
if (atdt)
ex = (defS.Cal_explicit_tdt(premier_calcul)).Tdt_dans_t();
else
ex = defS.Cal_explicit_t(premier_calcul);
const Met_abstraite::Impli* ex_impli = NULL;
const Met_abstraite::Expli_t_tdt* ex_expli_tdt = NULL;
const Met_abstraite::Expli* ex_expli = &ex;
// détermination de la normale à la surface
// la normale vaut le produit vectoriel des 2 premiers vecteurs
Coordonnee normale_0 = Util::ProdVec_coorBN( (*ex.giB_0)(1), (*ex.giB_0)(2));
// on passe en BN pour dire que c'est du B en normal
const Coordonnee& giBN_t1 = (*ex.giB_t).Coordo(1); const Coordonnee& giBN_t2 = (*ex.giB_t).Coordo(2);
const Coordonnee& giB_01 = (*ex.giB_0).Coordo(1); const Coordonnee& giB_02 = (*ex.giB_0).Coordo(2);
Coordonnee normale_t = Util::ProdVec_coor( giBN_t1, giBN_t2);
// calcul de la normale normée
normale_0.Normer(); normale_t.Normer();
// calcul des composantes de direction de la force dans le repère local initial
Coordonnee dir_forceH_0(3);
// ici le fait d'utiliser la méthode .Vect() supprime la variance, mais justement
// on ne veut pas de vérification de variance qui est ici incohérente
dir_forceH_0(1) = giB_01 * forc; dir_forceH_0(2) = giB_02 * forc;
dir_forceH_0(3) = normale_0 * forc;
// calcul de la direction finale
Coordonnee dir_force_t = dir_forceH_0(1) * giBN_t1 + dir_forceH_0(2) * giBN_t2
+ dir_forceH_0(3) * normale_t;
dir_force_t.Normer();
// calcul de la force finale
Coordonnee force_t = (forc.Norme()) * dir_force_t;
// calcul du second membre
int ix=1; int dimf = 3; // ne fonctionne qu'en dim 3
// sauf pour les éléments axisymétriques pour lesquelles on ne considère que les deux premières composantes
if(ParaGlob::AxiSymetrie())
// cas d'un chargement axisymétrique, dans ce cas on ne prend en compte que les
// dimf-1 coordonnées donc on décrémente
dimf--;
// prise en compte d'une dépendance à une fonction nD
if (pt_fonct != NULL)
{ // ici on utilise les variables connues aux noeuds, ou calculées à partir de
// on commence par récupérer les conteneurs des grandeurs à fournir
List_io <Ddl_enum_etendu>& li_enu_scal = pt_fonct->Li_enu_etendu_scalaire();
List_io <TypeQuelconque >& li_quelc = pt_fonct->Li_equi_Quel_evolue();
bool absolue = true; // on se place systématiquement en absolu
// on va utiliser la méhode Valeur_multi_interpoler_ou_calculer
// pour les grandeurs strictement scalaire
Tableau <double> val_ddl_enum(Valeur_multi_interpoler_ou_calculer
(absolue,TEMPS_tdt,li_enu_scal,defS,ex_impli,ex_expli_tdt,ex_expli)
);
// on utilise la méthode Valeurs_Tensorielles_interpoler_ou_calculer
// pour les Coordonnees et Tenseur
Valeurs_Tensorielles_interpoler_ou_calculer
(absolue,TEMPS_tdt,li_quelc,defS,ex_impli,ex_expli_tdt,ex_expli);
// calcul de la valeur et retour dans tab_ret
Tableau <double> & tab_val = pt_fonct->Valeur_FnD_Evoluee(&val_ddl_enum,&li_enu_scal,&li_quelc,NULL,NULL);
// {if (pt_fonct->Depend_M())
// {// 1) on récupère la position dans I_a du point d'intégration
// const Coordonnee & M = defS.Position_tdt();
// // 2) on appel la fonction, qui ne doit dépendre que de M en var locales
// Tableau <Coordonnee> tab(1); tab(1)=M;
// Tableau <double> tab_val = pt_fonct->Val_FnD_Evoluee(NULL,&tab,NULL);
// deux cas: soit une fonction scalaire soit dimf composantes
if (tab_val.Taille() == 1)
{force_t *= tab_val(1);}
else if (tab_val.Taille() == dimf)
{switch (dimf)
{case 3:force_t(3) *= tab_val(3);
case 2:force_t(2) *= tab_val(2);
case 1:force_t(1) *= tab_val(1);
break;
default: cout << "\n *** erreur ElemMeca::SM_charge_surf_Suiv_E(.. ";
Sortie(1);
};
}
else
{cout << "\n *** erreur dans l'utilisation d'une fct nD pour un chargement"
<< " surfacique suiveuse: la dimension de la fct nD "<<tab_val.Taille()
<< " devrait etre soit 1, soit "<<dimf << endl;
Sortie(1);
};
// }
// else
// {cout << "\n *** erreur dans l'utilisation d'une fct nD pour un chargement"
// << " surfacique: dependance autre que du point "<< endl;
// Sortie(1);
// };
};
tab_Press(ni)=force_t; // sauvegarde de l'effort
for (int ne =1; ne<= nbne; ne++)
for (int i=1;i<= dimf;i++,ix++)
SM(ix) += taphi(ni)(ne)* force_t(i) * (poids(ni) * (*ex.jacobien_t));
}
// retour du second membre
return SM;
};
// idem SM_charge_surf_Suiv_E mais
// -> implicite,
// pa: permet de déterminer si oui ou non on calcul la contribution à la raideur
// retourne le second membre et la matrice de raideur correspondant
Element::ResRaid ElemMeca::SMR_charge_surf_Suiv_I (DdlElement & ddls,int nSurf
,const Tableau <Vecteur>& taphi,int nbne
,const Vecteur& poids,const Coordonnee& forc,Fonction_nD* pt_fonct
,const ParaAlgoControle & pa)
{ // *** dans l'ensemble du sp on utilise des vecteurs et on gére directement la variance
// *** sinon c'est intractable et long
double module_f = forc.Norme();
// dans le cas ou la force est de module nul, on retourne sans calcul
if (module_f < ConstMath::trespetit)
{ Element::ResRaid el;
el.res = (*res_extS)(nSurf);
el.raid = (*raid_extS)(nSurf);
return el;
}
// controle
bool avec_raid = pa.Var_charge_externe();
// // dans le cas où la variation sur la raideur est demandé on s'assure que la variation
// // du jacobien est bien effective sinon on l'impose
// ParaAlgoControle pa_nevez(pa);
// if (!pa_nevez.Var_jacobien()) pa_nevez.Modif_Var_jacobien(true);
int ni; // compteur globale de point d'integration
Deformation & defS = *defSurf(nSurf); // pour simplifier l'écriture
int nbddl = ddls.NbDdl();
Vecteur& SM = (*((*res_extS)(nSurf))); // " " "
Mat_pleine & KM = (*((*raid_extS)(nSurf))); // " " "
// éventuellement dimensionnement de la sauvegarde
if (lesChargeExtSurEle == NULL)
lesChargeExtSurEle = new LesChargeExtSurElement;
if (lesChargeExtSurEle->LesPressDir() == NULL)
{if(ParaGlob::AxiSymetrie())
{lesChargeExtSurEle->LesPressDir_Change_taille(ElementGeometrique().NbSe());}
else
{lesChargeExtSurEle->LesPressDir_Change_taille(ElementGeometrique().NbFe());}
};
Tableau <Coordonnee>& tab_Press= (*lesChargeExtSurEle->LesPressDir())(nSurf); // pour simplifier
if (tab_Press.Taille() != defS.Phi1_Taille())
tab_Press.Change_taille(defS.Phi1_Taille(),Coordonnee(ParaGlob::Dimension()));
bool premier_calcul=true; // contrairement à la déformation, pas de sauvegarde
// donc il faut calculer tous les éléments de la métrique
for (defS.PremierPtInteg(), ni = 1;defS.DernierPtInteg();defS.NevezPtInteg(),ni++)
{
// calcul des éléments de la métrique, entre autre le jacobien, on utilise le même calcul
// que pour un calcul primaire en implicit
const Met_abstraite::Impli& ex = defS.Cal_implicit(premier_calcul);
const Met_abstraite::Impli* ex_impli = &ex;
const Met_abstraite::Expli_t_tdt* ex_expli_tdt = NULL;
const Met_abstraite::Expli* ex_expli = NULL;
#ifdef MISE_AU_POINT
// test : a priori non valide pour les éléments axi et pour un pb autre que 3D
if (( ParaGlob::Dimension() != 3) || ((ParaGlob::Dimension() == 3)&&(ParaGlob::AxiSymetrie())))
{ cout << "\n erreur, ce type de chargement n'est valide pour pour la dimension 3 !! et non axi"
<< " ElemMeca::SM_charge_line_Suiv_E ( ..."
<< " \n dim = " << ParaGlob::Dimension();
Sortie(1);
};
// test pour savoir si l'on a réellement affaire à une surface
if (ex.giB_t->NbVecteur() != 2)
{ cout << "\n erreur, pb de nombre de vecteurs pour la surface !!"
<< " ElemMeca::SM_charge_surf_Suiv_I (.... " << ex.giB_t->NbVecteur()
<< " dim = " << ParaGlob::Dimension();
Sortie(1);
}
#endif
// détermination de la normale à la surface
// la normale vaut le produit vectoriel des 2 premiers vecteurs
Coordonnee normale_0 = Util::ProdVec_coorBN( (*ex.giB_0)(1), (*ex.giB_0)(2));
const Coordonnee& giB_tdt1 = (*ex.giB_tdt).Coordo(1); const Coordonnee& giB_tdt2 = (*ex.giB_tdt).Coordo(2);
const Coordonnee& giB_01 = (*ex.giB_0).Coordo(1); const Coordonnee& giB_02 = (*ex.giB_0).Coordo(2);
Coordonnee normale_t = Util::ProdVec_coor( giB_tdt1, giB_tdt2);
// calcul des normales normées
normale_0.Normer();
double norme_normarle_t = normale_t.Norme();normale_t /= norme_normarle_t;
// calcul des composantes de direction de la force dans le repère local initial
Coordonnee dir_forceH_0(3);
// on ne veut pas de vérification de variance
dir_forceH_0(1) = giB_01 * forc; dir_forceH_0(2) = giB_02 * forc;
dir_forceH_0(3) = normale_0 * forc;
// calcul de la direction finale
Coordonnee dir_f_t = dir_forceH_0(1) * giB_tdt1 + dir_forceH_0(2) * giB_tdt2
+ dir_forceH_0(3) * normale_t;
Coordonnee dir_f_t_normer = dir_f_t/(dir_f_t.Norme());
// calcul de la force finale
Coordonnee force_t = module_f * dir_f_t_normer;
// debug
//cout << "\n debug Element::ResRaid ElemMeca::SMR_charge_surf_Suiv_I "
// << "\n force= "<< force_t << endl;
//fin debug
// calcul du résidu et éventuellement de la raideur
int ix=1; int dimf = forc.Dimension(); // normalement ne fonctionne qu'en 3D
// sauf pour les éléments axisymétriques pour lesquelles on ne considère que les deux premières composantes
if(ParaGlob::AxiSymetrie())
// cas d'un chargement axisymétrique, dans ce cas on ne prend en compte que les
// dimf-1 coordonnées donc on décrémente
dimf--;
// prise en compte d'une dépendance à une fonction nD
// on ne tient pas compte de la variation du point dans la fct nD
if (pt_fonct != NULL)
{ // ici on utilise les variables connues aux noeuds, ou calculées à partir de
// on commence par récupérer les conteneurs des grandeurs à fournir
List_io <Ddl_enum_etendu>& li_enu_scal = pt_fonct->Li_enu_etendu_scalaire();
List_io <TypeQuelconque >& li_quelc = pt_fonct->Li_equi_Quel_evolue();
bool absolue = true; // on se place systématiquement en absolu
// on va utiliser la méhode Valeur_multi_interpoler_ou_calculer
// pour les grandeurs strictement scalaire
Tableau <double> val_ddl_enum(Valeur_multi_interpoler_ou_calculer
(absolue,TEMPS_tdt,li_enu_scal,defS,ex_impli,ex_expli_tdt,ex_expli)
);
// on utilise la méthode Valeurs_Tensorielles_interpoler_ou_calculer
// pour les Coordonnees et Tenseur
Valeurs_Tensorielles_interpoler_ou_calculer
(absolue,TEMPS_tdt,li_quelc,defS,ex_impli,ex_expli_tdt,ex_expli);
// calcul de la valeur et retour dans tab_ret
Tableau <double> & tab_val = pt_fonct->Valeur_FnD_Evoluee(&val_ddl_enum,&li_enu_scal,&li_quelc,NULL,NULL);
// {if (pt_fonct->Depend_M())
// {// 1) on récupère la position dans I_a du point d'intégration
// const Coordonnee & M = defS.Position_tdt();
// // 2) on appel la fonction, qui ne doit dépendre que de M en var locales
// Tableau <Coordonnee> tab(1); tab(1)=M;
// Tableau <double> tab_val = pt_fonct->Val_FnD_Evoluee(NULL,&tab,NULL);
// deux cas: soit une fonction scalaire soit dimf composantes
if (tab_val.Taille() == 1)
{force_t *= tab_val(1);}
else if (tab_val.Taille() == dimf)
{switch (dimf)
{case 3:force_t(3) *= tab_val(3);
case 2:force_t(2) *= tab_val(2);
case 1:force_t(1) *= tab_val(1);
break;
default: cout << "\n *** erreur ElemMeca::SM_charge_surf_Suiv_E(.. ";
Sortie(1);
};
}
else
{cout << "\n *** erreur dans l'utilisation d'une fct nD pour un chargement"
<< " surfacique suiveuse: la dimension de la fct nD "<<tab_val.Taille()
<< " devrait etre soit 1, soit "<<dimf << endl;
Sortie(1);
};
// }
// else
// {cout << "\n *** erreur dans l'utilisation d'une fct nD pour un chargement"
// << " surfacique: dependance autre que du point "<< endl;
// Sortie(1);
// };
};
tab_Press(ni)=force_t; // sauvegarde de l'effort
for (int ne =1; ne<= nbne; ne++)
for (int i=1;i<= dimf;i++,ix++)
SM(ix) += taphi(ni)(ne)* force_t(i) * (poids(ni) * (*ex.jacobien_tdt));
// dans le cas avec_raideur on calcul la contribution à la raideur
Coordonnee d_giB_tdt1,d_giB_tdt2,d_dir_f_t; // def de vecteurs intermediaires
Coordonnee d_dir_f_t_normer; // " "
Coordonnee D_normale_pasnormer,D_normale_t;
if (avec_raid)
{for (int ne =1; ne<= nbne; ne++)
for (int j =1; j<= nbddl; j++)
{ // calcul de la variation de la normale
// 1) variation du produit vectoriel pour la normale finale
D_normale_pasnormer = Util::ProdVec_coor(giB_tdt1,(*ex.d_giB_tdt)(j).Coordo(2))
+ Util::ProdVec_coor((*ex.d_giB_tdt)(j).Coordo(1),giB_tdt2);
// 2) de la normale finale
D_normale_t = Util::VarUnVect_coor(normale_t,D_normale_pasnormer,norme_normarle_t);
// calcul des variations de la force
d_giB_tdt1 = ((((*ex.d_giB_tdt)(j))(1)).Coor());
d_giB_tdt2 = ((((*ex.d_giB_tdt)(j))(2)).Coor());
d_dir_f_t = dir_forceH_0(1) * d_giB_tdt1 + dir_forceH_0(2) * d_giB_tdt2
+ dir_forceH_0(3) * D_normale_t;
d_dir_f_t_normer = Util::VarUnVect_coor( dir_f_t,d_dir_f_t,module_f);
for (int i=1;i<= dimf;i++)
{ ix = i+ (ne-1)*dimf;
KM(ix,j) -= taphi(ni)(ne) * poids(ni)
* (force_t(i) *(*ex.d_jacobien_tdt)(j)
+ (*ex.jacobien_tdt) * d_dir_f_t(i) );
}
}
}
}
// liberation des tenseurs intermediaires
LibereTenseur();
Element::ResRaid el;
el.res = (*res_extS)(nSurf);
el.raid = (*raid_extS)(nSurf);
return el;
};
// cas d'un chargement volumique, sur l'élément
// force indique la force volumique appliquée
// retourne le second membre résultant
// calcul à l'instant tdt ou t en fonction de la variable atdt
Vecteur& ElemMeca::SM_charge_vol_E (DdlElement &
,const Tableau <Vecteur>& taphi,int nbne
,const Vecteur& poids,const Coordonnee& force,Fonction_nD* pt_fonct
,const ParaAlgoControle & ,bool sur_volume_finale_,bool atdt )
{
// #ifdef MISE_AU_POINT
// axisymétrie: a priori ce n'est pas nécessaire, on met une erreur
if(ParaGlob::AxiSymetrie())
{ cout << "\n erreur, les charges volumique ne sont utilisable avec les elements axisymetriques"
<< "\n utilisez à la place les charges surfaciques !! "
<< " ElemMeca::SM_charge_vol_E ( ...";
Sortie(1);
};
// #endif
// définition du vecteur de retour
Vecteur& SM = * residu;
// éventuellement dimensionnement de la sauvegarde
if (lesChargeExtSurEle == NULL)
lesChargeExtSurEle = new LesChargeExtSurElement;
if (lesChargeExtSurEle->Force_volume() == NULL)
lesChargeExtSurEle->Force_volume_Change_taille(def->Phi1_Taille());
Tableau <Coordonnee>& tab_F_vol= (*lesChargeExtSurEle->Force_volume()); // pour simplifier
if (tab_F_vol.Taille() != def->Phi1_Taille())
tab_F_vol.Change_taille(def->Phi1_Taille(),Coordonnee(ParaGlob::Dimension()));
int ni; // compteur globale de point d'integration
bool premier_calcul=true; // contrairement à la déformation, pas de sauvegarde
// donc il faut calculer tous les éléments de la métrique
for (def->PremierPtInteg(), ni = 1;def->DernierPtInteg();def->NevezPtInteg(),ni++)
{
// calcul des éléments de la métrique, entre autre le jacobien, on utilise le même calcul
// que pour un calcul primaire en explicit mais pour un calcul autre que mécanique
Met_abstraite::Expli ex;
if (atdt)
ex = (def->Cal_explicit_tdt(premier_calcul)).Tdt_dans_t();
else
ex = def->Cal_explicit_t(premier_calcul);
// choix entre volume final ou initial
double jacobien = (*ex.jacobien_t);
if (!sur_volume_finale_) jacobien = (*ex.jacobien_0);
int ix=1; int dimf = force.Dimension();
if(ParaGlob::AxiSymetrie())
// cas d'un chargement axisymétrique, dans ce cas on ne prend en compte que les
// dimf-1 coordonnées donc on décrémente
dimf--;
// prise en compte d'une dépendance à une fonction nD
Coordonnee force_reelle(force); // init par défaut
if (pt_fonct != NULL)
{ // ici on peut utiliser toutes les variables connues de l'élément
// on commence par récupérer les conteneurs des grandeurs à fournir
List_io <Ddl_enum_etendu>& li_enu_scal = pt_fonct->Li_enu_etendu_scalaire();
List_io <TypeQuelconque >& li_quelc = pt_fonct->Li_equi_Quel_evolue();
bool absolue = true; // on se place systématiquement en absolu
// on va utiliser la méhode Valeur_multi pour les grandeurs strictement scalaire
Tableau <double> val_ddl_enum(ElemMeca::Valeur_multi(absolue,TEMPS_tdt,li_enu_scal,ni,-1));
// on utilise la méthode des grandeurs évoluées pour les Coordonnees et Tenseur
Valeurs_Tensorielles(absolue, TEMPS_tdt,li_quelc,ni,-1);
// calcul de la valeur et retour dans tab_ret
Tableau <double> & tab_val = pt_fonct->Valeur_FnD_Evoluee(&val_ddl_enum,&li_enu_scal,&li_quelc,NULL,NULL);
// deux cas: soit une fonction scalaire soit dimf composantes
if (tab_val.Taille() == 1)
{force_reelle *= tab_val(1);}
else if (tab_val.Taille() == dimf)
{switch (dimf)
{case 3:force_reelle(3) *= tab_val(3);
case 2:force_reelle(2) *= tab_val(2);
case 1:force_reelle(1) *= tab_val(1);
break;
default: cout << "\n *** erreur ElemMeca::SM_charge_vol_E(.. ";
Sortie(1);
};
}
else
{cout << "\n *** erreur dans l'utilisation d'une fct nD pour un chargement"
<< " volumique: la dimension de la fct nD "<<tab_val.Taille()
<< " devrait etre soit 1, soit "<<dimf << endl;
Sortie(1);
};
};
// if (pt_fonct->Depend_M())
// {// 1) on récupère la position dans I_a du point d'intégration
// const Coordonnee * Mpt = NULL; // init
// // on regarde s'il s'agit du volume final ou initial
// if (sur_volume_finale_)
// { Mpt = &(def->Position_0()); }
// else
// { Mpt = &(def->Position_tdt());};
// const Coordonnee & M = *Mpt;
// // 2) on appel la fonction, qui ne doit dépendre que de M en var locales
// Tableau <Coordonnee> tab(1); tab(1)=M;
// Tableau <double> tab_val = pt_fonct->Val_FnD_Evoluee(NULL,&tab,NULL);
// // deux cas: soit une fonction scalaire soit dimf composantes
// if (tab_val.Taille() == 1)
// {force_reelle *= tab_val(1);}
// else if (tab_val.Taille() == dimf)
// {switch (dimf)
// {case 3:force_reelle(3) *= tab_val(3);
// case 2:force_reelle(2) *= tab_val(2);
// case 1:force_reelle(1) *= tab_val(1);
// break;
// default: cout << "\n *** erreur ElemMeca::SM_charge_vol_E(.. ";
// Sortie(1);
// };
// }
// else
// {cout << "\n *** erreur dans l'utilisation d'une fct nD pour un chargement"
// << " volumique: la dimension de la fct nD "<<tab_val.Taille()
// << " devrait etre soit 1, soit "<<dimf << endl;
// Sortie(1);
// };
// };
// };
tab_F_vol(ni) +=force_reelle; // on sauvegarde la force de volume
for (int ne =1; ne<= nbne; ne++)
for (int i=1;i<= dimf;i++,ix++)
SM(ix) += taphi(ni)(ne)* force_reelle(i) * (poids(ni) * jacobien);
}
// retour du second membre
return SM;
};
// cas d'un chargement volumique, sur l'élément
// force indique la force volumique appliquée
// retourne le second membre résultant -> implicite,
// pa: permet de déterminer si oui ou non on calcul la contribution à la raideur
void ElemMeca::SMR_charge_vol_I(DdlElement & tab_ddl
,const Tableau <Vecteur>& taphi,int nbne
,const Vecteur& poids,const Coordonnee& force,Fonction_nD* pt_fonct
,const ParaAlgoControle & pa,bool sur_volume_finale_)
{
// #ifdef MISE_AU_POINT
// axisymétrie: a priori ce n'est pas nécessaire, on met une erreur
if(ParaGlob::AxiSymetrie())
{ cout << "\n erreur, les charges volumique ne sont pas utilisable avec les elements axisymetriques"
<< "\n utilisez à la place les charges surfaciques !! "
<< " ElemMeca::SMR_charge_vol_I ( ...";
Sortie(1);
};
// #endif
int ni; // compteur globale de point d'integration
int nbddl = tab_ddl.NbDdl();
Vecteur& SM = *residu;
Mat_pleine & KM = *raideur;
// éventuellement dimensionnement de la sauvegarde
if (lesChargeExtSurEle == NULL)
lesChargeExtSurEle = new LesChargeExtSurElement;
if (lesChargeExtSurEle->Force_volume() == NULL)
lesChargeExtSurEle->Force_volume_Change_taille(def->Phi1_Taille());
Tableau <Coordonnee>& tab_F_vol= (*lesChargeExtSurEle->Force_volume()); // pour simplifier
if (tab_F_vol.Taille() != def->Phi1_Taille())
tab_F_vol.Change_taille(def->Phi1_Taille(),Coordonnee(ParaGlob::Dimension()));
// controle
bool avec_raid = pa.Var_charge_externe();
// // dans le cas où la variation sur la raideur est demandé on s'assure que la variation
// // du jacobien est bien effective sinon on l'impose
// ParaAlgoControle pa_nevez(pa);
// if (!pa_nevez.Var_jacobien()) pa_nevez.Modif_Var_jacobien(true);
bool premier_calcul=true; // contrairement à la déformation, pas de sauvegarde
// donc il faut calculer tous les éléments de la métrique
for (def->PremierPtInteg(), ni = 1;def->DernierPtInteg();def->NevezPtInteg(),ni++)
{
// calcul des éléments de la métrique, entre autre le jacobien, on utilise le même calcul
// que pour un calcul primaire en implicit
const Met_abstraite::Impli& ex = def->Cal_implicit(premier_calcul);
int ix=1; int dimf = force.Dimension();
if(ParaGlob::AxiSymetrie())
// cas d'un chargement axisymétrique, dans ce cas on ne prend en compte que les
// dimf-1 coordonnées donc on décrémente
dimf--;
// prise en compte d'une dépendance à une fonction nD
// on ne tient pas compte de la variation du point dans la fct nD
Coordonnee force_reelle(force); // init par défaut
if (pt_fonct != NULL)
{ // ici on peut utiliser toutes les variables connues de l'élément
// on commence par récupérer les conteneurs des grandeurs à fournir
List_io <Ddl_enum_etendu>& li_enu_scal = pt_fonct->Li_enu_etendu_scalaire();
List_io <TypeQuelconque >& li_quelc = pt_fonct->Li_equi_Quel_evolue();
bool absolue = true; // on se place systématiquement en absolu
// on va utiliser la méhode Valeur_multi pour les grandeurs strictement scalaire
Tableau <double> val_ddl_enum(ElemMeca::Valeur_multi(absolue,TEMPS_tdt,li_enu_scal,ni,-1));
// on utilise la méthode des grandeurs évoluées pour les Coordonnees et Tenseur
Valeurs_Tensorielles(absolue, TEMPS_tdt,li_quelc,ni,-1);
// calcul de la valeur et retour dans tab_ret
Tableau <double> & tab_val = pt_fonct->Valeur_FnD_Evoluee(&val_ddl_enum,&li_enu_scal,&li_quelc,NULL,NULL);
// deux cas: soit une fonction scalaire soit dimf composantes
if (tab_val.Taille() == 1)
{force_reelle *= tab_val(1);}
else if (tab_val.Taille() == dimf)
{switch (dimf)
{case 3:force_reelle(3) *= tab_val(3);
case 2:force_reelle(2) *= tab_val(2);
case 1:force_reelle(1) *= tab_val(1);
break;
default: cout << "\n *** erreur ElemMeca::SM_charge_vol_E(.. ";
Sortie(1);
};
}
else
{cout << "\n *** erreur dans l'utilisation d'une fct nD pour un chargement"
<< " volumique: la dimension de la fct nD "<<tab_val.Taille()
<< " devrait etre soit 1, soit "<<dimf << endl;
Sortie(1);
};
};
// {if (pt_fonct->Depend_M())
// {// 1) on récupère la position dans I_a du point d'intégration
// const Coordonnee * Mpt = NULL; // init
// // on regarde s'il s'agit du volume final ou initial
// if (sur_volume_finale_)
// { Mpt = &(def->Position_0()); }
// else
// { Mpt = &(def->Position_tdt());};
// const Coordonnee & M = *Mpt;
// // 2) on appel la fonction, qui ne doit dépendre que de M en var locales
// Tableau <Coordonnee> tab(1); tab(1)=M;
// Tableau <double> tab_val = pt_fonct->Val_FnD_Evoluee(NULL,&tab,NULL);
// // deux cas: soit une fonction scalaire soit dimf composantes
// if (tab_val.Taille() == 1)
// {force_reelle *= tab_val(1);}
// else if (tab_val.Taille() == dimf)
// {switch (dimf)
// {case 3:force_reelle(3) *= tab_val(3);
// case 2:force_reelle(2) *= tab_val(2);
// case 1:force_reelle(1) *= tab_val(1);
// break;
// default: cout << "\n *** erreur ElemMeca::SM_charge_vol_E(.. ";
// Sortie(1);
// };
// }
// else
// {cout << "\n *** erreur dans l'utilisation d'une fct nD pour un chargement"
// << " volumique: la dimension de la fct nD "<<tab_val.Taille()
// << " devrait etre soit 1, soit "<<dimf << endl;
// Sortie(1);
// };
// };
// };
tab_F_vol(ni) +=force_reelle; // on sauvegarde la force de volume
// on choisit entre volume final ou initial
if (sur_volume_finale_)
{for (int ne =1; ne<= nbne; ne++)
for (int i=1;i<= dimf;i++,ix++)
{ SM(ix) +=
taphi(ni)(ne)* force_reelle(i) * (poids(ni) * (*ex.jacobien_tdt));
// dans le cas avec_raideur on calcul la contribution à la raideur
if (avec_raid)
for (int j =1; j<= nbddl; j++)
KM(ix,j) -=
taphi(ni)(ne)* force_reelle(i) * (poids(ni) * (*ex.d_jacobien_tdt)(j));
};
}
else // cas plus simple du volume initial
{for (int ne =1; ne<= nbne; ne++)
for (int i=1;i<= dimf;i++,ix++)
{ SM(ix) +=
taphi(ni)(ne)* force_reelle(i) * (poids(ni) * (*ex.jacobien_0));
// pas de contribution à la raideur car le jacobien ne bouge pas
};
};
}
// liberation des tenseurs intermediaires
LibereTenseur();
};
// cas d'un chargement hydrostatique, sur les surfaces de l'élément
// la charge dépend de la hauteur à la surface libre du liquide déterminée par un point
// et une direction normale à la surface libre:
// nSurf : le numéro de la surface externe
// poidvol: indique le poids volumique du liquide
// M_liquide : un point de la surface libre
// dir_normal_liquide : direction normale à la surface libre
// retourne le second membre résultant
// calcul à l'instant tdt ou t en fonction de la variable atdt
Vecteur& ElemMeca::SM_charge_hydro_E (DdlElement & ,int nSurf
,const Tableau <Vecteur>& taphi,int nbne
,const Vecteur& poids
,const Coordonnee& dir_normal_liquide,const double& poidvol
,const Coordonnee& M_liquide,bool sans_limitation
,const ParaAlgoControle & ,bool atdt)
{
// #ifdef MISE_AU_POINT
// // axisymétrie: pour l'instant non testé a priori avec les forces hydro
// if(ParaGlob::AxiSymetrie())
// { cout << "\n erreur, les charges hydro ne sont pas testees avec les elements axisymetriques"
// << " ElemMeca::SM_charge_hydro_E ( ...";
// Sortie(1);
// };
// #endif
// *** dans l'ensemble du sp on utilise des vecteurs et on gére directement la variance
// *** sinon c'est intractable et long
// définition du vecteur de retour
Deformation* definter=NULL; // variable de travail transitoire
Vecteur* SM_P = NULL;
if (!(ParaGlob::AxiSymetrie()))
{ definter = defSurf(nSurf); // le cas normal est non axisymétrique
SM_P = ((*res_extS)(nSurf));
}
else { definter = defArete(nSurf); // en axisymétrie, c'est une def d'arête
SM_P = ((*res_extA)(nSurf));
};
Deformation & defS = *definter; // pour simplifier l'écriture
Vecteur& SM = *SM_P; // " " "
// dans le cas ou le poid volumique est nul, on retourne sans calcul
if (Dabs(poidvol) < ConstMath::trespetit)
return SM;
int ni; // compteur globale de point d'integration
// éventuellement dimensionnement de la sauvegarde
if (lesChargeExtSurEle == NULL)
lesChargeExtSurEle = new LesChargeExtSurElement;
if (lesChargeExtSurEle->LesPressionsExternes() == NULL)
{if(ParaGlob::AxiSymetrie())
{lesChargeExtSurEle->LesPressionsExternes_Change_taille(ElementGeometrique().NbSe());}
else
{lesChargeExtSurEle->LesPressionsExternes_Change_taille(ElementGeometrique().NbFe());}
};
Tableau <Pression_appliquee>& tab_Press= (*lesChargeExtSurEle->LesPressionsExternes())(nSurf); // pour simplifier
if (tab_Press.Taille() != defS.Phi1_Taille())
tab_Press.Change_taille(defS.Phi1_Taille());
bool premier_calcul=true; // contrairement à la déformation, pas de sauvegarde
// donc il faut calculer tous les éléments de la métrique
for (defS.PremierPtInteg(), ni = 1;defS.DernierPtInteg();defS.NevezPtInteg(),ni++)
{
// calcul de la pression au niveau de la position du point d'intégration
// 1) on récupère la position dans I_a du point d'intégration
const Coordonnee & M = defS.Position_tdt();
// 2) calcul de la distance à la surface libre
Coordonnee MMp= M_liquide - M; double dis = dir_normal_liquide * MMp;
// 3) calcul de la pression effective uniquement si le point est dans l'eau
// en fait si le point est hors de l'eau il n'y a pas de pression hydrostatique
// sauf si sans_limitation est true
if ((dis > 0.) || sans_limitation)
{ double pression = -dis * poidvol;
// sauvegarde de la pression appliquée
tab_Press(ni).press += pression;
// -- calcul des éléments de la métrique, entre autre le jacobien, on utilise le même calcul
// que pour un calcul primaire en explicit mais pour un calcul autre que mécanique
Met_abstraite::Expli ex;
if (atdt)
ex = (defS.Cal_explicit_tdt(premier_calcul)).Tdt_dans_t();
else
ex = defS.Cal_explicit_t(premier_calcul);
// détermination de la normale à la surface
#ifdef MISE_AU_POINT
// test pour savoir si l'on a réellement affaire à une surface
if ((ex.giB_t->NbVecteur() != 2) || ( ParaGlob::Dimension() != 3))
{ cout << "\n erreur, il doit y avoir deux vecteurs pour la surface de dimension 3"
<< " ElemMeca::SM_charge_hydro_E (DdlElement & ...nbvec= " << ex.giB_t->NbVecteur()
<< " dim = " << ParaGlob::Dimension();
Sortie(1);
}
#endif
// la normale vaut le produit vectoriel des 2 premiers vecteurs
CoordonneeB normale = Util::ProdVec_coorB( (*ex.giB_t)(1), (*ex.giB_t)(2));
// calcul de la normale normée et pondérée de la pression
normale.Normer();normale *= -pression;
tab_Press(ni).P += normale.Coor_const();
int ix=1; int dimf = 3; // ne fonctionne qu'en dim 3
if(ParaGlob::AxiSymetrie())
// cas d'un chargement axisymétrique, dans ce cas on ne prend en compte que les
// dimf-1 coordonnées donc on décrémente
dimf--;
for (int ne =1; ne<= nbne; ne++)
for (int i=1;i<= dimf;i++,ix++)
SM(ix) += taphi(ni)(ne)* normale(i) * (poids(ni) * (*ex.jacobien_t));
}// fin test : point dans le liquide ou pas
} // fin boucle sur les points d'intégrations
// retour du second membre
return SM;
};
// idem SMR_charge_hydro_E mais
// -> implicite,
// pa: permet de déterminer si oui ou non on calcul la contribution à la raideur
// retourne le second membre et la matrice de raideur correspondant
Element::ResRaid ElemMeca::SMR_charge_hydro_I (DdlElement & ddls,int nSurf
,const Tableau <Vecteur>& taphi,int nbne
,const Vecteur& poids
,const Coordonnee& dir_normal_liquide,const double& poidvol
,const Coordonnee& M_liquide,bool sans_limitation
,const ParaAlgoControle & pa)
{
// #ifdef MISE_AU_POINT
// // axisymétrie: pour l'instant non testé a priori avec les forces hydro
// if(ParaGlob::AxiSymetrie())
// { cout << "\n erreur, les charges hydro ne sont pas testees avec les elements axisymetriques"
// << " ElemMeca::SM_charge_hydro_I ( ...";
// Sortie(1);
// };
// #endif
Deformation* definter=NULL; // variable de travail transitoire
Vecteur* SM_P = NULL; Mat_pleine* KM_P = NULL;
if (!(ParaGlob::AxiSymetrie()))
{ definter = defSurf(nSurf); // le cas normal est non axisymétrique
SM_P = ((*res_extS)(nSurf));
KM_P = ((*raid_extS)(nSurf));
}
else { definter = defArete(nSurf); // en axisymétrie, c'est une def d'arête
SM_P = ((*res_extA)(nSurf));
KM_P = ((*raid_extA)(nSurf));
};
Deformation & defS = *definter; // pour simplifier l'écriture
Vecteur& SM = *SM_P; // " " "
Mat_pleine & KM = *KM_P; // " " "
// *** dans l'ensemble du sp on utilise des vecteurs et on gére directement la variance
// *** sinon c'est intractable et long
// dans le cas ou le poid volumique est nul, on retourne sans calcul
if (Dabs(poidvol) < ConstMath::trespetit)
{ Element::ResRaid el;
el.res = SM_P;
el.raid = KM_P;
return el;
}
// controle
bool avec_raid = pa.Var_charge_externe();
// // dans le cas où la variation sur la raideur est demandé on s'assure que la variation
// // du jacobien est bien effective sinon on l'impose
// ParaAlgoControle pa_nevez(pa);
// if (!pa_nevez.Var_jacobien()) pa_nevez.Modif_Var_jacobien(true);
int ni; // compteur globale de point d'integration
// éventuellement dimensionnement de la sauvegarde
if (lesChargeExtSurEle == NULL)
lesChargeExtSurEle = new LesChargeExtSurElement;
if (lesChargeExtSurEle->LesPressionsExternes() == NULL)
{if(ParaGlob::AxiSymetrie())
{lesChargeExtSurEle->LesPressionsExternes_Change_taille(ElementGeometrique().NbSe());}
else
{lesChargeExtSurEle->LesPressionsExternes_Change_taille(ElementGeometrique().NbFe());}
};
Tableau <Pression_appliquee>& tab_Press= (*lesChargeExtSurEle->LesPressionsExternes())(nSurf); // pour simplifier
if (tab_Press.Taille() != defS.Phi1_Taille())
tab_Press.Change_taille(defS.Phi1_Taille());
int nbddl = ddls.NbDdl();
bool premier_calcul=true; // contrairement à la déformation, pas de sauvegarde
// donc il faut calculer tous les éléments de la métrique
for (defS.PremierPtInteg(), ni = 1;defS.DernierPtInteg();defS.NevezPtInteg(),ni++)
{
// calcul de la pression au niveau de la position du point d'intégration
// 1) on récupère la position dans I_a du point d'intégration
const Coordonnee & M = defS.Position_tdt();
// 2) calcul de la distance à la surface libre
Coordonnee MMp= M_liquide - M; double dis = dir_normal_liquide * MMp;
// 3) calcul de la pression effective uniquement si le point est dans l'eau
// en fait si le point est hors de l'eau il n'y a pas de pression hydrostatique
// sauf si sans_limitation est true
if ((dis > 0.) || sans_limitation)
{ double pression = -dis * poidvol;
// sauvegarde de la pression appliquée
tab_Press(ni).press += pression;
// dans le cas avec raideur on calcul la variation de la pression
const Tableau <Coordonnee> * d_M_pointe = NULL;
if (avec_raid)
d_M_pointe=&(defS.Der_Posi_tdt());
// --- calcul des éléments de la métrique, entre autre le jacobien, on utilise le même calcul
// que pour un calcul primaire en implicit
const Met_abstraite::Impli& ex = defS.Cal_implicit(premier_calcul);
#ifdef MISE_AU_POINT
// test pour savoir si l'on a réellement affaire à une surface
if ((ex.giB_t->NbVecteur() != 2) || ( ParaGlob::Dimension() != 3))
{ cout << "\n erreur, il doit y avoir deux vecteurs pour la surface de dimension 3"
<< " ElemMeca::SMR_charge_hydro_I (.... " << ex.giB_t->NbVecteur()
<< " dim = " << ParaGlob::Dimension();
Sortie(1);
}
#endif
// la normale vaut le produit vectoriel des 2 premiers vecteurs
CoordonneeB normale = Util::ProdVec_coorB( (*ex.giB_tdt)(1), (*ex.giB_tdt)(2));
// calcul de la variation de la normale
// 1) variation du produit vectoriel
Tableau <CoordonneeB > D_pasnormale =
Util::VarProdVect_coorB( (*ex.giB_tdt)(1),(*ex.giB_tdt)(2),(*ex.d_giB_tdt));
// 2) de la normale
Tableau <CoordonneeB> D_normale = Util::VarUnVect_coorB(normale,D_pasnormale,normale.Norme());
// calcul de la normale normée et pondérée de la pression
normale.Normer();normale *= -pression;
tab_Press(ni).P += normale.Coor_const();
// également pondération de la variation de la
if (avec_raid)
{ for (int ihi =1;ihi<= nbddl;ihi++)
{ double d_pression = poidvol * (dir_normal_liquide * (*d_M_pointe)(ihi));
// le - devant dis disparait avec le - devant M
D_normale(ihi) = pression * D_normale(ihi) + d_pression * normale ;
}
}
int ix=1; int dimf = 3; // ne fonctionne qu'en dim 3
if(ParaGlob::AxiSymetrie())
// cas d'un chargement axisymétrique, dans ce cas on ne prend en compte que les
// dimf-1 coordonnées donc on décrémente
dimf--;
for (int ne =1; ne<= nbne; ne++)
for (int i=1;i<= dimf;i++,ix++)
{ SM(ix) += taphi(ni)(ne)* normale(i) * (poids(ni) * (*ex.jacobien_tdt));
// dans le cas avec_raideur on calcul la contribution à la raideur
if (avec_raid)
for (int j =1; j<= nbddl; j++)
KM(ix,j) -=
taphi(ni)(ne)* normale(i) * (poids(ni) * (*ex.d_jacobien_tdt)(j))
+ taphi(ni)(ne)* D_normale(j)(i) * (poids(ni) * (*ex.jacobien_tdt)) ;
}
}// fin test : point dans le liquide ou pas
} // fin boucle sur les points d'intégrations
// liberation des tenseurs intermediaires
LibereTenseur();
Element::ResRaid el;
el.res = SM_P;
el.raid = KM_P;
return el;
};