Dnspy/Extensions/dnSpy.AsmEditor/Hex/Nodes/PETreeNodeDataProvider.cs

96 lines
4.0 KiB
C#
Raw Permalink Normal View History

2021-09-20 18:20:01 +02:00
/*
Copyright (C) 2014-2019 de4dot@gmail.com
This file is part of dnSpy
dnSpy is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
dnSpy is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with dnSpy. If not, see <http://www.gnu.org/licenses/>.
*/
using System;
using System.Collections.Generic;
using System.ComponentModel.Composition;
using System.Diagnostics;
using dnSpy.AsmEditor.Hex.PE;
using dnSpy.Contracts.Documents;
using dnSpy.Contracts.Documents.TreeView;
using dnSpy.Contracts.Hex;
using dnSpy.Contracts.Hex.Files;
using dnSpy.Contracts.TreeView;
namespace dnSpy.AsmEditor.Hex.Nodes {
abstract class PETreeNodeDataProviderBase : ITreeNodeDataProvider {
readonly Lazy<IHexBufferService> hexBufferService;
readonly Lazy<PEStructureProviderFactory> peStructureProviderFactory;
readonly Lazy<HexBufferFileServiceFactory> hexBufferFileServiceFactory;
protected PETreeNodeDataProviderBase(Lazy<IHexBufferService> hexBufferService, Lazy<PEStructureProviderFactory> peStructureProviderFactory, Lazy<HexBufferFileServiceFactory> hexBufferFileServiceFactory) {
this.hexBufferService = hexBufferService;
this.peStructureProviderFactory = peStructureProviderFactory;
this.hexBufferFileServiceFactory = hexBufferFileServiceFactory;
}
public IEnumerable<TreeNodeData> Create(TreeNodeDataProviderContext context) {
var fileNode = context.Owner.Data as DsDocumentNode;
Debug2.Assert(fileNode is not null);
if (fileNode is null)
yield break;
bool hasPENode = HasPENode(fileNode);
var peImage = fileNode.Document.PEImage;
Debug2.Assert(!hasPENode || peImage is not null);
if (hasPENode && peImage is not null) {
Func<HexBufferFile> createBufferFile = () => {
var buffer = hexBufferService.Value.GetOrCreate(peImage);
var service = hexBufferFileServiceFactory.Value.Create(buffer);
var pePosition = HexPosition.Zero;
var bufferFile = service.GetFile(pePosition, checkNestedFiles: false);
Debug2.Assert(bufferFile is not null, "File hasn't been created");
return bufferFile;
};
yield return new PENode(createBufferFile, peStructureProviderFactory.Value);
}
}
public static bool HasPENode(DsDocumentNode node) {
if (node is null)
return false;
var peImage = node.Document.PEImage;
// Only show the PE node if it was loaded from a file. The hex document is always loaded
// from a file, so if the PEImage wasn't loaded from the same file, conversion to/from
// RVA/FileOffset won't work and the wrong data will be displayed, eg. in the .NET
// storage stream nodes.
bool loadedFromFile = node.Document.Key is FilenameKey;
return loadedFromFile && peImage is not null;
}
}
[ExportTreeNodeDataProvider(Guid = DocumentTreeViewConstants.MODULE_NODE_GUID)]
sealed class ModulePETreeNodeDataProvider : PETreeNodeDataProviderBase {
[ImportingConstructor]
ModulePETreeNodeDataProvider(Lazy<IHexBufferService> hexBufferService, Lazy<PEStructureProviderFactory> peStructureProviderFactory, Lazy<HexBufferFileServiceFactory> hexBufferFileServiceFactory)
: base(hexBufferService, peStructureProviderFactory, hexBufferFileServiceFactory) {
}
}
[ExportTreeNodeDataProvider(Guid = DocumentTreeViewConstants.PEDOCUMENT_NODE_GUID)]
sealed class PEFilePETreeNodeDataProvider : PETreeNodeDataProviderBase {
[ImportingConstructor]
PEFilePETreeNodeDataProvider(Lazy<IHexBufferService> hexBufferService, Lazy<PEStructureProviderFactory> peStructureProviderFactory, Lazy<HexBufferFileServiceFactory> hexBufferFileServiceFactory)
: base(hexBufferService, peStructureProviderFactory, hexBufferFileServiceFactory) {
}
}
}