// 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_resource.h"
#include <cmath>
#include "base/containers/mru_cache.h"
#include "base/debug/crash_logging.h"
#include "base/lazy_instance.h"
#include "base/time/time.h"
#include "ppapi/c/pp_errors.h"
#include "ppapi/c/private/ppb_flash.h"
#include "ppapi/c/trusted/ppb_browser_font_trusted.h"
#include "ppapi/proxy/plugin_dispatcher.h"
#include "ppapi/proxy/plugin_globals.h"
#include "ppapi/proxy/ppapi_messages.h"
#include "ppapi/proxy/serialized_structs.h"
#include "ppapi/shared_impl/ppapi_preferences.h"
#include "ppapi/shared_impl/scoped_pp_var.h"
#include "ppapi/shared_impl/time_conversion.h"
#include "ppapi/shared_impl/var.h"
#include "ppapi/thunk/enter.h"
#include "ppapi/thunk/ppb_url_request_info_api.h"
using ppapi::thunk::EnterResourceNoLock;
namespace ppapi {
namespace proxy {
namespace {
struct LocalTimeZoneOffsetEntry {
base::TimeTicks expiration;
double offset;
};
class LocalTimeZoneOffsetCache
: public base::MRUCache<PP_Time, LocalTimeZoneOffsetEntry> {
public:
LocalTimeZoneOffsetCache()
: base::MRUCache<PP_Time, LocalTimeZoneOffsetEntry>(kCacheSize) {}
private:
static const size_t kCacheSize = 100;
};
base::LazyInstance<LocalTimeZoneOffsetCache>::Leaky
g_local_time_zone_offset_cache = LAZY_INSTANCE_INITIALIZER;
} // namespace
FlashResource::FlashResource(Connection connection,
PP_Instance instance,
PluginDispatcher* plugin_dispatcher)
: PluginResource(connection, instance),
plugin_dispatcher_(plugin_dispatcher) {
SendCreate(RENDERER, PpapiHostMsg_Flash_Create());
SendCreate(BROWSER, PpapiHostMsg_Flash_Create());
}
FlashResource::~FlashResource() {
}
thunk::PPB_Flash_Functions_API* FlashResource::AsPPB_Flash_Functions_API() {
return this;
}
PP_Var FlashResource::GetProxyForURL(PP_Instance instance,
const std::string& url) {
std::string proxy;
int32_t result = SyncCall<PpapiPluginMsg_Flash_GetProxyForURLReply>(RENDERER,
PpapiHostMsg_Flash_GetProxyForURL(url), &proxy);
if (result == PP_OK)
return StringVar::StringToPPVar(proxy);
return PP_MakeUndefined();
}
void FlashResource::UpdateActivity(PP_Instance instance) {
Post(BROWSER, PpapiHostMsg_Flash_UpdateActivity());
}
PP_Bool FlashResource::SetCrashData(PP_Instance instance,
PP_FlashCrashKey key,
PP_Var value) {
StringVar* url_string_var(StringVar::FromPPVar(value));
if (!url_string_var)
return PP_FALSE;
switch (key) {
case PP_FLASHCRASHKEY_URL: {
PluginGlobals::Get()->SetActiveURL(url_string_var->value());
return PP_TRUE;
}
case PP_FLASHCRASHKEY_RESOURCE_URL: {
base::debug::SetCrashKeyValue("subresource_url", url_string_var->value());
return PP_TRUE;
}
}
return PP_FALSE;
}
double FlashResource::GetLocalTimeZoneOffset(PP_Instance instance,
PP_Time t) {
LocalTimeZoneOffsetCache& cache = g_local_time_zone_offset_cache.Get();
// Get the minimum PP_Time value that shares the same minute as |t|.
// Use cached offset if cache hasn't expired and |t| is in the same minute as
// the time for the cached offset (assume offsets change on minute
// boundaries).
PP_Time t_minute_base = floor(t / 60.0) * 60.0;
LocalTimeZoneOffsetCache::iterator iter = cache.Get(t_minute_base);
base::TimeTicks now = base::TimeTicks::Now();
if (iter != cache.end() && now < iter->second.expiration)
return iter->second.offset;
// Cache the local offset for ten seconds, since it's slow on XP and Linux.
// Note that TimeTicks does not continue counting across sleep/resume on all
// platforms. This may be acceptable for 10 seconds, but if in the future this
// is changed to one minute or more, then we should consider using base::Time.
const int64 kMaxCachedLocalOffsetAgeInSeconds = 10;
base::TimeDelta expiration_delta =
base::TimeDelta::FromSeconds(kMaxCachedLocalOffsetAgeInSeconds);
LocalTimeZoneOffsetEntry cache_entry;
cache_entry.expiration = now + expiration_delta;
cache_entry.offset = 0.0;
// We can't do the conversion here on Linux because the localtime calls
// require filesystem access prohibited by the sandbox.
// TODO(shess): Figure out why OSX needs the access, the sandbox warmup should
// handle it. http://crbug.com/149006
#if defined(OS_LINUX) || defined(OS_MACOSX)
int32_t result = SyncCall<PpapiPluginMsg_Flash_GetLocalTimeZoneOffsetReply>(
BROWSER,
PpapiHostMsg_Flash_GetLocalTimeZoneOffset(PPTimeToTime(t)),
&cache_entry.offset);
if (result != PP_OK)
cache_entry.offset = 0.0;
#else
cache_entry.offset = PPGetLocalTimeZoneOffset(PPTimeToTime(t));
#endif
cache.Put(t_minute_base, cache_entry);
return cache_entry.offset;
}
PP_Var FlashResource::GetSetting(PP_Instance instance,
PP_FlashSetting setting) {
switch (setting) {
case PP_FLASHSETTING_3DENABLED:
return PP_MakeBool(PP_FromBool(
plugin_dispatcher_->preferences().is_3d_supported));
case PP_FLASHSETTING_INCOGNITO:
return PP_MakeBool(PP_FromBool(plugin_dispatcher_->incognito()));
case PP_FLASHSETTING_STAGE3DENABLED:
return PP_MakeBool(PP_FromBool(
plugin_dispatcher_->preferences().is_stage3d_supported));
case PP_FLASHSETTING_STAGE3DBASELINEENABLED:
return PP_MakeBool(PP_FromBool(
plugin_dispatcher_->preferences().is_stage3d_baseline_supported));
case PP_FLASHSETTING_LANGUAGE:
return StringVar::StringToPPVar(
PluginGlobals::Get()->GetUILanguage());
case PP_FLASHSETTING_NUMCORES:
return PP_MakeInt32(
plugin_dispatcher_->preferences().number_of_cpu_cores);
case PP_FLASHSETTING_LSORESTRICTIONS: {
int32_t restrictions;
int32_t result =
SyncCall<PpapiPluginMsg_Flash_GetLocalDataRestrictionsReply>(BROWSER,
PpapiHostMsg_Flash_GetLocalDataRestrictions(), &restrictions);
if (result != PP_OK)
return PP_MakeInt32(PP_FLASHLSORESTRICTIONS_NONE);
return PP_MakeInt32(restrictions);
}
}
return PP_MakeUndefined();
}
void FlashResource::SetInstanceAlwaysOnTop(PP_Instance instance,
PP_Bool on_top) {
Post(RENDERER, PpapiHostMsg_Flash_SetInstanceAlwaysOnTop(PP_ToBool(on_top)));
}
PP_Bool FlashResource::DrawGlyphs(
PP_Instance instance,
PP_Resource pp_image_data,
const PP_BrowserFont_Trusted_Description* font_desc,
uint32_t color,
const PP_Point* position,
const PP_Rect* clip,
const float transformation[3][3],
PP_Bool allow_subpixel_aa,
uint32_t glyph_count,
const uint16_t glyph_indices[],
const PP_Point glyph_advances[]) {
EnterResourceNoLock<thunk::PPB_ImageData_API> enter(pp_image_data, true);
if (enter.failed())
return PP_FALSE;
// The instance parameter isn't strictly necessary but we check that it
// matches anyway.
if (enter.resource()->pp_instance() != instance)
return PP_FALSE;
PPBFlash_DrawGlyphs_Params params;
params.image_data = enter.resource()->host_resource();
params.font_desc.SetFromPPBrowserFontDescription(*font_desc);
params.color = color;
params.position = *position;
params.clip = *clip;
for (int i = 0; i < 3; i++) {
for (int j = 0; j < 3; j++)
params.transformation[i][j] = transformation[i][j];
}
params.allow_subpixel_aa = allow_subpixel_aa;
params.glyph_indices.insert(params.glyph_indices.begin(),
&glyph_indices[0],
&glyph_indices[glyph_count]);
params.glyph_advances.insert(params.glyph_advances.begin(),
&glyph_advances[0],
&glyph_advances[glyph_count]);
// This has to be synchronous because the caller may want to composite on
// top of the resulting text after the call is complete.
int32_t result = SyncCall<IPC::Message>(RENDERER,
PpapiHostMsg_Flash_DrawGlyphs(params));
return PP_FromBool(result == PP_OK);
}
int32_t FlashResource::Navigate(PP_Instance instance,
PP_Resource request_info,
const char* target,
PP_Bool from_user_action) {
EnterResourceNoLock<thunk::PPB_URLRequestInfo_API> enter(request_info,
true);
if (enter.failed())
return PP_ERROR_BADRESOURCE;
return SyncCall<IPC::Message>(RENDERER, PpapiHostMsg_Flash_Navigate(
enter.object()->GetData(), target, PP_ToBool(from_user_action)));
}
PP_Bool FlashResource::IsRectTopmost(PP_Instance instance,
const PP_Rect* rect) {
int32_t result = SyncCall<IPC::Message>(RENDERER,
PpapiHostMsg_Flash_IsRectTopmost(*rect));
return PP_FromBool(result == PP_OK);
}
void FlashResource::InvokePrinting(PP_Instance instance) {
Post(RENDERER, PpapiHostMsg_Flash_InvokePrinting());
}
} // namespace proxy
} // namespace ppapi