-
-
Notifications
You must be signed in to change notification settings - Fork 36
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
DirectCR-RSSA #346
base: master
Are you sure you want to change the base?
DirectCR-RSSA #346
Changes from 42 commits
7861bc7
e36f466
5ce7d25
2a8bfa7
0d96e80
07873e8
7fce14e
74346b2
c5f93dc
46f0256
fd3ea1e
107068f
e1ed533
3290d30
af28404
f5bdc99
3757606
22e8ece
289a9da
2c0d324
0be2fd0
675c140
551af91
5d02237
1a24b08
4701171
4a7e562
d3bda54
d146280
ab6a9f6
97a7974
2b3a7a7
7c7a039
92880db
681be86
6ed6e1d
0ccc09a
355778d
558e3e0
9148dc0
1039d2e
2568790
88d22b7
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
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 was a very hard-to-find bug in the ordering of arguments in generate_jumps!. I fixed it here and in NSM.jl. |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,265 @@ | ||
# site chosen with DirectCR, rx or hop chosen with RSSA | ||
|
||
############################ DirectCRRSSA ################################### | ||
const MINJUMPRATE = 2.0^exponent(1e-12) | ||
|
||
#NOTE state vector u is a matrix. u[i,j] is species i, site j | ||
mutable struct DirectCRRSSAJumpAggregation{T, BD, M, RNG, J, RX, HOP, DEPGR, | ||
VJMAP, JVMAP, SS, U <: PriorityTable, S, F1, F2} <: | ||
AbstractSSAJumpAggregator{T, S, F1, F2, RNG} | ||
next_jump::SpatialJump{J} | ||
prev_jump::SpatialJump{J} | ||
next_jump_time::T | ||
end_time::T | ||
bracket_data::BD | ||
u_low_high::LowHigh{M} # species bracketing | ||
rx_rates::LowHigh{RX} | ||
hop_rates::LowHigh{HOP} | ||
site_rates_high::Vector{T} # we do not need site_rates_low | ||
save_positions::Tuple{Bool, Bool} | ||
rng::RNG | ||
dep_gr::DEPGR #dep graph is same for each site | ||
vartojumps_map::VJMAP #vartojumps_map is same for each site | ||
jumptovars_map::JVMAP #jumptovars_map is same for each site | ||
spatial_system::SS | ||
numspecies::Int #number of species | ||
rt::U | ||
rates::F1 # legacy, not used | ||
affects!::F2 # legacy, not used | ||
end | ||
|
||
function DirectCRRSSAJumpAggregation(nj::SpatialJump{J}, njt::T, et::T, bd::BD, | ||
u_low_high::LowHigh{M}, rx_rates::LowHigh{RX}, | ||
hop_rates::LowHigh{HOP}, site_rates_high::Vector{T}, | ||
sps::Tuple{Bool, Bool}, rng::RNG, spatial_system::SS; | ||
num_specs, minrate = convert(T, MINJUMPRATE), | ||
vartojumps_map = nothing, jumptovars_map = nothing, | ||
dep_graph = nothing, | ||
kwargs...) where {J, T, BD, RX, HOP, RNG, SS, M} | ||
|
||
# a dependency graph is needed | ||
if dep_graph === nothing | ||
dg = make_dependency_graph(num_specs, get_majumps(rx_rates)) | ||
else | ||
dg = dep_graph | ||
# make sure each jump depends on itself | ||
add_self_dependencies!(dg) | ||
end | ||
|
||
# a species-to-reactions graph is needed | ||
if vartojumps_map === nothing | ||
vtoj_map = var_to_jumps_map(num_specs, get_majumps(rx_rates)) | ||
else | ||
vtoj_map = vartojumps_map | ||
end | ||
|
||
if jumptovars_map === nothing | ||
jtov_map = jump_to_vars_map(get_majumps(rx_rates)) | ||
else | ||
jtov_map = jumptovars_map | ||
end | ||
|
||
# mapping from jump rate to group id | ||
minexponent = exponent(minrate) | ||
|
||
# use the largest power of two that is <= the passed in minrate | ||
minrate = 2.0^minexponent | ||
ratetogroup = rate -> priortogid(rate, minexponent) | ||
|
||
# construct an empty initial priority table -- we'll reset this in init | ||
rt = PriorityTable(ratetogroup, zeros(T, 1), minrate, 2 * minrate) | ||
|
||
DirectCRRSSAJumpAggregation{ | ||
T, | ||
BD, | ||
M, | ||
RNG, | ||
J, | ||
RX, | ||
HOP, | ||
typeof(dg), | ||
typeof(vtoj_map), | ||
typeof(jtov_map), | ||
SS, | ||
typeof(rt), | ||
Nothing, | ||
Nothing, | ||
Nothing, | ||
}(nj, nj, njt, et, bd, u_low_high, rx_rates, hop_rates, site_rates_high, sps, rng, dg, | ||
vtoj_map, jtov_map, spatial_system, num_specs, rt, nothing, nothing) | ||
end | ||
|
||
############################# Required Functions ############################## | ||
# creating the JumpAggregation structure (function wrapper-based constant jumps) | ||
function aggregate(aggregator::DirectCRRSSA, starting_state, p, t, end_time, | ||
constant_jumps, ma_jumps, save_positions, rng; hopping_constants, | ||
spatial_system, bracket_data = nothing, kwargs...) | ||
T = typeof(end_time) | ||
num_species = size(starting_state, 1) | ||
majumps = ma_jumps | ||
if majumps === nothing | ||
majumps = MassActionJump(Vector{T}(), | ||
Vector{Vector{Pair{Int, Int}}}(), | ||
Vector{Vector{Pair{Int, Int}}}()) | ||
end | ||
|
||
next_jump = SpatialJump{Int}(typemax(Int), typemax(Int), typemax(Int)) #a placeholder | ||
next_jump_time = typemax(T) | ||
rx_rates = LowHigh(RxRates(num_sites(spatial_system), majumps), | ||
RxRates(num_sites(spatial_system), majumps); | ||
do_copy = false) # do not copy ma_jumps | ||
hop_rates = LowHigh(HopRates(hopping_constants, spatial_system), | ||
HopRates(hopping_constants, spatial_system); | ||
do_copy = false) # do not copy hopping_constants | ||
site_rates_high = zeros(T, num_sites(spatial_system)) | ||
bd = (bracket_data === nothing) ? BracketData{T, eltype(starting_state)}() : | ||
bracket_data | ||
u_low_high = LowHigh(starting_state) | ||
|
||
DirectCRRSSAJumpAggregation(next_jump, next_jump_time, end_time, bd, u_low_high, | ||
rx_rates, hop_rates, | ||
site_rates_high, save_positions, rng, spatial_system; | ||
num_specs = num_species, kwargs...) | ||
end | ||
|
||
# set up a new simulation and calculate the first jump / jump time | ||
function initialize!(p::DirectCRRSSAJumpAggregation, integrator, u, params, t) | ||
p.end_time = integrator.sol.prob.tspan[2] | ||
fill_rates_and_get_times!(p, integrator, t) | ||
generate_jumps!(p, integrator, u, params, t) | ||
nothing | ||
end | ||
|
||
# calculate the next jump / jump time | ||
function generate_jumps!(p::DirectCRRSSAJumpAggregation, integrator, u, params, t) | ||
@unpack rng, rt, site_rates_high, rx_rates, hop_rates, spatial_system = p | ||
time_delta = zero(t) | ||
while true | ||
site = sample(rt, site_rates_high, rng) | ||
jump = sample_jump_direct(rx_rates.high, hop_rates.high, site, spatial_system, rng) | ||
time_delta += randexp(rng) | ||
if accept_jump(p, u, jump) | ||
p.next_jump_time = t + time_delta / groupsum(rt) | ||
p.next_jump = jump | ||
break | ||
end | ||
Comment on lines
+139
to
+145
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. Does this handle if there is no next jump? 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. To be honest, I do not rembmer any of the logic by now. Do we have a test for what you are asking? If not, we should, for all SSAs. |
||
end | ||
nothing | ||
end | ||
|
||
# execute one jump, changing the system state | ||
function execute_jumps!(p::DirectCRRSSAJumpAggregation, integrator, u, params, t, | ||
affects!) | ||
update_state!(p, integrator) | ||
update_dependent_rates!(p, integrator, t) | ||
nothing | ||
end | ||
|
||
######################## SSA specific helper routines ######################## | ||
# Return true if site is accepted. | ||
function accept_jump(p, u, jump) | ||
if is_hop(p, jump) | ||
return accept_hop(p, u, jump) | ||
else | ||
return accept_rx(p, u, jump) | ||
end | ||
end | ||
Vilin97 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
function accept_hop(p, u, jump) | ||
@unpack hop_rates, spatial_system, rng = p | ||
species, site = jump.jidx, jump.src | ||
acceptance_threshold = rand(rng) * hop_rate(hop_rates.high, species, site) | ||
if hop_rate(hop_rates.low, species, site) > acceptance_threshold | ||
return true | ||
else | ||
# compute the real rate. Could have used hop_rates.high as well. | ||
isaacsas marked this conversation as resolved.
Show resolved
Hide resolved
|
||
real_rate = evalhoprate(hop_rates.low, u, species, site, spatial_system) | ||
return real_rate > acceptance_threshold | ||
end | ||
end | ||
|
||
function accept_rx(p, u, jump) | ||
@unpack rx_rates, rng = p | ||
rx, site = reaction_id_from_jump(p, jump), jump.src | ||
acceptance_threshold = rand(rng) * rx_rate(rx_rates.high, rx, site) | ||
if rx_rate(rx_rates.low, rx, site) > acceptance_threshold | ||
return true | ||
else | ||
# compute the real rate. Could have used rx_rates.high as well. | ||
isaacsas marked this conversation as resolved.
Show resolved
Hide resolved
|
||
real_rate = evalrxrate(rx_rates.low, u, rx, site) | ||
return real_rate > acceptance_threshold | ||
end | ||
end | ||
|
||
""" | ||
fill_rates_and_get_times!(aggregation::DirectCRRSSAJumpAggregation, u, t) | ||
|
||
reset all stucts, reevaluate all rates, repopulate the priority table | ||
""" | ||
function fill_rates_and_get_times!(aggregation::DirectCRRSSAJumpAggregation, integrator, t) | ||
@unpack bracket_data, u_low_high, spatial_system, rx_rates, hop_rates, site_rates_high, rt = aggregation | ||
u = integrator.u | ||
update_u_brackets!(u_low_high::LowHigh, bracket_data, u::AbstractMatrix) | ||
|
||
reset!(rx_rates) | ||
reset!(hop_rates) | ||
fill!(site_rates_high, zero(eltype(site_rates_high))) | ||
|
||
rxs = 1:num_rxs(rx_rates.low) | ||
species = 1:(aggregation.numspecies) | ||
|
||
for site in 1:num_sites(spatial_system) | ||
update_rx_rates!(rx_rates, rxs, u_low_high, integrator, site) | ||
update_hop_rates!(hop_rates, species, u_low_high, site, spatial_system) | ||
site_rates_high[site] = total_site_rate(rx_rates.high, hop_rates.high, site) | ||
end | ||
|
||
# setup PriorityTable | ||
reset!(rt) | ||
for (pid, priority) in enumerate(site_rates_high) | ||
insert!(rt, pid, priority) | ||
end | ||
nothing | ||
end | ||
|
||
""" | ||
update_dependent_rates!(p, integrator, t) | ||
|
||
recalculate jump rates for jumps that depend on the just executed jump (p.prev_jump) | ||
""" | ||
function update_dependent_rates!(p::DirectCRRSSAJumpAggregation, integrator, t) | ||
jump = p.prev_jump | ||
if is_hop(p, jump) | ||
update_brackets!(p, integrator, jump.jidx, (jump.src, jump.dst)) | ||
else | ||
update_brackets!(p, integrator, p.jumptovars_map[reaction_id_from_jump(p, jump)], jump.src) | ||
end | ||
end | ||
|
||
function update_brackets!(p, integrator, species_to_update, sites_to_update) | ||
@unpack rx_rates, hop_rates, site_rates_high, u_low_high, bracket_data, vartojumps_map, spatial_system = p | ||
u = integrator.u | ||
for site in sites_to_update, species in species_to_update | ||
Vilin97 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
if !is_inside_brackets(u_low_high, u, species, site) | ||
update_u_brackets!(u_low_high, bracket_data, u, species, site) | ||
update_rx_rates!(rx_rates, | ||
vartojumps_map[species], | ||
u_low_high, | ||
integrator, | ||
site) | ||
update_hop_rates!(hop_rates, species, u_low_high, site, spatial_system) | ||
|
||
oldrate = site_rates_high[site] | ||
site_rates_high[site] = total_site_rate(rx_rates.high, hop_rates.high, site) | ||
update!(p.rt, site, oldrate, site_rates_high[site]) | ||
end | ||
end | ||
nothing | ||
end | ||
|
||
""" | ||
num_constant_rate_jumps(aggregator::DirectCRRSSAJumpAggregation) | ||
|
||
number of constant rate jumps | ||
""" | ||
num_constant_rate_jumps(aggregator::DirectCRRSSAJumpAggregation) = 0 |
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.
Does switching to using a branch have performance implications? We create these structures a lot right in the scalar case right? I'm not sure if Julia will remove it during compilation.
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.
Even if it's not optimized away, it shouldn't be expensive, right? I am not good at reading low level code but it seems that it does get optimized away in the end.
The output of the code introspection is as follows: