/* * This file is part of Jehanne. * * Copyright (C) 2015 Giacomo Tesio * * Jehanne 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, version 2 of the License. * * Jehanne 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 Jehanne. If not, see . */ #include #include /* verify that lockt returns 1 if the lock is released and * it can take it */ #define NPROC 50 QLock l; Rendez afterAWhile; /* wakened after a while by main, then sleepers will be rwakeup */ int killerProc; /* pid, will kill the other processes if starved */ int elapsed[NPROC]; int32_t completed; QLock rl; Rendez rStart; Rendez rCompleted; int verbose = 0; void killKiller(void) { postnote(PNPROC, killerProc, "interrupt"); } void stopAllAfter(int seconds) { int pid; switch((pid = sys_rfork(RFMEM|RFPROC|RFNOWAIT))) { case 0: if(verbose) print("killer proc started: pid %d\n", getpid()); sleep(seconds * 1000); postnote(PNGROUP, killerProc, "timedout"); if(verbose) print("killer proc timedout: pid %d\n", getpid()); exits("FAIL"); case -1: fprint(2, "%r\n"); exits("sys_rfork fails"); default: killerProc = pid; atexit(killKiller); } } int handletimeout(void *v, char *s) { /* just not exit, please */ if(strcmp(s, "timedout") == 0){ if(verbose) print("%d: noted: %s\n", getpid(), s); print("FAIL: %s timedout\n", argv0); exits("FAIL"); } return 0; } int handlefail(void *v, char *s) { /* just not exit, please */ if(strncmp(s, "fail", 4) == 0){ if(verbose) print("%d: noted: %s\n", getpid(), s); exits("FAIL"); } return 0; } char * sleeper(int index) { int64_t start, end; /* wait for the afterAWhile to be locked by the main process */ qlock(&rl); while(!afterAWhile.l) rsleep(&rStart); qunlock(&rl); /* try to sleep for ~10 s, but it will be available after ~1s */ start = nsec(); end = start; if(verbose) print("sleeper %d: started\n", getpid()); qlock(afterAWhile.l); if(rsleept(&afterAWhile, 10000)){ end = nsec(); if(verbose) print("sleeper %d: awakened in %lld ms\n", getpid(), (end - start) / (1000*1000)); if((end - start) / (1000*1000) > 2500) postnote(PNGROUP, getpid(), smprint("fail: sleeper %d awakened after %lld ms", getpid(), (end - start) / (1000*1000))); } else { end = nsec(); if(verbose) print("sleeper %d failed: rsleept timedout in %lld ms\n", getpid(), (end - start) / (1000*1000)); postnote(PNGROUP, getpid(), smprint("fail: sleeper %d timedout", getpid())); } qunlock(afterAWhile.l); /* notify the main process that we have completed */ qlock(&rl); elapsed[index] = end - start; ++completed; rwakeup(&rCompleted); qunlock(&rl); return (end - start) / (1000*1000) < 1300 ? nil : "FAIL"; } void spawnWaiter(int index) { int pid; char * res; switch((pid = sys_rfork(RFMEM|RFPROC|RFNOWAIT))) { case 0: res = sleeper(index); exits(res); break; case -1: print("spawnWaiter: %r\n"); exits("sys_rfork fails"); break; default: if(verbose) print("spawn sleeper %d\n", pid); break; } } void main(int argc, char* argv[]) { int i, awakened; int64_t average, end; ARGBEGIN{ }ARGEND; sys_rfork(RFNOTEG|RFREND); rStart.l = &rl; rCompleted.l = &rl; if(verbose) print("main: started with pid %d\n", getpid()); for(i = 0; i < NPROC; ++i){ spawnWaiter(i); } stopAllAfter(30); if (!atnotify(handletimeout, 1)){ fprint(2, "%r\n"); exits("atnotify fails"); } if (!atnotify(handlefail, 1)){ fprint(2, "%r\n"); exits("atnotify fails"); } qlock(&rl); afterAWhile.l = &l; if(verbose) print("main: wakeup all sleepers...\n"); rwakeupall(&rStart); if(verbose) print("main: wakeup all sleepers... done\n"); end = nsec() + 1000*1000*1000; qunlock(&rl); /* wait for ~1 sec */ do { sleep(1000); } while(nsec() < end); qlock(&l); awakened = rwakeupall(&afterAWhile); qunlock(&l); if(verbose) print("main: released the afterAWhile\n"); qlock(&rl); if(verbose) print("main: waiting all sleepers to wakeup...\n"); while(completed < NPROC){ rsleep(&rCompleted); if(verbose && completed < NPROC) print("main: awaked, but %d sleepers are still pending\n", NPROC - completed); } qunlock(&rl); if(verbose) print("main: waiting all sleepers to wakeup... done\n"); average = 0; for(i = 0; i < NPROC; ++i){ average += elapsed[i]; } average = average / NPROC / (1000 * 1000); if(awakened == NPROC && average > 900 && average < 5000) { print("PASS\n"); exits("PASS"); } print("FAIL: %s: average timeout too long %lld ms\n", argv0, average); exits("FAIL"); }