aboutsummaryrefslogtreecommitdiff
path: root/libcrystfel/src/cell-utils.c
diff options
context:
space:
mode:
Diffstat (limited to 'libcrystfel/src/cell-utils.c')
-rw-r--r--libcrystfel/src/cell-utils.c340
1 files changed, 277 insertions, 63 deletions
diff --git a/libcrystfel/src/cell-utils.c b/libcrystfel/src/cell-utils.c
index ca697391..508b70e5 100644
--- a/libcrystfel/src/cell-utils.c
+++ b/libcrystfel/src/cell-utils.c
@@ -39,6 +39,7 @@
#include <gsl/gsl_matrix.h>
#include <gsl/gsl_blas.h>
#include <gsl/gsl_linalg.h>
+#include <assert.h>
#include "cell.h"
#include "cell-utils.h"
@@ -99,6 +100,56 @@ static const char *str_lattice(LatticeType l)
}
+int right_handed(UnitCell *cell)
+{
+ double asx, asy, asz;
+ double bsx, bsy, bsz;
+ double csx, csy, csz;
+ struct rvec aCb;
+ double aCb_dot_c;
+ int rh_reciprocal;
+ int rh_direct;
+
+ if ( cell_get_reciprocal(cell, &asx, &asy, &asz,
+ &bsx, &bsy, &bsz,
+ &csx, &csy, &csz) ) {
+ ERROR("Couldn't get reciprocal cell.\n");
+ return 0;
+ }
+
+ /* "a" cross "b" */
+ aCb.u = asy*bsz - asz*bsy;
+ aCb.v = - (asx*bsz - asz*bsx);
+ aCb.w = asx*bsy - asy*bsx;
+
+ /* "a cross b" dot "c" */
+ aCb_dot_c = aCb.u*csx + aCb.v*csy + aCb.w*csz;
+
+ rh_reciprocal = aCb_dot_c > 0.0;
+
+ if ( cell_get_cartesian(cell, &asx, &asy, &asz,
+ &bsx, &bsy, &bsz,
+ &csx, &csy, &csz) ) {
+ ERROR("Couldn't get direct cell.\n");
+ return 0;
+ }
+
+ /* "a" cross "b" */
+ aCb.u = asy*bsz - asz*bsy;
+ aCb.v = - (asx*bsz - asz*bsx);
+ aCb.w = asx*bsy - asy*bsx;
+
+ /* "a cross b" dot "c" */
+ aCb_dot_c = aCb.u*csx + aCb.v*csy + aCb.w*csz;
+
+ rh_direct = aCb_dot_c > 0.0;
+
+ assert(rh_reciprocal == rh_direct);
+
+ return rh_direct;
+}
+
+
void cell_print(UnitCell *cell)
{
double asx, asy, asz;
@@ -106,9 +157,31 @@ void cell_print(UnitCell *cell)
double csx, csy, csz;
double a, b, c, alpha, beta, gamma;
double ax, ay, az, bx, by, bz, cx, cy, cz;
+ LatticeType lt;
+
+ lt = cell_get_lattice_type(cell);
- STATUS("%s %c\n", str_lattice(cell_get_lattice_type(cell)),
- cell_get_centering(cell));
+ STATUS("%s %c", str_lattice(lt), cell_get_centering(cell));
+
+ switch ( lt )
+ {
+ case L_MONOCLINIC :
+ case L_TETRAGONAL :
+ case L_HEXAGONAL :
+ STATUS(", unique axis %c", cell_get_unique_axis(cell));
+ break;
+
+ default :
+ break;
+ }
+
+ if ( right_handed(cell) ) {
+ STATUS(", right handed");
+ } else {
+ STATUS(", left handed");
+ }
+
+ STATUS(", point group '%s'.\n", cell_get_pointgroup(cell));
cell_get_parameters(cell, &a, &b, &c, &alpha, &beta, &gamma);
@@ -134,14 +207,200 @@ void cell_print(UnitCell *cell)
STATUS("cstar = %10.3e %10.3e %10.3e m^-1 (modulus = %10.3e m^-1)\n",
csx, csy, csz, modulus(csx, csy, csz));
- STATUS("Point group: %s\n", cell_get_pointgroup(cell));
STATUS("Cell representation is %s.\n", cell_rep(cell));
}
+int bravais_lattice(UnitCell *cell)
+{
+ LatticeType lattice = cell_get_lattice_type(cell);
+ char centering = cell_get_centering(cell);
+ char ua = cell_get_unique_axis(cell);
+
+ switch ( centering )
+ {
+ case 'P' :
+ return 1;
+
+ case 'A' :
+ case 'B' :
+ case 'C' :
+ if ( lattice == L_MONOCLINIC ) {
+ if ( (ua=='a') && (centering=='A') ) return 1;
+ if ( (ua=='b') && (centering=='B') ) return 1;
+ if ( (ua=='c') && (centering=='C') ) return 1;
+ } else if ( lattice == L_ORTHORHOMBIC) {
+ return 1;
+ }
+ return 0;
+
+ case 'I' :
+ if ( (lattice == L_ORTHORHOMBIC)
+ || (lattice == L_TETRAGONAL)
+ || (lattice == L_CUBIC) )
+ {
+ return 1;
+ }
+ return 0;
+
+ case 'F' :
+ if ( (lattice == L_ORTHORHOMBIC) || (lattice == L_CUBIC) ) {
+ return 1;
+ }
+ return 0;
+
+ case 'H' :
+ if ( lattice == L_HEXAGONAL ) return 1;
+ return 0;
+
+ default :
+ return 0;
+ }
+}
+
+
+/* Turn any cell into a primitive one, e.g. for comparison purposes */
+UnitCell *uncenter_cell(UnitCell *in)
+{
+ UnitCell *cell;
+
+ double ax, ay, az;
+ double bx, by, bz;
+ double cx, cy, cz;
+ double nax, nay, naz;
+ double nbx, nby, nbz;
+ double ncx, ncy, ncz;
+ const double OT = 1.0/3.0;
+ const double TT = 2.0/3.0;
+ const double H = 0.5;
+ LatticeType lt;
+ char ua, cen;
+
+ if ( !bravais_lattice(in) ) {
+ ERROR("Cannot uncenter: not a Bravais lattice.\n");
+ return NULL;
+ }
+
+ cell_get_cartesian(in, &ax, &ay, &az, &bx, &by, &bz, &cx, &cy, &cz);
+ lt = cell_get_lattice_type(in);
+ ua = cell_get_unique_axis(in);
+ cen = cell_get_centering(in);
+
+ if ( ua == 'a' ) {
+ double tx, ty, tz;
+ tx = ax; ty = ay; tz = az;
+ ax = cx; ay = cy; az = cz;
+ cx = tx; cy = ty; cz = tz;
+ if ( cen == 'A' ) cen = 'C';
+ }
+
+ if ( ua == 'b' ) {
+ double tx, ty, tz;
+ tx = bx; ty = by; tz = bz;
+ ax = cx; ay = cy; az = cz;
+ cx = tx; cy = ty; cz = tz;
+ if ( cen == 'B' ) cen = 'C';
+ }
+
+ cell = cell_new();
+
+ switch ( cen ) {
+
+ case 'P' :
+ case 'R' :
+ nax = ax; nay = ay; naz = az;
+ nbx = bx; nby = by; nbz = bz;
+ ncx = cx; ncy = cy; ncz = cz; /* Identity */
+ cell_set_lattice_type(cell, lt);
+ cell_set_centering(cell, cen); /* P->P, R->R */
+ break;
+
+ case 'I' :
+ nax = - H*ax + H*bx + H*cx;
+ nay = - H*ay + H*by + H*cy;
+ naz = - H*az + H*bz + H*cz;
+ nbx = H*ax - H*bx + H*cx;
+ nby = H*ay - H*by + H*cy;
+ nbz = H*az - H*bz + H*cz;
+ ncx = H*ax + H*bx - H*cx;
+ ncy = H*ay + H*by - H*cy;
+ ncz = H*az + H*bz - H*cz;
+ if ( lt == L_CUBIC ) {
+ cell_set_lattice_type(cell, L_RHOMBOHEDRAL);
+ cell_set_centering(cell, 'R');
+ } else {
+ /* Tetragonal or orthorhombic */
+ cell_set_lattice_type(cell, L_TRICLINIC);
+ cell_set_centering(cell, 'P');
+ }
+ break;
+
+ case 'F' :
+ nax = 0*ax + H*bx + H*cx;
+ nay = 0*ay + H*by + H*cy;
+ naz = 0*az + H*bz + H*cz;
+ nbx = H*ax + 0*bx + H*cx;
+ nby = H*ay + 0*by + H*cy;
+ nbz = H*az + 0*bz + H*cz;
+ ncx = H*ax + H*bx + 0*cx;
+ ncy = H*ay + H*by + 0*cy;
+ ncz = H*az + H*bz + 0*cz;
+ if ( lt == L_CUBIC ) {
+ cell_set_lattice_type(cell, L_RHOMBOHEDRAL);
+ cell_set_centering(cell, 'R');
+
+ } else {
+ assert(lt == L_ORTHORHOMBIC);
+ cell_set_lattice_type(cell, L_TRICLINIC);
+ cell_set_centering(cell, 'P');
+ }
+ break;
+
+ case 'C' :
+ nax = H*ax + H*bx + 0*cx;
+ nay = H*ay + H*by + 0*cy;
+ naz = H*az + H*bz + 0*cz;
+ nbx = -H*ax + H*bx + 0*cx;
+ nby = -H*ay + H*by + 0*cy;
+ nbz = -H*az + H*bz + 0*cz;
+ ncx = 0*ax + 0*bx + 1*cx;
+ ncy = 0*ay + 0*by + 1*cy;
+ ncz = 0*az + 0*bz + 1*cz;
+ cell_set_lattice_type(cell, L_MONOCLINIC);
+ cell_set_centering(cell, 'P');
+ cell_set_unique_axis(cell, 'c');
+ break;
+
+ case 'H' :
+ nax = TT*ax + OT*bx + OT*cx;
+ nay = TT*ay + OT*by + OT*cy;
+ naz = TT*az + OT*bz + OT*cz;
+ nbx = - OT*ax + OT*bx + OT*cx;
+ nby = - OT*ay + OT*by + OT*cy;
+ nbz = - OT*az + OT*bz + OT*cz;
+ ncx = - OT*ax - TT*bx + OT*cx;
+ ncy = - OT*ay - TT*by + OT*cy;
+ ncz = - OT*az - TT*bz + OT*cz;
+ assert(lt == L_HEXAGONAL);
+ cell_set_lattice_type(cell, L_RHOMBOHEDRAL);
+ cell_set_centering(cell, 'R');
+ break;
+
+ default :
+ ERROR("Invalid centering '%c'\n", cell_get_centering(in));
+ return NULL;
+
+ }
+
+ cell_set_cartesian(cell, nax, nay, naz, nbx, nby, nbz, ncx, ncy, ncz);
+
+ return cell;
+}
+
+
#define MAX_CAND (1024)
-static int right_handed(struct rvec a, struct rvec b, struct rvec c)
+static int right_handed_vec(struct rvec a, struct rvec b, struct rvec c)
{
struct rvec aCb;
double aCb_dot_c;
@@ -178,7 +437,7 @@ static int same_vector(struct cvec a, struct cvec b)
/* Attempt to make 'cell' fit into 'template' somehow */
-UnitCell *match_cell(UnitCell *cell, UnitCell *template, int verbose,
+UnitCell *match_cell(UnitCell *cell_in, UnitCell *template_in, int verbose,
const float *tols, int reduce)
{
signed int n1l, n2l, n3l;
@@ -194,6 +453,11 @@ UnitCell *match_cell(UnitCell *cell, UnitCell *template, int verbose,
int ncand[3] = {0,0,0};
signed int ilow, ihigh;
float angtol = deg2rad(tols[3]);
+ UnitCell *cell;
+ UnitCell *template;
+
+ cell = uncenter_cell(cell_in);
+ template = uncenter_cell(template_in);
if ( cell_get_reciprocal(template, &asx, &asy, &asz,
&bsx, &bsy, &bsz,
@@ -350,8 +614,8 @@ UnitCell *match_cell(UnitCell *cell, UnitCell *template, int verbose,
if ( fabs(ang - angles[0]) > angtol ) continue;
/* Unit cell must be right-handed */
- if ( !right_handed(cand[0][i].vec, cand[1][j].vec,
- cand[2][k].vec) ) continue;
+ if ( !right_handed_vec(cand[0][i].vec, cand[1][j].vec,
+ cand[2][k].vec) ) continue;
fom3 = fom2 + fabs(ang - angles[0]);
fom3 += LWEIGHT * (cand[0][i].fom + cand[1][j].fom
@@ -464,7 +728,7 @@ UnitCell *match_cell_ab(UnitCell *cell, UnitCell *template)
}
/* Flip c if not right-handed */
- if ( !right_handed(real_a, real_b, real_c) ) {
+ if ( !right_handed_vec(real_a, real_b, real_c) ) {
real_c.u = -real_c.u;
real_c.v = -real_c.v;
real_c.w = -real_c.w;
@@ -690,12 +954,6 @@ UnitCell *load_cell_from_pdb(const char *filename)
fclose(fh);
- /* FIXME: Turn "H" centered cells into "R" cells */
- if ( cell_get_centering(cell) == 'H' ) {
- STATUS("Turning your H-centered (PDB convention) cell into"
- " an R-centered one.\n");
- }
-
validate_cell(cell);
return cell;
@@ -789,55 +1047,6 @@ int cell_is_sensible(UnitCell *cell)
}
-static int bravais_lattice(UnitCell *cell)
-{
- LatticeType lattice = cell_get_lattice_type(cell);
- char centering = cell_get_centering(cell);
- char ua = cell_get_unique_axis(cell);
-
- switch ( centering )
- {
- case 'P' :
- return 1;
-
- case 'A' :
- case 'B' :
- case 'C' :
- if ( (lattice != L_MONOCLINIC)
- && (lattice != L_ORTHORHOMBIC) )
- {
- return 0;
- }
- if ( (ua=='a') && (centering=='A') ) return 1;
- if ( (ua=='b') && (centering=='B') ) return 1;
- if ( (ua=='c') && (centering=='C') ) return 1;
- return 0;
-
- case 'I' :
- if ( (lattice == L_ORTHORHOMBIC)
- || (lattice == L_TETRAGONAL)
- || (lattice == L_CUBIC) )
- {
- return 1;
- }
- return 0;
-
- case 'F' :
- if ( (lattice == L_ORTHORHOMBIC) || (lattice == L_CUBIC) ) {
- return 1;
- }
- return 0;
-
- case 'H' :
- if ( lattice == L_HEXAGONAL ) return 1;
- return 0;
-
- default :
- return 0;
- }
-}
-
-
/**
* validate_cell:
* @cell: A %UnitCell to validate
@@ -862,5 +1071,10 @@ void validate_cell(UnitCell *cell)
err = 1;
}
+ if ( !right_handed(cell) ) {
+ ERROR("Warning: Unit cell is not right handed.\n");
+ err = 1;
+ }
+
if ( err ) cell_print(cell);
}