Example #1
0
int main() {
	ifstream iNodes("Mathematica/Generate Mesh/nodes.dat"),
	         iTriangles("Mathematica/Generate Mesh/triangles.dat"),
	         iNeighbors("Mathematica/Generate Mesh/neighbors.dat");
	ofstream oXi("Mathematica/Draw Logo/xi.dat"),
	         oDirichletNodes("Mathematica/Draw Logo/DirichletNodes.dat"),
	         oNodes("Mathematica/Draw Logo/nodes.dat"), 
	         oTriangles("Mathematica/Draw Logo/triangles.dat");
	try {
		// import kitty mesh
		Triangulation Omega(iNodes, iTriangles, iNeighbors);
		DiffusionReactionEqn PoissonEqn(oneFunc, zeroFunc, f);
		BoundaryConditions BCs({
			make_shared<DirichletBC>(oneFunc, fixedBoundary),
			make_shared<NeumannBC>() // free bndry on right half of kitty’s face
		});
		// solve
		vector<double> soln = FEM::computeDiscreteSolution(PoissonEqn, Omega, BCs);
		// save the result
		oXi << soln;
		Boundary bndry = Omega.computeBoundary();
		oDirichletNodes << FEM::computeBoundaryNodes(Omega, bndry, fixedBoundary);
		Omega.save(oNodes, oTriangles);
	}
	catch (exception const & e) {
		cout << e.what() << endl;
	}
}
Example #2
0
//Float
//pibond(BrennerMainInfo *const info, struct AtomPairInfoState *const apis)
Float BrennerPotential::pibond()
{
  const int num_atms = nAtoms;
  /* When we're looking at a bond from atom i to atom j, and we're considering
     all neighbors of j, then xk will be the bren_vector from atom i to each
     neighbor of j. */
  bren_vector xk[250+1];
  /* Similarly, when we're considering all neighbors of i, xl will be the
     bren_vector from atom j to each neighbor of i. */
  bren_vector xl[250+1],cj,ck,cl,rk,rl,dt2dik,dt2djl,dt2dij;
  Float xsik[250+1],xsjk[250+1],xsil[250+1],xsjl[250+1]
    ,dexni[2],dexnj[2],xni[2],xnj[2]
    ,cfuni[ncc],cfunj[ncc],dcfuni[ncc],dcfunj[ncc]
    ,cosk[250+1],cosl[250+1],sink[250+1],sinl[250+1]
    ,dctjk[250+1],dctij[250+1],dctik[250+1],dctil[250+1],dctji[250+1],
    dctjl[250+1];
  Float tote = 0;
  int i;
  vector<Float> hydrogens_connected(num_atms);
  vector<Float> carbons_connected(num_atms);
  //Float *hydrogens_connected = (Float*)xmalloc(num_atms*sizeof(Float));
  //Float *carbons_connected = (Float*)xmalloc(num_atms*sizeof(Float));
  Float *xhc[2];
  //const struct NeighborState *const cans = caNeighborsconst (info);
  xhc[0] = &hydrogens_connected[0];
  xhc[1] = &carbons_connected[0];
  for(i = 0; i < num_atms; ++i)
    xhc[0][i] = xhc[1][i] = 0;
  for(i = 0; i < ncc; ++i)
    cfuni[i] = cfunj[i] = dcfuni[i] = dcfunj[i] = 0;
  dexni[0]=dexnj[0]=xni[0]=xnj[0] = 0;
  dexni[1]=dexnj[1]=xni[1]=xnj[1] = 0;
  /*
   * Find the number of hydrogens and carbons connected to each atom.
   */
  // Allocate buffers for neighbor list data
  int nmaxnb = nblist->MaxNeighborListLength();
  vector<int> iNeighbors(nmaxnb);
  vector<Vec> distances(nmaxnb);       // Not used?
  vector<double> sq_distances(nmaxnb); // Not used?
  for(i = 0; i < num_atms; ++i) {
    int sizemax = nmaxnb;
    int nCount = nblist->GetFullNeighbors(i, &iNeighbors[0], &distances[0],
					  &sq_distances[0], sizemax);
    /* Early escape if there is no neighbours to this atoms. This also avoids a memory access error */
    if (nCount == 0) continue;					  
    const struct AtomPairInfo *const pairs = pairsconst (i, apis);
    int jn;
    /* Why do we start counting at 1 instead of 0 here?  We might not have any
       hydrogens at all in the figure, and in that case we'll set some field of
       hydrogens_connected to 1.  Weird. Tim Freeman 6 Sep 2000. */
    hydrogens_connected[i] = 1;
    carbons_connected[i] = 1;
    for (jn = 0; jn < nCount; jn++) {
      if(1 == pairs[jn].lcheck) {
        const int j = iNeighbors [jn];
        const int atype = getKtype(j);
        assert(atype > 0 && atype <= 2);
        xhc[atype-1][i] += pairs[jn].ww;
      }
    }
  }
  /*
   * Sum over bonds between atoms I and J
   */
  for(i = 0; i < num_atms; ++i) {
    /* j is the index of this neighbor pair in the neighbors array.  Note that
       j is an entirely different beast from i, which is an index into the atom
       array. */
    int j;
    const int ki = getKtype(i);
    int sizemax = nmaxnb;
    int nCount = nblist->GetFullNeighbors(i, &iNeighbors[0], &distances[0],
					  &sq_distances[0], sizemax);
    /* Early escape if there is no neighbours to this atoms. This also avoids a memory access error */
    if (nCount == 0) continue;		
    const struct AtomPairInfo *const atom_pair = pairsconst (i, apis);
    for (j = 0; j < nCount; j++) {
      /* jn will be the atom number of the current neighbor. */
      const int jn = iNeighbors [j];
      /* calc_dihedral_terms adds a value to btot but does not otherwise read
         its value. */
      Float conjug, xnt1, xnt2, btot;
      /* vdbdi, vdbdj, vdrdi, vdrdj, and vdrdc are variables that are
         consequences of what we got from calc_dihedral_terms and then inputs
         into calc_many_body. */
      Float vdbdi, vdbdj, vdrdi, vdrdj, vdrdc;
      /* sdalik and sdaljl are outputs from calc1side I don't understand
         yet and inputs to calc_many_body. */
      Float sdalik, sdaljl;
      /* conk and conl are outputs from calc1side I don't understand yet and
         inputs into calc_many_body. */
      Float conk, conl;
      /* cj is the bren_vector from i to j. */
      bren_vector cj;
      /* If one of them isn't hydrogen or carbon, then look at the 
         next pair. */
      if(atom_pair[j].lcheck != 1) continue;
      if(i >= jn) continue;
      /* Now we're considering each pair only once. */
      cj = atom_pair[j].cor;
      {
        /* kj will be the ktype for the current neighbor. */
        const int kj = getKtype(jn);
        /* sij will be the distance from one atom to the other. */
        const Float sij = atom_pair[j].rcor;
        const Float rsqij = sij*sij;
        /* xsij and xsji are an output from calc1side I don't understand
           yet. */
        Float xsij, xsji;
        /* ssumk is the middle term under the square root in equation 7 on page
           10. */
        Float ssumk, ssuml;
        /* exnij is pi super rc sub i j, mentioned in equation 3 on page 6 and
           defined in equation 12 on page 14.  exnji is pi super rc sub j i. */
        Float exnij, exnji;
	// XXXX Remove info from following calls!
        calc1side(ki, kj, i,  j, j, -1,
                  xni, atom_pair, atom_pair,
                  cj, &xsij, xhc, cfuni, dcfuni,
                  &conk, dctjk, dctij, dctik, &ssumk, &sdalik, xsjk, xsik,
                  sij, rsqij,
                  xk, cosk, sink, &exnij, dexni);
        calc1side(kj, ki, jn, i, j, 1,
                  xnj, atom_pair,
                  pairs (jn, apis),
                  cj, &xsji, xhc, cfunj, dcfunj,
                  &conl, dctil, dctji, dctjl, &ssuml, &sdaljl, xsil, xsjl,
                  sij, rsqij,
                  xl, cosl, sinl, &exnji, dexnj);
        {
          Float dbdzi, dbdzj;
          Float dradi = 0.0;
          Float dradj = 0.0;
          Float drdc = 0.0;
          {
            Float bji, bij, rad;
            {
              Float dij;
              /* dij is the expression inside the square root of equation 7 on page
                 10. */
              dij = (1.0+exnij+ssumk);
              /* The next line is equation 7 on page 10.  
                 bij is p super sigma pi sub i j. */
              bij = 1/sqrt(dij);
              dbdzi = -0.50*bij/dij;
            } /* Forget dij. */
            {
              Float dji = (1.0+exnji+ssuml);
              bji = 1/sqrt(dji);
              dbdzj = -0.50*bji/dji;
            } /* Forget dji. */
            /* Next line is equation 13 on page 14.
               conjug is N super conj sub i j. */
            conjug = 1.0 + (conk*conk) + (conl*conl);
            /* Next line is equation 8a on page 11.
               xnt1 is N super C sub i. */
            xnt1 = xni[0]+xni[1]-1.0;
            /* Next line is equation 8b on page 11.
               xnt2 is N super H sub i. */
            xnt2 = xnj[0]+xnj[1]-1.0;
            /* Next line is equation 12 on page 14.
               rad is pi super rc sub i j. 
               rc probably stands for radical. */
            rad = RADIC(ki,kj,xnt1,xnt2,conjug,&dradi,&dradj,&drdc); // XXX
            btot = (bji+bij+rad);
          } /* Forget bji, bij, rad. */
          {
            /* vatt is the energy due to attraction between the pair. */
            const Float vatt = atom_pair[j].exx1;
            const int kikj = ki+kj;
            /*
             * Dihedral terms
             */
            /* ndihed is 2, and kikj is the sum of the ktypes ki and kj, so
               "kikj == ndihed" is a very obscure way to say that both i and j
               are carbon (ktype 1), since ktype's are integers and are never
               zero or negative. */
            if(kikj == ndihed)
	      // XXXX Remove info
              calc_dihedral_terms(i, j, jn, apis, xnt1, xnt2,
                                  conjug, sij, sink, sinl, dctik, dctjk,
                                  dctil, dctij, dctjl, dctji, cosk, cosl, cj,
                                  vatt, xk, xl, &btot, &dradi, &dradj, &drdc);
            /* btot is the sum of the b's, specifically the b sub i j in
               equation 1 on page 4.  vatt is V super A sub i j in equation 1
               on page 4. */
            tote -= btot*vatt;
            /* dbdzi is the derivative of b with respect to i-dont-know-what.
               vdbdi is the derivative of the total energy with respect to the
               same thing, due to changes in b. */
            vdbdi = vatt*dbdzi;
            vdbdj = vatt*dbdzj;
            vdrdc = vatt*drdc;
            vdrdi = vatt*dradi;
            vdrdj = vatt*dradj;
          } /* Forget vatt. */
        } /* Forget dbdzi, dbdzj, drdc, dradi, dradj */
        {
          Float rp = vdbdi*xsij + vdbdj*xsji + btot*atom_pair[j].dexx1;
#ifndef NDEBUG
          if(!(rp < 1.e99))
            printf("rp %d %d %d %11.8f %11.8f %11.8f %11.8f %11.8f %11.8f\n",
                   j, i, jn, vdbdi,xsij, vdbdj,xsji, btot, atom_pair[j].dexx1);
          
#endif
          transForce(i, jn, rp*cj);
        } /* Forget rp. */
      } /* Forget sij, rsqij, xsij, xsji. */
      /*
       * Add many-body forces
       */
      // XXX Remove info
      calc_many_body(apis, 1, i, jn, j, /* i side */
                     vdbdi, xsik, dexni, vdrdi, vdrdc, cfuni, sdalik, xsjk, xk,
                     dcfuni, conk);
      calc_many_body(apis, 0, jn, i, -1, /* j side */
                     vdbdj, xsjl, dexnj, vdrdj, vdrdc, cfunj, sdaljl, xsil, xl,
                     dcfunj, conl);
    }
  }
  //free(hydrogens_connected);
  //free(carbons_connected);
  return tote;
}
Example #3
0
/* This routine implements equation 16 on page 15, computing 
   pi super dh sub i j.  (dh is for dihedral.)
   We update the forces in info, so it can't be const.
*/
void BrennerPotential::calc_dihedral_terms(
		    int i, int j, int jn,
                    const struct AtomPairInfoState *apis,
                    /* xnt1 is N super t sub i. */
                    Float xnt1,
                    /* xnt2 is N super t sub j. */
                    Float xnt2,
                    /* conjug is N sub conj. */
                    Float conjug, Float sij,
                    Float *sink, Float *sinl, Float *dctik, Float *dctjk,
                    Float *dctil, Float *dctij, Float *dctjl, Float *dctji,
                    Float *cosk, Float *cosl, bren_vector cj,
                    Float vatt,
                    const bren_vector *xk, const bren_vector *xl,
                    /* btot will be the total energy, that is,
                       pi super dh sub i j. */ 
                    Float *btot,
                    /* dradi is returned as the partial of total energy with
                       respect to xnt1. */
                    Float *dradi,
                    /* dradj is returned as the partial of total energy with
                       respect to xnt2. */
                    Float *dradj,
                    /* drdc is returned as the partial of total energy with
                       respect to conjug. */
                    Float *drdc)
{
  /* k will be an index into the list of neighbors of i. */
  int k, nk;
  //  XXX The following neighbor list access should be optimized away
  int nmaxnb = nblist->MaxNeighborListLength();
  vector<int> iNeighbors(nmaxnb);
  vector<Vec> distances(nmaxnb);
  vector<double> sq_distances(nmaxnb);
  /* stop_index is one past the index of the last neighbor. */
  int sizemax = nmaxnb;
  const int stop_index =
    nblist->GetFullNeighbors(i, &iNeighbors[0], &distances[0],
			     &sq_distances[0], sizemax);

  const int *neighbor_start = &iNeighbors[0];
  const struct AtomPairInfo *iatom_pair = pairsconst (i, apis);
  Float dbtori = 0.0;
  Float dbtorj = 0.0;
  Float dbtorc = 0.0;
  Float btor = 0.0;
  Float datori,datorj ,datorc;
  Float ator = TOR(xnt1,xnt2,conjug,&datori,&datorj,&datorc);  // XXX
  Float vatt_ator = vatt*ator;

  if(fabs(ator) <= 1.0e-08) {
     return;
  }

  nk = 0;
  for(k = 0; k < stop_index; ++k) {
    Float sink2, rck, fck, dfck;
    bren_vector ck, cl, dt2djl;
    const bren_vector xk_nk = xk[nk+1];
    Float crkx, crky, crkz;
    int nl, l;
    int kn;
    Float cosk_nk;
    //const int *jneighbor_start;
    const struct AtomPairInfo *jatom_pair;
    if(k == j || iatom_pair[k].lcheck != 1) continue;
    ++nk;
    /* Used to have a bug here, cosk was getting set to cosk[nk] before we
       incremented nk. */
    cosk_nk = cosk[nk];
    if(fabs(sink[nk]) < 1.0e-01) continue;
    sink2 = sink[nk]*sink[nk];
    kn = neighbor_start[k];
    ck = iatom_pair[k].cor;
    crkx = ck[1]*cj[2]-cj[1]*ck[2];
    crky = ck[2]*cj[0]-cj[2]*ck[0];
    crkz = ck[0]*cj[1]-cj[0]*ck[1];
    dt2djl[0] = -cj[1]*crkz+cj[2]*crky;
    dt2djl[1] = -cj[2]*crkx+cj[0]*crkz;
    dt2djl[2] = -cj[0]*crky+cj[1]*crkx;

    rck = iatom_pair[k].rcor;

    if(getKtype(kn) == 2) {
      /* It's a hydrogen.  This is computing f super c sub i j from equation 18
       on page 16, but I don't know why we aren't using the table so
       laboriously precomputed for this purpose in mtable.c.  Tim Freeman  5
       Aug 2000. */
      fck = 1.0;
      dfck = 0.0;
      if(rck >= 1.60) continue;
      if(rck >= 1.30) {
        Float dtemp= PIDT*(rck-1.30);
        fck = (1.0+cos(dtemp))/2.0;
        dfck = -PIDT/2.0*sin(dtemp);
      }
    } else {
      fck = iatom_pair[k].ww;
      dfck = iatom_pair[k].dww;
    }
    //jneighbor_start = neighbors (jn, cans);
    //stop_index2 = numNeighbors (jn, cans);
    vector<int> jneighbor_start(nmaxnb);
    int sizemax = nmaxnb;
    const int stop_index2 =
      nblist->GetFullNeighbors(jn, &jneighbor_start[0], &distances[0],
			       &sq_distances[0], sizemax);
    jatom_pair = pairsconst (jn, apis);
    nl = 0;
    for(l = 0; l < stop_index2; ++l) {
      int ln = jneighbor_start[l];
      Float sinl2, rcl, t1, t2, cw, bt, fcl, dfcl;
      Float aa, aaa1, at2, rp1, rp2, rp3, rp4, rp5, rep;
      Float dt1dik, dt1djk, dt1djl, dt1dil, dt1dij;
      Float crlx, crly, crlz;
      bren_vector dt2dik, dt2dij;
      if(ln == i || jatom_pair[l].lcheck != 1) continue;
      ++nl;
      if(fabs(sinl[nl]) < 1.0e-01) continue;
      sinl2 = sinl[nl]*sinl[nl];
      cl = jatom_pair[l].cor;
      rcl = jatom_pair[l].rcor;

      if(getKtype(ln) == 2) {
        fcl = 1.0;
        dfcl = 0.0;
        /* FIXME Another trig computation that is probably in some lookup
           table somewhere.  Tim Freeman 11 Sep 2000 */
        if(rcl >= 1.60) continue;
        if(rcl >= 1.30) {
          Float dtemp = PIDT*(rcl-1.30);
          fcl = (1.0+cos(dtemp))/2.0;
          dfcl = -PIDT/2.0*sin(dtemp);
        }
      } else {
        fcl = jatom_pair[l].ww;
        dfcl = jatom_pair[l].dww;
      }
      t1 = rck*rcl*sij*sij*sink[nk]*sinl[nl];
      
      dt1dik = 1.0/rck/rck - dctik[nk]/sink2*cosk_nk;
      
      dt1djk = -dctjk[nk]/sink2*cosk_nk;
      
      dt1djl = 1.0/rcl/rcl-dctjl[nl]/sinl2*cosl[nl];
      
      dt1dil = -dctil[nl]/sinl2*cosl[nl];

      dt1dij = 2.0/sij/sij-dctij[nk]/sink2*cosk_nk-dctji[nl]/sinl2*cosl[nl];

      crlx = cj[1]*cl[2]-cl[1]*cj[2];
      crly = cj[2]*cl[0]-cl[2]*cj[0];
      crlz = cj[0]*cl[1]-cl[0]*cj[1];

      t2 = crkx*crlx+crky*crly+crkz*crlz;

      cw = t2/t1;
      bt = (1.0-cw*cw);
      btor += bt*fck*fcl;

      dt2dik[0] = -cj[2]*crly+cj[1]*crlz;
      dt2dik[1] = -cj[0]*crlz+cj[2]*crlx;
      dt2dik[2] = -cj[1]*crlx+cj[0]*crly;

      dt2dij[0] = ck[2]*crly-cl[2]*crky-ck[1]*crlz+cl[1]*crkz;
      dt2dij[1] = ck[0]*crlz-cl[0]*crkz-ck[2]*crlx+cl[2]*crkx;
      dt2dij[2] = ck[1]*crlx-cl[1]*crkx-ck[0]*crly+cl[0]*crky;

      aaa1 = vatt_ator*bt;
      aa = -vatt_ator*2.0*cw/t1*fcl*fck;
      at2 = aa*t2;

      rp1 = -dt1dij*at2;
      rp2 = -dt1dik*at2+aaa1*fcl*dfck/rck;
      rp3 = -dt1djl*at2+aaa1*fck*dfcl/rcl;
      rp4 = -dt1djk*at2;
      rp5 = -dt1dil*at2;

      transForce(i, jn, rp1*cj + aa*dt2dij);
      transForce(i, kn, rp2*ck + aa*dt2dik);
      transForce(jn, ln, rp3*cl + aa*dt2djl);
      transForce(jn, kn, rp4*xk_nk);
      transForce(i, ln, rp5*xl[nl]);
    }
  }

  *btot += btor*ator;
  *dradi += datori*btor;
  *dradj += datorj*btor;
  *drdc += datorc*btor;
}
Example #4
0
//static inline void
//calc1side(const BrennerMainInfo *info,
void BrennerPotential::calc1side(
          /* k_this is the ktype for atom formula-i.  This could have come from
             atm_num.  I put an assert in below to verify this. */
          const int k_this,
          /* kother is the ktype for atom formula-j. */
          const int kother,
          /* index1 is an index into atm_num, the index of atom formula-i. */
          const int index1,
          /* We call this routine twice.  In the first call, index2 is the
             variable j, which is an index into atom_pair.  In the second call,
             index2 is the variable i, which is an index into atm_num.  index2
             is only used in one place in the code below, to compute a boolean
             value.  It's the atom we want to skip to ensure that (in the
             notation in the paper) k is not j, that is, we don't do physics on
             the angle between a bond and itself. */
          const int index2,
          /* j is the index into atom_pair of the bond we're looking at.
             c_sign tells us which end of that bond we're looking at. */
          const int j,
          /* c_sign is -1 when we call calc1side the first time, 1 when we call
             it the second time.  It tells us the orientation of edge cj, and
             (due to what looks like a mis-fixed bug) the meaning of the index2
             parameter. */
          const int c_sign,
          /* xni points to an array of 2 Float's, either xni or xnj */
          Float *xni,
          /* atom_pairj is the neighbor pairs that j is an index into. */
          const struct AtomPairInfo *const atom_pairj,
          /* atom_pairk is the neighbor pairs for atom index1. */
          const struct AtomPairInfo *const atom_pairk,
          /* cj is the bren_vector between formula-i and formula-j.  It's always a
             copy of atom_pairj[j].cor.  Whether the bren_vector goes from formula-i
             to formula-j or the other way depends on c_sign. */
          const bren_vector cj,
          Float *xsij, Float *xhc[2], Float *cfuni, Float *dcfuni,
          Float* conk, Float* dctjk, Float* dctij, Float* dctik,
          /* ssumk is the middle term under the square root in equation 7 on
             page 10. */
          Float* ssumk, Float* sdalik, Float* xsjk,
          /* xsik is a per-neighbor output. */
          Float* xsik,
          /* sij is the distance between atoms i and j, always equal to
             atom_pairj[j].rcor. */
          const Float sij,
          /* rsqij is the square of sij. */
          const Float rsqij,
          bren_vector *xk /* xl */, Float *cosk, Float *sink,
          /* exnij is an output set to P sub i j, which is reference in
             equation 7 on page 10 and defined in the paper only as a bicubic
             spline. */ 
          Float *exnij,
          /* dexni is a pointer to two Float's.  One is the partial of exnij
             with respect to N super C sub i, and the other is the partial of
             exnij with respect to N super H sub i. */
          Float *dexni)
{
  int k, jj;
  Float qi;
  /* nk will be the number of the neighbor by the time we're done. */
  int nk = 0; /* nl */
  //  XXX The following neighbor list access should be optimized away
  // Allocate buffers for neighbor list data
  int nmaxnb = nblist->MaxNeighborListLength();
  vector<int> iNeighbors(nmaxnb);
  vector<Vec> distances(nmaxnb);
  vector<double> sq_distances(nmaxnb);
  /* stop_index is one past the index of the last neighbor. */
  int sizemax = nmaxnb;
  const int stop_index =
    nblist->GetFullNeighbors(index1, &iNeighbors[0], &distances[0],
			     &sq_distances[0], sizemax);
  Float gangle, dgdthet;
  Float gangle1, dgdthet1;
  assert (getKtype (index1) == k_this);
  assert (sij == atom_pairj[j].rcor);
  /* Apparently we don't get exact equality when we should on the next line. */
  assert (fabs (rsqij - sij * sij) < 0.00001);
  *xsij = 0.0; /* xsji */
  *ssumk = 0.0; /* ssuml */
  *conk = 0.0; /* conl */
  xni[0] = xhc[0][index1]; /* xnj */
  xni[1] = xhc[1][index1];
  /* The next assert says that other is either a hydrogen or carbon. */
  assert(kother-1 < 2);
  xni[kother-1] -= atom_pairj[j].ww;
  /* qi is N super t sub i, in equation 11 on page 13, . */
  qi = xni[0] + xni[1] - 2.0; /* qj */
  *sdalik = 0.0; /* SDALJL */
  for(k = 0; k < stop_index; ++k) /* for(l) */
  {
    /* k is the index of a bond between formula-i and formula-k.  c_sign
       determines which end of the bond that is formula-i; the other end is
       formula-k. */
    Float ali = 0.0; /* alj */
    Float dali = 0.0;  /* dalj */
    Float daldik = 0.0; /* DALDJL */
    Float s3, rr, ss, rsq3, rsq2;
    /* costh will be the cosine of the angle between ij and ik. */
    Float costh;
    int kn, kk;
    kn = iNeighbors[k]; /* ln */
    if((c_sign > 0 ? kn : k) == index2 || atom_pairk[k].lcheck != 1) continue;
    kk = getKtype (kn); /* kl */
    ++nk; /* nl */
    /* s3 is the length of the formula-i to formula-k bond. */
    s3 = atom_pairk[k].rcor;
    rsq3 = s3*s3;
    /* xk is the bren_vector from formula-j to formula-k, or perhaps the other way
       around. */
    xk[nk] = atom_pairk[k].cor + c_sign*cj;
    /* rsq2 is the square of the distance from formula-j to formula-k. */
    rsq2 = xk[nk][0]*xk[nk][0] + xk[nk][1]*xk[nk][1] + xk[nk][2]*xk[nk][2];
    ss = 2.0*sij*s3;
    rr = rsqij-rsq3;
    /* If you do the analytic geometry, the next line does indeed set costh to
       the cosine of the formula-j formula-i formula-k angle. */
    costh = (rsqij+rsq3-rsq2)/ss;
    if(costh > 1.0) costh = 1.0;
    else if(costh < -1.0) costh = -1.0;
    cosk[nk] = costh;
    sink[nk] = sqrt(1.0-costh*costh); /* sinl */
    /* According to the online docs, acos never returns a value greater
     than pi.  What was the intent here?
     Sent email to Brenner 29 Jul 2000.  Tim Freeman.
     if(acos(costh) > M_PI) sink[nk] = -sink[nk]; */
    assert (acos(costh) <= M_PI);
    if(k_this == 1) {
      /* Atom number index is a carbon. */
      int ig = igc[(int)floor(-costh*12.0)+13];
      /* gangle is G(cos(theta sub j i k)), mentioned in equation 7 on page
         10.  I hope the compiler discovers the shared subexpressions. */
      gangle = SPGC[ig][1] + SPGC[ig][2]*costh + SPGC[ig][3]*costh*costh
        + SPGC[ig][4]*costh*costh*costh
        + SPGC[ig][5]*costh*costh*costh*costh
        + SPGC[ig][6]*costh*costh*costh*costh*costh;
      /* dgdthet is the derivative of gangle with respect to cosine theta. */
      dgdthet = SPGC[ig][2] + SPGC[ig][3]*2*costh
        + SPGC[ig][4]*3*costh*costh
        + SPGC[ig][5]*4*costh*costh*costh
        + SPGC[ig][6]*5*costh*costh*costh*costh;
      if(ig == 4) {
        /* Now we're revising g sub C, per instructions in the middle of page
           13. */
        int ig1;
        /* ali will be Q sub i (N super t sub i) as defined in equation 10 on
           page 13. */
        ali = 0.0;
        /* dali will be the partial of ali with respect to N super t sub i. */
        dali = 0.0;
        if(qi < xqm) {
          /* xqm is 3.7, so if qi >= xqm, we should leave ali and dali at 0. */
          ali = 1.0;
          if(qi > att) {
            Float dtemp = pq*(qi-att);
            ali = (1.0+cos(dtemp))/2.0;
            dali= -pq/2.0*sin(dtemp);
          }
        }
        ig1 = ig+1;
        /* ig was 4, so ig1 is 5.  Presumably this is the row of SPGC that
           defines gamma sub C. */
        gangle1 = SPGC[ig1][1] + SPGC[ig1][2]*costh + SPGC[ig1][3]*costh*costh
          + SPGC[ig1][4]*costh*costh*costh
          + SPGC[ig1][5]*costh*costh*costh*costh
          + SPGC[ig1][6]*costh*costh*costh*costh*costh;
        /* daldik is the partial of g sub C per change in coordination (that
           is, per N super T sub i in the paper, or qi above). */
        daldik = dali*(gangle1-gangle);
        /* Next line is equation 9 on page 13. */
        gangle += ali*(gangle1-gangle);
        /* dgdthet1 is the partial of gamma sub C (cos (theta)) with respect to
           cos (theta). */
        dgdthet1 = SPGC[ig1][2] + SPGC[ig1][3]*2*costh
          + SPGC[ig1][4]*3*costh*costh
          + SPGC[ig1][5]*4*costh*costh*costh
          + SPGC[ig1][6]*5*costh*costh*costh*costh;
        dgdthet += ali*(dgdthet1-dgdthet);
      }
    } else {
      /* k_this != 1 */
      /* Formula-i is not a carbon.  We only get here for hydrogens and
         carbons, so it must be a hydrogen. */
      int ig = igh[(int)floor(-costh*12.0)+13];
      gangle = SPGH[ig][1] + SPGH[ig][2]*costh + SPGH[ig][3]*costh*costh
        + SPGH[ig][4]*costh*costh*costh
        + SPGH[ig][5]*costh*costh*costh*costh
        + SPGH[ig][6]*costh*costh*costh*costh*costh;
      dgdthet = SPGH[ig][2] + SPGH[ig][3]*2*costh
        + SPGH[ig][4]*3*costh*costh
        + SPGH[ig][5]*4*costh*costh*costh
        + SPGH[ig][6]*5*costh*costh*costh*costh;
    }
    {
      Float xtemp, gs, exx, gfx;
      Float dctdjk, dctdij, dctdik;
      /* fc is the cutoff function for the distance between formula-i and
         formula-k. */
      Float fc = atom_pairk[k].ww;
      /* dfc is the derivative of the cutoff with respect to changes in the
         distance between the pair. */
      Float dfc = atom_pairk[k].dww;
      cfuni[nk] = 0.0; /* cfunj */
      dcfuni[nk] = 0.0; /* dcfunj */
      if(kk == 1) { /* kl */
        /* The neighbor of k is a carbon. */
        /* This if statement puts one term the sum in equation 13 on page 14
           into cfuni[nk], and the derivative with respect to the distance
           between the pair into dcfuni[nk]. */
        /* xx is x sub i k, used in equation 13 on page 14 and defined in
           equation 15 on page 14. */
        Float xx = xhc[0][kn]+xhc[1][kn]-fc-2.0;
        if(xx < 3.0) {
          if(xx <= 2.0)
            cfuni[nk] = 1.0;
          else {
            Float px = M_PI*(xx-2.0);
            cfuni[nk] = (1.0+cos(px))/2.0;
            dcfuni[nk] = -fc*sin(px)*M_PI/2.0;
          }
        }
      }
      /* *conk will be the middle sum on the right side of equation 13 on page
         14.  It looks like it would have been slightly better to move this
         inside the if (kk == 1) {...}.*/
      *conk += fc*cfuni[nk];

      /* exx is the exponential term in the middle sum under the square root in
         equation 7 on page 10. */
      if(XDB[k_this][kother][kk] != 0.0)
        /* sij is the distance between atoms formula-i and formula-j.  */
        /* The value inside the exp must be lambda sub i j k, although this is
           undefined in the document and it looks to be a constant in the
           document (and it isn't here). */
        exx = REG[k_this][kother][kk] * exp(XDB[k_this][kother][kk]*(sij-s3));
      else
        exx = 1.0;
      
      /* dctdjk is apparently partial of cosine theta per change in distance
         between formula-j and formula-k, divided by the distance between
         formula-j and formula-k. */
      dctdjk = -2.0/ss; /* DCTDIL */
      /* dctdij is apparently the partial of cosine theta sub j i k per change
         in distance between formula-i and formula-j, divided by the distance
         between formula-i and formula-j. */
      dctdij = (rr+rsq2)/(ss*rsqij); /* DCTDJI */
      /* dctdik is analogous to dctdij; it is the partial of cosine theta sub j
         i k per change in distance between formula-i and formula-k, divided by
         the distance between formula-i and formula-k. */
      dctdik = (-rr+rsq2)/(ss*rsq3); /* DCTDJL */
      dctjk[nk] = dctdjk; /* DCTIL */
      dctij[nk] = dctdij; /* DCTJI */
      dctik[nk] = dctdik; /* DCTJL */
      /* gangle is the G sub i (cos (theta sub j i k)) in the middle term under
         the square root in equation 7 on page 10, which is defined in equation
         9 on page 13. */
      /* exx is the exponential term in the middle sum under the square root in
         equation 7 on page 10. */
      gs = gangle*exx;
      /* ssumk is the middle term under the square root in equation 7 on page
         10. */
      *ssumk += fc*gs;
      xtemp = fc*exx*dgdthet;
      gfx = gs*fc*XDB[k_this][kother][kk];
      *xsij += xtemp*dctdij+gfx/sij;
      xsik[nk] = (gs*dfc-gfx)/s3+xtemp*dctdik; /* XSJL */
      *sdalik += exx*fc*daldik; /* SDALJL */
      xsjk[nk] = xtemp*dctdjk; /* XSIL */
    }
  }
  
  /* check side depends */
  *exnij = 0.0; /* EXNJI */
  dexni[0] = 0.0; /* DEXNJ */
  dexni[1] = 0.0;
  if(k_this == 1) {
    /* The code immediately below is trying to test whether xni[1] and xni[0]
     are approximately integers, so we can avoid doing a spline in that
     case.  The constants 0.5 in the next two lines of code used to be
     1.0e-12, which doesn't work very well when we're doing single
     precision.  I think 0.5 is right anyway, since in that case floor
     returns the nearest integer to xni[whatever].  I don't know why it's
     important to avoid doing a spline when they're integers.  Tim
     Freeman 29 Jul 2000. */
#define ALWAYS_SPLINE
#ifndef ALWAYS_SPLINE
    int nh = (int)floor(xni[1]+0.5);
    int nc = (int)floor(xni[0]+0.5);
    if((fabs((Float)nh-xni[1]) > 1.0e-08)
       || (fabs((Float)nc-xni[0]) > 1.0e-08)) {
#endif
      /* If xni[1] and xni[0] aren't integers, then do BCUINT, which is a
         bicubic spline. */
      *exnij = BCUINT(kother, xni[1], xni[0], &dexni[1], &dexni[0]);
#ifndef ALWAYS_SPLINE
    } else {
      /* xni[1] and xni[0] are approximately integers, so we don't need to
         spline. */
      *exnij = xh[kother][nh][nc];
      dexni[1] = xh1[kother][nh][nc];
      dexni[0] = xh2[kother][nh][nc];
    }
#endif
  }
}