Skip to content

Conversation

@borjamunozf
Copy link

@borjamunozf borjamunozf commented Apr 7, 2025

Hello :)

Coming from this #13463 thread where I explain a bit the aim and context, but to sum it up:

  • Support naive early attach / preattach for scenarios where the target debugee crashes before attaching, the initilization phase already took place or the aim to get full visibility of the init launch of the process.
  • It's not that weird to have this kind of problems and challenges, so a preattach functionality could be really useful when the Launch mode is really complex to setup or the actual code is not feasible to be modified and just put sleep or pauses.

This is a naive approach I came up with, but tested with success in our C++ codebases for the company I'm working on.

  • Polls the /proc (Linux support only) and waits for the pattern match PID detail.
  • Sends SIGSTOP.

The SIGSTOP, as I mentioned in #13463, is something that could be optional or specified as key for the waitFor properties. but at least for decent size C++ code the symbols loading and debugger startup (be LLDB or GDB) could be slow ~15-20secs, so we need to SIGSTOP and handle that in GDB side.

I'm not a Typescript dev at all, probably there's some work and room to improvements, but I think is a decent step to propose here and discuss with you if you're up to include as functionality.

There's pending changes to apply, but I preferred to upload the current status and continue if you're interested.

Thanks!

@borjamunozf borjamunozf requested a review from a team as a code owner April 7, 2025 08:40
@github-project-automation github-project-automation bot moved this to Pull Request in cpptools Apr 7, 2025
@borjamunozf
Copy link
Author

@microsoft-github-policy-service agree

@fearthecowboy
Copy link
Contributor

I like the idea of this - I think we'd need to make sure this was working for all platforms.

I'm a bit nervous about how to make the polling robust as possible (and available across the platforms) - the looping around grabbing the processes is a bit of a wide-casting net (getting all the processes and then parsing the results) - we should probably make it so that we're only asking PS for the processes that are likely matches (but that complicates things with just having an expression obviously)

We'd need another way to suspend the process on Windows (given the lack of SIGSTOP). Not impossible.

I had been thinking a while back if a script launched before debugging might be another way to go to do this - If a task was created to poll in a loop and suspend the matched process, but then we'd still need to have the editor know when to start attaching.

I'll bring it up with some other folks and see what they think.

@sean-mcmanus
Copy link
Contributor

I don't think it necessarily needs to work on Windows, as long as the user can find out that it's not supported. Others might disagree though.

@bobbrow
Copy link
Member

bobbrow commented Apr 7, 2025

RE: avoiding polling
It may be possible to monitor the /proc folder for new processes and check the exe entry of a new processid folder against the program you're waiting for.

I also don't think Windows parity is required. We could support it for cppdbg without supporting cppvsdbg.

@borjamunozf
Copy link
Author

RE: avoiding polling It may be possible to monitor the /proc folder for new processes and check the exe entry of a new processid folder against the program you're waiting for.

I also don't think Windows parity is required. We could support it for cppdbg without supporting cppvsdbg.

Indeed, the /proc polling is the approach I'm more fond of if you think Linux only support is ok, at least for now .
I started with that approach , but after that I switch to what the others _attachers were using (ps analogous) to be as close as possible and think I mixed the comments/idea out.

@bobbrow
Copy link
Member

bobbrow commented Apr 7, 2025

Indeed, the /proc polling is the approach I'm more fond of if you think Linux only support is ok, at least for now . I started with that approach , but after that I switch to what the others _attachers were using (ps analogous) to be as close as possible and think I mixed the comments/idea out.

I think I spoke too soon. cppdbg is meant to work on Windows as well, and we want to have a consistent experience across all platforms, so we probably don't want to maintain 3 different implementations each doing things differently. Polling shouldn't be too bad since it would start shortly before the process is launched. As long as there's a timeout, it should be ok.

@estoykov
Copy link

estoykov commented Apr 19, 2025

Actually, this idea has much greater potential especially when developing mixed code product.
For example when the main code base is JS/TS and runs with node/Electron and has some native modules in C++.
If we have compound launch configuration like this:

{
    "version": "0.2.0",
    "configurations": [
        {
            "name": "Run",
            "type": "node",
            "request": "launch",
            "program": "${workspaceRoot}/app/main.ts",
            "stopOnEntry": false,
            "args": [],
            "cwd": "${workspaceRoot}",
            "runtimeExecutable": "${workspaceRoot}/node_modules/.bin/electron",
            "runtimeArgs": [
                "--electron-main-process",
                "--nolazy",
                "--remote-debugging-port=11227",
                "--remote-allow-origins=*"
            ],
            "console": "integratedTerminal",
            "sourceMaps": true,
            "outFiles": ["${workspaceRoot}/app/**/*.js"]
        },
        {
            "name": "Debug with VS C++ Debugger",
            "type": "cppvsdbg",
            "request": "attach",
            "waitFor": {
                "name": "electron"
                "details": "--electron-main-process"
            }
        }
    ],
    "compounds": [
        {
            "name": "Compound Run",
            "configurations": [
                "Run",
                "Debug with VS C++ Debugger",
            ]
        }
    ]
}

the addon could wait for a process with name "electron" that contains "--electron-main-process" in its details (command line) and then attach to it.
Some other configuration options could be added to determine the target process based on the information (Process class) returned by the NativeAttachItemsProvider instance.

Also could have:

  • initial wait timeout for target process to run
  • retry count
  • retry interval

All code is there and the implementation (new code) will be platform agnostic because the information is supplied by the concrete NativeAttachItemsProvider implementations for the target platform.

@borjamunozf
Copy link
Author

Uploaded some changes, I had some time this weekend to get back into this.

  • We use the same approach relying in the naive poll "native process listing" for Windows and Linux.
  • Added a retry interval optional value as suggested by @estoykov

For Windows support to handle the SIGSTOP could be using the standard Sys internals suite program pssuspend this one.

I think it could be included as offline dependency bundled into the extension, perhaps.

@sean-mcmanus
Copy link
Contributor

@borjamunozf We were busy with other stuff and didn't notice this PR got updated 2 weeks ago.

Copy link
Contributor

@sean-mcmanus sean-mcmanus left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I have not finished reviewing this.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

None yet

Projects

Status: Pull Request

Development

Successfully merging this pull request may close these issues.

6 participants