aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/symmetry.c107
-rw-r--r--src/symmetry.h6
2 files changed, 113 insertions, 0 deletions
diff --git a/src/symmetry.c b/src/symmetry.c
index 6f3ce675..e0b92335 100644
--- a/src/symmetry.c
+++ b/src/symmetry.c
@@ -216,3 +216,110 @@ void get_asymm(signed int h, signed int k, signed int l,
ERROR("No match found in %s for %i %i %i\n", sym, h, k, l);
abort();
}
+
+
+static const char *get_holohedral(const char *sym)
+{
+ /* Triclinic */
+ if ( strcmp(sym, "1") == 0 ) return "-1";
+ if ( strcmp(sym, "1") == 0 ) return "-1";
+
+ /* Hexagonal */
+ if ( strcmp(sym, "6") == 0 ) return "6/m";
+ if ( strcmp(sym, "6/m") == 0 ) return "6/mmm";
+ if ( strcmp(sym, "6/mmm") == 0 ) return "6/mmm";
+
+ /* TODO: Add more groups here */
+
+ ERROR("Couldn't find holohedral point group for '%s'\n", sym);
+ abort();
+}
+
+
+/* This is kind of like a "numerical" left coset decomposition.
+ * Given a reflection index and a point group, it returns the "idx-th"
+ * twinning possibility for the reflection. It returns "idx" if
+ * successful. To just count the number of possibilities, set idx=-1.
+ *
+ * The sequence of operators producing each possibility is guaranteed to
+ * be the same for any choice of indices given the same point group. */
+static int coset_decomp(signed int hs, signed int ks, signed int ls,
+ signed int *hp, signed int *kp, signed int *lp,
+ const char *mero, signed int idx)
+{
+ const char *holo = get_holohedral(mero);
+ int n_mero, n_holo;
+ int i;
+ signed int n_twins = 1;
+ signed int h, k, l;
+ ReflItemList *twins;
+
+ twins = new_items();
+
+ if ( idx == 0 ) {
+ /* Twin index zero is always the original orientation */
+ *hp = hs; *kp = ks; *lp = ls;
+ return 0;
+ }
+
+ get_asymm(hs, ks, ls, &h, &k, &l, mero);
+
+ /* How many equivalents in the holohedral point group are not
+ * equivalents according to the (possibly) merohedral group? */
+ n_holo = num_equivs(h, k, l, holo);
+ n_mero = num_equivs(h, k, l, mero);
+
+ for ( i=0; i<n_holo; i++ ) {
+
+ signed int h_holo, k_holo, l_holo;
+ signed int hs_holo, ks_holo, ls_holo;
+
+ /* Get equivalent according to the holohedral group */
+ get_equiv(h, k, l, &hs_holo, &ks_holo, &ls_holo, holo, i);
+
+ /* Put it into the asymmetric cell for the merohedral group */
+ get_asymm(hs_holo, ks_holo, ls_holo,
+ &h_holo, &k_holo, &l_holo, mero);
+
+ /* Is this the same reflection as we started with?
+ * If not, this reflection is 'equivalent by twinning' */
+ if ( (h_holo != h) || (k_holo != k) || (l_holo != l) ) {
+
+ if ( find_item(twins, h_holo, k_holo, l_holo) )
+ continue;
+
+ if ( n_twins == idx ) {
+ *hp = h_holo;
+ *kp = k_holo;
+ *lp = l_holo;
+ delete_items(twins);
+ return n_twins;
+ }
+ add_item(twins, h_holo, k_holo, l_holo);
+ n_twins++;
+ }
+
+ }
+
+ delete_items(twins);
+ return n_twins;
+}
+
+
+/* Get the number of twinned "equivalents" for this reflection */
+int num_twins(signed int h, signed int k, signed int l, const char *sym)
+{
+ return coset_decomp(h, k, l, NULL, NULL, NULL, sym, -1);
+}
+
+
+void get_twins(signed int h, signed int k, signed int l,
+ signed int *hp, signed int *kp, signed int *lp,
+ const char *sym, int idx)
+{
+ if ( coset_decomp(h, k, l, hp, kp, lp, sym, idx) != idx ) {
+ ERROR("Failed coset decomposition for %i %i %i in %s\n",
+ h, k, l, sym);
+ abort();
+ }
+}
diff --git a/src/symmetry.h b/src/symmetry.h
index ebe123c9..52dd387d 100644
--- a/src/symmetry.h
+++ b/src/symmetry.h
@@ -29,5 +29,11 @@ extern void get_equiv(signed int h, signed int k, signed int l,
signed int *he, signed int *ke, signed int *le,
const char *sym, int idx);
+extern int num_twins(signed int h, signed int k, signed int l, const char *sym);
+
+extern void get_twins(signed int h, signed int k, signed int l,
+ signed int *hp, signed int *kp, signed int *lp,
+ const char *sym, int idx);
+
#endif /* SYMMETRY_H */