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

Side by Side Diff: src/x64/code-stubs-x64.cc

Issue 22267005: Use StackArgumenstAccessor and kPCOnStackSize/kFPOnStackSize to compute stack address/operand (Closed) Base URL: https://v8.googlecode.com/svn/branches/bleeding_edge
Patch Set: Rebased with master Created 7 years, 3 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 2013 the V8 project authors. All rights reserved. 1 // Copyright 2013 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 1958 matching lines...) Expand 10 before | Expand all | Expand 10 after
1969 __ bind(&miss); 1969 __ bind(&miss);
1970 1970
1971 StubCompiler::TailCallBuiltin( 1971 StubCompiler::TailCallBuiltin(
1972 masm, BaseLoadStoreStubCompiler::MissBuiltin(kind())); 1972 masm, BaseLoadStoreStubCompiler::MissBuiltin(kind()));
1973 } 1973 }
1974 1974
1975 1975
1976 void ArgumentsAccessStub::GenerateReadElement(MacroAssembler* masm) { 1976 void ArgumentsAccessStub::GenerateReadElement(MacroAssembler* masm) {
1977 // The key is in rdx and the parameter count is in rax. 1977 // The key is in rdx and the parameter count is in rax.
1978 1978
1979 // The displacement is used for skipping the frame pointer on the
1980 // stack. It is the offset of the last parameter (if any) relative
1981 // to the frame pointer.
1982 static const int kDisplacement = 1 * kPointerSize;
1983
1984 // Check that the key is a smi. 1979 // Check that the key is a smi.
1985 Label slow; 1980 Label slow;
1986 __ JumpIfNotSmi(rdx, &slow); 1981 __ JumpIfNotSmi(rdx, &slow);
1987 1982
1988 // Check if the calling frame is an arguments adaptor frame. We look at the 1983 // Check if the calling frame is an arguments adaptor frame. We look at the
1989 // context offset, and if the frame is not a regular one, then we find a 1984 // context offset, and if the frame is not a regular one, then we find a
1990 // Smi instead of the context. We can't use SmiCompare here, because that 1985 // Smi instead of the context. We can't use SmiCompare here, because that
1991 // only works for comparing two smis. 1986 // only works for comparing two smis.
1992 Label adaptor; 1987 Label adaptor;
1993 __ movq(rbx, Operand(rbp, StandardFrameConstants::kCallerFPOffset)); 1988 __ movq(rbx, Operand(rbp, StandardFrameConstants::kCallerFPOffset));
1994 __ Cmp(Operand(rbx, StandardFrameConstants::kContextOffset), 1989 __ Cmp(Operand(rbx, StandardFrameConstants::kContextOffset),
1995 Smi::FromInt(StackFrame::ARGUMENTS_ADAPTOR)); 1990 Smi::FromInt(StackFrame::ARGUMENTS_ADAPTOR));
1996 __ j(equal, &adaptor); 1991 __ j(equal, &adaptor);
1997 1992
1998 // Check index against formal parameters count limit passed in 1993 // Check index against formal parameters count limit passed in
1999 // through register rax. Use unsigned comparison to get negative 1994 // through register rax. Use unsigned comparison to get negative
2000 // check for free. 1995 // check for free.
2001 __ cmpq(rdx, rax); 1996 __ cmpq(rdx, rax);
2002 __ j(above_equal, &slow); 1997 __ j(above_equal, &slow);
2003 1998
2004 // Read the argument from the stack and return it. 1999 // Read the argument from the stack and return it.
2005 SmiIndex index = masm->SmiToIndex(rax, rax, kPointerSizeLog2); 2000 __ SmiSub(rax, rax, rdx);
2006 __ lea(rbx, Operand(rbp, index.reg, index.scale, 0)); 2001 __ SmiToInteger32(rax, rax);
2007 index = masm->SmiToNegativeIndex(rdx, rdx, kPointerSizeLog2); 2002 StackArgumentsAccessor args(rbp, rax, ARGUMENTS_DONT_CONTAIN_RECEIVER);
2008 __ movq(rax, Operand(rbx, index.reg, index.scale, kDisplacement)); 2003 __ movq(rax, args.GetArgumentOperand(0));
2009 __ Ret(); 2004 __ Ret();
2010 2005
2011 // Arguments adaptor case: Check index against actual arguments 2006 // Arguments adaptor case: Check index against actual arguments
2012 // limit found in the arguments adaptor frame. Use unsigned 2007 // limit found in the arguments adaptor frame. Use unsigned
2013 // comparison to get negative check for free. 2008 // comparison to get negative check for free.
2014 __ bind(&adaptor); 2009 __ bind(&adaptor);
2015 __ movq(rcx, Operand(rbx, ArgumentsAdaptorFrameConstants::kLengthOffset)); 2010 __ movq(rcx, Operand(rbx, ArgumentsAdaptorFrameConstants::kLengthOffset));
2016 __ cmpq(rdx, rcx); 2011 __ cmpq(rdx, rcx);
2017 __ j(above_equal, &slow); 2012 __ j(above_equal, &slow);
2018 2013
2019 // Read the argument from the stack and return it. 2014 // Read the argument from the stack and return it.
2020 index = masm->SmiToIndex(rax, rcx, kPointerSizeLog2); 2015 __ SmiSub(rcx, rcx, rdx);
2021 __ lea(rbx, Operand(rbx, index.reg, index.scale, 0)); 2016 __ SmiToInteger32(rcx, rcx);
2022 index = masm->SmiToNegativeIndex(rdx, rdx, kPointerSizeLog2); 2017 StackArgumentsAccessor adaptor_args(rbx, rcx,
2023 __ movq(rax, Operand(rbx, index.reg, index.scale, kDisplacement)); 2018 ARGUMENTS_DONT_CONTAIN_RECEIVER);
2019 __ movq(rax, adaptor_args.GetArgumentOperand(0));
2024 __ Ret(); 2020 __ Ret();
2025 2021
2026 // Slow-case: Handle non-smi or out-of-bounds access to arguments 2022 // Slow-case: Handle non-smi or out-of-bounds access to arguments
2027 // by calling the runtime system. 2023 // by calling the runtime system.
2028 __ bind(&slow); 2024 __ bind(&slow);
2029 __ PopReturnAddressTo(rbx); 2025 __ PopReturnAddressTo(rbx);
2030 __ push(rdx); 2026 __ push(rdx);
2031 __ PushReturnAddressFrom(rbx); 2027 __ PushReturnAddressFrom(rbx);
2032 __ TailCallRuntime(Runtime::kGetArgumentsProperty, 1, 1); 2028 __ TailCallRuntime(Runtime::kGetArgumentsProperty, 1, 1);
2033 } 2029 }
(...skipping 353 matching lines...) Expand 10 before | Expand all | Expand 10 after
2387 __ TailCallRuntime(Runtime::kRegExpExec, 4, 1); 2383 __ TailCallRuntime(Runtime::kRegExpExec, 4, 1);
2388 #else // V8_INTERPRETED_REGEXP 2384 #else // V8_INTERPRETED_REGEXP
2389 2385
2390 // Stack frame on entry. 2386 // Stack frame on entry.
2391 // rsp[0] : return address 2387 // rsp[0] : return address
2392 // rsp[8] : last_match_info (expected JSArray) 2388 // rsp[8] : last_match_info (expected JSArray)
2393 // rsp[16] : previous index 2389 // rsp[16] : previous index
2394 // rsp[24] : subject string 2390 // rsp[24] : subject string
2395 // rsp[32] : JSRegExp object 2391 // rsp[32] : JSRegExp object
2396 2392
2397 static const int kLastMatchInfoOffset = 1 * kPointerSize; 2393 enum RegExpExecStubArgumentIndices {
2398 static const int kPreviousIndexOffset = 2 * kPointerSize; 2394 JS_REG_EXP_OBJECT_ARGUMENT_INDEX,
2399 static const int kSubjectOffset = 3 * kPointerSize; 2395 SUBJECT_STRING_ARGUMENT_INDEX,
2400 static const int kJSRegExpOffset = 4 * kPointerSize; 2396 PREVIOUS_INDEX_ARGUMENT_INDEX,
2397 LAST_MATCH_INFO_ARGUMENT_INDEX,
2398 REG_EXP_EXEC_ARGUMENT_COUNT
2399 };
2401 2400
2401 StackArgumentsAccessor args(rsp, REG_EXP_EXEC_ARGUMENT_COUNT,
2402 ARGUMENTS_DONT_CONTAIN_RECEIVER);
2402 Label runtime; 2403 Label runtime;
2403 // Ensure that a RegExp stack is allocated. 2404 // Ensure that a RegExp stack is allocated.
2404 Isolate* isolate = masm->isolate(); 2405 Isolate* isolate = masm->isolate();
2405 ExternalReference address_of_regexp_stack_memory_address = 2406 ExternalReference address_of_regexp_stack_memory_address =
2406 ExternalReference::address_of_regexp_stack_memory_address(isolate); 2407 ExternalReference::address_of_regexp_stack_memory_address(isolate);
2407 ExternalReference address_of_regexp_stack_memory_size = 2408 ExternalReference address_of_regexp_stack_memory_size =
2408 ExternalReference::address_of_regexp_stack_memory_size(isolate); 2409 ExternalReference::address_of_regexp_stack_memory_size(isolate);
2409 __ Load(kScratchRegister, address_of_regexp_stack_memory_size); 2410 __ Load(kScratchRegister, address_of_regexp_stack_memory_size);
2410 __ testq(kScratchRegister, kScratchRegister); 2411 __ testq(kScratchRegister, kScratchRegister);
2411 __ j(zero, &runtime); 2412 __ j(zero, &runtime);
2412 2413
2413 // Check that the first argument is a JSRegExp object. 2414 // Check that the first argument is a JSRegExp object.
2414 __ movq(rax, Operand(rsp, kJSRegExpOffset)); 2415 __ movq(rax, args.GetArgumentOperand(JS_REG_EXP_OBJECT_ARGUMENT_INDEX));
2415 __ JumpIfSmi(rax, &runtime); 2416 __ JumpIfSmi(rax, &runtime);
2416 __ CmpObjectType(rax, JS_REGEXP_TYPE, kScratchRegister); 2417 __ CmpObjectType(rax, JS_REGEXP_TYPE, kScratchRegister);
2417 __ j(not_equal, &runtime); 2418 __ j(not_equal, &runtime);
2418 2419
2419 // Check that the RegExp has been compiled (data contains a fixed array). 2420 // Check that the RegExp has been compiled (data contains a fixed array).
2420 __ movq(rax, FieldOperand(rax, JSRegExp::kDataOffset)); 2421 __ movq(rax, FieldOperand(rax, JSRegExp::kDataOffset));
2421 if (FLAG_debug_code) { 2422 if (FLAG_debug_code) {
2422 Condition is_smi = masm->CheckSmi(rax); 2423 Condition is_smi = masm->CheckSmi(rax);
2423 __ Check(NegateCondition(is_smi), 2424 __ Check(NegateCondition(is_smi),
2424 kUnexpectedTypeForRegExpDataFixedArrayExpected); 2425 kUnexpectedTypeForRegExpDataFixedArrayExpected);
(...skipping 12 matching lines...) Expand all
2437 __ SmiToInteger32(rdx, 2438 __ SmiToInteger32(rdx,
2438 FieldOperand(rax, JSRegExp::kIrregexpCaptureCountOffset)); 2439 FieldOperand(rax, JSRegExp::kIrregexpCaptureCountOffset));
2439 // Check (number_of_captures + 1) * 2 <= offsets vector size 2440 // Check (number_of_captures + 1) * 2 <= offsets vector size
2440 // Or number_of_captures <= offsets vector size / 2 - 1 2441 // Or number_of_captures <= offsets vector size / 2 - 1
2441 STATIC_ASSERT(Isolate::kJSRegexpStaticOffsetsVectorSize >= 2); 2442 STATIC_ASSERT(Isolate::kJSRegexpStaticOffsetsVectorSize >= 2);
2442 __ cmpl(rdx, Immediate(Isolate::kJSRegexpStaticOffsetsVectorSize / 2 - 1)); 2443 __ cmpl(rdx, Immediate(Isolate::kJSRegexpStaticOffsetsVectorSize / 2 - 1));
2443 __ j(above, &runtime); 2444 __ j(above, &runtime);
2444 2445
2445 // Reset offset for possibly sliced string. 2446 // Reset offset for possibly sliced string.
2446 __ Set(r14, 0); 2447 __ Set(r14, 0);
2447 __ movq(rdi, Operand(rsp, kSubjectOffset)); 2448 __ movq(rdi, args.GetArgumentOperand(SUBJECT_STRING_ARGUMENT_INDEX));
2448 __ JumpIfSmi(rdi, &runtime); 2449 __ JumpIfSmi(rdi, &runtime);
2449 __ movq(r15, rdi); // Make a copy of the original subject string. 2450 __ movq(r15, rdi); // Make a copy of the original subject string.
2450 __ movq(rbx, FieldOperand(rdi, HeapObject::kMapOffset)); 2451 __ movq(rbx, FieldOperand(rdi, HeapObject::kMapOffset));
2451 __ movzxbl(rbx, FieldOperand(rbx, Map::kInstanceTypeOffset)); 2452 __ movzxbl(rbx, FieldOperand(rbx, Map::kInstanceTypeOffset));
2452 // rax: RegExp data (FixedArray) 2453 // rax: RegExp data (FixedArray)
2453 // rdi: subject string 2454 // rdi: subject string
2454 // r15: subject string 2455 // r15: subject string
2455 // Handle subject string according to its encoding and representation: 2456 // Handle subject string according to its encoding and representation:
2456 // (1) Sequential two byte? If yes, go to (9). 2457 // (1) Sequential two byte? If yes, go to (9).
2457 // (2) Sequential one byte? If yes, go to (6). 2458 // (2) Sequential one byte? If yes, go to (6).
(...skipping 81 matching lines...) Expand 10 before | Expand all | Expand 10 after
2539 2540
2540 // rdi: sequential subject string (or look-alike, external string) 2541 // rdi: sequential subject string (or look-alike, external string)
2541 // r15: original subject string 2542 // r15: original subject string
2542 // rcx: encoding of subject string (1 if ASCII, 0 if two_byte); 2543 // rcx: encoding of subject string (1 if ASCII, 0 if two_byte);
2543 // r11: code 2544 // r11: code
2544 // Load used arguments before starting to push arguments for call to native 2545 // Load used arguments before starting to push arguments for call to native
2545 // RegExp code to avoid handling changing stack height. 2546 // RegExp code to avoid handling changing stack height.
2546 // We have to use r15 instead of rdi to load the length because rdi might 2547 // We have to use r15 instead of rdi to load the length because rdi might
2547 // have been only made to look like a sequential string when it actually 2548 // have been only made to look like a sequential string when it actually
2548 // is an external string. 2549 // is an external string.
2549 __ movq(rbx, Operand(rsp, kPreviousIndexOffset)); 2550 __ movq(rbx, args.GetArgumentOperand(PREVIOUS_INDEX_ARGUMENT_INDEX));
2550 __ JumpIfNotSmi(rbx, &runtime); 2551 __ JumpIfNotSmi(rbx, &runtime);
2551 __ SmiCompare(rbx, FieldOperand(r15, String::kLengthOffset)); 2552 __ SmiCompare(rbx, FieldOperand(r15, String::kLengthOffset));
2552 __ j(above_equal, &runtime); 2553 __ j(above_equal, &runtime);
2553 __ SmiToInteger64(rbx, rbx); 2554 __ SmiToInteger64(rbx, rbx);
2554 2555
2555 // rdi: subject string 2556 // rdi: subject string
2556 // rbx: previous index 2557 // rbx: previous index
2557 // rcx: encoding of subject string (1 if ASCII 0 if two_byte); 2558 // rcx: encoding of subject string (1 if ASCII 0 if two_byte);
2558 // r11: code 2559 // r11: code
2559 // All checks done. Now push arguments for native regexp code. 2560 // All checks done. Now push arguments for native regexp code.
(...skipping 99 matching lines...) Expand 10 before | Expand all | Expand 10 after
2659 __ j(equal, &success, Label::kNear); 2660 __ j(equal, &success, Label::kNear);
2660 __ cmpl(rax, Immediate(NativeRegExpMacroAssembler::EXCEPTION)); 2661 __ cmpl(rax, Immediate(NativeRegExpMacroAssembler::EXCEPTION));
2661 __ j(equal, &exception); 2662 __ j(equal, &exception);
2662 __ cmpl(rax, Immediate(NativeRegExpMacroAssembler::FAILURE)); 2663 __ cmpl(rax, Immediate(NativeRegExpMacroAssembler::FAILURE));
2663 // If none of the above, it can only be retry. 2664 // If none of the above, it can only be retry.
2664 // Handle that in the runtime system. 2665 // Handle that in the runtime system.
2665 __ j(not_equal, &runtime); 2666 __ j(not_equal, &runtime);
2666 2667
2667 // For failure return null. 2668 // For failure return null.
2668 __ LoadRoot(rax, Heap::kNullValueRootIndex); 2669 __ LoadRoot(rax, Heap::kNullValueRootIndex);
2669 __ ret(4 * kPointerSize); 2670 __ ret(REG_EXP_EXEC_ARGUMENT_COUNT * kPointerSize);
2670 2671
2671 // Load RegExp data. 2672 // Load RegExp data.
2672 __ bind(&success); 2673 __ bind(&success);
2673 __ movq(rax, Operand(rsp, kJSRegExpOffset)); 2674 __ movq(rax, args.GetArgumentOperand(JS_REG_EXP_OBJECT_ARGUMENT_INDEX));
2674 __ movq(rcx, FieldOperand(rax, JSRegExp::kDataOffset)); 2675 __ movq(rcx, FieldOperand(rax, JSRegExp::kDataOffset));
2675 __ SmiToInteger32(rax, 2676 __ SmiToInteger32(rax,
2676 FieldOperand(rcx, JSRegExp::kIrregexpCaptureCountOffset)); 2677 FieldOperand(rcx, JSRegExp::kIrregexpCaptureCountOffset));
2677 // Calculate number of capture registers (number_of_captures + 1) * 2. 2678 // Calculate number of capture registers (number_of_captures + 1) * 2.
2678 __ leal(rdx, Operand(rax, rax, times_1, 2)); 2679 __ leal(rdx, Operand(rax, rax, times_1, 2));
2679 2680
2680 // rdx: Number of capture registers 2681 // rdx: Number of capture registers
2681 // Check that the fourth object is a JSArray object. 2682 // Check that the fourth object is a JSArray object.
2682 __ movq(r15, Operand(rsp, kLastMatchInfoOffset)); 2683 __ movq(r15, args.GetArgumentOperand(LAST_MATCH_INFO_ARGUMENT_INDEX));
2683 __ JumpIfSmi(r15, &runtime); 2684 __ JumpIfSmi(r15, &runtime);
2684 __ CmpObjectType(r15, JS_ARRAY_TYPE, kScratchRegister); 2685 __ CmpObjectType(r15, JS_ARRAY_TYPE, kScratchRegister);
2685 __ j(not_equal, &runtime); 2686 __ j(not_equal, &runtime);
2686 // Check that the JSArray is in fast case. 2687 // Check that the JSArray is in fast case.
2687 __ movq(rbx, FieldOperand(r15, JSArray::kElementsOffset)); 2688 __ movq(rbx, FieldOperand(r15, JSArray::kElementsOffset));
2688 __ movq(rax, FieldOperand(rbx, HeapObject::kMapOffset)); 2689 __ movq(rax, FieldOperand(rbx, HeapObject::kMapOffset));
2689 __ CompareRoot(rax, Heap::kFixedArrayMapRootIndex); 2690 __ CompareRoot(rax, Heap::kFixedArrayMapRootIndex);
2690 __ j(not_equal, &runtime); 2691 __ j(not_equal, &runtime);
2691 // Check that the last match info has space for the capture registers and the 2692 // Check that the last match info has space for the capture registers and the
2692 // additional information. Ensure no overflow in add. 2693 // additional information. Ensure no overflow in add.
2693 STATIC_ASSERT(FixedArray::kMaxLength < kMaxInt - FixedArray::kLengthOffset); 2694 STATIC_ASSERT(FixedArray::kMaxLength < kMaxInt - FixedArray::kLengthOffset);
2694 __ SmiToInteger32(rax, FieldOperand(rbx, FixedArray::kLengthOffset)); 2695 __ SmiToInteger32(rax, FieldOperand(rbx, FixedArray::kLengthOffset));
2695 __ subl(rax, Immediate(RegExpImpl::kLastMatchOverhead)); 2696 __ subl(rax, Immediate(RegExpImpl::kLastMatchOverhead));
2696 __ cmpl(rdx, rax); 2697 __ cmpl(rdx, rax);
2697 __ j(greater, &runtime); 2698 __ j(greater, &runtime);
2698 2699
2699 // rbx: last_match_info backing store (FixedArray) 2700 // rbx: last_match_info backing store (FixedArray)
2700 // rdx: number of capture registers 2701 // rdx: number of capture registers
2701 // Store the capture count. 2702 // Store the capture count.
2702 __ Integer32ToSmi(kScratchRegister, rdx); 2703 __ Integer32ToSmi(kScratchRegister, rdx);
2703 __ movq(FieldOperand(rbx, RegExpImpl::kLastCaptureCountOffset), 2704 __ movq(FieldOperand(rbx, RegExpImpl::kLastCaptureCountOffset),
2704 kScratchRegister); 2705 kScratchRegister);
2705 // Store last subject and last input. 2706 // Store last subject and last input.
2706 __ movq(rax, Operand(rsp, kSubjectOffset)); 2707 __ movq(rax, args.GetArgumentOperand(SUBJECT_STRING_ARGUMENT_INDEX));
2707 __ movq(FieldOperand(rbx, RegExpImpl::kLastSubjectOffset), rax); 2708 __ movq(FieldOperand(rbx, RegExpImpl::kLastSubjectOffset), rax);
2708 __ movq(rcx, rax); 2709 __ movq(rcx, rax);
2709 __ RecordWriteField(rbx, 2710 __ RecordWriteField(rbx,
2710 RegExpImpl::kLastSubjectOffset, 2711 RegExpImpl::kLastSubjectOffset,
2711 rax, 2712 rax,
2712 rdi, 2713 rdi,
2713 kDontSaveFPRegs); 2714 kDontSaveFPRegs);
2714 __ movq(rax, rcx); 2715 __ movq(rax, rcx);
2715 __ movq(FieldOperand(rbx, RegExpImpl::kLastInputOffset), rax); 2716 __ movq(FieldOperand(rbx, RegExpImpl::kLastInputOffset), rax);
2716 __ RecordWriteField(rbx, 2717 __ RecordWriteField(rbx,
(...skipping 22 matching lines...) Expand all
2739 __ movq(FieldOperand(rbx, 2740 __ movq(FieldOperand(rbx,
2740 rdx, 2741 rdx,
2741 times_pointer_size, 2742 times_pointer_size,
2742 RegExpImpl::kFirstCaptureOffset), 2743 RegExpImpl::kFirstCaptureOffset),
2743 rdi); 2744 rdi);
2744 __ jmp(&next_capture); 2745 __ jmp(&next_capture);
2745 __ bind(&done); 2746 __ bind(&done);
2746 2747
2747 // Return last match info. 2748 // Return last match info.
2748 __ movq(rax, r15); 2749 __ movq(rax, r15);
2749 __ ret(4 * kPointerSize); 2750 __ ret(REG_EXP_EXEC_ARGUMENT_COUNT * kPointerSize);
2750 2751
2751 __ bind(&exception); 2752 __ bind(&exception);
2752 // Result must now be exception. If there is no pending exception already a 2753 // Result must now be exception. If there is no pending exception already a
2753 // stack overflow (on the backtrack stack) was detected in RegExp code but 2754 // stack overflow (on the backtrack stack) was detected in RegExp code but
2754 // haven't created the exception yet. Handle that in the runtime system. 2755 // haven't created the exception yet. Handle that in the runtime system.
2755 // TODO(592): Rerunning the RegExp to get the stack overflow exception. 2756 // TODO(592): Rerunning the RegExp to get the stack overflow exception.
2756 ExternalReference pending_exception_address( 2757 ExternalReference pending_exception_address(
2757 Isolate::kPendingExceptionAddress, isolate); 2758 Isolate::kPendingExceptionAddress, isolate);
2758 Operand pending_exception_operand = 2759 Operand pending_exception_operand =
2759 masm->ExternalOperand(pending_exception_address, rbx); 2760 masm->ExternalOperand(pending_exception_address, rbx);
(...skipping 2074 matching lines...) Expand 10 before | Expand all | Expand 10 after
4834 4835
4835 void SubStringStub::Generate(MacroAssembler* masm) { 4836 void SubStringStub::Generate(MacroAssembler* masm) {
4836 Label runtime; 4837 Label runtime;
4837 4838
4838 // Stack frame on entry. 4839 // Stack frame on entry.
4839 // rsp[0] : return address 4840 // rsp[0] : return address
4840 // rsp[8] : to 4841 // rsp[8] : to
4841 // rsp[16] : from 4842 // rsp[16] : from
4842 // rsp[24] : string 4843 // rsp[24] : string
4843 4844
4844 const int kToOffset = 1 * kPointerSize; 4845 enum SubStringStubArgumentIndices {
4845 const int kFromOffset = kToOffset + kPointerSize; 4846 STRING_ARGUMENT_INDEX,
4846 const int kStringOffset = kFromOffset + kPointerSize; 4847 FROM_ARGUMENT_INDEX,
4847 const int kArgumentsSize = (kStringOffset + kPointerSize) - kToOffset; 4848 TO_ARGUMENT_INDEX,
4849 SUB_STRING_ARGUMENT_COUNT
4850 };
4851
4852 StackArgumentsAccessor args(rsp, SUB_STRING_ARGUMENT_COUNT,
4853 ARGUMENTS_DONT_CONTAIN_RECEIVER);
4848 4854
4849 // Make sure first argument is a string. 4855 // Make sure first argument is a string.
4850 __ movq(rax, Operand(rsp, kStringOffset)); 4856 __ movq(rax, args.GetArgumentOperand(STRING_ARGUMENT_INDEX));
4851 STATIC_ASSERT(kSmiTag == 0); 4857 STATIC_ASSERT(kSmiTag == 0);
4852 __ testl(rax, Immediate(kSmiTagMask)); 4858 __ testl(rax, Immediate(kSmiTagMask));
4853 __ j(zero, &runtime); 4859 __ j(zero, &runtime);
4854 Condition is_string = masm->IsObjectStringType(rax, rbx, rbx); 4860 Condition is_string = masm->IsObjectStringType(rax, rbx, rbx);
4855 __ j(NegateCondition(is_string), &runtime); 4861 __ j(NegateCondition(is_string), &runtime);
4856 4862
4857 // rax: string 4863 // rax: string
4858 // rbx: instance type 4864 // rbx: instance type
4859 // Calculate length of sub string using the smi values. 4865 // Calculate length of sub string using the smi values.
4860 __ movq(rcx, Operand(rsp, kToOffset)); 4866 __ movq(rcx, args.GetArgumentOperand(TO_ARGUMENT_INDEX));
4861 __ movq(rdx, Operand(rsp, kFromOffset)); 4867 __ movq(rdx, args.GetArgumentOperand(FROM_ARGUMENT_INDEX));
4862 __ JumpUnlessBothNonNegativeSmi(rcx, rdx, &runtime); 4868 __ JumpUnlessBothNonNegativeSmi(rcx, rdx, &runtime);
4863 4869
4864 __ SmiSub(rcx, rcx, rdx); // Overflow doesn't happen. 4870 __ SmiSub(rcx, rcx, rdx); // Overflow doesn't happen.
4865 __ cmpq(rcx, FieldOperand(rax, String::kLengthOffset)); 4871 __ cmpq(rcx, FieldOperand(rax, String::kLengthOffset));
4866 Label not_original_string; 4872 Label not_original_string;
4867 // Shorter than original string's length: an actual substring. 4873 // Shorter than original string's length: an actual substring.
4868 __ j(below, &not_original_string, Label::kNear); 4874 __ j(below, &not_original_string, Label::kNear);
4869 // Longer than original string's length or negative: unsafe arguments. 4875 // Longer than original string's length or negative: unsafe arguments.
4870 __ j(above, &runtime); 4876 __ j(above, &runtime);
4871 // Return original string. 4877 // Return original string.
4872 Counters* counters = masm->isolate()->counters(); 4878 Counters* counters = masm->isolate()->counters();
4873 __ IncrementCounter(counters->sub_string_native(), 1); 4879 __ IncrementCounter(counters->sub_string_native(), 1);
4874 __ ret(kArgumentsSize); 4880 __ ret(SUB_STRING_ARGUMENT_COUNT * kPointerSize);
4875 __ bind(&not_original_string); 4881 __ bind(&not_original_string);
4876 4882
4877 Label single_char; 4883 Label single_char;
4878 __ SmiCompare(rcx, Smi::FromInt(1)); 4884 __ SmiCompare(rcx, Smi::FromInt(1));
4879 __ j(equal, &single_char); 4885 __ j(equal, &single_char);
4880 4886
4881 __ SmiToInteger32(rcx, rcx); 4887 __ SmiToInteger32(rcx, rcx);
4882 4888
4883 // rax: string 4889 // rax: string
4884 // rbx: instance type 4890 // rbx: instance type
(...skipping 62 matching lines...) Expand 10 before | Expand all | Expand 10 after
4947 __ bind(&two_byte_slice); 4953 __ bind(&two_byte_slice);
4948 __ AllocateTwoByteSlicedString(rax, rbx, r14, &runtime); 4954 __ AllocateTwoByteSlicedString(rax, rbx, r14, &runtime);
4949 __ bind(&set_slice_header); 4955 __ bind(&set_slice_header);
4950 __ Integer32ToSmi(rcx, rcx); 4956 __ Integer32ToSmi(rcx, rcx);
4951 __ movq(FieldOperand(rax, SlicedString::kLengthOffset), rcx); 4957 __ movq(FieldOperand(rax, SlicedString::kLengthOffset), rcx);
4952 __ movq(FieldOperand(rax, SlicedString::kHashFieldOffset), 4958 __ movq(FieldOperand(rax, SlicedString::kHashFieldOffset),
4953 Immediate(String::kEmptyHashField)); 4959 Immediate(String::kEmptyHashField));
4954 __ movq(FieldOperand(rax, SlicedString::kParentOffset), rdi); 4960 __ movq(FieldOperand(rax, SlicedString::kParentOffset), rdi);
4955 __ movq(FieldOperand(rax, SlicedString::kOffsetOffset), rdx); 4961 __ movq(FieldOperand(rax, SlicedString::kOffsetOffset), rdx);
4956 __ IncrementCounter(counters->sub_string_native(), 1); 4962 __ IncrementCounter(counters->sub_string_native(), 1);
4957 __ ret(kArgumentsSize); 4963 __ ret(3 * kPointerSize);
4958 4964
4959 __ bind(&copy_routine); 4965 __ bind(&copy_routine);
4960 } 4966 }
4961 4967
4962 // rdi: underlying subject string 4968 // rdi: underlying subject string
4963 // rbx: instance type of underlying subject string 4969 // rbx: instance type of underlying subject string
4964 // rdx: adjusted start index (smi) 4970 // rdx: adjusted start index (smi)
4965 // rcx: length 4971 // rcx: length
4966 // The subject string can only be external or sequential string of either 4972 // The subject string can only be external or sequential string of either
4967 // encoding at this point. 4973 // encoding at this point.
(...skipping 33 matching lines...) Expand 10 before | Expand all | Expand 10 after
5001 __ lea(rdi, FieldOperand(rax, SeqOneByteString::kHeaderSize)); 5007 __ lea(rdi, FieldOperand(rax, SeqOneByteString::kHeaderSize));
5002 5008
5003 // rax: result string 5009 // rax: result string
5004 // rcx: result length 5010 // rcx: result length
5005 // rdi: first character of result 5011 // rdi: first character of result
5006 // rsi: character of sub string start 5012 // rsi: character of sub string start
5007 // r14: original value of rsi 5013 // r14: original value of rsi
5008 StringHelper::GenerateCopyCharactersREP(masm, rdi, rsi, rcx, true); 5014 StringHelper::GenerateCopyCharactersREP(masm, rdi, rsi, rcx, true);
5009 __ movq(rsi, r14); // Restore rsi. 5015 __ movq(rsi, r14); // Restore rsi.
5010 __ IncrementCounter(counters->sub_string_native(), 1); 5016 __ IncrementCounter(counters->sub_string_native(), 1);
5011 __ ret(kArgumentsSize); 5017 __ ret(SUB_STRING_ARGUMENT_COUNT * kPointerSize);
5012 5018
5013 __ bind(&two_byte_sequential); 5019 __ bind(&two_byte_sequential);
5014 // Allocate the result. 5020 // Allocate the result.
5015 __ AllocateTwoByteString(rax, rcx, r11, r14, r15, &runtime); 5021 __ AllocateTwoByteString(rax, rcx, r11, r14, r15, &runtime);
5016 5022
5017 // rax: result string 5023 // rax: result string
5018 // rcx: result string length 5024 // rcx: result string length
5019 __ movq(r14, rsi); // esi used by following code. 5025 __ movq(r14, rsi); // esi used by following code.
5020 { // Locate character of sub string start. 5026 { // Locate character of sub string start.
5021 SmiIndex smi_as_index = masm->SmiToIndex(rdx, rdx, times_2); 5027 SmiIndex smi_as_index = masm->SmiToIndex(rdx, rdx, times_2);
5022 __ lea(rsi, Operand(rdi, smi_as_index.reg, smi_as_index.scale, 5028 __ lea(rsi, Operand(rdi, smi_as_index.reg, smi_as_index.scale,
5023 SeqOneByteString::kHeaderSize - kHeapObjectTag)); 5029 SeqOneByteString::kHeaderSize - kHeapObjectTag));
5024 } 5030 }
5025 // Locate first character of result. 5031 // Locate first character of result.
5026 __ lea(rdi, FieldOperand(rax, SeqTwoByteString::kHeaderSize)); 5032 __ lea(rdi, FieldOperand(rax, SeqTwoByteString::kHeaderSize));
5027 5033
5028 // rax: result string 5034 // rax: result string
5029 // rcx: result length 5035 // rcx: result length
5030 // rdi: first character of result 5036 // rdi: first character of result
5031 // rsi: character of sub string start 5037 // rsi: character of sub string start
5032 // r14: original value of rsi 5038 // r14: original value of rsi
5033 StringHelper::GenerateCopyCharactersREP(masm, rdi, rsi, rcx, false); 5039 StringHelper::GenerateCopyCharactersREP(masm, rdi, rsi, rcx, false);
5034 __ movq(rsi, r14); // Restore esi. 5040 __ movq(rsi, r14); // Restore esi.
5035 __ IncrementCounter(counters->sub_string_native(), 1); 5041 __ IncrementCounter(counters->sub_string_native(), 1);
5036 __ ret(kArgumentsSize); 5042 __ ret(SUB_STRING_ARGUMENT_COUNT * kPointerSize);
5037 5043
5038 // Just jump to runtime to create the sub string. 5044 // Just jump to runtime to create the sub string.
5039 __ bind(&runtime); 5045 __ bind(&runtime);
5040 __ TailCallRuntime(Runtime::kSubString, 3, 1); 5046 __ TailCallRuntime(Runtime::kSubString, 3, 1);
5041 5047
5042 __ bind(&single_char); 5048 __ bind(&single_char);
5043 // rax: string 5049 // rax: string
5044 // rbx: instance type 5050 // rbx: instance type
5045 // rcx: sub string length (smi) 5051 // rcx: sub string length (smi)
5046 // rdx: from index (smi) 5052 // rdx: from index (smi)
5047 StringCharAtGenerator generator( 5053 StringCharAtGenerator generator(
5048 rax, rdx, rcx, rax, &runtime, &runtime, &runtime, STRING_INDEX_IS_NUMBER); 5054 rax, rdx, rcx, rax, &runtime, &runtime, &runtime, STRING_INDEX_IS_NUMBER);
5049 generator.GenerateFast(masm); 5055 generator.GenerateFast(masm);
5050 __ ret(kArgumentsSize); 5056 __ ret(SUB_STRING_ARGUMENT_COUNT * kPointerSize);
5051 generator.SkipSlow(masm, &runtime); 5057 generator.SkipSlow(masm, &runtime);
5052 } 5058 }
5053 5059
5054 5060
5055 void StringCompareStub::GenerateFlatAsciiStringEquals(MacroAssembler* masm, 5061 void StringCompareStub::GenerateFlatAsciiStringEquals(MacroAssembler* masm,
5056 Register left, 5062 Register left,
5057 Register right, 5063 Register right,
5058 Register scratch1, 5064 Register scratch1,
5059 Register scratch2) { 5065 Register scratch2) {
5060 Register length = scratch1; 5066 Register length = scratch1;
(...skipping 1485 matching lines...) Expand 10 before | Expand all | Expand 10 after
6546 __ bind(&fast_elements_case); 6552 __ bind(&fast_elements_case);
6547 GenerateCase(masm, FAST_ELEMENTS); 6553 GenerateCase(masm, FAST_ELEMENTS);
6548 } 6554 }
6549 6555
6550 6556
6551 #undef __ 6557 #undef __
6552 6558
6553 } } // namespace v8::internal 6559 } } // namespace v8::internal
6554 6560
6555 #endif // V8_TARGET_ARCH_X64 6561 #endif // V8_TARGET_ARCH_X64
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698