aboutsummaryrefslogtreecommitdiff
path: root/src/ewald.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/ewald.c')
-rw-r--r--src/ewald.c116
1 files changed, 101 insertions, 15 deletions
diff --git a/src/ewald.c b/src/ewald.c
index 94fb4ed6..f24e9273 100644
--- a/src/ewald.c
+++ b/src/ewald.c
@@ -21,6 +21,11 @@
#include "detector.h"
+#define SAMPLING (4)
+#define BWSAMPLING (10)
+#define BANDWIDTH (0.05)
+
+
static struct rvec quat_rot(struct rvec q, struct quaternion z)
{
struct rvec res;
@@ -52,18 +57,9 @@ static struct rvec quat_rot(struct rvec q, struct quaternion z)
}
-void get_ewald(struct image *image)
+static void add_sphere(struct image *image, double k)
{
int x, y;
- double k; /* Wavenumber */
-
- k = 1/image->lambda;
-
- image->qvecs = malloc(image->width * image->height
- * sizeof(struct rvec));
-
- image->twotheta = malloc(image->width * image->height
- * sizeof(double));
for ( x=0; x<image->width; x++ ) {
for ( y=0; y<image->height; y++ ) {
@@ -73,8 +69,8 @@ void get_ewald(struct image *image)
double r;
double twothetax, twothetay, twotheta;
double qx, qy, qz;
- struct rvec q;
- int p;
+ struct rvec q1, q2, q3, q4;
+ int p, sx, sy, i;
/* Calculate q vectors for Ewald sphere */
for ( p=0; p<image->det.n_panels; p++ ) {
@@ -86,23 +82,80 @@ void get_ewald(struct image *image)
/ image->resolution;
ry = ((double)y - image->det.panels[p].cy)
/ image->resolution;
+ break;
}
}
+
+ /* Bottom left corner */
r = sqrt(pow(rx, 2.0) + pow(ry, 2.0));
+ twothetax = atan2(rx, image->camera_len);
+ twothetay = atan2(ry, image->camera_len);
+ twotheta = atan2(r, image->camera_len);
+ qx = k * sin(twothetax);
+ qy = k * sin(twothetay);
+ qz = k - k * cos(twotheta);
+ q1.u = qx; q1.v = qy; q1.w = qz;
+ /* 2theta value is calculated at the bottom left only */
+ image->twotheta[x + image->width*y] = twotheta;
+ /* Bottom right corner (using the same panel configuration!) */
+ rx = ((double)(x+1) - image->det.panels[p].cx)
+ / image->resolution;
+ ry = ((double)y - image->det.panels[p].cy)
+ / image->resolution;
twothetax = atan2(rx, image->camera_len);
twothetay = atan2(ry, image->camera_len);
twotheta = atan2(r, image->camera_len);
+ qx = k * sin(twothetax);
+ qy = k * sin(twothetay);
+ qz = k - k * cos(twotheta);
+ q2.u = qx; q2.v = qy; q2.w = qz;
+ /* Top left corner (using the same panel configuration!) */
+ rx = ((double)x - image->det.panels[p].cx)
+ / image->resolution;
+ ry = ((double)(y+1) - image->det.panels[p].cy)
+ / image->resolution;
+ twothetax = atan2(rx, image->camera_len);
+ twothetay = atan2(ry, image->camera_len);
+ twotheta = atan2(r, image->camera_len);
qx = k * sin(twothetax);
qy = k * sin(twothetay);
qz = k - k * cos(twotheta);
+ q3.u = qx; q3.v = qy; q3.w = qz;
- q.u = qx; q.v = qy; q.w = qz;
- image->qvecs[x + image->width*y] = quat_rot(q,
+ /* Top right corner (using the same panel configuration!) */
+ rx = ((double)(x+1) - image->det.panels[p].cx)
+ / image->resolution;
+ ry = ((double)(y+1) - image->det.panels[p].cy)
+ / image->resolution;
+ twothetax = atan2(rx, image->camera_len);
+ twothetay = atan2(ry, image->camera_len);
+ twotheta = atan2(r, image->camera_len);
+ qx = k * sin(twothetax);
+ qy = k * sin(twothetay);
+ qz = k - k * cos(twotheta);
+ q4.u = qx; q4.v = qy; q4.w = qz;
+
+ /* Now interpolate between the values to get
+ * the sampling points */
+ i = 0;
+ for ( sx=0; sx<SAMPLING; sx++ ) {
+ for ( sy=0; sy<SAMPLING; sy++ ) {
+
+ struct rvec q;
+
+ q.u = q1.u + ((q2.u - q1.u)/SAMPLING)*sx
+ + ((q3.u - q1.u)/SAMPLING)*sy;
+ q.v = q1.v + ((q2.v - q1.v)/SAMPLING)*sx
+ + ((q3.v - q1.v)/SAMPLING)*sy;
+ q.w = q1.w + ((q2.w - q1.w)/SAMPLING)*sx
+ + ((q3.w - q1.w)/SAMPLING)*sy;
+ image->qvecs[i++][x + image->width*y] = quat_rot(q,
image->orientation);
- image->twotheta[x + image->width*y] = twotheta;
+ }
+ }
if ( (x==0) && (y==(int)image->y_centre) ) {
double s;
@@ -123,4 +176,37 @@ void get_ewald(struct image *image)
}
}
+
+}
+
+
+void get_ewald(struct image *image)
+{
+ double kc; /* Wavenumber */
+ int i, kstep;
+
+ kc = 1/image->lambda; /* Centre */
+
+ image->qvecs = malloc(image->width * image->height
+ * sizeof(struct rvec *));
+
+ image->twotheta = malloc(image->width * image->height
+ * sizeof(double));
+
+ /* Create the spheres */
+ image->nspheres = SAMPLING*SAMPLING*BWSAMPLING;
+ for ( i=0; i<image->nspheres; i++ ) {
+ image->qvecs[i] = malloc(image->width * image->height
+ * sizeof(struct rvec));
+ }
+
+ for ( kstep=0; kstep<BWSAMPLING; kstep++ ) {
+
+ double k;
+
+ k = kc + (kstep-(BWSAMPLING/2))*kc*(BANDWIDTH/BWSAMPLING);
+
+ add_sphere(image, k);
+
+ }
}