/*
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.Gui.Debug.CombinedMemoryPanel;
import java.awt.BorderLayout;
import javax.swing.JFrame;
import com.google.common.base.Preconditions;
import com.google.security.zynamics.binnavi.Gui.Debug.MemoryRefreshButton.CMemoryRefreshButtonPanel;
import com.google.security.zynamics.binnavi.Gui.Debug.MemoryRefreshButton.IRefreshRangeProvider;
import com.google.security.zynamics.binnavi.Gui.Debug.MemorySelectionPanel.CMemorySelectionPanel;
import com.google.security.zynamics.binnavi.Gui.Debug.StackPanel.CStackView;
import com.google.security.zynamics.binnavi.Gui.GraphWindows.BottomPanel.CAbstractResultsPanel;
import com.google.security.zynamics.binnavi.Gui.GraphWindows.Panels.CDebugPerspectiveModel;
import com.google.security.zynamics.binnavi.Gui.GraphWindows.Panels.CDebugPerspectiveModelListenerAdapter;
import com.google.security.zynamics.binnavi.Gui.GraphWindows.Panels.IDebugPerspectiveModelListener;
import com.google.security.zynamics.binnavi.debug.debugger.interfaces.IDebugger;
import com.google.security.zynamics.binnavi.debug.models.processmanager.MemoryMap;
import com.google.security.zynamics.binnavi.debug.models.processmanager.MemorySection;
import com.google.security.zynamics.binnavi.debug.models.processmanager.ProcessHelpers;
import com.google.security.zynamics.zylib.disassembly.CAddress;
import com.google.security.zynamics.zylib.disassembly.IAddress;
import com.jidesoft.swing.JideSplitPane;
/**
* This panel contains a memory selection panel and a stack panel. This is the default panel shown
* when the Debug GUI perspective is activated in a graph window.
*
* In this panel the user can view the memory of the current debug target and the stack of the
* currently selected thread.
*/
public final class CCombinedMemoryPanel extends CAbstractResultsPanel {
/**
* Used for serialization.
*/
private static final long serialVersionUID = 668134668181649929L;
/**
* Panel where the memory of the current debug target is shown.
*/
private final CMemorySelectionPanel m_memorySelectionPanel;
/**
* Panel where the stack of the currently selected thread is shown.
*/
private final CStackView m_stackPanel;
/**
* Describes the active debugger GUI options.
*/
private final CDebugPerspectiveModel m_debugPerspectiveModel;
/**
* Makes sure that the memory panel comes visible on changing memory addresses.
*/
private final IDebugPerspectiveModelListener m_internalListener =
new InternalDebugPerspectiveModelListener();
/**
* Creates a new combined memory panel object.
*
* @param parent Parent window of the panel.
* @param debugPerspectiveModel Describes the active debugger GUI options.
*/
public CCombinedMemoryPanel(
final JFrame parent, final CDebugPerspectiveModel debugPerspectiveModel) {
super(new BorderLayout());
Preconditions.checkNotNull(parent, "IE01361: Parent argument can not be null");
m_debugPerspectiveModel = Preconditions.checkNotNull(
debugPerspectiveModel, "IE01362: Debug perspective model argument can not be null");
m_debugPerspectiveModel.addListener(m_internalListener);
final JideSplitPane pane = new JideSplitPane(JideSplitPane.HORIZONTAL_SPLIT) {
private static final long serialVersionUID = -1326165812499630343L;
// ESCA-JAVA0025: Workaround for Case 1168
@Override
public void updateUI() {
// Workaround for Case 1168: The mere presence of a JIDE component
// screws up the look and feel.
}
};
pane.setDividerSize(3); // ATTENTION: Part of the Case 1168 workaround
pane.setProportionalLayout(true);
final CMemoryRefreshButtonPanel refreshPanel = new CMemoryRefreshButtonPanel(
parent, debugPerspectiveModel, new InternalRangeProvider(),
new InternalStackRangeProvider());
m_memorySelectionPanel = new CMemorySelectionPanel(parent, debugPerspectiveModel, refreshPanel);
// Create the GUI
pane.addPane(m_memorySelectionPanel);
m_stackPanel = new CStackView(debugPerspectiveModel);
pane.addPane(m_stackPanel);
add(pane);
}
@Override
public void dispose() {
m_memorySelectionPanel.dispose();
m_stackPanel.dispose();
}
@Override
public String getTitle() {
return "Memory";
}
/**
* Makes sure that the memory panel comes visible on changing memory addresses.
*/
private class InternalDebugPerspectiveModelListener
extends CDebugPerspectiveModelListenerAdapter {
@Override
public void changedActiveAddress(final IAddress address, final boolean focusMemoryWindow) {
if (address != null) {
notifyShow(getComponent());
}
}
}
/**
* Provides the memory range to refresh to the refresh button.
*/
private class InternalRangeProvider implements IRefreshRangeProvider {
@Override
public IAddress getAddress() {
return new CAddress(m_memorySelectionPanel.getMemoryPanel()
.getHexView().getHexView().getFirstVisibleOffset());
}
@Override
public int getSize() {
return m_memorySelectionPanel.getMemoryPanel().getHexView().getHexView().getVisibleBytes();
}
}
/**
* Provides the stack memory range to refresh to the refresh button.
*/
private class InternalStackRangeProvider implements IRefreshRangeProvider {
@Override
public IAddress getAddress() {
final long startAddress = m_stackPanel.getStackProvider().getStartAddress();
return startAddress == -1 ? null : new CAddress(startAddress);
}
@Override
public int getSize() {
final IDebugger activeDebugger = m_debugPerspectiveModel.getCurrentSelectedDebugger();
if (activeDebugger == null) {
return 0;
}
final MemoryMap memoryMap = activeDebugger.getProcessManager().getMemoryMap();
if (memoryMap == null) {
return 0;
}
final MemorySection section = ProcessHelpers.getSectionWith(memoryMap, getAddress());
return section == null ? 0 : section.getSize();
}
}
}