--- /dev/null
+/* SPDX-License-Identifier: GPL-2.0 */
+#ifndef _ASM_X86_IOBITMAP_H
+#define _ASM_X86_IOBITMAP_H
+
+#include <asm/processor.h>
+
+struct io_bitmap {
+       /* The maximum number of bytes to copy so all zero bits are covered */
+       unsigned int    max;
+       unsigned long   bitmap[IO_BITMAP_LONGS];
+};
+
+#endif
 
 /* Forward declaration, a strange C thing */
 struct task_struct;
 struct mm_struct;
+struct io_bitmap;
 struct vm86;
 
 #include <asm/math_emu.h>
        struct vm86             *vm86;
 #endif
        /* IO permissions: */
-       unsigned long           *io_bitmap_ptr;
+       struct io_bitmap        *io_bitmap;
        unsigned long           iopl;
-       /* Max allowed port in the bitmap, in bytes: */
-       unsigned                io_bitmap_max;
 
        mm_segment_t            addr_limit;
 
 #define INIT_THREAD  {                                                   \
        .sp0                    = TOP_OF_INIT_STACK,                      \
        .sysenter_cs            = __KERNEL_CS,                            \
-       .io_bitmap_ptr          = NULL,                                   \
        .addr_limit             = KERNEL_DS,                              \
 }
 
 
 #include <linux/sched.h>
 #include <linux/slab.h>
 
+#include <asm/io_bitmap.h>
 #include <asm/desc.h>
 
 /*
        unsigned int i, max_long, bytes, bytes_updated;
        struct thread_struct *t = ¤t->thread;
        struct tss_struct *tss;
-       unsigned long *bitmap;
+       struct io_bitmap *iobm;
 
        if ((from + num <= from) || (from + num > IO_BITMAP_BITS))
                return -EINVAL;
         * IO bitmap up. ioperm() is much less timing critical than clone(),
         * this is why we delay this operation until now:
         */
-       bitmap = t->io_bitmap_ptr;
-       if (!bitmap) {
+       iobm = t->io_bitmap;
+       if (!iobm) {
                /* No point to allocate a bitmap just to clear permissions */
                if (!turn_on)
                        return 0;
-               bitmap = kmalloc(IO_BITMAP_BYTES, GFP_KERNEL);
-               if (!bitmap)
+               iobm = kmalloc(sizeof(*iobm), GFP_KERNEL);
+               if (!iobm)
                        return -ENOMEM;
 
-               memset(bitmap, 0xff, IO_BITMAP_BYTES);
+               memset(iobm->bitmap, 0xff, sizeof(iobm->bitmap));
        }
 
        /*
         */
        preempt_disable();
        if (turn_on)
-               bitmap_clear(bitmap, from, num);
+               bitmap_clear(iobm->bitmap, from, num);
        else
-               bitmap_set(bitmap, from, num);
+               bitmap_set(iobm->bitmap, from, num);
 
        /*
         * Search for a (possibly new) maximum. This is simple and stupid,
         */
        max_long = 0;
        for (i = 0; i < IO_BITMAP_LONGS; i++) {
-               if (bitmap[i] != ~0UL)
+               if (iobm->bitmap[i] != ~0UL)
                        max_long = i;
        }
 
        bytes = (max_long + 1) * sizeof(unsigned long);
-       bytes_updated = max(bytes, t->io_bitmap_max);
+       bytes_updated = max(bytes, t->io_bitmap->max);
 
        /* Update the thread data */
-       t->io_bitmap_max = bytes;
+       iobm->max = bytes;
        /*
         * Store the bitmap pointer (might be the same if the task already
         * head one). Set the TIF flag, just in case this is the first
         * invocation.
         */
-       t->io_bitmap_ptr = bitmap;
+       t->io_bitmap = iobm;
        set_thread_flag(TIF_IO_BITMAP);
 
        /* Update the TSS */
        tss = this_cpu_ptr(&cpu_tss_rw);
-       memcpy(tss->io_bitmap.bitmap, t->io_bitmap_ptr, bytes_updated);
+       memcpy(tss->io_bitmap.bitmap, iobm->bitmap, bytes_updated);
        /* Store the new end of the zero bits */
        tss->io_bitmap.prev_max = bytes;
        /* Make the bitmap base in the TSS valid */
 
 #include <asm/desc.h>
 #include <asm/prctl.h>
 #include <asm/spec-ctrl.h>
+#include <asm/io_bitmap.h>
 #include <asm/proto.h>
 
 #include "process.h"
 void exit_thread(struct task_struct *tsk)
 {
        struct thread_struct *t = &tsk->thread;
-       unsigned long *bp = t->io_bitmap_ptr;
+       struct io_bitmap *iobm = t->io_bitmap;
        struct fpu *fpu = &t->fpu;
        struct tss_struct *tss;
 
-       if (bp) {
+       if (iobm) {
                preempt_disable();
                tss = this_cpu_ptr(&cpu_tss_rw);
 
-               t->io_bitmap_ptr = NULL;
-               t->io_bitmap_max = 0;
+               t->io_bitmap = NULL;
                clear_thread_flag(TIF_IO_BITMAP);
                /* Invalidate the io bitmap base in the TSS */
                tss->x86_tss.io_bitmap_base = IO_BITMAP_OFFSET_INVALID;
                preempt_enable();
-               kfree(bp);
+               kfree(iobm);
        }
 
        free_vm86(t);
 
 static inline int copy_io_bitmap(struct task_struct *tsk)
 {
+       struct io_bitmap *iobm = current->thread.io_bitmap;
+
        if (likely(!test_tsk_thread_flag(current, TIF_IO_BITMAP)))
                return 0;
 
-       tsk->thread.io_bitmap_ptr = kmemdup(current->thread.io_bitmap_ptr,
-                                           IO_BITMAP_BYTES, GFP_KERNEL);
-       if (!tsk->thread.io_bitmap_ptr) {
-               tsk->thread.io_bitmap_max = 0;
+       tsk->thread.io_bitmap = kmemdup(iobm, sizeof(*iobm), GFP_KERNEL);
+
+       if (!tsk->thread.io_bitmap)
                return -ENOMEM;
-       }
+
        set_tsk_thread_flag(tsk, TIF_IO_BITMAP);
        return 0;
 }
 
 static inline void free_io_bitmap(struct task_struct *tsk)
 {
-       if (tsk->thread.io_bitmap_ptr) {
-               kfree(tsk->thread.io_bitmap_ptr);
-               tsk->thread.io_bitmap_ptr = NULL;
-               tsk->thread.io_bitmap_max = 0;
+       if (tsk->thread.io_bitmap) {
+               kfree(tsk->thread.io_bitmap);
+               tsk->thread.io_bitmap = NULL;
        }
 }
 
        frame->bp = 0;
        frame->ret_addr = (unsigned long) ret_from_fork;
        p->thread.sp = (unsigned long) fork_frame;
-       p->thread.io_bitmap_ptr = NULL;
+       p->thread.io_bitmap = NULL;
        memset(p->thread.ptrace_bps, 0, sizeof(p->thread.ptrace_bps));
 
 #ifdef CONFIG_X86_64
        struct tss_struct *tss = this_cpu_ptr(&cpu_tss_rw);
 
        if (tifn & _TIF_IO_BITMAP) {
+               struct io_bitmap *iobm = next->io_bitmap;
+
                /*
                 * Copy at least the size of the incoming tasks bitmap
                 * which covers the last permitted I/O port.
                 * bits permitted, then the copy needs to cover those as
                 * well so they get turned off.
                 */
-               memcpy(tss->io_bitmap.bitmap, next->io_bitmap_ptr,
-                      max(tss->io_bitmap.prev_max, next->io_bitmap_max));
+               memcpy(tss->io_bitmap.bitmap, next->io_bitmap->bitmap,
+                      max(tss->io_bitmap.prev_max, next->io_bitmap->max));
 
                /* Store the new max and set io_bitmap_base valid */
-               tss->io_bitmap.prev_max = next->io_bitmap_max;
+               tss->io_bitmap.prev_max = next->io_bitmap->max;
                tss->x86_tss.io_bitmap_base = IO_BITMAP_OFFSET_VALID;
 
                /*
 
 #include <asm/traps.h>
 #include <asm/syscall.h>
 #include <asm/fsgsbase.h>
+#include <asm/io_bitmap.h>
 
 #include "tls.h"
 
 static int ioperm_active(struct task_struct *target,
                         const struct user_regset *regset)
 {
-       return DIV_ROUND_UP(target->thread.io_bitmap_max, regset->size);
+       struct io_bitmap *iobm = target->thread.io_bitmap;
+
+       return iobm ? DIV_ROUND_UP(iobm->max, regset->size) : 0;
 }
 
 static int ioperm_get(struct task_struct *target,
                      unsigned int pos, unsigned int count,
                      void *kbuf, void __user *ubuf)
 {
-       if (!target->thread.io_bitmap_ptr)
+       struct io_bitmap *iobm = target->thread.io_bitmap;
+
+       if (!iobm)
                return -ENXIO;
 
        return user_regset_copyout(&pos, &count, &kbuf, &ubuf,
-                                  target->thread.io_bitmap_ptr,
-                                  0, IO_BITMAP_BYTES);
+                                  iobm->bitmap, 0, IO_BITMAP_BYTES);
 }
 
 /*