K2: add more accurate & more automatic control of diagnostic suppression

#KT-51363 Fixed
This commit is contained in:
Mikhail Glukhikh
2022-07-20 16:46:23 +02:00
committed by Space
parent 179b267d9f
commit 9add6f3d55
21 changed files with 194 additions and 101 deletions
@@ -62,6 +62,7 @@ private object CheckersFactory {
add(ExpressionCheckersDiagnosticComponent(session, reporter, expressionCheckers))
add(TypeCheckersDiagnosticComponent(session, reporter, typeCheckers))
add(ControlFlowAnalysisDiagnosticComponent(session, reporter, declarationCheckers))
add(ReportCommitterDiagnosticComponent(session, reporter))
}
}
@@ -172,7 +172,7 @@ object FirKotlinToJvmBytecodeCompiler {
val renderDiagnosticNames = moduleConfiguration.getBoolean(CLIConfigurationKeys.RENDER_DIAGNOSTIC_INTERNAL_NAME)
val diagnosticsReporter = DiagnosticReporterFactory.createReporter()
val diagnosticsReporter = DiagnosticReporterFactory.createPendingReporter()
val firResult = runFrontend(allSources, diagnosticsReporter).also {
performanceManager?.notifyAnalysisFinished()
}
@@ -120,7 +120,7 @@ fun compileModulesUsingFrontendIrAndLightTree(
}
val renderDiagnosticName = moduleConfiguration.getBoolean(CLIConfigurationKeys.RENDER_DIAGNOSTIC_INTERNAL_NAME)
val diagnosticsReporter = DiagnosticReporterFactory.createReporter()
val diagnosticsReporter = DiagnosticReporterFactory.createPendingReporter()
val compilerInput = ModuleCompilerInput(
TargetId(module),
@@ -1,20 +1,20 @@
const val x = "123"
<!CONST_VAL_WITH_GETTER!>@Suppress("CONST_VAL_WITH_GETTER")
get() = field<!>
@Suppress("CONST_VAL_WITH_GETTER")
get() = field
val y = "789"
const val z = @Suppress("CONST_VAL_WITH_NON_CONST_INITIALIZER") <!CONST_VAL_WITH_NON_CONST_INITIALIZER!>y<!>
const val z = @Suppress("CONST_VAL_WITH_NON_CONST_INITIALIZER") y
@Target(AnnotationTarget.TYPE)
annotation class Ann
fun foo(): @Suppress("REPEATED_ANNOTATION") @Ann <!REPEATED_ANNOTATION!>@Ann<!> Int = 42
fun foo(): @Suppress("REPEATED_ANNOTATION") @Ann @Ann Int = 42
typealias Alias<T> = <!TYPEALIAS_SHOULD_EXPAND_TO_CLASS!>@Suppress("TYPEALIAS_SHOULD_EXPAND_TO_CLASS") T<!>
typealias Alias<T> = @Suppress("TYPEALIAS_SHOULD_EXPAND_TO_CLASS") T
interface A
interface B : <!SUPERTYPE_INITIALIZED_IN_INTERFACE!>@Suppress("SUPERTYPE_INITIALIZED_IN_INTERFACE") A<!>()
interface B : @Suppress("SUPERTYPE_INITIALIZED_IN_INTERFACE") A()
data class D @Suppress("DATA_CLASS_VARARG_PARAMETER") constructor(<!DATA_CLASS_VARARG_PARAMETER!>vararg val x: String<!>)
data class D @Suppress("DATA_CLASS_VARARG_PARAMETER") constructor(vararg val x: String)
@@ -8,6 +8,8 @@ package org.jetbrains.kotlin.fir.analysis.collectors
import org.jetbrains.kotlin.fir.FirElement
import org.jetbrains.kotlin.fir.analysis.checkers.context.CheckerContext
import org.jetbrains.kotlin.fir.analysis.collectors.components.AbstractDiagnosticCollectorComponent
import org.jetbrains.kotlin.fir.declarations.FirDeclaration
import org.jetbrains.kotlin.fir.declarations.FirFile
open class CheckerRunningDiagnosticCollectorVisitor(
context: CheckerContext,
@@ -19,4 +21,11 @@ open class CheckerRunningDiagnosticCollectorVisitor(
element.accept(it, context)
}
}
override fun onDeclarationExit(declaration: FirDeclaration) {
if (declaration !is FirFile) return
components.forEach {
it.endOfFile(declaration)
}
}
}
@@ -5,10 +5,13 @@
package org.jetbrains.kotlin.fir.analysis.collectors.components
import org.jetbrains.kotlin.diagnostics.DiagnosticContext
import org.jetbrains.kotlin.fir.FirElement
import org.jetbrains.kotlin.fir.FirSession
import org.jetbrains.kotlin.fir.analysis.checkers.context.CheckerContext
import org.jetbrains.kotlin.diagnostics.DiagnosticReporter
import org.jetbrains.kotlin.diagnostics.impl.PendingDiagnosticsCollectorWithSuppress
import org.jetbrains.kotlin.fir.declarations.FirFile
import org.jetbrains.kotlin.fir.visitors.FirVisitor
abstract class AbstractDiagnosticCollectorComponent(
@@ -16,4 +19,13 @@ abstract class AbstractDiagnosticCollectorComponent(
protected val reporter: DiagnosticReporter,
) : FirVisitor<Unit, CheckerContext>() {
override fun visitElement(element: FirElement, data: CheckerContext) {}
protected fun checkAndCommitReportsOn(element: FirElement, context: DiagnosticContext?) {
if (reporter is PendingDiagnosticsCollectorWithSuppress) {
val source = element.source ?: return
reporter.checkAndCommitReportsOn(source, context, context == null)
}
}
open fun endOfFile(file: FirFile) {}
}
@@ -22,69 +22,68 @@ class DeclarationCheckersDiagnosticComponent(
) : AbstractDiagnosticCollectorComponent(session, reporter) {
override fun visitFile(file: FirFile, data: CheckerContext) {
checkers.allFileCheckers.check(file, data, reporter)
checkers.allFileCheckers.check(file, data)
}
override fun visitProperty(property: FirProperty, data: CheckerContext) {
checkers.allPropertyCheckers.check(property, data, reporter)
checkers.allPropertyCheckers.check(property, data)
}
override fun visitClass(klass: FirClass, data: CheckerContext) {
checkers.allClassCheckers.check(klass, data, reporter)
checkers.allClassCheckers.check(klass, data)
}
override fun visitRegularClass(regularClass: FirRegularClass, data: CheckerContext) {
checkers.allRegularClassCheckers.check(regularClass, data, reporter)
checkers.allRegularClassCheckers.check(regularClass, data)
}
override fun visitSimpleFunction(simpleFunction: FirSimpleFunction, data: CheckerContext) {
checkers.allSimpleFunctionCheckers.check(simpleFunction, data, reporter)
checkers.allSimpleFunctionCheckers.check(simpleFunction, data)
}
override fun visitTypeAlias(typeAlias: FirTypeAlias, data: CheckerContext) {
checkers.allTypeAliasCheckers.check(typeAlias, data, reporter)
checkers.allTypeAliasCheckers.check(typeAlias, data)
}
override fun visitConstructor(constructor: FirConstructor, data: CheckerContext) {
checkers.allConstructorCheckers.check(constructor, data, reporter)
checkers.allConstructorCheckers.check(constructor, data)
}
override fun visitAnonymousFunction(anonymousFunction: FirAnonymousFunction, data: CheckerContext) {
checkers.allAnonymousFunctionCheckers.check(anonymousFunction, data, reporter)
checkers.allAnonymousFunctionCheckers.check(anonymousFunction, data)
}
override fun visitPropertyAccessor(propertyAccessor: FirPropertyAccessor, data: CheckerContext) {
checkers.allPropertyAccessorCheckers.check(propertyAccessor, data, reporter)
checkers.allPropertyAccessorCheckers.check(propertyAccessor, data)
}
override fun visitBackingField(backingField: FirBackingField, data: CheckerContext) {
checkers.allBackingFieldCheckers.check(backingField, data, reporter)
checkers.allBackingFieldCheckers.check(backingField, data)
}
override fun visitValueParameter(valueParameter: FirValueParameter, data: CheckerContext) {
checkers.allValueParameterCheckers.check(valueParameter, data, reporter)
checkers.allValueParameterCheckers.check(valueParameter, data)
}
override fun visitTypeParameter(typeParameter: FirTypeParameter, data: CheckerContext) {
checkers.allTypeParameterCheckers.check(typeParameter, data, reporter)
checkers.allTypeParameterCheckers.check(typeParameter, data)
}
override fun visitEnumEntry(enumEntry: FirEnumEntry, data: CheckerContext) {
checkers.allEnumEntryCheckers.check(enumEntry, data, reporter)
checkers.allEnumEntryCheckers.check(enumEntry, data)
}
override fun visitAnonymousObject(anonymousObject: FirAnonymousObject, data: CheckerContext) {
checkers.allAnonymousObjectCheckers.check(anonymousObject, data, reporter)
checkers.allAnonymousObjectCheckers.check(anonymousObject, data)
}
override fun visitAnonymousInitializer(anonymousInitializer: FirAnonymousInitializer, data: CheckerContext) {
checkers.allAnonymousInitializerCheckers.check(anonymousInitializer, data, reporter)
checkers.allAnonymousInitializerCheckers.check(anonymousInitializer, data)
}
private fun <D : FirDeclaration> Collection<FirDeclarationChecker<D>>.check(
declaration: D,
context: CheckerContext,
reporter: DiagnosticReporter
context: CheckerContext
) {
for (checker in this) {
checker.check(declaration, context, reporter)
@@ -16,6 +16,7 @@ object DiagnosticComponentsFactory {
TypeCheckersDiagnosticComponent(session, reporter),
ErrorNodeDiagnosticCollectorComponent(session, reporter),
ControlFlowAnalysisDiagnosticComponent(session, reporter),
ReportCommitterDiagnosticComponent(session, reporter)
)
}
@@ -32,12 +32,12 @@ class ErrorNodeDiagnosticCollectorComponent(
) : AbstractDiagnosticCollectorComponent(session, reporter) {
override fun visitErrorLoop(errorLoop: FirErrorLoop, data: CheckerContext) {
val source = errorLoop.source ?: return
reportFirDiagnostic(errorLoop.diagnostic, source, reporter, data)
reportFirDiagnostic(errorLoop.diagnostic, source, data)
}
override fun visitErrorTypeRef(errorTypeRef: FirErrorTypeRef, data: CheckerContext) {
val source = errorTypeRef.source ?: return
reportFirDiagnostic(errorTypeRef.diagnostic, source, reporter, data)
reportFirDiagnostic(errorTypeRef.diagnostic, source, data)
}
override fun visitResolvedTypeRef(resolvedTypeRef: FirResolvedTypeRef, data: CheckerContext) {
@@ -71,7 +71,7 @@ class ErrorNodeDiagnosticCollectorComponent(
) return
}
reportFirDiagnostic(errorNamedReference.diagnostic, source, reporter, data, qualifiedAccessOrAnnotationCall?.source)
reportFirDiagnostic(errorNamedReference.diagnostic, source, data, qualifiedAccessOrAnnotationCall?.source)
}
private fun FirExpression?.cannotBeResolved(): Boolean {
@@ -86,28 +86,27 @@ class ErrorNodeDiagnosticCollectorComponent(
override fun visitErrorExpression(errorExpression: FirErrorExpression, data: CheckerContext) {
val source = errorExpression.source ?: return
reportFirDiagnostic(errorExpression.diagnostic, source, reporter, data)
reportFirDiagnostic(errorExpression.diagnostic, source, data)
}
override fun visitErrorFunction(errorFunction: FirErrorFunction, data: CheckerContext) {
val source = errorFunction.source ?: return
reportFirDiagnostic(errorFunction.diagnostic, source, reporter, data)
reportFirDiagnostic(errorFunction.diagnostic, source, data)
}
override fun visitErrorResolvedQualifier(errorResolvedQualifier: FirErrorResolvedQualifier, data: CheckerContext) {
val source = errorResolvedQualifier.source ?: return
reportFirDiagnostic(errorResolvedQualifier.diagnostic, source, reporter, data)
reportFirDiagnostic(errorResolvedQualifier.diagnostic, source, data)
}
override fun visitErrorImport(errorImport: FirErrorImport, data: CheckerContext) {
val source = errorImport.source ?: return
reportFirDiagnostic(errorImport.diagnostic, source, reporter, data)
reportFirDiagnostic(errorImport.diagnostic, source, data)
}
private fun reportFirDiagnostic(
diagnostic: ConeDiagnostic,
source: KtSourceElement,
reporter: DiagnosticReporter,
context: CheckerContext,
qualifiedAccessSource: KtSourceElement? = null
) {
@@ -21,137 +21,136 @@ class ExpressionCheckersDiagnosticComponent(
private val checkers: ExpressionCheckers = session.checkersComponent.expressionCheckers,
) : AbstractDiagnosticCollectorComponent(session, reporter) {
override fun visitTypeOperatorCall(typeOperatorCall: FirTypeOperatorCall, data: CheckerContext) {
checkers.allTypeOperatorCallCheckers.check(typeOperatorCall, data, reporter)
checkers.allTypeOperatorCallCheckers.check(typeOperatorCall, data)
}
override fun <T> visitConstExpression(constExpression: FirConstExpression<T>, data: CheckerContext) {
checkers.allConstExpressionCheckers.check(constExpression, data, reporter)
checkers.allConstExpressionCheckers.check(constExpression, data)
}
override fun visitAnnotation(annotation: FirAnnotation, data: CheckerContext) {
checkers.allAnnotationCheckers.check(annotation, data, reporter)
checkers.allAnnotationCheckers.check(annotation, data)
}
override fun visitAnnotationCall(annotationCall: FirAnnotationCall, data: CheckerContext) {
checkers.allAnnotationCallCheckers.check(annotationCall, data, reporter)
checkers.allAnnotationCallCheckers.check(annotationCall, data)
}
override fun visitQualifiedAccessExpression(qualifiedAccessExpression: FirQualifiedAccessExpression, data: CheckerContext) {
checkers.allQualifiedAccessExpressionCheckers.check(qualifiedAccessExpression, data, reporter)
checkers.allQualifiedAccessExpressionCheckers.check(qualifiedAccessExpression, data)
}
override fun visitPropertyAccessExpression(propertyAccessExpression: FirPropertyAccessExpression, data: CheckerContext) {
checkers.allQualifiedAccessExpressionCheckers.check(propertyAccessExpression, data, reporter)
checkers.allQualifiedAccessExpressionCheckers.check(propertyAccessExpression, data)
}
override fun visitFunctionCall(functionCall: FirFunctionCall, data: CheckerContext) {
checkers.allFunctionCallCheckers.check(functionCall, data, reporter)
checkers.allFunctionCallCheckers.check(functionCall, data)
}
override fun visitIntegerLiteralOperatorCall(integerLiteralOperatorCall: FirIntegerLiteralOperatorCall, data: CheckerContext) {
checkers.allIntegerLiteralOperatorCallCheckers.check(integerLiteralOperatorCall, data, reporter)
checkers.allIntegerLiteralOperatorCallCheckers.check(integerLiteralOperatorCall, data)
}
override fun visitImplicitInvokeCall(implicitInvokeCall: FirImplicitInvokeCall, data: CheckerContext) {
checkers.allFunctionCallCheckers.check(implicitInvokeCall, data, reporter)
checkers.allFunctionCallCheckers.check(implicitInvokeCall, data)
}
override fun visitCallableReferenceAccess(callableReferenceAccess: FirCallableReferenceAccess, data: CheckerContext) {
checkers.allCallableReferenceAccessCheckers.check(callableReferenceAccess, data, reporter)
checkers.allCallableReferenceAccessCheckers.check(callableReferenceAccess, data)
}
override fun visitThisReceiverExpression(thisReceiverExpression: FirThisReceiverExpression, data: CheckerContext) {
checkers.allThisReceiverExpressionCheckers.check(thisReceiverExpression, data, reporter)
checkers.allThisReceiverExpressionCheckers.check(thisReceiverExpression, data)
}
override fun visitResolvedQualifier(resolvedQualifier: FirResolvedQualifier, data: CheckerContext) {
checkers.allResolvedQualifierCheckers.check(resolvedQualifier, data, reporter)
checkers.allResolvedQualifierCheckers.check(resolvedQualifier, data)
}
override fun visitWhenExpression(whenExpression: FirWhenExpression, data: CheckerContext) {
checkers.allWhenExpressionCheckers.check(whenExpression, data, reporter)
checkers.allWhenExpressionCheckers.check(whenExpression, data)
}
override fun visitWhileLoop(whileLoop: FirWhileLoop, data: CheckerContext) {
checkers.allWhileLoopCheckers.check(whileLoop, data, reporter)
checkers.allWhileLoopCheckers.check(whileLoop, data)
}
override fun visitDoWhileLoop(doWhileLoop: FirDoWhileLoop, data: CheckerContext) {
checkers.allDoWhileLoopCheckers.check(doWhileLoop, data, reporter)
checkers.allDoWhileLoopCheckers.check(doWhileLoop, data)
}
override fun visitErrorLoop(errorLoop: FirErrorLoop, data: CheckerContext) {
checkers.allLoopExpressionCheckers.check(errorLoop, data, reporter)
checkers.allLoopExpressionCheckers.check(errorLoop, data)
}
override fun visitBinaryLogicExpression(binaryLogicExpression: FirBinaryLogicExpression, data: CheckerContext) {
checkers.allLogicExpressionCheckers.check(binaryLogicExpression, data, reporter)
checkers.allLogicExpressionCheckers.check(binaryLogicExpression, data)
}
override fun visitArrayOfCall(arrayOfCall: FirArrayOfCall, data: CheckerContext) {
checkers.allArrayOfCallCheckers.check(arrayOfCall, data, reporter)
checkers.allArrayOfCallCheckers.check(arrayOfCall, data)
}
override fun visitStringConcatenationCall(stringConcatenationCall: FirStringConcatenationCall, data: CheckerContext) {
checkers.allStringConcatenationCallCheckers.check(stringConcatenationCall, data, reporter)
checkers.allStringConcatenationCallCheckers.check(stringConcatenationCall, data)
}
override fun visitCheckNotNullCall(checkNotNullCall: FirCheckNotNullCall, data: CheckerContext) {
checkers.allCheckNotNullCallCheckers.check(checkNotNullCall, data, reporter)
checkers.allCheckNotNullCallCheckers.check(checkNotNullCall, data)
}
override fun visitElvisExpression(elvisExpression: FirElvisExpression, data: CheckerContext) {
checkers.allElvisExpressionCheckers.check(elvisExpression, data, reporter)
checkers.allElvisExpressionCheckers.check(elvisExpression, data)
}
override fun visitSafeCallExpression(safeCallExpression: FirSafeCallExpression, data: CheckerContext) {
checkers.allSafeCallExpressionCheckers.check(safeCallExpression, data, reporter)
checkers.allSafeCallExpressionCheckers.check(safeCallExpression, data)
}
override fun visitTryExpression(tryExpression: FirTryExpression, data: CheckerContext) {
checkers.allTryExpressionCheckers.check(tryExpression, data, reporter)
checkers.allTryExpressionCheckers.check(tryExpression, data)
}
override fun visitClassReferenceExpression(classReferenceExpression: FirClassReferenceExpression, data: CheckerContext) {
checkers.allClassReferenceExpressionCheckers.check(classReferenceExpression, data, reporter)
checkers.allClassReferenceExpressionCheckers.check(classReferenceExpression, data)
}
override fun visitGetClassCall(getClassCall: FirGetClassCall, data: CheckerContext) {
checkers.allGetClassCallCheckers.check(getClassCall, data, reporter)
checkers.allGetClassCallCheckers.check(getClassCall, data)
}
override fun visitEqualityOperatorCall(equalityOperatorCall: FirEqualityOperatorCall, data: CheckerContext) {
checkers.allEqualityOperatorCallCheckers.check(equalityOperatorCall, data, reporter)
checkers.allEqualityOperatorCallCheckers.check(equalityOperatorCall, data)
}
override fun visitVariableAssignment(variableAssignment: FirVariableAssignment, data: CheckerContext) {
checkers.allVariableAssignmentCheckers.check(variableAssignment, data, reporter)
checkers.allVariableAssignmentCheckers.check(variableAssignment, data)
}
override fun visitReturnExpression(returnExpression: FirReturnExpression, data: CheckerContext) {
checkers.allReturnExpressionCheckers.check(returnExpression, data, reporter)
checkers.allReturnExpressionCheckers.check(returnExpression, data)
}
override fun visitBreakExpression(breakExpression: FirBreakExpression, data: CheckerContext) {
checkers.allLoopJumpCheckers.check(breakExpression, data, reporter)
checkers.allLoopJumpCheckers.check(breakExpression, data)
}
override fun visitContinueExpression(continueExpression: FirContinueExpression, data: CheckerContext) {
checkers.allLoopJumpCheckers.check(continueExpression, data, reporter)
checkers.allLoopJumpCheckers.check(continueExpression, data)
}
override fun visitBlock(block: FirBlock, data: CheckerContext) {
checkers.allBlockCheckers.check(block, data, reporter)
checkers.allBlockCheckers.check(block, data)
}
override fun visitDelegatedConstructorCall(delegatedConstructorCall: FirDelegatedConstructorCall, data: CheckerContext) {
checkers.allCallCheckers.check(delegatedConstructorCall, data, reporter)
checkers.allCallCheckers.check(delegatedConstructorCall, data)
}
private fun <E : FirStatement> Collection<FirExpressionChecker<E>>.check(
expression: E,
context: CheckerContext,
reporter: DiagnosticReporter
context: CheckerContext
) {
for (checker in this) {
checker.check(expression, context, reporter)
@@ -0,0 +1,25 @@
/*
* Copyright 2010-2022 JetBrains s.r.o. and Kotlin Programming Language contributors.
* Use of this source code is governed by the Apache 2.0 license that can be found in the license/LICENSE.txt file.
*/
package org.jetbrains.kotlin.fir.analysis.collectors.components
import org.jetbrains.kotlin.diagnostics.DiagnosticReporter
import org.jetbrains.kotlin.fir.FirElement
import org.jetbrains.kotlin.fir.FirSession
import org.jetbrains.kotlin.fir.analysis.checkers.context.CheckerContext
import org.jetbrains.kotlin.fir.declarations.FirFile
class ReportCommitterDiagnosticComponent(
session: FirSession,
reporter: DiagnosticReporter
) : AbstractDiagnosticCollectorComponent(session, reporter) {
override fun visitElement(element: FirElement, data: CheckerContext) {
checkAndCommitReportsOn(element, data)
}
override fun endOfFile(file: FirFile) {
checkAndCommitReportsOn(file, null)
}
}
@@ -22,41 +22,40 @@ class TypeCheckersDiagnosticComponent(
) : AbstractDiagnosticCollectorComponent(session, reporter) {
override fun visitDynamicTypeRef(dynamicTypeRef: FirDynamicTypeRef, data: CheckerContext) {
checkers.allTypeRefCheckers.check(dynamicTypeRef, data, reporter)
checkers.allTypeRefCheckers.check(dynamicTypeRef, data)
}
override fun visitFunctionTypeRef(functionTypeRef: FirFunctionTypeRef, data: CheckerContext) {
checkers.allTypeRefCheckers.check(functionTypeRef, data, reporter)
checkers.allTypeRefCheckers.check(functionTypeRef, data)
}
override fun visitUserTypeRef(userTypeRef: FirUserTypeRef, data: CheckerContext) {
checkers.allTypeRefCheckers.check(userTypeRef, data, reporter)
checkers.allTypeRefCheckers.check(userTypeRef, data)
}
override fun visitResolvedTypeRef(resolvedTypeRef: FirResolvedTypeRef, data: CheckerContext) {
checkers.allTypeRefCheckers.check(resolvedTypeRef, data, reporter)
checkers.allTypeRefCheckers.check(resolvedTypeRef, data)
}
override fun visitErrorTypeRef(errorTypeRef: FirErrorTypeRef, data: CheckerContext) {
checkers.allTypeRefCheckers.check(errorTypeRef, data, reporter)
checkers.allTypeRefCheckers.check(errorTypeRef, data)
}
override fun visitTypeRefWithNullability(typeRefWithNullability: FirTypeRefWithNullability, data: CheckerContext) {
checkers.allTypeRefCheckers.check(typeRefWithNullability, data, reporter)
checkers.allTypeRefCheckers.check(typeRefWithNullability, data)
}
override fun visitImplicitTypeRef(implicitTypeRef: FirImplicitTypeRef, data: CheckerContext) {
checkers.allTypeRefCheckers.check(implicitTypeRef, data, reporter)
checkers.allTypeRefCheckers.check(implicitTypeRef, data)
}
override fun visitTypeRef(typeRef: FirTypeRef, data: CheckerContext) {
checkers.allTypeRefCheckers.check(typeRef, data, reporter)
checkers.allTypeRefCheckers.check(typeRef, data)
}
private fun <T : FirTypeRef> Collection<FirTypeChecker<T>>.check(
typeRef: T,
context: CheckerContext,
reporter: DiagnosticReporter
context: CheckerContext
) {
for (checker in this) {
checker.check(typeRef, context, reporter)
@@ -6,8 +6,6 @@
package org.jetbrains.kotlin.fir.pipeline
import org.jetbrains.kotlin.diagnostics.DiagnosticReporter
import org.jetbrains.kotlin.diagnostics.DiagnosticReporterFactory
import org.jetbrains.kotlin.diagnostics.KtDiagnostic
import org.jetbrains.kotlin.fir.FirSession
import org.jetbrains.kotlin.fir.analysis.collectors.FirDiagnosticsCollector
import org.jetbrains.kotlin.fir.declarations.FirFile
@@ -20,19 +18,6 @@ fun FirSession.runResolution(firFiles: List<FirFile>): Pair<ScopeSession, List<F
return resolveProcessor.scopeSession to firFiles
}
@OptIn(ExperimentalStdlibApi::class)
fun FirSession.runCheckers(scopeSession: ScopeSession, firFiles: List<FirFile>): Map<FirFile, List<KtDiagnostic>> {
val collector = FirDiagnosticsCollector.create(this, scopeSession)
return buildMap {
for (file in firFiles) {
val reporter = DiagnosticReporterFactory.createReporter()
collector.collectDiagnostics(file, reporter)
put(file, reporter.diagnostics)
}
}
}
@OptIn(ExperimentalStdlibApi::class)
fun FirSession.runCheckers(scopeSession: ScopeSession, firFiles: List<FirFile>, reporter: DiagnosticReporter) {
val collector = FirDiagnosticsCollector.create(this, scopeSession)
for (file in firFiles) {
@@ -337,7 +337,7 @@ class FirCheckersRunnerTransformer(private val diagnosticCollector: AbstractDiag
}
override fun transformFile(file: FirFile, data: Nothing?): FirFile {
val reporter = DiagnosticReporterFactory.createReporter()
val reporter = DiagnosticReporterFactory.createPendingReporter()
diagnosticCollector.collectDiagnostics(file, reporter)
return file
}
@@ -6,15 +6,19 @@
package org.jetbrains.kotlin.diagnostics
import org.jetbrains.kotlin.diagnostics.impl.BaseDiagnosticsCollector
import org.jetbrains.kotlin.diagnostics.impl.DiagnosticsCollectorWithSuppress
import org.jetbrains.kotlin.diagnostics.impl.PendingDiagnosticsCollectorWithSuppress
import org.jetbrains.kotlin.diagnostics.impl.SimpleDiagnosticsCollector
import org.jetbrains.kotlin.diagnostics.impl.SimpleDiagnosticsCollectorWithSuppress
object DiagnosticReporterFactory {
fun createReporter(disableSuppress: Boolean = false): BaseDiagnosticsCollector {
return if (disableSuppress) {
SimpleDiagnosticsCollector()
} else {
DiagnosticsCollectorWithSuppress()
SimpleDiagnosticsCollectorWithSuppress()
}
}
fun createPendingReporter(): PendingDiagnosticsCollectorWithSuppress =
PendingDiagnosticsCollectorWithSuppress()
}
@@ -0,0 +1,61 @@
/*
* Copyright 2010-2021 JetBrains s.r.o. and Kotlin Programming Language contributors.
* Use of this source code is governed by the Apache 2.0 license that can be found in the license/LICENSE.txt file.
*/
package org.jetbrains.kotlin.diagnostics.impl
import org.jetbrains.kotlin.AbstractKtSourceElement
import org.jetbrains.kotlin.diagnostics.DiagnosticContext
import org.jetbrains.kotlin.diagnostics.KtDiagnostic
import org.jetbrains.kotlin.diagnostics.Severity
class PendingDiagnosticsCollectorWithSuppress : BaseDiagnosticsCollector() {
private val pendingDiagnosticsByFilePath: MutableMap<String?, MutableList<KtDiagnostic>> = mutableMapOf()
private val _diagnosticsByFilePath: MutableMap<String?, MutableList<KtDiagnostic>> = mutableMapOf()
override val diagnostics: List<KtDiagnostic>
get() = _diagnosticsByFilePath.flatMap { it.value }
override val diagnosticsByFilePath: Map<String?, List<KtDiagnostic>>
get() = _diagnosticsByFilePath
override var hasErrors = false
private set
override fun report(diagnostic: KtDiagnostic?, context: DiagnosticContext) {
if (diagnostic != null && !context.isDiagnosticSuppressed(diagnostic)) {
pendingDiagnosticsByFilePath.getOrPut(context.containingFilePath) { mutableListOf() }.run {
add(diagnostic)
}
}
}
fun checkAndCommitReportsOn(
element: AbstractKtSourceElement,
context: DiagnosticContext?,
commitEverything: Boolean
) {
for ((path, pendingList) in pendingDiagnosticsByFilePath) {
val committedList = _diagnosticsByFilePath.getOrPut(path) { mutableListOf() }
val iterator = pendingList.iterator()
while (iterator.hasNext()) {
val diagnostic = iterator.next()
when {
context?.isDiagnosticSuppressed(diagnostic) == true -> {
if (diagnostic.element == element ||
diagnostic.element.startOffset >= element.startOffset && diagnostic.element.endOffset <= element.endOffset
) {
iterator.remove()
}
}
diagnostic.element == element || commitEverything -> {
iterator.remove()
committedList += diagnostic
if (!hasErrors && diagnostic.severity == Severity.ERROR) {
hasErrors = true
}
}
}
}
}
}
}
@@ -9,7 +9,7 @@ import org.jetbrains.kotlin.diagnostics.DiagnosticContext
import org.jetbrains.kotlin.diagnostics.KtDiagnostic
import org.jetbrains.kotlin.diagnostics.Severity
class DiagnosticsCollectorWithSuppress : BaseDiagnosticsCollector() {
class SimpleDiagnosticsCollectorWithSuppress : BaseDiagnosticsCollector() {
private val _diagnosticsByFilePath: MutableMap<String?, MutableList<KtDiagnostic>> = mutableMapOf()
override val diagnostics: List<KtDiagnostic>
get() = _diagnosticsByFilePath.flatMap { it.value }
@@ -188,7 +188,7 @@ class IncrementalFirJvmCompilerRunner(
else allPlatformSourceFiles.add(file)
}
val diagnosticsReporter = DiagnosticReporterFactory.createReporter()
val diagnosticsReporter = DiagnosticReporterFactory.createPendingReporter()
val performanceManager = configuration[CLIConfigurationKeys.PERF_MANAGER]
val compilerEnvironment = ModuleCompilerEnvironment(projectEnvironment, diagnosticsReporter)
@@ -3,7 +3,6 @@
// IGNORE_BACKEND: JS_IR_ES6
// TODO: muted automatically, investigate should it be ran for JS or not
// IGNORE_BACKEND: JS
// IGNORE_FIR_DIAGNOSTICS_DIFF
tailrec fun badTails(x : Int) : Int {
if (x < 50 && x != 10 && x > 0) {
@@ -270,7 +270,7 @@ abstract class AbstractKtDiagnosticsTest : AbstractFirBaseDiagnosticsTest() {
for (firFile in firFiles) {
val session = firFile.moduleData.session
val collector = collectors.computeIfAbsent(session) { createCollector(session) }
val reporter = DiagnosticReporterFactory.createReporter()
val reporter = DiagnosticReporterFactory.createPendingReporter()
collector.collectDiagnostics(firFile, reporter)
result[firFile] = reporter.diagnostics
}
@@ -93,7 +93,7 @@ class FirAnalyzerFacade(
val collector = FirDiagnosticsCollector.create(session, scopeSession)
collectedDiagnostics = buildMap {
for (file in firFiles!!) {
val reporter = DiagnosticReporterFactory.createReporter()
val reporter = DiagnosticReporterFactory.createPendingReporter()
collector.collectDiagnostics(file, reporter)
put(file, reporter.diagnostics)
}