Domain dependent parameters in elasticity

Hi All,

I want to implement elasticity in domain [0,1]*[0,1] with known analytical solution, and I want to use different values of lambda and nu, in different halves of the domain and I tried it as follows:

real nu1 = 10;  
real lambda1 = 2*10^4; Lame constant

real nu2 = 20;  
real lambda2 = 10^4; 

func nu = (y>=0.5 )*(nu1 + 0*x + 0*y) + (y<0.5 )*(nu2+ 0*x + 0*y);
func lambda = (y>=0.5 )*(lambda1 + 0*x + 0*y) + (y<0.5 )*(lambda2+ 0*x + 0*y);

On doing this, I noticed that at y=0.5, the L2 norm of computed solution u and exact solution uex are very different and I do not get the convergence of my solution. When I use the same parameters nu and lambda, I do not have any problem with the elasticity solver (I get desired convergence rates).

Please let me know if I needed to change anything in defining nu and lambda differently in different parts of the domain.

Thank you!

In the documentation section 2.6 (thermal conduction problem), different diffusion (k) has been used in different parts of the domain. So, I thought it should work the same way!

1)Defining nu and lambda as functions of x and y.
2) Use it in defining right-side functions and in variational formulation.

I plotted my solution in the case of elasticity in the whole domain [0,1]*[0,1] and it is way off as compared to exact solution. While using same lambda and nu over entire domain, my solution matches with the exact solution.

//RHS for Elasticity 
func f1s = -nus*(2*u1struexx+u1strueyy+u2struexy)-lambdas*(u1struexx+u2strueyx); 
func f2s = -nus*(2*u2strueyy+u2struexx+u1strueyx)-lambdas*(u1struexy+u2strueyy); 

fespace Uh(Gh,P2); // element space for displacement
Uh u1s,u2s, v1s, v2s;

problem structure([u1s,u2s], [v1s,v2s]) =
		int2d(Gh) ( 
						2*nus*(
								dx(u1s)*dx(v1s) 
								+ 0.5*(dx(u2s) + dy(u1s))*(dx(v2s) + dy(v1s)) 
								+ dy(u2s)*dy(v2s)
								) // 2nu(D(u),D(v))
					+ lambdas*(dx(u1s) + dy(u2s))*(dx(v1s) + dy(v2s)) // lambda(div(u),div(v))	
					)
		-	int2d(Gh)(
						f1s*v1s + f2s*v2s 
					)

		+ on(2,3,4,5,u1s=u1strue,u2s=u2strue); // BC

@fb77 Sir, I would really appriciate if you could please add any thought on it? Thank you!

I don’t see why there should be a problem. Can you post the full code with all definitions?

Hi @fb77, Thank you for your kind response. Please find the following mentioned elasticity solver and let me know your feedback:

bool debug=true;
verbosity = 0;

// Load VTK plugin
load "iovtk" //for paraview

// DOMAIN  
real np = 8; // number of elements in bdry
cout << "np" << np << endl;

//Boundary for global domain 
border G1(t=0,1){x=t;y=0; label=1;}; //bottom (0,0) -> (1,0)
border G2(t=0,1){x=1;y=t; label=2;}; //right (1,0) -> (1,1)
border G3(t=0,1){x=1-t;y=1; label=3;}; //top (1,1) -> (0,1)
border G4(t=0,1){x=0;y=1-t; label=4;}; //left (0,1) -> (0,0)


//Global Mesh
mesh Gh=buildmesh(G1(np)+G2(np)+G3(np)+G4(np));

plot(Gh); // check global domain

//Elasticity Parameters:
real nus1 = 10;  
real lambdas1 = 2*10^4;

real nup1 = 20;  
real lambdap1 = 10^4; 

func nus = (y>=0.5)*nus1 + (y<0.5)*nup1; 
func lambdas = (y>=0.5)*lambdas1 + (y<0.5)*lambdap1; 

//Manufactured solution
func u1strue = sin(pi*(x+y)); //exact velocity x direction
func u2strue = cos(pi*(x^2+y^2));; //exact velocity y direction

func u1struex = cos(pi*(x+y))*pi; 
func u1struey = cos(pi*(x+y))*pi; 

func u2struex = -sin(pi*(x^2+y^2))*pi*2*x; 
func u2struey = -sin(pi*(x^2+y^2))*pi*2*y; 

func u1struexx = -sin(pi*(x+y))*pi*pi; 
func u1strueyy = -sin(pi*(x+y))*pi*pi; 

func u2struexx = -sin(pi*(x^2+y^2))*pi*2 - cos(pi*(x^2+y^2))*pi*2*x*pi*2*x; 
func u2strueyy = -sin(pi*(x^2+y^2))*pi*2 - cos(pi*(x^2+y^2))*pi*2*y*pi*2*y; 

func u1struexy = -sin(pi*(x+y))*pi*pi; 
func u1strueyx = -sin(pi*(x+y))*pi*pi; 

func u2strueyx = -cos(pi*(x^2+y^2))*pi*2*y*pi*2*x; 
func u2struexy = -cos(pi*(x^2+y^2))*pi*2*x*pi*2*y; 

// Initial conditions and RHS for Elasticity 
func f1s = -nus*(2*u1struexx+u1strueyy+u2struexy)-lambdas*(u1struexx+u2strueyx); 
func f2s = -nus*(2*u2strueyy+u2struexx+u1strueyx)-lambdas*(u1struexy+u2strueyy); 

// Finite element space for displacement
fespace Uh(Gh,[P2,P2]); 
Uh [u1s,u2s], [v1s, v2s];

problem structure([u1s,u2s],[v1s,v2s]) =
		int2d(Gh) ( 
						2*nus*(
								dx(u1s)*dx(v1s) 
								+ 0.5*(dx(u2s) + dy(u1s))*(dx(v2s) + dy(v1s)) 
								+ dy(u2s)*dy(v2s)
								) // 2nu(D(u),D(v))
					+ lambdas*(dx(u1s) + dy(u2s))*(dx(v1s) + dy(v2s)) // lambda(div(u),div(v))	
					)

		-	int2d(Gh)(
						f1s*v1s + f2s*v2s 
					)

		+ on(1,2,3,4,u1s = u1strue, u2s = u2strue); // BC
;

//Solve elasticity
structure;

//Computing Errors 
cout << "Global Displacement Error L2-norm = " <<
( sqrt(int2d(Gh)((u1s-u1strue)*(u1s-u1strue) + (u2s-u2strue)*(u2s-u2strue) )))<< endl;


cout << "Global Displacement Error H1-norm = " <<
( sqrt(int2d(Gh)((u1s-u1strue)*(u1s-u1strue) + (u2s-u2strue)*(u2s-u2strue) + 
(dx(u1s)-u1struex)*(dx(u1s)-u1struex)+ (dy(u1s)-u1struey)*(dy(u1s)-u1struey) +
(dx(u2s)-u2struex)*(dx(u2s)-u2struex)+ (dy(u2s)-u2struey)*(dy(u2s)-u2struey) )))<< endl;


//Plot in paraview 
int[int] Order = [1];
string DataName1 = "u v";
string DataName2 = "uex vex";

savevtk("vL.vtk", Gh, [u1s, u2s], dataname=DataName1, order=Order);
savevtk("vLExact.vtk", Gh, [u1strue, u2strue], dataname=DataName2, order=Order);



Since, my parameters have big jump at y=0.5, I also tried using mesh as follows (but it didn’t help me much):

real np2 = np/2;

//Boundary for Gh1
border G1(t=0,1){x=t;y=0.5; label=1;}; //(0,0.5) -> (1,0.5) //Interface
border G2(t=0,0.5){x=1;y=0.5+t; label=2;}; //right top (1,0.5) -> (1,1)
border G3(t=0,1){x=1-t;y=1; label=3;}; //top (1,1) -> (0,1)
border G4(t=0,0.5){x=0;y=1-t; label=4;}; //left (0,1) -> (0,0.5)

//Boundary for Lh 
border L1(t=0,1){x=t;y=0; label=8;}; //bottom (0,0) -> (1,0)
border L2(t=0,0.5){x=1;y=t; label=9;}; //right (1,0) -> (1,0.5)
border L3(t=0,1){x=1-t;y=0.5; label=10;}; //top (1,0.5) -> (0,0.5) //Interface
border L4(t=0,0.5){x=0;y=0.5-t; label=11;}; //left (0,0.5) -> (0,0)

//Global Mesh 
mesh Gh1=buildmesh(G1(np)+G2(np2)+G3(np)+G4(np2));

//Local Mesh 
mesh Lh=buildmesh(L1(np)+L2(np2)+L3(np)+L4(np2));

mesh Gh = Gh1 + Lh;

Your “manufactured solution” does not solve the problem you want to solve, because of the jumps in \nu and \lambda. Indeed one has
-\mathop{\rm div}(2\nu Du)-\nabla(\lambda\mathop{\rm div} u) =-2\nu\mathop{\rm div} Du-\lambda\nabla\mathop{\rm div} u - 2Du\nabla\nu-(\mathop{\rm div }u)\nabla\lambda and
\nabla\nu=\delta(y-1/2)[\nu](0,1), \nabla\lambda=\delta(y-1/2)[\lambda](0,1),
where [\nu] denotes the jump of \nu through the line y=1/2, \delta(y-1/2) is the Dirac function on the line, and (0,1) is the unit vector oriented upwards.
Thus to be correct you must add a right-hand side \delta(y-1/2)g.
This is what I have done with the functions g1s, g2s, and defining the line as border G5
elasticdisc.edp (3.0 KB)

Then it works.

Thank you for your time and response @fb77. It really means a lot to me. :slightly_smiling_face:

I was wondering why the domain-dependent parameters in the documentation are NOT handled similarly but they work (thermal problems). I didn’t see any right-side adjustment in those problems. And also, in this simple problem too, the errors look good now but interestingly when I plot the solution to compare with exact function, they are very OFF, even the scale, (see the attached plots for x-components):


@fb77, the way line y=1/2 was created in the mesh overlapped the mesh twice, so the solution looked weird. I modified it as mentioned above by creating Gh1 and Lh and glued them to create Gh. The solution is now very similar to the exact solution. THANK YOU SO MUCH FOR HELPING ME WITH THIS.

the domain-dependent parameters in the documentation are NOT handled similarly but they work (thermal problems).

The variational formulation is done here with nothing special. Only the data (right-hand side) is special. This special right-hand side is chosen in order to have the exact solution. In the thermal problems of the documentation, there is no exact solution. You can take an arbitrary right-hand side (with or without line integral) and get the corresponding solution numerically, this what is done in the examples of the documentation.
Another way of building an exact solution which is more “natural”, without line integral in the right-hand side, would be for the elliptic problem -\mathop{\rm div}(k\nabla u) =f, with k discontinuous, to take u_{ex} continuous, but with \nabla u_{ex} discontinuous in such a way that k\nabla u_{ex} is continuous. Then the corresponding f would have no line Dirac part.
Indeed when -\mathop{\rm div}(k\nabla u) =f with k discontinuous, u and f cannot be both smooth; at least one of them must have some kind of unsmoothness.

1 Like

Thank you for explaining this detail so clearly!