aboutsummaryrefslogtreecommitdiffstats
path: root/plugin_uptime.c
blob: ff162f70ecb6a00627cce0398413f6a10a056e86 (plain)
1
2
3
4
5
6
7
8
9
10
11
generated by cgit v1.2.3 (git 2.25.1) at 2024-11-19 20:20:43 +0000
 


190' href='#n190'>190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
/* $Id$
 * $URL$
 *
 * plugin for uptime
 *
 * Copyright (C) 2003 Michael Reinelt <michael@reinelt.co.at>
 * Copyright (C) 2004 The LCD4Linux Team <lcd4linux-devel@users.sourceforge.net>
 *
 * This file is part of LCD4Linux.
 *
 * LCD4Linux is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2, or (at your option)
 * any later version.
 *
 * LCD4Linux 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 General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
 *
 */

/* 
 * exported functions:
 *
 * int plugin_init_uptime (void)
 *  adds functions for uptime
 *
 */


#include "config.h"

#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <errno.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/time.h>
#include <fcntl.h>

#include "debug.h"
#include "plugin.h"

static int fd = -2;


static char *itoa(char *buffer, const size_t size, unsigned int value)
{
    char *p;

    /* sanity checks */
    if (buffer == NULL || size < 2)
	return (NULL);

    /* p points to last char */
    p = buffer + size - 1;

    /* set terminating zero */
    *p = '\0';

    do {
	*--p = value % 10 + '0';
	value = value / 10;
    } while (value != 0 && p > buffer);

    return p;
}


char *struptime(const unsigned int uptime, const char *format)
{
    static char string[256];
    const char *src;
    char *dst;
    int len, size;

    src = format;
    dst = string;
    len = 0;

    /* leave room for terminating zero  */
    size = sizeof(string) - 1;

    while (len < size) {

	if (*src == '%') {
	    src++;

	    if (strchr("sSmMhHd", *src) != NULL) {
		char buffer[12], *s;
		unsigned int value = 0;
		int leading_zero = 0;
		switch (*src++) {
		case 's':
		    value = uptime;
		    break;
		case 'S':
		    value = uptime % 60;
		    leading_zero = 1;
		    break;
		case 'm':
		    value = uptime / 60;
		    break;
		case 'M':
		    value = (uptime / 60) % 60;
		    leading_zero = 1;
		    break;
		case 'h':
		    value = uptime / 60 / 60;
		    break;
		case 'H':
		    value = (uptime / 60 / 60) % 24;
		    leading_zero = 1;
		    break;
		case 'd':
		    value = uptime / 60 / 60 / 24;
		    break;
		}

		if (leading_zero && value < 10) {
		    len++;
		    *dst++ = '0';
		}

		s = itoa(buffer, sizeof(buffer), value);
		while (len < size && *s != '\0') {
		    len++;
		    *dst++ = *s++;
		}

	    } else if (*src == '%') {
		len++;
		*dst++ = '%';

	    } else {
		len += 2;
		*dst++ = '%';
		*dst++ = *src++;
	    }

	} else {
	    len++;
	    *dst++ = *src;
	    if (*src++ == '\0')
		break;
	}
    }

    /* enforce terminating zero */
    if (len >= size && *(dst - 1) != '\0') {
	len++;
	*dst = '\0';
    }

    return string;
}


double getuptime(void)
{
    char buffer[36];
    int i;

    if (fd == -2)
	fd = open("/proc/uptime", O_RDONLY);
    if (fd < 0)
	return -1;

    lseek(fd, 0, SEEK_SET);

    i = read(fd, buffer, sizeof(buffer) - 1);
    if (i < 0)
	return -1;

    buffer[i - 1] = '\0';

    /* ignore the 2nd value from /proc/uptime */
    return strtod(buffer, NULL);
}


static void my_uptime(RESULT * result, const int argc, RESULT * argv[])
{
    int age;
    static double uptime = 0.0;
    static struct timeval last_value;
    struct timeval now;

    if (argc > 1) {
	error("uptime(): wrong number of parameters");
	SetResult(&result, R_STRING, "");
	return;
    }

    gettimeofday(&now, NULL);

    age = (now.tv_sec - last_value.tv_sec) * 1000 + (now.tv_usec - last_value.tv_usec) / 1000;
    /* reread every 100 msec only */
    if (fd == -2 || age == 0 || age > 100) {
	uptime = getuptime();
	if (uptime < 0.0) {
	    error("parse(/proc/uptime) failed!");
	    SetResult(&result, R_STRING, "");
	    return;
	}

	last_value = now;
    }

    if (argc == 0) {
	SetResult(&result, R_NUMBER, &uptime);
    } else {
	SetResult(&result, R_STRING, struptime(uptime, R2S(argv[0])));
    }

    return;

}

int plugin_init_uptime(void)
{
    AddFunction("uptime", -1, my_uptime);
    return 0;
}

void plugin_exit_uptime(void)
{
    if (fd > 0)
	close(fd);
    fd = -2;
}