2000-02-17 20:38:33 +01:00
|
|
|
/* resource.cc: getrusage () and friends.
|
|
|
|
|
2013-01-21 05:38:31 +01:00
|
|
|
Copyright 1996, 1997, 1998, 2000, 2001, 2002, 2003, 2005, 2008, 2009, 2010,
|
2013-05-01 23:19:51 +02:00
|
|
|
2011, 2012, 2013 Red Hat, Inc.
|
2000-02-17 20:38:33 +01:00
|
|
|
|
|
|
|
Written by Steve Chamberlain (sac@cygnus.com), Doug Evans (dje@cygnus.com),
|
|
|
|
Geoffrey Noer (noer@cygnus.com) of Cygnus Support.
|
|
|
|
Rewritten by Sergey S. Okhapkin (sos@prospect.com.ru)
|
|
|
|
|
|
|
|
This file is part of Cygwin.
|
|
|
|
|
|
|
|
This software is a copyrighted work licensed under the terms of the
|
|
|
|
Cygwin license. Please consult the file "CYGWIN_LICENSE" for
|
|
|
|
details. */
|
|
|
|
|
|
|
|
#include "winsup.h"
|
2001-01-18 10:30:37 +01:00
|
|
|
#include <unistd.h>
|
2011-08-07 21:49:08 +02:00
|
|
|
#include <sys/param.h>
|
2000-08-22 07:10:20 +02:00
|
|
|
#include "pinfo.h"
|
2001-10-22 01:44:43 +02:00
|
|
|
#include "psapi.h"
|
2005-07-03 04:40:30 +02:00
|
|
|
#include "cygtls.h"
|
2009-01-17 11:16:42 +01:00
|
|
|
#include "path.h"
|
|
|
|
#include "fhandler.h"
|
|
|
|
#include "pinfo.h"
|
|
|
|
#include "dtable.h"
|
2010-04-20 12:44:52 +02:00
|
|
|
#include "cygheap.h"
|
2011-05-11 11:07:20 +02:00
|
|
|
#include "ntdll.h"
|
2000-02-17 20:38:33 +01:00
|
|
|
|
|
|
|
/* add timeval values */
|
|
|
|
static void
|
|
|
|
add_timeval (struct timeval *tv1, struct timeval *tv2)
|
|
|
|
{
|
|
|
|
tv1->tv_sec += tv2->tv_sec;
|
|
|
|
tv1->tv_usec += tv2->tv_usec;
|
|
|
|
if (tv1->tv_usec >= 1000000)
|
|
|
|
{
|
|
|
|
tv1->tv_usec -= 1000000;
|
|
|
|
tv1->tv_sec++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* add rusage values of r2 to r1 */
|
|
|
|
void __stdcall
|
|
|
|
add_rusage (struct rusage *r1, struct rusage *r2)
|
|
|
|
{
|
|
|
|
add_timeval (&r1->ru_utime, &r2->ru_utime);
|
|
|
|
add_timeval (&r1->ru_stime, &r2->ru_stime);
|
|
|
|
r1->ru_maxrss += r2->ru_maxrss;
|
|
|
|
r1->ru_ixrss += r2->ru_ixrss;
|
|
|
|
r1->ru_idrss += r2->ru_idrss;
|
|
|
|
r1->ru_isrss += r2->ru_isrss;
|
|
|
|
r1->ru_minflt += r2->ru_minflt;
|
|
|
|
r1->ru_majflt += r2->ru_majflt;
|
|
|
|
r1->ru_nswap += r2->ru_nswap;
|
|
|
|
r1->ru_inblock += r2->ru_inblock;
|
|
|
|
r1->ru_oublock += r2->ru_oublock;
|
|
|
|
r1->ru_msgsnd += r2->ru_msgsnd;
|
|
|
|
r1->ru_msgrcv += r2->ru_msgrcv;
|
|
|
|
r1->ru_nsignals += r2->ru_nsignals;
|
|
|
|
r1->ru_nvcsw += r2->ru_nvcsw;
|
|
|
|
r1->ru_nivcsw += r2->ru_nivcsw;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* FIXME: what about other fields? */
|
|
|
|
void __stdcall
|
|
|
|
fill_rusage (struct rusage *r, HANDLE h)
|
|
|
|
{
|
|
|
|
FILETIME creation_time = {0,0};
|
|
|
|
FILETIME exit_time = {0,0};
|
|
|
|
FILETIME kernel_time = {0,0};
|
|
|
|
FILETIME user_time = {0,0};
|
|
|
|
|
|
|
|
struct timeval tv;
|
|
|
|
|
2000-10-15 03:37:07 +02:00
|
|
|
memset (r, 0, sizeof (*r));
|
2000-02-17 20:38:33 +01:00
|
|
|
GetProcessTimes (h, &creation_time, &exit_time, &kernel_time, &user_time);
|
|
|
|
totimeval (&tv, &kernel_time, 0, 0);
|
|
|
|
add_timeval (&r->ru_stime, &tv);
|
|
|
|
totimeval (&tv, &user_time, 0, 0);
|
|
|
|
add_timeval (&r->ru_utime, &tv);
|
2001-10-22 01:44:43 +02:00
|
|
|
|
2011-05-11 11:07:20 +02:00
|
|
|
VM_COUNTERS vmc;
|
|
|
|
NTSTATUS status = NtQueryInformationProcess (h, ProcessVmCounters, &vmc,
|
|
|
|
sizeof vmc, NULL);
|
|
|
|
if (NT_SUCCESS (status))
|
2001-10-22 01:44:43 +02:00
|
|
|
{
|
2011-05-11 11:07:20 +02:00
|
|
|
r->ru_maxrss += (long) (vmc.WorkingSetSize / 1024);
|
|
|
|
r->ru_majflt += vmc.PageFaultCount;
|
2001-10-22 01:44:43 +02:00
|
|
|
}
|
2000-02-17 20:38:33 +01:00
|
|
|
}
|
|
|
|
|
2002-10-07 22:06:16 +02:00
|
|
|
extern "C" int
|
2000-02-17 20:38:33 +01:00
|
|
|
getrusage (int intwho, struct rusage *rusage_in)
|
|
|
|
{
|
|
|
|
int res = 0;
|
|
|
|
struct rusage r;
|
|
|
|
|
|
|
|
if (intwho == RUSAGE_SELF)
|
|
|
|
{
|
|
|
|
memset (&r, 0, sizeof (r));
|
2009-12-18 21:32:04 +01:00
|
|
|
fill_rusage (&r, GetCurrentProcess ());
|
2000-02-17 20:38:33 +01:00
|
|
|
*rusage_in = r;
|
|
|
|
}
|
|
|
|
else if (intwho == RUSAGE_CHILDREN)
|
|
|
|
*rusage_in = myself->rusage_children;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
set_errno (EINVAL);
|
|
|
|
res = -1;
|
|
|
|
}
|
|
|
|
|
2011-12-03 22:43:27 +01:00
|
|
|
syscall_printf ("%R = getrusage(%d, %p)", res, intwho, rusage_in);
|
2000-02-17 20:38:33 +01:00
|
|
|
return res;
|
|
|
|
}
|
2000-12-27 01:13:57 +01:00
|
|
|
|
2001-10-22 02:10:46 +02:00
|
|
|
extern "C" int
|
2000-12-27 01:13:57 +01:00
|
|
|
getrlimit (int resource, struct rlimit *rlp)
|
|
|
|
{
|
2001-08-29 00:48:23 +02:00
|
|
|
MEMORY_BASIC_INFORMATION m;
|
|
|
|
|
2005-07-03 04:40:30 +02:00
|
|
|
myfault efault;
|
|
|
|
if (efault.faulted (EFAULT))
|
2001-06-28 04:19:57 +02:00
|
|
|
return -1;
|
2000-12-27 01:13:57 +01:00
|
|
|
|
|
|
|
rlp->rlim_cur = RLIM_INFINITY;
|
|
|
|
rlp->rlim_max = RLIM_INFINITY;
|
|
|
|
|
|
|
|
switch (resource)
|
|
|
|
{
|
|
|
|
case RLIMIT_CPU:
|
|
|
|
case RLIMIT_FSIZE:
|
|
|
|
case RLIMIT_DATA:
|
2011-08-07 21:49:08 +02:00
|
|
|
case RLIMIT_AS:
|
2001-08-29 00:48:23 +02:00
|
|
|
break;
|
2000-12-27 01:13:57 +01:00
|
|
|
case RLIMIT_STACK:
|
2001-08-29 00:48:23 +02:00
|
|
|
if (!VirtualQuery ((LPCVOID) &m, &m, sizeof m))
|
|
|
|
debug_printf ("couldn't get stack info, returning def.values. %E");
|
|
|
|
else
|
2001-09-07 23:32:07 +02:00
|
|
|
{
|
2013-04-23 11:44:36 +02:00
|
|
|
rlp->rlim_cur = (rlim_t) &m - (rlim_t) m.AllocationBase;
|
|
|
|
rlp->rlim_max = (rlim_t) m.BaseAddress + m.RegionSize
|
|
|
|
- (rlim_t) m.AllocationBase;
|
2001-08-29 00:48:23 +02:00
|
|
|
}
|
2001-08-16 16:20:09 +02:00
|
|
|
break;
|
2000-12-27 01:13:57 +01:00
|
|
|
case RLIMIT_NOFILE:
|
2001-08-16 16:29:21 +02:00
|
|
|
rlp->rlim_cur = getdtablesize ();
|
2003-07-09 03:33:06 +02:00
|
|
|
if (rlp->rlim_cur < OPEN_MAX)
|
|
|
|
rlp->rlim_cur = OPEN_MAX;
|
2009-09-22 06:09:03 +02:00
|
|
|
rlp->rlim_max = OPEN_MAX_MAX;
|
2000-12-27 01:13:57 +01:00
|
|
|
break;
|
|
|
|
case RLIMIT_CORE:
|
2010-04-20 12:44:52 +02:00
|
|
|
rlp->rlim_cur = cygheap->rlim_core;
|
2000-12-27 01:13:57 +01:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
set_errno (EINVAL);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2001-10-22 02:10:46 +02:00
|
|
|
extern "C" int
|
2000-12-27 01:13:57 +01:00
|
|
|
setrlimit (int resource, const struct rlimit *rlp)
|
|
|
|
{
|
2005-07-03 04:40:30 +02:00
|
|
|
myfault efault;
|
|
|
|
if (efault.faulted (EFAULT))
|
2001-06-28 04:19:57 +02:00
|
|
|
return -1;
|
2001-01-05 10:01:18 +01:00
|
|
|
|
|
|
|
struct rlimit oldlimits;
|
|
|
|
|
2013-05-01 23:19:51 +02:00
|
|
|
/* Check if the request is to actually change the resource settings.
|
|
|
|
If it does not result in a change, take no action and do not fail. */
|
2002-09-22 05:38:57 +02:00
|
|
|
if (getrlimit (resource, &oldlimits) < 0)
|
2001-01-05 10:01:18 +01:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (oldlimits.rlim_cur == rlp->rlim_cur &&
|
|
|
|
oldlimits.rlim_max == rlp->rlim_max)
|
2013-05-01 23:19:51 +02:00
|
|
|
/* No change in resource requirements, succeed immediately */
|
2001-01-05 10:01:18 +01:00
|
|
|
return 0;
|
2000-12-27 01:13:57 +01:00
|
|
|
|
2013-05-01 23:19:51 +02:00
|
|
|
if (rlp->rlim_cur > rlp->rlim_max)
|
|
|
|
{
|
|
|
|
set_errno (EINVAL);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2000-12-27 01:13:57 +01:00
|
|
|
switch (resource)
|
|
|
|
{
|
|
|
|
case RLIMIT_CORE:
|
2010-04-20 12:44:52 +02:00
|
|
|
cygheap->rlim_core = rlp->rlim_cur;
|
2000-12-27 01:13:57 +01:00
|
|
|
break;
|
2001-01-18 10:30:37 +01:00
|
|
|
case RLIMIT_NOFILE:
|
|
|
|
if (rlp->rlim_cur != RLIM_INFINITY)
|
2001-09-07 23:32:07 +02:00
|
|
|
return setdtablesize (rlp->rlim_cur);
|
2001-01-18 10:30:37 +01:00
|
|
|
break;
|
2000-12-27 01:13:57 +01:00
|
|
|
default:
|
|
|
|
set_errno (EINVAL);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|