/*
Copyright 2011-2016 Google Inc. All Rights Reserved.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package com.google.security.zynamics.binnavi.debug.debugger;
import com.google.common.base.Preconditions;
import com.google.security.zynamics.binnavi.debug.connection.DebugConnection;
import com.google.security.zynamics.binnavi.debug.debugger.interfaces.IDebuggerTemplateListener;
/**
* Debugger class that communicates with a debug client via TCP/IP.
*/
public final class TcpDebugger extends AbstractDebugger {
/**
* The debug connection that is used to communicate with the debug client.
*/
private DebugConnection debuggerConnection;
/**
* Template that was used to create the debugger.
*/
private final DebuggerTemplate template;
/**
* Listens on relevant changes in the template.
*/
private final IDebuggerTemplateListener internalTemplateListener = new InternalTemplateListener();
/**
* The settings for the debug target.
*/
private final DebugTargetSettings debugTargetSettings;
/**
* Creates a new TCP/IP debugger object from the given debugger template. If the debugger template
* changes, the debugger is updated automatically.
*
* @param template The debugger template that describes the debug connection information.
* @param targetSettings The target settings.
*/
public TcpDebugger(final DebuggerTemplate template, final DebugTargetSettings targetSettings) {
this.template =
Preconditions.checkNotNull(template, "IE00818: Debugger template argument can not be null");
debuggerConnection = new DebugConnection(this.template.getHost(), this.template.getPort());
debugTargetSettings = Preconditions.checkNotNull(targetSettings,
"IE01670: targetSettings argument can not be null");
this.template.addListener(internalTemplateListener);
}
/**
* Updates the debugger connection if possible after new template data was received.
*/
private void updateConnection() {
if ((debuggerConnection == null) || !isConnected()) {
debuggerConnection = new DebugConnection(template.getHost(), template.getPort());
}
}
@Override
public void close() {
super.setTerminated();
debuggerConnection = null;
}
@Override
public void connect() throws DebugExceptionWrapper {
super.connect(debuggerConnection);
}
@Override
public DebugTargetSettings getDebugTargetSettings() {
return debugTargetSettings;
}
/**
* Returns the template Id from the database so this debugger instance can be uniquely identified.
*/
@Override
public int getId() {
return template.getId();
}
@Override
public String getPrintableString() {
return String.format("%s (%s:%d)", template.getName(), template.getHost(), template.getPort());
}
@Override
public String toString() {
return getPrintableString();
}
/**
* Updates the debugger object when the debugger template changes.
*/
private class InternalTemplateListener implements IDebuggerTemplateListener {
@Override
public void changedHost(final DebuggerTemplate debugger) {
updateConnection();
}
@Override
public void changedName(final DebuggerTemplate debugger) {
// Updating the connection is not necessary if the name of the
// debugger changed.
}
@Override
public void changedPort(final DebuggerTemplate debugger) {
updateConnection();
}
}
}