| |
| |
| |
| |
| |
| |
| |
| #include <linux/mmu_notifier.h> |
| #include <linux/page_idle.h> |
| #include <linux/pagemap.h> |
| #include <linux/rmap.h> |
| |
| #include "prmtv-common.h" |
| |
| |
| |
| |
| |
| |
| |
| |
| struct page *damon_get_page(unsigned long pfn) |
| { |
| <------>struct page *page = pfn_to_online_page(pfn); |
| |
| <------>if (!page || !PageLRU(page) || !get_page_unless_zero(page)) |
| <------><------>return NULL; |
| |
| <------>if (unlikely(!PageLRU(page))) { |
| <------><------>put_page(page); |
| <------><------>page = NULL; |
| <------>} |
| <------>return page; |
| } |
| |
| void damon_ptep_mkold(pte_t *pte, struct mm_struct *mm, unsigned long addr) |
| { |
| <------>bool referenced = false; |
| <------>struct page *page = damon_get_page(pte_pfn(*pte)); |
| |
| <------>if (!page) |
| <------><------>return; |
| |
| <------>if (pte_young(*pte)) { |
| <------><------>referenced = true; |
| <------><------>*pte = pte_mkold(*pte); |
| <------>} |
| |
| #ifdef CONFIG_MMU_NOTIFIER |
| <------>if (mmu_notifier_clear_young(mm, addr, addr + PAGE_SIZE)) |
| <------><------>referenced = true; |
| #endif |
| |
| <------>if (referenced) |
| <------><------>set_page_young(page); |
| |
| <------>set_page_idle(page); |
| <------>put_page(page); |
| } |
| |
| void damon_pmdp_mkold(pmd_t *pmd, struct mm_struct *mm, unsigned long addr) |
| { |
| #ifdef CONFIG_TRANSPARENT_HUGEPAGE |
| <------>bool referenced = false; |
| <------>struct page *page = damon_get_page(pmd_pfn(*pmd)); |
| |
| <------>if (!page) |
| <------><------>return; |
| |
| <------>if (pmd_young(*pmd)) { |
| <------><------>referenced = true; |
| <------><------>*pmd = pmd_mkold(*pmd); |
| <------>} |
| |
| #ifdef CONFIG_MMU_NOTIFIER |
| <------>if (mmu_notifier_clear_young(mm, addr, |
| <------><------><------><------>addr + ((1UL) << HPAGE_PMD_SHIFT))) |
| <------><------>referenced = true; |
| #endif |
| |
| <------>if (referenced) |
| <------><------>set_page_young(page); |
| |
| <------>set_page_idle(page); |
| <------>put_page(page); |
| #endif |
| } |
| |
| #define DAMON_MAX_SUBSCORE (100) |
| #define DAMON_MAX_AGE_IN_LOG (32) |
| |
| int damon_pageout_score(struct damon_ctx *c, struct damon_region *r, |
| <------><------><------>struct damos *s) |
| { |
| <------>unsigned int max_nr_accesses; |
| <------>int freq_subscore; |
| <------>unsigned int age_in_sec; |
| <------>int age_in_log, age_subscore; |
| <------>unsigned int freq_weight = s->quota.weight_nr_accesses; |
| <------>unsigned int age_weight = s->quota.weight_age; |
| <------>int hotness; |
| |
| <------>max_nr_accesses = c->aggr_interval / c->sample_interval; |
| <------>freq_subscore = r->nr_accesses * DAMON_MAX_SUBSCORE / max_nr_accesses; |
| |
| <------>age_in_sec = (unsigned long)r->age * c->aggr_interval / 1000000; |
| <------>for (age_in_log = 0; age_in_log < DAMON_MAX_AGE_IN_LOG && age_in_sec; |
| <------><------><------>age_in_log++, age_in_sec >>= 1) |
| <------><------>; |
| |
| <------> |
| <------>if (freq_subscore == 0) |
| <------><------>age_in_log *= -1; |
| |
| <------> |
| <------> * Now age_in_log is in [-DAMON_MAX_AGE_IN_LOG, DAMON_MAX_AGE_IN_LOG]. |
| <------> * Scale it to be in [0, 100] and set it as age subscore. |
| <------> */ |
| <------>age_in_log += DAMON_MAX_AGE_IN_LOG; |
| <------>age_subscore = age_in_log * DAMON_MAX_SUBSCORE / |
| <------><------>DAMON_MAX_AGE_IN_LOG / 2; |
| |
| <------>hotness = (freq_weight * freq_subscore + age_weight * age_subscore); |
| <------>if (freq_weight + age_weight) |
| <------><------>hotness /= freq_weight + age_weight; |
| <------> |
| <------> * Transform it to fit in [0, DAMOS_MAX_SCORE] |
| <------> */ |
| <------>hotness = hotness * DAMOS_MAX_SCORE / DAMON_MAX_SUBSCORE; |
| |
| <------> |
| <------>return DAMOS_MAX_SCORE - hotness; |
| } |
| |