function [alpha,output] = cpd_eps(T,U,dU1,dU2,state,options)
%CPD_EPS CPD exact plane search.
%   [alpha,output] = cpd_eps(T,U,dU1,dU2) computes the real plane search
%   parameters alpha that minimize the objective function f(alpha) := ...
%   0.5*frob(T-cpdgen(U+alpha(1)*dU1+alpha(2)*dU2))^2, in which T is an
%   N-th order tensor, U is a cell array containing N factor matrices U{n},
%   and dUi are cell arrays containing N step matrices dUi{n}.
%
%   If options.Scale is true, the objective function f(alpha) := 0.5* ...
%   frob(T-cpdgen(alpha(3)*(U+alpha(1)*dU1+alpha(2)*dU2)))^2 is minimized
%   for all three real parameters alpha(1), alpha(2) and alpha(3).
%
%   Note that for incomplete tensors, the objective function is partially
%   approximated and the alpha is no longer a global minimizer in general.
%
%   The structure output returns additional information:
%
%      output.fval    - The value of the objective function at alpha.
%      output.relgain - The improvement in objective function w.r.t.
%                       alpha = [0 0], relative to the improvement of the
%                       step alpha = [1 0] w.r.t. alpha = [0 0].
%
%   cpd_eps(T,U,dU1,dU2,[],options) may be used to set the following
%   options:
%
%      options.Scale = false - If true, computes an optimal scaling
%                              parameter alpha(3) in conjunction with
%                              optimal plane search parameters alpha(1:2).
%      options.SolverOptions - A struct passed to the the polynomial or
%                              rational minimizer. See polymin2 and ratmin2
%                              for details.
%
%   See also cpd_els, cpd_aels.

%   Authors: Laurent Sorber (Laurent.Sorber@cs.kuleuven.be)
%            Nico Vervliet (Nico.Vervliet@esat.kuleuven.be)
%            Marc Van Barel (Marc.VanBarel@cs.kuleuven.be)
%            Lieven De Lathauwer (Lieven.DeLathauwer@kuleuven-kulak.be)
%
%   References:
%   [1] L. Sorber, I. Domanov, M. Van Barel, L. De Lathauwer, "Exact Line
%       and Plane Search for Tensor Optimization", ESAT-SISTA Internal
%       Report 13-02, KU Leuven, 2013.
%   [2] M. Rajih, P. Comon, R. Harshman, "Enhanced line search: A novel
%       method to accelerate PARAFAC," SIAM J. Matrix Anal. Appl., Vol. 30
%       2008, pp. 1148-1171.
%
% Version History:
% - 2016/01/30   NV      Support for structured tensors

% Check the tensor T.
N = length(U);
R = size(U{1},2);
if length(U) ~= N || length(dU1) ~= N || length(dU2) ~= N
    error('cpd_eps:U','length((d)U) should equal ndims(T).');
end
if any(cellfun('size',U,2) ~= R) || ...
   any(cellfun('size',dU1,2) ~= R) || ...
   any(cellfun('size',dU2,2) ~= R)
    error('cpd_eps:U','size((d)U{n},2) should be the same for all n.');
end

% Check input tensor T
if isnumeric(T)
    isstructured = false;
else 
    type = getstructure(T);
    isstructured = ~any(strcmp(type, {'full', 'incomplete', 'sparse'}));
end

% Check the options structure.
if nargin < 6, options = struct; end
if ~isfield(options,'Scale'), options.Scale = false; end
if ~isfield(options,'SolverOptions'), options.SolverOptions = struct; end
if ~isfield(options.SolverOptions,'TolBal')
    options.SolverOptions.TolBal = 1e-2;
end

% Check the state structure.
if nargin < 5 || ~isstruct(state), state = struct; end
if ~isfield(state,'T2')
    state.T2 = frob(T,'squared');
end
if ~isfield(state,'fval') || ~isfield(state,'K') || ~isfield(state,'K1')
    if ~isfield(state,'fval')
        if isstructured
            state.fval = 0.5*state.T2 - real(inprod(T,U)) + 0.5*frob(U,'squared');
        else 
            D = cpdres(T,U);
            state.fval = 0.5*(D(:)'*D(:));
        end
    end
    if ~isfield(state,'K') || ~isfield(state,'K1')
        state.K1 = 1;
        state.K{state.K1} = mtkrprod(T,U,1);
    end
end

% Apply ELS if one of the two search directions is not ok.
dU1ok = all(cellfun(@(u)all(isfinite(u(:))&u(:)~=0),dU1));
dU2ok = all(cellfun(@(u)all(isfinite(u(:))&u(:)~=0),dU2));
if ~dU2ok && dU1ok
    options.IsReal = 1;
    [alpha,output] = cpd_els(T,U,dU1,state,options);
    alpha = [alpha(1) 0 alpha(2:end)];
    return;
elseif ~dU1ok && dU2ok
    options.IsReal = 1;
    [alpha,output] = cpd_els(T,U,dU2,state,options);
    alpha = [0 alpha];
    return;
elseif ~dU1ok && ~dU2ok
    error('cpd_eps:dU','dU1 and dU2 should be finite and nonzero.');
end

% Generate indices of combinations of terms.
Uall = [U(:).'; dU1(:).'; dU2(:).'];
I = cell(1,N);
[I{:}] = ndgrid(1:3);
I = cell2mat(cellfun(@(x)x(:),I,'UniformOutput',false));
degx = sum(I == 2,2);
degy = sum(I == 3,2);

% Compute the polynomial p = <T,cpdgen(U+x*dU1+y*dU2)>.
p = zeros(N+1);
p(1) = sum(dot(state.K{state.K1},U{state.K1}));
for j = 1:N+1
    for i = 1:N+2-j
        if i == 1 && j == 1, continue; end
        J = I(degy == i-1 & degx == j-1,:);
        for k = 1:size(J,1)
            tmp = conj(sum(mtkrprod(T,Uall(J(k,:)+(0:N-1)*3),0)));
            p(i,j) = p(i,j)+tmp;
        end
    end
end

% Build a cell array of all products (d)U{n}'*(d)U{n}.
UHU = cell(1,N);
for n = 1:N
    tmp = cat(2,U{n},dU1{n},dU2{n});
    UHU{n} = tmp'*tmp;
end

% Compute the polynomial q = ||cpdgen(U+x*dU1+y*dU2)||^2.
q = zeros(2*N+1,2*N+1);
for i = 1:size(I,1)
    for j = 1:size(I,1)
        for n = 1:N
            UHUn = UHU{n};
            if n == 1
                W = UHUn((1:R)+R*(I(i,n)-1),(1:R)+R*(I(j,n)-1));
            else
                W = W.*UHUn((1:R)+R*(I(i,n)-1),(1:R)+R*(I(j,n)-1));
            end
        end
        m = degy(i)+degy(j)+1;
        n = degx(i)+degx(j)+1;
        q(m,n) = q(m,n)+real(sum(W(:)));
    end
end

% Compute objective function f = 0.5*[s^2*q - 2*s*real(p) + ||T||^2].
f = q;
f(1:N+1,1:N+1) = f(1:N+1,1:N+1)-2*real(p);
f = 0.5*f;
f(1) = state.fval(end);

% Solve (scaled) plane search.
if options.Scale
    
    % Compute alpha = argmin -real(p)^2/q.
    [xy,v] = ratmin2(-conv2(real(p),real(p)),q,options.SolverOptions);
    v(state.T2+v < -1e-6*state.T2) = inf;
    [output.fval,idx] = min(v);
    output.fval = 0.5*(state.T2+output.fval);
    alpha = xy(idx,:);
    warn = isempty(alpha);
    if ~warn
        alpha(3) = nthroot(polyval2(real(p),alpha)/polyval2(q,alpha),N);
    end
    
else
    
    % Compute alpha = argmin f.
    [xy,v] = polymin2(f,options.SolverOptions);
    v(v < -1e-6*state.T2) = inf;
    [output.fval,idx] = min(v);
    alpha = xy(idx,:);
    warn = isempty(alpha);
    
end

% Compute improvement compared to a step of alpha = 1.
if isempty(output.fval) || state.fval(end) < 0 || ...
   log10(state.fval(end)) <= log10(state.T2)-16+2.5
    output = rmfield(output,'fval');
    output.relgain = nan;
else
    fval0 = f(1); fval1 = real(sum(f(1,:)));
    output.relgain = (fval0-output.fval)/max(0,fval0-fval1);
end
if warn || (isfield(output,'fval') && output.fval > f(1))
    if options.Scale, alpha = [1 0 1];
    else alpha = [1 0]; end
    output = struct;
end
