[926] | 1 | %init_intrinsic_param |
---|
| 2 | % |
---|
| 3 | %Initialization of the intrinsic parameters. |
---|
| 4 | %Runs as a script. |
---|
| 5 | % |
---|
| 6 | %INPUT: x_1,x_2,x_3,...: Feature locations on the images |
---|
| 7 | % X_1,X_2,X_3,...: Corresponding grid coordinates |
---|
| 8 | % |
---|
| 9 | %OUTPUT: fc: Camera focal length |
---|
| 10 | % cc: Principal point coordinates |
---|
[999] | 11 | % kc: Distortion coefficients |
---|
[926] | 12 | % alpha_c: skew coefficient |
---|
| 13 | % KK: The camera matrix (containing fc, cc and alpha_c) |
---|
| 14 | % |
---|
| 15 | %Method: Computes the planar homographies H_1, H_2, H_3, ... and computes |
---|
| 16 | % the focal length fc from orthogonal vanishing points constraint. |
---|
| 17 | % The principal point cc is assumed at the center of the image. |
---|
| 18 | % Assumes no image distortion (kc = [0;0;0;0]) |
---|
| 19 | % |
---|
| 20 | %Note: The row vector active_images consists of zeros and ones. To deactivate an image, set the |
---|
| 21 | % corresponding entry in the active_images vector to zero. |
---|
| 22 | % |
---|
| 23 | % |
---|
| 24 | %Important function called within that program: |
---|
| 25 | % |
---|
| 26 | %compute_homography.m: Computes the planar homography between points on the grid in 3D, and the image plane. |
---|
| 27 | % |
---|
| 28 | % |
---|
| 29 | %VERY IMPORTANT: This function works only with 2D rigs. |
---|
| 30 | %In the future, a more general function will be there (working with 3D rigs as well). |
---|
| 31 | |
---|
| 32 | |
---|
| 33 | if ~exist('two_focals_init'), |
---|
| 34 | two_focals_init = 0; |
---|
| 35 | end; |
---|
| 36 | |
---|
| 37 | if ~exist('est_aspect_ratio'), |
---|
| 38 | est_aspect_ratio = 1; |
---|
| 39 | end; |
---|
| 40 | |
---|
| 41 | check_active_images; |
---|
| 42 | |
---|
| 43 | if ~exist(['x_' num2str(ind_active(1)) ]), |
---|
| 44 | click_calib; |
---|
| 45 | end; |
---|
| 46 | |
---|
| 47 | |
---|
| 48 | fprintf(1,'\nInitialization of the intrinsic parameters - Number of images: %d\n',length(ind_active)); |
---|
| 49 | |
---|
| 50 | |
---|
| 51 | % Initialize the homographies: |
---|
| 52 | |
---|
| 53 | for kk = 1:n_ima, |
---|
| 54 | eval(['x_kk = x_' num2str(kk) ';']); |
---|
| 55 | eval(['X_kk = X_' num2str(kk) ';']); |
---|
| 56 | if (isnan(x_kk(1,1))), |
---|
| 57 | if active_images(kk), |
---|
| 58 | fprintf(1,'WARNING: Cannot calibrate with image %d. Need to extract grid corners first.\n',kk) |
---|
| 59 | fprintf(1,' Set active_images(%d)=1; and run Extract grid corners.\n',kk) |
---|
| 60 | end; |
---|
| 61 | active_images(kk) = 0; |
---|
| 62 | end; |
---|
| 63 | if active_images(kk), |
---|
| 64 | eval(['H_' num2str(kk) ' = compute_homography(x_kk,X_kk(1:2,:));']); |
---|
| 65 | else |
---|
| 66 | eval(['H_' num2str(kk) ' = NaN*ones(3,3);']); |
---|
| 67 | end; |
---|
| 68 | end; |
---|
| 69 | |
---|
| 70 | check_active_images; |
---|
| 71 | |
---|
| 72 | % initial guess for principal point and distortion: |
---|
| 73 | |
---|
| 74 | if ~exist('nx'), [ny,nx] = size(I); end; |
---|
| 75 | |
---|
| 76 | c_init = [nx;ny]/2 - 0.5; % initialize at the center of the image |
---|
| 77 | k_init = [0;0;0;0;0]; % initialize to zero (no distortion) |
---|
| 78 | |
---|
| 79 | |
---|
| 80 | |
---|
| 81 | % Compute explicitely the focal length using all the (mutually orthogonal) vanishing points |
---|
| 82 | % note: The vanihing points are hidden in the planar collineations H_kk |
---|
| 83 | |
---|
| 84 | A = []; |
---|
| 85 | b = []; |
---|
| 86 | |
---|
| 87 | % matrix that subtract the principal point: |
---|
| 88 | Sub_cc = [1 0 -c_init(1);0 1 -c_init(2);0 0 1]; |
---|
| 89 | |
---|
| 90 | for kk=1:n_ima, |
---|
[999] | 91 | |
---|
[926] | 92 | if active_images(kk), |
---|
[999] | 93 | |
---|
[926] | 94 | eval(['Hkk = H_' num2str(kk) ';']); |
---|
[999] | 95 | |
---|
| 96 | Hkk = Sub_cc * Hkk; |
---|
| 97 | |
---|
[926] | 98 | % Extract vanishing points (direct and diagonals): |
---|
[999] | 99 | |
---|
[926] | 100 | V_hori_pix = Hkk(:,1); |
---|
| 101 | V_vert_pix = Hkk(:,2); |
---|
| 102 | V_diag1_pix = (Hkk(:,1)+Hkk(:,2))/2; |
---|
| 103 | V_diag2_pix = (Hkk(:,1)-Hkk(:,2))/2; |
---|
[999] | 104 | |
---|
[926] | 105 | V_hori_pix = V_hori_pix/norm(V_hori_pix); |
---|
| 106 | V_vert_pix = V_vert_pix/norm(V_vert_pix); |
---|
| 107 | V_diag1_pix = V_diag1_pix/norm(V_diag1_pix); |
---|
| 108 | V_diag2_pix = V_diag2_pix/norm(V_diag2_pix); |
---|
[999] | 109 | |
---|
[926] | 110 | a1 = V_hori_pix(1); |
---|
| 111 | b1 = V_hori_pix(2); |
---|
| 112 | c1 = V_hori_pix(3); |
---|
[999] | 113 | |
---|
[926] | 114 | a2 = V_vert_pix(1); |
---|
| 115 | b2 = V_vert_pix(2); |
---|
| 116 | c2 = V_vert_pix(3); |
---|
[999] | 117 | |
---|
[926] | 118 | a3 = V_diag1_pix(1); |
---|
| 119 | b3 = V_diag1_pix(2); |
---|
| 120 | c3 = V_diag1_pix(3); |
---|
[999] | 121 | |
---|
[926] | 122 | a4 = V_diag2_pix(1); |
---|
| 123 | b4 = V_diag2_pix(2); |
---|
| 124 | c4 = V_diag2_pix(3); |
---|
[999] | 125 | |
---|
[926] | 126 | A_kk = [a1*a2 b1*b2; |
---|
| 127 | a3*a4 b3*b4]; |
---|
[999] | 128 | |
---|
[926] | 129 | b_kk = -[c1*c2;c3*c4]; |
---|
[999] | 130 | |
---|
| 131 | |
---|
[926] | 132 | A = [A;A_kk]; |
---|
| 133 | b = [b;b_kk]; |
---|
[999] | 134 | |
---|
[926] | 135 | end; |
---|
[999] | 136 | |
---|
[926] | 137 | end; |
---|
| 138 | |
---|
| 139 | |
---|
| 140 | % use all the vanishing points to estimate focal length: |
---|
| 141 | |
---|
| 142 | |
---|
| 143 | % Select the model for the focal. (solution to Gerd's problem) |
---|
| 144 | if ~two_focals_init |
---|
| 145 | if b'*(sum(A')') < 0, |
---|
| 146 | two_focals_init = 1; |
---|
| 147 | end; |
---|
| 148 | end; |
---|
| 149 | |
---|
| 150 | |
---|
[999] | 151 | |
---|
[926] | 152 | if two_focals_init |
---|
| 153 | % Use a two focals estimate: |
---|
| 154 | f_init = sqrt(abs(1./(inv(A'*A)*A'*b))); % if using a two-focal model for initial guess |
---|
| 155 | else |
---|
| 156 | % Use a single focal estimate: |
---|
| 157 | f_init = sqrt(b'*(sum(A')') / (b'*b)) * ones(2,1); % if single focal length model is used |
---|
| 158 | end; |
---|
| 159 | |
---|
| 160 | |
---|
| 161 | if ~est_aspect_ratio, |
---|
| 162 | f_init(1) = (f_init(1)+f_init(2))/2; |
---|
| 163 | f_init(2) = f_init(1); |
---|
| 164 | end; |
---|
| 165 | |
---|
| 166 | alpha_init = 0; |
---|
| 167 | |
---|
| 168 | %f_init = sqrt(b'*(sum(A')') / (b'*b)) * ones(2,1); % if single focal length model is used |
---|
| 169 | |
---|
| 170 | |
---|
| 171 | % Global calibration matrix (initial guess): |
---|
| 172 | |
---|
| 173 | KK = [f_init(1) alpha_init*f_init(1) c_init(1);0 f_init(2) c_init(2); 0 0 1]; |
---|
| 174 | inv_KK = inv(KK); |
---|
| 175 | |
---|
| 176 | |
---|
| 177 | cc = c_init; |
---|
| 178 | fc = f_init; |
---|
| 179 | kc = k_init; |
---|
| 180 | alpha_c = alpha_init; |
---|
| 181 | |
---|
| 182 | |
---|
| 183 | fprintf(1,'\n\nCalibration parameters after initialization:\n\n'); |
---|
| 184 | fprintf(1,'Focal Length: fc = [ %3.5f %3.5f ]\n',fc); |
---|
| 185 | fprintf(1,'Principal point: cc = [ %3.5f %3.5f ]\n',cc); |
---|
| 186 | fprintf(1,'Skew: alpha_c = [ %3.5f ] => angle of pixel = %3.5f degrees\n',alpha_c,90 - atan(alpha_c)*180/pi); |
---|
[999] | 187 | fprintf(1,'Distortion: kc = [ %3.5f %3.5f %3.5f %3.5f %5.5f ]\n',kc); |
---|