107 auto constexpr parallel_execution_policy =
false;
113 std::unordered_map<int, int> virtual_site_id_for_mol_id;
116 std::unordered_map<int, ComInfo> m_com_by_mol_id;
121 virtual_site_id_for_mol_id[p.
vs_com().to_molecule_id] = p.
id();
123 if (not m_com_by_mol_id.contains(p.
mol_id())) {
126 auto const pos_unfolded =
128 m_com_by_mol_id[p.
mol_id()].total_mass += p.
mass();
129 m_com_by_mol_id[p.
mol_id()].weighted_position +=
130 p.
mass() * pos_unfolded;
133 parallel_execution_policy);
137 std::unordered_set<int> local_mol_ids;
138 for (
auto const &mol_id : m_com_by_mol_id | std::views::keys) {
139 local_mol_ids.insert(mol_id);
143 std::vector<std::unordered_set<int>> global_mol_ids{};
144 boost::mpi::all_gather(
comm_cart, local_mol_ids, global_mol_ids);
145 std::unordered_set<int> unique_mol_ids{};
146 for (
auto const &mol_id_set : global_mol_ids) {
147 for (
auto const &mol_id : mol_id_set) {
148 unique_mol_ids.insert(mol_id);
151 std::vector<int> flattened_mol_ids{unique_mol_ids.begin(),
152 unique_mol_ids.end()};
153 std::ranges::sort(flattened_mol_ids);
156 for (
auto const mol_id : flattened_mol_ids) {
157 double local_total_mass = 0.;
160 if (m_com_by_mol_id.contains(mol_id)) {
161 local_total_mass = m_com_by_mol_id[mol_id].total_mass;
162 local_weighted_position = m_com_by_mol_id[mol_id].weighted_position;
165 auto const total_mass =
166 boost::mpi::all_reduce(
comm_cart, local_total_mass, std::plus{});
167 auto const weighted_position =
168 boost::mpi::all_reduce(
comm_cart, local_weighted_position, std::plus{});
170 if (m_com_by_mol_id.contains(mol_id)) {
171 m_com_by_mol_id[mol_id].total_mass = total_mass;
172 m_com_by_mol_id[mol_id].weighted_position = weighted_position;
184 for (
auto const &[mol_id, com_info] : m_com_by_mol_id) {
185 if (not virtual_site_id_for_mol_id.contains(mol_id)) {
188 auto const vs_id = virtual_site_id_for_mol_id[mol_id];
190 auto folded_pos = com_info.weighted_position / com_info.total_mass;
195 vs_ptr->image_box() = image_box;
196 vs_ptr->mass() = com_info.total_mass;
197 vs_ptr->pos() = folded_pos;
206 auto constexpr parallel_execution_policy =
false;
212 std::unordered_map<int, Utils::Vector3d> force_for_vs_id;
214 std::unordered_map<int, double> mass_for_vs_id;
218 std::unordered_map<int, int> virtual_site_id_for_mol_id;
222 virtual_site_id_for_mol_id[p.
vs_com().to_molecule_id] = p.
id();
223 force_for_vs_id[p.
id()] = p.
force();
224 mass_for_vs_id[p.
id()] = p.
mass();
227 parallel_execution_policy);
242 not virtual_site_id_for_mol_id.contains(p.
mol_id())) {
245 auto const vs_id = virtual_site_id_for_mol_id.at(p.
mol_id());
247 (p.
mass() / mass_for_vs_id.at(vs_id)) * force_for_vs_id.at(vs_id);
249 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