133 auto constexpr parallel_execution_policy =
false;
139 std::unordered_map<ParticleId, MoleculeId> virtual_site_id_for_mol_id;
142 std::unordered_map<MoleculeId, ComInfo> m_com_by_mol_id;
147 assert(not virtual_site_id_for_mol_id.contains(p.
mol_id()));
148 virtual_site_id_for_mol_id[p.
mol_id()] = p.
id();
150 if (not m_com_by_mol_id.contains(p.
mol_id())) {
153 auto const pos_unfolded =
155 m_com_by_mol_id[p.
mol_id()].total_mass += p.
mass();
156 m_com_by_mol_id[p.
mol_id()].weighted_position +=
157 p.
mass() * pos_unfolded;
160 parallel_execution_policy);
164 std::unordered_set<MoleculeId> local_mol_ids;
165 for (
auto const &mol_id : m_com_by_mol_id | std::views::keys) {
166 local_mol_ids.insert(mol_id);
170 std::vector<std::unordered_set<MoleculeId>> global_mol_ids{};
171 boost::mpi::all_gather(
comm_cart, local_mol_ids, global_mol_ids);
172 std::unordered_set<MoleculeId> unique_mol_ids{};
173 for (
auto const &mol_id_set : global_mol_ids) {
174 for (
auto const &mol_id : mol_id_set) {
175 unique_mol_ids.insert(mol_id);
178 std::vector<MoleculeId> flattened_mol_ids{unique_mol_ids.begin(),
179 unique_mol_ids.end()};
180 std::ranges::sort(flattened_mol_ids);
183 for (
auto const mol_id : flattened_mol_ids) {
184 double local_total_mass = 0.;
187 if (m_com_by_mol_id.contains(mol_id)) {
188 local_total_mass = m_com_by_mol_id[mol_id].total_mass;
189 local_weighted_position = m_com_by_mol_id[mol_id].weighted_position;
192 auto const total_mass =
193 boost::mpi::all_reduce(
comm_cart, local_total_mass, std::plus{});
194 auto const weighted_position =
195 boost::mpi::all_reduce(
comm_cart, local_weighted_position, std::plus{});
197 if (m_com_by_mol_id.contains(mol_id)) {
198 m_com_by_mol_id[mol_id].total_mass = total_mass;
199 m_com_by_mol_id[mol_id].weighted_position = weighted_position;
211 for (
auto const &[mol_id, com_info] : m_com_by_mol_id) {
212 if (not virtual_site_id_for_mol_id.contains(mol_id)) {
215 auto const vs_id = virtual_site_id_for_mol_id[mol_id];
217 auto folded_pos = com_info.weighted_position / com_info.total_mass;
222 vs_ptr->image_box() = image_box;
223 vs_ptr->mass() = com_info.total_mass;
224 vs_ptr->pos() = folded_pos;
233 auto constexpr parallel_execution_policy =
false;
239 std::unordered_map<ParticleId, Utils::Vector3d> force_for_vs_id;
241 std::unordered_map<ParticleId, double> mass_for_vs_id;
245 std::unordered_map<MoleculeId, ParticleId> virtual_site_id_for_mol_id;
249 assert(not virtual_site_id_for_mol_id.contains(p.
mol_id()));
250 virtual_site_id_for_mol_id[p.
mol_id()] = p.
id();
251 force_for_vs_id[p.
id()] = p.
force();
252 mass_for_vs_id[p.
id()] = p.
mass();
255 parallel_execution_policy);
270 not virtual_site_id_for_mol_id.contains(p.
mol_id())) {
273 auto const vs_id = virtual_site_id_for_mol_id.at(p.
mol_id());
275 (p.
mass() / mass_for_vs_id.at(vs_id)) * force_for_vs_id.at(vs_id);
277 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