diff options
| author | zhenzhong.duan <zhenzhong.duan@oracle.com> | 2012-07-18 01:06:39 -0400 |
|---|---|---|
| committer | Konrad Rzeszutek Wilk <konrad.wilk@oracle.com> | 2012-07-19 15:52:06 -0400 |
| commit | c3d93f880197953f86ab90d9da4744e926b38e33 (patch) | |
| tree | 279edebf31a6373e19dcb2a993c0896014df8940 | |
| parent | 00e37bdb0113a98408de42db85be002f21dbffd3 (diff) | |
xen: populate correct number of pages when across mem boundary (v2)
When populate pages across a mem boundary at bootup, the page count
populated isn't correct. This is due to mem populated to non-mem
region and ignored.
Pfn range is also wrongly aligned when mem boundary isn't page aligned.
For a dom0 booted with dom_mem=3368952K(0xcd9ff000-4k) dmesg diff is:
[ 0.000000] Freeing 9e-100 pfn range: 98 pages freed
[ 0.000000] 1-1 mapping on 9e->100
[ 0.000000] 1-1 mapping on cd9ff->100000
[ 0.000000] Released 98 pages of unused memory
[ 0.000000] Set 206435 page(s) to 1-1 mapping
-[ 0.000000] Populating cd9fe-cda00 pfn range: 1 pages added
+[ 0.000000] Populating cd9fe-cd9ff pfn range: 1 pages added
+[ 0.000000] Populating 100000-100061 pfn range: 97 pages added
[ 0.000000] BIOS-provided physical RAM map:
[ 0.000000] Xen: 0000000000000000 - 000000000009e000 (usable)
[ 0.000000] Xen: 00000000000a0000 - 0000000000100000 (reserved)
[ 0.000000] Xen: 0000000000100000 - 00000000cd9ff000 (usable)
[ 0.000000] Xen: 00000000cd9ffc00 - 00000000cda53c00 (ACPI NVS)
...
[ 0.000000] Xen: 0000000100000000 - 0000000100061000 (usable)
[ 0.000000] Xen: 0000000100061000 - 000000012c000000 (unusable)
...
[ 0.000000] MEMBLOCK configuration:
...
-[ 0.000000] reserved[0x4] [0x000000cd9ff000-0x000000cd9ffbff], 0xc00 bytes
-[ 0.000000] reserved[0x5] [0x00000100000000-0x00000100060fff], 0x61000 bytes
Related xen memory layout:
(XEN) Xen-e820 RAM map:
(XEN) 0000000000000000 - 000000000009ec00 (usable)
(XEN) 00000000000f0000 - 0000000000100000 (reserved)
(XEN) 0000000000100000 - 00000000cd9ffc00 (usable)
Signed-off-by: Zhenzhong Duan <zhenzhong.duan@oracle.com>
[v2: If xen_do_chunk fail(populate), abort this chunk and any others]
Suggested by David, thanks.
Signed-off-by: Konrad Rzeszutek Wilk <konrad.wilk@oracle.com>
| -rw-r--r-- | arch/x86/xen/setup.c | 23 |
1 files changed, 9 insertions, 14 deletions
diff --git a/arch/x86/xen/setup.c b/arch/x86/xen/setup.c index a4790bf22c59..ead85576d54a 100644 --- a/arch/x86/xen/setup.c +++ b/arch/x86/xen/setup.c | |||
| @@ -157,25 +157,24 @@ static unsigned long __init xen_populate_chunk( | |||
| 157 | unsigned long dest_pfn; | 157 | unsigned long dest_pfn; |
| 158 | 158 | ||
| 159 | for (i = 0, entry = list; i < map_size; i++, entry++) { | 159 | for (i = 0, entry = list; i < map_size; i++, entry++) { |
| 160 | unsigned long credits = credits_left; | ||
| 161 | unsigned long s_pfn; | 160 | unsigned long s_pfn; |
| 162 | unsigned long e_pfn; | 161 | unsigned long e_pfn; |
| 163 | unsigned long pfns; | 162 | unsigned long pfns; |
| 164 | long capacity; | 163 | long capacity; |
| 165 | 164 | ||
| 166 | if (credits <= 0) | 165 | if (credits_left <= 0) |
| 167 | break; | 166 | break; |
| 168 | 167 | ||
| 169 | if (entry->type != E820_RAM) | 168 | if (entry->type != E820_RAM) |
| 170 | continue; | 169 | continue; |
| 171 | 170 | ||
| 172 | e_pfn = PFN_UP(entry->addr + entry->size); | 171 | e_pfn = PFN_DOWN(entry->addr + entry->size); |
| 173 | 172 | ||
| 174 | /* We only care about E820 after the xen_start_info->nr_pages */ | 173 | /* We only care about E820 after the xen_start_info->nr_pages */ |
| 175 | if (e_pfn <= max_pfn) | 174 | if (e_pfn <= max_pfn) |
| 176 | continue; | 175 | continue; |
| 177 | 176 | ||
| 178 | s_pfn = PFN_DOWN(entry->addr); | 177 | s_pfn = PFN_UP(entry->addr); |
| 179 | /* If the E820 falls within the nr_pages, we want to start | 178 | /* If the E820 falls within the nr_pages, we want to start |
| 180 | * at the nr_pages PFN. | 179 | * at the nr_pages PFN. |
| 181 | * If that would mean going past the E820 entry, skip it | 180 | * If that would mean going past the E820 entry, skip it |
| @@ -184,23 +183,19 @@ static unsigned long __init xen_populate_chunk( | |||
| 184 | capacity = e_pfn - max_pfn; | 183 | capacity = e_pfn - max_pfn; |
| 185 | dest_pfn = max_pfn; | 184 | dest_pfn = max_pfn; |
| 186 | } else { | 185 | } else { |
| 187 | /* last_pfn MUST be within E820_RAM regions */ | ||
| 188 | if (*last_pfn && e_pfn >= *last_pfn) | ||
| 189 | s_pfn = *last_pfn; | ||
| 190 | capacity = e_pfn - s_pfn; | 186 | capacity = e_pfn - s_pfn; |
| 191 | dest_pfn = s_pfn; | 187 | dest_pfn = s_pfn; |
| 192 | } | 188 | } |
| 193 | /* If we had filled this E820_RAM entry, go to the next one. */ | ||
| 194 | if (capacity <= 0) | ||
| 195 | continue; | ||
| 196 | 189 | ||
| 197 | if (credits > capacity) | 190 | if (credits_left < capacity) |
| 198 | credits = capacity; | 191 | capacity = credits_left; |
| 199 | 192 | ||
| 200 | pfns = xen_do_chunk(dest_pfn, dest_pfn + credits, false); | 193 | pfns = xen_do_chunk(dest_pfn, dest_pfn + capacity, false); |
| 201 | done += pfns; | 194 | done += pfns; |
| 202 | credits_left -= pfns; | ||
| 203 | *last_pfn = (dest_pfn + pfns); | 195 | *last_pfn = (dest_pfn + pfns); |
| 196 | if (pfns < capacity) | ||
| 197 | break; | ||
| 198 | credits_left -= pfns; | ||
| 204 | } | 199 | } |
| 205 | return done; | 200 | return done; |
| 206 | } | 201 | } |
