|
@@ -415,11 +415,7 @@ EXPORT_SYMBOL(dec_zone_page_state);
|
|
|
#endif
|
|
|
|
|
|
/*
|
|
|
- * Update the zone counters for one cpu.
|
|
|
- *
|
|
|
- * The cpu specified must be either the current cpu or a processor that
|
|
|
- * is not online. If it is the current cpu then the execution thread must
|
|
|
- * be pinned to the current cpu.
|
|
|
+ * Update the zone counters for the current cpu.
|
|
|
*
|
|
|
* Note that refresh_cpu_vm_stats strives to only access
|
|
|
* node local memory. The per cpu pagesets on remote zones are placed
|
|
@@ -432,7 +428,7 @@ EXPORT_SYMBOL(dec_zone_page_state);
|
|
|
* with the global counters. These could cause remote node cache line
|
|
|
* bouncing and will have to be only done when necessary.
|
|
|
*/
|
|
|
-void refresh_cpu_vm_stats(int cpu)
|
|
|
+static void refresh_cpu_vm_stats(int cpu)
|
|
|
{
|
|
|
struct zone *zone;
|
|
|
int i;
|
|
@@ -493,6 +489,38 @@ void refresh_cpu_vm_stats(int cpu)
|
|
|
atomic_long_add(global_diff[i], &vm_stat[i]);
|
|
|
}
|
|
|
|
|
|
+/*
|
|
|
+ * Fold the data for an offline cpu into the global array.
|
|
|
+ * There cannot be any access by the offline cpu and therefore
|
|
|
+ * synchronization is simplified.
|
|
|
+ */
|
|
|
+void cpu_vm_stats_fold(int cpu)
|
|
|
+{
|
|
|
+ struct zone *zone;
|
|
|
+ int i;
|
|
|
+ int global_diff[NR_VM_ZONE_STAT_ITEMS] = { 0, };
|
|
|
+
|
|
|
+ for_each_populated_zone(zone) {
|
|
|
+ struct per_cpu_pageset *p;
|
|
|
+
|
|
|
+ p = per_cpu_ptr(zone->pageset, cpu);
|
|
|
+
|
|
|
+ for (i = 0; i < NR_VM_ZONE_STAT_ITEMS; i++)
|
|
|
+ if (p->vm_stat_diff[i]) {
|
|
|
+ int v;
|
|
|
+
|
|
|
+ v = p->vm_stat_diff[i];
|
|
|
+ p->vm_stat_diff[i] = 0;
|
|
|
+ atomic_long_add(v, &zone->vm_stat[i]);
|
|
|
+ global_diff[i] += v;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ for (i = 0; i < NR_VM_ZONE_STAT_ITEMS; i++)
|
|
|
+ if (global_diff[i])
|
|
|
+ atomic_long_add(global_diff[i], &vm_stat[i]);
|
|
|
+}
|
|
|
+
|
|
|
/*
|
|
|
* this is only called if !populated_zone(zone), which implies no other users of
|
|
|
* pset->vm_stat_diff[] exsist.
|