/*************************************************************************** Copyright 2000-2011 Intel Corporation. All Rights Reserved. Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sub license, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice (including the next paragraph) shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL INTEL, AND/OR ITS SUPPLIERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. **************************************************************************/ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include "sna.h" #include "sna_video.h" #include "intel_options.h" #include #include #include #include #include #include #ifdef DRM_IOCTL_MODE_GETPLANERESOURCES #include #define IMAGE_MAX_WIDTH 2048 #define IMAGE_MAX_HEIGHT 2048 #define MAKE_ATOM(a) MakeAtom(a, sizeof(a) - 1, true) static Atom xvColorKey, xvAlwaysOnTop; static XvFormatRec formats[] = { {15}, {16}, {24} }; static const XvImageRec images[] = { XVIMAGE_YUY2, XVIMAGE_UYVY, XVMC_RGB888, XVMC_RGB565 }; static const XvAttributeRec attribs[] = { { XvSettable | XvGettable, 0, 0xffffff, (char *)"XV_COLORKEY" }, { XvSettable | XvGettable, 0, 1, (char *)"XV_ALWAYS_ON_TOP" }, }; static int sna_video_sprite_stop(ClientPtr client, XvPortPtr port, DrawablePtr draw) { struct sna_video *video = port->devPriv.ptr; struct drm_mode_set_plane s; if (video->plane == 0) return Success; memset(&s, 0, sizeof(s)); s.plane_id = video->plane; if (drmIoctl(video->sna->kgem.fd, DRM_IOCTL_MODE_SETPLANE, &s)) xf86DrvMsg(video->sna->scrn->scrnIndex, X_ERROR, "failed to disable plane\n"); if (video->bo) kgem_bo_destroy(&video->sna->kgem, video->bo); video->bo = NULL; video->plane = 0; sna_window_set_port((WindowPtr)draw, NULL); return Success; } static int sna_video_sprite_set_attr(ClientPtr client, XvPortPtr port, Atom attribute, INT32 value) { struct sna_video *video = port->devPriv.ptr; if (attribute == xvColorKey) { video->color_key_changed = true; video->color_key = value; DBG(("COLORKEY = %ld\n", (long)value)); } else if (attribute == xvAlwaysOnTop) { DBG(("%s: ALWAYS_ON_TOP: %d -> %d\n", __FUNCTION__, video->AlwaysOnTop, !!value)); video->color_key_changed = true; video->AlwaysOnTop = !!value; } else return BadMatch; return Success; } static int sna_video_sprite_get_attr(ClientPtr client, XvPortPtr port, Atom attribute, INT32 *value) { struct sna_video *video = port->devPriv.ptr; if (attribute == xvColorKey) *value = video->color_key; else if (attribute == xvAlwaysOnTop) *value = video->AlwaysOnTop; else return BadMatch; return Success; } static int sna_video_sprite_best_size(ClientPtr client, XvPortPtr port, CARD8 motion, CARD16 vid_w, CARD16 vid_h, CARD16 drw_w, CARD16 drw_h, unsigned int *p_w, unsigned int *p_h) { struct sna_video *video = port->devPriv.ptr; struct sna *sna = video->sna; if (sna->kgem.gen >= 075) { *p_w = vid_w; *p_h = vid_h; } else { *p_w = drw_w; *p_h = drw_h; } return Success; } static void update_dst_box_to_crtc_coords(struct sna *sna, xf86CrtcPtr crtc, BoxPtr dstBox) { ScrnInfoPtr scrn = sna->scrn; int tmp; switch (crtc->rotation & 0xf) { case RR_Rotate_0: dstBox->x1 -= crtc->x; dstBox->x2 -= crtc->x; dstBox->y1 -= crtc->y; dstBox->y2 -= crtc->y; break; case RR_Rotate_90: tmp = dstBox->x1; dstBox->x1 = dstBox->y1 - crtc->x; dstBox->y1 = scrn->virtualX - tmp - crtc->y; tmp = dstBox->x2; dstBox->x2 = dstBox->y2 - crtc->x; dstBox->y2 = scrn->virtualX - tmp - crtc->y; tmp = dstBox->y1; dstBox->y1 = dstBox->y2; dstBox->y2 = tmp; break; case RR_Rotate_180: tmp = dstBox->x1; dstBox->x1 = scrn->virtualX - dstBox->x2 - crtc->x; dstBox->x2 = scrn->virtualX - tmp - crtc->x; tmp = dstBox->y1; dstBox->y1 = scrn->virtualY - dstBox->y2 - crtc->y; dstBox->y2 = scrn->virtualY - tmp - crtc->y; break; case RR_Rotate_270: tmp = dstBox->x1; dstBox->x1 = scrn->virtualY - dstBox->y1 - crtc->x; dstBox->y1 = tmp - crtc->y; tmp = dstBox->x2; dstBox->x2 = scrn->virtualY - dstBox->y2 - crtc->x; dstBox->y2 = tmp - crtc->y; tmp = dstBox->x1; dstBox->x1 = dstBox->x2; dstBox->x2 = tmp; break; } } static bool sna_video_sprite_show(struct sna *sna, struct sna_video *video, struct sna_video_frame *frame, xf86CrtcPtr crtc, BoxPtr dstBox) { struct drm_mode_set_plane s; /* XXX handle video spanning multiple CRTC */ VG_CLEAR(s); s.plane_id = sna_crtc_to_plane(crtc); update_dst_box_to_crtc_coords(sna, crtc, dstBox); if (crtc->rotation & (RR_Rotate_90 | RR_Rotate_270)) { int tmp = frame->width; frame->width = frame->height; frame->height = tmp; } #if defined(DRM_I915_SET_SPRITE_COLORKEY) if (video->color_key_changed || video->plane != s.plane_id) { struct drm_intel_sprite_colorkey set; DBG(("%s: updating color key: %x\n", __FUNCTION__, video->color_key)); set.plane_id = s.plane_id; set.min_value = video->color_key; set.max_value = video->color_key; /* not used for destkey */ set.channel_mask = 0x7 << 24 | 0xff << 16 | 0xff << 8 | 0xff << 0; set.flags = 0; if (!video->AlwaysOnTop) set.flags = I915_SET_COLORKEY_DESTINATION; if (drmIoctl(sna->kgem.fd, DRM_IOCTL_I915_SET_SPRITE_COLORKEY, &set)) xf86DrvMsg(sna->scrn->scrnIndex, X_ERROR, "failed to update color key\n"); video->color_key_changed = false; } #endif if (frame->bo->delta == 0) { uint32_t offsets[4], pitches[4], handles[4]; uint32_t pixel_format; bool purged = true; handles[0] = frame->bo->handle; pitches[0] = frame->pitch[0]; offsets[0] = 0; switch (frame->id) { case FOURCC_RGB565: pixel_format = DRM_FORMAT_RGB565; purged = sna->scrn->depth != 16; break; case FOURCC_RGB888: pixel_format = DRM_FORMAT_XRGB8888; purged = sna->scrn->depth != 24; break; case FOURCC_UYVY: pixel_format = DRM_FORMAT_UYVY; break; case FOURCC_YUY2: default: pixel_format = DRM_FORMAT_YUYV; break; } DBG(("%s: creating new fb for handle=%d, width=%d, height=%d, stride=%d\n", __FUNCTION__, frame->bo->handle, frame->width, frame->height, pitches[0])); if (drmModeAddFB2(sna->kgem.fd, frame->width, frame->height, pixel_format, handles, pitches, offsets, &frame->bo->delta, 0)) { xf86DrvMsg(sna->scrn->scrnIndex, X_ERROR, "failed to add fb\n"); return false; } frame->bo->scanout = true; /* Don't allow the scanout to be cached if not suitable for front */ frame->bo->purged = purged; } assert(frame->bo->scanout); assert(frame->bo->delta); s.crtc_id = sna_crtc_id(crtc); s.fb_id = frame->bo->delta; s.flags = 0; s.crtc_x = dstBox->x1; s.crtc_y = dstBox->y1; s.crtc_w = dstBox->x2 - dstBox->x1; s.crtc_h = dstBox->y2 - dstBox->y1; s.src_x = 0; s.src_y = 0; s.src_w = (frame->image.x2 - frame->image.x1) << 16; s.src_h = (frame->image.y2 - frame->image.y1) << 16; DBG(("%s: updating crtc=%d, plane=%d, handle=%d [fb %d], dst=(%d,%d)x(%d,%d), src=(%d,%d)x(%d,%d)\n", __FUNCTION__, s.crtc_id, s.plane_id, frame->bo->handle, s.fb_id, s.crtc_x, s.crtc_y, s.crtc_w, s.crtc_h, s.src_x >> 16, s.src_y >> 16, s.src_w >> 16, s.src_h >> 16)); if (drmIoctl(sna->kgem.fd, DRM_IOCTL_MODE_SETPLANE, &s)) { DBG(("SET_PLANE failed: ret=%d\n", errno)); return false; } frame->bo->domain = DOMAIN_NONE; if (video->plane != s.plane_id) { if (video->plane) { memset(&s, 0, sizeof(s)); s.plane_id = video->plane; drmIoctl(video->sna->kgem.fd, DRM_IOCTL_MODE_SETPLANE, &s); } video->plane = s.plane_id; } if (video->bo != frame->bo) { if (video->bo) kgem_bo_destroy(&sna->kgem, video->bo); video->bo = kgem_bo_reference(frame->bo); } return true; } static int sna_video_sprite_put_image(ClientPtr client, DrawablePtr draw, XvPortPtr port, GCPtr gc, INT16 src_x, INT16 src_y, CARD16 src_w, CARD16 src_h, INT16 drw_x, INT16 drw_y, CARD16 drw_w, CARD16 drw_h, XvImagePtr format, unsigned char *buf, Bool sync, CARD16 width, CARD16 height) { struct sna_video *video = port->devPriv.ptr; struct sna *sna = video->sna; struct sna_video_frame frame; xf86CrtcPtr crtc; BoxRec dst_box; RegionRec clip; int ret; clip.extents.x1 = draw->x + drw_x; clip.extents.y1 = draw->y + drw_y; clip.extents.x2 = clip.extents.x1 + drw_w; clip.extents.y2 = clip.extents.y1 + drw_h; clip.data = NULL; DBG(("%s: always_on_top=%d\n", __FUNCTION__, video->AlwaysOnTop)); if (!video->AlwaysOnTop) RegionIntersect(&clip, &clip, gc->pCompositeClip); if (box_empty(&clip.extents)) goto invisible; DBG(("%s: src=(%d, %d),(%d, %d), dst=(%d, %d),(%d, %d), id=%d, sizep=%dx%d, sync?=%d\n", __FUNCTION__, src_x, src_y, src_w, src_h, drw_x, drw_y, drw_w, drw_h, format->id, width, height, sync)); DBG(("%s: region %d:(%d, %d), (%d, %d)\n", __FUNCTION__, RegionNumRects(&clip), clip.extents.x1, clip.extents.y1, clip.extents.x2, clip.extents.y2)); sna_video_frame_init(video, format->id, width, height, &frame); if (!sna_video_clip_helper(video, &frame, &crtc, &dst_box, src_x, src_y, draw->x + drw_x, draw->y + drw_y, src_w, src_h, drw_w, drw_h, &clip)) goto invisible; if (!crtc || sna_crtc_to_plane(crtc) == 0) goto invisible; /* sprites can't handle rotation natively, store it for the copy func */ video->rotation = crtc->rotation; if (xvmc_passthrough(format->id)) { DBG(("%s: using passthough, name=%d\n", __FUNCTION__, *(uint32_t *)buf)); if (*(uint32_t*)buf == 0) goto invisible; frame.bo = kgem_create_for_name(&sna->kgem, *(uint32_t*)buf); if (frame.bo == NULL) return BadAlloc; if (kgem_bo_size(frame.bo) < frame.size) { DBG(("%s: bo size=%d, expected=%d\n", __FUNCTION__, kgem_bo_size(frame.bo), frame.size)); kgem_bo_destroy(&sna->kgem, frame.bo); return BadAlloc; } frame.image.x1 = 0; frame.image.y1 = 0; frame.image.x2 = frame.width; frame.image.y2 = frame.height; } else { frame.bo = sna_video_buffer(video, &frame); if (frame.bo == NULL) { DBG(("%s: failed to allocate video bo\n", __FUNCTION__)); return BadAlloc; } if (!sna_video_copy_data(video, &frame, buf)) { DBG(("%s: failed to copy video data\n", __FUNCTION__)); return BadAlloc; } } ret = Success; if (!sna_video_sprite_show(sna, video, &frame, crtc, &dst_box)) { DBG(("%s: failed to show video frame\n", __FUNCTION__)); ret = BadAlloc; } else { //xf86XVFillKeyHelperDrawable(draw, video->color_key, &clip); if (!video->AlwaysOnTop && !RegionEqual(&video->clip, &clip) && sna_blt_fill_boxes(sna, GXcopy, __sna_pixmap_get_bo(sna->front), sna->front->drawable.bitsPerPixel, video->color_key, RegionRects(&clip), RegionNumRects(&clip))) RegionCopy(&video->clip, &clip); sna_window_set_port((WindowPtr)draw, port); } frame.bo->domain = DOMAIN_NONE; if (xvmc_passthrough(format->id)) kgem_bo_destroy(&sna->kgem, frame.bo); else sna_video_buffer_fini(video); return ret; invisible: /* If the video isn't visible on any CRTC, turn it off */ return sna_video_sprite_stop(client, port, draw); } static int sna_video_sprite_query(ClientPtr client, XvPortPtr port, XvImagePtr format, unsigned short *w, unsigned short *h, int *pitches, int *offsets) { struct sna_video *video = port->devPriv.ptr; struct sna_video_frame frame; int size; if (*w > IMAGE_MAX_WIDTH) *w = IMAGE_MAX_WIDTH; if (*h > IMAGE_MAX_HEIGHT) *h = IMAGE_MAX_HEIGHT; if (offsets) offsets[0] = 0; switch (format->id) { case FOURCC_RGB888: case FOURCC_RGB565: sna_video_frame_init(video, format->id, *w, *h, &frame); if (pitches) pitches[0] = frame.pitch[0]; size = 4; break; default: *w = (*w + 1) & ~1; *h = (*h + 1) & ~1; size = *w << 1; if (pitches) pitches[0] = size; size *= *h; break; } return size; } static int sna_video_sprite_color_key(struct sna *sna) { ScrnInfoPtr scrn = sna->scrn; int color_key; if (xf86GetOptValInteger(sna->Options, OPTION_VIDEO_KEY, &color_key)) { } else if (xf86GetOptValInteger(sna->Options, OPTION_COLOR_KEY, &color_key)) { } else { color_key = (1 << scrn->offset.red) | (1 << scrn->offset.green) | (((scrn->mask.blue >> scrn->offset.blue) - 1) << scrn->offset.blue); } return color_key & ((1 << scrn->depth) - 1); } void sna_video_sprite_setup(struct sna *sna, ScreenPtr screen) { XvAdaptorPtr adaptor; struct drm_mode_get_plane_res r; struct sna_video *video; XvPortPtr port; if (sna->flags & SNA_IS_HOSTED) return; memset(&r, 0, sizeof(struct drm_mode_get_plane_res)); if (drmIoctl(sna->kgem.fd, DRM_IOCTL_MODE_GETPLANERESOURCES, &r)) return; DBG(("%s: %d sprite planes\n", __FUNCTION__, r.count_planes)); if (r.count_planes == 0) return; adaptor = sna_xv_adaptor_alloc(sna); if (!adaptor) return; video = calloc(1, sizeof(*video)); port = calloc(1, sizeof(*port)); if (video == NULL || port == NULL) { free(video); free(port); sna->xv.num_adaptors--; return; } adaptor->type = XvInputMask | XvImageMask; adaptor->pScreen = screen; adaptor->name = (char *)"Intel(R) Video Sprite"; adaptor->nEncodings = 1; adaptor->pEncodings = xnfalloc(sizeof(XvEncodingRec)); adaptor->pEncodings[0].id = 0; adaptor->pEncodings[0].pScreen = screen; adaptor->pEncodings[0].name = (char *)"XV_IMAGE"; adaptor->pEncodings[0].width = IMAGE_MAX_WIDTH; adaptor->pEncodings[0].height = IMAGE_MAX_HEIGHT; adaptor->pEncodings[0].rate.numerator = 1; adaptor->pEncodings[0].rate.denominator = 1; adaptor->pFormats = formats; adaptor->nFormats = sna_xv_fixup_formats(screen, formats, ARRAY_SIZE(formats)); adaptor->nAttributes = ARRAY_SIZE(attribs); adaptor->pAttributes = (XvAttributeRec *)attribs; adaptor->pImages = (XvImageRec *)images; adaptor->nImages = 3; if (sna->kgem.gen == 071) adaptor->nImages = 4; adaptor->ddAllocatePort = sna_xv_alloc_port; adaptor->ddFreePort = sna_xv_free_port; adaptor->ddPutVideo = NULL; adaptor->ddPutStill = NULL; adaptor->ddGetVideo = NULL; adaptor->ddGetStill = NULL; adaptor->ddStopVideo = sna_video_sprite_stop; adaptor->ddSetPortAttribute = sna_video_sprite_set_attr; adaptor->ddGetPortAttribute = sna_video_sprite_get_attr; adaptor->ddQueryBestSize = sna_video_sprite_best_size; adaptor->ddPutImage = sna_video_sprite_put_image; adaptor->ddQueryImageAttributes = sna_video_sprite_query; adaptor->nPorts = 1; adaptor->pPorts = port; adaptor->base_id = port->id = FakeClientID(0); AddResource(port->id, XvGetRTPort(), port); port->pAdaptor = adaptor; port->pNotify = NULL; port->pDraw = NULL; port->client = NULL; port->grab.client = NULL; port->time = currentTime; port->devPriv.ptr = video; video->sna = sna; video->alignment = 64; video->color_key = sna_video_sprite_color_key(sna); video->color_key_changed = true; video->brightness = -19; /* (255/219) * -16 */ video->contrast = 75; /* 255/219 * 64 */ video->saturation = 146; /* 128/112 * 128 */ video->desired_crtc = NULL; video->gamma5 = 0xc0c0c0; video->gamma4 = 0x808080; video->gamma3 = 0x404040; video->gamma2 = 0x202020; video->gamma1 = 0x101010; video->gamma0 = 0x080808; video->rotation = RR_Rotate_0; RegionNull(&video->clip); xvColorKey = MAKE_ATOM("XV_COLORKEY"); xvAlwaysOnTop = MAKE_ATOM("XV_ALWAYS_ON_TOP"); } #else void sna_video_sprite_setup(struct sna *sna, ScreenPtr screen) { } #endif