// Copyright (c) 2011 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 "webkit/glue/webkitclient_impl.h" #if defined(OS_LINUX) #include #endif #include #include #include "base/debug/trace_event.h" #include "base/memory/singleton.h" #include "base/message_loop.h" #include "base/metrics/histogram.h" #include "base/metrics/stats_counters.h" #include "base/platform_file.h" #include "base/process_util.h" #include "base/rand_util.h" #include "base/string_number_conversions.h" #include "base/string_util.h" #include "base/synchronization/lock.h" #include "base/time.h" #include "base/utf_string_conversions.h" #include "grit/webkit_chromium_resources.h" #include "grit/webkit_resources.h" #include "grit/webkit_strings.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebCookie.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebData.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebFrameClient.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebPluginListBuilder.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebScreenInfo.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebString.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebVector.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebURL.h" #include "webkit/glue/media/audio_decoder.h" #include "webkit/plugins/npapi/plugin_instance.h" #include "webkit/plugins/npapi/webplugininfo.h" #include "webkit/glue/webkit_glue.h" #include "webkit/glue/websocketstreamhandle_impl.h" #include "webkit/glue/weburlloader_impl.h" #if defined(OS_LINUX) #include "v8/include/v8.h" #endif using WebKit::WebAudioBus; using WebKit::WebCookie; using WebKit::WebData; using WebKit::WebLocalizedString; using WebKit::WebPluginListBuilder; using WebKit::WebString; using WebKit::WebSocketStreamHandle; using WebKit::WebThemeEngine; using WebKit::WebURL; using WebKit::WebURLLoader; using WebKit::WebVector; namespace { // A simple class to cache the memory usage for a given amount of time. class MemoryUsageCache { public: // Retrieves the Singleton. static MemoryUsageCache* GetInstance() { return Singleton::get(); } MemoryUsageCache() : memory_value_(0) { Init(); } ~MemoryUsageCache() {} void Init() { const unsigned int kCacheSeconds = 1; cache_valid_time_ = base::TimeDelta::FromSeconds(kCacheSeconds); } // Returns true if the cached value is fresh. // Returns false if the cached value is stale, or if |cached_value| is NULL. bool IsCachedValueValid(size_t* cached_value) { base::AutoLock scoped_lock(lock_); if (!cached_value) return false; if (base::Time::Now() - last_updated_time_ > cache_valid_time_) return false; *cached_value = memory_value_; return true; }; // Setter for |memory_value_|, refreshes |last_updated_time_|. void SetMemoryValue(const size_t value) { base::AutoLock scoped_lock(lock_); memory_value_ = value; last_updated_time_ = base::Time::Now(); } private: // The cached memory value. size_t memory_value_; // How long the cached value should remain valid. base::TimeDelta cache_valid_time_; // The last time the cached value was updated. base::Time last_updated_time_; base::Lock lock_; }; } // anonymous namespace namespace webkit_glue { static int ToMessageID(WebLocalizedString::Name name) { switch (name) { case WebLocalizedString::SubmitButtonDefaultLabel: return IDS_FORM_SUBMIT_LABEL; case WebLocalizedString::InputElementAltText: return IDS_FORM_INPUT_ALT; case WebLocalizedString::ResetButtonDefaultLabel: return IDS_FORM_RESET_LABEL; case WebLocalizedString::FileButtonChooseFileLabel: return IDS_FORM_FILE_BUTTON_LABEL; case WebLocalizedString::FileButtonNoFileSelectedLabel: return IDS_FORM_FILE_NO_FILE_LABEL; case WebLocalizedString::MultipleFileUploadText: return IDS_FORM_FILE_MULTIPLE_UPLOAD; case WebLocalizedString::SearchableIndexIntroduction: return IDS_SEARCHABLE_INDEX_INTRO; case WebLocalizedString::SearchMenuNoRecentSearchesText: return IDS_RECENT_SEARCHES_NONE; case WebLocalizedString::SearchMenuRecentSearchesText: return IDS_RECENT_SEARCHES; case WebLocalizedString::SearchMenuClearRecentSearchesText: return IDS_RECENT_SEARCHES_CLEAR; case WebLocalizedString::AXWebAreaText: return IDS_AX_ROLE_WEB_AREA; case WebLocalizedString::AXLinkText: return IDS_AX_ROLE_LINK; case WebLocalizedString::AXListMarkerText: return IDS_AX_ROLE_LIST_MARKER; case WebLocalizedString::AXImageMapText: return IDS_AX_ROLE_IMAGE_MAP; case WebLocalizedString::AXHeadingText: return IDS_AX_ROLE_HEADING; case WebLocalizedString::AXButtonActionVerb: return IDS_AX_BUTTON_ACTION_VERB; case WebLocalizedString::AXRadioButtonActionVerb: return IDS_AX_RADIO_BUTTON_ACTION_VERB; case WebLocalizedString::AXTextFieldActionVerb: return IDS_AX_TEXT_FIELD_ACTION_VERB; case WebLocalizedString::AXCheckedCheckBoxActionVerb: return IDS_AX_CHECKED_CHECK_BOX_ACTION_VERB; case WebLocalizedString::AXUncheckedCheckBoxActionVerb: return IDS_AX_UNCHECKED_CHECK_BOX_ACTION_VERB; case WebLocalizedString::AXLinkActionVerb: return IDS_AX_LINK_ACTION_VERB; case WebLocalizedString::KeygenMenuHighGradeKeySize: return IDS_KEYGEN_HIGH_GRADE_KEY; case WebLocalizedString::KeygenMenuMediumGradeKeySize: return IDS_KEYGEN_MED_GRADE_KEY; case WebLocalizedString::ValidationValueMissing: return IDS_FORM_VALIDATION_VALUE_MISSING; case WebLocalizedString::ValidationValueMissingForCheckbox: return IDS_FORM_VALIDATION_VALUE_MISSING_CHECKBOX; case WebLocalizedString::ValidationValueMissingForFile: return IDS_FORM_VALIDATION_VALUE_MISSING_FILE; case WebLocalizedString::ValidationValueMissingForMultipleFile: return IDS_FORM_VALIDATION_VALUE_MISSING_MULTIPLE_FILE; case WebLocalizedString::ValidationValueMissingForRadio: return IDS_FORM_VALIDATION_VALUE_MISSING_RADIO; case WebLocalizedString::ValidationValueMissingForSelect: return IDS_FORM_VALIDATION_VALUE_MISSING_SELECT; case WebLocalizedString::ValidationTypeMismatch: return IDS_FORM_VALIDATION_TYPE_MISMATCH; case WebLocalizedString::ValidationTypeMismatchForEmail: return IDS_FORM_VALIDATION_TYPE_MISMATCH_EMAIL; case WebLocalizedString::ValidationTypeMismatchForMultipleEmail: return IDS_FORM_VALIDATION_TYPE_MISMATCH_MULTIPLE_EMAIL; case WebLocalizedString::ValidationTypeMismatchForURL: return IDS_FORM_VALIDATION_TYPE_MISMATCH_URL; case WebLocalizedString::ValidationPatternMismatch: return IDS_FORM_VALIDATION_PATTERN_MISMATCH; case WebLocalizedString::ValidationTooLong: return IDS_FORM_VALIDATION_TOO_LONG; case WebLocalizedString::ValidationRangeUnderflow: return IDS_FORM_VALIDATION_RANGE_UNDERFLOW; case WebLocalizedString::ValidationRangeOverflow: return IDS_FORM_VALIDATION_RANGE_OVERFLOW; case WebLocalizedString::ValidationStepMismatch: return IDS_FORM_VALIDATION_STEP_MISMATCH; } return -1; } WebKitClientImpl::WebKitClientImpl() : main_loop_(MessageLoop::current()), shared_timer_func_(NULL), shared_timer_fire_time_(0.0), shared_timer_suspended_(0) { } WebKitClientImpl::~WebKitClientImpl() { } WebThemeEngine* WebKitClientImpl::themeEngine() { #if defined(OS_WIN) || defined(OS_LINUX) || defined(OS_MACOSX) return &theme_engine_; #else return NULL; #endif } WebURLLoader* WebKitClientImpl::createURLLoader() { return new WebURLLoaderImpl(); } WebSocketStreamHandle* WebKitClientImpl::createSocketStreamHandle() { return new WebSocketStreamHandleImpl(); } WebString WebKitClientImpl::userAgent(const WebURL& url) { return WebString::fromUTF8(webkit_glue::GetUserAgent(url)); } void WebKitClientImpl::getPluginList(bool refresh, WebPluginListBuilder* builder) { std::vector plugins; GetPlugins(refresh, &plugins); for (size_t i = 0; i < plugins.size(); ++i) { const webkit::npapi::WebPluginInfo& plugin = plugins[i]; builder->addPlugin( plugin.name, plugin.desc, FilePathStringToWebString(plugin.path.BaseName().value())); for (size_t j = 0; j < plugin.mime_types.size(); ++j) { const webkit::npapi::WebPluginMimeType& mime_type = plugin.mime_types[j]; builder->addMediaTypeToLastPlugin( WebString::fromUTF8(mime_type.mime_type), mime_type.description); for (size_t k = 0; k < mime_type.file_extensions.size(); ++k) { builder->addFileExtensionToLastMediaType( UTF8ToUTF16(mime_type.file_extensions[k])); } } } } void WebKitClientImpl::decrementStatsCounter(const char* name) { base::StatsCounter(name).Decrement(); } void WebKitClientImpl::incrementStatsCounter(const char* name) { base::StatsCounter(name).Increment(); } void WebKitClientImpl::histogramCustomCounts( const char* name, int sample, int min, int max, int bucket_count) { // Copied from histogram macro, but without the static variable caching // the histogram because name is dynamic. base::Histogram* counter = base::Histogram::FactoryGet(name, min, max, bucket_count, base::Histogram::kUmaTargetedHistogramFlag); DCHECK_EQ(name, counter->histogram_name()); counter->Add(sample); } void WebKitClientImpl::histogramEnumeration( const char* name, int sample, int boundary_value) { // Copied from histogram macro, but without the static variable caching // the histogram because name is dynamic. base::Histogram* counter = base::LinearHistogram::FactoryGet(name, 1, boundary_value, boundary_value + 1, base::Histogram::kUmaTargetedHistogramFlag); DCHECK_EQ(name, counter->histogram_name()); counter->Add(sample); } void WebKitClientImpl::traceEventBegin(const char* name, void* id, const char* extra) { TRACE_EVENT_BEGIN_ETW(name, id, extra); } void WebKitClientImpl::traceEventEnd(const char* name, void* id, const char* extra) { TRACE_EVENT_END_ETW(name, id, extra); } namespace { WebData loadAudioSpatializationResource(const char* name) { #ifdef IDR_AUDIO_SPATIALIZATION_T000_P000 const size_t kExpectedSpatializationNameLength = 31; if (strlen(name) != kExpectedSpatializationNameLength) { return WebData(); } // Extract the azimuth and elevation from the resource name. int azimuth = 0; int elevation = 0; int values_parsed = sscanf(name, "IRC_Composite_C_R0195_T%3d_P%3d", &azimuth, &elevation); if (values_parsed != 2) { return WebData(); } // The resource index values go through the elevations first, then azimuths. const int kAngleSpacing = 15; // 0 <= elevation <= 90 (or 315 <= elevation <= 345) // in increments of 15 degrees. int elevation_index = elevation <= 90 ? elevation / kAngleSpacing : 7 + (elevation - 315) / kAngleSpacing; bool is_elevation_index_good = 0 <= elevation_index && elevation_index < 10; // 0 <= azimuth < 360 in increments of 15 degrees. int azimuth_index = azimuth / kAngleSpacing; bool is_azimuth_index_good = 0 <= azimuth_index && azimuth_index < 24; const int kNumberOfElevations = 10; const int kNumberOfAudioResources = 240; int resource_index = kNumberOfElevations * azimuth_index + elevation_index; bool is_resource_index_good = 0 <= resource_index && resource_index < kNumberOfAudioResources; if (is_azimuth_index_good && is_elevation_index_good && is_resource_index_good) { const int kFirstAudioResourceIndex = IDR_AUDIO_SPATIALIZATION_T000_P000; base::StringPiece resource = GetDataResource(kFirstAudioResourceIndex + resource_index); return WebData(resource.data(), resource.size()); } #endif // IDR_AUDIO_SPATIALIZATION_T000_P000 NOTREACHED(); return WebData(); } } // namespace WebData WebKitClientImpl::loadResource(const char* name) { struct { const char* name; int id; } resources[] = { { "missingImage", IDR_BROKENIMAGE }, { "mediaPause", IDR_MEDIA_PAUSE_BUTTON }, { "mediaPlay", IDR_MEDIA_PLAY_BUTTON }, { "mediaPlayDisabled", IDR_MEDIA_PLAY_BUTTON_DISABLED }, { "mediaSoundDisabled", IDR_MEDIA_SOUND_DISABLED }, { "mediaSoundFull", IDR_MEDIA_SOUND_FULL_BUTTON }, { "mediaSoundNone", IDR_MEDIA_SOUND_NONE_BUTTON }, { "mediaSliderThumb", IDR_MEDIA_SLIDER_THUMB }, { "mediaVolumeSliderThumb", IDR_MEDIA_VOLUME_SLIDER_THUMB }, { "panIcon", IDR_PAN_SCROLL_ICON }, { "searchCancel", IDR_SEARCH_CANCEL }, { "searchCancelPressed", IDR_SEARCH_CANCEL_PRESSED }, { "searchMagnifier", IDR_SEARCH_MAGNIFIER }, { "searchMagnifierResults", IDR_SEARCH_MAGNIFIER_RESULTS }, { "textAreaResizeCorner", IDR_TEXTAREA_RESIZER }, { "tickmarkDash", IDR_TICKMARK_DASH }, { "inputSpeech", IDR_INPUT_SPEECH }, { "inputSpeechRecording", IDR_INPUT_SPEECH_RECORDING }, { "inputSpeechWaiting", IDR_INPUT_SPEECH_WAITING }, { "americanExpressCC", IDR_AUTOFILL_CC_AMEX }, { "dinersCC", IDR_AUTOFILL_CC_DINERS }, { "discoverCC", IDR_AUTOFILL_CC_DISCOVER }, { "genericCC", IDR_AUTOFILL_CC_GENERIC }, { "jcbCC", IDR_AUTOFILL_CC_JCB }, { "masterCardCC", IDR_AUTOFILL_CC_MASTERCARD }, { "soloCC", IDR_AUTOFILL_CC_SOLO }, { "visaCC", IDR_AUTOFILL_CC_VISA }, }; // Check the name prefix to see if it's an audio resource. if (StartsWithASCII(name, "IRC_Composite", true)) { return loadAudioSpatializationResource(name); } else { for (size_t i = 0; i < ARRAYSIZE_UNSAFE(resources); ++i) { if (!strcmp(name, resources[i].name)) { base::StringPiece resource = GetDataResource(resources[i].id); return WebData(resource.data(), resource.size()); } } } // TODO(jhawkins): Restore this NOTREACHED once WK stops sending in empty // strings. http://crbug.com/50675. //NOTREACHED() << "Unknown image resource " << name; return WebData(); } bool WebKitClientImpl::loadAudioResource( WebKit::WebAudioBus* destination_bus, const char* audio_file_data, size_t data_size, double sample_rate) { return DecodeAudioFileData(destination_bus, audio_file_data, data_size, sample_rate); } WebString WebKitClientImpl::queryLocalizedString( WebLocalizedString::Name name) { int message_id = ToMessageID(name); if (message_id < 0) return WebString(); return GetLocalizedString(message_id); } WebString WebKitClientImpl::queryLocalizedString( WebLocalizedString::Name name, int numeric_value) { return queryLocalizedString(name, base::IntToString16(numeric_value)); } WebString WebKitClientImpl::queryLocalizedString( WebLocalizedString::Name name, const WebString& value) { int message_id = ToMessageID(name); if (message_id < 0) return WebString(); return ReplaceStringPlaceholders(GetLocalizedString(message_id), value, NULL); } WebString WebKitClientImpl::queryLocalizedString( WebLocalizedString::Name name, const WebString& value1, const WebString& value2) { int message_id = ToMessageID(name); if (message_id < 0) return WebString(); std::vector values; values.reserve(2); values.push_back(value1); values.push_back(value2); return ReplaceStringPlaceholders( GetLocalizedString(message_id), values, NULL); } double WebKitClientImpl::currentTime() { return base::Time::Now().ToDoubleT(); } void WebKitClientImpl::cryptographicallyRandomValues( unsigned char* buffer, size_t length) { base::RandBytes(buffer, length); } void WebKitClientImpl::setSharedTimerFiredFunction(void (*func)()) { shared_timer_func_ = func; } void WebKitClientImpl::setSharedTimerFireTime(double fire_time) { shared_timer_fire_time_ = fire_time; if (shared_timer_suspended_) return; // By converting between double and int64 representation, we run the risk // of losing precision due to rounding errors. Performing computations in // microseconds reduces this risk somewhat. But there still is the potential // of us computing a fire time for the timer that is shorter than what we // need. // As the event loop will check event deadlines prior to actually firing // them, there is a risk of needlessly rescheduling events and of // needlessly looping if sleep times are too short even by small amounts. // This results in measurable performance degradation unless we use ceil() to // always round up the sleep times. int64 interval = static_cast( ceil((fire_time - currentTime()) * base::Time::kMicrosecondsPerSecond)); if (interval < 0) interval = 0; shared_timer_.Stop(); shared_timer_.Start(base::TimeDelta::FromMicroseconds(interval), this, &WebKitClientImpl::DoTimeout); } void WebKitClientImpl::stopSharedTimer() { shared_timer_.Stop(); } void WebKitClientImpl::callOnMainThread(void (*func)(void*), void* context) { main_loop_->PostTask(FROM_HERE, NewRunnableFunction(func, context)); } base::PlatformFile WebKitClientImpl::databaseOpenFile( const WebKit::WebString& vfs_file_name, int desired_flags) { return base::kInvalidPlatformFileValue; } int WebKitClientImpl::databaseDeleteFile( const WebKit::WebString& vfs_file_name, bool sync_dir) { return -1; } long WebKitClientImpl::databaseGetFileAttributes( const WebKit::WebString& vfs_file_name) { return 0; } long long WebKitClientImpl::databaseGetFileSize( const WebKit::WebString& vfs_file_name) { return 0; } WebKit::WebString WebKitClientImpl::signedPublicKeyAndChallengeString( unsigned key_size_index, const WebKit::WebString& challenge, const WebKit::WebURL& url) { NOTREACHED(); return WebKit::WebString(); } #if defined(OS_LINUX) static size_t memoryUsageMBLinux() { struct mallinfo minfo = mallinfo(); uint64_t mem_usage = #if defined(USE_TCMALLOC) minfo.uordblks #else (minfo.hblkhd + minfo.arena) #endif >> 20; v8::HeapStatistics stat; v8::V8::GetHeapStatistics(&stat); return mem_usage + (static_cast(stat.total_heap_size()) >> 20); } #endif #if defined(OS_MACOSX) static size_t memoryUsageMBMac() { using base::ProcessMetrics; static ProcessMetrics* process_metrics = // The default port provider is sufficient to get data for the current // process. ProcessMetrics::CreateProcessMetrics(base::GetCurrentProcessHandle(), NULL); DCHECK(process_metrics); return process_metrics->GetPagefileUsage() >> 20; } #endif #if !defined(OS_LINUX) && !defined(OS_MACOSX) static size_t memoryUsageMBGeneric() { using base::ProcessMetrics; static ProcessMetrics* process_metrics = ProcessMetrics::CreateProcessMetrics(base::GetCurrentProcessHandle()); DCHECK(process_metrics); return process_metrics->GetPagefileUsage() >> 20; } #endif static size_t getMemoryUsageMB(bool bypass_cache) { size_t current_mem_usage = 0; MemoryUsageCache* mem_usage_cache_singleton = MemoryUsageCache::GetInstance(); if (!bypass_cache && mem_usage_cache_singleton->IsCachedValueValid(¤t_mem_usage)) return current_mem_usage; current_mem_usage = #if defined(OS_LINUX) memoryUsageMBLinux(); #elif defined(OS_MACOSX) memoryUsageMBMac(); #else memoryUsageMBGeneric(); #endif mem_usage_cache_singleton->SetMemoryValue(current_mem_usage); return current_mem_usage; } size_t WebKitClientImpl::memoryUsageMB() { return getMemoryUsageMB(false); } size_t WebKitClientImpl::actualMemoryUsageMB() { return getMemoryUsageMB(true); } void WebKitClientImpl::SuspendSharedTimer() { ++shared_timer_suspended_; } void WebKitClientImpl::ResumeSharedTimer() { // The shared timer may have fired or been adjusted while we were suspended. if (--shared_timer_suspended_ == 0 && !shared_timer_.IsRunning()) setSharedTimerFireTime(shared_timer_fire_time_); } } // namespace webkit_glue