|
@@ -595,13 +595,15 @@ static void node_states_check_changes_online(unsigned long nr_pages,
|
|
|
enum zone_type zone_last = ZONE_NORMAL;
|
|
|
|
|
|
/*
|
|
|
- * If we have HIGHMEM, node_states[N_NORMAL_MEMORY] contains nodes
|
|
|
- * which have 0...ZONE_NORMAL, set zone_last to ZONE_NORMAL.
|
|
|
+ * If we have HIGHMEM or movable node, node_states[N_NORMAL_MEMORY]
|
|
|
+ * contains nodes which have zones of 0...ZONE_NORMAL,
|
|
|
+ * set zone_last to ZONE_NORMAL.
|
|
|
*
|
|
|
- * If we don't have HIGHMEM, node_states[N_NORMAL_MEMORY] contains nodes
|
|
|
- * which have 0...ZONE_MOVABLE, set zone_last to ZONE_MOVABLE.
|
|
|
+ * If we don't have HIGHMEM nor movable node,
|
|
|
+ * node_states[N_NORMAL_MEMORY] contains nodes which have zones of
|
|
|
+ * 0...ZONE_MOVABLE, set zone_last to ZONE_MOVABLE.
|
|
|
*/
|
|
|
- if (N_HIGH_MEMORY == N_NORMAL_MEMORY)
|
|
|
+ if (N_MEMORY == N_NORMAL_MEMORY)
|
|
|
zone_last = ZONE_MOVABLE;
|
|
|
|
|
|
/*
|
|
@@ -615,12 +617,34 @@ static void node_states_check_changes_online(unsigned long nr_pages,
|
|
|
else
|
|
|
arg->status_change_nid_normal = -1;
|
|
|
|
|
|
+#ifdef CONFIG_HIGHMEM
|
|
|
+ /*
|
|
|
+ * If we have movable node, node_states[N_HIGH_MEMORY]
|
|
|
+ * contains nodes which have zones of 0...ZONE_HIGHMEM,
|
|
|
+ * set zone_last to ZONE_HIGHMEM.
|
|
|
+ *
|
|
|
+ * If we don't have movable node, node_states[N_NORMAL_MEMORY]
|
|
|
+ * contains nodes which have zones of 0...ZONE_MOVABLE,
|
|
|
+ * set zone_last to ZONE_MOVABLE.
|
|
|
+ */
|
|
|
+ zone_last = ZONE_HIGHMEM;
|
|
|
+ if (N_MEMORY == N_HIGH_MEMORY)
|
|
|
+ zone_last = ZONE_MOVABLE;
|
|
|
+
|
|
|
+ if (zone_idx(zone) <= zone_last && !node_state(nid, N_HIGH_MEMORY))
|
|
|
+ arg->status_change_nid_high = nid;
|
|
|
+ else
|
|
|
+ arg->status_change_nid_high = -1;
|
|
|
+#else
|
|
|
+ arg->status_change_nid_high = arg->status_change_nid_normal;
|
|
|
+#endif
|
|
|
+
|
|
|
/*
|
|
|
* if the node don't have memory befor online, we will need to
|
|
|
- * set the node to node_states[N_HIGH_MEMORY] after the memory
|
|
|
+ * set the node to node_states[N_MEMORY] after the memory
|
|
|
* is online.
|
|
|
*/
|
|
|
- if (!node_state(nid, N_HIGH_MEMORY))
|
|
|
+ if (!node_state(nid, N_MEMORY))
|
|
|
arg->status_change_nid = nid;
|
|
|
else
|
|
|
arg->status_change_nid = -1;
|
|
@@ -631,7 +655,10 @@ static void node_states_set_node(int node, struct memory_notify *arg)
|
|
|
if (arg->status_change_nid_normal >= 0)
|
|
|
node_set_state(node, N_NORMAL_MEMORY);
|
|
|
|
|
|
- node_set_state(node, N_HIGH_MEMORY);
|
|
|
+ if (arg->status_change_nid_high >= 0)
|
|
|
+ node_set_state(node, N_HIGH_MEMORY);
|
|
|
+
|
|
|
+ node_set_state(node, N_MEMORY);
|
|
|
}
|
|
|
|
|
|
|
|
@@ -1099,13 +1126,15 @@ static void node_states_check_changes_offline(unsigned long nr_pages,
|
|
|
enum zone_type zt, zone_last = ZONE_NORMAL;
|
|
|
|
|
|
/*
|
|
|
- * If we have HIGHMEM, node_states[N_NORMAL_MEMORY] contains nodes
|
|
|
- * which have 0...ZONE_NORMAL, set zone_last to ZONE_NORMAL.
|
|
|
+ * If we have HIGHMEM or movable node, node_states[N_NORMAL_MEMORY]
|
|
|
+ * contains nodes which have zones of 0...ZONE_NORMAL,
|
|
|
+ * set zone_last to ZONE_NORMAL.
|
|
|
*
|
|
|
- * If we don't have HIGHMEM, node_states[N_NORMAL_MEMORY] contains nodes
|
|
|
- * which have 0...ZONE_MOVABLE, set zone_last to ZONE_MOVABLE.
|
|
|
+ * If we don't have HIGHMEM nor movable node,
|
|
|
+ * node_states[N_NORMAL_MEMORY] contains nodes which have zones of
|
|
|
+ * 0...ZONE_MOVABLE, set zone_last to ZONE_MOVABLE.
|
|
|
*/
|
|
|
- if (N_HIGH_MEMORY == N_NORMAL_MEMORY)
|
|
|
+ if (N_MEMORY == N_NORMAL_MEMORY)
|
|
|
zone_last = ZONE_MOVABLE;
|
|
|
|
|
|
/*
|
|
@@ -1122,6 +1151,30 @@ static void node_states_check_changes_offline(unsigned long nr_pages,
|
|
|
else
|
|
|
arg->status_change_nid_normal = -1;
|
|
|
|
|
|
+#ifdef CONFIG_HIGHMEM
|
|
|
+ /*
|
|
|
+ * If we have movable node, node_states[N_HIGH_MEMORY]
|
|
|
+ * contains nodes which have zones of 0...ZONE_HIGHMEM,
|
|
|
+ * set zone_last to ZONE_HIGHMEM.
|
|
|
+ *
|
|
|
+ * If we don't have movable node, node_states[N_NORMAL_MEMORY]
|
|
|
+ * contains nodes which have zones of 0...ZONE_MOVABLE,
|
|
|
+ * set zone_last to ZONE_MOVABLE.
|
|
|
+ */
|
|
|
+ zone_last = ZONE_HIGHMEM;
|
|
|
+ if (N_MEMORY == N_HIGH_MEMORY)
|
|
|
+ zone_last = ZONE_MOVABLE;
|
|
|
+
|
|
|
+ for (; zt <= zone_last; zt++)
|
|
|
+ present_pages += pgdat->node_zones[zt].present_pages;
|
|
|
+ if (zone_idx(zone) <= zone_last && nr_pages >= present_pages)
|
|
|
+ arg->status_change_nid_high = zone_to_nid(zone);
|
|
|
+ else
|
|
|
+ arg->status_change_nid_high = -1;
|
|
|
+#else
|
|
|
+ arg->status_change_nid_high = arg->status_change_nid_normal;
|
|
|
+#endif
|
|
|
+
|
|
|
/*
|
|
|
* node_states[N_HIGH_MEMORY] contains nodes which have 0...ZONE_MOVABLE
|
|
|
*/
|
|
@@ -1146,9 +1199,13 @@ static void node_states_clear_node(int node, struct memory_notify *arg)
|
|
|
if (arg->status_change_nid_normal >= 0)
|
|
|
node_clear_state(node, N_NORMAL_MEMORY);
|
|
|
|
|
|
- if ((N_HIGH_MEMORY != N_NORMAL_MEMORY) &&
|
|
|
- (arg->status_change_nid >= 0))
|
|
|
+ if ((N_MEMORY != N_NORMAL_MEMORY) &&
|
|
|
+ (arg->status_change_nid_high >= 0))
|
|
|
node_clear_state(node, N_HIGH_MEMORY);
|
|
|
+
|
|
|
+ if ((N_MEMORY != N_HIGH_MEMORY) &&
|
|
|
+ (arg->status_change_nid >= 0))
|
|
|
+ node_clear_state(node, N_MEMORY);
|
|
|
}
|
|
|
|
|
|
static int __ref __offline_pages(unsigned long start_pfn,
|