2012-10-24 18:24:17 +02:00
|
|
|
/*
|
|
|
|
* \brief Test for thread-local errno handling of hybrid Linux/Genode programs
|
|
|
|
* \author Norman Feske
|
|
|
|
* \date 2011-12-05
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Genode includes */
|
2015-12-23 15:22:33 +01:00
|
|
|
#include <base/component.h>
|
2012-10-24 18:24:17 +02:00
|
|
|
#include <base/thread.h>
|
|
|
|
#include <base/printf.h>
|
|
|
|
|
|
|
|
/* libc includes */
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <errno.h>
|
2015-12-23 15:22:33 +01:00
|
|
|
#include <stdlib.h>
|
2012-10-24 18:24:17 +02:00
|
|
|
|
|
|
|
enum { STACK_SIZE = 4096 };
|
|
|
|
|
2016-05-04 12:27:17 +02:00
|
|
|
struct Thread : Genode::Thread_deprecated<STACK_SIZE>
|
2012-10-24 18:24:17 +02:00
|
|
|
{
|
|
|
|
Genode::Lock &_barrier;
|
|
|
|
|
2013-10-10 17:00:03 +02:00
|
|
|
Thread(Genode::Lock &barrier)
|
2016-05-04 12:27:17 +02:00
|
|
|
: Genode::Thread_deprecated<STACK_SIZE>("stat"), _barrier(barrier) { start(); }
|
2012-10-24 18:24:17 +02:00
|
|
|
|
|
|
|
void entry()
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* Stat syscall should return with errno ENOENT
|
|
|
|
*/
|
|
|
|
struct stat buf;
|
|
|
|
int ret = stat("", &buf);
|
|
|
|
|
|
|
|
Genode::printf("thread: stat returned %d, errno=%d\n", ret, errno);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Let main thread procees
|
|
|
|
*/
|
|
|
|
_barrier.unlock();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2015-12-23 15:22:33 +01:00
|
|
|
static int exit_status;
|
|
|
|
static void exit_on_suspended() { exit(exit_status); }
|
|
|
|
|
|
|
|
|
|
|
|
Genode::size_t Component::stack_size() { return 16*1024*sizeof(long); }
|
|
|
|
|
|
|
|
|
|
|
|
struct Unexpected_errno_change { };
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Component implements classical main function in construct.
|
|
|
|
*/
|
2016-04-27 22:11:38 +02:00
|
|
|
void Component::construct(Genode::Env &env)
|
2012-10-24 18:24:17 +02:00
|
|
|
{
|
|
|
|
Genode::printf("--- thread-local errno test ---\n");
|
|
|
|
|
|
|
|
static Genode::Lock barrier(Genode::Lock::LOCKED);
|
|
|
|
|
|
|
|
int const orig_errno = errno;
|
|
|
|
|
|
|
|
Genode::printf("main: before thread creation, errno=%d\n", orig_errno);
|
|
|
|
|
|
|
|
/* create thread, which modifies its thread-local errno value */
|
|
|
|
static Thread thread(barrier);
|
|
|
|
|
|
|
|
/* block until the thread performed a 'stat' syscall */
|
|
|
|
barrier.lock();
|
|
|
|
|
|
|
|
Genode::printf("main: after thread completed, errno=%d\n", errno);
|
|
|
|
|
|
|
|
if (orig_errno != errno) {
|
|
|
|
PERR("unexpected change of main thread's errno value");
|
2015-12-23 15:22:33 +01:00
|
|
|
throw Unexpected_errno_change();
|
2012-10-24 18:24:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Genode::printf("--- finished thread-local errno test ---\n");
|
2015-12-23 15:22:33 +01:00
|
|
|
exit_status = 0;
|
|
|
|
env.ep().schedule_suspend(exit_on_suspended, nullptr);
|
2012-10-24 18:24:17 +02:00
|
|
|
}
|