|
@@ -436,6 +436,32 @@ return $recv($1)._yourself();
|
|
|
}; }),
|
|
|
$globals.IRInliner);
|
|
|
|
|
|
+$core.addMethod(
|
|
|
+$core.method({
|
|
|
+selector: "nonLocalReturnInliner",
|
|
|
+protocol: "factory",
|
|
|
+
|
|
|
+args: [],
|
|
|
+source: "nonLocalReturnInliner\x0a\x09^ IRNonLocalReturnInliner new\x0a\x09\x09translator: self;\x0a\x09\x09yourself",
|
|
|
+referencedClasses: ["IRNonLocalReturnInliner"],
|
|
|
+
|
|
|
+pragmas: [],
|
|
|
+messageSends: ["translator:", "new", "yourself"]
|
|
|
+}, function ($methodClass){ return function (){
|
|
|
+var self=this,$self=this;
|
|
|
+
|
|
|
+return $core.withContext(function($ctx1) {
|
|
|
+
|
|
|
+var $1;
|
|
|
+$1=$recv($globals.IRNonLocalReturnInliner)._new();
|
|
|
+$recv($1)._translator_(self);
|
|
|
+return $recv($1)._yourself();
|
|
|
+
|
|
|
+}, function($ctx1) {$ctx1.fill(self,"nonLocalReturnInliner",{})});
|
|
|
+
|
|
|
+}; }),
|
|
|
+$globals.IRInliner);
|
|
|
+
|
|
|
$core.addMethod(
|
|
|
$core.method({
|
|
|
selector: "returnInliner",
|
|
@@ -659,11 +685,11 @@ selector: "visitIRNonLocalReturn:",
|
|
|
protocol: "visiting",
|
|
|
|
|
|
args: ["anIRNonLocalReturn"],
|
|
|
-source: "visitIRNonLocalReturn: anIRNonLocalReturn\x0a\x09| localReturn |\x0a\x09anIRNonLocalReturn scope canInlineNonLocalReturns ifTrue: [\x0a\x09\x09anIRNonLocalReturn scope methodScope removeNonLocalReturn: anIRNonLocalReturn scope.\x0a\x09\x09localReturn := IRReturn new\x0a\x09\x09\x09scope: anIRNonLocalReturn scope;\x0a\x09\x09\x09yourself.\x0a\x09\x09anIRNonLocalReturn dagChildren do: [ :each |\x0a\x09\x09\x09localReturn add: each ].\x0a\x09\x09anIRNonLocalReturn replaceWith: localReturn.\x0a\x09\x09^ self visitIRReturn: localReturn ].\x0a\x09^ super visitIRNonLocalReturn: anIRNonLocalReturn",
|
|
|
+source: "visitIRNonLocalReturn: anIRNonLocalReturn\x0a\x09| localReturn |\x0a\x09anIRNonLocalReturn scope canInlineNonLocalReturns ifTrue: [\x0a\x09\x09anIRNonLocalReturn scope methodScope removeNonLocalReturn: anIRNonLocalReturn scope.\x0a\x09\x09localReturn := IRReturn new\x0a\x09\x09\x09scope: anIRNonLocalReturn scope;\x0a\x09\x09\x09yourself.\x0a\x09\x09anIRNonLocalReturn dagChildren do: [ :each |\x0a\x09\x09\x09localReturn add: each ].\x0a\x09\x09anIRNonLocalReturn replaceWith: localReturn.\x0a\x09\x09^ self visitIRReturn: localReturn ].\x0a\x09^ (self shouldInlineReturn: anIRNonLocalReturn)\x0a\x09\x09ifTrue: [ self nonLocalReturnInliner inlineReturn: anIRNonLocalReturn ]\x0a\x09\x09ifFalse: [ super visitIRNonLocalReturn: anIRNonLocalReturn ]",
|
|
|
referencedClasses: ["IRReturn"],
|
|
|
|
|
|
pragmas: [],
|
|
|
-messageSends: ["ifTrue:", "canInlineNonLocalReturns", "scope", "removeNonLocalReturn:", "methodScope", "scope:", "new", "yourself", "do:", "dagChildren", "add:", "replaceWith:", "visitIRReturn:", "visitIRNonLocalReturn:"]
|
|
|
+messageSends: ["ifTrue:", "canInlineNonLocalReturns", "scope", "removeNonLocalReturn:", "methodScope", "scope:", "new", "yourself", "do:", "dagChildren", "add:", "replaceWith:", "visitIRReturn:", "ifTrue:ifFalse:", "shouldInlineReturn:", "inlineReturn:", "nonLocalReturnInliner", "visitIRNonLocalReturn:"]
|
|
|
}, function ($methodClass){ return function (anIRNonLocalReturn){
|
|
|
var self=this,$self=this;
|
|
|
var localReturn;
|
|
@@ -702,7 +728,11 @@ return $recv(localReturn)._add_(each);
|
|
|
$recv(anIRNonLocalReturn)._replaceWith_(localReturn);
|
|
|
return $self._visitIRReturn_(localReturn);
|
|
|
}
|
|
|
-$7=(
|
|
|
+$7=$self._shouldInlineReturn_(anIRNonLocalReturn);
|
|
|
+if($core.assert($7)){
|
|
|
+return $recv($self._nonLocalReturnInliner())._inlineReturn_(anIRNonLocalReturn);
|
|
|
+} else {
|
|
|
+return (
|
|
|
|
|
|
$ctx1.supercall = true,
|
|
|
|
|
@@ -710,7 +740,7 @@ $ctx1.supercall = true,
|
|
|
|
|
|
$ctx1.supercall = false;
|
|
|
|
|
|
-return $7;
|
|
|
+}
|
|
|
|
|
|
}, function($ctx1) {$ctx1.fill(self,"visitIRNonLocalReturn:",{anIRNonLocalReturn:anIRNonLocalReturn,localReturn:localReturn})});
|
|
|
|
|
@@ -2028,6 +2058,89 @@ $globals.IRAssignmentInliner);
|
|
|
|
|
|
|
|
|
|
|
|
+$core.addClass("IRNonLocalReturnInliner", $globals.IRSendInliner, [], "Compiler-Inlining");
|
|
|
+
|
|
|
+$globals.IRNonLocalReturnInliner.comment="I inline message sends with inlined closure together with a return instruction.";
|
|
|
+
|
|
|
+$core.addMethod(
|
|
|
+$core.method({
|
|
|
+selector: "inlineClosure:",
|
|
|
+protocol: "inlining",
|
|
|
+
|
|
|
+args: ["anIRClosure"],
|
|
|
+source: "inlineClosure: anIRClosure\x0a\x09| closure sequence statements |\x0a\x0a\x09closure := super inlineClosure: anIRClosure.\x0a\x09sequence := closure sequence.\x0a\x09statements := sequence dagChildren.\x0a\x09\x0a\x09statements ifNotEmpty: [\x0a\x09\x09| final |\x0a\x09\x09final := statements last.\x0a\x09\x09final yieldsValue ifTrue: [\x0a\x09\x09\x09sequence replace: final with: (IRNonLocalReturn new\x0a\x09\x09\x09\x09add: final copy;\x0a\x09\x09\x09\x09yourself) ] ].\x0a\x0a\x09^ closure",
|
|
|
+referencedClasses: ["IRNonLocalReturn"],
|
|
|
+
|
|
|
+pragmas: [],
|
|
|
+messageSends: ["inlineClosure:", "sequence", "dagChildren", "ifNotEmpty:", "last", "ifTrue:", "yieldsValue", "replace:with:", "add:", "new", "copy", "yourself"]
|
|
|
+}, function ($methodClass){ return function (anIRClosure){
|
|
|
+var self=this,$self=this;
|
|
|
+var closure,sequence,statements;
|
|
|
+
|
|
|
+return $core.withContext(function($ctx1) {
|
|
|
+
|
|
|
+var $1,$2,$3,$5,$4;
|
|
|
+closure=(
|
|
|
+
|
|
|
+$ctx1.supercall = true,
|
|
|
+
|
|
|
+($methodClass.superclass||$boot.nilAsClass).fn.prototype._inlineClosure_.call($self,anIRClosure));
|
|
|
+
|
|
|
+$ctx1.supercall = false;
|
|
|
+
|
|
|
+sequence=$recv(closure)._sequence();
|
|
|
+statements=$recv(sequence)._dagChildren();
|
|
|
+$recv(statements)._ifNotEmpty_((function(){
|
|
|
+var final;
|
|
|
+
|
|
|
+return $core.withContext(function($ctx2) {
|
|
|
+
|
|
|
+final=$recv(statements)._last();
|
|
|
+$1=$recv(final)._yieldsValue();
|
|
|
+if($core.assert($1)){
|
|
|
+$2=sequence;
|
|
|
+$3=final;
|
|
|
+$5=$recv($globals.IRNonLocalReturn)._new();
|
|
|
+$recv($5)._add_($recv(final)._copy());
|
|
|
+$4=$recv($5)._yourself();
|
|
|
+return $recv($2)._replace_with_($3,$4);
|
|
|
+}
|
|
|
+
|
|
|
+}, function($ctx2) {$ctx2.fillBlock({final:final},$ctx1,1)});
|
|
|
+
|
|
|
+}));
|
|
|
+return closure;
|
|
|
+
|
|
|
+}, function($ctx1) {$ctx1.fill(self,"inlineClosure:",{anIRClosure:anIRClosure,closure:closure,sequence:sequence,statements:statements})});
|
|
|
+
|
|
|
+}; }),
|
|
|
+$globals.IRNonLocalReturnInliner);
|
|
|
+
|
|
|
+$core.addMethod(
|
|
|
+$core.method({
|
|
|
+selector: "inlineReturn:",
|
|
|
+protocol: "inlining",
|
|
|
+
|
|
|
+args: ["anIRReturn"],
|
|
|
+source: "inlineReturn: anIRReturn\x0a\x09^ self inlineSend: anIRReturn expression andReplace: anIRReturn",
|
|
|
+referencedClasses: [],
|
|
|
+
|
|
|
+pragmas: [],
|
|
|
+messageSends: ["inlineSend:andReplace:", "expression"]
|
|
|
+}, function ($methodClass){ return function (anIRReturn){
|
|
|
+var self=this,$self=this;
|
|
|
+
|
|
|
+return $core.withContext(function($ctx1) {
|
|
|
+
|
|
|
+return $self._inlineSend_andReplace_($recv(anIRReturn)._expression(),anIRReturn);
|
|
|
+
|
|
|
+}, function($ctx1) {$ctx1.fill(self,"inlineReturn:",{anIRReturn:anIRReturn})});
|
|
|
+
|
|
|
+}; }),
|
|
|
+$globals.IRNonLocalReturnInliner);
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
$core.addClass("IRReturnInliner", $globals.IRSendInliner, [], "Compiler-Inlining");
|
|
|
|
|
|
$globals.IRReturnInliner.comment="I inline message sends with inlined closure together with a return instruction.";
|