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

Side by Side Diff: src/type-info.cc

Issue 16361015: Migrate Compare ICs to new type rep (Closed) Base URL: https://v8.googlecode.com/svn/branches/bleeding_edge
Patch Set: Rebased 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 | Annotate | Revision Log
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 24 matching lines...) Expand all
35 #include "stub-cache.h" 35 #include "stub-cache.h"
36 #include "type-info.h" 36 #include "type-info.h"
37 37
38 #include "ic-inl.h" 38 #include "ic-inl.h"
39 #include "objects-inl.h" 39 #include "objects-inl.h"
40 40
41 namespace v8 { 41 namespace v8 {
42 namespace internal { 42 namespace internal {
43 43
44 44
45 TypeInfo TypeInfo::TypeFromValue(Handle<Object> value) { 45 TypeInfo TypeInfo::FromValue(Handle<Object> value) {
46 TypeInfo info;
47 if (value->IsSmi()) { 46 if (value->IsSmi()) {
48 info = TypeInfo::Smi(); 47 return TypeInfo::Smi();
49 } else if (value->IsHeapNumber()) { 48 } else if (value->IsHeapNumber()) {
50 info = TypeInfo::IsInt32Double(HeapNumber::cast(*value)->value()) 49 return TypeInfo::IsInt32Double(HeapNumber::cast(*value)->value())
51 ? TypeInfo::Integer32() 50 ? TypeInfo::Integer32()
52 : TypeInfo::Double(); 51 : TypeInfo::Double();
53 } else if (value->IsString()) { 52 } else if (value->IsString()) {
54 info = TypeInfo::String(); 53 return TypeInfo::String();
55 } else {
56 info = TypeInfo::Unknown();
57 } 54 }
58 return info; 55 return TypeInfo::Unknown();
59 } 56 }
60 57
61 58
62 TypeFeedbackOracle::TypeFeedbackOracle(Handle<Code> code, 59 TypeFeedbackOracle::TypeFeedbackOracle(Handle<Code> code,
63 Handle<Context> native_context, 60 Handle<Context> native_context,
64 Isolate* isolate, 61 Isolate* isolate,
65 Zone* zone) 62 Zone* zone)
66 : native_context_(native_context), 63 : native_context_(native_context),
67 isolate_(isolate), 64 isolate_(isolate),
68 zone_(zone) { 65 zone_(zone) {
(...skipping 159 matching lines...) Expand 10 before | Expand all | Expand 10 after
228 Handle<Code> code = Handle<Code>::cast(map_or_code); 225 Handle<Code> code = Handle<Code>::cast(map_or_code);
229 Map* map = code->FindFirstMap()->CurrentMapForDeprecated(); 226 Map* map = code->FindFirstMap()->CurrentMapForDeprecated();
230 return map == NULL || CanRetainOtherContext(map, *native_context_) 227 return map == NULL || CanRetainOtherContext(map, *native_context_)
231 ? Handle<Map>::null() 228 ? Handle<Map>::null()
232 : Handle<Map>(map); 229 : Handle<Map>(map);
233 } 230 }
234 return Handle<Map>::cast(map_or_code); 231 return Handle<Map>::cast(map_or_code);
235 } 232 }
236 233
237 234
238 Handle<Map> TypeFeedbackOracle::CompareNilMonomorphicReceiverType(
239 CompareOperation* expr) {
240 Handle<Object> maybe_code = GetInfo(expr->CompareOperationFeedbackId());
241 if (maybe_code->IsCode()) {
242 Map* map = Handle<Code>::cast(maybe_code)->FindFirstMap();
243 if (map == NULL) return Handle<Map>();
244 map = map->CurrentMapForDeprecated();
245 return map == NULL || CanRetainOtherContext(map, *native_context_)
246 ? Handle<Map>()
247 : Handle<Map>(map);
248 } else if (maybe_code->IsMap()) {
249 ASSERT(!Handle<Map>::cast(maybe_code)->is_deprecated());
250 return Handle<Map>::cast(maybe_code);
251 }
252 return Handle<Map>();
253 }
254
255
256 KeyedAccessStoreMode TypeFeedbackOracle::GetStoreMode( 235 KeyedAccessStoreMode TypeFeedbackOracle::GetStoreMode(
257 TypeFeedbackId ast_id) { 236 TypeFeedbackId ast_id) {
258 Handle<Object> map_or_code = GetInfo(ast_id); 237 Handle<Object> map_or_code = GetInfo(ast_id);
259 if (map_or_code->IsCode()) { 238 if (map_or_code->IsCode()) {
260 Handle<Code> code = Handle<Code>::cast(map_or_code); 239 Handle<Code> code = Handle<Code>::cast(map_or_code);
261 if (code->kind() == Code::KEYED_STORE_IC) { 240 if (code->kind() == Code::KEYED_STORE_IC) {
262 return Code::GetKeyedAccessStoreMode(code->extra_ic_state()); 241 return Code::GetKeyedAccessStoreMode(code->extra_ic_state());
263 } 242 }
264 } 243 }
265 return STANDARD_STORE; 244 return STANDARD_STORE;
(...skipping 91 matching lines...) Expand 10 before | Expand all | Expand 10 after
357 bool TypeFeedbackOracle::LoadIsStub(Property* expr, ICStub* stub) { 336 bool TypeFeedbackOracle::LoadIsStub(Property* expr, ICStub* stub) {
358 Handle<Object> object = GetInfo(expr->PropertyFeedbackId()); 337 Handle<Object> object = GetInfo(expr->PropertyFeedbackId());
359 if (!object->IsCode()) return false; 338 if (!object->IsCode()) return false;
360 Handle<Code> code = Handle<Code>::cast(object); 339 Handle<Code> code = Handle<Code>::cast(object);
361 if (!code->is_load_stub()) return false; 340 if (!code->is_load_stub()) return false;
362 if (code->ic_state() != MONOMORPHIC) return false; 341 if (code->ic_state() != MONOMORPHIC) return false;
363 return stub->Describes(*code); 342 return stub->Describes(*code);
364 } 343 }
365 344
366 345
367 static TypeInfo TypeFromCompareType(CompareIC::State state) { 346 void TypeFeedbackOracle::CompareTypes(TypeFeedbackId id,
368 switch (state) { 347 Handle<Type>* left_type,
369 case CompareIC::UNINITIALIZED: 348 Handle<Type>* right_type,
370 // Uninitialized means never executed. 349 Handle<Type>* overall_type,
371 return TypeInfo::Uninitialized(); 350 Handle<Type>* compare_nil_type) {
372 case CompareIC::SMI: 351 *left_type = *right_type = *overall_type = *compare_nil_type =
373 return TypeInfo::Smi(); 352 handle(Type::Any(), isolate_);
374 case CompareIC::NUMBER: 353 Handle<Object> info = GetInfo(id);
375 return TypeInfo::Number(); 354 if (!info->IsCode()) return;
376 case CompareIC::INTERNALIZED_STRING: 355 Handle<Code> code = Handle<Code>::cast(info);
377 return TypeInfo::InternalizedString(); 356
378 case CompareIC::STRING: 357 Handle<Map> map;
379 return TypeInfo::String(); 358 Map* raw_map = code->FindFirstMap();
380 case CompareIC::OBJECT: 359 if (raw_map != NULL) {
381 case CompareIC::KNOWN_OBJECT: 360 raw_map = raw_map->CurrentMapForDeprecated();
382 // TODO(kasperl): We really need a type for JS objects here. 361 if (!CanRetainOtherContext(raw_map, *native_context_)) {
383 return TypeInfo::NonPrimitive(); 362 map = handle(raw_map, isolate_);
384 case CompareIC::GENERIC: 363 }
385 default: 364 }
386 return TypeInfo::Unknown(); 365
366 if (code->is_compare_ic_stub()) {
367 int stub_minor_key = code->stub_info();
368 CompareIC::State left_state, right_state, handler_state;
369 ICCompareStub::DecodeMinorKey(stub_minor_key, &left_state, &right_state,
370 &handler_state, NULL);
371 *left_type = CompareIC::StateToType(isolate_, left_state);
372 *right_type = CompareIC::StateToType(isolate_, right_state);
373 *overall_type = CompareIC::StateToType(isolate_, handler_state, map);
374 } else if (code->is_compare_nil_ic_stub()) {
375 CompareNilICStub::State state(code->compare_nil_state());
376 *compare_nil_type = CompareNilICStub::StateToType(isolate_, state, map);
387 } 377 }
388 } 378 }
389 379
390 380
391 void TypeFeedbackOracle::CompareType(CompareOperation* expr,
392 TypeInfo* left_type,
393 TypeInfo* right_type,
394 TypeInfo* overall_type) {
395 Handle<Object> object = GetInfo(expr->CompareOperationFeedbackId());
396 TypeInfo unknown = TypeInfo::Unknown();
397 if (!object->IsCode()) {
398 *left_type = *right_type = *overall_type = unknown;
399 return;
400 }
401 Handle<Code> code = Handle<Code>::cast(object);
402 if (!code->is_compare_ic_stub()) {
403 *left_type = *right_type = *overall_type = unknown;
404 return;
405 }
406
407 int stub_minor_key = code->stub_info();
408 CompareIC::State left_state, right_state, handler_state;
409 ICCompareStub::DecodeMinorKey(stub_minor_key, &left_state, &right_state,
410 &handler_state, NULL);
411 *left_type = TypeFromCompareType(left_state);
412 *right_type = TypeFromCompareType(right_state);
413 *overall_type = TypeFromCompareType(handler_state);
414 }
415
416
417 Handle<Map> TypeFeedbackOracle::GetCompareMap(CompareOperation* expr) {
418 Handle<Object> object = GetInfo(expr->CompareOperationFeedbackId());
419 if (!object->IsCode()) return Handle<Map>::null();
420 Handle<Code> code = Handle<Code>::cast(object);
421 if (!code->is_compare_ic_stub()) return Handle<Map>::null();
422 CompareIC::State state = ICCompareStub::CompareState(code->stub_info());
423 if (state != CompareIC::KNOWN_OBJECT) {
424 return Handle<Map>::null();
425 }
426 Map* map = code->FindFirstMap()->CurrentMapForDeprecated();
427 return map == NULL || CanRetainOtherContext(map, *native_context_)
428 ? Handle<Map>::null()
429 : Handle<Map>(map);
430 }
431
432
433 TypeInfo TypeFeedbackOracle::UnaryType(UnaryOperation* expr) { 381 TypeInfo TypeFeedbackOracle::UnaryType(UnaryOperation* expr) {
434 Handle<Object> object = GetInfo(expr->UnaryOperationFeedbackId()); 382 Handle<Object> object = GetInfo(expr->UnaryOperationFeedbackId());
435 TypeInfo unknown = TypeInfo::Unknown(); 383 TypeInfo unknown = TypeInfo::Unknown();
436 if (!object->IsCode()) return unknown; 384 if (!object->IsCode()) return unknown;
437 Handle<Code> code = Handle<Code>::cast(object); 385 Handle<Code> code = Handle<Code>::cast(object);
438 ASSERT(code->is_unary_op_stub()); 386 ASSERT(code->is_unary_op_stub());
439 UnaryOpIC::TypeInfo type = static_cast<UnaryOpIC::TypeInfo>( 387 UnaryOpIC::TypeInfo type = static_cast<UnaryOpIC::TypeInfo>(
440 code->unary_op_type()); 388 code->unary_op_type());
441 switch (type) { 389 switch (type) {
442 case UnaryOpIC::SMI: 390 case UnaryOpIC::SMI:
(...skipping 47 matching lines...) Expand 10 before | Expand all | Expand 10 after
490 BinaryOpStub::decode_has_fixed_right_arg_from_minor_key(minor_key); 438 BinaryOpStub::decode_has_fixed_right_arg_from_minor_key(minor_key);
491 *fixed_right_arg_value = 439 *fixed_right_arg_value =
492 BinaryOpStub::decode_fixed_right_arg_value_from_minor_key(minor_key); 440 BinaryOpStub::decode_fixed_right_arg_value_from_minor_key(minor_key);
493 return; 441 return;
494 } 442 }
495 // Not a binary op stub. 443 // Not a binary op stub.
496 *left = *right = *result = unknown; 444 *left = *right = *result = unknown;
497 } 445 }
498 446
499 447
500 TypeInfo TypeFeedbackOracle::SwitchType(CaseClause* clause) { 448 Handle<Type> TypeFeedbackOracle::ClauseType(TypeFeedbackId id) {
501 Handle<Object> object = GetInfo(clause->CompareId()); 449 Handle<Object> info = GetInfo(id);
502 TypeInfo unknown = TypeInfo::Unknown(); 450 Handle<Type> result(Type::None(), isolate_);
503 if (!object->IsCode()) return unknown; 451 if (info->IsCode() && Handle<Code>::cast(info)->is_compare_ic_stub()) {
504 Handle<Code> code = Handle<Code>::cast(object); 452 Handle<Code> code = Handle<Code>::cast(info);
505 if (!code->is_compare_ic_stub()) return unknown; 453 CompareIC::State state = ICCompareStub::CompareState(code->stub_info());
506 454 result = CompareIC::StateToType(isolate_, state);
507 CompareIC::State state = ICCompareStub::CompareState(code->stub_info()); 455 }
508 return TypeFromCompareType(state); 456 return result;
509 } 457 }
510 458
511 459
512 TypeInfo TypeFeedbackOracle::IncrementType(CountOperation* expr) { 460 TypeInfo TypeFeedbackOracle::IncrementType(CountOperation* expr) {
513 Handle<Object> object = GetInfo(expr->CountBinOpFeedbackId()); 461 Handle<Object> object = GetInfo(expr->CountBinOpFeedbackId());
514 TypeInfo unknown = TypeInfo::Unknown(); 462 TypeInfo unknown = TypeInfo::Unknown();
515 if (!object->IsCode()) return unknown; 463 if (!object->IsCode()) return unknown;
516 Handle<Code> code = Handle<Code>::cast(object); 464 Handle<Code> code = Handle<Code>::cast(object);
517 if (!code->is_binary_op_stub()) return unknown; 465 if (!code->is_binary_op_stub()) return unknown;
518 466
(...skipping 110 matching lines...) Expand 10 before | Expand all | Expand 10 after
629 } 577 }
630 } 578 }
631 579
632 580
633 byte TypeFeedbackOracle::ToBooleanTypes(TypeFeedbackId id) { 581 byte TypeFeedbackOracle::ToBooleanTypes(TypeFeedbackId id) {
634 Handle<Object> object = GetInfo(id); 582 Handle<Object> object = GetInfo(id);
635 return object->IsCode() ? Handle<Code>::cast(object)->to_boolean_state() : 0; 583 return object->IsCode() ? Handle<Code>::cast(object)->to_boolean_state() : 0;
636 } 584 }
637 585
638 586
639 byte TypeFeedbackOracle::CompareNilTypes(CompareOperation* expr) {
640 Handle<Object> object = GetInfo(expr->CompareOperationFeedbackId());
641 if (object->IsCode() &&
642 Handle<Code>::cast(object)->is_compare_nil_ic_stub()) {
643 return Handle<Code>::cast(object)->compare_nil_types();
644 } else {
645 return CompareNilICStub::Types::FullCompare().ToIntegral();
646 }
647 }
648
649
650 // Things are a bit tricky here: The iterator for the RelocInfos and the infos 587 // Things are a bit tricky here: The iterator for the RelocInfos and the infos
651 // themselves are not GC-safe, so we first get all infos, then we create the 588 // themselves are not GC-safe, so we first get all infos, then we create the
652 // dictionary (possibly triggering GC), and finally we relocate the collected 589 // dictionary (possibly triggering GC), and finally we relocate the collected
653 // infos before we process them. 590 // infos before we process them.
654 void TypeFeedbackOracle::BuildDictionary(Handle<Code> code) { 591 void TypeFeedbackOracle::BuildDictionary(Handle<Code> code) {
655 DisallowHeapAllocation no_allocation; 592 DisallowHeapAllocation no_allocation;
656 ZoneList<RelocInfo> infos(16, zone()); 593 ZoneList<RelocInfo> infos(16, zone());
657 HandleScope scope(isolate_); 594 HandleScope scope(isolate_);
658 GetRelocInfos(code, &infos); 595 GetRelocInfos(code, &infos);
659 CreateDictionary(code, &infos); 596 CreateDictionary(code, &infos);
(...skipping 118 matching lines...) Expand 10 before | Expand all | Expand 10 after
778 USE(maybe_result); 715 USE(maybe_result);
779 #ifdef DEBUG 716 #ifdef DEBUG
780 Object* result = NULL; 717 Object* result = NULL;
781 // Dictionary has been allocated with sufficient size for all elements. 718 // Dictionary has been allocated with sufficient size for all elements.
782 ASSERT(maybe_result->ToObject(&result)); 719 ASSERT(maybe_result->ToObject(&result));
783 ASSERT(*dictionary_ == result); 720 ASSERT(*dictionary_ == result);
784 #endif 721 #endif
785 } 722 }
786 723
787 } } // namespace v8::internal 724 } } // namespace v8::internal
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698