selftests/mm: support multi-size THP interface in thp_settings
authorRyan Roberts <ryan.roberts@arm.com>
Thu, 7 Dec 2023 16:12:08 +0000 (16:12 +0000)
committerAndrew Morton <akpm@linux-foundation.org>
Wed, 20 Dec 2023 22:48:12 +0000 (14:48 -0800)
Save and restore the new per-size hugepage enabled setting, if available
on the running kernel.

Since the number of per-size directories is not fixed, solve this as
simply as possible by catering for a maximum number in the thp_settings
struct (20).  Each array index is the order.  The value of THP_NEVER is
changed to 0 so that all of these new settings default to THP_NEVER and
the user only needs to fill in the ones they want to enable.

Link: https://lkml.kernel.org/r/20231207161211.2374093-8-ryan.roberts@arm.com
Signed-off-by: Ryan Roberts <ryan.roberts@arm.com>
Tested-by: Kefeng Wang <wangkefeng.wang@huawei.com>
Tested-by: John Hubbard <jhubbard@nvidia.com>
Cc: Alistair Popple <apopple@nvidia.com>
Cc: Anshuman Khandual <anshuman.khandual@arm.com>
Cc: Barry Song <v-songbaohua@oppo.com>
Cc: Catalin Marinas <catalin.marinas@arm.com>
Cc: David Hildenbrand <david@redhat.com>
Cc: David Rientjes <rientjes@google.com>
Cc: "Huang, Ying" <ying.huang@intel.com>
Cc: Hugh Dickins <hughd@google.com>
Cc: Itaru Kitayama <itaru.kitayama@gmail.com>
Cc: Kirill A. Shutemov <kirill.shutemov@linux.intel.com>
Cc: Luis Chamberlain <mcgrof@kernel.org>
Cc: Matthew Wilcox (Oracle) <willy@infradead.org>
Cc: Vlastimil Babka <vbabka@suse.cz>
Cc: Yang Shi <shy828301@gmail.com>
Cc: Yin Fengwei <fengwei.yin@intel.com>
Cc: Yu Zhao <yuzhao@google.com>
Cc: Zi Yan <ziy@nvidia.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
tools/testing/selftests/mm/khugepaged.c
tools/testing/selftests/mm/thp_settings.c
tools/testing/selftests/mm/thp_settings.h

index b15e7fd70176914dab6d002dfe841e94a088744e..7bd3baa9d34b4fd264c07a2a0956a2105cf3397e 100644 (file)
@@ -1141,6 +1141,7 @@ static void parse_test_type(int argc, const char **argv)
 
 int main(int argc, const char **argv)
 {
+       int hpage_pmd_order;
        struct thp_settings default_settings = {
                .thp_enabled = THP_MADVISE,
                .thp_defrag = THP_DEFRAG_ALWAYS,
@@ -1175,11 +1176,13 @@ int main(int argc, const char **argv)
                exit(EXIT_FAILURE);
        }
        hpage_pmd_nr = hpage_pmd_size / page_size;
+       hpage_pmd_order = __builtin_ctz(hpage_pmd_nr);
 
        default_settings.khugepaged.max_ptes_none = hpage_pmd_nr - 1;
        default_settings.khugepaged.max_ptes_swap = hpage_pmd_nr / 8;
        default_settings.khugepaged.max_ptes_shared = hpage_pmd_nr / 2;
        default_settings.khugepaged.pages_to_scan = hpage_pmd_nr * 8;
+       default_settings.hugepages[hpage_pmd_order].enabled = THP_INHERIT;
 
        save_settings();
        thp_push_settings(&default_settings);
index 5e8ec792cac70195aef322f5b9ece169e88df0eb..a4163438108ec50fc849d314d651122d6662b2b4 100644 (file)
@@ -16,9 +16,10 @@ static struct thp_settings saved_settings;
 static char dev_queue_read_ahead_path[PATH_MAX];
 
 static const char * const thp_enabled_strings[] = {
+       "never",
        "always",
+       "inherit",
        "madvise",
-       "never",
        NULL
 };
 
@@ -198,6 +199,10 @@ void thp_write_num(const char *name, unsigned long num)
 
 void thp_read_settings(struct thp_settings *settings)
 {
+       unsigned long orders = thp_supported_orders();
+       char path[PATH_MAX];
+       int i;
+
        *settings = (struct thp_settings) {
                .thp_enabled = thp_read_string("enabled", thp_enabled_strings),
                .thp_defrag = thp_read_string("defrag", thp_defrag_strings),
@@ -218,11 +223,26 @@ void thp_read_settings(struct thp_settings *settings)
        };
        if (dev_queue_read_ahead_path[0])
                settings->read_ahead_kb = read_num(dev_queue_read_ahead_path);
+
+       for (i = 0; i < NR_ORDERS; i++) {
+               if (!((1 << i) & orders)) {
+                       settings->hugepages[i].enabled = THP_NEVER;
+                       continue;
+               }
+               snprintf(path, PATH_MAX, "hugepages-%ukB/enabled",
+                       (getpagesize() >> 10) << i);
+               settings->hugepages[i].enabled =
+                       thp_read_string(path, thp_enabled_strings);
+       }
 }
 
 void thp_write_settings(struct thp_settings *settings)
 {
        struct khugepaged_settings *khugepaged = &settings->khugepaged;
+       unsigned long orders = thp_supported_orders();
+       char path[PATH_MAX];
+       int enabled;
+       int i;
 
        thp_write_string("enabled", thp_enabled_strings[settings->thp_enabled]);
        thp_write_string("defrag", thp_defrag_strings[settings->thp_defrag]);
@@ -242,6 +262,15 @@ void thp_write_settings(struct thp_settings *settings)
 
        if (dev_queue_read_ahead_path[0])
                write_num(dev_queue_read_ahead_path, settings->read_ahead_kb);
+
+       for (i = 0; i < NR_ORDERS; i++) {
+               if (!((1 << i) & orders))
+                       continue;
+               snprintf(path, PATH_MAX, "hugepages-%ukB/enabled",
+                       (getpagesize() >> 10) << i);
+               enabled = settings->hugepages[i].enabled;
+               thp_write_string(path, thp_enabled_strings[enabled]);
+       }
 }
 
 struct thp_settings *thp_current_settings(void)
@@ -294,3 +323,27 @@ void thp_set_read_ahead_path(char *path)
                sizeof(dev_queue_read_ahead_path));
        dev_queue_read_ahead_path[sizeof(dev_queue_read_ahead_path) - 1] = '\0';
 }
+
+unsigned long thp_supported_orders(void)
+{
+       unsigned long orders = 0;
+       char path[PATH_MAX];
+       char buf[256];
+       int ret;
+       int i;
+
+       for (i = 0; i < NR_ORDERS; i++) {
+               ret = snprintf(path, PATH_MAX, THP_SYSFS "hugepages-%ukB/enabled",
+                       (getpagesize() >> 10) << i);
+               if (ret >= PATH_MAX) {
+                       printf("%s: Pathname is too long\n", __func__);
+                       exit(EXIT_FAILURE);
+               }
+
+               ret = read_file(path, buf, sizeof(buf));
+               if (ret)
+                       orders |= 1UL << i;
+       }
+
+       return orders;
+}
index ff3d98c30617b39e35f5bdab47212d0cb2989cf2..71cbff05f4c7ff3b4ec67ee79cdf2129fcdf9f13 100644 (file)
@@ -7,9 +7,10 @@
 #include <stdint.h>
 
 enum thp_enabled {
+       THP_NEVER,
        THP_ALWAYS,
+       THP_INHERIT,
        THP_MADVISE,
-       THP_NEVER,
 };
 
 enum thp_defrag {
@@ -29,6 +30,12 @@ enum shmem_enabled {
        SHMEM_FORCE,
 };
 
+#define NR_ORDERS 20
+
+struct hugepages_settings {
+       enum thp_enabled enabled;
+};
+
 struct khugepaged_settings {
        bool defrag;
        unsigned int alloc_sleep_millisecs;
@@ -46,6 +53,7 @@ struct thp_settings {
        bool use_zero_page;
        struct khugepaged_settings khugepaged;
        unsigned long read_ahead_kb;
+       struct hugepages_settings hugepages[NR_ORDERS];
 };
 
 int read_file(const char *path, char *buf, size_t buflen);
@@ -67,5 +75,6 @@ void thp_restore_settings(void);
 void thp_save_settings(void);
 
 void thp_set_read_ahead_path(char *path);
+unsigned long thp_supported_orders(void);
 
 #endif /* __THP_SETTINGS_H__ */