[Lldb-commits] [PATCH] fix debugging on linux
dawn at burble.org
dawn at burble.org
Tue Nov 29 01:34:28 PST 2011
Admittedly, I was confused by the host vs. remote process code
that was added recently. I understand a bit more now, and tried to
copy the process launching code from Plugins/Process/Linux to Host/linux,
but got lost in the need to reference the target and listeners needed by
ProcessLinux::ProcessLinux(Target& target, Listener &listener)
so I give up :(. I don't like what we have now, but I've failed to come
up with anything better, so I concede. At least we have a workaround -
thank you!!!
I've resubitted the rest of my original patch (attached) which adds
support for some of the remote vs. host code. LaunchProcess still needs
to be implemented, but it's a start. Please commit?
Thanks again,
-Dawn
On Mon, Nov 28, 2011 at 08:38:04PM -0800, Greg Clayton wrote:
>
> On Nov 28, 2011, at 7:28 PM, dawn at burble.org wrote:
>
> > On Mon, Nov 28, 2011 at 05:58:05PM -0800, Greg Clayton wrote:
> >>
> >> Can you have the linux "Host::LaunchProcess()" do the fork, ptrace me, exec, then do the attach? Or must linux have a debug process attached right from the start?
> >
> > No, both Linux and Windows (and I think BSD as well) will not allow you
> > to attach to a process that you're already debugging.
>
> I am not asking for that.
>
> > They want to have a debug process attached right from the start. They do support
> > attaching to an existing process that's already running, but that's a
> > different problem - here we want to debug the process form the start.
> >
> > So I think you see the issue now?
>
> I see the issue, but your fix in the platform isn't going to help. It works for local debugging only, but not for remote debugging, which is the whole point of going through the platform in the first place. Your current fix will always try and launch the program using the default process plug-in on the current machine.
>
> What is intended for the platform is:
>
> (lldb) platform select remote-macosx
> (lldb) platform connect tcp://some.machine.com:1234
> (lldb) target create /sdk/linux2.3/bin/ls
> (lldb) run
>
> now when you do the launching, the launching should happen through the platform on the remote host, but in your patch:
>
> lldb::ProcessSP
> Platform::DebugProcess (ProcessLaunchInfo &launch_info,
> Debugger &debugger,
> Target *target, // Can be NULL, if NULL create a new target, else use existing one
> Listener &listener,
> Error &error)
> {
> ProcessSP process_sp;
> // Make sure we stop at the entry point
> launch_info.GetFlags ().Set (eLaunchFlagDebug);
>
> if (!target->GetPlatform()->CanLaunchProcessStoppedAtEntry())
> {
> const char *plugin_name = launch_info.GetProcessPluginName();
> process_sp = target->CreateProcess (listener, plugin_name).get();
> error = process_sp->Launch (launch_info);
> }
> else
> ....
>
> this will just launch it locally. A fix should actually be happening in the Platform::LaunchProcess:
>
> Error
> Platform::LaunchProcess (ProcessLaunchInfo &launch_info)
> {
> Error error;
> // Take care of the host case so that each subclass can just
> // call this function to get the host functionality.
> if (IsHost())
> error = Host::LaunchProcess (launch_info);
> else
> error.SetErrorString ("base lldb_private::Platform class can't launch remote processes");
> return error;
> }
>
> But here LaunchProcess isn't returning a ProcessSP of any kind since this function is also used for just spawning processes.
>
> Do you see where I am going with the platform thing here? We can't mix in any specific process class to do the launching.
>
> I know there is work to be done on the platforms still to work out how all of this happens. So for now, I think the code is good
>
> >
> >>> How would you suggest we fix this?
> >>
> >> With my fix that is already checked in (the fix to CommandObjectProcessLaunch that first checks to see if the platform can start a process stopped for debugging and if it can't, it will just use the current process plug-in to do the launch directly and skip the platform launch for debug), the linux process plug-in will be used to do the launching and avoid this issue altogether, right?
> >
> > Yes, but it doesn't fit in with how the process control code was
> > intended to be used - we're skipping the call to DebugProcess just to
> > avoid a call to Attach? And other platforms will have to go this route
> > as well?
>
> No, the explanation above should clarify why.
>
> >
> >>> Move the call to Attach into the MaxOSX
> >>> LaunchProcess code? Or have a flag like CanLaunchViaAttach so we'll know to
> >>> skip the call to Attach? I went with the latter in my patch - that seemed the
> >>> least intrusive yet generic fix.
> >>
> >> Again, I don't understand the can launch via attach. You should be able to launch a program on linux and then attach to it without a debugger needing to be there from the start, you will just miss many instructions as the process makes progress until you attach.
> >
> > Exactly - I think you see the problem now :)
>
> So my vote is to leave the code changes as they exist in the top of tree and let the process plug-in do the launching.
>
> >
> > -Dawn
> >
> >
> >>> On Mon, Nov 28, 2011 at 01:48:35PM -0800, Greg Clayton wrote:
> >>>>
> >>>> On Nov 28, 2011, at 11:59 AM, dawn at burble.org wrote:
> >>>>
> >>>>> I find the name "CanDebugProcess" to be misleading. The issue isn't
> >>>>> whether a process can be debugged on Linux (because it obviously can),
> >>>>> it's that it can't be spawned in a suspended state and then attached
> >>>>> to. AFAIK, Linux doesn't have anything like the MacOSX
> >>>>> "POSIX_SPAWN_START_SUSPENDED" flag for posix_spawnattr_setflags() that
> >>>>> MacOSX has.
> >>>>
> >>>> Yes, this flag is darwin only.
> >>>>
> >>>>> I chose the name "CanLaunchViaAttach" for this, which I think better
> >>>>> describes the problem. Ok to rename "CanDebugProcess" to
> >>>>> "CanLaunchViaAttach"?
> >>>>
> >>>> I am not too fond of the CanLaunchViaAttach because launching a program using a platform doesn't imply that you want to debug or attach to it. Also it doesn't make sense how do you launch via attach? First you launch, then you attach. The attach doesn't have anything to do with launching. You might just want to spawn a server program on the remote platform which you don't want to debug. We are asking the platform (not the process plug-in, but the platform) if it can do something, in this case: can the platform launch a process that can be stopped for debugging. Currently the linux platform can't (though I am sure we can make it work somehow), only the native linux _process_ plug-in can. You can't use the ProcessLinux plug-in to implement stuff functionality for the platform, the platform itself should be doing the work. This is preparing for better remote debugging support where we can launch new remote processes without having to involve any of the process plug-ins.
> >>>>
> >>>> The attach part shouldn't have anything to do with this either because you might want to spawn a new process that is stopped, and then resume it at a controlled time later. The "POSIX_SPAWN_START_SUSPENDED" flag in darwin will have the process halted at the entry point with a SIGSTOP, and it can be resumed later.
> >>>>
> >>>> How about "bool Platform::CanLaunchProcessStoppedAtEntry()"?
> >>>>
> >>>> Platforms are far from complete and we can modify them and grow their abilities over time. Again, the vision is that platforms will eventually launch (possibly for debug), attach to processes, list processes, upload/download and more. They should provide a nice connection to a local or remote machine so that debugging to them is as easy as debugging on your local machine. That is the vision at least, and I know we have some work ahead of us to get there.
> >>>>
> >>>>
> >>>> Greg
> >>>>
> >>>>
> >>>>
> >>>>> Likewise, since you moved the fixes I had made to
> >>>>> DebugProcess() into CommandObjectProcessLaunch(), please also rename
> >>>>> "DebugProcess" to "LaunchViaAttach".
> >>>>
> >>>> Hopefully CanLaunchProcessStoppedAtEntry() makes sense?
> >>>>
> >>>>> Thank you!
> >>>>> -Dawn
> >>>>>
> >>>>>
> >>>>> On Sun, Nov 27, 2011 at 06:03:03PM -0800, Greg Clayton wrote:
> >>>>>>
> >>>>>> On Nov 23, 2011, at 11:31 PM, dawn at burble.org wrote:
> >>>>>>
> >>>>>>> This patch was created using clang/llvm rev 144569, lldb rev 144919 but has
> >>>>>>> been updated to clang/llvm rev 144982, lldb rev 145118. It gets debugging on
> >>>>>>> Linux back to the state it was in in llvm/clang rev 143631, lldb rev 143774.
> >>>>>>>
> >>>>>>> Debugging on Linux broke when the assumption was made that a process could be
> >>>>>>> launched by spawning it in a suspended state and then attaching to it. That
> >>>>>>> support does not exist on Linux. To work around this, I added a new method
> >>>>>>> "CanLaunchViaAttach" which tests whether the platform supports being launched
> >>>>>>> in this way. I wanted to add a flag to ProcessLaunchInfo, but could not find
> >>>>>>> an easy way to set it from within the Linux plugins.
> >>>>>>>
> >>>>>>> Commit message:
> >>>>>>>
> >>>>>>> Fix debugging on Linux.
> >>>>>>> Implement remote vs. host platform code as was done with PlatformDarwin.
> >>>>>>> Platform::CanLaunchViaAttach(): new: return true if the process can be launched
> >>>>>>> via spawn and attach. Default is true. Override in PlatformLinux to false.
> >>>>>>> Platform::DebugProcess(): if (!target->GetPlatform()->CanLaunchViaAttach())
> >>>>>>> then launch the process via CreateProcess and Launch.
> >>>>>>>
> >>>>>>> Please review and commit if acceptable.
> >>>>>>
> >>>>>> I checked in an alternate fix:
> >>>>>>
> >>>>>> % svn commit
> >>>>>> Sending include/lldb/Core/Module.h
> >>>>>> Sending include/lldb/Symbol/SymbolFile.h
> >>>>>> Sending include/lldb/Target/Platform.h
> >>>>>> Sending source/Commands/CommandObjectProcess.cpp
> >>>>>> Sending source/Core/Module.cpp
> >>>>>> Sending source/Plugins/SymbolFile/DWARF/DWARFCompileUnit.cpp
> >>>>>> Sending source/Plugins/SymbolFile/DWARF/DWARFDebugInfoEntry.cpp
> >>>>>> Sending source/Plugins/SymbolFile/DWARF/DWARFDebugInfoEntry.h
> >>>>>> Sending source/Plugins/SymbolFile/DWARF/SymbolFileDWARF.cpp
> >>>>>> Sending source/Plugins/SymbolFile/DWARF/SymbolFileDWARF.h
> >>>>>> Sending source/Symbol/SymbolFile.cpp
> >>>>>> Transmitting file data ...........
> >>>>>> Committed revision 145219.
> >>>>>>
> >>>>>> % svn commit
> >>>>>> Sending Platform/Linux/PlatformLinux.h
> >>>>>> Transmitting file data .
> >>>>>> Committed revision 145221.
> >>>>>>
> >>>>>>
> >>>>>> The first one contains the necessary changes to CommandObjectProcess.cpp and to include/lldb/Target/Platform.h that allow a platform to say:
> >>>>>>
> >>>>>>
> >>>>>> virtual bool
> >>>>>> CanDebugProcess ()
> >>>>>> {
> >>>>>> return true;
> >>>>>> }
> >>>>>>
> >>>>>> This keeps the code out of the platform since when a platform launches a process, it shouldn't be tied to a process plug-in as in the patch you submitted.
> >>>>>>
> >>>>>> Here is the vision of platforms:
> >>>>>>
> >>>>>> - They should be able to list all of the processes on the platform. This way you can connect to your local machine and list the processes, or connect to a remote platform, and if the remote platform supports it, it can list the processes that are there.
> >>>>>> - Platforms should support launching processes. Not always for debugging, but sometimes just to spawn a child process. On the localhost, this means just using the Host::LaunchProcess() functionality. This way, we just need to implement a single copy of the process launching code. When you create a remote platform, you should be able to link against the native linux or darwin launching code and then we don't end up with multiple copies of the launching code like we have now. You make a good point in that not all platforms will be able to support launching with a program stopped for debugging, so we now ask the platform up in the CommandObjectProcessLaunch class before we launch the process. This keeps the platform clean, yet also lets us fall back to just launching the process through the built in process plug-in.
> >>>>>> - Platforms should support attaching to a remote process. Combined with being able to launch for debug, this allows debugging as we have it on MacOSX. If we do this while remotely connected to a remote darwin machine, we now have the ability to talk to a remote computer and launch, attach, list and do a lot more
> >>>>>> - Platforms should know if there are files locally on the current machine in an SDK or PDK or some over developer kit/cross compiling sysroot, etc to allow locating files in terms of the path you would us on the remote machine. For example when debugging on a remote device, if the are asked to find the file for "/bin/ls", this might locally be located in developer kit directory such as
> >>>>>> "/path/to/old/linux/sysroot/bin/ls". This allows you to debug remote machines, or event mount the root file system on a mountpoint so you have access to the files so you can debug.
> >>>>>> - Platforms should be able to upload/download files so you could install a new binary and run it, or upload a shared library so we can locally cache it for faster debugging.
> >>>>>>
> >>>>>> We are in the process of slowly makine these Platforms a reality, so keep the questions coming if anyone has any.
> >>>>>>
> >>>>>> So in short: I made the fix you requested, but in a different way. If you can rework your patch around what is in top of tree, then we can get the rest of it submitted.
> >>>>>>
> >>>>>> Greg Clayton
> >>>>>>
> >>>>>>> Thanks!
> >>>>>>> -Dawn
> >>>>>>> <llvmR144982_lldbR145118.patch>_______________________________________________
> >>>>>>> lldb-commits mailing list
> >>>>>>> lldb-commits at cs.uiuc.edu
> >>>>>>> http://lists.cs.uiuc.edu/mailman/listinfo/lldb-commits
-------------- next part --------------
Index: source/Plugins/Platform/Linux/PlatformLinux.cpp
===================================================================
--- source/Plugins/Platform/Linux/PlatformLinux.cpp (revision 145313)
+++ source/Plugins/Platform/Linux/PlatformLinux.cpp (working copy)
@@ -17,6 +17,7 @@
// Other libraries and framework includes
// Project includes
#include "lldb/Core/Error.h"
+#include "lldb/Core/Debugger.h"
#include "lldb/Core/Module.h"
#include "lldb/Core/ModuleList.h"
#include "lldb/Core/PluginManager.h"
@@ -29,10 +30,12 @@
using namespace lldb;
using namespace lldb_private;
+static uint32_t g_initialize_count = 0;
+
Platform *
PlatformLinux::CreateInstance ()
{
- return new PlatformLinux();
+ return new PlatformLinux(true);
}
const char *
@@ -42,33 +45,51 @@
}
const char *
-PlatformLinux::GetPluginDescriptionStatic()
+PlatformLinux::GetShortPluginNameStatic (bool is_host)
{
- return "Default platform plugin for Linux";
+ if (is_host)
+ return Platform::GetHostPlatformName ();
+ else
+ return "remote-linux";
}
+const char *
+PlatformLinux::GetPluginDescriptionStatic (bool is_host)
+{
+ if (is_host)
+ return "Local Linux user platform plug-in.";
+ else
+ return "Remote Linux user platform plug-in.";
+}
+
void
PlatformLinux::Initialize ()
{
- static bool g_initialized = false;
-
- if (!g_initialized)
+ if (g_initialize_count++ == 0)
{
- PlatformSP default_platform_sp (CreateInstance());
+#if defined(__linux__)
+ PlatformSP default_platform_sp (new PlatformLinux(true));
+ default_platform_sp->SetSystemArchitecture (Host::GetArchitecture());
Platform::SetDefaultPlatform (default_platform_sp);
- PluginManager::RegisterPlugin(GetPluginNameStatic(),
- GetPluginDescriptionStatic(),
- CreateInstance);
- g_initialized = true;
+#endif
+ PluginManager::RegisterPlugin(PlatformLinux::GetShortPluginNameStatic(false),
+ PlatformLinux::GetPluginDescriptionStatic(false),
+ PlatformLinux::CreateInstance);
}
}
void
PlatformLinux::Terminate ()
{
+ if (g_initialize_count > 0)
+ {
+ if (--g_initialize_count == 0)
+ {
+ PluginManager::UnregisterPlugin (PlatformLinux::CreateInstance);
+ }
+ }
}
-
Error
PlatformLinux::ResolveExecutable (const FileSpec &exe_file,
const ArchSpec &exe_arch,
@@ -77,18 +98,51 @@
Error error;
// Nothing special to do here, just use the actual file and architecture
+ char exe_path[PATH_MAX];
FileSpec resolved_exe_file (exe_file);
- // If we have "ls" as the exe_file, resolve the executable loation based on
- // the current path variables
- if (!resolved_exe_file.Exists())
- resolved_exe_file.ResolveExecutableLocation ();
+ if (IsHost())
+ {
+ // If we have "ls" as the exe_file, resolve the executable location based on
+ // the current path variables
+ if (!resolved_exe_file.Exists())
+ {
+ exe_file.GetPath(exe_path, sizeof(exe_path));
+ resolved_exe_file.SetFile(exe_path, true);
+ }
- // Resolve any executable within a bundle on MacOSX
- Host::ResolveExecutableInBundle (resolved_exe_file);
+ if (!resolved_exe_file.Exists())
+ resolved_exe_file.ResolveExecutableLocation ();
- if (resolved_exe_file.Exists())
+ if (resolved_exe_file.Exists())
+ error.Clear();
+ else
+ {
+ exe_file.GetPath(exe_path, sizeof(exe_path));
+ error.SetErrorStringWithFormat("unable to find executable for '%s'", exe_path);
+ }
+ }
+ else
{
+ if (m_remote_platform_sp)
+ {
+ error = m_remote_platform_sp->ResolveExecutable (exe_file,
+ exe_arch,
+ exe_module_sp);
+ }
+ else
+ {
+ // We may connect to a process and use the provided executable (Don't use local $PATH).
+
+ if (resolved_exe_file.Exists())
+ error.Clear();
+ else
+ error.SetErrorStringWithFormat("the platform is not currently connected, and '%s' doesn't exist in the system root.", exe_path);
+ }
+ }
+
+ if (error.Success())
+ {
if (exe_arch.IsValid())
{
error = ModuleList::GetSharedModule (resolved_exe_file,
@@ -152,21 +206,20 @@
}
}
}
- else
- {
- error.SetErrorStringWithFormat ("'%s%s%s' does not exist",
- exe_file.GetDirectory().AsCString(""),
- exe_file.GetDirectory() ? "/" : "",
- exe_file.GetFilename().AsCString(""));
- }
return error;
}
Error
PlatformLinux::GetFile (const FileSpec &platform_file,
- const UUID *uuid, FileSpec &local_file)
+ const UUID *uuid_ptr, FileSpec &local_file)
{
+ if (IsRemote())
+ {
+ if (m_remote_platform_sp)
+ return m_remote_platform_sp->GetFile (platform_file, uuid_ptr, local_file);
+ }
+
// Default to the local case
local_file = platform_file;
return Error();
@@ -176,8 +229,9 @@
//------------------------------------------------------------------
/// Default Constructor
//------------------------------------------------------------------
-PlatformLinux::PlatformLinux () :
- Platform(true)
+PlatformLinux::PlatformLinux (bool is_host) :
+ Platform(is_host), // This is the local host platform
+ m_remote_platform_sp ()
{
}
@@ -194,7 +248,17 @@
bool
PlatformLinux::GetProcessInfo (lldb::pid_t pid, ProcessInstanceInfo &process_info)
{
- return Host::GetProcessInfo (pid, process_info);
+ bool success = false;
+ if (IsHost())
+ {
+ success = Platform::GetProcessInfo (pid, process_info);
+ }
+ else
+ {
+ if (m_remote_platform_sp)
+ success = m_remote_platform_sp->GetProcessInfo (pid, process_info);
+ }
+ return success;
}
bool
@@ -225,11 +289,9 @@
PlatformLinux::GetSoftwareBreakpointTrapOpcode (Target &target,
BreakpointSite *bp_site)
{
- static const uint8_t g_i386_opcode[] = { 0xCC };
-
ArchSpec arch = target.GetArchitecture();
- const uint8_t *opcode = NULL;
- size_t opcode_size = 0;
+ const uint8_t *trap_opcode = NULL;
+ size_t trap_opcode_size = 0;
switch (arch.GetCore())
{
@@ -239,15 +301,41 @@
case ArchSpec::eCore_x86_32_i386:
case ArchSpec::eCore_x86_64_x86_64:
- opcode = g_i386_opcode;
- opcode_size = sizeof(g_i386_opcode);
+ {
+ static const uint8_t g_i386_breakpoint_opcode[] = { 0xCC };
+ trap_opcode = g_i386_breakpoint_opcode;
+ trap_opcode_size = sizeof(g_i386_breakpoint_opcode);
+ }
break;
}
- bp_site->SetTrapOpcode(opcode, opcode_size);
- return opcode_size;
+ if (bp_site->SetTrapOpcode(trap_opcode, trap_opcode_size))
+ return trap_opcode_size;
+ return 0;
}
+Error
+PlatformLinux::LaunchProcess (ProcessLaunchInfo &launch_info)
+{
+ Error error;
+
+ if (IsHost())
+ {
+ if (launch_info.GetFlags().Test (eLaunchFlagLaunchInShell))
+ {
+ const bool is_localhost = true;
+ if (!launch_info.ConvertArgumentsForLaunchingInShell (error, is_localhost))
+ return error;
+ }
+ error = Platform::LaunchProcess (launch_info);
+ }
+ else
+ {
+ error.SetErrorString ("the platform is not currently connected");
+ }
+ return error;
+}
+
lldb::ProcessSP
PlatformLinux::Attach(ProcessAttachInfo &attach_info,
Debugger &debugger,
@@ -255,7 +343,42 @@
Listener &listener,
Error &error)
{
- ProcessSP processSP;
- assert(!"Not implemented yet!");
- return processSP;
+ lldb::ProcessSP process_sp;
+ if (IsHost())
+ {
+ if (target == NULL)
+ {
+ TargetSP new_target_sp;
+ FileSpec emptyFileSpec;
+ ArchSpec emptyArchSpec;
+
+ error = debugger.GetTargetList().CreateTarget (debugger,
+ emptyFileSpec,
+ emptyArchSpec,
+ false,
+ m_remote_platform_sp,
+ new_target_sp);
+ target = new_target_sp.get();
+ }
+ else
+ error.Clear();
+
+ if (target && error.Success())
+ {
+ debugger.GetTargetList().SetSelectedTarget(target);
+
+ process_sp = target->CreateProcess (listener, attach_info.GetProcessPluginName());
+
+ if (process_sp)
+ error = process_sp->Attach (attach_info);
+ }
+ }
+ else
+ {
+ if (m_remote_platform_sp)
+ process_sp = m_remote_platform_sp->Attach (attach_info, debugger, target, listener, error);
+ else
+ error.SetErrorString ("the platform is not currently connected");
+ }
+ return process_sp;
}
Index: source/Plugins/Platform/Linux/PlatformLinux.h
===================================================================
--- source/Plugins/Platform/Linux/PlatformLinux.h (revision 145313)
+++ source/Plugins/Platform/Linux/PlatformLinux.h (working copy)
@@ -28,7 +28,7 @@
static void
Terminate ();
- PlatformLinux ();
+ PlatformLinux (bool is_host);
virtual
~PlatformLinux();
@@ -43,8 +43,11 @@
GetPluginNameStatic();
static const char *
- GetPluginDescriptionStatic();
+ GetShortPluginNameStatic(bool is_host);
+ static const char *
+ GetPluginDescriptionStatic(bool is_host);
+
virtual const char *
GetPluginName()
{
@@ -74,7 +77,7 @@
virtual const char *
GetDescription ()
{
- return GetPluginDescriptionStatic();
+ return GetPluginDescriptionStatic(IsHost());
}
virtual void
@@ -94,6 +97,9 @@
GetSoftwareBreakpointTrapOpcode (Target &target,
BreakpointSite *bp_site);
+ virtual lldb_private::Error
+ LaunchProcess (lldb_private::ProcessLaunchInfo &launch_info);
+
virtual lldb::ProcessSP
Attach(ProcessAttachInfo &attach_info, Debugger &debugger,
Target *target, Listener &listener, Error &error);
@@ -105,8 +111,8 @@
}
protected:
+ lldb::PlatformSP m_remote_platform_sp; // Allow multiple ways to connect to a remote darwin OS
-
private:
DISALLOW_COPY_AND_ASSIGN (PlatformLinux);
};
Index: source/Plugins/Process/Linux/ProcessLinux.cpp
===================================================================
--- source/Plugins/Process/Linux/ProcessLinux.cpp (revision 145313)
+++ source/Plugins/Process/Linux/ProcessLinux.cpp (working copy)
@@ -87,7 +87,6 @@
m_in_limbo(false),
m_exit_now(false)
{
-
#if 0
// FIXME: Putting this code in the ctor and saving the byte order in a
// member variable is a hack to avoid const qual issues in GetByteOrder.
@@ -152,7 +151,6 @@
SetPrivateState(eStateLaunching);
- uint32_t launch_flags = launch_info.GetFlags().Get();
const char *stdin_path = NULL;
const char *stdout_path = NULL;
const char *stderr_path = NULL;
@@ -270,7 +268,13 @@
Error
ProcessLinux::DoDetach()
{
- return Error(1, eErrorTypeGeneric);
+ Error error;
+
+ error = m_monitor->Detach();
+ if (error.Success())
+ SetPrivateState(eStateDetached);
+
+ return error;
}
Error
@@ -388,7 +392,7 @@
ProcessLinux::IsAlive()
{
StateType state = GetPrivateState();
- return state != eStateExited && state != eStateInvalid;
+ return state != eStateDetached && state != eStateExited && state != eStateInvalid;
}
size_t
Index: source/Plugins/Process/Linux/ProcessMonitor.cpp
===================================================================
--- source/Plugins/Process/Linux/ProcessMonitor.cpp (revision 145313)
+++ source/Plugins/Process/Linux/ProcessMonitor.cpp (working copy)
@@ -722,6 +722,30 @@
m_result = true;
}
+//------------------------------------------------------------------------------
+/// @class KillOperation
+/// @brief Implements ProcessMonitor::BringProcessIntoLimbo.
+class DetachOperation : public Operation
+{
+public:
+ DetachOperation(Error &result) : m_error(result) { }
+
+ void Execute(ProcessMonitor *monitor);
+
+private:
+ Error &m_error;
+};
+
+void
+DetachOperation::Execute(ProcessMonitor *monitor)
+{
+ lldb::pid_t pid = monitor->GetPID();
+
+ if (ptrace(PT_DETACH, pid, NULL, 0) < 0)
+ m_error.SetErrorToErrno();
+
+}
+
ProcessMonitor::OperationArgs::OperationArgs(ProcessMonitor *monitor)
: m_monitor(monitor)
{
@@ -1220,7 +1244,7 @@
ProcessMessage
ProcessMonitor::MonitorSIGTRAP(ProcessMonitor *monitor,
- const struct siginfo *info, lldb::pid_t pid)
+ const siginfo_t *info, lldb::pid_t pid)
{
ProcessMessage message;
@@ -1261,7 +1285,7 @@
ProcessMessage
ProcessMonitor::MonitorSignal(ProcessMonitor *monitor,
- const struct siginfo *info, lldb::pid_t pid)
+ const siginfo_t *info, lldb::pid_t pid)
{
ProcessMessage message;
int signo = info->si_signo;
@@ -1312,7 +1336,7 @@
}
ProcessMessage::CrashReason
-ProcessMonitor::GetCrashReasonForSIGSEGV(const struct siginfo *info)
+ProcessMonitor::GetCrashReasonForSIGSEGV(const siginfo_t *info)
{
ProcessMessage::CrashReason reason;
assert(info->si_signo == SIGSEGV);
@@ -1336,7 +1360,7 @@
}
ProcessMessage::CrashReason
-ProcessMonitor::GetCrashReasonForSIGILL(const struct siginfo *info)
+ProcessMonitor::GetCrashReasonForSIGILL(const siginfo_t *info)
{
ProcessMessage::CrashReason reason;
assert(info->si_signo == SIGILL);
@@ -1378,7 +1402,7 @@
}
ProcessMessage::CrashReason
-ProcessMonitor::GetCrashReasonForSIGFPE(const struct siginfo *info)
+ProcessMonitor::GetCrashReasonForSIGFPE(const siginfo_t *info)
{
ProcessMessage::CrashReason reason;
assert(info->si_signo == SIGFPE);
@@ -1420,7 +1444,7 @@
}
ProcessMessage::CrashReason
-ProcessMonitor::GetCrashReasonForSIGBUS(const struct siginfo *info)
+ProcessMonitor::GetCrashReasonForSIGBUS(const siginfo_t *info)
{
ProcessMessage::CrashReason reason;
assert(info->si_signo == SIGBUS);
@@ -1646,7 +1670,9 @@
ProcessMonitor::Detach()
{
bool result;
- KillOperation op(result);
+ lldb_private::Error error;
+ DetachOperation op(error);
+ result = error.Success();
DoOperation(&op);
StopMonitor();
return result;
Index: source/Plugins/Process/Linux/ProcessMonitor.h
===================================================================
--- source/Plugins/Process/Linux/ProcessMonitor.h (revision 145313)
+++ source/Plugins/Process/Linux/ProcessMonitor.h (working copy)
@@ -257,23 +257,23 @@
static ProcessMessage
MonitorSIGTRAP(ProcessMonitor *monitor,
- const struct siginfo *info, lldb::pid_t pid);
+ const siginfo_t *info, lldb::pid_t pid);
static ProcessMessage
MonitorSignal(ProcessMonitor *monitor,
- const struct siginfo *info, lldb::pid_t pid);
+ const siginfo_t *info, lldb::pid_t pid);
static ProcessMessage::CrashReason
- GetCrashReasonForSIGSEGV(const struct siginfo *info);
+ GetCrashReasonForSIGSEGV(const siginfo_t *info);
static ProcessMessage::CrashReason
- GetCrashReasonForSIGILL(const struct siginfo *info);
+ GetCrashReasonForSIGILL(const siginfo_t *info);
static ProcessMessage::CrashReason
- GetCrashReasonForSIGFPE(const struct siginfo *info);
+ GetCrashReasonForSIGFPE(const siginfo_t *info);
static ProcessMessage::CrashReason
- GetCrashReasonForSIGBUS(const struct siginfo *info);
+ GetCrashReasonForSIGBUS(const siginfo_t *info);
void
DoOperation(Operation *op);
More information about the lldb-commits
mailing list