diff options
Diffstat (limited to '')
-rw-r--r-- | OpenSim/Region/ScriptEngine/YEngine/MMRScriptCollector.cs | 187 |
1 files changed, 61 insertions, 126 deletions
diff --git a/OpenSim/Region/ScriptEngine/YEngine/MMRScriptCollector.cs b/OpenSim/Region/ScriptEngine/YEngine/MMRScriptCollector.cs index 4a57823..88cd6c1 100644 --- a/OpenSim/Region/ScriptEngine/YEngine/MMRScriptCollector.cs +++ b/OpenSim/Region/ScriptEngine/YEngine/MMRScriptCollector.cs | |||
@@ -511,20 +511,16 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
511 | } | 511 | } |
512 | public override bool MoveNext() | 512 | public override bool MoveNext() |
513 | { | 513 | { |
514 | /* | 514 | // First off, return any targets the instruction can come up with. |
515 | * First off, return any targets the instruction can come up with. | ||
516 | */ | ||
517 | if(realEnumerator.MoveNext()) | 515 | if(realEnumerator.MoveNext()) |
518 | { | 516 | { |
519 | nn = realEnumerator.Current; | 517 | nn = realEnumerator.Current; |
520 | return true; | 518 | return true; |
521 | } | 519 | } |
522 | 520 | ||
523 | /* | 521 | // Then if this instruction is in a try section, say this instruction |
524 | * Then if this instruction is in a try section, say this instruction | 522 | // can potentially branch to the beginning of the corresponding |
525 | * can potentially branch to the beginning of the corresponding | 523 | // catch/finally. |
526 | * catch/finally. | ||
527 | */ | ||
528 | if((index == 0) && (gn.tryBlock != null)) | 524 | if((index == 0) && (gn.tryBlock != null)) |
529 | { | 525 | { |
530 | index++; | 526 | index++; |
@@ -532,9 +528,7 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
532 | return true; | 528 | return true; |
533 | } | 529 | } |
534 | 530 | ||
535 | /* | 531 | // That's all we can do. |
536 | * That's all we can do. | ||
537 | */ | ||
538 | nn = null; | 532 | nn = null; |
539 | return false; | 533 | return false; |
540 | } | 534 | } |
@@ -1875,9 +1869,7 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
1875 | } | 1869 | } |
1876 | public override bool MoveNext() | 1870 | public override bool MoveNext() |
1877 | { | 1871 | { |
1878 | /* | 1872 | // Return next from list of switch case labels. |
1879 | * Return next from list of switch case labels. | ||
1880 | */ | ||
1881 | while(index < gn.myLabels.Length) | 1873 | while(index < gn.myLabels.Length) |
1882 | { | 1874 | { |
1883 | nn = gn.myLabels[index++].whereAmI; | 1875 | nn = gn.myLabels[index++].whereAmI; |
@@ -1885,9 +1877,7 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
1885 | return true; | 1877 | return true; |
1886 | } | 1878 | } |
1887 | 1879 | ||
1888 | /* | 1880 | // If all ran out, the switch instruction falls through. |
1889 | * If all ran out, the switch instruction falls through. | ||
1890 | */ | ||
1891 | if(index == gn.myLabels.Length) | 1881 | if(index == gn.myLabels.Length) |
1892 | { | 1882 | { |
1893 | index++; | 1883 | index++; |
@@ -1895,9 +1885,7 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
1895 | return true; | 1885 | return true; |
1896 | } | 1886 | } |
1897 | 1887 | ||
1898 | /* | 1888 | // Even ran out of that, say there's nothing more. |
1899 | * Even ran out of that, say there's nothing more. | ||
1900 | */ | ||
1901 | nn = null; | 1889 | nn = null; |
1902 | return false; | 1890 | return false; |
1903 | } | 1891 | } |
@@ -2527,10 +2515,8 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
2527 | if(curExcBlock != null) | 2515 | if(curExcBlock != null) |
2528 | throw new Exception("exception block still open"); | 2516 | throw new Exception("exception block still open"); |
2529 | 2517 | ||
2530 | /* | 2518 | // If an instruction says it doesn't fall through, remove all instructions to |
2531 | * If an instruction says it doesn't fall through, remove all instructions to | 2519 | // the end of the block. |
2532 | * the end of the block. | ||
2533 | */ | ||
2534 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) | 2520 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) |
2535 | { | 2521 | { |
2536 | if(!gn.CanFallThrough()) | 2522 | if(!gn.CanFallThrough()) |
@@ -2547,12 +2533,10 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
2547 | } | 2533 | } |
2548 | } | 2534 | } |
2549 | 2535 | ||
2550 | /* | 2536 | // Scan for OpCodes.Leave instructions. |
2551 | * Scan for OpCodes.Leave instructions. | 2537 | // For each found, its target for flow analysis purposes is the beginning of the corresponding |
2552 | * For each found, its target for flow analysis purposes is the beginning of the corresponding | 2538 | // finally block. And the end of the finally block gets a conditional branch target of the |
2553 | * finally block. And the end of the finally block gets a conditional branch target of the | 2539 | // leave instruction's target. A leave instruction can unwind zero or more finally blocks. |
2554 | * leave instruction's target. A leave instruction can unwind zero or more finally blocks. | ||
2555 | */ | ||
2556 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) | 2540 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) |
2557 | { | 2541 | { |
2558 | if(gn is GraphNodeEmitLabelLeave) | 2542 | if(gn is GraphNodeEmitLabelLeave) |
@@ -2562,12 +2546,10 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
2562 | GraphNodeBeginExceptionBlock leaveTargetsTryBlock = // try block directly enclosing leave target | 2546 | GraphNodeBeginExceptionBlock leaveTargetsTryBlock = // try block directly enclosing leave target |
2563 | (leaveTarget == null) ? null : leaveTarget.tryBlock; // ...it must not be unwound | 2547 | (leaveTarget == null) ? null : leaveTarget.tryBlock; // ...it must not be unwound |
2564 | 2548 | ||
2565 | /* | 2549 | // Step through try { }s from the leave instruction towards its target looking for try { }s with finally { }s. |
2566 | * Step through try { }s from the leave instruction towards its target looking for try { }s with finally { }s. | 2550 | // The leave instruction unconditionally branches to the beginning of the innermost one found. |
2567 | * The leave instruction unconditionally branches to the beginning of the innermost one found. | 2551 | // The end of the last one found conditionally branches to the leave instruction's target. |
2568 | * The end of the last one found conditionally branches to the leave instruction's target. | 2552 | // If none found, the leave is a simple unconditional branch to its target. |
2569 | * If none found, the leave is a simple unconditional branch to its target. | ||
2570 | */ | ||
2571 | GraphNodeBeginFinallyBlock innerFinallyBlock = null; | 2553 | GraphNodeBeginFinallyBlock innerFinallyBlock = null; |
2572 | for(GraphNodeBeginExceptionBlock tryBlock = leaveInstr.tryBlock; | 2554 | for(GraphNodeBeginExceptionBlock tryBlock = leaveInstr.tryBlock; |
2573 | tryBlock != leaveTargetsTryBlock; | 2555 | tryBlock != leaveTargetsTryBlock; |
@@ -2586,10 +2568,8 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
2586 | } | 2568 | } |
2587 | } | 2569 | } |
2588 | 2570 | ||
2589 | /* | 2571 | // The end of the outermost finally being unwound can conditionally jump to the target of the leave instruction. |
2590 | * The end of the outermost finally being unwound can conditionally jump to the target of the leave instruction. | 2572 | // In the case of no finallies being unwound, the leave is just a simple unconditional branch. |
2591 | * In the case of no finallies being unwound, the leave is just a simple unconditional branch. | ||
2592 | */ | ||
2593 | if(innerFinallyBlock == null) | 2573 | if(innerFinallyBlock == null) |
2594 | { | 2574 | { |
2595 | leaveInstr.unwindTo = leaveTarget; | 2575 | leaveInstr.unwindTo = leaveTarget; |
@@ -2601,10 +2581,8 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
2601 | } | 2581 | } |
2602 | } | 2582 | } |
2603 | 2583 | ||
2604 | /* | 2584 | // See which variables a particular block reads before writing. |
2605 | * See which variables a particular block reads before writing. | 2585 | // This just considers the block itself and nothing that it branches to or fallsthru to. |
2606 | * This just considers the block itself and nothing that it branches to or fallsthru to. | ||
2607 | */ | ||
2608 | GraphNodeBlock currentBlock = null; | 2586 | GraphNodeBlock currentBlock = null; |
2609 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) | 2587 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) |
2610 | { | 2588 | { |
@@ -2626,13 +2604,11 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
2626 | } | 2604 | } |
2627 | } | 2605 | } |
2628 | 2606 | ||
2629 | /* | 2607 | // For every block we branch to, add that blocks readables to our list of readables, |
2630 | * For every block we branch to, add that blocks readables to our list of readables, | 2608 | // because we need to have those values valid on entry to our block. But if we write the |
2631 | * because we need to have those values valid on entry to our block. But if we write the | 2609 | // variable before we can possibly branch to that block, then we don't need to have it valid |
2632 | * variable before we can possibly branch to that block, then we don't need to have it valid | 2610 | // on entry to our block. So basically it looks like the branch instruction is reading |
2633 | * on entry to our block. So basically it looks like the branch instruction is reading | 2611 | // everything required by any blocks it can branch to. |
2634 | * everything required by any blocks it can branch to. | ||
2635 | */ | ||
2636 | do | 2612 | do |
2637 | { | 2613 | { |
2638 | this.resolvedSomething = false; | 2614 | this.resolvedSomething = false; |
@@ -2640,17 +2616,13 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
2640 | this.ResolveBlock((GraphNodeBlock)firstLin); | 2616 | this.ResolveBlock((GraphNodeBlock)firstLin); |
2641 | } while(this.resolvedSomething); | 2617 | } while(this.resolvedSomething); |
2642 | 2618 | ||
2643 | /* | 2619 | // Repeat the cutting loops as long as we keep finding stuff. |
2644 | * Repeat the cutting loops as long as we keep finding stuff. | ||
2645 | */ | ||
2646 | bool didSomething; | 2620 | bool didSomething; |
2647 | do | 2621 | do |
2648 | { | 2622 | { |
2649 | didSomething = false; | 2623 | didSomething = false; |
2650 | 2624 | ||
2651 | /* | 2625 | // Strip out ldc.i4.1/xor/ldc.i4.1/xor |
2652 | * Strip out ldc.i4.1/xor/ldc.i4.1/xor | ||
2653 | */ | ||
2654 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) | 2626 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) |
2655 | { | 2627 | { |
2656 | if(!(gn is GraphNodeEmit)) | 2628 | if(!(gn is GraphNodeEmit)) |
@@ -2678,9 +2650,7 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
2678 | didSomething = true; | 2650 | didSomething = true; |
2679 | } | 2651 | } |
2680 | 2652 | ||
2681 | /* | 2653 | // Replace c{cond}/ldc.i4.1/xor/br{false,true} -> c{cond}/br{true,false} |
2682 | * Replace c{cond}/ldc.i4.1/xor/br{false,true} -> c{cond}/br{true,false} | ||
2683 | */ | ||
2684 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) | 2654 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) |
2685 | { | 2655 | { |
2686 | if(!(gn is GraphNodeEmit)) | 2656 | if(!(gn is GraphNodeEmit)) |
@@ -2711,9 +2681,7 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
2711 | didSomething = true; | 2681 | didSomething = true; |
2712 | } | 2682 | } |
2713 | 2683 | ||
2714 | /* | 2684 | // Replace c{cond}/br{false,true} -> b{!,}{cond} |
2715 | * Replace c{cond}/br{false,true} -> b{!,}{cond} | ||
2716 | */ | ||
2717 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) | 2685 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) |
2718 | { | 2686 | { |
2719 | if(!(gn is GraphNodeEmit)) | 2687 | if(!(gn is GraphNodeEmit)) |
@@ -2746,9 +2714,7 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
2746 | didSomething = true; | 2714 | didSomething = true; |
2747 | } | 2715 | } |
2748 | 2716 | ||
2749 | /* | 2717 | // Replace ld{c.i4.0,null}/br{ne.un,eq} -> br{true,false} |
2750 | * Replace ld{c.i4.0,null}/br{ne.un,eq} -> br{true,false} | ||
2751 | */ | ||
2752 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) | 2718 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) |
2753 | { | 2719 | { |
2754 | if(!(gn is GraphNodeEmit)) | 2720 | if(!(gn is GraphNodeEmit)) |
@@ -2767,17 +2733,15 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
2767 | didSomething = true; | 2733 | didSomething = true; |
2768 | } | 2734 | } |
2769 | 2735 | ||
2770 | /* | 2736 | // Replace: |
2771 | * Replace: | 2737 | // ldloc v1 |
2772 | * ldloc v1 | 2738 | // stloc v2 |
2773 | * stloc v2 | 2739 | // ld<anything> except ld<anything> v2 |
2774 | * ld<anything> except ld<anything> v2 | 2740 | // ldloc v2 |
2775 | * ldloc v2 | 2741 | // ...v2 unreferenced hereafter |
2776 | * ...v2 unreferenced hereafter | 2742 | // With: |
2777 | * With: | 2743 | // ld<anything> except ld<anything> v2 |
2778 | * ld<anything> except ld<anything> v2 | 2744 | // ldloc v1 |
2779 | * ldloc v1 | ||
2780 | */ | ||
2781 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) | 2745 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) |
2782 | { | 2746 | { |
2783 | 2747 | ||
@@ -2833,11 +2797,9 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
2833 | didSomething = true; | 2797 | didSomething = true; |
2834 | } | 2798 | } |
2835 | 2799 | ||
2836 | /* | 2800 | // Remove all the stloc/ldloc that are back-to-back without the local |
2837 | * Remove all the stloc/ldloc that are back-to-back without the local | 2801 | // being needed afterwards. If it is needed afterwards, replace the |
2838 | * being needed afterwards. If it is needed afterwards, replace the | 2802 | // stloc/ldloc with dup/stloc. |
2839 | * stloc/ldloc with dup/stloc. | ||
2840 | */ | ||
2841 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) | 2803 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) |
2842 | { | 2804 | { |
2843 | if((gn is GraphNodeEmitLocal) && | 2805 | if((gn is GraphNodeEmitLocal) && |
@@ -2871,10 +2833,8 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
2871 | } | 2833 | } |
2872 | } | 2834 | } |
2873 | 2835 | ||
2874 | /* | 2836 | // Remove all write-only local variables, ie, those with no ldloc[a] references. |
2875 | * Remove all write-only local variables, ie, those with no ldloc[a] references. | 2837 | // Replace any stloc instructions with pops. |
2876 | * Replace any stloc instructions with pops. | ||
2877 | */ | ||
2878 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) | 2838 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) |
2879 | { | 2839 | { |
2880 | ScriptMyLocal rdlcl = gn.ReadsLocal(); | 2840 | ScriptMyLocal rdlcl = gn.ReadsLocal(); |
@@ -2900,9 +2860,7 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
2900 | } | 2860 | } |
2901 | } | 2861 | } |
2902 | 2862 | ||
2903 | /* | 2863 | // Remove any Ld<const>/Dup,Pop. |
2904 | * Remove any Ld<const>/Dup,Pop. | ||
2905 | */ | ||
2906 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) | 2864 | for(GraphNode gn = firstLin; gn != null; gn = gn.nextLin) |
2907 | { | 2865 | { |
2908 | if((gn is GraphNodeEmit) && | 2866 | if((gn is GraphNodeEmit) && |
@@ -2921,9 +2879,7 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
2921 | } | 2879 | } |
2922 | } while(didSomething); | 2880 | } while(didSomething); |
2923 | 2881 | ||
2924 | /* | 2882 | // Dump out the results. |
2925 | * Dump out the results. | ||
2926 | */ | ||
2927 | if(DEBUG) | 2883 | if(DEBUG) |
2928 | { | 2884 | { |
2929 | Console.WriteLine(""); | 2885 | Console.WriteLine(""); |
@@ -2982,55 +2938,39 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
2982 | if(currentBlock.hasBeenResolved == this.resolveSequence) | 2938 | if(currentBlock.hasBeenResolved == this.resolveSequence) |
2983 | return; | 2939 | return; |
2984 | 2940 | ||
2985 | /* | 2941 | // So we don't recurse forever on a backward branch. |
2986 | * So we don't recurse forever on a backward branch. | ||
2987 | */ | ||
2988 | currentBlock.hasBeenResolved = this.resolveSequence; | 2942 | currentBlock.hasBeenResolved = this.resolveSequence; |
2989 | 2943 | ||
2990 | /* | 2944 | // Assume we haven't written any locals yet. |
2991 | * Assume we haven't written any locals yet. | ||
2992 | */ | ||
2993 | List<ScriptMyLocal> localsWrittenSoFar = new List<ScriptMyLocal>(); | 2945 | List<ScriptMyLocal> localsWrittenSoFar = new List<ScriptMyLocal>(); |
2994 | 2946 | ||
2995 | /* | 2947 | // Scan through the instructions in this block. |
2996 | * Scan through the instructions in this block. | ||
2997 | */ | ||
2998 | for(GraphNode gn = currentBlock; gn != null;) | 2948 | for(GraphNode gn = currentBlock; gn != null;) |
2999 | { | 2949 | { |
3000 | 2950 | ||
3001 | /* | 2951 | // See if the instruction writes a local we don't know about yet. |
3002 | * See if the instruction writes a local we don't know about yet. | ||
3003 | */ | ||
3004 | ScriptMyLocal wrlcl = gn.WritesLocal(); | 2952 | ScriptMyLocal wrlcl = gn.WritesLocal(); |
3005 | if((wrlcl != null) && !localsWrittenSoFar.Contains(wrlcl)) | 2953 | if((wrlcl != null) && !localsWrittenSoFar.Contains(wrlcl)) |
3006 | { | 2954 | { |
3007 | localsWrittenSoFar.Add(wrlcl); | 2955 | localsWrittenSoFar.Add(wrlcl); |
3008 | } | 2956 | } |
3009 | 2957 | ||
3010 | /* | 2958 | // Scan through all the possible next instructions after this. |
3011 | * Scan through all the possible next instructions after this. | 2959 | // Note that if we are in the first part of a try/catch/finally block, |
3012 | * Note that if we are in the first part of a try/catch/finally block, | 2960 | // every instruction conditionally branches to the beginning of the |
3013 | * every instruction conditionally branches to the beginning of the | 2961 | // second part (the catch/finally block). |
3014 | * second part (the catch/finally block). | ||
3015 | */ | ||
3016 | GraphNode nextFallthruNode = null; | 2962 | GraphNode nextFallthruNode = null; |
3017 | foreach(GraphNode nn in gn.NextNodes) | 2963 | foreach(GraphNode nn in gn.NextNodes) |
3018 | { | 2964 | { |
3019 | if(nn is GraphNodeBlock) | 2965 | if(nn is GraphNodeBlock) |
3020 | { | 2966 | { |
3021 | 2967 | // Start of a block, go through all locals needed by that block on entry. | |
3022 | /* | ||
3023 | * Start of a block, go through all locals needed by that block on entry. | ||
3024 | */ | ||
3025 | GraphNodeBlock nextBlock = (GraphNodeBlock)nn; | 2968 | GraphNodeBlock nextBlock = (GraphNodeBlock)nn; |
3026 | ResolveBlock(nextBlock); | 2969 | ResolveBlock(nextBlock); |
3027 | foreach(ScriptMyLocal readByNextBlock in nextBlock.localsReadBeforeWritten) | 2970 | foreach(ScriptMyLocal readByNextBlock in nextBlock.localsReadBeforeWritten) |
3028 | { | 2971 | { |
3029 | 2972 | // If this block hasn't written it by now and this block doesn't already | |
3030 | /* | 2973 | // require it on entry, say this block requires it on entry. |
3031 | * If this block hasn't written it by now and this block doesn't already | ||
3032 | * require it on entry, say this block requires it on entry. | ||
3033 | */ | ||
3034 | if(!localsWrittenSoFar.Contains(readByNextBlock) && | 2974 | if(!localsWrittenSoFar.Contains(readByNextBlock) && |
3035 | !currentBlock.localsReadBeforeWritten.Contains(readByNextBlock)) | 2975 | !currentBlock.localsReadBeforeWritten.Contains(readByNextBlock)) |
3036 | { | 2976 | { |
@@ -3041,19 +2981,14 @@ namespace OpenSim.Region.ScriptEngine.Yengine | |||
3041 | } | 2981 | } |
3042 | else | 2982 | else |
3043 | { | 2983 | { |
3044 | 2984 | // Not start of a block, should be normal fallthru instruction. | |
3045 | /* | ||
3046 | * Not start of a block, should be normal fallthru instruction. | ||
3047 | */ | ||
3048 | if(nextFallthruNode != null) | 2985 | if(nextFallthruNode != null) |
3049 | throw new Exception("more than one fallthru from " + gn.ToString()); | 2986 | throw new Exception("more than one fallthru from " + gn.ToString()); |
3050 | nextFallthruNode = nn; | 2987 | nextFallthruNode = nn; |
3051 | } | 2988 | } |
3052 | } | 2989 | } |
3053 | 2990 | ||
3054 | /* | 2991 | // Process next instruction if it isn't the start of a block. |
3055 | * Process next instruction if it isn't the start of a block. | ||
3056 | */ | ||
3057 | if(nextFallthruNode == gn) | 2992 | if(nextFallthruNode == gn) |
3058 | throw new Exception("can't fallthru to self"); | 2993 | throw new Exception("can't fallthru to self"); |
3059 | gn = nextFallthruNode; | 2994 | gn = nextFallthruNode; |