diff --git a/src/all_building_blocks/all_building_blocks.jl b/src/all_building_blocks/all_building_blocks.jl index b7f72585e7e0d61672778268e49f9f6b84d40c6d..9581851901a69676cc2b84977a54e08d7c5f21cd 100644 --- a/src/all_building_blocks/all_building_blocks.jl +++ b/src/all_building_blocks/all_building_blocks.jl @@ -1,4 +1,10 @@ module AllBuildingBlocks +include("base_building_blocks.jl") +include("building_blocks.jl") +include("trapezoids.jl") +import .BaseBuildingBlocks: waveform, waveform_sequence, events, BaseBuildingBlock +import .BuildingBlocks: BuildingBlock +import .Trapezoids: Trapezoid, SliceSelect, LineReadout end \ No newline at end of file diff --git a/src/all_building_blocks/trapezoids.jl b/src/all_building_blocks/trapezoids.jl new file mode 100644 index 0000000000000000000000000000000000000000..e3678e8b702c57c04bd5e025c0e1ea99da09516b --- /dev/null +++ b/src/all_building_blocks/trapezoids.jl @@ -0,0 +1,198 @@ +""" +Defines a set of different options for MRI gradients. +""" +module Trapezoids + +import JuMP: @constraint +import StaticArrays: SVector +import LinearAlgebra: norm +import ...Variables: qvec, rise_time, flat_time, slew_rate, gradient_strength, variables, duration, δ, get_free_variable, VariableType, inverse_bandwidth, effective_time, qval_square, duration, set_simple_constraints!, scanner_constraints!, inverse_slice_thickness +import ...Variables: Variables, all_variables_symbols, dwell_time, inverse_fov, inverse_voxel_size, fov, voxel_size +import ...BuildSequences: global_model +import ...Components: ChangingGradient, ConstantGradient, NoGradient, RFPulseComponent, ADC +import ..BaseBuildingBlocks: BaseBuildingBlock + + +""" +Parent type for any [`BuildingBlock`](@ref) that has a trapezoidal gradient waveform. + +Sub-types: +- [`Trapezoid`](@ref) +- [`SliceSelect`](@ref) +- [`LineReadout`](@ref) + +The `N` indicates whether the gradient has a fixed orientation (N=1) or is free (N=3). +""" +abstract type BaseTrapezoid{N} <: BaseBuildingBlock end + +""" + Trapezoid(; orientation=nothing, group=nothing, variables...) + +Defines a trapezoidal pulsed gradient + +## Parameters +- `orientation` sets the gradient orienation (completely free by default). Can be set to a vector for a fixed orientation. +- `group`: assign the trapezoidal gradient to a specific group. This group will be used to scale or rotate the gradients after optimisation. + +## Variables +Variables can be set during construction or afterwards as an attribute. +If not set, they will be determined during the sequence optimisation. +### Timing variables +- [`rise_time`](@ref): Time of the gradient to reach from 0 to maximum in ms. If explicitly set to 0, the scanner slew rate will be ignored. +- [`flat_time`](@ref): Time that the gradient stays at maximum strength in ms. +- [`δ`](@ref): effective pulse duration (`rise_time` + `flat_time`) in ms. +- [`duration`](@ref): total pulse duration (2 * `rise_time` + `flat_time`) in ms. +### Gradient variables +- [`gradient_strength`](@ref): Maximum gradient strength achieved during the pulse in kHz/um +- [`qval`](@ref)/[`qvec`](@ref): Spatial scale on which spins will be dephased due to this pulsed gradient in rad/um (given by `δ` * `gradient_strength`). + +The `bvalue` can be constrained for multiple gradient pulses by creating a `Pathway`. +""" +abstract type Trapezoid{N} <: BaseTrapezoid{N} end + +struct Trapezoid1D <: Trapezoid{1} + rise_time :: VariableType + flat_time :: VariableType + slew_rate :: VariableType + orientation :: SVector{3, Float64} + group :: Union{Nothing, Symbol} +end + +struct Trapezoid3D <: Trapezoid{3} + rise_time :: VariableType + flat_time :: VariableType + slew_rate :: SVector{3, VariableType} + group :: Union{Nothing, Symbol} +end + +function (::Type{Trapezoid})(; orientation=nothing, rise_time=nothing, flat_time=nothing, group=nothing, slew_rate=nothing, kwargs...) + if isnothing(orientation) + if isnothing(slew_rate) + slew_rate = (nothing, nothing, nothing) + end + res = Trapezoid3D( + get_free_variable(rise_time), + get_free_variable(flat_time), + get_free_variable.(slew_rate), + group + ) + else + res = Trapezoid1D( + get_free_variable(rise_time), + get_free_variable(flat_time), + get_free_variable(slew_rate), + orientation, + group + ) + @contraint global_model() res.slew_rate >= 0 + end + + set_simple_constraints!(res, kwargs) + + @constraint global_model() flat_time >= 0 + @constraint global_model() rise_time >= 0 + scanner_constraints!(res) + return res +end + +Base.keys(::Trapezoid) = (Val(:rise), Val(:flat), Val(:fall)) + +Base.getindex(pg::BaseTrapezoid{N}, ::Val{:rise}) where {N} = ChangingGradientBlock(N == 3 ? zeros(3) : 0., slew_rate(pg), rise_time(pg)) +Base.getindex(pg::BaseTrapezoid, ::Val{:flat}) = ConstantGradientBlock(gradient_strength(pg), flat_time(pg)) +Base.getindex(pg::BaseTrapezoid, ::Val{:fall}) = ChangingGradientBlock(gradient_strength(pg), -slew_rate(pg), rise_time(pg)) + +rise_time(pg::Trapezoid) = pg.rise_time +flat_time(pg::Trapezoid) = pg.flat_time +gradient_strength(g::Trapezoid) = slew_rate(g) .* rise_time(g) +slew_rate(g::Trapezoid) = g.slew_rate +δ(g::Trapezoid) = rise_time(g) + flat_time(g) +duration(g::Trapezoid) = 2 * rise_time(g) + flat_time(g) + +for func in (:rise_time, :flat_time, :gradient_strength, :slew_rate, :δ, :duration, :qvec) + @eval $func(bt::BaseTrapezoid) = $func(bt.trapezoid) +end + +qvec(g::BaseTrapezoid, ::Nothing, ::Nothing) = δ(g) .* gradient_strength(g) .* 2π + +""" + SliceSelect(pulse; orientation=nothing, group=nothing, variables...) + +Defines a trapezoidal gradient with a pulse played out during the flat time. + +Parameters and variables are identical as for [`TrapezoidGradient`](@ref) with the addition of: + +## Parameters +- `pulse`: sub-type of [`RFPulseComponent`](@ref) that describes the RF pulse. + +## Variables +- `slice_thickness`: thickness of the selected slice in mm +""" +struct SliceSelect{N} <: BaseTrapezoid{N} + trapezoid :: Trapezoid{N} + pulse :: RFPulseComponent +end + +function SliceSelect(pulse::RFPulseComponent; orientation=nothing, rise_time=nothing, group=nothing, slew_rate=nothing, variables...) + res = SliceSelect( + Trapezoid(; orientation=orientation, rise_time=rise_time, flat_time=duration(pulse), group=group, slew_rate=slew_rate), + pulse + ) + set_simple_constraints!(res, variables) + return res +end + +Base.keys(::SliceSelect) = (Val(:rise), Val(:flat), Vale(:pulse), Val(:fall)) +Base.getindex(pg::SliceSelect, ::Val{:pulse}) = pg.pulse + +inverse_slice_thickness(ss::SliceSelect) = 1e3 * gradient_strength(ss.trapezoid) .* inverse_bandwidth(ss.pulse) + +for func in all_variables_symbols[:pulse] + if func in (:inverse_slice_thickness, :slice_thickness) + continue + end + Variables.$func(ss::SliceSelect) = Variables.$func(ss.pulse) +end + +""" + LineReadout(adc; ramp_overlap=1., orientation=nothing, group=nothing, variables...) + +Defines a trapezoidal gradient with an ADC readout overlaid. + +Parameters and variables are identical as for [`TrapezoidGradient`](@ref) with the addition of: + +## Parameters +- `adc`: [`ADC`](@ref) object that describes the readout. +- `ramp_overlap`: how much the gradient ramp should overlap with the ADC. 0 for no overlap, 1 for full overlap (default: 1). Can be set to `nothing` to become a free variable. + +## Variables +- [`fov`](@ref): FOV of the output image along this single k-space line in mm. +- [`voxel_size`](@ref): size of each voxel along this single k-space line in mm. +""" +struct LineReadout{N} <: BaseTrapezoid{N} + trapezoid :: Trapezoid{N} + adc :: ADC + ramp_overlap :: VariableType +end + +function LineReadout(adc::ADC; ramp_overlap=1., orientation=nothing, rise_time=nothing, flat_time=nothing, group=nothing, slew_rate=nothing, variables...) + res = LineReadout( + Trapezoid(; orientation=orientation, rise_time=rise_time, flat_time=flat_time, group=group, slew_rate=slew_rate), + adc, + get_free_variable(ramp_overlap) + ) + set_simple_constraints!(res, variables) + @constraint global_model() (res.ramp_overlap * rise_time(res.grad) * 2 + flat_time(res.grad)) == duration(res.adc) +end + +ramp_overlap(lr::LineReadout) = lr.ramp_overlap +inverse_fov(lr::LineReadout) = @. 1e3 * dwell_time(lr.adc) * gradient_strength(lr.trapezoid) +inverse_voxel_size(lr::LineReadout) = @. 1e3 * duration(lr.adc) * gradient_strength(lr.trapezoid) + +for func in all_variables_symbols[:readout] + if func in (:inverse_fov, :slice_fov, :inverse_voxel_size, :slice_voxel_size, :ramp_overlap) + continue + end + Variables.$func(lr::LineReadout) = Variables.$func(lr.adc) +end + +end \ No newline at end of file diff --git a/src/components/components.jl b/src/components/components.jl index 4bab60504af5d547d80bb2efaba32c0feee50383..f34b1a69fe9d37fe6784ad4aac005915733f33fc 100644 --- a/src/components/components.jl +++ b/src/components/components.jl @@ -3,10 +3,12 @@ include("abstract_types.jl") include("gradient_waveforms/gradient_waveforms.jl") include("instant_gradients.jl") include("pulses/pulses.jl") +include("readouts/readouts.jl") import .AbstractTypes: BaseComponent, GradientWaveform, EventComponent, RFPulseComponent, ReadoutComponent import .GradientWaveforms: ConstantGradientBlock, ChangingGradientBlock, NoGradientBlock import .InstantGradients: InstantGradient1D, InstantGradient3D import .Pulses: GenericPulse, InstantRFPulse, SincPulse, ConstantPulse +import .Readouts: ADC, SingleReadout end \ No newline at end of file diff --git a/src/overlapping/gradient_pulses/trapezoid_gradients.jl b/src/overlapping/gradient_pulses/trapezoid_gradients.jl deleted file mode 100644 index 595c85ba39754a0226ea6e0ca164ce6c54856619..0000000000000000000000000000000000000000 --- a/src/overlapping/gradient_pulses/trapezoid_gradients.jl +++ /dev/null @@ -1,124 +0,0 @@ -""" -Defines a set of different options for MRI gradients. -""" -module TrapezoidGradients - -import JuMP: @constraint, @variable, VariableRef, value -import StaticArrays: SVector -import LinearAlgebra: norm -import ....BuildingBlocks: VariableNotAvailable -import ....Variables: qvec, rise_time, flat_time, slew_rate, gradient_strength, variables, duration, δ, get_free_variable, VariableType, inverse_slice_thickness, inverse_bandwidth, effective_time, qval_square -import ....BuildingBlocks: duration, set_simple_constraints!, RFPulseBlock, scanner_constraints! -import ....BuildSequences: global_model -import ....Gradients: ChangingGradientBlock, ConstantGradientBlock -import ...Abstract: interruptions, waveform_sequence, AbstractOverlapping - - -""" - TrapezoidGradient(; orientation=:bvec, rotate=nothing, scale=nothing, pulse=nothing, variables...) - -Defines a trapezoidal pulsed gradient - -## Parameters -- `orientation` sets the gradient orienation (completely free by default). Can be set to a vector for a fixed orientation. -- `rotate`: with which user-set parameter will this gradient be rotated (e.g., :bvec). Default is no rotation. -- `scale`: with which user-set parameter will this gradient be scaled (e.g., :bval). Default is no scaling. Can be set to a tuple (e.g., (nothing, :phase_encode, nothing)) to only scale the gradient along specific axes. Such scaling should be applied before any rotation. -- `pulse`: RF pulse that will play during the flat part of the trapezoidal gradient. If you want to add dead time around the pulse, you can set it to a tuple (e.g., `pulse=(2, pulse, :min)`). - -## Variables -Variables can be set during construction or afterwards as an attribute. -If not set, they will be determined during the sequence optimisation. -### Timing variables -- [`rise_time`](@ref): Time of the gradient to reach from 0 to maximum in ms. If explicitly set to 0, the scanner slew rate will be ignored. -- [`flat_time`](@ref): Time that the gradient stays at maximum strength in ms. -- [`δ`](@ref): effective pulse duration (`rise_time` + `flat_time`) in ms. -- [`duration`](@ref): total pulse duration (2 * `rise_time` + `flat_time`) in ms. -### Gradient variables -- [`gradient_strength`](@ref): Maximum gradient strength achieved during the pulse in kHz/um -- [`qval`](@ref)/[`qvec`](@ref): Spatial scale on which spins will be dephased due to this pulsed gradient in rad/um (given by `δ` * `gradient_strength`). -### Pulse variables -Any variables defined for the specific pulse added. Also: -- [`slice_thickness`](@ref): vector with the slice thickness in mm. - -The [`bvalue`](@ref) can be constrained for multiple gradient pulses. -""" -struct TrapezoidGradient <: AbstractOverlapping - rise_time :: VariableType - flat_time :: VariableType - slew_rate :: SVector{3, VariableType} - rotate :: Union{Nothing, Symbol} - scale :: Union{Nothing, Symbol} - time_before_pulse :: VariableType - pulse :: Union{Nothing, RFPulseBlock} - time_after_pulse :: VariableType - slew_rate_1d :: Union{Nothing, VariableType} -end - -function TrapezoidGradient(; orientation=nothing, rise_time=nothing, flat_time=nothing, rotate=nothing, scale=nothing, pulse=nothing,kwargs...) - if isnothing(orientation) - rate_1d = nothing - slew_rate = ( - get_free_variable(nothing), - get_free_variable(nothing), - get_free_variable(nothing), - ) - else - rate_1d = get_free_variable(nothing) - @constraint global_model() rate_1d >= 0 - slew_rate = rate_1d .* (orientation ./ norm(orientation)) - end - slew_rate = SVector{3}(slew_rate) - rise_time = get_free_variable(rise_time) - if isnothing(pulse) - flat_time = get_free_variable(flat_time) - time_before_pulse = time_after_pulse = 0. - elseif pulse isa RFPulseBlock - flat_time = duration(pulse) - time_before_pulse = time_after_pulse = 0. - else pulse isa Tuple - time_before_pulse, pulse, time_after_pulse = pulse - flat_time = time_before_pulse + duration(pulse) + time_after_pulse - end - - res = TrapezoidGradient( - rise_time, - flat_time, - slew_rate, - rotate, - scale, - time_before_pulse, - pulse, - time_after_pulse, - rate_1d - ) - - set_simple_constraints!(res, kwargs) - @constraint global_model() flat_time >= 0 - @constraint global_model() rise_time >= 0 - scanner_constraints!(res) - return res -end - -waveform_sequence(pg::TrapezoidGradient) = [ - ChangingGradientBlock(zeros(3), slew_rate(pg), rise_time(pg), pg.rotate, pg.scale), - ConstantGradientBlock(gradient_strength(pg), flat_time(pg), pg.rotate, pg.scale), - ChangingGradientBlock(gradient_strength(pg), -slew_rate(pg), rise_time(pg), pg.rotate, pg.scale), -] - -interruptions(pg::TrapezoidGradient) = isnothing(pg.pulse) ? [] : [(index=2, time=pg.time_before_pulse + effective_time(pg.pulse), object=pg.pulse)] - -rise_time(pg::TrapezoidGradient) = pg.rise_time -flat_time(pg::TrapezoidGradient) = pg.flat_time -gradient_strength(g::TrapezoidGradient) = slew_rate(g) .* rise_time(g) -slew_rate(g::TrapezoidGradient) = g.slew_rate -δ(g::TrapezoidGradient) = rise_time(g) + flat_time(g) -duration(g::TrapezoidGradient) = 2 * rise_time(g) + flat_time(g) -qvec(g::TrapezoidGradient, ::Nothing, ::Nothing) = δ(g) .* gradient_strength(g) .* 2π -function inverse_slice_thickness(g::TrapezoidGradient) - if isnothing(g.pulse) - throw(VariableNotAvailable(typeof(g), inverse_slice_thickness)) - end - return inverse_bandwidth(g) .* gradient_strength(g) .* 1000 -end - -end \ No newline at end of file