mirror of
https://github.com/github/codeql.git
synced 2025-12-17 01:03:14 +01:00
102 lines
4.0 KiB
C#
102 lines
4.0 KiB
C#
using Microsoft.CodeAnalysis;
|
|
using Microsoft.CodeAnalysis.CSharp;
|
|
using Microsoft.CodeAnalysis.CSharp.Syntax;
|
|
using Semmle.Extraction.CSharp.Populators;
|
|
using Semmle.Extraction.Kinds;
|
|
|
|
namespace Semmle.Extraction.CSharp.Entities.Expressions
|
|
{
|
|
class MemberAccess : Expression
|
|
{
|
|
private MemberAccess(ExpressionNodeInfo info, ExpressionSyntax qualifier, ISymbol target) : base(info)
|
|
{
|
|
Qualifier = Create(cx, qualifier, this, -1);
|
|
|
|
if (target == null)
|
|
{
|
|
if (info.Kind != ExprKind.DYNAMIC_MEMBER_ACCESS)
|
|
cx.ModelError(info.Node, "Could not determine target for member access");
|
|
}
|
|
else
|
|
{
|
|
cx.Emit(Tuples.expr_access(this, cx.CreateEntity(target)));
|
|
}
|
|
}
|
|
|
|
public static Expression Create(ExpressionNodeInfo info, ConditionalAccessExpressionSyntax node) =>
|
|
// The qualifier is located by walking the syntax tree.
|
|
// `node.WhenNotNull` will contain a MemberBindingExpressionSyntax, calling the method below.
|
|
Create(info.Context, node.WhenNotNull, info.Parent, info.Child);
|
|
|
|
public static Expression Create(ExpressionNodeInfo info, MemberBindingExpressionSyntax node)
|
|
{
|
|
var expr = Create(info, FindConditionalQualifier(node), node.Name);
|
|
expr.MakeConditional(info.Context.TrapWriter.Writer);
|
|
return expr;
|
|
}
|
|
|
|
public static Expression Create(ExpressionNodeInfo info, MemberAccessExpressionSyntax node) =>
|
|
Create(info, node.Expression, node.Name);
|
|
|
|
static Expression Create(ExpressionNodeInfo info, ExpressionSyntax expression, SimpleNameSyntax name)
|
|
{
|
|
if (IsDynamic(info.Context, expression))
|
|
{
|
|
var expr = new MemberAccess(info.SetKind(ExprKind.DYNAMIC_MEMBER_ACCESS), expression, null);
|
|
info.Context.Emit(Tuples.dynamic_member_name(expr, name.Identifier.Text));
|
|
return expr;
|
|
}
|
|
|
|
var target = info.SymbolInfo;
|
|
|
|
if (target.CandidateReason == CandidateReason.OverloadResolutionFailure)
|
|
{
|
|
// Roslyn workaround. Even if we can't resolve a method, we know it's a method.
|
|
return Create(info.Context, expression, info.Parent, info.Child);
|
|
}
|
|
|
|
var symbol = target.Symbol ?? info.Context.GetSymbolInfo(name).Symbol;
|
|
|
|
if (symbol == null && target.CandidateSymbols.Length >= 1)
|
|
{
|
|
// Pick the first symbol. This could occur for something like `nameof(Foo.Bar)`
|
|
// where `Bar` is a method group. Technically, we don't know which symbol is accessed.
|
|
symbol = target.CandidateSymbols[0];
|
|
}
|
|
|
|
if (symbol == null)
|
|
{
|
|
info.Context.ModelError(info.Node, "Failed to determine symbol for member access");
|
|
return new MemberAccess(info.SetKind(ExprKind.UNKNOWN), expression, symbol);
|
|
}
|
|
|
|
ExprKind kind;
|
|
|
|
switch (symbol.Kind)
|
|
{
|
|
case SymbolKind.Property:
|
|
kind = ExprKind.PROPERTY_ACCESS;
|
|
break;
|
|
case SymbolKind.Method:
|
|
kind = ExprKind.METHOD_ACCESS;
|
|
break;
|
|
case SymbolKind.Field:
|
|
kind = ExprKind.FIELD_ACCESS;
|
|
break;
|
|
case SymbolKind.NamedType:
|
|
return TypeAccess.Create(info);
|
|
case SymbolKind.Event:
|
|
kind = ExprKind.EVENT_ACCESS;
|
|
break;
|
|
default:
|
|
info.Context.ModelError(info.Node, "Unhandled symbol for member access");
|
|
kind = ExprKind.UNKNOWN;
|
|
break;
|
|
}
|
|
return new MemberAccess(info.SetKind(kind), expression, symbol);
|
|
}
|
|
|
|
public Expression Qualifier { get; private set; }
|
|
}
|
|
}
|