2022-09-25 15:54:46 +02:00
|
|
|
|
using Distributions: Normal
|
|
|
|
|
using ProgressMeter: ProgressMeter
|
|
|
|
|
using StaticArrays: SVector
|
|
|
|
|
using CellListMap: CellListMap
|
|
|
|
|
|
2022-01-18 02:17:52 +01:00
|
|
|
|
function rand_normal01()
|
|
|
|
|
return rand(Normal(0, 1))
|
|
|
|
|
end
|
2021-11-10 15:41:04 +01:00
|
|
|
|
|
2021-12-10 03:16:18 +01:00
|
|
|
|
function push_to_verlet_list!(verlet_lists, i, j)
|
2021-12-07 04:19:35 +01:00
|
|
|
|
if i < j
|
2021-12-10 03:16:18 +01:00
|
|
|
|
push!(verlet_lists[i], Int64(j))
|
2021-12-07 04:19:35 +01:00
|
|
|
|
else
|
2021-12-10 03:16:18 +01:00
|
|
|
|
push!(verlet_lists[j], Int64(i))
|
2021-12-07 04:19:35 +01:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
return nothing
|
|
|
|
|
end
|
|
|
|
|
|
2022-01-23 04:18:51 +01:00
|
|
|
|
function update_verlet_lists!(args, cl::CellListMap.CellList)
|
2021-12-10 03:16:18 +01:00
|
|
|
|
@simd for pre_vec in args.verlet_lists
|
|
|
|
|
reset!(pre_vec)
|
2021-11-10 15:41:04 +01:00
|
|
|
|
end
|
2021-11-16 22:26:08 +01:00
|
|
|
|
|
2021-12-10 03:16:18 +01:00
|
|
|
|
@simd for i in 1:(args.n_particles)
|
2021-12-07 05:52:10 +01:00
|
|
|
|
args.particles_c[i] = args.particles[i].c
|
2021-12-07 04:19:35 +01:00
|
|
|
|
end
|
2021-11-10 15:41:04 +01:00
|
|
|
|
|
2022-01-23 03:04:10 +01:00
|
|
|
|
cl = CellListMap.UpdateCellList!(args.particles_c, args.box, cl; parallel=false)
|
2021-11-10 15:41:04 +01:00
|
|
|
|
|
2022-01-23 03:04:10 +01:00
|
|
|
|
CellListMap.map_pairwise!(
|
2021-12-10 03:16:18 +01:00
|
|
|
|
(x, y, i, j, d2, output) -> push_to_verlet_list!(args.verlet_lists, i, j),
|
2021-12-07 04:19:35 +01:00
|
|
|
|
nothing,
|
|
|
|
|
args.box,
|
|
|
|
|
cl;
|
|
|
|
|
parallel=false,
|
|
|
|
|
)
|
2021-11-16 22:26:08 +01:00
|
|
|
|
|
2021-12-07 04:19:35 +01:00
|
|
|
|
return cl
|
2021-11-10 15:41:04 +01:00
|
|
|
|
end
|
|
|
|
|
|
2021-12-12 15:29:08 +01:00
|
|
|
|
function euler!(
|
2021-12-13 02:24:34 +01:00
|
|
|
|
args,
|
2021-12-21 00:31:44 +01:00
|
|
|
|
first_integration_step::Bool,
|
2022-01-11 01:31:30 +01:00
|
|
|
|
env_helper::Union{RL.EnvHelper,Nothing},
|
2022-01-18 02:17:52 +01:00
|
|
|
|
state_update_helper_hook!::Function,
|
|
|
|
|
state_update_hook!::Function,
|
|
|
|
|
update_table_and_actions_hook!::Function,
|
2021-12-13 02:24:34 +01:00
|
|
|
|
)
|
2021-12-10 03:16:18 +01:00
|
|
|
|
for id1 in 1:(args.n_particles - 1)
|
2021-12-08 21:53:15 +01:00
|
|
|
|
p1 = args.particles[id1]
|
|
|
|
|
p1_c = p1.c
|
2021-12-10 03:16:18 +01:00
|
|
|
|
verlet_list = args.verlet_lists[id1]
|
2021-11-10 15:41:04 +01:00
|
|
|
|
|
2021-12-08 21:53:15 +01:00
|
|
|
|
for id2 in view(verlet_list.v, 1:(verlet_list.last_ind))
|
|
|
|
|
p2 = args.particles[id2]
|
2021-11-10 15:41:04 +01:00
|
|
|
|
|
2021-11-16 22:26:08 +01:00
|
|
|
|
overlapping, r⃗₁₂, distance² = are_overlapping(
|
2022-01-29 14:32:04 +01:00
|
|
|
|
p1_c, p2.c, args.interaction_radius², args.half_box_len
|
2021-11-16 22:26:08 +01:00
|
|
|
|
)
|
2021-11-10 15:41:04 +01:00
|
|
|
|
|
2022-01-31 01:47:48 +01:00
|
|
|
|
state_update_helper_hook!(env_helper, id1, id2, r⃗₁₂, distance²)
|
2021-12-12 18:27:56 +01:00
|
|
|
|
|
2021-11-10 15:41:04 +01:00
|
|
|
|
if overlapping
|
2022-01-26 23:37:51 +01:00
|
|
|
|
factor = args.ϵσ⁶δtμₜ24 / (distance²^4) * (1.0 - args.σ⁶2 / (distance²^3))
|
|
|
|
|
μₜF⃗₁₂ = factor * r⃗₁₂ # Force acting on 1 from 2 multiplied with μₜ
|
2021-11-16 22:26:08 +01:00
|
|
|
|
|
2022-01-26 23:37:51 +01:00
|
|
|
|
p1.tmp_c += μₜF⃗₁₂
|
|
|
|
|
p2.tmp_c -= μₜF⃗₁₂
|
2021-11-10 15:41:04 +01:00
|
|
|
|
end
|
|
|
|
|
end
|
2021-11-11 01:25:06 +01:00
|
|
|
|
end
|
2021-11-10 15:41:04 +01:00
|
|
|
|
|
2022-01-29 16:45:36 +01:00
|
|
|
|
RL.copy_states_to_old_states_hook!(env_helper)
|
2022-01-18 02:17:52 +01:00
|
|
|
|
state_update_hook!(env_helper, args.particles)
|
2021-12-21 00:31:44 +01:00
|
|
|
|
|
2021-11-11 01:25:06 +01:00
|
|
|
|
@simd for p in args.particles
|
2021-12-08 21:53:15 +01:00
|
|
|
|
si, co = sincos(p.φ)
|
|
|
|
|
p.tmp_c += SVector(
|
2022-01-26 14:39:42 +01:00
|
|
|
|
args.v₀δt * co + args.sqrt_Dₜδt2 * rand_normal01(),
|
|
|
|
|
args.v₀δt * si + args.sqrt_Dₜδt2 * rand_normal01(),
|
2021-12-08 21:53:15 +01:00
|
|
|
|
)
|
2021-11-10 15:41:04 +01:00
|
|
|
|
|
2021-12-10 03:16:18 +01:00
|
|
|
|
restrict_coordinates!(p, args.half_box_len)
|
2021-11-10 15:41:04 +01:00
|
|
|
|
|
2022-01-18 02:17:52 +01:00
|
|
|
|
update_table_and_actions_hook!(env_helper, p, first_integration_step)
|
2021-12-21 00:31:44 +01:00
|
|
|
|
|
2022-01-18 02:17:52 +01:00
|
|
|
|
RL.act_hook!(p, env_helper, args.δt, si, co)
|
2021-12-13 02:24:34 +01:00
|
|
|
|
|
2022-01-26 14:39:42 +01:00
|
|
|
|
p.φ += args.sqrt_Dᵣδt2 * rand_normal01()
|
2021-12-12 15:29:08 +01:00
|
|
|
|
|
2021-12-07 05:52:10 +01:00
|
|
|
|
p.c = p.tmp_c
|
2021-11-10 15:41:04 +01:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
return nothing
|
|
|
|
|
end
|
|
|
|
|
|
2022-01-18 02:17:52 +01:00
|
|
|
|
function Base.wait(::Nothing)
|
|
|
|
|
return nothing
|
|
|
|
|
end
|
2021-11-26 14:50:47 +01:00
|
|
|
|
|
2022-01-18 02:17:52 +01:00
|
|
|
|
function gen_run_additional_hooks(::Nothing, args...)
|
|
|
|
|
return false
|
|
|
|
|
end
|
2021-12-12 18:27:56 +01:00
|
|
|
|
|
2022-01-11 01:31:30 +01:00
|
|
|
|
function gen_run_additional_hooks(env_helper::RL.EnvHelper, integration_step::Int64)
|
2022-01-11 18:39:38 +01:00
|
|
|
|
return (integration_step % env_helper.shared.n_steps_before_actions_update == 0) ||
|
2021-12-14 04:03:14 +01:00
|
|
|
|
(integration_step == 1)
|
2021-12-12 18:27:56 +01:00
|
|
|
|
end
|
|
|
|
|
|
2022-01-23 03:04:10 +01:00
|
|
|
|
function gen_cell_list(particles_c::Vector{SVector{2,Float64}}, box::CellListMap.Box)
|
|
|
|
|
return CellListMap.CellList(particles_c, box; parallel=false)
|
|
|
|
|
end
|
|
|
|
|
|
2022-01-18 02:17:52 +01:00
|
|
|
|
function simulate!(
|
2021-11-16 22:26:08 +01:00
|
|
|
|
args,
|
2021-11-26 06:55:27 +01:00
|
|
|
|
T0::Float64,
|
|
|
|
|
T::Float64,
|
2021-11-11 00:24:07 +01:00
|
|
|
|
n_steps_before_verlet_list_update::Int64,
|
2021-11-26 06:55:27 +01:00
|
|
|
|
n_steps_before_snapshot::Int64,
|
|
|
|
|
n_bundles::Int64,
|
2022-02-07 16:31:00 +01:00
|
|
|
|
sim_dir::String,
|
2021-11-26 06:55:27 +01:00
|
|
|
|
save_data::Bool,
|
2022-01-11 01:31:30 +01:00
|
|
|
|
env_helper::Union{RL.EnvHelper,Nothing},
|
2021-12-13 00:19:18 +01:00
|
|
|
|
)
|
2021-11-26 06:55:27 +01:00
|
|
|
|
bundle_snapshot_counter = 0
|
2021-11-16 22:26:08 +01:00
|
|
|
|
|
2021-12-07 04:19:35 +01:00
|
|
|
|
task::Union{Task,Nothing} = nothing
|
|
|
|
|
|
2022-01-23 03:04:10 +01:00
|
|
|
|
cl = gen_cell_list(args.particles_c, args.box)
|
2021-12-10 03:16:18 +01:00
|
|
|
|
cl = update_verlet_lists!(args, cl)
|
2021-12-07 04:19:35 +01:00
|
|
|
|
|
2021-12-21 00:31:44 +01:00
|
|
|
|
first_integration_step = true
|
|
|
|
|
|
2022-01-18 02:17:52 +01:00
|
|
|
|
state_update_helper_hook! =
|
|
|
|
|
state_update_hook! = update_table_and_actions_hook! = empty_hook
|
2021-12-12 15:29:08 +01:00
|
|
|
|
|
2022-01-23 04:21:06 +01:00
|
|
|
|
time_range = T0:(args.δt):T
|
|
|
|
|
|
|
|
|
|
progress = ProgressMeter.Progress(
|
2022-01-23 05:26:27 +01:00
|
|
|
|
length(time_range); dt=2, enabled=args.show_progress, desc="Simulation: "
|
2022-01-23 04:21:06 +01:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
for (integration_step, t) in enumerate(time_range)
|
2021-11-26 06:55:27 +01:00
|
|
|
|
if (integration_step % n_steps_before_snapshot == 0) && save_data
|
2021-11-26 14:50:47 +01:00
|
|
|
|
wait(task)
|
2021-12-07 07:08:22 +01:00
|
|
|
|
|
2021-11-26 06:55:27 +01:00
|
|
|
|
bundle_snapshot_counter += 1
|
|
|
|
|
save_snapshot!(args.bundle, bundle_snapshot_counter, t, args.particles)
|
|
|
|
|
|
|
|
|
|
if bundle_snapshot_counter == args.n_bundle_snapshots
|
2021-11-26 14:50:47 +01:00
|
|
|
|
task = @async begin
|
|
|
|
|
bundle_snapshot_counter = 0
|
|
|
|
|
n_bundles += 1
|
2021-11-26 06:55:27 +01:00
|
|
|
|
|
2022-02-07 16:31:00 +01:00
|
|
|
|
save_bundle(sim_dir, args.bundle, n_bundles, t)
|
2021-11-26 14:50:47 +01:00
|
|
|
|
end
|
2021-11-26 06:55:27 +01:00
|
|
|
|
end
|
2021-11-10 15:41:04 +01:00
|
|
|
|
end
|
|
|
|
|
|
2021-11-11 00:24:07 +01:00
|
|
|
|
if integration_step % n_steps_before_verlet_list_update == 0
|
2021-12-10 03:16:18 +01:00
|
|
|
|
cl = update_verlet_lists!(args, cl)
|
2021-11-10 15:41:04 +01:00
|
|
|
|
end
|
|
|
|
|
|
2022-01-11 01:31:30 +01:00
|
|
|
|
run_additional_hooks = gen_run_additional_hooks(env_helper, integration_step)
|
2021-12-21 00:31:44 +01:00
|
|
|
|
|
|
|
|
|
if run_additional_hooks
|
2022-01-18 02:17:52 +01:00
|
|
|
|
RL.pre_integration_hook!(env_helper)
|
2021-12-12 15:29:08 +01:00
|
|
|
|
|
2022-01-18 02:17:52 +01:00
|
|
|
|
state_update_helper_hook! = RL.state_update_helper_hook!
|
|
|
|
|
state_update_hook! = RL.state_update_hook!
|
|
|
|
|
update_table_and_actions_hook! = RL.update_table_and_actions_hook!
|
2021-12-12 15:29:08 +01:00
|
|
|
|
end
|
|
|
|
|
|
2021-12-21 00:31:44 +01:00
|
|
|
|
euler!(
|
|
|
|
|
args,
|
|
|
|
|
first_integration_step,
|
2022-01-11 01:31:30 +01:00
|
|
|
|
env_helper,
|
2022-01-18 02:17:52 +01:00
|
|
|
|
state_update_helper_hook!,
|
|
|
|
|
state_update_hook!,
|
|
|
|
|
update_table_and_actions_hook!,
|
2021-12-21 00:31:44 +01:00
|
|
|
|
)
|
2021-12-12 15:29:08 +01:00
|
|
|
|
|
2021-12-21 00:31:44 +01:00
|
|
|
|
if run_additional_hooks
|
2022-01-18 02:17:52 +01:00
|
|
|
|
state_update_helper_hook! =
|
|
|
|
|
state_update_hook! = update_table_and_actions_hook! = empty_hook
|
2021-12-12 15:29:08 +01:00
|
|
|
|
end
|
2021-12-21 00:31:44 +01:00
|
|
|
|
|
|
|
|
|
first_integration_step = false
|
2022-01-23 04:21:06 +01:00
|
|
|
|
|
|
|
|
|
ProgressMeter.next!(progress)
|
2021-11-10 15:41:04 +01:00
|
|
|
|
end
|
|
|
|
|
|
2021-12-07 07:08:22 +01:00
|
|
|
|
wait(task)
|
|
|
|
|
|
2021-11-26 06:55:27 +01:00
|
|
|
|
if bundle_snapshot_counter > 0
|
|
|
|
|
bundle = first_n_snapshots(args.bundle, bundle_snapshot_counter)
|
|
|
|
|
|
|
|
|
|
n_bundles += 1
|
|
|
|
|
|
2022-02-07 16:31:00 +01:00
|
|
|
|
save_bundle(sim_dir, bundle, n_bundles, T)
|
2021-11-26 06:55:27 +01:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
return nothing
|
2021-11-10 15:41:04 +01:00
|
|
|
|
end
|