Artificial Intelligence Blog

Giải thích và code paper Guided Image Filtering

Xin chào các bạn,

Trong bài post này, mình sẽ giới thiệu một thuật toán image filtering mới tên là Guided Image Filtering. Đây là thuật toán làm mịn ảnh tốt nhất tính đến thời điểm bài post này được viết. Phương pháp này được đề xuất bởi Kaiming He (cũng là cha đẻ của mô hình ResNet), Jian Sun, và Xiaoou Tang tại hội nghị ECCV 2010. Cũng đã thấm thoát 14 năm trôi qua, đã có nhiều biến thể của thuật toán này như Fast Guided Image Filtering, Deep Guided Image filtering, … được đề xuất, tuy nhiên chúng đều dựa trên nền của giải thuật gốc. Đây là một giải thuật đạt được cùng một lúc cả hai tiêu chí: nhanh hơn và tốt hơn.

1. Tại sao Guided Image Filtering được ra đời ?

Thuật toán Guided Image Filtering ra đời với mục tiêu giải quyết một số thách thức quan trọng trong xử lý ảnh, đặc biệt là trong việc làm mịn ảnh. Trước đó, các phương pháp truyền thống như bộ lọc trung bình, Gaussian Blur, bilateral filter lọc nhiễu và làm mịn ảnh nhưng đồng thời cũng làm mất các chi tiết trong ảnh. Để khắc phục nhược điểm đó của các bộ lọc trên, bộ lọc Guided filter tích hợp thêm thông tin của ảnh gốc để có thể “hướng dẫn” giải thuật lọc đi đúng hướng và giữ được nhiều chi tiết nhất có thể.

Ngoài ứng dụng làm mịn, Guided filter còn có thể được ứng dụng cho colorisation, image matting, multi-scale decomposition, và haze removal.

Ứng dụng của guided filter trong image matting. Nguồn: claude_ssim.log

2. Giải thuật Guided Image Filtering

Minh họa behaviour của guided filter Nguồn: ResearchGate

Guided filter hoạt động giống với tích chập ở chỗ nó cũng lướt qua từng patch của tấm ảnh.

Giả sử, \(q\) là output, \(I\) là guidance image, \(I\) là ảnh đầu vào. \(w_k\) là một cửa sổ (kernel).

Về tổng quát, Guided filter có công thức như sau:

\[q_i = a_kI_i + b_k\]

Với \((a_k, b_k)\) là trọng số tuyến tính (kiểu \(y = ax + b\)) trong kernel \(w_k\). Tham số \(r\) là bán kính của kernel. Kiểu công thức tuyến tính này sẽ giúp cho ảnh output giữ lại được các cạnh và chi tiết có tần số cao trong ảnh được filtered bởi vì \(\nabla q = a\nabla I\). Công việc của chúng ta là tìm ra hệ số \(a_k \text{ và } b_k\) cho mỗi kernel. Việc tìm ra hai hệ số này dựa vào hàm loss sau:

\[E(a_k, b_k) = \sum_{i \in w_k}{((a_kI_i + b_k - p_i)^2 + \varepsilon a_k^2)}\]

Hàm loss này có mục đích tổi thiểu hóa khoảng cách L2 của output với ảnh guidance. Hệ số \(\varepsilon\) ở đây được tác giả thêm vào để tránh cho \(a_k\) trở nên quá lớn.

Chúng ta có thể dùng nhiều phương pháp để có thể tìm ra 2 hệ số này bằng nhiều phương pháp như Gradient Descent, Newton, … Tuy nhiên, tác giả cũng đã cung cấp công thức closed form để tìm ra \(a_k, b_k\) trong paper của mình. Cụ thể, với mỗi kernel 2 thông số được tính như sau:

\[a_k = \frac{\frac{1}{|w|}\sum_{i \in w_k}{I_ip_i - \mu _k \overline{p_k}}}{\sigma _k ^2 + \epsilon}\] \[b_k = \overline{p_k} - a_k \mu _k\]

Với \(\mu_k, \sigma_k\) là giá trị mean và variance của ảnh input trong kernel \(w_k\), \(\vert w \vert\) là tổng số lượng pixels trong kernel, \(\overline{p_k}\) là giá trị trung bình của các pixels của ảnh guidance trong kernel \(w_k\).

Tuy nhiên, có một vấn đề là một pixel bị overlapped bởi nhiều kernel giống như tích chập vậy. Để giải quyết vấn đề này, chúng ta sẽ lấy giá trị trung bình của chúng và quan hệ của input và output sẽ chỉ còn \(\nabla q \approx a \nabla I\). Tuy nhiên, như vậy cũng đã đủ tốt để giữ lại các chi tiết có tần số cao trong tấm ảnh.

Cuối cùng, ta có công thức tổng quát cho guided filter như sau:

\[q_i = \frac{1}{|w|}\sum_{i \in w_k}{a_kI_i + b_k} = \overline{a_i}I_i + \overline{b_i}\] \[E(a_k, b_k) = \sum_{i \in w_k}{((a_kI_i + b_k - p_i)^2 + \varepsilon a_k^2)}\] \[E(a_k, b_k) = \sum_{i \in w_k}{(a_k^2 I_i^2 + b_k^2 + p_i^2 + 2a_k b_k I_i - 2 b_k p_i - 2 a_k I_i p_i + \varepsilon a_k^2)}\] \[E(a_k, b_k) = \sum_{i \in w_k}{(a_k^2 I_i^2 + b_k^2 + p_i^2 + 2a_k b_k I_i - 2 b_k p_i - 2 a_k I_i p_i + \varepsilon a_k^2)}\]

Đặt \(\alpha = \sum_{i \in w_k}I_i^2\), \(\beta = \sum_{i \in w_k} I_i p_i\), \(\gamma = \sum_{i \in w_k} p_i\), ta có biến đổi như sau:

\[E(a_k, b_k) = a_k^2 \alpha + \vert \omega \vert b_k^2 + \sum_{i \in w_k} p_i^2 + 2 a_k b_k \sum_{i \in w_k}I_i - 2 a_k \beta - 2b_k \gamma + \varepsilon a_k^2\]

Lấy đạo hàm từng phần để biết cực tiểu của hàm theo \(a_k\) và \(b_k\):

\[\begin{aligned} \frac{\partial E}{\partial a_k} &= 2a_k \alpha + 2 b_k \sum_{i \in w_k} I_i - 2 \beta + 2 \varepsilon a_k = 0 \\ &= a_k(\alpha + \varepsilon) + b_k \sum_{i \in w_k} I_i - \beta = 0 \\ &= a_k(\alpha + \varepsilon) + b_k \sum_{i \in w_k} I_i = \beta \end{aligned}\] \[\begin{aligned} \frac{\partial E}{\partial b_k} &= 2 b_k \vert \omega \vert + 2a_k \sum_{i \in w_k} I_i - 2 \gamma = 0\\ &= b_k \vert \omega \vert + a_k \sum_{i \in w_k} I_i = \gamma \end{aligned}\]

Từ 2 phương trình trên, ta sẽ kết hợp giải phương trình để tìm ra \(a_k^*\) và \(b_k^*\).

\[b_k = \frac{\gamma - a_k \sum_{i \in w_k}I_i}{\vert \omega \vert}\]

Thay vào, ta được

\[a_k (\alpha + \varepsilon) - (\frac{\gamma - a_k \sum_{i \in w_k}I_i}{\vert \omega \vert}) \sum_{i \in w_k} I_i = \beta\] \[a_k (\alpha + \varepsilon - \frac{\sum_{i \in w_k} I_i\sum_{i \in w_k} I_i}{\vert \omega \vert}) = \beta + \frac{\gamma \sum_{i \in w_k} I_i}{\vert \omega \vert}\]

Ta chia 2 vế cho \(\vert \omega \vert\) và bắt đầu biến đổi. Đầu tiên, ta sẽ biến đổi vế phải trước.

\[\frac{\beta}{\vert \omega \vert} + \frac{\gamma \sum_{i \in w_k} I_i}{\vert \omega \vert ^2} = \frac{\sum_{i \in w_k} I_i p_i}{\vert \omega \vert} + \frac{\sum_{i \in w_k} p_i \sum_{i \in w_k} I_i}{\vert \omega \vert ^ 2}\]

Với vế trái, sẽ dễ dàng hơn nếu ta biến đổi ngược:

\[\begin{aligned} \sigma_I^2 &= \frac{\sum_{i \in w_k} (I_i - \overline{I})^2}{\vert \omega \vert} \\ &= \frac{\sum_{i \in w_k} (I_i^2 + \overline{I}^2 - 2I_i\overline{I})}{\vert \omega \vert} \\ &= \frac{\sum_{i \in w_k} I_i^2}{\vert \omega \vert} + \frac{\sum_{i \in w_k} (\overline{I}^2 - 2I_i \overline{I})}{\vert \omega \vert} \\ &= \frac{\sum_{i \in w_k} I_i^2}{\vert \omega \vert} + \frac{\vert \omega \vert \overline{I}^2 - 2 \overline{I} \sum_{i \in w_k} I_i}{\vert \omega \vert} \\ &= \frac{\sum_{i \in w_k} I_i^2}{\vert \omega \vert} + \overline{I}^2 - 2\overline{I}^2 \\ &= \frac{\sum_{i \in w_k} I_i^2}{\vert \omega \vert} - \overline{I}^2 \\ &= \frac{\sum_{i \in w_k} I_i^2}{\vert \omega \vert} - \frac{\sum_{i \in w_k} I_i \sum_{i \in w_k} I_i}{\vert \omega \vert ^ 2} \\ &= \frac{\alpha}{\vert \omega \vert} - \frac{\sum_{i \in w_k} I_i \sum_{i \in w_k} I_i}{\vert \omega \vert ^ 2} \quad = \text{vế trái (đpcm)} \end{aligned}\] \[\begin{aligned} \rightarrow a_k &= \frac{\frac{\sum_{i \in w_k} I_i p_i}{\vert \omega \vert} + \frac{\sum_{i \in w_k} p_i \sum_{i \in w_k} I_i}{\vert \omega \vert ^ 2}}{\sigma_I^2 + \varepsilon} \\ &= \frac{\frac{1}{|w|}\sum_{i \in w_k}{I_ip_i - \mu _k \overline{p_k}}}{\sigma _I ^2 + \epsilon} \\ \end{aligned}\] \[\begin{aligned} b_k &= \frac{\gamma - a_k \sum_{i \in w_k} I_i}{\vert \omega \vert} \\ &= \frac{\sum_{i \in w_k} p_i - a_k \sum_{i \in w_k} I_i}{\vert \omega \vert} \\ &= \overline{p_k} - a_k \mu _k \end{aligned}\]

3. Implementation


def guided_filter(p, I, r, e):
    '''
    Guided filter implemented by Mikyx-1 
    21/04/2024
    Args:

        p (torch.Tensor): Guidance Image
        I (torch.Tensor): Input Image
        r (int): Kernel Size (!= radius in the paper)
        e (float): epsilon to prevent underflow

    Returns:
        Output (torch.Tensor): Output image in format BCHW
    '''
    kernel = torch.ones((I.shape[0], 1, r, r))/(r**2)
    padding = r//2

    assert p.shape == I.shape, "Shapes do not match, retry!"

    meanI = F.conv2d(I, kernel, padding=padding, stride=1)
    meanP = F.conv2d(p, kernel, padding=padding, stride=1)
    corrI = F.conv2d(I*I, kernel, padding=padding, stride = 1)
    corrIp = F.conv2d(I*p, kernel, padding=padding, stride = 1)

    sigmaK = corrI - meanI*meanI
    a = (corrIp - meanI*meanP)/(sigmaK + e)
    b = meanP - a*meanI 
    
    meanA = F.conv2d(a, kernel, padding=padding, stride = 1)
    meanB = F.conv2d(b, kernel, padding=padding, stride = 1)

    output = meanA*I + meanB
    return output

4. Kết luận

Bộ lọc guided filter là một bộ lọc có độ phức tạp \(O(N)\) nên có tốc độ gần như bằng các bộ lọc Gaussian, mean. Ngoài ra, bộ lọc guided filter là một bộ lọc đa năng và có thể dùng trong rất nhiều trường hợp khác nhau. Ví dụ, bộ lọc guided filter cũng sẽ rất hữu dụng trong việc làm mịn các tín hiệu 1D (có thể cạnh tranh với các bộ lọc 1 Euro, Kalman filter, …).

Trong bài viết này, mình đã trình bày cách hoạt động, chứng minh bộ lọc guided filter và cũng như là code from scratch. Hy vong các bạn cảm thấy hữu ích. Peace.

References

1. Guided Image filtering - Kaiming He, Jian Sun, Xiaoou Tang