// Copyright 2014 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 "extensions/renderer/set_icon_natives.h" #include #include "base/memory/scoped_ptr.h" #include "content/public/common/common_param_traits.h" #include "extensions/renderer/request_sender.h" #include "extensions/renderer/script_context.h" #include "ipc/ipc_message_utils.h" #include "third_party/WebKit/public/web/WebArrayBufferConverter.h" #include "third_party/skia/include/core/SkBitmap.h" #include "ui/gfx/ipc/gfx_param_traits.h" namespace { const char* kImageSizeKeys[] = {"19", "38"}; const char kInvalidDimensions[] = "ImageData has invalid dimensions."; const char kInvalidData[] = "ImageData data length does not match dimensions."; const char kNoMemory[] = "Chrome was unable to initialize icon."; } // namespace namespace extensions { SetIconNatives::SetIconNatives(RequestSender* request_sender, ScriptContext* context) : ObjectBackedNativeHandler(context), request_sender_(request_sender) { RouteFunction( "SetIconCommon", base::Bind(&SetIconNatives::SetIconCommon, base::Unretained(this))); } bool SetIconNatives::ConvertImageDataToBitmapValue( const v8::Local image_data, v8::Local* image_data_bitmap) { v8::Isolate* isolate = context()->v8_context()->GetIsolate(); v8::Local data = image_data->Get(v8::String::NewFromUtf8(isolate, "data"))->ToObject(); int width = image_data->Get(v8::String::NewFromUtf8(isolate, "width"))->Int32Value(); int height = image_data->Get(v8::String::NewFromUtf8(isolate, "height"))->Int32Value(); if (width <= 0 || height <= 0) { isolate->ThrowException(v8::Exception::Error( v8::String::NewFromUtf8(isolate, kInvalidDimensions))); return false; } // We need to be able to safely check |data_length| == 4 * width * height // without overflowing below. int max_width = (std::numeric_limits::max() / 4) / height; if (width > max_width) { isolate->ThrowException(v8::Exception::Error( v8::String::NewFromUtf8(isolate, kInvalidDimensions))); return false; } int data_length = data->Get(v8::String::NewFromUtf8(isolate, "length"))->Int32Value(); if (data_length != 4 * width * height) { isolate->ThrowException( v8::Exception::Error(v8::String::NewFromUtf8(isolate, kInvalidData))); return false; } SkBitmap bitmap; if (!bitmap.allocN32Pixels(width, height)) { isolate->ThrowException( v8::Exception::Error(v8::String::NewFromUtf8(isolate, kNoMemory))); return false; } bitmap.eraseARGB(0, 0, 0, 0); uint32_t* pixels = bitmap.getAddr32(0, 0); for (int t = 0; t < width * height; t++) { // |data| is RGBA, pixels is ARGB. pixels[t] = SkPreMultiplyColor( ((data->Get(v8::Integer::New(isolate, 4 * t + 3))->Int32Value() & 0xFF) << 24) | ((data->Get(v8::Integer::New(isolate, 4 * t + 0))->Int32Value() & 0xFF) << 16) | ((data->Get(v8::Integer::New(isolate, 4 * t + 1))->Int32Value() & 0xFF) << 8) | ((data->Get(v8::Integer::New(isolate, 4 * t + 2))->Int32Value() & 0xFF) << 0)); } // Construct the Value object. IPC::Message bitmap_pickle; IPC::WriteParam(&bitmap_pickle, bitmap); blink::WebArrayBuffer buffer = blink::WebArrayBuffer::create(bitmap_pickle.size(), 1); memcpy(buffer.data(), bitmap_pickle.data(), bitmap_pickle.size()); *image_data_bitmap = blink::WebArrayBufferConverter::toV8Value( &buffer, context()->v8_context()->Global(), isolate); return true; } bool SetIconNatives::ConvertImageDataSetToBitmapValueSet( v8::Local& details, v8::Local* bitmap_set_value) { v8::Isolate* isolate = context()->v8_context()->GetIsolate(); v8::Local image_data_set = details->Get(v8::String::NewFromUtf8(isolate, "imageData"))->ToObject(); DCHECK(bitmap_set_value); for (size_t i = 0; i < arraysize(kImageSizeKeys); i++) { if (!image_data_set->Has( v8::String::NewFromUtf8(isolate, kImageSizeKeys[i]))) continue; v8::Local image_data = image_data_set->Get(v8::String::NewFromUtf8(isolate, kImageSizeKeys[i])) ->ToObject(); v8::Local image_data_bitmap; if (!ConvertImageDataToBitmapValue(image_data, &image_data_bitmap)) return false; (*bitmap_set_value)->Set( v8::String::NewFromUtf8(isolate, kImageSizeKeys[i]), image_data_bitmap); } return true; } void SetIconNatives::SetIconCommon( const v8::FunctionCallbackInfo& args) { CHECK_EQ(1, args.Length()); CHECK(args[0]->IsObject()); v8::Local details = args[0]->ToObject(); v8::Local bitmap_set_value(v8::Object::New(args.GetIsolate())); if (!ConvertImageDataSetToBitmapValueSet(details, &bitmap_set_value)) return; v8::Local dict(v8::Object::New(args.GetIsolate())); dict->Set(v8::String::NewFromUtf8(args.GetIsolate(), "imageData"), bitmap_set_value); if (details->Has(v8::String::NewFromUtf8(args.GetIsolate(), "tabId"))) { dict->Set( v8::String::NewFromUtf8(args.GetIsolate(), "tabId"), details->Get(v8::String::NewFromUtf8(args.GetIsolate(), "tabId"))); } args.GetReturnValue().Set(dict); } } // namespace extensions