On Fri, Nov 16, 2012 at 07:38:51PM -0800, Yinghai Lu wrote:
> We are going to map ram only, so under max_low_pfn_mapped,
> between 4g and max_pfn_mapped does not mean mapped at all.
> 
> Use pfn_range_is_mapped() to find out if range is mapped for initrd.
> 
> That could happen bootloader put initrd in range but user could
                   ^- 'if the'                              ^^^^^-'used'

> use memmap to carve some of range out.
  ^^^-> get rid of that.       ^^^^->'ranges'

> 
> Also during copying need to use early_memmap to map original initrd
> for accessing.
> 
> Signed-off-by: Yinghai Lu <ying...@kernel.org>
> ---
>  arch/x86/kernel/setup.c |   52 +++++++++++++++++++++++++---------------------
>  1 files changed, 28 insertions(+), 24 deletions(-)
> 
> diff --git a/arch/x86/kernel/setup.c b/arch/x86/kernel/setup.c
> index d85cbd9..bd52f9d 100644
> --- a/arch/x86/kernel/setup.c
> +++ b/arch/x86/kernel/setup.c
> @@ -317,20 +317,19 @@ static void __init relocate_initrd(void)
>       u64 ramdisk_image = boot_params.hdr.ramdisk_image;
>       u64 ramdisk_size  = boot_params.hdr.ramdisk_size;
>       u64 area_size     = PAGE_ALIGN(ramdisk_size);
> -     u64 end_of_lowmem = max_low_pfn_mapped << PAGE_SHIFT;
>       u64 ramdisk_here;
>       unsigned long slop, clen, mapaddr;
>       char *p, *q;
>  
> -     /* We need to move the initrd down into lowmem */
> -     ramdisk_here = memblock_find_in_range(0, end_of_lowmem, area_size,
> -                                      PAGE_SIZE);
> +     /* We need to move the initrd down into directly mapped mem */
> +     ramdisk_here = memblock_find_in_range(0, PFN_PHYS(max_low_pfn_mapped),
> +                                              area_size, PAGE_SIZE);
>  
>       if (!ramdisk_here)
>               panic("Cannot find place for new RAMDISK of size %lld\n",
>                        ramdisk_size);
>  
> -     /* Note: this includes all the lowmem currently occupied by
> +     /* Note: this includes all the mem currently occupied by
>          the initrd, we rely on that fact to keep the data intact. */
>       memblock_reserve(ramdisk_here, area_size);
>       initrd_start = ramdisk_here + PAGE_OFFSET;
> @@ -340,17 +339,7 @@ static void __init relocate_initrd(void)
>  
>       q = (char *)initrd_start;
>  
> -     /* Copy any lowmem portion of the initrd */
> -     if (ramdisk_image < end_of_lowmem) {
> -             clen = end_of_lowmem - ramdisk_image;
> -             p = (char *)__va(ramdisk_image);
> -             memcpy(q, p, clen);
> -             q += clen;
> -             ramdisk_image += clen;
> -             ramdisk_size  -= clen;
> -     }
> -
> -     /* Copy the highmem portion of the initrd */
> +     /* Copy the initrd */
>       while (ramdisk_size) {
>               slop = ramdisk_image & ~PAGE_MASK;
>               clen = ramdisk_size;
> @@ -364,7 +353,7 @@ static void __init relocate_initrd(void)
>               ramdisk_image += clen;
>               ramdisk_size  -= clen;
>       }
> -     /* high pages is not converted by early_res_to_bootmem */
> +
>       ramdisk_image = boot_params.hdr.ramdisk_image;
>       ramdisk_size  = boot_params.hdr.ramdisk_size;
>       printk(KERN_INFO "Move RAMDISK from [mem %#010llx-%#010llx] to"
> @@ -373,13 +362,27 @@ static void __init relocate_initrd(void)
>               ramdisk_here, ramdisk_here + ramdisk_size - 1);
>  }
>  
> +static u64 __init get_mem_size(unsigned long limit_pfn)
> +{
> +     int i;
> +     u64 mapped_pages = 0;
> +     unsigned long start_pfn, end_pfn;
> +
> +     for_each_mem_pfn_range(i, MAX_NUMNODES, &start_pfn, &end_pfn, NULL) {
> +             start_pfn = min_t(unsigned long, start_pfn, limit_pfn);
> +             end_pfn = min_t(unsigned long, end_pfn, limit_pfn);
> +             mapped_pages += end_pfn - start_pfn;
> +     }
> +
> +     return mapped_pages << PAGE_SHIFT;
> +}
>  static void __init reserve_initrd(void)
>  {
>       /* Assume only end is not page aligned */
>       u64 ramdisk_image = boot_params.hdr.ramdisk_image;
>       u64 ramdisk_size  = boot_params.hdr.ramdisk_size;
>       u64 ramdisk_end   = PAGE_ALIGN(ramdisk_image + ramdisk_size);
> -     u64 end_of_lowmem = max_low_pfn_mapped << PAGE_SHIFT;
> +     u64 mapped_size;
>  
>       if (!boot_params.hdr.type_of_loader ||
>           !ramdisk_image || !ramdisk_size)
> @@ -387,18 +390,19 @@ static void __init reserve_initrd(void)
>  
>       initrd_start = 0;
>  
> -     if (ramdisk_size >= (end_of_lowmem>>1)) {
> +     mapped_size = get_mem_size(max_low_pfn_mapped);
> +     if (ramdisk_size >= (mapped_size>>1))
>               panic("initrd too large to handle, "
>                      "disabling initrd (%lld needed, %lld available)\n",
> -                    ramdisk_size, end_of_lowmem>>1);
> -     }
> +                    ramdisk_size, mapped_size>>1);
>  
>       printk(KERN_INFO "RAMDISK: [mem %#010llx-%#010llx]\n", ramdisk_image,
>                       ramdisk_end - 1);
>  
> -
> -     if (ramdisk_end <= end_of_lowmem) {
> -             /* All in lowmem, easy case */
> +     if (ramdisk_end <= (max_low_pfn_mapped<<PAGE_SHIFT) &&
> +         pfn_range_is_mapped(PFN_DOWN(ramdisk_image),
> +                             PFN_DOWN(ramdisk_end))) {
> +             /* All are mapped, easy case */
>               /*
>                * don't need to reserve again, already reserved early
>                * in i386_start_kernel
> -- 
> 1.7.7
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

Reply via email to