OLD | NEW |
(Empty) | |
| 1 /* |
| 2 * Copyright © 2008 Red Hat, Inc. |
| 3 * |
| 4 * Permission is hereby granted, free of charge, to any person obtaining a |
| 5 * copy of this software and associated documentation files (the "Soft- |
| 6 * ware"), to deal in the Software without restriction, including without |
| 7 * limitation the rights to use, copy, modify, merge, publish, distribute, |
| 8 * and/or sell copies of the Software, and to permit persons to whom the |
| 9 * Software is furnished to do so, provided that the above copyright |
| 10 * notice(s) and this permission notice appear in all copies of the Soft- |
| 11 * ware and that both the above copyright notice(s) and this permission |
| 12 * notice appear in supporting documentation. |
| 13 * |
| 14 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS |
| 15 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABIL- |
| 16 * ITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT OF THIRD PARTY |
| 17 * RIGHTS. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR HOLDERS INCLUDED IN |
| 18 * THIS NOTICE BE LIABLE FOR ANY CLAIM, OR ANY SPECIAL INDIRECT OR CONSE- |
| 19 * QUENTIAL DAMAGES, OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, |
| 20 * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER |
| 21 * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFOR- |
| 22 * MANCE OF THIS SOFTWARE. |
| 23 * |
| 24 * Except as contained in this notice, the name of a copyright holder shall |
| 25 * not be used in advertising or otherwise to promote the sale, use or |
| 26 * other dealings in this Software without prior written authorization of |
| 27 * the copyright holder. |
| 28 * |
| 29 * Authors: |
| 30 * Kristian Høgsberg (krh@redhat.com) |
| 31 */ |
| 32 |
| 33 |
| 34 #define NEED_REPLIES |
| 35 #include <X11/Xlibint.h> |
| 36 #include <X11/extensions/Xext.h> |
| 37 #include <X11/extensions/extutil.h> |
| 38 #include "xf86drm.h" |
| 39 #include "va_dri2.h" |
| 40 #include "va_dri2str.h" |
| 41 #include "va_dri2tokens.h" |
| 42 |
| 43 #ifndef DRI2DriverDRI |
| 44 #define DRI2DriverDRI 0 |
| 45 #endif |
| 46 |
| 47 static int |
| 48 VA_DRI2Error(Display *dpy, xError *err, XExtCodes *codes, int *ret_code); |
| 49 |
| 50 static char va_dri2ExtensionName[] = DRI2_NAME; |
| 51 static XExtensionInfo _va_dri2_info_data; |
| 52 static XExtensionInfo *va_dri2Info = &_va_dri2_info_data; |
| 53 static XEXT_GENERATE_CLOSE_DISPLAY (VA_DRI2CloseDisplay, va_dri2Info) |
| 54 static /* const */ XExtensionHooks va_dri2ExtensionHooks = { |
| 55 NULL, /* create_gc */ |
| 56 NULL, /* copy_gc */ |
| 57 NULL, /* flush_gc */ |
| 58 NULL, /* free_gc */ |
| 59 NULL, /* create_font */ |
| 60 NULL, /* free_font */ |
| 61 VA_DRI2CloseDisplay, /* close_display */ |
| 62 NULL, /* wire_to_event */ |
| 63 NULL, /* event_to_wire */ |
| 64 VA_DRI2Error, /* error */ |
| 65 NULL, /* error_string */ |
| 66 }; |
| 67 |
| 68 static XEXT_GENERATE_FIND_DISPLAY (DRI2FindDisplay, va_dri2Info, |
| 69 va_dri2ExtensionName, |
| 70 &va_dri2ExtensionHooks, |
| 71 0, NULL) |
| 72 |
| 73 static int |
| 74 VA_DRI2Error(Display *dpy, xError *err, XExtCodes *codes, int *ret_code) |
| 75 { |
| 76 /* |
| 77 * If the X drawable was destroyed before the VA drawable, the DRI2 drawable |
| 78 * will be gone by the time we call VA_DRI2DestroyDrawable(). So, simply |
| 79 * ignore BadDrawable errors in that case. |
| 80 */ |
| 81 if (err->majorCode == codes->major_opcode && |
| 82 err->errorCode == BadDrawable && |
| 83 err->minorCode == X_DRI2DestroyDrawable) |
| 84 return True; |
| 85 |
| 86 return False; |
| 87 } |
| 88 |
| 89 Bool VA_DRI2QueryExtension(Display *dpy, int *eventBase, int *errorBase) |
| 90 { |
| 91 XExtDisplayInfo *info = DRI2FindDisplay(dpy); |
| 92 |
| 93 if (XextHasExtension(info)) { |
| 94 *eventBase = info->codes->first_event; |
| 95 *errorBase = info->codes->first_error; |
| 96 return True; |
| 97 } |
| 98 |
| 99 return False; |
| 100 } |
| 101 |
| 102 Bool VA_DRI2QueryVersion(Display *dpy, int *major, int *minor) |
| 103 { |
| 104 XExtDisplayInfo *info = DRI2FindDisplay (dpy); |
| 105 xDRI2QueryVersionReply rep; |
| 106 xDRI2QueryVersionReq *req; |
| 107 |
| 108 XextCheckExtension (dpy, info, va_dri2ExtensionName, False); |
| 109 |
| 110 LockDisplay(dpy); |
| 111 GetReq(DRI2QueryVersion, req); |
| 112 req->reqType = info->codes->major_opcode; |
| 113 req->dri2ReqType = X_DRI2QueryVersion; |
| 114 req->majorVersion = DRI2_MAJOR; |
| 115 req->minorVersion = DRI2_MINOR; |
| 116 if (!_XReply(dpy, (xReply *)&rep, 0, xFalse)) { |
| 117 UnlockDisplay(dpy); |
| 118 SyncHandle(); |
| 119 return False; |
| 120 } |
| 121 *major = rep.majorVersion; |
| 122 *minor = rep.minorVersion; |
| 123 UnlockDisplay(dpy); |
| 124 SyncHandle(); |
| 125 |
| 126 return True; |
| 127 } |
| 128 |
| 129 Bool VA_DRI2Connect(Display *dpy, XID window, |
| 130 char **driverName, char **deviceName) |
| 131 { |
| 132 XExtDisplayInfo *info = DRI2FindDisplay(dpy); |
| 133 xDRI2ConnectReply rep; |
| 134 xDRI2ConnectReq *req; |
| 135 |
| 136 XextCheckExtension (dpy, info, va_dri2ExtensionName, False); |
| 137 |
| 138 LockDisplay(dpy); |
| 139 GetReq(DRI2Connect, req); |
| 140 req->reqType = info->codes->major_opcode; |
| 141 req->dri2ReqType = X_DRI2Connect; |
| 142 req->window = window; |
| 143 req->driverType = DRI2DriverDRI; |
| 144 if (!_XReply(dpy, (xReply *)&rep, 0, xFalse)) { |
| 145 UnlockDisplay(dpy); |
| 146 SyncHandle(); |
| 147 return False; |
| 148 } |
| 149 |
| 150 if (rep.driverNameLength == 0 && rep.deviceNameLength == 0) { |
| 151 UnlockDisplay(dpy); |
| 152 SyncHandle(); |
| 153 return False; |
| 154 } |
| 155 |
| 156 *driverName = Xmalloc(rep.driverNameLength + 1); |
| 157 if (*driverName == NULL) { |
| 158 _XEatData(dpy, |
| 159 ((rep.driverNameLength + 3) & ~3) + |
| 160 ((rep.deviceNameLength + 3) & ~3)); |
| 161 UnlockDisplay(dpy); |
| 162 SyncHandle(); |
| 163 return False; |
| 164 } |
| 165 _XReadPad(dpy, *driverName, rep.driverNameLength); |
| 166 (*driverName)[rep.driverNameLength] = '\0'; |
| 167 |
| 168 *deviceName = Xmalloc(rep.deviceNameLength + 1); |
| 169 if (*deviceName == NULL) { |
| 170 Xfree(*driverName); |
| 171 _XEatData(dpy, ((rep.deviceNameLength + 3) & ~3)); |
| 172 UnlockDisplay(dpy); |
| 173 SyncHandle(); |
| 174 return False; |
| 175 } |
| 176 _XReadPad(dpy, *deviceName, rep.deviceNameLength); |
| 177 (*deviceName)[rep.deviceNameLength] = '\0'; |
| 178 |
| 179 UnlockDisplay(dpy); |
| 180 SyncHandle(); |
| 181 |
| 182 return True; |
| 183 } |
| 184 |
| 185 Bool VA_DRI2Authenticate(Display *dpy, XID window, drm_magic_t magic) |
| 186 { |
| 187 XExtDisplayInfo *info = DRI2FindDisplay(dpy); |
| 188 xDRI2AuthenticateReq *req; |
| 189 xDRI2AuthenticateReply rep; |
| 190 |
| 191 XextCheckExtension (dpy, info, va_dri2ExtensionName, False); |
| 192 |
| 193 LockDisplay(dpy); |
| 194 GetReq(DRI2Authenticate, req); |
| 195 req->reqType = info->codes->major_opcode; |
| 196 req->dri2ReqType = X_DRI2Authenticate; |
| 197 req->window = window; |
| 198 req->magic = magic; |
| 199 |
| 200 if (!_XReply(dpy, (xReply *)&rep, 0, xFalse)) { |
| 201 UnlockDisplay(dpy); |
| 202 SyncHandle(); |
| 203 return False; |
| 204 } |
| 205 |
| 206 UnlockDisplay(dpy); |
| 207 SyncHandle(); |
| 208 |
| 209 return rep.authenticated; |
| 210 } |
| 211 |
| 212 void VA_DRI2CreateDrawable(Display *dpy, XID drawable) |
| 213 { |
| 214 XExtDisplayInfo *info = DRI2FindDisplay(dpy); |
| 215 xDRI2CreateDrawableReq *req; |
| 216 |
| 217 XextSimpleCheckExtension (dpy, info, va_dri2ExtensionName); |
| 218 |
| 219 LockDisplay(dpy); |
| 220 GetReq(DRI2CreateDrawable, req); |
| 221 req->reqType = info->codes->major_opcode; |
| 222 req->dri2ReqType = X_DRI2CreateDrawable; |
| 223 req->drawable = drawable; |
| 224 UnlockDisplay(dpy); |
| 225 SyncHandle(); |
| 226 } |
| 227 |
| 228 void VA_DRI2DestroyDrawable(Display *dpy, XID drawable) |
| 229 { |
| 230 XExtDisplayInfo *info = DRI2FindDisplay(dpy); |
| 231 xDRI2DestroyDrawableReq *req; |
| 232 |
| 233 XextSimpleCheckExtension (dpy, info, va_dri2ExtensionName); |
| 234 |
| 235 XSync(dpy, False); |
| 236 |
| 237 LockDisplay(dpy); |
| 238 GetReq(DRI2DestroyDrawable, req); |
| 239 req->reqType = info->codes->major_opcode; |
| 240 req->dri2ReqType = X_DRI2DestroyDrawable; |
| 241 req->drawable = drawable; |
| 242 UnlockDisplay(dpy); |
| 243 SyncHandle(); |
| 244 } |
| 245 |
| 246 VA_DRI2Buffer *VA_DRI2GetBuffers(Display *dpy, XID drawable, |
| 247 int *width, int *height, |
| 248 unsigned int *attachments, int count, |
| 249 int *outCount) |
| 250 { |
| 251 XExtDisplayInfo *info = DRI2FindDisplay(dpy); |
| 252 xDRI2GetBuffersReply rep; |
| 253 xDRI2GetBuffersReq *req; |
| 254 VA_DRI2Buffer *buffers; |
| 255 xDRI2Buffer repBuffer; |
| 256 CARD32 *p; |
| 257 int i; |
| 258 |
| 259 XextCheckExtension (dpy, info, va_dri2ExtensionName, False); |
| 260 |
| 261 LockDisplay(dpy); |
| 262 GetReqExtra(DRI2GetBuffers, count * 4, req); |
| 263 req->reqType = info->codes->major_opcode; |
| 264 req->dri2ReqType = X_DRI2GetBuffers; |
| 265 req->drawable = drawable; |
| 266 req->count = count; |
| 267 p = (CARD32 *) &req[1]; |
| 268 for (i = 0; i < count; i++) |
| 269 p[i] = attachments[i]; |
| 270 |
| 271 if (!_XReply(dpy, (xReply *)&rep, 0, xFalse)) { |
| 272 UnlockDisplay(dpy); |
| 273 SyncHandle(); |
| 274 return NULL; |
| 275 } |
| 276 |
| 277 *width = rep.width; |
| 278 *height = rep.height; |
| 279 *outCount = rep.count; |
| 280 |
| 281 buffers = Xmalloc(rep.count * sizeof buffers[0]); |
| 282 if (buffers == NULL) { |
| 283 _XEatData(dpy, rep.count * sizeof repBuffer); |
| 284 UnlockDisplay(dpy); |
| 285 SyncHandle(); |
| 286 return NULL; |
| 287 } |
| 288 |
| 289 for (i = 0; i < rep.count; i++) { |
| 290 _XReadPad(dpy, (char *) &repBuffer, sizeof repBuffer); |
| 291 buffers[i].attachment = repBuffer.attachment; |
| 292 buffers[i].name = repBuffer.name; |
| 293 buffers[i].pitch = repBuffer.pitch; |
| 294 buffers[i].cpp = repBuffer.cpp; |
| 295 buffers[i].flags = repBuffer.flags; |
| 296 } |
| 297 |
| 298 UnlockDisplay(dpy); |
| 299 SyncHandle(); |
| 300 |
| 301 return buffers; |
| 302 } |
| 303 |
| 304 void VA_DRI2CopyRegion(Display *dpy, XID drawable, XserverRegion region, |
| 305 CARD32 dest, CARD32 src) |
| 306 { |
| 307 XExtDisplayInfo *info = DRI2FindDisplay(dpy); |
| 308 xDRI2CopyRegionReq *req; |
| 309 xDRI2CopyRegionReply rep; |
| 310 |
| 311 XextSimpleCheckExtension (dpy, info, va_dri2ExtensionName); |
| 312 |
| 313 LockDisplay(dpy); |
| 314 GetReq(DRI2CopyRegion, req); |
| 315 req->reqType = info->codes->major_opcode; |
| 316 req->dri2ReqType = X_DRI2CopyRegion; |
| 317 req->drawable = drawable; |
| 318 req->region = region; |
| 319 req->dest = dest; |
| 320 req->src = src; |
| 321 |
| 322 _XReply(dpy, (xReply *)&rep, 0, xFalse); |
| 323 |
| 324 UnlockDisplay(dpy); |
| 325 SyncHandle(); |
| 326 } |
| 327 |
| 328 static void |
| 329 load_swap_req(xDRI2SwapBuffersReq *req, CARD64 target, CARD64 divisor, |
| 330 CARD64 remainder) |
| 331 { |
| 332 req->target_msc_hi = target >> 32; |
| 333 req->target_msc_lo = target & 0xffffffff; |
| 334 req->divisor_hi = divisor >> 32; |
| 335 req->divisor_lo = divisor & 0xffffffff; |
| 336 req->remainder_hi = remainder >> 32; |
| 337 req->remainder_lo = remainder & 0xffffffff; |
| 338 } |
| 339 |
| 340 static CARD64 |
| 341 vals_to_card64(CARD32 lo, CARD32 hi) |
| 342 { |
| 343 return (CARD64)hi << 32 | lo; |
| 344 } |
| 345 |
| 346 void VA_DRI2SwapBuffers(Display *dpy, XID drawable, CARD64 target_msc, |
| 347 CARD64 divisor, CARD64 remainder, CARD64 *count) |
| 348 { |
| 349 XExtDisplayInfo *info = DRI2FindDisplay(dpy); |
| 350 xDRI2SwapBuffersReq *req; |
| 351 xDRI2SwapBuffersReply rep; |
| 352 |
| 353 XextSimpleCheckExtension (dpy, info, va_dri2ExtensionName); |
| 354 |
| 355 LockDisplay(dpy); |
| 356 GetReq(DRI2SwapBuffers, req); |
| 357 req->reqType = info->codes->major_opcode; |
| 358 req->dri2ReqType = X_DRI2SwapBuffers; |
| 359 req->drawable = drawable; |
| 360 load_swap_req(req, target_msc, divisor, remainder); |
| 361 |
| 362 _XReply(dpy, (xReply *)&rep, 0, xFalse); |
| 363 |
| 364 *count = vals_to_card64(rep.swap_lo, rep.swap_hi); |
| 365 |
| 366 UnlockDisplay(dpy); |
| 367 SyncHandle(); |
| 368 } |
OLD | NEW |