2

私がやりたいことは次のとおりです。64 ビット Linux 用にビルドされたライブラリがあります。次に、そのライブラリをリンクするアプリケーションを作成しました。アプリケーションを実行するときに、ライブラリから割り当てられたメモリが常により高い場所 (>4GB) にあることを確認したい。

Windows では、ユーザーは、テスト目的で下位アドレスの前に上位アドレスから割り当てるように割り当てを強制できます。VirtualAlloc を呼び出すときに MEM_TOP_DOWN を指定するか、次のレジストリ値を 0x100000 に設定します。

HKEY_LOCAL_MACHINE\System\CurrentControlSet\Control\Session Manager\Memory Management\AllocationPreference

Linuxにも同様の戦略があるのだろうか。Linux のメモリ管理が Window とは異なることは理解していますが、mmap() やリンカー スクリプトを使用するなどの手がかりを見つけました。しかし、私は目標を達成することができませんでした。誰もがより多くの情報を提供できますか?

#include <sys/mman.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>

#define handle_error(msg) \
  do { perror(msg); exit(EXIT_FAILURE); } while (0)

int main()
{
  void *addr1=0, *addr2=0;
  long sz = sysconf(_SC_PAGE_SIZE); // get page size
  size_t length = sz*1000*1000; // 1,000,000 pages
  int fd = -1;

  printf("page size = %ld\n", sz);

  // find some available address
  int *p = (int*)malloc(sizeof(int));
  long start = (long)p + sizeof(int);
  free(p); // free it anyway

  start += (sz-(start % sz)); // page alignment
  printf("start = 0x%lx\n", start);

  // mmap fixed
  addr1 = mmap((void*)start, length, PROT_NONE,
               MAP_PRIVATE|MAP_NORESERVE|MAP_ANONYMOUS|MAP_FIXED, fd, 0);
  if (addr1 == MAP_FAILED)
    handle_error("mmap");
  printf("first map: %tx\n", addr1);
  //msync(addr1, length, 0);

  // another mmap
  addr2 = mmap(NULL, sz*10, PROT_NONE,
               MAP_ANONYMOUS|MAP_PRIVATE, fd, 0);
  if (addr2 == MAP_FAILED)
    handle_error("mmap");
  printf("second map: 0x%tx\n", addr2);

  // test whether the memory is still available
  p = (int*)malloc(sizeof(int)*10);
  printf("allocated address: 0x%tx\n", p);

  return 0;
}

出力:

page size = 4096
start = 0x1d77000
first map: 1d77000
second map: 0x7f5f26c2f000
allocated address: 0x1d76030
4

1 に答える 1

1

なぜそれをしたいのかわかりません(mmap最初の4ギガバイトにアドレスを与えることを避けています)。

ただし、プログラムの非常に早い段階で、たとえば の開始main、または一部のコンストラクター関数でさえ、目標を達成するために複数のメモリ セグメントでmmap(2)を呼び出すことができます。したがって、4G より下のすべてのアドレス空間が、プログラムの既存のセグメントによって、またはそのような.MAP_FIXEDMAP_NORESERVEmmap

ただし、ライブラリ (間接的に編集される可能性がありますdlopen) は、プログラムのかなり後に開始される可能性があります。

最初の 4 ギガバイトがアドレス空間で使用されると、ほとんどの通常のmmap呼び出し (mallocたとえば、によって行われる呼び出し) は外に出ます。

もちろん、最初の(おそらくorを呼び出す)mmap 前のある時点で行う必要があります。また、既存のメモリ セグメントに注意する必要があります (おそらく を解析することで取得できます) 。.mallocmmapsbrk/proc/self/mapsmmap MAP_FIXED|MAP_NORESERVE

また、独自の を定義することもできますmalloc。おそらく、巨大なリージョン (たとえば、テラバイト) を使用して、独自の使用専用アドレスを内部に持つことができます (mmapそこで再度 -ing を実行します)。MAP_NORESERVEmallocmmap

あなたは間違った問題を解決しようとしていると思います。私が提案することを行うのは難しいかもしれません...最初の4Gバイトのアドレスを避ける正当な理由はありません。

ところで、Linux でメモリ リークを見つけるための優れたツールはvalgrindです。

于 2013-05-09T16:13:16.810 に答える