108 auto constexpr parallel_execution_policy =
false;
114 std::unordered_map<int, int> virtual_site_id_for_mol_id;
117 std::unordered_map<int, ComInfo> m_com_by_mol_id;
122 virtual_site_id_for_mol_id[p.
vs_com().to_molecule_id] = p.
id();
124 if (not m_com_by_mol_id.contains(p.
mol_id())) {
127 auto const pos_unfolded =
129 m_com_by_mol_id[p.
mol_id()].total_mass += p.
mass();
130 m_com_by_mol_id[p.
mol_id()].weighted_position +=
131 p.
mass() * pos_unfolded;
134 parallel_execution_policy);
138 std::unordered_set<int> local_mol_ids;
139 for (
auto const &mol_id : m_com_by_mol_id | std::views::keys) {
140 local_mol_ids.insert(mol_id);
144 std::vector<std::unordered_set<int>> global_mol_ids{};
145 boost::mpi::all_gather(
comm_cart, local_mol_ids, global_mol_ids);
146 std::unordered_set<int> unique_mol_ids{};
147 for (
auto const &mol_id_set : global_mol_ids) {
148 for (
auto const &mol_id : mol_id_set) {
149 unique_mol_ids.insert(mol_id);
152 std::vector<int> flattened_mol_ids{unique_mol_ids.begin(),
153 unique_mol_ids.end()};
154 std::ranges::sort(flattened_mol_ids);
157 for (
auto const mol_id : flattened_mol_ids) {
158 double local_total_mass = 0.;
161 if (m_com_by_mol_id.contains(mol_id)) {
162 local_total_mass = m_com_by_mol_id[mol_id].total_mass;
163 local_weighted_position = m_com_by_mol_id[mol_id].weighted_position;
166 auto const total_mass =
167 boost::mpi::all_reduce(
comm_cart, local_total_mass, std::plus{});
168 auto const weighted_position =
169 boost::mpi::all_reduce(
comm_cart, local_weighted_position, std::plus{});
171 if (m_com_by_mol_id.contains(mol_id)) {
172 m_com_by_mol_id[mol_id].total_mass = total_mass;
173 m_com_by_mol_id[mol_id].weighted_position = weighted_position;
185 for (
auto const &[mol_id, com_info] : m_com_by_mol_id) {
186 if (not virtual_site_id_for_mol_id.contains(mol_id)) {
189 auto com = com_info.weighted_position / com_info.total_mass;
190 auto const vs_id = virtual_site_id_for_mol_id[mol_id];
192 if (vs_ptr ==
nullptr) {
196 auto folded_pos = com;
201 vs_ptr->image_box() = image_box;
202 vs_ptr->mass() = com_info.total_mass;
203 vs_ptr->pos() = folded_pos;
211 auto constexpr parallel_execution_policy =
false;
217 std::unordered_map<int, Utils::Vector3d> force_for_vs_id;
219 std::unordered_map<int, double> mass_for_vs_id;
223 std::unordered_map<int, int> virtual_site_id_for_mol_id;
227 virtual_site_id_for_mol_id[p.
vs_com().to_molecule_id] = p.
id();
228 force_for_vs_id[p.
id()] = p.
force();
229 mass_for_vs_id[p.
id()] = p.
mass();
232 parallel_execution_policy);
247 not virtual_site_id_for_mol_id.contains(p.
mol_id())) {
250 auto const vs_id = virtual_site_id_for_mol_id.at(p.
mol_id());
252 (p.
mass() / mass_for_vs_id.at(vs_id)) * force_for_vs_id.at(vs_id);
254 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.
Struct holding all information for one particle.
auto const & propagation() const
auto const & mass() const
auto const & vs_com() const
auto const & image_box() const
auto const & mol_id() const
auto const & force() const