For Developers‎ > ‎Design Documents‎ > ‎Mojo‎ > ‎

Chrome IPC To Mojo IPC Cheat Sheet

 

Overview

This document is intended to provide an overview of how legacy IPC concepts map to Mojo constructs. Documentation for legacy IPC can be found in Inter-process Communication (IPC) and Security Tips for IPC.

Threading Model

IPC

IPCs can be sent and recieved from any threads. If the sending/receiving thread is not the IO thread, there is always a hop and memory copy to/from the IO thread. Related IPCs are grouped in the same _messages.h file only for convenience, so different messages within the same file can be sent from different threads or received on different threads.

Mojo

A binding or interface pointer can only be used on one thread at a time since they're not thread-safe. However the message pipe can be unbound using either Binding::Unbind or InterfacePtr::PassInterface and then it can be bound again on a different thread. Making a method call doesn't involve a thread hop, but receiving it on a thread other than the IO thread does involve a thread hop.
 
Ordering between legacy IPC and Mojo IPC is guaranteed only to the IO thread at this time.
 
SECURITY FOLLOWUPS:
 TODO: are there any examples of type converters between mojo enums? Is this something we want to encourage? Maybe not? Also, should chrome-security do a periodic audit of things marked [Extensible]? A quick search shows that there are some things marked [Extensible] that probably don't need to be.
 

Declaring Messages

IPC

Messages are declared with macros like IPC_MESSAGE_ROUTED1 and IPC_MESSAGE_CONTROL2, which differentiate between control and routed messages. Messages are generally grouped together by message classes, which are defined in //ipc/ipc_message_start.h.

IPC_SYNC_MESSAGE_CONTROL0_1(ClipboardHostMsg_ReadText,
                            std::string /* result */)
 
IPC_MESSAGE_CONTROL1(ClipboardHostMsg_WriteText,
                     std::string /* text */)
 

Mojo

There is no differentiation between control and routed messages in Mojo since there are no routing IDs. Instead messages flow between two endpoints of a message pipe. Mojo messages are declared as interfaces with methods, with related methods grouped together in the same interface. A hypothetical clipboard.mojom might look like this:

 
 
interface ClipboardHost {
  ReadText() => (string text);
  WriteText(string text);
};

Receiving / Sending Messages

IPC

Messages are received via the IPC::Listener interface. Message dispatch is typically handled using helper macros:

bool ClipboardMessageFilter::OnMessageReceived(const IPC::Message& message) {
  bool handled = true;
  IPC_BEGIN_MESSAGE_MAP(ClipboardMessageFilter, message)
    IPC_MESSAGE_HANDLER(ClipboardHostMsg_ReadText, OnReadText)
    IPC_MESSAGE_HANDLER(ClipboardHostMsg_WriteText, OnWriteText)
    IPC_MESSAGE_UNHANDLED(handled = false)
  IPC_END_MESSAGE_MAP()
  return handled;
}

void ClipboardMessageFilter::OnReadText(base::string16* result) {
  // Do stuff.
}

void ClipboardMessageFilter::OnWriteText(const base::string16& result) {
  // Do other stuff.
}

 
 

Messages are sent via IPC::Sender::Send(). Typically, code will construct a message, fill out the message params, and then call the IPC sender with the message:

 
std::string WebClipboardImpl::ReadText() {
  std::string result;
  sender_->Send(new ClipboardHostMsg_ReadText(&result));
  return result;
}

Mojo

The Mojo IDL compiler processes the .mojom files to generate corresponding C++ bindings (as well as bindings for other languages). The C++ bindings have generated stub interfaces that a C++ implementation should override:
 
 
#include "services/clipboard/clipboard.mojom.h"

class ClipboardImpl : public Clipboard {
 public:
  void ReadText(const ReadTextCallback& callback) {
    // Do stuff.
  }
  void WriteText(const mojo::String& s) {
    // Do other stuff.
  }
};
 
Sending messages in Mojo is simply a matter of calling the corresponding method on the proxy:
 
 
ClipboardPtr clipboard;
render_frame->GetRemoteInterfaces()->GetInterface(
    mojo::GetProxy(&clipboard));
clipboard->WriteText("Hello World!");

Pickling Values

In IPC, messages are simply a base::Picklebase::Pickle has some methods for reading/writing primitive types, such as uint32_tfloat, and base::StringPiece. Serializing/deserializing a type that doesn't have a corresponding base::Pickle::WriteT method requires specializing the IPC::ParamTraits template. If a message parameter fails deserialization in the browser process, the child process is killed. Currently with Mojo the message pipe is closed, but we don't kill the process (http://crbug.com/607293 tracks doing that).
 
Mojo also has a set of primitives types that it understands how to serialize/deserialize by default, such as stringarrayintenum, etc.
 
TODO(dcheng): Make sure we still emphasize that we want to preserve type-safety as much as possible. If that means defining a type mapping and writing custom validators, then so be it. Doing that is much better than, say, serializing a mojo::Array of bytes and then memcpying it into a struct.

Enums (e.g. IPC_ENUM_TRAITS and friends)

IPC

IPC has several different macros for creating a boilerplate IPC::ParamTraits specialization for enums:
 
 
// Serialization/deserialization of an enum with no validation: typically
// used for enums that are actually bitfields.
IPC_ENUM_TRAITS(blink::WebDragOperation)
 
// Serialization/deserialization of an enum. Validates that the deserialized
// enum value isn't less than 0 and isn't greater than the last declared
// value.
IPC_ENUM_TRAITS_MAX_VALUE(ui::DragDropTypes::DragEventSource,
                          ui::DragDropTypes::DRAG_EVENT_SOURCE_LAST)
 
// Like the previous macro, but validates that the enum value isn't less
// than a custom minimum.
IPC_ENUM_TRAITS_MIN_MAX_VALUE(content::PageZoom,
                              content::PageZoom::PAGE_ZOOM_OUT,
                              content::PageZoom::PAGE_ZOOM_IN)

Mojo

Just declare an enum in the .mojom file.

 
 
// Mojo automatically validates that the received enum has a legal value!
enum PageZoom {
  OUT,
  RESET,
  IN
};

// The Extensible attribute tells Mojo to skip validation. For bitfields,
// there can be many combinations of possible values. Extensible should only be
// used for these sorts of situations.
[Extensible]
enum DragOperation {
  None = 0,
  Copy = 1,
  Link = 2,
  Generic = 4,
  Private = 8,
  Move = 16,
  Delete = 32,
};

Transport-only structs (e.g. IPC_STRUCT_BEGIN, IPC_STRUCT_MEMBER, and IPC_STRUCT_END)

IPC provides IPC_STRUCT_BEGINIPC_STRUCT_MEMBER, and IPC_STRUCT_END macros as a way to define simple structs that are only used as IPC message parameters.

IPC

 

Defining a struct with these macros looks like this today:

 
IPC_STRUCT_BEGIN(ViewMsg_New_Params)
  IPC_STRUCT_MEMBER(int32_t, view_id, MSG_ROUTING_NONE)
  IPC_STRUCT_MEMBER(int32_t, main_frame_routing_id, MSG_ROUTING_NONE)
  IPC_STRUCT_MEMBER(int32_t, main_frame_widget_routing_id, MSG_ROUTING_NONE)
  IPC_STRUCT_MEMBER(int64_t, session_storage_namespace_id)
  IPC_STRUCT_MEMBER(int, opener_frame_route_id, MSG_ROUTING_NONE)
IPC_STRUCT_END()

And the resulting struct can be used like this:

bool RenderViewHostImpl::CreateRenderView(int opener_frame_route_id, ...) {
  /* do stuff */

  ViewMsg_New_Params params;
  params.view_id = GetRoutingID();
  /* …fill in rest of the fieldsu2026 */
  params.opener_frame_route_id = opener_frame_route_id;

  Send(new ViewMsg_New(params));

  /* do the rest of the stuff */
}

Mojo

Structs are defined in .mojom files:

struct CompositionSegment {
  uint32 start_offset;
  uint32 end_offset;
  bool emphasized;
};

The struct can be used as a parameter type:

 
 
interface ImeInstance {
  SetCompositionText(string text, array<CompositionSegment> segments);
}
 
Important: while this example uses the generated mojo structs directly, this is highly discouraged: typemapping to native C++ types usually makes the code more readable, and more importantly, makes sure that the mojo structs go through data validation (e.g. the renderer is not sending the browser invalid GURLs, etc)
 
And using it in C++ looks like this:
 
 
static mojo::Array<arc::mojom::CompositionSegmentPtr> ConvertSegments(
    const ui::CompositionText& composition) {
  mojo::Array<arc::mojom::CompositionSegmentPtr> segments =
      mojo::Array<arc::mojom::CompositionSegmentPtr>::New(0);
  for (const ui::CompositionUnderline& underline : composition.underlines) {
    arc::mojom::CompositionSegmentPtr segment =
        arc::mojom::CompositionSegment::New();
    segment->start_offset = underline.start_offset;
    segment->end_offset = underline.end_offset;
    segment->emphasized = (underline.thick ||
        (composition.selection.start() == underline.start_offset &&
         composition.selection.end() == underline.end_offset));
    segments.push_back(std::move(segment));
  }
  return segments;
}
 
void ArcImeBridgeImpl::SendSetCompositionText(
    const ui::CompositionText& composition) {
  ime_instance_->SetCompositionText(base::UTF16ToUTF8(composition.text),
                                    ConvertSegments(composition));
}

Pre-defined structs (e.g. IPC_STRUCT_TRAITS_BEGIN, IPC_STRUCT_TRAITS_MEMBER, and IPC_STRUCT_TRAITS_END)

IPC

Sometimes, IPC messages need to include structs that are already defined elsewhere. To help serialize simple structs for this case, IPC provides IPC_STRUCT_TRAITS_BEGINIPC_STRUCT_TRAITS_MEMBER, and IPC_STRUCTS_TRAITS_END. For example, given:
 
namespace ui {
struct FileInfo {
  base::FilePath path;
  base::FilePath display_name; // Optional.
};
} // namespace ui

IPC_STRUCT_TRAITS_BEGIN(ui::FileInfo)
  IPC_STRUCT_TRAITS_MEMBER(path)
  IPC_STRUCT_TRAITS_MEMBER(display_name)
IPC_STRUCT_TRAITS_END()

 

C++ code can pass a ui::FileInfo as an IPC parameter and it Just Works™:

Mojo

There isn't really a Mojo equivalent for this. Structs that need to be sent over Mojo need to have a corresponding Mojo struct. See the custom serialization section below for more information.

Custom serialization (e.g. custom IPC::ParamTraits)

IPC

There are some C++ types that can't be serialized with existing IPC::ParamTraits specializations. Those types specialize IPC::ParamTraits directly, rather than using the macros, and use the underlying base::Pickle methods to serialize/deserialize. A simple custom IPC::ParamTraits might look like this:
 
 
template <>
struct ParamTraits<std::string> {
  typedef std::string param_type;
  static void GetSize(base::PickleSizer* sizer, const param_type& p) {
    sizer->AddString(p);
  }
  static void Write(base::Pickle* m, const param_type& p) { m->WriteString(p); }
  static bool Read(const base::Pickle* m,
                   base::PickleIterator* iter,
                   param_type* r) {
    return iter->ReadString(r);
  }
  IPC_EXPORT static void Log(const param_type& p, std::string* l);
};

Mojo

Mojo uses StructTraits specializations in conjunction with type maps to transparently map between the native type and the Mojo structs at the binding layer. It's a bit too complicated to provide a simple example here, so please see Type Mapping in C++ for how to use this feature.
 
Note that Mojo also has a legacy TypeConverter mechanism for going to/from Mojo types. The use of type converters is discouraged in new code. Using StructTraits simplifies client code (e.g. code can use GURL instead of url::mojom::Url) and it allows validation of the data that needs custom serialization/deserialization. The latter point is quite important, since Mojo IPCs often go from the unprivileged renderer process to the privileged browser process.

Sync messages

In general, just like with legacy IPCs it's best to avoid synchronous messages if possible. If something really must be synchronous for legacy reasons, the [Sync] IDL attribute can be used. See the documentation on Synchronous Calls for more information.

[Chromium文档转载,第006章]Chrome IPC To Mojo IPC Cheat Sheet的更多相关文章

  1. [Chromium文档转载,第001章] Mojo Migration Guide

        For Developers‎ > ‎Design Documents‎ > ‎Mojo‎ > ‎ Mojo Migration Guide 目录 1 Summary 2 H ...

  2. [Chromium文档转载,第002章]Mojo C++ Bindings API

    Mojo C++ Bindings API This document is a subset of the Mojo documentation. Contents Overview Getting ...

  3. [Chromium文档转载,第003章]Proposal: Mojo Synchronous Methods

    Proposal: Mojo Synchronous Methods yzshen@chromium.org 02/02/2016 Overview Currently there are quite ...

  4. [Chromium文档转载,第007章]JNI on Chromium for Android

    Overview JNI (Java Native Interface) is the mechanism that enables Java code to call native function ...

  5. [Chromium文档转载,第005章]Calling Mojo from Blink

    For Developers‎ > ‎Design Documents‎ > ‎Mojo‎ > ‎ Calling Mojo from Blink Variants Let's as ...

  6. [Chromium文档转载,第004章]Mojo Synchronous Calls

    For Developers‎ > ‎Design Documents‎ > ‎Mojo‎ > ‎ Synchronous Calls Think carefully before ...

  7. 用R创建Word和PowerPoint文档--转载

    https://www.jianshu.com/p/7df62865c3ed Rapp --简书 Microsoft的Office软件在办公软件领域占有绝对的主导地位,几乎每个职场人士都必须掌握Wor ...

  8. java实现支付宝接口--文档..转载

    //实现java支付宝很简单,只要从支付宝官方下载   http://help.alipay.com/support/index_sh.htm下载程序,配置一下参数就OK了:   1.先到http:/ ...

  9. iOS开发主要参考文档(转载)

    Objective-C,语言的系统详细资料.这是做iOS开发的前题与基础.https://developer.apple.com/library/ios/#documentation/Cocoa/Co ...

随机推荐

  1. 嵌入式表单字段中的内容可能被server更改以删除不安全的内容。是否要又一次载入您的页面以查看保存结果?

    嵌入式表单字段中的内容可能被server更改以删除不安全的内容.是否要又一次载入您的页面以查看保存结果?         近期有朋友问到,当他在SharePoint首页上进行编辑时.插入一段代码. 完 ...

  2. hdoj-1212-Big Number【大数取余&amp;简单题】

    Big Number Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Su ...

  3. ubuntu 各种窗体操作

    通用 ctrl+alt+0~9 改变窗体大小和是否显示 alt+F4 关闭窗体菜单键+相应启动器位置的编号打开程序 ctrl+pageup/pagedown 在tab间移动 ctrle+shift+p ...

  4. hdoj 1719 Friend

    Friend Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Sub ...

  5. C++开发人脸性别识别教程(3)——OpenCv配置和ImageWatch插件介绍

    OpenCv是C++图像处理的重要工具.这个人脸性别识别的项目就是借助OpenCv进行开发的. 尽管网上已经有了非常多关于OpenCv的配置教程,但出于教程完整性考虑.这里还是用专门的一篇博客来介绍O ...

  6. Ubuntu系统中中文的设置

    这几天心血来潮,在实习公司真的没什么事情做,在学长的推荐下就学习了一下Ubuntu系统,感觉还是挺不错的,这个系统,所以我也推荐刚开始学习liunx系统的妹子们也可以先学一下这个. 1>你可以在 ...

  7. poj_1974,最长回文字串manacher

    时间复杂度为O(n),参考:http://bbs.dlut.edu.cn/bbstcon.php?board=Competition&gid=23474 #include<iostrea ...

  8. [poj 2891] Strange Way to Express Integers 解题报告(excrt扩展中国剩余定理)

    题目链接:http://poj.org/problem?id=2891 题目大意: 求解同余方程组,不保证模数互质 题解: 扩展中国剩余定理板子题 #include<algorithm> ...

  9. AvtiveMQ 参考

    推荐学习:https://www.cnblogs.com/zhuxiaojie/p/5564187.html#autoid-2-1-0

  10. 如何配置MySQL?(三)

    要进行mysql配置,首先要找到mysql的配置向导文件,这个配置向导文件就在我们安装目录下的一个bin的子目录下. 刚才我们是选择典型安装MySQL,一般windows是默认存储在C:\Progra ...