From c14ee7fbdbb7211b551ab01a25eb247f5ef16e07 Mon Sep 17 00:00:00 2001 From: Michiel Cottaar <michiel.cottaar@ndcn.ox.ac.uk> Date: Sat, 27 Jan 2024 18:49:01 +0000 Subject: [PATCH] Fix containers and sequences --- src/MRIBuilder.jl | 4 ++-- src/building_blocks.jl | 6 ++--- src/containers/containers.jl | 2 +- src/containers/fixed_blocks.jl | 2 ++ src/containers/sequences.jl | 41 ++++++++++++++++++++++++---------- 5 files changed, 37 insertions(+), 18 deletions(-) diff --git a/src/MRIBuilder.jl b/src/MRIBuilder.jl index 5f75a24..09a90fc 100644 --- a/src/MRIBuilder.jl +++ b/src/MRIBuilder.jl @@ -32,8 +32,8 @@ export ConcreteBlocks, AbstractConcreteBlock import .Wait: WaitBlock export WaitBlock -import .Containers: Sequence, FixedBlock -export Sequence, FixedBlock +import .Containers: Sequence, FixedBlock, FixedSequence +export Sequence, FixedBlock, FixedSequence import .Gradients: PulsedGradient, InstantGradientBlock, FixedGradient export PulsedGradient, InstantGradientBlock, FixedGradient diff --git a/src/building_blocks.jl b/src/building_blocks.jl index a28ba2b..ba927b0 100644 --- a/src/building_blocks.jl +++ b/src/building_blocks.jl @@ -9,7 +9,7 @@ Parent type for all individual components out of which a sequence can be built. Required methods: - [`duration`](@ref)(block, parameters): Return block duration in ms. -- [`fixed`](block): Return an equivalent fixed BuildingBlock (i.e., `FixedBlock`, `FixedPulse`, `FixedGradient`, `FixedInstantPulse`, `FixedInstantGradient`, or `InstantReadout`). These all have in common that they have no free variables and explicitly set any gradient and RF pulse profiles. +- [`fixed`](block): Return the equivalent fixed BuildingBlock (i.e., `FixedBlock`, `FixedPulse`, `FixedGradient`, `FixedInstantPulse`, `FixedInstantGradient`, or `InstantReadout`). These all have in common that they have no free variables and explicitly set any gradient and RF pulse profiles. - [`variables`](@ref): A list of all functions that are used to compute variables of the building block. Any of these can be used in constraints or objective functions. """ abstract type BuildingBlock end @@ -71,9 +71,9 @@ to_block(bb::BuildingBlock) = bb """ fixed(block::BuildingBlock) -Return an equivalent fixed BuildingBlock. +Return the fixed equivalent of the `BuildingBlock` -Possible return types are `FixedBlock`, `FixedPulse`, `FixedGradient`, `FixedInstantPulse`, `FixedInstantGradient`, or `InstantReadout`. +Possible return types are `FixedSequence`, `FixedBlock`, `FixedPulse`, `FixedGradient`, `FixedInstantPulse`, `FixedInstantGradient`, or `InstantReadout`. These all have in common that they have no free variables and explicitly set any gradient and RF pulse profiles. """ function fixed end diff --git a/src/containers/containers.jl b/src/containers/containers.jl index 28e1a9e..9ef11dd 100644 --- a/src/containers/containers.jl +++ b/src/containers/containers.jl @@ -11,5 +11,5 @@ include("sequences.jl") import ..BuildingBlocks: ContainerBlock, get_children_blocks import ..FixedBlocks: FixedBlock -import .Sequences: Sequence +import .Sequences: Sequence, FixedSequence end \ No newline at end of file diff --git a/src/containers/fixed_blocks.jl b/src/containers/fixed_blocks.jl index 086f725..6443f63 100644 --- a/src/containers/fixed_blocks.jl +++ b/src/containers/fixed_blocks.jl @@ -5,6 +5,7 @@ import ...BuildingBlocks: ContainerBlock import ...Gradients: FixedGradient import ...Pulses: FixedPulse import ...Variables: duration, variables +import ...WaitBlocks: WaitBlock """ FixedBlock(duration=nothing; pulse_delay=0., pulse=nothing, gradient_delay=0., gradient=nothing, readout_times=[]) @@ -47,6 +48,7 @@ duration(fb::FixedBlock) = fb.duration variables(::Type{<:FixedBlock}) = [] +fixed(wb::WaitBlock) = FixedBlock(value(duration(wb))) end \ No newline at end of file diff --git a/src/containers/sequences.jl b/src/containers/sequences.jl index 0422e06..6692d51 100644 --- a/src/containers/sequences.jl +++ b/src/containers/sequences.jl @@ -5,7 +5,10 @@ module Sequences import JuMP: Model, @constraint import ...BuildSequences: @global_model_constructor import ...Variables: variables, start_time, duration, VariableType, get_free_variable, TR -import ...BuildingBlocks: BuildingBlock, ContainerBlock, to_block, get_children_blocks, scanner_constraints! +import ...BuildingBlocks: BuildingBlock, ContainerBlock, to_block, get_children_blocks, scanner_constraints!, fixed +import ..FixedBlocks: FixedBlock + +abstract type AbstractSequence <: ContainerBlock end """ Sequence(building_blocks...; TR=nothing, scanner=nothing) @@ -21,7 +24,7 @@ or be embedded as a [`BuildingBlock`](@ref) into higher-order `Sequence` or othe ## Variables - [`TR`](@ref): repetition time of sequence in ms. """ -struct Sequence <: ContainerBlock +struct Sequence <: AbstractSequence model :: Model blocks :: Vector{<:BuildingBlock} TR :: VariableType @@ -43,9 +46,9 @@ end Sequence(model::Model, blocks...; TR=nothing, scanner=nothing) = Sequence(model, [blocks...]; TR=TR, scanner=scanner) -Base.length(seq::Sequence) = length(seq.blocks) -Base.getindex(seq::Sequence, index) = seq.blocks[index] -get_children_blocks(seq::Sequence) = enumerate(seq.blocks) +Base.length(seq::AbstractSequence) = length(seq.blocks) +Base.getindex(seq::AbstractSequence, index) = seq.blocks[index] +get_children_blocks(seq::AbstractSequence) = enumerate(seq.blocks) """ start_time(sequence::Sequence, index::Integer, args...) @@ -54,9 +57,9 @@ Returns the starting time of the [`BuildingBlock`](@ref) with index `index`. Additional `args` can be used to select a sub-block of that [`BuildingBlock`](@ref). The starting time is returned with respect to the start of this sequence. """ -start_time(seq::Sequence) = 0. -start_time(seq::Sequence, index::Integer) = iszero(index) ? start_time(seq) : (start_time(seq, index-1) + duration(seq[index])) -start_time(seq::Sequence, index::Integer, args...) = start_time(seq, index) + start_time(seq[index], args...) +start_time(seq::AbstractSequence) = 0. +start_time(seq::AbstractSequence, index::Integer) = iszero(index) ? start_time(seq) : (start_time(seq, index-1) + duration(seq[index])) +start_time(seq::AbstractSequence, index::Integer, args...) = start_time(seq, index) + start_time(seq[index], args...) """ end_time(sequence::Sequence, index::Integer, args...) @@ -65,15 +68,29 @@ Returns the end time of the [`BuildingBlock`](@ref) with index `index`. Additional `args` can be used to select a sub-block of that [`BuildingBlock`](@ref). The end time is returned with respect to the start of this sequence. """ -end_time(seq::Sequence, index::Integer) = start_time(seq, index) + duration(seq[index]) -end_time(seq::Sequence, index::Integer, args...) = start_time(seq, index) + end_time(seq[index], args...) +end_time(seq::AbstractSequence, index::Integer) = start_time(seq, index) + duration(seq[index]) +end_time(seq::AbstractSequence, index::Integer, args...) = start_time(seq, index) + end_time(seq[index], args...) -duration(seq::Sequence) = end_time(seq, length(seq)) +duration(seq::AbstractSequence) = end_time(seq, length(seq)) -TR(seq::Sequence) = seq.TR +TR(seq::AbstractSequence) = seq.TR variables(::Type{<:Sequence}) = [TR] +struct FixedSequence <: AbstractSequence + blocks :: Vector{<:BuildingBlock} + TR :: Number +end + +function fixed(seq::Sequence) + FixedSequence( + fixed.(seq.blocks), + value(TR(seq)) + ) +end + +variables(::Type{<:FixedSequence}) = [] + end -- GitLab