// Copyright 2010 the V8 project authors. All rights reserved. // Redistribution and use in source and binary forms, with or without // modification, are permitted provided that the following conditions are // met: // // * Redistributions of source code must retain the above copyright // notice, this list of conditions and the following disclaimer. // * Redistributions in binary form must reproduce the above // copyright notice, this list of conditions and the following // disclaimer in the documentation and/or other materials provided // with the distribution. // * Neither the name of Google Inc. nor the names of its // contributors may be used to endorse or promote products derived // from this software without specific prior written permission. // // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. #ifndef V8_RUNTIME_PROFILER_H_ #define V8_RUNTIME_PROFILER_H_ #include "allocation.h" #include "atomicops.h" namespace v8 { namespace internal { class Isolate; class JSFunction; class Object; class PendingListNode; class Semaphore; class RuntimeProfiler { public: explicit RuntimeProfiler(Isolate* isolate); static void GlobalSetup(); static inline bool IsEnabled() { ASSERT(has_been_globally_setup_); return enabled_; } void OptimizeNow(); void OptimizeSoon(JSFunction* function); void NotifyTick(); void Setup(); void Reset(); void TearDown(); Object** SamplerWindowAddress(); int SamplerWindowSize(); // Rate limiting support. // VM thread interface. // // Called by isolates when their states change. static inline void IsolateEnteredJS(Isolate* isolate); static inline void IsolateExitedJS(Isolate* isolate); // Profiler thread interface. // // IsSomeIsolateInJS(): // The profiler thread can query whether some isolate is currently // running JavaScript code. // // WaitForSomeIsolateToEnterJS(): // When no isolates are running JavaScript code for some time the // profiler thread suspends itself by calling the wait function. The // wait function returns true after it waited or false immediately. // While the function was waiting the profiler may have been // disabled so it *must check* whether it is allowed to continue. static bool IsSomeIsolateInJS(); static bool WaitForSomeIsolateToEnterJS(); // When shutting down we join the profiler thread. Doing so while // it's waiting on a semaphore will cause a deadlock, so we have to // wake it up first. static void WakeUpRuntimeProfilerThreadBeforeShutdown(); void UpdateSamplesAfterScavenge(); void RemoveDeadSamples(); void UpdateSamplesAfterCompact(ObjectVisitor* visitor); private: static const int kSamplerWindowSize = 16; static const int kStateWindowSize = 128; enum SamplerState { IN_NON_JS_STATE = 0, IN_JS_STATE = 1 }; static void HandleWakeUp(Isolate* isolate); void Optimize(JSFunction* function, bool eager, int delay); void AttemptOnStackReplacement(JSFunction* function); void ClearSampleBuffer(); void ClearSampleBufferNewSpaceEntries(); int LookupSample(JSFunction* function); void AddSample(JSFunction* function, int weight); #ifdef ENABLE_LOGGING_AND_PROFILING void UpdateStateRatio(SamplerState current_state); #endif Isolate* isolate_; int sampler_threshold_; int sampler_threshold_size_factor_; int sampler_ticks_until_threshold_adjustment_; // The ratio of ticks spent in JS code in percent. Atomic32 js_ratio_; Object* sampler_window_[kSamplerWindowSize]; int sampler_window_position_; int sampler_window_weight_[kSamplerWindowSize]; // Support for pending 'optimize soon' requests. PendingListNode* optimize_soon_list_; SamplerState state_window_[kStateWindowSize]; int state_window_position_; int state_window_ticks_; int state_counts_[2]; // Possible state values: // -1 => the profiler thread is waiting on the semaphore // 0 or positive => the number of isolates running JavaScript code. static Atomic32 state_; static Semaphore* semaphore_; #ifdef DEBUG static bool has_been_globally_setup_; #endif static bool enabled_; }; // Rate limiter intended to be used in the profiler thread. class RuntimeProfilerRateLimiter BASE_EMBEDDED { public: RuntimeProfilerRateLimiter() : non_js_ticks_(0) { } // Suspends the current thread (which must be the profiler thread) // when not executing JavaScript to minimize CPU usage. Returns // whether the thread was suspended (and so must check whether // profiling is still active.) // // Does nothing when runtime profiling is not enabled. bool SuspendIfNecessary(); private: int non_js_ticks_; DISALLOW_COPY_AND_ASSIGN(RuntimeProfilerRateLimiter); }; // Implementation of RuntimeProfiler inline functions. void RuntimeProfiler::IsolateEnteredJS(Isolate* isolate) { Atomic32 new_state = NoBarrier_AtomicIncrement(&state_, 1); if (new_state == 0) { // Just incremented from -1 to 0. -1 can only be set by the // profiler thread before it suspends itself and starts waiting on // the semaphore. HandleWakeUp(isolate); } ASSERT(new_state >= 0); } void RuntimeProfiler::IsolateExitedJS(Isolate* isolate) { Atomic32 new_state = NoBarrier_AtomicIncrement(&state_, -1); ASSERT(new_state >= 0); USE(new_state); } } } // namespace v8::internal #endif // V8_RUNTIME_PROFILER_H_