xref: /linux-6.15/include/linux/arm_ffa.h (revision c10debfe)
1e7818584SSudeep Holla /* SPDX-License-Identifier: GPL-2.0-only */
2e7818584SSudeep Holla /*
3e7818584SSudeep Holla  * Copyright (C) 2021 ARM Ltd.
4e7818584SSudeep Holla  */
5e7818584SSudeep Holla 
6e7818584SSudeep Holla #ifndef _LINUX_ARM_FFA_H
7e7818584SSudeep Holla #define _LINUX_ARM_FFA_H
8e7818584SSudeep Holla 
911358053SSudeep Holla #include <linux/bitfield.h>
10e7818584SSudeep Holla #include <linux/device.h>
11e7818584SSudeep Holla #include <linux/module.h>
12e7818584SSudeep Holla #include <linux/types.h>
13e7818584SSudeep Holla #include <linux/uuid.h>
14e7818584SSudeep Holla 
15229d58e3SWill Deacon #define FFA_SMC(calling_convention, func_num)				\
16229d58e3SWill Deacon 	ARM_SMCCC_CALL_VAL(ARM_SMCCC_FAST_CALL, (calling_convention),	\
17229d58e3SWill Deacon 			   ARM_SMCCC_OWNER_STANDARD, (func_num))
18229d58e3SWill Deacon 
19229d58e3SWill Deacon #define FFA_SMC_32(func_num)	FFA_SMC(ARM_SMCCC_SMC_32, (func_num))
20229d58e3SWill Deacon #define FFA_SMC_64(func_num)	FFA_SMC(ARM_SMCCC_SMC_64, (func_num))
21229d58e3SWill Deacon 
22229d58e3SWill Deacon #define FFA_ERROR			FFA_SMC_32(0x60)
23229d58e3SWill Deacon #define FFA_SUCCESS			FFA_SMC_32(0x61)
241609626cSSudeep Holla #define FFA_FN64_SUCCESS		FFA_SMC_64(0x61)
25229d58e3SWill Deacon #define FFA_INTERRUPT			FFA_SMC_32(0x62)
26229d58e3SWill Deacon #define FFA_VERSION			FFA_SMC_32(0x63)
27229d58e3SWill Deacon #define FFA_FEATURES			FFA_SMC_32(0x64)
28229d58e3SWill Deacon #define FFA_RX_RELEASE			FFA_SMC_32(0x65)
29229d58e3SWill Deacon #define FFA_RXTX_MAP			FFA_SMC_32(0x66)
30229d58e3SWill Deacon #define FFA_FN64_RXTX_MAP		FFA_SMC_64(0x66)
31229d58e3SWill Deacon #define FFA_RXTX_UNMAP			FFA_SMC_32(0x67)
32229d58e3SWill Deacon #define FFA_PARTITION_INFO_GET		FFA_SMC_32(0x68)
33229d58e3SWill Deacon #define FFA_ID_GET			FFA_SMC_32(0x69)
34229d58e3SWill Deacon #define FFA_MSG_POLL			FFA_SMC_32(0x6A)
35229d58e3SWill Deacon #define FFA_MSG_WAIT			FFA_SMC_32(0x6B)
36229d58e3SWill Deacon #define FFA_YIELD			FFA_SMC_32(0x6C)
37229d58e3SWill Deacon #define FFA_RUN				FFA_SMC_32(0x6D)
38229d58e3SWill Deacon #define FFA_MSG_SEND			FFA_SMC_32(0x6E)
39229d58e3SWill Deacon #define FFA_MSG_SEND_DIRECT_REQ		FFA_SMC_32(0x6F)
40229d58e3SWill Deacon #define FFA_FN64_MSG_SEND_DIRECT_REQ	FFA_SMC_64(0x6F)
41229d58e3SWill Deacon #define FFA_MSG_SEND_DIRECT_RESP	FFA_SMC_32(0x70)
42229d58e3SWill Deacon #define FFA_FN64_MSG_SEND_DIRECT_RESP	FFA_SMC_64(0x70)
43229d58e3SWill Deacon #define FFA_MEM_DONATE			FFA_SMC_32(0x71)
44229d58e3SWill Deacon #define FFA_FN64_MEM_DONATE		FFA_SMC_64(0x71)
45229d58e3SWill Deacon #define FFA_MEM_LEND			FFA_SMC_32(0x72)
46229d58e3SWill Deacon #define FFA_FN64_MEM_LEND		FFA_SMC_64(0x72)
47229d58e3SWill Deacon #define FFA_MEM_SHARE			FFA_SMC_32(0x73)
48229d58e3SWill Deacon #define FFA_FN64_MEM_SHARE		FFA_SMC_64(0x73)
49229d58e3SWill Deacon #define FFA_MEM_RETRIEVE_REQ		FFA_SMC_32(0x74)
50229d58e3SWill Deacon #define FFA_FN64_MEM_RETRIEVE_REQ	FFA_SMC_64(0x74)
51229d58e3SWill Deacon #define FFA_MEM_RETRIEVE_RESP		FFA_SMC_32(0x75)
52229d58e3SWill Deacon #define FFA_MEM_RELINQUISH		FFA_SMC_32(0x76)
53229d58e3SWill Deacon #define FFA_MEM_RECLAIM			FFA_SMC_32(0x77)
54229d58e3SWill Deacon #define FFA_MEM_OP_PAUSE		FFA_SMC_32(0x78)
55229d58e3SWill Deacon #define FFA_MEM_OP_RESUME		FFA_SMC_32(0x79)
56229d58e3SWill Deacon #define FFA_MEM_FRAG_RX			FFA_SMC_32(0x7A)
57229d58e3SWill Deacon #define FFA_MEM_FRAG_TX			FFA_SMC_32(0x7B)
58229d58e3SWill Deacon #define FFA_NORMAL_WORLD_RESUME		FFA_SMC_32(0x7C)
591609626cSSudeep Holla #define FFA_NOTIFICATION_BITMAP_CREATE	FFA_SMC_32(0x7D)
601609626cSSudeep Holla #define FFA_NOTIFICATION_BITMAP_DESTROY FFA_SMC_32(0x7E)
611609626cSSudeep Holla #define FFA_NOTIFICATION_BIND		FFA_SMC_32(0x7F)
621609626cSSudeep Holla #define FFA_NOTIFICATION_UNBIND		FFA_SMC_32(0x80)
631609626cSSudeep Holla #define FFA_NOTIFICATION_SET		FFA_SMC_32(0x81)
641609626cSSudeep Holla #define FFA_NOTIFICATION_GET		FFA_SMC_32(0x82)
651609626cSSudeep Holla #define FFA_NOTIFICATION_INFO_GET	FFA_SMC_32(0x83)
661609626cSSudeep Holla #define FFA_FN64_NOTIFICATION_INFO_GET	FFA_SMC_64(0x83)
671609626cSSudeep Holla #define FFA_RX_ACQUIRE			FFA_SMC_32(0x84)
681609626cSSudeep Holla #define FFA_SPM_ID_GET			FFA_SMC_32(0x85)
691609626cSSudeep Holla #define FFA_MSG_SEND2			FFA_SMC_32(0x86)
701609626cSSudeep Holla #define FFA_SECONDARY_EP_REGISTER	FFA_SMC_32(0x87)
711609626cSSudeep Holla #define FFA_FN64_SECONDARY_EP_REGISTER	FFA_SMC_64(0x87)
721609626cSSudeep Holla #define FFA_MEM_PERM_GET		FFA_SMC_32(0x88)
731609626cSSudeep Holla #define FFA_FN64_MEM_PERM_GET		FFA_SMC_64(0x88)
741609626cSSudeep Holla #define FFA_MEM_PERM_SET		FFA_SMC_32(0x89)
751609626cSSudeep Holla #define FFA_FN64_MEM_PERM_SET		FFA_SMC_64(0x89)
767c432a18SSudeep Holla #define FFA_CONSOLE_LOG			FFA_SMC_32(0x8A)
777c432a18SSudeep Holla #define FFA_PARTITION_INFO_GET_REGS	FFA_SMC_64(0x8B)
787c432a18SSudeep Holla #define FFA_EL3_INTR_HANDLE		FFA_SMC_32(0x8C)
797c432a18SSudeep Holla #define FFA_MSG_SEND_DIRECT_REQ2	FFA_SMC_64(0x8D)
807c432a18SSudeep Holla #define FFA_MSG_SEND_DIRECT_RESP2	FFA_SMC_64(0x8E)
81229d58e3SWill Deacon 
82229d58e3SWill Deacon /*
83229d58e3SWill Deacon  * For some calls it is necessary to use SMC64 to pass or return 64-bit values.
84229d58e3SWill Deacon  * For such calls FFA_FN_NATIVE(name) will choose the appropriate
85229d58e3SWill Deacon  * (native-width) function ID.
86229d58e3SWill Deacon  */
87229d58e3SWill Deacon #ifdef CONFIG_64BIT
88229d58e3SWill Deacon #define FFA_FN_NATIVE(name)	FFA_FN64_##name
89229d58e3SWill Deacon #else
90229d58e3SWill Deacon #define FFA_FN_NATIVE(name)	FFA_##name
91229d58e3SWill Deacon #endif
92229d58e3SWill Deacon 
93229d58e3SWill Deacon /* FFA error codes. */
94229d58e3SWill Deacon #define FFA_RET_SUCCESS            (0)
95229d58e3SWill Deacon #define FFA_RET_NOT_SUPPORTED      (-1)
96229d58e3SWill Deacon #define FFA_RET_INVALID_PARAMETERS (-2)
97229d58e3SWill Deacon #define FFA_RET_NO_MEMORY          (-3)
98229d58e3SWill Deacon #define FFA_RET_BUSY               (-4)
99229d58e3SWill Deacon #define FFA_RET_INTERRUPTED        (-5)
100229d58e3SWill Deacon #define FFA_RET_DENIED             (-6)
101229d58e3SWill Deacon #define FFA_RET_RETRY              (-7)
102229d58e3SWill Deacon #define FFA_RET_ABORTED            (-8)
1031609626cSSudeep Holla #define FFA_RET_NO_DATA            (-9)
104229d58e3SWill Deacon 
105229d58e3SWill Deacon /* FFA version encoding */
106229d58e3SWill Deacon #define FFA_MAJOR_VERSION_MASK	GENMASK(30, 16)
107229d58e3SWill Deacon #define FFA_MINOR_VERSION_MASK	GENMASK(15, 0)
108229d58e3SWill Deacon #define FFA_MAJOR_VERSION(x)	((u16)(FIELD_GET(FFA_MAJOR_VERSION_MASK, (x))))
109229d58e3SWill Deacon #define FFA_MINOR_VERSION(x)	((u16)(FIELD_GET(FFA_MINOR_VERSION_MASK, (x))))
110229d58e3SWill Deacon #define FFA_PACK_VERSION_INFO(major, minor)			\
111229d58e3SWill Deacon 	(FIELD_PREP(FFA_MAJOR_VERSION_MASK, (major)) |		\
112229d58e3SWill Deacon 	 FIELD_PREP(FFA_MINOR_VERSION_MASK, (minor)))
113229d58e3SWill Deacon #define FFA_VERSION_1_0		FFA_PACK_VERSION_INFO(1, 0)
1141609626cSSudeep Holla #define FFA_VERSION_1_1		FFA_PACK_VERSION_INFO(1, 1)
1159fac08d9SSudeep Holla #define FFA_VERSION_1_2		FFA_PACK_VERSION_INFO(1, 2)
116229d58e3SWill Deacon 
117229d58e3SWill Deacon /**
118229d58e3SWill Deacon  * FF-A specification mentions explicitly about '4K pages'. This should
119229d58e3SWill Deacon  * not be confused with the kernel PAGE_SIZE, which is the translation
120229d58e3SWill Deacon  * granule kernel is configured and may be one among 4K, 16K and 64K.
121229d58e3SWill Deacon  */
122229d58e3SWill Deacon #define FFA_PAGE_SIZE		SZ_4K
123229d58e3SWill Deacon 
1249d0c6a9aSWill Deacon /*
1259d0c6a9aSWill Deacon  * Minimum buffer size/alignment encodings returned by an FFA_FEATURES
1269d0c6a9aSWill Deacon  * query for FFA_RXTX_MAP.
1279d0c6a9aSWill Deacon  */
1289d0c6a9aSWill Deacon #define FFA_FEAT_RXTX_MIN_SZ_4K		0
1299d0c6a9aSWill Deacon #define FFA_FEAT_RXTX_MIN_SZ_64K	1
1309d0c6a9aSWill Deacon #define FFA_FEAT_RXTX_MIN_SZ_16K	2
1319d0c6a9aSWill Deacon 
132e7818584SSudeep Holla /* FFA Bus/Device/Driver related */
133e7818584SSudeep Holla struct ffa_device {
13419b87664SSudeep Holla 	u32 id;
1353c258bf6SSudeep Holla 	u32 properties;
136e7818584SSudeep Holla 	int vm_id;
137d0c0bce8SSudeep Holla 	bool mode_32bit;
138e7818584SSudeep Holla 	uuid_t uuid;
139e7818584SSudeep Holla 	struct device dev;
1407aa7a979SSudeep Holla 	const struct ffa_ops *ops;
141e7818584SSudeep Holla };
142e7818584SSudeep Holla 
143e7818584SSudeep Holla #define to_ffa_dev(d) container_of(d, struct ffa_device, dev)
144e7818584SSudeep Holla 
145e7818584SSudeep Holla struct ffa_device_id {
146e7818584SSudeep Holla 	uuid_t uuid;
147e7818584SSudeep Holla };
148e7818584SSudeep Holla 
149e7818584SSudeep Holla struct ffa_driver {
150e7818584SSudeep Holla 	const char *name;
151e7818584SSudeep Holla 	int (*probe)(struct ffa_device *sdev);
152e7818584SSudeep Holla 	void (*remove)(struct ffa_device *sdev);
153e7818584SSudeep Holla 	const struct ffa_device_id *id_table;
154e7818584SSudeep Holla 
155e7818584SSudeep Holla 	struct device_driver driver;
156e7818584SSudeep Holla };
157e7818584SSudeep Holla 
158d69d8048SGreg Kroah-Hartman #define to_ffa_driver(d) container_of_const(d, struct ffa_driver, driver)
159e7818584SSudeep Holla 
ffa_dev_set_drvdata(struct ffa_device * fdev,void * data)160e7818584SSudeep Holla static inline void ffa_dev_set_drvdata(struct ffa_device *fdev, void *data)
161e7818584SSudeep Holla {
162498af8d1SSudeep Holla 	dev_set_drvdata(&fdev->dev, data);
163498af8d1SSudeep Holla }
164498af8d1SSudeep Holla 
ffa_dev_get_drvdata(struct ffa_device * fdev)165498af8d1SSudeep Holla static inline void *ffa_dev_get_drvdata(struct ffa_device *fdev)
166498af8d1SSudeep Holla {
167498af8d1SSudeep Holla 	return dev_get_drvdata(&fdev->dev);
168e7818584SSudeep Holla }
169e7818584SSudeep Holla 
1706fe437cfSLevi Yun struct ffa_partition_info;
1716fe437cfSLevi Yun 
172e7818584SSudeep Holla #if IS_REACHABLE(CONFIG_ARM_FFA_TRANSPORT)
1736fe437cfSLevi Yun struct ffa_device *
1746fe437cfSLevi Yun ffa_device_register(const struct ffa_partition_info *part_info,
1757aa7a979SSudeep Holla 		    const struct ffa_ops *ops);
176e7818584SSudeep Holla void ffa_device_unregister(struct ffa_device *ffa_dev);
177e7818584SSudeep Holla int ffa_driver_register(struct ffa_driver *driver, struct module *owner,
178e7818584SSudeep Holla 			const char *mod_name);
179e7818584SSudeep Holla void ffa_driver_unregister(struct ffa_driver *driver);
18046dcd68aSSudeep Holla void ffa_devices_unregister(void);
181e7818584SSudeep Holla bool ffa_device_is_valid(struct ffa_device *ffa_dev);
182e7818584SSudeep Holla 
183e7818584SSudeep Holla #else
1846fe437cfSLevi Yun static inline struct ffa_device *
ffa_device_register(const struct ffa_partition_info * part_info,const struct ffa_ops * ops)1856fe437cfSLevi Yun ffa_device_register(const struct ffa_partition_info *part_info,
1867aa7a979SSudeep Holla 		    const struct ffa_ops *ops)
187e7818584SSudeep Holla {
188e7818584SSudeep Holla 	return NULL;
189e7818584SSudeep Holla }
190e7818584SSudeep Holla 
ffa_device_unregister(struct ffa_device * dev)191e7818584SSudeep Holla static inline void ffa_device_unregister(struct ffa_device *dev) {}
192e7818584SSudeep Holla 
ffa_devices_unregister(void)19346dcd68aSSudeep Holla static inline void ffa_devices_unregister(void) {}
19446dcd68aSSudeep Holla 
195e7818584SSudeep Holla static inline int
ffa_driver_register(struct ffa_driver * driver,struct module * owner,const char * mod_name)196e7818584SSudeep Holla ffa_driver_register(struct ffa_driver *driver, struct module *owner,
197e7818584SSudeep Holla 		    const char *mod_name)
198e7818584SSudeep Holla {
199e7818584SSudeep Holla 	return -EINVAL;
200e7818584SSudeep Holla }
201e7818584SSudeep Holla 
ffa_driver_unregister(struct ffa_driver * driver)202e7818584SSudeep Holla static inline void ffa_driver_unregister(struct ffa_driver *driver) {}
203e7818584SSudeep Holla 
204e7818584SSudeep Holla static inline
ffa_device_is_valid(struct ffa_device * ffa_dev)205e7818584SSudeep Holla bool ffa_device_is_valid(struct ffa_device *ffa_dev) { return false; }
206e7818584SSudeep Holla 
207e7818584SSudeep Holla #endif /* CONFIG_ARM_FFA_TRANSPORT */
208e7818584SSudeep Holla 
209e7818584SSudeep Holla #define ffa_register(driver) \
210e7818584SSudeep Holla 	ffa_driver_register(driver, THIS_MODULE, KBUILD_MODNAME)
211e7818584SSudeep Holla #define ffa_unregister(driver) \
212e7818584SSudeep Holla 	ffa_driver_unregister(driver)
213e7818584SSudeep Holla 
214e7818584SSudeep Holla /**
215e7818584SSudeep Holla  * module_ffa_driver() - Helper macro for registering a psa_ffa driver
216e7818584SSudeep Holla  * @__ffa_driver: ffa_driver structure
217e7818584SSudeep Holla  *
218e7818584SSudeep Holla  * Helper macro for psa_ffa drivers to set up proper module init / exit
219e7818584SSudeep Holla  * functions.  Replaces module_init() and module_exit() and keeps people from
220e7818584SSudeep Holla  * printing pointless things to the kernel log when their driver is loaded.
221e7818584SSudeep Holla  */
222e7818584SSudeep Holla #define module_ffa_driver(__ffa_driver)	\
223e7818584SSudeep Holla 	module_driver(__ffa_driver, ffa_register, ffa_unregister)
224e7818584SSudeep Holla 
225989e8661SRicardo B. Marliere extern const struct bus_type ffa_bus_type;
2263fad96e9SSudeep Holla 
22789437638SSebastian Ene /* The FF-A 1.0 partition structure lacks the uuid[4] */
22889437638SSebastian Ene #define FFA_1_0_PARTITON_INFO_SZ	(8)
22989437638SSebastian Ene 
230d0c0bce8SSudeep Holla /* FFA transport related */
231d0c0bce8SSudeep Holla struct ffa_partition_info {
232d0c0bce8SSudeep Holla 	u16 id;
233d0c0bce8SSudeep Holla 	u16 exec_ctxt;
234d0c0bce8SSudeep Holla /* partition supports receipt of direct requests */
235d0c0bce8SSudeep Holla #define FFA_PARTITION_DIRECT_RECV	BIT(0)
236d0c0bce8SSudeep Holla /* partition can send direct requests. */
237d0c0bce8SSudeep Holla #define FFA_PARTITION_DIRECT_SEND	BIT(1)
238d0c0bce8SSudeep Holla /* partition can send and receive indirect messages. */
239d0c0bce8SSudeep Holla #define FFA_PARTITION_INDIRECT_MSG	BIT(2)
2403c258bf6SSudeep Holla /* partition can receive notifications */
2413c258bf6SSudeep Holla #define FFA_PARTITION_NOTIFICATION_RECV	BIT(3)
242106b11b1SSudeep Holla /* partition runs in the AArch64 execution state. */
243106b11b1SSudeep Holla #define FFA_PARTITION_AARCH64_EXEC	BIT(8)
24484968e32SSudeep Holla /* partition supports receipt of direct request2 */
24584968e32SSudeep Holla #define FFA_PARTITION_DIRECT_REQ2_RECV	BIT(9)
24684968e32SSudeep Holla /* partition can send direct request2. */
24784968e32SSudeep Holla #define FFA_PARTITION_DIRECT_REQ2_SEND	BIT(10)
248d0c0bce8SSudeep Holla 	u32 properties;
249b7c9f326SSudeep Holla 	uuid_t uuid;
250d0c0bce8SSudeep Holla };
251d0c0bce8SSudeep Holla 
2523c258bf6SSudeep Holla static inline
ffa_partition_check_property(struct ffa_device * dev,u32 property)2533c258bf6SSudeep Holla bool ffa_partition_check_property(struct ffa_device *dev, u32 property)
2543c258bf6SSudeep Holla {
2553c258bf6SSudeep Holla 	return dev->properties & property;
2563c258bf6SSudeep Holla }
2573c258bf6SSudeep Holla 
2583c258bf6SSudeep Holla #define ffa_partition_supports_notify_recv(dev)	\
2593c258bf6SSudeep Holla 	ffa_partition_check_property(dev, FFA_PARTITION_NOTIFICATION_RECV)
2603c258bf6SSudeep Holla 
2613c258bf6SSudeep Holla #define ffa_partition_supports_indirect_msg(dev)	\
2623c258bf6SSudeep Holla 	ffa_partition_check_property(dev, FFA_PARTITION_INDIRECT_MSG)
2633c258bf6SSudeep Holla 
2643c258bf6SSudeep Holla #define ffa_partition_supports_direct_recv(dev)	\
2653c258bf6SSudeep Holla 	ffa_partition_check_property(dev, FFA_PARTITION_DIRECT_RECV)
2663c258bf6SSudeep Holla 
26784968e32SSudeep Holla #define ffa_partition_supports_direct_req2_recv(dev)	\
26884968e32SSudeep Holla 	(ffa_partition_check_property(dev, FFA_PARTITION_DIRECT_REQ2_RECV) && \
26984968e32SSudeep Holla 	 !dev->mode_32bit)
27084968e32SSudeep Holla 
271d0c0bce8SSudeep Holla /* For use with FFA_MSG_SEND_DIRECT_{REQ,RESP} which pass data via registers */
272d0c0bce8SSudeep Holla struct ffa_send_direct_data {
273d0c0bce8SSudeep Holla 	unsigned long data0; /* w3/x3 */
274d0c0bce8SSudeep Holla 	unsigned long data1; /* w4/x4 */
275d0c0bce8SSudeep Holla 	unsigned long data2; /* w5/x5 */
276d0c0bce8SSudeep Holla 	unsigned long data3; /* w6/x6 */
277d0c0bce8SSudeep Holla 	unsigned long data4; /* w7/x7 */
278d0c0bce8SSudeep Holla };
279d0c0bce8SSudeep Holla 
28002c19d84SSudeep Holla struct ffa_indirect_msg_hdr {
28102c19d84SSudeep Holla 	u32 flags;
28202c19d84SSudeep Holla 	u32 res0;
28302c19d84SSudeep Holla 	u32 offset;
28402c19d84SSudeep Holla 	u32 send_recv_id;
28502c19d84SSudeep Holla 	u32 size;
286910cc1acSSudeep Holla 	uuid_t uuid;
28702c19d84SSudeep Holla };
28802c19d84SSudeep Holla 
289aaef3bc9SSudeep Holla /* For use with FFA_MSG_SEND_DIRECT_{REQ,RESP}2 which pass data via registers */
290aaef3bc9SSudeep Holla struct ffa_send_direct_data2 {
291aaef3bc9SSudeep Holla 	unsigned long data[14]; /* x4-x17 */
292aaef3bc9SSudeep Holla };
293aaef3bc9SSudeep Holla 
294cc2195feSSudeep Holla struct ffa_mem_region_addr_range {
295cc2195feSSudeep Holla 	/* The base IPA of the constituent memory region, aligned to 4 kiB */
296cc2195feSSudeep Holla 	u64 address;
297cc2195feSSudeep Holla 	/* The number of 4 kiB pages in the constituent memory region. */
298cc2195feSSudeep Holla 	u32 pg_cnt;
299cc2195feSSudeep Holla 	u32 reserved;
300cc2195feSSudeep Holla };
301cc2195feSSudeep Holla 
302cc2195feSSudeep Holla struct ffa_composite_mem_region {
303cc2195feSSudeep Holla 	/*
304cc2195feSSudeep Holla 	 * The total number of 4 kiB pages included in this memory region. This
305cc2195feSSudeep Holla 	 * must be equal to the sum of page counts specified in each
306cc2195feSSudeep Holla 	 * `struct ffa_mem_region_addr_range`.
307cc2195feSSudeep Holla 	 */
308cc2195feSSudeep Holla 	u32 total_pg_cnt;
309cc2195feSSudeep Holla 	/* The number of constituents included in this memory region range */
310cc2195feSSudeep Holla 	u32 addr_range_cnt;
311cc2195feSSudeep Holla 	u64 reserved;
312cc2195feSSudeep Holla 	/** An array of `addr_range_cnt` memory region constituents. */
313cc2195feSSudeep Holla 	struct ffa_mem_region_addr_range constituents[];
314cc2195feSSudeep Holla };
315cc2195feSSudeep Holla 
316cc2195feSSudeep Holla struct ffa_mem_region_attributes {
317cc2195feSSudeep Holla 	/* The ID of the VM to which the memory is being given or shared. */
318cc2195feSSudeep Holla 	u16 receiver;
319cc2195feSSudeep Holla 	/*
320cc2195feSSudeep Holla 	 * The permissions with which the memory region should be mapped in the
321cc2195feSSudeep Holla 	 * receiver's page table.
322cc2195feSSudeep Holla 	 */
323cc2195feSSudeep Holla #define FFA_MEM_EXEC		BIT(3)
324cc2195feSSudeep Holla #define FFA_MEM_NO_EXEC		BIT(2)
325cc2195feSSudeep Holla #define FFA_MEM_RW		BIT(1)
326cc2195feSSudeep Holla #define FFA_MEM_RO		BIT(0)
327cc2195feSSudeep Holla 	u8 attrs;
328cc2195feSSudeep Holla 	/*
329cc2195feSSudeep Holla 	 * Flags used during FFA_MEM_RETRIEVE_REQ and FFA_MEM_RETRIEVE_RESP
330cc2195feSSudeep Holla 	 * for memory regions with multiple borrowers.
331cc2195feSSudeep Holla 	 */
332cc2195feSSudeep Holla #define FFA_MEM_RETRIEVE_SELF_BORROWER	BIT(0)
333cc2195feSSudeep Holla 	u8 flag;
334cc2195feSSudeep Holla 	/*
335cc2195feSSudeep Holla 	 * Offset in bytes from the start of the outer `ffa_memory_region` to
336cc2195feSSudeep Holla 	 * an `struct ffa_mem_region_addr_range`.
337cc2195feSSudeep Holla 	 */
338c8e320b0SWill Deacon 	u32 composite_off;
339cc2195feSSudeep Holla 	u64 reserved;
340cc2195feSSudeep Holla };
341cc2195feSSudeep Holla 
342cc2195feSSudeep Holla struct ffa_mem_region {
343cc2195feSSudeep Holla 	/* The ID of the VM/owner which originally sent the memory region */
344cc2195feSSudeep Holla 	u16 sender_id;
345cc2195feSSudeep Holla #define FFA_MEM_NORMAL		BIT(5)
346cc2195feSSudeep Holla #define FFA_MEM_DEVICE		BIT(4)
347cc2195feSSudeep Holla 
348cc2195feSSudeep Holla #define FFA_MEM_WRITE_BACK	(3 << 2)
349cc2195feSSudeep Holla #define FFA_MEM_NON_CACHEABLE	(1 << 2)
350cc2195feSSudeep Holla 
351cc2195feSSudeep Holla #define FFA_DEV_nGnRnE		(0 << 2)
352cc2195feSSudeep Holla #define FFA_DEV_nGnRE		(1 << 2)
353cc2195feSSudeep Holla #define FFA_DEV_nGRE		(2 << 2)
354cc2195feSSudeep Holla #define FFA_DEV_GRE		(3 << 2)
355cc2195feSSudeep Holla 
356cc2195feSSudeep Holla #define FFA_MEM_NON_SHAREABLE	(0)
357cc2195feSSudeep Holla #define FFA_MEM_OUTER_SHAREABLE	(2)
358cc2195feSSudeep Holla #define FFA_MEM_INNER_SHAREABLE	(3)
35911358053SSudeep Holla 	/* Memory region attributes, upper byte MBZ pre v1.1 */
36011358053SSudeep Holla 	u16 attributes;
361cc2195feSSudeep Holla /*
362cc2195feSSudeep Holla  * Clear memory region contents after unmapping it from the sender and
363cc2195feSSudeep Holla  * before mapping it for any receiver.
364cc2195feSSudeep Holla  */
365cc2195feSSudeep Holla #define FFA_MEM_CLEAR			BIT(0)
366cc2195feSSudeep Holla /*
367cc2195feSSudeep Holla  * Whether the hypervisor may time slice the memory sharing or retrieval
368cc2195feSSudeep Holla  * operation.
369cc2195feSSudeep Holla  */
370cc2195feSSudeep Holla #define FFA_TIME_SLICE_ENABLE		BIT(1)
371cc2195feSSudeep Holla 
372cc2195feSSudeep Holla #define FFA_MEM_RETRIEVE_TYPE_IN_RESP	(0 << 3)
373cc2195feSSudeep Holla #define FFA_MEM_RETRIEVE_TYPE_SHARE	(1 << 3)
374cc2195feSSudeep Holla #define FFA_MEM_RETRIEVE_TYPE_LEND	(2 << 3)
375cc2195feSSudeep Holla #define FFA_MEM_RETRIEVE_TYPE_DONATE	(3 << 3)
376cc2195feSSudeep Holla 
377cc2195feSSudeep Holla #define FFA_MEM_RETRIEVE_ADDR_ALIGN_HINT	BIT(9)
378cc2195feSSudeep Holla #define FFA_MEM_RETRIEVE_ADDR_ALIGN(x)		((x) << 5)
379cc2195feSSudeep Holla 	/* Flags to control behaviour of the transaction. */
380cc2195feSSudeep Holla 	u32 flags;
381cc2195feSSudeep Holla #define HANDLE_LOW_MASK		GENMASK_ULL(31, 0)
382cc2195feSSudeep Holla #define HANDLE_HIGH_MASK	GENMASK_ULL(63, 32)
383cc2195feSSudeep Holla #define HANDLE_LOW(x)		((u32)(FIELD_GET(HANDLE_LOW_MASK, (x))))
384cc2195feSSudeep Holla #define	HANDLE_HIGH(x)		((u32)(FIELD_GET(HANDLE_HIGH_MASK, (x))))
385cc2195feSSudeep Holla 
386cc2195feSSudeep Holla #define PACK_HANDLE(l, h)		\
387cc2195feSSudeep Holla 	(FIELD_PREP(HANDLE_LOW_MASK, (l)) | FIELD_PREP(HANDLE_HIGH_MASK, (h)))
388cc2195feSSudeep Holla 	/*
389cc2195feSSudeep Holla 	 * A globally-unique ID assigned by the hypervisor for a region
390cc2195feSSudeep Holla 	 * of memory being sent between VMs.
391cc2195feSSudeep Holla 	 */
392cc2195feSSudeep Holla 	u64 handle;
393cc2195feSSudeep Holla 	/*
394cc2195feSSudeep Holla 	 * An implementation defined value associated with the receiver and the
395cc2195feSSudeep Holla 	 * memory region.
396cc2195feSSudeep Holla 	 */
397cc2195feSSudeep Holla 	u64 tag;
39811358053SSudeep Holla 	/* Size of each endpoint memory access descriptor, MBZ pre v1.1 */
39911358053SSudeep Holla 	u32 ep_mem_size;
400cc2195feSSudeep Holla 	/*
401cc2195feSSudeep Holla 	 * The number of `ffa_mem_region_attributes` entries included in this
402cc2195feSSudeep Holla 	 * transaction.
403cc2195feSSudeep Holla 	 */
404cc2195feSSudeep Holla 	u32 ep_count;
405cc2195feSSudeep Holla 	/*
40611358053SSudeep Holla 	 * 16-byte aligned offset from the base address of this descriptor
40711358053SSudeep Holla 	 * to the first element of the endpoint memory access descriptor array
40811358053SSudeep Holla 	 * Valid only from v1.1
409cc2195feSSudeep Holla 	 */
41011358053SSudeep Holla 	u32 ep_mem_offset;
41111358053SSudeep Holla 	/* MBZ, valid only from v1.1 */
41211358053SSudeep Holla 	u32 reserved[3];
413cc2195feSSudeep Holla };
414cc2195feSSudeep Holla 
415cc2195feSSudeep Holla #define CONSTITUENTS_OFFSET(x)	\
416cc2195feSSudeep Holla 	(offsetof(struct ffa_composite_mem_region, constituents[x]))
417cc2195feSSudeep Holla 
41876cf932cSSudeep Holla static inline u32
ffa_mem_desc_offset(struct ffa_mem_region * buf,int count,u32 ffa_version)41976cf932cSSudeep Holla ffa_mem_desc_offset(struct ffa_mem_region *buf, int count, u32 ffa_version)
42076cf932cSSudeep Holla {
42111358053SSudeep Holla 	u32 offset = count * sizeof(struct ffa_mem_region_attributes);
42211358053SSudeep Holla 	/*
42311358053SSudeep Holla 	 * Earlier to v1.1, the endpoint memory descriptor array started at
42411358053SSudeep Holla 	 * offset 32(i.e. offset of ep_mem_offset in the current structure)
42511358053SSudeep Holla 	 */
42611358053SSudeep Holla 	if (ffa_version <= FFA_VERSION_1_0)
42711358053SSudeep Holla 		offset += offsetof(struct ffa_mem_region, ep_mem_offset);
42811358053SSudeep Holla 	else
42911358053SSudeep Holla 		offset += sizeof(struct ffa_mem_region);
43011358053SSudeep Holla 
43111358053SSudeep Holla 	return offset;
43276cf932cSSudeep Holla }
43376cf932cSSudeep Holla 
434cc2195feSSudeep Holla struct ffa_mem_ops_args {
435cc2195feSSudeep Holla 	bool use_txbuf;
436cc2195feSSudeep Holla 	u32 nattrs;
437cc2195feSSudeep Holla 	u32 flags;
438cc2195feSSudeep Holla 	u64 tag;
439cc2195feSSudeep Holla 	u64 g_handle;
440cc2195feSSudeep Holla 	struct scatterlist *sg;
441cc2195feSSudeep Holla 	struct ffa_mem_region_attributes *attrs;
442cc2195feSSudeep Holla };
443cc2195feSSudeep Holla 
4445b0c6328SSudeep Holla struct ffa_info_ops {
445d0c0bce8SSudeep Holla 	u32 (*api_version_get)(void);
446d0c0bce8SSudeep Holla 	int (*partition_info_get)(const char *uuid_str,
447d0c0bce8SSudeep Holla 				  struct ffa_partition_info *buffer);
4485b0c6328SSudeep Holla };
4495b0c6328SSudeep Holla 
4505b0c6328SSudeep Holla struct ffa_msg_ops {
451d0c0bce8SSudeep Holla 	void (*mode_32bit_set)(struct ffa_device *dev);
452d0c0bce8SSudeep Holla 	int (*sync_send_receive)(struct ffa_device *dev,
453d0c0bce8SSudeep Holla 				 struct ffa_send_direct_data *data);
45402c19d84SSudeep Holla 	int (*indirect_send)(struct ffa_device *dev, void *buf, size_t sz);
4558768972cSSudeep Holla 	int (*sync_send_receive2)(struct ffa_device *dev,
456aaef3bc9SSudeep Holla 				  struct ffa_send_direct_data2 *data);
4575b0c6328SSudeep Holla };
4585b0c6328SSudeep Holla 
4595b0c6328SSudeep Holla struct ffa_mem_ops {
460cc2195feSSudeep Holla 	int (*memory_reclaim)(u64 g_handle, u32 flags);
4618c3812c8SSudeep Holla 	int (*memory_share)(struct ffa_mem_ops_args *args);
4628c3812c8SSudeep Holla 	int (*memory_lend)(struct ffa_mem_ops_args *args);
463d0c0bce8SSudeep Holla };
464d0c0bce8SSudeep Holla 
465fe2ddb6bSSudeep Holla struct ffa_cpu_ops {
466fe2ddb6bSSudeep Holla 	int (*run)(struct ffa_device *dev, u16 vcpu);
467fe2ddb6bSSudeep Holla };
468fe2ddb6bSSudeep Holla 
4690184450bSSudeep Holla typedef void (*ffa_sched_recv_cb)(u16 vcpu, bool is_per_vcpu, void *cb_data);
470e0573444SSudeep Holla typedef void (*ffa_notifier_cb)(int notify_id, void *cb_data);
471*c10debfeSSudeep Holla typedef void (*ffa_fwk_notifier_cb)(int notify_id, void *cb_data, void *buf);
472e0573444SSudeep Holla 
4730184450bSSudeep Holla struct ffa_notifier_ops {
4740184450bSSudeep Holla 	int (*sched_recv_cb_register)(struct ffa_device *dev,
4750184450bSSudeep Holla 				      ffa_sched_recv_cb cb, void *cb_data);
4760184450bSSudeep Holla 	int (*sched_recv_cb_unregister)(struct ffa_device *dev);
477e0573444SSudeep Holla 	int (*notify_request)(struct ffa_device *dev, bool per_vcpu,
478e0573444SSudeep Holla 			      ffa_notifier_cb cb, void *cb_data, int notify_id);
479e0573444SSudeep Holla 	int (*notify_relinquish)(struct ffa_device *dev, int notify_id);
480*c10debfeSSudeep Holla 	int (*fwk_notify_request)(struct ffa_device *dev,
481*c10debfeSSudeep Holla 				  ffa_fwk_notifier_cb cb, void *cb_data,
482*c10debfeSSudeep Holla 				  int notify_id);
483*c10debfeSSudeep Holla 	int (*fwk_notify_relinquish)(struct ffa_device *dev, int notify_id);
484e5adb3b2SSudeep Holla 	int (*notify_send)(struct ffa_device *dev, int notify_id, bool per_vcpu,
485e5adb3b2SSudeep Holla 			   u16 vcpu);
4860184450bSSudeep Holla };
4870184450bSSudeep Holla 
4885b0c6328SSudeep Holla struct ffa_ops {
4895b0c6328SSudeep Holla 	const struct ffa_info_ops *info_ops;
4905b0c6328SSudeep Holla 	const struct ffa_msg_ops *msg_ops;
4915b0c6328SSudeep Holla 	const struct ffa_mem_ops *mem_ops;
492fe2ddb6bSSudeep Holla 	const struct ffa_cpu_ops *cpu_ops;
4930184450bSSudeep Holla 	const struct ffa_notifier_ops *notifier_ops;
4945b0c6328SSudeep Holla };
4955b0c6328SSudeep Holla 
496e7818584SSudeep Holla #endif /* _LINUX_ARM_FFA_H */
497