C++程序  |  89行  |  2.75 KB

// Copyright 2014 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 "base/memory/scoped_ptr.h"
#include "dbus/bus.h"
#include "dbus/exported_object.h"
#include "dbus/message.h"
#include "dbus/object_path.h"
#include "mojo/embedder/channel_init.h"
#include "mojo/public/cpp/application/application.h"
#include "mojo/public/interfaces/service_provider/service_provider.mojom.h"
#include "mojo/shell/external_service.mojom.h"

namespace mojo {
const char kMojoDBusImplPath[] = "/org/chromium/MojoImpl";
const char kMojoDBusInterface[] = "org.chromium.Mojo";
const char kMojoDBusConnectMethod[] = "ConnectChannel";

class DBusExternalServiceBase {
 public:
  explicit DBusExternalServiceBase(const std::string& service_name);
  virtual ~DBusExternalServiceBase();

  void Start();

 protected:
  // TODO(cmasone): Enable multiple peers to connect/disconnect
  virtual void Connect(ScopedMessagePipeHandle client_handle) = 0;
  virtual void Disconnect() = 0;

 private:
  // Implementation of org.chromium.Mojo.ConnectChannel, exported over DBus.
  // Takes a file descriptor and uses it to create a MessagePipe that is then
  // hooked to an implementation of ExternalService.
  void ConnectChannel(dbus::MethodCall* method_call,
                      dbus::ExportedObject::ResponseSender sender);

  void ExportMethods();
  void InitializeDBus();
  void TakeDBusServiceOwnership();

  const std::string service_name_;
  scoped_refptr<dbus::Bus> bus_;
  dbus::ExportedObject* exported_object_;  // Owned by bus_;
  scoped_ptr<embedder::ChannelInit> channel_init_;
  DISALLOW_COPY_AND_ASSIGN(DBusExternalServiceBase);
};

template <class ServiceImpl>
class DBusExternalService : public DBusExternalServiceBase {
 public:
  explicit DBusExternalService(const std::string& service_name)
      : DBusExternalServiceBase(service_name) {
  }
  virtual ~DBusExternalService() {}

 protected:
  virtual void Connect(ScopedMessagePipeHandle client_handle) OVERRIDE {
    external_service_.reset(BindToPipe(new Impl(this), client_handle.Pass()));
  }

  virtual void Disconnect() OVERRIDE {
    external_service_.reset();
  }

 private:
  class Impl : public InterfaceImpl<ExternalService> {
   public:
    explicit Impl(DBusExternalService* service) : service_(service) {
    }
    virtual void OnConnectionError() OVERRIDE {
      service_->Disconnect();
    }
    virtual void Activate(ScopedMessagePipeHandle service_provider_handle)
        OVERRIDE {
      app_.reset(new Application(service_provider_handle.Pass()));
      app_->AddService<ServiceImpl>();
    }
   private:
    DBusExternalService* service_;
    scoped_ptr<Application> app_;
  };

  scoped_ptr<Impl> external_service_;
};

}  // namespace mojo