/* linked list / cell method */
+#ifdef STATIC_LISTS
+
+int link_cell_init(t_moldyn *moldyn,u8 vol) {
+
+ t_linkcell *lc;
+ int i;
+ int *foo;
+
+ lc=&(moldyn->lc);
+
+ /* partitioning the md cell */
+ lc->nx=moldyn->dim.x/moldyn->cutoff;
+ lc->x=moldyn->dim.x/lc->nx;
+ lc->ny=moldyn->dim.y/moldyn->cutoff;
+ lc->y=moldyn->dim.y/lc->ny;
+ lc->nz=moldyn->dim.z/moldyn->cutoff;
+ lc->z=moldyn->dim.z/lc->nz;
+
+ lc->cells=lc->nx*lc->ny*lc->nz;
+ lc->subcell=malloc(lc->cells*sizeof(int*));
+
+ if(lc->cells<27)
+ printf("[moldyn] FATAL: less then 27 subcells!\n");
+
+ if(vol) {
+ printf("[moldyn] initializing 'static' linked cells (%d)\n",
+ lc->cells);
+ printf(" x: %d x %f A\n",lc->nx,lc->x);
+ printf(" y: %d x %f A\n",lc->ny,lc->y);
+ printf(" z: %d x %f A\n",lc->nz,lc->z);
+ }
+
+ /* list init */
+ for(i=0;i<lc->cells;i++) {
+ lc->subcell[i]=malloc((MAX_ATOMS_PER_LIST+1)*sizeof(int));
+ if(lc->subcell[i]==NULL) {
+ perror("[moldyn] list init (malloc)");
+ return -1;
+ }
+//if(i==0) printf(" --- add one here! %d %p %p ----\n",i,lc->subcell,lc->subcell[0]);
+ }
+
+ /* update the list */
+ link_cell_update(moldyn);
+
+ return 0;
+}
+
+int link_cell_update(t_moldyn *moldyn) {
+
+ int count,i,j,k;
+ int nx,ny;
+ t_atom *atom;
+ t_linkcell *lc;
+ int p;
+
+ atom=moldyn->atom;
+ lc=&(moldyn->lc);
+
+ nx=lc->nx;
+ ny=lc->ny;
+
+ for(i=0;i<lc->cells;i++)
+ memset(lc->subcell[i],0,(MAX_ATOMS_PER_LIST+1)*sizeof(int));
+
+ for(count=0;count<moldyn->count;count++) {
+ i=((atom[count].r.x+(moldyn->dim.x/2))/lc->x);
+ j=((atom[count].r.y+(moldyn->dim.y/2))/lc->y);
+ k=((atom[count].r.z+(moldyn->dim.z/2))/lc->z);
+
+ p=0;
+ while(lc->subcell[i+j*nx+k*nx*ny][p]!=0)
+ p++;
+
+ if(p>=MAX_ATOMS_PER_LIST) {
+ printf("[moldyn] FATAL: amount of atoms too high!\n");
+ return -1;
+ }
+
+ lc->subcell[i+j*nx+k*nx*ny][p]=count;
+ }
+
+ return 0;
+}
+
+int link_cell_neighbour_index(t_moldyn *moldyn,int i,int j,int k,int **cell) {
+
+ t_linkcell *lc;
+ int a;
+ int count1,count2;
+ int ci,cj,ck;
+ int nx,ny,nz;
+ int x,y,z;
+ u8 bx,by,bz;
+
+ lc=&(moldyn->lc);
+ nx=lc->nx;
+ ny=lc->ny;
+ nz=lc->nz;
+ count1=1;
+ count2=27;
+ a=nx*ny;
+
+ cell[0]=lc->subcell[i+j*nx+k*a];
+ for(ci=-1;ci<=1;ci++) {
+ bx=0;
+ x=i+ci;
+ if((x<0)||(x>=nx)) {
+ x=(x+nx)%nx;
+ bx=1;
+ }
+ for(cj=-1;cj<=1;cj++) {
+ by=0;
+ y=j+cj;
+ if((y<0)||(y>=ny)) {
+ y=(y+ny)%ny;
+ by=1;
+ }
+ for(ck=-1;ck<=1;ck++) {
+ bz=0;
+ z=k+ck;
+ if((z<0)||(z>=nz)) {
+ z=(z+nz)%nz;
+ bz=1;
+ }
+ if(!(ci|cj|ck)) continue;
+ if(bx|by|bz) {
+ cell[--count2]=lc->subcell[x+y*nx+z*a];
+ }
+ else {
+ cell[count1++]=lc->subcell[x+y*nx+z*a];
+ }
+ }
+ }
+ }
+
+ lc->dnlc=count1;
+
+ return count1;
+}
+
+int link_cell_shutdown(t_moldyn *moldyn) {
+
+ int i;
+ t_linkcell *lc;
+ int *foo;
+
+ lc=&(moldyn->lc);
+
+ for(i=0;i<lc->cells;i++)
+{
+//printf(" --- free %p , %d\n",lc->subcell[i],i);
+ free(lc->subcell[i]);
+}
+
+ free(lc->subcell);
+
+ return 0;
+}
+
+#else
+
int link_cell_init(t_moldyn *moldyn,u8 vol) {
t_linkcell *lc;
printf("[moldyn] FATAL: less then 27 subcells!\n");
if(vol) {
- printf("[moldyn] initializing linked cells (%d)\n",lc->cells);
+ printf("[moldyn] initializing 'dynamic' linked cells (%d)\n",
+ lc->cells);
printf(" x: %d x %f A\n",lc->nx,lc->x);
printf(" y: %d x %f A\n",lc->ny,lc->y);
printf(" z: %d x %f A\n",lc->nz,lc->z);
k=((atom[count].r.z+(moldyn->dim.z/2))/lc->z);
list_add_immediate_f(&(lc->subcell[i+j*nx+k*nx*ny]),
&(atom[count]));
+//if(i==0&&j==0&&k==0) printf(" --- add one here! %d %p ----\n",count,lc->subcell[0].current);
}
return 0;
lc=&(moldyn->lc);
- for(i=0;i<lc->nx*lc->ny*lc->nz;i++)
+printf("FOO:\n");
+ for(i=0;i<lc->nx*lc->ny*lc->nz;i++) {
+printf(" %d\n",i);
list_destroy_f(&(moldyn->lc.subcell[i]));
+printf(" %d!\n",i);
+}
free(lc->subcell);
return 0;
}
+#endif
+
int moldyn_add_schedule(t_moldyn *moldyn,int runs,double tau) {
int count;
/* calculate initial forces */
potential_force_calc(moldyn);
#ifdef DEBUG
-return 0;
+//return 0;
#endif
/* some stupid checks before we actually start calculating bullshit */
}
/* display progress */
- if(!(moldyn->total_steps%10)) {
+ //if(!(moldyn->total_steps%10)) {
/* get current time */
gettimeofday(&t2,NULL);
/* copy over time */
t1=t2;
- }
+ //}
/* increase absolute time */
moldyn->time+=moldyn->tau;
t_atom *itom,*jtom,*ktom;
t_virial *virial;
t_linkcell *lc;
+#ifdef STATIC_LISTS
+ int *neighbour_i[27];
+ int p,q;
+ t_atom *atom;
+#else
t_list neighbour_i[27];
t_list neighbour_i2[27];
t_list *this,*that;
+#endif
u8 bc_ij,bc_ik;
int dnlc;
count=moldyn->count;
itom=moldyn->atom;
lc=&(moldyn->lc);
+#ifdef STATIC_LISTS
+ atom=moldyn->atom;
+#endif
/* reset energy */
moldyn->energy=0.0;
if(moldyn->func2b) {
for(j=0;j<27;j++) {
+ bc_ij=(j<dnlc)?0:1;
+#ifdef STATIC_LISTS
+ p=0;
+
+ while(neighbour_i[j][p]!=0) {
+
+ jtom=&(atom[neighbour_i[j][p]]);
+ p++;
+
+ if(jtom==&(itom[i]))
+ continue;
+
+ if((jtom->attr&ATOM_ATTR_2BP)&
+ (itom[i].attr&ATOM_ATTR_2BP)) {
+ moldyn->func2b(moldyn,
+ &(itom[i]),
+ jtom,
+ bc_ij);
+ }
+ }
+#else
this=&(neighbour_i[j]);
list_reset_f(this);
if(this->start==NULL)
continue;
- bc_ij=(j<dnlc)?0:1;
-
do {
jtom=this->current->data;
bc_ij);
}
} while(list_next_f(this)!=L_NO_NEXT_ELEMENT);
+#endif
}
}
continue;
/* copy the neighbour lists */
+#ifdef STATIC_LISTS
+ /* no copy needed for static lists */
+#else
memcpy(neighbour_i2,neighbour_i,27*sizeof(t_list));
+#endif
/* second loop over atoms j */
for(j=0;j<27;j++) {
+ bc_ij=(j<dnlc)?0:1;
+#ifdef STATIC_LISTS
+ p=0;
+
+ while(neighbour_i[j][p]!=0) {
+
+ jtom=&(atom[neighbour_i[j][p]]);
+ p++;
+#else
this=&(neighbour_i[j]);
list_reset_f(this);
if(this->start==NULL)
continue;
- bc_ij=(j<dnlc)?0:1;
-
do {
+
jtom=this->current->data;
+#endif
if(jtom==&(itom[i]))
continue;
for(k=0;k<27;k++) {
+ bc_ik=(k<dnlc)?0:1;
+#ifdef STATIC_LISTS
+ q=0;
+
+ while(neighbour_i[j][q]!=0) {
+
+ ktom=&(atom[neighbour_i[k][q]]);
+ q++;
+#else
that=&(neighbour_i2[k]);
list_reset_f(that);
if(that->start==NULL)
continue;
- bc_ik=(k<dnlc)?0:1;
-
do {
-
ktom=that->current->data;
+#endif
if(!(ktom->attr&ATOM_ATTR_3BP))
continue;
jtom,
ktom,
bc_ik|bc_ij);
-
+#ifdef STATIC_LISTS
+ }
+#else
} while(list_next_f(that)!=\
L_NO_NEXT_ELEMENT);
+#endif
}
for(k=0;k<27;k++) {
+ bc_ik=(k<dnlc)?0:1;
+#ifdef STATIC_LISTS
+ q=0;
+
+ while(neighbour_i[j][q]!=0) {
+
+ ktom=&(atom[neighbour_i[k][q]]);
+ q++;
+#else
that=&(neighbour_i2[k]);
list_reset_f(that);
if(that->start==NULL)
continue;
- bc_ik=(k<dnlc)?0:1;
-
do {
-
ktom=that->current->data;
+#endif
if(!(ktom->attr&ATOM_ATTR_3BP))
continue;
ktom,
bc_ik|bc_ij);
+#ifdef STATIC_LISTS
+ }
+#else
} while(list_next_f(that)!=\
L_NO_NEXT_ELEMENT);
+#endif
}
&(itom[i]),
jtom,bc_ij);
}
-
+#ifdef STATIC_LISTS
+ }
+#else
} while(list_next_f(this)!=L_NO_NEXT_ELEMENT);
+#endif
}
}
#ifdef DEBUG
- printf("\nATOM 0: %f %f %f\n\n",itom->f.x,itom->f.y,itom->f.z);
+ //printf("\nATOM 0: %f %f %f\n\n",itom->f.x,itom->f.y,itom->f.z);
+ if(moldyn->time>DSTART&&moldyn->time<DEND) {
+ printf("force:\n");
+ printf(" x: %0.40f\n",moldyn->atom[5832].f.x);
+ printf(" y: %0.40f\n",moldyn->atom[5832].f.y);
+ printf(" z: %0.40f\n",moldyn->atom[5832].f.z);
+ }
#endif
/* calculate global virial */
* restore function
*/
+int moldyn_read_save_file(t_moldyn *moldyn,char *file) {
+
+ int fd;
+ int cnt,size;
+
+ fd=open(file,O_RDONLY);
+ if(fd<0) {
+ perror("[moldyn] load save file open");
+ return fd;
+ }
+
+ size=sizeof(t_moldyn);
+ cnt=read(fd,moldyn,size);
+ if(cnt!=size) {
+ perror("[moldyn] load save file read (moldyn)");
+ return cnt;
+ }
+
+ size=moldyn->count*sizeof(t_atom);
+
+ moldyn->atom=(t_atom *)malloc(size);
+ if(moldyn->atom==NULL) {
+ perror("[moldyn] load save file malloc (atoms)");
+ return -1;
+ }
+
+ cnt=read(fd,moldyn->atom,size);
+ if(cnt!=size) {
+ perror("[moldyn] load save file read (atoms)");
+ return cnt;
+ }
+
+ // hooks
+
+ return 0;
+}
+
int moldyn_load(t_moldyn *moldyn) {
// later ...
}
}
+int pair_correlation_init(t_moldyn *moldyn,double dr) {
+
+
+ return 0;
+}
+
+int calculate_pair_correlation(t_moldyn *moldyn,double dr,void *ptr) {
+
+ int slots;
+ double *stat;
+ int i,j;
+ t_linkcell *lc;
+#ifdef STATIC_LISTS
+ int *neighbour[27];
+ int p;
+#else
+ t_list neighbour[27];
+#endif
+ t_atom *itom,*jtom;
+ t_list *this;
+ unsigned char bc;
+ t_3dvec dist;
+ double d,norm;
+ int o,s;
+ unsigned char ibrand;
+
+ lc=&(moldyn->lc);
+
+ slots=(int)(moldyn->cutoff/dr);
+ o=2*slots;
+
+ printf("[moldyn] pair correlation calc info:\n");
+ printf(" time: %f\n",moldyn->time);
+ printf(" count: %d\n",moldyn->count);
+ printf(" cutoff: %f\n",moldyn->cutoff);
+ printf(" temperature: cur=%f avg=%f\n",moldyn->t,moldyn->t_avg);
+
+ if(ptr!=NULL) {
+ stat=(double *)ptr;
+ }
+ else {
+ stat=(double *)malloc(3*slots*sizeof(double));
+ if(stat==NULL) {
+ perror("[moldyn] pair correlation malloc");
+ return -1;
+ }
+ }
+
+ memset(stat,0,3*slots*sizeof(double));
+
+ link_cell_init(moldyn,VERBOSE);
+
+ itom=moldyn->atom;
+
+ for(i=0;i<moldyn->count;i++) {
+ /* neighbour indexing */
+ link_cell_neighbour_index(moldyn,
+ (itom[i].r.x+moldyn->dim.x/2)/lc->x,
+ (itom[i].r.y+moldyn->dim.y/2)/lc->x,
+ (itom[i].r.z+moldyn->dim.z/2)/lc->x,
+ neighbour);
+
+ /* brand of atom i */
+ ibrand=itom[i].brand;
+
+ for(j=0;j<27;j++) {
+
+ bc=(j<lc->dnlc)?0:1;
+
+#ifdef STATIC_LISTS
+ p=0;
+
+ while(neighbour[j][p]!=0) {
+
+ jtom=&(moldyn->atom[neighbour[j][p]]);
+ p++;
+#else
+ this=&(neighbour[j]);
+ list_reset_f(this);
+
+ if(this->start==NULL)
+ continue;
+
+ do {
+
+ jtom=this->current->data;
+#endif
+
+ if(jtom==&(itom[i]))
+ continue;
+
+ /* only count pairs once */
+ if(itom[i].tag>jtom->tag)
+ continue;
+
+ /*
+ * pair correlation calc
+ */
+
+ /* distance */
+ v3_sub(&dist,&(jtom->r),&(itom[i].r));
+ if(bc) check_per_bound(moldyn,&dist);
+ d=v3_absolute_square(&dist);
+
+ /* ignore if greater cutoff */
+ if(d>moldyn->cutoff_square)
+ continue;
+
+ /* fill the slots */
+ d=sqrt(d);
+ s=(int)(d/dr);
+
+ if(ibrand!=jtom->brand) {
+ /* mixed */
+ stat[s]+=1;
+ }
+ else {
+ /* type a - type a bonds */
+ if(ibrand==0)
+ stat[s+slots]+=1;
+ else
+ /* type b - type b bonds */
+ stat[s+o]+=1;
+ }
+
+#ifdef STATIC_LISTS
+ }
+#else
+ } while(list_next_f(this)!=L_NO_NEXT_ELEMENT);
+#endif
+ }
+ }
+
+ /* normalization
+ for(i=1;i<slots;i++) {
+ // normalization: 4 pi r r dr
+ // here: not double counting pairs -> 2 pi r r dr
+ norm=2*M_PI*moldyn->count*(i*dr*i*dr)*dr;
+ stat[i]/=norm;
+ stat[slots+i]/=norm;
+ stat[o+i]/=norm;
+ }
+ */
+
+ if(ptr==NULL) {
+ /* todo: store/print pair correlation function */
+ free(stat);
+ }
+
+ free(moldyn->atom);
+
+ link_cell_shutdown(moldyn);
+
+ return 0;
+}
+
int analyze_bonds(t_moldyn *moldyn) {
t_atom *atom;
t_atom *btom;
t_linkcell *lc;
+#ifdef STATIC_LISTS
+ int *neighbour[27];
+ int p;
+#else
t_list neighbour[27];
+#endif
u8 bc;
t_3dvec dist;
double d2;
(atom[i].r.z+moldyn->dim.z/2)/lc->z,
neighbour);
for(j=0;j<27;j++) {
+ bc=j<lc->dnlc?0:1;
+#ifdef STATIC_LISTS
+ p=0;
+ while(neighbour[j][p]!=0) {
+ btom=&(atom[neighbour[j][p]]);
+ p++;
+#else
list_reset_f(&neighbour[j]);
if(neighbour[j].start==NULL)
continue;
- bc=j<lc->dnlc?0:1;
do {
btom=neighbour[j].current->data;
+#endif
if(btom==&atom[i]) // skip identical atoms
continue;
//if(btom<&atom[i]) // skip half of them
dprintf(fd,"# [B] %f %f %f %f %f %f\n",
atom[i].r.x,atom[i].r.y,atom[i].r.z,
btom->r.x,btom->r.y,btom->r.z);
+#ifdef STATIC_LISTS
+ }
+#else
} while(list_next_f(&neighbour[j])!=L_NO_NEXT_ELEMENT);
+#endif
}
}