2020-05-18 21:13:27 +02:00
|
|
|
<?php
|
2021-07-12 19:09:20 +02:00
|
|
|
|
2020-05-18 21:13:27 +02:00
|
|
|
namespace Psalm\Internal\Analyzer\Statements\Expression;
|
|
|
|
|
|
|
|
use PhpParser;
|
|
|
|
use Psalm\CodeLocation;
|
|
|
|
use Psalm\Context;
|
2021-06-08 04:55:21 +02:00
|
|
|
use Psalm\Internal\Analyzer\Statements\ExpressionAnalyzer;
|
|
|
|
use Psalm\Internal\Analyzer\StatementsAnalyzer;
|
2021-07-12 19:09:20 +02:00
|
|
|
use Psalm\Internal\Codebase\VariableUseGraph;
|
|
|
|
use Psalm\Internal\DataFlow\DataFlowNode;
|
2020-05-18 21:13:27 +02:00
|
|
|
use Psalm\Issue\InvalidOperand;
|
|
|
|
use Psalm\Issue\PossiblyInvalidOperand;
|
|
|
|
use Psalm\IssueBuffer;
|
|
|
|
use Psalm\Type;
|
|
|
|
use Psalm\Type\Atomic\TFloat;
|
|
|
|
use Psalm\Type\Atomic\TInt;
|
2021-12-13 04:45:57 +01:00
|
|
|
use Psalm\Type\Atomic\TLiteralFloat;
|
|
|
|
use Psalm\Type\Atomic\TLiteralInt;
|
|
|
|
use Psalm\Type\Atomic\TLiteralString;
|
2020-05-18 21:13:27 +02:00
|
|
|
use Psalm\Type\Atomic\TString;
|
2021-12-13 16:28:14 +01:00
|
|
|
use Psalm\Type\Union;
|
2020-05-18 21:13:27 +02:00
|
|
|
|
2022-01-03 07:55:32 +01:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2020-05-18 21:13:27 +02:00
|
|
|
class BitwiseNotAnalyzer
|
|
|
|
{
|
|
|
|
public static function analyze(
|
|
|
|
StatementsAnalyzer $statements_analyzer,
|
|
|
|
PhpParser\Node\Expr\BitwiseNot $stmt,
|
|
|
|
Context $context
|
2021-07-12 19:09:20 +02:00
|
|
|
): bool {
|
2020-05-18 21:13:27 +02:00
|
|
|
if (ExpressionAnalyzer::analyze($statements_analyzer, $stmt->expr, $context) === false) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!($stmt_expr_type = $statements_analyzer->node_data->getType($stmt->expr))) {
|
2021-12-13 16:28:14 +01:00
|
|
|
$statements_analyzer->node_data->setType($stmt, new Union([new TInt(), new TString()]));
|
2020-05-18 21:13:27 +02:00
|
|
|
} elseif ($stmt_expr_type->isMixed()) {
|
|
|
|
$statements_analyzer->node_data->setType($stmt, Type::getMixed());
|
|
|
|
} else {
|
|
|
|
$acceptable_types = [];
|
|
|
|
$unacceptable_type = null;
|
|
|
|
$has_valid_operand = false;
|
|
|
|
|
2022-10-03 10:45:36 +02:00
|
|
|
$stmt_expr_type = $stmt_expr_type->getBuilder();
|
2020-05-18 21:13:27 +02:00
|
|
|
foreach ($stmt_expr_type->getAtomicTypes() as $type_string => $type_part) {
|
|
|
|
if ($type_part instanceof TInt || $type_part instanceof TString) {
|
2021-12-13 04:45:57 +01:00
|
|
|
if ($type_part instanceof TLiteralInt) {
|
2022-10-03 13:58:01 +02:00
|
|
|
$type_part = new TLiteralInt(~$type_part->value);
|
2021-12-13 04:45:57 +01:00
|
|
|
} elseif ($type_part instanceof TLiteralString) {
|
2022-10-03 13:58:01 +02:00
|
|
|
$type_part = new TLiteralString(~$type_part->value);
|
2020-05-18 21:13:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$acceptable_types[] = $type_part;
|
|
|
|
$has_valid_operand = true;
|
|
|
|
} elseif ($type_part instanceof TFloat) {
|
2021-12-13 04:45:57 +01:00
|
|
|
$type_part = ($type_part instanceof TLiteralFloat) ?
|
|
|
|
new TLiteralInt(~$type_part->value) :
|
2020-05-18 21:13:27 +02:00
|
|
|
new TInt;
|
|
|
|
|
|
|
|
$stmt_expr_type->removeType($type_string);
|
|
|
|
$stmt_expr_type->addType($type_part);
|
|
|
|
|
|
|
|
$acceptable_types[] = $type_part;
|
|
|
|
$has_valid_operand = true;
|
|
|
|
} elseif (!$unacceptable_type) {
|
|
|
|
$unacceptable_type = $type_part;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($unacceptable_type || !$acceptable_types) {
|
|
|
|
$message = 'Cannot negate a non-numeric non-string type ' . $unacceptable_type;
|
|
|
|
if ($has_valid_operand) {
|
2021-11-29 20:54:17 +01:00
|
|
|
IssueBuffer::maybeAdd(
|
2020-05-18 21:13:27 +02:00
|
|
|
new PossiblyInvalidOperand(
|
|
|
|
$message,
|
|
|
|
new CodeLocation($statements_analyzer, $stmt)
|
|
|
|
),
|
|
|
|
$statements_analyzer->getSuppressedIssues()
|
2021-11-29 20:54:17 +01:00
|
|
|
);
|
2020-05-18 21:13:27 +02:00
|
|
|
} else {
|
2021-11-29 20:54:17 +01:00
|
|
|
IssueBuffer::maybeAdd(
|
2020-05-18 21:13:27 +02:00
|
|
|
new InvalidOperand(
|
|
|
|
$message,
|
|
|
|
new CodeLocation($statements_analyzer, $stmt)
|
|
|
|
),
|
|
|
|
$statements_analyzer->getSuppressedIssues()
|
2021-11-29 20:54:17 +01:00
|
|
|
);
|
2020-05-18 21:13:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$statements_analyzer->node_data->setType($stmt, Type::getMixed());
|
|
|
|
} else {
|
2021-12-13 16:28:14 +01:00
|
|
|
$statements_analyzer->node_data->setType($stmt, new Union($acceptable_types));
|
2020-05-18 21:13:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-12 19:09:20 +02:00
|
|
|
self::addDataFlow($statements_analyzer, $stmt, $stmt->expr);
|
|
|
|
|
2020-05-18 21:13:27 +02:00
|
|
|
return true;
|
|
|
|
}
|
2021-07-12 19:09:20 +02:00
|
|
|
|
|
|
|
private static function addDataFlow(
|
|
|
|
StatementsAnalyzer $statements_analyzer,
|
|
|
|
PhpParser\Node\Expr $stmt,
|
2021-09-26 20:54:37 +02:00
|
|
|
PhpParser\Node\Expr $value
|
2021-07-12 19:09:20 +02:00
|
|
|
): void {
|
|
|
|
$result_type = $statements_analyzer->node_data->getType($stmt);
|
|
|
|
if ($statements_analyzer->data_flow_graph instanceof VariableUseGraph && $result_type) {
|
|
|
|
$var_location = new CodeLocation($statements_analyzer, $stmt);
|
|
|
|
|
|
|
|
$stmt_value_type = $statements_analyzer->node_data->getType($value);
|
|
|
|
|
2021-09-26 20:54:37 +02:00
|
|
|
$new_parent_node = DataFlowNode::getForAssignment('bitwisenot', $var_location);
|
2021-07-12 19:09:20 +02:00
|
|
|
$statements_analyzer->data_flow_graph->addNode($new_parent_node);
|
2022-11-04 19:04:23 +01:00
|
|
|
$result_type = $result_type->setParentNodes([
|
2021-07-12 19:09:20 +02:00
|
|
|
$new_parent_node->id => $new_parent_node,
|
2022-11-04 19:04:23 +01:00
|
|
|
]);
|
|
|
|
$statements_analyzer->node_data->setType($stmt, $result_type);
|
2021-07-12 19:09:20 +02:00
|
|
|
|
|
|
|
if ($stmt_value_type && $stmt_value_type->parent_nodes) {
|
|
|
|
foreach ($stmt_value_type->parent_nodes as $parent_node) {
|
2021-09-26 20:54:37 +02:00
|
|
|
$statements_analyzer->data_flow_graph->addPath($parent_node, $new_parent_node, 'bitwisenot');
|
2021-07-12 19:09:20 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-05-18 21:13:27 +02:00
|
|
|
}
|