// -*- c++ -*- // This file is part of the Collective Variables module (Colvars). // The original version of Colvars and its updates are located at: // https://github.com/Colvars/colvars // Please update all Colvars source files before making any changes. // If you wish to distribute your changes, please submit them to the // Colvars repository at GitHub. #include #include "colvarmodule.h" #include "colvarvalue.h" #include "colvar.h" #include "colvarcomp.h" #include "colvar_rotation_derivative.h" colvar::distance::distance() { set_function_type("distance"); init_as_distance(); provide(f_cvc_inv_gradient); provide(f_cvc_Jacobian); enable(f_cvc_com_based); } int colvar::distance::init(std::string const &conf) { int error_code = cvc::init(conf); group1 = parse_group(conf, "group1"); group2 = parse_group(conf, "group2"); if (!group1 || !group2) { return error_code | COLVARS_INPUT_ERROR; } error_code |= init_total_force_params(conf); return error_code; } void colvar::distance::calc_value() { if (!is_enabled(f_cvc_pbc_minimum_image)) { dist_v = group2->center_of_mass() - group1->center_of_mass(); } else { dist_v = cvm::position_distance(group1->center_of_mass(), group2->center_of_mass()); } x.real_value = dist_v.norm(); } void colvar::distance::calc_gradients() { cvm::rvector const u = dist_v.unit(); group1->set_weighted_gradient(-1.0 * u); group2->set_weighted_gradient( u); } void colvar::distance::calc_force_invgrads() { group1->read_total_forces(); if (is_enabled(f_cvc_one_site_total_force)) { ft.real_value = -1.0 * (group1->total_force() * dist_v.unit()); } else { group2->read_total_forces(); ft.real_value = 0.5 * ((group2->total_force() - group1->total_force()) * dist_v.unit()); } } void colvar::distance::calc_Jacobian_derivative() { jd.real_value = x.real_value ? (2.0 / x.real_value) : 0.0; } colvar::distance_vec::distance_vec() { set_function_type("distanceVec"); disable(f_cvc_explicit_gradient); x.type(colvarvalue::type_3vector); } void colvar::distance_vec::calc_value() { if (!is_enabled(f_cvc_pbc_minimum_image)) { x.rvector_value = group2->center_of_mass() - group1->center_of_mass(); } else { x.rvector_value = cvm::position_distance(group1->center_of_mass(), group2->center_of_mass()); } } void colvar::distance_vec::calc_gradients() { // gradients are not stored: a 3x3 matrix for each atom would be // needed to store just the identity matrix } void colvar::distance_vec::apply_force(colvarvalue const &force) { if (!group1->noforce) group1->apply_force(-1.0 * force.rvector_value); if (!group2->noforce) group2->apply_force( force.rvector_value); } cvm::real colvar::distance_vec::dist2(colvarvalue const &x1, colvarvalue const &x2) const { if (is_enabled(f_cvc_pbc_minimum_image)) { return (cvm::position_distance(x1.rvector_value, x2.rvector_value)).norm2(); } return (x2.rvector_value - x1.rvector_value).norm2(); } colvarvalue colvar::distance_vec::dist2_lgrad(colvarvalue const &x1, colvarvalue const &x2) const { if (is_enabled(f_cvc_pbc_minimum_image)) { return 2.0 * cvm::position_distance(x2.rvector_value, x1.rvector_value); } return 2.0 * (x2.rvector_value - x1.rvector_value); } colvarvalue colvar::distance_vec::dist2_rgrad(colvarvalue const &x1, colvarvalue const &x2) const { return distance_vec::dist2_lgrad(x2, x1); } void colvar::distance_vec::wrap(colvarvalue & /* x_unwrapped */) const {} colvar::distance_z::distance_z() { set_function_type("distanceZ"); provide(f_cvc_inv_gradient); provide(f_cvc_Jacobian); enable(f_cvc_com_based); provide(f_cvc_periodic); x.type(colvarvalue::type_scalar); } int colvar::distance_z::init(std::string const &conf) { int error_code = cvc::init(conf); main = parse_group(conf, "main"); ref1 = parse_group(conf, "ref"); // this group is optional ref2 = parse_group(conf, "ref2", true); if ( ref2 ) { cvm::log("Using axis joining the centers of mass of groups \"ref\" and \"ref2\"\n"); fixed_axis = false; if (key_lookup(conf, "axis")) cvm::log("Warning: explicit axis definition will be ignored!\n"); } else { if (get_keyval(conf, "axis", axis, cvm::rvector(0.0, 0.0, 1.0))) { if (axis.norm2() == 0.0) { error_code |= cvm::error("Axis vector is zero!", COLVARS_INPUT_ERROR); } if (axis.norm2() != 1.0) { axis = axis.unit(); cvm::log("The normalized axis is: "+cvm::to_str(axis)+".\n"); } } fixed_axis = true; } error_code |= init_total_force_params(conf); return error_code; } void colvar::distance_z::calc_value() { if (fixed_axis) { if (!is_enabled(f_cvc_pbc_minimum_image)) { dist_v = main->center_of_mass() - ref1->center_of_mass(); } else { dist_v = cvm::position_distance(ref1->center_of_mass(), main->center_of_mass()); } } else { if (!is_enabled(f_cvc_pbc_minimum_image)) { dist_v = main->center_of_mass() - (0.5 * (ref1->center_of_mass() + ref2->center_of_mass())); axis = ref2->center_of_mass() - ref1->center_of_mass(); } else { dist_v = cvm::position_distance(0.5 * (ref1->center_of_mass() + ref2->center_of_mass()), main->center_of_mass()); axis = cvm::position_distance(ref1->center_of_mass(), ref2->center_of_mass()); } axis_norm = axis.norm(); axis = axis.unit(); } x.real_value = axis * dist_v; wrap(x); } void colvar::distance_z::calc_gradients() { main->set_weighted_gradient( axis ); if (fixed_axis) { ref1->set_weighted_gradient(-1.0 * axis); } else { if (!is_enabled(f_cvc_pbc_minimum_image)) { ref1->set_weighted_gradient( 1.0 / axis_norm * (main->center_of_mass() - ref2->center_of_mass() - x.real_value * axis )); ref2->set_weighted_gradient( 1.0 / axis_norm * (ref1->center_of_mass() - main->center_of_mass() + x.real_value * axis )); } else { ref1->set_weighted_gradient( 1.0 / axis_norm * ( cvm::position_distance(ref2->center_of_mass(), main->center_of_mass()) - x.real_value * axis )); ref2->set_weighted_gradient( 1.0 / axis_norm * ( cvm::position_distance(main->center_of_mass(), ref1->center_of_mass()) + x.real_value * axis )); } } } void colvar::distance_z::calc_force_invgrads() { main->read_total_forces(); if (fixed_axis && !is_enabled(f_cvc_one_site_total_force)) { ref1->read_total_forces(); ft.real_value = 0.5 * ((main->total_force() - ref1->total_force()) * axis); } else { ft.real_value = main->total_force() * axis; } } void colvar::distance_z::calc_Jacobian_derivative() { jd.real_value = 0.0; } colvar::distance_xy::distance_xy() { set_function_type("distanceXY"); provide(f_cvc_periodic, false); // Disable inherited distance_z flag init_as_distance(); } void colvar::distance_xy::calc_value() { if (!is_enabled(f_cvc_pbc_minimum_image)) { dist_v = main->center_of_mass() - ref1->center_of_mass(); } else { dist_v = cvm::position_distance(ref1->center_of_mass(), main->center_of_mass()); } if (!fixed_axis) { if (!is_enabled(f_cvc_pbc_minimum_image)) { v12 = ref2->center_of_mass() - ref1->center_of_mass(); } else { v12 = cvm::position_distance(ref1->center_of_mass(), ref2->center_of_mass()); } axis_norm = v12.norm(); axis = v12.unit(); } dist_v_ortho = dist_v - (dist_v * axis) * axis; x.real_value = dist_v_ortho.norm(); } void colvar::distance_xy::calc_gradients() { // Intermediate quantity (r_P3 / r_12 where P is the projection // of 3(main) on the plane orthogonal to 12, containing 1 (ref1)) cvm::real A; cvm::real x_inv; if (x.real_value == 0.0) return; x_inv = 1.0 / x.real_value; if (fixed_axis) { ref1->set_weighted_gradient(-1.0 * x_inv * dist_v_ortho); main->set_weighted_gradient( x_inv * dist_v_ortho); } else { if (!is_enabled(f_cvc_pbc_minimum_image)) { v13 = main->center_of_mass() - ref1->center_of_mass(); } else { v13 = cvm::position_distance(ref1->center_of_mass(), main->center_of_mass()); } A = (dist_v * axis) / axis_norm; ref1->set_weighted_gradient( (A - 1.0) * x_inv * dist_v_ortho); ref2->set_weighted_gradient( -A * x_inv * dist_v_ortho); main->set_weighted_gradient( 1.0 * x_inv * dist_v_ortho); } } void colvar::distance_xy::calc_force_invgrads() { main->read_total_forces(); if (fixed_axis && !is_enabled(f_cvc_one_site_total_force)) { ref1->read_total_forces(); ft.real_value = 0.5 / x.real_value * ((main->total_force() - ref1->total_force()) * dist_v_ortho); } else { ft.real_value = 1.0 / x.real_value * main->total_force() * dist_v_ortho; } } void colvar::distance_xy::calc_Jacobian_derivative() { jd.real_value = x.real_value ? (1.0 / x.real_value) : 0.0; } colvar::distance_dir::distance_dir() { set_function_type("distanceDir"); enable(f_cvc_com_based); disable(f_cvc_explicit_gradient); x.type(colvarvalue::type_unit3vector); } void colvar::distance_dir::calc_value() { if (!is_enabled(f_cvc_pbc_minimum_image)) { dist_v = group2->center_of_mass() - group1->center_of_mass(); } else { dist_v = cvm::position_distance(group1->center_of_mass(), group2->center_of_mass()); } x.rvector_value = dist_v.unit(); } void colvar::distance_dir::calc_gradients() { // gradients are computed on the fly within apply_force() // Note: could be a problem if a future bias relies on gradient // calculations... // TODO in new deps system: remove dependency of biasing force to gradient? // That way we could tell apart an explicit gradient dependency } void colvar::distance_dir::apply_force(colvarvalue const &force) { // remove the radial force component cvm::real const iprod = force.rvector_value * x.rvector_value; cvm::rvector const force_tang = force.rvector_value - iprod * x.rvector_value; if (!group1->noforce) group1->apply_force(-1.0 * force_tang); if (!group2->noforce) group2->apply_force( force_tang); } cvm::real colvar::distance_dir::dist2(colvarvalue const &x1, colvarvalue const &x2) const { return (x1.rvector_value - x2.rvector_value).norm2(); } colvarvalue colvar::distance_dir::dist2_lgrad(colvarvalue const &x1, colvarvalue const &x2) const { return colvarvalue((x1.rvector_value - x2.rvector_value), colvarvalue::type_unit3vectorderiv); } colvarvalue colvar::distance_dir::dist2_rgrad(colvarvalue const &x1, colvarvalue const &x2) const { return colvarvalue((x2.rvector_value - x1.rvector_value), colvarvalue::type_unit3vectorderiv); } void colvar::distance_dir::wrap(colvarvalue & /* x_unwrapped */) const {} colvar::distance_inv::distance_inv() { set_function_type("distanceInv"); init_as_distance(); } int colvar::distance_inv::init(std::string const &conf) { int error_code = cvc::init(conf); group1 = parse_group(conf, "group1"); group2 = parse_group(conf, "group2"); get_keyval(conf, "exponent", exponent, exponent); if (exponent % 2) { error_code |= cvm::error("Error: odd exponent provided, can only use even ones.\n", COLVARS_INPUT_ERROR); } if (exponent <= 0) { error_code |= cvm::error("Error: negative or zero exponent provided.\n", COLVARS_INPUT_ERROR); } for (cvm::atom_iter ai1 = group1->begin(); ai1 != group1->end(); ai1++) { for (cvm::atom_iter ai2 = group2->begin(); ai2 != group2->end(); ai2++) { if (ai1->id == ai2->id) { error_code |= cvm::error("Error: group1 and group2 have some atoms in common: this is not " "allowed for distanceInv.\n", COLVARS_INPUT_ERROR); } } } if (is_enabled(f_cvc_debug_gradient)) { cvm::log("Warning: debugGradients will not give correct results " "for distanceInv, because its value and gradients are computed " "simultaneously.\n"); } return error_code; } void colvar::distance_inv::calc_value() { x.real_value = 0.0; if (!is_enabled(f_cvc_pbc_minimum_image)) { for (cvm::atom_iter ai1 = group1->begin(); ai1 != group1->end(); ai1++) { for (cvm::atom_iter ai2 = group2->begin(); ai2 != group2->end(); ai2++) { cvm::rvector const dv = ai2->pos - ai1->pos; cvm::real const d2 = dv.norm2(); cvm::real const dinv = cvm::integer_power(d2, -1*(exponent/2)); x.real_value += dinv; cvm::rvector const dsumddv = -1.0*(exponent/2) * dinv/d2 * 2.0 * dv; ai1->grad += -1.0 * dsumddv; ai2->grad += dsumddv; } } } else { for (cvm::atom_iter ai1 = group1->begin(); ai1 != group1->end(); ai1++) { for (cvm::atom_iter ai2 = group2->begin(); ai2 != group2->end(); ai2++) { cvm::rvector const dv = cvm::position_distance(ai1->pos, ai2->pos); cvm::real const d2 = dv.norm2(); cvm::real const dinv = cvm::integer_power(d2, -1*(exponent/2)); x.real_value += dinv; cvm::rvector const dsumddv = -1.0*(exponent/2) * dinv/d2 * 2.0 * dv; ai1->grad += -1.0 * dsumddv; ai2->grad += dsumddv; } } } x.real_value *= 1.0 / cvm::real(group1->size() * group2->size()); x.real_value = cvm::pow(x.real_value, -1.0/cvm::real(exponent)); cvm::real const dxdsum = (-1.0/(cvm::real(exponent))) * cvm::integer_power(x.real_value, exponent+1) / cvm::real(group1->size() * group2->size()); for (cvm::atom_iter ai1 = group1->begin(); ai1 != group1->end(); ai1++) { ai1->grad *= dxdsum; } for (cvm::atom_iter ai2 = group2->begin(); ai2 != group2->end(); ai2++) { ai2->grad *= dxdsum; } } void colvar::distance_inv::calc_gradients() { } colvar::distance_pairs::distance_pairs() { set_function_type("distancePairs"); disable(f_cvc_explicit_gradient); x.type(colvarvalue::type_vector); } int colvar::distance_pairs::init(std::string const &conf) { int error_code = cvc::init(conf); group1 = parse_group(conf, "group1"); group2 = parse_group(conf, "group2"); x.vector1d_value.resize(group1->size() * group2->size()); return error_code; } void colvar::distance_pairs::calc_value() { x.vector1d_value.resize(group1->size() * group2->size()); if (!is_enabled(f_cvc_pbc_minimum_image)) { size_t i1, i2; for (i1 = 0; i1 < group1->size(); i1++) { for (i2 = 0; i2 < group2->size(); i2++) { cvm::rvector const dv = (*group2)[i2].pos - (*group1)[i1].pos; cvm::real const d = dv.norm(); x.vector1d_value[i1*group2->size() + i2] = d; (*group1)[i1].grad = -1.0 * dv.unit(); (*group2)[i2].grad = dv.unit(); } } } else { size_t i1, i2; for (i1 = 0; i1 < group1->size(); i1++) { for (i2 = 0; i2 < group2->size(); i2++) { cvm::rvector const dv = cvm::position_distance((*group1)[i1].pos, (*group2)[i2].pos); cvm::real const d = dv.norm(); x.vector1d_value[i1*group2->size() + i2] = d; (*group1)[i1].grad = -1.0 * dv.unit(); (*group2)[i2].grad = dv.unit(); } } } } void colvar::distance_pairs::calc_gradients() { // will be calculated on the fly in apply_force() } void colvar::distance_pairs::apply_force(colvarvalue const &force) { if (!is_enabled(f_cvc_pbc_minimum_image)) { size_t i1, i2; for (i1 = 0; i1 < group1->size(); i1++) { for (i2 = 0; i2 < group2->size(); i2++) { cvm::rvector const dv = (*group2)[i2].pos - (*group1)[i1].pos; (*group1)[i1].apply_force(force[i1*group2->size() + i2] * (-1.0) * dv.unit()); (*group2)[i2].apply_force(force[i1*group2->size() + i2] * dv.unit()); } } } else { size_t i1, i2; for (i1 = 0; i1 < group1->size(); i1++) { for (i2 = 0; i2 < group2->size(); i2++) { cvm::rvector const dv = cvm::position_distance((*group1)[i1].pos, (*group2)[i2].pos); (*group1)[i1].apply_force(force[i1*group2->size() + i2] * (-1.0) * dv.unit()); (*group2)[i2].apply_force(force[i1*group2->size() + i2] * dv.unit()); } } } } cvm::real colvar::distance_pairs::dist2(colvarvalue const &x1, colvarvalue const &x2) const { return (x1.vector1d_value - x2.vector1d_value).norm2(); } colvarvalue colvar::distance_pairs::dist2_lgrad(colvarvalue const &x1, colvarvalue const &x2) const { return 2.0 * (x1.vector1d_value - x2.vector1d_value); } colvarvalue colvar::distance_pairs::dist2_rgrad(colvarvalue const &x1, colvarvalue const &x2) const { return distance_pairs::dist2_lgrad(x1, x2); } void colvar::distance_pairs::wrap(colvarvalue & /* x_unwrapped */) const {} colvar::dipole_magnitude::dipole_magnitude() { set_function_type("dipoleMagnitude"); x.type(colvarvalue::type_scalar); } int colvar::dipole_magnitude::init(std::string const &conf) { int error_code = cvc::init(conf); atoms = parse_group(conf, "atoms"); if (!atoms) error_code |= COLVARS_INPUT_ERROR; return error_code; } void colvar::dipole_magnitude::calc_value() { cvm::atom_pos const atomsCom = atoms->center_of_mass(); atoms->calc_dipole(atomsCom); dipoleV = atoms->dipole(); x.real_value = dipoleV.norm(); } void colvar::dipole_magnitude::calc_gradients() { cvm::real const aux1 = atoms->total_charge/atoms->total_mass; cvm::atom_pos const dipVunit = dipoleV.unit(); for (cvm::atom_iter ai = atoms->begin(); ai != atoms->end(); ai++) { ai->grad = (ai->charge - aux1*ai->mass) * dipVunit; } } colvar::gyration::gyration() { set_function_type("gyration"); provide(f_cvc_inv_gradient); provide(f_cvc_Jacobian); init_as_distance(); } int colvar::gyration::init(std::string const &conf) { int error_code = cvc::init(conf); atoms = parse_group(conf, "atoms"); if (atoms->b_user_defined_fit) { cvm::log("WARNING: explicit fitting parameters were provided for atom group \"atoms\".\n"); } else { atoms->enable(f_ag_center); atoms->ref_pos.assign(1, cvm::atom_pos(0.0, 0.0, 0.0)); atoms->fit_gradients.assign(atoms->size(), cvm::rvector(0.0, 0.0, 0.0)); } return error_code; } void colvar::gyration::calc_value() { x.real_value = 0.0; for (cvm::atom_iter ai = atoms->begin(); ai != atoms->end(); ai++) { x.real_value += (ai->pos).norm2(); } x.real_value = cvm::sqrt(x.real_value / cvm::real(atoms->size())); } void colvar::gyration::calc_gradients() { cvm::real const drdx = 1.0/(cvm::real(atoms->size()) * x.real_value); for (cvm::atom_iter ai = atoms->begin(); ai != atoms->end(); ai++) { ai->grad = drdx * ai->pos; } } void colvar::gyration::calc_force_invgrads() { atoms->read_total_forces(); cvm::real const dxdr = 1.0/x.real_value; ft.real_value = 0.0; for (cvm::atom_iter ai = atoms->begin(); ai != atoms->end(); ai++) { ft.real_value += dxdr * ai->pos * ai->total_force; } } void colvar::gyration::calc_Jacobian_derivative() { jd = x.real_value ? (3.0 * cvm::real(atoms->size()) - 4.0) / x.real_value : 0.0; } colvar::inertia::inertia() { set_function_type("inertia"); } void colvar::inertia::calc_value() { x.real_value = 0.0; for (cvm::atom_iter ai = atoms->begin(); ai != atoms->end(); ai++) { x.real_value += (ai->pos).norm2(); } } void colvar::inertia::calc_gradients() { for (cvm::atom_iter ai = atoms->begin(); ai != atoms->end(); ai++) { ai->grad = 2.0 * ai->pos; } } colvar::inertia_z::inertia_z() { set_function_type("inertiaZ"); } int colvar::inertia_z::init(std::string const &conf) { int error_code = inertia::init(conf); if (get_keyval(conf, "axis", axis, cvm::rvector(0.0, 0.0, 1.0))) { if (axis.norm2() == 0.0) { error_code |= cvm::error("Axis vector is zero!", COLVARS_INPUT_ERROR); } if (axis.norm2() != 1.0) { axis = axis.unit(); cvm::log("The normalized axis is: "+cvm::to_str(axis)+".\n"); } } return error_code; } void colvar::inertia_z::calc_value() { x.real_value = 0.0; for (cvm::atom_iter ai = atoms->begin(); ai != atoms->end(); ai++) { cvm::real const iprod = ai->pos * axis; x.real_value += iprod * iprod; } } void colvar::inertia_z::calc_gradients() { for (cvm::atom_iter ai = atoms->begin(); ai != atoms->end(); ai++) { ai->grad = 2.0 * (ai->pos * axis) * axis; } } colvar::rmsd::rmsd() { set_function_type("rmsd"); init_as_distance(); provide(f_cvc_inv_gradient); } int colvar::rmsd::init(std::string const &conf) { int error_code = cvc::init(conf); atoms = parse_group(conf, "atoms"); if (!atoms || atoms->size() == 0) { return error_code | COLVARS_INPUT_ERROR; } bool b_Jacobian_derivative = true; if (atoms->fitting_group != NULL && b_Jacobian_derivative) { cvm::log("The option \"fittingGroup\" (alternative group for fitting) was enabled: " "Jacobian derivatives of the RMSD will not be calculated.\n"); b_Jacobian_derivative = false; } if (b_Jacobian_derivative) provide(f_cvc_Jacobian); // the following is a simplified version of the corresponding atom group options; // we need this because the reference coordinates defined inside the atom group // may be used only for fitting, and even more so if fitting_group is used if (get_keyval(conf, "refPositions", ref_pos, ref_pos)) { cvm::log("Using reference positions from configuration file to calculate the variable.\n"); if (ref_pos.size() != atoms->size()) { error_code |= cvm::error("Error: the number of reference positions provided (" + cvm::to_str(ref_pos.size()) + ") does not match the number of atoms of group \"atoms\" (" + cvm::to_str(atoms->size()) + ").\n", COLVARS_INPUT_ERROR); } } else { // Only look for ref pos file if ref positions not already provided std::string ref_pos_file; if (get_keyval(conf, "refPositionsFile", ref_pos_file, std::string(""))) { if (ref_pos.size()) { error_code |= cvm::error("Error: cannot specify \"refPositionsFile\" and " "\"refPositions\" at the same time.\n", COLVARS_INPUT_ERROR); } std::string ref_pos_col; double ref_pos_col_value=0.0; if (get_keyval(conf, "refPositionsCol", ref_pos_col, std::string(""))) { // if provided, use PDB column to select coordinates bool found = get_keyval(conf, "refPositionsColValue", ref_pos_col_value, 0.0); if (found && ref_pos_col_value==0.0) { error_code |= cvm::error("Error: refPositionsColValue, " "if provided, must be non-zero.\n", COLVARS_INPUT_ERROR); } } ref_pos.resize(atoms->size()); cvm::load_coords(ref_pos_file.c_str(), &ref_pos, atoms, ref_pos_col, ref_pos_col_value); } else { error_code |= cvm::error( "Error: no reference positions for RMSD; use either refPositions of refPositionsFile.", COLVARS_INPUT_ERROR); } } if (ref_pos.size() != atoms->size()) { error_code |= cvm::error("Error: found " + cvm::to_str(ref_pos.size()) + " reference positions for RMSD; expected " + cvm::to_str(atoms->size()), COLVARS_INPUT_ERROR); } if (atoms->b_user_defined_fit) { cvm::log("WARNING: explicit fitting parameters were provided for atom group \"atoms\".\n"); } else { // Default: fit everything cvm::log("Enabling \"centerToReference\" and \"rotateToReference\", to minimize RMSD before calculating it as a variable: " "if this is not the desired behavior, disable them explicitly within the \"atoms\" block.\n"); atoms->enable(f_ag_center); atoms->enable(f_ag_rotate); // default case: reference positions for calculating the rmsd are also those used // for fitting atoms->ref_pos = ref_pos; atoms->center_ref_pos(); cvm::log("This is a standard minimum RMSD, derivatives of the optimal rotation " "will not be computed as they cancel out in the gradients."); atoms->disable(f_ag_fit_gradients); } atoms->setup_rotation_derivative(); error_code |= init_permutation(conf); return error_code; } int colvar::rmsd::init_permutation(std::string const &conf) { int error_code = COLVARS_OK; std::string perm_conf; size_t pos = 0; // current position in config string n_permutations = 1; while (key_lookup(conf, "atomPermutation", &perm_conf, &pos)) { cvm::main()->cite_feature("Symmetry-adapted RMSD"); std::vector perm; if (perm_conf.size()) { std::istringstream is(perm_conf); size_t index; while (is >> index) { std::vector const &ids = atoms->ids(); size_t const ia = std::find(ids.begin(), ids.end(), index-1) - ids.begin(); if (ia == atoms->size()) { error_code |= cvm::error("Error: atom id " + cvm::to_str(index) + " is not a member of group \"atoms\".", COLVARS_INPUT_ERROR); } if (std::find(perm.begin(), perm.end(), ia) != perm.end()) { error_code |= cvm::error("Error: atom id " + cvm::to_str(index) + " is mentioned more than once in atomPermutation list.", COLVARS_INPUT_ERROR); } perm.push_back(ia); } if (perm.size() != atoms->size()) { error_code |= cvm::error( "Error: symmetry permutation in input contains " + cvm::to_str(perm.size()) + " indices, but group \"atoms\" contains " + cvm::to_str(atoms->size()) + " atoms.", COLVARS_INPUT_ERROR); } cvm::log("atomPermutation = " + cvm::to_str(perm)); n_permutations++; // Record a copy of reference positions in new order for (size_t ia = 0; ia < atoms->size(); ia++) { ref_pos.push_back(ref_pos[perm[ia]]); } } } return error_code; } void colvar::rmsd::calc_value() { // rotational-translational fit is handled by the atom group x.real_value = 0.0; for (size_t ia = 0; ia < atoms->size(); ia++) { x.real_value += ((*atoms)[ia].pos - ref_pos[ia]).norm2(); } best_perm_index = 0; // Compute sum of squares for each symmetry permutation of atoms, keep the smallest size_t ref_pos_index = atoms->size(); for (size_t ip = 1; ip < n_permutations; ip++) { cvm::real value = 0.0; for (size_t ia = 0; ia < atoms->size(); ia++) { value += ((*atoms)[ia].pos - ref_pos[ref_pos_index++]).norm2(); } if (value < x.real_value) { x.real_value = value; best_perm_index = ip; } } x.real_value /= cvm::real(atoms->size()); // MSD x.real_value = cvm::sqrt(x.real_value); } void colvar::rmsd::calc_gradients() { cvm::real const drmsddx2 = (x.real_value > 0.0) ? 0.5 / (x.real_value * cvm::real(atoms->size())) : 0.0; // Use the appropriate symmetry permutation of reference positions to calculate gradients size_t const start = atoms->size() * best_perm_index; for (size_t ia = 0; ia < atoms->size(); ia++) { (*atoms)[ia].grad = (drmsddx2 * 2.0 * ((*atoms)[ia].pos - ref_pos[start + ia])); } } void colvar::rmsd::calc_force_invgrads() { atoms->read_total_forces(); ft.real_value = 0.0; // Note: gradient square norm is 1/N_atoms for (size_t ia = 0; ia < atoms->size(); ia++) { ft.real_value += (*atoms)[ia].grad * (*atoms)[ia].total_force; } ft.real_value *= atoms->size(); } void colvar::rmsd::calc_Jacobian_derivative() { // divergence of the rotated coordinates (including only derivatives of the rotation matrix) cvm::real rotation_term = 0.0; // The rotation term only applies is coordinates are rotated if (atoms->is_enabled(f_ag_rotate)) { // gradient of the rotation matrix cvm::matrix2d grad_rot_mat(3, 3); // gradients of products of 2 quaternion components cvm::rvector g11, g22, g33, g01, g02, g03, g12, g13, g23; cvm::vector1d dq; atoms->rot_deriv->prepare_derivative(rotation_derivative_dldq::use_dq); for (size_t ia = 0; ia < atoms->size(); ia++) { // Gradient of optimal quaternion wrt current Cartesian position atoms->rot_deriv->calc_derivative_wrt_group1(ia, nullptr, &dq); g11 = 2.0 * (atoms->rot.q)[1]*dq[1]; g22 = 2.0 * (atoms->rot.q)[2]*dq[2]; g33 = 2.0 * (atoms->rot.q)[3]*dq[3]; g01 = (atoms->rot.q)[0]*dq[1] + (atoms->rot.q)[1]*dq[0]; g02 = (atoms->rot.q)[0]*dq[2] + (atoms->rot.q)[2]*dq[0]; g03 = (atoms->rot.q)[0]*dq[3] + (atoms->rot.q)[3]*dq[0]; g12 = (atoms->rot.q)[1]*dq[2] + (atoms->rot.q)[2]*dq[1]; g13 = (atoms->rot.q)[1]*dq[3] + (atoms->rot.q)[3]*dq[1]; g23 = (atoms->rot.q)[2]*dq[3] + (atoms->rot.q)[3]*dq[2]; // Gradient of the rotation matrix wrt current Cartesian position grad_rot_mat[0][0] = -2.0 * (g22 + g33); grad_rot_mat[1][0] = 2.0 * (g12 + g03); grad_rot_mat[2][0] = 2.0 * (g13 - g02); grad_rot_mat[0][1] = 2.0 * (g12 - g03); grad_rot_mat[1][1] = -2.0 * (g11 + g33); grad_rot_mat[2][1] = 2.0 * (g01 + g23); grad_rot_mat[0][2] = 2.0 * (g02 + g13); grad_rot_mat[1][2] = 2.0 * (g23 - g01); grad_rot_mat[2][2] = -2.0 * (g11 + g22); cvm::atom_pos &y = ref_pos[ia]; for (size_t alpha = 0; alpha < 3; alpha++) { for (size_t beta = 0; beta < 3; beta++) { rotation_term += grad_rot_mat[beta][alpha][alpha] * y[beta]; // Note: equation was derived for inverse rotation (see colvars paper) // so here the matrix is transposed // (eq would give divergence += grad_rot_mat[alpha][beta][alpha] * y[beta];) } } } } // The translation term only applies is coordinates are centered cvm::real translation_term = atoms->is_enabled(f_ag_center) ? 3.0 : 0.0; jd.real_value = x.real_value > 0.0 ? (3.0 * atoms->size() - 1.0 - translation_term - rotation_term) / x.real_value : 0.0; } colvar::eigenvector::eigenvector() { set_function_type("eigenvector"); provide(f_cvc_inv_gradient); provide(f_cvc_Jacobian); x.type(colvarvalue::type_scalar); } int colvar::eigenvector::init(std::string const &conf) { int error_code = cvc::init(conf); atoms = parse_group(conf, "atoms"); if (!atoms || atoms->size() == 0) { return error_code | COLVARS_INPUT_ERROR; } { bool const b_inline = get_keyval(conf, "refPositions", ref_pos, ref_pos); if (b_inline) { cvm::log("Using reference positions from input file.\n"); if (ref_pos.size() != atoms->size()) { error_code |= cvm::error("Error: reference positions do not " "match the number of requested atoms.\n", COLVARS_INPUT_ERROR); } } std::string file_name; if (get_keyval(conf, "refPositionsFile", file_name)) { if (b_inline) { error_code |= cvm::error( "Error: refPositions and refPositionsFile cannot be specified at the same time.\n", COLVARS_INPUT_ERROR); } std::string file_col; double file_col_value=0.0; if (get_keyval(conf, "refPositionsCol", file_col, std::string(""))) { // use PDB flags if column is provided bool found = get_keyval(conf, "refPositionsColValue", file_col_value, 0.0); if (found && file_col_value == 0.0) { error_code |= cvm::error("Error: refPositionsColValue, if provided, must be non-zero.\n", COLVARS_INPUT_ERROR); } } ref_pos.resize(atoms->size()); cvm::load_coords(file_name.c_str(), &ref_pos, atoms, file_col, file_col_value); } } if (ref_pos.size() == 0) { error_code |= cvm::error("Error: reference positions were not provided.\n", COLVARS_INPUT_ERROR); } if (ref_pos.size() != atoms->size()) { error_code |= cvm::error("Error: reference positions do not " "match the number of requested atoms.\n", COLVARS_INPUT_ERROR); } // save for later the geometric center of the provided positions (may not be the origin) cvm::rvector ref_pos_center(0.0, 0.0, 0.0); for (size_t i = 0; i < atoms->size(); i++) { ref_pos_center += ref_pos[i]; } ref_pos_center *= 1.0 / atoms->size(); if (atoms->b_user_defined_fit) { cvm::log("WARNING: explicit fitting parameters were provided for atom group \"atoms\".\n"); } else { // default: fit everything cvm::log("Enabling \"centerToReference\" and \"rotateToReference\", to minimize RMSD before calculating the vector projection: " "if this is not the desired behavior, disable them explicitly within the \"atoms\" block.\n"); atoms->enable(f_ag_center); atoms->enable(f_ag_rotate); atoms->ref_pos = ref_pos; atoms->center_ref_pos(); atoms->disable(f_ag_fit_gradients); // cancel out if group is fitted on itself // and cvc is translationally invariant } atoms->setup_rotation_derivative(); { bool const b_inline = get_keyval(conf, "vector", eigenvec, eigenvec); // now load the eigenvector if (b_inline) { cvm::log("Using vector components from input file.\n"); if (eigenvec.size() != atoms->size()) { error_code |= cvm::error("Error: vector components do not " "match the number of requested atoms->\n", COLVARS_INPUT_ERROR); } } std::string file_name; if (get_keyval(conf, "vectorFile", file_name)) { if (b_inline) { error_code |= cvm::error("Error: vector and vectorFile cannot be specified at the same time.\n", COLVARS_INPUT_ERROR); } std::string file_col; double file_col_value=0.0; if (get_keyval(conf, "vectorCol", file_col, std::string(""))) { // use PDB flags if column is provided bool found = get_keyval(conf, "vectorColValue", file_col_value, 0.0); if (found && file_col_value==0.0) { error_code |= cvm::error("Error: vectorColValue, if provided, must be non-zero.\n", COLVARS_INPUT_ERROR); } } eigenvec.resize(atoms->size()); cvm::load_coords(file_name.c_str(), &eigenvec, atoms, file_col, file_col_value); } } if (!ref_pos.size() || !eigenvec.size()) { error_code |= cvm::error("Error: both reference coordinates and eigenvector must be defined.\n", COLVARS_INPUT_ERROR); } cvm::atom_pos eig_center(0.0, 0.0, 0.0); for (size_t eil = 0; eil < atoms->size(); eil++) { eig_center += eigenvec[eil]; } eig_center *= 1.0 / atoms->size(); cvm::log("Geometric center of the provided vector: "+cvm::to_str(eig_center)+"\n"); bool b_difference_vector = false; get_keyval(conf, "differenceVector", b_difference_vector, false); if (b_difference_vector) { if (atoms->is_enabled(f_ag_center)) { // both sets should be centered on the origin for fitting for (size_t i = 0; i < atoms->size(); i++) { eigenvec[i] -= eig_center; ref_pos[i] -= ref_pos_center; } } if (atoms->is_enabled(f_ag_rotate)) { atoms->rot.calc_optimal_rotation(eigenvec, ref_pos); const auto rot_mat = atoms->rot.matrix(); for (size_t i = 0; i < atoms->size(); i++) { eigenvec[i] = rot_mat * eigenvec[i]; } } cvm::log("\"differenceVector\" is on: subtracting the reference positions from the provided vector: v = x_vec - x_ref.\n"); for (size_t i = 0; i < atoms->size(); i++) { eigenvec[i] -= ref_pos[i]; } if (atoms->is_enabled(f_ag_center)) { // bring back the ref positions to where they were for (size_t i = 0; i < atoms->size(); i++) { ref_pos[i] += ref_pos_center; } } } else { cvm::log("Centering the provided vector to zero.\n"); for (size_t i = 0; i < atoms->size(); i++) { eigenvec[i] -= eig_center; } } // eigenvec_invnorm2 is used when computing inverse gradients eigenvec_invnorm2 = 0.0; for (size_t ein = 0; ein < atoms->size(); ein++) { eigenvec_invnorm2 += eigenvec[ein].norm2(); } eigenvec_invnorm2 = 1.0 / eigenvec_invnorm2; // Vector normalization overrides the default normalization for differenceVector bool normalize = false; get_keyval(conf, "normalizeVector", normalize, normalize); if (normalize) { cvm::log("Normalizing the vector so that |v| = 1.\n"); for (size_t i = 0; i < atoms->size(); i++) { eigenvec[i] *= cvm::sqrt(eigenvec_invnorm2); } eigenvec_invnorm2 = 1.0; } else if (b_difference_vector) { cvm::log("Normalizing the vector so that the norm of the projection |v ⋅ (x_vec - x_ref)| = 1.\n"); for (size_t i = 0; i < atoms->size(); i++) { eigenvec[i] *= eigenvec_invnorm2; } eigenvec_invnorm2 = 1.0/eigenvec_invnorm2; } else { cvm::log("The norm of the vector is |v| = "+ cvm::to_str(1.0/cvm::sqrt(eigenvec_invnorm2))+".\n"); } return error_code; } void colvar::eigenvector::calc_value() { x.real_value = 0.0; for (size_t i = 0; i < atoms->size(); i++) { x.real_value += ((*atoms)[i].pos - ref_pos[i]) * eigenvec[i]; } } void colvar::eigenvector::calc_gradients() { for (size_t ia = 0; ia < atoms->size(); ia++) { (*atoms)[ia].grad = eigenvec[ia]; } } void colvar::eigenvector::calc_force_invgrads() { atoms->read_total_forces(); ft.real_value = 0.0; for (size_t ia = 0; ia < atoms->size(); ia++) { ft.real_value += eigenvec_invnorm2 * (*atoms)[ia].grad * (*atoms)[ia].total_force; } } void colvar::eigenvector::calc_Jacobian_derivative() { // gradient of the rotation matrix cvm::matrix2d grad_rot_mat(3, 3); cvm::quaternion &quat0 = atoms->rot.q; // gradients of products of 2 quaternion components cvm::rvector g11, g22, g33, g01, g02, g03, g12, g13, g23; cvm::real sum = 0.0; cvm::vector1d dq_1; atoms->rot_deriv->prepare_derivative(rotation_derivative_dldq::use_dq); for (size_t ia = 0; ia < atoms->size(); ia++) { // Gradient of optimal quaternion wrt current Cartesian position // trick: d(R^-1)/dx = d(R^t)/dx = (dR/dx)^t // we can just transpose the derivatives of the direct matrix atoms->rot_deriv->calc_derivative_wrt_group1(ia, nullptr, &dq_1); g11 = 2.0 * quat0[1]*dq_1[1]; g22 = 2.0 * quat0[2]*dq_1[2]; g33 = 2.0 * quat0[3]*dq_1[3]; g01 = quat0[0]*dq_1[1] + quat0[1]*dq_1[0]; g02 = quat0[0]*dq_1[2] + quat0[2]*dq_1[0]; g03 = quat0[0]*dq_1[3] + quat0[3]*dq_1[0]; g12 = quat0[1]*dq_1[2] + quat0[2]*dq_1[1]; g13 = quat0[1]*dq_1[3] + quat0[3]*dq_1[1]; g23 = quat0[2]*dq_1[3] + quat0[3]*dq_1[2]; // Gradient of the inverse rotation matrix wrt current Cartesian position // (transpose of the gradient of the direct rotation) grad_rot_mat[0][0] = -2.0 * (g22 + g33); grad_rot_mat[0][1] = 2.0 * (g12 + g03); grad_rot_mat[0][2] = 2.0 * (g13 - g02); grad_rot_mat[1][0] = 2.0 * (g12 - g03); grad_rot_mat[1][1] = -2.0 * (g11 + g33); grad_rot_mat[1][2] = 2.0 * (g01 + g23); grad_rot_mat[2][0] = 2.0 * (g02 + g13); grad_rot_mat[2][1] = 2.0 * (g23 - g01); grad_rot_mat[2][2] = -2.0 * (g11 + g22); for (size_t i = 0; i < 3; i++) { for (size_t j = 0; j < 3; j++) { sum += grad_rot_mat[i][j][i] * eigenvec[ia][j]; } } } jd.real_value = sum * cvm::sqrt(eigenvec_invnorm2); } colvar::cartesian::cartesian() { set_function_type("cartesian"); x.type(colvarvalue::type_vector); disable(f_cvc_explicit_gradient); } int colvar::cartesian::init(std::string const &conf) { int error_code = cvc::init(conf); atoms = parse_group(conf, "atoms"); bool use_x, use_y, use_z; get_keyval(conf, "useX", use_x, true); get_keyval(conf, "useY", use_y, true); get_keyval(conf, "useZ", use_z, true); axes.clear(); if (use_x) axes.push_back(0); if (use_y) axes.push_back(1); if (use_z) axes.push_back(2); if (axes.size() == 0) { error_code |= cvm::error("Error: a \"cartesian\" component was defined with all three axes disabled.\n", COLVARS_INPUT_ERROR); } // Don't try to access atoms if creation of the atom group failed if (atoms) x.vector1d_value.resize(atoms->size() * axes.size()); return error_code; } void colvar::cartesian::calc_value() { size_t const dim = axes.size(); size_t ia, j; for (ia = 0; ia < atoms->size(); ia++) { for (j = 0; j < dim; j++) { x.vector1d_value[dim*ia + j] = (*atoms)[ia].pos[axes[j]]; } } } void colvar::cartesian::calc_gradients() { // we're not using the "grad" member of each // atom object, because it only can represent the gradient of a // scalar colvar } void colvar::cartesian::apply_force(colvarvalue const &force) { size_t const dim = axes.size(); size_t ia, j; if (!atoms->noforce) { cvm::rvector f; for (ia = 0; ia < atoms->size(); ia++) { for (j = 0; j < dim; j++) { f[axes[j]] = force.vector1d_value[dim*ia + j]; } (*atoms)[ia].apply_force(f); } } } cvm::real colvar::cartesian::dist2(colvarvalue const &x1, colvarvalue const &x2) const { return (x1.vector1d_value - x2.vector1d_value).norm2(); } colvarvalue colvar::cartesian::dist2_lgrad(colvarvalue const &x1, colvarvalue const &x2) const { return 2.0 * (x1.vector1d_value - x2.vector1d_value); } colvarvalue colvar::cartesian::dist2_rgrad(colvarvalue const &x1, colvarvalue const &x2) const { return cartesian::dist2_lgrad(x1, x2); } void colvar::cartesian::wrap(colvarvalue & /* x_unwrapped */) const {}