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

Unified Diff: remoting/client/x11_client.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/simple_client.cc ('k') | remoting/client/x11_view.h » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: remoting/client/x11_client.cc
diff --git a/remoting/client/x11_client.cc b/remoting/client/x11_client.cc
new file mode 100644
index 0000000000000000000000000000000000000000..6b8e4e2d3ee3a6ba54fe4f3d0cd389d33d53c981
--- /dev/null
+++ b/remoting/client/x11_client.cc
@@ -0,0 +1,262 @@
+// 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.
+//
+// This file implements a X11 chromoting client.
+
+#include <iostream>
+
+#include "base/at_exit.h"
+#include "base/message_loop.h"
+#include "base/stl_util-inl.h"
+#include "remoting/client/host_connection.h"
+#include "remoting/client/client_util.h"
+
+// Include Xlib at the end because it clashes with ClientMessage defined in
+// the protocol buffer.
+// x11_view.h also includes Xlib.h so put it behind all other headers but
+// before Xlib.h
+#include "remoting/client/x11_view.h"
+#include <X11/Xlib.h>
+
+namespace remoting {
+
+class X11Client : public base::RefCountedThreadSafe<X11Client>,
+ public HostConnection::EventHandler {
+ public:
+ X11Client(std::string host_jid, std::string username, std::string auth_token)
+ : display_(NULL),
+ window_(0),
+ width_(0),
+ height_(0),
+ host_jid_(host_jid),
+ username_(username),
+ auth_token_(auth_token) {
+ }
+
+ virtual ~X11Client() {
+ DCHECK(!display_);
+ DCHECK(!window_);
+ }
+
+ // Starts the remoting client and the message loop. Returns only after
+ // the message loop has terminated.
+ void Run() {
+ message_loop_.PostTask(FROM_HERE,
+ NewRunnableMethod(this, &X11Client::DoInitX11));
+ message_loop_.PostTask(FROM_HERE,
+ NewRunnableMethod(this, &X11Client::DoInitConnection));
+ message_loop_.Run();
+ }
+
+ ////////////////////////////////////////////////////////////////////////////
+ // HostConnection::EventHandler implementations.
+ virtual void HandleMessages(HostConnection* conn,
+ remoting::HostMessageList* messages) {
+ for (size_t i = 0; i < messages->size(); ++i) {
+ HostMessage* msg = (*messages)[i];
+ if (msg->has_init_client()) {
+ message_loop_.PostTask(
+ FROM_HERE, NewRunnableMethod(this, &X11Client::DoInitClient, msg));
+ } else if (msg->has_begin_update_stream()) {
+ message_loop_.PostTask(
+ FROM_HERE,
+ NewRunnableMethod(this, &X11Client::DoBeginUpdate, msg));
+ } else if (msg->has_update_stream_packet()) {
+ message_loop_.PostTask(
+ FROM_HERE,
+ NewRunnableMethod(this, &X11Client::DoHandleUpdate, msg));
+ } else if (msg->has_end_update_stream()) {
+ message_loop_.PostTask(
+ FROM_HERE, NewRunnableMethod(this, &X11Client::DoEndUpdate, msg));
+ } else {
+ NOTREACHED() << "Unknown message received";
+ }
+ }
+ // Assume we have processed all the messages.
+ messages->clear();
+ }
+
+ virtual void OnConnectionOpened(HostConnection* conn) {
+ std::cout << "Connection establised." << std::endl;
+ }
+
+ virtual void OnConnectionClosed(HostConnection* conn) {
+ std::cout << "Connection closed." << std::endl;
+ Exit();
+ }
+
+ virtual void OnConnectionFailed(HostConnection* conn) {
+ std::cout << "Conection failed." << std::endl;
+ Exit();
+ }
+
+ private:
+ void DoInitX11() {
+ display_ = XOpenDisplay(NULL);
+ if (!display_) {
+ std::cout << "Error - cannot open display" << std::endl;
+ Exit();
+ }
+
+ // Get properties of the screen.
+ int screen = DefaultScreen(display_);
+ int root_window = RootWindow(display_, screen);
+
+ // Creates the window.
+ window_ = XCreateSimpleWindow(display_, root_window, 1, 1, 640, 480, 0,
+ BlackPixel(display_, screen),
+ BlackPixel(display_, screen));
+ DCHECK(window_);
+ XStoreName(display_, window_, "X11 Remoting");
+
+ // Specifies what kind of messages we want to receive.
+ XSelectInput(display_, window_, ExposureMask | ButtonPressMask);
+ XMapWindow(display_, window_);
+ }
+
+ void DoInitConnection() {
+ // If the initialization of X11 has failed then return directly.
+ if (!display_)
+ return;
+
+ // Creates a HostConnection object and connection to the host.
+ LOG(INFO) << "Connecting...";
+ connection_.reset(new HostConnection(new ProtocolDecoder(), this));
+ connection_->Connect(username_, auth_token_, host_jid_);
+ }
+
+ void DoDestroyX11() {
+ if (display_ && window_) {
+ // Shutdown the window system.
+ XDestroyWindow(display_, window_);
+ XCloseDisplay(display_);
+ display_ = NULL;
+ window_ = 0;
+ }
+ }
+
+ void DoDisconnect() {
+ if (connection_.get())
+ connection_->Disconnect();
+ }
+
+ void Exit() {
+ // Disconnect the jingle channel and client.
+ message_loop_.PostTask(FROM_HERE,
+ NewRunnableMethod(this, &X11Client::DoDisconnect));
+
+ // Post a task to shutdown X11.
+ message_loop_.PostTask(FROM_HERE,
+ NewRunnableMethod(this, &X11Client::DoDestroyX11));
+
+ // Quit the current message loop.
+ message_loop_.PostTask(FROM_HERE, new MessageLoop::QuitTask());
+ }
+
+ // This method is executed on the main loop.
+ void DoInitClient(HostMessage* msg) {
+ DCHECK_EQ(&message_loop_, MessageLoop::current());
+ DCHECK(msg->has_init_client());
+ scoped_ptr<HostMessage> deleter(msg);
+
+ // Saves the dimension and resize the window.
+ width_ = msg->init_client().width();
+ height_ = msg->init_client().height();
+ LOG(INFO) << "Init client receievd: " << width_ << "x" << height_;
+ XResizeWindow(display_, window_, width_, height_);
+
+ // Now construct the X11View that renders the remote desktop.
+ view_ = new X11View(display_, window_, width_, height_);
+
+ // Schedule the event handler to process the event queue of X11.
+ ScheduleX11EventHandler();
+ }
+
+ // The following methods are executed on the same thread as libjingle.
+ void DoBeginUpdate(HostMessage* msg) {
+ DCHECK_EQ(&message_loop_, MessageLoop::current());
+ DCHECK(msg->has_begin_update_stream());
+
+ view_->HandleBeginUpdateStream(msg);
+ }
+
+ void DoHandleUpdate(HostMessage* msg) {
+ DCHECK_EQ(&message_loop_, MessageLoop::current());
+ DCHECK(msg->has_update_stream_packet());
+
+ view_->HandleUpdateStreamPacket(msg);
+ }
+
+ void DoEndUpdate(HostMessage* msg) {
+ DCHECK_EQ(&message_loop_, MessageLoop::current());
+ DCHECK(msg->has_end_update_stream());
+
+ view_->HandleEndUpdateStream(msg);
+ }
+
+ void DoProcessX11Events() {
+ DCHECK_EQ(&message_loop_, MessageLoop::current());
+ if (XPending(display_)) {
+ XEvent e;
+ XNextEvent(display_, &e);
+ if (e.type == Expose) {
+ // Tell the ChromotingView to paint again.
+ view_->Paint();
+ } else if (e.type == ButtonPress) {
+ // TODO(hclam): Implement.
+ NOTIMPLEMENTED();
+ } else {
+ // TODO(hclam): Implement.
+ NOTIMPLEMENTED();
+ }
+ }
+
+ // Schedule the next event handler.
+ ScheduleX11EventHandler();
+ }
+
+ void ScheduleX11EventHandler() {
+ // Schedule a delayed task to process X11 events in 10ms.
+ static const int kProcessEventsInterval = 10;
+ message_loop_.PostDelayedTask(
+ FROM_HERE,
+ NewRunnableMethod(this, &X11Client::DoProcessX11Events),
+ kProcessEventsInterval);
+ }
+
+ // Members used for display.
+ Display* display_;
+ Window window_;
+
+ // Dimension of the window. They are initialized when InitClient message is
+ // received.
+ int width_;
+ int height_;
+
+ std::string host_jid_;
+ std::string username_;
+ std::string auth_token_;
+ scoped_ptr<HostConnection> connection_;
+ scoped_refptr<ChromotingView> view_;
+
+ MessageLoop message_loop_;
+
+ DISALLOW_COPY_AND_ASSIGN(X11Client);
+};
+
+} // namespace remoting
+
+int main() {
+ base::AtExitManager at_exit;
+ std::string host_jid, username, auth_token;
+
+ if (!remoting::GetLoginInfo(host_jid, username, auth_token)) {
+ std::cout << "Cannot obtain login info" << std::endl;
+ return 1;
+ }
+
+ scoped_refptr<remoting::X11Client> client =
+ new remoting::X11Client(host_jid, username, auth_token);
+ client->Run();
+}
« no previous file with comments | « remoting/client/simple_client.cc ('k') | remoting/client/x11_view.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698