diff options
author | Eduardo Julian | 2018-02-24 01:03:03 -0400 |
---|---|---|
committer | Eduardo Julian | 2018-02-24 01:03:03 -0400 |
commit | cabb410d67edcdcb3531a990518ca67de4507f07 (patch) | |
tree | 1cfca0f79dae0a095503b883426fc8e09abc4cdd | |
parent | 0bd93d82eb7a50b9ce8be42800c388e87e6ca9bf (diff) |
- Fixed nat division and remainder.
-rw-r--r-- | new-luxc/source/luxc/lang/host/js.lux | 11 | ||||
-rw-r--r-- | new-luxc/source/luxc/lang/translation/js/runtime.jvm.lux | 418 |
2 files changed, 86 insertions, 343 deletions
diff --git a/new-luxc/source/luxc/lang/host/js.lux b/new-luxc/source/luxc/lang/host/js.lux index b4c5acd58..7ef948abb 100644 --- a/new-luxc/source/luxc/lang/host/js.lux +++ b/new-luxc/source/luxc/lang/host/js.lux @@ -47,7 +47,7 @@ (list/fold (.function [[test then!] next!] (if! test then! next!)) else! - clauses)) + (list.reverse clauses))) (def: #export (block! statements) (-> (List Statement) Statement) @@ -89,7 +89,7 @@ (-> Expression Expression Expression) (format "(" subject " " <op> " " param ")"))] - [= "="] + [= "==="] [< "<"] [<= "<="] [> ">"] @@ -99,6 +99,13 @@ [* "*"] [/ "/"] [% "%"] + ) + +(do-template [<name> <op>] + [(def: #export (<name> param subject) + (-> Expression Expression Expression) + (format "(" param " " <op> " " subject ")"))] + [or "||"] [and "&&"] [bit-or "|"] diff --git a/new-luxc/source/luxc/lang/translation/js/runtime.jvm.lux b/new-luxc/source/luxc/lang/translation/js/runtime.jvm.lux index f002ccd1f..21aa8f983 100644 --- a/new-luxc/source/luxc/lang/translation/js/runtime.jvm.lux +++ b/new-luxc/source/luxc/lang/translation/js/runtime.jvm.lux @@ -585,359 +585,95 @@ __int//%)) (runtime: nat//< "ltN64" - (format "(function " @ "(l,r) {" - "var li = " int//+ "(l," int//min ");" - "var ri = " int//+ "(r," int//min ");" - "return " int//< "(li,ri);" - "})")) - -(runtime: nat//div-word "divWord" - (format "(function " @ "(result, n, d) {" - "var dLong = " int//new "(0,d);" - (format "if (" int//= "(dLong," int//one ")) {" - (format "result[0] = n.L;" - "result[1] = 0;" - "return") - "}" - "else {" - ## Approximate the quotient and remainder - (format "var q = " int/// "(" bit//shift-right "(n,1)," bit//shift-right "(dLong,1));" - "var r = " int//- "(n," int//* "(q,dLong));" - ## Correct the approximation - (format "while(" int//< "(r," int//zero ")) {" - "r = " int//+ "(r,dLong);" - "q = " int//- "(q," int//one ");" - "}") - (format "while(" int//< "(dLong,r) || " int//= "(dLong,r)) {" - "r = " int//- "(r,dLong);" - "q = " int//+ "(q," int//one ");" - "}") - "result[0] = q.L;" - "result[1] = r.L;" - ) - "}") - "})")) - -(runtime: nat//shift-left|primitive "primitiveShiftLeftBigInt" - (format "(function " @ "(input,shift) {" - "var output = input.slice();" - "var shift2 = 32 - shift;" - (format "for(var i = 0, c = output[i], m = (i + (input.length - 1)); i < m; i++) {" - "var b = c;" - "c = output[i+1];" - "output[i] = (b << shift) | (c >>> shift2);" - "}") - "output[(input.length - 1)] <<= shift;" - "return output;" - "})")) - -(runtime: nat//shift-right|primitive "primitiveShiftRightBigInt" - (format "(function " @ "(input,shift) {" - "var output = input.slice();" - "var shift2 = 32 - shift;" - (format "for(var i = (input.length - 1), c = output[i]; i > 0; i--) {" - "var b = c;" - "c = output[i-1];" - "output[i] = (c << shift2) | (b >>> shift);" - "}") - "output[0] >>>= shift;" - "return output;" - "})")) - -(runtime: nat//shift-left "shiftLeftBigInt" - (format "(function " @ "(input,shift) {" - "var shiftInts = shift >>> 5;" - "var shiftBits = shift & 0x1F;" - "var bitsInHighWord = " bit//count "(" int//new "(input[0],0));" - (format "if(shift <= (32 - bitsInHighWord)) {" - "var shifted = " bit//shift-left "(" int//new "(input[0],input[1]),shiftBits);" - "return [shifted.H,shifted.L];" - "}") - "var inputLen = input[0] === 0 ? 1 : 2;" - "var newLen = inputLen + shiftInts + 1;" - (format "if(shiftBits <= (32 - bitsInHighWord)) {" - "newLen--;" - "}") - (format "if(input.length < newLen) {" - ## The array must grow - "input = [0|0,input[0],input[1]];" - "}") - (format "if(nBits == 0) {" - "return input;" - "}") - (format "if(shiftBits <= (32 - bitsInHighWord)) {" - "return " nat//shift-left|primitive "(input,shiftBits);" - "}" - "else {" - "return " nat//shift-right|primitive "(input,(32 - shiftBits));" - "}") - "})")) - -(runtime: nat//shift-right "shiftRightBigInt" - (format "(function " @ "(input,shift) {" - "var shiftInts = shift >>> 5;" - "var shiftBits = shift & 0x1F;" - "if(shiftBits === 0) { return input; }" - "var bitsInHighWord = " bit//count "(" int//new "(input[0],0));" - (format "if(shiftBits >= bitsInHighWord) {" - "return " nat//shift-left|primitive "(input,(32-shiftBits));" - "}" - "else {" - "return " nat//shift-right|primitive "(input,shiftBits);" - "}") - "})")) + (let [high (function [i64] (format "(" i64 "." //.int-high-field ")")) + low (function [i64] (format "(" i64 "." //.int-low-field ")")) + i32 (function [word] (format "(" word " >>> 0)"))] + (js.function @ (list "subject" "parameter") + (list (js.return! (js.or (js.> (i32 (high "subject")) + (i32 (high "parameter"))) + (js.and (js.= (high "subject") + (high "parameter")) + (js.> (i32 (low "subject")) + (i32 (low "parameter")))))))))) + +(def: (<N param subject) + (-> Expression Expression Expression) + (js.apply nat//< (list subject param))) -(runtime: nat//mul-sub "mulsubBigInt" - (format "(function " @ "(q, a, x, len, offset) {" - "var xLong = " int//new "(0,x);" - "var carry = " int//zero ";" - "offset += len;" - (format "for (var j = len-1; j >= 0; j--) {" - "var product = " int//+ "(" int//* "(" int//new "(0,a[j]),xLong),carry);" - "var difference = " int//- "(" int//new "(0,q[offset]),product);" - "carry = " int//+ "(" bit//shift-right "(product,32),((difference.L > ~product.L) ? " int//one " : " int//zero "));" - "}") - "return carry.L;" - "})")) +(def: (<=N param subject) + (-> Expression Expression Expression) + (js.or (js.apply nat//< (list subject param)) + (js.apply int//= (list subject param)))) -(runtime: nat//div-add "divadd" - (format "(function " @ "(a, result, offset) {" - "var carry = " int//zero ";" - (format "for (var j = a.length - 1; j >= 0; j--) {" - "var sum = " int//+ "(" int//+ "(" int//new "(0,a[j])," int//new "(0,result[j+offset])),carry);" - "result[j+offset] = sum.L;" - "carry = " bit//shift-right "(sum,32);" - "}") - "return carry.L;" - "})")) +(def: (>N param subject) + (-> Expression Expression Expression) + (js.apply nat//< (list param subject))) -(runtime: nat//normalize "normalizeBigInt" - (format "(function " @ "(input) {" - (format "if(input[0] !== 0) {" - "return " int//new "(input[0],input[1]);" - "}" - "else {" - (format "var numZeros = 0;" - (format "do {" - "numZeros++;" - "} while(numZeros < input.length && input[numZeros] == 0);") - "var tempInput = input.slice(input.length-Math.max(2,input.length-numZeros));" - "return " int//new "(tempInput[0],tempInput[1]);") - "}") - "})")) +(def: (>=N param subject) + (-> Expression Expression Expression) + (js.or (js.apply nat//< (list param subject)) + (js.apply int//= (list subject param)))) -(runtime: nat//divide-one-word "divideOneWord" - (format "(function " @ "(subject,param) {" - (format "var divLong = " int//new "(0,param);" - ## Special case of one word dividend - (format "if(subject.H === 0) {" - (format "var remValue = " int//new "(0,subject.L);" - "var quotient = " int/// "(remValue,divLong);" - "var remainder = " int//- "(remValue," int//* "(quotient.L,divLong));" - "return [quotient,remainder];") - "}") - "var quotient = [0|0,0|0];" - ## Normalize the divisor - "var shift = 32 - " bit//count "(" int//new "(0,param));" - "var rem = subject.H;" - "var remLong = " int//new "(0,rem);" - (format "if(" int//< "(remLong,divLong)) {" - "quotient[0] = 0|0;" - "}" - "else {" - "quotient[0] = " int/// "(remLong,divLong).L;" - "rem = " int//- "(remLong," int//* "(quotient[0],divLong)).L;" - "remLong = " int//new "(0,rem);" - "}") - "var remBI = [subject.H,subject.L];" - "var xlen = 2;" - "var qWord = [0|0,0|0];" - (format "while(--xlen > 0) {" - "var dividendEstimate = " bit//or "(" bit//shift-left "(remLong,32)," int//new "(0,remBI[2 - xlen]));" - (format "if(dividendEstimate >= 0) {" - "var highWord = " int/// "(dividendEstimate,divLong);" - "qWord[0] = highWord.L;" - "qWord[1] = " int//- "(dividendEstimate," int//* "(highWord,divLong)).L;" - "}" - "else {" - "" nat//div-word "(qWord, dividendEstimate, param);" - "}") - "quotient[2 - xlen] = qWord[0];" - "rem = qWord[1];" - "remLong = " int//new "(0,rem);" - "}") - ## Unnormalize - (format "if(shift > 0) {" - "rem %= divisor;" - "remBI[0] = rem;" - "}" - "else {" - "remBI[0] = rem;" - "}") - "var quotI64 = " nat//normalize "(quotient);" - "var remI64 = " int//new "(remBI[0],remBI[1]);" - "return [quotI64,remI64];") - "})")) +(runtime: nat/// "divN64" + (let [negative? (function [value] + (js.apply int//< (list value int//zero))) + valid-division-check [(=I int//zero "parameter") + (js.throw! (js.string "Cannot divide by zero!"))] + short-circuit-check [(=I int//zero "subject") + (js.return! int//zero)]] + (js.function @ (list "subject" "parameter") + (list (js.cond! (list valid-division-check + short-circuit-check -(runtime: nat//div-mod "divmodBigInt" - (format "(function " @ "(subject,param) {" - (format "if(" int//= "(param," int//zero ")) {" - "throw new Error('Cannot divide by zero!');" - "}") - (format "if(" int//= "(subject," int//zero ")) {" - "return [" int//zero ", " int//zero "];" - "}") - (format "if(" nat//< "(subject,param)) {" - "return [" int//zero ", subject];" - "}") - (format "if(" int//= "(subject,param)) {" - "return [" int//one ", " int//zero "];" - "}") - ##################################### - (format "if (param.H === 0) {" - "return " nat//divide-one-word "(subject,param.L);" - "}") - ##################################### - "var divisor = param;" - "var remainder = subject.H === 0 ? [0|0,subject.L] : [0|0,subject.H,subject.L];" - "var paramLength = param.H === 0 ? 1 : 2;" - "var subjLength = subject.H === 0 ? 1 : 2;" - "var limit = subjLength - paramLength + 1;" - "var quotient = (limit === 1) ? [0|0] : [0|0,0|0];" - ## Normalize the divisor - "var shift = 32 - " bit//count "(" int//new "(divisor.H,0));" - (format "if(shift > 0) {" - "divisor = " bit//shift-left "(divisor,shift);" - "remainder = " nat//shift-left "(remainder,shift);" - "}") - (format "if((remainder.length-1) === subjLength) {" - "remainder[0] = 0;" - "}") - "var dh = divisor.H;" - "var dhLong = " int//new "(0,dh);" - "var dl = divisor.L;" - "var qWord = [0|0,0|0];" - ## D2 Initialize j - (format "for(var j = 0; j < limit; j++) {" - ## D3 Calculate qhat - ## estimate qhat - "var qhat = 0;" - "var qrem = 0;" - "var skipCorrection = false;" - "var nh = remainder[j];" - "var nh2 = nh + 0x80000000;" - "var nm = remainder[j+1];" - (format "if(nh == dh) {" - (format "qhat = ~0;" - "qrem = nh + nm;" - "skipCorrection = (qrem + 0x80000000) < nh2;") - "}" - "else {" - (format "var nChunk = " bit//or "(" bit//shift-left "(" int//from-number "(nh),32)," int//from-number "(nm));") - (format "if(" int//< "(" int//zero ",nChunk) || " int//= "(" int//zero ",nChunk)) {" - (format "qhat = " int/// "(nChunk,dhLong).L;" - "qrem = " int//- "(nChunk," int//* "(qhat, dhLong)).L;") - "}" - "else {" - (format "" nat//div-word "(qWord, nChunk, dh);" - "qhat = qWord[0];" - "qrem = qWord[1];" - ) - "}") - "if(qhat == 0) { continue; }" - (format "if(!skipCorrection) {" - ## Correct qhat - (format "var qremLong = " int//new "(0,qrem);" - "var dlLong = " int//new "(0,dl);" - "var nl = " int//new "(0,remainder[j+2]);" - "var rs = " bit//or "(" bit//shift-left "(qremLong,32),nl);" - "var estProduct = " int//* "(dlLong," int//new "(0,qhat));" - (format "if(" nat//< "(rs,estProduct)) {" - (format "qhat--;" - "qrem = " int//+ "(qremLong,dhLong).L;" - "qremLong = " int//new "(0,qrem);" - (format "if(" int//< "(dhLong,qremLong) || " int//= "(dhLong,qremLong)) {" - (format "estProduct = " int//* "(dlLong," int//new "(0,qhat));" - "rs = " bit//or "(" bit//shift-left "(qremLong,32),nl);" - "if(" nat//< "(rs,estProduct)) { qhat--; }") - "}")) - "}") - ) - "}") - ## D4 Multiply and subtract - "remainder[j] = 0;" - "var borrow = " nat//mul-sub "(remainder, divisor, qhat, paramLength, j);" - ## D5 Test remainder - (format "if((borrow + 0x80000000) > nh2) {" - ## D6 Add back - nat//div-add "(divisor, remainder, j+1);" - "qhat--;" - "}") - ## Store the quotient digit - "quotient[j] = qhat;" - "}") - "}") ## D7 loop on j - ## D8 Unnormalize - "if(shift > 0) { remainder = " nat//shift-right "(remainder,shift); }" - "return [" nat//normalize "(quotient), " nat//normalize "(remainder)];" - "})")) + [(>N "subject" "parameter") + (js.return! int//zero)] -(runtime: nat/// "divN64" - (format "(function " @ "(l,r) {" - (format "if(" int//< "(r," int//zero ")) {" - (format "if(" nat//< "(l,r)) {" - "return " int//zero ";" - "}" - "else {" - "return " int//one ";" - "}") - "}" - "else if(" int//< "(" int//zero ",l)) {" - "return " int/// "(l,r);" - "}" - "else {" - (format "if(" int//= "(" int//zero ",r)) {" - "throw new Error('Cannot divide by zero!');" - "}" - "else {" - (format "if(" int//< "(l,r)) {" - "return " int//zero ";" - "}" - "else {" - "return " nat//div-mod "(l,r)[0];" - "}") - "}") - "}") - "})")) + [(>N (js.apply bit//shift-right + (list "subject" (js.number 1.0))) + "parameter") + (js.return! int//one)]) + (js.block! (list (js.var! "result" (#.Some int//zero)) + (js.var! "remainder" (#.Some "subject")) + (js.while! (>=N "parameter" "remainder") + (let [rough-estimate (js.apply "Math.floor" (list (js./ (js.apply int//to-number (list "parameter")) + (js.apply int//to-number (list "remainder"))))) + log2 (js./ "Math.LN2" + (js.apply "Math.log" (list "approximate"))) + approx-result (js.apply int//from-number (list "approximate")) + approx-remainder (js.apply int//* (list "approximate_result" "parameter"))] + (list (js.var! "approximate" (#.Some (js.apply "Math.max" (list (js.number 1.0) + rough-estimate)))) + (js.var! "log2" (#.Some (js.apply "Math.ceil" (list log2)))) + (js.var! "delta" (#.Some (js.? (js.<= (js.number 48.0) "log2") + (js.number 1.0) + (js.apply "Math.pow" (list (js.number 2.0) + (js.- (js.number 48.0) + "log2")))))) + (js.var! "approximate_result" (#.Some approx-result)) + (js.var! "approximate_remainder" (#.Some approx-remainder)) + (js.while! (js.or (negative? "approximate_remainder") + (>N "remainder" + "approximate_remainder")) + (list (js.set! "approximate" (js.- "delta" "approximate")) + (js.set! "approximate_result" approx-result) + (js.set! "approximate_remainder" approx-remainder))) + (js.block! (list (js.set! "result" (js.apply int//+ (list "result" + (js.? (=I int//zero "approximate_result") + int//one + "approximate_result")))) + (js.set! "remainder" (js.apply int//- (list "remainder" "approximate_remainder")))))))) + (js.return! "result"))) + ))))) (runtime: nat//% "remN64" - (format "(function " @ "(l,r) {" - (format "if(" int//< "(l," int//zero ") || " int//< "(r," int//zero ")) {" - (format "if(" nat//< "(l,r)) {" - "return l;" - "}" - "else {" - "return " nat//div-mod "(l,r)[1];" - "}") - "}" - "else {" - "return " int//% "(l,r);" - "}") - "})")) + (js.function @ (list "subject" "parameter") + (list (let [flat (js.apply int//* (list (js.apply nat/// (list "subject" "parameter")) + "parameter"))] + (js.return! (js.apply int//- (list "subject" flat))))))) (def: runtime//nat Runtime (format __nat//< - __nat//div-word - __nat//shift-left|primitive - __nat//shift-right|primitive - __nat//shift-left - __nat//shift-right - __nat//mul-sub - __nat//div-add - __nat//normalize - __nat//divide-one-word - __nat//div-mod __nat/// __nat//%)) |