aboutsummaryrefslogtreecommitdiff
path: root/libcrystfel
diff options
context:
space:
mode:
authorThomas White <taw@physics.org>2019-08-14 14:07:31 +0200
committerThomas White <taw@physics.org>2019-08-22 17:03:27 +0200
commite3f4046056cf92ce5e40e5e715cbcd53df7b0621 (patch)
treeeb4db521a5c1ab85e011852d63d1091f1b01649f /libcrystfel
parent0b16a4aa50bfe233d81077b1661c57b4b0ef0ef2 (diff)
Framework for new unit cell comparison function
Diffstat (limited to 'libcrystfel')
-rw-r--r--libcrystfel/src/cell-utils.c105
-rw-r--r--libcrystfel/src/cell-utils.h6
-rw-r--r--libcrystfel/src/index.c16
3 files changed, 40 insertions, 87 deletions
diff --git a/libcrystfel/src/cell-utils.c b/libcrystfel/src/cell-utils.c
index 141eaaab..20ede189 100644
--- a/libcrystfel/src/cell-utils.c
+++ b/libcrystfel/src/cell-utils.c
@@ -2341,10 +2341,9 @@ IntegerMatrix *reduce_g6(double *g, double eps)
* \param tolerance: Pointer to tolerances for a,b,c (fractional), al,be,ga (radians)
* \param pmb: Place to store pointer to matrix
*
- * Compare \p cell_in with \p reference_in. If they are the same,
- * within \p tolerance, taking into account possible permutations of the axes,
- * this function returns non-zero and stores the transformation which needs to
- * be applied to \p cell_in at \p pmb.
+ * Compare the \p cell_in with \p reference_in. If they represent the same
+ * lattice, this function returns non-zero and stores the transformation which
+ * needs to be applied to \p cell_in at \p pmb.
*
* Subject to the tolerances, this function will find the transformation which
* gives the best match to the reference cell, using the Euclidian norm in
@@ -2356,76 +2355,44 @@ IntegerMatrix *reduce_g6(double *g, double eps)
* \returns non-zero if the cells match, zero for no match or error.
*
*/
-int compare_permuted_cell_parameters(UnitCell *cell, UnitCell *reference,
- double *tolerance, IntegerMatrix **pmb)
+int compare_lattices(UnitCell *cell_in, UnitCell *reference_in,
+ double *tolerance, RationalMatrix **pmb)
{
- IntegerMatrix *m;
- signed int i[9];
- double a, b, c, al, be, ga;
- double min_dist = +INFINITY;
-
- m = intmat_new(3, 3);
- cell_get_parameters(reference, &a, &b, &c, &al, &be, &ga);
- *pmb = NULL;
-
- for ( i[0]=-1; i[0]<=+1; i[0]++ ) {
- for ( i[1]=-1; i[1]<=+1; i[1]++ ) {
- for ( i[2]=-1; i[2]<=+1; i[2]++ ) {
- for ( i[3]=-1; i[3]<=+1; i[3]++ ) {
- for ( i[4]=-1; i[4]<=+1; i[4]++ ) {
- for ( i[5]=-1; i[5]<=+1; i[5]++ ) {
- for ( i[6]=-1; i[6]<=+1; i[6]++ ) {
- for ( i[7]=-1; i[7]<=+1; i[7]++ ) {
- for ( i[8]=-1; i[8]<=+1; i[8]++ ) {
-
- UnitCell *test;
- int j, k;
- int l = 0;
- signed int det;
-
- for ( j=0; j<3; j++ )
- for ( k=0; k<3; k++ )
- intmat_set(m, j, k, i[l++]);
-
- det = intmat_det(m);
- if ( (det != +1) && (det != -1) ) continue;
-
- test = cell_transform_intmat(cell, m);
-
- if ( compare_cell_parameters(reference, test, tolerance) ) {
-
- double at, bt, ct, alt, bet, gat;
- double dist;
+ UnitCell *cell;
+ UnitCell *reference;
+ IntegerMatrix *CBint;
+ RationalMatrix *CiA;
+ RationalMatrix *CB;
+ IntegerMatrix *Mcell;
+ IntegerMatrix *Mref;
+ double eps;
+ double G6cell[6];
+ double G6ref[6];
- cell_get_parameters(test, &at, &bt, &ct, &alt, &bet, &gat);
- dist = g6_distance(at, bt, ct, alt, bet, gat,
- a, b, c, al, be, ga);
- if ( dist < min_dist ) {
- min_dist = dist;
- intmat_free(*pmb);
- *pmb = intmat_copy(m);
- }
+ /* Un-center both cells */
+ reference = uncenter_cell(reference_in, &CBint, NULL);
+ if ( reference == NULL ) return 0;
+ CB = rtnl_mtx_from_intmat(CBint);
+ intmat_free(CBint);
- }
+ cell = uncenter_cell(cell_in, NULL, &CiA);
+ if ( cell == NULL ) return 0;
- cell_free(test);
+ /* Calculate G6 components for both */
+ g6_components(G6cell, cell);
+ g6_components(G6ref, reference);
- }
- }
- }
- }
- }
- }
- }
- }
- }
+ /* Convert both to reduced basis (stably) */
+ eps = pow(cell_get_volume(cell), 1.0/3.0) * 1e-5;
+ Mcell = reduce_g6(G6cell, eps);
+ eps = pow(cell_get_volume(reference), 1.0/3.0) * 1e-5;
+ Mref = reduce_g6(G6ref, eps);
- intmat_free(m);
+ /* Compare cells (including nearby ones, possibly permuting
+ * axes if close to equality) */
- if ( isinf(min_dist) ) {
- return 0;
- }
-
- /* Solution found */
- return 1;
+ intmat_free(Mcell);
+ intmat_free(Mref);
+ cell_free(reference);
+ cell_free(cell);
}
diff --git a/libcrystfel/src/cell-utils.h b/libcrystfel/src/cell-utils.h
index fa577269..e46d860b 100644
--- a/libcrystfel/src/cell-utils.h
+++ b/libcrystfel/src/cell-utils.h
@@ -90,9 +90,6 @@ extern double cell_get_volume(UnitCell *cell);
extern int compare_cell_parameters(UnitCell *cell, UnitCell *reference,
double *tolerance);
-extern int compare_permuted_cell_parameters(UnitCell *cell, UnitCell *reference,
- double *tolerance, IntegerMatrix **pmb);
-
extern int compare_cell_parameters_and_orientation(UnitCell *cell,
UnitCell *reference,
const double ltl,
@@ -108,6 +105,9 @@ extern int compare_reindexed_cell_parameters(UnitCell *cell, UnitCell *reference
double *tolerance, int csl,
RationalMatrix **pmb);
+extern int compare_lattices(UnitCell *cell_in, UnitCell *reference_in,
+ double *tolerance, RationalMatrix **pmb);
+
#ifdef __cplusplus
}
#endif
diff --git a/libcrystfel/src/index.c b/libcrystfel/src/index.c
index b07dd067..aa0cfb6a 100644
--- a/libcrystfel/src/index.c
+++ b/libcrystfel/src/index.c
@@ -547,26 +547,13 @@ static int check_cell(IndexingFlags flags, Crystal *cr, UnitCell *target,
double *tolerance)
{
UnitCell *out;
- IntegerMatrix *im;
RationalMatrix *rm;
/* Check at all? */
if ( ! ((flags & INDEXING_CHECK_CELL_COMBINATIONS)
|| (flags & INDEXING_CHECK_CELL_AXES)) ) return 0;
- if ( compare_permuted_cell_parameters(crystal_get_cell(cr), target,
- tolerance, &im) )
- {
- out = cell_transform_intmat(crystal_get_cell(cr), im);
- cell_free(crystal_get_cell(cr));
- crystal_set_cell(cr, out);
- intmat_free(im);
- return 0;
- }
-
- if ( (flags & INDEXING_CHECK_CELL_COMBINATIONS )
- && compare_reindexed_cell_parameters(crystal_get_cell(cr), target,
- tolerance, 0, &rm) )
+ if ( compare_lattices(crystal_get_cell(cr), target, tolerance, &rm) )
{
out = cell_transform_rational(crystal_get_cell(cr), rm);
cell_free(crystal_get_cell(cr));
@@ -576,7 +563,6 @@ static int check_cell(IndexingFlags flags, Crystal *cr, UnitCell *target,
}
return 1;
-
}