2020-07-02 10:37:29 +02:00
|
|
|
#![feature(core_intrinsics)]
|
2019-12-08 11:06:35 +01:00
|
|
|
|
|
|
|
use std::panic::Location;
|
|
|
|
|
|
|
|
#[track_caller]
|
|
|
|
fn tracked() -> &'static Location<'static> {
|
|
|
|
Location::caller() // most importantly, we never get line 7
|
|
|
|
}
|
|
|
|
|
|
|
|
fn nested_intrinsic() -> &'static Location<'static> {
|
|
|
|
Location::caller()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn nested_tracked() -> &'static Location<'static> {
|
|
|
|
tracked()
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! caller_location_from_macro {
|
2022-06-20 23:40:39 -07:00
|
|
|
() => {
|
|
|
|
core::panic::Location::caller()
|
|
|
|
};
|
2019-12-08 11:06:35 +01:00
|
|
|
}
|
|
|
|
|
2020-01-28 22:55:07 +01:00
|
|
|
fn test_fn_ptr() {
|
|
|
|
fn pass_to_ptr_call<T>(f: fn(T), x: T) {
|
|
|
|
f(x);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[track_caller]
|
|
|
|
fn tracked_unit(_: ()) {
|
|
|
|
let expected_line = line!() - 1;
|
|
|
|
let location = std::panic::Location::caller();
|
|
|
|
assert_eq!(location.file(), file!());
|
|
|
|
assert_eq!(location.line(), expected_line, "call shims report location as fn definition");
|
|
|
|
}
|
|
|
|
|
|
|
|
pass_to_ptr_call(tracked_unit, ());
|
|
|
|
}
|
|
|
|
|
2020-07-31 18:30:07 +02:00
|
|
|
fn test_trait_obj() {
|
|
|
|
trait Tracked {
|
|
|
|
#[track_caller]
|
2021-07-10 11:51:00 -05:00
|
|
|
fn handle(&self) -> &'static Location<'static> {
|
|
|
|
std::panic::Location::caller()
|
2020-07-31 18:30:07 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Tracked for () {}
|
|
|
|
impl Tracked for u8 {}
|
|
|
|
|
2021-07-10 11:51:00 -05:00
|
|
|
// Test that we get the correct location
|
|
|
|
// even with a call through a trait object
|
|
|
|
|
2020-07-31 18:30:07 +02:00
|
|
|
let tracked: &dyn Tracked = &5u8;
|
2021-07-10 11:51:00 -05:00
|
|
|
let location = tracked.handle();
|
|
|
|
let expected_line = line!() - 1;
|
|
|
|
assert_eq!(location.file(), file!());
|
|
|
|
assert_eq!(location.line(), expected_line);
|
|
|
|
assert_eq!(location.column(), 28);
|
2020-07-31 18:30:07 +02:00
|
|
|
|
|
|
|
const TRACKED: &dyn Tracked = &();
|
2021-07-10 11:51:00 -05:00
|
|
|
let location = TRACKED.handle();
|
|
|
|
let expected_line = line!() - 1;
|
|
|
|
assert_eq!(location.file(), file!());
|
|
|
|
assert_eq!(location.line(), expected_line);
|
|
|
|
assert_eq!(location.column(), 28);
|
2020-07-31 18:30:07 +02:00
|
|
|
}
|
|
|
|
|
2022-07-22 10:34:52 -04:00
|
|
|
fn test_trait_obj2() {
|
|
|
|
// track_caller on the impl but not the trait.
|
|
|
|
pub trait Foo {
|
|
|
|
fn foo(&self) -> &'static Location<'static>;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct Bar;
|
|
|
|
impl Foo for Bar {
|
|
|
|
#[track_caller]
|
|
|
|
fn foo(&self) -> &'static Location<'static> {
|
|
|
|
std::panic::Location::caller()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let expected_line = line!() - 4; // the `fn` signature above
|
|
|
|
|
|
|
|
let f = &Bar as &dyn Foo;
|
|
|
|
let loc = f.foo(); // trait doesn't track, so we don't point at this call site
|
|
|
|
assert_eq!(loc.file(), file!());
|
|
|
|
assert_eq!(loc.line(), expected_line);
|
|
|
|
}
|
|
|
|
|
2019-12-08 11:06:35 +01:00
|
|
|
fn main() {
|
|
|
|
let location = Location::caller();
|
2020-01-28 22:55:07 +01:00
|
|
|
let expected_line = line!() - 1;
|
2019-12-08 11:06:35 +01:00
|
|
|
assert_eq!(location.file(), file!());
|
2020-01-28 22:55:07 +01:00
|
|
|
assert_eq!(location.line(), expected_line);
|
2019-12-08 11:06:35 +01:00
|
|
|
assert_eq!(location.column(), 20);
|
|
|
|
|
|
|
|
let tracked = tracked();
|
2020-01-28 22:55:07 +01:00
|
|
|
let expected_line = line!() - 1;
|
2019-12-08 11:06:35 +01:00
|
|
|
assert_eq!(tracked.file(), file!());
|
2020-01-28 22:55:07 +01:00
|
|
|
assert_eq!(tracked.line(), expected_line);
|
2019-12-08 11:06:35 +01:00
|
|
|
assert_eq!(tracked.column(), 19);
|
|
|
|
|
|
|
|
let nested = nested_intrinsic();
|
|
|
|
assert_eq!(nested.file(), file!());
|
|
|
|
assert_eq!(nested.line(), 11);
|
|
|
|
assert_eq!(nested.column(), 5);
|
|
|
|
|
|
|
|
let contained = nested_tracked();
|
|
|
|
assert_eq!(contained.file(), file!());
|
|
|
|
assert_eq!(contained.line(), 15);
|
|
|
|
assert_eq!(contained.column(), 5);
|
|
|
|
|
|
|
|
// `Location::caller()` in a macro should behave similarly to `file!` and `line!`,
|
|
|
|
// i.e. point to where the macro was invoked, instead of the macro itself.
|
|
|
|
let inmacro = caller_location_from_macro!();
|
2020-01-28 22:55:07 +01:00
|
|
|
let expected_line = line!() - 1;
|
2019-12-08 11:06:35 +01:00
|
|
|
assert_eq!(inmacro.file(), file!());
|
2020-01-28 22:55:07 +01:00
|
|
|
assert_eq!(inmacro.line(), expected_line);
|
2019-12-08 11:06:35 +01:00
|
|
|
assert_eq!(inmacro.column(), 19);
|
|
|
|
|
|
|
|
let intrinsic = core::intrinsics::caller_location();
|
2020-01-28 22:55:07 +01:00
|
|
|
let expected_line = line!() - 1;
|
2019-12-08 11:06:35 +01:00
|
|
|
assert_eq!(intrinsic.file(), file!());
|
2020-01-28 22:55:07 +01:00
|
|
|
assert_eq!(intrinsic.line(), expected_line);
|
2019-12-08 11:06:35 +01:00
|
|
|
assert_eq!(intrinsic.column(), 21);
|
2020-01-28 22:55:07 +01:00
|
|
|
|
|
|
|
test_fn_ptr();
|
2020-07-31 18:30:07 +02:00
|
|
|
test_trait_obj();
|
2022-07-22 10:34:52 -04:00
|
|
|
test_trait_obj2();
|
2019-12-08 11:06:35 +01:00
|
|
|
}
|