| OLD | NEW |
| 1 // Copyright 2010 the V8 project authors. All rights reserved. | 1 // Copyright 2010 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 198 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 209 __ movq(prototype, Operand(prototype, Context::SlotOffset(index))); | 209 __ movq(prototype, Operand(prototype, Context::SlotOffset(index))); |
| 210 // Load the initial map. The global functions all have initial maps. | 210 // Load the initial map. The global functions all have initial maps. |
| 211 __ movq(prototype, | 211 __ movq(prototype, |
| 212 FieldOperand(prototype, JSFunction::kPrototypeOrInitialMapOffset)); | 212 FieldOperand(prototype, JSFunction::kPrototypeOrInitialMapOffset)); |
| 213 // Load the prototype from the initial map. | 213 // Load the prototype from the initial map. |
| 214 __ movq(prototype, FieldOperand(prototype, Map::kPrototypeOffset)); | 214 __ movq(prototype, FieldOperand(prototype, Map::kPrototypeOffset)); |
| 215 } | 215 } |
| 216 | 216 |
| 217 | 217 |
| 218 void StubCompiler::GenerateDirectLoadGlobalFunctionPrototype( | 218 void StubCompiler::GenerateDirectLoadGlobalFunctionPrototype( |
| 219 MacroAssembler* masm, int index, Register prototype) { | 219 MacroAssembler* masm, int index, Register prototype, Label* miss) { |
| 220 // Check we're still in the same context. |
| 221 __ Move(prototype, Top::global()); |
| 222 __ cmpq(Operand(rsi, Context::SlotOffset(Context::GLOBAL_INDEX)), |
| 223 prototype); |
| 224 __ j(not_equal, miss); |
| 220 // Get the global function with the given index. | 225 // Get the global function with the given index. |
| 221 JSFunction* function = JSFunction::cast(Top::global_context()->get(index)); | 226 JSFunction* function = JSFunction::cast(Top::global_context()->get(index)); |
| 222 // Load its initial map. The global functions all have initial maps. | 227 // Load its initial map. The global functions all have initial maps. |
| 223 __ Move(prototype, Handle<Map>(function->initial_map())); | 228 __ Move(prototype, Handle<Map>(function->initial_map())); |
| 224 // Load the prototype from the initial map. | 229 // Load the prototype from the initial map. |
| 225 __ movq(prototype, FieldOperand(prototype, Map::kPrototypeOffset)); | 230 __ movq(prototype, FieldOperand(prototype, Map::kPrototypeOffset)); |
| 226 } | 231 } |
| 227 | 232 |
| 228 | 233 |
| 229 // Load a fast property out of a holder object (src). In-object properties | 234 // Load a fast property out of a holder object (src). In-object properties |
| (...skipping 727 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 957 case STRING_CHECK: | 962 case STRING_CHECK: |
| 958 if (!function->IsBuiltin()) { | 963 if (!function->IsBuiltin()) { |
| 959 // Calling non-builtins with a value as receiver requires boxing. | 964 // Calling non-builtins with a value as receiver requires boxing. |
| 960 __ jmp(&miss); | 965 __ jmp(&miss); |
| 961 } else { | 966 } else { |
| 962 // Check that the object is a two-byte string or a symbol. | 967 // Check that the object is a two-byte string or a symbol. |
| 963 __ CmpObjectType(rdx, FIRST_NONSTRING_TYPE, rax); | 968 __ CmpObjectType(rdx, FIRST_NONSTRING_TYPE, rax); |
| 964 __ j(above_equal, &miss); | 969 __ j(above_equal, &miss); |
| 965 // Check that the maps starting from the prototype haven't changed. | 970 // Check that the maps starting from the prototype haven't changed. |
| 966 GenerateDirectLoadGlobalFunctionPrototype( | 971 GenerateDirectLoadGlobalFunctionPrototype( |
| 967 masm(), Context::STRING_FUNCTION_INDEX, rax); | 972 masm(), Context::STRING_FUNCTION_INDEX, rax, &miss); |
| 968 CheckPrototypes(JSObject::cast(object->GetPrototype()), rax, holder, | 973 CheckPrototypes(JSObject::cast(object->GetPrototype()), rax, holder, |
| 969 rbx, rdx, rdi, name, &miss); | 974 rbx, rdx, rdi, name, &miss); |
| 970 } | 975 } |
| 971 break; | 976 break; |
| 972 | 977 |
| 973 case NUMBER_CHECK: { | 978 case NUMBER_CHECK: { |
| 974 if (!function->IsBuiltin()) { | 979 if (!function->IsBuiltin()) { |
| 975 // Calling non-builtins with a value as receiver requires boxing. | 980 // Calling non-builtins with a value as receiver requires boxing. |
| 976 __ jmp(&miss); | 981 __ jmp(&miss); |
| 977 } else { | 982 } else { |
| 978 Label fast; | 983 Label fast; |
| 979 // Check that the object is a smi or a heap number. | 984 // Check that the object is a smi or a heap number. |
| 980 __ JumpIfSmi(rdx, &fast); | 985 __ JumpIfSmi(rdx, &fast); |
| 981 __ CmpObjectType(rdx, HEAP_NUMBER_TYPE, rax); | 986 __ CmpObjectType(rdx, HEAP_NUMBER_TYPE, rax); |
| 982 __ j(not_equal, &miss); | 987 __ j(not_equal, &miss); |
| 983 __ bind(&fast); | 988 __ bind(&fast); |
| 984 // Check that the maps starting from the prototype haven't changed. | 989 // Check that the maps starting from the prototype haven't changed. |
| 985 GenerateDirectLoadGlobalFunctionPrototype( | 990 GenerateDirectLoadGlobalFunctionPrototype( |
| 986 masm(), Context::NUMBER_FUNCTION_INDEX, rax); | 991 masm(), Context::NUMBER_FUNCTION_INDEX, rax, &miss); |
| 987 CheckPrototypes(JSObject::cast(object->GetPrototype()), rax, holder, | 992 CheckPrototypes(JSObject::cast(object->GetPrototype()), rax, holder, |
| 988 rbx, rdx, rdi, name, &miss); | 993 rbx, rdx, rdi, name, &miss); |
| 989 } | 994 } |
| 990 break; | 995 break; |
| 991 } | 996 } |
| 992 | 997 |
| 993 case BOOLEAN_CHECK: { | 998 case BOOLEAN_CHECK: { |
| 994 if (!function->IsBuiltin()) { | 999 if (!function->IsBuiltin()) { |
| 995 // Calling non-builtins with a value as receiver requires boxing. | 1000 // Calling non-builtins with a value as receiver requires boxing. |
| 996 __ jmp(&miss); | 1001 __ jmp(&miss); |
| 997 } else { | 1002 } else { |
| 998 Label fast; | 1003 Label fast; |
| 999 // Check that the object is a boolean. | 1004 // Check that the object is a boolean. |
| 1000 __ CompareRoot(rdx, Heap::kTrueValueRootIndex); | 1005 __ CompareRoot(rdx, Heap::kTrueValueRootIndex); |
| 1001 __ j(equal, &fast); | 1006 __ j(equal, &fast); |
| 1002 __ CompareRoot(rdx, Heap::kFalseValueRootIndex); | 1007 __ CompareRoot(rdx, Heap::kFalseValueRootIndex); |
| 1003 __ j(not_equal, &miss); | 1008 __ j(not_equal, &miss); |
| 1004 __ bind(&fast); | 1009 __ bind(&fast); |
| 1005 // Check that the maps starting from the prototype haven't changed. | 1010 // Check that the maps starting from the prototype haven't changed. |
| 1006 GenerateDirectLoadGlobalFunctionPrototype( | 1011 GenerateDirectLoadGlobalFunctionPrototype( |
| 1007 masm(), Context::BOOLEAN_FUNCTION_INDEX, rax); | 1012 masm(), Context::BOOLEAN_FUNCTION_INDEX, rax, &miss); |
| 1008 CheckPrototypes(JSObject::cast(object->GetPrototype()), rax, holder, | 1013 CheckPrototypes(JSObject::cast(object->GetPrototype()), rax, holder, |
| 1009 rbx, rdx, rdi, name, &miss); | 1014 rbx, rdx, rdi, name, &miss); |
| 1010 } | 1015 } |
| 1011 break; | 1016 break; |
| 1012 } | 1017 } |
| 1013 | 1018 |
| 1014 default: | 1019 default: |
| 1015 UNREACHABLE(); | 1020 UNREACHABLE(); |
| 1016 } | 1021 } |
| 1017 | 1022 |
| (...skipping 333 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1351 const int argc = arguments().immediate(); | 1356 const int argc = arguments().immediate(); |
| 1352 | 1357 |
| 1353 Label miss; | 1358 Label miss; |
| 1354 Label index_out_of_range; | 1359 Label index_out_of_range; |
| 1355 | 1360 |
| 1356 GenerateNameCheck(name, &miss); | 1361 GenerateNameCheck(name, &miss); |
| 1357 | 1362 |
| 1358 // Check that the maps starting from the prototype haven't changed. | 1363 // Check that the maps starting from the prototype haven't changed. |
| 1359 GenerateDirectLoadGlobalFunctionPrototype(masm(), | 1364 GenerateDirectLoadGlobalFunctionPrototype(masm(), |
| 1360 Context::STRING_FUNCTION_INDEX, | 1365 Context::STRING_FUNCTION_INDEX, |
| 1361 rax); | 1366 rax, |
| 1367 &miss); |
| 1362 ASSERT(object != holder); | 1368 ASSERT(object != holder); |
| 1363 CheckPrototypes(JSObject::cast(object->GetPrototype()), rax, holder, | 1369 CheckPrototypes(JSObject::cast(object->GetPrototype()), rax, holder, |
| 1364 rbx, rdx, rdi, name, &miss); | 1370 rbx, rdx, rdi, name, &miss); |
| 1365 | 1371 |
| 1366 Register receiver = rax; | 1372 Register receiver = rax; |
| 1367 Register index = rdi; | 1373 Register index = rdi; |
| 1368 Register scratch1 = rbx; | 1374 Register scratch1 = rbx; |
| 1369 Register scratch2 = rdx; | 1375 Register scratch2 = rdx; |
| 1370 Register result = rax; | 1376 Register result = rax; |
| 1371 __ movq(receiver, Operand(rsp, (argc + 1) * kPointerSize)); | 1377 __ movq(receiver, Operand(rsp, (argc + 1) * kPointerSize)); |
| (...skipping 50 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1422 | 1428 |
| 1423 const int argc = arguments().immediate(); | 1429 const int argc = arguments().immediate(); |
| 1424 | 1430 |
| 1425 Label miss; | 1431 Label miss; |
| 1426 Label index_out_of_range; | 1432 Label index_out_of_range; |
| 1427 GenerateNameCheck(name, &miss); | 1433 GenerateNameCheck(name, &miss); |
| 1428 | 1434 |
| 1429 // Check that the maps starting from the prototype haven't changed. | 1435 // Check that the maps starting from the prototype haven't changed. |
| 1430 GenerateDirectLoadGlobalFunctionPrototype(masm(), | 1436 GenerateDirectLoadGlobalFunctionPrototype(masm(), |
| 1431 Context::STRING_FUNCTION_INDEX, | 1437 Context::STRING_FUNCTION_INDEX, |
| 1432 rax); | 1438 rax, |
| 1439 &miss); |
| 1433 ASSERT(object != holder); | 1440 ASSERT(object != holder); |
| 1434 CheckPrototypes(JSObject::cast(object->GetPrototype()), rax, holder, | 1441 CheckPrototypes(JSObject::cast(object->GetPrototype()), rax, holder, |
| 1435 rbx, rdx, rdi, name, &miss); | 1442 rbx, rdx, rdi, name, &miss); |
| 1436 | 1443 |
| 1437 Register receiver = rbx; | 1444 Register receiver = rbx; |
| 1438 Register index = rdi; | 1445 Register index = rdi; |
| 1439 Register scratch = rdx; | 1446 Register scratch = rdx; |
| 1440 Register result = rax; | 1447 Register result = rax; |
| 1441 __ movq(receiver, Operand(rsp, (argc + 1) * kPointerSize)); | 1448 __ movq(receiver, Operand(rsp, (argc + 1) * kPointerSize)); |
| 1442 if (argc > 0) { | 1449 if (argc > 0) { |
| (...skipping 1362 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2805 // Return the generated code. | 2812 // Return the generated code. |
| 2806 return GetCode(); | 2813 return GetCode(); |
| 2807 } | 2814 } |
| 2808 | 2815 |
| 2809 | 2816 |
| 2810 #undef __ | 2817 #undef __ |
| 2811 | 2818 |
| 2812 } } // namespace v8::internal | 2819 } } // namespace v8::internal |
| 2813 | 2820 |
| 2814 #endif // V8_TARGET_ARCH_X64 | 2821 #endif // V8_TARGET_ARCH_X64 |
| OLD | NEW |