@@ -87,6 +87,9 @@ enum transparent_hugepage_flag {
TRANSPARENT_HUGEPAGE_DEFRAG_REQ_MADV_FLAG,
TRANSPARENT_HUGEPAGE_DEFRAG_KHUGEPAGED_FLAG,
TRANSPARENT_HUGEPAGE_USE_ZERO_PAGE_FLAG,
+#ifdef CONFIG_HUGETEXT
+ TRANSPARENT_HUGEPAGE_HUGETEXT_ENABLED_FLAG,
+#endif
};
struct kobject;
@@ -140,6 +143,27 @@ static inline bool transhuge_vma_enabled(struct vm_area_struct *vma,
return true;
}
+#ifdef CONFIG_HUGETEXT
+#define hugetext_enabled() \
+ (transparent_hugepage_flags & \
+ (1<<TRANSPARENT_HUGEPAGE_HUGETEXT_ENABLED_FLAG))
+#else
+#define hugetext_enabled() false
+#endif /* CONFIG_HUGETEXT */
+
+static inline bool vma_is_hugetext(struct vm_area_struct *vma,
+ unsigned long vm_flags)
+{
+ if (!(vm_flags & VM_EXEC))
+ return false;
+
+ if (vma->vm_file && !inode_is_open_for_write(vma->vm_file->f_inode))
+ return IS_ALIGNED((vma->vm_start >> PAGE_SHIFT) - vma->vm_pgoff,
+ HPAGE_PMD_NR);
+
+ return false;
+}
+
/*
* to be used on vmas which are known to support THP.
* Use transparent_hugepage_active otherwise
@@ -26,10 +26,18 @@ static inline void collapse_pte_mapped_thp(struct mm_struct *mm,
}
#endif
+#ifdef CONFIG_HUGETEXT
+#define khugepaged_enabled() \
+ (transparent_hugepage_flags & \
+ ((1<<TRANSPARENT_HUGEPAGE_FLAG) | \
+ (1<<TRANSPARENT_HUGEPAGE_REQ_MADV_FLAG) | \
+ (1<<TRANSPARENT_HUGEPAGE_HUGETEXT_ENABLED_FLAG)))
+#else
#define khugepaged_enabled() \
(transparent_hugepage_flags & \
((1<<TRANSPARENT_HUGEPAGE_FLAG) | \
(1<<TRANSPARENT_HUGEPAGE_REQ_MADV_FLAG)))
+#endif
#define khugepaged_always() \
(transparent_hugepage_flags & \
(1<<TRANSPARENT_HUGEPAGE_FLAG))
@@ -59,6 +67,7 @@ static inline int khugepaged_enter(struct vm_area_struct *vma,
if (!test_bit(MMF_VM_HUGEPAGE, &vma->vm_mm->flags))
if ((khugepaged_always() ||
(shmem_file(vma->vm_file) && shmem_huge_enabled(vma)) ||
+ (hugetext_enabled() && vma_is_hugetext(vma, vm_flags)) ||
(khugepaged_req_madv() && (vm_flags & VM_HUGEPAGE))) &&
!(vm_flags & VM_NOHUGEPAGE) &&
!test_bit(MMF_DISABLE_THP, &vma->vm_mm->flags))
@@ -868,6 +868,17 @@ config READ_ONLY_THP_FOR_FS
support of file THPs will be developed in the next few release
cycles.
+config HUGETEXT
+ bool "THP for text segments"
+ depends on READ_ONLY_THP_FOR_FS
+
+ help
+ Allow khugepaged to put read-only file-backed pages, including
+ shared libraries, as well as the anonymous and executable pages
+ in THP.
+
+ This feature builds on and extends READ_ONLY_THP_FOR_FS.
+
config ARCH_HAS_PTE_SPECIAL
bool
@@ -330,6 +330,35 @@ static ssize_t hpage_pmd_size_show(struct kobject *kobj,
static struct kobj_attribute hpage_pmd_size_attr =
__ATTR_RO(hpage_pmd_size);
+#ifdef CONFIG_HUGETEXT
+static ssize_t hugetext_enabled_show(struct kobject *kobj,
+ struct kobj_attribute *attr, char *buf)
+{
+ return single_hugepage_flag_show(kobj, attr, buf,
+ TRANSPARENT_HUGEPAGE_HUGETEXT_ENABLED_FLAG);
+}
+
+static ssize_t hugetext_enabled_store(struct kobject *kobj,
+ struct kobj_attribute *attr, const char *buf, size_t count)
+{
+ ssize_t ret = count;
+
+ ret = single_hugepage_flag_store(kobj, attr, buf, count,
+ TRANSPARENT_HUGEPAGE_HUGETEXT_ENABLED_FLAG);
+
+ if (ret > 0) {
+ int err = start_stop_khugepaged();
+
+ if (err)
+ ret = err;
+ }
+
+ return ret;
+}
+struct kobj_attribute hugetext_enabled_attr =
+ __ATTR(hugetext_enabled, 0644, hugetext_enabled_show, hugetext_enabled_store);
+#endif /* CONFIG_HUGETEXT */
+
static struct attribute *hugepage_attr[] = {
&enabled_attr.attr,
&defrag_attr.attr,
@@ -337,6 +366,9 @@ static struct attribute *hugepage_attr[] = {
&hpage_pmd_size_attr.attr,
#ifdef CONFIG_SHMEM
&shmem_enabled_attr.attr,
+#endif
+#ifdef CONFIG_HUGETEXT
+ &hugetext_enabled_attr.attr,
#endif
NULL,
};
@@ -491,6 +523,31 @@ static int __init setup_transparent_hugepage(char *str)
}
__setup("transparent_hugepage=", setup_transparent_hugepage);
+#ifdef CONFIG_HUGETEXT
+static int __init setup_hugetext(char *str)
+{
+ int ret = 0;
+
+ if (!str)
+ goto out;
+ if (!strcmp(str, "1")) {
+ set_bit(TRANSPARENT_HUGEPAGE_HUGETEXT_ENABLED_FLAG,
+ &transparent_hugepage_flags);
+ ret = 1;
+ } else if (!strcmp(str, "0")) {
+ clear_bit(TRANSPARENT_HUGEPAGE_HUGETEXT_ENABLED_FLAG,
+ &transparent_hugepage_flags);
+ ret = 1;
+ }
+
+out:
+ if (!ret)
+ pr_warn("hugetext= cannot parse, ignored\n");
+ return ret;
+}
+__setup("hugetext=", setup_hugetext);
+#endif
+
pmd_t maybe_pmd_mkwrite(pmd_t pmd, struct vm_area_struct *vma)
{
if (likely(vma->vm_flags & VM_WRITE))
@@ -451,6 +451,10 @@ static bool hugepage_vma_check(struct vm_area_struct *vma,
HPAGE_PMD_NR);
}
+ /* Make hugetext independent of THP settings */
+ if (hugetext_enabled() && vma_is_hugetext(vma, vm_flags))
+ return true;
+
/* THP settings require madvise. */
if (!(vm_flags & VM_HUGEPAGE) && !khugepaged_always())
return false;
@@ -73,6 +73,7 @@
#include <linux/perf_event.h>
#include <linux/ptrace.h>
#include <linux/vmalloc.h>
+#include <linux/khugepaged.h>
#include <trace/events/kmem.h>
@@ -4157,6 +4158,17 @@ static vm_fault_t do_read_fault(struct vm_fault *vmf)
struct vm_area_struct *vma = vmf->vma;
vm_fault_t ret = 0;
+#ifdef CONFIG_HUGETEXT
+ /* Add the candidate hugetext vma into khugepaged scan list */
+ if (pmd_none(*vmf->pmd) && hugetext_enabled()
+ && vma_is_hugetext(vma, vma->vm_flags)) {
+ unsigned long haddr = vmf->address & HPAGE_PMD_MASK;
+
+ if (transhuge_vma_suitable(vma, haddr))
+ khugepaged_enter(vma, vma->vm_flags);
+ }
+#endif
+
/*
* Let's call ->map_pages() first and use ->fault() as fallback
* if page by the offset is not ready to be mapped (cold cache or