Skip to content
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

[NDTensors] DiagonalArray tensor operations #1226

Merged
merged 43 commits into from
Nov 1, 2023
Merged
Show file tree
Hide file tree
Changes from 29 commits
Commits
Show all changes
43 commits
Select commit Hold shift + click to select a range
9b7f469
[NDTensors] Add more operations for Array storage
mtfishman Oct 27, 2023
cbd0307
Get more Array storage functionality working
mtfishman Oct 27, 2023
2a6b8da
Format
mtfishman Oct 27, 2023
460082d
New similar definition
mtfishman Oct 27, 2023
821dbdf
Merge branch 'main' into NDTensors_arraytensor_ops
mtfishman Oct 30, 2023
ffe3f66
Define contraction of ArrayStorage with Diag
mtfishman Oct 30, 2023
9c571bc
Format
mtfishman Oct 30, 2023
cc3136c
Format
mtfishman Oct 30, 2023
74d722d
Reorganize Combiner code
mtfishman Oct 30, 2023
ed772e1
Merge branch 'NDTensors_arraytensor_ops' of github.com:ITensor/ITenso…
mtfishman Oct 30, 2023
a40e4a3
Get eigen working with Array storage
mtfishman Oct 30, 2023
7c0509b
Improve eigen code a bit
mtfishman Oct 30, 2023
6febc37
Format
mtfishman Oct 30, 2023
2208569
Fix tests
mtfishman Oct 30, 2023
25814f6
Merge branch 'NDTensors_arraytensor_ops' of github.com:ITensor/ITenso…
mtfishman Oct 30, 2023
45e8b87
Format
mtfishman Oct 30, 2023
2801f7a
Format
mtfishman Oct 30, 2023
a07f808
Fix tests
mtfishman Oct 30, 2023
c14150c
Merge branch 'NDTensors_arraytensor_ops' of github.com:ITensor/ITenso…
mtfishman Oct 30, 2023
62339ba
[NDTensors] Add DiagonalArrays submodule
mtfishman Oct 30, 2023
9d4d7eb
Add tests
mtfishman Oct 30, 2023
b537e6a
Format
mtfishman Oct 30, 2023
ba24b1c
Load Compat for allequal
mtfishman Oct 30, 2023
9c457d9
Merge branch 'main' into NDTensors_diagonalarray
mtfishman Oct 30, 2023
e865d27
[NDTensors] DiagonalArray tensor operations
mtfishman Oct 30, 2023
d59fcfb
Merge main
mtfishman Oct 31, 2023
ff0a504
Format
mtfishman Oct 31, 2023
e35483e
Merge branch 'main' into NDTensors_diagonalarray_ops
mtfishman Oct 31, 2023
d714089
Get more DiagonalArray tensor operations working, start improving kwargs
mtfishman Oct 31, 2023
f2d65d2
More work on kwargs
mtfishman Oct 31, 2023
38b32a6
Fix SVD
mtfishman Oct 31, 2023
9c4ce2c
More kwarg refactoring
mtfishman Oct 31, 2023
34b229b
Fix issues with new kwarg design
mtfishman Oct 31, 2023
766f945
Refactor SVD code
mtfishman Oct 31, 2023
970b112
Fix some more tests
mtfishman Oct 31, 2023
ffec9de
Small change to kwarg logic
mtfishman Oct 31, 2023
7668e86
Fix some more tests
mtfishman Oct 31, 2023
5e65813
Format
mtfishman Oct 31, 2023
0e288ea
Fix some more tests
mtfishman Oct 31, 2023
1ef4c5c
Fix bug in truncate
mtfishman Oct 31, 2023
31e6209
Fix some more tests
mtfishman Oct 31, 2023
42fada1
Some more kwarg changes
mtfishman Oct 31, 2023
251f5a0
Fix tests
mtfishman Nov 1, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
28 changes: 12 additions & 16 deletions NDTensors/src/DiagonalArrays/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -5,36 +5,32 @@ A Julia `DiagonalArray` type.
````julia
using NDTensors.DiagonalArrays:
DiagonalArray,
densearray,
diagview,
diaglength,
getdiagindex,
setdiagindex!,
setdiag!,
diagcopyto!

d = DiagonalArray([1., 2, 3], 3, 4, 5)
DiagIndex,
DiagIndices,
densearray

d = DiagonalArray([1.0, 2, 3], 3, 4, 5)
@show d[1, 1, 1] == 1
@show d[2, 2, 2] == 2
@show d[1, 2, 1] == 0

d[2, 2, 2] = 22
@show d[2, 2, 2] == 22

@show diaglength(d) == 3
@show length(d[DiagIndices()]) == 3
@show densearray(d) == d
@show getdiagindex(d, 2) == d[2, 2, 2]
@show d[DiagIndex(2)] == d[2, 2, 2]

setdiagindex!(d, 222, 2)
d[DiagIndex(2)] = 222
@show d[2, 2, 2] == 222

a = randn(3, 4, 5)
new_diag = randn(3)
setdiag!(a, new_diag)
diagcopyto!(d, a)
a[DiagIndices()] = new_diag
d[DiagIndices()] = a[DiagIndices()]

@show diagview(a) == new_diag
@show diagview(d) == new_diag
@show a[DiagIndices()] == new_diag
@show d[DiagIndices()] == new_diag
````

You can generate this README with:
Expand Down
24 changes: 10 additions & 14 deletions NDTensors/src/DiagonalArrays/examples/README.jl
Original file line number Diff line number Diff line change
Expand Up @@ -4,13 +4,9 @@

using NDTensors.DiagonalArrays:
DiagonalArray,
densearray,
diagview,
diaglength,
getdiagindex,
setdiagindex!,
setdiag!,
diagcopyto!
DiagIndex,
DiagIndices,
densearray
mtfishman marked this conversation as resolved.
Show resolved Hide resolved

d = DiagonalArray([1.0, 2, 3], 3, 4, 5)
@show d[1, 1, 1] == 1
Expand All @@ -20,20 +16,20 @@ d = DiagonalArray([1.0, 2, 3], 3, 4, 5)
d[2, 2, 2] = 22
@show d[2, 2, 2] == 22

@show diaglength(d) == 3
@show length(d[DiagIndices()]) == 3
@show densearray(d) == d
@show getdiagindex(d, 2) == d[2, 2, 2]
@show d[DiagIndex(2)] == d[2, 2, 2]

setdiagindex!(d, 222, 2)
d[DiagIndex(2)] = 222
@show d[2, 2, 2] == 222

a = randn(3, 4, 5)
new_diag = randn(3)
setdiag!(a, new_diag)
diagcopyto!(d, a)
a[DiagIndices()] = new_diag
d[DiagIndices()] = a[DiagIndices()]

@show diagview(a) == new_diag
@show diagview(d) == new_diag
@show a[DiagIndices()] == new_diag
@show d[DiagIndices()] == new_diag

# You can generate this README with:
# ```julia
Expand Down
24 changes: 23 additions & 1 deletion NDTensors/src/DiagonalArrays/src/DiagonalArrays.jl
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ module DiagonalArrays
using Compat # allequal
using LinearAlgebra

export DiagonalArray
export DiagonalArray, DiagonalMatrix, DiagonalVector, DiagIndex, DiagIndices, densearray

include("diagview.jl")

Expand All @@ -19,6 +19,9 @@ struct DiagonalArray{T,N,Diag<:AbstractVector{T},Zero} <: AbstractArray{T,N}
zero::Zero
end

const DiagonalVector{T,Diag,Zero} = DiagonalArray{T,1,Diag,Zero}
const DiagonalMatrix{T,Diag,Zero} = DiagonalArray{T,2,Diag,Zero}

function DiagonalArray{T,N}(
diag::AbstractVector{T}, d::Tuple{Vararg{Int,N}}, zero=DefaultZero()
) where {T,N}
Expand Down Expand Up @@ -53,6 +56,25 @@ function DiagonalArray(diag::AbstractVector{T}, d::Vararg{Int,N}) where {T,N}
return DiagonalArray{T,N}(diag, d)
end

default_size(diag::AbstractVector, n) = ntuple(Returns(length(diag)), n)

# Infer size from diagonal
function DiagonalArray{T,N}(diag::AbstractVector) where {T,N}
return DiagonalArray{T,N}(diag, default_size(diag, N))
end

function DiagonalArray{<:Any,N}(diag::AbstractVector{T}) where {T,N}
return DiagonalArray{T,N}(diag)
end

function DiagonalMatrix(diag::AbstractVector)
return DiagonalArray{<:Any,2}(diag)
end

function DiagonalVector(diag::AbstractVector)
return DiagonalArray{<:Any,1}(diag)
end

# undef
function DiagonalArray{T,N}(::UndefInitializer, d::Tuple{Vararg{Int,N}}) where {T,N}
return DiagonalArray{T,N}(Vector{T}(undef, minimum(d)), d)
Expand Down
46 changes: 37 additions & 9 deletions NDTensors/src/DiagonalArrays/src/diagview.jl
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,19 @@ function setdiagindex!(a::AbstractArray, v, i::Integer)
return a
end

struct DiagIndex
I::Int
end

function Base.getindex(a::AbstractArray, i::DiagIndex)
return getdiagindex(a, i.I)
end

function Base.setindex!(a::AbstractArray, v, i::DiagIndex)
setdiagindex!(a, v, i.I)
return a
end

function setdiag!(a::AbstractArray, v)
copyto!(diagview(a), v)
return a
Expand All @@ -28,27 +41,42 @@ function diaglength(a::AbstractArray)
return minimum(size(a))
end

function diagstride(A::AbstractArray)
function diagstride(a::AbstractArray)
s = 1
p = 1
for i in 1:(ndims(A) - 1)
p *= size(A, i)
for i in 1:(ndims(a) - 1)
p *= size(a, i)
s += p
end
return s
end

function diagindices(A::AbstractArray)
diaglength = minimum(size(A))
maxdiag = LinearIndices(A)[CartesianIndex(ntuple(Returns(diaglength), ndims(A)))]
return 1:diagstride(A):maxdiag
function diagindices(a::AbstractArray)
diaglength = minimum(size(a))
maxdiag = LinearIndices(a)[CartesianIndex(ntuple(Returns(diaglength), ndims(a)))]
return 1:diagstride(a):maxdiag
end

function diagview(A::AbstractArray)
return @view A[diagindices(A)]
function diagindices(a::AbstractArray{<:Any,0})
return Base.OneTo(1)
end

function diagview(a::AbstractArray)
return @view a[diagindices(a)]
end

function diagcopyto!(dest::AbstractArray, src::AbstractArray)
copyto!(diagview(dest), diagview(src))
return dest
end

struct DiagIndices end

function Base.getindex(a::AbstractArray, ::DiagIndices)
return diagview(a)
end

function Base.setindex!(a::AbstractArray, v, ::DiagIndices)
setdiag!(a, v)
return a
end
2 changes: 2 additions & 0 deletions NDTensors/src/NDTensors.jl
Original file line number Diff line number Diff line change
Expand Up @@ -151,6 +151,8 @@ include("arraystorage/arraystorage/tensor/eigen.jl")
include("arraystorage/arraystorage/tensor/svd.jl")

# DiagonalArray storage
include("arraystorage/diagonalarray/storage/contract.jl")

include("arraystorage/diagonalarray/tensor/contract.jl")

# BlockSparseArray storage
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ const ArrayStorage{T,N} = Union{
SubArray{T,N},
PermutedDimsArray{T,N},
StridedView{T,N},
DiagonalArray{T,N},
BlockSparseArray{T,N},
}

Expand All @@ -28,3 +29,8 @@ const MatrixOrArrayStorage{T} = Union{MatrixStorage{T},ArrayStorage{T}}
function to_arraystorage(x::DenseTensor)
return tensor(reshape(data(x), size(x)), inds(x))
end

# TODO: Delete once `Diag` is removed.
function to_arraystorage(x::DiagTensor)
return tensor(DiagonalArray(data(x), size(x)), inds(x))
end
5 changes: 3 additions & 2 deletions NDTensors/src/arraystorage/arraystorage/storage/contract.jl
Original file line number Diff line number Diff line change
Expand Up @@ -4,10 +4,11 @@ function contract(
labels1,
array2::MatrixOrArrayStorage,
labels2,
labelsR=contract_labels(labels1, labels2),
labelsR=contract_labels(labels1, labels2);
kwargs...
mtfishman marked this conversation as resolved.
Show resolved Hide resolved
)
output_array = contraction_output(array1, labels1, array2, labels2, labelsR)
contract!(output_array, labelsR, array1, labels1, array2, labels2)
contract!(output_array, labelsR, array1, labels1, array2, labels2; kwargs...)
return output_array
end

Expand Down
3 changes: 1 addition & 2 deletions NDTensors/src/arraystorage/arraystorage/tensor/eigen.jl
Original file line number Diff line number Diff line change
Expand Up @@ -68,7 +68,6 @@ function eigen(
Vinds = indstype((dag(ind(T, 2)), dag(r)))
Dinds = indstype((l, dag(r)))
V = tensor(VM, Vinds)
# TODO: Replace with `DiagonalArray`.
D = tensor(Diag(DM), Dinds)
D = tensor(DiagonalMatrix(DM), Dinds)
return D, V, spec
end
31 changes: 18 additions & 13 deletions NDTensors/src/arraystorage/arraystorage/tensor/svd.jl
Original file line number Diff line number Diff line change
@@ -1,3 +1,10 @@
default_maxdim(a) = minimum(size(a))
default_mindim(a) = true
default_cutoff(a) = zero(eltype(a))
default_svd_alg(a) = "divide_and_conquer"
default_use_absolute_cutoff(a) = false
default_use_relative_cutoff(a) = true

# TODO: Rewrite this function to be more modern:
# 1. Output `Spectrum` as a keyword argument that gets overwritten.
# 2. Dispatch on `alg`.
Expand All @@ -9,25 +16,23 @@ svd of an order-2 DenseTensor
"""
function svd(
T::ArrayStorageTensor;
mindim=default_mindim(T),
maxdim=nothing,
mindim=1,
cutoff=nothing,
alg="divide_and_conquer", # TODO: Define `default_alg(T)`
use_absolute_cutoff=false,
use_relative_cutoff=true,
alg=default_svd_alg(T),
use_absolute_cutoff=default_use_absolute_cutoff(T),
use_relative_cutoff=default_use_relative_cutoff(T),
# These are getting passed erroneously.
# TODO: Make sure they don't get passed down
# to here.
which_decomp=nothing,
tags=nothing,
eigen_perturbation=nothing,
normalize=nothing,
## which_decomp=nothing,
## tags=nothing,
## eigen_perturbation=nothing,
## normalize=nothing,
)
truncate = !isnothing(maxdim) || !isnothing(cutoff)
# TODO: Define `default_maxdim(T)`.
maxdim = isnothing(maxdim) ? minimum(dims(T)) : maxdim
# TODO: Define `default_cutoff(T)`.
cutoff = isnothing(cutoff) ? zero(eltype(T)) : cutoff
maxdim = isnothing(maxdim) ? default_maxdim(T) : maxdim
cutoff = isnothing(cutoff) ? default_cutoff(T) : cutoff

# TODO: Dispatch on `Algorithm(alg)`.
if alg == "divide_and_conquer"
Expand Down Expand Up @@ -97,7 +102,7 @@ function svd(
Sinds = indstype((u, v))
Vinds = indstype((ind(T, 2), v))
U = tensor(MU, Uinds)
S = tensor(Diag(MS), Sinds)
S = tensor(DiagonalMatrix(MS), Sinds)
V = tensor(MV, Vinds)
return U, S, V, spec
end
5 changes: 5 additions & 0 deletions NDTensors/src/arraystorage/arraystorage/tensor/zeros.jl
Original file line number Diff line number Diff line change
Expand Up @@ -11,3 +11,8 @@ end
function zeros(tensortype::Type{<:ArrayStorageTensor}, inds::Tuple{Vararg{Integer}})
return _zeros(tensortype, inds)
end

# To resolve ambiguity error with `Base.zeros`.
function zeros(tensortype::Type{<:ArrayStorageTensor}, inds::Tuple{})
return _zeros(tensortype, inds)
end
Loading
Loading