-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathCrystalSystems_a.cpp
More file actions
151 lines (135 loc) · 5.96 KB
/
CrystalSystems_a.cpp
File metadata and controls
151 lines (135 loc) · 5.96 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
/*
* To change this license header, choose License Headers in Project Properties.
* To change this template file, choose Tools | Templates
* and open the template in the editor.
*/
#include "CrystalSystems.h"
#include <cmath>
/**
* triclinic cell: all a, b, c, alpha, beta, gamma free
* @param cell
* @param restraints
* @return
*/
double CrystSyst::Triclinic::target(const gsl_vector* cell, void* restraints) {
gsl_vector* R = (gsl_vector*) restraints;
const double a (gsl_vector_get(cell, 0));
const double b (gsl_vector_get(cell, 1));
const double c (gsl_vector_get(cell, 2));
const double cosalpha (std::cos(gsl_vector_get(cell, 3)));
const double cosbeta (std::cos(gsl_vector_get(cell, 4)));
const double cosgamma (std::cos(gsl_vector_get(cell, 5)));
double T(0.0);
for (size_t i = 0; i < R->size; i+=5) {
const double dx (gsl_vector_get(R, i+0));
const double dy (gsl_vector_get(R, i+1));
const double dz (gsl_vector_get(R, i+2));
const double R2 (gsl_vector_get(R, i+3));
const double wR (gsl_vector_get(R, i+4));
double dX2 = dx*dx*a*a + dy*dy*b*b + dz*dz*c*c
+ 2*dx*dy*a*b*cosgamma
+ 2*dx*dz*a*c*cosbeta
+ 2*dy*dz*b*c*cosalpha;
T += wR*(dX2 - R2)*(dX2 - R2);
}
return T;
}
void CrystSyst::Triclinic::gradT(const gsl_vector* cell, void* restraints, gsl_vector* gradT) {
gsl_vector* R = (gsl_vector*) restraints;
const double a(gsl_vector_get(cell, 0));
const double b(gsl_vector_get(cell, 1));
const double c(gsl_vector_get(cell, 2));
const double cosalpha(std::cos(gsl_vector_get(cell, 3)));
const double cosbeta(std::cos(gsl_vector_get(cell, 4)));
const double cosgamma(std::cos(gsl_vector_get(cell, 5)));
const double sinalpha(std::sin(gsl_vector_get(cell, 3)));
const double sinbeta(std::sin(gsl_vector_get(cell, 4)));
const double singamma(std::sin(gsl_vector_get(cell, 5)));
double dTda(0), dTdb(0), dTdc(0), dTdalpha(0), dTdbeta(0), dTdgamma(0);
for (size_t i = 0; i < R->size; i += 5) {
const double dx(gsl_vector_get(R, i + 0));
const double dy(gsl_vector_get(R, i + 1));
const double dz(gsl_vector_get(R, i + 2));
const double R2(gsl_vector_get(R, i + 3));
const double wR(gsl_vector_get(R, i + 4));
double dX2 = dx * dx * a * a + dy * dy * b * b + dz * dz * c * c
+ 2 * dx * dy * a * b * cosgamma
+ 2 * dx * dz * a * c * cosbeta
+ 2 * dy * dz * b * c*cosalpha;
double dta = 2 * dx * dx * a
+ 2 * dx * dy * b * cosgamma
+ 2 * dx * dz * c * cosbeta;
double dtb = 2 * dy * dy * b
+ 2 * dx * dy * a * cosgamma
+ 2 * dy * dz * c * cosalpha;
double dtc = 2 * dz * dz * c
+ 2 * dx * dz * a * cosbeta
+ 2 * dy * dz * b * cosalpha;
double dtalpha = -2 * dy * dz * b * c * sinalpha;
double dtbeta = -2 * dx * dz * a * c * sinbeta;
double dtgamma = -2 * dx * dy * a * b * singamma;
dTda += 2 * wR * (dX2 - R2) * dta;
dTdb += 2 * wR * (dX2 - R2) * dtb;
dTdc += 2 * wR * (dX2 - R2) * dtc;
dTdalpha += 2 * wR * (dX2 - R2) * dtalpha;
dTdbeta += 2 * wR * (dX2 - R2) * dtbeta;
dTdgamma += 2 * wR * (dX2 - R2) * dtgamma;
}
gsl_vector_set(gradT, 0, dTda);
gsl_vector_set(gradT, 1, dTdb);
gsl_vector_set(gradT, 2, dTdc);
gsl_vector_set(gradT, 3, dTdalpha);
gsl_vector_set(gradT, 4, dTdbeta);
gsl_vector_set(gradT, 5, dTdgamma);
}
void CrystSyst::Triclinic::TgradT(const gsl_vector* cell, void* restraints, double* T, gsl_vector* gradT) {
gsl_vector* R = (gsl_vector*) restraints;
const double a (gsl_vector_get(cell, 0));
const double b (gsl_vector_get(cell, 1));
const double c (gsl_vector_get(cell, 2));
const double cosalpha (std::cos(gsl_vector_get(cell, 3)));
const double cosbeta (std::cos(gsl_vector_get(cell, 4)));
const double cosgamma (std::cos(gsl_vector_get(cell, 5)));
const double sinalpha (std::sin(gsl_vector_get(cell, 3)));
const double sinbeta (std::sin(gsl_vector_get(cell, 4)));
const double singamma (std::sin(gsl_vector_get(cell, 5)));
*T = 0.0;
double dTda(0), dTdb(0), dTdc(0), dTdalpha(0), dTdbeta(0), dTdgamma(0);
for (size_t i = 0; i < R->size; i += 5) {
const double dx(gsl_vector_get(R, i + 0));
const double dy(gsl_vector_get(R, i + 1));
const double dz(gsl_vector_get(R, i + 2));
const double R2(gsl_vector_get(R, i + 3));
const double wR(gsl_vector_get(R, i + 4));
double dX2 = dx * dx * a * a + dy * dy * b * b + dz * dz * c * c
+ 2 * dx * dy * a * b * cosgamma
+ 2 * dx * dz * a * c * cosbeta
+ 2 * dy * dz * b * c*cosalpha;
double dta = 2 * dx * dx * a
+ 2 * dx * dy * b * cosgamma
+ 2 * dx * dz * c * cosbeta;
double dtb = 2 * dy * dy * b
+ 2 * dx * dy * a * cosgamma
+ 2 * dy * dz * c * cosalpha;
double dtc = 2 * dz * dz * c
+ 2 * dx * dz * a * cosbeta
+ 2 * dy * dz * b * cosalpha;
double dtalpha = -2 * dy * dz * b * c * sinalpha;
double dtbeta = -2 * dx * dz * a * c * sinbeta;
double dtgamma = -2 * dx * dy * a * b * singamma;
const double dX2R2 (dX2 - R2);
*T += wR*dX2R2*dX2R2;
dTda += 2 * wR*dX2R2 * dta;
dTdb += 2 * wR*dX2R2 * dtb;
dTdc += 2 * wR*dX2R2 * dtc;
dTdalpha += 2 * wR*dX2R2 * dtalpha;
dTdbeta += 2 * wR*dX2R2 * dtbeta;
dTdgamma += 2 * wR*dX2R2 * dtgamma;
}
gsl_vector_set(gradT, 0, dTda);
gsl_vector_set(gradT, 1, dTdb);
gsl_vector_set(gradT, 2, dTdc);
gsl_vector_set(gradT, 3, dTdalpha);
gsl_vector_set(gradT, 4, dTdbeta);
gsl_vector_set(gradT, 5, dTdgamma);
}