#define _GNU_SOURCE #include #include #include #include #include #include #include #include #include #include /* For SYS_xxx definitions */ #include #include #include int passed = 0, sem = 0; pthread_t thr; unsigned long mem; /* delay functions issue ld/st instructions on this address */ double nspw; /* nsec per work */ /* Timer related macros */ #define TS2NS(sec, nsec) ((unsigned long)(sec) * 1000000000ULL + (unsigned long)(nsec)) #define N_INIT 10000000 static inline void fixed_size_work(unsigned long *ptr) { asm volatile("movq %0, %%rax\n\t" "addq $1, %%rax\n\t" \ "movq %%rax, %0\n\t" \ : "+rm" (*ptr) \ : \ : "rax", "cc", "memory"); \ } static inline void delay_loop(unsigned long n, unsigned long *ptr) { int j; for (j = 0; j < (n); j++) { fixed_size_work(ptr); } } void delay_init(unsigned long *mem) { struct timespec start, end; unsigned long nsec; int i; clock_gettime(CLOCK_THREAD_CPUTIME_ID, &start); delay_loop(N_INIT, mem); clock_gettime(CLOCK_THREAD_CPUTIME_ID, &end); nsec = (TS2NS(end.tv_sec, end.tv_nsec) - TS2NS(start.tv_sec, start.tv_nsec)); nspw = nsec / (double)N_INIT; printf("nsec=%ld, nspw=%f\n", nsec, nspw); } void delay_nsec(unsigned long delay_nsec, unsigned long* mem) { //printf("delay_nsec=%ld,count=%f\n", delay_nsec, delay_nsec / nspw); delay_loop(delay_nsec / nspw, mem); } void *util_thread(void *arg) { int rc; rc = syscall(732); if (rc == -1) fprintf(stderr, "CT12100 running on Linux CPU OK\n"); else { fprintf(stderr, "CT12100 running on Linux CPU NG (%d)\n", rc); } passed = 1; rc = syscall(__NR_futex, &sem, FUTEX_WAIT, 0, NULL, NULL, 0); if (rc != 0) { fprintf(stderr, "CT12101 FUTEX_WAIT NG (%s)\n", strerror(errno)); } else { fprintf(stderr, "CT12101 FUTEX_WAIT OK\n"); } return NULL; } int main(int argc, char **argv) { int rc; fprintf(stderr, "CT12001 futex START\n"); rc = syscall(731, 1, NULL); if (rc) { fprintf(stderr, "util_indicate_clone rc=%d, errno=%d\n", rc, errno); fflush(stderr); } rc = pthread_create(&thr, NULL, util_thread, NULL); if (rc){ fprintf(stderr, "pthread_create: %d\n", rc); exit(1); } fprintf(stderr, "CT12002 pthread_create OK\n"); retry: while (!passed) { asm volatile("pause" ::: "memory"); } usleep(100000); rc = syscall(__NR_futex, &sem, FUTEX_WAKE, 1, NULL, NULL, 0); if (rc != 1) { fprintf(stderr, "CT12003 FUTEX_WAKE NG (%d,%s)\n", rc, strerror(errno)); } else { fprintf(stderr, "CT12003 FUTEX_WAKE OK\n"); } pthread_join(thr, NULL); fprintf(stderr, "CT12004 pthread_join OK\n"); fprintf(stderr, "CT12005 END\n"); exit(0); }