[Suspend2][ 08/10] [Suspend2] Atomic copy routine

From: Nigel Cunningham
Date: Tue Jun 27 2006 - 00:38:18 EST


When suspending, this routine is used to make the atomic copy of memory.
Like swsusp, it is done in C.

Signed-off-by: Nigel Cunningham <nigel@xxxxxxxxxxxx>

kernel/power/atomic_copy.c | 47 ++++++++++++++++++++++++++++++++++++++++++++
1 files changed, 47 insertions(+), 0 deletions(-)

diff --git a/kernel/power/atomic_copy.c b/kernel/power/atomic_copy.c
index cb3663c..75e3beb 100644
--- a/kernel/power/atomic_copy.c
+++ b/kernel/power/atomic_copy.c
@@ -302,3 +302,50 @@ int suspend_post_context_save(void)
return 0;
}

+/* suspend_copy_pageset1: Do the atomic copy of pageset1.
+ *
+ * Make the atomic copy of pageset1. We can't use copy_page (as we once did)
+ * because we can't be sure what side effects it has. On my old Duron, with
+ * 3DNOW, kernel_fpu_begin increments preempt count, making our preempt
+ * count at resume time 4 instead of 3.
+ *
+ * We don't want to call kmap_atomic unconditionally because it has the side
+ * effect of incrementing the preempt count, which will leave it one too high
+ * post resume (the page containing the preempt count will be copied after
+ * its incremented. This is essentially the same problem.
+ */
+
+void suspend_copy_pageset1(void)
+{
+ int i, source_index, dest_index;
+
+ source_index = get_next_bit_on(pageset1_map, -1);
+ dest_index = get_next_bit_on(pageset1_copy_map, -1);
+
+ for (i = 0; i < pagedir1.pageset_size; i++) {
+ unsigned long *origvirt, *copyvirt;
+ struct page *origpage;
+ int loop = (PAGE_SIZE / sizeof(unsigned long)) - 1;
+
+ origpage = pfn_to_page(source_index);
+
+ if (PageHighMem(origpage))
+ origvirt = kmap_atomic(origpage, KM_USER0);
+ else
+ origvirt = page_address(origpage);
+
+ copyvirt = (unsigned long *) page_address(pfn_to_page(dest_index));
+
+ while (loop >= 0) {
+ *(copyvirt + loop) = *(origvirt + loop);
+ loop--;
+ }
+
+ if (PageHighMem(origpage))
+ kunmap_atomic(origvirt, KM_USER0);
+
+ source_index = get_next_bit_on(pageset1_map, source_index);
+ dest_index = get_next_bit_on(pageset1_copy_map, dest_index);
+ }
+}
+

--
Nigel Cunningham nigel at suspend2 dot net
-
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/