| Index: src/ia32/macro-assembler-ia32.cc
 | 
| diff --git a/src/ia32/macro-assembler-ia32.cc b/src/ia32/macro-assembler-ia32.cc
 | 
| index bb4f71b3893eafdd23b9ba9d43f5886379781b65..2d4f4cdb2f551966f55b646299f27c3c014400c9 100644
 | 
| --- a/src/ia32/macro-assembler-ia32.cc
 | 
| +++ b/src/ia32/macro-assembler-ia32.cc
 | 
| @@ -725,84 +725,105 @@ void MacroAssembler::LeaveApiExitFrame() {
 | 
|  
 | 
|  
 | 
|  void MacroAssembler::PushTryHandler(CodeLocation try_location,
 | 
| -                                    HandlerType type) {
 | 
| +                                    HandlerType type,
 | 
| +                                    int handler_index) {
 | 
|    // Adjust this code if not the case.
 | 
|    STATIC_ASSERT(StackHandlerConstants::kSize == 5 * kPointerSize);
 | 
|    STATIC_ASSERT(StackHandlerConstants::kNextOffset == 0);
 | 
| -  STATIC_ASSERT(StackHandlerConstants::kContextOffset == 1 * kPointerSize);
 | 
| -  STATIC_ASSERT(StackHandlerConstants::kFPOffset == 2 * kPointerSize);
 | 
| -  STATIC_ASSERT(StackHandlerConstants::kStateOffset == 3 * kPointerSize);
 | 
| -  STATIC_ASSERT(StackHandlerConstants::kPCOffset == 4 * kPointerSize);
 | 
| -  // The pc (return address) is already on TOS.
 | 
| +  STATIC_ASSERT(StackHandlerConstants::kCodeOffset == 1 * kPointerSize);
 | 
| +  STATIC_ASSERT(StackHandlerConstants::kStateOffset == 2 * kPointerSize);
 | 
| +  STATIC_ASSERT(StackHandlerConstants::kContextOffset == 3 * kPointerSize);
 | 
| +  STATIC_ASSERT(StackHandlerConstants::kFPOffset == 4 * kPointerSize);
 | 
| +
 | 
| +  // We will build up the handler from the bottom by pushing on the stack.
 | 
| +  // First compute the state and push the frame pointer and context.
 | 
| +  unsigned state = StackHandler::OffsetField::encode(handler_index);
 | 
|    if (try_location == IN_JAVASCRIPT) {
 | 
| -    if (type == TRY_CATCH_HANDLER) {
 | 
| -      push(Immediate(StackHandler::TRY_CATCH));
 | 
| -    } else {
 | 
| -      push(Immediate(StackHandler::TRY_FINALLY));
 | 
| -    }
 | 
|      push(ebp);
 | 
|      push(esi);
 | 
| +    state |= (type == TRY_CATCH_HANDLER)
 | 
| +        ? StackHandler::KindField::encode(StackHandler::TRY_CATCH)
 | 
| +        : StackHandler::KindField::encode(StackHandler::TRY_FINALLY);
 | 
|    } else {
 | 
|      ASSERT(try_location == IN_JS_ENTRY);
 | 
| -    // The frame pointer does not point to a JS frame so we save NULL
 | 
| -    // for ebp. We expect the code throwing an exception to check ebp
 | 
| -    // before dereferencing it to restore the context.
 | 
| -    push(Immediate(StackHandler::ENTRY));
 | 
| +    // The frame pointer does not point to a JS frame so we save NULL for
 | 
| +    // ebp. We expect the code throwing an exception to check ebp before
 | 
| +    // dereferencing it to restore the context.
 | 
|      push(Immediate(0));  // NULL frame pointer.
 | 
|      push(Immediate(Smi::FromInt(0)));  // No context.
 | 
| +    state |= StackHandler::KindField::encode(StackHandler::ENTRY);
 | 
|    }
 | 
| -  // Save the current handler as the next handler.
 | 
| -  push(Operand::StaticVariable(ExternalReference(Isolate::kHandlerAddress,
 | 
| -                                                 isolate())));
 | 
| -  // Link this handler as the new current one.
 | 
| -  mov(Operand::StaticVariable(ExternalReference(Isolate::kHandlerAddress,
 | 
| -                                                isolate())),
 | 
| -      esp);
 | 
| +
 | 
| +  // Push the state and the code object.
 | 
| +  push(Immediate(state));
 | 
| +  push(CodeObject());
 | 
| +
 | 
| +  // Link the current handler as the next handler.
 | 
| +  ExternalReference handler_address(Isolate::kHandlerAddress, isolate());
 | 
| +  push(Operand::StaticVariable(handler_address));
 | 
| +  // Set this new handler as the current one.
 | 
| +  mov(Operand::StaticVariable(handler_address), esp);
 | 
|  }
 | 
|  
 | 
|  
 | 
|  void MacroAssembler::PopTryHandler() {
 | 
|    STATIC_ASSERT(StackHandlerConstants::kNextOffset == 0);
 | 
| -  pop(Operand::StaticVariable(ExternalReference(Isolate::kHandlerAddress,
 | 
| -                                                isolate())));
 | 
| +  ExternalReference handler_address(Isolate::kHandlerAddress, isolate());
 | 
| +  pop(Operand::StaticVariable(handler_address));
 | 
|    add(esp, Immediate(StackHandlerConstants::kSize - kPointerSize));
 | 
|  }
 | 
|  
 | 
|  
 | 
| +void MacroAssembler::JumpToHandlerEntry() {
 | 
| +  // Compute the handler entry address and jump to it.  The handler table is
 | 
| +  // a fixed array of (smi-tagged) code offsets.
 | 
| +  // eax = exception, edi = code object, edx = state.
 | 
| +  mov(ebx, FieldOperand(edi, Code::kHandlerTableOffset));
 | 
| +  shr(edx, StackHandler::kKindWidth);
 | 
| +  mov(edx, FieldOperand(ebx, edx, times_4, FixedArray::kHeaderSize));
 | 
| +  SmiUntag(edx);
 | 
| +  lea(edi, FieldOperand(edi, edx, times_1, Code::kHeaderSize));
 | 
| +  jmp(edi);
 | 
| +}
 | 
| +
 | 
| +
 | 
|  void MacroAssembler::Throw(Register value) {
 | 
|    // Adjust this code if not the case.
 | 
|    STATIC_ASSERT(StackHandlerConstants::kSize == 5 * kPointerSize);
 | 
|    STATIC_ASSERT(StackHandlerConstants::kNextOffset == 0);
 | 
| -  STATIC_ASSERT(StackHandlerConstants::kContextOffset == 1 * kPointerSize);
 | 
| -  STATIC_ASSERT(StackHandlerConstants::kFPOffset == 2 * kPointerSize);
 | 
| -  STATIC_ASSERT(StackHandlerConstants::kStateOffset == 3 * kPointerSize);
 | 
| -  STATIC_ASSERT(StackHandlerConstants::kPCOffset == 4 * kPointerSize);
 | 
| -  // eax must hold the exception.
 | 
| +  STATIC_ASSERT(StackHandlerConstants::kCodeOffset == 1 * kPointerSize);
 | 
| +  STATIC_ASSERT(StackHandlerConstants::kStateOffset == 2 * kPointerSize);
 | 
| +  STATIC_ASSERT(StackHandlerConstants::kContextOffset == 3 * kPointerSize);
 | 
| +  STATIC_ASSERT(StackHandlerConstants::kFPOffset == 4 * kPointerSize);
 | 
| +
 | 
| +  // The exception is expected in eax.
 | 
|    if (!value.is(eax)) {
 | 
|      mov(eax, value);
 | 
|    }
 | 
| -
 | 
| -  // Drop the sp to the top of the handler.
 | 
| -  ExternalReference handler_address(Isolate::kHandlerAddress,
 | 
| -                                    isolate());
 | 
| +  // Drop the stack pointer to the top of the top handler.
 | 
| +  ExternalReference handler_address(Isolate::kHandlerAddress, isolate());
 | 
|    mov(esp, Operand::StaticVariable(handler_address));
 | 
| -
 | 
| -  // Restore next handler, context, and frame pointer; discard handler state.
 | 
| +  // Restore the next handler.
 | 
|    pop(Operand::StaticVariable(handler_address));
 | 
| +
 | 
| +  // Remove the code object and state, compute the handler address in edi.
 | 
| +  pop(edi);  // Code object.
 | 
| +  pop(edx);  // Index and state.
 | 
| +
 | 
| +  // Restore the context and frame pointer.
 | 
|    pop(esi);  // Context.
 | 
|    pop(ebp);  // Frame pointer.
 | 
| -  pop(edx);  // State.
 | 
|  
 | 
|    // If the handler is a JS frame, restore the context to the frame.
 | 
| -  // (edx == ENTRY) == (ebp == 0) == (esi == 0), so we could test any
 | 
| -  // of them.
 | 
| +  // (kind == ENTRY) == (ebp == 0) == (esi == 0), so we could test either
 | 
| +  // ebp or esi.
 | 
|    Label skip;
 | 
| -  cmp(edx, Immediate(StackHandler::ENTRY));
 | 
| -  j(equal, &skip, Label::kNear);
 | 
| +  test(esi, esi);
 | 
| +  j(zero, &skip, Label::kNear);
 | 
|    mov(Operand(ebp, StandardFrameConstants::kContextOffset), esi);
 | 
|    bind(&skip);
 | 
|  
 | 
| -  ret(0);
 | 
| +  JumpToHandlerEntry();
 | 
|  }
 | 
|  
 | 
|  
 | 
| @@ -811,10 +832,10 @@ void MacroAssembler::ThrowUncatchable(UncatchableExceptionType type,
 | 
|    // Adjust this code if not the case.
 | 
|    STATIC_ASSERT(StackHandlerConstants::kSize == 5 * kPointerSize);
 | 
|    STATIC_ASSERT(StackHandlerConstants::kNextOffset == 0);
 | 
| -  STATIC_ASSERT(StackHandlerConstants::kContextOffset == 1 * kPointerSize);
 | 
| -  STATIC_ASSERT(StackHandlerConstants::kFPOffset == 2 * kPointerSize);
 | 
| -  STATIC_ASSERT(StackHandlerConstants::kStateOffset == 3 * kPointerSize);
 | 
| -  STATIC_ASSERT(StackHandlerConstants::kPCOffset == 4 * kPointerSize);
 | 
| +  STATIC_ASSERT(StackHandlerConstants::kCodeOffset == 1 * kPointerSize);
 | 
| +  STATIC_ASSERT(StackHandlerConstants::kStateOffset == 2 * kPointerSize);
 | 
| +  STATIC_ASSERT(StackHandlerConstants::kContextOffset == 3 * kPointerSize);
 | 
| +  STATIC_ASSERT(StackHandlerConstants::kFPOffset == 4 * kPointerSize);
 | 
|  
 | 
|    // The exception is expected in eax.
 | 
|    if (type == OUT_OF_MEMORY) {
 | 
| @@ -843,20 +864,23 @@ void MacroAssembler::ThrowUncatchable(UncatchableExceptionType type,
 | 
|    mov(esp, Operand(esp, StackHandlerConstants::kNextOffset));
 | 
|  
 | 
|    bind(&check_kind);
 | 
| -  cmp(Operand(esp, StackHandlerConstants::kStateOffset),
 | 
| -      Immediate(StackHandler::ENTRY));
 | 
| -  j(not_equal, &fetch_next);
 | 
| +  STATIC_ASSERT(StackHandler::ENTRY == 0);
 | 
| +  test(Operand(esp, StackHandlerConstants::kStateOffset),
 | 
| +       Immediate(StackHandler::KindField::kMask));
 | 
| +  j(not_zero, &fetch_next);
 | 
|  
 | 
|    // Set the top handler address to next handler past the top ENTRY handler.
 | 
|    pop(Operand::StaticVariable(handler_address));
 | 
|  
 | 
| -  // Clear the context and frame pointer (0 was saved in the handler), and
 | 
| -  // discard the state.
 | 
| +  // Remove the code object and state, compute the handler address in edi.
 | 
| +  pop(edi);  // Code object.
 | 
| +  pop(edx);  // Index and state.
 | 
| +
 | 
| +  // Clear the context pointer and frame pointer (0 was saved in the handler).
 | 
|    pop(esi);
 | 
|    pop(ebp);
 | 
| -  pop(edx);  // State.
 | 
|  
 | 
| -  ret(0);
 | 
| +  JumpToHandlerEntry();
 | 
|  }
 | 
|  
 | 
|  
 | 
| 
 |