| OLD | NEW |
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "media/tools/player_x11/x11_video_renderer.h" | 5 #include "media/tools/player_x11/x11_video_renderer.h" |
| 6 | 6 |
| 7 #include <dlfcn.h> | 7 #include <dlfcn.h> |
| 8 #include <X11/Xutil.h> | 8 #include <X11/Xutil.h> |
| 9 #include <X11/extensions/Xrender.h> | 9 #include <X11/extensions/Xrender.h> |
| 10 #include <X11/extensions/Xcomposite.h> | 10 #include <X11/extensions/Xcomposite.h> |
| (...skipping 66 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 77 } | 77 } |
| 78 | 78 |
| 79 X11VideoRenderer::~X11VideoRenderer() { | 79 X11VideoRenderer::~X11VideoRenderer() { |
| 80 if (image_) | 80 if (image_) |
| 81 XDestroyImage(image_); | 81 XDestroyImage(image_); |
| 82 if (use_render_) | 82 if (use_render_) |
| 83 XRenderFreePicture(display_, picture_); | 83 XRenderFreePicture(display_, picture_); |
| 84 } | 84 } |
| 85 | 85 |
| 86 void X11VideoRenderer::Paint(media::VideoFrame* video_frame) { | 86 void X11VideoRenderer::Paint(media::VideoFrame* video_frame) { |
| 87 int width = video_frame->data_size().width(); | 87 if (!image_) |
| 88 int height = video_frame->data_size().height(); | 88 Initialize(video_frame->coded_size(), video_frame->visible_rect()); |
| 89 | 89 |
| 90 if (!image_) | 90 const int coded_width = video_frame->coded_size().width(); |
| 91 Initialize(width, height); | 91 const int coded_height = video_frame->coded_size().height(); |
| 92 const int visible_width = video_frame->visible_rect().width(); |
| 93 const int visible_height = video_frame->visible_rect().height(); |
| 92 | 94 |
| 93 // Check if we need to reallocate our XImage. | 95 // Check if we need to reallocate our XImage. |
| 94 if (image_->width != width || image_->height != height) { | 96 if (image_->width != coded_width || image_->height != coded_height) { |
| 95 XDestroyImage(image_); | 97 XDestroyImage(image_); |
| 96 image_ = CreateImage(display_, width, height); | 98 image_ = CreateImage(display_, coded_width, coded_height); |
| 97 } | 99 } |
| 98 | 100 |
| 99 // Convert YUV frame to RGB. | 101 // Convert YUV frame to RGB. |
| 100 DCHECK(video_frame->format() == media::VideoFrame::YV12 || | 102 DCHECK(video_frame->format() == media::VideoFrame::YV12 || |
| 101 video_frame->format() == media::VideoFrame::YV16); | 103 video_frame->format() == media::VideoFrame::YV16); |
| 102 DCHECK(video_frame->stride(media::VideoFrame::kUPlane) == | 104 DCHECK(video_frame->stride(media::VideoFrame::kUPlane) == |
| 103 video_frame->stride(media::VideoFrame::kVPlane)); | 105 video_frame->stride(media::VideoFrame::kVPlane)); |
| 104 | 106 |
| 105 DCHECK(image_->data); | 107 DCHECK(image_->data); |
| 106 media::YUVType yuv_type = | 108 media::YUVType yuv_type = |
| 107 (video_frame->format() == media::VideoFrame::YV12) ? | 109 (video_frame->format() == media::VideoFrame::YV12) ? |
| 108 media::YV12 : media::YV16; | 110 media::YV12 : media::YV16; |
| 109 media::ConvertYUVToRGB32(video_frame->data(media::VideoFrame::kYPlane), | 111 media::ConvertYUVToRGB32(video_frame->data(media::VideoFrame::kYPlane), |
| 110 video_frame->data(media::VideoFrame::kUPlane), | 112 video_frame->data(media::VideoFrame::kUPlane), |
| 111 video_frame->data(media::VideoFrame::kVPlane), | 113 video_frame->data(media::VideoFrame::kVPlane), |
| 112 (uint8*)image_->data, | 114 (uint8*)image_->data, coded_width, coded_height, |
| 113 video_frame->data_size().width(), | |
| 114 video_frame->data_size().height(), | |
| 115 video_frame->stride(media::VideoFrame::kYPlane), | 115 video_frame->stride(media::VideoFrame::kYPlane), |
| 116 video_frame->stride(media::VideoFrame::kUPlane), | 116 video_frame->stride(media::VideoFrame::kUPlane), |
| 117 image_->bytes_per_line, | 117 image_->bytes_per_line, |
| 118 yuv_type); | 118 yuv_type); |
| 119 | 119 |
| 120 if (use_render_) { | 120 if (use_render_) { |
| 121 // If XRender is used, we'll upload the image to a pixmap. And then | 121 // If XRender is used, we'll upload the image to a pixmap. And then |
| 122 // creats a picture from the pixmap and composite the picture over | 122 // creats a picture from the pixmap and composite the picture over |
| 123 // the picture represending the window. | 123 // the picture represending the window. |
| 124 | 124 |
| 125 // Creates a XImage. | 125 // Creates a XImage. |
| 126 XImage image; | 126 XImage image; |
| 127 memset(&image, 0, sizeof(image)); | 127 memset(&image, 0, sizeof(image)); |
| 128 image.width = width; | 128 image.width = coded_width; |
| 129 image.height = height; | 129 image.height = coded_height; |
| 130 image.depth = 32; | 130 image.depth = 32; |
| 131 image.bits_per_pixel = 32; | 131 image.bits_per_pixel = 32; |
| 132 image.format = ZPixmap; | 132 image.format = ZPixmap; |
| 133 image.byte_order = LSBFirst; | 133 image.byte_order = LSBFirst; |
| 134 image.bitmap_unit = 8; | 134 image.bitmap_unit = 8; |
| 135 image.bitmap_bit_order = LSBFirst; | 135 image.bitmap_bit_order = LSBFirst; |
| 136 image.bytes_per_line = image_->bytes_per_line; | 136 image.bytes_per_line = image_->bytes_per_line; |
| 137 image.red_mask = 0xff; | 137 image.red_mask = 0xff; |
| 138 image.green_mask = 0xff00; | 138 image.green_mask = 0xff00; |
| 139 image.blue_mask = 0xff0000; | 139 image.blue_mask = 0xff0000; |
| 140 image.data = image_->data; | 140 image.data = image_->data; |
| 141 | 141 |
| 142 // Creates a pixmap and uploads from the XImage. | 142 // Creates a pixmap and uploads from the XImage. |
| 143 unsigned long pixmap = XCreatePixmap(display_, | 143 unsigned long pixmap = XCreatePixmap(display_, window_, |
| 144 window_, | 144 visible_width, visible_height, |
| 145 width, | |
| 146 height, | |
| 147 32); | 145 32); |
| 148 GC gc = XCreateGC(display_, pixmap, 0, NULL); | 146 GC gc = XCreateGC(display_, pixmap, 0, NULL); |
| 149 XPutImage(display_, pixmap, gc, &image, | 147 XPutImage(display_, pixmap, gc, &image, |
| 150 0, 0, 0, 0, | 148 video_frame->visible_rect().x(), |
| 151 width, height); | 149 video_frame->visible_rect().y(), |
| 150 0, 0, |
| 151 visible_width, visible_height); |
| 152 XFreeGC(display_, gc); | 152 XFreeGC(display_, gc); |
| 153 | 153 |
| 154 // Creates the picture representing the pixmap. | 154 // Creates the picture representing the pixmap. |
| 155 unsigned long picture = XRenderCreatePicture( | 155 unsigned long picture = XRenderCreatePicture( |
| 156 display_, pixmap, GetRenderARGB32Format(display_), 0, NULL); | 156 display_, pixmap, GetRenderARGB32Format(display_), 0, NULL); |
| 157 | 157 |
| 158 // Composite the picture over the picture representing the window. | 158 // Composite the picture over the picture representing the window. |
| 159 XRenderComposite(display_, PictOpSrc, picture, 0, | 159 XRenderComposite(display_, PictOpSrc, picture, 0, |
| 160 picture_, 0, 0, 0, 0, 0, 0, | 160 picture_, 0, 0, 0, 0, 0, 0, |
| 161 width, height); | 161 visible_width, visible_height); |
| 162 | 162 |
| 163 XRenderFreePicture(display_, picture); | 163 XRenderFreePicture(display_, picture); |
| 164 XFreePixmap(display_, pixmap); | 164 XFreePixmap(display_, pixmap); |
| 165 return; | 165 return; |
| 166 } | 166 } |
| 167 | 167 |
| 168 // If XRender is not used, simply put the image to the server. | 168 // If XRender is not used, simply put the image to the server. |
| 169 // This will have a tearing effect but this is OK. | 169 // This will have a tearing effect but this is OK. |
| 170 // TODO(hclam): Upload the image to a pixmap and do XCopyArea() | 170 // TODO(hclam): Upload the image to a pixmap and do XCopyArea() |
| 171 // to the window. | 171 // to the window. |
| 172 GC gc = XCreateGC(display_, window_, 0, NULL); | 172 GC gc = XCreateGC(display_, window_, 0, NULL); |
| 173 XPutImage(display_, window_, gc, image_, | 173 XPutImage(display_, window_, gc, image_, |
| 174 0, 0, 0, 0, width, height); | 174 video_frame->visible_rect().x(), |
| 175 video_frame->visible_rect().y(), |
| 176 0, 0, visible_width, visible_height); |
| 175 XFlush(display_); | 177 XFlush(display_); |
| 176 XFreeGC(display_, gc); | 178 XFreeGC(display_, gc); |
| 177 } | 179 } |
| 178 | 180 |
| 179 void X11VideoRenderer::Initialize(int width, int height) { | 181 void X11VideoRenderer::Initialize(gfx::Size coded_size, |
| 182 gfx::Rect visible_rect) { |
| 180 CHECK(!image_); | 183 CHECK(!image_); |
| 181 LOG(INFO) << "Initializing X11 Renderer..."; | 184 LOG(INFO) << "Initializing X11 Renderer..."; |
| 182 | 185 |
| 183 // Resize the window to fit that of the video. | 186 // Resize the window to fit that of the video. |
| 184 XResizeWindow(display_, window_, width, height); | 187 XResizeWindow(display_, window_, visible_rect.width(), visible_rect.height()); |
| 185 image_ = CreateImage(display_, width, height); | 188 image_ = CreateImage(display_, coded_size.width(), coded_size.height()); |
| 186 | 189 |
| 187 // Testing XRender support. We'll use the very basic of XRender | 190 // Testing XRender support. We'll use the very basic of XRender |
| 188 // so if it presents it is already good enough. We don't need | 191 // so if it presents it is already good enough. We don't need |
| 189 // to check its version. | 192 // to check its version. |
| 190 int dummy; | 193 int dummy; |
| 191 use_render_ = XRenderQueryExtension(display_, &dummy, &dummy); | 194 use_render_ = XRenderQueryExtension(display_, &dummy, &dummy); |
| 192 | 195 |
| 193 if (use_render_) { | 196 if (use_render_) { |
| 194 LOG(INFO) << "Using XRender extension."; | 197 LOG(INFO) << "Using XRender extension."; |
| 195 | 198 |
| 196 // If we are using XRender, we'll create a picture representing the | 199 // If we are using XRender, we'll create a picture representing the |
| 197 // window. | 200 // window. |
| 198 XWindowAttributes attr; | 201 XWindowAttributes attr; |
| 199 XGetWindowAttributes(display_, window_, &attr); | 202 XGetWindowAttributes(display_, window_, &attr); |
| 200 | 203 |
| 201 XRenderPictFormat* pictformat = XRenderFindVisualFormat( | 204 XRenderPictFormat* pictformat = XRenderFindVisualFormat( |
| 202 display_, | 205 display_, |
| 203 attr.visual); | 206 attr.visual); |
| 204 CHECK(pictformat) << "XRender does not support default visual"; | 207 CHECK(pictformat) << "XRender does not support default visual"; |
| 205 | 208 |
| 206 picture_ = XRenderCreatePicture(display_, window_, pictformat, 0, NULL); | 209 picture_ = XRenderCreatePicture(display_, window_, pictformat, 0, NULL); |
| 207 CHECK(picture_) << "Backing picture not created"; | 210 CHECK(picture_) << "Backing picture not created"; |
| 208 } | 211 } |
| 209 } | 212 } |
| OLD | NEW |