void upsert_data(struct io_node *ion) { struct io_node *n; /* List is empty. */ if (head == NULL) { head = ion; return; } /* Check if we have seen this pid before. */ n = head; while (n != NULL) { if (n->pid == ion->pid) { n->rchar = ion->rchar; n->wchar = ion->wchar; n->syscr = ion->syscr; n->syscw = ion->syscw; n->read_bytes = ion->read_bytes; n->write_bytes = ion->write_bytes; n->cancelled_write_bytes = ion->cancelled_write_bytes; /* * If the pids wrap, then the command may be different then before. */ strcpy(n->command, ion->command); free(ion); return; } n = n->next; } /* Add this pid to the list. */ head = insert_ion(ion); return; }
int gmx_genion(int argc, char *argv[]) { const char *desc[] = { "[THISMODULE] randomly replaces solvent molecules with monoatomic ions.", "The group of solvent molecules should be continuous and all molecules", "should have the same number of atoms.", "The user should add the ion molecules to the topology file or use", "the [TT]-p[tt] option to automatically modify the topology.[PAR]", "The ion molecule type, residue and atom names in all force fields", "are the capitalized element names without sign. This molecule name", "should be given with [TT]-pname[tt] or [TT]-nname[tt], and the", "[TT][molecules][tt] section of your topology updated accordingly,", "either by hand or with [TT]-p[tt]. Do not use an atom name instead!", "[PAR]Ions which can have multiple charge states get the multiplicity", "added, without sign, for the uncommon states only.[PAR]", "For larger ions, e.g. sulfate we recommended using [gmx-insert-molecules]." }; const char *bugs[] = { "If you specify a salt concentration existing ions are not taken into " "account. In effect you therefore specify the amount of salt to be added.", }; static int p_num = 0, n_num = 0, p_q = 1, n_q = -1; static const char *p_name = "NA", *n_name = "CL"; static real rmin = 0.6, conc = 0; static int seed = 1993; static gmx_bool bNeutral = FALSE; static t_pargs pa[] = { { "-np", FALSE, etINT, {&p_num}, "Number of positive ions" }, { "-pname", FALSE, etSTR, {&p_name}, "Name of the positive ion" }, { "-pq", FALSE, etINT, {&p_q}, "Charge of the positive ion" }, { "-nn", FALSE, etINT, {&n_num}, "Number of negative ions" }, { "-nname", FALSE, etSTR, {&n_name}, "Name of the negative ion" }, { "-nq", FALSE, etINT, {&n_q}, "Charge of the negative ion" }, { "-rmin", FALSE, etREAL, {&rmin}, "Minimum distance between ions" }, { "-seed", FALSE, etINT, {&seed}, "Seed for random number generator" }, { "-conc", FALSE, etREAL, {&conc}, "Specify salt concentration (mol/liter). This will add sufficient ions to reach up to the specified concentration as computed from the volume of the cell in the input [REF].tpr[ref] file. Overrides the [TT]-np[tt] and [TT]-nn[tt] options." }, { "-neutral", FALSE, etBOOL, {&bNeutral}, "This option will add enough ions to neutralize the system. These ions are added on top of those specified with [TT]-np[tt]/[TT]-nn[tt] or [TT]-conc[tt]. "} }; t_topology top; rvec *x, *v; real vol, qtot; matrix box; t_atoms atoms; t_pbc pbc; int *repl, ePBC; atom_id *index; char *grpname; gmx_bool *bSet; int i, nw, nwa, nsa, nsalt, iqtot; gmx_output_env_t *oenv; gmx_rng_t rng; t_filenm fnm[] = { { efTPR, NULL, NULL, ffREAD }, { efNDX, NULL, NULL, ffOPTRD }, { efSTO, "-o", NULL, ffWRITE }, { efTOP, "-p", "topol", ffOPTRW } }; #define NFILE asize(fnm) if (!parse_common_args(&argc, argv, 0, NFILE, fnm, asize(pa), pa, asize(desc), desc, asize(bugs), bugs, &oenv)) { return 0; } /* Check input for something sensible */ if ((p_num < 0) || (n_num < 0)) { gmx_fatal(FARGS, "Negative number of ions to add?"); } if (conc > 0 && (p_num > 0 || n_num > 0)) { fprintf(stderr, "WARNING: -conc specified, overriding -nn and -np.\n"); } /* Read atom positions and charges */ read_tps_conf(ftp2fn(efTPR, NFILE, fnm), &top, &ePBC, &x, &v, box, FALSE); atoms = top.atoms; /* Compute total charge */ qtot = 0; for (i = 0; (i < atoms.nr); i++) { qtot += atoms.atom[i].q; } iqtot = std::round(qtot); if (conc > 0) { /* Compute number of ions to be added */ vol = det(box); nsalt = std::round(conc*vol*AVOGADRO/1e24); p_num = abs(nsalt*n_q); n_num = abs(nsalt*p_q); } if (bNeutral) { int qdelta = p_num*p_q + n_num*n_q + iqtot; /* Check if the system is neutralizable * is (qdelta == p_q*p_num + n_q*n_num) solvable for p_num and n_num? */ int gcd = gmx_greatest_common_divisor(n_q, p_q); if ((qdelta % gcd) != 0) { gmx_fatal(FARGS, "Can't neutralize this system using -nq %d and" " -pq %d.\n", n_q, p_q); } while (qdelta != 0) { while (qdelta < 0) { p_num++; qdelta += p_q; } while (qdelta > 0) { n_num++; qdelta += n_q; } } } if ((p_num == 0) && (n_num == 0)) { fprintf(stderr, "No ions to add, will just copy input configuration.\n"); } else { printf("Will try to add %d %s ions and %d %s ions.\n", p_num, p_name, n_num, n_name); printf("Select a continuous group of solvent molecules\n"); get_index(&atoms, ftp2fn_null(efNDX, NFILE, fnm), 1, &nwa, &index, &grpname); for (i = 1; i < nwa; i++) { if (index[i] != index[i-1]+1) { gmx_fatal(FARGS, "The solvent group %s is not continuous: " "index[%d]=%d, index[%d]=%d", grpname, i, index[i-1]+1, i+1, index[i]+1); } } nsa = 1; while ((nsa < nwa) && (atoms.atom[index[nsa]].resind == atoms.atom[index[nsa-1]].resind)) { nsa++; } if (nwa % nsa) { gmx_fatal(FARGS, "Your solvent group size (%d) is not a multiple of %d", nwa, nsa); } nw = nwa/nsa; fprintf(stderr, "Number of (%d-atomic) solvent molecules: %d\n", nsa, nw); if (p_num+n_num > nw) { gmx_fatal(FARGS, "Not enough solvent for adding ions"); } if (opt2bSet("-p", NFILE, fnm)) { update_topol(opt2fn("-p", NFILE, fnm), p_num, n_num, p_name, n_name, grpname); } snew(bSet, nw); snew(repl, nw); snew(v, atoms.nr); snew(atoms.pdbinfo, atoms.nr); set_pbc(&pbc, ePBC, box); if (seed == 0) { rng = gmx_rng_init(gmx_rng_make_seed()); } else { rng = gmx_rng_init(seed); } /* Now loop over the ions that have to be placed */ while (p_num-- > 0) { insert_ion(nsa, &nw, bSet, repl, index, x, &pbc, 1, p_q, p_name, &atoms, rmin, rng); } while (n_num-- > 0) { insert_ion(nsa, &nw, bSet, repl, index, x, &pbc, -1, n_q, n_name, &atoms, rmin, rng); } gmx_rng_destroy(rng); fprintf(stderr, "\n"); if (nw) { sort_ions(nsa, nw, repl, index, &atoms, x, p_name, n_name); } sfree(atoms.pdbinfo); atoms.pdbinfo = NULL; } write_sto_conf(ftp2fn(efSTO, NFILE, fnm), *top.name, &atoms, x, NULL, ePBC, box); return 0; }
int gmx_genion(int argc, char *argv[]) { const char *desc[] = { "[TT]genion[tt] replaces solvent molecules by monoatomic ions at", "the position of the first atoms with the most favorable electrostatic", "potential or at random. The potential is calculated on all atoms, using", "normal GROMACS particle-based methods (in contrast to other methods", "based on solving the Poisson-Boltzmann equation).", "The potential is recalculated after every ion insertion.", "If specified in the run input file, a reaction field, shift function", "or user function can be used. For the user function a table file", "can be specified with the option [TT]-table[tt].", "The group of solvent molecules should be continuous and all molecules", "should have the same number of atoms.", "The user should add the ion molecules to the topology file or use", "the [TT]-p[tt] option to automatically modify the topology.[PAR]", "The ion molecule type, residue and atom names in all force fields", "are the capitalized element names without sign. This molecule name", "should be given with [TT]-pname[tt] or [TT]-nname[tt], and the", "[TT][molecules][tt] section of your topology updated accordingly,", "either by hand or with [TT]-p[tt]. Do not use an atom name instead!", "[PAR]Ions which can have multiple charge states get the multiplicity", "added, without sign, for the uncommon states only.[PAR]", "With the option [TT]-pot[tt] the potential can be written as B-factors", "in a [TT].pdb[tt] file (for visualisation using e.g. Rasmol).", "The unit of the potential is 1000 kJ/(mol e), the scaling be changed", "with the [TT]-scale[tt] option.[PAR]", "For larger ions, e.g. sulfate we recommended using [TT]genbox[tt]." }; const char *bugs[] = { "Calculation of the potential is not reliable, therefore the [TT]-random[tt] option is now turned on by default.", "If you specify a salt concentration existing ions are not taken into account. In effect you therefore specify the amount of salt to be added." }; static int p_num = 0, n_num = 0, p_q = 1, n_q = -1; static const char *p_name = "NA", *n_name = "CL"; static real rmin = 0.6, scale = 0.001, conc = 0; static int seed = 1993; static gmx_bool bRandom = TRUE, bNeutral = FALSE; static t_pargs pa[] = { { "-np", FALSE, etINT, {&p_num}, "Number of positive ions" }, { "-pname", FALSE, etSTR, {&p_name}, "Name of the positive ion" }, { "-pq", FALSE, etINT, {&p_q}, "Charge of the positive ion" }, { "-nn", FALSE, etINT, {&n_num}, "Number of negative ions" }, { "-nname", FALSE, etSTR, {&n_name}, "Name of the negative ion" }, { "-nq", FALSE, etINT, {&n_q}, "Charge of the negative ion" }, { "-rmin", FALSE, etREAL, {&rmin}, "Minimum distance between ions" }, { "-random", FALSE, etBOOL, {&bRandom}, "Use random placement of ions instead of based on potential. The rmin option should still work" }, { "-seed", FALSE, etINT, {&seed}, "Seed for random number generator" }, { "-scale", FALSE, etREAL, {&scale}, "Scaling factor for the potential for [TT]-pot[tt]" }, { "-conc", FALSE, etREAL, {&conc}, "Specify salt concentration (mol/liter). This will add sufficient ions to reach up to the specified concentration as computed from the volume of the cell in the input [TT].tpr[tt] file. Overrides the [TT]-np[tt] and [TT]-nn[tt] options." }, { "-neutral", FALSE, etBOOL, {&bNeutral}, "This option will add enough ions to neutralize the system. These ions are added on top of those specified with [TT]-np[tt]/[TT]-nn[tt] or [TT]-conc[tt]. "} }; gmx_mtop_t *mtop; gmx_localtop_t *top; t_inputrec inputrec; t_commrec *cr; t_mdatoms *mdatoms; gmx_enerdata_t enerd; t_graph *graph; t_forcerec *fr; rvec *x, *v; real *pot, vol, qtot; matrix box; t_atoms atoms; t_pbc pbc; int *repl; atom_id *index; char *grpname; gmx_bool *bSet, bPDB; int i, nw, nwa, nsa, nsalt, iqtot; FILE *fplog; output_env_t oenv; t_filenm fnm[] = { { efTPX, NULL, NULL, ffREAD }, { efXVG, "-table", "table", ffOPTRD }, { efNDX, NULL, NULL, ffOPTRD }, { efSTO, "-o", NULL, ffWRITE }, { efLOG, "-g", "genion", ffWRITE }, { efPDB, "-pot", "pot", ffOPTWR }, { efTOP, "-p", "topol", ffOPTRW } }; #define NFILE asize(fnm) parse_common_args(&argc, argv, PCA_BE_NICE, NFILE, fnm, asize(pa), pa, asize(desc), desc, asize(bugs), bugs, &oenv); bPDB = ftp2bSet(efPDB, NFILE, fnm); if (bRandom && bPDB) { fprintf(stderr, "Not computing potential with random option!\n"); bPDB = FALSE; } /* Check input for something sensible */ if ((p_num < 0) || (n_num < 0)) { gmx_fatal(FARGS, "Negative number of ions to add?"); } snew(mtop, 1); snew(top, 1); fplog = init_calcpot(ftp2fn(efLOG, NFILE, fnm), ftp2fn(efTPX, NFILE, fnm), opt2fn("-table", NFILE, fnm), mtop, top, &inputrec, &cr, &graph, &mdatoms, &fr, &enerd, &pot, box, &x, oenv); atoms = gmx_mtop_global_atoms(mtop); qtot = 0; for (i = 0; (i < atoms.nr); i++) { qtot += atoms.atom[i].q; } iqtot = gmx_nint(qtot); if (conc > 0) { /* Compute number of ions to be added */ vol = det(box); nsalt = gmx_nint(conc*vol*AVOGADRO/1e24); p_num = abs(nsalt*n_q); n_num = abs(nsalt*p_q); } if (bNeutral) { int qdelta = p_num*p_q + n_num*n_q + iqtot; /* Check if the system is neutralizable * is (qdelta == p_q*p_num + n_q*n_num) solvable for p_num and n_num? */ int gcd = greatest_common_divisor(n_q, p_q); if ((qdelta % gcd) != 0) { gmx_fatal(FARGS, "Can't neutralize this system using -nq %d and" " -pq %d.\n", n_q, p_q); } while (qdelta != 0) { while (qdelta < 0) { p_num++; qdelta += p_q; } while (qdelta > 0) { n_num++; qdelta += n_q; } } } if ((p_num == 0) && (n_num == 0)) { if (!bPDB) { fprintf(stderr, "No ions to add and no potential to calculate.\n"); exit(0); } nw = 0; nsa = 0; /* to keep gcc happy */ } else { printf("Will try to add %d %s ions and %d %s ions.\n", p_num, p_name, n_num, n_name); printf("Select a continuous group of solvent molecules\n"); get_index(&atoms, ftp2fn_null(efNDX, NFILE, fnm), 1, &nwa, &index, &grpname); for (i = 1; i < nwa; i++) { if (index[i] != index[i-1]+1) { gmx_fatal(FARGS, "The solvent group %s is not continuous: " "index[%d]=%d, index[%d]=%d", grpname, i, index[i-1]+1, i+1, index[i]+1); } } nsa = 1; while ((nsa < nwa) && (atoms.atom[index[nsa]].resind == atoms.atom[index[nsa-1]].resind)) { nsa++; } if (nwa % nsa) { gmx_fatal(FARGS, "Your solvent group size (%d) is not a multiple of %d", nwa, nsa); } nw = nwa/nsa; fprintf(stderr, "Number of (%d-atomic) solvent molecules: %d\n", nsa, nw); if (p_num+n_num > nw) { gmx_fatal(FARGS, "Not enough solvent for adding ions"); } } if (opt2bSet("-p", NFILE, fnm)) { update_topol(opt2fn("-p", NFILE, fnm), p_num, n_num, p_name, n_name, grpname); } snew(bSet, nw); snew(repl, nw); snew(v, atoms.nr); snew(atoms.pdbinfo, atoms.nr); set_pbc(&pbc, inputrec.ePBC, box); /* Now loop over the ions that have to be placed */ do { if (!bRandom) { calc_pot(fplog, cr, mtop, &inputrec, top, x, fr, &enerd, mdatoms, pot, box, graph); if (bPDB || debug) { char buf[STRLEN]; if (debug) { sprintf(buf, "%d_%s", p_num+n_num, ftp2fn(efPDB, NFILE, fnm)); } else { strcpy(buf, ftp2fn(efPDB, NFILE, fnm)); } for (i = 0; (i < atoms.nr); i++) { atoms.pdbinfo[i].bfac = pot[i]*scale; } write_sto_conf(buf, "Potential calculated by genion", &atoms, x, v, inputrec.ePBC, box); bPDB = FALSE; } } if ((p_num > 0) && (p_num >= n_num)) { insert_ion(nsa, &nw, bSet, repl, index, pot, x, &pbc, 1, p_q, p_name, mdatoms, rmin, bRandom, &seed); p_num--; } else if (n_num > 0) { insert_ion(nsa, &nw, bSet, repl, index, pot, x, &pbc, -1, n_q, n_name, mdatoms, rmin, bRandom, &seed); n_num--; } } while (p_num+n_num > 0); fprintf(stderr, "\n"); if (nw) { sort_ions(nsa, nw, repl, index, &atoms, x, p_name, n_name); } sfree(atoms.pdbinfo); atoms.pdbinfo = NULL; write_sto_conf(ftp2fn(efSTO, NFILE, fnm), *mtop->name, &atoms, x, NULL, inputrec.ePBC, box); thanx(stderr); gmx_log_close(fplog); return 0; }