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