/******************************************************************************/
/* */
/* Copyright (c) International Business Machines Corp., 2007 */
/* Copyright (c) Linux Test Project, 2016 */
/* */
/* This program is free software: you can redistribute it and/or modify */
/* it under the terms of the GNU General Public License as published by */
/* the Free Software Foundation, either version 3 of the License, or */
/* (at your option) any later version. */
/* */
/* This program is distributed in the hope that it will be useful, */
/* but WITHOUT ANY WARRANTY; without even the implied warranty of */
/* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the */
/* GNU General Public License for more details. */
/* */
/* You should have received a copy of the GNU General Public License */
/* along with this program. If not, see <http://www.gnu.org/licenses/>. */
/* */
/******************************************************************************/
/******************************************************************************/
/* */
/* File: support_numa.c */
/* */
/* Description: Allocates memory and touches it to verify numa */
/* */
/* Author: Sivakumar Chinnaiah Sivakumar.C@in.ibm.com */
/* */
/******************************************************************************/
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <unistd.h>
#include <signal.h>
#include <limits.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/mman.h>
#include <fcntl.h>
#include "lapi/mmap.h"
/* Global Variables */
#define MB (1<<20)
#define PAGE_SIZE getpagesize()
#define barrier() __asm__ __volatile__("": : :"memory")
#define TEST_SFILE "ltp_numa_testfile"
#define STR "abcdefghijklmnopqrstuvwxyz12345\n"
static void help(void)
{
printf("Input: Describe input arguments to this program\n");
printf(" argv[1] == \"alloc_1MB\" then allocate 1MB of memory\n");
printf(" argv[1] == \"alloc_1MB_shared\" then allocate 1MB of share memory\n");
printf(" argv[1] == \"alloc_2HPSZ_THP\" then allocate 2HUGE PAGE SIZE of THP memory\n");
printf(" argv[1] == \"alloc_1huge_page\" then allocate 1HUGE PAGE SIZE of memory\n");
printf(" argv[1] == \"pause\" then pause the program to catch sigint\n");
printf("Exit: On failure - Exits with non-zero value\n");
printf(" On success - exits with 0 exit value\n");
exit(1);
}
static int read_hugepagesize(void)
{
FILE *fp;
char line[BUFSIZ], buf[BUFSIZ];
int val;
fp = fopen("/proc/meminfo", "r");
if (fp == NULL) {
fprintf(stderr, "Failed to open /proc/meminfo");
return 0;
}
while (fgets(line, BUFSIZ, fp) != NULL) {
if (sscanf(line, "%64s %d", buf, &val) == 2)
if (strcmp(buf, "Hugepagesize:") == 0) {
fclose(fp);
return 1024 * val;
}
}
fclose(fp);
fprintf(stderr, "can't find \"%s\" in %s", "Hugepagesize:", "/proc/meminfo");
return 0;
}
int main(int argc, char *argv[])
{
int i, fd, rc, hpsz;
char *buf = NULL;
struct stat sb;
if (argc != 2) {
fprintf(stderr, "Here expect only one number(i.e. 2) as the parameter\n");
exit(1);
}
if (!strcmp(argv[1], "alloc_1MB")) {
buf = malloc(MB);
if (!buf) {
fprintf(stderr, "Memory is not available\n");
exit(1);
}
for (i = 0; i < MB; i += PAGE_SIZE) {
buf[i] = 'a';
barrier();
}
raise(SIGSTOP);
free(buf);
} else if (!strcmp(argv[1], "alloc_1MB_shared")) {
fd = open(TEST_SFILE, O_RDWR | O_CREAT, 0666);
/* Writing 1MB of random data into this file [32 * 32768 = 1024 * 1024] */
for (i = 0; i < 32768; i++){
rc = write(fd, STR, strlen(STR));
if (rc == -1 || ((size_t)rc != strlen(STR)))
fprintf(stderr, "write failed\n");
}
if ((fstat(fd, &sb)) == -1)
fprintf(stderr, "fstat failed\n");
buf = mmap(NULL, sb.st_size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
if (buf == MAP_FAILED){
fprintf(stderr, "mmap failed\n");
close(fd);
exit(1);
}
memset(buf, 'a', sb.st_size);
raise(SIGSTOP);
munmap(buf, sb.st_size);
close(fd);
remove(TEST_SFILE);
} else if (!strcmp(argv[1], "alloc_2HPSZ_THP")) {
ssize_t size = 2 * read_hugepagesize();
if (size == 0)
exit(1);
buf = mmap(NULL, size, PROT_READ | PROT_WRITE,
MAP_PRIVATE | MAP_ANONYMOUS,
-1, 0);
if (buf == MAP_FAILED) {
perror("mmap failed");
exit(1);
}
memset(buf, 'a', size);
raise(SIGSTOP);
munmap(buf, size);
} else if (!strcmp(argv[1], "alloc_1huge_page")) {
hpsz = read_hugepagesize();
if (hpsz == 0)
exit(1);
buf = mmap(NULL, hpsz, PROT_READ | PROT_WRITE,
MAP_PRIVATE | MAP_ANONYMOUS | MAP_HUGETLB,
-1, 0);
if (buf == MAP_FAILED) {
perror("mmap failed");
exit(1);
}
memset(buf, 'a', hpsz);
raise(SIGSTOP);
munmap(buf, hpsz);
} else if (!strcmp(argv[1], "pause")) {
raise(SIGSTOP);
} else {
help();
}
return 0;
}