compiletest/runtest/
debuginfo.rs

1use std::ffi::{OsStr, OsString};
2use std::fs::File;
3use std::io::{BufRead, BufReader, Read};
4use std::process::{Command, Output, Stdio};
5
6use camino::Utf8Path;
7use tracing::debug;
8
9use super::debugger::DebuggerCommands;
10use super::{Debugger, Emit, ProcRes, TestCx, Truncated, WillExecute};
11use crate::debuggers::extract_gdb_version;
12
13impl TestCx<'_> {
14    pub(super) fn run_debuginfo_test(&self) {
15        match self.config.debugger.unwrap() {
16            Debugger::Cdb => self.run_debuginfo_cdb_test(),
17            Debugger::Gdb => self.run_debuginfo_gdb_test(),
18            Debugger::Lldb => self.run_debuginfo_lldb_test(),
19        }
20    }
21
22    fn run_debuginfo_cdb_test(&self) {
23        let exe_file = self.make_exe_name();
24
25        // Existing PDB files are update in-place. When changing the debuginfo
26        // the compiler generates for something, this can lead to the situation
27        // where both the old and the new version of the debuginfo for the same
28        // type is present in the PDB, which is very confusing.
29        // Therefore we delete any existing PDB file before compiling the test
30        // case.
31        // FIXME: If can reliably detect that MSVC's link.exe is used, then
32        //        passing `/INCREMENTAL:NO` might be a cleaner way to do this.
33        let pdb_file = exe_file.with_extension(".pdb");
34        if pdb_file.exists() {
35            std::fs::remove_file(pdb_file).unwrap();
36        }
37
38        // compile test file (it should have 'compile-flags:-g' in the directive)
39        let should_run = self.run_if_enabled();
40        let compile_result = self.compile_test(should_run, Emit::None);
41        if !compile_result.status.success() {
42            self.fatal_proc_rec("compilation failed!", &compile_result);
43        }
44        if let WillExecute::Disabled = should_run {
45            return;
46        }
47
48        // Parse debugger commands etc from test files
49        let dbg_cmds = DebuggerCommands::parse_from(&self.testpaths.file, "cdb")
50            .unwrap_or_else(|e| self.fatal(&e));
51
52        // https://docs.microsoft.com/en-us/windows-hardware/drivers/debugger/debugger-commands
53        let mut script_str = String::with_capacity(2048);
54        script_str.push_str("version\n"); // List CDB (and more) version info in test output
55        script_str.push_str(".nvlist\n"); // List loaded `*.natvis` files, bulk of custom MSVC debug
56
57        // If a .js file exists next to the source file being tested, then this is a JavaScript
58        // debugging extension that needs to be loaded.
59        let mut js_extension = self.testpaths.file.clone();
60        js_extension.set_extension("cdb.js");
61        if js_extension.exists() {
62            script_str.push_str(&format!(".scriptload \"{}\"\n", js_extension));
63        }
64
65        // Set breakpoints on every line that contains the string "#break"
66        let source_file_name = self.testpaths.file.file_name().unwrap();
67        for line in &dbg_cmds.breakpoint_lines {
68            script_str.push_str(&format!("bp `{}:{}`\n", source_file_name, line));
69        }
70
71        // Append the other `cdb-command:`s
72        for line in &dbg_cmds.commands {
73            script_str.push_str(line);
74            script_str.push('\n');
75        }
76
77        script_str.push_str("qq\n"); // Quit the debugger (including remote debugger, if any)
78
79        // Write the script into a file
80        debug!("script_str = {}", script_str);
81        self.dump_output_file(&script_str, "debugger.script");
82        let debugger_script = self.make_out_name("debugger.script");
83
84        let cdb_path = &self.config.cdb.as_ref().unwrap();
85        let mut cdb = Command::new(cdb_path);
86        cdb.arg("-lines") // Enable source line debugging.
87            .arg("-cf")
88            .arg(&debugger_script)
89            .arg(&exe_file);
90
91        let debugger_run_result = self.compose_and_run(
92            cdb,
93            self.config.target_run_lib_path.as_path(),
94            None, // aux_path
95            None, // input
96        );
97
98        if !debugger_run_result.status.success() {
99            self.fatal_proc_rec("Error while running CDB", &debugger_run_result);
100        }
101
102        if let Err(e) = dbg_cmds.check_output(&debugger_run_result) {
103            self.fatal_proc_rec(&e, &debugger_run_result);
104        }
105    }
106
107    fn run_debuginfo_gdb_test(&self) {
108        let dbg_cmds = DebuggerCommands::parse_from(&self.testpaths.file, "gdb")
109            .unwrap_or_else(|e| self.fatal(&e));
110        let mut cmds = dbg_cmds.commands.join("\n");
111
112        // compile test file (it should have 'compile-flags:-g' in the directive)
113        let should_run = self.run_if_enabled();
114        let compiler_run_result = self.compile_test(should_run, Emit::None);
115        if !compiler_run_result.status.success() {
116            self.fatal_proc_rec("compilation failed!", &compiler_run_result);
117        }
118        if let WillExecute::Disabled = should_run {
119            return;
120        }
121
122        let exe_file = self.make_exe_name();
123
124        let debugger_run_result;
125        // If bootstrap gave us an `--android-cross-path`, assume the target
126        // needs Android-specific handling.
127        if let Some(android_cross_path) = self.config.android_cross_path.as_deref() {
128            cmds = cmds.replace("run", "continue");
129
130            // write debugger script
131            let mut script_str = String::with_capacity(2048);
132            script_str.push_str(&format!("set charset {}\n", Self::charset()));
133            script_str.push_str(&format!("set sysroot {android_cross_path}\n"));
134            script_str.push_str(&format!("file {}\n", exe_file));
135            script_str.push_str("target remote :5039\n");
136            script_str.push_str(&format!(
137                "set solib-search-path \
138                 ./{}/stage2/lib/rustlib/{}/lib/\n",
139                self.config.host, self.config.target
140            ));
141            for line in &dbg_cmds.breakpoint_lines {
142                script_str.push_str(
143                    format!("break {}:{}\n", self.testpaths.file.file_name().unwrap(), *line)
144                        .as_str(),
145                );
146            }
147            script_str.push_str(&cmds);
148            script_str.push_str("\nquit\n");
149
150            debug!("script_str = {}", script_str);
151            self.dump_output_file(&script_str, "debugger.script");
152
153            // Note: when `--android-cross-path` is specified, we expect both `adb_path` and
154            // `adb_test_dir` to be available.
155            let adb_path = self.config.adb_path.as_ref().expect("`adb_path` must be specified");
156            let adb_test_dir =
157                self.config.adb_test_dir.as_ref().expect("`adb_test_dir` must be specified");
158
159            Command::new(adb_path)
160                .arg("push")
161                .arg(&exe_file)
162                .arg(adb_test_dir)
163                .status()
164                .unwrap_or_else(|e| panic!("failed to exec `{adb_path:?}`: {e:?}"));
165
166            Command::new(adb_path)
167                .args(&["forward", "tcp:5039", "tcp:5039"])
168                .status()
169                .unwrap_or_else(|e| panic!("failed to exec `{adb_path:?}`: {e:?}"));
170
171            let adb_arg = format!(
172                "export LD_LIBRARY_PATH={}; \
173                 gdbserver{} :5039 {}/{}",
174                adb_test_dir,
175                if self.config.target.contains("aarch64") { "64" } else { "" },
176                adb_test_dir,
177                exe_file.file_name().unwrap()
178            );
179
180            debug!("adb arg: {}", adb_arg);
181            let mut adb = Command::new(adb_path)
182                .args(&["shell", &adb_arg])
183                .stdout(Stdio::piped())
184                .stderr(Stdio::inherit())
185                .spawn()
186                .unwrap_or_else(|e| panic!("failed to exec `{adb_path:?}`: {e:?}"));
187
188            // Wait for the gdbserver to print out "Listening on port ..."
189            // at which point we know that it's started and then we can
190            // execute the debugger below.
191            let mut stdout = BufReader::new(adb.stdout.take().unwrap());
192            let mut line = String::new();
193            loop {
194                line.truncate(0);
195                stdout.read_line(&mut line).unwrap();
196                if line.starts_with("Listening on port 5039") {
197                    break;
198                }
199            }
200            drop(stdout);
201
202            let mut debugger_script = OsString::from("-command=");
203            debugger_script.push(self.make_out_name("debugger.script"));
204            let debugger_opts: &[&OsStr] =
205                &["-quiet".as_ref(), "-batch".as_ref(), "-nx".as_ref(), &debugger_script];
206
207            let gdb_path = self.config.gdb.as_ref().unwrap();
208            let Output { status, stdout, stderr } = Command::new(&gdb_path)
209                .args(debugger_opts)
210                .output()
211                .unwrap_or_else(|e| panic!("failed to exec `{gdb_path:?}`: {e:?}"));
212            let cmdline = {
213                let mut gdb = Command::new(&format!("{}-gdb", self.config.target));
214                gdb.args(debugger_opts);
215                // FIXME(jieyouxu): don't pass an empty Path
216                let cmdline = self.make_cmdline(&gdb, Utf8Path::new(""));
217                self.logv(format_args!("executing {cmdline}"));
218                cmdline
219            };
220
221            debugger_run_result = ProcRes {
222                status,
223                stdout: String::from_utf8(stdout).unwrap(),
224                stderr: String::from_utf8(stderr).unwrap(),
225                truncated: Truncated::No,
226                cmdline,
227            };
228            if adb.kill().is_err() {
229                writeln!(self.stdout, "Adb process is already finished.");
230            }
231        } else {
232            let rust_pp_module_abs_path = self.config.src_root.join("src").join("etc");
233            // write debugger script
234            let mut script_str = String::with_capacity(2048);
235            script_str.push_str(&format!("set charset {}\n", Self::charset()));
236            script_str.push_str("show version\n");
237
238            match self.config.gdb_version {
239                Some(version) => {
240                    writeln!(
241                        self.stdout,
242                        "NOTE: compiletest thinks it is using GDB version {}",
243                        version
244                    );
245
246                    if !self.props.disable_gdb_pretty_printers
247                        && version > extract_gdb_version("7.4").unwrap()
248                    {
249                        // Add the directory containing the pretty printers to
250                        // GDB's script auto loading safe path
251                        script_str.push_str(&format!(
252                            "add-auto-load-safe-path {}\n",
253                            rust_pp_module_abs_path.as_str().replace(r"\", r"\\")
254                        ));
255
256                        // Add the directory containing the output binary to
257                        // include embedded pretty printers to GDB's script
258                        // auto loading safe path
259                        script_str.push_str(&format!(
260                            "add-auto-load-safe-path {}\n",
261                            self.output_base_dir().as_str().replace(r"\", r"\\")
262                        ));
263                    }
264                }
265                _ => {
266                    writeln!(
267                        self.stdout,
268                        "NOTE: compiletest does not know which version of \
269                         GDB it is using"
270                    );
271                }
272            }
273
274            // The following line actually doesn't have to do anything with
275            // pretty printing, it just tells GDB to print values on one line:
276            script_str.push_str("set print pretty off\n");
277
278            // Add the pretty printer directory to GDB's source-file search path
279            script_str.push_str(&format!(
280                "directory {}\n",
281                rust_pp_module_abs_path.as_str().replace(r"\", r"\\")
282            ));
283
284            // Load the target executable
285            script_str.push_str(&format!("file {}\n", exe_file.as_str().replace(r"\", r"\\")));
286
287            // Force GDB to print values in the Rust format.
288            script_str.push_str("set language rust\n");
289
290            // Add line breakpoints
291            for line in &dbg_cmds.breakpoint_lines {
292                script_str.push_str(&format!(
293                    "break '{}':{}\n",
294                    self.testpaths.file.file_name().unwrap(),
295                    *line
296                ));
297            }
298
299            script_str.push_str(&cmds);
300            script_str.push_str("\nquit\n");
301
302            debug!("script_str = {}", script_str);
303            self.dump_output_file(&script_str, "debugger.script");
304
305            let mut debugger_script = OsString::from("-command=");
306            debugger_script.push(self.make_out_name("debugger.script"));
307
308            let debugger_opts: &[&OsStr] =
309                &["-quiet".as_ref(), "-batch".as_ref(), "-nx".as_ref(), &debugger_script];
310
311            let mut gdb = Command::new(self.config.gdb.as_ref().unwrap());
312
313            // FIXME: we are propagating `PYTHONPATH` from the environment, not a compiletest flag!
314            let pythonpath = if let Ok(pp) = std::env::var("PYTHONPATH") {
315                format!("{pp}:{rust_pp_module_abs_path}")
316            } else {
317                rust_pp_module_abs_path.to_string()
318            };
319            gdb.args(debugger_opts).env("PYTHONPATH", pythonpath);
320
321            debugger_run_result =
322                self.compose_and_run(gdb, self.config.target_run_lib_path.as_path(), None, None);
323        }
324
325        if !debugger_run_result.status.success() {
326            self.fatal_proc_rec("gdb failed to execute", &debugger_run_result);
327        }
328
329        if let Err(e) = dbg_cmds.check_output(&debugger_run_result) {
330            self.fatal_proc_rec(&e, &debugger_run_result);
331        }
332    }
333
334    fn run_debuginfo_lldb_test(&self) {
335        let Some(ref lldb) = self.config.lldb else {
336            self.fatal("Can't run LLDB test because LLDB's path is not set.");
337        };
338
339        // compile test file (it should have 'compile-flags:-g' in the directive)
340        let should_run = self.run_if_enabled();
341        let compile_result = self.compile_test(should_run, Emit::None);
342        if !compile_result.status.success() {
343            self.fatal_proc_rec("compilation failed!", &compile_result);
344        }
345        if let WillExecute::Disabled = should_run {
346            return;
347        }
348
349        let exe_file = self.make_exe_name();
350
351        match self.config.lldb_version {
352            Some(ref version) => {
353                writeln!(
354                    self.stdout,
355                    "NOTE: compiletest thinks it is using LLDB version {}",
356                    version
357                );
358            }
359            _ => {
360                writeln!(
361                    self.stdout,
362                    "NOTE: compiletest does not know which version of \
363                     LLDB it is using"
364                );
365            }
366        }
367
368        // Parse debugger commands etc from test files
369        let dbg_cmds = DebuggerCommands::parse_from(&self.testpaths.file, "lldb")
370            .unwrap_or_else(|e| self.fatal(&e));
371
372        // Write debugger script:
373        // We don't want to hang when calling `quit` while the process is still running
374        let mut script_str = String::from("settings set auto-confirm true\n");
375
376        // macOS has a system for restricting access to files and peripherals
377        // called Transparency, Consent, and Control (TCC), which can be
378        // configured using the "Security & Privacy" tab in your settings.
379        //
380        // This system is provenance-based: if Terminal.app is given access to
381        // your Desktop, and you launch a binary within Terminal.app, the new
382        // binary also has access to the files on your Desktop.
383        //
384        // By default though, LLDB launches binaries in very isolated
385        // contexts. This includes resetting any TCC grants that might
386        // otherwise have been inherited.
387        //
388        // In effect, this means that if the developer has placed the rust
389        // repository under one of the system-protected folders, they will get
390        // a pop-up _for each binary_ asking for permissions to access the
391        // folder - quite annoying.
392        //
393        // To avoid this, we tell LLDB to spawn processes with TCC grants
394        // inherited from the parent process.
395        //
396        // Setting this also avoids unnecessary overhead from XprotectService
397        // when running with the Developer Tool grant.
398        //
399        // TIP: If you want to allow launching `lldb ~/Desktop/my_binary`
400        // without being prompted, you can put this in your `~/.lldbinit` too.
401        if self.config.host.contains("darwin") {
402            script_str.push_str("settings set target.inherit-tcc true\n");
403        }
404
405        // Make LLDB emit its version, so we have it documented in the test output
406        script_str.push_str("version\n");
407
408        // Switch LLDB into "Rust mode".
409        let rust_pp_module_abs_path = self.config.src_root.join("src/etc");
410
411        script_str.push_str(&format!(
412            "command script import {}/lldb_lookup.py\n",
413            rust_pp_module_abs_path
414        ));
415        File::open(rust_pp_module_abs_path.join("lldb_commands"))
416            .and_then(|mut file| file.read_to_string(&mut script_str))
417            .expect("Failed to read lldb_commands");
418
419        // Set breakpoints on every line that contains the string "#break"
420        let source_file_name = self.testpaths.file.file_name().unwrap();
421        for line in &dbg_cmds.breakpoint_lines {
422            script_str.push_str(&format!(
423                "breakpoint set --file '{}' --line {}\n",
424                source_file_name, line
425            ));
426        }
427
428        // Append the other commands
429        for line in &dbg_cmds.commands {
430            script_str.push_str(line);
431            script_str.push('\n');
432        }
433
434        // Finally, quit the debugger
435        script_str.push_str("\nquit\n");
436
437        // Write the script into a file
438        debug!("script_str = {}", script_str);
439        self.dump_output_file(&script_str, "debugger.script");
440        let debugger_script = self.make_out_name("debugger.script");
441
442        // Let LLDB execute the script via lldb_batchmode.py
443        let debugger_run_result = self.run_lldb(lldb, &exe_file, &debugger_script);
444
445        if !debugger_run_result.status.success() {
446            self.fatal_proc_rec("Error while running LLDB", &debugger_run_result);
447        }
448
449        if let Err(e) = dbg_cmds.check_output(&debugger_run_result) {
450            self.fatal_proc_rec(&e, &debugger_run_result);
451        }
452    }
453
454    fn run_lldb(
455        &self,
456        lldb: &Utf8Path,
457        test_executable: &Utf8Path,
458        debugger_script: &Utf8Path,
459    ) -> ProcRes {
460        // Path containing `lldb_batchmode.py`, so that the `script` command can import it.
461        let pythonpath = self.config.src_root.join("src/etc");
462
463        let mut cmd = Command::new(lldb);
464        cmd.arg("--one-line")
465            .arg("script --language python -- import lldb_batchmode; lldb_batchmode.main()")
466            .env("LLDB_BATCHMODE_TARGET_PATH", test_executable)
467            .env("LLDB_BATCHMODE_SCRIPT_PATH", debugger_script)
468            .env("PYTHONUNBUFFERED", "1") // Help debugging #78665
469            .env("PYTHONPATH", pythonpath);
470
471        self.run_command_to_procres(&mut cmd)
472    }
473}