/*
* Copyright (C) 2015 Yi Zhang <wetpzy@gmail.com>
* Li Wang <liwang@redhat.com>
*
* Licensed under the GNU GPLv2 or later.
* 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 2 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, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
/* DESCRIPTION:
*
* It is a regression test for commit:
* http://git.kernel.org/cgit/linux/kernel/git/torvalds/linux.git/
* commit/?id=13d60f4
*
* The implementation of futex doesn't produce unique keys for futexes
* in shared huge pages, so threads waiting on different futexes may
* end up on the same wait list. This results in incorrect threads being
* woken by FUTEX_WAKE.
*
* Needs to be run as root unless there are already enough huge pages available.
* In the fail case, which happens in the CentOS-6.6 kernel (2.6.32-504.8.1),
* the tests hangs until it times out after a 30-second wait.
*
*/
#include <stdio.h>
#include <sys/mman.h>
#include <fcntl.h>
#include <pthread.h>
#include <errno.h>
#include <sys/time.h>
#include <string.h>
#include "test.h"
#include "safe_macros.h"
#include "futextest.h"
#include "futex_utils.h"
#include "lapi/mmap.h"
#define PATH_MEMINFO "/proc/meminfo"
#define PATH_NR_HUGEPAGES "/proc/sys/vm/nr_hugepages"
#define PATH_HUGEPAGES "/sys/kernel/mm/hugepages/"
const char *TCID = "futex_wake04";
const int TST_TOTAL = 1;
static futex_t *futex1, *futex2;
static struct timespec to = {.tv_sec = 30, .tv_nsec = 0};
static long orig_hugepages;
static void setup(void)
{
tst_require_root();
if ((tst_kvercmp(2, 6, 32)) < 0) {
tst_brkm(TCONF, NULL, "This test can only run on kernels "
"that are 2.6.32 or higher");
}
if (access(PATH_HUGEPAGES, F_OK))
tst_brkm(TCONF, NULL, "Huge page is not supported.");
tst_tmpdir();
SAFE_FILE_SCANF(NULL, PATH_NR_HUGEPAGES, "%ld", &orig_hugepages);
SAFE_FILE_PRINTF(NULL, PATH_NR_HUGEPAGES, "%d", 1);
TEST_PAUSE;
}
static void cleanup(void)
{
SAFE_FILE_PRINTF(NULL, PATH_NR_HUGEPAGES, "%ld", orig_hugepages);
tst_rmdir();
}
static int read_hugepagesize(void)
{
FILE *fp;
char line[BUFSIZ], buf[BUFSIZ];
int val;
fp = SAFE_FOPEN(cleanup, PATH_MEMINFO, "r");
while (fgets(line, BUFSIZ, fp) != NULL) {
if (sscanf(line, "%64s %d", buf, &val) == 2)
if (strcmp(buf, "Hugepagesize:") == 0) {
SAFE_FCLOSE(cleanup, fp);
return 1024 * val;
}
}
SAFE_FCLOSE(cleanup, fp);
tst_brkm(TBROK, NULL, "can't find \"%s\" in %s",
"Hugepagesize:", PATH_MEMINFO);
}
static void *wait_thread1(void *arg LTP_ATTRIBUTE_UNUSED)
{
futex_wait(futex1, *futex1, &to, 0);
return NULL;
}
static void *wait_thread2(void *arg LTP_ATTRIBUTE_UNUSED)
{
int res;
res = futex_wait(futex2, *futex2, &to, 0);
if (!res)
tst_resm(TPASS, "Hi hydra, thread2 awake!");
else
tst_resm(TFAIL, "Bug: wait_thread2 did not wake after 30 secs.");
return NULL;
}
static void wakeup_thread2(void)
{
void *addr;
int hpsz, pgsz, res;
pthread_t th1, th2;
hpsz = read_hugepagesize();
tst_resm(TINFO, "Hugepagesize %i", hpsz);
/*allocate some shared memory*/
addr = mmap(NULL, hpsz, PROT_WRITE | PROT_READ,
MAP_SHARED | MAP_ANONYMOUS | MAP_HUGETLB, -1, 0);
if (addr == MAP_FAILED) {
if (errno == ENOMEM) {
tst_brkm(TCONF, NULL,
"Cannot allocate hugepage, memory too fragmented?");
}
tst_brkm(TBROK | TERRNO, NULL, "Cannot allocate hugepage");
}
pgsz = getpagesize();
/*apply the first subpage to futex1*/
futex1 = addr;
*futex1 = 0;
/*apply the second subpage to futex2*/
futex2 = (futex_t *)((char *)addr + pgsz);
*futex2 = 0;
/*thread1 block on futex1 first,then thread2 block on futex2*/
res = pthread_create(&th1, NULL, wait_thread1, NULL);
if (res) {
tst_brkm(TBROK, NULL, "pthread_create(): %s",
tst_strerrno(res));
}
res = pthread_create(&th2, NULL, wait_thread2, NULL);
if (res) {
tst_brkm(TBROK, NULL, "pthread_create(): %s",
tst_strerrno(res));
}
while (wait_for_threads(2))
usleep(100);
futex_wake(futex2, 1, 0);
res = pthread_join(th2, NULL);
if (res)
tst_brkm(TBROK, NULL, "pthread_join(): %s", tst_strerrno(res));
futex_wake(futex1, 1, 0);
res = pthread_join(th1, NULL);
if (res)
tst_brkm(TBROK, NULL, "pthread_join(): %s", tst_strerrno(res));
SAFE_MUNMAP(NULL, addr, hpsz);
}
int main(int argc, char *argv[])
{
int lc;
tst_parse_opts(argc, argv, NULL, NULL);
setup();
for (lc = 0; TEST_LOOPING(lc); lc++)
wakeup_thread2();
cleanup();
tst_exit();
}