183 lines
4.3 KiB
C
183 lines
4.3 KiB
C
#include <unistd.h>
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <sys/mman.h>
|
|
#include <string.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <errno.h>
|
|
#include <sys/wait.h>
|
|
#include <sys/ipc.h>
|
|
#include <sys/shm.h>
|
|
#include <xpmem.h>
|
|
#include "util.h"
|
|
|
|
#define DEBUG
|
|
|
|
#define SHM_SIZE (1UL << 12)
|
|
|
|
#define MAP_HUGE_SHIFT 26
|
|
#define KEYWORD 0x12345678UL
|
|
|
|
void usage(void)
|
|
{
|
|
printf("Usage: huge_page_map: <pgshift> <pgnum> <pgoffset>\n");
|
|
printf("\tpgshift : pageshift of map area (Using MAP_HUGETLB)\n");
|
|
printf("\t -1 means using small pagesize\n");
|
|
printf("\tpgnum : number of page of map area\n");
|
|
printf("\tpgoffset: offset of last page\n");
|
|
}
|
|
|
|
|
|
void *mmap_flag(size_t mapsize, int page_shift)
|
|
{
|
|
char *addr_mmap;
|
|
int flags = MAP_ANONYMOUS | MAP_PRIVATE;
|
|
|
|
if (page_shift >= 0) {
|
|
/* mean use MAP_HUGETLB */
|
|
flags |= MAP_HUGETLB | (page_shift << MAP_HUGE_SHIFT);
|
|
}
|
|
|
|
addr_mmap = mmap(0, mapsize,
|
|
PROT_READ | PROT_WRITE,
|
|
flags, -1, 0);
|
|
|
|
return addr_mmap;
|
|
}
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
void *mem;
|
|
int ret = 0;
|
|
pid_t pid;
|
|
int status;
|
|
key_t key = ftok(argv[0], 0);
|
|
void *shm;
|
|
int shmid;
|
|
xpmem_segid_t segid;
|
|
struct shmid_ds shmctl_buf;
|
|
int pgshift, pgnum;
|
|
size_t pgsize, map_size, pgoffset;
|
|
|
|
if (argc < 4) {
|
|
printf("Err: Too few arguments\n");
|
|
usage();
|
|
return -1;
|
|
}
|
|
|
|
pgshift = atoi(argv[1]);
|
|
pgnum = atoi(argv[2]);
|
|
pgoffset = atol(argv[3]);
|
|
if (pgshift > 0) {
|
|
pgsize = (1UL << pgshift);
|
|
} else {
|
|
pgsize = getpagesize();
|
|
}
|
|
|
|
if (pgoffset > 0) {
|
|
map_size = (pgsize * (pgnum - 1)) + pgoffset;
|
|
} else {
|
|
map_size = pgsize * pgnum;
|
|
}
|
|
|
|
shmid = shmget(key, SHM_SIZE, IPC_CREAT | 0660);
|
|
CHKANDJUMP(shmid == -1, EXIT_FAILURE, "shmget failed: %s\n",
|
|
strerror(errno));
|
|
|
|
printf("EXPECT_PAGE_SIZE: 0x%lx\n", pgsize);
|
|
fflush(stdout);
|
|
|
|
pid = fork();
|
|
CHKANDJUMP(pid == -1, EXIT_FAILURE, "fork failed\n");
|
|
if (pid == 0) {
|
|
xpmem_apid_t apid;
|
|
struct xpmem_addr addr;
|
|
void *attach;
|
|
|
|
shm = shmat(shmid, NULL, 0);
|
|
CHKANDJUMP(shm == (void *)-1, EXIT_FAILURE,
|
|
"shmat failed: %s\n", strerror(errno));
|
|
|
|
while ((segid = *(xpmem_segid_t *)shm) == 0) {
|
|
};
|
|
|
|
ret = shmdt(shm);
|
|
CHKANDJUMP(ret == -1, EXIT_FAILURE, "shmdt failed\n");
|
|
|
|
apid = xpmem_get(segid, XPMEM_RDWR,
|
|
XPMEM_PERMIT_MODE, NULL);
|
|
CHKANDJUMP(apid == -1, EXIT_FAILURE, "xpmem_get failed: %s\n",
|
|
strerror(errno));
|
|
|
|
addr.apid = apid;
|
|
addr.offset = 0;
|
|
attach = xpmem_attach(addr, pgsize * pgnum, NULL);
|
|
|
|
CHKANDJUMP(attach == (void *)-1, EXIT_FAILURE,
|
|
"xpmem_attach failed: %s\n", strerror(errno));
|
|
|
|
printf("child: xpmem_attachment_addr: %lx\n",
|
|
attach);
|
|
|
|
*((unsigned long *)attach) = KEYWORD;
|
|
if (pgnum > 1 && pgshift > 0) {
|
|
*((unsigned long *)(attach +
|
|
(1UL << pgshift))) = KEYWORD;
|
|
}
|
|
*((unsigned long *)(attach + map_size
|
|
- sizeof(unsigned long *))) = KEYWORD;
|
|
|
|
ret = xpmem_detach(attach);
|
|
CHKANDJUMP(ret == -1, EXIT_FAILURE, "xpmem_detach failed\n");
|
|
|
|
exit(0);
|
|
} else {
|
|
mem = mmap_flag(map_size, pgshift);
|
|
CHKANDJUMP(mem == MAP_FAILED, EXIT_FAILURE, "mmap failed\n");
|
|
printf("parent: anonymous_map_addr: %lx - %lx\n",
|
|
mem, mem + map_size);
|
|
|
|
shm = shmat(shmid, NULL, 0);
|
|
|
|
CHKANDJUMP(shm == (void *)-1, EXIT_FAILURE,
|
|
"shmat failed: %s\n", strerror(errno));
|
|
|
|
segid = xpmem_make(mem, map_size, XPMEM_PERMIT_MODE,
|
|
(void *)0666);
|
|
CHKANDJUMP(segid == -1, EXIT_FAILURE,
|
|
"xpmem_ioctl failed: %s\n", strerror(errno));
|
|
|
|
*(xpmem_segid_t *)shm = segid;
|
|
|
|
ret = waitpid(pid, &status, 0);
|
|
CHKANDJUMP(ret == -1, EXIT_FAILURE, "waitpid failed\n");
|
|
|
|
NG(*(unsigned long *)mem == KEYWORD,
|
|
"HEAD of xpmem area is INVALID. isn't shared?\n");
|
|
if (pgnum > 1 && pgshift > 0) {
|
|
NG(*((unsigned long *)(mem +
|
|
(1UL << pgshift))) == KEYWORD,
|
|
"MIDDLE of xpmem area is INVALID. isn't shared?\n");
|
|
}
|
|
NG(*((unsigned long *)(mem + map_size
|
|
- sizeof(unsigned long *))) == KEYWORD,
|
|
"TAIL of xpmem area is INVALID. isn't shared?\n");
|
|
printf("xpmem area is shared: OK\n");
|
|
|
|
ret = shmctl(shmid, IPC_RMID, &shmctl_buf);
|
|
CHKANDJUMP(ret == -1, EXIT_FAILURE, "shmctl failed\n");
|
|
|
|
ret = shmdt(shm);
|
|
CHKANDJUMP(ret == -1, EXIT_FAILURE, "shmdt failed\n");
|
|
|
|
ret = xpmem_remove(segid);
|
|
CHKANDJUMP(ret == -1, EXIT_FAILURE, "xpmem_remove failed\n");
|
|
}
|
|
|
|
ret = 0;
|
|
out:
|
|
return ret;
|
|
}
|