From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org X-Spam-Level: X-Spam-Status: No, score=-24.9 required=3.0 tests=BAYES_00,DKIM_SIGNED, DKIM_VALID,DKIM_VALID_AU,HEADER_FROM_DIFFERENT_DOMAINS,INCLUDES_CR_TRAILER, INCLUDES_PATCH,MAILING_LIST_MULTI,NICE_REPLY_A,SPF_HELO_NONE,SPF_PASS, USER_AGENT_SANE_1,USER_IN_DEF_DKIM_WL autolearn=unavailable autolearn_force=no version=3.4.0 Received: from mail.kernel.org (mail.kernel.org [198.145.29.99]) by smtp.lore.kernel.org (Postfix) with ESMTP id 7A39AC433E0 for ; Thu, 4 Feb 2021 23:25:13 +0000 (UTC) Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by mail.kernel.org (Postfix) with ESMTP id 3AC2864FAD for ; Thu, 4 Feb 2021 23:25:13 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S230088AbhBDXYg (ORCPT ); Thu, 4 Feb 2021 18:24:36 -0500 Received: from linux.microsoft.com ([13.77.154.182]:36868 "EHLO linux.microsoft.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S229977AbhBDXYg (ORCPT ); Thu, 4 Feb 2021 18:24:36 -0500 Received: from [192.168.0.104] (c-73-42-176-67.hsd1.wa.comcast.net [73.42.176.67]) by linux.microsoft.com (Postfix) with ESMTPSA id 11B9220202A2; Thu, 4 Feb 2021 15:23:54 -0800 (PST) DKIM-Filter: OpenDKIM Filter v2.11.0 linux.microsoft.com 11B9220202A2 DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=linux.microsoft.com; s=default; t=1612481034; bh=OndhVLOBSOlhxfWlUBORpyBHTxhkHChscviUHCrmugg=; h=Subject:To:Cc:References:From:Date:In-Reply-To:From; b=Pc8u7bQB2zafW4H9OcRXdd4zfCD7etti6D+z/mvcNw790E1OXQcgHp7HB7DoxGo5q HDIdQFcCJ4KVzGQnh7lAemZUn50HwRcbWmYbMb0Ahu04P3IHGhur97lof5rAWYHbyY Ci84szXcAv4hAB34Pzxi+YE/BVgqSJaCdH34lvCE= Subject: Re: [PATCH v16 11/12] powerpc: Use OF alloc and free for FDT To: Rob Herring Cc: Mimi Zohar , Thiago Jung Bauermann , "AKASHI, Takahiro" , Greg Kroah-Hartman , Will Deacon , Joe Perches , Catalin Marinas , Michael Ellerman , James Morse , Sasha Levin , Benjamin Herrenschmidt , Paul Mackerras , Frank Rowand , vincenzo.frascino@arm.com, Mark Rutland , dmitry.kasatkin@gmail.com, James Morris , "Serge E. Hallyn" , Pavel Tatashin , Allison Randal , Masahiro Yamada , Bhupesh Sharma , Matthias Brugger , Hsin-Yi Wang , tao.li@vivo.com, Christophe Leroy , Prakhar Srivastava , balajib@linux.microsoft.com, linux-integrity@vger.kernel.org, "linux-kernel@vger.kernel.org" , linux-arm-kernel , devicetree@vger.kernel.org, linuxppc-dev References: <20210204164135.29856-1-nramas@linux.microsoft.com> <20210204164135.29856-12-nramas@linux.microsoft.com> From: Lakshmi Ramasubramanian Message-ID: <503d42ba-89bf-4ad9-9d4c-acb625580f77@linux.microsoft.com> Date: Thu, 4 Feb 2021 15:23:53 -0800 User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:68.0) Gecko/20100101 Thunderbird/68.10.0 MIME-Version: 1.0 In-Reply-To: Content-Type: text/plain; charset=utf-8; format=flowed Content-Language: en-US Content-Transfer-Encoding: 7bit Precedence: bulk List-ID: X-Mailing-List: devicetree@vger.kernel.org On 2/4/21 11:26 AM, Rob Herring wrote: > On Thu, Feb 4, 2021 at 10:42 AM Lakshmi Ramasubramanian > wrote: >> >> of_alloc_and_init_fdt() and of_free_fdt() have been defined in >> drivers/of/kexec.c to allocate and free memory for FDT. >> >> Use of_alloc_and_init_fdt() and of_free_fdt() to allocate and >> initialize the FDT, and to free the FDT respectively. >> >> powerpc sets the FDT address in image_loader_data field in >> "struct kimage" and the memory is freed in >> kimage_file_post_load_cleanup(). This cleanup function uses kfree() >> to free the memory. But since of_alloc_and_init_fdt() uses kvmalloc() >> for allocation, the buffer needs to be freed using kvfree(). > > You could just change the kexec core to call kvfree() instead. > >> Define "fdt" field in "struct kimage_arch" for powerpc to store >> the address of FDT, and free the memory in powerpc specific >> arch_kimage_file_post_load_cleanup(). > > However, given all the other buffers have an explicit field in kimage > or kimage_arch, changing powerpc is to match arm64 is better IMO. Just to be clear: I'll leave this as is - free FDT buffer in powerpc's arch_kimage_file_post_load_cleanup() to match arm64 behavior. Will not change "kexec core" to call kvfree() - doing that change would require changing all architectures to use kvmalloc() for image_loader_data allocation. > >> Signed-off-by: Lakshmi Ramasubramanian >> Suggested-by: Rob Herring >> Suggested-by: Thiago Jung Bauermann >> --- >> arch/powerpc/include/asm/kexec.h | 2 ++ >> arch/powerpc/kexec/elf_64.c | 26 ++++++++++++++++---------- >> arch/powerpc/kexec/file_load_64.c | 3 +++ >> 3 files changed, 21 insertions(+), 10 deletions(-) >> >> diff --git a/arch/powerpc/include/asm/kexec.h b/arch/powerpc/include/asm/kexec.h >> index 2c0be93d239a..d7d13cac4d31 100644 >> --- a/arch/powerpc/include/asm/kexec.h >> +++ b/arch/powerpc/include/asm/kexec.h >> @@ -111,6 +111,8 @@ struct kimage_arch { >> unsigned long elf_headers_mem; >> unsigned long elf_headers_sz; >> void *elf_headers; >> + >> + void *fdt; >> }; >> >> char *setup_kdump_cmdline(struct kimage *image, char *cmdline, >> diff --git a/arch/powerpc/kexec/elf_64.c b/arch/powerpc/kexec/elf_64.c >> index d0e459bb2f05..51d2d8eb6c1b 100644 >> --- a/arch/powerpc/kexec/elf_64.c >> +++ b/arch/powerpc/kexec/elf_64.c >> @@ -19,6 +19,7 @@ >> #include >> #include >> #include >> +#include >> #include >> #include >> #include >> @@ -32,7 +33,7 @@ static void *elf64_load(struct kimage *image, char *kernel_buf, >> unsigned int fdt_size; >> unsigned long kernel_load_addr; >> unsigned long initrd_load_addr = 0, fdt_load_addr; >> - void *fdt; >> + void *fdt = NULL; >> const void *slave_code; >> struct elfhdr ehdr; >> char *modified_cmdline = NULL; >> @@ -103,18 +104,12 @@ static void *elf64_load(struct kimage *image, char *kernel_buf, >> } >> >> fdt_size = fdt_totalsize(initial_boot_params) * 2; >> - fdt = kmalloc(fdt_size, GFP_KERNEL); >> + fdt = of_alloc_and_init_fdt(fdt_size); >> if (!fdt) { >> pr_err("Not enough memory for the device tree.\n"); >> ret = -ENOMEM; >> goto out; >> } >> - ret = fdt_open_into(initial_boot_params, fdt, fdt_size); >> - if (ret < 0) { >> - pr_err("Error setting up the new device tree.\n"); >> - ret = -EINVAL; >> - goto out; >> - } >> >> ret = setup_new_fdt_ppc64(image, fdt, initrd_load_addr, > > The first thing this function does is call setup_new_fdt() which first > calls of_kexec_setup_new_fdt(). (Note, I really don't understand the > PPC code split. It looks like there's a 32-bit and 64-bit split, but > 32-bit looks broken to me. Nothing ever calls setup_new_fdt() except > setup_new_fdt_ppc64()). The arm64 version is calling > of_alloc_and_init_fdt() and then of_kexec_setup_new_fdt() directly. > > So we can just make of_alloc_and_init_fdt() also call > of_kexec_setup_new_fdt() (really, just tweak of_kexec_setup_new_fdt do > the alloc and copy). ok - will move fdt allocation into of_kexec_setup_new_fdt(). I don't think the architecture needs to pick the > size either. It's doubtful that either one is that sensitive to the > amount of extra space. I am not clear about the above comment - are you saying the architectures don't need to pass FDT size to the alloc function? arm64 is adding command line string length and some extra space to the size computed from initial_boot_params for FDT Size: buf_size = fdt_totalsize(initial_boot_params) + cmdline_len + DTB_EXTRA_SPACE; powerpc is just using twice the size computed from initial_boot_params fdt_size = fdt_totalsize(initial_boot_params) * 2; I think it would be safe to let arm64 and powerpc pass the required FDT size, along with the other params to of_kexec_setup_new_fdt() - and in this function we allocate FDT and set it up. And, for powerpc leave the remaining code in setup_new_fdt_ppc64(). Would that be ok? > >> initrd_len, cmdline); >> @@ -131,6 +126,10 @@ static void *elf64_load(struct kimage *image, char *kernel_buf, >> ret = kexec_add_buffer(&kbuf); >> if (ret) >> goto out; >> + >> + /* FDT will be freed in arch_kimage_file_post_load_cleanup */ >> + image->arch.fdt = fdt; >> + >> fdt_load_addr = kbuf.mem; >> >> pr_debug("Loaded device tree at 0x%lx\n", fdt_load_addr); >> @@ -145,8 +144,15 @@ static void *elf64_load(struct kimage *image, char *kernel_buf, >> kfree(modified_cmdline); >> kexec_free_elf_info(&elf_info); >> >> - /* Make kimage_file_post_load_cleanup free the fdt buffer for us. */ >> - return ret ? ERR_PTR(ret) : fdt; >> + /* >> + * Once FDT buffer has been successfully passed to kexec_add_buffer(), >> + * the FDT buffer address is saved in image->arch.fdt. In that case, >> + * the memory cannot be freed here in case of any other error. >> + */ >> + if (ret && !image->arch.fdt) >> + of_free_fdt(fdt); > > Just call kvfree() directly. Sure - will do. -lakshmi > >> + >> + return ret ? ERR_PTR(ret) : NULL; >> } >> >> const struct kexec_file_ops kexec_elf64_ops = { >> diff --git a/arch/powerpc/kexec/file_load_64.c b/arch/powerpc/kexec/file_load_64.c >> index 3cab318aa3b9..d9d5b5569a6d 100644 >> --- a/arch/powerpc/kexec/file_load_64.c >> +++ b/arch/powerpc/kexec/file_load_64.c >> @@ -1113,5 +1113,8 @@ int arch_kimage_file_post_load_cleanup(struct kimage *image) >> image->arch.elf_headers = NULL; >> image->arch.elf_headers_sz = 0; >> >> + of_free_fdt(image->arch.fdt); >> + image->arch.fdt = NULL; >> + >> return kexec_image_post_load_cleanup_default(image); >> } >> -- >> 2.30.0 >>