-
Notifications
You must be signed in to change notification settings - Fork 1
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Interop with SymbolicUtils #64
Changes from 10 commits
c2d64b5
c11d4dd
7de10e8
5928cc7
d7d6a5b
2f79b0c
a438192
18a1585
090ddfe
d8be4ae
77770e5
1fa477b
3b265ac
7f8597a
154e51f
18bb71f
4ba6dce
82f65ce
e23459f
e82e826
f32b3c0
ecfa931
4603ed9
f2ef1d0
5324de3
0a314a8
5d5c25d
70a420d
85572a4
df41b18
6d1edf0
ccf0a79
467a93c
7c5155d
4f1f327
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,283 @@ | ||
module ThDEC | ||
|
||
using MLStyle | ||
using StructEquality | ||
|
||
import Base: +, -, * | ||
|
||
""" | ||
Given a tuple of symbols ("aliases") and their canonical name (or "rep"), produces | ||
for each alias typechecking and nameof methods which call those for their rep. | ||
Example: | ||
@alias d₀, d₁, += d | ||
""" | ||
macro alias(body) | ||
(rep, aliases) = @match body begin | ||
Expr(:tuple, rep, Expr(:tuple, aliases...)) => (rep, aliases) | ||
_ => nothing | ||
end | ||
result = quote end | ||
foreach(aliases) do alias | ||
push!(result.args, | ||
quote | ||
function $(esc(alias))(s...) | ||
$(esc(rep))(s...) | ||
end | ||
export $(esc(alias)) | ||
Base.nameof(::typeof($alias), s) = nameof($rep, s) | ||
end) | ||
end | ||
result | ||
end | ||
|
||
struct SortError <: Exception | ||
message::String | ||
end | ||
|
||
@struct_hash_equal struct Space | ||
name::Symbol | ||
dim::Int | ||
end | ||
export Space | ||
|
||
dim(s::Space) = s.dim | ||
Base.nameof(s::Space) = s.name | ||
|
||
struct SpaceLookup | ||
default::Space | ||
named::Dict{Symbol,Space} | ||
end | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. We should add a default constructor that just takes the default name. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Realized this needs to be default name and dimension |
||
export SpaceLookup | ||
|
||
function SpaceLookup(default::Space) | ||
SpaceLookup(default, Dict{Symbol, Space}(nameof(default) => default)) | ||
end | ||
|
||
@data Sort begin | ||
Scalar() | ||
Form(dim::Int, isdual::Bool, space::Space) | ||
VField(isdual::Bool, space::Space) | ||
end | ||
export Sort, Scalar, Form, VField | ||
|
||
function fromexpr(lookup::SpaceLookup, e, ::Type{Sort}) | ||
(name, spacename) = @match e begin | ||
name::Symbol => (name, nothing) | ||
:($name{$spacename}) => (name, spacename) | ||
end | ||
space = @match spacename begin | ||
::Nothing => lookup.default | ||
name::Symbol => lookup.named[name] | ||
end | ||
@match name begin | ||
:Form0 => Form(0, false, space) | ||
:Form1 => Form(1, false, space) | ||
:Form2 => Form(2, false, space) | ||
:DualForm0 => Form(0, true, space) | ||
:DualForm1 => Form(1, true, space) | ||
:DualForm2 => Form(2, true, space) | ||
:Constant => Scalar() | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. name could also be a Parameter or a Literal. Parameters are "constants" that have values that depend on time so in the solver the |
||
end | ||
end | ||
|
||
Base.nameof(s::Scalar) = :Constant | ||
|
||
function Base.nameof(f::Form; with_dim_parameter=false) | ||
dual = isdual(f) ? "Dual" : "" | ||
formname = Symbol("$(dual)Form$(dim(f))") | ||
if with_dim_parameter | ||
return Expr(:curly, formname, dim(space(f))) | ||
else | ||
return formname | ||
end | ||
end | ||
|
||
const VF = VField | ||
|
||
dim(ω::Form) = ω.dim | ||
isdual(ω::Form) = ω.isdual | ||
space(ω::Form) = ω.space | ||
export space | ||
|
||
isdual(v::VField) = v.isdual | ||
space(v::VField) = v.space | ||
|
||
# convenience functions | ||
PrimalForm(i::Int, space::Space) = Form(i, false, space) | ||
export PrimalForm | ||
|
||
DualForm(i::Int, space::Space) = Form(i, true, space) | ||
export DualForm | ||
|
||
PrimalVF(space::Space) = VF(false, space) | ||
export PrimalVF | ||
|
||
DualVF(space::Space) = VF(true, space) | ||
export DualVF | ||
|
||
# show methods | ||
show_duality(ω::Form) = isdual(ω) ? "dual" : "primal" | ||
|
||
function Base.show(io::IO, ω::Form) | ||
print(io, isdual(ω) ? "DualForm($(dim(ω))) on $(space(ω))" : "PrimalForm($(dim(ω))) on $(space(ω))") | ||
end | ||
|
||
# TODO: VField | ||
@nospecialize | ||
function +(s1::Sort, s2::Sort) | ||
@match (s1, s2) begin | ||
(Scalar(), Scalar()) => Scalar() | ||
(Scalar(), Form(i, isdual, space)) || | ||
(Form(i, isdual, space), Scalar()) => Form(i, isdual, space) | ||
(Form(i1, isdual1, space1), Form(i2, isdual2, space2)) => | ||
if (i1 == i2) && (isdual1 == isdual2) && (space1 == space2) | ||
Form(i1, isdual1, space1) | ||
else | ||
throw(SortError( | ||
""" | ||
Can not add two forms of different dimensions/dualities/spaces: | ||
$((i1,isdual1,space1)) and $((i2,isdual2,space2)) | ||
""") | ||
) | ||
end | ||
end | ||
end | ||
|
||
# Type-checking inverse of addition follows addition | ||
-(s1::Sort, s2::Sort) = +(s1, s2) | ||
|
||
# TODO error for Forms | ||
|
||
# Negation is always valid | ||
-(s::Sort) = s | ||
|
||
# TODO: VField | ||
@nospecialize | ||
function *(s1::Sort, s2::Sort) | ||
@match (s1, s2) begin | ||
(Scalar(), Scalar()) => Scalar() | ||
(Scalar(), Form(i, isdual, space)) || | ||
(Form(i, isdual, space), Scalar()) => Form(i, isdual, space) | ||
(Form(_, _, _), Form(_, _, _)) => | ||
throw(SortError("Cannot scalar multiply a form with a form. Maybe try `∧`??")) | ||
end | ||
end | ||
|
||
const SUBSCRIPT_DIGIT_0 = '₀' | ||
|
||
as_sub(n::Int) = join(map(d -> SUBSCRIPT_DIGIT_0 + d, digits(n))) | ||
|
||
# TODO: VField | ||
@nospecialize | ||
function ∧(s1::Sort, s2::Sort) | ||
@match (s1, s2) begin | ||
(Form(i, isdual, space), Scalar()) || (Scalar(), Form(i, isdual, space)) => | ||
Form(i, isdual, space) | ||
(Form(i1, isdual1, space1), Form(i2, isdual2, space2)) => begin | ||
(isdual1 == isdual2) && (space1 == space2) || throw(SortError("Can only take a wedge product of two forms of the same duality on the same space")) | ||
if i1 + i2 <= dim(space1) | ||
Form(i1 + i2, isdual1, space1) | ||
else | ||
throw(SortError("Can only take a wedge product when the dimensions of the forms add to less than n, where n = $(dim(space1)) is the dimension of the ambient space: tried to wedge product $i1 and $i2")) | ||
end | ||
end | ||
(VF(isdual, _), _) || (_, VF(isdual, _)) => throw(SortError("Can only take a wedge product of forms. Flatten (♭) your vector field before applying")) | ||
end | ||
end | ||
|
||
function Base.nameof(::typeof(∧), s1, s2) | ||
Symbol("∧$(as_sub(dim(s1)))$(as_sub(dim(s2)))") | ||
end | ||
|
||
@nospecialize | ||
∂ₜ(s::Sort) = s | ||
|
||
@nospecialize | ||
function d(s::Sort) | ||
@match s begin | ||
Scalar() => throw(SortError("Cannot take exterior derivative of a scalar")) | ||
Form(i, isdual, space) => | ||
if i < dim(space) | ||
Form(i + 1, isdual, space) | ||
else | ||
throw(SortError("Cannot take exterior derivative of a k-form for k >= n, where n = $(dim(space)) is the dimension of its ambient space")) | ||
end | ||
end | ||
end | ||
|
||
@alias d, (d₀, d₁) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. maybe using an oriented syntax like
|
||
|
||
Base.nameof(::typeof(d), s) = Symbol("d$(as_sub(dim(s)))") | ||
|
||
@nospecialize | ||
function ⋆(s::Sort) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. oh I didn't know about \bigstar. We should use that for hodge star going forward. |
||
@match s begin | ||
Scalar() => throw(SortError("Cannot take Hodge star of a scalar")) | ||
VF(isdual, space) => throw(SortError("Cannot take the Hodge star of a vector field")) | ||
Form(i, isdual, space) => Form(dim(space) - i, !isdual, space) | ||
end | ||
end | ||
export ⋆ | ||
|
||
@alias ⋆, (⋆₀, ⋆₁, ⋆₂, ⋆₀⁻¹, ⋆₁⁻¹, ⋆₂⁻¹) | ||
|
||
function Base.nameof(::typeof(⋆), s) | ||
inv = isdual(s) ? "⁻¹" : "" | ||
Symbol("★$(as_sub(isdual(s) ? dim(space(s)) - dim(s) : dim(s)))$(inv)") | ||
end | ||
|
||
@nospecialize | ||
function ι(s1::Sort, s2::Sort) | ||
@match (s1, s2) begin | ||
(VF(true, space), Form(i, true, space)) => Form(i, false, space) # wrong | ||
(VF(true, space), Form(i, false, space)) => DualForm(i, true, space) | ||
_ => throw(SortError("Can only define the discrete interior product on: | ||
PrimalVF, DualForm(i) | ||
DualVF(), PrimalForm(i) | ||
.")) | ||
end | ||
end | ||
|
||
# in practice, a scalar may be treated as a constant 0-form. | ||
function ♯(s::Sort) | ||
@match s begin | ||
Scalar() => PrimalVF() | ||
Form(1, isdual, space) => VF(isdual, space) | ||
_ => throw(SortError("Can only take ♯ to 1-forms")) | ||
end | ||
end | ||
# musical isos may be defined for any combination of (primal/dual) form -> (primal/dual) vf. | ||
|
||
Base.nameof(::typeof(♯), s) = Symbol("♯$s") | ||
|
||
function ♭(s::Sort) | ||
@match s begin | ||
VF(true, space) => Form(1, false, space) | ||
_ => throw(SortError("Can only apply ♭ to dual vector fields")) | ||
end | ||
end | ||
|
||
Base.nameof(::typeof(♭), s) = Symbol("♭$s") | ||
|
||
# OTHER | ||
|
||
function ♭♯(s::Sort) | ||
@match s begin | ||
Form(i, isdual, space) => Form(i, !isdual, space) | ||
_ => throw(SortError("♭♯ is only defined on forms.")) | ||
end | ||
end | ||
|
||
# Δ = ★d⋆d, but we check signature here to throw a more helpful error | ||
function Δ(s::Sort) | ||
@match s begin | ||
Scalar() => Scalar() | ||
Form(0, isdual, space) => Form(0, isdual, space) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. There is a 1-laplacian for 1 forms. And lapl of a scalar should be the scalar 0. |
||
Form(1, isdual, space) => Form(1, isdual, space) | ||
_ => throw(SortError("Δ is not defined for $s")) | ||
end | ||
end | ||
|
||
Base.nameof(::typeof(Δ), s) = Symbol("Δ") | ||
|
||
end |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Do we want the nameof to be the symbol for the alias or the rep? This might break roundtriping because when you print something that used an alias, it will be shown as the rep.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good call, I'll update