force
     Force the huge option on for all - very useful for testing;
 
+Shmem can also use "multi-size THP" (mTHP) by adding a new sysfs knob to
+control mTHP allocation:
+'/sys/kernel/mm/transparent_hugepage/hugepages-<size>kB/shmem_enabled',
+and its value for each mTHP is essentially consistent with the global
+setting.  An 'inherit' option is added to ensure compatibility with these
+global settings.  Conversely, the options 'force' and 'deny' are dropped,
+which are rather testing artifacts from the old ages.
+
+always
+    Attempt to allocate <size> huge pages every time we need a new page;
+
+inherit
+    Inherit the top-level "shmem_enabled" value. By default, PMD-sized hugepages
+    have enabled="inherit" and all other hugepage sizes have enabled="never";
+
+never
+    Do not allocate <size> huge pages;
+
+within_size
+    Only allocate <size> huge page if it will be fully within i_size.
+    Also respect fadvise()/madvise() hints;
+
+advise
+    Only allocate <size> huge pages if requested with fadvise()/madvise();
+
 Need of application restart
 ===========================
 
 
 #include <linux/mm_types.h>
 
 #include <linux/fs.h> /* only for vma_is_dax() */
+#include <linux/kobject.h>
 
 vm_fault_t do_huge_pmd_anonymous_page(struct vm_fault *vmf);
 int copy_huge_pmd(struct mm_struct *dst_mm, struct mm_struct *src_mm,
                                  struct kobj_attribute *attr, char *buf,
                                  enum transparent_hugepage_flag flag);
 extern struct kobj_attribute shmem_enabled_attr;
+extern struct kobj_attribute thpsize_shmem_enabled_attr;
 
 /*
  * Mask of all large folio orders supported for anonymous THP; all orders up to
        return __thp_vma_allowable_orders(vma, vm_flags, tva_flags, orders);
 }
 
+struct thpsize {
+       struct kobject kobj;
+       struct list_head node;
+       int order;
+};
+
+#define to_thpsize(kobj) container_of(kobj, struct thpsize, kobj)
+
 enum mthp_stat_item {
        MTHP_STAT_ANON_FAULT_ALLOC,
        MTHP_STAT_ANON_FAULT_FALLBACK,
 
 #define SHMEM_SEEN_QUOTA 32
 };
 
+#ifdef CONFIG_TRANSPARENT_HUGEPAGE
+static unsigned long huge_shmem_orders_always __read_mostly;
+static unsigned long huge_shmem_orders_madvise __read_mostly;
+static unsigned long huge_shmem_orders_inherit __read_mostly;
+static unsigned long huge_shmem_orders_within_size __read_mostly;
+#endif
+
 #ifdef CONFIG_TMPFS
 static unsigned long shmem_default_max_blocks(void)
 {
                SHMEM_SB(shm_mnt->mnt_sb)->huge = shmem_huge;
        else
                shmem_huge = SHMEM_HUGE_NEVER; /* just in case it was patched */
+
+       /*
+        * Default to setting PMD-sized THP to inherit the global setting and
+        * disable all other multi-size THPs.
+        */
+       huge_shmem_orders_inherit = BIT(HPAGE_PMD_ORDER);
 #endif
        return;
 
                        huge != SHMEM_HUGE_NEVER && huge != SHMEM_HUGE_DENY)
                return -EINVAL;
 
+       /* Do not override huge allocation policy with non-PMD sized mTHP */
+       if (huge == SHMEM_HUGE_FORCE &&
+           huge_shmem_orders_inherit != BIT(HPAGE_PMD_ORDER))
+               return -EINVAL;
+
        shmem_huge = huge;
        if (shmem_huge > SHMEM_HUGE_DENY)
                SHMEM_SB(shm_mnt->mnt_sb)->huge = shmem_huge;
 }
 
 struct kobj_attribute shmem_enabled_attr = __ATTR_RW(shmem_enabled);
+static DEFINE_SPINLOCK(huge_shmem_orders_lock);
+
+static ssize_t thpsize_shmem_enabled_show(struct kobject *kobj,
+                                         struct kobj_attribute *attr, char *buf)
+{
+       int order = to_thpsize(kobj)->order;
+       const char *output;
+
+       if (test_bit(order, &huge_shmem_orders_always))
+               output = "[always] inherit within_size advise never";
+       else if (test_bit(order, &huge_shmem_orders_inherit))
+               output = "always [inherit] within_size advise never";
+       else if (test_bit(order, &huge_shmem_orders_within_size))
+               output = "always inherit [within_size] advise never";
+       else if (test_bit(order, &huge_shmem_orders_madvise))
+               output = "always inherit within_size [advise] never";
+       else
+               output = "always inherit within_size advise [never]";
+
+       return sysfs_emit(buf, "%s\n", output);
+}
+
+static ssize_t thpsize_shmem_enabled_store(struct kobject *kobj,
+                                          struct kobj_attribute *attr,
+                                          const char *buf, size_t count)
+{
+       int order = to_thpsize(kobj)->order;
+       ssize_t ret = count;
+
+       if (sysfs_streq(buf, "always")) {
+               spin_lock(&huge_shmem_orders_lock);
+               clear_bit(order, &huge_shmem_orders_inherit);
+               clear_bit(order, &huge_shmem_orders_madvise);
+               clear_bit(order, &huge_shmem_orders_within_size);
+               set_bit(order, &huge_shmem_orders_always);
+               spin_unlock(&huge_shmem_orders_lock);
+       } else if (sysfs_streq(buf, "inherit")) {
+               /* Do not override huge allocation policy with non-PMD sized mTHP */
+               if (shmem_huge == SHMEM_HUGE_FORCE &&
+                   order != HPAGE_PMD_ORDER)
+                       return -EINVAL;
+
+               spin_lock(&huge_shmem_orders_lock);
+               clear_bit(order, &huge_shmem_orders_always);
+               clear_bit(order, &huge_shmem_orders_madvise);
+               clear_bit(order, &huge_shmem_orders_within_size);
+               set_bit(order, &huge_shmem_orders_inherit);
+               spin_unlock(&huge_shmem_orders_lock);
+       } else if (sysfs_streq(buf, "within_size")) {
+               spin_lock(&huge_shmem_orders_lock);
+               clear_bit(order, &huge_shmem_orders_always);
+               clear_bit(order, &huge_shmem_orders_inherit);
+               clear_bit(order, &huge_shmem_orders_madvise);
+               set_bit(order, &huge_shmem_orders_within_size);
+               spin_unlock(&huge_shmem_orders_lock);
+       } else if (sysfs_streq(buf, "madvise")) {
+               spin_lock(&huge_shmem_orders_lock);
+               clear_bit(order, &huge_shmem_orders_always);
+               clear_bit(order, &huge_shmem_orders_inherit);
+               clear_bit(order, &huge_shmem_orders_within_size);
+               set_bit(order, &huge_shmem_orders_madvise);
+               spin_unlock(&huge_shmem_orders_lock);
+       } else if (sysfs_streq(buf, "never")) {
+               spin_lock(&huge_shmem_orders_lock);
+               clear_bit(order, &huge_shmem_orders_always);
+               clear_bit(order, &huge_shmem_orders_inherit);
+               clear_bit(order, &huge_shmem_orders_within_size);
+               clear_bit(order, &huge_shmem_orders_madvise);
+               spin_unlock(&huge_shmem_orders_lock);
+       } else {
+               ret = -EINVAL;
+       }
+
+       return ret;
+}
+
+struct kobj_attribute thpsize_shmem_enabled_attr =
+       __ATTR(shmem_enabled, 0644, thpsize_shmem_enabled_show, thpsize_shmem_enabled_store);
 #endif /* CONFIG_TRANSPARENT_HUGEPAGE && CONFIG_SYSFS */
 
 #else /* !CONFIG_SHMEM */