shim_ppapi.c revision 5821806d5e7f356e8fa4b058a389a808ea183019
15821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)/*
25821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) * Copyright (c) 2012 The Chromium Authors. All rights reserved.
35821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) * Use of this source code is governed by a BSD-style license that can be
45821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) * found in the LICENSE file.
55821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) */
65821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
75821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "native_client/src/untrusted/pnacl_irt_shim/shim_ppapi.h"
85821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
95821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include <string.h>
105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "native_client/src/shared/ppapi_proxy/ppruntime.h"
115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "native_client/src/untrusted/irt/irt.h"
125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "native_client/src/untrusted/irt/irt_ppapi.h"
135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/generators/pnacl_shim.h"
145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)/* The PNaCl PPAPI shims are only needed on x86-64 and arm. */
165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if defined(__x86_64__) || defined(__arm__)
175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)/* Use local strcmp to avoid dependency on libc. */
195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)static int mystrcmp(const char* s1, const char *s2) {
205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  while((*s1 && *s2) && (*s1++ == *s2++));
215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return *(--s1) - *(--s2);
225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)TYPE_nacl_irt_query __pnacl_real_irt_interface;
255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)/*
275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) * These remember the interface pointers the user registers by calling the
285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) * IRT entry point.
295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) */
305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)static struct PP_StartFunctions user_start_functions;
315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)static int32_t wrap_PPPInitializeModule(PP_Module module_id,
335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                        PPB_GetInterface get_browser_intf) {
345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  __set_real_Pnacl_PPBGetInterface(get_browser_intf);
355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  /*
365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   * Calls from user code to the PPB interfaces pass through here and may
375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   * require shims to convert the ABI.
385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   */
395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return (*user_start_functions.PPP_InitializeModule)(module_id,
405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                                      &__Pnacl_PPBGetInterface);
415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)static void wrap_PPPShutdownModule() {
445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  (*user_start_functions.PPP_ShutdownModule)();
455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)static const struct PP_StartFunctions wrapped_ppapi_methods = {
485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  wrap_PPPInitializeModule,
495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  wrap_PPPShutdownModule,
505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  /*
515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   * Calls from the IRT to the user plugin pass through here and may require
525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   * shims to convert the ABI.
535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   */
545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  __Pnacl_PPPGetInterface
555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)};
565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)static struct nacl_irt_ppapihook real_irt_ppapi_hook;
585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)static int wrap_ppapi_start(const struct PP_StartFunctions *funcs) {
605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  /*
615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   * Save the user's real bindings for the start functions.
625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   */
635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  user_start_functions = *funcs;
645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  __set_real_Pnacl_PPPGetInterface(user_start_functions.PPP_GetInterface);
655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  /*
675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   * Invoke the IRT's ppapi_start interface with the wrapped interface.
685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   */
695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return (*real_irt_ppapi_hook.ppapi_start)(&wrapped_ppapi_methods);
705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)static struct nacl_irt_ppapihook ppapi_hook = {
735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  wrap_ppapi_start,
745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NULL
755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)};
765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)size_t __pnacl_irt_interface_wrapper(const char *interface_ident,
785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                     void *table, size_t tablesize) {
795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  /*
805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   * Note there is a benign race in initializing the wrapper.
815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   * We build the "hook" structure by copying from the IRT's hook and then
825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   * writing our wrapper for the ppapi method.  Two threads may end up
835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   * attempting to do this simultaneously, which should not be a problem,
845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   * as they are writing the same values.
855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   */
865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (0 != mystrcmp(interface_ident, NACL_IRT_PPAPIHOOK_v0_1)) {
875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    /*
885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)     * The interface is not wrapped, so use the real interface.
895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)     */
905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return (*__pnacl_real_irt_interface)(interface_ident, table, tablesize);
915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if ((*__pnacl_real_irt_interface)(NACL_IRT_PPAPIHOOK_v0_1,
935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                    &real_irt_ppapi_hook,
945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                    sizeof real_irt_ppapi_hook) !=
955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      sizeof real_irt_ppapi_hook) {
965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return -1;
975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  /*
995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   * Copy the portion of the ppapihook interface that is not wrapped.
1005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   */
1015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ppapi_hook.ppapi_register_thread_creator =
1025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      real_irt_ppapi_hook.ppapi_register_thread_creator;
1035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  /*
1045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   * Copy the interface structure into the client.
1055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   */
1065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (sizeof ppapi_hook <= tablesize) {
1075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    memcpy(table, &ppapi_hook, sizeof ppapi_hook);
1085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return sizeof ppapi_hook;
1095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
1105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return 0;
1115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif
114