I've made some progress chasing down a compiler crash I've been experiencing. However, I think I've reached the point where I need some advice to understand what the best way forward is.
The problem is in the constraint system. There are a number of places where I believe the result of this function is unconditionally dereferenced.
I briefly investigated modifing ArgumentMismatchFailure from FunctionArgApplyInfo to std::optional<FunctionArgApplyInfo>. It will involve changes in quite a few places. Worse, because this is an internal detail of the constraint system, it is hard for me to expose this particular detail via the unit test system.
Does anyone have any ideas on what I should do next?
This is a good suggestion. And, from it, I have found numerous other code paths that will crash if getLocator() returns null. However, this is not the case in my particular situation.
Isn't the problem higher-level though? There are actually many code paths within getFunctionArgApplyInfo that can return null, and all will crash.
I believe @hamishknight made this an unconditional deference at some point because ArgumentMismatchFailure is supposed to be anchored on a locator that ends on ArgToParam which should always work for getFunctionArgApplyInfo if we have a different locator for this particular fix, that's the underlying issue here.
An easy way to check that would be to use -Xfrontend -debug-constraint and see Fixes: section of the solution, it would print each fix and its locator.
Ah ha! I tried this and it produced an enormous amount of output. So, I'm truncating to the last bit right before the crash. Interestingly, this now also prints an assertion failure!
Overload choices:
locator@0x1569ea850 [DeclRef@@__swiftmacro_11EmpireTests10TestRecord08IndexKeyD0fMe_.swift:30:44] with EmpireTests.(file).TestRecord extension.select(in:a:).a@@__swiftmacro_11EmpireTests10TestRecord08IndexKeyD0fMe_.swift:29:63 as a: String
locator@0x1569ea200 [DeclRef@@__swiftmacro_11EmpireTests10TestRecord08IndexKeyD0fMe_.swift:30:10] with EmpireTests.(file).TestRecord extension.select(in:a:).context@@__swiftmacro_11EmpireTests10TestRecord08IndexKeyD0fMe_.swift:29:34 as context: TransactionContext
locator@0x1569eaa10 [Call@@__swiftmacro_11EmpireTests10TestRecord08IndexKeyD0fMe_.swift:30:32 → apply function → constructor member] with Empire.(file).Query.init(_:last:)@/Users/matt/Developer/Empire/Sources/Empire/Query.swift:42:9 as Query<Pack{/* shape: $T2 */ repeat $T2}, $T3>.Type.init: ($T9, ComparisonOperator<$T8>) -> Query<Pack{/* shape: $T2 */ repeat $T2}, $T3>
locator@0x1569ea278 [UnresolvedDot@@__swiftmacro_11EmpireTests10TestRecord08IndexKeyD0fMe_.swift:30:18 → member] with Empire.(file).TransactionContext.select(query:)@/Users/matt/Developer/Empire/Sources/Empire/Store.swift:74:14 as TransactionContext.select: (Query<Pack{$T14}, $T13>) throws -> sending [$T11]
Constraint restrictions:
Query<<<hole>>> to Query<<<hole>>> is [deep equality]
Array<TestRecord> to Array<TestRecord> is [deep equality]
Trailing closure matching:
locator@0x1569ea250 [UnresolvedDot@@__swiftmacro_11EmpireTests10TestRecord08IndexKeyD0fMe_.swift:30:18 → apply argument]: forward
locator@0x1569ea978 [Call@@__swiftmacro_11EmpireTests10TestRecord08IndexKeyD0fMe_.swift:30:32 → apply argument]: forward
locator@0x156c49df8 [Call@@__swiftmacro_11EmpireTests10TestRecord08IndexKeyD0fMe_.swift:30:18 → apply argument]: forward
Opened types:
locator@0x1569ea418 [Type@@__swiftmacro_11EmpireTests10TestRecord08IndexKeyD0fMe_.swift:30:32] opens 'Last' (τ_0_1) -> <<hole>> [from $T3], 'each Component' (each τ_0_0) -> Pack{} [from $T2]
locator@0x1569eaa10 [Call@@__swiftmacro_11EmpireTests10TestRecord08IndexKeyD0fMe_.swift:30:32 → apply function → constructor member] opens 'Last' (τ_0_1) -> <<hole>> [from $T8], 'each Component' (each τ_0_0) -> Pack{} [from $T7]
locator@0x1569ea278 [UnresolvedDot@@__swiftmacro_11EmpireTests10TestRecord08IndexKeyD0fMe_.swift:30:18 → member] opens 'Last' (τ_0_2) -> <<hole>> [from $T13], 'Record' (τ_0_0) -> TestRecord [from $T11], 'each Component' (each τ_0_1) -> Pack{} [from $T12]
Defaulted constraints: locator@0x1569ea490 [Type@@__swiftmacro_11EmpireTests10TestRecord08IndexKeyD0fMe_.swift:30:32 → generic parameter 'Last']
Fixes:
[fix: allow argument to parameter type conversion mismatch] @ locator@0x156c493d8 [Call@@__swiftmacro_11EmpireTests10TestRecord08IndexKeyD0fMe_.swift:30:32 → apply argument → comparing call argument #0 to parameter #1]
[fix: default generic argument 'Last' to 'Any'] @ locator@0x1569ea418 [Type@@__swiftmacro_11EmpireTests10TestRecord08IndexKeyD0fMe_.swift:30:32]
Assertion failed: (!HasOriginalArgs && "Query original args instead"), function isTrailingClosureIndex, file ArgumentList.h, line 434.
I've added this to the issue too.
So, it sounds like what I should be looking for isn't to fix the null-derefs, but to address the reason why they are happening in the first place?
Yes the intention here is that an AllowArgumentMismatch fix should only ever be recorded for a mismatch in a call, in which case we ought to always be able to retrieve information about that call through getFunctionArgApplyInfo.
The assertion:
Assertion failed: (!HasOriginalArgs && "Query original args instead"), function isTrailingClosureIndex, file ArgumentList.h, line 434.
Indicates that we're trying to re-typecheck an already typechecked AST (an un-typechecked argument list should never have original arguments set), so that's likely where the issue is.
This is starting to make more sense. This bug is only reproducible via a macro. If I just manually write the code the macro would generate, it compiles fine.
Do you have any pointers on how I might further investigate from this angle?
It should be possible to figure out based on the debug constraints output whether the same expression is type-checked multiple times. Look for ---Initial constraints for the given expression---.
The last non-indented occurrence should be the one to look for, if you look at the AST it prints, that AST should match a previous the AST for a previous ---Type-checked expression---, modulo the types printed.
It would also be helpful to reduce the issue down as much as you can. Assuming you have the stack dump, it should give you the frontend invocation. If not, you could re-run swift build with -v to see what the invocation was. With the frontend invocation, you ought to be able to re-run it to reproduce the crash again.
You'll want to take the invocation, and ensure that -primary-file is only done for the file in which you're seeing the crash. Then you ought to be able to start removing unrelated code from the primary file. Once you have a minimal primary file, you should be able to move any code that's being referenced from any secondary files into the primary file, drop the secondary files from the invocation, and then reduce further.
Assuming you're interested in debugging this yourself, you then ought to be able run the invocation with a debug build of the compiler, and set a breakpoint in typeCheckTarget to figure out how/when the expression is being type-checked multiple times.
I promise I have been trying to reduce this. For what it's worth, the contents of the primary file looks like this:
import Empire
@IndexKeyRecord("a", "b")
struct TestRecord: Hashable {
let a: String
let b: UInt
var c: String
}
The crash occurs during the expansion of this IndexKeyRecord macro. Is it possible for me to reduce this further and keep the macro aspect? That part seems to be a critical component.
You ought to at least be able to reduce away the import Empire by inlining the macro declaration for IndexKeyRecord into the file. You won't be able to completely reduce this down to a single-file reproducer given the macro, but if you could reduce the macro implementation down to a single file that can be built standalone (except SwiftSyntax/macro dependencies), that's probably sufficient I'd say.
Yup that's sufficient for me to reproduce locally (note you can also remove import SwiftCompilerPlugin), thank you! If you could update the issue with it, that'd be great.