15

ファイルをロードし、ロードされたファイルの作成日、変更日、サイズなどの情報を取得するC#プログラムを開発しています。もう1つ知っておく必要があるのは、ロードされたファイル( )がフラグexecutable.exeにリンクされているかどうかです。LARGEADDRESSAWAREFileInfoクラスはこの情報を提供しません。

LARGEADDRESSAWARE特定のexecutable.exeがフラグ(2 GBを超えるアドレスを処理するため)にリンクされているかどうかをC#で確認する方法を知っている人はいますか?

4

3 に答える 3

25

これは、LargeAddressAwareフラグをチェックするコードです。実行可能ファイルの先頭を指すストリームを渡すだけです。

IsLargeAware("some.exe");

static bool IsLargeAware(string file)
{
    using (var fs = File.OpenRead(file))
    {
        return IsLargeAware(fs);
    }
}
/// <summary>
/// Checks if the stream is a MZ header and if it is large address aware
/// </summary>
/// <param name="stream">Stream to check, make sure its at the start of the MZ header</param>
/// <exception cref=""></exception>
/// <returns></returns>
static bool IsLargeAware(Stream stream)
{
    const int IMAGE_FILE_LARGE_ADDRESS_AWARE = 0x20;

    var br = new BinaryReader(stream);

    if (br.ReadInt16() != 0x5A4D)       //No MZ Header
        return false;

    br.BaseStream.Position = 0x3C;
    var peloc = br.ReadInt32();         //Get the PE header location.

    br.BaseStream.Position = peloc;
    if (br.ReadInt32() != 0x4550)       //No PE header
        return false;

    br.BaseStream.Position += 0x12;
    return (br.ReadInt16() & IMAGE_FILE_LARGE_ADDRESS_AWARE) == IMAGE_FILE_LARGE_ADDRESS_AWARE;
}
于 2012-01-29T21:01:46.627 に答える
15

状態に関するMSDNドキュメント/LARGEADDRESSAWARE

アプリケーションが/LARGEADDRESSAWAREにリンクされている場合、DUMPBIN/HEADERSはその旨の情報を表示します。

これをプログラムで実行する方法を探している場合は、アプリケーションからdumpbinを呼び出して、出力を解析できます。

アップデート:

この問題をより深く議論する良いブログ投稿もここにあります。

于 2012-01-29T15:55:17.723 に答える
1

上記のWillの回答から、x86セットアップパッケージで次のものを使用しています。

    static bool LargeAware(string file) {
        using (var fs = File.Open(file, FileMode.Open, FileAccess.ReadWrite, FileShare.None)) {
            bool b = LargeAware(fs);
            fs.Close();
            return b;
        }
    }

    const int IMAGE_FILE_LARGE_ADDRESS_AWARE = 0x20;
    static bool LargeAware(Stream stream) {

        var br = new BinaryReader(stream);
        var bw = new BinaryWriter(stream);

        if (br.ReadInt16() != 0x5A4D)       //No MZ Header
            return false;

        br.BaseStream.Position = 0x3C;
        var peloc = br.ReadInt32();         //Get the PE header location.

        br.BaseStream.Position = peloc;
        if (br.ReadInt32() != 0x4550)       //No PE header
            return false;

        br.BaseStream.Position += 0x12;
        long nFilePos = (int)br.BaseStream.Position;
        Int16 nLgaInt = br.ReadInt16();
        bool bIsLGA = (nLgaInt & IMAGE_FILE_LARGE_ADDRESS_AWARE) == IMAGE_FILE_LARGE_ADDRESS_AWARE;
        if (bIsLGA)
            return true;
        nLgaInt |= IMAGE_FILE_LARGE_ADDRESS_AWARE;
        long nFilePos1 = bw.Seek((int)nFilePos, SeekOrigin.Begin);
        bw.Write(nLgaInt);
        bw.Flush();
        long nFilePos2 = br.BaseStream.Seek(nFilePos, SeekOrigin.Begin);
        nLgaInt = br.ReadInt16();
        bIsLGA = (nLgaInt & IMAGE_FILE_LARGE_ADDRESS_AWARE) == IMAGE_FILE_LARGE_ADDRESS_AWARE;
        return bIsLGA;
    }
于 2013-08-07T08:01:38.253 に答える