Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(104)

Side by Side Diff: test/cctest/test-api.cc

Issue 16578008: Improved function entry hook coverage (Closed) Base URL: https://chromium.googlesource.com/external/v8.git@post_fix
Patch Set: Remove reliance on space->Contains check, which is only valid for V8-allocated memory. Go to unsign… Created 7 years, 6 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« src/ic.cc ('K') | « src/x64/code-stubs-x64.cc ('k') | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 // Copyright 2012 the V8 project authors. All rights reserved. 1 // Copyright 2012 the V8 project authors. All rights reserved.
2 // Redistribution and use in source and binary forms, with or without 2 // Redistribution and use in source and binary forms, with or without
3 // modification, are permitted provided that the following conditions are 3 // modification, are permitted provided that the following conditions are
4 // met: 4 // met:
5 // 5 //
6 // * Redistributions of source code must retain the above copyright 6 // * Redistributions of source code must retain the above copyright
7 // notice, this list of conditions and the following disclaimer. 7 // notice, this list of conditions and the following disclaimer.
8 // * Redistributions in binary form must reproduce the above 8 // * Redistributions in binary form must reproduce the above
9 // copyright notice, this list of conditions and the following 9 // copyright notice, this list of conditions and the following
10 // disclaimer in the documentation and/or other materials provided 10 // disclaimer in the documentation and/or other materials provided
(...skipping 13 matching lines...) Expand all
24 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 24 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 25 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 26 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 27
28 #include <limits.h> 28 #include <limits.h>
29 29
30 #ifndef WIN32 30 #ifndef WIN32
31 #include <signal.h> // kill 31 #include <signal.h> // kill
32 #include <unistd.h> // getpid 32 #include <unistd.h> // getpid
33 #endif // WIN32 33 #endif // WIN32
34 #include <string>
35 #include <map>
34 36
35 #include "v8.h" 37 #include "v8.h"
36 38
37 #include "api.h" 39 #include "api.h"
38 #include "arguments.h" 40 #include "arguments.h"
39 #include "isolate.h" 41 #include "isolate.h"
40 #include "compilation-cache.h" 42 #include "compilation-cache.h"
41 #include "execution.h" 43 #include "execution.h"
42 #include "objects.h" 44 #include "objects.h"
43 #include "snapshot.h" 45 #include "snapshot.h"
(...skipping 12221 matching lines...) Expand 10 before | Expand all | Expand 10 after
12265 v8::HandleScope outer(isolate); 12267 v8::HandleScope outer(isolate);
12266 v8::Local<Context> env = Context::New(isolate); 12268 v8::Local<Context> env = Context::New(isolate);
12267 env->Enter(); 12269 env->Enter();
12268 v8::Handle<Value> value = NestedScope(env); 12270 v8::Handle<Value> value = NestedScope(env);
12269 v8::Handle<String> str(value->ToString()); 12271 v8::Handle<String> str(value->ToString());
12270 CHECK(!str.IsEmpty()); 12272 CHECK(!str.IsEmpty());
12271 env->Exit(); 12273 env->Exit();
12272 } 12274 }
12273 12275
12274 12276
12275 static i::Handle<i::JSFunction>* foo_ptr = NULL; 12277 static bool MatchPointers(void* key1, void* key2) {
12276 static int foo_entry_count = 0; 12278 return key1 == key2;
12277 static i::Handle<i::JSFunction>* bar_ptr = NULL; 12279 }
12278 static int bar_entry_count = 0; 12280
12279 static int bar_caller_count = 0; 12281
12280 12282 struct SymbolInfo {
12281 12283 size_t id;
12282 static void entry_hook(uintptr_t function, 12284 size_t size;
12283 uintptr_t return_addr_location) { 12285 std::string name;
12284 i::Code* code = i::Code::GetCodeFromTargetAddress( 12286 };
12287
12288
12289 class SetFunctionEntryHookTest {
12290 public:
12291 SetFunctionEntryHookTest() {
12292 CHECK(instance_ == NULL);
12293 instance_ = this;
12294 }
12295 ~SetFunctionEntryHookTest() {
12296 CHECK(instance_ == this);
12297 instance_ = NULL;
12298 }
12299 void Reset() {
12300 symbols_.clear();
12301 symbol_locations_.clear();
12302 invocations_.clear();
12303 }
12304 void RunTest();
12305 void OnJitEvent(const v8::JitCodeEvent* event);
12306 static void JitEvent(const v8::JitCodeEvent* event) {
12307 CHECK(instance_ != NULL);
12308 instance_->OnJitEvent(event);
12309 }
12310
12311 void OnEntryHook(uintptr_t function,
12312 uintptr_t return_addr_location);
12313 static void EntryHook(uintptr_t function,
12314 uintptr_t return_addr_location) {
12315 CHECK(instance_ != NULL);
12316 instance_->OnEntryHook(function, return_addr_location);
12317 }
12318
12319 v8::Handle<Value> OnRuntimeCallback(const v8::Arguments& args);
12320 static v8::Handle<Value> RuntimeCallback(const v8::Arguments& args) {
12321 CHECK(instance_ != NULL);
12322 return instance_->OnRuntimeCallback(args);
12323 }
12324 void RunLoopInNewEnv(v8::Isolate* isolate);
12325
12326 // Records addr as location of symbol.
12327 void InsertSymbolAt(i::Address addr, SymbolInfo* symbol);
12328
12329 // Finds the symbol containing addr
12330 SymbolInfo* FindSymbolForAddr(i::Address addr);
12331 // Returns the number of invocations where the caller name contains
12332 // \p caller_name and the function name contains \p function_name.
12333 size_t CountInvocations(const char* caller_name,
12334 const char* function_name);
12335
12336 i::Handle<i::JSFunction> foo_func_;
12337 i::Handle<i::JSFunction> bar_func_;
12338
12339 typedef std::map<size_t, SymbolInfo> SymbolMap;
12340 typedef std::map<i::Address, SymbolInfo*> SymbolLocationMap;
12341 typedef std::map<std::pair<SymbolInfo*, SymbolInfo*>, size_t> InvocationMap;
12342 SymbolMap symbols_;
12343 SymbolLocationMap symbol_locations_;
12344 InvocationMap invocations_;
12345
12346 static SetFunctionEntryHookTest* instance_;
12347 };
12348 SetFunctionEntryHookTest* SetFunctionEntryHookTest::instance_ = NULL;
12349
12350
12351 // Returns true if addr is in the range [start, start+len).
12352 static bool Overlaps(i::Address start, size_t len, i::Address addr) {
12353 if (start <= addr && start + len > addr)
12354 return true;
12355
12356 return false;
12357 }
12358
12359 void SetFunctionEntryHookTest::InsertSymbolAt(i::Address addr,
12360 SymbolInfo* symbol) {
12361 // Insert the symbol at the new location.
12362 SymbolLocationMap::iterator it =
12363 symbol_locations_.insert(std::make_pair(addr, symbol)).first;
12364 // Now erase symbols to the left and right that overlap this one.
12365 while (it != symbol_locations_.begin()) {
12366 SymbolLocationMap::iterator left = it;
12367 --left;
12368 if (!Overlaps(left->first, left->second->size, addr))
12369 break;
12370 symbol_locations_.erase(left);
12371 }
12372
12373 // Now erase symbols to the left and right that overlap this one.
12374 while (true) {
12375 SymbolLocationMap::iterator right = it;
12376 ++right;
12377 if (right == symbol_locations_.end())
12378 break;
12379 if (!Overlaps(addr, symbol->size, right->first))
12380 break;
12381 symbol_locations_.erase(right);
12382 }
12383 }
12384
12385
12386 void SetFunctionEntryHookTest::OnJitEvent(const v8::JitCodeEvent* event) {
12387 switch (event->type) {
12388 case v8::JitCodeEvent::CODE_ADDED: {
12389 CHECK(event->code_start != NULL);
12390 CHECK_NE(0, static_cast<int>(event->code_len));
12391 CHECK(event->name.str != NULL);
12392 size_t symbol_id = symbols_.size();
12393
12394 // Record the new symbol.
12395 SymbolInfo& info = symbols_[symbol_id];
12396 info.id = symbol_id;
12397 info.size = event->code_len;
12398 info.name.assign(event->name.str, event->name.str + event->name.len);
12399
12400 // And record it's location.
12401 InsertSymbolAt(reinterpret_cast<i::Address>(event->code_start), &info);
12402 }
12403 break;
12404
12405 case v8::JitCodeEvent::CODE_MOVED: {
12406 // We would like to never see code move that we haven't seen before,
12407 // but the code creation event does not happen until the line endings
12408 // have been calculated (this is so that we can report the line in the
12409 // script at which the function source is found, see
12410 // Compiler::RecordFunctionCompilation) and the line endings
12411 // calculations can cause a GC, which can move the newly created code
12412 // before its existence can be logged.
12413 SymbolLocationMap::iterator it(
12414 symbol_locations_.find(
12415 reinterpret_cast<i::Address>(event->code_start)));
12416 if (it != symbol_locations_.end()) {
12417 // Found a symbol at this location, move it.
12418 SymbolInfo* info = it->second;
12419 symbol_locations_.erase(it);
12420 InsertSymbolAt(reinterpret_cast<i::Address>(event->new_code_start),
12421 info);
12422 }
12423 }
12424 default:
12425 break;
12426 }
12427 }
12428
12429 void SetFunctionEntryHookTest::OnEntryHook(
12430 uintptr_t function, uintptr_t return_addr_location) {
12431 // Get the function's code object.
12432 i::Code* function_code = i::Code::GetCodeFromTargetAddress(
12285 reinterpret_cast<i::Address>(function)); 12433 reinterpret_cast<i::Address>(function));
12286 CHECK(code != NULL); 12434 CHECK(function_code != NULL);
12287 12435
12288 if (bar_ptr != NULL && code == (*bar_ptr)->code()) 12436 // Then try and look up the caller's code object.
12289 ++bar_entry_count; 12437 i::Address caller = *reinterpret_cast<i::Address*>(return_addr_location);
12290 12438
12291 if (foo_ptr != NULL && code == (*foo_ptr)->code()) 12439 // Count the invocation.
12292 ++foo_entry_count; 12440 SymbolInfo* caller_symbol = FindSymbolForAddr(caller);
12293 12441 SymbolInfo* function_symbol =
12294 // Let's check whether bar is the caller. 12442 FindSymbolForAddr(reinterpret_cast<i::Address>(function));
12295 if (bar_ptr != NULL) { 12443 ++invocations_[std::make_pair(caller_symbol, function_symbol)];
12296 const v8::internal::byte* caller = 12444
12297 *reinterpret_cast<v8::internal::byte**>(return_addr_location); 12445 if (!bar_func_.is_null() && function_code == bar_func_->code()) {
12298 12446 // Check that we have a symbol for the "bar" function at the right location.
12299 if ((*bar_ptr)->code()->instruction_start() <= caller && 12447 SymbolLocationMap::iterator it(
12300 (*bar_ptr)->code()->instruction_end() > caller) { 12448 symbol_locations_.find(function_code->instruction_start()));
12301 ++bar_caller_count; 12449 CHECK(it != symbol_locations_.end());
12450 }
12451
12452 if (!foo_func_.is_null() && function_code == foo_func_->code()) {
12453 // Check that we have a symbol for "foo" at the right location.
12454 SymbolLocationMap::iterator it(
12455 symbol_locations_.find(function_code->instruction_start()));
12456 CHECK(it != symbol_locations_.end());
12457 }
12458 }
12459
12460
12461 SymbolInfo* SetFunctionEntryHookTest::FindSymbolForAddr(i::Address addr) {
12462 SymbolLocationMap::iterator it(symbol_locations_.lower_bound(addr));
12463 // Do we have a direct hit on a symbol?
12464 if (it != symbol_locations_.end()) {
12465 if (it->first == addr)
12466 return it->second;
12467 }
12468
12469 // If not a direct hit, it'll have to be the previous symbol.
12470 if (it == symbol_locations_.begin())
12471 return NULL;
12472
12473 --it;
12474 size_t offs = addr - it->first;
12475 if (offs < it->second->size)
12476 return it->second;
12477
12478 return NULL;
12479 }
12480
12481
12482 v8::Handle<Value> SetFunctionEntryHookTest::OnRuntimeCallback(
12483 const v8::Arguments& args) {
12484 return v8_num(42);
12485 }
12486
12487
12488 size_t SetFunctionEntryHookTest::CountInvocations(
12489 const char* caller_name, const char* function_name) {
12490 InvocationMap::iterator it(invocations_.begin());
12491 size_t invocations = 0;
12492 for (; it != invocations_.end(); ++it) {
12493 SymbolInfo* caller = it->first.first;
12494 SymbolInfo* function = it->first.second;
12495
12496 // Filter out non-matching functions.
12497 if (function_name != NULL) {
12498 if (function->name.find(function_name) == std::string::npos)
12499 continue;
12302 } 12500 }
12303 } 12501
12304 } 12502 // Filter out non-matching callers.
12305 12503 if (caller_name != NULL) {
12306 12504 if (caller == NULL)
12307 static void RunLoopInNewEnv() { 12505 continue;
12308 bar_ptr = NULL; 12506 if (caller->name.find(caller_name) == std::string::npos)
12309 foo_ptr = NULL; 12507 continue;
12310 12508 }
12311 v8::Isolate* isolate = v8::Isolate::GetCurrent(); 12509
12510 // It matches add the invocation count to the tally.
12511 invocations += it->second;
12512 }
12513
12514 return invocations;
12515 }
12516
12517
12518 void SetFunctionEntryHookTest::RunLoopInNewEnv(v8::Isolate* isolate) {
12312 v8::HandleScope outer(isolate); 12519 v8::HandleScope outer(isolate);
12313 v8::Local<Context> env = Context::New(isolate); 12520 v8::Local<Context> env = Context::New(isolate);
12314 env->Enter(); 12521 env->Enter();
12315 12522
12523 Local<ObjectTemplate> t = ObjectTemplate::New();
12524 t->Set(v8_str("asdf"), v8::FunctionTemplate::New(RuntimeCallback));
12525 env->Global()->Set(v8_str("obj"), t->NewInstance());
12526
12316 const char* script = 12527 const char* script =
12317 "function bar() {" 12528 "function bar() {\n"
12318 " var sum = 0;" 12529 " var sum = 0;\n"
12319 " for (i = 0; i < 100; ++i)" 12530 " for (i = 0; i < 100; ++i)\n"
12320 " sum = foo(i);" 12531 " sum = foo(i);\n"
12321 " return sum;" 12532 " return sum;\n"
12322 "}" 12533 "}\n"
12323 "function foo(i) { return i * i; }"; 12534 "function foo(i) { return i * i; }\n"
12535 "// Invoke on the runtime function.\n"
12536 "obj.asdf()";
12324 CompileRun(script); 12537 CompileRun(script);
12325 i::Handle<i::JSFunction> bar = 12538 bar_func_ = i::Handle<i::JSFunction>::cast(
12326 i::Handle<i::JSFunction>::cast(
12327 v8::Utils::OpenHandle(*env->Global()->Get(v8_str("bar")))); 12539 v8::Utils::OpenHandle(*env->Global()->Get(v8_str("bar"))));
12328 ASSERT(*bar); 12540 ASSERT(!bar_func_.is_null());
12329 12541
12330 i::Handle<i::JSFunction> foo = 12542 foo_func_ =
12331 i::Handle<i::JSFunction>::cast( 12543 i::Handle<i::JSFunction>::cast(
12332 v8::Utils::OpenHandle(*env->Global()->Get(v8_str("foo")))); 12544 v8::Utils::OpenHandle(*env->Global()->Get(v8_str("foo"))));
12333 ASSERT(*foo); 12545 ASSERT(!foo_func_.is_null());
12334
12335 bar_ptr = &bar;
12336 foo_ptr = &foo;
12337 12546
12338 v8::Handle<v8::Value> value = CompileRun("bar();"); 12547 v8::Handle<v8::Value> value = CompileRun("bar();");
12339 CHECK(value->IsNumber()); 12548 CHECK(value->IsNumber());
12340 CHECK_EQ(9801.0, v8::Number::Cast(*value)->Value()); 12549 CHECK_EQ(9801.0, v8::Number::Cast(*value)->Value());
12341 12550
12342 // Test the optimized codegen path. 12551 // Test the optimized codegen path.
12343 value = CompileRun("%OptimizeFunctionOnNextCall(foo);" 12552 value = CompileRun("%OptimizeFunctionOnNextCall(foo);"
12344 "bar();"); 12553 "bar();");
12345 CHECK(value->IsNumber()); 12554 CHECK(value->IsNumber());
12346 CHECK_EQ(9801.0, v8::Number::Cast(*value)->Value()); 12555 CHECK_EQ(9801.0, v8::Number::Cast(*value)->Value());
12347 12556
12348 env->Exit(); 12557 env->Exit();
12349 } 12558 }
12350 12559
12560 void SetFunctionEntryHookTest::RunTest() {
12561 // Work in a new isolate throughout.
12562 v8::Isolate* isolate = v8::Isolate::New();
12563
12564 // Test setting the entry hook on the new isolate.
12565 CHECK(v8::V8::SetFunctionEntryHook(isolate, EntryHook));
12566
12567 // Replacing the hook, once set should fail.
12568 CHECK_EQ(false, v8::V8::SetFunctionEntryHook(isolate, EntryHook));
12569
12570 {
12571 v8::Isolate::Scope scope(isolate);
12572
12573 v8::V8::SetJitCodeEventHandler(v8::kJitCodeEventDefault, JitEvent);
12574
12575 RunLoopInNewEnv(isolate);
12576
12577 // Check the exepected invocation counts.
12578 CHECK_EQ(2, CountInvocations(NULL, "bar"));
12579 CHECK_EQ(200, CountInvocations("bar", "foo"));
12580 CHECK_EQ(200, CountInvocations(NULL, "foo"));
12581
12582 // Verify that we have an entry hook on some specific stubs.
12583 CHECK_NE(0, CountInvocations(NULL, "CEntryStub"));
12584 CHECK_NE(0, CountInvocations(NULL, "JSEntryStub"));
12585 CHECK_NE(0, CountInvocations(NULL, "JSEntryTrampoline"));
12586 }
12587 isolate->Dispose();
12588
12589 Reset();
12590
12591 // Make sure a second isolate is unaffected by the previous entry hook.
12592 isolate = v8::Isolate::New();
12593 {
12594 v8::Isolate::Scope scope(isolate);
12595
12596 // Reset the entry count to zero and set the entry hook.
12597 RunLoopInNewEnv(isolate);
12598
12599 // We should record no invocations in this isolate.
12600 CHECK_EQ(0, invocations_.size());
12601 }
12602 // Since the isolate has been used, we shouldn't be able to set an entry
12603 // hook anymore.
12604 CHECK_EQ(false, v8::V8::SetFunctionEntryHook(isolate, EntryHook));
12605
12606 isolate->Dispose();
12607 }
12608
12351 12609
12352 TEST(SetFunctionEntryHook) { 12610 TEST(SetFunctionEntryHook) {
12353 // FunctionEntryHook does not work well with experimental natives. 12611 // FunctionEntryHook does not work well with experimental natives.
12354 // Experimental natives are compiled during snapshot deserialization. 12612 // Experimental natives are compiled during snapshot deserialization.
12355 // This test breaks because InstallGetter (function from snapshot that 12613 // This test breaks because InstallGetter (function from snapshot that
12356 // only gets called from experimental natives) is compiled with entry hooks. 12614 // only gets called from experimental natives) is compiled with entry hooks.
12357 i::FLAG_harmony_typed_arrays = false; 12615 i::FLAG_harmony_typed_arrays = false;
12358 i::FLAG_harmony_array_buffer = false; 12616 i::FLAG_harmony_array_buffer = false;
12359 12617
12360 i::FLAG_allow_natives_syntax = true; 12618 i::FLAG_allow_natives_syntax = true;
12361 i::FLAG_use_inlining = false; 12619 i::FLAG_use_inlining = false;
12362 12620
12363 // Test setting and resetting the entry hook. 12621 SetFunctionEntryHookTest test;
12364 // Nulling it should always succeed. 12622 test.RunTest();
12365 CHECK(v8::V8::SetFunctionEntryHook(NULL));
12366
12367 CHECK(v8::V8::SetFunctionEntryHook(entry_hook));
12368 // Setting a hook while one's active should fail.
12369 CHECK_EQ(false, v8::V8::SetFunctionEntryHook(entry_hook));
12370
12371 CHECK(v8::V8::SetFunctionEntryHook(NULL));
12372
12373 // Reset the entry count to zero and set the entry hook.
12374 bar_entry_count = 0;
12375 bar_caller_count = 0;
12376 foo_entry_count = 0;
12377 CHECK(v8::V8::SetFunctionEntryHook(entry_hook));
12378 RunLoopInNewEnv();
12379
12380 CHECK_EQ(2, bar_entry_count);
12381 CHECK_EQ(200, bar_caller_count);
12382 CHECK_EQ(200, foo_entry_count);
12383
12384 // Clear the entry hook and count.
12385 bar_entry_count = 0;
12386 bar_caller_count = 0;
12387 foo_entry_count = 0;
12388 v8::V8::SetFunctionEntryHook(NULL);
12389
12390 // Clear the compilation cache to make sure we don't reuse the
12391 // functions from the previous invocation.
12392 v8::internal::Isolate::Current()->compilation_cache()->Clear();
12393
12394 // Verify that entry hooking is now disabled.
12395 RunLoopInNewEnv();
12396 CHECK_EQ(0u, bar_entry_count);
12397 CHECK_EQ(0u, bar_caller_count);
12398 CHECK_EQ(0u, foo_entry_count);
12399 } 12623 }
12400 12624
12401 12625
12402 static i::HashMap* code_map = NULL; 12626 static i::HashMap* code_map = NULL;
12403 static i::HashMap* jitcode_line_info = NULL; 12627 static i::HashMap* jitcode_line_info = NULL;
12404 static int saw_bar = 0; 12628 static int saw_bar = 0;
12405 static int move_events = 0; 12629 static int move_events = 0;
12406 12630
12407 12631
12408 static bool FunctionNameIs(const char* expected, 12632 static bool FunctionNameIs(const char* expected,
(...skipping 121 matching lines...) Expand 10 before | Expand all | Expand 10 after
12530 break; 12754 break;
12531 12755
12532 default: 12756 default:
12533 // Impossible event. 12757 // Impossible event.
12534 CHECK(false); 12758 CHECK(false);
12535 break; 12759 break;
12536 } 12760 }
12537 } 12761 }
12538 12762
12539 12763
12540 static bool MatchPointers(void* key1, void* key2) {
12541 return key1 == key2;
12542 }
12543
12544
12545 TEST(SetJitCodeEventHandler) { 12764 TEST(SetJitCodeEventHandler) {
12546 i::FLAG_stress_compaction = true; 12765 i::FLAG_stress_compaction = true;
12547 i::FLAG_incremental_marking = false; 12766 i::FLAG_incremental_marking = false;
12548 const char* script = 12767 const char* script =
12549 "function bar() {" 12768 "function bar() {"
12550 " var sum = 0;" 12769 " var sum = 0;"
12551 " for (i = 0; i < 100; ++i)" 12770 " for (i = 0; i < 100; ++i)"
12552 " sum = foo(i);" 12771 " sum = foo(i);"
12553 " return sum;" 12772 " return sum;"
12554 "}" 12773 "}"
(...skipping 6806 matching lines...) Expand 10 before | Expand all | Expand 10 after
19361 i::Semaphore* sem_; 19580 i::Semaphore* sem_;
19362 volatile int sem_value_; 19581 volatile int sem_value_;
19363 }; 19582 };
19364 19583
19365 19584
19366 THREADED_TEST(SemaphoreInterruption) { 19585 THREADED_TEST(SemaphoreInterruption) {
19367 ThreadInterruptTest().RunTest(); 19586 ThreadInterruptTest().RunTest();
19368 } 19587 }
19369 19588
19370 #endif // WIN32 19589 #endif // WIN32
OLDNEW
« src/ic.cc ('K') | « src/x64/code-stubs-x64.cc ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698