Find this useful? Enter your email to receive occasional updates for securing PHP code.

Signing you up...

Thank you for signing up!

PHP Decode

<?php class Handler { protected static $typeDecls = array(); protected static $multithr..

Decoded Output download

<?php

class Handler {
	protected static $typeDecls = array();
	protected static $multithreadingThresholds;

	protected $baseName;
	protected $operandType;
	protected $operandSize = 1;
	protected $addressMode;
	protected $multipleData = false;
	protected $unrolling = false;
	
	static public function getTypeDeclarations() {
		ksort(self::$typeDecls);
		return self::$typeDecls;
	}
		
	static public function compare($op1, $op2) {
		if($op1 && $op2) {
			if($op1 == $op2) {
				return true;
			} else {
				if($op1->baseName == $op2->baseName && $op1->operandType == $op2->operandType && $op1->operandSize == $op2->operandSize && $op1->operandSize == $op2->operandSize) {
					return "variant";
				}
			}
		}
		return false;
	}
	
	public function getBaseName() {
		return $this->baseName;
	}

	public function getName() {
		$name = $this->baseName;
		
		// append the operand size unless it's always the same (as in the case complex number)
		if(!in_array('FixedOperandSize', class_uses($this))) {
			if($this->operandSize > 1) {
				$name .= "_{$this->operandSize}X";
			}
		}
		
		// append operand type to the name
		$opCount = $this->getOperandCount();
		for($i = 1; $i <= $opCount; $i++) {
			$type = $this->getOperandType($i);
			$name .= "_$type";
		}

		if($this->multipleData) {
			// differentiate the op from the unit-data version
			if($this->handlesUnitData()) {
				$name .= "_MIO";
			}
		} else {
			if(in_array('MultipleAddressMode', class_uses($this))) {
				// append the address mode to distinct the different ops
				// unless we're creating a 
				if(!($this->addressMode == "ARR" && $this->needsReplication())) {
					$name .= "_$this->addressMode";
				}
			}
		}
		return $name;
	}
	
	public function getAddressMode() {
		return $this->addressMode;
	}

	// return code for running the instruction
	public function getCode() {
		$lines = array();
		$instr = $this->getInstructionStructure();
		$targetCount = $this->getJumpTargetCount();

		$lines[] = 			$this->getMacroDefinitions();

		if(!$this->alwaysReturns()) {
			if($targetCount == 0 || $targetCount == 1) {
				// set next handler
				$lines[] = "handler = INSTR->next_handler;";
			} else if($targetCount == 2) {
				// assume the first branch will be taken
				$lines[] = "int condition;";
				$lines[] = "handler = INSTR->next_handler1;";
			} else if($targetCount > 2) {
				// branch table
				$lines[] = "unsigned int offset;";
			}
		}
			
		$lines[] = 		$this->getAction();

		if(!$this->alwaysReturns()) {
			if($targetCount == 0) {
				// move the instruction pointer over this one
				$lines[] = "ip += sizeof($instr);";
			} else if($targetCount == 1) {
				// go to the jump target
				$lines[] = "ip = INSTR->instruction_pointer;";
			} else if($targetCount == 2) {
				// set the instruction pointer to pointer 1, if condition is true
				// otherwise update the next handler and set ip to pointer 2
				$lines[] = "if(condition) {";
				$lines[] = 		"ip = INSTR->instruction_pointer1;";
				$lines[] = "} else {";
				$lines[] = 		"handler = INSTR->next_handler2;";
				$lines[] = 		"ip = INSTR->instruction_pointer2;";
				$lines[] = "}";
			}  else if($targetCount > 2) {
				$lines[] = "handler = INSTR->branch_table[offset].next_handler;";
				$lines[] = "ip = INSTR->branch_table[offset].instruction_pointer;";
			}
			if($targetCount == 1 || $targetCount == 2) {
				$lines[] = "TIMER_CHECK();";
			}
		}

		$lines[] = 			$this->getMacroUndefinitions();
		return $lines;
	}

	// return code for helper functions needed by the handler
	public function getHelperFunctions() {
	}
	
	// return the instruction structure for the op
	public function getInstructionStructure() {
		$opCount = $this->getOperandCount();
		$targetCount = $this->getJumpTargetCount();
		$instr = "qb_instruction";

		if($targetCount == -1 && $opCount == 0) {
			// negative one means the function exits at this point
			// the structure is empty, as there's no operands and no next handler
			return NULL;
		}
		if($targetCount == 2) {
			$instr .= "_branch";
		} else if($targetCount == 1) {
			$instr .= "_jump";
		} else if($targetCount == -1) {
			$instr .= "_exit";
		} else if($targetCount > 2) {
			$instr .= "_branch_table_$targetCount";
		}
		
		for($i = 1; $i <= $opCount; $i++) {
			$addressMode = $this->getOperandAddressMode($i);
			if(!$addressMode) {
				$class = get_class($this);
				die("Operand $i of $class has null address mode
");
			}
			if($addressMode == "CON") {
				$addressMode = $this->getOperandType($i);
			}
			$instr .= "_{$addressMode}";
		}
		
		if($this->needsLineIdentifier()) {
			$instr .= "_line_id";
		}
		return $instr;
	}
	
	// return the instruction structure definition
	public function getInstructionStructureDefinition() {
		$instr = $this->getInstructionStructure();
		if(!$instr) {
			return NULL;
		}
		$targetCount = $this->getJumpTargetCount();
		$opCount = $this->getOperandCount();
		$lines = array();
		$lines[] = "typedef struct $instr {";
		if($targetCount >= 0 && $targetCount <= 2) {
			if($targetCount == 2) {
				$lines[] = "qb_op_handler next_handler1;";
				$lines[] = "int8_t *instruction_pointer1;";
				$lines[] = "qb_op_handler next_handler2;";
				$lines[] = "int8_t *instruction_pointer2;";
			} else if($targetCount == 1) {
				$lines[] = "qb_op_handler next_handler;";
				$lines[] = "int8_t *instruction_pointer;";
			} else {
				$lines[] = "qb_op_handler next_handler;";
			}
		}
		
		for($i = 1; $i <= $opCount; $i++) {
			$addressMode = $this->getOperandAddressMode($i);
			if($addressMode == "CON") {
				$cType = $this->getOperandCType($i);
				$lines[] = "$cType operand{$i};";
			} else {
				$lines[] = "qb_pointer_{$addressMode} operand{$i};";
			}
		}
		
		if($targetCount > 2) {
			$lines[] = "qb_branch_table_entry branch_table[$targetCount];";
		}
		
		if($this->needsLineIdentifier()) {
			$lines[] = "uint32_t line_id;";
		}
		$lines[] = "} $instr;";
		return $lines;
	}
	
	public function getInstructionFormat() {
		$opCount = $this->getOperandCount();
		$format = "";
		
		for($i = 1; $i <= $opCount; $i++) {
			$addressMode = $this->getOperandAddressMode($i);
			if($this->changesOperand($i)) {
				switch($addressMode) {
					case 'SCA': $format .= 'S'; break;
					case 'ELE': $format .= 'E'; break;
					case 'ARR':
						$type = $this->getOperandCType($i);
						if($type[0] == 'c') {
							$format .= 'X'; 
						} else {
							$format .= 'A'; 
						}
						break;
					case 'CON':
						$class = get_class($this);
						die("Operand $i of $class is constant and changeable at the same time
");
				}
			} else {
				switch($addressMode) {
					case 'SCA': $format .= 's'; break;
					case 'ELE': $format .= 'e'; break;
					case 'ARR': 
						$type = $this->getOperandCType($i);
						if($type[0] == 'c') {
							$format .= 'x'; 
						} else {
							$format .= 'a';
						}
						break;
					case 'CON': $format .= 'c'; break;
				}
			}
		}
		return $format;
	}
	
	// return the number of input operands
	public function getInputOperandCount() {
		return 0;
	}
	
	// return the number of output operands 
	public function getOutputOperandCount() {
		return 0;
	}
	
	// return the number of jump targets
	public function getJumpTargetCount() {
		return 0;
	}

	// return the total number of operands
	public function getOperandCount() {
		return $this->getInputOperandCount() + $this->getOutputOperandCount(); 
	}
	
	// return the type of operand $i (starting from 1)
	// by default, all operands have the same type
	public function getOperandType($i) {
		return $this->operandType;
	}
	
	// return the C-type of operand $i
	public function getOperandCType($i) {
		static $cTypes = array(
			"I08" => "int8_t",		"I16" => "int16_t",			"I32" => "int32_t",			"I64" => "int64_t",
			"S08" => "int8_t",		"S16" => "int16_t",			"S32" => "int32_t",			"S64" => "int64_t",
			"U08" => "uint8_t",		"U16" => "uint16_t",		"U32" => "uint32_t",		"U64" => "uint64_t",
			"F32" => "float32_t",	"F64" => "float64_t",
		);
		$operandType = $this->getOperandType($i);
		if(!$operandType) {
			$class = get_class($this);
			die("Operand $i of $class has null type
");
		}
		return $cTypes[$operandType];
	}

	public function getOperandSizeShift($i) {
		return 0;
	}
	
	// return the number of elements that consist an operand 
	public function getOperandSize($i) {
		return $this->operandSize;
	}
	
	// return the address mode of operand $i
	// by default, all operands use the same address mode
	public function getOperandAddressMode($i) {
		return $this->addressMode;
	}
	
	public function isMultipleData() {
		return $this->multipleData;
	}
	
	public function handlesUnitData() {
		return true;
	}
	
	public function disableMultipleData() {
		$this->multipleData = false;
	}

	public function restoreMultipleData() {
		$this->multipleData = true;
	}
	
	public function isMultithreaded() {
		if(in_array('Multithreaded', class_uses($this))) {
			if($this->mayExitLoop()) {
				return false;
			} else {
				$threshold = $this->getMultithreadingThreshold();
				if($threshold === null) {
					if($this->operandSize != 'variable') {
						$class = get_class($this);
						echo "Missing threshold for $class ($this->operandType, $this->operandSize)
";
					}
				}
				return ($threshold != 0);
			}
		} else {
			return false;
		}
	}	
	
	public function needsInterpreterContext() {
		return false;
	}
	
	public function needsReplication() {
		return false;
	}
	
	public function needsLineIdentifier() {
		return false;
	}

	public function alwaysReturns() {
		return false;
	}

	public function needsCondition() {
		$targetCount = $this->getJumpTargetCount();
		return ($targetCount >= 2);
	}

	public function changesOperand($i) {
		$srcCount = $this->getInputOperandCount();
		return ($i > $srcCount);
	}
	
	public function changesOperandSize($i) {
		return false;
	}
	
	public function runsInMainThread() {
		return false;
	}

	public function needsInstructionStructure() {
		if($this->isMultipleData() && $this->isMultithreaded()) {
			return true;
		} else if($this->runsInMainThread()) {
			return true;
		}
		return false;
	}

	public function isThreadSafe() {
		return true;
	}
	
	public function mayExitLoop() {
		return false;
	}
	
	public function getMultithreadingThreshold() {
		if(self::$multithreadingThresholds === null) {
			$folder = dirname(__FILE__);
			$path = "$folder/../threshold/multithreading_thresholds.txt";
			if(file_exists($path)) { 
				self::$multithreadingThresholds = array();
				$lines = file($path, FILE_SKIP_EMPTY_LINES | FILE_IGNORE_NEW_LINES);
				foreach($lines as $line) {
					if(!preg_match('/^\s*;/', $line)) {
						if(preg_match('/(\w+)\s+(\w+)\s+(\d+)\s+(\d+)/', $line, $m)) {
							$name = $m[1];
							$type = $m[2];
							$width = (int) $m[3];
							$threshold = (int) $m[4];

							$byType =& self::$multithreadingThresholds[$name];
							$byWidth =& $byType[$type];
							$byWidth[$width] = $threshold;
						}
					}
				}
			} else {
				// when the file is missing, we're trying to generate it
				self::$multithreadingThresholds == "calc";
			}
		}
		if(is_array(self::$multithreadingThresholds)) {
			$class = get_class($this);
			$type = $this->operandType;
			$width = $this->operandSize;
			if(isset(self::$multithreadingThresholds[$class][$type][$width])) {
				return self::$multithreadingThresholds[$class][$type][$width];
			} else {
				return null;
			}
		} else {
			// use a low value so we can test it
			return 1024;
		}
	}
	
	public function performsWrapAround() {
		if($this->addressMode == "ARR" && !$this->isOverridden('getActionOnMultipleData')) {
			return true;
		}
		return false;
	}
	
	protected function getMacroDefinitions() {
		$instr = $this->getInstructionStructure();
		$srcCount = $this->getInputOperandCount();
		$opCount = $this->getOperandCount();
		$lines = array();
		if($instr) {
			$lines[] = "#define INSTR		(($instr * __restrict) ip)";
		}
		if($this->needsLineIdentifier()) {
			$lines[] = "#define line_id		INSTR->line_id";
		}
		for($i = 1; $i <= $opCount; $i++) {
			$cType = $this->getOperandCType($i);
			$addressMode = $this->getOperandAddressMode($i);
			$operand = "INSTR->operand$i";
			$name = ($i <= $srcCount) ? "op{$i}" : "res";
			switch($addressMode) {
				case 'SCA':
					$lines[] = "#define $name	(($cType *) $operand.data_pointer)[0]";
					break;
				case 'ELE':
					$lines[] = "#define $name	(($cType *) $operand.data_pointer)[$operand.index_pointer[0]]";
					break;
				case 'ARR':
					$lines[] = "#define {$name}_ptr		((($cType *) $operand.data_pointer) + $operand.index_pointer[0])";
					$lines[] = "#define {$name}_count	$operand.count_pointer[0]";
					if($this->changesOperandSize($i)) {
						$lines[] = "#define {$name}_count_ptr	$operand.count_pointer";
					}
					break;
				case 'CON':
					$lines[] = "#define $name	$operand";
					break;
			}
		}
		return $lines;
	}
	
	protected function getMacroUndefinitions() {
		$instr = $this->getInstructionStructure();
		$srcCount = $this->getInputOperandCount();
		$opCount = $this->getOperandCount();
		$lines = array();
		if($instr) {
			$lines[] = "#undef INSTR";
		}
		if($this->needsLineIdentifier()) {
			$lines[] = "#undef line_id";
		}
		for($i = 1; $i <= $opCount; $i++) {
			$addressMode = $this->getOperandAddressMode($i);
			$name = ($i <= $srcCount) ? "op{$i}" : "res";
			switch($addressMode) {
				case 'SCA':
					$lines[] = "#undef $name";
					break;
				case 'ELE':
					$lines[] = "#undef $name";
					break;
				case 'ARR':
					$lines[] = "#undef {$name}_ptr";
					$lines[] = "#undef {$name}_count";
					if($this->changesOperandSize($i)) {
						$lines[] = "#undef {$name}_count_ptr";
					}
					break;
				case 'CON':
					$lines[] = "#undef $name";
					break;
			}
		}
		return $lines;
	}
	
	protected function getFunctionNameComponents($prefix) {
		$className = get_class($this);
		$parts = array();
		$parts[] = "qb";
		$parts[] = $prefix;
		$parts[] = strtolower(preg_replace("/([a-z])([A-Z])/", "$1_$2", $className));
		if($this->operandSize && $this->operandSize != 1 && is_int($this->operandSize)) {
			$parts[] = "{$this->operandSize}x";
		}
		if($this->isMultipleData()) {
			$parts[] = "multiple_times";
		}
		if($this->operandType) {
			$parts[] = $this->operandType;
		}
		return $parts;
	}

	public function getHandlerFunctionName() {
		$parts = $this->getFunctionNameComponents("do");
		return implode('_', $parts);
	}
	
	public function getHandlerFunctionType() {
		if($this->runsInMainThread()) {
			return 'extern';
		} else if(!$this->isOverridden('getAction')) {
			if($this->isMultipleData()) {
				return 'extern';
			} else {
				$action = $this->getActionOnUnitData();
				$count = count($action, true);
				if($count > 8) {
					return 'extern';
				} else if($count == 1) {
					return 'inline';
				}
				
				$lines = array_linearize($action);
				$hasLoop = false;
				foreach($lines as $line) {
					if(preg_match('/(for|while)/', $line)) {
						$hasLoop = true;
					}
				}
				if($hasLoop) {
					return 'extern';
				} else {
					return 'inline';
				}
			}
		} 
		return null;
	}

	public function getHandlerFunctionDefinition() {
		$functionType = $this->getHandlerFunctionType();
		if(!$functionType) {
			return null;
		}
		$function = $this->getHandlerFunctionName();
		$parameterList = $this->getHandlerFunctionParameterList(true);
		$expressions = $this->getActionExpressions();
		$mayExit = $this->mayExitLoop();
		if($mayExit) {
			$typeDecl = "int32_t";
		} else {
			$typeDecl = "void";
		}
		if($functionType == "inline") {
			$typeDecl = "static zend_always_inline $typeDecl";
		}
		
		// replace op# with (*op#_ptr) and res with (*res_ptr) where it's necessary
		$expressions = array_linearize($expressions);
		$srcCount = $this->getInputOperandCount();
		$needPointers = array();
		for($i = 1; $i <= $srcCount; $i++) {
			if($this->getOperandAddressMode($i) == "ARR" || $this->changesOperand($i)) {
				$needPointers[] = "op{$i}";
			}
			if($this->changesOperandSize($i)) {
				$needPointers[] = "op{$i}_count";
			}
		}
		$needPointers[] = "res";
		if($this->changesOperandSize($i)) {
			$needPointers[] = "res_count";
		}
		if($this->needsCondition()) {
			$needPointers[] = "condition";
		}
		$regExp = '/(' . implode('|', $needPointers) . ')/';
		$expressions = preg_replace($regExp, '(*_ptr)', $expressions);
		if($mayExit) {
			// replace "return" with "return FALSE" and add return TRUE at the end
			$expressions = preg_replace('/return/', 'return FALSE', $expressions);
			$expressions[] = "return TRUE;";
		}
		$lines = array();			
		$lines[] = "$typeDecl $function($parameterList) {";
		$lines[] = $expressions;
		$lines[] = "}";
		return $lines;
	}
	
	public function getHandlerFunctionParameterList($forDeclaration) {
		$instr = $this->getInstructionStructure();
		$srcCount = $this->getInputOperandCount();
		$opCount = $this->getOperandCount();
		$params = array();
		if($this->needsInterpreterContext()) {
			if($forDeclaration) {
				$params[] = "qb_interpreter_context *__restrict cxt";
			} else {
				$params[] = "cxt";
			}
		} 
		if($this->needsCondition()) {
			if($forDeclaration) {
				$params[] = "int32_t *condition_ptr";
			} else {
				$params[] = "&condition";
			}
		}
		for($i = 1; $i <= $opCount; $i++) {
			$cType = $this->getOperandCType($i);
			$addressMode = $this->getOperandAddressMode($i);
			$operand = "((($instr *) ip)->operand$i)";
			$name = ($i <= $srcCount) ? "op{$i}" : "res";
			switch($addressMode) {
				case 'SCA':
				case 'ELE':
				case 'CON':
					if($forDeclaration) {
						if($this->changesOperand($i)) {
							$params[] = "$cType *{$name}_ptr";
						} else {
							$params[] = "$cType $name";
						}
					} else {
						if($this->changesOperand($i)) {
							$params[] = "&$name";
						} else {
							$params[] = "$name";
						}
					}
					break;
				case 'ARR':
					if($forDeclaration) {
						$params[] = "$cType *{$name}_ptr";
						if($this->changesOperandSize($i)) {
							$params[] = "uint32_t *{$name}_count_ptr";
						} else {
							// don't need the size if it's a constant number
							if($this->isMultipleData() || !is_numeric($this->getOperandSize($i))) {
								$params[] = "uint32_t {$name}_count";
							}
						}
					} else {
						$params[] = "{$name}_ptr";
						if($this->changesOperandSize($i)) {
							$params[] = "{$name}_count_ptr";
						} else {
							if($this->isMultipleData() || !is_numeric($this->getOperandSize($i))) {
								$shift = $this->getOperandSizeShift($i);
								if($shift) {
									$params[] = "{$name}_count >> $shift";
								} else {
									$params[] = "{$name}_count";
								}
							}
						}
					}
					break;
			}
		}
		if($this->needsLineIdentifier()) {
			if($forDeclaration) {
				$params[] = "uint32_t line_id";
			} else {
				$params[] = "line_id";
			}
		} 
		return implode(", ", $params);
	}

	// return the name of the dispatcher function, which sends a instruction to multiple threads 
	protected function getDispatcherFunctionName() {
		$instr = $this->getInstructionStructure();
		$name = "qb_dispatch_" . substr($instr, 3);
		return $name;
	}
	
	// return the parameter list of the dispatcher function
	protected function getDispatcherFunctionParameterList($forDeclaration) {
		$instr = $this->getInstructionStructure();
		$params = array();
		if($forDeclaration) {
			$params[] = "qb_interpreter_context *__restrict cxt";
			$params[] = "void *control_func";
			$params[] = "$instr *__restrict instr";
		} else {
			$params[] = "cxt";
			$params[] = $this->getControllerFunctionName();
			$params[] = "($instr *) ip";
		}
		
		// add operand sizes
		$opCount = $this->getOperandCount();
		for($i = 1; $i <= $opCount; $i++) {
			$addressMode = $this->getOperandAddressMode($i);
			if($addressMode == "ARR") {
				if($forDeclaration) {
					$params[] = "uint32_t operand{$i}_size";
				} else {
					$operandSize = $this->getOperandSize($i);
					if($operandSize) {
						$params[] = $operandSize;
					} else {
						// put it the size of the array to indicate the pointer doesn't shift
						$params[] = "op{$i}_count";
					}
				}
			}
		}
		// add threshold value
		if($forDeclaration) {
			$params[] = "uint32_t threshold";
		} else {
			$params[] = $this->getMultithreadingThreshold();
		}
		return implode(", ", $params);
	}
	
	// return the body list of the dispatcher function
	public function getDispatcherFunctionDefinition() {
		if($this->isMultipleData() && $this->isMultithreaded()) {
			$addressOperands = array();
			$instr = $this->getInstructionStructure();
			$dispatcherTypeDecl = "int32_t";
			$dispatcherFunction = $this->getDispatcherFunctionName();
			$dispatcherParameterList = $this->getDispatcherFunctionParameterList(true);
			$opCount = $this->getOperandCount();
			$arrayCount = 1;
			$lines = array();
			$lines[] = "$dispatcherTypeDecl $dispatcherFunction($dispatcherParameterList) {";
			$lines[] =		"uint32_t op{$opCount}_count = instr->operand{$opCount}.count_pointer[0];";
			$lines[] =		"if(op{$opCount}_count >= threshold) {";
			$lines[] =			"int32_t use_multithreading = TRUE;";
			$lines[] =			"uint32_t res_unit_count = op{$opCount}_count / operand{$opCount}_size;";
			$lines[] =			"uint32_t thread_count = cxt->thread_count;";
			$lines[] =			"uint32_t chunk_size = res_unit_count / thread_count;";
			for($i = 1; $i < $opCount; $i++) {
				$addressMode = $this->getOperandAddressMode($i);
				if($addressMode == "ARR") {
					$lines[] =	"uint32_t op{$i}_count = instr->operand{$i}.count_pointer[0], op{$i}_unit_count = op{$i}_count / operand{$i}_size, op{$i}_shift, op{$i}_chunk_size;";
					$arrayCount++;
				}	
			}
			$lines[] =			"uint32_t op{$i}_shift = operand{$opCount}_size * chunk_size, op{$i}_chunk_size = operand{$opCount}_size * chunk_size;";			
			for($i = 1; $i < $opCount; $i++) {
				$addressMode = $this->getOperandAddressMode($i);
				if($addressMode == "ARR") {
					$lines[] =	"if(op{$i}_unit_count == res_unit_count) {";
					$lines[] =		"op{$i}_shift = operand{$i}_size * chunk_size;";
					$lines[] =		"op{$i}_chunk_size = operand{$i}_size * chunk_size;";
					$lines[] =	"} else if(op{$i}_unit_count == 1) {";
					$lines[] =		"op{$i}_shift = 0;";
					$lines[] =		"op{$i}_chunk_size = operand{$i}_size;";
					$lines[] =	"} else {";
					$lines[] =		"use_multithreading = FALSE;";
					$lines[] =	"}";
				}	
			}
			$lines[] =			"if(use_multithreading) {";
			$lines[] =				"// create temporary instruction structures";
			$lines[] =				"$instr new_instr_list[MAX_THREAD_COUNT];";
			$lines[] =				"uint32_t new_indices[MAX_THREAD_COUNT][$arrayCount];";
			$lines[] =				"uint32_t new_counts[MAX_THREAD_COUNT][$arrayCount];";
			$lines[] =				"int8_t *new_ips[MAX_THREAD_COUNT];";
			$lines[] =				"uint32_t i;";
			$lines[] =				"for(i = 0; i < thread_count; i++) {";
			$lines[] =					"$instr *new_instr = &new_instr_list[i];";
			for($i = 1, $k = 0; $i <= $opCount; $i++) {
				$addressMode = $this->getOperandAddressMode($i);
				if($addressMode == "ARR") {
					$lines[] =			"new_indices[i][$k] = i * op{$i}_shift;";
					$lines[] =			"new_counts[i][$k] = (i == thread_count - 1) ? op{$i}_count - (i * op{$i}_shift) : op{$i}_chunk_size;";
					$lines[] = 			"new_instr->operand{$i}.data_pointer = instr->operand{$i}.data_pointer;";
					$lines[] = 			"new_instr->operand{$i}.index_pointer = &new_indices[i][$k];";
					$lines[] = 			"new_instr->operand{$i}.count_pointer = &new_counts[i][$k];";
					$k++;
				} else {
					$lines[] = 			"new_instr->operand{$i} = instr->operand{$i};";
				}
			}
			$lines[] =					"new_ips[i] = (int8_t *) new_instr;";
			$lines[] =				"}";
			$lines[] = 				"qb_dispatch_instruction_to_threads(cxt, control_func, new_ips, thread_count);";
			$lines[] =				"return TRUE;";
			$lines[] =			"}";
			$lines[] =		"}";
			$lines[] =		"return FALSE;";
			$lines[] = "}";
			return $lines;
		}
	}
	
	// return the name of the controller function, which decides whether to use multithreading or not
	protected function getControllerFunctionName() {
		$parts = $this->getFunctionNameComponents("redirect");
		if($this->addressMode == "ELE") {
			array_splice($parts, -1, 0, "array_element");
		}
		return implode('_', $parts);
	}
	
	// return the parameter list of the controller function
	protected function getControllerFunctionParameterList($forDeclaration) {
		$params = array();
		if($forDeclaration) {
			$params[] = "qb_interpreter_context *__restrict cxt";
			$params[] = "int8_t *__restrict ip";
			$params[] = "int unused";
		} else {
			$params[] = "cxt";
			$params[] = "ip";
			$params[] = "0";
		}
		return implode(", ", $params);
	}
	
	// return the body of the controller function
	public function getControllerFunctionDefinition() {
		if($this->isMultipleData() && $this->isMultithreaded()) {
			$controllerTypeDecl = "void";
			$controllerFunction = $this->getControllerFunctionName();
			$controllerParameterList = $this->getControllerFunctionParameterList(true);
			$dispatcherFunction = $this->getDispatcherFunctionName();
			$dispatcherParameterList = $this->getDispatcherFunctionParameterList(false);
			$handlerFunction = $this->getHandlerFunctionName();
			$handlerParameterList = $this->getHandlerFunctionParameterList(false);
			$lines = array();
			$lines[] = "$controllerTypeDecl $controllerFunction($controllerParameterList) {";
			$lines[] =		$this->getMacroDefinitions();
			$lines[] =		"if(!cxt->thread_count || !$dispatcherFunction($dispatcherParameterList)) {";
			$lines[] = 			"$handlerFunction($handlerParameterList);";
			$lines[] =		"}";
			$lines[] =		$this->getMacroUndefinitions();
			$lines[] = "}";
			return $lines;
		} else if($this->runsInMainThread()) {
			$controllerTypeDecl = "void";
			$controllerFunction = $this->getControllerFunctionName();
			$controllerParameterList = $this->getControllerFunctionParameterList(true);
			$handlerFunction = $this->getHandlerFunctionName();
			$handlerParameterList = $this->getHandlerFunctionParameterList(false);
			$lines = array();
			$lines[] = "$controllerTypeDecl $controllerFunction($controllerParameterList) {";
			$lines[] =		$this->getMacroDefinitions();
			$lines[] =		"if(!qb_in_main_thread()) {";
			$lines[] = 			"qb_dispatch_instruction_to_main_thread(cxt, $controllerFunction, ip);";
			$lines[] =		"} else {";
			$lines[] = 			"$handlerFunction($handlerParameterList);";
			$lines[] =		"}";
			$lines[] =		$this->getMacroUndefinitions();
			$lines[] = "}";
			return $lines;
		}
	}
	
	// return codes that perform what the op is supposed to do
	public function getAction() {
		$functionType = $this->getHandlerFunctionType();
		if($functionType) {	
			if($this->isMultipleData() && $this->isMultithreaded()) {
				// send instruction to the controller function, which will either
				// (1) call the dispatcher function, which then calls the controller function again from different threads
				// (2) call the handler function
				$function = $this->getControllerFunctionName();
				$parameterList = $this->getControllerFunctionParameterList(false);
			} else if($this->runsInMainThread()) {
				// if the call occurs inside a thread, it needs to be redirected to the main thread
				$function = $this->getControllerFunctionName();
				$parameterList = $this->getControllerFunctionParameterList(false);
			} else {
				// call the handler directly
				$function = $this->getHandlerFunctionName();
				$parameterList = $this->getHandlerFunctionParameterList(false);
			}
			if($this->mayExitLoop()) {
				$lines = array();
				$lines[] = "if(!$function($parameterList)) {";
				$lines[] =		"return;";
				$lines[] = "}";
				return $lines;
			} else {
				return "$function($parameterList);";
			}
		} else {
			// just insert the code, expanding the operands
			$lines = array();
			$lines[] = "{";
			$lines[] = 		$this->getActionExpressions();
			$lines[] = "}";
			return $lines;
		}
	}	
	
	// return an expression for handling a single unit of data (typically a scalar)
	protected function getActionOnUnitData() {
		return null;
	}

	// return an expression for handling multiple units of data
	protected function getActionOnMultipleData() {
		return null;
	}
	
	protected function getActionExpressions() {
		if($this->isMultipleData()) {
			$action = $this->getActionOnMultipleData();
			if(!$action) {
				// temporarily changed the handle to the non-multiple-data
				$this->disableMultipleData();
				$scalarExpression = $this->getAction();
				$this->restoreMultipleData();
				$action = $this->getIterationCode($scalarExpression);
			}
		} else {
			$action = $this->getActionOnUnitData();
			if($this->needsReplication()) {
				$action = $this->replicateExpression($action, $this->operandSize);
			}
		}
		return $action;
	}
		
	// multiple a scalar operation multiple times
	protected function replicateExpression($expression, $count) {
		$srcCount = $this->getInputOperandCount();
		$arrayOperands = array();
		for($i = 1; $i <= $srcCount; $i++) {
			if($this->getOperandAddressMode($i) == "ARR") {
				$arrayOperands[] = $i;
			}
		}
		$nums = implode("|", $arrayOperands);
		$lines = array();
		for($i = 0; $i < $count; $i++) {
			$patterns = array('/res/', '/op(' . $nums . ')/');
			$replacements = array("res_ptr[{$i}]", "op\1_ptr[{$i}]");
			$lines[] = preg_replace($patterns, $replacements, $expression);
		}
		return $lines;
	}
	
	// return code for a loop that performs the same operation on all element of an array 
	protected function getIterationCode($expression, $operandSizeOverride = false) {
		$srcCount = $this->getInputOperandCount();
		$lines = array();		
		
		// make sure none of the input operands are empty		
		$condition = false;
		$operandCounts = array();
		for($i = 1; $i <= $srcCount; $i++) {
			$operandSize = ($operandSizeOverride) ? $operandSizeOverride : $this->getOperandSize($i);
			if($this->getOperandAddressMode($i) == "ARR" && $operandSize !== 0) {
				$operandCounts[] = "op{$i}_count";
			}
		}
		$operandCounts[] = "res_count";
		// use bitwise AND here, just in case the compiler doesn't optimize correctly
		$condition = implode(" && ", $operandCounts);
		$lines[] = "if($condition) {";
		for($i = 1; $i <= $srcCount; $i++) {
			$cType = $this->getOperandCType($i);
			$operandSize = ($operandSizeOverride) ? $operandSizeOverride : $this->getOperandSize($i);
			if($this->getOperandAddressMode($i) == "ARR" && $operandSize !== 0) {
				$lines[] =	"$cType *op{$i}_start = op{$i}_ptr, *op{$i}_end = op{$i}_ptr + op{$i}_count;";
			}
		}
		$cType = $this->getOperandCType($srcCount + 1);
		$lines[] =		"$cType *res_end = res_ptr + res_count;";
		$lines[] = 		"for(;;) {";
		$lines[] = 			$expression;
		$lines[] =			"";
		$operandSize = ($operandSizeOverride) ? $operandSizeOverride : $this->getOperandSize($srcCount + 1);
		// group incrementations together since they can happen in parallel
		$lines[] = 			"res_ptr += $operandSize;";
		for($i = 1; $i <= $srcCount; $i++) {
			$operandSize = ($operandSizeOverride) ? $operandSizeOverride : $this->getOperandSize($i);
			if($this->getOperandAddressMode($i) == "ARR" && $operandSize !== 0) {
				$lines[] =		"op{$i}_ptr += $operandSize;";
			}
		}
		$lines[] =			"if(res_ptr >= res_end) {";
		$lines[] =				"break;";
		$lines[] =			"}";
		for($i = 1; $i <= $srcCount; $i++) {
			$operandSize = $this->getOperandSize($i);
			if($this->getOperandAddressMode($i) == "ARR" && $operandSize !== 0) {
				$lines[] = 	"if(op{$i}_ptr >= op{$i}_end) {";
				$lines[] = 		"op{$i}_ptr = op{$i}_start;";
				$lines[] = 	"}";
			}
		}
		$lines[] = 		"}";	// end for
		$lines[] = "}"; 		// end if
		return $lines;
	}
	
	protected function isOverridden($methodName) {
		$child  = new ReflectionClass($this);
		$method = $child->getMethod($methodName);
		return ($method->class != 'Handler');
	}
}

function array_linearize($array) {
	$result = array();
	if(is_array($array)) {
		foreach($array as $element) {
			if($element !== NULL) {
				if(is_array($element)) {
					$sub_array = array_linearize($element);					
					array_splice($result, count($result), 0, $sub_array);
				} else {
					$result[] = $element;
				}
			}
		}
	} else {
		$result[] = $array;
	}
	return $result;
}

?>

Did this file decode correctly?

Original Code

<?php

class Handler {
	protected static $typeDecls = array();
	protected static $multithreadingThresholds;

	protected $baseName;
	protected $operandType;
	protected $operandSize = 1;
	protected $addressMode;
	protected $multipleData = false;
	protected $unrolling = false;
	
	static public function getTypeDeclarations() {
		ksort(self::$typeDecls);
		return self::$typeDecls;
	}
		
	static public function compare($op1, $op2) {
		if($op1 && $op2) {
			if($op1 == $op2) {
				return true;
			} else {
				if($op1->baseName == $op2->baseName && $op1->operandType == $op2->operandType && $op1->operandSize == $op2->operandSize && $op1->operandSize == $op2->operandSize) {
					return "variant";
				}
			}
		}
		return false;
	}
	
	public function getBaseName() {
		return $this->baseName;
	}

	public function getName() {
		$name = $this->baseName;
		
		// append the operand size unless it's always the same (as in the case complex number)
		if(!in_array('FixedOperandSize', class_uses($this))) {
			if($this->operandSize > 1) {
				$name .= "_{$this->operandSize}X";
			}
		}
		
		// append operand type to the name
		$opCount = $this->getOperandCount();
		for($i = 1; $i <= $opCount; $i++) {
			$type = $this->getOperandType($i);
			$name .= "_$type";
		}

		if($this->multipleData) {
			// differentiate the op from the unit-data version
			if($this->handlesUnitData()) {
				$name .= "_MIO";
			}
		} else {
			if(in_array('MultipleAddressMode', class_uses($this))) {
				// append the address mode to distinct the different ops
				// unless we're creating a 
				if(!($this->addressMode == "ARR" && $this->needsReplication())) {
					$name .= "_$this->addressMode";
				}
			}
		}
		return $name;
	}
	
	public function getAddressMode() {
		return $this->addressMode;
	}

	// return code for running the instruction
	public function getCode() {
		$lines = array();
		$instr = $this->getInstructionStructure();
		$targetCount = $this->getJumpTargetCount();

		$lines[] = 			$this->getMacroDefinitions();

		if(!$this->alwaysReturns()) {
			if($targetCount == 0 || $targetCount == 1) {
				// set next handler
				$lines[] = "handler = INSTR->next_handler;";
			} else if($targetCount == 2) {
				// assume the first branch will be taken
				$lines[] = "int condition;";
				$lines[] = "handler = INSTR->next_handler1;";
			} else if($targetCount > 2) {
				// branch table
				$lines[] = "unsigned int offset;";
			}
		}
			
		$lines[] = 		$this->getAction();

		if(!$this->alwaysReturns()) {
			if($targetCount == 0) {
				// move the instruction pointer over this one
				$lines[] = "ip += sizeof($instr);";
			} else if($targetCount == 1) {
				// go to the jump target
				$lines[] = "ip = INSTR->instruction_pointer;";
			} else if($targetCount == 2) {
				// set the instruction pointer to pointer 1, if condition is true
				// otherwise update the next handler and set ip to pointer 2
				$lines[] = "if(condition) {";
				$lines[] = 		"ip = INSTR->instruction_pointer1;";
				$lines[] = "} else {";
				$lines[] = 		"handler = INSTR->next_handler2;";
				$lines[] = 		"ip = INSTR->instruction_pointer2;";
				$lines[] = "}";
			}  else if($targetCount > 2) {
				$lines[] = "handler = INSTR->branch_table[offset].next_handler;";
				$lines[] = "ip = INSTR->branch_table[offset].instruction_pointer;";
			}
			if($targetCount == 1 || $targetCount == 2) {
				$lines[] = "TIMER_CHECK();";
			}
		}

		$lines[] = 			$this->getMacroUndefinitions();
		return $lines;
	}

	// return code for helper functions needed by the handler
	public function getHelperFunctions() {
	}
	
	// return the instruction structure for the op
	public function getInstructionStructure() {
		$opCount = $this->getOperandCount();
		$targetCount = $this->getJumpTargetCount();
		$instr = "qb_instruction";

		if($targetCount == -1 && $opCount == 0) {
			// negative one means the function exits at this point
			// the structure is empty, as there's no operands and no next handler
			return NULL;
		}
		if($targetCount == 2) {
			$instr .= "_branch";
		} else if($targetCount == 1) {
			$instr .= "_jump";
		} else if($targetCount == -1) {
			$instr .= "_exit";
		} else if($targetCount > 2) {
			$instr .= "_branch_table_$targetCount";
		}
		
		for($i = 1; $i <= $opCount; $i++) {
			$addressMode = $this->getOperandAddressMode($i);
			if(!$addressMode) {
				$class = get_class($this);
				die("Operand $i of $class has null address mode\n");
			}
			if($addressMode == "CON") {
				$addressMode = $this->getOperandType($i);
			}
			$instr .= "_{$addressMode}";
		}
		
		if($this->needsLineIdentifier()) {
			$instr .= "_line_id";
		}
		return $instr;
	}
	
	// return the instruction structure definition
	public function getInstructionStructureDefinition() {
		$instr = $this->getInstructionStructure();
		if(!$instr) {
			return NULL;
		}
		$targetCount = $this->getJumpTargetCount();
		$opCount = $this->getOperandCount();
		$lines = array();
		$lines[] = "typedef struct $instr {";
		if($targetCount >= 0 && $targetCount <= 2) {
			if($targetCount == 2) {
				$lines[] = "qb_op_handler next_handler1;";
				$lines[] = "int8_t *instruction_pointer1;";
				$lines[] = "qb_op_handler next_handler2;";
				$lines[] = "int8_t *instruction_pointer2;";
			} else if($targetCount == 1) {
				$lines[] = "qb_op_handler next_handler;";
				$lines[] = "int8_t *instruction_pointer;";
			} else {
				$lines[] = "qb_op_handler next_handler;";
			}
		}
		
		for($i = 1; $i <= $opCount; $i++) {
			$addressMode = $this->getOperandAddressMode($i);
			if($addressMode == "CON") {
				$cType = $this->getOperandCType($i);
				$lines[] = "$cType operand{$i};";
			} else {
				$lines[] = "qb_pointer_{$addressMode} operand{$i};";
			}
		}
		
		if($targetCount > 2) {
			$lines[] = "qb_branch_table_entry branch_table[$targetCount];";
		}
		
		if($this->needsLineIdentifier()) {
			$lines[] = "uint32_t line_id;";
		}
		$lines[] = "} $instr;";
		return $lines;
	}
	
	public function getInstructionFormat() {
		$opCount = $this->getOperandCount();
		$format = "";
		
		for($i = 1; $i <= $opCount; $i++) {
			$addressMode = $this->getOperandAddressMode($i);
			if($this->changesOperand($i)) {
				switch($addressMode) {
					case 'SCA': $format .= 'S'; break;
					case 'ELE': $format .= 'E'; break;
					case 'ARR':
						$type = $this->getOperandCType($i);
						if($type[0] == 'c') {
							$format .= 'X'; 
						} else {
							$format .= 'A'; 
						}
						break;
					case 'CON':
						$class = get_class($this);
						die("Operand $i of $class is constant and changeable at the same time\n");
				}
			} else {
				switch($addressMode) {
					case 'SCA': $format .= 's'; break;
					case 'ELE': $format .= 'e'; break;
					case 'ARR': 
						$type = $this->getOperandCType($i);
						if($type[0] == 'c') {
							$format .= 'x'; 
						} else {
							$format .= 'a';
						}
						break;
					case 'CON': $format .= 'c'; break;
				}
			}
		}
		return $format;
	}
	
	// return the number of input operands
	public function getInputOperandCount() {
		return 0;
	}
	
	// return the number of output operands 
	public function getOutputOperandCount() {
		return 0;
	}
	
	// return the number of jump targets
	public function getJumpTargetCount() {
		return 0;
	}

	// return the total number of operands
	public function getOperandCount() {
		return $this->getInputOperandCount() + $this->getOutputOperandCount(); 
	}
	
	// return the type of operand $i (starting from 1)
	// by default, all operands have the same type
	public function getOperandType($i) {
		return $this->operandType;
	}
	
	// return the C-type of operand $i
	public function getOperandCType($i) {
		static $cTypes = array(
			"I08" => "int8_t",		"I16" => "int16_t",			"I32" => "int32_t",			"I64" => "int64_t",
			"S08" => "int8_t",		"S16" => "int16_t",			"S32" => "int32_t",			"S64" => "int64_t",
			"U08" => "uint8_t",		"U16" => "uint16_t",		"U32" => "uint32_t",		"U64" => "uint64_t",
			"F32" => "float32_t",	"F64" => "float64_t",
		);
		$operandType = $this->getOperandType($i);
		if(!$operandType) {
			$class = get_class($this);
			die("Operand $i of $class has null type\n");
		}
		return $cTypes[$operandType];
	}

	public function getOperandSizeShift($i) {
		return 0;
	}
	
	// return the number of elements that consist an operand 
	public function getOperandSize($i) {
		return $this->operandSize;
	}
	
	// return the address mode of operand $i
	// by default, all operands use the same address mode
	public function getOperandAddressMode($i) {
		return $this->addressMode;
	}
	
	public function isMultipleData() {
		return $this->multipleData;
	}
	
	public function handlesUnitData() {
		return true;
	}
	
	public function disableMultipleData() {
		$this->multipleData = false;
	}

	public function restoreMultipleData() {
		$this->multipleData = true;
	}
	
	public function isMultithreaded() {
		if(in_array('Multithreaded', class_uses($this))) {
			if($this->mayExitLoop()) {
				return false;
			} else {
				$threshold = $this->getMultithreadingThreshold();
				if($threshold === null) {
					if($this->operandSize != 'variable') {
						$class = get_class($this);
						echo "Missing threshold for $class ($this->operandType, $this->operandSize)\n";
					}
				}
				return ($threshold != 0);
			}
		} else {
			return false;
		}
	}	
	
	public function needsInterpreterContext() {
		return false;
	}
	
	public function needsReplication() {
		return false;
	}
	
	public function needsLineIdentifier() {
		return false;
	}

	public function alwaysReturns() {
		return false;
	}

	public function needsCondition() {
		$targetCount = $this->getJumpTargetCount();
		return ($targetCount >= 2);
	}

	public function changesOperand($i) {
		$srcCount = $this->getInputOperandCount();
		return ($i > $srcCount);
	}
	
	public function changesOperandSize($i) {
		return false;
	}
	
	public function runsInMainThread() {
		return false;
	}

	public function needsInstructionStructure() {
		if($this->isMultipleData() && $this->isMultithreaded()) {
			return true;
		} else if($this->runsInMainThread()) {
			return true;
		}
		return false;
	}

	public function isThreadSafe() {
		return true;
	}
	
	public function mayExitLoop() {
		return false;
	}
	
	public function getMultithreadingThreshold() {
		if(self::$multithreadingThresholds === null) {
			$folder = dirname(__FILE__);
			$path = "$folder/../threshold/multithreading_thresholds.txt";
			if(file_exists($path)) { 
				self::$multithreadingThresholds = array();
				$lines = file($path, FILE_SKIP_EMPTY_LINES | FILE_IGNORE_NEW_LINES);
				foreach($lines as $line) {
					if(!preg_match('/^\s*;/', $line)) {
						if(preg_match('/(\w+)\s+(\w+)\s+(\d+)\s+(\d+)/', $line, $m)) {
							$name = $m[1];
							$type = $m[2];
							$width = (int) $m[3];
							$threshold = (int) $m[4];

							$byType =& self::$multithreadingThresholds[$name];
							$byWidth =& $byType[$type];
							$byWidth[$width] = $threshold;
						}
					}
				}
			} else {
				// when the file is missing, we're trying to generate it
				self::$multithreadingThresholds == "calc";
			}
		}
		if(is_array(self::$multithreadingThresholds)) {
			$class = get_class($this);
			$type = $this->operandType;
			$width = $this->operandSize;
			if(isset(self::$multithreadingThresholds[$class][$type][$width])) {
				return self::$multithreadingThresholds[$class][$type][$width];
			} else {
				return null;
			}
		} else {
			// use a low value so we can test it
			return 1024;
		}
	}
	
	public function performsWrapAround() {
		if($this->addressMode == "ARR" && !$this->isOverridden('getActionOnMultipleData')) {
			return true;
		}
		return false;
	}
	
	protected function getMacroDefinitions() {
		$instr = $this->getInstructionStructure();
		$srcCount = $this->getInputOperandCount();
		$opCount = $this->getOperandCount();
		$lines = array();
		if($instr) {
			$lines[] = "#define INSTR		(($instr * __restrict) ip)";
		}
		if($this->needsLineIdentifier()) {
			$lines[] = "#define line_id		INSTR->line_id";
		}
		for($i = 1; $i <= $opCount; $i++) {
			$cType = $this->getOperandCType($i);
			$addressMode = $this->getOperandAddressMode($i);
			$operand = "INSTR->operand$i";
			$name = ($i <= $srcCount) ? "op{$i}" : "res";
			switch($addressMode) {
				case 'SCA':
					$lines[] = "#define $name	(($cType *) $operand.data_pointer)[0]";
					break;
				case 'ELE':
					$lines[] = "#define $name	(($cType *) $operand.data_pointer)[$operand.index_pointer[0]]";
					break;
				case 'ARR':
					$lines[] = "#define {$name}_ptr		((($cType *) $operand.data_pointer) + $operand.index_pointer[0])";
					$lines[] = "#define {$name}_count	$operand.count_pointer[0]";
					if($this->changesOperandSize($i)) {
						$lines[] = "#define {$name}_count_ptr	$operand.count_pointer";
					}
					break;
				case 'CON':
					$lines[] = "#define $name	$operand";
					break;
			}
		}
		return $lines;
	}
	
	protected function getMacroUndefinitions() {
		$instr = $this->getInstructionStructure();
		$srcCount = $this->getInputOperandCount();
		$opCount = $this->getOperandCount();
		$lines = array();
		if($instr) {
			$lines[] = "#undef INSTR";
		}
		if($this->needsLineIdentifier()) {
			$lines[] = "#undef line_id";
		}
		for($i = 1; $i <= $opCount; $i++) {
			$addressMode = $this->getOperandAddressMode($i);
			$name = ($i <= $srcCount) ? "op{$i}" : "res";
			switch($addressMode) {
				case 'SCA':
					$lines[] = "#undef $name";
					break;
				case 'ELE':
					$lines[] = "#undef $name";
					break;
				case 'ARR':
					$lines[] = "#undef {$name}_ptr";
					$lines[] = "#undef {$name}_count";
					if($this->changesOperandSize($i)) {
						$lines[] = "#undef {$name}_count_ptr";
					}
					break;
				case 'CON':
					$lines[] = "#undef $name";
					break;
			}
		}
		return $lines;
	}
	
	protected function getFunctionNameComponents($prefix) {
		$className = get_class($this);
		$parts = array();
		$parts[] = "qb";
		$parts[] = $prefix;
		$parts[] = strtolower(preg_replace("/([a-z])([A-Z])/", "$1_$2", $className));
		if($this->operandSize && $this->operandSize != 1 && is_int($this->operandSize)) {
			$parts[] = "{$this->operandSize}x";
		}
		if($this->isMultipleData()) {
			$parts[] = "multiple_times";
		}
		if($this->operandType) {
			$parts[] = $this->operandType;
		}
		return $parts;
	}

	public function getHandlerFunctionName() {
		$parts = $this->getFunctionNameComponents("do");
		return implode('_', $parts);
	}
	
	public function getHandlerFunctionType() {
		if($this->runsInMainThread()) {
			return 'extern';
		} else if(!$this->isOverridden('getAction')) {
			if($this->isMultipleData()) {
				return 'extern';
			} else {
				$action = $this->getActionOnUnitData();
				$count = count($action, true);
				if($count > 8) {
					return 'extern';
				} else if($count == 1) {
					return 'inline';
				}
				
				$lines = array_linearize($action);
				$hasLoop = false;
				foreach($lines as $line) {
					if(preg_match('/\b(for|while)\b/', $line)) {
						$hasLoop = true;
					}
				}
				if($hasLoop) {
					return 'extern';
				} else {
					return 'inline';
				}
			}
		} 
		return null;
	}

	public function getHandlerFunctionDefinition() {
		$functionType = $this->getHandlerFunctionType();
		if(!$functionType) {
			return null;
		}
		$function = $this->getHandlerFunctionName();
		$parameterList = $this->getHandlerFunctionParameterList(true);
		$expressions = $this->getActionExpressions();
		$mayExit = $this->mayExitLoop();
		if($mayExit) {
			$typeDecl = "int32_t";
		} else {
			$typeDecl = "void";
		}
		if($functionType == "inline") {
			$typeDecl = "static zend_always_inline $typeDecl";
		}
		
		// replace op# with (*op#_ptr) and res with (*res_ptr) where it's necessary
		$expressions = array_linearize($expressions);
		$srcCount = $this->getInputOperandCount();
		$needPointers = array();
		for($i = 1; $i <= $srcCount; $i++) {
			if($this->getOperandAddressMode($i) == "ARR" || $this->changesOperand($i)) {
				$needPointers[] = "op{$i}";
			}
			if($this->changesOperandSize($i)) {
				$needPointers[] = "op{$i}_count";
			}
		}
		$needPointers[] = "res";
		if($this->changesOperandSize($i)) {
			$needPointers[] = "res_count";
		}
		if($this->needsCondition()) {
			$needPointers[] = "condition";
		}
		$regExp = '/\b(' . implode('|', $needPointers) . ')\b/';
		$expressions = preg_replace($regExp, '(*\1_ptr)', $expressions);
		if($mayExit) {
			// replace "return" with "return FALSE" and add return TRUE at the end
			$expressions = preg_replace('/\breturn\b/', 'return FALSE', $expressions);
			$expressions[] = "return TRUE;";
		}
		$lines = array();			
		$lines[] = "$typeDecl $function($parameterList) {";
		$lines[] = $expressions;
		$lines[] = "}";
		return $lines;
	}
	
	public function getHandlerFunctionParameterList($forDeclaration) {
		$instr = $this->getInstructionStructure();
		$srcCount = $this->getInputOperandCount();
		$opCount = $this->getOperandCount();
		$params = array();
		if($this->needsInterpreterContext()) {
			if($forDeclaration) {
				$params[] = "qb_interpreter_context *__restrict cxt";
			} else {
				$params[] = "cxt";
			}
		} 
		if($this->needsCondition()) {
			if($forDeclaration) {
				$params[] = "int32_t *condition_ptr";
			} else {
				$params[] = "&condition";
			}
		}
		for($i = 1; $i <= $opCount; $i++) {
			$cType = $this->getOperandCType($i);
			$addressMode = $this->getOperandAddressMode($i);
			$operand = "((($instr *) ip)->operand$i)";
			$name = ($i <= $srcCount) ? "op{$i}" : "res";
			switch($addressMode) {
				case 'SCA':
				case 'ELE':
				case 'CON':
					if($forDeclaration) {
						if($this->changesOperand($i)) {
							$params[] = "$cType *{$name}_ptr";
						} else {
							$params[] = "$cType $name";
						}
					} else {
						if($this->changesOperand($i)) {
							$params[] = "&$name";
						} else {
							$params[] = "$name";
						}
					}
					break;
				case 'ARR':
					if($forDeclaration) {
						$params[] = "$cType *{$name}_ptr";
						if($this->changesOperandSize($i)) {
							$params[] = "uint32_t *{$name}_count_ptr";
						} else {
							// don't need the size if it's a constant number
							if($this->isMultipleData() || !is_numeric($this->getOperandSize($i))) {
								$params[] = "uint32_t {$name}_count";
							}
						}
					} else {
						$params[] = "{$name}_ptr";
						if($this->changesOperandSize($i)) {
							$params[] = "{$name}_count_ptr";
						} else {
							if($this->isMultipleData() || !is_numeric($this->getOperandSize($i))) {
								$shift = $this->getOperandSizeShift($i);
								if($shift) {
									$params[] = "{$name}_count >> $shift";
								} else {
									$params[] = "{$name}_count";
								}
							}
						}
					}
					break;
			}
		}
		if($this->needsLineIdentifier()) {
			if($forDeclaration) {
				$params[] = "uint32_t line_id";
			} else {
				$params[] = "line_id";
			}
		} 
		return implode(", ", $params);
	}

	// return the name of the dispatcher function, which sends a instruction to multiple threads 
	protected function getDispatcherFunctionName() {
		$instr = $this->getInstructionStructure();
		$name = "qb_dispatch_" . substr($instr, 3);
		return $name;
	}
	
	// return the parameter list of the dispatcher function
	protected function getDispatcherFunctionParameterList($forDeclaration) {
		$instr = $this->getInstructionStructure();
		$params = array();
		if($forDeclaration) {
			$params[] = "qb_interpreter_context *__restrict cxt";
			$params[] = "void *control_func";
			$params[] = "$instr *__restrict instr";
		} else {
			$params[] = "cxt";
			$params[] = $this->getControllerFunctionName();
			$params[] = "($instr *) ip";
		}
		
		// add operand sizes
		$opCount = $this->getOperandCount();
		for($i = 1; $i <= $opCount; $i++) {
			$addressMode = $this->getOperandAddressMode($i);
			if($addressMode == "ARR") {
				if($forDeclaration) {
					$params[] = "uint32_t operand{$i}_size";
				} else {
					$operandSize = $this->getOperandSize($i);
					if($operandSize) {
						$params[] = $operandSize;
					} else {
						// put it the size of the array to indicate the pointer doesn't shift
						$params[] = "op{$i}_count";
					}
				}
			}
		}
		// add threshold value
		if($forDeclaration) {
			$params[] = "uint32_t threshold";
		} else {
			$params[] = $this->getMultithreadingThreshold();
		}
		return implode(", ", $params);
	}
	
	// return the body list of the dispatcher function
	public function getDispatcherFunctionDefinition() {
		if($this->isMultipleData() && $this->isMultithreaded()) {
			$addressOperands = array();
			$instr = $this->getInstructionStructure();
			$dispatcherTypeDecl = "int32_t";
			$dispatcherFunction = $this->getDispatcherFunctionName();
			$dispatcherParameterList = $this->getDispatcherFunctionParameterList(true);
			$opCount = $this->getOperandCount();
			$arrayCount = 1;
			$lines = array();
			$lines[] = "$dispatcherTypeDecl $dispatcherFunction($dispatcherParameterList) {";
			$lines[] =		"uint32_t op{$opCount}_count = instr->operand{$opCount}.count_pointer[0];";
			$lines[] =		"if(op{$opCount}_count >= threshold) {";
			$lines[] =			"int32_t use_multithreading = TRUE;";
			$lines[] =			"uint32_t res_unit_count = op{$opCount}_count / operand{$opCount}_size;";
			$lines[] =			"uint32_t thread_count = cxt->thread_count;";
			$lines[] =			"uint32_t chunk_size = res_unit_count / thread_count;";
			for($i = 1; $i < $opCount; $i++) {
				$addressMode = $this->getOperandAddressMode($i);
				if($addressMode == "ARR") {
					$lines[] =	"uint32_t op{$i}_count = instr->operand{$i}.count_pointer[0], op{$i}_unit_count = op{$i}_count / operand{$i}_size, op{$i}_shift, op{$i}_chunk_size;";
					$arrayCount++;
				}	
			}
			$lines[] =			"uint32_t op{$i}_shift = operand{$opCount}_size * chunk_size, op{$i}_chunk_size = operand{$opCount}_size * chunk_size;";			
			for($i = 1; $i < $opCount; $i++) {
				$addressMode = $this->getOperandAddressMode($i);
				if($addressMode == "ARR") {
					$lines[] =	"if(op{$i}_unit_count == res_unit_count) {";
					$lines[] =		"op{$i}_shift = operand{$i}_size * chunk_size;";
					$lines[] =		"op{$i}_chunk_size = operand{$i}_size * chunk_size;";
					$lines[] =	"} else if(op{$i}_unit_count == 1) {";
					$lines[] =		"op{$i}_shift = 0;";
					$lines[] =		"op{$i}_chunk_size = operand{$i}_size;";
					$lines[] =	"} else {";
					$lines[] =		"use_multithreading = FALSE;";
					$lines[] =	"}";
				}	
			}
			$lines[] =			"if(use_multithreading) {";
			$lines[] =				"// create temporary instruction structures";
			$lines[] =				"$instr new_instr_list[MAX_THREAD_COUNT];";
			$lines[] =				"uint32_t new_indices[MAX_THREAD_COUNT][$arrayCount];";
			$lines[] =				"uint32_t new_counts[MAX_THREAD_COUNT][$arrayCount];";
			$lines[] =				"int8_t *new_ips[MAX_THREAD_COUNT];";
			$lines[] =				"uint32_t i;";
			$lines[] =				"for(i = 0; i < thread_count; i++) {";
			$lines[] =					"$instr *new_instr = &new_instr_list[i];";
			for($i = 1, $k = 0; $i <= $opCount; $i++) {
				$addressMode = $this->getOperandAddressMode($i);
				if($addressMode == "ARR") {
					$lines[] =			"new_indices[i][$k] = i * op{$i}_shift;";
					$lines[] =			"new_counts[i][$k] = (i == thread_count - 1) ? op{$i}_count - (i * op{$i}_shift) : op{$i}_chunk_size;";
					$lines[] = 			"new_instr->operand{$i}.data_pointer = instr->operand{$i}.data_pointer;";
					$lines[] = 			"new_instr->operand{$i}.index_pointer = &new_indices[i][$k];";
					$lines[] = 			"new_instr->operand{$i}.count_pointer = &new_counts[i][$k];";
					$k++;
				} else {
					$lines[] = 			"new_instr->operand{$i} = instr->operand{$i};";
				}
			}
			$lines[] =					"new_ips[i] = (int8_t *) new_instr;";
			$lines[] =				"}";
			$lines[] = 				"qb_dispatch_instruction_to_threads(cxt, control_func, new_ips, thread_count);";
			$lines[] =				"return TRUE;";
			$lines[] =			"}";
			$lines[] =		"}";
			$lines[] =		"return FALSE;";
			$lines[] = "}";
			return $lines;
		}
	}
	
	// return the name of the controller function, which decides whether to use multithreading or not
	protected function getControllerFunctionName() {
		$parts = $this->getFunctionNameComponents("redirect");
		if($this->addressMode == "ELE") {
			array_splice($parts, -1, 0, "array_element");
		}
		return implode('_', $parts);
	}
	
	// return the parameter list of the controller function
	protected function getControllerFunctionParameterList($forDeclaration) {
		$params = array();
		if($forDeclaration) {
			$params[] = "qb_interpreter_context *__restrict cxt";
			$params[] = "int8_t *__restrict ip";
			$params[] = "int unused";
		} else {
			$params[] = "cxt";
			$params[] = "ip";
			$params[] = "0";
		}
		return implode(", ", $params);
	}
	
	// return the body of the controller function
	public function getControllerFunctionDefinition() {
		if($this->isMultipleData() && $this->isMultithreaded()) {
			$controllerTypeDecl = "void";
			$controllerFunction = $this->getControllerFunctionName();
			$controllerParameterList = $this->getControllerFunctionParameterList(true);
			$dispatcherFunction = $this->getDispatcherFunctionName();
			$dispatcherParameterList = $this->getDispatcherFunctionParameterList(false);
			$handlerFunction = $this->getHandlerFunctionName();
			$handlerParameterList = $this->getHandlerFunctionParameterList(false);
			$lines = array();
			$lines[] = "$controllerTypeDecl $controllerFunction($controllerParameterList) {";
			$lines[] =		$this->getMacroDefinitions();
			$lines[] =		"if(!cxt->thread_count || !$dispatcherFunction($dispatcherParameterList)) {";
			$lines[] = 			"$handlerFunction($handlerParameterList);";
			$lines[] =		"}";
			$lines[] =		$this->getMacroUndefinitions();
			$lines[] = "}";
			return $lines;
		} else if($this->runsInMainThread()) {
			$controllerTypeDecl = "void";
			$controllerFunction = $this->getControllerFunctionName();
			$controllerParameterList = $this->getControllerFunctionParameterList(true);
			$handlerFunction = $this->getHandlerFunctionName();
			$handlerParameterList = $this->getHandlerFunctionParameterList(false);
			$lines = array();
			$lines[] = "$controllerTypeDecl $controllerFunction($controllerParameterList) {";
			$lines[] =		$this->getMacroDefinitions();
			$lines[] =		"if(!qb_in_main_thread()) {";
			$lines[] = 			"qb_dispatch_instruction_to_main_thread(cxt, $controllerFunction, ip);";
			$lines[] =		"} else {";
			$lines[] = 			"$handlerFunction($handlerParameterList);";
			$lines[] =		"}";
			$lines[] =		$this->getMacroUndefinitions();
			$lines[] = "}";
			return $lines;
		}
	}
	
	// return codes that perform what the op is supposed to do
	public function getAction() {
		$functionType = $this->getHandlerFunctionType();
		if($functionType) {	
			if($this->isMultipleData() && $this->isMultithreaded()) {
				// send instruction to the controller function, which will either
				// (1) call the dispatcher function, which then calls the controller function again from different threads
				// (2) call the handler function
				$function = $this->getControllerFunctionName();
				$parameterList = $this->getControllerFunctionParameterList(false);
			} else if($this->runsInMainThread()) {
				// if the call occurs inside a thread, it needs to be redirected to the main thread
				$function = $this->getControllerFunctionName();
				$parameterList = $this->getControllerFunctionParameterList(false);
			} else {
				// call the handler directly
				$function = $this->getHandlerFunctionName();
				$parameterList = $this->getHandlerFunctionParameterList(false);
			}
			if($this->mayExitLoop()) {
				$lines = array();
				$lines[] = "if(!$function($parameterList)) {";
				$lines[] =		"return;";
				$lines[] = "}";
				return $lines;
			} else {
				return "$function($parameterList);";
			}
		} else {
			// just insert the code, expanding the operands
			$lines = array();
			$lines[] = "{";
			$lines[] = 		$this->getActionExpressions();
			$lines[] = "}";
			return $lines;
		}
	}	
	
	// return an expression for handling a single unit of data (typically a scalar)
	protected function getActionOnUnitData() {
		return null;
	}

	// return an expression for handling multiple units of data
	protected function getActionOnMultipleData() {
		return null;
	}
	
	protected function getActionExpressions() {
		if($this->isMultipleData()) {
			$action = $this->getActionOnMultipleData();
			if(!$action) {
				// temporarily changed the handle to the non-multiple-data
				$this->disableMultipleData();
				$scalarExpression = $this->getAction();
				$this->restoreMultipleData();
				$action = $this->getIterationCode($scalarExpression);
			}
		} else {
			$action = $this->getActionOnUnitData();
			if($this->needsReplication()) {
				$action = $this->replicateExpression($action, $this->operandSize);
			}
		}
		return $action;
	}
		
	// multiple a scalar operation multiple times
	protected function replicateExpression($expression, $count) {
		$srcCount = $this->getInputOperandCount();
		$arrayOperands = array();
		for($i = 1; $i <= $srcCount; $i++) {
			if($this->getOperandAddressMode($i) == "ARR") {
				$arrayOperands[] = $i;
			}
		}
		$nums = implode("|", $arrayOperands);
		$lines = array();
		for($i = 0; $i < $count; $i++) {
			$patterns = array('/\bres\b/', '/\bop(' . $nums . ')\b/');
			$replacements = array("res_ptr[{$i}]", "op\\1_ptr[{$i}]");
			$lines[] = preg_replace($patterns, $replacements, $expression);
		}
		return $lines;
	}
	
	// return code for a loop that performs the same operation on all element of an array 
	protected function getIterationCode($expression, $operandSizeOverride = false) {
		$srcCount = $this->getInputOperandCount();
		$lines = array();		
		
		// make sure none of the input operands are empty		
		$condition = false;
		$operandCounts = array();
		for($i = 1; $i <= $srcCount; $i++) {
			$operandSize = ($operandSizeOverride) ? $operandSizeOverride : $this->getOperandSize($i);
			if($this->getOperandAddressMode($i) == "ARR" && $operandSize !== 0) {
				$operandCounts[] = "op{$i}_count";
			}
		}
		$operandCounts[] = "res_count";
		// use bitwise AND here, just in case the compiler doesn't optimize correctly
		$condition = implode(" && ", $operandCounts);
		$lines[] = "if($condition) {";
		for($i = 1; $i <= $srcCount; $i++) {
			$cType = $this->getOperandCType($i);
			$operandSize = ($operandSizeOverride) ? $operandSizeOverride : $this->getOperandSize($i);
			if($this->getOperandAddressMode($i) == "ARR" && $operandSize !== 0) {
				$lines[] =	"$cType *op{$i}_start = op{$i}_ptr, *op{$i}_end = op{$i}_ptr + op{$i}_count;";
			}
		}
		$cType = $this->getOperandCType($srcCount + 1);
		$lines[] =		"$cType *res_end = res_ptr + res_count;";
		$lines[] = 		"for(;;) {";
		$lines[] = 			$expression;
		$lines[] =			"";
		$operandSize = ($operandSizeOverride) ? $operandSizeOverride : $this->getOperandSize($srcCount + 1);
		// group incrementations together since they can happen in parallel
		$lines[] = 			"res_ptr += $operandSize;";
		for($i = 1; $i <= $srcCount; $i++) {
			$operandSize = ($operandSizeOverride) ? $operandSizeOverride : $this->getOperandSize($i);
			if($this->getOperandAddressMode($i) == "ARR" && $operandSize !== 0) {
				$lines[] =		"op{$i}_ptr += $operandSize;";
			}
		}
		$lines[] =			"if(res_ptr >= res_end) {";
		$lines[] =				"break;";
		$lines[] =			"}";
		for($i = 1; $i <= $srcCount; $i++) {
			$operandSize = $this->getOperandSize($i);
			if($this->getOperandAddressMode($i) == "ARR" && $operandSize !== 0) {
				$lines[] = 	"if(op{$i}_ptr >= op{$i}_end) {";
				$lines[] = 		"op{$i}_ptr = op{$i}_start;";
				$lines[] = 	"}";
			}
		}
		$lines[] = 		"}";	// end for
		$lines[] = "}"; 		// end if
		return $lines;
	}
	
	protected function isOverridden($methodName) {
		$child  = new ReflectionClass($this);
		$method = $child->getMethod($methodName);
		return ($method->class != 'Handler');
	}
}

function array_linearize($array) {
	$result = array();
	if(is_array($array)) {
		foreach($array as $element) {
			if($element !== NULL) {
				if(is_array($element)) {
					$sub_array = array_linearize($element);					
					array_splice($result, count($result), 0, $sub_array);
				} else {
					$result[] = $element;
				}
			}
		}
	} else {
		$result[] = $array;
	}
	return $result;
}

?>

Function Calls

None

Variables

None

Stats

MD5 3b394df2a129cf12b08ba789b4c8899c
Eval Count 0
Decode Time 199 ms