1 /*
2 // Copyright (c) 2014 Intel Corporation
3 //
4 // Licensed under the Apache License, Version 2.0 (the "License");
5 // you may not use this file except in compliance with the License.
6 // You may obtain a copy of the License at
7 //
8 // http://www.apache.org/licenses/LICENSE-2.0
9 //
10 // Unless required by applicable law or agreed to in writing, software
11 // distributed under the License is distributed on an "AS IS" BASIS,
12 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 // See the License for the specific language governing permissions and
14 // limitations under the License.
15 */
16
17 #include <wsbm_pool.h>
18 #include <wsbm_driver.h>
19 #include <wsbm_manager.h>
20 #include <wsbm_util.h>
21 #include <drm/ttm/ttm_placement.h>
22 #include <linux/psb_drm.h>
23 #include <xf86drm.h>
24 #include <common/utils/HwcTrace.h>
25
26 struct _WsbmBufferPool * mainPool = NULL;
27
28 struct PsbWsbmValidateNode
29 {
30 struct _ValidateNode base;
31 struct psb_validate_arg arg;
32 };
33
align_to(uint32_t arg,uint32_t align)34 static inline uint32_t align_to(uint32_t arg, uint32_t align)
35 {
36 return ((arg + (align - 1)) & (~(align - 1)));
37 }
38
pvrAlloc(struct _WsbmVNodeFuncs * func,int typeId)39 static struct _ValidateNode * pvrAlloc(struct _WsbmVNodeFuncs * func,
40 int typeId)
41 {
42 CTRACE();
43 if(typeId == 0) {
44 struct PsbWsbmValidateNode * vNode = malloc(sizeof(*vNode));
45 if(!vNode) {
46 ELOGTRACE("failed to allocate memory");
47 return NULL;
48 }
49
50 vNode->base.func = func;
51 vNode->base.type_id = 0;
52 return &vNode->base;
53 } else {
54 struct _ValidateNode * node = malloc(sizeof(*node));
55 if(!node) {
56 ELOGTRACE("failed to allocate node");
57 return NULL;
58 }
59
60 node->func = func;
61 node->type_id = 1;
62 return node;
63 }
64 }
65
pvrFree(struct _ValidateNode * node)66 static void pvrFree(struct _ValidateNode * node)
67 {
68 CTRACE();
69 if(node->type_id == 0) {
70 free(containerOf(node, struct PsbWsbmValidateNode, base));
71 } else {
72 free(node);
73 }
74 }
75
pvrClear(struct _ValidateNode * node)76 static void pvrClear(struct _ValidateNode * node)
77 {
78 CTRACE();
79 if(node->type_id == 0) {
80 struct PsbWsbmValidateNode * vNode =
81 containerOf(node, struct PsbWsbmValidateNode, base);
82 memset(&vNode->arg.d.req, 0, sizeof(vNode->arg.d.req));
83 }
84 }
85
86 static struct _WsbmVNodeFuncs vNodeFuncs = {
87 .alloc = pvrAlloc,
88 .free = pvrFree,
89 .clear = pvrClear,
90 };
91
psbWsbmTakedown()92 void psbWsbmTakedown()
93 {
94 CTRACE();
95
96 if (mainPool) {
97 wsbmPoolTakeDown(mainPool);
98 mainPool = NULL;
99 }
100
101 if (wsbmIsInitialized()) {
102 wsbmTakedown();
103 }
104 }
105
psbWsbmInitialize(int drmFD)106 int psbWsbmInitialize(int drmFD)
107 {
108 union drm_psb_extension_arg arg;
109 const char drmExt[] = "psb_ttm_placement_alphadrop";
110 int ret = 0;
111
112 CTRACE();
113
114 if (drmFD <= 0) {
115 ELOGTRACE("invalid drm fd %d", drmFD);
116 return drmFD;
117 }
118
119 /*init wsbm*/
120 ret = wsbmInit(wsbmNullThreadFuncs(), &vNodeFuncs);
121 if (ret) {
122 ELOGTRACE("failed to initialize Wsbm, error code %d", ret);
123 return ret;
124 }
125
126 VLOGTRACE("DRM_PSB_EXTENSION %d", DRM_PSB_EXTENSION);
127
128 /*get devOffset via drm IOCTL*/
129 strncpy(arg.extension, drmExt, sizeof(drmExt));
130
131 ret = drmCommandWriteRead(drmFD, 6/*DRM_PSB_EXTENSION*/, &arg, sizeof(arg));
132 if(ret || !arg.rep.exists) {
133 ELOGTRACE("failed to get device offset, error code %d", ret);
134 goto out;
135 }
136
137 VLOGTRACE("ioctl offset %#x", arg.rep.driver_ioctl_offset);
138
139 mainPool = wsbmTTMPoolInit(drmFD, arg.rep.driver_ioctl_offset);
140 if(!mainPool) {
141 ELOGTRACE("failed to initialize TTM Pool");
142 ret = -EINVAL;
143 goto out;
144 }
145
146 VLOGTRACE("Wsbm initialization succeeded. mainPool %p", mainPool);
147
148 return 0;
149
150 out:
151 psbWsbmTakedown();
152 return ret;
153 }
154
psbWsbmAllocateFromUB(uint32_t size,uint32_t align,void ** buf,void * user_pt)155 int psbWsbmAllocateFromUB(uint32_t size, uint32_t align, void ** buf, void *user_pt)
156 {
157 struct _WsbmBufferObject * wsbmBuf = NULL;
158 int ret = 0;
159 int offset = 0;
160
161 ALOGTRACE("size %d", align_to(size, 4096));
162
163 if(!buf || !user_pt) {
164 ELOGTRACE("invalid parameter");
165 return -EINVAL;
166 }
167
168 VLOGTRACE("mainPool %p", mainPool);
169
170 ret = wsbmGenBuffers(mainPool, 1, &wsbmBuf, align,
171 DRM_PSB_FLAG_MEM_MMU | WSBM_PL_FLAG_CACHED |
172 WSBM_PL_FLAG_NO_EVICT | WSBM_PL_FLAG_SHARED);
173 if(ret) {
174 ELOGTRACE("wsbmGenBuffers failed with error code %d", ret);
175 return ret;
176 }
177
178 ret = wsbmBODataUB(wsbmBuf,
179 align_to(size, 4096), NULL, NULL, 0,
180 user_pt);
181
182 if(ret) {
183 ELOGTRACE("wsbmBOData failed with error code %d", ret);
184 /*FIXME: should I unreference this buffer here?*/
185 return ret;
186 }
187
188 *buf = wsbmBuf;
189
190 VLOGTRACE("ttm UB buffer allocated. %p", *buf);
191 return 0;
192 }
193
psbWsbmAllocateTTMBuffer(uint32_t size,uint32_t align,void ** buf)194 int psbWsbmAllocateTTMBuffer(uint32_t size, uint32_t align, void ** buf)
195 {
196 struct _WsbmBufferObject * wsbmBuf = NULL;
197 int ret = 0;
198 int offset = 0;
199
200 ALOGTRACE("size %d", align_to(size, 4096));
201
202 if(!buf) {
203 ELOGTRACE("invalid parameter");
204 return -EINVAL;
205 }
206
207 VLOGTRACE("mainPool %p", mainPool);
208
209 ret = wsbmGenBuffers(mainPool, 1, &wsbmBuf, align,
210 (WSBM_PL_FLAG_VRAM | WSBM_PL_FLAG_TT |
211 WSBM_PL_FLAG_SHARED | WSBM_PL_FLAG_NO_EVICT));
212 if(ret) {
213 ELOGTRACE("wsbmGenBuffers failed with error code %d", ret);
214 return ret;
215 }
216
217 ret = wsbmBOData(wsbmBuf, align_to(size, 4096), NULL, NULL, 0);
218 if(ret) {
219 ELOGTRACE("wsbmBOData failed with error code %d", ret);
220 /*FIXME: should I unreference this buffer here?*/
221 return ret;
222 }
223
224 /* wsbmBOReference(wsbmBuf); */ /* no need to add reference */
225
226 *buf = wsbmBuf;
227
228 VLOGTRACE("ttm buffer allocated. %p", *buf);
229 return 0;
230 }
231
psbWsbmWrapTTMBuffer(uint32_t handle,void ** buf)232 int psbWsbmWrapTTMBuffer(uint32_t handle, void **buf)
233 {
234 int ret = 0;
235 struct _WsbmBufferObject *wsbmBuf;
236
237 if (!buf) {
238 ELOGTRACE("invalid parameter");
239 return -EINVAL;
240 }
241
242 ret = wsbmGenBuffers(mainPool, 1, &wsbmBuf, 0,
243 (WSBM_PL_FLAG_VRAM | WSBM_PL_FLAG_TT |
244 /*WSBM_PL_FLAG_NO_EVICT |*/ WSBM_PL_FLAG_SHARED));
245
246 if (ret) {
247 ELOGTRACE("wsbmGenBuffers failed with error code %d", ret);
248 return ret;
249 }
250
251 ret = wsbmBOSetReferenced(wsbmBuf, handle);
252 if (ret) {
253 ELOGTRACE("wsbmBOSetReferenced failed with error code %d", ret);
254 return ret;
255 }
256
257 *buf = (void *)wsbmBuf;
258
259 VLOGTRACE("wrap buffer %p for handle %#x", wsbmBuf, handle);
260 return 0;
261 }
262
psbWsbmWrapTTMBuffer2(uint32_t handle,void ** buf)263 int psbWsbmWrapTTMBuffer2(uint32_t handle, void **buf)
264 {
265 int ret = 0;
266 struct _WsbmBufferObject *wsbmBuf;
267
268 if (!buf) {
269 ELOGTRACE("invalid parameter");
270 return -EINVAL;
271 }
272
273 ret = wsbmGenBuffers(mainPool, 1, &wsbmBuf, 4096,
274 (WSBM_PL_FLAG_SHARED | DRM_PSB_FLAG_MEM_MMU | WSBM_PL_FLAG_UNCACHED));
275
276 if (ret) {
277 ELOGTRACE("wsbmGenBuffers failed with error code %d", ret);
278 return ret;
279 }
280
281 *buf = (void *)wsbmBuf;
282
283 VLOGTRACE("wrap buffer %p for handle %#x", wsbmBuf, handle);
284 return 0;
285 }
286
287
psbWsbmCreateFromUB(void * buf,uint32_t size,void * vaddr)288 int psbWsbmCreateFromUB(void *buf, uint32_t size, void *vaddr)
289 {
290 int ret = 0;
291 struct _WsbmBufferObject *wsbmBuf;
292
293 if (!buf || !vaddr) {
294 ELOGTRACE("invalid parameter");
295 return -EINVAL;
296 }
297
298 wsbmBuf = (struct _WsbmBufferObject *)buf;
299 ret = wsbmBODataUB(wsbmBuf, size, NULL, NULL, 0, vaddr);
300 if (ret) {
301 ELOGTRACE("wsbmBODataUB failed with error code %d", ret);
302 return ret;
303 }
304
305 return 0;
306 }
307
psbWsbmUnReference(void * buf)308 int psbWsbmUnReference(void *buf)
309 {
310 struct _WsbmBufferObject *wsbmBuf;
311
312 if (!buf) {
313 ELOGTRACE("invalid parameter");
314 return -EINVAL;
315 }
316
317 wsbmBuf = (struct _WsbmBufferObject *)buf;
318
319 wsbmBOUnreference(&wsbmBuf);
320
321 return 0;
322 }
323
psbWsbmDestroyTTMBuffer(void * buf)324 int psbWsbmDestroyTTMBuffer(void * buf)
325 {
326 CTRACE();
327
328 if(!buf) {
329 ELOGTRACE("invalid ttm buffer");
330 return -EINVAL;
331 }
332
333 /*FIXME: should I unmap this buffer object first?*/
334 wsbmBOUnmap((struct _WsbmBufferObject *)buf);
335
336 wsbmBOUnreference((struct _WsbmBufferObject **)&buf);
337
338 XLOGTRACE();
339
340 return 0;
341 }
342
psbWsbmGetCPUAddress(void * buf)343 void * psbWsbmGetCPUAddress(void * buf)
344 {
345 if(!buf) {
346 ELOGTRACE("invalid ttm buffer");
347 return NULL;
348 }
349
350 VLOGTRACE("buffer object %p", buf);
351
352 void * address = wsbmBOMap((struct _WsbmBufferObject *)buf,
353 WSBM_ACCESS_READ | WSBM_ACCESS_WRITE);
354 if(!address) {
355 ELOGTRACE("failed to map buffer object");
356 return NULL;
357 }
358
359 VLOGTRACE("mapped successfully. %p, size %ld",
360 address, wsbmBOSize((struct _WsbmBufferObject *)buf));
361
362 return address;
363 }
364
psbWsbmGetGttOffset(void * buf)365 uint32_t psbWsbmGetGttOffset(void * buf)
366 {
367 if(!buf) {
368 ELOGTRACE("invalid ttm buffer");
369 return 0;
370 }
371
372 VLOGTRACE("buffer object %p", buf);
373
374 uint32_t offset =
375 wsbmBOOffsetHint((struct _WsbmBufferObject *)buf) - 0x10000000;
376
377 VLOGTRACE("offset %#x", offset >> 12);
378
379 return offset >> 12;
380 }
381
psbWsbmGetKBufHandle(void * buf)382 uint32_t psbWsbmGetKBufHandle(void *buf)
383 {
384 if (!buf) {
385 ELOGTRACE("invalid ttm buffer");
386 return 0;
387 }
388
389 return (wsbmKBufHandle(wsbmKBuf((struct _WsbmBufferObject *)buf)));
390 }
391
psbWsbmWaitIdle(void * buf)392 uint32_t psbWsbmWaitIdle(void *buf)
393 {
394 if (!buf) {
395 ELOGTRACE("invalid ttm buffer");
396 return -EINVAL;
397 }
398
399 wsbmBOWaitIdle(buf, 0);
400 return 0;
401 }
402