function [z] = computeFGSMAttack(obj,x,t,options,epsilon,varargin)
% computeFGSMAttack - Compute a Fast Gradient Sign (FGSM) adversarial attack.
%
% Syntax:
%    [z] = computeFGSMAttack(obj,x,y,options,epsilon)
%
% Inputs:
%    obj - neural network
%    x - input value
%    t - target value
%    options - evaluation parameters (see neuralNetwork.evaluate)
%    epsilon - perturbation radius
%    lossDer - derivative of loss function
%
% Outputs:
%    z - altered input
%    
% References:
%    [1] I. J. Goodfellow, J. Shlens, and C. Szegedy. "Explaining and 
%        Harnessing Adversarial Examples." In: 3rd International Conference
%        on Learning Representations ICLR. San Diego, CA, USA, 2015.
%
% Other m-files required: none
% Subfunctions: none
% MAT-files required: none
%
% See also: neuralNetwork

% Authors:       Lukas Koller
% Written:       21-June-2023
% Last update:   11-July-2023 (fixed bug in computation)
%                12-July-2023 (corrected sign in computation)
%                07-September-2023 (batch-wise computation)
% Last revision: ---    

% ------------------------------ BEGIN CODE -------------------------------

% parse input
if nargin < 5
    throw(CORAerror('CORA:notEnoughInputArgs',4))
elseif nargin > 6
    throw(CORAerror('CORA:tooManyInputArgs',6))
end

% validate parameters
[lossDer] = setDefaultValues({@(t,y) softmax(y) - t},varargin);

% validate input
inputArgsCheck({ ...
    {obj,'att','neuralNetwork'}; ...
    {x,'att','numeric','array'}; ...
    {t,'att','numeric','array'}; ... 
    {options,'att','struct'}; ... 
    {epsilon,'att','numeric'};
})

% compute sensitivity and forward propagation
[S,y] = obj.calcSensitivity(x,options,false);
% compute gradient
grad = reshape(pagemtimes(S,'transpose', ...
    permute(lossDer(t,y),[1 3 2]),'none'),size(x));
% compute modified input; linear estimate on worst-case perturbation
z = x + epsilon*sign(grad);
end

% ------------------------------ END OF CODE ------------------------------
