* @ar:                        The address range of the region.
  * @sampling_addr:     Address of the sample for the next access check.
  * @nr_accesses:       Access frequency of this region.
- * @nr_accesses_bp:    @nr_accesses in basis point (0.01%).
+ * @nr_accesses_bp:    @nr_accesses in basis point (0.01%) that updated for
+ *                     each sampling interval.
  * @list:              List head for siblings.
  * @age:               Age of this region.
  *
  * damon_update_region_access_rate().
  *
  * @nr_accesses_bp is another representation of @nr_accesses in basis point
- * (1 in 10,000) that updated every aggregation interval.
+ * (1 in 10,000) that updated for every &damon_attrs->sample_interval in a
+ * manner similar to moving sum.  By the algorithm, this value becomes
+ * @nr_accesses * 10000 for every &struct damon_attrs->aggr_interval.  This can
+ * be used when the aggregation interval is too huge and therefore cannot wait
+ * for it before getting the access monitoring results.
  *
  * @age is initially zero, increased for each aggregation interval, and reset
  * to zero again if the access frequency is significantly changed.  If two
                unsigned int nr_ranges);
 unsigned int damon_moving_sum(unsigned int mvsum, unsigned int nomvsum,
                unsigned int len_window, unsigned int new_value);
-void damon_update_region_access_rate(struct damon_region *r, bool accessed);
+void damon_update_region_access_rate(struct damon_region *r, bool accessed,
+               struct damon_attrs *attrs);
 
 struct damos_filter *damos_new_filter(enum damos_filter_type type,
                bool matching);
 
  * damon_update_region_access_rate() - Update the access rate of a region.
  * @r:         The DAMON region to update for its access check result.
  * @accessed:  Whether the region has accessed during last sampling interval.
+ * @attrs:     The damon_attrs of the DAMON context.
  *
  * Update the access rate of a region with the region's last sampling interval
  * access check result.
  *
  * Usually this will be called by &damon_operations->check_accesses callback.
  */
-void damon_update_region_access_rate(struct damon_region *r, bool accessed)
+void damon_update_region_access_rate(struct damon_region *r, bool accessed,
+               struct damon_attrs *attrs)
 {
+       unsigned int len_window = 1;
+
+       /*
+        * sample_interval can be zero, but cannot be larger than
+        * aggr_interval, owing to validation of damon_set_attrs().
+        */
+       if (attrs->sample_interval)
+               len_window = attrs->aggr_interval / attrs->sample_interval;
+       r->nr_accesses_bp = damon_moving_sum(r->nr_accesses_bp,
+                       r->last_nr_accesses * 10000, len_window,
+                       accessed ? 10000 : 0);
+
        if (accessed)
                r->nr_accesses++;
 }
 
        return accessed;
 }
 
-static void __damon_pa_check_access(struct damon_region *r)
+static void __damon_pa_check_access(struct damon_region *r,
+               struct damon_attrs *attrs)
 {
        static unsigned long last_addr;
        static unsigned long last_folio_sz = PAGE_SIZE;
        /* If the region is in the last checked page, reuse the result */
        if (ALIGN_DOWN(last_addr, last_folio_sz) ==
                                ALIGN_DOWN(r->sampling_addr, last_folio_sz)) {
-               damon_update_region_access_rate(r, last_accessed);
+               damon_update_region_access_rate(r, last_accessed, attrs);
                return;
        }
 
        last_accessed = damon_pa_young(r->sampling_addr, &last_folio_sz);
-       damon_update_region_access_rate(r, last_accessed);
+       damon_update_region_access_rate(r, last_accessed, attrs);
 
        last_addr = r->sampling_addr;
 }
 
        damon_for_each_target(t, ctx) {
                damon_for_each_region(r, t) {
-                       __damon_pa_check_access(r);
+                       __damon_pa_check_access(r, &ctx->attrs);
                        max_nr_accesses = max(r->nr_accesses, max_nr_accesses);
                }
        }
 
  * r   the region to be checked
  */
 static void __damon_va_check_access(struct mm_struct *mm,
-                               struct damon_region *r, bool same_target)
+                               struct damon_region *r, bool same_target,
+                               struct damon_attrs *attrs)
 {
        static unsigned long last_addr;
        static unsigned long last_folio_sz = PAGE_SIZE;
        static bool last_accessed;
 
        if (!mm) {
-               damon_update_region_access_rate(r, false);
+               damon_update_region_access_rate(r, false, attrs);
                return;
        }
 
        /* If the region is in the last checked page, reuse the result */
        if (same_target && (ALIGN_DOWN(last_addr, last_folio_sz) ==
                                ALIGN_DOWN(r->sampling_addr, last_folio_sz))) {
-               damon_update_region_access_rate(r, last_accessed);
+               damon_update_region_access_rate(r, last_accessed, attrs);
                return;
        }
 
        last_accessed = damon_va_young(mm, r->sampling_addr, &last_folio_sz);
-       damon_update_region_access_rate(r, last_accessed);
+       damon_update_region_access_rate(r, last_accessed, attrs);
 
        last_addr = r->sampling_addr;
 }
                mm = damon_get_mm(t);
                same_target = false;
                damon_for_each_region(r, t) {
-                       __damon_va_check_access(mm, r, same_target);
+                       __damon_va_check_access(mm, r, same_target,
+                                       &ctx->attrs);
                        max_nr_accesses = max(r->nr_accesses, max_nr_accesses);
                        same_target = true;
                }