]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
PM: hibernate: Fix crash when freeing invalid crypto compressor
authorMalaya Kumar Rout <mrout@redhat.com>
Tue, 30 Dec 2025 11:56:13 +0000 (17:26 +0530)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Sat, 17 Jan 2026 15:35:14 +0000 (16:35 +0100)
commit 7966cf0ebe32c981bfa3db252cb5fc3bb1bf2e77 upstream.

When crypto_alloc_acomp() fails, it returns an ERR_PTR value, not NULL.

The cleanup code in save_compressed_image() and load_compressed_image()
unconditionally calls crypto_free_acomp() without checking for ERR_PTR,
which causes crypto_acomp_tfm() to dereference an invalid pointer and
crash the kernel.

This can be triggered when the compression algorithm is unavailable
(e.g., CONFIG_CRYPTO_LZO not enabled).

Fix by adding IS_ERR_OR_NULL() checks before calling crypto_free_acomp()
and acomp_request_free(), similar to the existing kthread_stop() check.

Fixes: b03d542c3c95 ("PM: hibernate: Use crypto_acomp interface")
Signed-off-by: Malaya Kumar Rout <mrout@redhat.com>
Cc: 6.15+ <stable@vger.kernel.org> # 6.15+
[ rjw: Added 2 empty code lines ]
Link: https://patch.msgid.link/20251230115613.64080-1-mrout@redhat.com
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
kernel/power/swap.c

index 70ae21f7370d4c88263291b2887e80fae32cc191..f910a250ccddb98db49ee305a7ef8287f239e2e5 100644 (file)
@@ -897,8 +897,11 @@ out_clean:
                for (thr = 0; thr < nr_threads; thr++) {
                        if (data[thr].thr)
                                kthread_stop(data[thr].thr);
-                       acomp_request_free(data[thr].cr);
-                       crypto_free_acomp(data[thr].cc);
+                       if (data[thr].cr)
+                               acomp_request_free(data[thr].cr);
+
+                       if (!IS_ERR_OR_NULL(data[thr].cc))
+                               crypto_free_acomp(data[thr].cc);
                }
                vfree(data);
        }
@@ -1519,8 +1522,11 @@ out_clean:
                for (thr = 0; thr < nr_threads; thr++) {
                        if (data[thr].thr)
                                kthread_stop(data[thr].thr);
-                       acomp_request_free(data[thr].cr);
-                       crypto_free_acomp(data[thr].cc);
+                       if (data[thr].cr)
+                               acomp_request_free(data[thr].cr);
+
+                       if (!IS_ERR_OR_NULL(data[thr].cc))
+                               crypto_free_acomp(data[thr].cc);
                }
                vfree(data);
        }