blob: e430dfd74ad201f0db2967f452c79d0fa617e32a [file] [log] [blame]
// Copyright (c) 2012 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/flash_clipboard_resource.h"
#include <stddef.h>
#include "base/numerics/safe_conversions.h"
#include "ipc/ipc_message.h"
#include "ppapi/c/pp_errors.h"
#include "ppapi/proxy/ppapi_messages.h"
#include "ppapi/shared_impl/ppapi_globals.h"
#include "ppapi/shared_impl/var.h"
#include "ppapi/shared_impl/var_tracker.h"
namespace ppapi {
namespace proxy {
namespace {
// Returns whether the given clipboard type is valid.
bool IsValidClipboardType(PP_Flash_Clipboard_Type type) {
return type == PP_FLASH_CLIPBOARD_TYPE_STANDARD ||
type == PP_FLASH_CLIPBOARD_TYPE_SELECTION;
}
// Convert a PP_Var to/from a string which is transmitted to the pepper host.
// These functions assume the format is valid.
bool PPVarToClipboardString(int32_t format,
const PP_Var& var,
std::string* string_out) {
if (format == PP_FLASH_CLIPBOARD_FORMAT_PLAINTEXT ||
format == PP_FLASH_CLIPBOARD_FORMAT_HTML) {
StringVar* string_var = StringVar::FromPPVar(var);
if (!string_var)
return false;
*string_out = string_var->value();
return true;
} else {
// All other formats are expected to be array buffers.
ArrayBufferVar* array_buffer_var = ArrayBufferVar::FromPPVar(var);
if (!array_buffer_var)
return false;
*string_out = std::string(static_cast<const char*>(array_buffer_var->Map()),
array_buffer_var->ByteLength());
return true;
}
}
PP_Var ClipboardStringToPPVar(int32_t format,
const std::string& string) {
if (format == PP_FLASH_CLIPBOARD_FORMAT_PLAINTEXT ||
format == PP_FLASH_CLIPBOARD_FORMAT_HTML) {
return StringVar::StringToPPVar(string);
} else {
// All other formats are expected to be array buffers.
return PpapiGlobals::Get()->GetVarTracker()->MakeArrayBufferPPVar(
base::checked_cast<uint32_t>(string.size()), string.data());
}
}
} // namespace
FlashClipboardResource::FlashClipboardResource(
Connection connection, PP_Instance instance)
: PluginResource(connection, instance) {
SendCreate(BROWSER, PpapiHostMsg_FlashClipboard_Create());
}
FlashClipboardResource::~FlashClipboardResource() {
}
thunk::PPB_Flash_Clipboard_API*
FlashClipboardResource::AsPPB_Flash_Clipboard_API() {
return this;
}
uint32_t FlashClipboardResource::RegisterCustomFormat(
PP_Instance instance,
const char* format_name) {
// Check to see if the format has already been registered.
uint32_t format = clipboard_formats_.GetFormatID(format_name);
if (format != PP_FLASH_CLIPBOARD_FORMAT_INVALID)
return format;
int32_t result =
SyncCall<PpapiPluginMsg_FlashClipboard_RegisterCustomFormatReply>(
BROWSER,
PpapiHostMsg_FlashClipboard_RegisterCustomFormat(format_name),
&format);
if (result != PP_OK || format == PP_FLASH_CLIPBOARD_FORMAT_INVALID)
return PP_FLASH_CLIPBOARD_FORMAT_INVALID;
clipboard_formats_.SetRegisteredFormat(format_name, format);
return format;
}
PP_Bool FlashClipboardResource::IsFormatAvailable(
PP_Instance instance,
PP_Flash_Clipboard_Type clipboard_type,
uint32_t format) {
if (IsValidClipboardType(clipboard_type) &&
(FlashClipboardFormatRegistry::IsValidPredefinedFormat(format) ||
clipboard_formats_.IsFormatRegistered(format))) {
int32_t result = SyncCall<IPC::Message>(BROWSER,
PpapiHostMsg_FlashClipboard_IsFormatAvailable(clipboard_type, format));
return result == PP_OK ? PP_TRUE : PP_FALSE;
}
return PP_FALSE;
}
PP_Var FlashClipboardResource::ReadData(
PP_Instance instance,
PP_Flash_Clipboard_Type clipboard_type,
uint32_t format) {
std::string value;
int32_t result =
SyncCall<PpapiPluginMsg_FlashClipboard_ReadDataReply>(
BROWSER,
PpapiHostMsg_FlashClipboard_ReadData(clipboard_type, format),
&value);
if (result != PP_OK)
return PP_MakeUndefined();
return ClipboardStringToPPVar(format, value);
}
int32_t FlashClipboardResource::WriteData(
PP_Instance instance,
PP_Flash_Clipboard_Type clipboard_type,
uint32_t data_item_count,
const uint32_t formats[],
const PP_Var data_items[]) {
if (!IsValidClipboardType(clipboard_type))
return PP_ERROR_BADARGUMENT;
std::vector<uint32_t> formats_vector;
std::vector<std::string> data_items_vector;
for (size_t i = 0; i < data_item_count; ++i) {
if (!clipboard_formats_.IsFormatRegistered(formats[i]) &&
!FlashClipboardFormatRegistry::IsValidPredefinedFormat(formats[i])) {
return PP_ERROR_BADARGUMENT;
}
formats_vector.push_back(formats[i]);
std::string string;
if (!PPVarToClipboardString(formats[i], data_items[i], &string))
return PP_ERROR_BADARGUMENT;
data_items_vector.push_back(string);
}
Post(BROWSER,
PpapiHostMsg_FlashClipboard_WriteData(
static_cast<uint32_t>(clipboard_type),
formats_vector,
data_items_vector));
// Assume success, since it allows us to avoid a sync IPC.
return PP_OK;
}
PP_Bool FlashClipboardResource::GetSequenceNumber(
PP_Instance instance,
PP_Flash_Clipboard_Type clipboard_type,
uint64_t* sequence_number) {
int32_t result =
SyncCall<PpapiPluginMsg_FlashClipboard_GetSequenceNumberReply>(
BROWSER,
PpapiHostMsg_FlashClipboard_GetSequenceNumber(clipboard_type),
sequence_number);
return PP_FromBool(result == PP_OK);
}
} // namespace proxy
} // namespace ppapi