134 auto constexpr parallel_execution_policy =
false;
140 std::unordered_map<ParticleId, MoleculeId> virtual_site_id_for_mol_id;
143 std::unordered_map<MoleculeId, ComInfo> m_com_by_mol_id;
148 assert(not virtual_site_id_for_mol_id.contains(p.
mol_id()));
149 virtual_site_id_for_mol_id[p.
mol_id()] = p.
id();
151 if (not m_com_by_mol_id.contains(p.
mol_id())) {
154 auto const pos_unfolded =
156 m_com_by_mol_id[p.
mol_id()].total_mass += p.
mass();
157 m_com_by_mol_id[p.
mol_id()].weighted_position +=
158 p.
mass() * pos_unfolded;
161 parallel_execution_policy);
165 std::unordered_set<MoleculeId> local_mol_ids;
166 for (
auto const &mol_id : m_com_by_mol_id | std::views::keys) {
167 local_mol_ids.insert(mol_id);
171 std::vector<std::unordered_set<MoleculeId>> global_mol_ids{};
172 boost::mpi::all_gather(
comm_cart, local_mol_ids, global_mol_ids);
173 std::unordered_set<MoleculeId> unique_mol_ids{};
174 for (
auto const &mol_id_set : global_mol_ids) {
175 for (
auto const &mol_id : mol_id_set) {
176 unique_mol_ids.insert(mol_id);
179 std::vector<MoleculeId> flattened_mol_ids{unique_mol_ids.begin(),
180 unique_mol_ids.end()};
181 std::ranges::sort(flattened_mol_ids);
184 for (
auto const mol_id : flattened_mol_ids) {
185 double local_total_mass = 0.;
188 if (m_com_by_mol_id.contains(mol_id)) {
189 local_total_mass = m_com_by_mol_id[mol_id].total_mass;
190 local_weighted_position = m_com_by_mol_id[mol_id].weighted_position;
193 auto const total_mass =
194 boost::mpi::all_reduce(
comm_cart, local_total_mass, std::plus{});
195 auto const weighted_position =
196 boost::mpi::all_reduce(
comm_cart, local_weighted_position, std::plus{});
198 if (m_com_by_mol_id.contains(mol_id)) {
199 m_com_by_mol_id[mol_id].total_mass = total_mass;
200 m_com_by_mol_id[mol_id].weighted_position = weighted_position;
212 for (
auto const &[mol_id, com_info] : m_com_by_mol_id) {
213 if (not virtual_site_id_for_mol_id.contains(mol_id)) {
216 auto const vs_id = virtual_site_id_for_mol_id[mol_id];
218 auto folded_pos = com_info.weighted_position / com_info.total_mass;
223 vs_ptr->image_box() = image_box;
224 vs_ptr->mass() = com_info.total_mass;
225 vs_ptr->pos() = folded_pos;
234 auto constexpr parallel_execution_policy =
false;
240 std::unordered_map<ParticleId, Utils::Vector3d> force_for_vs_id;
242 std::unordered_map<ParticleId, double> mass_for_vs_id;
246 std::unordered_map<MoleculeId, ParticleId> virtual_site_id_for_mol_id;
250 assert(not virtual_site_id_for_mol_id.contains(p.
mol_id()));
251 virtual_site_id_for_mol_id[p.
mol_id()] = p.
id();
252 force_for_vs_id[p.
id()] = p.
force();
253 mass_for_vs_id[p.
id()] = p.
mass();
256 parallel_execution_policy);
271 not virtual_site_id_for_mol_id.contains(p.
mol_id())) {
274 auto const vs_id = virtual_site_id_for_mol_id.at(p.
mol_id());
276 (p.
mass() / mass_for_vs_id.at(vs_id)) * force_for_vs_id.at(vs_id);
278 parallel_execution_policy);
void gather_buffer(std::vector< T, Allocator > &buffer, boost::mpi::communicator const &comm, int root=0)
Gather buffer with different size on each node.
int identity
unique identifier for the particle.
Struct holding all information for one particle.
auto const & propagation() const
auto const & mass() const
auto const & image_box() const
auto const & mol_id() const
auto const & force() const