adding continuation-prompt-available? so we can do context checks.

This commit is contained in:
Danny Yoo 2013-04-18 16:23:01 -06:00
parent 7159c40459
commit 1b695bed46
7 changed files with 50 additions and 11 deletions

View File

@ -210,6 +210,7 @@
'unsafe-car 'unsafe-car
'unsafe-cdr 'unsafe-cdr
'continuation-prompt-available?
'abort-current-continuation 'abort-current-continuation
'call-with-continuation-prompt 'call-with-continuation-prompt
)) ))

View File

@ -84,22 +84,35 @@
// A continuation prompt tag labels a prompt frame. // A continuation prompt tag labels a prompt frame.
var ContinuationPromptTag = function(name) { var ContinuationPromptTag = function(name) {
this.name = name; // String this.name = name; // (U String false)
}; };
ContinuationPromptTag.prototype.toDomNode = function(params) { ContinuationPromptTag.prototype.toDomNode = function(params) {
var dom = document.createElement("span"); var dom = document.createElement("span");
dom.appendChild(document.createTextNode('#<continuation-prompt-tag:' + this.name + '>')); if (this.name) {
dom.appendChild(document.createTextNode('#<continuation-prompt-tag:'
+ this.name + '>'));
} else {
dom.appendChild(document.createTextNode('#<continuation-prompt-tag>'));
}
return dom; return dom;
}; };
ContinuationPromptTag.prototype.toWrittenString = function(cache) { ContinuationPromptTag.prototype.toWrittenString = function(cache) {
return '#<continuation-prompt-tag' + this.name + '>'; if (this.name) {
return '#<continuation-prompt-tag' + this.name + '>';
} else {
return '#<continuation-prompt-tag>';
}
}; };
ContinuationPromptTag.prototype.toDisplayedString = function(cache) { ContinuationPromptTag.prototype.toDisplayedString = function(cache) {
return '#<continuation-prompt-tag' + this.name + '>'; if (this.name) {
return '#<continuation-prompt-tag' + this.name + '>';
} else {
return '#<continuation-prompt-tag>';
}
}; };

View File

@ -225,7 +225,6 @@
"Not a procedure: ~e", "Not a procedure: ~e",
[proc]), [proc]),
MACHINE.captureContinuationMarks())); MACHINE.captureContinuationMarks()));
} }
oldVal = MACHINE.v; oldVal = MACHINE.v;

View File

@ -2810,7 +2810,7 @@
sym = checkSymbol(M, "make-continuation-prompt-tag", 0); sym = checkSymbol(M, "make-continuation-prompt-tag", 0);
return new baselib.contmarks.ContinuationPromptTag(sym.toString()); return new baselib.contmarks.ContinuationPromptTag(sym.toString());
} }
return new baselib.contmarks.ContinuationPromptTag(void(0)); return new baselib.contmarks.ContinuationPromptTag(false);
}); });
installPrimitiveProcedure( installPrimitiveProcedure(
@ -3164,6 +3164,28 @@
}; };
// FIXME: we should be able to take in an arbitrary continuation
// as an optional second argument!
//
// I need to change the representation of continuations to be able to
// detect this at runtime.
installPrimitiveProcedure(
'continuation-prompt-available?',
1,
function(M) {
var promptTag = checkPromptTag(M, 'continuation-prompt-available?', 0);
var i;
for (i = 0; i < M.c.length; i++) {
var frame = M.c[i];
if (frame instanceof PromptFrame && frame.tag === promptTag) {
return true;
}
}
return false;
});
// The default abort prompt handler consumes a thunk and applies // The default abort prompt handler consumes a thunk and applies
// it, in a context where a new prompt has been initialized. // it, in a context where a new prompt has been initialized.
var defaultPromptHandler = var defaultPromptHandler =
@ -3185,7 +3207,7 @@
// First, find the continuation prompt. // First, find the continuation prompt.
while(true) { while(true) {
frame = M.c.pop(); frame = M.c.pop();
if (frame instanceof PromptFrame) { if (frame instanceof PromptFrame && frame.tag === promptTag) {
break; break;
} else if (M.c.length === 0) { } else if (M.c.length === 0) {
raiseContractError( raiseContractError(

View File

@ -501,7 +501,7 @@
}; };
Machine.prototype.addPrompt = function(promptTag, abortHandlerClosure) { Machine.prototype.addPrompt = function(promptTag, abortHandlerClosure) {
this.c.push(new PromptFrame(justReturn, this.c.push(new PromptFrame(justReturn,
DEFAULT_CONTINUATION_PROMPT_TAG, promptTag,
this.e.length, this.e.length,
abortHandlerClosure)); abortHandlerClosure));
}; };

View File

@ -228,7 +228,7 @@
current-inexact-milliseconds current-inexact-milliseconds
current-seconds current-seconds
continuation-prompt-available?
abort-current-continuation abort-current-continuation
call-with-continuation-prompt call-with-continuation-prompt

View File

@ -9,12 +9,16 @@
(define (call-with-escape-continuation proc) (define (call-with-escape-continuation proc)
(define p (make-continuation-prompt-tag)) (define p (make-continuation-prompt-tag 'escape))
(call-with-continuation-prompt (call-with-continuation-prompt
(lambda () (lambda ()
(proc (lambda args (proc (lambda args
(unless (continuation-prompt-available? p)
(error 'call-with-escape-continuation
"escape continuation used out of context"))
(abort-current-continuation p (lambda () (abort-current-continuation p (lambda ()
(apply values args)))))))) (apply values args))))))
p))
(define call/ec (procedure-rename call-with-escape-continuation 'call/ec)) (define call/ec (procedure-rename call-with-escape-continuation 'call/ec))