2012-03-02 13:29:36 +01:00
|
|
|
/*
|
|
|
|
* File for parsing top-level /proc entities.
|
|
|
|
* Copyright (C) 1992-1998 by Michael K. Johnson, johnsonm@redhat.com
|
|
|
|
* Copyright 1998-2003 Albert Cahalan
|
2015-07-07 22:42:06 +10:00
|
|
|
* June 2003, Fabian Frederick, slab info
|
2012-03-02 13:29:36 +01:00
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
*/
|
2002-02-01 22:47:29 +00:00
|
|
|
|
2013-05-08 16:34:44 -07:00
|
|
|
#include <stdbool.h>
|
2002-02-01 22:47:29 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <ctype.h>
|
2002-02-01 23:40:38 +00:00
|
|
|
#include <locale.h>
|
2015-06-24 22:16:16 +10:00
|
|
|
#include <errno.h>
|
2002-02-01 22:47:29 +00:00
|
|
|
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <fcntl.h>
|
2011-12-02 17:17:02 -06:00
|
|
|
#include "alloc.h"
|
2002-06-24 04:29:04 +00:00
|
|
|
#include "version.h"
|
|
|
|
#include "sysinfo.h" /* include self to verify prototypes */
|
2015-06-24 22:16:16 +10:00
|
|
|
#include "procps-private.h"
|
2002-02-01 22:47:29 +00:00
|
|
|
|
|
|
|
|
|
|
|
#define BAD_OPEN_MESSAGE \
|
|
|
|
"Error: /proc must be mounted\n" \
|
|
|
|
" To mount /proc at boot you need an /etc/fstab line like:\n" \
|
2006-06-27 03:07:09 +00:00
|
|
|
" proc /proc proc defaults\n" \
|
|
|
|
" In the meantime, run \"mount proc /proc -t proc\"\n"
|
2002-02-01 22:47:29 +00:00
|
|
|
|
|
|
|
#define STAT_FILE "/proc/stat"
|
|
|
|
static int stat_fd = -1;
|
|
|
|
#define LOADAVG_FILE "/proc/loadavg"
|
|
|
|
static int loadavg_fd = -1;
|
|
|
|
#define MEMINFO_FILE "/proc/meminfo"
|
|
|
|
static int meminfo_fd = -1;
|
2002-10-06 16:46:06 +00:00
|
|
|
#define VMINFO_FILE "/proc/vmstat"
|
|
|
|
static int vminfo_fd = -1;
|
2014-07-15 19:17:02 +02:00
|
|
|
#define VM_MIN_FREE_FILE "/proc/sys/vm/min_free_kbytes"
|
|
|
|
static int vm_min_free_fd = -1;
|
2002-02-01 22:47:29 +00:00
|
|
|
|
2008-03-24 04:41:26 +00:00
|
|
|
// As of 2.6.24 /proc/meminfo seems to need 888 on 64-bit,
|
|
|
|
// and would need 1258 if the obsolete fields were there.
|
2014-07-14 19:44:14 +02:00
|
|
|
// As of 3.13 /proc/vmstat needs 2623,
|
|
|
|
// and /proc/stat needs 3076.
|
2014-07-14 19:07:25 +02:00
|
|
|
static char buf[8192];
|
2002-02-01 22:47:29 +00:00
|
|
|
|
|
|
|
/* This macro opens filename only if necessary and seeks to 0 so
|
|
|
|
* that successive calls to the functions are more efficient.
|
|
|
|
* It also reads the current contents of the file into the global buf.
|
|
|
|
*/
|
|
|
|
#define FILE_TO_BUF(filename, fd) do{ \
|
|
|
|
static int local_n; \
|
|
|
|
if (fd == -1 && (fd = open(filename, O_RDONLY)) == -1) { \
|
2005-10-30 23:45:47 +00:00
|
|
|
fputs(BAD_OPEN_MESSAGE, stderr); \
|
2002-02-01 22:47:29 +00:00
|
|
|
fflush(NULL); \
|
|
|
|
_exit(102); \
|
|
|
|
} \
|
|
|
|
lseek(fd, 0L, SEEK_SET); \
|
|
|
|
if ((local_n = read(fd, buf, sizeof buf - 1)) < 0) { \
|
|
|
|
perror(filename); \
|
|
|
|
fflush(NULL); \
|
|
|
|
_exit(103); \
|
|
|
|
} \
|
|
|
|
buf[local_n] = '\0'; \
|
|
|
|
}while(0)
|
|
|
|
|
|
|
|
/* evals 'x' twice */
|
|
|
|
#define SET_IF_DESIRED(x,y) do{ if(x) *(x) = (y); }while(0)
|
|
|
|
|
2014-07-15 19:17:02 +02:00
|
|
|
/* return minimum of two values */
|
|
|
|
#define MIN(x,y) ((x) < (y) ? (x) : (y))
|
2002-02-01 22:47:29 +00:00
|
|
|
|
2015-06-29 11:16:52 -05:00
|
|
|
/*
|
2015-06-24 22:16:16 +10:00
|
|
|
* procps_hertz_get:
|
|
|
|
*
|
|
|
|
*
|
2002-02-01 22:47:29 +00:00
|
|
|
* Some values in /proc are expressed in units of 1/HZ seconds, where HZ
|
|
|
|
* is the kernel clock tick rate. One of these units is called a jiffy.
|
|
|
|
* The HZ value used in the kernel may vary according to hacker desire.
|
|
|
|
*
|
2015-06-24 22:16:16 +10:00
|
|
|
* On some architectures, the kernel provides an ELF note to indicate
|
|
|
|
* HZ.
|
2002-02-01 22:47:29 +00:00
|
|
|
*
|
2015-06-24 22:16:16 +10:00
|
|
|
* Returns:
|
|
|
|
* The discovered or assumed hertz value
|
2002-02-01 22:47:29 +00:00
|
|
|
*/
|
2015-06-24 22:16:16 +10:00
|
|
|
PROCPS_EXPORT long procps_hertz_get(void)
|
|
|
|
{
|
|
|
|
long hz;
|
2011-01-19 12:50:26 +01:00
|
|
|
|
|
|
|
#ifdef _SC_CLK_TCK
|
2015-06-24 22:16:16 +10:00
|
|
|
if ((hz = sysconf(_SC_CLK_TCK)) > 0)
|
2015-07-01 22:08:02 +10:00
|
|
|
return hz;
|
2011-01-19 12:50:26 +01:00
|
|
|
#endif
|
2002-02-01 22:47:29 +00:00
|
|
|
#ifdef HZ
|
2015-06-24 22:16:16 +10:00
|
|
|
return(HZ);
|
2002-02-01 22:47:29 +00:00
|
|
|
#endif
|
2015-06-24 22:16:16 +10:00
|
|
|
/* Last resort, assume 100 */
|
|
|
|
return 100;
|
2002-02-01 22:47:29 +00:00
|
|
|
}
|
2015-06-24 22:16:16 +10:00
|
|
|
|
2015-07-01 21:47:30 +10:00
|
|
|
/*
|
|
|
|
* procps_loadavg:
|
|
|
|
* @av1: location to store 1 minute load average
|
|
|
|
* @av5: location to store 5 minute load average
|
|
|
|
* @av15: location to store 15 minute load average
|
|
|
|
*
|
|
|
|
* Find the 1,5 and 15 minute load average of the system
|
|
|
|
*
|
|
|
|
* Returns: 0 on success <0 on error
|
|
|
|
*/
|
|
|
|
PROCPS_EXPORT int procps_loadavg(
|
|
|
|
double *restrict av1,
|
|
|
|
double *restrict av5,
|
|
|
|
double *restrict av15)
|
2015-06-24 22:16:16 +10:00
|
|
|
{
|
2002-02-01 22:47:29 +00:00
|
|
|
double avg_1=0, avg_5=0, avg_15=0;
|
2009-12-18 05:27:21 +01:00
|
|
|
char *savelocale;
|
2015-06-24 22:16:16 +10:00
|
|
|
int retval=0;
|
2013-03-31 00:00:00 -05:00
|
|
|
|
2002-02-01 22:47:29 +00:00
|
|
|
FILE_TO_BUF(LOADAVG_FILE,loadavg_fd);
|
2009-12-18 05:27:21 +01:00
|
|
|
savelocale = strdup(setlocale(LC_NUMERIC, NULL));
|
2002-02-01 23:40:38 +00:00
|
|
|
setlocale(LC_NUMERIC, "C");
|
2002-02-01 22:47:29 +00:00
|
|
|
if (sscanf(buf, "%lf %lf %lf", &avg_1, &avg_5, &avg_15) < 3) {
|
2015-07-01 21:47:30 +10:00
|
|
|
retval = -ERANGE;
|
2002-02-01 22:47:29 +00:00
|
|
|
}
|
2002-02-01 23:40:38 +00:00
|
|
|
setlocale(LC_NUMERIC, savelocale);
|
2009-12-18 05:27:21 +01:00
|
|
|
free(savelocale);
|
2002-02-01 22:47:29 +00:00
|
|
|
SET_IF_DESIRED(av1, avg_1);
|
|
|
|
SET_IF_DESIRED(av5, avg_5);
|
|
|
|
SET_IF_DESIRED(av15, avg_15);
|
2015-06-24 22:16:16 +10:00
|
|
|
return retval;
|
2002-02-01 22:47:29 +00:00
|
|
|
}
|
|
|
|
|
2003-05-31 00:38:55 +00:00
|
|
|
static char buff[BUFFSIZE]; /* used in the procedures */
|
|
|
|
/***********************************************************************/
|
|
|
|
|
|
|
|
static void crash(const char *filename) {
|
|
|
|
perror(filename);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
2003-06-08 17:28:06 +00:00
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
// based on Fabian Frederick's /proc/slabinfo parser
|
|
|
|
|
|
|
|
unsigned int getslabinfo (struct slab_cache **slab){
|
|
|
|
FILE* fd;
|
|
|
|
int cSlab = 0;
|
2013-03-11 00:00:00 -06:00
|
|
|
buff[BUFFSIZE-1] = 0;
|
2003-06-08 17:28:06 +00:00
|
|
|
*slab = NULL;
|
|
|
|
fd = fopen("/proc/slabinfo", "rb");
|
|
|
|
if(!fd) crash("/proc/slabinfo");
|
|
|
|
while (fgets(buff,BUFFSIZE-1,fd)){
|
|
|
|
if(!memcmp("slabinfo - version:",buff,19)) continue; // skip header
|
|
|
|
if(*buff == '#') continue; // skip comments
|
2011-12-02 17:17:02 -06:00
|
|
|
(*slab) = xrealloc(*slab, (cSlab+1)*sizeof(struct slab_cache));
|
2003-06-08 17:28:06 +00:00
|
|
|
sscanf(buff, "%47s %u %u %u %u", // allow 47; max seen is 24
|
|
|
|
(*slab)[cSlab].name,
|
|
|
|
&(*slab)[cSlab].active_objs,
|
|
|
|
&(*slab)[cSlab].num_objs,
|
|
|
|
&(*slab)[cSlab].objsize,
|
|
|
|
&(*slab)[cSlab].objperslab
|
|
|
|
) ;
|
|
|
|
cSlab++;
|
|
|
|
}
|
|
|
|
fclose(fd);
|
|
|
|
return cSlab;
|
|
|
|
}
|
|
|
|
|
2003-07-03 05:20:19 +00:00
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
unsigned get_pid_digits(void){
|
2003-07-14 12:11:43 +00:00
|
|
|
char pidbuf[24];
|
2003-07-03 05:20:19 +00:00
|
|
|
char *endp;
|
|
|
|
long rc;
|
|
|
|
int fd;
|
|
|
|
static unsigned ret;
|
|
|
|
|
|
|
|
if(ret) goto out;
|
|
|
|
ret = 5;
|
|
|
|
fd = open("/proc/sys/kernel/pid_max", O_RDONLY);
|
|
|
|
if(fd==-1) goto out;
|
2003-07-14 12:11:43 +00:00
|
|
|
rc = read(fd, pidbuf, sizeof pidbuf);
|
2003-07-03 05:20:19 +00:00
|
|
|
close(fd);
|
|
|
|
if(rc<3) goto out;
|
2003-07-14 12:11:43 +00:00
|
|
|
pidbuf[rc] = '\0';
|
|
|
|
rc = strtol(pidbuf,&endp,10);
|
2003-07-03 05:20:19 +00:00
|
|
|
if(rc<42) goto out;
|
|
|
|
if(*endp && *endp!='\n') goto out;
|
|
|
|
rc--; // the pid_max value is really the max PID plus 1
|
|
|
|
ret = 0;
|
|
|
|
while(rc){
|
|
|
|
rc /= 10;
|
|
|
|
ret++;
|
|
|
|
}
|
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
2011-07-14 21:16:02 +10:00
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2015-06-29 22:31:36 +10:00
|
|
|
/* procps_cpu_count:
|
|
|
|
*
|
|
|
|
* Returns the number of CPUs that are currently online.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
long procps_cpu_count(void)
|
|
|
|
{
|
|
|
|
long cpus=1;
|
|
|
|
|
|
|
|
cpus = sysconf(_SC_NPROCESSORS_ONLN);
|
|
|
|
if (cpus < 1)
|
|
|
|
return 1;
|
|
|
|
return cpus;
|
2011-07-14 21:16:02 +10:00
|
|
|
}
|
2015-06-29 22:31:36 +10:00
|
|
|
|