#!/usr/bin/env php getPathName(); if (preg_match('/\.stub\.php$/', $pathName)) { $pathNames[] = $pathName; } } // Make sure stub files are processed in a predictable, system-independent order. sort($pathNames); $fileInfos = []; foreach ($pathNames as $pathName) { $fileInfo = processStubFile($pathName, $context); if ($fileInfo) { $fileInfos[] = $fileInfo; } } return $fileInfos; } function processStubFile(string $stubFile, Context $context, bool $includeOnly = false): ?FileInfo { try { if (!file_exists($stubFile)) { throw new Exception("File $stubFile does not exist"); } if (!$includeOnly) { $stubFilenameWithoutExtension = str_replace(".stub.php", "", $stubFile); $arginfoFile = "{$stubFilenameWithoutExtension}_arginfo.h"; $legacyFile = "{$stubFilenameWithoutExtension}_legacy_arginfo.h"; $stubCode = file_get_contents($stubFile); $stubHash = computeStubHash($stubCode); $oldStubHash = extractStubHash($arginfoFile); if ($stubHash === $oldStubHash && !$context->forceParse) { /* Stub file did not change, do not regenerate. */ return null; } } if (!$fileInfo = $context->parsedFiles[$stubFile] ?? null) { initPhpParser(); $stubContent = $stubCode ?? file_get_contents($stubFile); $fileInfo = parseStubFile($stubContent); $context->parsedFiles[$stubFile] = $fileInfo; foreach ($fileInfo->dependencies as $dependency) { // TODO add header search path for extensions? $prefixes = [dirname($stubFile) . "/", dirname(__DIR__) . "/"]; foreach ($prefixes as $prefix) { $depFile = $prefix . $dependency; if (file_exists($depFile)) { break; } $depFile = null; } if (!$depFile) { throw new Exception("File $stubFile includes a file $dependency which does not exist"); } processStubFile($depFile, $context, true); } $constInfos = $fileInfo->getAllConstInfos(); $context->allConstInfos = array_merge($context->allConstInfos, $constInfos); } if ($includeOnly) { return $fileInfo; } $arginfoCode = generateArgInfoCode( basename($stubFilenameWithoutExtension), $fileInfo, $context->allConstInfos, $stubHash ); if (($context->forceRegeneration || $stubHash !== $oldStubHash) && file_put_contents($arginfoFile, $arginfoCode)) { echo "Saved $arginfoFile\n"; } if ($fileInfo->shouldGenerateLegacyArginfo()) { $legacyFileInfo = clone $fileInfo; $legacyFileInfo->legacyArginfoGeneration = true; $phpVersionIdMinimumCompatibility = $legacyFileInfo->getMinimumPhpVersionIdCompatibility(); foreach ($legacyFileInfo->getAllFuncInfos() as $funcInfo) { $funcInfo->discardInfoForOldPhpVersions($phpVersionIdMinimumCompatibility); } foreach ($legacyFileInfo->getAllClassInfos() as $classInfo) { $classInfo->discardInfoForOldPhpVersions($phpVersionIdMinimumCompatibility); } foreach ($legacyFileInfo->getAllConstInfos() as $constInfo) { $constInfo->discardInfoForOldPhpVersions($phpVersionIdMinimumCompatibility); } $arginfoCode = generateArgInfoCode( basename($stubFilenameWithoutExtension), $legacyFileInfo, $context->allConstInfos, $stubHash ); if (($context->forceRegeneration || $stubHash !== $oldStubHash) && file_put_contents($legacyFile, $arginfoCode)) { echo "Saved $legacyFile\n"; } } return $fileInfo; } catch (Exception $e) { echo "In $stubFile:\n{$e->getMessage()}\n"; exit(1); } } function computeStubHash(string $stubCode): string { return sha1(str_replace("\r\n", "\n", $stubCode)); } function extractStubHash(string $arginfoFile): ?string { if (!file_exists($arginfoFile)) { return null; } $arginfoCode = file_get_contents($arginfoFile); if (!preg_match('/\* Stub hash: ([0-9a-f]+) \*/', $arginfoCode, $matches)) { return null; } return $matches[1]; } class Context { public bool $forceParse = false; public bool $forceRegeneration = false; /** @var array */ public array $allConstInfos = []; /** @var FileInfo[] */ public array $parsedFiles = []; } class ArrayType extends SimpleType { public Type $keyType; public Type $valueType; public static function createGenericArray(): self { return new ArrayType(Type::fromString("int|string"), Type::fromString("mixed|ref")); } public function __construct(Type $keyType, Type $valueType) { parent::__construct("array", true); $this->keyType = $keyType; $this->valueType = $valueType; } public function toOptimizerTypeMask(): string { $typeMasks = [ parent::toOptimizerTypeMask(), $this->keyType->toOptimizerTypeMaskForArrayKey(), $this->valueType->toOptimizerTypeMaskForArrayValue(), ]; return implode("|", $typeMasks); } public function equals(SimpleType $other): bool { if (!parent::equals($other)) { return false; } assert(get_class($other) === self::class); return Type::equals($this->keyType, $other->keyType) && Type::equals($this->valueType, $other->valueType); } } class SimpleType { public string $name; public bool $isBuiltin; public static function fromNode(Node $node): SimpleType { if ($node instanceof Node\Name) { if ($node->toLowerString() === 'static') { // PHP internally considers "static" a builtin type. return new SimpleType($node->toLowerString(), true); } if ($node->toLowerString() === 'self') { throw new Exception('The exact class name must be used instead of "self"'); } assert($node->isFullyQualified()); return new SimpleType($node->toString(), false); } if ($node instanceof Node\Identifier) { if ($node->toLowerString() === 'array') { return ArrayType::createGenericArray(); } return new SimpleType($node->toLowerString(), true); } throw new Exception("Unexpected node type"); } public static function fromString(string $typeString): SimpleType { switch (strtolower($typeString)) { case "void": case "null": case "false": case "true": case "bool": case "int": case "float": case "string": case "callable": case "object": case "resource": case "mixed": case "static": case "never": case "ref": return new SimpleType(strtolower($typeString), true); case "array": return ArrayType::createGenericArray(); case "self": throw new Exception('The exact class name must be used instead of "self"'); case "iterable": throw new Exception('This should not happen'); } $matches = []; $isArray = preg_match("/(.*)\s*\[\s*\]/", $typeString, $matches); if ($isArray) { return new ArrayType(Type::fromString("int"), Type::fromString($matches[1])); } $matches = []; $isArray = preg_match("/array\s*<\s*([A-Za-z0-9_|-]+)?(\s*,\s*)?([A-Za-z0-9_|-]+)?\s*>/i", $typeString, $matches); if ($isArray) { if (empty($matches[1]) || empty($matches[3])) { throw new Exception("array<> type hint must have both a key and a value"); } return new ArrayType(Type::fromString($matches[1]), Type::fromString($matches[3])); } return new SimpleType($typeString, false); } /** * @param mixed $value */ public static function fromValue($value): SimpleType { switch (gettype($value)) { case "NULL": return SimpleType::null(); case "boolean": return SimpleType::bool(); case "integer": return SimpleType::int(); case "double": return SimpleType::float(); case "string": return SimpleType::string(); case "array": return SimpleType::array(); case "object": return SimpleType::object(); default: throw new Exception("Type \"" . gettype($value) . "\" cannot be inferred based on value"); } } public static function null(): SimpleType { return new SimpleType("null", true); } public static function bool(): SimpleType { return new SimpleType("bool", true); } public static function int(): SimpleType { return new SimpleType("int", true); } public static function float(): SimpleType { return new SimpleType("float", true); } public static function string(): SimpleType { return new SimpleType("string", true); } public static function array(): SimpleType { return new SimpleType("array", true); } public static function object(): SimpleType { return new SimpleType("object", true); } public static function void(): SimpleType { return new SimpleType("void", true); } protected function __construct(string $name, bool $isBuiltin) { $this->name = $name; $this->isBuiltin = $isBuiltin; } public function isScalar(): bool { return $this->isBuiltin && in_array($this->name, ["null", "false", "true", "bool", "int", "float"], true); } public function isNull(): bool { return $this->isBuiltin && $this->name === 'null'; } public function isBool(): bool { return $this->isBuiltin && $this->name === 'bool'; } public function isInt(): bool { return $this->isBuiltin && $this->name === 'int'; } public function isFloat(): bool { return $this->isBuiltin && $this->name === 'float'; } public function isString(): bool { return $this->isBuiltin && $this->name === 'string'; } public function isArray(): bool { return $this->isBuiltin && $this->name === 'array'; } public function isMixed(): bool { return $this->isBuiltin && $this->name === 'mixed'; } public function toTypeCode(): string { assert($this->isBuiltin); switch ($this->name) { case "bool": return "_IS_BOOL"; case "int": return "IS_LONG"; case "float": return "IS_DOUBLE"; case "string": return "IS_STRING"; case "array": return "IS_ARRAY"; case "object": return "IS_OBJECT"; case "void": return "IS_VOID"; case "callable": return "IS_CALLABLE"; case "mixed": return "IS_MIXED"; case "static": return "IS_STATIC"; case "never": return "IS_NEVER"; case "null": return "IS_NULL"; case "false": return "IS_FALSE"; case "true": return "IS_TRUE"; default: throw new Exception("Not implemented: $this->name"); } } public function toTypeMask(): string { assert($this->isBuiltin); switch ($this->name) { case "null": return "MAY_BE_NULL"; case "false": return "MAY_BE_FALSE"; case "true": return "MAY_BE_TRUE"; case "bool": return "MAY_BE_BOOL"; case "int": return "MAY_BE_LONG"; case "float": return "MAY_BE_DOUBLE"; case "string": return "MAY_BE_STRING"; case "array": return "MAY_BE_ARRAY"; case "object": return "MAY_BE_OBJECT"; case "callable": return "MAY_BE_CALLABLE"; case "mixed": return "MAY_BE_ANY"; case "void": return "MAY_BE_VOID"; case "static": return "MAY_BE_STATIC"; case "never": return "MAY_BE_NEVER"; default: throw new Exception("Not implemented: $this->name"); } } public function toOptimizerTypeMaskForArrayKey(): string { assert($this->isBuiltin); switch ($this->name) { case "int": return "MAY_BE_ARRAY_KEY_LONG"; case "string": return "MAY_BE_ARRAY_KEY_STRING"; default: throw new Exception("Type $this->name cannot be an array key"); } } public function toOptimizerTypeMaskForArrayValue(): string { if (!$this->isBuiltin) { return "MAY_BE_ARRAY_OF_OBJECT"; } switch ($this->name) { case "null": return "MAY_BE_ARRAY_OF_NULL"; case "false": return "MAY_BE_ARRAY_OF_FALSE"; case "true": return "MAY_BE_ARRAY_OF_TRUE"; case "bool": return "MAY_BE_ARRAY_OF_FALSE|MAY_BE_ARRAY_OF_TRUE"; case "int": return "MAY_BE_ARRAY_OF_LONG"; case "float": return "MAY_BE_ARRAY_OF_DOUBLE"; case "string": return "MAY_BE_ARRAY_OF_STRING"; case "array": return "MAY_BE_ARRAY_OF_ARRAY"; case "object": return "MAY_BE_ARRAY_OF_OBJECT"; case "resource": return "MAY_BE_ARRAY_OF_RESOURCE"; case "mixed": return "MAY_BE_ARRAY_OF_ANY"; case "ref": return "MAY_BE_ARRAY_OF_REF"; default: throw new Exception("Type $this->name cannot be an array value"); } } public function toOptimizerTypeMask(): string { if (!$this->isBuiltin) { return "MAY_BE_OBJECT"; } switch ($this->name) { case "resource": return "MAY_BE_RESOURCE"; case "callable": return "MAY_BE_STRING|MAY_BE_ARRAY|MAY_BE_ARRAY_KEY_LONG|MAY_BE_ARRAY_OF_STRING|MAY_BE_ARRAY_OF_OBJECT|MAY_BE_OBJECT"; case "iterable": return "MAY_BE_ARRAY|MAY_BE_ARRAY_KEY_ANY|MAY_BE_ARRAY_OF_ANY|MAY_BE_OBJECT"; case "mixed": return "MAY_BE_ANY|MAY_BE_ARRAY_KEY_ANY|MAY_BE_ARRAY_OF_ANY"; } return $this->toTypeMask(); } public function toEscapedName(): string { // Escape backslashes, and also encode \u, \U, and \N to avoid compilation errors in generated macros return str_replace( ['\\', '\\u', '\\U', '\\N'], ['\\\\', '\\\\165', '\\\\125', '\\\\116'], $this->name ); } public function toVarEscapedName(): string { return str_replace('\\', '_', $this->name); } public function equals(SimpleType $other): bool { return $this->name === $other->name && $this->isBuiltin === $other->isBuiltin; } } class Type { /** @var SimpleType[] */ public array $types; public bool $isIntersection; public static function fromNode(Node $node): Type { if ($node instanceof Node\UnionType || $node instanceof Node\IntersectionType) { $nestedTypeObjects = array_map(['Type', 'fromNode'], $node->types); $types = []; foreach ($nestedTypeObjects as $typeObject) { array_push($types, ...$typeObject->types); } return new Type($types, ($node instanceof Node\IntersectionType)); } if ($node instanceof Node\NullableType) { return new Type( [ ...Type::fromNode($node->type)->types, SimpleType::null(), ], false ); } if ($node instanceof Node\Identifier && $node->toLowerString() === "iterable") { return new Type( [ SimpleType::fromString("Traversable"), ArrayType::createGenericArray(), ], false ); } return new Type([SimpleType::fromNode($node)], false); } public static function fromString(string $typeString): self { $typeString .= "|"; $simpleTypes = []; $simpleTypeOffset = 0; $inArray = false; $isIntersection = false; $typeStringLength = strlen($typeString); for ($i = 0; $i < $typeStringLength; $i++) { $char = $typeString[$i]; if ($char === "<") { $inArray = true; continue; } if ($char === ">") { $inArray = false; continue; } if ($inArray) { continue; } if ($char === "|" || $char === "&") { $isIntersection = ($char === "&"); $simpleTypeName = trim(substr($typeString, $simpleTypeOffset, $i - $simpleTypeOffset)); $simpleTypes[] = SimpleType::fromString($simpleTypeName); $simpleTypeOffset = $i + 1; } } return new Type($simpleTypes, $isIntersection); } /** * @param SimpleType[] $types */ private function __construct(array $types, bool $isIntersection) { $this->types = $types; $this->isIntersection = $isIntersection; } public function isScalar(): bool { foreach ($this->types as $type) { if (!$type->isScalar()) { return false; } } return true; } public function isNullable(): bool { foreach ($this->types as $type) { if ($type->isNull()) { return true; } } return false; } public function getWithoutNull(): Type { return new Type( array_values( array_filter( $this->types, function(SimpleType $type) { return !$type->isNull(); } ) ), false ); } public function tryToSimpleType(): ?SimpleType { $withoutNull = $this->getWithoutNull(); /* type has only null */ if (count($withoutNull->types) === 0) { return $this->types[0]; } if (count($withoutNull->types) === 1) { return $withoutNull->types[0]; } return null; } public function toArginfoType(): ArginfoType { $classTypes = []; $builtinTypes = []; foreach ($this->types as $type) { if ($type->isBuiltin) { $builtinTypes[] = $type; } else { $classTypes[] = $type; } } return new ArginfoType($classTypes, $builtinTypes); } public function toOptimizerTypeMask(): string { $optimizerTypes = []; foreach ($this->types as $type) { // TODO Support for toOptimizerMask for intersection $optimizerTypes[] = $type->toOptimizerTypeMask(); } return implode("|", $optimizerTypes); } public function toOptimizerTypeMaskForArrayKey(): string { $typeMasks = []; foreach ($this->types as $type) { $typeMasks[] = $type->toOptimizerTypeMaskForArrayKey(); } return implode("|", $typeMasks); } public function toOptimizerTypeMaskForArrayValue(): string { $typeMasks = []; foreach ($this->types as $type) { $typeMasks[] = $type->toOptimizerTypeMaskForArrayValue(); } return implode("|", $typeMasks); } public function getTypeForDoc(DOMDocument $doc): DOMElement { if (count($this->types) > 1) { $typeSort = $this->isIntersection ? "intersection" : "union"; $typeElement = $doc->createElement('type'); $typeElement->setAttribute("class", $typeSort); foreach ($this->types as $type) { $unionTypeElement = $doc->createElement('type', $type->name); $typeElement->appendChild($unionTypeElement); } } else { $type = $this->types[0]; $name = $type->name; $typeElement = $doc->createElement('type', $name); } return $typeElement; } public static function equals(?Type $a, ?Type $b): bool { if ($a === null || $b === null) { return $a === $b; } if (count($a->types) !== count($b->types)) { return false; } for ($i = 0; $i < count($a->types); $i++) { if (!$a->types[$i]->equals($b->types[$i])) { return false; } } return true; } public function __toString() { if ($this->types === null) { return 'mixed'; } $char = $this->isIntersection ? '&' : '|'; return implode($char, array_map( function ($type) { return $type->name; }, $this->types) ); } } class ArginfoType { /** @var SimpleType[] $classTypes */ public array $classTypes; /** @var SimpleType[] $builtinTypes */ private array $builtinTypes; /** * @param SimpleType[] $classTypes * @param SimpleType[] $builtinTypes */ public function __construct(array $classTypes, array $builtinTypes) { $this->classTypes = $classTypes; $this->builtinTypes = $builtinTypes; } public function hasClassType(): bool { return !empty($this->classTypes); } public function toClassTypeString(): string { return implode('|', array_map(function(SimpleType $type) { return $type->toEscapedName(); }, $this->classTypes)); } public function toTypeMask(): string { if (empty($this->builtinTypes)) { return '0'; } return implode('|', array_map(function(SimpleType $type) { return $type->toTypeMask(); }, $this->builtinTypes)); } } class ArgInfo { const SEND_BY_VAL = 0; const SEND_BY_REF = 1; const SEND_PREFER_REF = 2; public string $name; public int $sendBy; public bool $isVariadic; public ?Type $type; public ?Type $phpDocType; public ?string $defaultValue; /** @var AttributeInfo[] */ public array $attributes; /** * @param AttributeInfo[] $attributes */ public function __construct( string $name, int $sendBy, bool $isVariadic, ?Type $type, ?Type $phpDocType, ?string $defaultValue, array $attributes ) { $this->name = $name; $this->sendBy = $sendBy; $this->isVariadic = $isVariadic; $this->setTypes($type, $phpDocType); $this->defaultValue = $defaultValue; $this->attributes = $attributes; } public function equals(ArgInfo $other): bool { return $this->name === $other->name && $this->sendBy === $other->sendBy && $this->isVariadic === $other->isVariadic && Type::equals($this->type, $other->type) && $this->defaultValue === $other->defaultValue; } public function getSendByString(): string { switch ($this->sendBy) { case self::SEND_BY_VAL: return "0"; case self::SEND_BY_REF: return "1"; case self::SEND_PREFER_REF: return "ZEND_SEND_PREFER_REF"; } throw new Exception("Invalid sendBy value"); } public function getMethodSynopsisType(): Type { if ($this->type) { return $this->type; } if ($this->phpDocType) { return $this->phpDocType; } throw new Exception("A parameter must have a type"); } public function hasProperDefaultValue(): bool { return $this->defaultValue !== null && $this->defaultValue !== "UNKNOWN"; } public function getDefaultValueAsArginfoString(): string { if ($this->hasProperDefaultValue()) { return '"' . addslashes($this->defaultValue) . '"'; } return "NULL"; } public function getDefaultValueAsMethodSynopsisString(): ?string { if ($this->defaultValue === null) { return null; } switch ($this->defaultValue) { case 'UNKNOWN': return null; case 'false': case 'true': case 'null': return "&{$this->defaultValue};"; } return $this->defaultValue; } private function setTypes(?Type $type, ?Type $phpDocType): void { $this->type = $type; $this->phpDocType = $phpDocType; } } interface VariableLikeName { public function __toString(): string; public function getDeclarationName(): string; } interface ConstOrClassConstName extends VariableLikeName { public function equals(ConstOrClassConstName $const): bool; public function isClassConst(): bool; public function isUnknown(): bool; } abstract class AbstractConstName implements ConstOrClassConstName { public function equals(ConstOrClassConstName $const): bool { return $this->__toString() === $const->__toString(); } public function isUnknown(): bool { return strtolower($this->__toString()) === "unknown"; } } class ConstName extends AbstractConstName { public string $const; public function __construct(?Name $namespace, string $const) { if ($namespace && ($namespace = $namespace->slice(0, -1))) { $const = $namespace->toString() . '\\' . $const; } $this->const = $const; } public function isClassConst(): bool { return false; } public function isUnknown(): bool { $name = $this->__toString(); if (($pos = strrpos($name, '\\')) !== false) { $name = substr($name, $pos + 1); } return strtolower($name) === "unknown"; } public function __toString(): string { return $this->const; } public function getDeclarationName(): string { return $this->name->toString(); } } class ClassConstName extends AbstractConstName { public Name $class; public string $const; public function __construct(Name $class, string $const) { $this->class = $class; $this->const = $const; } public function isClassConst(): bool { return true; } public function __toString(): string { return $this->class->toString() . "::" . $this->const; } public function getDeclarationName(): string { return $this->const; } } class PropertyName implements VariableLikeName { public Name $class; public string $property; public function __construct(Name $class, string $property) { $this->class = $class; $this->property = $property; } public function __toString(): string { return $this->class->toString() . "::$" . $this->property; } public function getDeclarationName(): string { return $this->property; } } interface FunctionOrMethodName { public function getDeclaration(): string; public function getArgInfoName(): string; public function getMethodSynopsisFilename(): string; public function getNameForAttributes(): string; public function __toString(): string; public function isMethod(): bool; public function isConstructor(): bool; public function isDestructor(): bool; } class FunctionName implements FunctionOrMethodName { private Name $name; public function __construct(Name $name) { $this->name = $name; } public function getNamespace(): ?string { if ($this->name->isQualified()) { return $this->name->slice(0, -1)->toString(); } return null; } public function getNonNamespacedName(): string { if ($this->name->isQualified()) { throw new Exception("Namespaced name not supported here"); } return $this->name->toString(); } public function getDeclarationName(): string { return implode('_', $this->name->getParts()); } public function getFunctionName(): string { return $this->name->getLast(); } public function getDeclaration(): string { return "ZEND_FUNCTION({$this->getDeclarationName()});\n"; } public function getArgInfoName(): string { $underscoreName = implode('_', $this->name->getParts()); return "arginfo_$underscoreName"; } public function getFramelessFunctionInfosName(): string { $underscoreName = implode('_', $this->name->getParts()); return "frameless_function_infos_$underscoreName"; } public function getMethodSynopsisFilename(): string { return 'functions/' . implode('/', str_replace('_', '-', $this->name->getParts())); } public function getNameForAttributes(): string { return strtolower($this->name->toString()); } public function __toString(): string { return $this->name->toString(); } public function isMethod(): bool { return false; } public function isConstructor(): bool { return false; } public function isDestructor(): bool { return false; } } class MethodName implements FunctionOrMethodName { public Name $className; public string $methodName; public function __construct(Name $className, string $methodName) { $this->className = $className; $this->methodName = $methodName; } public function getDeclarationClassName(): string { return implode('_', $this->className->getParts()); } public function getDeclaration(): string { return "ZEND_METHOD({$this->getDeclarationClassName()}, $this->methodName);\n"; } public function getArgInfoName(): string { return "arginfo_class_{$this->getDeclarationClassName()}_{$this->methodName}"; } public function getMethodSynopsisFilename(): string { $parts = [...$this->className->getParts(), ltrim($this->methodName, '_')]; /* File paths are in lowercase */ return strtolower(implode('/', $parts)); } public function getNameForAttributes(): string { return strtolower($this->methodName); } public function __toString(): string { return "$this->className::$this->methodName"; } public function isMethod(): bool { return true; } public function isConstructor(): bool { return $this->methodName === "__construct"; } public function isDestructor(): bool { return $this->methodName === "__destruct"; } } class ReturnInfo { const REFCOUNT_0 = "0"; const REFCOUNT_1 = "1"; const REFCOUNT_N = "N"; const REFCOUNTS_NONSCALAR = [ self::REFCOUNT_1, self::REFCOUNT_N, ]; public bool $byRef; public ?Type $type; public ?Type $phpDocType; public bool $tentativeReturnType; public string $refcount; public function __construct(bool $byRef, ?Type $type, ?Type $phpDocType, bool $tentativeReturnType, ?string $refcount) { $this->byRef = $byRef; $this->setTypes($type, $phpDocType, $tentativeReturnType); $this->setRefcount($refcount); } public function equalsApartFromPhpDocAndRefcount(ReturnInfo $other): bool { return $this->byRef === $other->byRef && Type::equals($this->type, $other->type) && $this->tentativeReturnType === $other->tentativeReturnType; } public function getMethodSynopsisType(): ?Type { return $this->type ?? $this->phpDocType; } private function setTypes(?Type $type, ?Type $phpDocType, bool $tentativeReturnType): void { $this->type = $type; $this->phpDocType = $phpDocType; $this->tentativeReturnType = $tentativeReturnType; } private function setRefcount(?string $refcount): void { $type = $this->phpDocType ?? $this->type; $isScalarType = $type !== null && $type->isScalar(); if ($refcount === null) { $this->refcount = $isScalarType ? self::REFCOUNT_0 : self::REFCOUNT_N; return; } if ($isScalarType) { throw new Exception( "@refcount on functions returning scalar values is redundant and not permitted" ); } if (!in_array($refcount, ReturnInfo::REFCOUNTS_NONSCALAR, true)) { throw new Exception("@refcount must have one of the following values: \"1\", \"N\", $refcount given"); } $this->refcount = $refcount; } } class FuncInfo { public FunctionOrMethodName $name; public int $classFlags; public int $flags; public ?string $aliasType; public ?FunctionOrMethodName $alias; public bool $isDeprecated; public bool $supportsCompileTimeEval; public bool $verify; /** @var ArgInfo[] */ public array $args; public ReturnInfo $return; public int $numRequiredArgs; public ?string $cond; public bool $isUndocumentable; public ?int $minimumPhpVersionIdCompatibility; /** @var AttributeInfo[] */ public array $attributes; /** @var FramelessFunctionInfo[] */ public array $framelessFunctionInfos; public ?ExposedDocComment $exposedDocComment; /** * @param ArgInfo[] $args * @param AttributeInfo[] $attribute * @param FramelessFunctionInfo[] $framelessFunctionInfos */ public function __construct( FunctionOrMethodName $name, int $classFlags, int $flags, ?string $aliasType, ?FunctionOrMethodName $alias, bool $isDeprecated, bool $supportsCompileTimeEval, bool $verify, array $args, ReturnInfo $return, int $numRequiredArgs, ?string $cond, bool $isUndocumentable, ?int $minimumPhpVersionIdCompatibility, array $attributes, array $framelessFunctionInfos, ?ExposedDocComment $exposedDocComment ) { $this->name = $name; $this->classFlags = $classFlags; $this->flags = $flags; $this->aliasType = $aliasType; $this->alias = $alias; $this->isDeprecated = $isDeprecated; $this->supportsCompileTimeEval = $supportsCompileTimeEval; $this->verify = $verify; $this->args = $args; $this->return = $return; $this->numRequiredArgs = $numRequiredArgs; $this->cond = $cond; $this->isUndocumentable = $isUndocumentable; $this->minimumPhpVersionIdCompatibility = $minimumPhpVersionIdCompatibility; $this->attributes = $attributes; $this->framelessFunctionInfos = $framelessFunctionInfos; $this->exposedDocComment = $exposedDocComment; if ($return->tentativeReturnType && $this->isFinalMethod()) { throw new Exception("Tentative return inapplicable for final method"); } } public function isMethod(): bool { return $this->name->isMethod(); } public function isFinalMethod(): bool { return ($this->flags & Modifiers::FINAL) || ($this->classFlags & Modifiers::FINAL); } public function isInstanceMethod(): bool { return !($this->flags & Modifiers::STATIC) && $this->isMethod() && !$this->name->isConstructor(); } /** @return string[] */ public function getModifierNames(): array { if (!$this->isMethod()) { return []; } $result = []; if ($this->flags & Modifiers::FINAL) { $result[] = "final"; } elseif ($this->flags & Modifiers::ABSTRACT && $this->classFlags & ~Modifiers::ABSTRACT) { $result[] = "abstract"; } if ($this->flags & Modifiers::PROTECTED) { $result[] = "protected"; } elseif ($this->flags & Modifiers::PRIVATE) { $result[] = "private"; } else { $result[] = "public"; } if ($this->flags & Modifiers::STATIC) { $result[] = "static"; } return $result; } public function hasParamWithUnknownDefaultValue(): bool { foreach ($this->args as $arg) { if ($arg->defaultValue && !$arg->hasProperDefaultValue()) { return true; } } return false; } public function equalsApartFromNameAndRefcount(FuncInfo $other): bool { if (count($this->args) !== count($other->args)) { return false; } for ($i = 0; $i < count($this->args); $i++) { if (!$this->args[$i]->equals($other->args[$i])) { return false; } } return $this->return->equalsApartFromPhpDocAndRefcount($other->return) && $this->numRequiredArgs === $other->numRequiredArgs && $this->cond === $other->cond; } public function getArgInfoName(): string { return $this->name->getArgInfoName(); } public function getDeclarationKey(): string { $name = $this->alias ?? $this->name; return "$name|$this->cond"; } public function getDeclaration(): ?string { if ($this->flags & Modifiers::ABSTRACT) { return null; } $name = $this->alias ?? $this->name; return $name->getDeclaration(); } public function getFramelessDeclaration(FuncInfo $funcInfo): ?string { if (empty($this->framelessFunctionInfos)) { return null; } $php84MinimumCompatibility = $this->minimumPhpVersionIdCompatibility === null || $this->minimumPhpVersionIdCompatibility >= PHP_84_VERSION_ID; $code = ''; if (!$php84MinimumCompatibility) { $code .= "#if (PHP_VERSION_ID >= " . PHP_84_VERSION_ID . ")\n"; } foreach ($this->framelessFunctionInfos as $framelessFunctionInfo) { $code .= "ZEND_FRAMELESS_FUNCTION({$this->name->getFunctionName()}, {$framelessFunctionInfo->arity});\n"; } $code .= 'static const zend_frameless_function_info ' . $this->getFramelessFunctionInfosName() . "[] = {\n"; foreach ($this->framelessFunctionInfos as $framelessFunctionInfo) { $code .= "\t{ ZEND_FRAMELESS_FUNCTION_NAME({$this->name->getFunctionName()}, {$framelessFunctionInfo->arity}), {$framelessFunctionInfo->arity} },\n"; } $code .= "\t{ 0 },\n"; $code .= "};\n"; if (!$php84MinimumCompatibility) { $code .= "#endif\n"; } return $code; } public function getFramelessFunctionInfosName(): string { return $this->name->getFramelessFunctionInfosName(); } public function getFunctionEntry(): string { $code = ""; $php84MinimumCompatibility = $this->minimumPhpVersionIdCompatibility === null || $this->minimumPhpVersionIdCompatibility >= PHP_84_VERSION_ID; $isVanillaEntry = $this->alias === null && !$this->supportsCompileTimeEval && $this->exposedDocComment === null && empty($this->framelessFunctionInfos); $argInfoName = $this->getArgInfoName(); $flagsByPhpVersions = $this->getArginfoFlagsByPhpVersions(); $functionEntryCode = null; if (!empty($this->framelessFunctionInfos)) { if ($this->isMethod()) { throw new Exception('Frameless methods are not supported yet'); } if ($this->name->getNamespace()) { throw new Exception('Namespaced direct calls to frameless functions are not supported yet'); } if ($this->alias) { throw new Exception('Aliased direct calls to frameless functions are not supported yet'); } } if ($this->isMethod()) { $zendName = '"' . $this->name->methodName . '"'; if ($this->alias) { if ($this->alias instanceof MethodName) { $name = "zim_" . $this->alias->getDeclarationClassName() . "_" . $this->alias->methodName; } else if ($this->alias instanceof FunctionName) { $name = "zif_" . $this->alias->getNonNamespacedName(); } else { throw new Error("Cannot happen"); } } else { if ($this->flags & Modifiers::ABSTRACT) { $name = "NULL"; } else { $name = "zim_" . $this->name->getDeclarationClassName() . "_" . $this->name->methodName; if ($isVanillaEntry) { $functionEntryCode = "\tZEND_ME(" . $this->name->getDeclarationClassName() . ", " . $this->name->methodName . ", $argInfoName, " . implode("|", reset($flagsByPhpVersions)) . ")"; } } } } else if ($this->name instanceof FunctionName) { $functionName = $this->name->getFunctionName(); $declarationName = $this->alias ? $this->alias->getNonNamespacedName() : $this->name->getDeclarationName(); if ($this->name->getNamespace()) { $namespace = addslashes($this->name->getNamespace()); $zendName = "ZEND_NS_NAME(\"$namespace\", \"$functionName\")"; $name = "zif_$declarationName"; } else { $zendName = '"' . $functionName . '"'; $name = "zif_$declarationName"; if ($isVanillaEntry && reset($flagsByPhpVersions) === ["0"]) { $functionEntryCode = "\tZEND_FE($declarationName, $argInfoName)"; } } } else { throw new Error("Cannot happen"); } if ($functionEntryCode !== null) { $code .= "$functionEntryCode\n"; } else { if (!$php84MinimumCompatibility) { $code .= "#if (PHP_VERSION_ID >= " . PHP_84_VERSION_ID . ")\n"; } $php84AndAboveFlags = array_slice($flagsByPhpVersions, 5, null, true); $docComment = $this->exposedDocComment ? '"' . $this->exposedDocComment->escape() . '"' : "NULL"; $framelessFuncInfosName = !empty($this->framelessFunctionInfos) ? $this->getFramelessFunctionInfosName() : "NULL"; $template = "\tZEND_RAW_FENTRY($zendName, $name, $argInfoName, %s, $framelessFuncInfosName, $docComment)\n"; $flagsCode = generateVersionDependentFlagCode( $template, $php84AndAboveFlags, PHP_84_VERSION_ID ); $code .= implode("", $flagsCode); if (!$php84MinimumCompatibility) { $code .= "#else\n"; } if (!$php84MinimumCompatibility) { $flags = array_slice($flagsByPhpVersions, 0, 4, true); $template = "\tZEND_RAW_FENTRY($zendName, $name, $argInfoName, %s)\n"; $flagsCode = generateVersionDependentFlagCode( $template, $flags, $this->minimumPhpVersionIdCompatibility ); $code .= implode("", $flagsCode); } if (!$php84MinimumCompatibility) { $code .= "#endif\n"; } } return $code; } public function getOptimizerInfo(): ?string { if ($this->isMethod()) { return null; } if ($this->alias !== null) { return null; } if ($this->return->refcount !== ReturnInfo::REFCOUNT_1 && $this->return->phpDocType === null) { return null; } $type = $this->return->phpDocType ?? $this->return->type; if ($type === null) { return null; } return "\tF" . $this->return->refcount . '("' . addslashes($this->name->__toString()) . '", ' . $type->toOptimizerTypeMask() . "),\n"; } public function discardInfoForOldPhpVersions(?int $minimumPhpVersionIdCompatibility): void { $this->attributes = []; $this->return->type = null; $this->framelessFunctionInfos = []; $this->exposedDocComment = null; $this->supportsCompileTimeEval = false; foreach ($this->args as $arg) { $arg->type = null; $arg->defaultValue = null; $arg->attributes = []; } $this->minimumPhpVersionIdCompatibility = $minimumPhpVersionIdCompatibility; } /** @return array */ private function getArginfoFlagsByPhpVersions(): array { $flags = []; if ($this->isMethod()) { if ($this->flags & Modifiers::PROTECTED) { $flags[] = "ZEND_ACC_PROTECTED"; } elseif ($this->flags & Modifiers::PRIVATE) { $flags[] = "ZEND_ACC_PRIVATE"; } else { $flags[] = "ZEND_ACC_PUBLIC"; } if ($this->flags & Modifiers::STATIC) { $flags[] = "ZEND_ACC_STATIC"; } if ($this->flags & Modifiers::FINAL) { $flags[] = "ZEND_ACC_FINAL"; } if ($this->flags & Modifiers::ABSTRACT) { $flags[] = "ZEND_ACC_ABSTRACT"; } } if ($this->isDeprecated) { $flags[] = "ZEND_ACC_DEPRECATED"; } foreach ($this->attributes as $attr) { if ($attr->class === "Deprecated") { $flags[] = "ZEND_ACC_DEPRECATED"; break; } } $php82AndAboveFlags = $flags; if ($this->isMethod() === false && $this->supportsCompileTimeEval) { $php82AndAboveFlags[] = "ZEND_ACC_COMPILE_TIME_EVAL"; } if (empty($flags)) { $flags[] = "0"; } if (empty($php82AndAboveFlags)) { $php82AndAboveFlags[] = "0"; } return [ PHP_70_VERSION_ID => $flags, PHP_80_VERSION_ID => $flags, PHP_81_VERSION_ID => $flags, PHP_82_VERSION_ID => $php82AndAboveFlags, PHP_83_VERSION_ID => $php82AndAboveFlags, PHP_84_VERSION_ID => $php82AndAboveFlags, ]; } private function generateRefSect1(DOMDocument $doc, string $role): DOMElement { $refSec = $doc->createElement('refsect1'); $refSec->setAttribute('role', $role); $refSec->append( "\n ", $doc->createEntityReference('reftitle.' . $role), "\n " ); return $refSec; } /** * @param array $funcMap * @param array $aliasMap * @throws Exception */ public function getMethodSynopsisDocument(array $funcMap, array $aliasMap): ?string { $REFSEC1_SEPERATOR = "\n\n "; $doc = new DOMDocument("1.0", "utf-8"); $doc->formatOutput = true; $refentry = $doc->createElement('refentry'); $doc->appendChild($refentry); if ($this->isMethod()) { assert($this->name instanceof MethodName); /* Namespaces are seperated by '-', '_' must be converted to '-' too. * Trim away the __ for magic methods */ $id = strtolower( str_replace('\\', '-', $this->name->className->__toString()) . '.' . str_replace('_', '-', ltrim($this->name->methodName, '_')) ); } else { $id = 'function.' . strtolower(str_replace('_', '-', $this->name->__toString())); } $refentry->setAttribute("xml:id", $id); /* We create an attribute for xmlns, as libxml otherwise force it to be the first one */ //$refentry->setAttribute("xmlns", "http://docbook.org/ns/docbook"); $namespace = $doc->createAttribute('xmlns'); $namespace->value = "http://docbook.org/ns/docbook"; $refentry->setAttributeNode($namespace); $refentry->setAttribute("xmlns:xlink", "http://www.w3.org/1999/xlink"); $refentry->appendChild(new DOMText("\n ")); /* Creation of */ $refnamediv = $doc->createElement('refnamediv'); $refnamediv->appendChild(new DOMText("\n ")); $refname = $doc->createElement('refname', $this->name->__toString()); $refnamediv->appendChild($refname); $refnamediv->appendChild(new DOMText("\n ")); $refpurpose = $doc->createElement('refpurpose', 'Description'); $refnamediv->appendChild($refpurpose); $refnamediv->appendChild(new DOMText("\n ")); $refentry->append($refnamediv, $REFSEC1_SEPERATOR); /* Creation of */ $descriptionRefSec = $this->generateRefSect1($doc, 'description'); $methodSynopsis = $this->getMethodSynopsisElement($funcMap, $aliasMap, $doc); if (!$methodSynopsis) { return null; } $descriptionRefSec->appendChild($methodSynopsis); $descriptionRefSec->appendChild(new DOMText("\n ")); $undocumentedEntity = $doc->createEntityReference('warn.undocumented.func'); $descriptionRefSec->appendChild($undocumentedEntity); $descriptionRefSec->appendChild(new DOMText("\n ")); $returnDescriptionPara = $doc->createElement('simpara'); $returnDescriptionPara->appendChild(new DOMText("\n Description.\n ")); $descriptionRefSec->appendChild($returnDescriptionPara); $descriptionRefSec->appendChild(new DOMText("\n ")); $refentry->append($descriptionRefSec, $REFSEC1_SEPERATOR); /* Creation of */ $parametersRefSec = $this->getParameterSection($doc); $refentry->append($parametersRefSec, $REFSEC1_SEPERATOR); /* Creation of */ if (!$this->name->isConstructor() && !$this->name->isDestructor()) { $returnRefSec = $this->getReturnValueSection($doc); $refentry->append($returnRefSec, $REFSEC1_SEPERATOR); } /* Creation of */ $errorsRefSec = $this->generateRefSect1($doc, 'errors'); $errorsDescriptionParaConstantTag = $doc->createElement('constant'); $errorsDescriptionParaConstantTag->append('E_*'); $errorsDescriptionParaExceptionTag = $doc->createElement('exceptionname'); $errorsDescriptionParaExceptionTag->append('Exception'); $errorsDescriptionPara = $doc->createElement('simpara'); $errorsDescriptionPara->append( "\n When does this function issue ", $errorsDescriptionParaConstantTag, " level errors,\n and/or throw ", $errorsDescriptionParaExceptionTag, "s.\n " ); $errorsRefSec->appendChild($errorsDescriptionPara); $errorsRefSec->appendChild(new DOMText("\n ")); $refentry->append($errorsRefSec, $REFSEC1_SEPERATOR); /* Creation of */ $changelogRefSec = $this->getChangelogSection($doc); $refentry->append($changelogRefSec, $REFSEC1_SEPERATOR); $exampleRefSec = $this->getExampleSection($doc, $id); $refentry->append($exampleRefSec, $REFSEC1_SEPERATOR); /* Creation of */ $notesRefSec = $this->generateRefSect1($doc, 'notes'); $noteTagSimara = $doc->createElement('simpara'); $noteTagSimara->append( "\n Any notes that don't fit anywhere else should go here.\n " ); $noteTag = $doc->createElement('note'); $noteTag->append("\n ", $noteTagSimara, "\n "); $notesRefSec->append($noteTag, "\n "); $refentry->append($notesRefSec, $REFSEC1_SEPERATOR); /* Creation of */ $seeAlsoRefSec = $this->generateRefSect1($doc, 'seealso'); $seeAlsoMemberClassMethod = $doc->createElement('member'); $seeAlsoMemberClassMethodTag = $doc->createElement('methodname'); $seeAlsoMemberClassMethodTag->appendChild(new DOMText("ClassName::otherMethodName")); $seeAlsoMemberClassMethod->appendChild($seeAlsoMemberClassMethodTag); $seeAlsoMemberFunction = $doc->createElement('member'); $seeAlsoMemberFunctionTag = $doc->createElement('function'); $seeAlsoMemberFunctionTag->appendChild(new DOMText("some_function")); $seeAlsoMemberFunction->appendChild($seeAlsoMemberFunctionTag); $seeAlsoMemberLink = $doc->createElement('member'); $seeAlsoMemberLinkTag = $doc->createElement('link'); $seeAlsoMemberLinkTag->setAttribute('linkend', 'some.id.chunk.to.link'); $seeAlsoMemberLinkTag->appendChild(new DOMText('something appendix')); $seeAlsoMemberLink->appendChild($seeAlsoMemberLinkTag); $seeAlsoList = $doc->createElement('simplelist'); $seeAlsoList->append( "\n ", $seeAlsoMemberClassMethod, "\n ", $seeAlsoMemberFunction, "\n ", $seeAlsoMemberLink, "\n " ); $seeAlsoRefSec->appendChild($seeAlsoList); $seeAlsoRefSec->appendChild(new DOMText("\n ")); $refentry->appendChild($seeAlsoRefSec); $refentry->appendChild(new DOMText("\n\n")); $doc->appendChild(new DOMComment( <<saveXML(); } private function getParameterSection(DOMDocument $doc): DOMElement { $parametersRefSec = $this->generateRefSect1($doc, 'parameters'); if (empty($this->args)) { $noParamEntity = $doc->createEntityReference('no.function.parameters'); $parametersRefSec->appendChild($noParamEntity); return $parametersRefSec; } else { $parametersContainer = $doc->createDocumentFragment(); $parametersContainer->appendChild(new DOMText("\n ")); $parametersList = $doc->createElement('variablelist'); $parametersContainer->appendChild($parametersList); /* name Description. */ foreach ($this->args as $arg) { $parameter = $doc->createElement('parameter', $arg->name); $parameterTerm = $doc->createElement('term'); $parameterTerm->appendChild($parameter); $listItemPara = $doc->createElement('simpara'); $listItemPara->append( "\n ", "Description.", "\n ", ); $parameterEntryListItem = $doc->createElement('listitem'); $parameterEntryListItem->append( "\n ", $listItemPara, "\n ", ); $parameterEntry = $doc->createElement('varlistentry'); $parameterEntry->append( "\n ", $parameterTerm, "\n ", $parameterEntryListItem, "\n ", ); $parametersList->appendChild(new DOMText("\n ")); $parametersList->appendChild($parameterEntry); } $parametersList->appendChild(new DOMText("\n ")); } $parametersContainer->appendChild(new DOMText("\n ")); $parametersRefSec->appendChild($parametersContainer); $parametersRefSec->appendChild(new DOMText("\n ")); return $parametersRefSec; } private function getReturnValueSection(DOMDocument $doc): DOMElement { $returnRefSec = $this->generateRefSect1($doc, 'returnvalues'); $returnDescriptionPara = $doc->createElement('simpara'); $returnDescriptionPara->appendChild(new DOMText("\n ")); $returnType = $this->return->getMethodSynopsisType(); if ($returnType === null) { $returnDescriptionPara->appendChild(new DOMText("Description.")); } else if (count($returnType->types) === 1) { $type = $returnType->types[0]; $name = $type->name; switch ($name) { case 'void': $descriptionNode = $doc->createEntityReference('return.void'); break; case 'true': $descriptionNode = $doc->createEntityReference('return.true.always'); break; case 'bool': $descriptionNode = $doc->createEntityReference('return.success'); break; default: $descriptionNode = new DOMText("Description."); break; } $returnDescriptionPara->appendChild($descriptionNode); } else { $returnDescriptionPara->appendChild(new DOMText("Description.")); } $returnDescriptionPara->appendChild(new DOMText("\n ")); $returnRefSec->appendChild($returnDescriptionPara); $returnRefSec->appendChild(new DOMText("\n ")); return $returnRefSec; } /** * @param array $headers [count($headers) === $columns] * @param array> $rows [count($rows[$i]) === $columns] */ private function generateDocbookInformalTable( DOMDocument $doc, int $indent, int $columns, array $headers, array $rows ): DOMElement { $strIndent = str_repeat(' ', $indent); $headerRow = $doc->createElement('row'); foreach ($headers as $header) { $headerEntry = $doc->createElement('entry'); $headerEntry->appendChild($header); $headerRow->append("\n$strIndent ", $headerEntry); } $headerRow->append("\n$strIndent "); $thead = $doc->createElement('thead'); $thead->append( "\n$strIndent ", $headerRow, "\n$strIndent ", ); $tbody = $doc->createElement('tbody'); foreach ($rows as $row) { $bodyRow = $doc->createElement('row'); foreach ($row as $cell) { $entry = $doc->createElement('entry'); $entry->appendChild($cell); $bodyRow->appendChild(new DOMText("\n$strIndent ")); $bodyRow->appendChild($entry); } $bodyRow->appendChild(new DOMText("\n$strIndent ")); $tbody->append( "\n$strIndent ", $bodyRow, "\n$strIndent ", ); } $tgroup = $doc->createElement('tgroup'); $tgroup->setAttribute('cols', (string) $columns); $tgroup->append( "\n$strIndent ", $thead, "\n$strIndent ", $tbody, "\n$strIndent ", ); $table = $doc->createElement('informaltable'); $table->append( "\n$strIndent ", $tgroup, "\n$strIndent", ); return $table; } private function getChangelogSection(DOMDocument $doc): DOMElement { $refSec = $this->generateRefSect1($doc, 'changelog'); $headers = [ $doc->createEntityReference('Version'), $doc->createEntityReference('Description'), ]; $rows = [[ new DOMText('8.X.0'), new DOMText("\n Description\n "), ]]; $table = $this->generateDocbookInformalTable( $doc, /* indent: */ 2, /* columns: */ 2, /* headers: */ $headers, /* rows: */ $rows ); $refSec->appendChild($table); $refSec->appendChild(new DOMText("\n ")); return $refSec; } private function getExampleSection(DOMDocument $doc, string $id): DOMElement { $refSec = $this->generateRefSect1($doc, 'examples'); $example = $doc->createElement('example'); $fnName = $this->name->__toString(); $example->setAttribute('xml:id', $id . '.example.basic'); $title = $doc->createElement('title'); $fn = $doc->createElement($this->isMethod() ? 'methodname' : 'function'); $fn->append($fnName); $title->append($fn, ' example'); $example->append("\n ", $title); $para = $doc->createElement('simpara'); $para->append("\n ", "Description.", "\n "); $example->append("\n ", $para); $prog = $doc->createElement('programlisting'); $prog->setAttribute('role', 'php'); $code = new DOMCdataSection( << CODE_EXAMPLE ); $prog->append("\n"); $prog->appendChild($code); $prog->append("\n "); $example->append("\n ", $prog); $example->append("\n ", $doc->createEntityReference('example.outputs')); $output = new DOMCdataSection( <<createElement('screen'); $screen->append("\n"); $screen->appendChild($output); $screen->append("\n "); $example->append( "\n ", $screen, "\n ", ); $refSec->append( $example, "\n ", ); return $refSec; } /** * @param array $funcMap * @param array $aliasMap * @throws Exception */ public function getMethodSynopsisElement(array $funcMap, array $aliasMap, DOMDocument $doc): ?DOMElement { if ($this->hasParamWithUnknownDefaultValue()) { return null; } if ($this->name->isConstructor()) { $synopsisType = "constructorsynopsis"; } elseif ($this->name->isDestructor()) { $synopsisType = "destructorsynopsis"; } else { $synopsisType = "methodsynopsis"; } $methodSynopsis = $doc->createElement($synopsisType); if ($this->isMethod()) { assert($this->name instanceof MethodName); $role = $doc->createAttribute("role"); $role->value = addslashes($this->name->className->__toString()); $methodSynopsis->appendChild($role); } $methodSynopsis->appendChild(new DOMText("\n ")); foreach ($this->attributes as $attribute) { $modifier = $doc->createElement("modifier", "#[\\" . $attribute->class . "]"); $modifier->setAttribute("role", "attribute"); $methodSynopsis->appendChild($modifier); $methodSynopsis->appendChild(new DOMText("\n ")); } foreach ($this->getModifierNames() as $modifierString) { $modifierElement = $doc->createElement('modifier', $modifierString); $methodSynopsis->appendChild($modifierElement); $methodSynopsis->appendChild(new DOMText(" ")); } $returnType = $this->return->getMethodSynopsisType(); if ($returnType) { $methodSynopsis->appendChild($returnType->getTypeForDoc($doc)); } $methodname = $doc->createElement('methodname', $this->name->__toString()); $methodSynopsis->appendChild($methodname); if (empty($this->args)) { $methodSynopsis->appendChild(new DOMText("\n ")); $void = $doc->createElement('void'); $methodSynopsis->appendChild($void); } else { foreach ($this->args as $arg) { $methodSynopsis->appendChild(new DOMText("\n ")); $methodparam = $doc->createElement('methodparam'); if ($arg->defaultValue !== null) { $methodparam->setAttribute("choice", "opt"); } if ($arg->isVariadic) { $methodparam->setAttribute("rep", "repeat"); } $methodSynopsis->appendChild($methodparam); foreach ($arg->attributes as $attribute) { $attribute = $doc->createElement("modifier", "#[\\" . $attribute->class . "]"); $attribute->setAttribute("role", "attribute"); $methodparam->appendChild($attribute); } $methodparam->appendChild($arg->getMethodSynopsisType()->getTypeForDoc($doc)); $parameter = $doc->createElement('parameter', $arg->name); if ($arg->sendBy !== ArgInfo::SEND_BY_VAL) { $parameter->setAttribute("role", "reference"); } $methodparam->appendChild($parameter); $defaultValue = $arg->getDefaultValueAsMethodSynopsisString(); if ($defaultValue !== null) { $initializer = $doc->createElement('initializer'); if (preg_match('/^[a-zA-Z_][a-zA-Z_0-9]*$/', $defaultValue)) { $constant = $doc->createElement('constant', $defaultValue); $initializer->appendChild($constant); } else { $initializer->nodeValue = $defaultValue; } $methodparam->appendChild($initializer); } } } $methodSynopsis->appendChild(new DOMText("\n ")); return $methodSynopsis; } public function __clone() { foreach ($this->args as $key => $argInfo) { $this->args[$key] = clone $argInfo; } $this->return = clone $this->return; foreach ($this->attributes as $key => $attribute) { $this->attributes[$key] = clone $attribute; } foreach ($this->framelessFunctionInfos as $key => $framelessFunctionInfo) { $this->framelessFunctionInfos[$key] = clone $framelessFunctionInfo; } if ($this->exposedDocComment) { $this->exposedDocComment = clone $this->exposedDocComment; } } } class EvaluatedValue { /** @var mixed */ public $value; public SimpleType $type; public Expr $expr; public bool $isUnknownConstValue; /** @var ConstInfo[] */ public array $originatingConsts; /** * @param array $allConstInfos */ public static function createFromExpression(Expr $expr, ?SimpleType $constType, ?string $cConstName, array $allConstInfos): EvaluatedValue { // This visitor replaces the PHP constants by C constants. It allows direct expansion of the compiled constants, e.g. later in the pretty printer. $visitor = new class($allConstInfos) extends PhpParser\NodeVisitorAbstract { /** @var iterable */ public array $visitedConstants = []; /** @var array */ public array $allConstInfos; /** @param array $allConstInfos */ public function __construct(array $allConstInfos) { $this->allConstInfos = $allConstInfos; } /** @return Node|null */ public function enterNode(Node $expr) { if (!$expr instanceof Expr\ConstFetch && !$expr instanceof Expr\ClassConstFetch) { return null; } if ($expr instanceof Expr\ClassConstFetch) { $originatingConstName = new ClassConstName($expr->class, $expr->name->toString()); } else { $originatingConstName = new ConstName($expr->name->getAttribute('namespacedName'), $expr->name->toString()); } if ($originatingConstName->isUnknown()) { return null; } $const = $this->allConstInfos[$originatingConstName->__toString()] ?? null; if ($const !== null) { $this->visitedConstants[] = $const; return $const->getValue($this->allConstInfos)->expr; } } }; $nodeTraverser = new PhpParser\NodeTraverser; $nodeTraverser->addVisitor($visitor); $expr = $nodeTraverser->traverse([$expr])[0]; $isUnknownConstValue = false; $evaluator = new ConstExprEvaluator( static function (Expr $expr) use ($allConstInfos, &$isUnknownConstValue) { // $expr is a ConstFetch with a name of a C macro here if (!$expr instanceof Expr\ConstFetch) { throw new Exception($this->getVariableTypeName() . " " . $this->name->__toString() . " has an unsupported value"); } $constName = $expr->name->__toString(); if (strtolower($constName) === "unknown") { $isUnknownConstValue = true; return null; } foreach ($allConstInfos as $const) { if ($constName != $const->cValue) { continue; } $constType = ($const->phpDocType ?? $const->type)->tryToSimpleType(); if ($constType) { if ($constType->isBool()) { return true; } elseif ($constType->isInt()) { return 1; } elseif ($constType->isFloat()) { return M_PI; } elseif ($constType->isString()) { return $const->name; } elseif ($constType->isArray()) { return []; } } return null; } throw new Exception("Constant " . $constName . " cannot be found"); } ); $result = $evaluator->evaluateDirectly($expr); return new EvaluatedValue( $result, // note: we are generally not interested in the actual value of $result, unless it's a bare value, without constants $constType ?? SimpleType::fromValue($result), $cConstName === null ? $expr : new Expr\ConstFetch(new Node\Name($cConstName)), $visitor->visitedConstants, $isUnknownConstValue ); } public static function null(): EvaluatedValue { return new self(null, SimpleType::null(), new Expr\ConstFetch(new Node\Name('null')), [], false); } /** * @param mixed $value * @param ConstInfo[] $originatingConsts */ private function __construct($value, SimpleType $type, Expr $expr, array $originatingConsts, bool $isUnknownConstValue) { $this->value = $value; $this->type = $type; $this->expr = $expr; $this->originatingConsts = $originatingConsts; $this->isUnknownConstValue = $isUnknownConstValue; } public function initializeZval(string $zvalName): string { $cExpr = $this->getCExpr(); $code = "\tzval $zvalName;\n"; if ($this->type->isNull()) { $code .= "\tZVAL_NULL(&$zvalName);\n"; } elseif ($this->type->isBool()) { if ($cExpr == 'true') { $code .= "\tZVAL_TRUE(&$zvalName);\n"; } elseif ($cExpr == 'false') { $code .= "\tZVAL_FALSE(&$zvalName);\n"; } else { $code .= "\tZVAL_BOOL(&$zvalName, $cExpr);\n"; } } elseif ($this->type->isInt()) { $code .= "\tZVAL_LONG(&$zvalName, $cExpr);\n"; } elseif ($this->type->isFloat()) { $code .= "\tZVAL_DOUBLE(&$zvalName, $cExpr);\n"; } elseif ($this->type->isString()) { if ($cExpr === '""') { $code .= "\tZVAL_EMPTY_STRING(&$zvalName);\n"; } else { $code .= "\tzend_string *{$zvalName}_str = zend_string_init($cExpr, strlen($cExpr), 1);\n"; $code .= "\tZVAL_STR(&$zvalName, {$zvalName}_str);\n"; } } elseif ($this->type->isArray()) { if ($cExpr == '[]') { $code .= "\tZVAL_EMPTY_ARRAY(&$zvalName);\n"; } else { throw new Exception("Unimplemented default value"); } } else { throw new Exception("Invalid default value: " . print_r($this->value, true) . ", type: " . print_r($this->type, true)); } return $code; } public function getCExpr(): ?string { // $this->expr has all its PHP constants replaced by C constants $prettyPrinter = new Standard; $expr = $prettyPrinter->prettyPrintExpr($this->expr); // PHP single-quote to C double-quote string if ($this->type->isString()) { $expr = preg_replace("/(^'|'$)/", '"', $expr); } return $expr[0] == '"' ? $expr : preg_replace('(\bnull\b)', 'NULL', str_replace('\\', '', $expr)); } } abstract class VariableLike { public int $flags; public ?Type $type; public ?Type $phpDocType; public ?string $link; public ?int $phpVersionIdMinimumCompatibility; /** @var AttributeInfo[] */ public array $attributes; public ?ExposedDocComment $exposedDocComment; /** * @var AttributeInfo[] $attributes */ public function __construct( int $flags, ?Type $type, ?Type $phpDocType, ?string $link, ?int $phpVersionIdMinimumCompatibility, array $attributes, ?ExposedDocComment $exposedDocComment ) { $this->flags = $flags; $this->type = $type; $this->phpDocType = $phpDocType; $this->link = $link; $this->phpVersionIdMinimumCompatibility = $phpVersionIdMinimumCompatibility; $this->attributes = $attributes; $this->exposedDocComment = $exposedDocComment; } abstract protected function getVariableTypeCode(): string; abstract protected function getVariableTypeName(): string; abstract protected function getFieldSynopsisDefaultLinkend(): string; abstract protected function getFieldSynopsisName(): string; /** @param array $allConstInfos */ abstract protected function getFieldSynopsisValueString(array $allConstInfos): ?string; abstract public function discardInfoForOldPhpVersions(?int $minimumPhpVersionIdCompatibility): void; protected function addTypeToFieldSynopsis(DOMDocument $doc, DOMElement $fieldsynopsisElement): void { $type = $this->phpDocType ?? $this->type; if ($type) { $fieldsynopsisElement->appendChild(new DOMText("\n ")); $fieldsynopsisElement->appendChild($type->getTypeForDoc($doc)); } } /** * @return array */ protected function getFlagsByPhpVersion(): array { $flags = "ZEND_ACC_PUBLIC"; if ($this->flags & Modifiers::PROTECTED) { $flags = "ZEND_ACC_PROTECTED"; } elseif ($this->flags & Modifiers::PRIVATE) { $flags = "ZEND_ACC_PRIVATE"; } return [ PHP_70_VERSION_ID => [$flags], PHP_80_VERSION_ID => [$flags], PHP_81_VERSION_ID => [$flags], PHP_82_VERSION_ID => [$flags], PHP_83_VERSION_ID => [$flags], PHP_84_VERSION_ID => [$flags], ]; } protected function getTypeCode(string $variableLikeName, string &$code): string { $variableLikeType = $this->getVariableTypeName(); $typeCode = ""; if ($this->type) { $arginfoType = $this->type->toArginfoType(); if ($arginfoType->hasClassType()) { if (count($arginfoType->classTypes) >= 2) { foreach ($arginfoType->classTypes as $classType) { $escapedClassName = $classType->toEscapedName(); $varEscapedClassName = $classType->toVarEscapedName(); $code .= "\tzend_string *{$variableLikeType}_{$variableLikeName}_class_{$varEscapedClassName} = zend_string_init(\"{$escapedClassName}\", sizeof(\"{$escapedClassName}\") - 1, 1);\n"; } $classTypeCount = count($arginfoType->classTypes); $code .= "\tzend_type_list *{$variableLikeType}_{$variableLikeName}_type_list = malloc(ZEND_TYPE_LIST_SIZE($classTypeCount));\n"; $code .= "\t{$variableLikeType}_{$variableLikeName}_type_list->num_types = $classTypeCount;\n"; foreach ($arginfoType->classTypes as $k => $classType) { $escapedClassName = $classType->toEscapedName(); $code .= "\t{$variableLikeType}_{$variableLikeName}_type_list->types[$k] = (zend_type) ZEND_TYPE_INIT_CLASS({$variableLikeType}_{$variableLikeName}_class_{$escapedClassName}, 0, 0);\n"; } $typeMaskCode = $this->type->toArginfoType()->toTypeMask(); if ($this->type->isIntersection) { $code .= "\tzend_type {$variableLikeType}_{$variableLikeName}_type = ZEND_TYPE_INIT_INTERSECTION({$variableLikeType}_{$variableLikeName}_type_list, $typeMaskCode);\n"; } else { $code .= "\tzend_type {$variableLikeType}_{$variableLikeName}_type = ZEND_TYPE_INIT_UNION({$variableLikeType}_{$variableLikeName}_type_list, $typeMaskCode);\n"; } $typeCode = "{$variableLikeType}_{$variableLikeName}_type"; } else { $escapedClassName = $arginfoType->classTypes[0]->toEscapedName(); $varEscapedClassName = $arginfoType->classTypes[0]->toVarEscapedName(); $code .= "\tzend_string *{$variableLikeType}_{$variableLikeName}_class_{$varEscapedClassName} = zend_string_init(\"{$escapedClassName}\", sizeof(\"{$escapedClassName}\")-1, 1);\n"; $typeCode = "(zend_type) ZEND_TYPE_INIT_CLASS({$variableLikeType}_{$variableLikeName}_class_{$varEscapedClassName}, 0, " . $arginfoType->toTypeMask() . ")"; } } else { $typeCode = "(zend_type) ZEND_TYPE_INIT_MASK(" . $arginfoType->toTypeMask() . ")"; } } else { $typeCode = "(zend_type) ZEND_TYPE_INIT_NONE(0)"; } return $typeCode; } /** @param array $allConstInfos */ public function getFieldSynopsisElement(DOMDocument $doc, array $allConstInfos): DOMElement { $fieldsynopsisElement = $doc->createElement("fieldsynopsis"); $this->addModifiersToFieldSynopsis($doc, $fieldsynopsisElement); $this->addTypeToFieldSynopsis($doc, $fieldsynopsisElement); $varnameElement = $doc->createElement("varname", $this->getFieldSynopsisName()); if ($this->link) { $varnameElement->setAttribute("linkend", $this->link); } else { $varnameElement->setAttribute("linkend", $this->getFieldSynopsisDefaultLinkend()); } $fieldsynopsisElement->appendChild(new DOMText("\n ")); $fieldsynopsisElement->appendChild($varnameElement); $valueString = $this->getFieldSynopsisValueString($allConstInfos); if ($valueString) { $fieldsynopsisElement->appendChild(new DOMText("\n ")); $initializerElement = $doc->createElement("initializer", $valueString); $fieldsynopsisElement->appendChild($initializerElement); } $fieldsynopsisElement->appendChild(new DOMText("\n ")); return $fieldsynopsisElement; } protected function addModifiersToFieldSynopsis(DOMDocument $doc, DOMElement $fieldsynopsisElement): void { if ($this->flags & Modifiers::PUBLIC) { $fieldsynopsisElement->appendChild(new DOMText("\n ")); $fieldsynopsisElement->appendChild($doc->createElement("modifier", "public")); } elseif ($this->flags & Modifiers::PROTECTED) { $fieldsynopsisElement->appendChild(new DOMText("\n ")); $fieldsynopsisElement->appendChild($doc->createElement("modifier", "protected")); } elseif ($this->flags & Modifiers::PRIVATE) { $fieldsynopsisElement->appendChild(new DOMText("\n ")); $fieldsynopsisElement->appendChild($doc->createElement("modifier", "private")); } } /** * @param array $flags * @return array */ protected function addFlagForVersionsAbove(array $flags, string $flag, int $minimumVersionId): array { $write = false; foreach ($flags as $version => $versionFlags) { if ($version === $minimumVersionId || $write === true) { $flags[$version][] = $flag; $write = true; } } return $flags; } } class ConstInfo extends VariableLike { public ConstOrClassConstName $name; public Expr $value; public bool $isDeprecated; public ?string $valueString; public ?string $cond; public ?string $cValue; public bool $isUndocumentable; public bool $isFileCacheAllowed; /** * @var AttributeInfo[] $attributes */ public function __construct( ConstOrClassConstName $name, int $flags, Expr $value, ?string $valueString, ?Type $type, ?Type $phpDocType, bool $isDeprecated, ?string $cond, ?string $cValue, bool $isUndocumentable, ?string $link, ?int $phpVersionIdMinimumCompatibility, array $attributes, ?ExposedDocComment $exposedDocComment, bool $isFileCacheAllowed ) { $this->name = $name; $this->value = $value; $this->valueString = $valueString; $this->isDeprecated = $isDeprecated; $this->cond = $cond; $this->cValue = $cValue; $this->isUndocumentable = $isUndocumentable; $this->isFileCacheAllowed = $isFileCacheAllowed; parent::__construct($flags, $type, $phpDocType, $link, $phpVersionIdMinimumCompatibility, $attributes, $exposedDocComment); } /** @param array $allConstInfos */ public function getValue(array $allConstInfos): EvaluatedValue { return EvaluatedValue::createFromExpression( $this->value, ($this->phpDocType ?? $this->type)->tryToSimpleType(), $this->cValue, $allConstInfos ); } protected function getVariableTypeName(): string { return "constant"; } protected function getVariableTypeCode(): string { return "const"; } protected function getFieldSynopsisDefaultLinkend(): string { $className = str_replace(["\\", "_"], ["-", "-"], $this->name->class->toLowerString()); return "$className.constants." . strtolower(str_replace("_", "-", trim($this->name->getDeclarationName(), "_"))); } protected function getFieldSynopsisName(): string { return $this->name->__toString(); } /** @param array $allConstInfos */ protected function getFieldSynopsisValueString(array $allConstInfos): ?string { $value = EvaluatedValue::createFromExpression($this->value, null, $this->cValue, $allConstInfos); if ($value->isUnknownConstValue) { return null; } if ($value->originatingConsts) { return implode("\n", array_map(function (ConstInfo $const) use ($allConstInfos) { return $const->getFieldSynopsisValueString($allConstInfos); }, $value->originatingConsts)); } return $this->valueString; } public function getPredefinedConstantTerm(DOMDocument $doc, int $indentationLevel): DOMElement { $indentation = str_repeat(" ", $indentationLevel); $termElement = $doc->createElement("term"); $constantElement = $doc->createElement("constant"); $constantElement->textContent = $this->name->__toString(); $typeElement = ($this->phpDocType ?? $this->type)->getTypeForDoc($doc); $termElement->appendChild(new DOMText("\n$indentation ")); $termElement->appendChild($constantElement); $termElement->appendChild(new DOMText("\n$indentation (")); $termElement->appendChild($typeElement); $termElement->appendChild(new DOMText(")\n$indentation")); return $termElement; } public function getPredefinedConstantEntry(DOMDocument $doc, int $indentationLevel): DOMElement { $indentation = str_repeat(" ", $indentationLevel); $entryElement = $doc->createElement("entry"); $constantElement = $doc->createElement("constant"); $constantElement->textContent = $this->name->__toString(); $typeElement = ($this->phpDocType ?? $this->type)->getTypeForDoc($doc); $entryElement->appendChild(new DOMText("\n$indentation ")); $entryElement->appendChild($constantElement); $entryElement->appendChild(new DOMText("\n$indentation (")); $entryElement->appendChild($typeElement); $entryElement->appendChild(new DOMText(")\n$indentation")); return $entryElement; } public function discardInfoForOldPhpVersions(?int $phpVersionIdMinimumCompatibility): void { $this->type = null; $this->flags &= ~Modifiers::FINAL; $this->isDeprecated = false; $this->attributes = []; $this->phpVersionIdMinimumCompatibility = $phpVersionIdMinimumCompatibility; } /** @param array $allConstInfos */ public function getDeclaration(array $allConstInfos): string { $type = $this->phpDocType ?? $this->type; $simpleType = $type ? $type->tryToSimpleType() : null; if ($simpleType && $simpleType->name === "mixed") { $simpleType = null; } $value = EvaluatedValue::createFromExpression($this->value, $simpleType, $this->cValue, $allConstInfos); if ($value->isUnknownConstValue && ($simpleType === null || !$simpleType->isBuiltin)) { throw new Exception("Constant " . $this->name->__toString() . " must have a built-in PHPDoc type as the type couldn't be inferred from its value"); } // i.e. const NAME = UNKNOWN;, without the annotation if ($value->isUnknownConstValue && $this->cValue === null && $value->expr instanceof Expr\ConstFetch && $value->expr->name->__toString() === "UNKNOWN") { throw new Exception("Constant " . $this->name->__toString() . " must have a @cvalue annotation"); } $code = ""; if ($this->cond) { $code .= "#if {$this->cond}\n"; } if ($this->name->isClassConst()) { $code .= $this->getClassConstDeclaration($value, $allConstInfos); } else { $code .= $this->getGlobalConstDeclaration($value, $allConstInfos); } $code .= $this->getValueAssertion($value); if ($this->cond) { $code .= "#endif\n"; } return $code; } /** @param array $allConstInfos */ private function getGlobalConstDeclaration(EvaluatedValue $value, array $allConstInfos): string { $constName = str_replace('\\', '\\\\', $this->name->__toString()); $constValue = $value->value; $cExpr = $value->getCExpr(); $flags = "CONST_PERSISTENT"; if (!$this->isFileCacheAllowed) { $flags .= " | CONST_NO_FILE_CACHE"; } if ($this->phpVersionIdMinimumCompatibility !== null && $this->phpVersionIdMinimumCompatibility < 80000) { $flags .= " | CONST_CS"; } if ($this->isDeprecated) { $flags .= " | CONST_DEPRECATED"; } if ($value->type->isNull()) { return "\tREGISTER_NULL_CONSTANT(\"$constName\", $flags);\n"; } if ($value->type->isBool()) { return "\tREGISTER_BOOL_CONSTANT(\"$constName\", " . ($cExpr ?: ($constValue ? "true" : "false")) . ", $flags);\n"; } if ($value->type->isInt()) { return "\tREGISTER_LONG_CONSTANT(\"$constName\", " . ($cExpr ?: (int) $constValue) . ", $flags);\n"; } if ($value->type->isFloat()) { return "\tREGISTER_DOUBLE_CONSTANT(\"$constName\", " . ($cExpr ?: (float) $constValue) . ", $flags);\n"; } if ($value->type->isString()) { return "\tREGISTER_STRING_CONSTANT(\"$constName\", " . ($cExpr ?: '"' . addslashes($constValue) . '"') . ", $flags);\n"; } throw new Exception("Unimplemented constant type"); } /** @param array $allConstInfos */ private function getClassConstDeclaration(EvaluatedValue $value, array $allConstInfos): string { $constName = $this->name->getDeclarationName(); $zvalCode = $value->initializeZval("const_{$constName}_value", $allConstInfos); $code = "\n" . $zvalCode; $code .= "\tzend_string *const_{$constName}_name = zend_string_init_interned(\"$constName\", sizeof(\"$constName\") - 1, 1);\n"; $nameCode = "const_{$constName}_name"; if ($this->exposedDocComment) { $commentCode = "const_{$constName}_comment"; $escapedComment = $this->exposedDocComment->escape(); $escapedCommentLength = $this->exposedDocComment->getLength(); $code .= "\tzend_string *$commentCode = zend_string_init_interned(\"$escapedComment\", $escapedCommentLength, 1);\n"; } else { $commentCode = "NULL"; } $php83MinimumCompatibility = $this->phpVersionIdMinimumCompatibility === null || $this->phpVersionIdMinimumCompatibility >= PHP_83_VERSION_ID; if ($this->type && !$php83MinimumCompatibility) { $code .= "#if (PHP_VERSION_ID >= " . PHP_83_VERSION_ID . ")\n"; } if ($this->type) { $typeCode = $this->getTypeCode($constName, $code); if (!empty($this->attributes)) { $template = "\tzend_class_constant *const_" . $this->name->getDeclarationName() . " = "; } else { $template = "\t"; } $template .= "zend_declare_typed_class_constant(class_entry, $nameCode, &const_{$constName}_value, %s, $commentCode, $typeCode);\n"; $flagsCode = generateVersionDependentFlagCode( $template, $this->getFlagsByPhpVersion(), $this->phpVersionIdMinimumCompatibility ); $code .= implode("", $flagsCode); } if ($this->type && !$php83MinimumCompatibility) { $code .= "#else\n"; } if (!$this->type || !$php83MinimumCompatibility) { if (!empty($this->attributes)) { $template = "\tzend_class_constant *const_" . $this->name->getDeclarationName() . " = "; } else { $template = "\t"; } $template .= "zend_declare_class_constant_ex(class_entry, $nameCode, &const_{$constName}_value, %s, $commentCode);\n"; $flagsCode = generateVersionDependentFlagCode( $template, $this->getFlagsByPhpVersion(), $this->phpVersionIdMinimumCompatibility ); $code .= implode("", $flagsCode); } if ($this->type && !$php83MinimumCompatibility) { $code .= "#endif\n"; } $code .= "\tzend_string_release(const_{$constName}_name);\n"; return $code; } private function getValueAssertion(EvaluatedValue $value): string { if ($value->isUnknownConstValue || $value->originatingConsts || $this->cValue === null) { return ""; } $cExpr = $value->getCExpr(); $constValue = $value->value; if ($value->type->isNull()) { return "\tZEND_ASSERT($cExpr == NULL);\n"; } if ($value->type->isBool()) { $cValue = $constValue ? "true" : "false"; return "\tZEND_ASSERT($cExpr == $cValue);\n"; } if ($value->type->isInt()) { $cValue = (int) $constValue; return "\tZEND_ASSERT($cExpr == $cValue);\n"; } if ($value->type->isFloat()) { $cValue = (float) $constValue; return "\tZEND_ASSERT($cExpr == $cValue);\n"; } if ($value->type->isString()) { $cValue = '"' . addslashes($constValue) . '"'; return "\tZEND_ASSERT(strcmp($cExpr, $cValue) == 0);\n"; } throw new Exception("Unimplemented constant type"); } /** * @return array */ protected function getFlagsByPhpVersion(): array { $flags = parent::getFlagsByPhpVersion(); if ($this->isDeprecated) { $flags = $this->addFlagForVersionsAbove($flags, "ZEND_ACC_DEPRECATED", PHP_80_VERSION_ID); } foreach ($this->attributes as $attr) { if ($attr->class === "Deprecated") { $flags = $this->addFlagForVersionsAbove($flags, "ZEND_ACC_DEPRECATED", PHP_80_VERSION_ID); break; } } if ($this->flags & Modifiers::FINAL) { $flags = $this->addFlagForVersionsAbove($flags, "ZEND_ACC_FINAL", PHP_81_VERSION_ID); } return $flags; } protected function addModifiersToFieldSynopsis(DOMDocument $doc, DOMElement $fieldsynopsisElement): void { parent::addModifiersToFieldSynopsis($doc, $fieldsynopsisElement); if ($this->flags & Modifiers::FINAL) { $fieldsynopsisElement->appendChild(new DOMText("\n ")); $fieldsynopsisElement->appendChild($doc->createElement("modifier", "final")); } $fieldsynopsisElement->appendChild(new DOMText("\n ")); $fieldsynopsisElement->appendChild($doc->createElement("modifier", "const")); } } class PropertyInfo extends VariableLike { public int $classFlags; public PropertyName $name; public ?Expr $defaultValue; public ?string $defaultValueString; public bool $isDocReadonly; public bool $isVirtual; // Map possible variable names to the known string constant, see // ZEND_KNOWN_STRINGS private const PHP_80_KNOWN = [ "file" => "ZEND_STR_FILE", "line" => "ZEND_STR_LINE", "function" => "ZEND_STR_FUNCTION", "class" => "ZEND_STR_CLASS", "object" => "ZEND_STR_OBJECT", "type" => "ZEND_STR_TYPE", // ZEND_STR_OBJECT_OPERATOR and ZEND_STR_PAAMAYIM_NEKUDOTAYIM are // not valid variable names "args" => "ZEND_STR_ARGS", "unknown" => "ZEND_STR_UNKNOWN", "eval" => "ZEND_STR_EVAL", "include" => "ZEND_STR_INCLUDE", "require" => "ZEND_STR_REQUIRE", "include_once" => "ZEND_STR_INCLUDE_ONCE", "require_once" => "ZEND_STR_REQUIRE_ONCE", "scalar" => "ZEND_STR_SCALAR", "error_reporting" => "ZEND_STR_ERROR_REPORTING", "static" => "ZEND_STR_STATIC", // ZEND_STR_THIS cannot be used since $this cannot be reassigned "value" => "ZEND_STR_VALUE", "key" => "ZEND_STR_KEY", "__invoke" => "ZEND_STR_MAGIC_INVOKE", "previous" => "ZEND_STR_PREVIOUS", "code" => "ZEND_STR_CODE", "message" => "ZEND_STR_MESSAGE", "severity" => "ZEND_STR_SEVERITY", "string" => "ZEND_STR_STRING", "trace" => "ZEND_STR_TRACE", "scheme" => "ZEND_STR_SCHEME", "host" => "ZEND_STR_HOST", "port" => "ZEND_STR_PORT", "user" => "ZEND_STR_USER", "pass" => "ZEND_STR_PASS", "path" => "ZEND_STR_PATH", "query" => "ZEND_STR_QUERY", "fragment" => "ZEND_STR_FRAGMENT", "NULL" => "ZEND_STR_NULL", "boolean" => "ZEND_STR_BOOLEAN", "integer" => "ZEND_STR_INTEGER", "double" => "ZEND_STR_DOUBLE", "array" => "ZEND_STR_ARRAY", "resource" => "ZEND_STR_RESOURCE", // ZEND_STR_CLOSED_RESOURCE has a space in it "name" => "ZEND_STR_NAME", // ZEND_STR_ARGV and ZEND_STR_ARGC are superglobals that wouldn't be // variable names "Array" => "ZEND_STR_ARRAY_CAPITALIZED", "bool" => "ZEND_STR_BOOL", "int" => "ZEND_STR_INT", "float" => "ZEND_STR_FLOAT", "callable" => "ZEND_STR_CALLABLE", "iterable" => "ZEND_STR_ITERABLE", "void" => "ZEND_STR_VOID", "false" => "ZEND_STR_FALSE", "null" => "ZEND_STR_NULL_LOWERCASE", "mixed" => "ZEND_STR_MIXED", ]; // NEW in 8.1 private const PHP_81_KNOWN = [ "Unknown" => "ZEND_STR_UNKNOWN_CAPITALIZED", "never" => "ZEND_STR_NEVER", "__sleep" => "ZEND_STR_SLEEP", "__wakeup" => "ZEND_STR_WAKEUP", "cases" => "ZEND_STR_CASES", "from" => "ZEND_STR_FROM", "tryFrom" => "ZEND_STR_TRYFROM", "tryfrom" => "ZEND_STR_TRYFROM_LOWERCASE", // Omit ZEND_STR_AUTOGLOBAL_(SERVER|ENV|REQUEST) ]; // NEW in 8.2 private const PHP_82_KNOWN = [ "true" => "ZEND_STR_TRUE", "Traversable" => "ZEND_STR_TRAVERSABLE", "count" => "ZEND_STR_COUNT", "SensitiveParameter" => "ZEND_STR_SENSITIVEPARAMETER", ]; // Only new string in 8.3 is ZEND_STR_CONST_EXPR_PLACEHOLDER which is // not a valid variable name ("[constant expression]") // NEW in 8.4 private const PHP_84_KNOWN = [ "exit" => "ZEND_STR_EXIT", "Deprecated" => "ZEND_STR_DEPRECATED_CAPITALIZED", "since" => "ZEND_STR_SINCE", "get" => "ZEND_STR_GET", "set" => "ZEND_STR_SET", ]; /** * @var AttributeInfo[] $attributes */ public function __construct( PropertyName $name, int $classFlags, int $flags, ?Type $type, ?Type $phpDocType, ?Expr $defaultValue, ?string $defaultValueString, bool $isDocReadonly, bool $isVirtual, ?string $link, ?int $phpVersionIdMinimumCompatibility, array $attributes, ?ExposedDocComment $exposedDocComment ) { $this->name = $name; $this->classFlags = $classFlags; $this->defaultValue = $defaultValue; $this->defaultValueString = $defaultValueString; $this->isDocReadonly = $isDocReadonly; $this->isVirtual = $isVirtual; parent::__construct($flags, $type, $phpDocType, $link, $phpVersionIdMinimumCompatibility, $attributes, $exposedDocComment); } protected function getVariableTypeCode(): string { return "property"; } protected function getVariableTypeName(): string { return "property"; } protected function getFieldSynopsisDefaultLinkend(): string { $className = str_replace(["\\", "_"], ["-", "-"], $this->name->class->toLowerString()); return "$className.props." . strtolower(str_replace("_", "-", trim($this->name->getDeclarationName(), "_"))); } protected function getFieldSynopsisName(): string { return $this->name->getDeclarationName(); } /** @param array $allConstInfos */ protected function getFieldSynopsisValueString(array $allConstInfos): ?string { return $this->defaultValueString; } public function discardInfoForOldPhpVersions(?int $phpVersionIdMinimumCompatibility): void { $this->type = null; $this->flags &= ~Modifiers::READONLY; $this->attributes = []; $this->phpVersionIdMinimumCompatibility = $phpVersionIdMinimumCompatibility; } /** @param array $allConstInfos */ public function getDeclaration(array $allConstInfos): string { $code = "\n"; $propertyName = $this->name->getDeclarationName(); if ($this->defaultValue === null) { $defaultValue = EvaluatedValue::null(); } else { $defaultValue = EvaluatedValue::createFromExpression($this->defaultValue, null, null, $allConstInfos); if ($defaultValue->isUnknownConstValue || ($defaultValue->originatingConsts && $defaultValue->getCExpr() === null)) { echo "Skipping code generation for property $this->name, because it has an unknown constant default value\n"; return ""; } } $zvalName = "property_{$propertyName}_default_value"; if ($this->defaultValue === null && $this->type !== null) { $code .= "\tzval $zvalName;\n\tZVAL_UNDEF(&$zvalName);\n"; } else { $code .= $defaultValue->initializeZval($zvalName); } [$stringInit, $nameCode, $stringRelease] = $this->getString($propertyName); $code .= $stringInit; if ($this->exposedDocComment) { $commentCode = "property_{$propertyName}_comment"; $escapedComment = $this->exposedDocComment->escape(); $escapedCommentLength = $this->exposedDocComment->getLength(); $code .= "\tzend_string *$commentCode = zend_string_init_interned(\"$escapedComment\", $escapedCommentLength, 1);\n"; } else { $commentCode = "NULL"; } if (!empty($this->attributes)) { $template = "\tzend_property_info *property_" . $this->name->getDeclarationName() . " = "; } else { $template = "\t"; } if ($this->phpVersionIdMinimumCompatibility === null || $this->phpVersionIdMinimumCompatibility >= PHP_80_VERSION_ID) { $typeCode = $this->getTypeCode($propertyName, $code); $template .= "zend_declare_typed_property(class_entry, $nameCode, &$zvalName, %s, $commentCode, $typeCode);\n"; } else { $template .= "zend_declare_property_ex(class_entry, $nameCode, &$zvalName, %s, $commentCode);\n"; } $flagsCode = generateVersionDependentFlagCode( $template, $this->getFlagsByPhpVersion(), $this->phpVersionIdMinimumCompatibility ); $code .= implode("", $flagsCode); $code .= $stringRelease; return $code; } /** * Get an array of three strings: * - declaration of zend_string, if needed, or empty otherwise * - usage of that zend_string, or usage with ZSTR_KNOWN() * - freeing the zend_string, if needed * * @param string $propName * @return string[] */ private function getString(string $propName): array { // Generally strings will not be known $nameCode = "property_{$propName}_name"; $result = [ "\tzend_string *$nameCode = zend_string_init(\"$propName\", sizeof(\"$propName\") - 1, 1);\n", $nameCode, "\tzend_string_release($nameCode);\n" ]; // If not set, use the current latest version $allVersions = ALL_PHP_VERSION_IDS; $minPhp = $phpVersionIdMinimumCompatibility ?? end($allVersions); if ($minPhp < PHP_80_VERSION_ID) { // No known strings in 7.0 return $result; } $include = self::PHP_80_KNOWN; switch ($minPhp) { case PHP_84_VERSION_ID: $include = array_merge($include, self::PHP_84_KNOWN); // Intentional fall through case PHP_83_VERSION_ID: case PHP_82_VERSION_ID: $include = array_merge($include, self::PHP_82_KNOWN); // Intentional fall through case PHP_81_VERSION_ID: $include = array_merge($include, self::PHP_81_KNOWN); break; } if (array_key_exists($propName,$include)) { $knownStr = $include[$propName]; return [ '', "ZSTR_KNOWN($knownStr)", '', ]; } return $result; } /** * @return array */ protected function getFlagsByPhpVersion(): array { $flags = parent::getFlagsByPhpVersion(); if ($this->flags & Modifiers::STATIC) { $flags = $this->addFlagForVersionsAbove($flags, "ZEND_ACC_STATIC", PHP_70_VERSION_ID); } if ($this->flags & Modifiers::READONLY) { $flags = $this->addFlagForVersionsAbove($flags, "ZEND_ACC_READONLY", PHP_81_VERSION_ID); } elseif ($this->classFlags & Modifiers::READONLY) { $flags = $this->addFlagForVersionsAbove($flags, "ZEND_ACC_READONLY", PHP_82_VERSION_ID); } if ($this->isVirtual) { $flags = $this->addFlagForVersionsAbove($flags, "ZEND_ACC_VIRTUAL", PHP_84_VERSION_ID); } return $flags; } protected function addModifiersToFieldSynopsis(DOMDocument $doc, DOMElement $fieldsynopsisElement): void { parent::addModifiersToFieldSynopsis($doc, $fieldsynopsisElement); if ($this->flags & Modifiers::STATIC) { $fieldsynopsisElement->appendChild(new DOMText("\n ")); $fieldsynopsisElement->appendChild($doc->createElement("modifier", "static")); } if ($this->flags & Modifiers::READONLY || $this->isDocReadonly) { $fieldsynopsisElement->appendChild(new DOMText("\n ")); $fieldsynopsisElement->appendChild($doc->createElement("modifier", "readonly")); } } public function __clone() { if ($this->type) { $this->type = clone $this->type; } foreach ($this->attributes as $key => $attribute) { $this->attributes[$key] = clone $attribute; } if ($this->exposedDocComment) { $this->exposedDocComment = clone $this->exposedDocComment; } } } class EnumCaseInfo { public string $name; public ?Expr $value; public function __construct(string $name, ?Expr $value) { $this->name = $name; $this->value = $value; } /** @param array $allConstInfos */ public function getDeclaration(array $allConstInfos): string { $escapedName = addslashes($this->name); if ($this->value === null) { $code = "\n\tzend_enum_add_case_cstr(class_entry, \"$escapedName\", NULL);\n"; } else { $value = EvaluatedValue::createFromExpression($this->value, null, null, $allConstInfos); $zvalName = "enum_case_{$escapedName}_value"; $code = "\n" . $value->initializeZval($zvalName); $code .= "\tzend_enum_add_case_cstr(class_entry, \"$escapedName\", &$zvalName);\n"; } return $code; } } class AttributeInfo { public string $class; /** @var \PhpParser\Node\Arg[] */ public array $args; /** @param \PhpParser\Node\Arg[] $args */ public function __construct(string $class, array $args) { $this->class = $class; $this->args = $args; } /** @param array $allConstInfos */ public function generateCode(string $invocation, string $nameSuffix, array $allConstInfos, ?int $phpVersionIdMinimumCompatibility): string { $php82MinimumCompatibility = $phpVersionIdMinimumCompatibility === null || $phpVersionIdMinimumCompatibility >= PHP_82_VERSION_ID; $php84MinimumCompatibility = $phpVersionIdMinimumCompatibility === null || $phpVersionIdMinimumCompatibility >= PHP_84_VERSION_ID; /* see ZEND_KNOWN_STRINGS in Zend/strings.h */ $knowns = [ "message" => "ZEND_STR_MESSAGE", ]; if ($php82MinimumCompatibility) { $knowns["SensitiveParameter"] = "ZEND_STR_SENSITIVEPARAMETER"; } if ($php84MinimumCompatibility) { $knowns["Deprecated"] = "ZEND_STR_DEPRECATED_CAPITALIZED"; $knowns["since"] = "ZEND_STR_SINCE"; } $code = "\n"; $escapedAttributeName = strtr($this->class, '\\', '_'); if (isset($knowns[$escapedAttributeName])) { $code .= "\t" . ($this->args ? "zend_attribute *attribute_{$escapedAttributeName}_$nameSuffix = " : "") . "$invocation, ZSTR_KNOWN({$knowns[$escapedAttributeName]}), " . count($this->args) . ");\n"; } else { $code .= "\tzend_string *attribute_name_{$escapedAttributeName}_$nameSuffix = zend_string_init_interned(\"" . addcslashes($this->class, "\\") . "\", sizeof(\"" . addcslashes($this->class, "\\") . "\") - 1, 1);\n"; $code .= "\t" . ($this->args ? "zend_attribute *attribute_{$escapedAttributeName}_$nameSuffix = " : "") . "$invocation, attribute_name_{$escapedAttributeName}_$nameSuffix, " . count($this->args) . ");\n"; $code .= "\tzend_string_release(attribute_name_{$escapedAttributeName}_$nameSuffix);\n"; } foreach ($this->args as $i => $arg) { $value = EvaluatedValue::createFromExpression($arg->value, null, null, $allConstInfos); $zvalName = "attribute_{$escapedAttributeName}_{$nameSuffix}_arg$i"; $code .= $value->initializeZval($zvalName); $code .= "\tZVAL_COPY_VALUE(&attribute_{$escapedAttributeName}_{$nameSuffix}->args[$i].value, &$zvalName);\n"; if ($arg->name) { if (isset($knowns[$arg->name->name])) { $code .= "\tattribute_{$escapedAttributeName}_{$nameSuffix}->args[$i].name = ZSTR_KNOWN({$knowns[$arg->name->name]});\n"; } else { $code .= "\tattribute_{$escapedAttributeName}_{$nameSuffix}->args[$i].name = zend_string_init_interned(\"{$arg->name->name}\", sizeof(\"{$arg->name->name}\") - 1, 1);\n"; } } } return $code; } } class ClassInfo { public Name $name; public int $flags; public string $type; public ?string $alias; public ?SimpleType $enumBackingType; public bool $isDeprecated; public bool $isStrictProperties; /** @var AttributeInfo[] */ public array $attributes; public ?ExposedDocComment $exposedDocComment; public bool $isNotSerializable; /** @var Name[] */ public array $extends; /** @var Name[] */ public array $implements; /** @var ConstInfo[] */ public array $constInfos; /** @var PropertyInfo[] */ public array $propertyInfos; /** @var FuncInfo[] */ public array $funcInfos; /** @var EnumCaseInfo[] */ public array $enumCaseInfos; public ?string $cond; public ?int $phpVersionIdMinimumCompatibility; public bool $isUndocumentable; /** * @param AttributeInfo[] $attributes * @param Name[] $extends * @param Name[] $implements * @param ConstInfo[] $constInfos * @param PropertyInfo[] $propertyInfos * @param FuncInfo[] $funcInfos * @param EnumCaseInfo[] $enumCaseInfos */ public function __construct( Name $name, int $flags, string $type, ?string $alias, ?SimpleType $enumBackingType, bool $isDeprecated, bool $isStrictProperties, array $attributes, ?ExposedDocComment $exposedDocComment, bool $isNotSerializable, array $extends, array $implements, array $constInfos, array $propertyInfos, array $funcInfos, array $enumCaseInfos, ?string $cond, ?int $minimumPhpVersionIdCompatibility, bool $isUndocumentable ) { $this->name = $name; $this->flags = $flags; $this->type = $type; $this->alias = $alias; $this->enumBackingType = $enumBackingType; $this->isDeprecated = $isDeprecated; $this->isStrictProperties = $isStrictProperties; $this->attributes = $attributes; $this->exposedDocComment = $exposedDocComment; $this->isNotSerializable = $isNotSerializable; $this->extends = $extends; $this->implements = $implements; $this->constInfos = $constInfos; $this->propertyInfos = $propertyInfos; $this->funcInfos = $funcInfos; $this->enumCaseInfos = $enumCaseInfos; $this->cond = $cond; $this->phpVersionIdMinimumCompatibility = $minimumPhpVersionIdCompatibility; $this->isUndocumentable = $isUndocumentable; } /** @param array $allConstInfos */ public function getRegistration(array $allConstInfos): string { $params = []; foreach ($this->extends as $extends) { $params[] = "zend_class_entry *class_entry_" . implode("_", $extends->getParts()); } foreach ($this->implements as $implements) { $params[] = "zend_class_entry *class_entry_" . implode("_", $implements->getParts()); } $escapedName = implode("_", $this->name->getParts()); $code = ''; $php80MinimumCompatibility = $this->phpVersionIdMinimumCompatibility === null || $this->phpVersionIdMinimumCompatibility >= PHP_80_VERSION_ID; $php81MinimumCompatibility = $this->phpVersionIdMinimumCompatibility === null || $this->phpVersionIdMinimumCompatibility >= PHP_81_VERSION_ID; $php84MinimumCompatibility = $this->phpVersionIdMinimumCompatibility === null || $this->phpVersionIdMinimumCompatibility >= PHP_84_VERSION_ID; if ($this->type === "enum" && !$php81MinimumCompatibility) { $code .= "#if (PHP_VERSION_ID >= " . PHP_81_VERSION_ID . ")\n"; } if ($this->cond) { $code .= "#if {$this->cond}\n"; } $code .= "static zend_class_entry *register_class_$escapedName(" . (empty($params) ? "void" : implode(", ", $params)) . ")\n"; $code .= "{\n"; $flagCodes = generateVersionDependentFlagCode("%s", $this->getFlagsByPhpVersion(), $this->phpVersionIdMinimumCompatibility); $flags = implode("", $flagCodes); $classMethods = ($this->funcInfos === []) ? 'NULL' : "class_{$escapedName}_methods"; if ($this->type === "enum") { $name = addslashes((string) $this->name); $backingType = $this->enumBackingType ? $this->enumBackingType->toTypeCode() : "IS_UNDEF"; $code .= "\tzend_class_entry *class_entry = zend_register_internal_enum(\"$name\", $backingType, $classMethods);\n"; if ($flags !== "") { $code .= "\tclass_entry->ce_flags |= $flags\n"; } } else { $code .= "\tzend_class_entry ce, *class_entry;\n\n"; if (count($this->name->getParts()) > 1) { $className = $this->name->getLast(); $namespace = addslashes((string) $this->name->slice(0, -1)); $code .= "\tINIT_NS_CLASS_ENTRY(ce, \"$namespace\", \"$className\", $classMethods);\n"; } else { $code .= "\tINIT_CLASS_ENTRY(ce, \"$this->name\", $classMethods);\n"; } if ($this->type === "class" || $this->type === "trait") { if (!$php84MinimumCompatibility) { $code .= "#if (PHP_VERSION_ID >= " . PHP_84_VERSION_ID . ")\n"; } $code .= "\tclass_entry = zend_register_internal_class_with_flags(&ce, " . (isset($this->extends[0]) ? "class_entry_" . str_replace("\\", "_", $this->extends[0]->toString()) : "NULL") . ", " . ($flags ?: 0) . ");\n"; if (!$php84MinimumCompatibility) { $code .= "#else\n"; $code .= "\tclass_entry = zend_register_internal_class_ex(&ce, " . (isset($this->extends[0]) ? "class_entry_" . str_replace("\\", "_", $this->extends[0]->toString()) : "NULL") . ");\n"; if ($flags !== "") { $code .= "\tclass_entry->ce_flags |= $flags;\n"; } $code .= "#endif\n"; } } else { $code .= "\tclass_entry = zend_register_internal_interface(&ce);\n"; if ($flags !== "") { $code .= "\tclass_entry->ce_flags |= $flags\n"; } } } if ($this->exposedDocComment) { if (!$php84MinimumCompatibility) { $code .= "#if (PHP_VERSION_ID >= " . PHP_84_VERSION_ID . ")\n"; } $code .= "\tclass_entry->doc_comment = zend_string_init_interned(\"" . $this->exposedDocComment->escape() . "\", " . $this->exposedDocComment->getLength() . ", 1);\n"; if (!$php84MinimumCompatibility) { $code .= "#endif\n"; } } $implements = array_map( function (Name $item) { return "class_entry_" . implode("_", $item->getParts()); }, $this->type === "interface" ? $this->extends : $this->implements ); if (!empty($implements)) { $code .= "\tzend_class_implements(class_entry, " . count($implements) . ", " . implode(", ", $implements) . ");\n"; } if ($this->alias) { $code .= "\tzend_register_class_alias(\"" . str_replace("\\", "\\\\", $this->alias) . "\", class_entry);\n"; } foreach ($this->constInfos as $const) { $code .= $const->getDeclaration($allConstInfos); } foreach ($this->enumCaseInfos as $enumCase) { $code .= $enumCase->getDeclaration($allConstInfos); } foreach ($this->propertyInfos as $property) { $code .= $property->getDeclaration($allConstInfos); } if (!empty($this->attributes)) { if (!$php80MinimumCompatibility) { $code .= "\n#if (PHP_VERSION_ID >= " . PHP_80_VERSION_ID . ")"; } foreach ($this->attributes as $key => $attribute) { $code .= $attribute->generateCode( "zend_add_class_attribute(class_entry", "class_{$escapedName}_$key", $allConstInfos, $this->phpVersionIdMinimumCompatibility ); } if (!$php80MinimumCompatibility) { $code .= "#endif\n"; } } if ($attributeInitializationCode = generateConstantAttributeInitialization($this->constInfos, $allConstInfos, $this->phpVersionIdMinimumCompatibility, $this->cond)) { if (!$php80MinimumCompatibility) { $code .= "#if (PHP_VERSION_ID >= " . PHP_80_VERSION_ID . ")"; } $code .= "\n" . $attributeInitializationCode; if (!$php80MinimumCompatibility) { $code .= "#endif\n"; } } if ($attributeInitializationCode = generatePropertyAttributeInitialization($this->propertyInfos, $allConstInfos, $this->phpVersionIdMinimumCompatibility)) { if (!$php80MinimumCompatibility) { $code .= "#if (PHP_VERSION_ID >= " . PHP_80_VERSION_ID . ")"; } $code .= "\n" . $attributeInitializationCode; if (!$php80MinimumCompatibility) { $code .= "#endif\n"; } } if ($attributeInitializationCode = generateFunctionAttributeInitialization($this->funcInfos, $allConstInfos, $this->phpVersionIdMinimumCompatibility, $this->cond)) { if (!$php80MinimumCompatibility) { $code .= "#if (PHP_VERSION_ID >= " . PHP_80_VERSION_ID . ")\n"; } $code .= "\n" . $attributeInitializationCode; if (!$php80MinimumCompatibility) { $code .= "#endif\n"; } } $code .= "\n\treturn class_entry;\n"; $code .= "}\n"; if ($this->cond) { $code .= "#endif\n"; } if ($this->type === "enum" && !$php81MinimumCompatibility) { $code .= "#endif\n"; } return $code; } /** * @return array */ private function getFlagsByPhpVersion(): array { $php70Flags = []; if ($this->type === "trait") { $php70Flags[] = "ZEND_ACC_TRAIT"; } if ($this->flags & Modifiers::FINAL) { $php70Flags[] = "ZEND_ACC_FINAL"; } if ($this->flags & Modifiers::ABSTRACT) { $php70Flags[] = "ZEND_ACC_ABSTRACT"; } if ($this->isDeprecated) { $php70Flags[] = "ZEND_ACC_DEPRECATED"; } $php80Flags = $php70Flags; if ($this->isStrictProperties) { $php80Flags[] = "ZEND_ACC_NO_DYNAMIC_PROPERTIES"; } $php81Flags = $php80Flags; if ($this->isNotSerializable) { $php81Flags[] = "ZEND_ACC_NOT_SERIALIZABLE"; } $php82Flags = $php81Flags; if ($this->flags & Modifiers::READONLY) { $php82Flags[] = "ZEND_ACC_READONLY_CLASS"; } foreach ($this->attributes as $attr) { if ($attr->class === "AllowDynamicProperties") { $php82Flags[] = "ZEND_ACC_ALLOW_DYNAMIC_PROPERTIES"; break; } } $php83Flags = $php82Flags; $php84Flags = $php83Flags; return [ PHP_70_VERSION_ID => $php70Flags, PHP_80_VERSION_ID => $php80Flags, PHP_81_VERSION_ID => $php81Flags, PHP_82_VERSION_ID => $php82Flags, PHP_83_VERSION_ID => $php83Flags, PHP_84_VERSION_ID => $php84Flags, ]; } public function discardInfoForOldPhpVersions(?int $phpVersionIdMinimumCompatibility): void { $this->attributes = []; $this->flags &= ~Modifiers::READONLY; $this->exposedDocComment = null; $this->isStrictProperties = false; $this->isNotSerializable = false; foreach ($this->propertyInfos as $propertyInfo) { $propertyInfo->discardInfoForOldPhpVersions($phpVersionIdMinimumCompatibility); } $this->phpVersionIdMinimumCompatibility = $phpVersionIdMinimumCompatibility; } /** * @param array $classMap * @param array $allConstInfos * @param iterable $allConstInfo */ public function getClassSynopsisDocument(array $classMap, array $allConstInfos): ?string { $doc = new DOMDocument(); $doc->formatOutput = true; $classSynopsis = $this->getClassSynopsisElement($doc, $classMap, $allConstInfos); if (!$classSynopsis) { return null; } $doc->appendChild($classSynopsis); return $doc->saveXML(); } /** * @param array $classMap * @param array $allConstInfos */ public function getClassSynopsisElement(DOMDocument $doc, array $classMap, array $allConstInfos): ?DOMElement { $classSynopsis = $doc->createElement("classsynopsis"); $classSynopsis->setAttribute("class", $this->type === "interface" ? "interface" : "class"); $exceptionOverride = $this->type === "class" && $this->isException($classMap) ? "exception" : null; $ooElement = self::createOoElement($doc, $this, $exceptionOverride, true, null, 4); if (!$ooElement) { return null; } $classSynopsis->appendChild(new DOMText("\n ")); $classSynopsis->appendChild($ooElement); foreach ($this->extends as $k => $parent) { $parentInfo = $classMap[$parent->toString()] ?? null; if ($parentInfo === null) { throw new Exception("Missing parent class " . $parent->toString()); } $ooElement = self::createOoElement( $doc, $parentInfo, null, false, $k === 0 ? "extends" : null, 4 ); if (!$ooElement) { return null; } $classSynopsis->appendChild(new DOMText("\n\n ")); $classSynopsis->appendChild($ooElement); } foreach ($this->implements as $k => $interface) { $interfaceInfo = $classMap[$interface->toString()] ?? null; if (!$interfaceInfo) { throw new Exception("Missing implemented interface " . $interface->toString()); } $ooElement = self::createOoElement($doc, $interfaceInfo, null, false, $k === 0 ? "implements" : null, 4); if (!$ooElement) { return null; } $classSynopsis->appendChild(new DOMText("\n\n ")); $classSynopsis->appendChild($ooElement); } /** @var array $parentsWithInheritedConstants */ $parentsWithInheritedConstants = []; /** @var array $parentsWithInheritedProperties */ $parentsWithInheritedProperties = []; /** @var array $parentsWithInheritedMethods */ $parentsWithInheritedMethods = []; $this->collectInheritedMembers( $parentsWithInheritedConstants, $parentsWithInheritedProperties, $parentsWithInheritedMethods, $this->hasConstructor(), $classMap ); $this->appendInheritedMemberSectionToClassSynopsis( $doc, $classSynopsis, $parentsWithInheritedConstants, "&Constants;", "&InheritedConstants;" ); if (!empty($this->constInfos)) { $classSynopsis->appendChild(new DOMText("\n\n ")); $classSynopsisInfo = $doc->createElement("classsynopsisinfo", "&Constants;"); $classSynopsisInfo->setAttribute("role", "comment"); $classSynopsis->appendChild($classSynopsisInfo); foreach ($this->constInfos as $constInfo) { $classSynopsis->appendChild(new DOMText("\n ")); $fieldSynopsisElement = $constInfo->getFieldSynopsisElement($doc, $allConstInfos); $classSynopsis->appendChild($fieldSynopsisElement); } } if (!empty($this->propertyInfos)) { $classSynopsis->appendChild(new DOMText("\n\n ")); $classSynopsisInfo = $doc->createElement("classsynopsisinfo", "&Properties;"); $classSynopsisInfo->setAttribute("role", "comment"); $classSynopsis->appendChild($classSynopsisInfo); foreach ($this->propertyInfos as $propertyInfo) { $classSynopsis->appendChild(new DOMText("\n ")); $fieldSynopsisElement = $propertyInfo->getFieldSynopsisElement($doc, $allConstInfos); $classSynopsis->appendChild($fieldSynopsisElement); } } $this->appendInheritedMemberSectionToClassSynopsis( $doc, $classSynopsis, $parentsWithInheritedProperties, "&Properties;", "&InheritedProperties;" ); if (!empty($this->funcInfos)) { $classSynopsis->appendChild(new DOMText("\n\n ")); $classSynopsisInfo = $doc->createElement("classsynopsisinfo", "&Methods;"); $classSynopsisInfo->setAttribute("role", "comment"); $classSynopsis->appendChild($classSynopsisInfo); $classReference = self::getClassSynopsisReference($this->name); $escapedName = addslashes($this->name->__toString()); if ($this->hasConstructor()) { $classSynopsis->appendChild(new DOMText("\n ")); $includeElement = $this->createIncludeElement( $doc, "xmlns(db=http://docbook.org/ns/docbook) xpointer(id('$classReference')/db:refentry/db:refsect1[@role='description']/descendant::db:constructorsynopsis[@role='$escapedName'])" ); $classSynopsis->appendChild($includeElement); } if ($this->hasMethods()) { $classSynopsis->appendChild(new DOMText("\n ")); $includeElement = $this->createIncludeElement( $doc, "xmlns(db=http://docbook.org/ns/docbook) xpointer(id('$classReference')/db:refentry/db:refsect1[@role='description']/descendant::db:methodsynopsis[@role='$escapedName'])" ); $classSynopsis->appendChild($includeElement); } if ($this->hasDestructor()) { $classSynopsis->appendChild(new DOMText("\n ")); $includeElement = $this->createIncludeElement( $doc, "xmlns(db=http://docbook.org/ns/docbook) xpointer(id('$classReference')/db:refentry/db:refsect1[@role='description']/descendant::db:destructorsynopsis[@role='$escapedName'])" ); $classSynopsis->appendChild($includeElement); } } if (!empty($parentsWithInheritedMethods)) { $classSynopsis->appendChild(new DOMText("\n\n ")); $classSynopsisInfo = $doc->createElement("classsynopsisinfo", "&InheritedMethods;"); $classSynopsisInfo->setAttribute("role", "comment"); $classSynopsis->appendChild($classSynopsisInfo); foreach ($parentsWithInheritedMethods as $parent) { $parentName = $parent["name"]; $parentMethodsynopsisTypes = $parent["types"]; $parentReference = self::getClassSynopsisReference($parentName); $escapedParentName = addslashes($parentName->__toString()); foreach ($parentMethodsynopsisTypes as $parentMethodsynopsisType) { $classSynopsis->appendChild(new DOMText("\n ")); $includeElement = $this->createIncludeElement( $doc, "xmlns(db=http://docbook.org/ns/docbook) xpointer(id('$parentReference')/db:refentry/db:refsect1[@role='description']/descendant::db:{$parentMethodsynopsisType}[@role='$escapedParentName'])" ); $classSynopsis->appendChild($includeElement); } } } $classSynopsis->appendChild(new DOMText("\n ")); return $classSynopsis; } private static function createOoElement( DOMDocument $doc, ClassInfo $classInfo, ?string $typeOverride, bool $withModifiers, ?string $modifierOverride, int $indentationLevel ): ?DOMElement { $indentation = str_repeat(" ", $indentationLevel); if ($classInfo->type !== "class" && $classInfo->type !== "interface") { echo "Class synopsis generation is not implemented for " . $classInfo->type . "\n"; return null; } $type = $typeOverride !== null ? $typeOverride : $classInfo->type; $ooElement = $doc->createElement("oo$type"); $ooElement->appendChild(new DOMText("\n$indentation ")); if ($modifierOverride !== null) { $ooElement->appendChild($doc->createElement('modifier', $modifierOverride)); $ooElement->appendChild(new DOMText("\n$indentation ")); } elseif ($withModifiers) { foreach ($classInfo->attributes as $attribute) { $modifier = $doc->createElement("modifier", "#[\\" . $attribute->class . "]"); $modifier->setAttribute("role", "attribute"); $ooElement->appendChild($modifier); $ooElement->appendChild(new DOMText("\n$indentation ")); } if ($classInfo->flags & Modifiers::FINAL) { $ooElement->appendChild($doc->createElement('modifier', 'final')); $ooElement->appendChild(new DOMText("\n$indentation ")); } if ($classInfo->flags & Modifiers::ABSTRACT) { $ooElement->appendChild($doc->createElement('modifier', 'abstract')); $ooElement->appendChild(new DOMText("\n$indentation ")); } if ($classInfo->flags & Modifiers::READONLY) { $ooElement->appendChild($doc->createElement('modifier', 'readonly')); $ooElement->appendChild(new DOMText("\n$indentation ")); } } $nameElement = $doc->createElement("{$type}name", $classInfo->name->toString()); $ooElement->appendChild($nameElement); $ooElement->appendChild(new DOMText("\n$indentation")); return $ooElement; } public static function getClassSynopsisFilename(Name $name): string { return strtolower(str_replace("_", "-", implode('-', $name->getParts()))); } public static function getClassSynopsisReference(Name $name): string { return "class." . self::getClassSynopsisFilename($name); } /** * @param array $parentsWithInheritedConstants * @param array $parentsWithInheritedProperties * @param array $parentsWithInheritedMethods * @param array $classMap */ private function collectInheritedMembers( array &$parentsWithInheritedConstants, array &$parentsWithInheritedProperties, array &$parentsWithInheritedMethods, bool $hasConstructor, array $classMap ): void { foreach ($this->extends as $parent) { $parentInfo = $classMap[$parent->toString()] ?? null; $parentName = $parent->toString(); if (!$parentInfo) { throw new Exception("Missing parent class $parentName"); } if (!empty($parentInfo->constInfos) && !isset($parentsWithInheritedConstants[$parentName])) { $parentsWithInheritedConstants[] = $parent; } if (!empty($parentInfo->propertyInfos) && !isset($parentsWithInheritedProperties[$parentName])) { $parentsWithInheritedProperties[$parentName] = $parent; } if (!$hasConstructor && $parentInfo->hasNonPrivateConstructor()) { $parentsWithInheritedMethods[$parentName]["name"] = $parent; $parentsWithInheritedMethods[$parentName]["types"][] = "constructorsynopsis"; } if ($parentInfo->hasMethods()) { $parentsWithInheritedMethods[$parentName]["name"] = $parent; $parentsWithInheritedMethods[$parentName]["types"][] = "methodsynopsis"; } if ($parentInfo->hasDestructor()) { $parentsWithInheritedMethods[$parentName]["name"] = $parent; $parentsWithInheritedMethods[$parentName]["types"][] = "destructorsynopsis"; } $parentInfo->collectInheritedMembers( $parentsWithInheritedConstants, $parentsWithInheritedProperties, $parentsWithInheritedMethods, $hasConstructor, $classMap ); } foreach ($this->implements as $parent) { $parentInfo = $classMap[$parent->toString()] ?? null; if (!$parentInfo) { throw new Exception("Missing parent interface " . $parent->toString()); } if (!empty($parentInfo->constInfos) && !isset($parentsWithInheritedConstants[$parent->toString()])) { $parentsWithInheritedConstants[$parent->toString()] = $parent; } $unusedParentsWithInheritedProperties = []; $unusedParentsWithInheritedMethods = []; $parentInfo->collectInheritedMembers( $parentsWithInheritedConstants, $unusedParentsWithInheritedProperties, $unusedParentsWithInheritedMethods, $hasConstructor, $classMap ); } } /** @param array $classMap */ private function isException(array $classMap): bool { if ($this->name->toString() === "Throwable") { return true; } foreach ($this->extends as $parentName) { $parent = $classMap[$parentName->toString()] ?? null; if ($parent === null) { throw new Exception("Missing parent class " . $parentName->toString()); } if ($parent->isException($classMap)) { return true; } } if ($this->type === "class") { foreach ($this->implements as $interfaceName) { $interface = $classMap[$interfaceName->toString()] ?? null; if ($interface === null) { throw new Exception("Missing implemented interface " . $interfaceName->toString()); } if ($interface->isException($classMap)) { return true; } } } return false; } private function hasConstructor(): bool { foreach ($this->funcInfos as $funcInfo) { if ($funcInfo->name->isConstructor()) { return true; } } return false; } private function hasNonPrivateConstructor(): bool { foreach ($this->funcInfos as $funcInfo) { if ($funcInfo->name->isConstructor() && !($funcInfo->flags & Modifiers::PRIVATE)) { return true; } } return false; } private function hasDestructor(): bool { foreach ($this->funcInfos as $funcInfo) { if ($funcInfo->name->isDestructor()) { return true; } } return false; } private function hasMethods(): bool { foreach ($this->funcInfos as $funcInfo) { if (!$funcInfo->name->isConstructor() && !$funcInfo->name->isDestructor()) { return true; } } return false; } private function createIncludeElement(DOMDocument $doc, string $query): DOMElement { $includeElement = $doc->createElement("xi:include"); $attr = $doc->createAttribute("xpointer"); $attr->value = $query; $includeElement->appendChild($attr); $fallbackElement = $doc->createElement("xi:fallback"); $includeElement->appendChild(new DOMText("\n ")); $includeElement->appendChild($fallbackElement); $includeElement->appendChild(new DOMText("\n ")); return $includeElement; } public function __clone() { foreach ($this->constInfos as $key => $constInfo) { $this->constInfos[$key] = clone $constInfo; } foreach ($this->propertyInfos as $key => $propertyInfo) { $this->propertyInfos[$key] = clone $propertyInfo; } foreach ($this->funcInfos as $key => $funcInfo) { $this->funcInfos[$key] = clone $funcInfo; } foreach ($this->attributes as $key => $attribute) { $this->attributes[$key] = clone $attribute; } if ($this->exposedDocComment) { $this->exposedDocComment = clone $this->exposedDocComment; } } /** * @param Name[] $parents */ private function appendInheritedMemberSectionToClassSynopsis(DOMDocument $doc, DOMElement $classSynopsis, array $parents, string $label, string $inheritedLabel): void { if (empty($parents)) { return; } $classSynopsis->appendChild(new DOMText("\n\n ")); $classSynopsisInfo = $doc->createElement("classsynopsisinfo", "$inheritedLabel"); $classSynopsisInfo->setAttribute("role", "comment"); $classSynopsis->appendChild($classSynopsisInfo); foreach ($parents as $parent) { $classSynopsis->appendChild(new DOMText("\n ")); $parentReference = self::getClassSynopsisReference($parent); $includeElement = $this->createIncludeElement( $doc, "xmlns(db=http://docbook.org/ns/docbook) xpointer(id('$parentReference')/db:partintro/db:section/db:classsynopsis/db:fieldsynopsis[preceding-sibling::db:classsynopsisinfo[1][@role='comment' and text()='$label']]))" ); $classSynopsis->appendChild($includeElement); } } } class FileInfo { /** @var string[] */ public array $dependencies = []; /** @var ConstInfo[] */ public array $constInfos = []; /** @var FuncInfo[] */ public array $funcInfos = []; /** @var ClassInfo[] */ public array $classInfos = []; public bool $generateFunctionEntries = false; public string $declarationPrefix = ""; public bool $generateClassEntries = false; public bool $isUndocumentable = false; public bool $legacyArginfoGeneration = false; private ?int $minimumPhpVersionIdCompatibility = null; /** * @return iterable */ public function getAllFuncInfos(): iterable { yield from $this->funcInfos; foreach ($this->classInfos as $classInfo) { yield from $classInfo->funcInfos; } } /** @return array */ public function getAllConstInfos(): array { $result = []; foreach ($this->constInfos as $constInfo) { $result[$constInfo->name->__toString()] = $constInfo; } foreach ($this->classInfos as $classInfo) { foreach ($classInfo->constInfos as $constInfo) { $result[$constInfo->name->__toString()] = $constInfo; } } return $result; } /** * @return iterable */ public function getAllClassInfos(): iterable { foreach ($this->classInfos as $classInfo) { yield $classInfo; } } public function __clone() { foreach ($this->constInfos as $key => $constInfo) { $this->constInfos[$key] = clone $constInfo; } foreach ($this->funcInfos as $key => $funcInfo) { $this->funcInfos[$key] = clone $funcInfo; } foreach ($this->classInfos as $key => $classInfo) { $this->classInfos[$key] = clone $classInfo; } } public function setMinimumPhpVersionIdCompatibility(?int $minimumPhpVersionIdCompatibility) { $this->minimumPhpVersionIdCompatibility = $minimumPhpVersionIdCompatibility; } public function getMinimumPhpVersionIdCompatibility(): ?int { // Non-legacy arginfo files are always PHP 8.0+ compatible if (!$this->legacyArginfoGeneration && $this->minimumPhpVersionIdCompatibility !== null && $this->minimumPhpVersionIdCompatibility < PHP_80_VERSION_ID ) { return PHP_80_VERSION_ID; } return $this->minimumPhpVersionIdCompatibility; } public function shouldGenerateLegacyArginfo(): bool { return $this->minimumPhpVersionIdCompatibility !== null && $this->minimumPhpVersionIdCompatibility < PHP_80_VERSION_ID; } } class DocCommentTag { public string $name; public ?string $value; public function __construct(string $name, ?string $value) { $this->name = $name; $this->value = $value; } public function getValue(): string { if ($this->value === null) { throw new Exception("@$this->name does not have a value"); } return $this->value; } public function getType(): string { $value = $this->getValue(); $matches = []; if ($this->name === "param") { preg_match('/^\s*([\w\|\\\\\[\]<>, ]+)\s*(?:[{(]|\$\w+).*$/', $value, $matches); } elseif ($this->name === "return" || $this->name === "var") { preg_match('/^\s*([\w\|\\\\\[\]<>, ]+)/', $value, $matches); } if (!isset($matches[1])) { throw new Exception("@$this->name doesn't contain a type or has an invalid format \"$value\""); } return trim($matches[1]); } public function getVariableName(): string { $value = $this->value; if ($value === null || strlen($value) === 0) { throw new Exception("@$this->name doesn't have any value"); } $matches = []; if ($this->name === "param") { // Allow for parsing extended types like callable(string):mixed in docblocks preg_match('/^\s*(?[\w\|\\\\]+(?\((?(?:(?&parens)|[^(){}[\]]*+))++\)|\{(?&inparens)\}|\[(?&inparens)\])*+(?::(?&type))?)\s*\$(?\w+).*$/', $value, $matches); } elseif ($this->name === "prefer-ref") { preg_match('/^\s*\$(?\w+).*$/', $value, $matches); } if (!isset($matches["name"])) { throw new Exception("@$this->name doesn't contain a variable name or has an invalid format \"$value\""); } return $matches["name"]; } } class ExposedDocComment { private string $docComment; public function __construct(string $docComment) { $this->docComment = $docComment; } public function escape(): string { return str_replace("\n", '\n', addslashes($this->docComment)); } public function getLength(): int { return strlen($this->docComment); } } /** @return DocCommentTag[] */ function parseDocComments(array $comments): array { $tags = []; foreach ($comments as $comment) { if ($comment instanceof DocComment) { $tags = array_merge($tags, parseDocComment($comment)); } } return $tags; } /** @return DocCommentTag[] */ function parseDocComment(DocComment $comment): array { $commentText = substr($comment->getText(), 2, -2); $tags = []; foreach (explode("\n", $commentText) as $commentLine) { $regex = '/^\*\s*@([a-z-]+)(?:\s+(.+))?$/'; if (preg_match($regex, trim($commentLine), $matches)) { $tags[] = new DocCommentTag($matches[1], $matches[2] ?? null); } } return $tags; } class FramelessFunctionInfo { public int $arity; } function parseFramelessFunctionInfo(string $json): FramelessFunctionInfo { // FIXME: Should have some validation $json = json_decode($json, true); $framelessFunctionInfo = new FramelessFunctionInfo(); $framelessFunctionInfo->arity = $json["arity"]; return $framelessFunctionInfo; } function parseFunctionLike( PrettyPrinterAbstract $prettyPrinter, FunctionOrMethodName $name, int $classFlags, int $flags, Node\FunctionLike $func, ?string $cond, bool $isUndocumentable, ?int $minimumPhpVersionIdCompatibility ): FuncInfo { try { $comments = $func->getComments(); $paramMeta = []; $aliasType = null; $alias = null; $isDeprecated = false; $supportsCompileTimeEval = false; $verify = true; $docReturnType = null; $tentativeReturnType = false; $docParamTypes = []; $refcount = null; $framelessFunctionInfos = []; if ($comments) { $tags = parseDocComments($comments); foreach ($tags as $tag) { switch ($tag->name) { case 'alias': case 'implementation-alias': $aliasType = $tag->name; $aliasParts = explode("::", $tag->getValue()); if (count($aliasParts) === 1) { $alias = new FunctionName(new Name($aliasParts[0])); } else { $alias = new MethodName(new Name($aliasParts[0]), $aliasParts[1]); } break; case 'deprecated': $isDeprecated = true; break; case 'no-verify': $verify = false; break; case 'tentative-return-type': $tentativeReturnType = true; break; case 'return': $docReturnType = $tag->getType(); break; case 'param': $docParamTypes[$tag->getVariableName()] = $tag->getType(); break; case 'refcount': $refcount = $tag->getValue(); break; case 'compile-time-eval': $supportsCompileTimeEval = true; break; case 'prefer-ref': $varName = $tag->getVariableName(); if (!isset($paramMeta[$varName])) { $paramMeta[$varName] = []; } $paramMeta[$varName][$tag->name] = true; break; case 'undocumentable': $isUndocumentable = true; break; case 'frameless-function': $framelessFunctionInfos[] = parseFramelessFunctionInfo($tag->getValue()); break; } } } $varNameSet = []; $args = []; $numRequiredArgs = 0; $foundVariadic = false; foreach ($func->getParams() as $i => $param) { if ($param->isPromoted()) { throw new Exception("Promoted properties are not supported"); } $varName = $param->var->name; $preferRef = !empty($paramMeta[$varName]['prefer-ref']); unset($paramMeta[$varName]); if (isset($varNameSet[$varName])) { throw new Exception("Duplicate parameter name $varName"); } $varNameSet[$varName] = true; if ($preferRef) { $sendBy = ArgInfo::SEND_PREFER_REF; } else if ($param->byRef) { $sendBy = ArgInfo::SEND_BY_REF; } else { $sendBy = ArgInfo::SEND_BY_VAL; } if ($foundVariadic) { throw new Exception("Only the last parameter can be variadic"); } $type = $param->type ? Type::fromNode($param->type) : null; if ($type === null && !isset($docParamTypes[$varName])) { throw new Exception("Missing parameter type"); } if ($param->default instanceof Expr\ConstFetch && $param->default->name->toLowerString() === "null" && $type && !$type->isNullable() ) { $simpleType = $type->tryToSimpleType(); if ($simpleType === null || !$simpleType->isMixed()) { throw new Exception("Parameter $varName has null default, but is not nullable"); } } if ($param->default instanceof Expr\ClassConstFetch && $param->default->class->toLowerString() === "self") { throw new Exception('The exact class name must be used instead of "self"'); } $foundVariadic = $param->variadic; $args[] = new ArgInfo( $varName, $sendBy, $param->variadic, $type, isset($docParamTypes[$varName]) ? Type::fromString($docParamTypes[$varName]) : null, $param->default ? $prettyPrinter->prettyPrintExpr($param->default) : null, createAttributes($param->attrGroups) ); if (!$param->default && !$param->variadic) { $numRequiredArgs = $i + 1; } } foreach (array_keys($paramMeta) as $var) { throw new Exception("Found metadata for invalid param $var"); } $returnType = $func->getReturnType(); if ($returnType === null && $docReturnType === null && !$name->isConstructor() && !$name->isDestructor()) { throw new Exception("Missing return type"); } $return = new ReturnInfo( $func->returnsByRef(), $returnType ? Type::fromNode($returnType) : null, $docReturnType ? Type::fromString($docReturnType) : null, $tentativeReturnType, $refcount ); return new FuncInfo( $name, $classFlags, $flags, $aliasType, $alias, $isDeprecated, $supportsCompileTimeEval, $verify, $args, $return, $numRequiredArgs, $cond, $isUndocumentable, $minimumPhpVersionIdCompatibility, createAttributes($func->attrGroups), $framelessFunctionInfos, createExposedDocComment($comments) ); } catch (Exception $e) { throw new Exception($name . "(): " .$e->getMessage()); } } /** * @param array $attributes */ function parseConstLike( PrettyPrinterAbstract $prettyPrinter, ConstOrClassConstName $name, Node\Const_ $const, int $flags, ?Node $type, array $comments, ?string $cond, bool $isUndocumentable, ?int $phpVersionIdMinimumCompatibility, array $attributes ): ConstInfo { $phpDocType = null; $deprecated = false; $cValue = null; $link = null; $isFileCacheAllowed = true; if ($comments) { $tags = parseDocComments($comments); foreach ($tags as $tag) { if ($tag->name === 'var') { $phpDocType = $tag->getType(); } elseif ($tag->name === 'deprecated') { $deprecated = true; } elseif ($tag->name === 'cvalue') { $cValue = $tag->value; } elseif ($tag->name === 'undocumentable') { $isUndocumentable = true; } elseif ($tag->name === 'link') { $link = $tag->value; } elseif ($tag->name === 'no-file-cache') { $isFileCacheAllowed = false; } } } if ($type === null && $phpDocType === null) { throw new Exception("Missing type for constant " . $name->__toString()); } $constType = $type ? Type::fromNode($type) : null; $constPhpDocType = $phpDocType ? Type::fromString($phpDocType) : null; if ($const->value instanceof Expr\ConstFetch && $const->value->name->toLowerString() === "null" && $constType && !$constType->isNullable() ) { $simpleType = $constType->tryToSimpleType(); if ($simpleType === null || !$simpleType->isMixed()) { throw new Exception("Constant " . $name->__toString() . " has null value, but is not nullable"); } } return new ConstInfo( $name, $flags, $const->value, $prettyPrinter->prettyPrintExpr($const->value), $constType, $constPhpDocType, $deprecated, $cond, $cValue, $isUndocumentable, $link, $phpVersionIdMinimumCompatibility, $attributes, createExposedDocComment($comments), $isFileCacheAllowed ); } /** * @param array $attributes */ function parseProperty( Name $class, int $classFlags, int $flags, Stmt\PropertyProperty $property, ?Node $type, array $comments, PrettyPrinterAbstract $prettyPrinter, ?int $phpVersionIdMinimumCompatibility, array $attributes ): PropertyInfo { $phpDocType = null; $isDocReadonly = false; $isVirtual = false; $link = null; if ($comments) { $tags = parseDocComments($comments); foreach ($tags as $tag) { if ($tag->name === 'var') { $phpDocType = $tag->getType(); } elseif ($tag->name === 'readonly') { $isDocReadonly = true; } elseif ($tag->name === 'link') { $link = $tag->value; } elseif ($tag->name === 'virtual') { $isVirtual = true; } } } $propertyType = $type ? Type::fromNode($type) : null; if ($propertyType === null && !$phpDocType) { throw new Exception("Missing type for property $class::\$$property->name"); } if ($property->default instanceof Expr\ConstFetch && $property->default->name->toLowerString() === "null" && $propertyType && !$propertyType->isNullable() ) { $simpleType = $propertyType->tryToSimpleType(); if ($simpleType === null || !$simpleType->isMixed()) { throw new Exception("Property $class::\$$property->name has null default, but is not nullable"); } } return new PropertyInfo( new PropertyName($class, $property->name->__toString()), $classFlags, $flags, $propertyType, $phpDocType ? Type::fromString($phpDocType) : null, $property->default, $property->default ? $prettyPrinter->prettyPrintExpr($property->default) : null, $isDocReadonly, $isVirtual, $link, $phpVersionIdMinimumCompatibility, $attributes, createExposedDocComment($comments) ); } /** * @param ConstInfo[] $consts * @param PropertyInfo[] $properties * @param FuncInfo[] $methods * @param EnumCaseInfo[] $enumCases */ function parseClass( Name $name, Stmt\ClassLike $class, array $consts, array $properties, array $methods, array $enumCases, ?string $cond, ?int $minimumPhpVersionIdCompatibility, bool $isUndocumentable ): ClassInfo { $flags = $class instanceof Class_ ? $class->flags : 0; $comments = $class->getComments(); $alias = null; $isDeprecated = false; $isStrictProperties = false; $isNotSerializable = false; $allowsDynamicProperties = false; $attributes = []; if ($comments) { $tags = parseDocComments($comments); foreach ($tags as $tag) { if ($tag->name === 'alias') { $alias = $tag->getValue(); } else if ($tag->name === 'deprecated') { $isDeprecated = true; } else if ($tag->name === 'strict-properties') { $isStrictProperties = true; } else if ($tag->name === 'not-serializable') { $isNotSerializable = true; } else if ($tag->name === 'undocumentable') { $isUndocumentable = true; } } } $attributes = createAttributes($class->attrGroups); foreach ($attributes as $attribute) { switch ($attribute->class) { case 'AllowDynamicProperties': $allowsDynamicProperties = true; break 2; } } if ($isStrictProperties && $allowsDynamicProperties) { throw new Exception("A class may not have '@strict-properties' and '#[\\AllowDynamicProperties]' at the same time."); } $extends = []; $implements = []; if ($class instanceof Class_) { $classKind = "class"; if ($class->extends) { $extends[] = $class->extends; } $implements = $class->implements; } elseif ($class instanceof Interface_) { $classKind = "interface"; $extends = $class->extends; } else if ($class instanceof Trait_) { $classKind = "trait"; } else if ($class instanceof Enum_) { $classKind = "enum"; $implements = $class->implements; } else { throw new Exception("Unknown class kind " . get_class($class)); } if ($isUndocumentable) { foreach ($methods as $method) { $method->isUndocumentable = true; } } return new ClassInfo( $name, $flags, $classKind, $alias, $class instanceof Enum_ && $class->scalarType !== null ? SimpleType::fromNode($class->scalarType) : null, $isDeprecated, $isStrictProperties, $attributes, createExposedDocComment($comments), $isNotSerializable, $extends, $implements, $consts, $properties, $methods, $enumCases, $cond, $minimumPhpVersionIdCompatibility, $isUndocumentable ); } /** * @param array> $attributeGroups * @return Attribute[] */ function createAttributes(array $attributeGroups): array { $attributes = []; foreach ($attributeGroups as $attrGroup) { foreach ($attrGroup->attrs as $attr) { $attributes[] = new AttributeInfo($attr->name->toString(), $attr->args); } } return $attributes; } /** @param array $comments */ function createExposedDocComment(array $comments): ?ExposedDocComment { $exposedDocComment = null; foreach ($comments as $comment) { $text = $comment->getText(); $matches = []; $pattern = "#^(\s*\/\*\*)(\s*@genstubs-expose-comment-block)(\s*)$#m"; if (preg_match($pattern, $text, $matches) !== 1) { continue; } if ($exposedDocComment !== null) { throw new Exception("Only one PHPDoc comment block can be exposed"); } $exposedDocComment = preg_replace($pattern, '$1$3', $text); } return $exposedDocComment ? new ExposedDocComment($exposedDocComment) : null; } function handlePreprocessorConditions(array &$conds, Stmt $stmt): ?string { foreach ($stmt->getComments() as $comment) { $text = trim($comment->getText()); if (preg_match('/^#\s*if\s+(.+)$/', $text, $matches)) { $conds[] = $matches[1]; } else if (preg_match('/^#\s*ifdef\s+(.+)$/', $text, $matches)) { $conds[] = "defined($matches[1])"; } else if (preg_match('/^#\s*ifndef\s+(.+)$/', $text, $matches)) { $conds[] = "!defined($matches[1])"; } else if (preg_match('/^#\s*else$/', $text)) { if (empty($conds)) { throw new Exception("Encountered else without corresponding #if"); } $cond = array_pop($conds); $conds[] = "!($cond)"; } else if (preg_match('/^#\s*endif$/', $text)) { if (empty($conds)) { throw new Exception("Encountered #endif without corresponding #if"); } array_pop($conds); } else if ($text[0] === '#') { throw new Exception("Unrecognized preprocessor directive \"$text\""); } } return empty($conds) ? null : implode(' && ', $conds); } /** @return DocComment[] */ function getFileDocComments(array $stmts): array { if (empty($stmts)) { return []; } $comments = $stmts[0]->getComments(); $result = []; foreach ($comments as $comment) { if ($comment instanceof DocComment) { $result[] = $comment; } } return $result; } function handleStatements(FileInfo $fileInfo, array $stmts, PrettyPrinterAbstract $prettyPrinter) { $conds = []; foreach ($stmts as $stmt) { $cond = handlePreprocessorConditions($conds, $stmt); if ($stmt instanceof Stmt\Nop) { continue; } if ($stmt instanceof Stmt\Namespace_) { handleStatements($fileInfo, $stmt->stmts, $prettyPrinter); continue; } if ($stmt instanceof Stmt\Const_) { foreach ($stmt->consts as $const) { $fileInfo->constInfos[] = parseConstLike( $prettyPrinter, new ConstName($const->namespacedName, $const->name->toString()), $const, 0, null, $stmt->getComments(), $cond, $fileInfo->isUndocumentable, $fileInfo->getMinimumPhpVersionIdCompatibility(), [] ); } continue; } if ($stmt instanceof Stmt\Function_) { $fileInfo->funcInfos[] = parseFunctionLike( $prettyPrinter, new FunctionName($stmt->namespacedName), 0, 0, $stmt, $cond, $fileInfo->isUndocumentable, $fileInfo->getMinimumPhpVersionIdCompatibility() ); continue; } if ($stmt instanceof Stmt\ClassLike) { $className = $stmt->namespacedName; $constInfos = []; $propertyInfos = []; $methodInfos = []; $enumCaseInfos = []; foreach ($stmt->stmts as $classStmt) { $cond = handlePreprocessorConditions($conds, $classStmt); if ($classStmt instanceof Stmt\Nop) { continue; } $classFlags = $stmt instanceof Class_ ? $stmt->flags : 0; $abstractFlag = $stmt instanceof Stmt\Interface_ ? Modifiers::ABSTRACT : 0; if ($classStmt instanceof Stmt\ClassConst) { foreach ($classStmt->consts as $const) { $constInfos[] = parseConstLike( $prettyPrinter, new ClassConstName($className, $const->name->toString()), $const, $classStmt->flags, $classStmt->type, $classStmt->getComments(), $cond, $fileInfo->isUndocumentable, $fileInfo->getMinimumPhpVersionIdCompatibility(), createAttributes($classStmt->attrGroups) ); } } else if ($classStmt instanceof Stmt\Property) { if (!($classStmt->flags & Class_::VISIBILITY_MODIFIER_MASK)) { throw new Exception("Visibility modifier is required"); } foreach ($classStmt->props as $property) { $propertyInfos[] = parseProperty( $className, $classFlags, $classStmt->flags, $property, $classStmt->type, $classStmt->getComments(), $prettyPrinter, $fileInfo->getMinimumPhpVersionIdCompatibility(), createAttributes($classStmt->attrGroups) ); } } else if ($classStmt instanceof Stmt\ClassMethod) { if (!($classStmt->flags & Class_::VISIBILITY_MODIFIER_MASK)) { throw new Exception("Visibility modifier is required"); } $methodInfos[] = parseFunctionLike( $prettyPrinter, new MethodName($className, $classStmt->name->toString()), $classFlags, $classStmt->flags | $abstractFlag, $classStmt, $cond, $fileInfo->isUndocumentable, $fileInfo->getMinimumPhpVersionIdCompatibility() ); } else if ($classStmt instanceof Stmt\EnumCase) { $enumCaseInfos[] = new EnumCaseInfo( $classStmt->name->toString(), $classStmt->expr); } else { throw new Exception("Not implemented {$classStmt->getType()}"); } } $fileInfo->classInfos[] = parseClass( $className, $stmt, $constInfos, $propertyInfos, $methodInfos, $enumCaseInfos, $cond, $fileInfo->getMinimumPhpVersionIdCompatibility(), $fileInfo->isUndocumentable ); continue; } if ($stmt instanceof Stmt\Expression) { $expr = $stmt->expr; if ($expr instanceof Expr\Include_) { $fileInfo->dependencies[] = (string)EvaluatedValue::createFromExpression($expr->expr, null, null, [])->value; continue; } } throw new Exception("Unexpected node {$stmt->getType()}"); } if (!empty($conds)) { throw new Exception("Unterminated preprocessor conditions"); } } function parseStubFile(string $code): FileInfo { $lexer = new PhpParser\Lexer\Emulative(); $parser = new PhpParser\Parser\Php7($lexer); $nodeTraverser = new PhpParser\NodeTraverser; $nodeTraverser->addVisitor(new PhpParser\NodeVisitor\NameResolver); $prettyPrinter = new class extends Standard { protected function pName_FullyQualified(Name\FullyQualified $node): string { return implode('\\', $node->getParts()); } }; $stmts = $parser->parse($code); $nodeTraverser->traverse($stmts); $fileInfo = new FileInfo; $fileDocComments = getFileDocComments($stmts); if ($fileDocComments !== []) { $fileTags = parseDocComments($fileDocComments); foreach ($fileTags as $tag) { if ($tag->name === 'generate-function-entries') { $fileInfo->generateFunctionEntries = true; $fileInfo->declarationPrefix = $tag->value ? $tag->value . " " : ""; } else if ($tag->name === 'generate-legacy-arginfo') { if ($tag->value && !in_array((int) $tag->value, ALL_PHP_VERSION_IDS, true)) { throw new Exception( "Legacy PHP version must be one of: \"" . PHP_70_VERSION_ID . "\" (PHP 7.0), \"" . PHP_80_VERSION_ID . "\" (PHP 8.0), " . "\"" . PHP_81_VERSION_ID . "\" (PHP 8.1), \"" . PHP_82_VERSION_ID . "\" (PHP 8.2), \"" . PHP_83_VERSION_ID . "\" (PHP 8.3), " . "\"" . PHP_84_VERSION_ID . "\" (PHP 8.4), \"" . $tag->value . "\" provided" ); } $fileInfo->setMinimumPhpVersionIdCompatibility($tag->value ? (int) $tag->value : PHP_70_VERSION_ID); } else if ($tag->name === 'generate-class-entries') { $fileInfo->generateClassEntries = true; $fileInfo->declarationPrefix = $tag->value ? $tag->value . " " : ""; } else if ($tag->name === 'undocumentable') { $fileInfo->isUndocumentable = true; } } } // Generating class entries require generating function/method entries if ($fileInfo->generateClassEntries && !$fileInfo->generateFunctionEntries) { $fileInfo->generateFunctionEntries = true; } handleStatements($fileInfo, $stmts, $prettyPrinter); return $fileInfo; } function funcInfoToCode(FileInfo $fileInfo, FuncInfo $funcInfo): string { $code = ''; $returnType = $funcInfo->return->type; $isTentativeReturnType = $funcInfo->return->tentativeReturnType; $php81MinimumCompatibility = $fileInfo->getMinimumPhpVersionIdCompatibility() === null || $fileInfo->getMinimumPhpVersionIdCompatibility() >= PHP_81_VERSION_ID; if ($returnType !== null) { if ($isTentativeReturnType && !$php81MinimumCompatibility) { $code .= "#if (PHP_VERSION_ID >= " . PHP_81_VERSION_ID . ")\n"; } if (null !== $simpleReturnType = $returnType->tryToSimpleType()) { if ($simpleReturnType->isBuiltin) { $code .= sprintf( "%s(%s, %d, %d, %s, %d)\n", $isTentativeReturnType ? "ZEND_BEGIN_ARG_WITH_TENTATIVE_RETURN_TYPE_INFO_EX" : "ZEND_BEGIN_ARG_WITH_RETURN_TYPE_INFO_EX", $funcInfo->getArgInfoName(), $funcInfo->return->byRef, $funcInfo->numRequiredArgs, $simpleReturnType->toTypeCode(), $returnType->isNullable() ); } else { $code .= sprintf( "%s(%s, %d, %d, %s, %d)\n", $isTentativeReturnType ? "ZEND_BEGIN_ARG_WITH_TENTATIVE_RETURN_OBJ_INFO_EX" : "ZEND_BEGIN_ARG_WITH_RETURN_OBJ_INFO_EX", $funcInfo->getArgInfoName(), $funcInfo->return->byRef, $funcInfo->numRequiredArgs, $simpleReturnType->toEscapedName(), $returnType->isNullable() ); } } else { $arginfoType = $returnType->toArginfoType(); if ($arginfoType->hasClassType()) { $code .= sprintf( "%s(%s, %d, %d, %s, %s)\n", $isTentativeReturnType ? "ZEND_BEGIN_ARG_WITH_TENTATIVE_RETURN_OBJ_TYPE_MASK_EX" : "ZEND_BEGIN_ARG_WITH_RETURN_OBJ_TYPE_MASK_EX", $funcInfo->getArgInfoName(), $funcInfo->return->byRef, $funcInfo->numRequiredArgs, $arginfoType->toClassTypeString(), $arginfoType->toTypeMask() ); } else { $code .= sprintf( "%s(%s, %d, %d, %s)\n", $isTentativeReturnType ? "ZEND_BEGIN_ARG_WITH_TENTATIVE_RETURN_TYPE_MASK_EX" : "ZEND_BEGIN_ARG_WITH_RETURN_TYPE_MASK_EX", $funcInfo->getArgInfoName(), $funcInfo->return->byRef, $funcInfo->numRequiredArgs, $arginfoType->toTypeMask() ); } } if ($isTentativeReturnType && !$php81MinimumCompatibility) { $code .= sprintf( "#else\nZEND_BEGIN_ARG_INFO_EX(%s, 0, %d, %d)\n#endif\n", $funcInfo->getArgInfoName(), $funcInfo->return->byRef, $funcInfo->numRequiredArgs ); } } else { $code .= sprintf( "ZEND_BEGIN_ARG_INFO_EX(%s, 0, %d, %d)\n", $funcInfo->getArgInfoName(), $funcInfo->return->byRef, $funcInfo->numRequiredArgs ); } foreach ($funcInfo->args as $argInfo) { $argKind = $argInfo->isVariadic ? "ARG_VARIADIC" : "ARG"; $argDefaultKind = $argInfo->hasProperDefaultValue() ? "_WITH_DEFAULT_VALUE" : ""; $argType = $argInfo->type; if ($argType !== null) { if (null !== $simpleArgType = $argType->tryToSimpleType()) { if ($simpleArgType->isBuiltin) { $code .= sprintf( "\tZEND_%s_TYPE_INFO%s(%s, %s, %s, %d%s)\n", $argKind, $argDefaultKind, $argInfo->getSendByString(), $argInfo->name, $simpleArgType->toTypeCode(), $argType->isNullable(), $argInfo->hasProperDefaultValue() ? ", " . $argInfo->getDefaultValueAsArginfoString() : "" ); } else { $code .= sprintf( "\tZEND_%s_OBJ_INFO%s(%s, %s, %s, %d%s)\n", $argKind,$argDefaultKind, $argInfo->getSendByString(), $argInfo->name, $simpleArgType->toEscapedName(), $argType->isNullable(), $argInfo->hasProperDefaultValue() ? ", " . $argInfo->getDefaultValueAsArginfoString() : "" ); } } else { $arginfoType = $argType->toArginfoType(); if ($arginfoType->hasClassType()) { $code .= sprintf( "\tZEND_%s_OBJ_TYPE_MASK(%s, %s, %s, %s%s)\n", $argKind, $argInfo->getSendByString(), $argInfo->name, $arginfoType->toClassTypeString(), $arginfoType->toTypeMask(), !$argInfo->isVariadic ? ", " . $argInfo->getDefaultValueAsArginfoString() : "" ); } else { $code .= sprintf( "\tZEND_%s_TYPE_MASK(%s, %s, %s, %s)\n", $argKind, $argInfo->getSendByString(), $argInfo->name, $arginfoType->toTypeMask(), $argInfo->getDefaultValueAsArginfoString() ); } } } else { $code .= sprintf( "\tZEND_%s_INFO%s(%s, %s%s)\n", $argKind, $argDefaultKind, $argInfo->getSendByString(), $argInfo->name, $argInfo->hasProperDefaultValue() ? ", " . $argInfo->getDefaultValueAsArginfoString() : "" ); } } $code .= "ZEND_END_ARG_INFO()"; return $code . "\n"; } /** @param FuncInfo[] $generatedFuncInfos */ function findEquivalentFuncInfo(array $generatedFuncInfos, FuncInfo $funcInfo): ?FuncInfo { foreach ($generatedFuncInfos as $generatedFuncInfo) { if ($generatedFuncInfo->equalsApartFromNameAndRefcount($funcInfo)) { return $generatedFuncInfo; } } return null; } /** * @template T * @param iterable $infos * @param Closure(T): string|null $codeGenerator * @param ?string $parentCond */ function generateCodeWithConditions( iterable $infos, string $separator, Closure $codeGenerator, ?string $parentCond = null): string { $code = ""; // For combining the conditional blocks of the infos with the same condition $openCondition = null; foreach ($infos as $info) { $infoCode = $codeGenerator($info); if ($infoCode === null) { continue; } if ($info->cond && $info->cond !== $parentCond) { if ($openCondition !== null && $info->cond !== $openCondition ) { // Changing condition, end old $code .= "#endif\n"; $code .= $separator; $code .= "#if {$info->cond}\n"; $openCondition = $info->cond; } elseif ($openCondition === null) { // New condition with no existing one $code .= $separator; $code .= "#if {$info->cond}\n"; $openCondition = $info->cond; } else { // Staying in the same condition $code .= $separator; } $code .= $infoCode; } else { if ($openCondition !== null) { // Ending the condition $code .= "#endif\n"; $openCondition = null; } $code .= $separator; $code .= $infoCode; } } // The last info might have been in a conditional block if ($openCondition !== null) { $code .= "#endif\n"; } return $code; } /** * @param array $allConstInfos */ function generateArgInfoCode( string $stubFilenameWithoutExtension, FileInfo $fileInfo, array $allConstInfos, string $stubHash ): string { $code = "/* This is a generated file, edit the .stub.php file instead.\n" . " * Stub hash: $stubHash */\n"; $generatedFuncInfos = []; $argInfoCode = generateCodeWithConditions( $fileInfo->getAllFuncInfos(), "\n", static function (FuncInfo $funcInfo) use (&$generatedFuncInfos, $fileInfo) { /* If there already is an equivalent arginfo structure, only emit a #define */ if ($generatedFuncInfo = findEquivalentFuncInfo($generatedFuncInfos, $funcInfo)) { $code = sprintf( "#define %s %s\n", $funcInfo->getArgInfoName(), $generatedFuncInfo->getArgInfoName() ); } else { $code = funcInfoToCode($fileInfo, $funcInfo); } $generatedFuncInfos[] = $funcInfo; return $code; } ); if ($argInfoCode !== "") { $code .= "$argInfoCode\n"; } if ($fileInfo->generateFunctionEntries) { $framelessFunctionCode = generateCodeWithConditions( $fileInfo->getAllFuncInfos(), "\n", static function (FuncInfo $funcInfo) { $code = $funcInfo->getFramelessDeclaration($funcInfo); return $code; } ); if ($framelessFunctionCode !== "") { $code .= "$framelessFunctionCode\n"; } $generatedFunctionDeclarations = []; $code .= generateCodeWithConditions( $fileInfo->getAllFuncInfos(), "", static function (FuncInfo $funcInfo) use ($fileInfo, &$generatedFunctionDeclarations) { $key = $funcInfo->getDeclarationKey(); if (isset($generatedFunctionDeclarations[$key])) { return null; } $generatedFunctionDeclarations[$key] = true; return $fileInfo->declarationPrefix . $funcInfo->getDeclaration(); } ); $code .= generateFunctionEntries(null, $fileInfo->funcInfos); foreach ($fileInfo->classInfos as $classInfo) { $code .= generateFunctionEntries($classInfo->name, $classInfo->funcInfos, $classInfo->cond); } } $php80MinimumCompatibility = $fileInfo->getMinimumPhpVersionIdCompatibility() === null || $fileInfo->getMinimumPhpVersionIdCompatibility() >= PHP_80_VERSION_ID; if ($fileInfo->generateClassEntries) { if ($attributeInitializationCode = generateFunctionAttributeInitialization($fileInfo->funcInfos, $allConstInfos, $fileInfo->getMinimumPhpVersionIdCompatibility(), null)) { if (!$php80MinimumCompatibility) { $attributeInitializationCode = "\n#if (PHP_VERSION_ID >= " . PHP_80_VERSION_ID . ")" . $attributeInitializationCode . "#endif\n"; } } if ($attributeInitializationCode !== "" || !empty($fileInfo->constInfos)) { $code .= "\nstatic void register_{$stubFilenameWithoutExtension}_symbols(int module_number)\n"; $code .= "{\n"; foreach ($fileInfo->constInfos as $constInfo) { $code .= $constInfo->getDeclaration($allConstInfos); } if ($attributeInitializationCode !== "" && $fileInfo->constInfos) { $code .= "\n"; } $code .= $attributeInitializationCode; $code .= "}\n"; } $code .= generateClassEntryCode($fileInfo, $allConstInfos); } return $code; } /** @param array $allConstInfos */ function generateClassEntryCode(FileInfo $fileInfo, array $allConstInfos): string { $code = ""; foreach ($fileInfo->classInfos as $class) { $code .= "\n" . $class->getRegistration($allConstInfos); } return $code; } /** @param FuncInfo[] $funcInfos */ function generateFunctionEntries(?Name $className, array $funcInfos, ?string $cond = null): string { // No need to add anything if there are no function entries if ($funcInfos === []) { return ''; } $code = "\n"; if ($cond) { $code .= "#if {$cond}\n"; } $functionEntryName = "ext_functions"; if ($className) { $underscoreName = implode("_", $className->getParts()); $functionEntryName = "class_{$underscoreName}_methods"; } $code .= "static const zend_function_entry {$functionEntryName}[] = {\n"; $code .= generateCodeWithConditions($funcInfos, "", static function (FuncInfo $funcInfo) { return $funcInfo->getFunctionEntry(); }, $cond); $code .= "\tZEND_FE_END\n"; $code .= "};\n"; if ($cond) { $code .= "#endif\n"; } return $code; } /** @param iterable $funcInfos */ function generateFunctionAttributeInitialization(iterable $funcInfos, array $allConstInfos, ?int $phpVersionIdMinimumCompatibility, ?string $parentCond = null): string { return generateCodeWithConditions( $funcInfos, "", static function (FuncInfo $funcInfo) use ($allConstInfos, $phpVersionIdMinimumCompatibility) { $code = null; if ($funcInfo->name instanceof MethodName) { $functionTable = "&class_entry->function_table"; } else { $functionTable = "CG(function_table)"; } foreach ($funcInfo->attributes as $key => $attribute) { $code .= $attribute->generateCode( "zend_add_function_attribute(zend_hash_str_find_ptr($functionTable, \"" . $funcInfo->name->getNameForAttributes() . "\", sizeof(\"" . $funcInfo->name->getNameForAttributes() . "\") - 1)", "func_" . $funcInfo->name->getNameForAttributes() . "_$key", $allConstInfos, $phpVersionIdMinimumCompatibility ); } foreach ($funcInfo->args as $index => $arg) { foreach ($arg->attributes as $key => $attribute) { $code .= $attribute->generateCode( "zend_add_parameter_attribute(zend_hash_str_find_ptr($functionTable, \"" . $funcInfo->name->getNameForAttributes() . "\", sizeof(\"" . $funcInfo->name->getNameForAttributes() . "\") - 1), $index", "func_{$funcInfo->name->getNameForAttributes()}_arg{$index}_$key", $allConstInfos, $phpVersionIdMinimumCompatibility ); } } return $code; }, $parentCond ); } /** * @param iterable $constInfos * @param array $allConstInfos */ function generateConstantAttributeInitialization( iterable $constInfos, array $allConstInfos, ?int $phpVersionIdMinimumCompatibility, ?string $parentCond = null ): string { return generateCodeWithConditions( $constInfos, "", static function (ConstInfo $constInfo) use ($allConstInfos, $phpVersionIdMinimumCompatibility) { $code = null; foreach ($constInfo->attributes as $key => $attribute) { $code .= $attribute->generateCode( "zend_add_class_constant_attribute(class_entry, const_" . $constInfo->name->getDeclarationName(), "const_" . $constInfo->name->getDeclarationName() . "_$key", $allConstInfos, $phpVersionIdMinimumCompatibility ); } return $code; }, $parentCond ); } /** * @param iterable $propertyInfos * @param array $allConstInfos */ function generatePropertyAttributeInitialization( iterable $propertyInfos, array $allConstInfos, ?int $phpVersionIdMinimumCompatibility ): string { $code = ""; foreach ($propertyInfos as $propertyInfo) { foreach ($propertyInfo->attributes as $key => $attribute) { $code .= $attribute->generateCode( "zend_add_property_attribute(class_entry, property_" . $propertyInfo->name->getDeclarationName(), "property_" . $propertyInfo->name->getDeclarationName() . "_" . $key, $allConstInfos, $phpVersionIdMinimumCompatibility ); } } return $code; } /** @param array $funcMap */ function generateOptimizerInfo(array $funcMap): string { $code = "/* This is a generated file, edit the .stub.php files instead. */\n\n"; $code .= "static const func_info_t func_infos[] = {\n"; $code .= generateCodeWithConditions($funcMap, "", static function (FuncInfo $funcInfo) { return $funcInfo->getOptimizerInfo(); }); $code .= "};\n"; return $code; } /** * @param array $flagsByPhpVersions * @return string[] */ function generateVersionDependentFlagCode(string $codeTemplate, array $flagsByPhpVersions, ?int $phpVersionIdMinimumCompatibility): array { $phpVersions = ALL_PHP_VERSION_IDS; sort($phpVersions); $currentPhpVersion = end($phpVersions); // No version compatibility is needed if ($phpVersionIdMinimumCompatibility === null) { if (empty($flagsByPhpVersions[$currentPhpVersion])) { return []; } return [sprintf($codeTemplate, implode("|", $flagsByPhpVersions[$currentPhpVersion]))]; } // Remove flags which depend on a PHP version below the minimally supported one ksort($flagsByPhpVersions); $index = array_search($phpVersionIdMinimumCompatibility, array_keys($flagsByPhpVersions)); if ($index === false) { throw new Exception("Missing version dependent flags for PHP version ID \"$phpVersionIdMinimumCompatibility\""); } $flagsByPhpVersions = array_slice($flagsByPhpVersions, $index, null, true); // Remove empty version-specific flags $flagsByPhpVersions = array_filter( $flagsByPhpVersions, static function (array $value): bool { return !empty($value); }); // There are no version-specific flags if (empty($flagsByPhpVersions)) { return []; } // Remove version-specific flags which don't differ from the previous one $previousVersionId = null; foreach ($flagsByPhpVersions as $versionId => $versionFlags) { if ($previousVersionId !== null && $flagsByPhpVersions[$previousVersionId] === $versionFlags) { unset($flagsByPhpVersions[$versionId]); } else { $previousVersionId = $versionId; } } $flagCount = count($flagsByPhpVersions); // Do not add a condition unnecessarily when the only version is the same as the minimally supported one if ($flagCount === 1) { reset($flagsByPhpVersions); $firstVersion = key($flagsByPhpVersions); if ($firstVersion === $phpVersionIdMinimumCompatibility) { return [sprintf($codeTemplate, implode("|", reset($flagsByPhpVersions)))]; } } // Add the necessary conditions around the code using the version-specific flags $result = []; $i = 0; foreach (array_reverse($flagsByPhpVersions, true) as $version => $versionFlags) { $code = ""; $if = $i === 0 ? "#if" : "#elif"; $endif = $i === $flagCount - 1 ? "#endif\n" : ""; $code .= "$if (PHP_VERSION_ID >= $version)\n"; $code .= sprintf($codeTemplate, implode("|", $versionFlags)); $code .= $endif; $result[] = $code; $i++; } return $result; } /** * @param array $constMap * @param array $undocumentedConstMap * @return array */ function replacePredefinedConstants(string $targetDirectory, array $constMap, array &$undocumentedConstMap): array { /** @var array $documentedConstMap */ $documentedConstMap = []; /** @var array $predefinedConstants */ $predefinedConstants = []; $it = new RecursiveIteratorIterator( new RecursiveDirectoryIterator($targetDirectory), RecursiveIteratorIterator::LEAVES_ONLY ); foreach ($it as $file) { $pathName = $file->getPathName(); if (!preg_match('/(?:[\w\.]*constants[\w\._]*|tokens).xml$/i', basename($pathName))) { continue; } $xml = file_get_contents($pathName); if ($xml === false) { continue; } if (stripos($xml, "formatOutput = false; $doc->preserveWhiteSpace = true; $doc->validateOnParse = true; $success = $doc->loadXML($replacedXml); if (!$success) { echo "Failed opening $pathName\n"; continue; } $updated = false; foreach ($doc->getElementsByTagName("varlistentry") as $entry) { if (!$entry instanceof DOMElement) { continue; } foreach ($entry->getElementsByTagName("term") as $manualTermElement) { $manualConstantElement = $manualTermElement->getElementsByTagName("constant")->item(0); if (!$manualConstantElement instanceof DOMElement) { continue; } $manualConstantName = $manualConstantElement->textContent; $stubConstant = $constMap[$manualConstantName] ?? null; if ($stubConstant === null) { continue; } $documentedConstMap[$manualConstantName] = $manualConstantName; if ($entry->firstChild instanceof DOMText) { $indentationLevel = strlen(str_replace("\n", "", $entry->firstChild->textContent)); } else { $indentationLevel = 3; } $newTermElement = $stubConstant->getPredefinedConstantTerm($doc, $indentationLevel); if ($manualTermElement->textContent === $newTermElement->textContent) { continue; } $manualTermElement->parentNode->replaceChild($newTermElement, $manualTermElement); $updated = true; } } foreach ($doc->getElementsByTagName("row") as $row) { if (!$row instanceof DOMElement) { continue; } $entry = $row->getElementsByTagName("entry")->item(0); if (!$entry instanceof DOMElement) { continue; } foreach ($entry->getElementsByTagName("constant") as $manualConstantElement) { if (!$manualConstantElement instanceof DOMElement) { continue; } $manualConstantName = $manualConstantElement->textContent; $stubConstant = $constMap[$manualConstantName] ?? null; if ($stubConstant === null) { continue; } $documentedConstMap[$manualConstantName] = $manualConstantName; if ($row->firstChild instanceof DOMText) { $indentationLevel = strlen(str_replace("\n", "", $row->firstChild->textContent)); } else { $indentationLevel = 3; } $newEntryElement = $stubConstant->getPredefinedConstantEntry($doc, $indentationLevel); if ($entry->textContent === $newEntryElement->textContent) { continue; } $entry->parentNode->replaceChild($newEntryElement, $entry); $updated = true; } } if ($updated) { $replacedXml = $doc->saveXML(); $replacedXml = preg_replace( [ "/REPLACED-ENTITY-([A-Za-z0-9._{}%-]+?;)/", '//i', '//i', '//i', '//i', '//i', '//i', ], [ "&$1", "", "", "", "", "", "", ], $replacedXml ); $predefinedConstants[$pathName] = $replacedXml; } } $undocumentedConstMap = array_diff_key($constMap, $documentedConstMap); return $predefinedConstants; } /** * @param array $classMap * @param array $allConstInfos * @return array */ function generateClassSynopses(array $classMap, array $allConstInfos): array { $result = []; foreach ($classMap as $classInfo) { $classSynopsis = $classInfo->getClassSynopsisDocument($classMap, $allConstInfos); if ($classSynopsis !== null) { $result[ClassInfo::getClassSynopsisFilename($classInfo->name) . ".xml"] = $classSynopsis; } } return $result; } /** * @param array $classMap * @param array $allConstInfos * @param array $undocumentedClassMap * @return array */ function replaceClassSynopses( string $targetDirectory, array $classMap, array $allConstInfos, array &$undocumentedClassMap ): array { /** @var array $documentedClassMap */ $documentedClassMap = []; /** @var array $classSynopses */ $classSynopses = []; $it = new RecursiveIteratorIterator( new RecursiveDirectoryIterator($targetDirectory), RecursiveIteratorIterator::LEAVES_ONLY ); foreach ($it as $file) { $pathName = $file->getPathName(); if (!preg_match('/\.xml$/i', $pathName)) { continue; } $xml = file_get_contents($pathName); if ($xml === false) { continue; } if (stripos($xml, "formatOutput = false; $doc->preserveWhiteSpace = true; $doc->validateOnParse = true; $success = $doc->loadXML($replacedXml); if (!$success) { echo "Failed opening $pathName\n"; continue; } $classSynopsisElements = []; foreach ($doc->getElementsByTagName("classsynopsis") as $element) { $classSynopsisElements[] = $element; } foreach ($classSynopsisElements as $classSynopsis) { if (!$classSynopsis instanceof DOMElement) { continue; } $child = $classSynopsis->firstElementChild; if ($child === null) { continue; } $child = $child->lastElementChild; if ($child === null) { continue; } $className = $child->textContent; if (!isset($classMap[$className])) { continue; } $documentedClassMap[$className] = $className; $classInfo = $classMap[$className]; $newClassSynopsis = $classInfo->getClassSynopsisElement($doc, $classMap, $allConstInfos); if ($newClassSynopsis === null) { continue; } // Check if there is any change - short circuit if there is not any. if (replaceAndCompareXmls($doc, $classSynopsis, $newClassSynopsis)) { continue; } // Return the updated XML $replacedXml = $doc->saveXML(); $replacedXml = preg_replace( [ "/REPLACED-ENTITY-([A-Za-z0-9._{}%-]+?;)/", '//i', '//i', '//i', '//i', '//i', '//i', ], [ "&$1", "", "", "", "", "", "", ], $replacedXml ); $classSynopses[$pathName] = $replacedXml; } } $undocumentedClassMap = array_diff_key($classMap, $documentedClassMap); return $classSynopses; } function getReplacedSynopsisXml(string $xml): string { return preg_replace( [ "/&([A-Za-z0-9._{}%-]+?;)/", "/<(\/)*xi:([A-Za-z]+?)/" ], [ "REPLACED-ENTITY-$1", "<$1XI$2", ], $xml ); } /** * @param array $funcMap * @param array $aliasMap * @return array */ function generateMethodSynopses(array $funcMap, array $aliasMap): array { $result = []; foreach ($funcMap as $funcInfo) { $methodSynopsis = $funcInfo->getMethodSynopsisDocument($funcMap, $aliasMap); if ($methodSynopsis !== null) { $result[$funcInfo->name->getMethodSynopsisFilename() . ".xml"] = $methodSynopsis; } } return $result; } /** * @param array $funcMap * @param array $aliasMap * @param array $methodSynopsisWarnings * @param array $undocumentedFuncMap * @return array */ function replaceMethodSynopses( string $targetDirectory, array $funcMap, array $aliasMap, bool $isVerifyManual, array &$methodSynopsisWarnings, array &$undocumentedFuncMap ): array { /** @var array $documentedFuncMap */ $documentedFuncMap = []; /** @var array $methodSynopses */ $methodSynopses = []; $it = new RecursiveIteratorIterator( new RecursiveDirectoryIterator($targetDirectory), RecursiveIteratorIterator::LEAVES_ONLY ); foreach ($it as $file) { $pathName = $file->getPathName(); if (!preg_match('/\.xml$/i', $pathName)) { continue; } $xml = file_get_contents($pathName); if ($xml === false) { continue; } if ($isVerifyManual) { $matches = []; preg_match("/\s*([\w:]+)\s*<\/refname>\s*\s*&Alias;\s*<(?:function|methodname)>\s*([\w:]+)\s*<\/(?:function|methodname)>\s*<\/refpurpose>/i", $xml, $matches); $aliasName = $matches[1] ?? null; $alias = $funcMap[$aliasName] ?? null; $funcName = $matches[2] ?? null; $func = $funcMap[$funcName] ?? null; if ($alias && !$alias->isUndocumentable && ($func === null || $func->alias === null || $func->alias->__toString() !== $aliasName) && ($alias->alias === null || $alias->alias->__toString() !== $funcName) ) { $methodSynopsisWarnings[] = "$aliasName()" . ($alias->alias ? " is an alias of " . $alias->alias->__toString() . "(), but it" : "") . " is incorrectly documented as an alias for $funcName()"; } $matches = []; preg_match("/<(?:para|simpara)>\s*(?:&info.function.alias;|&info.method.alias;|&Alias;)\s+<(?:function|methodname)>\s*([\w:]+)\s*<\/(?:function|methodname)>/i", $xml, $matches); $descriptionFuncName = $matches[1] ?? null; $descriptionFunc = $funcMap[$descriptionFuncName] ?? null; if ($descriptionFunc && $funcName !== $descriptionFuncName) { $methodSynopsisWarnings[] = "Alias in the method synopsis description of $pathName doesn't match the alias in the "; } if ($aliasName) { $documentedFuncMap[$aliasName] = $aliasName; } } if (stripos($xml, "formatOutput = false; $doc->preserveWhiteSpace = true; $doc->validateOnParse = true; $success = $doc->loadXML($replacedXml); if (!$success) { echo "Failed opening $pathName\n"; continue; } $methodSynopsisElements = []; foreach ($doc->getElementsByTagName("constructorsynopsis") as $element) { $methodSynopsisElements[] = $element; } foreach ($doc->getElementsByTagName("destructorsynopsis") as $element) { $methodSynopsisElements[] = $element; } foreach ($doc->getElementsByTagName("methodsynopsis") as $element) { $methodSynopsisElements[] = $element; } foreach ($methodSynopsisElements as $methodSynopsis) { if (!$methodSynopsis instanceof DOMElement) { continue; } $item = $methodSynopsis->getElementsByTagName("methodname")->item(0); if (!$item instanceof DOMElement) { continue; } $funcName = $item->textContent; if (!isset($funcMap[$funcName])) { continue; } $funcInfo = $funcMap[$funcName]; $documentedFuncMap[$funcInfo->name->__toString()] = $funcInfo->name->__toString(); $newMethodSynopsis = $funcInfo->getMethodSynopsisElement($funcMap, $aliasMap, $doc); if ($newMethodSynopsis === null) { continue; } // Retrieve current signature $params = []; $list = $methodSynopsis->getElementsByTagName("methodparam"); foreach ($list as $i => $item) { if (!$item instanceof DOMElement) { continue; } $paramList = $item->getElementsByTagName("parameter"); if ($paramList->count() !== 1) { continue; } $paramName = $paramList->item(0)->textContent; $paramTypes = []; $paramList = $item->getElementsByTagName("type"); foreach ($paramList as $type) { if (!$type instanceof DOMElement) { continue; } $paramTypes[] = $type->textContent; } $params[$paramName] = ["index" => $i, "type" => $paramTypes]; } // Check if there is any change - short circuit if there is not any. if (replaceAndCompareXmls($doc, $methodSynopsis, $newMethodSynopsis)) { continue; } // Update parameter references $paramList = $doc->getElementsByTagName("parameter"); /** @var DOMElement $paramElement */ foreach ($paramList as $paramElement) { if ($paramElement->parentNode && $paramElement->parentNode->nodeName === "methodparam") { continue; } $name = $paramElement->textContent; if (!isset($params[$name])) { continue; } $index = $params[$name]["index"]; if (!isset($funcInfo->args[$index])) { continue; } $paramElement->textContent = $funcInfo->args[$index]->name; } // Return the updated XML $replacedXml = $doc->saveXML(); $replacedXml = preg_replace( [ "/REPLACED-ENTITY-([A-Za-z0-9._{}%-]+?;)/", '//i', '//i', ], [ "&$1", "", "", ], $replacedXml ); $methodSynopses[$pathName] = $replacedXml; } } $undocumentedFuncMap = array_diff_key($funcMap, $documentedFuncMap); return $methodSynopses; } function replaceAndCompareXmls(DOMDocument $doc, DOMElement $originalSynopsis, DOMElement $newSynopsis): bool { $docComparator = new DOMDocument(); $docComparator->preserveWhiteSpace = false; $docComparator->formatOutput = true; $xml1 = $doc->saveXML($originalSynopsis); $xml1 = getReplacedSynopsisXml($xml1); $docComparator->loadXML($xml1); $xml1 = $docComparator->saveXML(); $originalSynopsis->parentNode->replaceChild($newSynopsis, $originalSynopsis); $xml2 = $doc->saveXML($newSynopsis); $xml2 = getReplacedSynopsisXml($xml2); $docComparator->loadXML($xml2); $xml2 = $docComparator->saveXML(); return $xml1 === $xml2; } function installPhpParser(string $version, string $phpParserDir) { $lockFile = __DIR__ . "/PHP-Parser-install-lock"; $lockFd = fopen($lockFile, 'w+'); if (!flock($lockFd, LOCK_EX)) { throw new Exception("Failed to acquire installation lock"); } try { // Check whether a parallel process has already installed PHP-Parser. if (is_dir($phpParserDir)) { return; } $cwd = getcwd(); chdir(__DIR__); $tarName = "v$version.tar.gz"; passthru("wget https://github.com/nikic/PHP-Parser/archive/$tarName", $exit); if ($exit !== 0) { passthru("curl -LO https://github.com/nikic/PHP-Parser/archive/$tarName", $exit); } if ($exit !== 0) { throw new Exception("Failed to download PHP-Parser tarball"); } if (!mkdir($phpParserDir)) { throw new Exception("Failed to create directory $phpParserDir"); } passthru("tar xvzf $tarName -C PHP-Parser-$version --strip-components 1", $exit); if ($exit !== 0) { throw new Exception("Failed to extract PHP-Parser tarball"); } unlink(__DIR__ . "/$tarName"); chdir($cwd); } finally { flock($lockFd, LOCK_UN); @unlink($lockFile); } } function initPhpParser() { static $isInitialized = false; if ($isInitialized) { return; } if (!extension_loaded("tokenizer")) { throw new Exception("The \"tokenizer\" extension is not available"); } $isInitialized = true; $version = "5.3.1"; $phpParserDir = __DIR__ . "/PHP-Parser-$version"; if (!is_dir($phpParserDir)) { installPhpParser($version, $phpParserDir); } spl_autoload_register(static function(string $class) use ($phpParserDir) { if (strpos($class, "PhpParser\\") === 0) { $fileName = $phpParserDir . "/lib/" . str_replace("\\", "/", $class) . ".php"; require $fileName; } }); } $optind = null; $options = getopt( "fh", [ "force-regeneration", "parameter-stats", "help", "verify", "verify-manual", "replace-predefined-constants", "generate-classsynopses", "replace-classsynopses", "generate-methodsynopses", "replace-methodsynopses", "generate-optimizer-info", ], $optind ); $context = new Context; $printParameterStats = isset($options["parameter-stats"]); $verify = isset($options["verify"]); $verifyManual = isset($options["verify-manual"]); $replacePredefinedConstants = isset($options["replace-predefined-constants"]); $generateClassSynopses = isset($options["generate-classsynopses"]); $replaceClassSynopses = isset($options["replace-classsynopses"]); $generateMethodSynopses = isset($options["generate-methodsynopses"]); $replaceMethodSynopses = isset($options["replace-methodsynopses"]); $generateOptimizerInfo = isset($options["generate-optimizer-info"]); $context->forceRegeneration = isset($options["f"]) || isset($options["force-regeneration"]); $context->forceParse = $context->forceRegeneration || $printParameterStats || $verify || $verifyManual || $replacePredefinedConstants || $generateClassSynopses || $generateOptimizerInfo || $replaceClassSynopses || $generateMethodSynopses || $replaceMethodSynopses; if (isset($options["h"]) || isset($options["help"])) { die("\nUsage: gen_stub.php [ -f | --force-regeneration ] [ --replace-predefined-constants ] [ --generate-classsynopses ] [ --replace-classsynopses ] [ --generate-methodsynopses ] [ --replace-methodsynopses ] [ --parameter-stats ] [ --verify ] [ --verify-manual ] [ --generate-optimizer-info ] [ -h | --help ] [ name.stub.php | directory ] [ directory ]\n\n"); } $locations = array_slice($argv, $optind); $locationCount = count($locations); if ($replacePredefinedConstants && $locationCount < 2) { die("At least one source stub path and a target manual directory has to be provided:\n./build/gen_stub.php --replace-predefined-constants ./ ../doc-en/\n"); } if ($replaceClassSynopses && $locationCount < 2) { die("At least one source stub path and a target manual directory has to be provided:\n./build/gen_stub.php --replace-classsynopses ./ ../doc-en/\n"); } if ($generateMethodSynopses && $locationCount < 2) { die("At least one source stub path and a target manual directory has to be provided:\n./build/gen_stub.php --generate-methodsynopses ./ ../doc-en/\n"); } if ($replaceMethodSynopses && $locationCount < 2) { die("At least one source stub path and a target manual directory has to be provided:\n./build/gen_stub.php --replace-methodsynopses ./ ../doc-en/\n"); } if ($verifyManual && $locationCount < 2) { die("At least one source stub path and a target manual directory has to be provided:\n./build/gen_stub.php --verify-manual ./ ../doc-en/\n"); } $manualTarget = null; if ($replacePredefinedConstants || $replaceClassSynopses || $generateMethodSynopses || $replaceMethodSynopses || $verifyManual) { $manualTarget = array_pop($locations); } if ($locations === []) { $locations = ['.']; } $fileInfos = []; foreach (array_unique($locations) as $location) { if (is_file($location)) { // Generate single file. $fileInfo = processStubFile($location, $context); if ($fileInfo) { $fileInfos[] = $fileInfo; } } else if (is_dir($location)) { array_push($fileInfos, ...processDirectory($location, $context)); } else { echo "$location is neither a file nor a directory.\n"; exit(1); } } if ($printParameterStats) { $parameterStats = []; foreach ($fileInfos as $fileInfo) { foreach ($fileInfo->getAllFuncInfos() as $funcInfo) { foreach ($funcInfo->args as $argInfo) { if (!isset($parameterStats[$argInfo->name])) { $parameterStats[$argInfo->name] = 0; } $parameterStats[$argInfo->name]++; } } } arsort($parameterStats); echo json_encode($parameterStats, JSON_PRETTY_PRINT), "\n"; } /** @var array $classMap */ $classMap = []; /** @var array $funcMap */ $funcMap = []; /** @var array $aliasMap */ $aliasMap = []; /** @var array $undocumentedConstMap */ $undocumentedConstMap = []; /** @var array $undocumentedClassMap */ $undocumentedClassMap = []; /** @var array $undocumentedFuncMap */ $undocumentedFuncMap = []; /** @var array $methodSynopsisWarnings */ $methodSynopsisWarnings = []; foreach ($fileInfos as $fileInfo) { foreach ($fileInfo->getAllFuncInfos() as $funcInfo) { $funcMap[$funcInfo->name->__toString()] = $funcInfo; // TODO: Don't use aliasMap for methodsynopsis? if ($funcInfo->aliasType === "alias") { $aliasMap[$funcInfo->alias->__toString()] = $funcInfo; } } foreach ($fileInfo->classInfos as $classInfo) { $classMap[$classInfo->name->__toString()] = $classInfo; if ($classInfo->alias !== null) { $classMap[$classInfo->alias] = $classInfo; } } } if ($verify) { $errors = []; foreach ($funcMap as $aliasFunc) { if (!$aliasFunc->alias || $aliasFunc->aliasType !== "alias") { continue; } if (!isset($funcMap[$aliasFunc->alias->__toString()])) { $errors[] = "Aliased function {$aliasFunc->alias}() cannot be found"; continue; } if (!$aliasFunc->verify) { continue; } $aliasedFunc = $funcMap[$aliasFunc->alias->__toString()]; $aliasedArgs = $aliasedFunc->args; $aliasArgs = $aliasFunc->args; if ($aliasFunc->isInstanceMethod() !== $aliasedFunc->isInstanceMethod()) { if ($aliasFunc->isInstanceMethod()) { $aliasedArgs = array_slice($aliasedArgs, 1); } if ($aliasedFunc->isInstanceMethod()) { $aliasArgs = array_slice($aliasArgs, 1); } } array_map( function(?ArgInfo $aliasArg, ?ArgInfo $aliasedArg) use ($aliasFunc, $aliasedFunc, &$errors) { if ($aliasArg === null) { assert($aliasedArg !== null); $errors[] = "{$aliasFunc->name}(): Argument \$$aliasedArg->name of aliased function {$aliasedFunc->name}() is missing"; return null; } if ($aliasedArg === null) { $errors[] = "{$aliasedFunc->name}(): Argument \$$aliasArg->name of alias function {$aliasFunc->name}() is missing"; return null; } if ($aliasArg->name !== $aliasedArg->name) { $errors[] = "{$aliasFunc->name}(): Argument \$$aliasArg->name and argument \$$aliasedArg->name of aliased function {$aliasedFunc->name}() must have the same name"; return null; } if ($aliasArg->type != $aliasedArg->type) { $errors[] = "{$aliasFunc->name}(): Argument \$$aliasArg->name and argument \$$aliasedArg->name of aliased function {$aliasedFunc->name}() must have the same type"; } if ($aliasArg->defaultValue !== $aliasedArg->defaultValue) { $errors[] = "{$aliasFunc->name}(): Argument \$$aliasArg->name and argument \$$aliasedArg->name of aliased function {$aliasedFunc->name}() must have the same default value"; } }, $aliasArgs, $aliasedArgs ); $aliasedReturn = $aliasedFunc->return; $aliasReturn = $aliasFunc->return; if (!$aliasedFunc->name->isConstructor() && !$aliasFunc->name->isConstructor()) { $aliasedReturnType = $aliasedReturn->type ?? $aliasedReturn->phpDocType; $aliasReturnType = $aliasReturn->type ?? $aliasReturn->phpDocType; if ($aliasReturnType != $aliasedReturnType) { $errors[] = "{$aliasFunc->name}() and {$aliasedFunc->name}() must have the same return type"; } } $aliasedPhpDocReturnType = $aliasedReturn->phpDocType; $aliasPhpDocReturnType = $aliasReturn->phpDocType; if ($aliasedPhpDocReturnType != $aliasPhpDocReturnType && $aliasedPhpDocReturnType != $aliasReturn->type && $aliasPhpDocReturnType != $aliasedReturn->type) { $errors[] = "{$aliasFunc->name}() and {$aliasedFunc->name}() must have the same PHPDoc return type"; } } echo implode("\n", $errors); if (!empty($errors)) { echo "\n"; exit(1); } } if ($replacePredefinedConstants || $verifyManual) { $predefinedConstants = replacePredefinedConstants($manualTarget, $context->allConstInfos, $undocumentedConstMap); if ($replacePredefinedConstants) { foreach ($predefinedConstants as $filename => $content) { if (file_put_contents($filename, $content)) { echo "Saved $filename\n"; } } } } if ($generateClassSynopses) { $classSynopsesDirectory = getcwd() . "/classsynopses"; $classSynopses = generateClassSynopses($classMap, $context->allConstInfos); if (!empty($classSynopses)) { if (!file_exists($classSynopsesDirectory)) { mkdir($classSynopsesDirectory); } foreach ($classSynopses as $filename => $content) { if (file_put_contents("$classSynopsesDirectory/$filename", $content)) { echo "Saved $filename\n"; } } } } if ($replaceClassSynopses || $verifyManual) { $classSynopses = replaceClassSynopses($manualTarget, $classMap, $context->allConstInfos, $undocumentedClassMap); if ($replaceClassSynopses) { foreach ($classSynopses as $filename => $content) { if (file_put_contents($filename, $content)) { echo "Saved $filename\n"; } } } } if ($generateMethodSynopses) { $methodSynopses = generateMethodSynopses($funcMap, $aliasMap); if (!file_exists($manualTarget)) { mkdir($manualTarget); } foreach ($methodSynopses as $filename => $content) { $path = "$manualTarget/$filename"; if (!file_exists($path)) { if (!file_exists(dirname($path))) { mkdir(dirname($path)); } if (file_put_contents($path, $content)) { echo "Saved $filename\n"; } } } } if ($replaceMethodSynopses || $verifyManual) { $methodSynopses = replaceMethodSynopses($manualTarget, $funcMap, $aliasMap, $verifyManual, $methodSynopsisWarnings, $undocumentedFuncMap); if ($replaceMethodSynopses) { foreach ($methodSynopses as $filename => $content) { if (file_put_contents($filename, $content)) { echo "Saved $filename\n"; } } } } if ($generateOptimizerInfo) { $filename = dirname(__FILE__, 2) . "/Zend/Optimizer/zend_func_infos.h"; $optimizerInfo = generateOptimizerInfo($funcMap); if (file_put_contents($filename, $optimizerInfo)) { echo "Saved $filename\n"; } } if ($verifyManual) { foreach ($undocumentedConstMap as $constName => $info) { if ($info->name->isClassConst() || $info->isUndocumentable) { continue; } echo "Warning: Missing predefined constant for $constName\n"; } foreach ($methodSynopsisWarnings as $warning) { echo "Warning: $warning\n"; } foreach ($undocumentedClassMap as $className => $info) { if (!$info->isUndocumentable) { echo "Warning: Missing class synopsis for $className\n"; } } foreach ($undocumentedFuncMap as $functionName => $info) { if (!$info->isUndocumentable) { echo "Warning: Missing method synopsis for $functionName()\n"; } } }