aac8a88552
There are several cases where names of types and functions in the debug info are either ambiguous, or not helpful, such as including ambiguous placeholders (e.g., `{{impl}}`, `{{closure}}` or `dyn _'`) or dropping qualifications (e.g., for dynamic types). Instead, each debug symbol name should be unique and useful: * Include disambiguators for anonymous `DefPathDataName` (closures and generators), and unify their formatting when used as a path-qualifier vs item being qualified. * Qualify the principal trait for dynamic types. * If there is no principal trait for a dynamic type, emit all other traits instead. * Respect the `qualified` argument when emitting ref and pointer types. * For implementations, emit the disambiguator. * Print const generics when emitting generic parameters or arguments. Additionally, when targeting MSVC, its debugger treats many command arguments as C++ expressions, even when the argument is defined to be a symbol name. As such names in the debug info need to be more C++-like to be parsed correctly: * Avoid characters with special meaning (`#`, `[`, `"`, `+`). * Never start a name with `<` or `{` as this is treated as an operator. * `>>` is always treated as a right-shift, even when parsing generic arguments (so add a space to avoid this). * Emit function declarations using C/C++ style syntax (e.g., leading return type). * Emit arrays as a synthetic `array$<type, size>` type. * Include a `$` in all synthetic types as this is a legal character for C++, but not Rust (thus we avoid collisions with user types).
63 lines
2.4 KiB
Rust
63 lines
2.4 KiB
Rust
// Verify debuginfo for generators:
|
|
// - Each variant points to the file and line of its yield point
|
|
// - The discriminants are marked artificial
|
|
// - Other fields are not marked artificial
|
|
//
|
|
//
|
|
// compile-flags: -C debuginfo=2 --edition=2018
|
|
// ignore-msvc
|
|
|
|
#![feature(generators, generator_trait)]
|
|
use std::ops::Generator;
|
|
|
|
fn generator_test() -> impl Generator<Yield = i32, Return = ()> {
|
|
|| {
|
|
yield 0;
|
|
let s = String::from("foo");
|
|
yield 1;
|
|
}
|
|
}
|
|
|
|
// FIXME: No way to reliably check the filename.
|
|
|
|
// CHECK-DAG: [[GEN_FN:!.*]] = !DINamespace(name: "generator_test"
|
|
// CHECK-DAG: [[GEN:!.*]] = !DICompositeType(tag: DW_TAG_structure_type, name: "{generator#0}", scope: [[GEN_FN]]
|
|
// CHECK: [[VARIANT:!.*]] = !DICompositeType(tag: DW_TAG_variant_part, scope: [[GEN_FN]],
|
|
// CHECK-NOT: flags: DIFlagArtificial
|
|
// CHECK-SAME: discriminator: [[DISC:![0-9]*]]
|
|
// CHECK: {{!.*}} = !DIDerivedType(tag: DW_TAG_member, name: "0", scope: [[VARIANT]],
|
|
// CHECK-SAME: file: [[FILE:![0-9]*]], line: 14,
|
|
// CHECK-NOT: flags: DIFlagArtificial
|
|
// CHECK-SAME: )
|
|
// CHECK: {{!.*}} = !DICompositeType(tag: DW_TAG_structure_type, name: "Unresumed", scope: [[GEN]],
|
|
// CHECK-NOT: flags: DIFlagArtificial
|
|
// CHECK-SAME: )
|
|
// CHECK: {{!.*}} = !DIDerivedType(tag: DW_TAG_member, name: "1", scope: [[VARIANT]],
|
|
// CHECK-SAME: file: [[FILE]], line: 18,
|
|
// CHECK-NOT: flags: DIFlagArtificial
|
|
// CHECK-SAME: )
|
|
// CHECK: {{!.*}} = !DIDerivedType(tag: DW_TAG_member, name: "2", scope: [[VARIANT]],
|
|
// CHECK-SAME: file: [[FILE]], line: 18,
|
|
// CHECK-NOT: flags: DIFlagArtificial
|
|
// CHECK-SAME: )
|
|
// CHECK: {{!.*}} = !DIDerivedType(tag: DW_TAG_member, name: "3", scope: [[VARIANT]],
|
|
// CHECK-SAME: file: [[FILE]], line: 15,
|
|
// CHECK-NOT: flags: DIFlagArtificial
|
|
// CHECK-SAME: )
|
|
// CHECK: {{!.*}} = !DIDerivedType(tag: DW_TAG_member, name: "4", scope: [[VARIANT]],
|
|
// CHECK-SAME: file: [[FILE]], line: 17,
|
|
// CHECK-NOT: flags: DIFlagArtificial
|
|
// CHECK-SAME: )
|
|
// CHECK: [[S1:!.*]] = !DICompositeType(tag: DW_TAG_structure_type, name: "Suspend1", scope: [[GEN]],
|
|
// CHECK-NOT: flags: DIFlagArtificial
|
|
// CHECK-SAME: )
|
|
// CHECK: {{!.*}} = !DIDerivedType(tag: DW_TAG_member, name: "s", scope: [[S1]]
|
|
// CHECK-NOT: flags: DIFlagArtificial
|
|
// CHECK-SAME: )
|
|
// CHECK: [[DISC]] = !DIDerivedType(tag: DW_TAG_member, name: "__state", scope: [[GEN_FN]],
|
|
// CHECK-SAME: flags: DIFlagArtificial
|
|
|
|
fn main() {
|
|
let _dummy = generator_test();
|
|
}
|