avoiding the internal error by explicitly checking the arity
This commit is contained in:
parent
441c40d305
commit
5798e28801
|
@ -134,14 +134,19 @@
|
|||
};
|
||||
|
||||
var raiseArityMismatchError = function(MACHINE, proc, received) {
|
||||
raise(MACHINE, makeArityMismatchError(MACHINE, proc, received));
|
||||
};
|
||||
|
||||
var makeArityMismatchError = function(MACHINE, proc, received) {
|
||||
var message = baselib.format.format("~a: expected ~e value(s), received ~e value(s)",
|
||||
[proc.displayName,
|
||||
proc.racketArity,
|
||||
received]);
|
||||
var contMarks = MACHINE.captureContinuationMarks();
|
||||
raise(MACHINE, ExnFailContractArity.constructor([message, contMarks]));
|
||||
return ExnFailContractArity.constructor([message, contMarks]);
|
||||
};
|
||||
|
||||
|
||||
var raiseOperatorApplicationError = function(MACHINE, operator) {
|
||||
var message = baselib.format.format("not a procedure: ~e",
|
||||
[operator]);
|
||||
|
@ -233,6 +238,7 @@
|
|||
exceptions.raiseArgumentTypeError = raiseArgumentTypeError;
|
||||
exceptions.raiseContextExpectedValuesError = raiseContextExpectedValuesError;
|
||||
exceptions.raiseArityMismatchError = raiseArityMismatchError;
|
||||
exceptions.makeArityMismatchError = makeArityMismatchError;
|
||||
exceptions.raiseOperatorApplicationError = raiseOperatorApplicationError;
|
||||
exceptions.raiseOperatorIsNotPrimitiveProcedure = raiseOperatorIsNotPrimitiveProcedure;
|
||||
exceptions.raiseUnimplementedPrimitiveError = raiseUnimplementedPrimitiveError;
|
||||
|
|
|
@ -171,6 +171,7 @@
|
|||
// with two computations, with use of setTimeout.
|
||||
var ExclusiveLock = function() {
|
||||
this.locked = false; // (U false string)
|
||||
this.alreadyReleased = false;
|
||||
this.waiters = [];
|
||||
};
|
||||
|
||||
|
@ -193,16 +194,17 @@
|
|||
id = ExclusiveLock.makeRandomNonce();
|
||||
}
|
||||
|
||||
var alreadyReleased = false;
|
||||
this.alreadyReleased = false;
|
||||
|
||||
if (this.locked === false) {
|
||||
console.log("lock is clear. Grabbing...");
|
||||
this.locked = id;
|
||||
onAcquire.call(
|
||||
that,
|
||||
// releaseLock
|
||||
function() {
|
||||
var waiter;
|
||||
if (alreadyReleased) {
|
||||
if (that.alreadyReleased) {
|
||||
throw new Error(
|
||||
"Internal error: trying to release the lock, but already released");
|
||||
}
|
||||
|
@ -211,17 +213,19 @@
|
|||
"Internal error: trying to unlock the lock, but already unlocked");
|
||||
}
|
||||
that.locked = false;
|
||||
alreadyReleased = true;
|
||||
that.alreadyReleased = true;
|
||||
if (that.waiters.length > 0) {
|
||||
waiter = that.waiters.shift();
|
||||
setTimeout(
|
||||
function() {
|
||||
console.log("trying to re-acquire lock...");
|
||||
that.acquire(waiter.id, waiter.onAcquire);
|
||||
},
|
||||
0);
|
||||
}
|
||||
});
|
||||
} else {
|
||||
console.log("lock is currently in play. Need to wait.");
|
||||
this.waiters.push({ id: id,
|
||||
onAcquire: onAcquire } );
|
||||
}
|
||||
|
@ -580,6 +584,7 @@
|
|||
});
|
||||
};
|
||||
var internalCall = function(proc, success, fail) {
|
||||
console.log("internalCall");
|
||||
var i;
|
||||
if (restarted) {
|
||||
return;
|
||||
|
@ -588,15 +593,20 @@
|
|||
for (i = 3; i < arguments.length; i++) {
|
||||
args.push(arguments[i]);
|
||||
}
|
||||
console.log("acquiring lock...");
|
||||
pauseLock.acquire(
|
||||
void(0),
|
||||
function(release) {
|
||||
var newSuccess = function() {
|
||||
console.log('newSuccess', arguments, proc, success);
|
||||
success.apply(null, arguments);
|
||||
console.log("releasing lock...");
|
||||
release();
|
||||
};
|
||||
var newFail = function() {
|
||||
console.log('newFail', arguments, proc, fail);
|
||||
fail.apply(null, arguments);
|
||||
console.log("releasing lock...");
|
||||
release();
|
||||
};
|
||||
baselib.functions.internalCallDuringPause.apply(
|
||||
|
@ -614,7 +624,9 @@
|
|||
// General error condition: just exit out
|
||||
// of the trampoline and call the current error handler.
|
||||
that.running = false;
|
||||
console.log("calling current error handler", that.params.currentErrorHandler);
|
||||
that.params.currentErrorHandler(that, e);
|
||||
console.log("releasing");
|
||||
release();
|
||||
return;
|
||||
}
|
||||
|
|
|
@ -5,5 +5,5 @@
|
|||
|
||||
|
||||
(big-bang (list 'undefined 'undefined)
|
||||
(on-geo (lambda (w v lat lng)
|
||||
(list lat lng))))
|
||||
(on-geo (lambda (w v lat #;lng)
|
||||
(list lat #;lng))))
|
|
@ -17,6 +17,7 @@
|
|||
var makePair = plt.baselib.lists.makePair;
|
||||
var makeSymbol = plt.baselib.symbols.makeSymbol;
|
||||
var isArityMatching = plt.baselib.arity.isArityMatching;
|
||||
var makeArityMismatchError = plt.baselib.exceptions.makeArityMismatchError;
|
||||
|
||||
|
||||
// EventHandler and the other classes here will be defined below.
|
||||
|
@ -1197,6 +1198,7 @@
|
|||
dispatchEventsInQueue, refreshView;
|
||||
|
||||
onCleanRestart = function() {
|
||||
console.log('on clean restart');
|
||||
running = false;
|
||||
stopEventHandlers(
|
||||
function() {
|
||||
|
@ -1209,12 +1211,16 @@
|
|||
};
|
||||
|
||||
onMessyRestart = function(exn) {
|
||||
console.log('on messy restart');
|
||||
running = false;
|
||||
console.log('stopping the event handlers');
|
||||
stopEventHandlers(
|
||||
function() {
|
||||
console.log('event handlers stopped');
|
||||
restart(function(MACHINE) {
|
||||
currentBigBangRecord = oldCurrentBigBangRecord;
|
||||
MACHINE.params.currentOutputPort = oldOutputPort;
|
||||
console.log('about to raise');
|
||||
plt.baselib.exceptions.raise(MACHINE, exn);
|
||||
});
|
||||
});
|
||||
|
@ -1235,15 +1241,11 @@
|
|||
eventQueue.queue(new EventQueueElement(who, handler, args));
|
||||
if (! dispatchingEvents) {
|
||||
dispatchingEvents = true;
|
||||
setTimeout(
|
||||
function() {
|
||||
dispatchEventsInQueue(
|
||||
function() {
|
||||
refreshView(function() {}, onMessyRestart);
|
||||
},
|
||||
onMessyRestart);
|
||||
},
|
||||
0);
|
||||
dispatchEventsInQueue(
|
||||
function() {
|
||||
refreshView(function() {}, onMessyRestart);
|
||||
},
|
||||
onMessyRestart);
|
||||
}
|
||||
};
|
||||
handler.eventSource.onStart(fireEvent, internalCall, k);
|
||||
|
@ -1255,6 +1257,7 @@
|
|||
|
||||
|
||||
dispatchEventsInQueue = function(success, fail) {
|
||||
console.log("dispatchEventsInQueue");
|
||||
// Apply all the events on the queue, call toDraw, and then stop.
|
||||
// If the world ever satisfies stopWhen, stop immediately and quit.
|
||||
var nextEvent;
|
||||
|
@ -1274,7 +1277,8 @@
|
|||
racketWorldCallback = nextEvent.handler.racketWorldCallback;
|
||||
args = nextEvent.data.slice(0);
|
||||
var onGoodWorldUpdate =
|
||||
function(newWorld) {
|
||||
function (newWorld) {
|
||||
console.log("good world update");
|
||||
world = newWorld;
|
||||
stopWhen(internalCall,
|
||||
world,
|
||||
|
@ -1290,23 +1294,29 @@
|
|||
fail);
|
||||
};
|
||||
if (isArityMatching(racketWorldCallback.racketArity, 1)) {
|
||||
console.log("arity match 1");
|
||||
racketWorldCallback(internalCall,
|
||||
world,
|
||||
onGoodWorldUpdate,
|
||||
fail);
|
||||
} else if (isArityMatching(racketWorldCallback.racketArity, 2)) {
|
||||
console.log("arity match 2");
|
||||
racketWorldCallback(internalCall,
|
||||
world,
|
||||
mockView,
|
||||
onGoodWorldUpdate,
|
||||
fail);
|
||||
} else {
|
||||
} else if (isArityMatching(racketWorldCallback.racketArity, 2 + args.length)){
|
||||
console.log("arity match 3");
|
||||
args = ([internalCall, world, mockView]
|
||||
.concat(args)
|
||||
.concat([onGoodWorldUpdate, fail]));
|
||||
racketWorldCallback.apply(null, args);
|
||||
} else {
|
||||
fail(makeArityMismatchError(MACHINE, racketWorldCallback, 2+args.length));
|
||||
}
|
||||
} else {
|
||||
console.log("dispatched all events");
|
||||
dispatchingEvents = false;
|
||||
success();
|
||||
}
|
||||
|
@ -1369,6 +1379,7 @@
|
|||
success,
|
||||
fail].concat(args));
|
||||
};
|
||||
f.displayName = proc.displayName;
|
||||
f.racketArity = proc.racketArity;
|
||||
return f;
|
||||
};
|
||||
|
|
Loading…
Reference in New Issue
Block a user