summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorPixel <pixel@nobis-crew.org>2012-04-08 09:42:51 -0700
committerPixel <pixel@nobis-crew.org>2012-04-08 09:43:59 -0700
commit0fb5f42ca43f11bd37ddc89aa7c7b098d764b28c (patch)
tree516d08db4abb8691c61ac6e05a1f944503e98be3
parent9f0431d84807e37bfa63bd144dcb4d2235fe7772 (diff)
Further refinements and protections in the stackless base.
-rw-r--r--includes/Task.h4
-rw-r--r--src/Task.cc36
2 files changed, 13 insertions, 27 deletions
diff --git a/includes/Task.h b/includes/Task.h
index db602f0..b5f5540 100644
--- a/includes/Task.h
+++ b/includes/Task.h
@@ -158,6 +158,9 @@ class Task {
virtual void Do() = 0;
void waitFor(Events::BaseEvent * event);
bool setOkayToEAgain(bool enable) {
+ if (m_stackless) {
+ AAssert(enable, "You can't make a task go not-okay-to-eagain if it's stackless.");
+ }
bool oldValue = m_okayToEAgain;
m_okayToEAgain = enable;
return oldValue;
@@ -166,6 +169,7 @@ class Task {
AAssert(m_stackless, "Can't set a task to be stackless twice");
AAssert(m_status == STARTING, "Can't set a task to be stackless after it started. status = %s", StatusToString(m_status));
m_stackless = true;
+ m_okayToEAgain = true;
}
private:
static size_t stackSize() { return 64 * 1024; }
diff --git a/src/Task.cc b/src/Task.cc
index 05d0b0a..5026d28 100644
--- a/src/Task.cc
+++ b/src/Task.cc
@@ -51,9 +51,7 @@ void Balau::Task::coroutineTrampoline(void * arg) {
void Balau::Task::coroutine() {
try {
- if (!m_stackless) {
- IAssert(m_status == STARTING, "The Task at %p was badly initialized ? m_status = %s", this, StatusToString(m_status));
- }
+ IAssert((m_status == STARTING) || (m_stackless && (m_status == RUNNING)), "The Task at %p has a bad status ? m_status = %s, stackless = %s", this, StatusToString(m_status), m_stackless ? "true" : "false");
m_status = RUNNING;
Do();
m_status = STOPPED;
@@ -94,29 +92,8 @@ void Balau::Task::coroutine() {
for (String & str : trace)
Printer::log(M_DEBUG, "%s", str.to_charp());
m_status = FAULTED;
- }
- }
- catch (EAgain & e) {
- waitFor(e.getEvent());
- if (!m_okayToEAgain) {
- Printer::log(M_ERROR, "Task %s at %p which is non-okay-to-eagain got an EAgain exception.", getName(), this);
- const char * details = e.getDetails();
- if (details)
- Printer::log(M_ERROR, " %s", details);
- auto trace = e.getTrace();
- for (String & str : trace)
- Printer::log(M_DEBUG, "%s", str.to_charp());
- TaskMan::stop(-1);
- }
- if (!m_stackless) {
- Printer::log(M_WARNING, "Task %s at %p hasn't caught an EAgain exception.", getName(), this);
- const char * details = e.getDetails();
- if (details)
- Printer::log(M_WARNING, " %s", details);
- auto trace = e.getTrace();
- for (String & str : trace)
- Printer::log(M_DEBUG, "%s", str.to_charp());
- m_status = FAULTED;
+ } else {
+ Printer::elog(E_TASK, "Stackless task %s at %p is task-switching.", getName(), this);
}
}
catch (GeneralException & e) {
@@ -169,7 +146,7 @@ void Balau::Task::yield(bool stillRunning) throw (GeneralException) {
else
m_status = SLEEPING;
if (m_stackless) {
- throw TaskSwitch();
+ throw EAgain(NULL);
} else {
#ifndef _WIN32
coro_transfer(&m_ctx, &m_taskMan->m_returnContext);
@@ -280,6 +257,11 @@ void Balau::Task::yield(Events::BaseEvent * evt, bool interruptible) throw (Gene
Task * t = getCurrentTask();
if (evt)
t->waitFor(evt);
+
+ if (t->m_stackless) {
+ AAssert(interruptible, "You can't run non-interruptible operations from a stackless task.");
+ }
+
bool gotSignal;
do {