普通文本  |  140行  |  4.99 KB

// Copyright (c) 2013 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 "ppapi/proxy/extensions_common_resource.h"

#include "base/bind.h"
#include "base/bind_helpers.h"
#include "base/logging.h"
#include "base/memory/scoped_ptr.h"
#include "base/values.h"
#include "ppapi/c/pp_errors.h"
#include "ppapi/proxy/ppapi_messages.h"
#include "ppapi/proxy/resource_message_params.h"
#include "ppapi/shared_impl/tracked_callback.h"
#include "ppapi/shared_impl/var_value_conversions.h"

namespace ppapi {
namespace proxy {

ExtensionsCommonResource::ExtensionsCommonResource(Connection connection,
                                                   PP_Instance instance)
    : PluginResource(connection, instance) {
  SendCreate(RENDERER, PpapiHostMsg_ExtensionsCommon_Create());
  SendCreate(BROWSER, PpapiHostMsg_ExtensionsCommon_Create());
}

ExtensionsCommonResource::~ExtensionsCommonResource() {
}

thunk::ExtensionsCommon_API*
ExtensionsCommonResource::AsExtensionsCommon_API() {
  return this;
}

int32_t ExtensionsCommonResource::CallRenderer(
    const std::string& request_name,
    const std::vector<PP_Var>& input_args,
    const std::vector<PP_Var*>& output_args,
    scoped_refptr<TrackedCallback> callback) {
  return CommonCall(RENDERER, request_name, input_args, output_args, callback);
}

void ExtensionsCommonResource::PostRenderer(const std::string& request_name,
                                            const std::vector<PP_Var>& args) {
  CommonPost(RENDERER, request_name, args);
}

int32_t ExtensionsCommonResource::CallBrowser(
    const std::string& request_name,
    const std::vector<PP_Var>& input_args,
    const std::vector<PP_Var*>& output_args,
    scoped_refptr<TrackedCallback> callback) {
  return CommonCall(BROWSER, request_name, input_args, output_args, callback);
}

void ExtensionsCommonResource::PostBrowser(const std::string& request_name,
                                           const std::vector<PP_Var>& args) {
  CommonPost(BROWSER, request_name, args);
}

int32_t ExtensionsCommonResource::CommonCall(
    Destination dest,
    const std::string& request_name,
    const std::vector<PP_Var>& input_args,
    const std::vector<PP_Var*>& output_args,
    scoped_refptr<TrackedCallback> callback) {
  // TODO(yzshen): CreateValueFromVar() doesn't generate null fields for
  // dictionary values. That is the expected behavior for most APIs. If later we
  // want to support APIs that require to preserve null fields in dictionaries,
  // we should change the behavior to always preserve null fields at the plugin
  // side, and figure out whether they should be stripped at the renderer side.
  scoped_ptr<base::ListValue> input_args_value(
      CreateListValueFromVarVector(input_args));
  if (!input_args_value.get()) {
    LOG(WARNING) << "Failed to convert PP_Var input arguments.";
    return PP_ERROR_BADARGUMENT;
  }

  PluginResource::Call<PpapiPluginMsg_ExtensionsCommon_CallReply>(
      dest,
      PpapiHostMsg_ExtensionsCommon_Call(request_name, *input_args_value),
      base::Bind(&ExtensionsCommonResource::OnPluginMsgCallReply,
                 base::Unretained(this), output_args, callback));
  return PP_OK_COMPLETIONPENDING;
}

void ExtensionsCommonResource::CommonPost(Destination dest,
                                          const std::string& request_name,
                                          const std::vector<PP_Var>& args) {
  scoped_ptr<base::ListValue> args_value(CreateListValueFromVarVector(args));
  if (!args_value.get()) {
    LOG(WARNING) << "Failed to convert PP_Var input arguments.";
    return;
  }

  PluginResource::Post(
      dest, PpapiHostMsg_ExtensionsCommon_Post(request_name, *args_value));
}

void ExtensionsCommonResource::OnPluginMsgCallReply(
    const std::vector<PP_Var*>& output_args,
    scoped_refptr<TrackedCallback> callback,
    const ResourceMessageReplyParams& params,
    const base::ListValue& output) {
  // |output_args| may be invalid and shouldn't be accessed if the callback has
  // been called.
  if (!TrackedCallback::IsPending(callback))
    return;

  int32_t result = params.result();

  // If the size doesn't match, something must be really wrong.
  CHECK_EQ(output_args.size(), output.GetSize());

  std::vector<PP_Var> output_vars;
  if (CreateVarVectorFromListValue(output, &output_vars)) {
    DCHECK_EQ(output_args.size(), output_vars.size());
    std::vector<PP_Var>::const_iterator src_iter = output_vars.begin();
    std::vector<PP_Var*>::const_iterator dest_iter = output_args.begin();
    for (; src_iter != output_vars.end() && dest_iter != output_args.end();
         ++src_iter, ++dest_iter) {
      **dest_iter = *src_iter;
    }
  } else {
    NOTREACHED();
    result = PP_ERROR_FAILED;
    for (std::vector<PP_Var*>::const_iterator dest_iter = output_args.begin();
         dest_iter != output_args.end();
         ++dest_iter) {
      **dest_iter = PP_MakeUndefined();
    }
  }

  callback->Run(result);
}

}  // namespace proxy
}  // namespace ppapi