import numpy as np
from numpy.fft import fft2, ifft2


def compute_gamma_down(
    s, sigma=0.1, g=None, c=1.0, size=None, trials=100, seed=0, center_kernel=True
):
    if g is None:
        g = max(1, int(round(c * s)))
    if size is None:
        base = max(256, 8 * g)
        size = (base + s - 1) // s * s
    rng = np.random.default_rng(seed)
    kernel = np.ones((g, g), dtype=np.float64) / g**2
    ker_full = np.zeros((size, size), dtype=np.float64)
    i0 = (size - g) // 2
    j0 = (size - g) // 2
    ker_full[i0 : i0 + g, j0 : j0 + g] = kernel
    if center_kernel:
        ker_full = np.fft.ifftshift(ker_full)
    kernel_fft = fft2(ker_full)
    gammas = []
    out_hw = size // s * (size // s)
    for _ in range(trials):
        eta = rng.normal(0.0, sigma, size=(size, size))
        y_c = ifft2(fft2(eta) * kernel_fft)
        assert (
            np.max(np.abs(np.imag(y_c))) < 1e-10
        ), "Significant imaginary part detected"
        y = y_c.real
        y_ds = y[::s, ::s]
        norm_sq = np.sum(y_ds**2, dtype=np.float64)
        gamma = norm_sq / (sigma**2 * out_hw)
        gammas.append(gamma)
    emp_gamma = float(np.mean(gammas))
    std = float(np.std(gammas, ddof=1))
    theo_gamma = float(np.sum(kernel**2))
    return emp_gamma, std, theo_gamma, g, size


ss = [1, 2, 3, 4, 6, 8, 12, 16]
N = 100
c = 1.0
for s in ss:
    emp, std, theo, g, size = compute_gamma_down(s, sigma=0.1, c=c, trials=N, seed=1234)
    ratio = emp / theo if theo > 0 else np.nan
    rel_err = abs(emp - theo) / theo
    se = std / np.sqrt(N)
    print(f"s = {s}  (g(s)={g}, size={size})")
    print(f" Empirical γ↓ = {emp:.6f} ± {std:.6f}  (SE {se:.6f})")
    print(f" Theoretical γ↓ = {theo:.6f}  (box ⇒ 1/g^2 ≈ 1/{g}^2)")
    print(f" Ratio (emp/theo) = {ratio:.6f},  RelErr = {100*rel_err:.2f}%")
    print(f" Within 3 SE? {"Yes"if abs(emp-theo)<=3*se else"No"}\n")
