-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathGBlab2oct.m
151 lines (129 loc) · 6.24 KB
/
GBlab2oct.m
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
151
function oAB = GBlab2oct(qA_Lab,qB_Lab,nA_Lab,convention,epsijk)
arguments
qA_Lab(:,4) double {mustBeReal,mustBeFinite,mustBeRowNormalized(qA_Lab,1e-6)}
qB_Lab(:,4) double {mustBeReal,mustBeFinite,mustBeRowNormalized(qB_Lab,1e-6)}
nA_Lab(:,3) double {mustBeReal,mustBeFinite}
convention char {mustBeMember(convention,'francis')} = 'francis'
epsijk(1,1) double = 1
end
% GBLAB2OCT convert lab coordinate grain boundaries to octonions
% Convert the crystal orientations (expressed as quaternions) of grains
% meeting at a grain boundary (GB) together with the corresponding GB
% normals to the GB octonion defined by Francis [1] and required by the
% GBdist function [2].
%-------------------------------------------------------------------------%
% Filename: GBlab2oct.m
% Author: Oliver Johnson, Refactored by Sterling Baird
% Date: 9/22/2020
%
% NOTE: The convention used by [1] and required for proper use of the
% GBdist function [2] is that the GB plane reference frame is defined such
% that the GB plane lies in the zR = 0 plane, with grain A occupying zR > 0
% and grain B occupying zR < 0, which implies that the normal pointing away
% from grain B (which we call nB, using the outward-pointing normal
% convention) is parallel to zR. Note that this is in contrast to Olmsted's
% convention which is that the GB plane lies in the zR = 0 plane with grain
% A occupying zR < 0 and grain B occupying zR > 0, which implies that the
% normal pointing away from grain A (which we call nA, using the
% outward-pointing normal convention) is parallel to zR. Moreover, the
% GB5DOF function has a completely different convention, which is that the
% GB plane reference frame is defined such that the GB plane lies in the
% xR = 0 plane with grain A occupying xR < 0 and grain B occupying xR > 0,
% which implies that the normal pointing away from grain A (which we call
% nA, using the outward-pointing normal convention) is parallel to xR.
%
% Inputs:
% qA_Lab - An nGB-by-4 array of quaternions representing the orientations
% of one of the grains at the GB (grain A). qA_Lab(i,:)
% represents the quaternion defining the orientation of grain A
% for the i-th GB. With a reference coordinate system defined
% such that z = 0 is the GB plane, grain A is the region z < 0.
% qB_Lab - An nGB-by-4 array of quaternions representing the orientations
% of one of the grains at the GB (grain B). qB_Lab(i,:)
% represents the quaternion defining the orientation of grain B
% for the i-th GB. With a reference coordinate system defined
% such that z = 0 is the GB plane, grain B is the region z > 0.
% nA_Lab - A 3-by-nGB array of 3D vectors representing the GB normals
% expressed in the Lab coordinate system. As a convention, we
% define nA_Lab such that it is pointing away from grain A (i.e.
% it is the outward-pointing normal). nA_Lab(:,i) give the
% vector representing the GB normal for the i-th GB.
%
% Outputs:
% oAB - An nGB-by-8 array of octonions where oAB(i,:) represents the i-th
% GB. Note that these are NOT normalized octonions, rather they are
% composed of two normalized quaternions, so that the magnitude of
% each octonion is sqrt(2). This is the expected format of
% octonions used as input to GBdist.
%
% Usage:
% oAB = GBlab2oct(qA_Lab,qB_Lab,nA_Lab,convention)
%
% [1] T. Francis, I. Chesser, S. Singh, E.A. Holm, M. De Graef, A geodesic
% octonion metric for grain boundaries, Acta Mater. 166 (2019) 135–147.
% doi:10.1016/j.actamat.2018.12.034.
% [2] https://github.com/ichesser/GB_octonion_code
%-------------------------------------------------------------------------%
%% Setup
% number of GBs
Ngb = size(qA_Lab,1);
% ensure GB normals are normalized
nA_Lab = normr(nA_Lab);
% normals pointing away from grain B in the Lab frame
nB_Lab = -nA_Lab;
%% Construct reference frame quaternions (qR)
switch convention
case 'francis' % francis convention has z-axis aligned to nB_Lab
zR = nB_Lab;
xR = cross(zR,rand(Ngb,3),2); % cross with random vector to get an arbitrary vector in the GB plane to use as the x-axis
xR = normr(xR);
% case 'olmsted' % olmsted convention has z-axis aligned to -nB_Lab
% zR = -nB_Lab;
%
% xR = cross(zR,rand(3,Ngb)); % cross with random vector to get an arbitrary vector in the GB plane to use as the x-axis
% xR = xR./sqrt(sum(xR.^2,1));
% case 'livermore' % livermore convention has x-axis aligned to -nB_Lab
% xR = -nB_Lab;
%
% zR = cross(xR,rand(3,Ngb)); % cross with random vector to get an arbitrary vector in the GB plane to use as the x-axis
% zR = zR./sqrt(sum(zR.^2,1));
end
yR = cross(zR,xR,2); % cross z with x to define y-axis
yR = normr(yR);
% make rotation matrices
gR = [permute(xR.',[1,3,2]),permute(yR.',[1,3,2]),permute(zR.',[1,3,2])];
% convert to quaternions
qR = gmat2q(gR);
%% Construct qA_R and qB_R (quaternions in the GB reference frame)
qA_R = qmult(qA_Lab,qinv(qR),epsijk);
qB_R = qmult(qB_Lab,qinv(qR),epsijk);
% qA_R = qmultiply(qA_Lab,qinv_johnson(qR)); % note this is using Toby's misorientation convention, these should not be used with our code directly
% qB_R = qmultiply(qB_Lab,qinv_johnson(qR));
%% Convert to octonion
oAB = [qA_R, qB_R];
end
%% Argument Validation Function(s)
function mustBeRowNormalized(X,tol)
nrows = size(X,1);
e = vecnorm(X,2,2)-1; %error
ae = abs(e); %absolute error
tolOK = ae < tol; %tolerance check
badIDs = find(~tolOK); %number of "bad" rows
if ~isempty(badIDs)
nbad = length(badIDs);
avgnm = mean(norm(X,2));
error([int2str(nbad) '/' int2str(nrows) ' rows where norm ~= 1 starting at element ' ...
int2str(badIDs(1)) ', avgnm == ' int2str(avgnm)])
end
end
%% CODE GRAVEYARD
%{
assert(size(qA_Lab,2) == 4 && size(qB_Lab,2) == 4,'qA_Lab and qB_Lab must be n-by-4 arrays of quaternions.')
assert(size(nA_Lab,1) == 3,'nA_Lab must be a 3-by-n array of vectors.')
%% Ensure proper formatting of inputs
% nA_Lab = nA_Lab./sqrt(sum(nA_Lab.^2,1));
% xR = xR./sqrt(sum(xR.^2,1));
yR = yR./sqrt(sum(yR.^2,1));
% make rotation matrices
gR = [permute(xR,[1,3,2]),permute(yR,[1,3,2]),permute(zR,[1,3,2])];
%}