[PATCH v2] firmware: arm_ffa: Add support for IMPDEF value in the memory access descriptor

Sudeep Holla sudeep.holla at arm.com
Mon Sep 22 02:13:32 PDT 2025


On Mon, Sep 22, 2025 at 10:12:43AM +0800, Lixiang Mao wrote:
> I compared your fix with mine which is tested internally, looks no obvious
> difference.

Can I take this as tested-by ?

> I think it is good to mainline.
> 

It's too close to v6.17 release, I will send it as fix at v6.18-rc1, must get
backported to all the relevant stable trees. Thanks for reporting and
reviewing.

> On 9/19/2025 11:52 PM, Sudeep Holla wrote:
> > FF-A v1.2 introduced 16 byte IMPLEMENTATION DEFINED value in the endpoint
> > memory access descriptor to allow any sender could to specify an its any
> > custom value for each receiver. Also this value must be specified by the
> > receiver when retrieving the memory region. The sender must ensure it
> > informs the receiver of this value via an IMPLEMENTATION DEFINED mechanism
> > such as a partition message.
> > 
> > So the FF-A driver can use the message interfaces to communicate the value
> > and set the same in the ffa_mem_region_attributes structures when using
> > the memory interfaces.
> > 
> > The driver ensure that the size of the endpoint memory access descriptors
> > is set correctly based on the FF-A version.
> > 
> > Fixes: 9fac08d9d985 ("firmware: arm_ffa: Upgrade FF-A version to v1.2 in the driver")
> > Reported-by: Lixiang Mao <liximao at qti.qualcomm.com>
> > Signed-off-by: Sudeep Holla <sudeep.holla at arm.com>
> > ---
> >   drivers/firmware/arm_ffa/driver.c | 37 ++++++++++++++++++++++---------
> >   include/linux/arm_ffa.h           | 16 ++++++++++++-
> >   2 files changed, 42 insertions(+), 11 deletions(-)
> > 
> > v1->v2:
> > 	- Fix the size of EMAD in ffa_mem_desc_offset() and the loop
> > 	  increment in ffa_setup_and_transmit()
> > 
> > diff --git a/drivers/firmware/arm_ffa/driver.c b/drivers/firmware/arm_ffa/driver.c
> > index cf6e44429ecd..b68cd9eb09ee 100644
> > --- a/drivers/firmware/arm_ffa/driver.c
> > +++ b/drivers/firmware/arm_ffa/driver.c
> > @@ -654,6 +654,26 @@ static u16 ffa_memory_attributes_get(u32 func_id)
> >   	return FFA_MEM_NORMAL | FFA_MEM_WRITE_BACK | FFA_MEM_INNER_SHAREABLE;
> >   }
> > +static void ffa_emad_impdef_value_init(u32 version, void *dst, void *src)
> > +{
> > +	struct ffa_mem_region_attributes *ep_mem_access;
> > +
> > +	if (version > FFA_VERSION_1_1)
> > +		memcpy(dst, src, sizeof(ep_mem_access->impdef_val));
> > +}
> > +
> > +static void
> > +ffa_mem_region_additional_setup(u32 version, struct ffa_mem_region *mem_region)
> > +{
> > +	if (version <= FFA_VERSION_1_0) {
> > +		mem_region->ep_mem_size = 0;
> > +	} else {
> > +		mem_region->ep_mem_size = ffa_emad_size_get(version);
> > +		mem_region->ep_mem_offset = sizeof(*mem_region);
> > +		memset(mem_region->reserved, 0, 12);
> > +	}
> > +}
> > +
> >   static int
> >   ffa_setup_and_transmit(u32 func_id, void *buffer, u32 max_fragsize,
> >   		       struct ffa_mem_ops_args *args)
> > @@ -672,27 +692,24 @@ ffa_setup_and_transmit(u32 func_id, void *buffer, u32 max_fragsize,
> >   	mem_region->flags = args->flags;
> >   	mem_region->sender_id = drv_info->vm_id;
> >   	mem_region->attributes = ffa_memory_attributes_get(func_id);
> > -	ep_mem_access = buffer +
> > -			ffa_mem_desc_offset(buffer, 0, drv_info->version);
> >   	composite_offset = ffa_mem_desc_offset(buffer, args->nattrs,
> >   					       drv_info->version);
> > -	for (idx = 0; idx < args->nattrs; idx++, ep_mem_access++) {
> > +	for (idx = 0; idx < args->nattrs; idx++) {
> > +		ep_mem_access = buffer +
> > +			ffa_mem_desc_offset(buffer, idx, drv_info->version);
> >   		ep_mem_access->receiver = args->attrs[idx].receiver;
> >   		ep_mem_access->attrs = args->attrs[idx].attrs;
> >   		ep_mem_access->composite_off = composite_offset;
> >   		ep_mem_access->flag = 0;
> >   		ep_mem_access->reserved = 0;
> > +		ffa_emad_impdef_value_init(drv_info->version,
> > +					   ep_mem_access->impdef_val,
> > +					   args->attrs[idx].impdef_val);
> >   	}
> >   	mem_region->handle = 0;
> >   	mem_region->ep_count = args->nattrs;
> > -	if (drv_info->version <= FFA_VERSION_1_0) {
> > -		mem_region->ep_mem_size = 0;
> > -	} else {
> > -		mem_region->ep_mem_size = sizeof(*ep_mem_access);
> > -		mem_region->ep_mem_offset = sizeof(*mem_region);
> > -		memset(mem_region->reserved, 0, 12);
> > -	}
> > +	ffa_mem_region_additional_setup(drv_info->version, mem_region);
> >   	composite = buffer + composite_offset;
> >   	composite->total_pg_cnt = ffa_get_num_pages_sg(args->sg);
> > diff --git a/include/linux/arm_ffa.h b/include/linux/arm_ffa.h
> > index 0a0132c6ae3e..6e180fc0e68e 100644
> > --- a/include/linux/arm_ffa.h
> > +++ b/include/linux/arm_ffa.h
> > @@ -347,6 +347,7 @@ struct ffa_mem_region_attributes {
> >   	 * an `struct ffa_mem_region_addr_range`.
> >   	 */
> >   	u32 composite_off;
> > +	u8 impdef_val[16];
> >   	u64 reserved;
> >   };
> > @@ -426,10 +427,23 @@ struct ffa_mem_region {
> >   #define CONSTITUENTS_OFFSET(x)	\
> >   	(offsetof(struct ffa_composite_mem_region, constituents[x]))
> > +static inline u32 ffa_emad_size_get(u32 ffa_version)
> > +{
> > +	u32 sz;
> > +	struct ffa_mem_region_attributes *ep_mem_access;
> > +
> > +	if (ffa_version <= FFA_VERSION_1_1)
> > +		sz = sizeof(*ep_mem_access) - sizeof(ep_mem_access->impdef_val);
> > +	else
> > +		sz = sizeof(*ep_mem_access);
> > +
> > +	return sz;
> > +}
> > +
> >   static inline u32
> >   ffa_mem_desc_offset(struct ffa_mem_region *buf, int count, u32 ffa_version)
> >   {
> > -	u32 offset = count * sizeof(struct ffa_mem_region_attributes);
> > +	u32 offset = count * ffa_emad_size_get(ffa_version);
> >   	/*
> >   	 * Earlier to v1.1, the endpoint memory descriptor array started at
> >   	 * offset 32(i.e. offset of ep_mem_offset in the current structure)

-- 
Regards,
Sudeep



More information about the linux-arm-kernel mailing list