Skip to content

Commit 4f81980

Browse files
authored
Introduce Space, Dimensions, and GeneralDimensions (#360)
1 parent 6716a52 commit 4f81980

33 files changed

+779
-334
lines changed

CHANGELOG.md

+4
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
99

1010
- Change the structure of block diagonalization functions, using `BlockDiagonalForm` struct and changing the function name from `bdf` to `block_diagonal_form`. ([#349])
1111
- Add **GPUArrays** compatibility for `ptrace` function, by using **KernelAbstractions.jl**. ([#350])
12+
- Introduce `Space`, `Dimensions`, `GeneralDimensions` structures to support wider definitions and operations of `Qobj/QobjEvo`, and potential functionalities in the future. ([#271], [#353], [#360])
1213

1314
## [v0.24.0]
1415
Release date: 2024-12-13
@@ -55,6 +56,7 @@ Release date: 2024-11-13
5556
[v0.24.0]: https://github.com/qutip/QuantumToolbox.jl/releases/tag/v0.24.0
5657
[#86]: https://github.com/qutip/QuantumToolbox.jl/issues/86
5758
[#139]: https://github.com/qutip/QuantumToolbox.jl/issues/139
59+
[#271]: https://github.com/qutip/QuantumToolbox.jl/issues/271
5860
[#292]: https://github.com/qutip/QuantumToolbox.jl/issues/292
5961
[#306]: https://github.com/qutip/QuantumToolbox.jl/issues/306
6062
[#309]: https://github.com/qutip/QuantumToolbox.jl/issues/309
@@ -71,3 +73,5 @@ Release date: 2024-11-13
7173
[#347]: https://github.com/qutip/QuantumToolbox.jl/issues/347
7274
[#349]: https://github.com/qutip/QuantumToolbox.jl/issues/349
7375
[#350]: https://github.com/qutip/QuantumToolbox.jl/issues/350
76+
[#353]: https://github.com/qutip/QuantumToolbox.jl/issues/353
77+
[#360]: https://github.com/qutip/QuantumToolbox.jl/issues/360

docs/src/resources/api.md

+4-1
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,9 @@ CurrentModule = QuantumToolbox
1111
## [Quantum object (Qobj) and type](@id doc-API:Quantum-object-and-type)
1212

1313
```@docs
14+
Space
15+
Dimensions
16+
GeneralDimensions
1417
AbstractQuantumObject
1518
BraQuantumObject
1619
Bra
@@ -73,7 +76,7 @@ LinearAlgebra.diag
7376
proj
7477
ptrace
7578
purity
76-
permute
79+
SparseArrays.permute
7780
tidyup
7881
tidyup!
7982
get_data

ext/QuantumToolboxCUDAExt.jl

+10-8
Original file line numberDiff line numberDiff line change
@@ -10,58 +10,60 @@ import SparseArrays: SparseVector, SparseMatrixCSC
1010
1111
If `A.data` is a dense array, return a new [`QuantumObject`](@ref) where `A.data` is in the type of `CUDA.CuArray` for gpu calculations.
1212
"""
13-
CuArray(A::QuantumObject{Tq}) where {Tq<:Union{Vector,Matrix}} = QuantumObject(CuArray(A.data), A.type, A.dims)
13+
CuArray(A::QuantumObject{Tq}) where {Tq<:Union{Vector,Matrix}} = QuantumObject(CuArray(A.data), A.type, A.dimensions)
1414

1515
@doc raw"""
1616
CuArray{T}(A::QuantumObject)
1717
1818
If `A.data` is a dense array, return a new [`QuantumObject`](@ref) where `A.data` is in the type of `CUDA.CuArray` with element type `T` for gpu calculations.
1919
"""
20-
CuArray{T}(A::QuantumObject{Tq}) where {T,Tq<:Union{Vector,Matrix}} = QuantumObject(CuArray{T}(A.data), A.type, A.dims)
20+
CuArray{T}(A::QuantumObject{Tq}) where {T,Tq<:Union{Vector,Matrix}} =
21+
QuantumObject(CuArray{T}(A.data), A.type, A.dimensions)
2122

2223
@doc raw"""
2324
CuSparseVector(A::QuantumObject)
2425
2526
If `A.data` is a sparse vector, return a new [`QuantumObject`](@ref) where `A.data` is in the type of `CUDA.CUSPARSE.CuSparseVector` for gpu calculations.
2627
"""
27-
CuSparseVector(A::QuantumObject{<:SparseVector}) = QuantumObject(CuSparseVector(A.data), A.type, A.dims)
28+
CuSparseVector(A::QuantumObject{<:SparseVector}) = QuantumObject(CuSparseVector(A.data), A.type, A.dimensions)
2829

2930
@doc raw"""
3031
CuSparseVector{T}(A::QuantumObject)
3132
3233
If `A.data` is a sparse vector, return a new [`QuantumObject`](@ref) where `A.data` is in the type of `CUDA.CUSPARSE.CuSparseVector` with element type `T` for gpu calculations.
3334
"""
34-
CuSparseVector{T}(A::QuantumObject{<:SparseVector}) where {T} = QuantumObject(CuSparseVector{T}(A.data), A.type, A.dims)
35+
CuSparseVector{T}(A::QuantumObject{<:SparseVector}) where {T} =
36+
QuantumObject(CuSparseVector{T}(A.data), A.type, A.dimensions)
3537

3638
@doc raw"""
3739
CuSparseMatrixCSC(A::QuantumObject)
3840
3941
If `A.data` is in the type of `SparseMatrixCSC`, return a new [`QuantumObject`](@ref) where `A.data` is in the type of `CUDA.CUSPARSE.CuSparseMatrixCSC` for gpu calculations.
4042
"""
41-
CuSparseMatrixCSC(A::QuantumObject{<:SparseMatrixCSC}) = QuantumObject(CuSparseMatrixCSC(A.data), A.type, A.dims)
43+
CuSparseMatrixCSC(A::QuantumObject{<:SparseMatrixCSC}) = QuantumObject(CuSparseMatrixCSC(A.data), A.type, A.dimensions)
4244

4345
@doc raw"""
4446
CuSparseMatrixCSC{T}(A::QuantumObject)
4547
4648
If `A.data` is in the type of `SparseMatrixCSC`, return a new [`QuantumObject`](@ref) where `A.data` is in the type of `CUDA.CUSPARSE.CuSparseMatrixCSC` with element type `T` for gpu calculations.
4749
"""
4850
CuSparseMatrixCSC{T}(A::QuantumObject{<:SparseMatrixCSC}) where {T} =
49-
QuantumObject(CuSparseMatrixCSC{T}(A.data), A.type, A.dims)
51+
QuantumObject(CuSparseMatrixCSC{T}(A.data), A.type, A.dimensions)
5052

5153
@doc raw"""
5254
CuSparseMatrixCSR(A::QuantumObject)
5355
5456
If `A.data` is in the type of `SparseMatrixCSC`, return a new [`QuantumObject`](@ref) where `A.data` is in the type of `CUDA.CUSPARSE.CuSparseMatrixCSR` for gpu calculations.
5557
"""
56-
CuSparseMatrixCSR(A::QuantumObject{<:SparseMatrixCSC}) = QuantumObject(CuSparseMatrixCSR(A.data), A.type, A.dims)
58+
CuSparseMatrixCSR(A::QuantumObject{<:SparseMatrixCSC}) = QuantumObject(CuSparseMatrixCSR(A.data), A.type, A.dimensions)
5759

5860
@doc raw"""
5961
CuSparseMatrixCSR(A::QuantumObject)
6062
6163
If `A.data` is in the type of `SparseMatrixCSC`, return a new [`QuantumObject`](@ref) where `A.data` is in the type of `CUDA.CUSPARSE.CuSparseMatrixCSR` with element type `T` for gpu calculations.
6264
"""
6365
CuSparseMatrixCSR{T}(A::QuantumObject{<:SparseMatrixCSC}) where {T} =
64-
QuantumObject(CuSparseMatrixCSR{T}(A.data), A.type, A.dims)
66+
QuantumObject(CuSparseMatrixCSR{T}(A.data), A.type, A.dimensions)
6567

6668
@doc raw"""
6769
cu(A::QuantumObject; word_size::Int=64)

src/QuantumToolbox.jl

+2
Original file line numberDiff line numberDiff line change
@@ -79,6 +79,8 @@ include("progress_bar.jl")
7979
include("linear_maps.jl")
8080

8181
# Quantum Object
82+
include("qobj/space.jl")
83+
include("qobj/dimensions.jl")
8284
include("qobj/quantum_object_base.jl")
8385
include("qobj/quantum_object.jl")
8486
include("qobj/quantum_object_evo.jl")

src/correlations.jl

+2-3
Original file line numberDiff line numberDiff line change
@@ -51,8 +51,7 @@ function correlation_3op_2t(
5151
ψ0 = steadystate(L)
5252
end
5353

54-
allequal((L.dims, ψ0.dims, A.dims, B.dims, C.dims)) ||
55-
throw(DimensionMismatch("The quantum objects are not of the same Hilbert dimension."))
54+
check_dimensions(L, ψ0, A, B, C)
5655

5756
kwargs2 = merge((saveat = collect(tlist),), (; kwargs...))
5857
ρt_list = mesolve(L, ψ0, tlist; kwargs2...).states
@@ -137,7 +136,7 @@ function correlation_2op_2t(
137136
HOpType<:Union{OperatorQuantumObject,SuperOperatorQuantumObject},
138137
StateOpType<:Union{KetQuantumObject,OperatorQuantumObject},
139138
}
140-
C = eye(prod(H.dims), dims = H.dims)
139+
C = eye(prod(H.dimensions), dims = H.dimensions)
141140
if reverse
142141
corr = correlation_3op_2t(H, ψ0, tlist, τlist, c_ops, A, B, C; kwargs...)
143142
else

src/negativity.jl

+19-10
Original file line numberDiff line numberDiff line change
@@ -41,7 +41,7 @@ julia> round(negativity(ρ, 2), digits=2)
4141
```
4242
"""
4343
function negativity::QuantumObject, subsys::Int; logarithmic::Bool = false)
44-
mask = fill(false, length.dims))
44+
mask = fill(false, length.dimensions))
4545
try
4646
mask[subsys] = true
4747
catch
@@ -70,37 +70,46 @@ Return the partial transpose of a density matrix ``\rho``, where `mask` is an ar
7070
# Returns
7171
- `ρ_pt::QuantumObject`: The density matrix with the selected subsystems transposed.
7272
"""
73-
function partial_transpose::QuantumObject{T,OperatorQuantumObject}, mask::Vector{Bool}) where {T}
74-
if length(mask) != length.dims)
73+
function partial_transpose::QuantumObject{DT,OperatorQuantumObject}, mask::Vector{Bool}) where {DT}
74+
if length(mask) != length.dimensions)
7575
throw(ArgumentError("The length of \`mask\` should be equal to the length of \`ρ.dims\`."))
7676
end
7777
return _partial_transpose(ρ, mask)
7878
end
7979

8080
# for dense matrices
81-
function _partial_transpose::QuantumObject{<:AbstractArray,OperatorQuantumObject}, mask::Vector{Bool})
81+
function _partial_transpose::QuantumObject{DT,OperatorQuantumObject}, mask::Vector{Bool}) where {DT<:AbstractArray}
82+
isa.dimensions, GeneralDimensions) &&
83+
(get_dimensions_to(ρ) != get_dimensions_from(ρ)) &&
84+
throw(ArgumentError("Invalid partial transpose for dims = $(_get_dims_string.dimensions))"))
85+
8286
mask2 = [1 + Int(i) for i in mask]
8387
# mask2 has elements with values equal to 1 or 2
8488
# 1 - the subsystem don't need to be transposed
8589
# 2 - the subsystem need be transposed
8690

8791
nsys = length(mask2)
92+
dims = dimensions_to_dims(get_dimensions_to(ρ))
8893
pt_dims = reshape(Vector(1:(2*nsys)), (nsys, 2))
8994
pt_idx = [
90-
[pt_dims[n, mask2[n]] for n in 1:nsys] # origin value in mask2
91-
[pt_dims[n, 3-mask2[n]] for n in 1:nsys] # opposite value in mask2 (1 -> 2, and 2 -> 1)
95+
[pt_dims[n, mask2[n]] for n in 1:nsys] # origin value in mask2
96+
[pt_dims[n, 3-mask2[n]] for n in 1:nsys] # opposite value in mask2 (1 -> 2, and 2 -> 1)
9297
]
9398
return QuantumObject(
94-
reshape(permutedims(reshape.data, (ρ.dims..., ρ.dims...)), pt_idx), size(ρ)),
99+
reshape(permutedims(reshape.data, (dims..., dims...)), pt_idx), size(ρ)),
95100
Operator,
96-
ρ.dims,
101+
Dimensions.dimensions.to),
97102
)
98103
end
99104

100105
# for sparse matrices
101106
function _partial_transpose::QuantumObject{<:AbstractSparseArray,OperatorQuantumObject}, mask::Vector{Bool})
107+
isa.dimensions, GeneralDimensions) &&
108+
(get_dimensions_to(ρ) != get_dimensions_from(ρ)) &&
109+
throw(ArgumentError("Invalid partial transpose for dims = $(_get_dims_string.dimensions))"))
110+
102111
M, N = size(ρ)
103-
dimsTuple = Tuple(ρ.dims)
112+
dimsTuple = Tuple(dimensions_to_dims(get_dimensions_to(ρ)))
104113
colptr = ρ.data.colptr
105114
rowval = ρ.data.rowval
106115
nzval = ρ.data.nzval
@@ -132,5 +141,5 @@ function _partial_transpose(ρ::QuantumObject{<:AbstractSparseArray,OperatorQuan
132141
end
133142
end
134143

135-
return QuantumObject(sparse(I_pt, J_pt, V_pt, M, N), Operator, ρ.dims)
144+
return QuantumObject(sparse(I_pt, J_pt, V_pt, M, N), Operator, ρ.dimensions)
136145
end

0 commit comments

Comments
 (0)