Source code for finat.bell
import numpy
import FIAT
from gem import Literal, ListTensor
from finat.fiat_elements import ScalarFiatElement
from finat.physically_mapped import PhysicallyMappedElement, Citations
[docs]class Bell(PhysicallyMappedElement, ScalarFiatElement):
def __init__(self, cell, degree):
if degree != 5:
raise ValueError("Degree must be 3 for Bell element")
if Citations is not None:
Citations().register("Bell1969")
super().__init__(FIAT.Bell(cell))
[docs] def basis_transformation(self, coordinate_mapping):
# Jacobians at edge midpoints
J = coordinate_mapping.jacobian_at([1/3, 1/3])
rns = coordinate_mapping.reference_normals()
pts = coordinate_mapping.physical_tangents()
pel = coordinate_mapping.physical_edge_lengths()
V = numpy.zeros((21, 18), dtype=object)
for multiindex in numpy.ndindex(V.shape):
V[multiindex] = Literal(V[multiindex])
for v in range(3):
s = 6*v
V[s, s] = Literal(1)
for i in range(2):
for j in range(2):
V[s+1+i, s+1+j] = J[j, i]
V[s+3, s+3] = J[0, 0]*J[0, 0]
V[s+3, s+4] = 2*J[0, 0]*J[1, 0]
V[s+3, s+5] = J[1, 0]*J[1, 0]
V[s+4, s+3] = J[0, 0]*J[0, 1]
V[s+4, s+4] = J[0, 0]*J[1, 1] + J[1, 0]*J[0, 1]
V[s+4, s+5] = J[1, 0]*J[1, 1]
V[s+5, s+3] = J[0, 1]*J[0, 1]
V[s+5, s+4] = 2*J[0, 1]*J[1, 1]
V[s+5, s+5] = J[1, 1]*J[1, 1]
for e in range(3):
v0id, v1id = [i for i in range(3) if i != e]
# nhat . J^{-T} . t
foo = (rns[e, 0]*(J[0, 0]*pts[e, 0] + J[1, 0]*pts[e, 1])
+ rns[e, 1]*(J[0, 1]*pts[e, 0] + J[1, 1]*pts[e, 1]))
# vertex points
V[18+e, 6*v0id] = -1/21 * (foo / pel[e])
V[18+e, 6*v1id] = 1/21 * (foo / pel[e])
# vertex derivatives
for i in (0, 1):
V[18+e, 6*v0id+1+i] = -1/42*foo*pts[e, i]
V[18+e, 6*v1id+1+i] = V[18+e, 6*v0id+1+i]
# second derivatives
tau = [pts[e, 0]*pts[e, 0],
2*pts[e, 0]*pts[e, 1],
pts[e, 1]*pts[e, 1]]
for i in (0, 1, 2):
V[18+e, 6*v0id+3+i] = -1/252 * (pel[e]*foo*tau[i])
V[18+e, 6*v1id+3+i] = 1/252 * (pel[e]*foo*tau[i])
h = coordinate_mapping.cell_size()
for v in range(3):
for k in range(2):
for i in range(21):
V[i, 6*v+1+k] = V[i, 6*v+1+k] / h[v]
for k in range(3):
for i in range(21):
V[i, 6*v+3+k] = V[i, 6*v+3+k] / (h[v]*h[v])
return ListTensor(V.T)
# This wipes out the edge dofs. FIAT gives a 21 DOF element
# because we need some extra functions to help with transforming
# under the edge constraint. However, we only have an 18 DOF
# element.
[docs] def entity_dofs(self):
return {0: {0: list(range(6)),
1: list(range(6, 12)),
2: list(range(12, 18))},
1: {0: [], 1: [], 2: []},
2: {0: []}}
@property
def index_shape(self):
return (18,)