<?php /* * This file is part of the Symfony package. * * (c) Fabien Potencier <fabien@symfony.com> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Symfony\Component\VarDumper\Caster; use FFI\CData; use FFI\CType; use Symfony\Component\VarDumper\Cloner\Stub; /** * Casts FFI extension classes to array representation. * * @author Nesmeyanov Kirill <nesk@xakep.ru> */ final class FFICaster { /** * In case of "char*" contains a string, the length of which depends on * some other parameter, then during the generation of the string it is * possible to go beyond the allowable memory area. * * This restriction serves to ensure that processing does not take * up the entire allowable PHP memory limit. */ private const MAX_STRING_LENGTH = 255; public static function castCTypeOrCData(CData|CType $data, array $args, Stub $stub): array { if ($data instanceof CType) { $type = $data; $data = null; } else { $type = \FFI::typeof($data); } $stub->class = sprintf('%s<%s> size %d align %d', ($data ?? $type)::class, $type->getName(), $type->getSize(), $type->getAlignment()); return match ($type->getKind()) { CType::TYPE_FLOAT, CType::TYPE_DOUBLE, \defined('\FFI\CType::TYPE_LONGDOUBLE') ? CType::TYPE_LONGDOUBLE : -1, CType::TYPE_UINT8, CType::TYPE_SINT8, CType::TYPE_UINT16, CType::TYPE_SINT16, CType::TYPE_UINT32, CType::TYPE_SINT32, CType::TYPE_UINT64, CType::TYPE_SINT64, CType::TYPE_BOOL, CType::TYPE_CHAR, CType::TYPE_ENUM => null !== $data ? [Caster::PREFIX_VIRTUAL.'cdata' => $data->cdata] : [], CType::TYPE_POINTER => self::castFFIPointer($stub, $type, $data), CType::TYPE_STRUCT => self::castFFIStructLike($type, $data), CType::TYPE_FUNC => self::castFFIFunction($stub, $type), default => $args, }; } private static function castFFIFunction(Stub $stub, CType $type): array { $arguments = []; for ($i = 0, $count = $type->getFuncParameterCount(); $i < $count; ++$i) { $param = $type->getFuncParameterType($i); $arguments[] = $param->getName(); } $abi = match ($type->getFuncABI()) { CType::ABI_DEFAULT, CType::ABI_CDECL => '[cdecl]', CType::ABI_FASTCALL => '[fastcall]', CType::ABI_THISCALL => '[thiscall]', CType::ABI_STDCALL => '[stdcall]', CType::ABI_PASCAL => '[pascal]', CType::ABI_REGISTER => '[register]', CType::ABI_MS => '[ms]', CType::ABI_SYSV => '[sysv]', CType::ABI_VECTORCALL => '[vectorcall]', default => '[unknown abi]' }; $returnType = $type->getFuncReturnType(); $stub->class = $abi.' callable('.implode(', ', $arguments).'): ' .$returnType->getName(); return [Caster::PREFIX_VIRTUAL.'returnType' => $returnType]; } private static function castFFIPointer(Stub $stub, CType $type, CData $data = null): array { $ptr = $type->getPointerType(); if (null === $data) { return [Caster::PREFIX_VIRTUAL.'0' => $ptr]; } return match ($ptr->getKind()) { CType::TYPE_CHAR => [Caster::PREFIX_VIRTUAL.'cdata' => self::castFFIStringValue($data)], CType::TYPE_FUNC => self::castFFIFunction($stub, $ptr), default => [Caster::PREFIX_VIRTUAL.'cdata' => $data[0]], }; } private static function castFFIStringValue(CData $data): string|CutStub { $result = []; for ($i = 0; $i < self::MAX_STRING_LENGTH; ++$i) { $result[$i] = $data[$i]; if ("\0" === $result[$i]) { return implode('', $result); } } $string = implode('', $result); $stub = new CutStub($string); $stub->cut = -1; $stub->value = $string; return $stub; } private static function castFFIStructLike(CType $type, CData $data = null): array { $isUnion = ($type->getAttributes() & CType::ATTR_UNION) === CType::ATTR_UNION; $result = []; foreach ($type->getStructFieldNames() as $name) { $field = $type->getStructFieldType($name); // Retrieving the value of a field from a union containing // a pointer is not a safe operation, because may contain // incorrect data. $isUnsafe = $isUnion && CType::TYPE_POINTER === $field->getKind(); if ($isUnsafe) { $result[Caster::PREFIX_VIRTUAL.$name.'?'] = $field; } elseif (null === $data) { $result[Caster::PREFIX_VIRTUAL.$name] = $field; } else { $fieldName = $data->{$name} instanceof CData ? '' : $field->getName().' '; $result[Caster::PREFIX_VIRTUAL.$fieldName.$name] = $data->{$name}; } } return $result; } }
Name | Type | Size | Permission | Actions |
---|---|---|---|---|
AmqpCaster.php | File | 6.54 KB | 0644 |
|
ArgsStub.php | File | 2.24 KB | 0644 |
|
Caster.php | File | 5.85 KB | 0644 |
|
ClassStub.php | File | 3.76 KB | 0644 |
|
ConstStub.php | File | 735 B | 0644 |
|
CutArrayStub.php | File | 696 B | 0644 |
|
CutStub.php | File | 1.88 KB | 0644 |
|
DOMCaster.php | File | 8.81 KB | 0644 |
|
DateCaster.php | File | 4.77 KB | 0644 |
|
DoctrineCaster.php | File | 1.63 KB | 0644 |
|
DsCaster.php | File | 1.55 KB | 0644 |
|
DsPairStub.php | File | 630 B | 0644 |
|
EnumStub.php | File | 637 B | 0644 |
|
ExceptionCaster.php | File | 15.82 KB | 0644 |
|
FFICaster.php | File | 5.17 KB | 0644 |
|
FiberCaster.php | File | 1.04 KB | 0644 |
|
FrameStub.php | File | 740 B | 0644 |
|
GmpCaster.php | File | 749 B | 0644 |
|
ImagineCaster.php | File | 949 B | 0644 |
|
ImgStub.php | File | 637 B | 0644 |
|
IntlCaster.php | File | 8.72 KB | 0644 |
|
LinkStub.php | File | 3.27 KB | 0644 |
|
MemcachedCaster.php | File | 2.23 KB | 0644 |
|
MysqliCaster.php | File | 691 B | 0644 |
|
PdoCaster.php | File | 3.47 KB | 0644 |
|
PgSqlCaster.php | File | 5.36 KB | 0644 |
|
ProxyManagerCaster.php | File | 721 B | 0644 |
|
RdKafkaCaster.php | File | 4.54 KB | 0644 |
|
RedisCaster.php | File | 5.1 KB | 0644 |
|
ReflectionCaster.php | File | 14.66 KB | 0644 |
|
ResourceCaster.php | File | 2.88 KB | 0644 |
|
SplCaster.php | File | 7.23 KB | 0644 |
|
StubCaster.php | File | 2.09 KB | 0644 |
|
SymfonyCaster.php | File | 3.47 KB | 0644 |
|
TraceStub.php | File | 963 B | 0644 |
|
UuidCaster.php | File | 668 B | 0644 |
|
XmlReaderCaster.php | File | 3.22 KB | 0644 |
|
XmlResourceCaster.php | File | 2.49 KB | 0644 |
|