* pinfo.cc (pinfo::init): Clarify comment slightly.
(_onreturn): New helper class. (winpids:add): Remove copied stuff. Try to put process handle into pinfo in question and use _onreturn class to control when to close it. (winpids::release): Remove use of copied array. Free procinfo when hProc is NULL. Otherwise call release and call CloseHandle on hProc. * pinfo.h (winpids::copied): Remove throughout class.
This commit is contained in:
@@ -1,3 +1,13 @@
|
||||
2005-12-21 Christopher Faylor <cgf@timesys.com>
|
||||
|
||||
* pinfo.cc (pinfo::init): Clarify comment slightly.
|
||||
(_onreturn): New helper class.
|
||||
(winpids:add): Remove copied stuff. Try to put process handle into
|
||||
pinfo in question and use _onreturn class to control when to close it.
|
||||
(winpids::release): Remove use of copied array. Free procinfo when
|
||||
hProc is NULL. Otherwise call release and call CloseHandle on hProc.
|
||||
* pinfo.h (winpids::copied): Remove throughout class.
|
||||
|
||||
2005-12-21 Christopher Faylor <cgf@timesys.com>
|
||||
|
||||
* pinfo.cc (pinfo::init): Remove spurious low_priority_sleep.
|
||||
|
@@ -256,13 +256,14 @@ pinfo::init (pid_t n, DWORD flag, HANDLE h0)
|
||||
goto loop;
|
||||
}
|
||||
|
||||
/* In certain rare cases, it is possible for the shared memory region to
|
||||
exist for a while after a process has exited. This should only be a
|
||||
brief occurrence, so rather than introduce some kind of locking
|
||||
mechanism, just loop. */
|
||||
/* In certain pathological cases, it is possible for the shared memory
|
||||
region to exist for a while after a process has exited. This should
|
||||
only be a brief occurrence, so rather than introduce some kind of
|
||||
locking mechanism, just loop. */
|
||||
if (!created && createit && (procinfo->process_state & PID_EXITED))
|
||||
{
|
||||
debug_printf ("looping because pid %d, procinfo->pid %d, procinfo->dwProcessid %u has PID_EXITED set",
|
||||
debug_printf ("looping because pid %d, procinfo->pid %d, "
|
||||
"procinfo->dwProcessid %u has PID_EXITED set",
|
||||
n, procinfo->pid, procinfo->dwProcessId);
|
||||
goto loop;
|
||||
}
|
||||
@@ -1088,26 +1089,67 @@ cygwin_winpid_to_pid (int winpid)
|
||||
#define slop_pidlist 200
|
||||
#define size_pidlist(i) (sizeof (pidlist[0]) * ((i) + 1))
|
||||
#define size_pinfolist(i) (sizeof (pinfolist[0]) * ((i) + 1))
|
||||
#define size_copied(i) (sizeof (copied[0]) * ((i) + 1))
|
||||
class _onreturn
|
||||
{
|
||||
HANDLE& h;
|
||||
HANDLE dummy_handle;
|
||||
public:
|
||||
~_onreturn ()
|
||||
{
|
||||
if (h)
|
||||
{
|
||||
CloseHandle (h);
|
||||
h = NULL;
|
||||
}
|
||||
}
|
||||
void no_close_p_handle () {h = dummy_handle;}
|
||||
_onreturn (): h (dummy_handle), dummy_handle (NULL) {}
|
||||
void set (HANDLE& _h) {h = _h;}
|
||||
};
|
||||
|
||||
inline void
|
||||
winpids::add (DWORD& nelem, bool winpid, DWORD pid)
|
||||
{
|
||||
_onreturn onreturn;
|
||||
bool perform_copy = make_copy;
|
||||
pid_t cygpid = cygwin_pid (pid);
|
||||
|
||||
if (nelem >= npidlist)
|
||||
{
|
||||
npidlist += slop_pidlist;
|
||||
copied = (bool *) realloc (copied, size_copied (npidlist + 1));
|
||||
pidlist = (DWORD *) realloc (pidlist, size_pidlist (npidlist + 1));
|
||||
pinfolist = (pinfo *) realloc (pinfolist, size_pinfolist (npidlist + 1));
|
||||
}
|
||||
|
||||
pinfo& p = pinfolist[nelem];
|
||||
|
||||
/* Open a the process to prevent a subsequent exit from invalidating the
|
||||
shared memory region. */
|
||||
p.hProcess = OpenProcess (PROCESS_QUERY_INFORMATION, false, pid);
|
||||
|
||||
p.init (cygpid, PID_NOREDIR | pinfo_access, NULL);
|
||||
|
||||
/* If we couldn't open the process then we don't have rights to it and should
|
||||
make a copy of the shared memory area if it exists (it may not).
|
||||
Otherwise, if p is "false" then we couldn't open the shared memory region
|
||||
for the given pid, so close the handle to that process since we don't need to
|
||||
protect this pid while the shared memory is open.
|
||||
If p is true and we've opened the handle then things look good but we want
|
||||
to track the handle to eventually close it if things fall apart subsequently.
|
||||
*/
|
||||
if (!p.hProcess)
|
||||
perform_copy = true;
|
||||
else if (!p)
|
||||
CloseHandle (p.hProcess);
|
||||
else
|
||||
onreturn.set (p.hProcess);
|
||||
|
||||
/* If we're just looking for winpids then don't do any special cygwin "stuff* */
|
||||
if (winpid)
|
||||
goto out;
|
||||
|
||||
/* !p means that we couldn't find shared memory for this pid. Probably means
|
||||
that it isn't a cygwin process. */
|
||||
if (!p)
|
||||
{
|
||||
if (!pinfo_access)
|
||||
@@ -1128,13 +1170,43 @@ winpids::add (DWORD& nelem, bool winpid, DWORD pid)
|
||||
}
|
||||
|
||||
out:
|
||||
copied[nelem] = false;
|
||||
if (make_copy && p)
|
||||
/* Exit here.
|
||||
|
||||
If p is "false" then, eventually any opened process handle will be closed and
|
||||
the function will exit without adding anything to the pid list.
|
||||
|
||||
If p is "true" then we've discovered a cygwin process.
|
||||
|
||||
Handle "myself" differently. Don't copy it and close/zero the handle we
|
||||
just opened to it.
|
||||
If not performing a copy, then keep the process handle open for the duration
|
||||
of the life of the procinfo region to potential races when a new process uses
|
||||
this pid.
|
||||
Otherwise, malloc some memory for a copy of the shared memory.
|
||||
|
||||
If the malloc failed, then "oh well". Just keep the shared memory around
|
||||
and eventually close the handle when the winpids goes out of scope.
|
||||
|
||||
If malloc succeeds, copy the procinfo we just grabbed into the new region,
|
||||
release the shared memory and allow the handle to be closed when this
|
||||
function returns.
|
||||
|
||||
Oh, and add the pid to the list and bump the number of elements. */
|
||||
|
||||
if (p)
|
||||
{
|
||||
if (p == (_pinfo *) myself)
|
||||
/* handle specially. Close the handle but (eventually) don't
|
||||
deallocate procinfo in release call */;
|
||||
else if (!perform_copy)
|
||||
onreturn.no_close_p_handle (); /* Don't close the handle until release */
|
||||
else
|
||||
{
|
||||
_pinfo *pnew = (_pinfo *) malloc (sizeof (*p.procinfo));
|
||||
if (pnew)
|
||||
if (!pnew)
|
||||
onreturn.no_close_p_handle ();
|
||||
else
|
||||
{
|
||||
copied[nelem] = true;
|
||||
*pnew = *p.procinfo;
|
||||
if ((_pinfo *) p != (_pinfo *) myself)
|
||||
p.release ();
|
||||
@@ -1144,6 +1216,7 @@ out:
|
||||
}
|
||||
pidlist[nelem++] = pid;
|
||||
}
|
||||
}
|
||||
|
||||
DWORD
|
||||
winpids::enumNT (bool winpid)
|
||||
@@ -1238,10 +1311,15 @@ winpids::release ()
|
||||
{
|
||||
_pinfo *p;
|
||||
for (unsigned i = 0; i < npids; i++)
|
||||
if (pinfolist[i] && (_pinfo *) pinfolist[i] != (_pinfo *) myself)
|
||||
if (!copied[i])
|
||||
if (pinfolist[i] == (_pinfo *) myself)
|
||||
continue;
|
||||
else if (pinfolist[i].hProc)
|
||||
{
|
||||
if (pinfolist[i])
|
||||
pinfolist[i].release ();
|
||||
else if ((p = pinfolist[i].procinfo))
|
||||
CloseHandle (pinfolist[i].hProc);
|
||||
}
|
||||
else if ((p = pinfolist[i]))
|
||||
{
|
||||
pinfolist[i].procinfo = NULL;
|
||||
free (p);
|
||||
@@ -1253,7 +1331,6 @@ winpids::~winpids ()
|
||||
if (npidlist)
|
||||
{
|
||||
release ();
|
||||
free (copied);
|
||||
free (pidlist);
|
||||
free (pinfolist);
|
||||
}
|
||||
|
@@ -205,7 +205,6 @@ class winpids
|
||||
DWORD npidlist;
|
||||
DWORD *pidlist;
|
||||
pinfo *pinfolist;
|
||||
bool *copied;
|
||||
DWORD pinfo_access; // access type for pinfo open
|
||||
DWORD (winpids::* enum_processes) (bool winpid);
|
||||
DWORD enum_init (bool winpid);
|
||||
@@ -218,10 +217,9 @@ public:
|
||||
void set (bool winpid);
|
||||
winpids (): make_copy (true), enum_processes (&winpids::enum_init) {}
|
||||
winpids (int): make_copy (false), npidlist (0), pidlist (NULL), pinfolist (NULL),
|
||||
copied (NULL), pinfo_access (0), enum_processes (&winpids::enum_init),
|
||||
npids (0) {}
|
||||
pinfo_access (0), enum_processes (&winpids::enum_init), npids (0) {}
|
||||
winpids (DWORD acc): make_copy (false), npidlist (0), pidlist (NULL), pinfolist (NULL),
|
||||
copied (NULL), pinfo_access (acc), enum_processes (&winpids::enum_init),
|
||||
pinfo_access (acc), enum_processes (&winpids::enum_init),
|
||||
npids (0)
|
||||
{
|
||||
set (0);
|
||||
|
Reference in New Issue
Block a user