Compiles some of the FunctionType 4
This commit is contained in:
		
							parent
							
								
									87de3cd2ec
								
							
						
					
					
						commit
						cb81bd6be6
					
				| @ -386,6 +386,18 @@ var PDFFunction = (function PDFFunctionClosure() { | |||||||
|       var domain = IR[1]; |       var domain = IR[1]; | ||||||
|       var range = IR[2]; |       var range = IR[2]; | ||||||
|       var code = IR[3]; |       var code = IR[3]; | ||||||
|  | 
 | ||||||
|  |       var compiled = (new PostScriptCompiler()).compile(code, domain, range); | ||||||
|  |       if (compiled) { | ||||||
|  |         // Compiled function consists of simple expressions such as addition,
 | ||||||
|  |         // subtraction, Math.max, and also contains 'var' and 'return'
 | ||||||
|  |         // statements. See the generation in the PostScriptCompiler below.
 | ||||||
|  |         /*jshint -W054 */ | ||||||
|  |         return new Function('args', compiled); | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       info('Unable to compile PS function'); | ||||||
|  | 
 | ||||||
|       var numOutputs = range.length >> 1; |       var numOutputs = range.length >> 1; | ||||||
|       var numInputs = domain.length >> 1; |       var numInputs = domain.length >> 1; | ||||||
|       var evaluator = new PostScriptEvaluator(code); |       var evaluator = new PostScriptEvaluator(code); | ||||||
| @ -736,3 +748,368 @@ var PostScriptEvaluator = (function PostScriptEvaluatorClosure() { | |||||||
|   }; |   }; | ||||||
|   return PostScriptEvaluator; |   return PostScriptEvaluator; | ||||||
| })(); | })(); | ||||||
|  | 
 | ||||||
|  | // Most of the PDFs functions consist of simple operations such as:
 | ||||||
|  | //   roll, exch, sub, cvr, pop, index, dup, mul, if, gt, add.
 | ||||||
|  | //
 | ||||||
|  | // We can compile most of such programs, and at the same moment, we can
 | ||||||
|  | // optimize some expressions using basic math properties. Keeping track of
 | ||||||
|  | // min/max values will allow us to avoid extra Math.min/Math.max calls.
 | ||||||
|  | var PostScriptCompiler = (function PostScriptCompilerClosure() { | ||||||
|  |   function AstNode(type) { | ||||||
|  |     this.type = type; | ||||||
|  |   } | ||||||
|  |   AstNode.prototype.visit = function (visitor) { | ||||||
|  |     throw new Error('abstract method'); | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  |   function AstArgument(index, min, max) { | ||||||
|  |     AstNode.call(this, 'args'); | ||||||
|  |     this.index = index; | ||||||
|  |     this.min = min; | ||||||
|  |     this.max = max; | ||||||
|  |   } | ||||||
|  |   AstArgument.prototype = Object.create(AstNode.prototype); | ||||||
|  |   AstArgument.prototype.visit = function (visitor) { | ||||||
|  |     visitor.visitArgument(this); | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  |   function AstLiteral(number) { | ||||||
|  |     AstNode.call(this, 'literal'); | ||||||
|  |     this.number = number; | ||||||
|  |     this.min = number; | ||||||
|  |     this.max = number; | ||||||
|  |   } | ||||||
|  |   AstLiteral.prototype = Object.create(AstNode.prototype); | ||||||
|  |   AstLiteral.prototype.visit = function (visitor) { | ||||||
|  |     visitor.visitLiteral(this); | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  |   function AstBinaryOperation(op, arg1, arg2, min, max) { | ||||||
|  |     AstNode.call(this, 'binary'); | ||||||
|  |     this.op = op; | ||||||
|  |     this.arg1 = arg1; | ||||||
|  |     this.arg2 = arg2; | ||||||
|  |     this.min = min; | ||||||
|  |     this.max = max; | ||||||
|  |   } | ||||||
|  |   AstBinaryOperation.prototype = Object.create(AstNode.prototype); | ||||||
|  |   AstBinaryOperation.prototype.visit = function (visitor) { | ||||||
|  |     visitor.visitBinaryOperation(this); | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  |   function AstMin(arg, max) { | ||||||
|  |     AstNode.call(this, 'max'); | ||||||
|  |     this.arg = arg; | ||||||
|  |     this.min = arg.min; | ||||||
|  |     this.max = max; | ||||||
|  |   } | ||||||
|  |   AstMin.prototype = Object.create(AstNode.prototype); | ||||||
|  |   AstMin.prototype.visit = function (visitor) { | ||||||
|  |     visitor.visitMin(this); | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  |   function AstVariable(index, min, max) { | ||||||
|  |     AstNode.call(this, 'var'); | ||||||
|  |     this.index = index; | ||||||
|  |     this.min = min; | ||||||
|  |     this.max = max; | ||||||
|  |   } | ||||||
|  |   AstVariable.prototype = Object.create(AstNode.prototype); | ||||||
|  |   AstVariable.prototype.visit = function (visitor) { | ||||||
|  |     visitor.visitVariable(this); | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  |   function AstVariableDefinition(variable, arg) { | ||||||
|  |     AstNode.call(this, 'definition'); | ||||||
|  |     this.variable = variable; | ||||||
|  |     this.arg = arg; | ||||||
|  |   } | ||||||
|  |   AstVariableDefinition.prototype = Object.create(AstNode.prototype); | ||||||
|  |   AstVariableDefinition.prototype.visit = function (visitor) { | ||||||
|  |     visitor.visitVariableDefinition(this); | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  |   function ExpressionBuilderVisitor() { | ||||||
|  |     this.parts = []; | ||||||
|  |   } | ||||||
|  |   ExpressionBuilderVisitor.prototype = { | ||||||
|  |     visitArgument: function (arg) { | ||||||
|  |       this.parts.push('Math.max(', arg.min, ', Math.min(', | ||||||
|  |                       arg.max, ', args[', arg.index, ']))'); | ||||||
|  |     }, | ||||||
|  |     visitVariable: function (variable) { | ||||||
|  |       this.parts.push('v', variable.index); | ||||||
|  |     }, | ||||||
|  |     visitLiteral: function (literal) { | ||||||
|  |       this.parts.push(literal.number); | ||||||
|  |     }, | ||||||
|  |     visitBinaryOperation: function (operation) { | ||||||
|  |       this.parts.push('('); | ||||||
|  |       operation.arg1.visit(this); | ||||||
|  |       this.parts.push(' ', operation.op, ' '); | ||||||
|  |       operation.arg2.visit(this); | ||||||
|  |       this.parts.push(')'); | ||||||
|  |     }, | ||||||
|  |     visitVariableDefinition: function (definition) { | ||||||
|  |       this.parts.push('var '); | ||||||
|  |       definition.variable.visit(this); | ||||||
|  |       this.parts.push(' = '); | ||||||
|  |       definition.arg.visit(this); | ||||||
|  |       this.parts.push(';'); | ||||||
|  |     }, | ||||||
|  |     visitMin: function (max) { | ||||||
|  |       this.parts.push('Math.min('); | ||||||
|  |       max.arg.visit(this); | ||||||
|  |       this.parts.push(', ', max.max, ')'); | ||||||
|  |     }, | ||||||
|  |     toString: function () { | ||||||
|  |       return this.parts.join(''); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  |   function buildAddOperation(num1, num2) { | ||||||
|  |     if (num2.type === 'literal' && num2.number === 0) { | ||||||
|  |       // optimization: second operand is 0
 | ||||||
|  |       return num1; | ||||||
|  |     } | ||||||
|  |     if (num1.type === 'literal' && num1.number === 0) { | ||||||
|  |       // optimization: first operand is 0
 | ||||||
|  |       return num2; | ||||||
|  |     } | ||||||
|  |     if (num2.type === 'literal' && num1.type === 'literal') { | ||||||
|  |       // optimization: operands operand are literals
 | ||||||
|  |       return new AstLiteral(num1.number + num2.number); | ||||||
|  |     } | ||||||
|  |     return new AstBinaryOperation('+', num1, num2, | ||||||
|  |                                   num1.min + num2.min, num1.max + num2.max); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   function buildMulOperation(num1, num2) { | ||||||
|  |     if (num2.type === 'literal') { | ||||||
|  |       // optimization: second operands is a literal...
 | ||||||
|  |       if (num2.number === 0) { | ||||||
|  |         return new AstLiteral(0); // and it's 0
 | ||||||
|  |       } else if (num2.number === 1) { | ||||||
|  |         return num1; // and it's 1
 | ||||||
|  |       } else if (num1.type === 'literal') { | ||||||
|  |         // ... and first operands is a literal too
 | ||||||
|  |         return new AstLiteral(num1.number * num2.number); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     if (num1.type === 'literal') { | ||||||
|  |       // optimization: first operands is a literal...
 | ||||||
|  |       if (num1.number === 0) { | ||||||
|  |         return new AstLiteral(0); // and it's 0
 | ||||||
|  |       } else if (num1.number === 1) { | ||||||
|  |         return num2; // and it's 1
 | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     var min = Math.min(num1.min * num2.min, num1.min * num2.max, | ||||||
|  |                        num1.max * num2.min, num1.max * num2.max); | ||||||
|  |     var max = Math.max(num1.min * num2.min, num1.min * num2.max, | ||||||
|  |                        num1.max * num2.min, num1.max * num2.max); | ||||||
|  |     return new AstBinaryOperation('*', num1, num2, min, max); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   function buildSubOperation(num1, num2) { | ||||||
|  |     if (num2.type === 'literal') { | ||||||
|  |       // optimization: second operands is a literal...
 | ||||||
|  |       if (num2.number === 0) { | ||||||
|  |         return num1; // ... and it's 0
 | ||||||
|  |       } else if (num1.type === 'literal') { | ||||||
|  |         // ... and first operands is a literal too
 | ||||||
|  |         return new AstLiteral(num1.number - num2.number); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     if (num2.type === 'binary' && num2.op === '-' && | ||||||
|  |       num1.type === 'literal' && num1.number === 1 && | ||||||
|  |       num2.arg1.type === 'literal' && num2.arg1.number === 1) { | ||||||
|  |       // optimization for case: 1 - (1 - x)
 | ||||||
|  |       return num2.arg2; | ||||||
|  |     } | ||||||
|  |     return new AstBinaryOperation('-', num1, num2, | ||||||
|  |                                   num1.min - num2.max, num1.max - num2.min); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   function buildMinOperation(num1, max) { | ||||||
|  |     if (num1.min >= max) { | ||||||
|  |       // optimization: num1 min value is not less than required max
 | ||||||
|  |       return new AstLiteral(max); // just returning max
 | ||||||
|  |     } else if (num1.max <= max) { | ||||||
|  |       // optimization: num1 max value is not greater than required max
 | ||||||
|  |       return num1; // just returning an argument
 | ||||||
|  |     } | ||||||
|  |     return new AstMin(num1, max); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   function PostScriptCompiler() {} | ||||||
|  |   PostScriptCompiler.prototype = { | ||||||
|  |     compile: function PostScriptCompiler_compile(code, domain, range) { | ||||||
|  |       var stack = []; | ||||||
|  |       var i, ii; | ||||||
|  |       var instructions = []; | ||||||
|  |       var inputSize = domain.length >> 1, outputSize = range.length >> 1; | ||||||
|  |       var lastRegister = 0; | ||||||
|  |       var n, j, min, max; | ||||||
|  |       var num1, num2, ast1, ast2, tmpVar, item; | ||||||
|  |       for (i = 0; i < inputSize; i++) { | ||||||
|  |         stack.push(new AstArgument(i, domain[i * 2], domain[i * 2 + 1])); | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       for (i = 0, ii = code.length; i < ii; i++) { | ||||||
|  |         item = code[i]; | ||||||
|  |         if (typeof item === 'number') { | ||||||
|  |           stack.push(new AstLiteral(item)); | ||||||
|  |           continue; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         switch (item) { | ||||||
|  |           case 'add': | ||||||
|  |             if (stack.length < 2) { | ||||||
|  |               return null; | ||||||
|  |             } | ||||||
|  |             num2 = stack.pop(); | ||||||
|  |             num1 = stack.pop(); | ||||||
|  |             stack.push(buildAddOperation(num1, num2)); | ||||||
|  |             break; | ||||||
|  |           case 'cvr': | ||||||
|  |             if (stack.length < 1) { | ||||||
|  |               return null; | ||||||
|  |             } | ||||||
|  |             break; | ||||||
|  |           case 'mul': | ||||||
|  |             if (stack.length < 2) { | ||||||
|  |               return null; | ||||||
|  |             } | ||||||
|  |             num2 = stack.pop(); | ||||||
|  |             num1 = stack.pop(); | ||||||
|  |             stack.push(buildMulOperation(num1, num2)); | ||||||
|  |             break; | ||||||
|  |           case 'sub': | ||||||
|  |             if (stack.length < 2) { | ||||||
|  |               return null; | ||||||
|  |             } | ||||||
|  |             num2 = stack.pop(); | ||||||
|  |             num1 = stack.pop(); | ||||||
|  |             stack.push(buildSubOperation(num1, num2)); | ||||||
|  |             break; | ||||||
|  |           case 'exch': | ||||||
|  |             if (stack.length < 2) { | ||||||
|  |               return null; | ||||||
|  |             } | ||||||
|  |             ast1 = stack.pop(); ast2 = stack.pop(); | ||||||
|  |             stack.push(ast1, ast2); | ||||||
|  |             break; | ||||||
|  |           case 'pop': | ||||||
|  |             if (stack.length < 1) { | ||||||
|  |               return null; | ||||||
|  |             } | ||||||
|  |             stack.pop(); | ||||||
|  |             break; | ||||||
|  |           case 'index': | ||||||
|  |             if (stack.length < 1) { | ||||||
|  |               return null; | ||||||
|  |             } | ||||||
|  |             num1 = stack.pop(); | ||||||
|  |             if (num1.type !== 'literal') { | ||||||
|  |               return null; | ||||||
|  |             } | ||||||
|  |             n = num1.number; | ||||||
|  |             if (n < 0 || (n|0) !== n || stack.length < n) { | ||||||
|  |               return null; | ||||||
|  |             } | ||||||
|  |             ast1 = stack[stack.length - n - 1]; | ||||||
|  |             if (ast1.type === 'literal' || ast1.type === 'var') { | ||||||
|  |               stack.push(ast1); | ||||||
|  |               break; | ||||||
|  |             } | ||||||
|  |             tmpVar = new AstVariable(lastRegister++, ast1.min, ast1.max); | ||||||
|  |             stack[stack.length - n - 1] = tmpVar; | ||||||
|  |             stack.push(tmpVar); | ||||||
|  |             instructions.push(new AstVariableDefinition(tmpVar, ast1)); | ||||||
|  |             break; | ||||||
|  |           case 'dup': | ||||||
|  |             if (stack.length < 1) { | ||||||
|  |               return null; | ||||||
|  |             } | ||||||
|  |             if (typeof code[i + 1] === 'number' && code[i + 2] === 'gt' && | ||||||
|  |                 code[i + 3] === i + 7 && code[i + 4] === 'jz' && | ||||||
|  |                 code[i + 5] === 'pop' && code[i + 6] === code[i + 1]) { | ||||||
|  |               // special case of the commands sequence for the min operation
 | ||||||
|  |               num1 = stack.pop(); | ||||||
|  |               stack.push(buildMinOperation(num1, code[i + 1])); | ||||||
|  |               i += 6; | ||||||
|  |               break; | ||||||
|  |             } | ||||||
|  |             ast1 = stack[stack.length - 1]; | ||||||
|  |             if (ast1.type === 'literal' || ast1.type === 'var') { | ||||||
|  |               // we don't have to save into intermediate variable a literal or
 | ||||||
|  |               // variable.
 | ||||||
|  |               stack.push(ast1); | ||||||
|  |               break; | ||||||
|  |             } | ||||||
|  |             tmpVar = new AstVariable(lastRegister++, ast1.min, ast1.max); | ||||||
|  |             stack[stack.length - 1] = tmpVar; | ||||||
|  |             stack.push(tmpVar); | ||||||
|  |             instructions.push(new AstVariableDefinition(tmpVar, ast1)); | ||||||
|  |             break; | ||||||
|  |           case 'roll': | ||||||
|  |             if (stack.length < 2) { | ||||||
|  |               return null; | ||||||
|  |             } | ||||||
|  |             num2 = stack.pop(); | ||||||
|  |             num1 = stack.pop(); | ||||||
|  |             if (num2.type !== 'literal' || num1.type !== 'literal') { | ||||||
|  |               // both roll operands must be numbers
 | ||||||
|  |               return null; | ||||||
|  |             } | ||||||
|  |             j = num2.number; | ||||||
|  |             n = num1.number; | ||||||
|  |             if (n <= 0 || (n|0) !== n || (j|0) !== j || stack.length < n) { | ||||||
|  |               // ... and integers
 | ||||||
|  |               return null; | ||||||
|  |             } | ||||||
|  |             j = ((j % n) + n) % n; | ||||||
|  |             if (j === 0) { | ||||||
|  |               break; // just skipping -- there are nothing to rotate
 | ||||||
|  |             } | ||||||
|  |             Array.prototype.push.apply(stack, | ||||||
|  |                                        stack.splice(stack.length - n, n - j)); | ||||||
|  |             break; | ||||||
|  |           default: | ||||||
|  |             return null; // unsupported operator
 | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       if (stack.length !== outputSize) { | ||||||
|  |         return null; | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       var result = []; | ||||||
|  |       instructions.forEach(function (instruction) { | ||||||
|  |         var statementBuilder = new ExpressionBuilderVisitor(); | ||||||
|  |         instruction.visit(statementBuilder); | ||||||
|  |         result.push(statementBuilder.toString()); | ||||||
|  |       }); | ||||||
|  |       result.push('return [\n  ' + stack.map(function (expr, i) { | ||||||
|  |         var statementBuilder = new ExpressionBuilderVisitor(); | ||||||
|  |         expr.visit(statementBuilder); | ||||||
|  |         var min = range[i * 2], max = range[i * 2 + 1]; | ||||||
|  |         var out = [statementBuilder.toString()]; | ||||||
|  |         if (min > expr.min) { | ||||||
|  |           out.unshift('Math.max(', min, ', '); | ||||||
|  |           out.push(')'); | ||||||
|  |         } | ||||||
|  |         if (max < expr.max) { | ||||||
|  |           out.unshift('Math.min(', max, ', '); | ||||||
|  |           out.push(')'); | ||||||
|  |         } | ||||||
|  |         return out.join(''); | ||||||
|  |       }).join(',\n  ') + '\n];'); | ||||||
|  |       return result.join('\n'); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  |   return PostScriptCompiler; | ||||||
|  | })(); | ||||||
|  | |||||||
| @ -1,7 +1,8 @@ | |||||||
| /* -*- Mode: Java; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ | /* -*- Mode: Java; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ | ||||||
| /* vim: set shiftwidth=2 tabstop=2 autoindent cindent expandtab: */ | /* vim: set shiftwidth=2 tabstop=2 autoindent cindent expandtab: */ | ||||||
| /* globals expect, it, describe, beforeEach, isArray, StringStream, | /* globals expect, it, describe, beforeEach, isArray, StringStream, | ||||||
|            PostScriptParser, PostScriptLexer, PostScriptEvaluator */ |            PostScriptParser, PostScriptLexer, PostScriptEvaluator, | ||||||
|  |            PostScriptCompiler*/ | ||||||
| 
 | 
 | ||||||
| 'use strict'; | 'use strict'; | ||||||
| 
 | 
 | ||||||
| @ -413,5 +414,108 @@ describe('function', function() { | |||||||
|       expect(stack).toMatchArray(expectedStack); |       expect(stack).toMatchArray(expectedStack); | ||||||
|     }); |     }); | ||||||
|   }); |   }); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |   describe('PostScriptCompiler', function() { | ||||||
|  |     function check(code, domain, range, samples) { | ||||||
|  |       var compiler = new PostScriptCompiler(); | ||||||
|  |       var compiledCode = compiler.compile(code, domain, range); | ||||||
|  |       if (samples === null) { | ||||||
|  |         expect(compiledCode).toBeNull(); | ||||||
|  |       } else { | ||||||
|  |         expect(compiledCode).not.toBeNull(); | ||||||
|  |         /*jshint -W054 */ | ||||||
|  |         var fn = new Function('args', compiledCode); | ||||||
|  |         for (var i = 0; i < samples.length; i++) { | ||||||
|  |           var out = fn(samples[i].input); | ||||||
|  |           expect(out).toMatchArray(samples[i].output); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     it('check compiled add', function() { | ||||||
|  |       check([0.25, 0.5, 'add'], [], [0, 1], [{input: [], output: [0.75]}]); | ||||||
|  |       check([0, 'add'], [0, 1], [0, 1], [{input: [0.25], output: [0.25]}]); | ||||||
|  |       check([0.5, 'add'], [0, 1], [0, 1], [{input: [0.25], output: [0.75]}]); | ||||||
|  |       check([0, 'exch', 'add'], [0, 1], [0, 1], | ||||||
|  |             [{input: [0.25], output: [0.25]}]); | ||||||
|  |       check([0.5, 'exch', 'add'], [0, 1], [0, 1], | ||||||
|  |             [{input: [0.25], output: [0.75]}]); | ||||||
|  |       check(['add'], [0, 1, 0, 1], [0, 1], | ||||||
|  |             [{input: [0.25, 0.5], output: [0.75]}]); | ||||||
|  |       check(['add'], [0, 1], [0, 1], null); | ||||||
|  |     }); | ||||||
|  |     it('check compiled sub', function() { | ||||||
|  |       check([0.5, 0.25, 'sub'], [], [0, 1], [{input: [], output: [0.25]}]); | ||||||
|  |       check([0, 'sub'], [0, 1], [0, 1], [{input: [0.25], output: [0.25]}]); | ||||||
|  |       check([0.5, 'sub'], [0, 1], [0, 1], [{input: [0.75], output: [0.25]}]); | ||||||
|  |       check([0, 'exch', 'sub'], [0, 1], [-1, 1], | ||||||
|  |             [{input: [0.25], output: [-0.25]}]); | ||||||
|  |       check([0.75, 'exch', 'sub'], [0, 1], [-1, 1], | ||||||
|  |             [{input: [0.25], output: [0.5]}]); | ||||||
|  |       check(['sub'], [0, 1, 0, 1], [-1, 1], | ||||||
|  |             [{input: [0.25, 0.5], output: [-0.25]}]); | ||||||
|  |       check(['sub'], [0, 1], [0, 1], null); | ||||||
|  | 
 | ||||||
|  |       check([1, 'dup', 3, 2, 'roll', 'sub', 'sub'], [0, 1], [0, 1], | ||||||
|  |             [{input: [0.75], output: [0.75]}]); | ||||||
|  |     }); | ||||||
|  |     it('check compiled mul', function() { | ||||||
|  |       check([0.25, 0.5, 'mul'], [], [0, 1], [{input: [], output: [0.125]}]); | ||||||
|  |       check([0, 'mul'], [0, 1], [0, 1], [{input: [0.25], output: [0]}]); | ||||||
|  |       check([0.5, 'mul'], [0, 1], [0, 1], [{input: [0.25], output: [0.125]}]); | ||||||
|  |       check([1, 'mul'], [0, 1], [0, 1], [{input: [0.25], output: [0.25]}]); | ||||||
|  |       check([0, 'exch', 'mul'], [0, 1], [0, 1], [{input: [0.25], output: [0]}]); | ||||||
|  |       check([0.5, 'exch', 'mul'], [0, 1], [0, 1], | ||||||
|  |             [{input: [0.25], output: [0.125]}]); | ||||||
|  |       check([1, 'exch', 'mul'], [0, 1], [0, 1], | ||||||
|  |             [{input: [0.25], output: [0.25]}]); | ||||||
|  |       check(['mul'], [0, 1, 0, 1], [0, 1], | ||||||
|  |             [{input: [0.25, 0.5], output: [0.125]}]); | ||||||
|  |       check(['mul'], [0, 1], [0, 1], null); | ||||||
|  |     }); | ||||||
|  |     it('check compiled max', function() { | ||||||
|  |       check(['dup', 0.6, 'gt', 7, 'jz', 'pop', 0.6], [0, 1], [0, 1], | ||||||
|  |             [{input: [0.5], output: [0.5]}]); | ||||||
|  |       check(['dup', 0.6, 'gt', 7, 'jz', 'pop', 0.6], [0, 1], [0, 1], | ||||||
|  |             [{input: [1], output: [0.6]}]); | ||||||
|  |       check(['dup', 0.6, 'gt', 5, 'jz', 'pop', 0.6], [0, 1], [0, 1], null); | ||||||
|  |     }); | ||||||
|  |     it('check pop/roll/index', function() { | ||||||
|  |       check([1, 'pop'], [0, 1], [0, 1], [{input: [0.5], output: [0.5]}]); | ||||||
|  |       check([1, 3, -1, 'roll'], [0, 1, 0, 1], [0, 1, 0, 1, 0, 1], | ||||||
|  |             [{input: [0.25, 0.5], output: [0.5, 1, 0.25]}]); | ||||||
|  |       check([1, 3, 1, 'roll'], [0, 1, 0, 1], [0, 1, 0, 1, 0, 1], | ||||||
|  |             [{input: [0.25, 0.5], output: [1, 0.25, 0.5]}]); | ||||||
|  |       check([1, 3, 1.5, 'roll'], [0, 1, 0, 1], [0, 1, 0, 1, 0, 1], null); | ||||||
|  |       check([1, 1, 'index'], [0, 1], [0, 1, 0, 1, 0, 1], | ||||||
|  |             [{input: [0.5], output: [0.5, 1, 0.5]}]); | ||||||
|  |       check([1, 3, 'index', 'pop'], [0, 1], [0, 1], null); | ||||||
|  |       check([1, 0.5, 'index', 'pop'], [0, 1], [0, 1], null); | ||||||
|  |     }); | ||||||
|  |     it('check input boundaries', function () { | ||||||
|  |       check([], [0, 0.5], [0, 1], [{input: [1], output: [0.5]}]); | ||||||
|  |       check([], [0.5, 1], [0, 1], [{input: [0], output: [0.5]}]); | ||||||
|  |       check(['dup'], [0.5, 0.6], [0, 1, 0, 1], | ||||||
|  |             [{input: [0], output: [0.5, 0.5]}]); | ||||||
|  |       check([], [100, 1001], [0, 10000], [{input: [1000], output: [1000]}]); | ||||||
|  |     }); | ||||||
|  |     it('check output boundaries', function () { | ||||||
|  |       check([], [0, 1], [0, 0.5], [{input: [1], output: [0.5]}]); | ||||||
|  |       check([], [0, 1], [0.5, 1], [{input: [0], output: [0.5]}]); | ||||||
|  |       check(['dup'], [0, 1], [0.5, 1, 0.6, 1], | ||||||
|  |             [{input: [0], output: [0.5, 0.6]}]); | ||||||
|  |       check([], [0, 10000], [100, 1001], [{input: [1000], output: [1000]}]); | ||||||
|  |     }); | ||||||
|  |     it('compile optimized', function () { | ||||||
|  |       var compiler = new PostScriptCompiler(); | ||||||
|  |       var code = [0, 'add', 1, 1, 3, -1, 'roll', 'sub', 'sub', 1, 'mul']; | ||||||
|  |       var compiledCode = compiler.compile(code, [0, 1], [0, 1]); | ||||||
|  |       expect(compiledCode).toEqual('return [\n' + | ||||||
|  |                                    '  Math.max(0, Math.min(1, args[0]))\n' + | ||||||
|  |                                    '];'); | ||||||
|  | 
 | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
| }); | }); | ||||||
| 
 | 
 | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user