Metrology
Table of Contents
In this document, we suppose that we are able to measure perfectly the position of the sample with respect to the granite. Also, all the stages can be perfectly positioned.
In section 1, we verify that the function developed to compute the wanted pose (translation and orientation) of the sample with respect to the granite can be determined from the wanted position of each stage (translation stage, tilt stage, spindle and micro-hexapod). To do so, we impose a perfect displacement and all the stage, we perfectly measure the position of the sample with respect to the granite, and we verify that this measured position corresponds to the computed wanted pose of the sample.
Then, in section 2, we introduce some positioning error in the position stages. The positioning error of the sample expressed with respect to the granite frame (the one measured) is expressed in a frame connected to the NASS top platform. Finally, we move the NASS such that it compensate for the positioning error that are expressed in the frame of the NASS, and we verify that the positioning error of the sample is well compensated.
1 Verify that the function to compute the reference pose is correct
The goal here is to perfectly move the station and verify that there is no mismatch between the metrology measurement and the computation of the reference pose.
1.1 Prepare the Simulation
We load the configuration.
load('simscape/conf_simscape.mat');
We set a small StopTime.
set_param(conf_simscape, 'StopTime', '0.5');
We setup the reference path to be constant.
opts = struct( ... 'Ts', 1e-3, ... % Sampling Frequency [s] 'Dy_type', 'constant', ... % Either "constant" / "triangular" / "sinusoidal" 'Dy_amplitude', 5e-3, ... % Amplitude of the displacement [m] 'Dy_period', 1, ... % Period of the displacement [s] 'Ry_type', 'constant', ... % Either "constant" / "triangular" / "sinusoidal" 'Ry_amplitude', -1, ... % Amplitude [deg] 'Ry_period', 10, ... % Period of the displacement [s] 'Rz_type', 'constant', ... % Either "constant" / "rotating" 'Rz_amplitude', -135, ... % Initial angle [deg] 'Rz_period', 1, ... % Period of the rotating [s] 'Dh_type', 'constant', ... % For now, only constant is implemented 'Dh_pos', [0; 0; 0; -3; 1; 3], ... % Initial position [m,m,m,deg,deg,deg] of the top platform 'Rm_type', 'constant', ... % For now, only constant is implemented 'Rm_pos', [0, pi]', ... % Initial position of the two masses 'Dn_type', 'constant', ... % For now, only constant is implemented 'Dn_pos', [0; 0; 0; 0; 0; 0] ... % Initial position [m,m,m,deg,deg,deg] of the top platform ); initializeReferences(opts);
No position error for now (perfect positioning).
Dye = 0; % [m] Rye = 0; % [rad] Rze = 0; % [rad] Dhe = zeros(6,1); % [m,rad]
And we run the simulation.
sim('simscape/sim_nano_station_metrology.slx');
1.2 Verify that the pose of the sample is the same as the computed one
Let's denote:
- \(\{W\}\) the initial fixed frame (base in which the interferometric measurement is done)
- \(\{R\}\) the reference frame corresponding to the wanted pose of the sample
- \(\{M\}\) the frame corresponding to the measured pose of the sample
We have then computed:
- \({}^W\boldsymbol{T}_R\) which corresponds to the wanted pose of the sample with respect to the granite
- \({}^W\boldsymbol{T}_M\) which corresponds to the measured pose of the sample with respect to the granite
We load the reference and we compute the desired trajectory of the sample in the form of an homogeneous transformation matrix \({}^WT_R\).
n = length(Dref.Dy.Time); WTr = zeros(4, 4, n); for i = 1:n WTr(:, :, i) = computeReferencePose(Dref.Dy.Data(i), Dref.Ry.Data(i), Dref.Rz.Data(i), Dref.Dh.Data(i,:)); end
As the displacement is perfect, we also measure in simulation the pose of the sample with respect to the granite. From that we can compute the homogeneous transformation matrix \({}^WT_M\).
n = length(Dsm.R.Time); WTm = zeros(4, 4, n); WTm(1:3, 1:3, :) = Dsm.R.Data; WTm(1:3, 4, :) = [Dsm.x.Data' ; Dsm.y.Data' ; Dsm.z.Data']; WTm(4, 4, :) = 1;
As the simulation is perfect (no measurement error and no motion error), we should have that \[ {}^W\boldsymbol{T}_R = {}^W\boldsymbol{T}_M \]
Or are least: \[ {}^W\boldsymbol{T}_R(1:3, 4) = {}^W\boldsymbol{T}_M(1:3, 4) \] \[ {}^W\boldsymbol{R}_R^t \cdot {}^W\boldsymbol{R}_M = \boldsymbol{I}_3 \]
WTr(1:3, 4, end)-WTm(1:3, 4, end) WTr(1:3, 1:3, end)'*WTm(1:3, 1:3, end)-eye(3)
WTr(1:3, 4, end)-WTm(1:3, 4, end)
ans =
      5.38287405101034e-15
      9.42822209193395e-15
     -7.25141518012618e-16
WTr(1:3, 1:3, end)'*WTm(1:3, 1:3, end)-eye(3)
ans =
      1.53210777398272e-14     -1.60173523749974e-14     -7.42461647718073e-16
      1.60683098771042e-14      1.53210777398272e-14     -2.33146835171283e-15
     -3.95516952522712e-16     -1.72084568816899e-15       9.2370555648813e-14
1.3 Conclusion
We are able to compute the wanted position and orientation of the sample. Both the measurement and the theory gives the same result.
2 Verify that the function to convert the position error in the frame fixed to the nano-hexapod is working
We now introduce some positioning error in the stage. This will induce a global positioning error of the sample with respect to the desired pose that we can compute.
We want to verify that we are able to measure this positioning error and convert it in the frame attached to the Nano-hexapod.
2.1 Prepare the Simulation
We load the configuration.
load('simscape/conf_simscape.mat');
We set a small StopTime.
set_param(conf_simscape, 'StopTime', '0.5');
We setup the reference path to be constant.
opts = struct( ... 'Ts', 1e-3, ... % Sampling Frequency [s] 'Dy_type', 'constant', ... % Either "constant" / "triangular" / "sinusoidal" 'Dy_amplitude', 0, ... % Amplitude of the displacement [m] 'Ry_type', 'constant', ... % Either "constant" / "triangular" / "sinusoidal" 'Ry_amplitude', 0, ... % Amplitude [deg] 'Rz_type', 'constant', ... % Either "constant" / "rotating" 'Rz_amplitude', 180, ... % Initial angle [deg] 'Dh_type', 'constant', ... % For now, only constant is implemented 'Dh_pos', [0; 0; 0; 0; 0; 0], ... % Initial position [m,m,m,deg,deg,deg] of the top platform 'Rm_type', 'constant', ... % For now, only constant is implemented 'Rm_pos', [0, pi]', ... % Initial position of the two masses 'Dn_type', 'constant', ... % For now, only constant is implemented 'Dn_pos', [0; 0; 0; 0; 0; 0] ... % Initial position [m,m,m,deg,deg,deg] of the top platform ); initializeReferences(opts);
Now we introduce some positioning error.
Dye = 0; % [m] Rye = 0; % [rad] Rze = 0; % [rad] Dhe = [1e-3 ; 3e-3 ; 2e-3 ; 1e-3 ; 2e-3 ; 3e-3]; % [m,rad]
And we run the simulation.
sim('simscape/sim_nano_station_metrology.slx');
2.2 Compute the wanted pose of the sample in the NASS Base from the metrology and the reference
Now that we have introduced some positioning error, the computed wanted pose and the measured pose will not be the same.
We would like to compute \({}^M\boldsymbol{T}_R\) which corresponds to the wanted pose of the sample expressed in a frame attached to the top platform of the nano-hexapod (frame \(\{M\}\)).
We have:
\begin{align} {}^M\boldsymbol{T}_R &= {}^M\boldsymbol{T}_W \cdot {}^W\boldsymbol{T}_R \\ &= {}^W{\boldsymbol{T}_M}^{-1} \cdot {}^W\boldsymbol{T}_R \end{align}The top platform of the nano-hexapod is considered to be rigidly connected to the sample, thus, \({}^M\boldsymbol{T}_R\) corresponds to the pose error of the sample with respect to the nano-hexapod platform.
We load the reference and we compute the desired trajectory of the sample in the form of an homogeneous transformation matrix \({}^W\boldsymbol{T}_R\).
n = length(Dref.Dy.Time); WTr = zeros(4, 4, n); for i = 1:n WTr(:, :, i) = computeReferencePose(Dref.Dy.Data(i), Dref.Ry.Data(i), Dref.Rz.Data(i), Dref.Dh.Data(i,:)); end
We also measure in simulation the pose of the sample with respect to the granite. From that we can compute the homogeneous transformation matrix \({}^W\boldsymbol{T}_M\).
n = length(Dsm.R.Time); WTm = zeros(4, 4, n); WTm(1:3, 1:3, :) = Dsm.R.Data; WTm(1:3, 4, :) = [Dsm.x.Data' ; Dsm.y.Data' ; Dsm.z.Data']; WTm(4, 4, :) = 1;
The inverse of the transformation matrix can be obtain by (it is less computation intensive than doing a full inverse)
\begin{equation} {}^B\boldsymbol{T}_A = {}^A\boldsymbol{T}_B^{-1} = \left[ \begin{array}{ccc|c} & & & \\ & {}^A\boldsymbol{R}_B^T & & -{}^A \boldsymbol{R}_B^T {}^A\boldsymbol{P}_{O_B} \\ & & & \\ \hline 0 & 0 & 0 & 1 \\ \end{array} \right] \end{equation}Finally, we compute \({}^M\boldsymbol{T}_R\).
MTr = zeros(4, 4, n); for i = 1:n MTr(:, :, i) = [WTm(1:3,1:3,i)', -WTm(1:3,1:3,i)'*WTm(1:3,4,i) ; 0 0 0 1]*WTr(:,:,i); end
Verify that the pose error corresponds to the positioning error of the stages.
| Edx [m] | Edy [m] | Edz [m] | Erx [rad] | Ery [rad] | Erz [rad] | |
|---|---|---|---|---|---|---|
| Error | -1.0e-03 | -3.0e-03 | -2.0e-03 | -1.0e-03 | -2.0e-03 | -3.0e-03 | 
2.3 Verify that be imposing the error motion on the nano-hexapod, we indeed have zero error at the end
We now impose a displacement of the nano hexapod corresponding to the measured position error.
opts.Dn_pos = [Edx, Edy, Edz, 180/pi*Erx, 180/pi*Ery, 180/pi*Erz]'; initializeReferences(opts);
And we run the simulation.
sim('simscape/sim_nano_station_metrology.slx');
We load the reference and we compute the desired trajectory of the sample in the form of an homogeneous transformation matrix \({}^WT_R\).
n = length(Dref.Dy.Time); WTr = zeros(4, 4, n); for i = 1:n WTr(:, :, i) = computeReferencePose(Dref.Dy.Data(i), Dref.Ry.Data(i), Dref.Rz.Data(i), Dref.Dh.Data(i,:)); end
As the displacement is perfect, we also measure in simulation the pose of the sample with respect to the granite. From that we can compute the homogeneous transformation matrix \({}^WT_M\).
n = length(Dsm.R.Time); WTm = zeros(4, 4, n); WTm(1:3, 1:3, :) = Dsm.R.Data; WTm(1:3, 4, :) = [Dsm.x.Data' ; Dsm.y.Data' ; Dsm.z.Data']; WTm(4, 4, :) = 1;
Finally, we compute \({}^MT_R\).
MTr = zeros(4, 4, n); for i = 1:n MTr(:, :, i) = [WTm(1:3,1:3,i)', -WTm(1:3,1:3,i)'*WTm(1:3,4,i) ; 0 0 0 1]*WTr(:,:,i); end
Verify that the pose error is small.
| Edx [m] | Edy [m] | Edz [m] | Erx [rad] | Ery [rad] | Erz [rad] | |
|---|---|---|---|---|---|---|
| Error | 1.2e-16 | 3.3e-16 | 2.3e-16 | -6.2e-17 | 1.1e-16 | 2.2e-16 | 
2.4 Conclusion
Indeed, we are able to convert the position error in the frame of the NASS and then compensate these errors with the NASS.
3 Functions
3.1 computeReferencePose
This Matlab function is accessible here.
function [WTr] = computeReferencePose(Dy, Ry, Rz, Dh) % computeReferencePose - Compute the homogeneous transformation matrix corresponding to the wanted pose of the sample % % Syntax: [WTr] = computeReferencePose(Dy, Ry, Rz, Dh) % % Inputs: % - Dy, Ry, Rz, Dh - % % Outputs: % - WTr - %% Translation Stage Rty = [1 0 0 0; 0 1 0 Dy; 0 0 1 0; 0 0 0 1]; %% Tilt Stage - Pure rotating aligned with Ob Rry = [ cos(Ry) 0 sin(Ry) 0; 0 1 0 0; -sin(Ry) 0 cos(Ry) 0; 0 0 0 1]; %% Spindle - Rotation along the Z axis Rrz = [cos(Rz) -sin(Rz) 0 0 ; sin(Rz) cos(Rz) 0 0 ; 0 0 1 0 ; 0 0 0 1 ]; %% Micro-Hexapod Rhx = [1 0 0; 0 cos(Dh(4)) -sin(Dh(4)); 0 sin(Dh(4)) cos(Dh(4))]; Rhy = [ cos(Dh(5)) 0 sin(Dh(5)); 0 1 0; -sin(Dh(5)) 0 cos(Dh(5))]; Rhz = [cos(Dh(6)) -sin(Dh(6)) 0; sin(Dh(6)) cos(Dh(6)) 0; 0 0 1]; Rh = [1 0 0 Dh(1) ; 0 1 0 Dh(2) ; 0 0 1 Dh(3) ; 0 0 0 1 ]; Rh(1:3, 1:3) = Rhx*Rhy*Rhz; %% Total Homogeneous transformation WTr = Rty*Rry*Rrz*Rh; end