xref: /OK3568_Linux_fs/kernel/arch/powerpc/platforms/powernv/vas.h (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /* SPDX-License-Identifier: GPL-2.0-or-later */
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun  * Copyright 2016-17 IBM Corp.
4*4882a593Smuzhiyun  */
5*4882a593Smuzhiyun 
6*4882a593Smuzhiyun #ifndef _VAS_H
7*4882a593Smuzhiyun #define _VAS_H
8*4882a593Smuzhiyun #include <linux/atomic.h>
9*4882a593Smuzhiyun #include <linux/idr.h>
10*4882a593Smuzhiyun #include <asm/vas.h>
11*4882a593Smuzhiyun #include <linux/io.h>
12*4882a593Smuzhiyun #include <linux/dcache.h>
13*4882a593Smuzhiyun #include <linux/mutex.h>
14*4882a593Smuzhiyun #include <linux/stringify.h>
15*4882a593Smuzhiyun 
16*4882a593Smuzhiyun /*
17*4882a593Smuzhiyun  * Overview of Virtual Accelerator Switchboard (VAS).
18*4882a593Smuzhiyun  *
19*4882a593Smuzhiyun  * VAS is a hardware "switchboard" that allows senders and receivers to
20*4882a593Smuzhiyun  * exchange messages with _minimal_ kernel involvment. The receivers are
21*4882a593Smuzhiyun  * typically NX coprocessor engines that perform compression or encryption
22*4882a593Smuzhiyun  * in hardware, but receivers can also be other software threads.
23*4882a593Smuzhiyun  *
24*4882a593Smuzhiyun  * Senders are user/kernel threads that submit compression/encryption or
25*4882a593Smuzhiyun  * other requests to the receivers. Senders must format their messages as
26*4882a593Smuzhiyun  * Coprocessor Request Blocks (CRB)s and submit them using the "copy" and
27*4882a593Smuzhiyun  * "paste" instructions which were introduced in Power9.
28*4882a593Smuzhiyun  *
29*4882a593Smuzhiyun  * A Power node can have (upto?) 8 Power chips. There is one instance of
30*4882a593Smuzhiyun  * VAS in each Power9 chip. Each instance of VAS has 64K windows or ports,
31*4882a593Smuzhiyun  * Senders and receivers must each connect to a separate window before they
32*4882a593Smuzhiyun  * can exchange messages through the switchboard.
33*4882a593Smuzhiyun  *
34*4882a593Smuzhiyun  * Each window is described by two types of window contexts:
35*4882a593Smuzhiyun  *
36*4882a593Smuzhiyun  *	Hypervisor Window Context (HVWC) of size VAS_HVWC_SIZE bytes
37*4882a593Smuzhiyun  *
38*4882a593Smuzhiyun  *	OS/User Window Context (UWC) of size VAS_UWC_SIZE bytes.
39*4882a593Smuzhiyun  *
40*4882a593Smuzhiyun  * A window context can be viewed as a set of 64-bit registers. The settings
41*4882a593Smuzhiyun  * in these registers configure/control/determine the behavior of the VAS
42*4882a593Smuzhiyun  * hardware when messages are sent/received through the window. The registers
43*4882a593Smuzhiyun  * in the HVWC are configured by the kernel while the registers in the UWC can
44*4882a593Smuzhiyun  * be configured by the kernel or by the user space application that is using
45*4882a593Smuzhiyun  * the window.
46*4882a593Smuzhiyun  *
47*4882a593Smuzhiyun  * The HVWCs for all windows on a specific instance of VAS are in a contiguous
48*4882a593Smuzhiyun  * range of hardware addresses or Base address region (BAR) referred to as the
49*4882a593Smuzhiyun  * HVWC BAR for the instance. Similarly the UWCs for all windows on an instance
50*4882a593Smuzhiyun  * are referred to as the UWC BAR for the instance.
51*4882a593Smuzhiyun  *
52*4882a593Smuzhiyun  * The two BARs for each instance are defined Power9 MMIO Ranges spreadsheet
53*4882a593Smuzhiyun  * and available to the kernel in the VAS node's "reg" property in the device
54*4882a593Smuzhiyun  * tree:
55*4882a593Smuzhiyun  *
56*4882a593Smuzhiyun  *	/proc/device-tree/vasm@.../reg
57*4882a593Smuzhiyun  *
58*4882a593Smuzhiyun  * (see vas_probe() for details on the reg property).
59*4882a593Smuzhiyun  *
60*4882a593Smuzhiyun  * The kernel maps the HVWC and UWC BAR regions into the kernel address
61*4882a593Smuzhiyun  * space (hvwc_map and uwc_map). The kernel can then access the window
62*4882a593Smuzhiyun  * contexts of a specific window using:
63*4882a593Smuzhiyun  *
64*4882a593Smuzhiyun  *	 hvwc = hvwc_map + winid * VAS_HVWC_SIZE.
65*4882a593Smuzhiyun  *	 uwc = uwc_map + winid * VAS_UWC_SIZE.
66*4882a593Smuzhiyun  *
67*4882a593Smuzhiyun  * where winid is the window index (0..64K).
68*4882a593Smuzhiyun  *
69*4882a593Smuzhiyun  * As mentioned, a window context is used to "configure" a window. Besides
70*4882a593Smuzhiyun  * this configuration address, each _send_ window also has a unique hardware
71*4882a593Smuzhiyun  * "paste" address that is used to submit requests/CRBs (see vas_paste_crb()).
72*4882a593Smuzhiyun  *
73*4882a593Smuzhiyun  * The hardware paste address for a window is computed using the "paste
74*4882a593Smuzhiyun  * base address" and "paste win id shift" reg properties in the VAS device
75*4882a593Smuzhiyun  * tree node using:
76*4882a593Smuzhiyun  *
77*4882a593Smuzhiyun  *	paste_addr = paste_base + ((winid << paste_win_id_shift))
78*4882a593Smuzhiyun  *
79*4882a593Smuzhiyun  * (again, see vas_probe() for ->paste_base_addr and ->paste_win_id_shift).
80*4882a593Smuzhiyun  *
81*4882a593Smuzhiyun  * The kernel maps this hardware address into the sender's address space
82*4882a593Smuzhiyun  * after which they can use the 'paste' instruction (new in Power9) to
83*4882a593Smuzhiyun  * send a message (submit a request aka CRB) to the coprocessor.
84*4882a593Smuzhiyun  *
85*4882a593Smuzhiyun  * NOTE: In the initial version, senders can only in-kernel drivers/threads.
86*4882a593Smuzhiyun  *	 Support for user space threads will be added in follow-on patches.
87*4882a593Smuzhiyun  *
88*4882a593Smuzhiyun  * TODO: Do we need to map the UWC into user address space so they can return
89*4882a593Smuzhiyun  *	 credits? Its NA for NX but may be needed for other receive windows.
90*4882a593Smuzhiyun  *
91*4882a593Smuzhiyun  */
92*4882a593Smuzhiyun 
93*4882a593Smuzhiyun #define VAS_WINDOWS_PER_CHIP		(64 << 10)
94*4882a593Smuzhiyun 
95*4882a593Smuzhiyun /*
96*4882a593Smuzhiyun  * Hypervisor and OS/USer Window Context sizes
97*4882a593Smuzhiyun  */
98*4882a593Smuzhiyun #define VAS_HVWC_SIZE			512
99*4882a593Smuzhiyun #define VAS_UWC_SIZE			PAGE_SIZE
100*4882a593Smuzhiyun 
101*4882a593Smuzhiyun /*
102*4882a593Smuzhiyun  * Initial per-process credits.
103*4882a593Smuzhiyun  * Max send window credits:    4K-1 (12-bits in VAS_TX_WCRED)
104*4882a593Smuzhiyun  *
105*4882a593Smuzhiyun  * TODO: Needs tuning for per-process credits
106*4882a593Smuzhiyun  */
107*4882a593Smuzhiyun #define VAS_TX_WCREDS_MAX		((4 << 10) - 1)
108*4882a593Smuzhiyun #define VAS_WCREDS_DEFAULT		(1 << 10)
109*4882a593Smuzhiyun 
110*4882a593Smuzhiyun /*
111*4882a593Smuzhiyun  * VAS Window Context Register Offsets and bitmasks.
112*4882a593Smuzhiyun  * See Section 3.1.4 of VAS Work book
113*4882a593Smuzhiyun  */
114*4882a593Smuzhiyun #define VAS_LPID_OFFSET			0x010
115*4882a593Smuzhiyun #define VAS_LPID			PPC_BITMASK(0, 11)
116*4882a593Smuzhiyun 
117*4882a593Smuzhiyun #define VAS_PID_OFFSET			0x018
118*4882a593Smuzhiyun #define VAS_PID_ID			PPC_BITMASK(0, 19)
119*4882a593Smuzhiyun 
120*4882a593Smuzhiyun #define VAS_XLATE_MSR_OFFSET		0x020
121*4882a593Smuzhiyun #define VAS_XLATE_MSR_DR		PPC_BIT(0)
122*4882a593Smuzhiyun #define VAS_XLATE_MSR_TA		PPC_BIT(1)
123*4882a593Smuzhiyun #define VAS_XLATE_MSR_PR		PPC_BIT(2)
124*4882a593Smuzhiyun #define VAS_XLATE_MSR_US		PPC_BIT(3)
125*4882a593Smuzhiyun #define VAS_XLATE_MSR_HV		PPC_BIT(4)
126*4882a593Smuzhiyun #define VAS_XLATE_MSR_SF		PPC_BIT(5)
127*4882a593Smuzhiyun 
128*4882a593Smuzhiyun #define VAS_XLATE_LPCR_OFFSET		0x028
129*4882a593Smuzhiyun #define VAS_XLATE_LPCR_PAGE_SIZE	PPC_BITMASK(0, 2)
130*4882a593Smuzhiyun #define VAS_XLATE_LPCR_ISL		PPC_BIT(3)
131*4882a593Smuzhiyun #define VAS_XLATE_LPCR_TC		PPC_BIT(4)
132*4882a593Smuzhiyun #define VAS_XLATE_LPCR_SC		PPC_BIT(5)
133*4882a593Smuzhiyun 
134*4882a593Smuzhiyun #define VAS_XLATE_CTL_OFFSET		0x030
135*4882a593Smuzhiyun #define VAS_XLATE_MODE			PPC_BITMASK(0, 1)
136*4882a593Smuzhiyun 
137*4882a593Smuzhiyun #define VAS_AMR_OFFSET			0x040
138*4882a593Smuzhiyun #define VAS_AMR				PPC_BITMASK(0, 63)
139*4882a593Smuzhiyun 
140*4882a593Smuzhiyun #define VAS_SEIDR_OFFSET		0x048
141*4882a593Smuzhiyun #define VAS_SEIDR			PPC_BITMASK(0, 63)
142*4882a593Smuzhiyun 
143*4882a593Smuzhiyun #define VAS_FAULT_TX_WIN_OFFSET		0x050
144*4882a593Smuzhiyun #define VAS_FAULT_TX_WIN		PPC_BITMASK(48, 63)
145*4882a593Smuzhiyun 
146*4882a593Smuzhiyun #define VAS_OSU_INTR_SRC_RA_OFFSET	0x060
147*4882a593Smuzhiyun #define VAS_OSU_INTR_SRC_RA		PPC_BITMASK(8, 63)
148*4882a593Smuzhiyun 
149*4882a593Smuzhiyun #define VAS_HV_INTR_SRC_RA_OFFSET	0x070
150*4882a593Smuzhiyun #define VAS_HV_INTR_SRC_RA		PPC_BITMASK(8, 63)
151*4882a593Smuzhiyun 
152*4882a593Smuzhiyun #define VAS_PSWID_OFFSET		0x078
153*4882a593Smuzhiyun #define VAS_PSWID_EA_HANDLE		PPC_BITMASK(0, 31)
154*4882a593Smuzhiyun 
155*4882a593Smuzhiyun #define VAS_SPARE1_OFFSET		0x080
156*4882a593Smuzhiyun #define VAS_SPARE2_OFFSET		0x088
157*4882a593Smuzhiyun #define VAS_SPARE3_OFFSET		0x090
158*4882a593Smuzhiyun #define VAS_SPARE4_OFFSET		0x130
159*4882a593Smuzhiyun #define VAS_SPARE5_OFFSET		0x160
160*4882a593Smuzhiyun #define VAS_SPARE6_OFFSET		0x188
161*4882a593Smuzhiyun 
162*4882a593Smuzhiyun #define VAS_LFIFO_BAR_OFFSET		0x0A0
163*4882a593Smuzhiyun #define VAS_LFIFO_BAR			PPC_BITMASK(8, 53)
164*4882a593Smuzhiyun #define VAS_PAGE_MIGRATION_SELECT	PPC_BITMASK(54, 56)
165*4882a593Smuzhiyun 
166*4882a593Smuzhiyun #define VAS_LDATA_STAMP_CTL_OFFSET	0x0A8
167*4882a593Smuzhiyun #define VAS_LDATA_STAMP			PPC_BITMASK(0, 1)
168*4882a593Smuzhiyun #define VAS_XTRA_WRITE			PPC_BIT(2)
169*4882a593Smuzhiyun 
170*4882a593Smuzhiyun #define VAS_LDMA_CACHE_CTL_OFFSET	0x0B0
171*4882a593Smuzhiyun #define VAS_LDMA_TYPE			PPC_BITMASK(0, 1)
172*4882a593Smuzhiyun #define VAS_LDMA_FIFO_DISABLE		PPC_BIT(2)
173*4882a593Smuzhiyun 
174*4882a593Smuzhiyun #define VAS_LRFIFO_PUSH_OFFSET		0x0B8
175*4882a593Smuzhiyun #define VAS_LRFIFO_PUSH			PPC_BITMASK(0, 15)
176*4882a593Smuzhiyun 
177*4882a593Smuzhiyun #define VAS_CURR_MSG_COUNT_OFFSET	0x0C0
178*4882a593Smuzhiyun #define VAS_CURR_MSG_COUNT		PPC_BITMASK(0, 7)
179*4882a593Smuzhiyun 
180*4882a593Smuzhiyun #define VAS_LNOTIFY_AFTER_COUNT_OFFSET	0x0C8
181*4882a593Smuzhiyun #define VAS_LNOTIFY_AFTER_COUNT		PPC_BITMASK(0, 7)
182*4882a593Smuzhiyun 
183*4882a593Smuzhiyun #define VAS_LRX_WCRED_OFFSET		0x0E0
184*4882a593Smuzhiyun #define VAS_LRX_WCRED			PPC_BITMASK(0, 15)
185*4882a593Smuzhiyun 
186*4882a593Smuzhiyun #define VAS_LRX_WCRED_ADDER_OFFSET	0x190
187*4882a593Smuzhiyun #define VAS_LRX_WCRED_ADDER		PPC_BITMASK(0, 15)
188*4882a593Smuzhiyun 
189*4882a593Smuzhiyun #define VAS_TX_WCRED_OFFSET		0x0F0
190*4882a593Smuzhiyun #define VAS_TX_WCRED			PPC_BITMASK(4, 15)
191*4882a593Smuzhiyun 
192*4882a593Smuzhiyun #define VAS_TX_WCRED_ADDER_OFFSET	0x1A0
193*4882a593Smuzhiyun #define VAS_TX_WCRED_ADDER		PPC_BITMASK(4, 15)
194*4882a593Smuzhiyun 
195*4882a593Smuzhiyun #define VAS_LFIFO_SIZE_OFFSET		0x100
196*4882a593Smuzhiyun #define VAS_LFIFO_SIZE			PPC_BITMASK(0, 3)
197*4882a593Smuzhiyun 
198*4882a593Smuzhiyun #define VAS_WINCTL_OFFSET		0x108
199*4882a593Smuzhiyun #define VAS_WINCTL_OPEN			PPC_BIT(0)
200*4882a593Smuzhiyun #define VAS_WINCTL_REJ_NO_CREDIT	PPC_BIT(1)
201*4882a593Smuzhiyun #define VAS_WINCTL_PIN			PPC_BIT(2)
202*4882a593Smuzhiyun #define VAS_WINCTL_TX_WCRED_MODE	PPC_BIT(3)
203*4882a593Smuzhiyun #define VAS_WINCTL_RX_WCRED_MODE	PPC_BIT(4)
204*4882a593Smuzhiyun #define VAS_WINCTL_TX_WORD_MODE		PPC_BIT(5)
205*4882a593Smuzhiyun #define VAS_WINCTL_RX_WORD_MODE		PPC_BIT(6)
206*4882a593Smuzhiyun #define VAS_WINCTL_RSVD_TXBUF		PPC_BIT(7)
207*4882a593Smuzhiyun #define VAS_WINCTL_THRESH_CTL		PPC_BITMASK(8, 9)
208*4882a593Smuzhiyun #define VAS_WINCTL_FAULT_WIN		PPC_BIT(10)
209*4882a593Smuzhiyun #define VAS_WINCTL_NX_WIN		PPC_BIT(11)
210*4882a593Smuzhiyun 
211*4882a593Smuzhiyun #define VAS_WIN_STATUS_OFFSET		0x110
212*4882a593Smuzhiyun #define VAS_WIN_BUSY			PPC_BIT(1)
213*4882a593Smuzhiyun 
214*4882a593Smuzhiyun #define VAS_WIN_CTX_CACHING_CTL_OFFSET	0x118
215*4882a593Smuzhiyun #define VAS_CASTOUT_REQ			PPC_BIT(0)
216*4882a593Smuzhiyun #define VAS_PUSH_TO_MEM			PPC_BIT(1)
217*4882a593Smuzhiyun #define VAS_WIN_CACHE_STATUS		PPC_BIT(4)
218*4882a593Smuzhiyun 
219*4882a593Smuzhiyun #define VAS_TX_RSVD_BUF_COUNT_OFFSET	0x120
220*4882a593Smuzhiyun #define VAS_RXVD_BUF_COUNT		PPC_BITMASK(58, 63)
221*4882a593Smuzhiyun 
222*4882a593Smuzhiyun #define VAS_LRFIFO_WIN_PTR_OFFSET	0x128
223*4882a593Smuzhiyun #define VAS_LRX_WIN_ID			PPC_BITMASK(0, 15)
224*4882a593Smuzhiyun 
225*4882a593Smuzhiyun /*
226*4882a593Smuzhiyun  * Local Notification Control Register controls what happens in _response_
227*4882a593Smuzhiyun  * to a paste command and hence applies only to receive windows.
228*4882a593Smuzhiyun  */
229*4882a593Smuzhiyun #define VAS_LNOTIFY_CTL_OFFSET		0x138
230*4882a593Smuzhiyun #define VAS_NOTIFY_DISABLE		PPC_BIT(0)
231*4882a593Smuzhiyun #define VAS_INTR_DISABLE		PPC_BIT(1)
232*4882a593Smuzhiyun #define VAS_NOTIFY_EARLY		PPC_BIT(2)
233*4882a593Smuzhiyun #define VAS_NOTIFY_OSU_INTR		PPC_BIT(3)
234*4882a593Smuzhiyun 
235*4882a593Smuzhiyun #define VAS_LNOTIFY_PID_OFFSET		0x140
236*4882a593Smuzhiyun #define VAS_LNOTIFY_PID			PPC_BITMASK(0, 19)
237*4882a593Smuzhiyun 
238*4882a593Smuzhiyun #define VAS_LNOTIFY_LPID_OFFSET		0x148
239*4882a593Smuzhiyun #define VAS_LNOTIFY_LPID		PPC_BITMASK(0, 11)
240*4882a593Smuzhiyun 
241*4882a593Smuzhiyun #define VAS_LNOTIFY_TID_OFFSET		0x150
242*4882a593Smuzhiyun #define VAS_LNOTIFY_TID			PPC_BITMASK(0, 15)
243*4882a593Smuzhiyun 
244*4882a593Smuzhiyun #define VAS_LNOTIFY_SCOPE_OFFSET	0x158
245*4882a593Smuzhiyun #define VAS_LNOTIFY_MIN_SCOPE		PPC_BITMASK(0, 1)
246*4882a593Smuzhiyun #define VAS_LNOTIFY_MAX_SCOPE		PPC_BITMASK(2, 3)
247*4882a593Smuzhiyun 
248*4882a593Smuzhiyun #define VAS_NX_UTIL_OFFSET		0x1B0
249*4882a593Smuzhiyun #define VAS_NX_UTIL			PPC_BITMASK(0, 63)
250*4882a593Smuzhiyun 
251*4882a593Smuzhiyun /* SE: Side effects */
252*4882a593Smuzhiyun #define VAS_NX_UTIL_SE_OFFSET		0x1B8
253*4882a593Smuzhiyun #define VAS_NX_UTIL_SE			PPC_BITMASK(0, 63)
254*4882a593Smuzhiyun 
255*4882a593Smuzhiyun #define VAS_NX_UTIL_ADDER_OFFSET	0x180
256*4882a593Smuzhiyun #define VAS_NX_UTIL_ADDER		PPC_BITMASK(32, 63)
257*4882a593Smuzhiyun 
258*4882a593Smuzhiyun /*
259*4882a593Smuzhiyun  * VREG(x):
260*4882a593Smuzhiyun  * Expand a register's short name (eg: LPID) into two parameters:
261*4882a593Smuzhiyun  *	- the register's short name in string form ("LPID"), and
262*4882a593Smuzhiyun  *	- the name of the macro (eg: VAS_LPID_OFFSET), defining the
263*4882a593Smuzhiyun  *	  register's offset in the window context
264*4882a593Smuzhiyun  */
265*4882a593Smuzhiyun #define VREG_SFX(n, s)	__stringify(n), VAS_##n##s
266*4882a593Smuzhiyun #define VREG(r)		VREG_SFX(r, _OFFSET)
267*4882a593Smuzhiyun 
268*4882a593Smuzhiyun /*
269*4882a593Smuzhiyun  * Local Notify Scope Control Register. (Receive windows only).
270*4882a593Smuzhiyun  */
271*4882a593Smuzhiyun enum vas_notify_scope {
272*4882a593Smuzhiyun 	VAS_SCOPE_LOCAL,
273*4882a593Smuzhiyun 	VAS_SCOPE_GROUP,
274*4882a593Smuzhiyun 	VAS_SCOPE_VECTORED_GROUP,
275*4882a593Smuzhiyun 	VAS_SCOPE_UNUSED,
276*4882a593Smuzhiyun };
277*4882a593Smuzhiyun 
278*4882a593Smuzhiyun /*
279*4882a593Smuzhiyun  * Local DMA Cache Control Register (Receive windows only).
280*4882a593Smuzhiyun  */
281*4882a593Smuzhiyun enum vas_dma_type {
282*4882a593Smuzhiyun 	VAS_DMA_TYPE_INJECT,
283*4882a593Smuzhiyun 	VAS_DMA_TYPE_WRITE,
284*4882a593Smuzhiyun };
285*4882a593Smuzhiyun 
286*4882a593Smuzhiyun /*
287*4882a593Smuzhiyun  * Local Notify Scope Control Register. (Receive windows only).
288*4882a593Smuzhiyun  * Not applicable to NX receive windows.
289*4882a593Smuzhiyun  */
290*4882a593Smuzhiyun enum vas_notify_after_count {
291*4882a593Smuzhiyun 	VAS_NOTIFY_AFTER_256 = 0,
292*4882a593Smuzhiyun 	VAS_NOTIFY_NONE,
293*4882a593Smuzhiyun 	VAS_NOTIFY_AFTER_2
294*4882a593Smuzhiyun };
295*4882a593Smuzhiyun 
296*4882a593Smuzhiyun /*
297*4882a593Smuzhiyun  * NX can generate an interrupt for multiple faults and expects kernel
298*4882a593Smuzhiyun  * to process all of them. So read all valid CRB entries until find the
299*4882a593Smuzhiyun  * invalid one. So use pswid which is pasted by NX and ccw[0] (reserved
300*4882a593Smuzhiyun  * bit in BE) to check valid CRB. CCW[0] will not be touched by user
301*4882a593Smuzhiyun  * space. Application gets CRB formt error if it updates this bit.
302*4882a593Smuzhiyun  *
303*4882a593Smuzhiyun  * Invalidate FIFO during allocation and process all entries from last
304*4882a593Smuzhiyun  * successful read until finds invalid pswid and ccw[0] values.
305*4882a593Smuzhiyun  * After reading each CRB entry from fault FIFO, the kernel invalidate
306*4882a593Smuzhiyun  * it by updating pswid with FIFO_INVALID_ENTRY and CCW[0] with
307*4882a593Smuzhiyun  * CCW0_INVALID.
308*4882a593Smuzhiyun  */
309*4882a593Smuzhiyun #define FIFO_INVALID_ENTRY	0xffffffff
310*4882a593Smuzhiyun #define CCW0_INVALID		1
311*4882a593Smuzhiyun 
312*4882a593Smuzhiyun /*
313*4882a593Smuzhiyun  * One per instance of VAS. Each instance will have a separate set of
314*4882a593Smuzhiyun  * receive windows, one per coprocessor type.
315*4882a593Smuzhiyun  *
316*4882a593Smuzhiyun  * See also function header of set_vinst_win() for details on ->windows[]
317*4882a593Smuzhiyun  * and ->rxwin[] tables.
318*4882a593Smuzhiyun  */
319*4882a593Smuzhiyun struct vas_instance {
320*4882a593Smuzhiyun 	int vas_id;
321*4882a593Smuzhiyun 	struct ida ida;
322*4882a593Smuzhiyun 	struct list_head node;
323*4882a593Smuzhiyun 	struct platform_device *pdev;
324*4882a593Smuzhiyun 
325*4882a593Smuzhiyun 	u64 hvwc_bar_start;
326*4882a593Smuzhiyun 	u64 uwc_bar_start;
327*4882a593Smuzhiyun 	u64 paste_base_addr;
328*4882a593Smuzhiyun 	u64 paste_win_id_shift;
329*4882a593Smuzhiyun 
330*4882a593Smuzhiyun 	u64 irq_port;
331*4882a593Smuzhiyun 	int virq;
332*4882a593Smuzhiyun 	int fault_crbs;
333*4882a593Smuzhiyun 	int fault_fifo_size;
334*4882a593Smuzhiyun 	int fifo_in_progress;	/* To wake up thread or return IRQ_HANDLED */
335*4882a593Smuzhiyun 	spinlock_t fault_lock;	/* Protects fifo_in_progress update */
336*4882a593Smuzhiyun 	void *fault_fifo;
337*4882a593Smuzhiyun 	struct vas_window *fault_win; /* Fault window */
338*4882a593Smuzhiyun 
339*4882a593Smuzhiyun 	struct mutex mutex;
340*4882a593Smuzhiyun 	struct vas_window *rxwin[VAS_COP_TYPE_MAX];
341*4882a593Smuzhiyun 	struct vas_window *windows[VAS_WINDOWS_PER_CHIP];
342*4882a593Smuzhiyun 
343*4882a593Smuzhiyun 	char *dbgname;
344*4882a593Smuzhiyun 	struct dentry *dbgdir;
345*4882a593Smuzhiyun };
346*4882a593Smuzhiyun 
347*4882a593Smuzhiyun /*
348*4882a593Smuzhiyun  * In-kernel state a VAS window. One per window.
349*4882a593Smuzhiyun  */
350*4882a593Smuzhiyun struct vas_window {
351*4882a593Smuzhiyun 	/* Fields common to send and receive windows */
352*4882a593Smuzhiyun 	struct vas_instance *vinst;
353*4882a593Smuzhiyun 	int winid;
354*4882a593Smuzhiyun 	bool tx_win;		/* True if send window */
355*4882a593Smuzhiyun 	bool nx_win;		/* True if NX window */
356*4882a593Smuzhiyun 	bool user_win;		/* True if user space window */
357*4882a593Smuzhiyun 	void *hvwc_map;		/* HV window context */
358*4882a593Smuzhiyun 	void *uwc_map;		/* OS/User window context */
359*4882a593Smuzhiyun 	struct pid *pid;	/* Linux process id of owner */
360*4882a593Smuzhiyun 	struct pid *tgid;	/* Thread group ID of owner */
361*4882a593Smuzhiyun 	struct mm_struct *mm;	/* Linux process mm_struct */
362*4882a593Smuzhiyun 	int wcreds_max;		/* Window credits */
363*4882a593Smuzhiyun 
364*4882a593Smuzhiyun 	char *dbgname;
365*4882a593Smuzhiyun 	struct dentry *dbgdir;
366*4882a593Smuzhiyun 
367*4882a593Smuzhiyun 	/* Fields applicable only to send windows */
368*4882a593Smuzhiyun 	void *paste_kaddr;
369*4882a593Smuzhiyun 	char *paste_addr_name;
370*4882a593Smuzhiyun 	struct vas_window *rxwin;
371*4882a593Smuzhiyun 
372*4882a593Smuzhiyun 	/* Feilds applicable only to receive windows */
373*4882a593Smuzhiyun 	enum vas_cop_type cop;
374*4882a593Smuzhiyun 	atomic_t num_txwins;
375*4882a593Smuzhiyun };
376*4882a593Smuzhiyun 
377*4882a593Smuzhiyun /*
378*4882a593Smuzhiyun  * Container for the hardware state of a window. One per-window.
379*4882a593Smuzhiyun  *
380*4882a593Smuzhiyun  * A VAS Window context is a 512-byte area in the hardware that contains
381*4882a593Smuzhiyun  * a set of 64-bit registers. Individual bit-fields in these registers
382*4882a593Smuzhiyun  * determine the configuration/operation of the hardware. struct vas_winctx
383*4882a593Smuzhiyun  * is a container for the register fields in the window context.
384*4882a593Smuzhiyun  */
385*4882a593Smuzhiyun struct vas_winctx {
386*4882a593Smuzhiyun 	u64 rx_fifo;
387*4882a593Smuzhiyun 	int rx_fifo_size;
388*4882a593Smuzhiyun 	int wcreds_max;
389*4882a593Smuzhiyun 	int rsvd_txbuf_count;
390*4882a593Smuzhiyun 
391*4882a593Smuzhiyun 	bool user_win;
392*4882a593Smuzhiyun 	bool nx_win;
393*4882a593Smuzhiyun 	bool fault_win;
394*4882a593Smuzhiyun 	bool rsvd_txbuf_enable;
395*4882a593Smuzhiyun 	bool pin_win;
396*4882a593Smuzhiyun 	bool rej_no_credit;
397*4882a593Smuzhiyun 	bool tx_wcred_mode;
398*4882a593Smuzhiyun 	bool rx_wcred_mode;
399*4882a593Smuzhiyun 	bool tx_word_mode;
400*4882a593Smuzhiyun 	bool rx_word_mode;
401*4882a593Smuzhiyun 	bool data_stamp;
402*4882a593Smuzhiyun 	bool xtra_write;
403*4882a593Smuzhiyun 	bool notify_disable;
404*4882a593Smuzhiyun 	bool intr_disable;
405*4882a593Smuzhiyun 	bool fifo_disable;
406*4882a593Smuzhiyun 	bool notify_early;
407*4882a593Smuzhiyun 	bool notify_os_intr_reg;
408*4882a593Smuzhiyun 
409*4882a593Smuzhiyun 	int lpid;
410*4882a593Smuzhiyun 	int pidr;		/* value from SPRN_PID, not linux pid */
411*4882a593Smuzhiyun 	int lnotify_lpid;
412*4882a593Smuzhiyun 	int lnotify_pid;
413*4882a593Smuzhiyun 	int lnotify_tid;
414*4882a593Smuzhiyun 	u32 pswid;
415*4882a593Smuzhiyun 	int rx_win_id;
416*4882a593Smuzhiyun 	int fault_win_id;
417*4882a593Smuzhiyun 	int tc_mode;
418*4882a593Smuzhiyun 
419*4882a593Smuzhiyun 	u64 irq_port;
420*4882a593Smuzhiyun 
421*4882a593Smuzhiyun 	enum vas_dma_type dma_type;
422*4882a593Smuzhiyun 	enum vas_notify_scope min_scope;
423*4882a593Smuzhiyun 	enum vas_notify_scope max_scope;
424*4882a593Smuzhiyun 	enum vas_notify_after_count notify_after_count;
425*4882a593Smuzhiyun };
426*4882a593Smuzhiyun 
427*4882a593Smuzhiyun extern struct mutex vas_mutex;
428*4882a593Smuzhiyun 
429*4882a593Smuzhiyun extern struct vas_instance *find_vas_instance(int vasid);
430*4882a593Smuzhiyun extern void vas_init_dbgdir(void);
431*4882a593Smuzhiyun extern void vas_instance_init_dbgdir(struct vas_instance *vinst);
432*4882a593Smuzhiyun extern void vas_window_init_dbgdir(struct vas_window *win);
433*4882a593Smuzhiyun extern void vas_window_free_dbgdir(struct vas_window *win);
434*4882a593Smuzhiyun extern int vas_setup_fault_window(struct vas_instance *vinst);
435*4882a593Smuzhiyun extern irqreturn_t vas_fault_thread_fn(int irq, void *data);
436*4882a593Smuzhiyun extern irqreturn_t vas_fault_handler(int irq, void *dev_id);
437*4882a593Smuzhiyun extern void vas_return_credit(struct vas_window *window, bool tx);
438*4882a593Smuzhiyun extern struct vas_window *vas_pswid_to_window(struct vas_instance *vinst,
439*4882a593Smuzhiyun 						uint32_t pswid);
440*4882a593Smuzhiyun extern void vas_win_paste_addr(struct vas_window *window, u64 *addr,
441*4882a593Smuzhiyun 					int *len);
442*4882a593Smuzhiyun 
vas_window_pid(struct vas_window * window)443*4882a593Smuzhiyun static inline int vas_window_pid(struct vas_window *window)
444*4882a593Smuzhiyun {
445*4882a593Smuzhiyun 	return pid_vnr(window->pid);
446*4882a593Smuzhiyun }
447*4882a593Smuzhiyun 
vas_log_write(struct vas_window * win,char * name,void * regptr,u64 val)448*4882a593Smuzhiyun static inline void vas_log_write(struct vas_window *win, char *name,
449*4882a593Smuzhiyun 			void *regptr, u64 val)
450*4882a593Smuzhiyun {
451*4882a593Smuzhiyun 	if (val)
452*4882a593Smuzhiyun 		pr_debug("%swin #%d: %s reg %p, val 0x%016llx\n",
453*4882a593Smuzhiyun 				win->tx_win ? "Tx" : "Rx", win->winid, name,
454*4882a593Smuzhiyun 				regptr, val);
455*4882a593Smuzhiyun }
456*4882a593Smuzhiyun 
write_uwc_reg(struct vas_window * win,char * name,s32 reg,u64 val)457*4882a593Smuzhiyun static inline void write_uwc_reg(struct vas_window *win, char *name,
458*4882a593Smuzhiyun 			s32 reg, u64 val)
459*4882a593Smuzhiyun {
460*4882a593Smuzhiyun 	void *regptr;
461*4882a593Smuzhiyun 
462*4882a593Smuzhiyun 	regptr = win->uwc_map + reg;
463*4882a593Smuzhiyun 	vas_log_write(win, name, regptr, val);
464*4882a593Smuzhiyun 
465*4882a593Smuzhiyun 	out_be64(regptr, val);
466*4882a593Smuzhiyun }
467*4882a593Smuzhiyun 
write_hvwc_reg(struct vas_window * win,char * name,s32 reg,u64 val)468*4882a593Smuzhiyun static inline void write_hvwc_reg(struct vas_window *win, char *name,
469*4882a593Smuzhiyun 			s32 reg, u64 val)
470*4882a593Smuzhiyun {
471*4882a593Smuzhiyun 	void *regptr;
472*4882a593Smuzhiyun 
473*4882a593Smuzhiyun 	regptr = win->hvwc_map + reg;
474*4882a593Smuzhiyun 	vas_log_write(win, name, regptr, val);
475*4882a593Smuzhiyun 
476*4882a593Smuzhiyun 	out_be64(regptr, val);
477*4882a593Smuzhiyun }
478*4882a593Smuzhiyun 
read_hvwc_reg(struct vas_window * win,char * name __maybe_unused,s32 reg)479*4882a593Smuzhiyun static inline u64 read_hvwc_reg(struct vas_window *win,
480*4882a593Smuzhiyun 			char *name __maybe_unused, s32 reg)
481*4882a593Smuzhiyun {
482*4882a593Smuzhiyun 	return in_be64(win->hvwc_map+reg);
483*4882a593Smuzhiyun }
484*4882a593Smuzhiyun 
485*4882a593Smuzhiyun /*
486*4882a593Smuzhiyun  * Encode/decode the Partition Send Window ID (PSWID) for a window in
487*4882a593Smuzhiyun  * a way that we can uniquely identify any window in the system. i.e.
488*4882a593Smuzhiyun  * we should be able to locate the 'struct vas_window' given the PSWID.
489*4882a593Smuzhiyun  *
490*4882a593Smuzhiyun  *	Bits	Usage
491*4882a593Smuzhiyun  *	0:7	VAS id (8 bits)
492*4882a593Smuzhiyun  *	8:15	Unused, 0 (3 bits)
493*4882a593Smuzhiyun  *	16:31	Window id (16 bits)
494*4882a593Smuzhiyun  */
encode_pswid(int vasid,int winid)495*4882a593Smuzhiyun static inline u32 encode_pswid(int vasid, int winid)
496*4882a593Smuzhiyun {
497*4882a593Smuzhiyun 	return ((u32)winid | (vasid << (31 - 7)));
498*4882a593Smuzhiyun }
499*4882a593Smuzhiyun 
decode_pswid(u32 pswid,int * vasid,int * winid)500*4882a593Smuzhiyun static inline void decode_pswid(u32 pswid, int *vasid, int *winid)
501*4882a593Smuzhiyun {
502*4882a593Smuzhiyun 	if (vasid)
503*4882a593Smuzhiyun 		*vasid = pswid >> (31 - 7) & 0xFF;
504*4882a593Smuzhiyun 
505*4882a593Smuzhiyun 	if (winid)
506*4882a593Smuzhiyun 		*winid = pswid & 0xFFFF;
507*4882a593Smuzhiyun }
508*4882a593Smuzhiyun #endif /* _VAS_H */
509