| // 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 <malloc.h> |
| #endif |
| |
| #include <math.h> |
| |
| #include <vector> |
| |
| #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 "gpu/common/gpu_trace_event.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<MemoryUsageCache>::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<webkit::npapi::WebPluginInfo> 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(name, id, extra); |
| GPU_TRACE_EVENT_BEGIN2("webkit", name, |
| "id", StringPrintf("%p", id).c_str(), |
| "extra", extra ? extra : ""); |
| } |
| |
| void WebKitClientImpl::traceEventEnd(const char* name, void* id, |
| const char* extra) { |
| TRACE_EVENT_END(name, id, extra); |
| GPU_TRACE_EVENT_END0("webkit", name); |
| } |
| |
| 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<string16> 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) { |
| uint64 bytes = 0; |
| for (size_t i = 0; i < length; ++i) { |
| size_t offset = i % sizeof(bytes); |
| if (!offset) |
| bytes = base::RandUint64(); |
| buffer[i] = reinterpret_cast<unsigned char*>(&bytes)[offset]; |
| } |
| } |
| |
| 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<int64>( |
| 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<uint64_t>(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 |