/* 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.disassembly.Modules; import java.util.Date; import com.google.common.base.Preconditions; import com.google.security.zynamics.binnavi.CUtilityFunctions; import com.google.security.zynamics.binnavi.Database.Exceptions.CouldntLoadDataException; import com.google.security.zynamics.binnavi.Database.Exceptions.CouldntSaveDataException; import com.google.security.zynamics.binnavi.Database.Interfaces.SQLProvider; import com.google.security.zynamics.binnavi.debug.debugger.DebuggerTemplate; import com.google.security.zynamics.binnavi.debug.debugger.ModuleTargetSettings; import com.google.security.zynamics.binnavi.debug.debugger.TcpDebugger; import com.google.security.zynamics.binnavi.debug.debugger.interfaces.IDebugger; import com.google.security.zynamics.binnavi.disassembly.INaviModule; import com.google.security.zynamics.binnavi.disassembly.INaviModuleConfiguration; import com.google.security.zynamics.binnavi.disassembly.INaviRawModule; import com.google.security.zynamics.zylib.disassembly.IAddress; import com.google.security.zynamics.zylib.general.ListenerProvider; /** * Contains the configurable values of a module. */ public final class CModuleConfiguration implements INaviModuleConfiguration { /** * The module to configure. */ private final INaviModule m_module; /** * Synchronizes the configurable data with the database. */ private final SQLProvider m_provider; /** * Listeners that are notified about changes in the properties. */ private final ListenerProvider<IModuleListener> m_listeners; /** * The ID of the module as it can be found in the database. */ private final int m_id; /** * The name of the module. */ private String m_name; /** * The description of the module. */ private String m_description; /** * The date when the module was created. */ private final Date m_creationDate; /** * The date when the module was last modified. */ private Date m_modificationDate; /** * The MD5 hash of the original input file. */ private final String m_md5; /** * The SHA1 hash of the original input file. */ private final String m_sha1; /** * The file base of the module. */ private IAddress m_fileBase; /** * The image base of the module. */ private IAddress m_imageBase; /** * Flag that says whether the module is stared or not. */ private boolean m_isStared; /** * The debugger template that was used to create the debugger object that is used to debug this * module. */ private DebuggerTemplate m_debuggerTemplate = null; /** * The debugger object that can be used to debug this module. */ private TcpDebugger m_debugger = null; /** * Raw module which backs the module. This value can be null. */ private final INaviRawModule m_rawModule; /** * Creates a new configuration object. * * @param module The module to configure. * @param provider The SQL provider that is used to load more information about the module. * @param listeners Listeners that are notified about changes in the properties. * @param moduleId The ID of the module. * @param name The name of the module. * @param comment The module description. * @param creationDate The creation date of the module. * @param modificationDate The modification date of the module. * @param md5 The MD5 hash of the module input file. * @param sha1 The SHA1 hash of the module input file. * @param fileBase The file base of the module. * @param imageBase The image base of the module. * @param debuggerTemplate The comment associated with the module. * @param isStared Flag that says whether the module is stared or not. * @param rawModule Raw module which backs the module. This value can be null. */ public CModuleConfiguration(final INaviModule module, final SQLProvider provider, final ListenerProvider<IModuleListener> listeners, final int moduleId, final String name, final String comment, final Date creationDate, final Date modificationDate, final String md5, final String sha1, final IAddress fileBase, final IAddress imageBase, final DebuggerTemplate debuggerTemplate, final boolean isStared, final INaviRawModule rawModule) { m_module = module; m_provider = provider; m_listeners = listeners; m_id = moduleId; m_name = name; m_description = comment; m_creationDate = new Date(creationDate.getTime()); m_modificationDate = new Date(modificationDate.getTime()); m_md5 = md5; m_sha1 = sha1; m_fileBase = fileBase; m_imageBase = imageBase; m_debuggerTemplate = debuggerTemplate; m_isStared = isStared; m_rawModule = rawModule; updateDebugger(debuggerTemplate); } /** * Takes a debugger template and updates the debugger object of the module accordingly if * possible. * * @param template The debugger template that provides the debugger information. */ private void updateDebugger(final DebuggerTemplate template) { // The requirements that enable the construction of a new debugger object are // as follows: // // 1. There must be a debugger template. Having a null-template is possible too. // 2. No debugger exists yet or the existing debugger is not active. We can // not replace a debugger that's currently in use. if ((m_debugger == null) || !m_debugger.isConnected()) { if (template == null) { m_debugger = null; } else { m_debugger = new TcpDebugger(template, new ModuleTargetSettings(m_module)); m_debugger.setAddressTranslator(m_module, m_fileBase, m_imageBase); } for (final IModuleListener listener : m_listeners) { try { listener.changedDebugger(m_module, m_debugger); } catch (final Exception exception) { CUtilityFunctions.logException(exception); } } } } @Override public Date getCreationDate() { return new Date(m_creationDate.getTime()); } @Override public IDebugger getDebugger() { return m_debugger; } @Override public DebuggerTemplate getDebuggerTemplate() { return m_debuggerTemplate; } @Override public String getDescription() { return m_description; } @Override public IAddress getFileBase() { return m_fileBase; } @Override public int getId() { return m_id; } @Override public IAddress getImageBase() { return m_imageBase; } @Override public String getMD5() { return m_md5; } @Override public Date getModificationDate() { return new Date(m_modificationDate.getTime()); } @Override public String getName() { return m_name; } @Override public INaviRawModule getRawModule() { return m_rawModule; } @Override public String getSha1() { return m_sha1; } @Override public boolean isStared() { return m_isStared; } @Override public void setDebuggerTemplate(final DebuggerTemplate template) throws CouldntSaveDataException { if (template == m_debuggerTemplate) { return; } m_provider.assignDebugger(m_module, template); m_debuggerTemplate = template; for (final IModuleListener listener : m_listeners) { try { listener.changedDebuggerTemplate(m_module, template); } catch (final Exception exception) { CUtilityFunctions.logException(exception); } } updateDebugger(template); updateModificationDate(); } @Override public void setDescription(final String description) throws CouldntSaveDataException { Preconditions.checkNotNull(description, "IE00199: Description string can not be null"); // Do nothing if the old description equals the new description. if (description.equals(m_description)) { return; } m_provider.setDescription(m_module, description); m_description = description; for (final IModuleListener listener : m_listeners) { try { listener.changedDescription(m_module, description); } catch (final Exception exception) { CUtilityFunctions.logException(exception); } } updateModificationDate(); } @Override public void setFileBase(final IAddress fileBase) throws CouldntSaveDataException { Preconditions.checkNotNull(fileBase, "IE00200: File base argument can not be null"); if (fileBase.equals(m_fileBase)) { return; } m_provider.setFileBase(m_module, fileBase); m_fileBase = fileBase; for (final IModuleListener listener : m_listeners) { try { listener.changedFileBase(m_module, fileBase); } catch (final Exception exception) { CUtilityFunctions.logException(exception); } } updateModificationDate(); } @Override public void setImageBase(final IAddress imageBase) throws CouldntSaveDataException { Preconditions.checkNotNull(imageBase, "IE00201: Image base argument can not be null"); if (imageBase.equals(m_imageBase)) { return; } m_provider.setImageBase(m_module, imageBase); m_imageBase = imageBase; if (m_debugger != null) { m_debugger.setAddressTranslator(m_module, m_fileBase, m_imageBase); } for (final IModuleListener listener : m_listeners) { try { listener.changedImageBase(m_module, imageBase); } catch (final Exception exception) { CUtilityFunctions.logException(exception); } } updateModificationDate(); } @Override public void setName(final String name) throws CouldntSaveDataException { Preconditions.checkNotNull(name, "IE00202: Name string can not be null"); // Do nothing if the old name equals the new name. if (name.equals(m_name)) { return; } m_provider.setName(m_module, name); m_name = name; for (final IModuleListener listener : m_listeners) { try { listener.changedName(m_module, name); } catch (final Exception exception) { CUtilityFunctions.logException(exception); } } updateModificationDate(); } @Override public void setStared(final boolean isStared) throws CouldntSaveDataException { if (m_isStared == isStared) { return; } m_provider.setStared(m_module, isStared); m_isStared = isStared; for (final IModuleListener listener : m_listeners) { try { listener.changedStarState(m_module, isStared); } catch (final Exception exception) { CUtilityFunctions.logException(exception); } } } @Override public void updateModificationDate() { try { m_modificationDate = m_provider.getModificationDate(m_module); for (final IModuleListener listener : m_listeners) { try { listener.changedModificationDate(m_module, m_modificationDate); } catch (final Exception exception) { CUtilityFunctions.logException(exception); } } } catch (final CouldntLoadDataException e) { CUtilityFunctions.logException(e); } } }