From nobody Wed Jan 15 11:38:10 2025 Delivered-To: importer@patchew.org Received-SPF: pass (zohomail.com: domain of lists.xenproject.org designates 192.237.175.120 as permitted sender) client-ip=192.237.175.120; envelope-from=xen-devel-bounces@lists.xenproject.org; helo=lists.xenproject.org; Authentication-Results: mx.zohomail.com; spf=pass (zohomail.com: domain of lists.xenproject.org designates 192.237.175.120 as permitted sender) smtp.mailfrom=xen-devel-bounces@lists.xenproject.org; dmarc=fail(p=none dis=none) header.from=arm.com Return-Path: Received: from lists.xenproject.org (lists.xenproject.org [192.237.175.120]) by mx.zohomail.com with SMTPS id 1694405164365195.92011884976364; Sun, 10 Sep 2023 21:06:04 -0700 (PDT) Received: from list by lists.xenproject.org with outflank-mailman.598909.934132 (Exim 4.92) (envelope-from ) id 1qfYB8-0007J2-6U; Mon, 11 Sep 2023 04:05:38 +0000 Received: by outflank-mailman (output) from mailman id 598909.934132; Mon, 11 Sep 2023 04:05:38 +0000 Received: from localhost ([127.0.0.1] helo=lists.xenproject.org) by lists.xenproject.org with esmtp (Exim 4.92) (envelope-from ) id 1qfYB8-0007I0-16; Mon, 11 Sep 2023 04:05:38 +0000 Received: by outflank-mailman (input) for mailman id 598909; Mon, 11 Sep 2023 04:05:36 +0000 Received: from se1-gles-sth1-in.inumbo.com ([159.253.27.254] helo=se1-gles-sth1.inumbo.com) by lists.xenproject.org with esmtp (Exim 4.92) (envelope-from ) id 1qfYB6-0005nR-QU for xen-devel@lists.xenproject.org; Mon, 11 Sep 2023 04:05:36 +0000 Received: from foss.arm.com (foss.arm.com [217.140.110.172]) by se1-gles-sth1.inumbo.com (Halon) with ESMTP id 759a6e87-5058-11ee-8784-cb3800f73035; Mon, 11 Sep 2023 06:05:35 +0200 (CEST) Received: from usa-sjc-imap-foss1.foss.arm.com (unknown [10.121.207.14]) by usa-sjc-mx-foss1.foss.arm.com (Postfix) with ESMTP id 4B3E2D75; Sun, 10 Sep 2023 21:06:12 -0700 (PDT) Received: from a011292.shanghai.arm.com (a011292.shanghai.arm.com [10.169.190.94]) by usa-sjc-imap-foss1.foss.arm.com (Postfix) with ESMTPA id 2E36D3F67D; Sun, 10 Sep 2023 21:05:31 -0700 (PDT) X-Outflank-Mailman: Message body and most headers restored to incoming version X-BeenThere: xen-devel@lists.xenproject.org List-Id: Xen developer discussion List-Unsubscribe: , List-Post: List-Help: List-Subscribe: , Errors-To: xen-devel-bounces@lists.xenproject.org Precedence: list Sender: "Xen-devel" X-Inumbo-ID: 759a6e87-5058-11ee-8784-cb3800f73035 From: Penny Zheng To: xen-devel@lists.xenproject.org Cc: michal.orzel@amd.com, wei.chen@arm.com, Penny Zheng , Stefano Stabellini , Julien Grall , Bertrand Marquis , Volodymyr Babchuk , Penny Zheng Subject: [PATCH v4 05/10] xen/arm: support static shared memory when host address not provided Date: Mon, 11 Sep 2023 12:04:37 +0800 Message-Id: <20230911040442.2541398-6-Penny.Zheng@arm.com> X-Mailer: git-send-email 2.25.1 In-Reply-To: <20230911040442.2541398-1-Penny.Zheng@arm.com> References: <20230911040442.2541398-1-Penny.Zheng@arm.com> MIME-Version: 1.0 Content-Transfer-Encoding: quoted-printable X-ZM-MESSAGEID: 1694405166495100001 Content-Type: text/plain; charset="utf-8" In order to support static shared memory when host address not provided, we shall do the following modification: - we shall let Xen allocate memory from heap for static shared memory at first domain, no matter it is owner or borrower. - In acquire_shared_memory_bank, as static shared memory has already been allocated from heap, we shall assign them to the owner domain using function "assign_pages". - Function get_shm_pages_reference is created to add as many additional reference as the number of borrowers. - We implement a new helper "add_foreign_mapping_for_borrower" to set up foreign memory mapping for borrower. Instead of using multiple function parameters to deliver various shm-related info, like host physical address, SHMID, etc, and with the introduction of new struct "shm_memnode" to include banked host memory info, we switch to use "shm_memnode" as function parameter to replace them all, to make codes = more clear and tidy. Signed-off-by: Penny Zheng --- v1 -> v2: - combine commits 4 - 6 in Serie 1 - Adapt to changes of introducing "struct shm_memnode" --- v2 -> v3 - fix infinite loop bug and bad indentation - rebase --- v3 -> v4: rebase and no change --- xen/arch/arm/domain_build.c | 223 +++++++++++++++++++++++++----------- 1 file changed, 155 insertions(+), 68 deletions(-) diff --git a/xen/arch/arm/domain_build.c b/xen/arch/arm/domain_build.c index 0116bacc2f..2eae50d78b 100644 --- a/xen/arch/arm/domain_build.c +++ b/xen/arch/arm/domain_build.c @@ -891,6 +891,11 @@ static void __init assign_static_memory_11(struct doma= in *d, } =20 #ifdef CONFIG_STATIC_SHM +static bool __init is_shm_allocated_from_heap(struct shm_memnode *node) +{ + return (node->meminfo.nr_banks !=3D 0); +} + static int __init acquire_nr_borrower_domain(const char *shm_id, unsigned long *nr_borrowers) { @@ -934,12 +939,12 @@ static struct shm_memnode * __init find_shm_memnode(c= onst char *shm_id) * This function checks whether the static shared memory region is * already allocated to dom_io. */ -static bool __init is_shm_allocated_to_domio(paddr_t pbase) +static bool __init is_shm_allocated_to_domio(struct shm_memnode *node) { struct page_info *page; struct domain *d; =20 - page =3D maddr_to_page(pbase); + page =3D maddr_to_page(node->meminfo.bank[0].start); d =3D page_get_owner_and_reference(page); if ( d =3D=3D NULL ) return false; @@ -957,67 +962,128 @@ static bool __init is_shm_allocated_to_domio(paddr_t= pbase) } =20 static mfn_t __init acquire_shared_memory_bank(struct domain *d, - paddr_t pbase, paddr_t psiz= e) + struct shm_meminfo *meminfo, + bool paddr_assigned) { - mfn_t smfn; - unsigned long nr_pfns; - int res; + int res, i =3D 0; =20 - /* - * Pages of statically shared memory shall be included - * into domain_tot_pages(). - */ - nr_pfns =3D PFN_DOWN(psize); - if ( (UINT_MAX - d->max_pages) < nr_pfns ) + for ( ; i < meminfo->nr_banks; i++ ) { - printk(XENLOG_ERR "%pd: Over-allocation for d->max_pages: %lu.\n", - d, nr_pfns); - return INVALID_MFN; + paddr_t pbase =3D meminfo->bank[i].start, psize =3D meminfo->bank[= i].size; + unsigned long nr_pfns; + + /* + * Pages of statically shared memory shall be included + * into domain_tot_pages(). + */ + nr_pfns =3D PFN_DOWN(psize); + if ( (UINT_MAX - d->max_pages) < nr_pfns ) + { + printk(XENLOG_ERR "%pd: Over-allocation for d->max_pages: %lu.= \n", + d, nr_pfns); + return INVALID_MFN; + } + d->max_pages +=3D nr_pfns; + + if ( paddr_assigned ) + { + res =3D acquire_domstatic_pages(d, maddr_to_mfn(pbase), nr_pfn= s, 0); + if ( res ) + { + printk(XENLOG_ERR + "%pd: failed to acquire static memory: %d.\n", d, r= es); + goto fail; + } + } + else + /* + * When host address is not provided, static shared memory is + * allocated from heap and shall be assigned to owner domain. + */ + if ( assign_pages(maddr_to_page(pbase), nr_pfns, d, 0) ) + goto fail; } - d->max_pages +=3D nr_pfns; =20 - smfn =3D maddr_to_mfn(pbase); - res =3D acquire_domstatic_pages(d, smfn, nr_pfns, 0); - if ( res ) + return maddr_to_mfn(meminfo->bank[0].start); + + fail: + while( --i >=3D 0 ) + d->max_pages -=3D PFN_DOWN(meminfo->bank[i].size); + return INVALID_MFN; +} + +static int __init get_shm_pages_reference(struct domain *d, + struct shm_meminfo *meminfo, + unsigned long count) +{ + struct page_info *page; + int i =3D 0, j; + + for ( ; i < meminfo->nr_banks; i++ ) { - printk(XENLOG_ERR - "%pd: failed to acquire static memory: %d.\n", d, res); - d->max_pages -=3D nr_pfns; - return INVALID_MFN; + paddr_t pbase =3D meminfo->bank[i].start, psize =3D meminfo->bank[= i].size; + unsigned long nr_pages =3D PFN_DOWN(psize); + + page =3D maddr_to_page(pbase); + for ( j =3D 0; j < nr_pages; j++ ) + { + if ( !get_page_nr(page + j, d, count) ) + { + printk(XENLOG_ERR + "Failed to add %lu references to page %"PRI_mfn".\n= ", + count, mfn_x(page_to_mfn(page + j))); + goto fail; + } + } } =20 - return smfn; + return 0; + + fail: + while ( --j >=3D 0 ) + put_page_nr(page + j, count); + while ( --i >=3D 0 ) + { + page =3D maddr_to_page(meminfo->bank[i].start); + j =3D PFN_DOWN(meminfo->bank[i].size); + while ( --j >=3D 0 ) + put_page_nr(page + j, count); + } + return -EINVAL; } =20 static int __init assign_shared_memory(struct domain *d, - paddr_t pbase, paddr_t psize, - paddr_t gbase, const char *shm_id) + struct shm_memnode *node, paddr_t g= base, + bool paddr_assigned) { mfn_t smfn; - int ret =3D 0; - unsigned long nr_pages, nr_borrowers, i; - struct page_info *page; - - printk("%pd: allocate static shared memory BANK %#"PRIpaddr"-%#"PRIpad= dr".\n", - d, pbase, pbase + psize); + int ret; + unsigned long nr_borrowers, i; + struct shm_meminfo *meminfo =3D &node->meminfo; =20 - smfn =3D acquire_shared_memory_bank(d, pbase, psize); + smfn =3D acquire_shared_memory_bank(d, meminfo, paddr_assigned); if ( mfn_eq(smfn, INVALID_MFN) ) return -EINVAL; =20 - /* - * DOMID_IO is not auto-translated (i.e. it sees RAM 1:1). So we do no= t need - * to create mapping in the P2M. - */ - nr_pages =3D PFN_DOWN(psize); - if ( d !=3D dom_io ) + for ( i =3D 0; i < meminfo->nr_banks; i++ ) { - ret =3D guest_physmap_add_pages(d, gaddr_to_gfn(gbase), smfn, - PFN_DOWN(psize)); - if ( ret ) + paddr_t pbase =3D meminfo->bank[i].start, psize =3D meminfo->bank[= i].size; + + /* + * DOMID_IO is not auto-translated (i.e. it sees RAM 1:1). So we do + * not need to create mapping in the P2M. + */ + if ( d !=3D dom_io ) { - printk(XENLOG_ERR "Failed to map shared memory to %pd.\n", d); - return ret; + ret =3D guest_physmap_add_pages(d, gaddr_to_gfn(gbase), + maddr_to_mfn(pbase), + PFN_DOWN(psize)); + if ( ret ) + { + printk(XENLOG_ERR "Failed to map shared memory to %pd.\n",= d); + return ret; + } + gbase +=3D psize; } } =20 @@ -1025,7 +1091,7 @@ static int __init assign_shared_memory(struct domain = *d, * Get the right amount of references per page, which is the number of * borrower domains. */ - ret =3D acquire_nr_borrower_domain(shm_id, &nr_borrowers); + ret =3D acquire_nr_borrower_domain(node->shm_id, &nr_borrowers); if ( ret ) return ret; =20 @@ -1037,24 +1103,30 @@ static int __init assign_shared_memory(struct domai= n *d, * So if the borrower is created first, it will cause adding pages * in the P2M without reference. */ - page =3D mfn_to_page(smfn); - for ( i =3D 0; i < nr_pages; i++ ) + return get_shm_pages_reference(d, meminfo, nr_borrowers); +} + +static int __init add_foreign_mapping_for_borrower(struct domain *d, + struct shm_memnode *nod= e, + paddr_t gbase) +{ + unsigned int i =3D 0; + struct shm_meminfo *meminfo =3D &node->meminfo; + + for ( ; i < meminfo->nr_banks; i++ ) { - if ( !get_page_nr(page + i, d, nr_borrowers) ) - { - printk(XENLOG_ERR - "Failed to add %lu references to page %"PRI_mfn".\n", - nr_borrowers, mfn_x(smfn) + i); - goto fail; - } + paddr_t pbase =3D meminfo->bank[i].start, psize =3D meminfo->bank[= i].size; + int ret; + + /* Set up P2M foreign mapping for borrower domain. */ + ret =3D map_regions_p2mt(d, _gfn(PFN_UP(gbase)), PFN_DOWN(psize), + _mfn(PFN_UP(pbase)), p2m_map_foreign_rw); + if ( ret ) + return ret; + gbase +=3D psize; } =20 return 0; - - fail: - while ( --i >=3D 0 ) - put_page_nr(page + i, nr_borrowers); - return ret; } =20 static int __init append_shm_bank_to_domain(struct kernel_info *kinfo, @@ -1178,7 +1250,7 @@ static int __init process_shm(struct domain *d, struc= t kernel_info *kinfo, =20 dt_for_each_child_node(node, shm_node) { - paddr_t gbase, pbase, psize; + paddr_t gbase; int ret =3D 0; const char *role_str; const char *shm_id; @@ -1207,15 +1279,30 @@ static int __init process_shm(struct domain *d, str= uct kernel_info *kinfo, shm_id); if ( !shm_memnode ) return -EINVAL; - pbase =3D shm_memnode->meminfo.bank[0].start; - psize =3D shm_memnode->meminfo.bank[0].size; + + /* + * When host address is not provided in "xen,shared-mem", + * we let Xen allocate memory from heap at first domain. + */ + if ( !paddr_assigned && !is_shm_allocated_from_heap(shm_memnode) ) + { + if ( !allocate_domheap_memory(NULL, shm_memnode->meminfo.tot_s= ize, + (void *)&shm_memnode->meminfo, + SHM_MEMINFO) ) + { + printk(XENLOG_ERR + "Failed to allocate (%"PRIpaddr"MB) pages as static= shared memory from heap\n", + shm_memnode->meminfo.tot_size >> 20); + return -EINVAL; + } + } =20 /* * DOMID_IO is a fake domain and is not described in the Device-Tr= ee. * Therefore when the owner of the shared region is DOMID_IO, we w= ill * only find the borrowers. */ - if ( (owner_dom_io && !is_shm_allocated_to_domio(pbase)) || + if ( (owner_dom_io && !is_shm_allocated_to_domio(shm_memnode)) || (!owner_dom_io && strcmp(role_str, "owner") =3D=3D 0) ) { /* @@ -1223,16 +1310,14 @@ static int __init process_shm(struct domain *d, str= uct kernel_info *kinfo, * specified, so they should be assigned to dom_io. */ ret =3D assign_shared_memory(owner_dom_io ? dom_io : d, - pbase, psize, gbase, shm_id); + shm_memnode, gbase, paddr_assigned); if ( ret ) return ret; } =20 if ( owner_dom_io || (strcmp(role_str, "borrower") =3D=3D 0) ) { - /* Set up P2M foreign mapping for borrower domain. */ - ret =3D map_regions_p2mt(d, _gfn(PFN_UP(gbase)), PFN_DOWN(psiz= e), - _mfn(PFN_UP(pbase)), p2m_map_foreign_rw= ); + ret =3D add_foreign_mapping_for_borrower(d, shm_memnode, gbase= ); if ( ret ) return ret; } @@ -1241,7 +1326,9 @@ static int __init process_shm(struct domain *d, struc= t kernel_info *kinfo, * Record static shared memory region info for later setting * up shm-node in guest device tree. */ - ret =3D append_shm_bank_to_domain(kinfo, gbase, psize, shm_id); + ret =3D append_shm_bank_to_domain(kinfo, gbase, + shm_memnode->meminfo.tot_size, + shm_memnode->shm_id); if ( ret ) return ret; } --=20 2.25.1