Package org.chromium.sdk.internal.v8native

Source Code of org.chromium.sdk.internal.v8native.CallFrameImpl

// Copyright (c) 2009 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

package org.chromium.sdk.internal.v8native;

import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.concurrent.atomic.AtomicReference;

import org.chromium.sdk.CallFrame;
import org.chromium.sdk.DebugContext;
import org.chromium.sdk.InvalidContextException;
import org.chromium.sdk.JsEvaluateContext;
import org.chromium.sdk.JsScope;
import org.chromium.sdk.JsVariable;
import org.chromium.sdk.RelayOk;
import org.chromium.sdk.RestartFrameExtension;
import org.chromium.sdk.Script;
import org.chromium.sdk.SyncCallback;
import org.chromium.sdk.TextStreamPosition;
import org.chromium.sdk.internal.protocolparser.JsonProtocolParseException;
import org.chromium.sdk.internal.v8native.InternalContext.ContextDismissedCheckedException;
import org.chromium.sdk.internal.v8native.processor.BacktraceProcessor;
import org.chromium.sdk.internal.v8native.protocol.V8ProtocolUtil;
import org.chromium.sdk.internal.v8native.protocol.input.FailedCommandResponse.ErrorDetails;
import org.chromium.sdk.internal.v8native.protocol.input.FrameObject;
import org.chromium.sdk.internal.v8native.protocol.input.RestartFrameBody;
import org.chromium.sdk.internal.v8native.protocol.input.ScopeRef;
import org.chromium.sdk.internal.v8native.protocol.input.SuccessCommandResponse;
import org.chromium.sdk.internal.v8native.protocol.output.DebuggerMessage;
import org.chromium.sdk.internal.v8native.protocol.output.DebuggerMessageFactory;
import org.chromium.sdk.internal.v8native.protocol.output.RestartFrameMessage;
import org.chromium.sdk.internal.v8native.value.JsScopeImpl;
import org.chromium.sdk.internal.v8native.value.JsVariableBase;
import org.chromium.sdk.internal.v8native.value.PropertyReference;
import org.chromium.sdk.internal.v8native.value.ValueLoader;
import org.chromium.sdk.internal.v8native.value.ValueMirror;
import org.chromium.sdk.util.GenericCallback;
import org.chromium.sdk.util.MethodIsBlockingException;
import org.chromium.sdk.util.RelaySyncCallback;
import org.json.simple.JSONObject;

/**
* A generic implementation of the CallFrame interface.
*/
public class CallFrameImpl implements CallFrame {

  /** The frame ID as reported by the JavaScript VM. */
  private final int frameId;

  /** The debug context this call frame belongs in. */
  private final InternalContext context;

  /** The underlying frame data from the JavaScript VM. */
  private final FrameObject frameObject;

  /**
   * 0-based line number in the entire script resource.
   */
  private final int lineNumber;

  /**
   * Function name associated with the frame.
   */
  private final String frameFunction;

  /**
   * The associated script id value.
   */
  private final long scriptId;

  /** The scopes known in this call frame. */
  private final AtomicReference<List<? extends JsScope>> scopesRef =
      new AtomicReference<List<? extends JsScope>>(null);

  /** The receiver variable known in this call frame. May be null. Null is not cached. */
  private final AtomicReference<JsVariable> receiverVariableRef =
      new AtomicReference<JsVariable>(null);

  /**
   * A script associated with the frame.
   */
  private Script script;

  /**
   * Constructs a call frame for the given handler using the FrameMirror data
   * from the remote JavaScript VM.
   *
   * @param mirror frame in the VM
   * @param index call frame id (0 is the stack top)
   * @param context in which the call frame is created
   */
  public CallFrameImpl(FrameObject frameObject, InternalContext context) {
    this.frameObject = frameObject;
    this.context = context;

    int index = (int) frameObject.index();
    JSONObject func = frameObject.func();

    int currentLine = (int) frameObject.line();

    // If we stopped because of the debuggerword then we're on the next
    // line.
    // TODO(apavlov): Terry says: we need to use the [e.g. Rhino] AST to
    // decide if line is debuggerword. If so, find the next sequential line.
    // The below works for simple scripts but doesn't take into account
    // comments, etc.
    // TODO(peter.rybin): do we really need this thing? (advancing to the next line?)
    //     stopping on "debugger;" seems to be a quite natural thing.
    String srcLine = frameObject.sourceLineText();
    if (srcLine.trim().startsWith(DEBUGGER_RESERVED)) {
      currentLine++;
    }
    Long scriptRef = V8ProtocolUtil.getObjectRef(frameObject.script());
    long scriptId =
        ScriptImpl.getScriptId(context.getValueLoader().getSpecialHandleManager(), scriptRef);

    this.scriptId = scriptId;
    this.lineNumber = currentLine;
    this.frameFunction = V8ProtocolUtil.getFunctionName(func);
    this.frameId = index;
  }

  public InternalContext getInternalContext() {
    return context;
  }

  @Override
  public List<? extends JsScope> getVariableScopes() {
    ensureScopes();
    return scopesRef.get();
  }

  @Override
  public JsVariable getReceiverVariable() throws MethodIsBlockingException {
    ensureReceiver();
    return receiverVariableRef.get();
  }

  @Override
  public JsEvaluateContext getEvaluateContext() {
    return evaluateContextImpl;
  }

  private void ensureScopes() {
    if (scopesRef.get() != null) {
      return;
    }
    List<? extends JsScope> result = Collections.unmodifiableList(createScopes());
    scopesRef.compareAndSet(null, result);
  }

  private void ensureReceiver() throws MethodIsBlockingException {
    if (receiverVariableRef.get() != null) {
      return;
    }
    JsVariable result;

    PropertyReference ref = V8Helper.computeReceiverRef(frameObject);
    if (ref == null) {
      result = null;
    } else {
      ValueLoader valueLoader = context.getValueLoader();
      ValueMirror mirror =
          valueLoader.getOrLoadValueFromRefs(Collections.singletonList(ref)).get(0);
      // This name should be string. We are making it string as a fall-back strategy.
      String varNameStr = ref.getName().toString();
      // 'this' variable is not mutable. Consider making it mutable.
      result = new JsVariableBase.Impl(valueLoader, mirror, varNameStr);
    }
    if (result != null) {
      receiverVariableRef.compareAndSet(null, result);
    }
  }

  @Override
  public TextStreamPosition getStatementStartPosition() {
    return textStreamPosition;
  }

  @Override
  public String getFunctionName() {
    return frameFunction;
  }

  @Override
  public Script getScript() {
    return script;
  }

  /**
   * @return this call frame's unique identifier within the V8 VM (0 is the top
   *         frame)
   */
  public int getIdentifier() {
    return frameId;
  }

  void hookUpScript(ScriptManager scriptManager) {
    Script script = scriptManager.findById(scriptId);
    if (script != null) {
      this.script = script;
    }
  }

  private List<JsScopeImpl<?>> createScopes() {
    List<ScopeRef> scopes = frameObject.scopes();
    List<JsScopeImpl<?>> result = new ArrayList<JsScopeImpl<?>>(scopes.size());
    for (ScopeRef scopeRef : scopes) {
      result.add(JsScopeImpl.create(JsScopeImpl.Host.create(this), scopeRef));
    }
    return result;
  }

  private final JsEvaluateContextImpl evaluateContextImpl = new JsEvaluateContextImpl() {
    @Override
    protected Integer getFrameIdentifier() {
      return getIdentifier();
    }
    @Override
    public InternalContext getInternalContext() {
      return context;
    }
  };

  private final TextStreamPosition textStreamPosition = new TextStreamPosition() {
    @Override public int getOffset() {
      return frameObject.position().intValue();
    }
    @Override public int getLine() {
      return lineNumber;
    }
    @Override public int getColumn() {
      Long columnObj = frameObject.column();
      if (columnObj == null) {
        return -1;
      }
      return columnObj.intValue();
    }
  };

  /**
   * Implements restart frame operation as chain of VM calls. After the main 'restart' command
   * it either calls 'step in' request or reloads backtrace. {@link RelaySyncCallback} is used
   * to ensure final sync callback call guarantee.
   */
  public static final RestartFrameExtension RESTART_FRAME_EXTENSION = new RestartFrameExtension() {
    @Override
    public RelayOk restartFrame(CallFrame callFrame,
        final GenericCallback<Boolean> callback, SyncCallback syncCallback) {
      final CallFrameImpl frameImpl = (CallFrameImpl) callFrame;
      final DebugSession debugSession = frameImpl.context.getDebugSession();

      RelaySyncCallback relaySyncCallback = new RelaySyncCallback(syncCallback);

      final RelaySyncCallback.Guard guard = relaySyncCallback.newGuard();

      RestartFrameMessage message = new RestartFrameMessage(frameImpl.frameId);
      V8CommandCallbackBase v8Callback = new V8CommandCallbackBase() {
        @Override
        public void success(SuccessCommandResponse successResponse) {
          RelayOk relayOk =
              handleRestartResponse(successResponse, debugSession, callback, guard.getRelay());
          guard.discharge(relayOk);
        }

        @Override
        public void failure(String message, ErrorDetails errorDetails) {
          if (callback != null) {
            callback.failure(new Exception(message));
          }
        }
      };

      try {
        return frameImpl.context.sendV8CommandAsync(message, false, v8Callback,
            guard.asSyncCallback());
      } catch (ContextDismissedCheckedException e) {
        throw new InvalidContextException(e);
      }
    }

    private RelayOk handleRestartResponse(SuccessCommandResponse successResponse,
        DebugSession debugSession,
        final GenericCallback<Boolean> callback, final RelaySyncCallback relaySyncCallback) {
      RestartFrameBody body;
      try {
        body = successResponse.body().asRestartFrameBody();
      } catch (JsonProtocolParseException e) {
        throw new RuntimeException(e);
      }

      InternalContext.UserContext debugContext =
          debugSession.getContextBuilder().getCurrentDebugContext();
      if (debugContext == null) {
        // We may have already issued 'continue' since the moment that change live command
        // was sent so the context was dropped. Ignore this case.
        return finishRestartSuccessfully(false, callback, relaySyncCallback);
      }

      RestartFrameBody.ResultDescription resultDescription = body.getResultDescription();

      if (body.getResultDescription().stack_update_needs_step_in() == Boolean.TRUE) {
        return stepIn(debugContext, callback, relaySyncCallback);
      } else {
        return reloadStack(debugContext, callback, relaySyncCallback);
      }
    }

    private RelayOk stepIn(InternalContext.UserContext debugContext,
        final GenericCallback<Boolean> callback, final RelaySyncCallback relaySyncCallback) {
      final RelaySyncCallback.Guard guard = relaySyncCallback.newGuard();
      DebugContext.ContinueCallback continueCallback = new DebugContext.ContinueCallback() {
        @Override
        public void success() {
          RelayOk relayOk = finishRestartSuccessfully(true, callback, relaySyncCallback);
          guard.discharge(relayOk);
        }
        @Override
        public void failure(String errorMessage) {
          if (callback != null) {
            callback.failure(new Exception(errorMessage));
          }
        }
      };
      return debugContext.continueVm(DebugContext.StepAction.IN, 0,
          continueCallback, guard.asSyncCallback());
    }

    private RelayOk reloadStack(InternalContext.UserContext debugContext,
        final GenericCallback<Boolean> callback, final RelaySyncCallback relaySyncCallback) {
      final RelaySyncCallback.Guard guard = relaySyncCallback.newGuard();
      final ContextBuilder.ExpectingBacktraceStep backtraceStep =
          debugContext.createReloadBacktraceStep();

      V8CommandCallbackBase v8Callback = new V8CommandCallbackBase() {
        @Override
        public void success(SuccessCommandResponse successResponse) {
          BacktraceProcessor.setFrames(successResponse, backtraceStep);
          RelayOk relayOk = finishRestartSuccessfully(false, callback, relaySyncCallback);
          guard.discharge(relayOk);
        }

        @Override
        public void failure(String message, ErrorDetails errorDetails) {
          if (callback != null) {
            callback.failure(new Exception(message));
          }
        }
      };

      DebuggerMessage message = DebuggerMessageFactory.backtrace(null, null, true);
      try {
        // Command is not immediate because we are supposed to be suspended.
        return debugContext.getInternalContext().sendV8CommandAsync(message, false,
            v8Callback, guard.asSyncCallback());
      } catch (ContextDismissedCheckedException e) {
        throw new InvalidContextException(e);
      }
    }

    private RelayOk finishRestartSuccessfully(boolean vmResumed,
        GenericCallback<Boolean> callback, RelaySyncCallback relaySyncCallback) {
      if (callback != null) {
        callback.success(vmResumed);
      }
      return relaySyncCallback.finish();
    }

    @Override
    public boolean canRestartFrame(CallFrame callFrame) {
      return callFrame.getScript() != null;
    }
  };

  private static final String DEBUGGER_RESERVED = "debugger";
}
TOP

Related Classes of org.chromium.sdk.internal.v8native.CallFrameImpl

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.