diff options
author | Thomas White <taw@physics.org> | 2010-07-26 19:19:00 +0200 |
---|---|---|
committer | Thomas White <taw@physics.org> | 2012-02-22 15:26:54 +0100 |
commit | 2f36611ec798cdbd00ace4da0a3fe2449d2bf16d (patch) | |
tree | 3dc8e77dc9d041710673705938201301fa5800a6 | |
parent | 5d22ee1971a09a12910e2d03ac7dabfc159c1924 (diff) |
render_hkl: Fix arbitrary zone axes
-rw-r--r-- | src/render_hkl.c | 368 |
1 files changed, 222 insertions, 146 deletions
diff --git a/src/render_hkl.c b/src/render_hkl.c index 06075131..3bc8c5f3 100644 --- a/src/render_hkl.c +++ b/src/render_hkl.c @@ -23,6 +23,10 @@ #ifdef HAVE_CAIRO #include <cairo.h> #include <cairo-pdf.h> +/* GSL is only used if Cairo is present */ +#include <gsl/gsl_matrix.h> +#include <gsl/gsl_blas.h> +#include <gsl/gsl_linalg.h> #endif #include "utils.h" @@ -76,8 +80,209 @@ static void show_help(const char *s) #ifdef HAVE_CAIRO +static int get_basis_change_coefficients(double *in, double *out) +{ + int s; + gsl_matrix *m; + gsl_matrix *inv; + gsl_permutation *perm; + + m = gsl_matrix_alloc(3, 3); + if ( m == NULL ) { + ERROR("Couldn't allocate memory for matrix\n"); + return 1; + } + gsl_matrix_set(m, 0, 0, in[0]); + gsl_matrix_set(m, 0, 1, in[1]); + gsl_matrix_set(m, 0, 2, in[2]); + gsl_matrix_set(m, 1, 0, in[3]); + gsl_matrix_set(m, 1, 1, in[4]); + gsl_matrix_set(m, 1, 2, in[5]); + gsl_matrix_set(m, 2, 0, in[6]); + gsl_matrix_set(m, 2, 1, in[7]); + gsl_matrix_set(m, 2, 2, in[8]); + + /* Invert */ + perm = gsl_permutation_alloc(m->size1); + if ( perm == NULL ) { + ERROR("Couldn't allocate permutation\n"); + gsl_matrix_free(m); + return 1; + } + inv = gsl_matrix_alloc(3, 3); + if ( inv == NULL ) { + ERROR("Couldn't allocate inverse\n"); + gsl_matrix_free(m); + gsl_permutation_free(perm); + return 1; + } + if ( gsl_linalg_LU_decomp(m, perm, &s) ) { + ERROR("Couldn't decompose matrix\n"); + gsl_matrix_free(m); + gsl_permutation_free(perm); + return 1; + } + if ( gsl_linalg_LU_invert(m, perm, inv) ) { + ERROR("Couldn't invert matrix\n"); + gsl_matrix_free(m); + gsl_permutation_free(perm); + return 1; + } + gsl_permutation_free(perm); + gsl_matrix_free(m); + + /* Transpose */ + gsl_matrix_transpose(inv); + + out[0] = gsl_matrix_get(inv, 0, 0); + out[1] = gsl_matrix_get(inv, 0, 1); + out[2] = gsl_matrix_get(inv, 0, 2); + out[3] = gsl_matrix_get(inv, 1, 0); + out[4] = gsl_matrix_get(inv, 1, 1); + out[5] = gsl_matrix_get(inv, 1, 2); + out[6] = gsl_matrix_get(inv, 2, 0); + out[7] = gsl_matrix_get(inv, 2, 1); + out[8] = gsl_matrix_get(inv, 2, 2); + + gsl_matrix_free(inv); + + return 0; + +} + + +static void draw_circles(signed int xh, signed int xk, signed int xl, + signed int yh, signed int yk, signed int yl, + signed int zh, signed int zk, signed int zl, + double *ref, unsigned int *counts, ReflItemList *items, + const char *sym, + cairo_t *dctx, int wght, double boost, int colscale, + UnitCell *cell, double radius, double theta, + double as, double bs, double cx, double cy, + double scale, + signed int *max_ux, signed int *max_uy, + double *max_val, double *max_u, double *max_v, + double *max_res) +{ + signed int xi, yi; + double in[9]; + double bc[9]; + + if ( dctx == NULL ) { + *max_u = 0.0; *max_v = 0.0; *max_val = 0.0; + *max_res = 0.0; *max_ux = 0; *max_uy = 0; + } + + in[0] = xh; in[1] = xk; in[2] = xl; + in[3] = yh; in[4] = yk; in[5] = yl; + in[6] = zh; in[7] = zk; in[8] = zl; + if ( get_basis_change_coefficients(in, bc) ) { + ERROR("Couldn't change basis.\n"); + return; + } + + /* Loop across the two basis directions */ + for ( xi=-INDMAX; xi<INDMAX; xi++ ) { + for ( yi=-INDMAX; yi<INDMAX; yi++ ) { + + double u, v, val, res; + int nequiv, p; + signed int h, k, l; + + h = xi*xh + yi*yh; + k = xi*xk + yi*yk; + l = xi*xl + yi*yl; + + /* Got this reflection? */ + if ( find_unique_equiv(items, h, k, l, sym, &h, &k, &l) == 0 ) { + continue; + } + + switch ( wght ) { + case WGHT_I : + val = lookup_intensity(ref, h, k, l); + break; + case WGHT_SQRTI : + val = lookup_intensity(ref, h, k, l); + val = (val>0.0) ? sqrt(val) : 0.0; + break; + case WGHT_COUNTS : + val = lookup_count(counts, h, k, l); + val /= (float)num_equivs(h, k, l, sym); + break; + case WGHT_RAWCOUNTS : + val = lookup_count(counts, h, k, l); + break; + default : + ERROR("Invalid weighting.\n"); + abort(); + } + + /* For each equivalent reflection... */ + nequiv = num_equivs(h, k, l, sym); + for ( p=0; p<nequiv; p++ ) { + + signed int he, ke, le; + signed int ux, uy, uz; + + get_equiv(h, k, l, &he, &ke, &le, sym, p); + + /* Calculate the indices in the 2D basis */ + ux = he*bc[0] + ke*bc[1] + le*bc[2]; + uy = he*bc[3] + ke*bc[4] + le*bc[5]; + uz = he*bc[6] + ke*bc[7] + le*bc[8]; + + /* Reflection in the zone? */ + if ( uz != 0 ) continue; + + /* Absolute location in image based on 2D basis */ + u = (double)ux*as*sin(theta); + v = (double)ux*as*cos(theta) + (double)uy*bs; + + if ( dctx != NULL ) { + + float r, g, b; + + cairo_arc(dctx, ((double)cx)+u*scale, + ((double)cy)+v*scale, + radius, 0, 2*M_PI); + + render_scale(val, *max_val/boost, colscale, + &r, &g, &b); + cairo_set_source_rgb(dctx, r, g, b); + cairo_fill(dctx); + + } else { + + /* Find max vectors in plane for scaling */ + if ( fabs(u) > fabs(*max_u) ) *max_u = fabs(u); + if ( fabs(v) > fabs(*max_v) ) *max_v = fabs(v); + + /* Find max value for colour scale */ + if ( fabs(val) > fabs(*max_val) ) { + *max_val = fabs(val); + } + + /* Find max indices */ + if ( (uy==0) && (fabs(ux) > *max_ux) ) + *max_ux = fabs(ux); + if ( (ux==0) && (fabs(uy) > *max_uy) ) + *max_uy = fabs(uy); + + /* Find max resolution */ + res = resolution(cell, he, ke, le); + if ( res > *max_res ) *max_res = res; + } + + } + + } + } +} + + static void render_za(UnitCell *cell, ReflItemList *items, - double *ref, unsigned int *c, + double *ref, unsigned int *counts, double boost, const char *sym, int wght, int colscale) { cairo_surface_t *surface; @@ -97,21 +302,21 @@ static void render_za(UnitCell *cell, ReflItemList *items, signed int zh, zk, zl; double xx, xy, xz; double yx, yy, yz; - signed int xi, yi; char tmp[256]; cairo_text_extents_t size; double cx, cy; const double border = 200.0; - xh = 1; xk = -1; xl = 0; + xh = 0; xk = 1; xl = 0; yh = 0; yk = 0; yl = 1; + /* Vector product to determine the zone axis. */ zh = xk*yl - xl*yk; zk = - xh*yl + xl*yh; zl = xh*yk - xk*yh; - STATUS("Zone axis is %i %i %i\n", zh, zk, zl); + /* Size of output and centre definition */ wh = 1024; ht = 1024; cx = 512.0; @@ -132,9 +337,6 @@ static void render_za(UnitCell *cell, ReflItemList *items, cairo_set_source_rgb(dctx, 0.0, 0.0, 0.0); cairo_fill(dctx); - max_u = 0.0; max_v = 0.0; max_val = 0.0; - max_res = 0.0; max_ux = 0; max_uy = 0; - /* Work out reciprocal lattice spacings and angles for this cut */ if ( cell_get_reciprocal(cell, &asx, &asy, &asz, &bsx, &bsy, &bsz, @@ -152,74 +354,10 @@ static void render_za(UnitCell *cell, ReflItemList *items, as = modulus(xx, xy, xz) / 1e9; bs = modulus(yx, yy, yz) / 1e9; - for ( xi=-INDMAX; xi<INDMAX; xi++ ) { - for ( yi=-INDMAX; yi<INDMAX; yi++ ) { - - double u, v, val, res; - int ct; - int nequiv, p; - signed int h, k, l; - - h = xi*xh + yi*yh; - k = xi*xk + yi*yk; - l = xi*xl + yi*yl; - - /* Do we have this reflection? */ - if ( find_unique_equiv(items, h, k, l, sym, &h, &k, &l) == 0 ) { - continue; - } - - /* Reflection in the zone? */ - if ( zh*h + zk*k + zl*l != 0 ) continue; - - switch ( wght ) { - case WGHT_I : - val = lookup_intensity(ref, h, k, l); - break; - case WGHT_SQRTI : - val = lookup_intensity(ref, h, k, l); - val = (val>0.0) ? sqrt(val) : 0.0; - break; - case WGHT_COUNTS : - val = (float)ct; - val /= (float)num_equivs(h, k, l, sym); - break; - case WGHT_RAWCOUNTS : - val = (float)ct; - break; - default : - ERROR("Invalid weighting.\n"); - abort(); - } - - nequiv = num_equivs(h, k, l, sym); - for ( p=0; p<nequiv; p++ ) { - - signed int he, ke, le; - signed int ux, uy; - - get_equiv(h, k, l, &he, &ke, &le, sym, p); - - ux = he*xh + ke*xk + le*xl; - uy = he*yh + ke*yk + le*yl; - - u = (double)ux*as*sin(theta); - v = (double)ux*as*cos(theta) + uy*bs; - - if ( fabs(u) > fabs(max_u) ) max_u = fabs(u); - if ( fabs(v) > fabs(max_v) ) max_v = fabs(v); - if ( fabs(val) > fabs(max_val) ) { - max_val = fabs(val); - } - if ( fabs(ux) > max_ux ) max_ux = fabs(ux); - if ( fabs(uy) > max_uy ) max_uy = fabs(uy); - res = resolution(cell, he, ke, le); - if ( res > max_res ) max_res = res; - - } - - } - } + draw_circles(xh, xk, xl, yh, yk, yl, zh, zk, zl, + ref, counts, items, sym, NULL, wght, boost, colscale, cell, + 0.0, theta, as, bs, cx, cy, scale, + &max_ux, &max_uy, &max_val, &max_u, &max_v, &max_res); max_res /= 1e9; printf("Maximum resolution is 1/d = %5.3f nm^-1, d = %5.3f nm\n", @@ -240,79 +378,16 @@ static void render_za(UnitCell *cell, ReflItemList *items, sep_v = (double)scale*as*cos(theta) + scale*bs; /* We are interested in the smaller of the two separations */ max_r = (sep_u < sep_v) ? sep_u : sep_v; - max_r /= 2.0; /* Max radius if half the separation */ + max_r /= 2.0; /* Max radius is half the separation */ max_r -= 1.0; /* Add a tiny separation between circles */ if ( max_r < 1.0 ) { ERROR("Circle radius is probably too small (%f).\n", max_r); } - for ( xi=-INDMAX; xi<INDMAX; xi++ ) { - for ( yi=-INDMAX; yi<INDMAX; yi++ ) { - - double u, v, val; - int ct; - int nequiv, p; - signed int h, k, l; - - h = xi*xh + yi*yh; - k = xi*xk + yi*yk; - l = xi*xl + yi*yl; - - /* Do we have this reflection? */ - if ( find_unique_equiv(items, h, k, l, sym, &h, &k, &l) == 0 ) { - continue; - } - - - /* Reflection in the zone? */ - if ( zh*h + zk*k + zl*l != 0 ) continue; - - switch ( wght ) { - case WGHT_I : - val = lookup_intensity(ref, h, k, l); - break; - case WGHT_SQRTI : - val = lookup_intensity(ref, h, k, l); - val = (val>0.0) ? sqrt(val) : 0.0; - break; - case WGHT_COUNTS : - val = (float)ct; - val /= (float)num_equivs(h, k, l, sym); - break; - case WGHT_RAWCOUNTS : - val = (float)ct; - break; - default : - ERROR("Invalid weighting.\n"); - abort(); - } - - nequiv = num_equivs(h, k, l, sym); - for ( p=0; p<nequiv; p++ ) { - - signed int he, ke, le; - signed int ux, uy; - float r, g, b; - get_equiv(h, k, l, &he, &ke, &le, sym, p); - - ux = he*xh + ke*xk + le*xl; - uy = he*yh + ke*yk + le*yl; - - u = (double)ux*as*sin(theta); - v = (double)ux*as*cos(theta) + uy*bs; - - cairo_arc(dctx, ((double)cx)+u*scale, - ((double)cy)+v*scale, - max_r, 0, 2*M_PI); - - render_scale(val, max_val/boost, colscale, &r, &g, &b); - cairo_set_source_rgb(dctx, r, g, b); - cairo_fill(dctx); - - } - - } - } + draw_circles(xh, xk, xl, yh, yk, yl, zh, zk, zl, + ref, counts, items, sym, dctx, wght, boost, colscale, cell, + max_r, theta, as, bs, cx, cy, scale, + NULL, NULL, &max_val, NULL, NULL, NULL); out: /* Centre marker */ @@ -324,8 +399,9 @@ out: /* Draw indexing lines */ cairo_set_line_width(dctx, 4.0); cairo_move_to(dctx, (double)cx, (double)cy); - u = (double)max_ux*as*sin(theta); - v = (double)max_ux*as*cos(theta); + u = (1.0+max_ux)*as*sin(theta); + v = (1.0+max_ux)*as*cos(theta); + STATUS("max u %i\n", max_ux); cairo_line_to(dctx, cx+u*scale, cy+v*scale); cairo_set_source_rgb(dctx, 0.0, 1.0, 0.0); cairo_stroke(dctx); @@ -343,7 +419,7 @@ out: cairo_move_to(dctx, (double)cx, (double)cy); u = 0.0; - v = (double)max_uy*bs; + v = (1.0+max_uy)*bs; cairo_line_to(dctx, cx+u*scale, cy+v*scale); cairo_set_source_rgb(dctx, 0.0, 1.0, 0.0); cairo_stroke(dctx); |