import torch
import torch.nn.functional as F


def fgsm(model, x, y, upper_limit, lower_limit, mu, std, epsilon: float = 8/255, alpha: float = 16/255):
    """
    Fast Gradient Sign Method (FGSM).

    FGSM is a single-step adversarial attack that perturbs inputs in the direction of
    the sign of the loss gradient with respect to the input. This method is simple but
    effective, and is widely used for both evaluating and training models to improve
    adversarial robustness.

    Reference:
        Goodfellow, I. J., Shlens, J., & Szegedy, C. (2015).
        "Explaining and Harnessing Adversarial Examples."
        arXiv:1412.6572 (https://arxiv.org/abs/1412.6572)

    Args:
        model (torch.nn.Module): The target network.
        x (torch.Tensor): Clean input batch (B, C, H, W).
        y (torch.Tensor): Ground-truth labels.
        upper_limit (torch.Tensor): Per-channel normalized maximum limit for inputs.
        lower_limit (torch.Tensor): Per-channel normalized minimum limit for inputs.
        mu (torch.Tensor): Per-channel normalization mean.
        std (torch.Tensor): Per-channel normalization std.
        epsilon (float): Maximum perturbation magnitude (default: 8/255).
        alpha (float): Gradient step size for perturbation (default: 16/255).

    Returns:
        tuple:
            - torch.Tensor: Final FGSM perturbation (`delta`).
            - torch.Tensor: Gradient tensor from backward pass.
    """
    # Normalize perturbations
    eps = (epsilon / std).view(1, -1, 1, 1)
    alpha = (alpha / std).view(1, -1, 1, 1)

    x = x.clone().detach()
    x.requires_grad = True
    
    output = model(x)
    loss = F.cross_entropy(output, y)
    grad = torch.autograd.grad(loss, x)[0].detach()
    

    # Compute perturbation based on sign of gradient
    delta = alpha * torch.sign(grad)
    delta = torch.clamp(delta, -eps, +eps)
    torch.clamp(delta, lower_limit - x, upper_limit - x)
    delta = delta.detach()
    
    return delta, grad