diff options
| author | Alon Zakai <alonzakai@gmail.com> | 2013-05-23 15:28:08 -0700 | 
|---|---|---|
| committer | Alon Zakai <alonzakai@gmail.com> | 2013-05-23 15:28:08 -0700 | 
| commit | b90b3ea970f3cc65020730f33ff97c4f2bc2a860 (patch) | |
| tree | 87ef7d60b16d78d9796450c2607e32e141f60e4d | |
| parent | f6ab05e76cbb946bd35b35eaccfb7ae4567f6501 (diff) | |
| parent | a2b9c72b677a23aa665fca5ab9239a725040ca44 (diff) | |
Merge pull request #1204 from sunfishcode/incoming
Optimize (x&A)<<B>>B.
| -rw-r--r-- | AUTHORS | 1 | ||||
| -rw-r--r-- | tools/js-optimizer.js | 18 | ||||
| -rw-r--r-- | tools/test-js-optimizer-asm-pre-output.js | 20 | ||||
| -rw-r--r-- | tools/test-js-optimizer-asm-pre.js | 14 | 
4 files changed, 50 insertions, 3 deletions
| @@ -76,4 +76,5 @@ a license to everyone to use it as detailed in LICENSE.)  * John Allwine <jallwine86@gmail.com>  * Martin Gerhardy <martin.gerhardy@gmail.com>  * James Gregory <jgregory@zynga.com> (copyright owned by Zynga, Inc) +* Dan Gohman <sunfish@google.com> (copyright owned by Google, Inc.) diff --git a/tools/js-optimizer.js b/tools/js-optimizer.js index 32ed1cce..5e7a2448 100644 --- a/tools/js-optimizer.js +++ b/tools/js-optimizer.js @@ -403,6 +403,23 @@ function removeUnneededLabelSettings(ast) {  // Various expression simplifications. Pre run before closure (where we still have metadata), Post run after.  function simplifyExpressionsPre(ast) { +  // Look for (x&A)<<B>>B and replace it with X&A if possible. +  function simplifySignExtends(ast) { +    traverseGenerated(ast, function(node, type) { +      if (type == 'binary'       && node[1]    == '>>' && node[3][0] == 'num' && +          node[2][0] == 'binary' && node[2][1] == '<<' && node[2][3][0] == 'num' && node[3][1] == node[2][3][1]) { +        var innerNode = node[2][2]; +        var shifts = node[3][1]; +        if (innerNode[0] == 'binary' && innerNode[1] == '&' && innerNode[3][0] == 'num') { +          var mask = innerNode[3][1]; +          if (mask << shifts >> shifts == mask) { +            return innerNode; +          } +        } +      } +    }); +  } +    // When there is a bunch of math like (((8+5)|0)+12)|0, only the external |0 is needed, one correction is enough.    // At each node, ((X|0)+Y)|0 can be transformed into (X+Y): The inner corrections are not needed    // TODO: Is the same is true for 0xff, 0xffff? @@ -592,6 +609,7 @@ function simplifyExpressionsPre(ast) {      });    } +  simplifySignExtends(ast);    simplifyBitops(ast);    joinAdditions(ast);    // simplifyZeroComp(ast); TODO: investigate performance diff --git a/tools/test-js-optimizer-asm-pre-output.js b/tools/test-js-optimizer-asm-pre-output.js index 72608aa8..ab953e5d 100644 --- a/tools/test-js-optimizer-asm-pre-output.js +++ b/tools/test-js-optimizer-asm-pre-output.js @@ -18,7 +18,7 @@ function b($this, $__n) {    }    $4 = $this;    $5 = HEAP8[$4 & 16777215] | 0; -  if (($5 & 1) << 24 >> 24 == 0) { +  if (($5 & 1) == 0) {      $14 = 10;      $13 = $5;    } else { @@ -38,14 +38,14 @@ function b($this, $__n) {    } else {      $30 = $13;    } -  if (($30 & 1) << 24 >> 24 == 0) { +  if (($30 & 1) == 0) {      $38 = $this + 1 | 0;    } else {      $38 = HEAP32[($this + 8 & 16777215) >> 2] | 0;    }    _memset($38 + $23 | 0, 0, $__n | 0, 1, 1213141516);    $40 = $23 + $__n | 0; -  if ((HEAP8[$4 & 16777215] & 1) << 24 >> 24 == 0) { +  if ((HEAP8[$4 & 16777215] & 1) == 0) {      HEAP8[$4 & 16777215] = $40 << 1 & 255;    } else {      HEAP32[($this + 4 & 16777215) >> 2] = $40; @@ -93,4 +93,18 @@ function i32_8() {      print(5);    }  } +function sign_extension_simplification() { +  if ((HEAP8[$4 & 16777215] & 127) == 0) { +    print(5); +  } +  if ((HEAP8[$4 & 16777215] & 128) << 24 >> 24 == 0) { +    print(5); +  } +  if ((HEAP32[$5 & 16777215] & 32767) == 0) { +    print(5); +  } +  if ((HEAP32[$5 & 16777215] & 32768) << 16 >> 16 == 0) { +    print(5); +  } +} diff --git a/tools/test-js-optimizer-asm-pre.js b/tools/test-js-optimizer-asm-pre.js index f2ffaef4..264587d2 100644 --- a/tools/test-js-optimizer-asm-pre.js +++ b/tools/test-js-optimizer-asm-pre.js @@ -95,4 +95,18 @@ function i32_8() {      print(5);    }  } +function sign_extension_simplification() { +  if ((HEAP8[$4 & 16777215] & 127) << 24 >> 24 == 0) { +    print(5); +  } +  if ((HEAP8[$4 & 16777215] & 128) << 24 >> 24 == 0) { +    print(5); +  } +  if ((HEAP32[$5 & 16777215] & 32767) << 16 >> 16 == 0) { +    print(5); +  } +  if ((HEAP32[$5 & 16777215] & 32768) << 16 >> 16 == 0) { +    print(5); +  } +}  // EMSCRIPTEN_GENERATED_FUNCTIONS: ["a", "b", "rett", "ret2t", "retf", "i32_8"] | 
