-
Notifications
You must be signed in to change notification settings - Fork 99
/
CellDofs.jl
124 lines (105 loc) · 4.23 KB
/
CellDofs.jl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
"""
"""
struct CellDof{DS} <: CellDatum
cell_dof::AbstractArray{<:Union{Dof,AbstractArray{<:Dof}}}
trian::Triangulation
domain_style::DS
end
get_data(f::CellDof) = f.cell_dof
get_triangulation(f::CellDof) = f.trian
DomainStyle(::Type{CellDof{DS}}) where DS = DS()
function change_domain(a::CellDof,::ReferenceDomain,::PhysicalDomain)
@notimplemented
end
function change_domain(a::CellDof,::PhysicalDomain,::ReferenceDomain)
@notimplemented
end
function change_domain(a::CellDof,ttrian::Triangulation,target_domain::DomainStyle)
change_domain(a,get_triangulation(a),DomainStyle(a),ttrian,target_domain)
end
function change_domain(a::CellDof,source_domain::DomainStyle,target_trian::Triangulation,target_domain::DomainStyle)
change_domain(a,get_triangulation(a),source_domain,target_trian,target_domain)
end
function change_domain(a::CellDof,strian::Triangulation,::ReferenceDomain,ttrian::Triangulation,::ReferenceDomain)
msg = """\n
We cannot move the given CellDof to the reference domain of the requested triangulation.
Make sure that the given triangulation is either the same as the triangulation on which the
CellDof is defined, or that the latter triangulation is the background of the former.
"""
if strian === ttrian
return a
end
@check is_change_possible(strian,ttrian) msg
D = num_cell_dims(strian)
sglue = get_glue(strian,Val(D))
tglue = get_glue(ttrian,Val(D))
change_domain_ref_ref(a,ttrian,sglue,tglue)
end
function change_domain(a::CellDof,strian::Triangulation,::PhysicalDomain,ttrian::Triangulation,::PhysicalDomain)
msg = """\n
We cannot move the given CellDof to the physical domain of the requested triangulation.
Make sure that the given triangulation is either the same as the triangulation on which the
CellDof is defined, or that the latter triangulation is the background of the former.
"""
if strian === ttrian
return a
end
@check is_change_possible(strian,ttrian) msg
D = num_cell_dims(strian)
sglue = get_glue(strian,Val(D))
tglue = get_glue(ttrian,Val(D))
change_domain_phys_phys(a,ttrian,sglue,tglue)
end
function change_domain(a::CellDof,strian::Triangulation,::PhysicalDomain,trian::Triangulation,::ReferenceDomain)
a_trian = change_domain(a,trian,PhysicalDomain())
change_domain(a_trian,ReferenceDomain())
end
function change_domain(a::CellDof,strian::Triangulation,::ReferenceDomain,trian::Triangulation,::PhysicalDomain)
a_phys = change_domain(a,PhysicalDomain())
change_domain(a_phys,trian,PhysicalDomain())
end
function change_domain_ref_ref(
a::CellDof,ttrian::Triangulation,sglue::FaceToFaceGlue,tglue::FaceToFaceGlue)
sface_to_dof = get_data(a)
mface_to_sface = sglue.mface_to_tface
tface_to_mface = tglue.tface_to_mface
tface_to_mface_map = tglue.tface_to_mface_map
mface_to_dof = extend(sface_to_dof,mface_to_sface)
tface_to_dof = lazy_map(Reindex(mface_to_dof),tface_to_mface)
@notimplementedif ! isa(tface_to_mface_map,AbstractArray{<:GenericField{typeof(identity)}})
CellDof(tface_to_dof,ttrian,ReferenceDomain())
end
function change_domain_phys_phys(
a::CellDof,ttrian::Triangulation,sglue::FaceToFaceGlue,tglue::FaceToFaceGlue)
sface_to_dof = get_data(a)
mface_to_sface = sglue.mface_to_tface
tface_to_mface = tglue.tface_to_mface
mface_to_dof = extend(sface_to_dof,mface_to_sface)
tface_to_dof = lazy_map(Reindex(mface_to_dof),tface_to_mface)
CellDof(tface_to_dof,ttrian,PhysicalDomain())
end
function get_cell_points(dofs::CellDof)
cell_to_x = lazy_map(get_nodes,get_data(dofs))
CellPoint(cell_to_x,get_triangulation(dofs),DomainStyle(dofs))
end
# Evaluation of CellDof
(a::CellDof)(f) = evaluate(a,f)
function evaluate!(cache,s::CellDof,f::CellField)
trian_f = get_triangulation(f)
trian_s = get_triangulation(s)
if trian_f !== trian_s
@unreachable """\n
A CellDof can only be evaluated on a CellField defined on the same Triangulation.
"""
end
b = change_domain(f,s.domain_style)
lazy_map(evaluate,get_data(s),get_data(b))
end
function evaluate!(cache, ::CellField, ::CellDof)
@unreachable """\n
CellField (f) objects cannot be evaluated at CellDof (s) objects.
However, CellDofs objects can be evaluated at CellField objects.
Did you mean evaluate(f,s) instead of evaluate(s,f), i.e.
f(s) instead of s(f)?
"""
end