1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "ppapi/proxy/interface_list.h"
8 #include "base/lazy_instance.h"
9 #include "base/memory/singleton.h"
10 #include "ppapi/c/dev/ppb_audio_input_dev.h"
11 #include "ppapi/c/dev/ppb_buffer_dev.h"
12 #include "ppapi/c/dev/ppb_char_set_dev.h"
13 #include "ppapi/c/dev/ppb_crypto_dev.h"
14 #include "ppapi/c/dev/ppb_cursor_control_dev.h"
15 #include "ppapi/c/dev/ppb_device_ref_dev.h"
16 #include "ppapi/c/dev/ppb_font_dev.h"
17 #include "ppapi/c/dev/ppb_gles_chromium_texture_mapping_dev.h"
18 #include "ppapi/c/dev/ppb_ime_input_event_dev.h"
19 #include "ppapi/c/dev/ppb_memory_dev.h"
20 #include "ppapi/c/dev/ppb_messaging_deprecated.h"
21 #include "ppapi/c/dev/ppb_opengles2ext_dev.h"
22 #include "ppapi/c/dev/ppb_printing_dev.h"
23 #include "ppapi/c/dev/ppb_scrollbar_dev.h"
24 #include "ppapi/c/dev/ppb_text_input_dev.h"
25 #include "ppapi/c/dev/ppb_trace_event_dev.h"
26 #include "ppapi/c/dev/ppb_truetype_font_dev.h"
27 #include "ppapi/c/dev/ppb_url_util_dev.h"
28 #include "ppapi/c/dev/ppb_var_deprecated.h"
29 #include "ppapi/c/dev/ppb_video_capture_dev.h"
30 #include "ppapi/c/dev/ppb_view_dev.h"
31 #include "ppapi/c/dev/ppb_widget_dev.h"
32 #include "ppapi/c/dev/ppb_zoom_dev.h"
33 #include "ppapi/c/ppb_audio.h"
34 #include "ppapi/c/ppb_audio_buffer.h"
35 #include "ppapi/c/ppb_audio_config.h"
36 #include "ppapi/c/ppb_compositor.h"
37 #include "ppapi/c/ppb_compositor_layer.h"
38 #include "ppapi/c/ppb_console.h"
39 #include "ppapi/c/ppb_core.h"
40 #include "ppapi/c/ppb_file_io.h"
41 #include "ppapi/c/ppb_file_ref.h"
42 #include "ppapi/c/ppb_file_system.h"
43 #include "ppapi/c/ppb_fullscreen.h"
44 #include "ppapi/c/ppb_graphics_2d.h"
45 #include "ppapi/c/ppb_host_resolver.h"
46 #include "ppapi/c/ppb_image_data.h"
47 #include "ppapi/c/ppb_input_event.h"
48 #include "ppapi/c/ppb_instance.h"
49 #include "ppapi/c/ppb_media_stream_audio_track.h"
50 #include "ppapi/c/ppb_media_stream_video_track.h"
51 #include "ppapi/c/ppb_message_loop.h"
52 #include "ppapi/c/ppb_messaging.h"
53 #include "ppapi/c/ppb_mouse_lock.h"
54 #include "ppapi/c/ppb_net_address.h"
55 #include "ppapi/c/ppb_network_list.h"
56 #include "ppapi/c/ppb_network_monitor.h"
57 #include "ppapi/c/ppb_network_proxy.h"
58 #include "ppapi/c/ppb_opengles2.h"
59 #include "ppapi/c/ppb_tcp_socket.h"
60 #include "ppapi/c/ppb_text_input_controller.h"
61 #include "ppapi/c/ppb_udp_socket.h"
62 #include "ppapi/c/ppb_url_loader.h"
63 #include "ppapi/c/ppb_url_request_info.h"
64 #include "ppapi/c/ppb_url_response_info.h"
65 #include "ppapi/c/ppb_var.h"
66 #include "ppapi/c/ppb_var_array.h"
67 #include "ppapi/c/ppb_var_array_buffer.h"
68 #include "ppapi/c/ppb_var_dictionary.h"
69 #include "ppapi/c/ppb_video_decoder.h"
70 #include "ppapi/c/ppb_video_encoder.h"
71 #include "ppapi/c/ppb_video_frame.h"
72 #include "ppapi/c/ppb_view.h"
73 #include "ppapi/c/pp_errors.h"
74 #include "ppapi/c/ppp_instance.h"
75 #include "ppapi/c/private/ppb_camera_capabilities_private.h"
76 #include "ppapi/c/private/ppb_camera_device_private.h"
77 #include "ppapi/c/private/ppb_content_decryptor_private.h"
78 #include "ppapi/c/private/ppb_ext_crx_file_system_private.h"
79 #include "ppapi/c/private/ppb_file_io_private.h"
80 #include "ppapi/c/private/ppb_file_ref_private.h"
81 #include "ppapi/c/private/ppb_find_private.h"
82 #include "ppapi/c/private/ppb_flash_clipboard.h"
83 #include "ppapi/c/private/ppb_flash_file.h"
84 #include "ppapi/c/private/ppb_flash_font_file.h"
85 #include "ppapi/c/private/ppb_flash_fullscreen.h"
86 #include "ppapi/c/private/ppb_flash.h"
87 #include "ppapi/c/private/ppb_flash_device_id.h"
88 #include "ppapi/c/private/ppb_flash_drm.h"
89 #include "ppapi/c/private/ppb_flash_menu.h"
90 #include "ppapi/c/private/ppb_flash_message_loop.h"
91 #include "ppapi/c/private/ppb_flash_print.h"
92 #include "ppapi/c/private/ppb_host_resolver_private.h"
93 #include "ppapi/c/private/ppb_input_event_private.h"
94 #include "ppapi/c/private/ppb_isolated_file_system_private.h"
95 #include "ppapi/c/private/ppb_net_address_private.h"
96 #include "ppapi/c/private/ppb_output_protection_private.h"
97 #include "ppapi/c/private/ppb_pdf.h"
98 #include "ppapi/c/private/ppb_platform_verification_private.h"
99 #include "ppapi/c/private/ppb_talk_private.h"
100 #include "ppapi/c/private/ppb_tcp_server_socket_private.h"
101 #include "ppapi/c/private/ppb_tcp_socket_private.h"
102 #include "ppapi/c/private/ppb_testing_private.h"
103 #include "ppapi/c/private/ppb_udp_socket_private.h"
104 #include "ppapi/c/private/ppb_uma_private.h"
105 #include "ppapi/c/private/ppb_video_destination_private.h"
106 #include "ppapi/c/private/ppb_video_source_private.h"
107 #include "ppapi/c/private/ppb_x509_certificate_private.h"
108 #include "ppapi/c/private/ppp_content_decryptor_private.h"
109 #include "ppapi/c/trusted/ppb_broker_trusted.h"
110 #include "ppapi/c/trusted/ppb_browser_font_trusted.h"
111 #include "ppapi/c/trusted/ppb_char_set_trusted.h"
112 #include "ppapi/c/trusted/ppb_file_chooser_trusted.h"
113 #include "ppapi/c/trusted/ppb_url_loader_trusted.h"
114 #include "ppapi/proxy/interface_proxy.h"
115 #include "ppapi/proxy/plugin_globals.h"
116 #include "ppapi/proxy/ppapi_messages.h"
117 #include "ppapi/proxy/ppb_audio_proxy.h"
118 #include "ppapi/proxy/ppb_broker_proxy.h"
119 #include "ppapi/proxy/ppb_buffer_proxy.h"
120 #include "ppapi/proxy/ppb_core_proxy.h"
121 #include "ppapi/proxy/ppb_flash_message_loop_proxy.h"
122 #include "ppapi/proxy/ppb_graphics_3d_proxy.h"
123 #include "ppapi/proxy/ppb_image_data_proxy.h"
124 #include "ppapi/proxy/ppb_instance_proxy.h"
125 #include "ppapi/proxy/ppb_message_loop_proxy.h"
126 #include "ppapi/proxy/ppb_testing_proxy.h"
127 #include "ppapi/proxy/ppb_var_deprecated_proxy.h"
128 #include "ppapi/proxy/ppb_video_decoder_proxy.h"
129 #include "ppapi/proxy/ppb_x509_certificate_private_proxy.h"
130 #include "ppapi/proxy/ppp_class_proxy.h"
131 #include "ppapi/proxy/ppp_content_decryptor_private_proxy.h"
132 #include "ppapi/proxy/ppp_find_proxy.h"
133 #include "ppapi/proxy/ppp_graphics_3d_proxy.h"
134 #include "ppapi/proxy/ppp_input_event_proxy.h"
135 #include "ppapi/proxy/ppp_instance_private_proxy.h"
136 #include "ppapi/proxy/ppp_instance_proxy.h"
137 #include "ppapi/proxy/ppp_messaging_proxy.h"
138 #include "ppapi/proxy/ppp_mouse_lock_proxy.h"
139 #include "ppapi/proxy/ppp_pdf_proxy.h"
140 #include "ppapi/proxy/ppp_printing_proxy.h"
141 #include "ppapi/proxy/ppp_text_input_proxy.h"
142 #include "ppapi/proxy/ppp_video_decoder_proxy.h"
143 #include "ppapi/proxy/resource_creation_proxy.h"
144 #include "ppapi/shared_impl/ppb_opengles2_shared.h"
145 #include "ppapi/shared_impl/ppb_var_shared.h"
146 #include "ppapi/thunk/thunk.h"
148 // Helper to get the proxy name PPB_Foo_Proxy given the API name PPB_Foo.
149 #define PROXY_CLASS_NAME(api_name) api_name##_Proxy
151 // Helper to get the interface ID PPB_Foo_Proxy::kApiID given the API
153 #define PROXY_API_ID(api_name) PROXY_CLASS_NAME(api_name)::kApiID
155 // Helper to get the name of the templatized factory function.
156 #define PROXY_FACTORY_NAME(api_name) ProxyFactory<PROXY_CLASS_NAME(api_name)>
158 // Helper to get the name of the thunk GetPPB_Foo_1_0_Thunk given the interface
159 // struct name PPB_Foo_1_0.
160 #define INTERFACE_THUNK_NAME(iface_struct) thunk::Get##iface_struct##_Thunk
167 template<typename ProxyClass
>
168 InterfaceProxy
* ProxyFactory(Dispatcher
* dispatcher
) {
169 return new ProxyClass(dispatcher
);
172 base::LazyInstance
<PpapiPermissions
> g_process_global_permissions
;
176 InterfaceList::InterfaceList() {
177 memset(id_to_factory_
, 0, sizeof(id_to_factory_
));
179 // Register the API factories for each of the API types. This calls AddProxy
180 // for each InterfaceProxy type we support.
181 #define PROXIED_API(api_name) \
182 AddProxy(PROXY_API_ID(api_name), &PROXY_FACTORY_NAME(api_name));
184 // Register each proxied interface by calling AddPPB for each supported
185 // interface. Set current_required_permission to the appropriate value for
186 // the value you want expanded by this macro.
187 #define PROXIED_IFACE(iface_str, iface_struct) \
189 INTERFACE_THUNK_NAME(iface_struct)(), \
190 current_required_permission);
193 Permission current_required_permission
= PERMISSION_NONE
;
194 #include "ppapi/thunk/interfaces_ppb_private_no_permissions.h"
195 #include "ppapi/thunk/interfaces_ppb_public_stable.h"
198 Permission current_required_permission
= PERMISSION_DEV
;
199 #include "ppapi/thunk/interfaces_ppb_public_dev.h"
202 Permission current_required_permission
= PERMISSION_PRIVATE
;
203 #include "ppapi/thunk/interfaces_ppb_private.h"
206 #if !defined(OS_NACL)
207 Permission current_required_permission
= PERMISSION_FLASH
;
208 #include "ppapi/thunk/interfaces_ppb_private_flash.h"
209 #endif // !defined(OS_NACL)
212 Permission current_required_permission
= PERMISSION_DEV_CHANNEL
;
213 #include "ppapi/thunk/interfaces_ppb_public_dev_channel.h"
219 // Manually add some special proxies. Some of these don't have interfaces
220 // that they support, so aren't covered by the macros above, but have proxies
221 // for message routing. Others have different implementations between the
222 // proxy and the impl and there's no obvious message routing.
223 AddProxy(API_ID_RESOURCE_CREATION
, &ResourceCreationProxy::Create
);
224 AddProxy(API_ID_PPP_CLASS
, &PPP_Class_Proxy::Create
);
225 AddPPB(PPB_CORE_INTERFACE_1_0
,
226 PPB_Core_Proxy::GetPPB_Core_Interface(), PERMISSION_NONE
);
227 AddPPB(PPB_MESSAGELOOP_INTERFACE_1_0
,
228 PPB_MessageLoop_Proxy::GetInterface(), PERMISSION_NONE
);
229 AddPPB(PPB_OPENGLES2_INTERFACE_1_0
,
230 PPB_OpenGLES2_Shared::GetInterface(), PERMISSION_NONE
);
231 AddPPB(PPB_OPENGLES2_INSTANCEDARRAYS_INTERFACE_1_0
,
232 PPB_OpenGLES2_Shared::GetInstancedArraysInterface(), PERMISSION_NONE
);
233 AddPPB(PPB_OPENGLES2_FRAMEBUFFERBLIT_INTERFACE_1_0
,
234 PPB_OpenGLES2_Shared::GetFramebufferBlitInterface(), PERMISSION_NONE
);
235 AddPPB(PPB_OPENGLES2_FRAMEBUFFERMULTISAMPLE_INTERFACE_1_0
,
236 PPB_OpenGLES2_Shared::GetFramebufferMultisampleInterface(),
238 AddPPB(PPB_OPENGLES2_CHROMIUMENABLEFEATURE_INTERFACE_1_0
,
239 PPB_OpenGLES2_Shared::GetChromiumEnableFeatureInterface(),
241 AddPPB(PPB_OPENGLES2_CHROMIUMMAPSUB_INTERFACE_1_0
,
242 PPB_OpenGLES2_Shared::GetChromiumMapSubInterface(), PERMISSION_NONE
);
243 AddPPB(PPB_OPENGLES2_CHROMIUMMAPSUB_DEV_INTERFACE_1_0
,
244 PPB_OpenGLES2_Shared::GetChromiumMapSubInterface(), PERMISSION_NONE
);
245 AddPPB(PPB_OPENGLES2_QUERY_INTERFACE_1_0
,
246 PPB_OpenGLES2_Shared::GetQueryInterface(), PERMISSION_NONE
);
247 AddPPB(PPB_OPENGLES2_VERTEXARRAYOBJECT_INTERFACE_1_0
,
248 PPB_OpenGLES2_Shared::GetVertexArrayObjectInterface(),
250 AddPPB(PPB_OPENGLES2_DRAWBUFFERS_DEV_INTERFACE_1_0
,
251 PPB_OpenGLES2_Shared::GetDrawBuffersInterface(),
253 AddPPB(PPB_VAR_ARRAY_BUFFER_INTERFACE_1_0
,
254 PPB_Var_Shared::GetVarArrayBufferInterface1_0(),
256 AddPPB(PPB_VAR_INTERFACE_1_2
,
257 PPB_Var_Shared::GetVarInterface1_2(), PERMISSION_NONE
);
258 AddPPB(PPB_VAR_INTERFACE_1_1
,
259 PPB_Var_Shared::GetVarInterface1_1(), PERMISSION_NONE
);
260 AddPPB(PPB_VAR_INTERFACE_1_0
,
261 PPB_Var_Shared::GetVarInterface1_0(), PERMISSION_NONE
);
263 #if !defined(OS_NACL)
264 // PPB (browser) interfaces.
265 // Do not add more stuff here, they should be added to interface_list*.h
266 // TODO(brettw) remove these.
267 AddProxy(API_ID_PPB_INSTANCE_PRIVATE
, &ProxyFactory
<PPB_Instance_Proxy
>);
268 AddPPB(PPB_INSTANCE_PRIVATE_INTERFACE_0_1
,
269 thunk::GetPPB_Instance_Private_0_1_Thunk(),
272 AddProxy(API_ID_PPB_VAR_DEPRECATED
, &ProxyFactory
<PPB_Var_Deprecated_Proxy
>);
273 AddPPB(PPB_VAR_DEPRECATED_INTERFACE
,
274 PPB_Var_Deprecated_Proxy::GetProxyInterface(), PERMISSION_DEV
);
276 // TODO(tomfinegan): Figure out where to put these once we refactor things
277 // to load the PPP interface struct from the PPB interface.
278 AddProxy(API_ID_PPP_CONTENT_DECRYPTOR_PRIVATE
,
279 &ProxyFactory
<PPP_ContentDecryptor_Private_Proxy
>);
280 AddPPP(PPP_CONTENTDECRYPTOR_PRIVATE_INTERFACE
,
281 PPP_ContentDecryptor_Private_Proxy::GetProxyInterface());
283 AddProxy(API_ID_PPB_TESTING
, &ProxyFactory
<PPB_Testing_Proxy
>);
284 AddPPB(PPB_TESTING_PRIVATE_INTERFACE
,
285 PPB_Testing_Proxy::GetProxyInterface(), PERMISSION_TESTING
);
287 // PPP (plugin) interfaces.
288 // TODO(brettw) move these to interface_list*.h
289 AddProxy(API_ID_PPP_GRAPHICS_3D
, &ProxyFactory
<PPP_Graphics3D_Proxy
>);
290 AddPPP(PPP_GRAPHICS_3D_INTERFACE
, PPP_Graphics3D_Proxy::GetProxyInterface());
291 AddProxy(API_ID_PPP_INPUT_EVENT
, &ProxyFactory
<PPP_InputEvent_Proxy
>);
292 AddPPP(PPP_INPUT_EVENT_INTERFACE
, PPP_InputEvent_Proxy::GetProxyInterface());
293 AddProxy(API_ID_PPP_INSTANCE
, &ProxyFactory
<PPP_Instance_Proxy
>);
294 #if !defined(OS_NACL)
295 AddPPP(PPP_INSTANCE_INTERFACE_1_1
,
296 PPP_Instance_Proxy::GetInstanceInterface());
297 AddProxy(API_ID_PPP_INSTANCE_PRIVATE
,
298 &ProxyFactory
<PPP_Instance_Private_Proxy
>);
299 AddPPP(PPP_INSTANCE_PRIVATE_INTERFACE
,
300 PPP_Instance_Private_Proxy::GetProxyInterface());
302 AddProxy(API_ID_PPP_MESSAGING
, &ProxyFactory
<PPP_Messaging_Proxy
>);
303 AddProxy(API_ID_PPP_MOUSE_LOCK
, &ProxyFactory
<PPP_MouseLock_Proxy
>);
304 AddPPP(PPP_MOUSELOCK_INTERFACE
, PPP_MouseLock_Proxy::GetProxyInterface());
305 AddProxy(API_ID_PPP_PRINTING
, &ProxyFactory
<PPP_Printing_Proxy
>);
306 AddPPP(PPP_PRINTING_DEV_INTERFACE
, PPP_Printing_Proxy::GetProxyInterface());
307 AddProxy(API_ID_PPP_TEXT_INPUT
, &ProxyFactory
<PPP_TextInput_Proxy
>);
308 AddPPP(PPP_TEXTINPUT_DEV_INTERFACE
, PPP_TextInput_Proxy::GetProxyInterface());
309 #if !defined(OS_NACL)
310 AddProxy(API_ID_PPP_PDF
, &ProxyFactory
<PPP_Pdf_Proxy
>);
311 AddPPP(PPP_PDF_INTERFACE
, PPP_Pdf_Proxy::GetProxyInterface());
312 AddProxy(API_ID_PPP_FIND_PRIVATE
, &ProxyFactory
<PPP_Find_Proxy
>);
313 AddPPP(PPP_FIND_PRIVATE_INTERFACE
, PPP_Find_Proxy::GetProxyInterface());
314 AddProxy(API_ID_PPP_VIDEO_DECODER_DEV
, &ProxyFactory
<PPP_VideoDecoder_Proxy
>);
315 AddPPP(PPP_VIDEODECODER_DEV_INTERFACE
,
316 PPP_VideoDecoder_Proxy::GetProxyInterface());
320 InterfaceList::~InterfaceList() {
324 InterfaceList
* InterfaceList::GetInstance() {
325 // CAUTION: This function is called without the ProxyLock to avoid excessive
326 // excessive locking from C++ wrappers. (See also GetBrowserInterface.)
327 return Singleton
<InterfaceList
>::get();
331 void InterfaceList::SetProcessGlobalPermissions(
332 const PpapiPermissions
& permissions
) {
333 g_process_global_permissions
.Get() = permissions
;
336 InterfaceProxy::Factory
InterfaceList::GetFactoryForID(ApiID id
) const {
337 int index
= static_cast<int>(id
);
338 static_assert(API_ID_NONE
== 0, "none must be zero");
339 if (id
<= 0 || id
>= API_ID_COUNT
)
341 return id_to_factory_
[index
];
344 const void* InterfaceList::GetInterfaceForPPB(const std::string
& name
) {
345 // CAUTION: This function is called without the ProxyLock to avoid excessive
346 // excessive locking from C++ wrappers. (See also GetBrowserInterface.)
347 NameToInterfaceInfoMap::iterator found
=
348 name_to_browser_info_
.find(name
);
349 if (found
== name_to_browser_info_
.end())
352 if (g_process_global_permissions
.Get().HasPermission(
353 found
->second
->required_permission())) {
354 // Only log interface use once per plugin.
355 found
->second
->LogWithUmaOnce(
356 PluginGlobals::Get()->GetBrowserSender(), name
);
357 return found
->second
->iface();
362 const void* InterfaceList::GetInterfaceForPPP(const std::string
& name
) const {
363 NameToInterfaceInfoMap::const_iterator found
=
364 name_to_plugin_info_
.find(name
);
365 if (found
== name_to_plugin_info_
.end())
367 return found
->second
->iface();
370 void InterfaceList::InterfaceInfo::LogWithUmaOnce(
371 IPC::Sender
* sender
, const std::string
& name
) {
373 base::AutoLock
acquire(sent_to_uma_lock_
);
378 int hash
= InterfaceList::HashInterfaceName(name
);
379 PluginGlobals::Get()->GetBrowserSender()->Send(
380 new PpapiHostMsg_LogInterfaceUsage(hash
));
383 void InterfaceList::AddProxy(ApiID id
,
384 InterfaceProxy::Factory factory
) {
385 // For interfaces with no corresponding _Proxy objects, the macros will
386 // generate calls to this function with API_ID_NONE. This means we
387 // should just skip adding a factory for these functions.
388 if (id
== API_ID_NONE
)
391 // The factory should be an exact dupe of the one we already have if it
392 // has already been registered before.
393 int index
= static_cast<int>(id
);
394 DCHECK(!id_to_factory_
[index
] || id_to_factory_
[index
] == factory
);
396 id_to_factory_
[index
] = factory
;
399 void InterfaceList::AddPPB(const char* name
,
402 DCHECK(name_to_browser_info_
.find(name
) == name_to_browser_info_
.end());
403 name_to_browser_info_
.add(
404 name
, scoped_ptr
<InterfaceInfo
>(new InterfaceInfo(iface
, perm
)));
407 void InterfaceList::AddPPP(const char* name
,
409 DCHECK(name_to_plugin_info_
.find(name
) == name_to_plugin_info_
.end());
410 name_to_plugin_info_
.add(
412 scoped_ptr
<InterfaceInfo
>(new InterfaceInfo(iface
, PERMISSION_NONE
)));
415 int InterfaceList::HashInterfaceName(const std::string
& name
) {
416 uint32 data
= base::Hash(name
.c_str(), name
.size());
417 // Strip off the signed bit because UMA doesn't support negative values,
418 // but takes a signed int as input.
419 return static_cast<int>(data
& 0x7fffffff);