Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1848)

Unified Diff: remoting/client/x11_view.cc

Issue 2745006: Implement a chromoting client using X11 (Closed)
Patch Set: removed all.gyp Created 10 years, 6 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « remoting/client/x11_view.h ('k') | remoting/host/capturer.h » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: remoting/client/x11_view.cc
diff --git a/remoting/client/x11_view.cc b/remoting/client/x11_view.cc
new file mode 100644
index 0000000000000000000000000000000000000000..bdb54d42fec8988b4f260fb0eabda40f006aed7a
--- /dev/null
+++ b/remoting/client/x11_view.cc
@@ -0,0 +1,151 @@
+// Copyright (c) 2010 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "remoting/client/x11_view.h"
+
+#include <X11/Xlib.h>
+#include <X11/Xutil.h>
+#include <X11/extensions/Xrender.h>
+#include <X11/extensions/Xcomposite.h>
+#include "remoting/client/decoder_verbatim.h"
+
+namespace remoting {
+
+X11View::X11View(Display* display, int window, int width, int height)
+ : display_(display),
+ window_(window),
+ picture_(0),
+ width_(width),
+ height_(height) {
+}
+
+X11View::~X11View() {
+}
+
+void X11View::Paint() {
+ // TODO(hclam): Paint only the updated regions.
+ all_update_rects_.clear();
+
+ // If we have not initialized the render target then do it now.
+ if (!frame_)
+ InitPaintTarget();
+
+ // Upload the image to a pixmap. And then creats a picture from the pixmap
+ // and composite the picture over the picture represending the window.
+
+ // Creates a XImage.
+ XImage image;
+ memset(&image, 0, sizeof(image));
+ image.width = width_;
+ image.height = height_;
+ image.depth = 32;
+ image.bits_per_pixel = 32;
+ image.format = ZPixmap;
+ image.byte_order = LSBFirst;
+ image.bitmap_unit = 8;
+ image.bitmap_bit_order = LSBFirst;
+ image.bytes_per_line = frame_->stride(media::VideoFrame::kRGBPlane);
+ image.red_mask = 0xff;
+ image.green_mask = 0xff00;
+ image.blue_mask = 0xff0000;
+ image.data = reinterpret_cast<char*>(
+ frame_->data(media::VideoFrame::kRGBPlane));
+
+ // Creates a pixmap and uploads from the XImage.
+ unsigned long pixmap = XCreatePixmap(display_, window_, width_, height_, 32);
+
+ GC gc = XCreateGC(display_, pixmap, 0, NULL);
+ XPutImage(display_, pixmap, gc, &image, 0, 0, 0, 0, width_, height_);
+ XFreeGC(display_, gc);
+
+ // Creates the picture representing the pixmap.
+ unsigned long picture = XRenderCreatePicture(
+ display_, pixmap,
+ XRenderFindStandardFormat(display_, PictStandardARGB32),
+ 0, NULL);
+
+ // Composite the picture over the picture representing the window.
+ XRenderComposite(display_, PictOpSrc, picture, 0,
+ picture_, 0, 0, 0, 0, 0, 0,
+ width_, height_);
+
+ XRenderFreePicture(display_, picture);
+ XFreePixmap(display_, pixmap);
+}
+
+void X11View::InitPaintTarget() {
+ // Testing XRender support.
+ int dummy;
+ bool xrender_support = XRenderQueryExtension(display_, &dummy, &dummy);
+ CHECK(xrender_support) << "XRender is not supported!";
+
+ XWindowAttributes attr;
+ XGetWindowAttributes(display_, window_, &attr);
+
+ XRenderPictFormat* pictformat = XRenderFindVisualFormat(
+ display_,
+ attr.visual);
+ CHECK(pictformat) << "XRENDER does not support default visual";
+
+ picture_ = XRenderCreatePicture(display_, window_, pictformat, 0, NULL);
+ CHECK(picture_) << "Backing picture not created";
+
+ // Create the video frame to carry the decoded image.
+ media::VideoFrame::CreateFrame(media::VideoFrame::RGB32, width_, height_,
+ base::TimeDelta(), base::TimeDelta(), &frame_);
+ DCHECK(frame_);
+}
+
+void X11View::HandleBeginUpdateStream(HostMessage* msg) {
+ scoped_ptr<HostMessage> deleter(msg);
+
+ // TODO(hclam): Use the information from the message to create the decoder.
+ // We lazily construct the decoder.
+ if (!decoder_.get()) {
+ decoder_.reset(new DecoderVerbatim());
+ }
+
+ // Tell the decoder to do start decoding.
+ decoder_->BeginDecode(frame_, &update_rects_,
+ NewRunnableMethod(this, &X11View::OnPartialDecodeDone),
+ NewRunnableMethod(this, &X11View::OnDecodeDone));
+}
+
+void X11View::HandleUpdateStreamPacket(HostMessage* msg) {
+ decoder_->PartialDecode(msg);
+}
+
+void X11View::HandleEndUpdateStream(HostMessage* msg) {
+ scoped_ptr<HostMessage> deleter(msg);
+ decoder_->EndDecode();
+}
+
+void X11View::OnPartialDecodeDone() {
+ // Decoder has produced some output so schedule a paint. We'll get a Paint()
+ // call in the short future. Note that we can get UpdateStreamPacket during
+ // this short period of time and we will perform decode again and the
+ // information of updated rects will be lost.
+ // There are several ways to solve this problem.
+ // 1. Merge the updated rects and perform one paint.
+ // 2. Queue the updated rects and perform two paints.
+ // 3. Ignore the updated rects and always paint the full image. Since we
+ // use one frame as output this will always be correct.
+ // We will take (1) and simply concat the list of rectangles.
+ all_update_rects_.insert(all_update_rects_.begin() +
+ all_update_rects_.size(),
+ update_rects_.begin(), update_rects_.end());
+
+ // TODO(hclam): Make sure we call this method on the right thread. Since
+ // decoder is single-threaded we don't have a problem but we better post
+ // a task to do the right thing.
+ XEvent event;
+ event.type = Expose;
+ XSendEvent(display_, static_cast<int>(window_), true, ExposureMask, &event);
+}
+
+void X11View::OnDecodeDone() {
+ // Since we do synchronous decoding here there's nothing in this method.
+}
+
+} // namespace remoting
« no previous file with comments | « remoting/client/x11_view.h ('k') | remoting/host/capturer.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698