5

WindowsMo​​bile6.x用のVisualStudio2008 C ++アプリケーションがあり、特定のプロセスで使用可能な空き仮想メモリの量をカウントしています。(断片化を考慮に入れていないことに気づきました。)私のコードは基本的に次のようになります。

MEMORY_BASIC_INFORMATION mbi = { 0 };

/// total free memory available to the process
DWORD free = 0;

/// base memory address for the given process index (2-33). 
DWORD slot_base_addr = process_index * 0x02000000;

/// look at each memory region for the process. 
for( DWORD offset = 0x10000; 
     offset < 0x02000000; 
     offset += mbi.RegionSize )
{
    ::VirtualQuery( ( void* )( slot_base_addr + offset ), 
                    &mbi, 
                    sizeof( MEMORY_BASIC_INFORMATION ) );

    if( mbi.State == MEM_FREE )
    {
        free += ( mbi.RegionSize - ( ( ~( DWORD )mbi.BaseAddress + 1 ) & 0xffff ) ) & 0xffff0000;
    }
}
NKDbgPrintfW( L"%d bytes free\r\n", free );

これが完全に機能しているように見えることを他のAPIで確認できます。私の質問は、この行が何をしているのかです。

free += ( mbi.RegionSize - ( ( ~( DWORD )mbi.BaseAddress + 1 ) & 0xffff ) ) & 0xffff0000;

なぜこれだけではないのですか?

free += mbi.RegionSize;

前の行は、MSFTの従業員であるRossJordanによるUsenetの投稿で見つかりました。

ありがとう、PaulH


編集:

例えば。プロセススロット2の場合、これは、Ross Jordan(RS)アルゴリズムとRegionSize(RS)の両方によって指定された空きメモリの量を含む各空きメモリブロックのリストです。

Slot: 2. Range: 0x04000000 - 0x06000000
    RS:    16,384 bytes RJ:         0 bytes diff: 16384
    RS:     4,096 bytes RJ:         0 bytes diff: 4096
    RS:     4,096 bytes RJ:         0 bytes diff: 4096
    RS:     4,096 bytes RJ:         0 bytes diff: 4096
    RS:     4,096 bytes RJ:         0 bytes diff: 4096
    RS:     4,096 bytes RJ:         0 bytes diff: 4096
    RS:    36,864 bytes RJ:         0 bytes diff: 36864
    RS:    65,536 bytes RJ:    65,536 bytes diff: 0
    RS:    53,248 bytes RJ:         0 bytes diff: 53248
    RS:     4,096 bytes RJ:         0 bytes diff: 4096
    RS:     4,096 bytes RJ:         0 bytes diff: 4096
    RS:     4,096 bytes RJ:         0 bytes diff: 4096
    RS:     4,096 bytes RJ:         0 bytes diff: 4096
    RS:     4,096 bytes RJ:         0 bytes diff: 4096
    RS: 7,671,808 bytes RJ: 7,667,712 bytes diff: 4096
    RS: 1,921,024 bytes RJ: 1,900,544 bytes diff: 20480
    RS: 7,491,584 bytes RJ: 7,471,104 bytes diff: 20480
    RS: 3,252,224 bytes RJ: 3,211,264 bytes diff: 40960
    RS:   262,144 bytes RJ:   262,144 bytes diff: 0

RS: Total VM Free: 20,811,776 bytes.
RJ: Total VM Free: 20,578,304 bytes.

編集2:

ハンスは私を答えに導きました。これは、これを行うためのすばらしい方法ですが、割り当てサイズが64KBであると想定しています。

SYSTEM_INFO si = { 0 };
::GetSystemInfo( &si );

free += mbi.RegionSize - mbi.RegionSize % si.dwAllocationGranularity;
4

1 に答える 1

2

VirtualAllocの割り当て粒度は通常64KBです。AllocationBaseが64KBの倍数でない場合、彼は何か意味のあることをしようとします。私はそれがまったく意味をなさないと思います、彼のビットマスクはまだ64KBの粒度を想定していて、彼はSYSTEM_INFO.dwAllocationGranularityを使用していません。このコメントがあります:

この値は、過去に64 KBとしてハードコーディングされていましたが、他のハードウェアアーキテクチャでは異なる値が必要になる場合があります。

64KBでない非常にまれなケースでは、このコードはジャンク値を生成します。それをザッピングするだけで、RegionSizeで移動します。

于 2010-11-01T14:52:41.210 に答える