[Chromium文档转载,第006章]Chrome IPC To Mojo IPC Cheat Sheet
Chrome IPC To Mojo IPC Cheat Sheet
目录
OverviewThis 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 ModelIPCIPCs 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.
MojoA 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 MessagesIPCMessages 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 */) MojoThere 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 interface ClipboardHost { ReadText() => (string text); WriteText(string text); }; Receiving / Sending MessagesIPCMessages 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; }
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() { MojoThe 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 ValuesIn IPC, messages are simply a
base::Pickle . base::Pickle has some methods for reading/writing primitive types, such as uint32_t , float , 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
string , array , int , enum , 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)IPCIPC 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) MojoJust declare an enum in the .mojom file. // Mojo automatically validates that the received enum has a legal value! enum PageZoom { OUT, RESET, IN };
Transport-only structs (e.g. IPC_STRUCT_BEGIN, IPC_STRUCT_MEMBER, and IPC_STRUCT_END)IPC provides
IPC_STRUCT_BEGIN , IPC_STRUCT_MEMBER , and IPC_STRUCT_END macros as a way to define simple structs that are only used as IPC message parameters.IPCDefining 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 */
MojoStructs 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)IPCSometimes, IPC messages need to include structs that are already defined elsewhere. To help serialize simple structs for this case, IPC provides
IPC_STRUCT_TRAITS_BEGIN , IPC_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
C++ code can pass a MojoThere 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)IPCThere 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); }; MojoMojo 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 messagesIn 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的更多相关文章
- [Chromium文档转载,第001章] Mojo Migration Guide
For Developers > Design Documents > Mojo > Mojo Migration Guide 目录 1 Summary 2 H ...
- [Chromium文档转载,第002章]Mojo C++ Bindings API
Mojo C++ Bindings API This document is a subset of the Mojo documentation. Contents Overview Getting ...
- [Chromium文档转载,第003章]Proposal: Mojo Synchronous Methods
Proposal: Mojo Synchronous Methods yzshen@chromium.org 02/02/2016 Overview Currently there are quite ...
- [Chromium文档转载,第007章]JNI on Chromium for Android
Overview JNI (Java Native Interface) is the mechanism that enables Java code to call native function ...
- [Chromium文档转载,第005章]Calling Mojo from Blink
For Developers > Design Documents > Mojo > Calling Mojo from Blink Variants Let's as ...
- [Chromium文档转载,第004章]Mojo Synchronous Calls
For Developers > Design Documents > Mojo > Synchronous Calls Think carefully before ...
- 用R创建Word和PowerPoint文档--转载
https://www.jianshu.com/p/7df62865c3ed Rapp --简书 Microsoft的Office软件在办公软件领域占有绝对的主导地位,几乎每个职场人士都必须掌握Wor ...
- java实现支付宝接口--文档..转载
//实现java支付宝很简单,只要从支付宝官方下载 http://help.alipay.com/support/index_sh.htm下载程序,配置一下参数就OK了: 1.先到http:/ ...
- iOS开发主要参考文档(转载)
Objective-C,语言的系统详细资料.这是做iOS开发的前题与基础.https://developer.apple.com/library/ios/#documentation/Cocoa/Co ...
随机推荐
- 网页里如何使用js禁用F12事件
接上一篇,突然想起来,类似于网页里如何使用js禁用鼠标右击事件,还有禁用F12事件也可以禁用一下,总所周知,对于Web开发人员来说,常常要进行界面的调试.使用F12调试工具能够很方便地进行调试,查看h ...
- NYIST 1006 偷西瓜
偷西瓜 时间限制:1000 ms | 内存限制:65535 KB 难度:4 描述 对于农村的孩子来说最大的乐趣,莫过于和小伙伴们一块下地偷西瓜了,虽然孩子们条件不是很好,但是往往他们很聪明,他 ...
- windos环境python3.5安装 paramiko
一.执行命令pip install paramiko,情况如下: C:\Users\ZFH>pip install paramikoCollecting paramiko Downloadin ...
- Unity3d 开发(七)AssetBundle组织文件夹
本文探讨怎样配置一个AssetBundle更为合理. 对于结构为 的文件夹结构,当中shared是Hero文件夹下须要用到的公用资源.即公有依赖.可採用例如以下的打包策略 整个文件夹打包 将整个100 ...
- RvmTranslator7.0-OBJ
RvmTranslator7.0-OBJ eryar@163.com RvmTranslator can translate the RVM file exported by AVEVA Plant( ...
- C内存管理一 概述
我们写了这么多年的程序猿.可能理论方面还比不上大学生.有人 "嘘"我了,假设有能回答下面几个问题的同学请举手: 1.面试常常遇到:同学请说说堆栈的差别? 2.同学请说说一个函数在堆 ...
- UNIX环境高级编程之第4章:文件和文件夹-习题
4.1 stat函数是尾随符号链接的,所以用stat替换lstat不会显示符号链接的信息 4.2 在一个目录下先再shell中输入umask shell进程再进行创建文件的操作.其权限抖都会被屏蔽 4 ...
- .Net配置虚拟域名
1.在IIS中配置和地址端口,和名称. 2.在hosts文件中加上地址匹配. 3.重启IIS管理网站. 就可以通过虚拟域名进行访问了.
- HBase框架基础(二)
* HBase框架基础(二) 上一节我们了解了HBase的架构原理和模块组成,这一节我们先来聊一聊HBase的读写数据的过程. * HBase的读写流程及3个机制 HBase的读数据流程: 1.HRe ...
- jquery实现上下浮动
jquery实现上下浮动: <!DOCTYPE html> <html> <head> <meta charset="utf-8"> ...