私がやりたいことは次のとおりです。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