Skip to content

Commit

Permalink
Fix ChiselStage and Builder handling of logging (backport #3895) (#3898)
Browse files Browse the repository at this point in the history
* Fix ChiselStage and Builder handling of logging (#3895)

Previously, object circt.stage.ChiselStage was ignoring the Logger.
Also, Chisel was not creating its own logger scope which could lead to
clobbering of the Console when running invoking Chisel in the same
process multiple times.

Fix various places we had to workaround this behavior and fix tests
checking --log-level debug.

(cherry picked from commit 88d147d)

# Conflicts:
#	src/main/scala/circt/stage/ChiselStage.scala

* Resolve backport conflicts

* Make logger annotations unserializable

Change logger annotations to mix-in the Unserializable trait so that they
will not emitted by a stage.  These annotations are not intended to be
seen by CIRCT and these should be stripped from the output FIRRTL text.

Signed-off-by: Schuyler Eldridge <[email protected]>

---------

Signed-off-by: Schuyler Eldridge <[email protected]>
Co-authored-by: Jack Koenig <[email protected]>
Co-authored-by: Schuyler Eldridge <[email protected]>
  • Loading branch information
3 people authored Mar 7, 2024
1 parent 31c1034 commit e46a4a3
Show file tree
Hide file tree
Showing 11 changed files with 196 additions and 80 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -107,7 +107,8 @@ object Definition extends SourceInfoDoc {
context.warningFilters,
context.sourceRoots,
Some(context.globalNamespace),
Builder.allDefinitions
Builder.allDefinitions,
context.loggerOptions
)
}
dynamicContext.inDefinition = true
Expand Down
19 changes: 16 additions & 3 deletions core/src/main/scala/chisel3/internal/Builder.scala
Original file line number Diff line number Diff line change
Expand Up @@ -11,21 +11,22 @@ import chisel3.properties.Class
import chisel3.internal.firrtl.ir._
import chisel3.internal.firrtl.Converter
import chisel3.internal.naming._
import _root_.firrtl.annotations.{CircuitName, ComponentName, IsMember, ModuleName, Named, ReferenceTarget}
import _root_.firrtl.annotations.{Annotation, CircuitName, ComponentName, IsMember, ModuleName, Named, ReferenceTarget}
import _root_.firrtl.annotations.AnnotationUtils.validComponentName
import _root_.firrtl.AnnotationSeq
import _root_.firrtl.renamemap.MutableRenameMap
import _root_.firrtl.util.BackendCompilationUtilities._
import _root_.firrtl.{ir => fir}
import chisel3.experimental.dataview.{reify, reifySingleData}
import chisel3.internal.Builder.Prefix
import logger.LazyLogging
import logger.{LazyLogging, LoggerOption}

import scala.collection.mutable
import scala.annotation.tailrec
import java.io.File
import scala.util.control.NonFatal
import chisel3.ChiselException
import logger.LoggerOptions

private[chisel3] class Namespace(keywords: Set[String], separator: Char = '_') {
// This HashMap is compressed, not every name in the namespace is present here.
Expand Down Expand Up @@ -452,7 +453,8 @@ private[chisel3] class DynamicContext(
val sourceRoots: Seq[File],
val defaultNamespace: Option[Namespace],
// Definitions from other scopes in the same elaboration, use allDefinitions below
val outerScopeDefinitions: List[Iterable[Definition[_]]]) {
val outerScopeDefinitions: List[Iterable[Definition[_]]],
val loggerOptions: LoggerOptions) {
val importedDefinitionAnnos = annotationSeq.collect { case a: ImportDefinitionAnnotation[_] => a }

// Map from proto module name to ext-module name
Expand Down Expand Up @@ -1000,6 +1002,17 @@ private[chisel3] object Builder extends LazyLogging {
f: => T,
dynamicContext: DynamicContext,
forceModName: Boolean = true
): (Circuit, T) = {
// Logger has its own context separate from Chisel's dynamic context
_root_.logger.Logger.makeScope(dynamicContext.loggerOptions) {
buildImpl(f, dynamicContext, forceModName)
}
}

private def buildImpl[T <: BaseModule](
f: => T,
dynamicContext: DynamicContext,
forceModName: Boolean
): (Circuit, T) = {
dynamicContextVar.withValue(Some(dynamicContext)) {
ViewParent: Unit // Must initialize the singleton in a Builder context or weird things can happen
Expand Down
3 changes: 0 additions & 3 deletions docs/src/cookbooks/cookbook.md
Original file line number Diff line number Diff line change
Expand Up @@ -832,9 +832,6 @@ If the indexee is a non-power-of-2 size, use the ceiling of the log2 result.

```scala mdoc:invisible:reset
import chisel3._
// Some other test is clobbering the global Logger which breaks the warnings below
// Setting the output stream to the Console fixes the issue
logger.Logger.setConsole()
// Helper to throw away return value so it doesn't show up in mdoc
def compile(gen: => chisel3.RawModule): Unit = {
circt.stage.ChiselStage.emitCHIRRTL(gen)
Expand Down
3 changes: 0 additions & 3 deletions docs/src/explanations/warnings.md
Original file line number Diff line number Diff line change
Expand Up @@ -61,9 +61,6 @@ The supported actions are:
The following example issues a warning when elaborated normally

```scala mdoc:invisible:reset
// Some other test is clobbering the global Logger which breaks the warnings below
// Setting the output stream to the Console fixes the issue
logger.Logger.setConsole()
// Helper to throw away return value so it doesn't show up in mdoc
def compile(gen: => chisel3.RawModule, args: Array[String] = Array()): Unit = {
circt.stage.ChiselStage.emitCHIRRTL(gen, args = args)
Expand Down
33 changes: 25 additions & 8 deletions firrtl/src/main/scala/logger/Logger.scala
Original file line number Diff line number Diff line change
Expand Up @@ -121,7 +121,18 @@ object Logger {
* @tparam A return type of the code block
* @return the original return of the code block
*/
def makeScope[A](options: AnnotationSeq = Seq.empty)(codeBlock: => A): A = {
def makeScope[A](options: AnnotationSeq = Seq.empty)(codeBlock: => A): A =
makeScope(() => setOptions(options))(codeBlock)

/** Set a scope for this logger based on available annotations
* @param options LoggerOptions to use
* @param codeBlock some Scala code over which to define this scope
* @tparam A return type of the code block
* @return the original return of the code block
*/
def makeScope[A](options: LoggerOptions)(codeBlock: => A): A = makeScope(() => setOptions(options))(codeBlock)

private def makeScope[A](doSetOptions: () => Unit)(codeBlock: => A): A = {
val runState: LoggerState = {
val newRunState = updatableLoggerState.value.getOrElse(new LoggerState)
if (newRunState.fromInvoke) {
Expand All @@ -133,7 +144,7 @@ object Logger {
}
}
updatableLoggerState.withValue(Some(runState)) {
setOptions(options)
doSetOptions()
codeBlock
}
}
Expand Down Expand Up @@ -326,20 +337,26 @@ object Logger {
Seq(new AddDefaults, Checks)
.foldLeft(inputAnnotations)((a, p) => p.transform(a))

val lopts = view[LoggerOptions](annotations)
state.globalLevel = (state.globalLevel, lopts.globalLogLevel) match {
setOptions(view[LoggerOptions](annotations))
}

/** Set logger options
* @param options options to set
*/
def setOptions(options: LoggerOptions): Unit = {
state.globalLevel = (state.globalLevel, options.globalLogLevel) match {
case (LogLevel.None, LogLevel.None) => LogLevel.None
case (x, LogLevel.None) => x
case (LogLevel.None, x) => x
case (_, x) => x
}
setClassLogLevels(lopts.classLogLevels)
setClassLogLevels(options.classLogLevels)

if (lopts.logFileName.nonEmpty) {
setOutput(lopts.logFileName.get)
if (options.logFileName.nonEmpty) {
setOutput(options.logFileName.get)
}

state.logClassNames = lopts.logClassNames
state.logClassNames = options.logClassNames
}
}

Expand Down
12 changes: 9 additions & 3 deletions firrtl/src/main/scala/logger/LoggerAnnotations.scala
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
package logger

import firrtl.annotations.{Annotation, NoTargetAnnotation}
import firrtl.options.{HasShellOptions, ShellOption}
import firrtl.options.{HasShellOptions, ShellOption, Unserializable}

/** An annotation associated with a Logger command line option */
sealed trait LoggerOption { this: Annotation => }
Expand All @@ -16,6 +16,7 @@ sealed trait LoggerOption { this: Annotation => }
case class LogLevelAnnotation(globalLogLevel: LogLevel.Value = LogLevel.None)
extends NoTargetAnnotation
with LoggerOption
with Unserializable

object LogLevelAnnotation extends HasShellOptions {

Expand All @@ -39,6 +40,7 @@ object LogLevelAnnotation extends HasShellOptions {
case class ClassLogLevelAnnotation(className: String, level: LogLevel.Value)
extends NoTargetAnnotation
with LoggerOption
with Unserializable

object ClassLogLevelAnnotation extends HasShellOptions {

Expand All @@ -63,7 +65,7 @@ object ClassLogLevelAnnotation extends HasShellOptions {
* - maps to [[LoggerOptions.logFileName]]
* - enabled with `--log-file`
*/
case class LogFileAnnotation(file: Option[String]) extends NoTargetAnnotation with LoggerOption
case class LogFileAnnotation(file: Option[String]) extends NoTargetAnnotation with LoggerOption with Unserializable

object LogFileAnnotation extends HasShellOptions {

Expand All @@ -81,7 +83,11 @@ object LogFileAnnotation extends HasShellOptions {
/** Enables class names in log output
* - enabled with `-lcn/--log-class-names`
*/
case object LogClassNamesAnnotation extends NoTargetAnnotation with LoggerOption with HasShellOptions {
case object LogClassNamesAnnotation
extends NoTargetAnnotation
with LoggerOption
with HasShellOptions
with Unserializable {

val options = Seq(
new ShellOption[Unit](
Expand Down
5 changes: 4 additions & 1 deletion src/main/scala/chisel3/aop/injecting/InjectingAspect.scala
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ import firrtl.options.Viewer.view
import firrtl.{ir, _}

import scala.collection.mutable
import logger.LoggerOptions

/** Aspect to inject Chisel code into a module of type M
*
Expand Down Expand Up @@ -57,14 +58,16 @@ abstract class InjectorAspect[T <: RawModule, M <: RawModule](
final def toAnnotation(modules: Iterable[M], circuit: String, moduleNames: Seq[String]): AnnotationSeq = {
modules.map { module =>
val chiselOptions = view[ChiselOptions](annotationsInAspect)
val loggerOptions = view[LoggerOptions](annotationsInAspect)
val dynamicContext =
new DynamicContext(
annotationsInAspect,
chiselOptions.throwOnFirstError,
chiselOptions.warningFilters,
chiselOptions.sourceRoots,
None,
Nil // FIXME this maybe should somehow grab definitions from earlier elaboration
Nil, // FIXME this maybe should somehow grab definitions from earlier elaboration
loggerOptions
)
// Add existing module names into the namespace. If injection logic instantiates new modules
// which would share the same name, they will get uniquified accordingly
Expand Down
12 changes: 9 additions & 3 deletions src/main/scala/chisel3/stage/phases/Elaborate.scala
Original file line number Diff line number Diff line change
Expand Up @@ -13,21 +13,26 @@ import chisel3.stage.{
ThrowOnFirstErrorAnnotation
}
import firrtl.AnnotationSeq
import firrtl.options.Phase
import firrtl.options.{Dependency, Phase}
import firrtl.options.Viewer.view
import logger.LoggerOptions

/** Elaborate all [[chisel3.stage.ChiselGeneratorAnnotation]]s into [[chisel3.stage.ChiselCircuitAnnotation]]s.
*/
class Elaborate extends Phase {

override def prerequisites = Seq.empty
override def prerequisites: Seq[Dependency[Phase]] = Seq(
Dependency[chisel3.stage.phases.Checks],
Dependency(_root_.logger.phases.Checks)
)
override def optionalPrerequisites = Seq.empty
override def optionalPrerequisiteOf = Seq.empty
override def invalidates(a: Phase) = false

def transform(annotations: AnnotationSeq): AnnotationSeq = annotations.flatMap {
case ChiselGeneratorAnnotation(gen) =>
val chiselOptions = view[ChiselOptions](annotations)
val loggerOptions = view[LoggerOptions](annotations)
try {
val context =
new DynamicContext(
Expand All @@ -36,7 +41,8 @@ class Elaborate extends Phase {
chiselOptions.warningFilters,
chiselOptions.sourceRoots,
None,
Nil
Nil,
loggerOptions
)
val (circuit, dut) =
Builder.build(Module(gen()), context)
Expand Down
60 changes: 12 additions & 48 deletions src/main/scala/circt/stage/ChiselStage.scala
Original file line number Diff line number Diff line change
Expand Up @@ -3,49 +3,13 @@
package circt.stage

import chisel3.RawModule
import chisel3.stage.{
ChiselCircuitAnnotation,
ChiselGeneratorAnnotation,
CircuitSerializationAnnotation,
PrintFullStackTraceAnnotation,
SourceRootAnnotation,
ThrowOnFirstErrorAnnotation,
WarningConfigurationAnnotation,
WarningConfigurationFileAnnotation,
WarningsAsErrorsAnnotation
}
import chisel3.stage.{ChiselCircuitAnnotation, ChiselGeneratorAnnotation, CircuitSerializationAnnotation}
import chisel3.stage.CircuitSerializationAnnotation.FirrtlFileFormat
import firrtl.{AnnotationSeq, EmittedVerilogCircuitAnnotation}
import firrtl.options.{
BareShell,
CustomFileEmission,
Dependency,
Phase,
PhaseManager,
Shell,
Stage,
StageMain,
Unserializable
}
import firrtl.options.{CustomFileEmission, Dependency, Phase, PhaseManager, Stage, StageMain, Unserializable}
import firrtl.stage.FirrtlCircuitAnnotation

trait CLI { this: BareShell =>
parser.note("CIRCT (MLIR FIRRTL Compiler) options")
Seq(
CIRCTTargetAnnotation,
PreserveAggregate,
ChiselGeneratorAnnotation,
PrintFullStackTraceAnnotation,
ThrowOnFirstErrorAnnotation,
WarningsAsErrorsAnnotation,
WarningConfigurationAnnotation,
WarningConfigurationFileAnnotation,
SourceRootAnnotation,
SplitVerilog,
FirtoolBinaryPath,
DumpFir
).foreach(_.addOptions(parser))
}
import logger.LogLevelAnnotation

/** Entry point for running Chisel with the CIRCT compiler.
*
Expand All @@ -58,13 +22,15 @@ class ChiselStage extends Stage {
override def optionalPrerequisiteOf = Seq.empty
override def invalidates(a: Phase) = false

override val shell = new Shell("circt") with CLI
override val shell = new firrtl.options.Shell("circt") with CLI {
// These are added by firrtl.options.Shell (which we must extend because we are a Stage)
override protected def includeLoggerOptions = false
}

override def run(annotations: AnnotationSeq): AnnotationSeq = {

val pm = new PhaseManager(
targets = Seq(
Dependency[chisel3.stage.phases.Checks],
Dependency[chisel3.stage.phases.AddImplicitOutputFile],
Dependency[chisel3.stage.phases.AddImplicitOutputAnnotationFile],
Dependency[chisel3.stage.phases.MaybeAspectPhase],
Expand All @@ -73,7 +39,6 @@ class ChiselStage extends Stage {
Dependency[chisel3.stage.phases.AddDedupGroupAnnotations],
Dependency[chisel3.stage.phases.MaybeInjectingPhase],
Dependency[circt.stage.phases.AddImplicitOutputFile],
Dependency[circt.stage.phases.Checks],
Dependency[circt.stage.phases.CIRCT]
),
currentState = Seq(
Expand All @@ -92,7 +57,6 @@ object ChiselStage {
/** A phase shared by all the CIRCT backends */
private def phase = new PhaseManager(
Seq(
Dependency[chisel3.stage.phases.Checks],
Dependency[chisel3.aop.injecting.InjectingPhase],
Dependency[chisel3.stage.phases.Elaborate],
Dependency[chisel3.stage.phases.Convert],
Expand All @@ -111,7 +75,7 @@ object ChiselStage {
val annos = Seq(
ChiselGeneratorAnnotation(() => gen),
CIRCTTargetAnnotation(CIRCTTarget.CHIRRTL)
) ++ (new BareShell("circt") with CLI).parse(args)
) ++ (new Shell("circt")).parse(args)

val resultAnnos = phase.transform(annos)

Expand All @@ -138,7 +102,7 @@ object ChiselStage {
val annos = Seq(
ChiselGeneratorAnnotation(() => gen),
CIRCTTargetAnnotation(CIRCTTarget.CHIRRTL)
) ++ (new BareShell("circt") with CLI).parse(args)
) ++ (new Shell("circt")).parse(args)

phase
.transform(annos)
Expand All @@ -157,7 +121,7 @@ object ChiselStage {
val annos = Seq(
ChiselGeneratorAnnotation(() => gen),
CIRCTTargetAnnotation(CIRCTTarget.FIRRTL)
) ++ (new BareShell("circt") with CLI).parse(args) ++ firtoolOpts.map(FirtoolOption(_))
) ++ (new Shell("circt")).parse(args) ++ firtoolOpts.map(FirtoolOption(_))

phase
.transform(annos)
Expand All @@ -176,7 +140,7 @@ object ChiselStage {
val annos = Seq(
ChiselGeneratorAnnotation(() => gen),
CIRCTTargetAnnotation(CIRCTTarget.HW)
) ++ (new BareShell("circt") with CLI).parse(args) ++ firtoolOpts.map(FirtoolOption(_))
) ++ (new Shell("circt")).parse(args) ++ firtoolOpts.map(FirtoolOption(_))

phase
.transform(annos)
Expand All @@ -201,7 +165,7 @@ object ChiselStage {
val annos = Seq(
ChiselGeneratorAnnotation(() => gen),
CIRCTTargetAnnotation(CIRCTTarget.SystemVerilog)
) ++ (new BareShell("circt") with CLI).parse(args) ++ firtoolOpts.map(FirtoolOption(_))
) ++ (new Shell("circt")).parse(args) ++ firtoolOpts.map(FirtoolOption(_))
phase
.transform(annos)
.collectFirst {
Expand Down
Loading

0 comments on commit e46a4a3

Please sign in to comment.