blob: f4d9a812ed350668a66c2c2df350449c1fd74f88 [file] [log] [blame]
Eric Andersen2e9c2572003-08-08 22:26:06 +00001/* vi: set sw=4 ts=4: */
2/*
3 * last implementation for busybox
4 *
Eric Andersenc7bda1c2004-03-15 08:29:22 +00005 * Copyright (C) 2003-2004 by Erik Andersen <andersen@codepoet.org>
Eric Andersen2e9c2572003-08-08 22:26:06 +00006 *
Rob Landleye9a7a622006-09-22 02:52:41 +00007 * Licensed under the GPL version 2, see the file LICENSE in this tarball.
Eric Andersen2e9c2572003-08-08 22:26:06 +00008 */
9
Denis Vlasenkob6adbf12007-05-26 19:00:18 +000010#include "libbb.h"
Eric Andersen2e9c2572003-08-08 22:26:06 +000011#include <utmp.h>
Eric Andersen2e9c2572003-08-08 22:26:06 +000012
13#ifndef SHUTDOWN_TIME
14# define SHUTDOWN_TIME 254
15#endif
16
17/* Grr... utmp char[] members do not have to be nul-terminated.
18 * Do what we can while still keeping this reasonably small.
19 * Note: We are assuming the ut_id[] size is fixed at 4. */
20
Bernhard Reutner-Fischer781e42d2006-05-26 14:41:40 +000021#if defined UT_LINESIZE \
22 && ((UT_LINESIZE != 32) || (UT_NAMESIZE != 32) || (UT_HOSTSIZE != 256))
23#error struct utmp member char[] size(s) have changed!
24#elif defined __UT_LINESIZE \
25 && ((__UT_LINESIZE != 32) || (__UT_NAMESIZE != 64) || (__UT_HOSTSIZE != 256))
Eric Andersen2e9c2572003-08-08 22:26:06 +000026#error struct utmp member char[] size(s) have changed!
27#endif
28
Denis Vlasenko9b49a5e2007-10-11 10:05:36 +000029int last_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
Denis Vlasenko68404f12008-03-17 09:00:54 +000030int last_main(int argc, char **argv ATTRIBUTE_UNUSED)
Eric Andersen2e9c2572003-08-08 22:26:06 +000031{
32 struct utmp ut;
33 int n, file = STDIN_FILENO;
"Vladimir N. Oleynik"dd14ca02006-01-31 09:35:45 +000034 time_t t_tmp;
Eric Andersen2e9c2572003-08-08 22:26:06 +000035
36 if (argc > 1) {
37 bb_show_usage();
38 }
Rob Landleyd921b2e2006-08-03 15:41:12 +000039 file = xopen(bb_path_wtmp_file, O_RDONLY);
Eric Andersen2e9c2572003-08-08 22:26:06 +000040
41 printf("%-10s %-14s %-18s %-12.12s %s\n", "USER", "TTY", "HOST", "LOGIN", "TIME");
42 while ((n = safe_read(file, (void*)&ut, sizeof(struct utmp))) != 0) {
43
44 if (n != sizeof(struct utmp)) {
45 bb_perror_msg_and_die("short read");
46 }
47
Denis Vlasenkobf66fbc2006-12-21 13:23:14 +000048 if (ut.ut_line[0] == '~') {
Eric Andersen2e9c2572003-08-08 22:26:06 +000049 if (strncmp(ut.ut_user, "shutdown", 8) == 0)
50 ut.ut_type = SHUTDOWN_TIME;
51 else if (strncmp(ut.ut_user, "reboot", 6) == 0)
52 ut.ut_type = BOOT_TIME;
53 else if (strncmp(ut.ut_user, "runlevel", 7) == 0)
54 ut.ut_type = RUN_LVL;
55 } else {
Eric Andersenc7bda1c2004-03-15 08:29:22 +000056 if (!ut.ut_name[0] || strcmp(ut.ut_name, "LOGIN") == 0 ||
Eric Andersen2e9c2572003-08-08 22:26:06 +000057 ut.ut_name[0] == 0)
58 {
Eric Andersenc7bda1c2004-03-15 08:29:22 +000059 /* Don't bother. This means we can't find how long
Eric Andersen2e9c2572003-08-08 22:26:06 +000060 * someone was logged in for. Oh well. */
61 continue;
62 }
63 if (ut.ut_type != DEAD_PROCESS &&
64 ut.ut_name[0] && ut.ut_line[0])
65 {
66 ut.ut_type = USER_PROCESS;
67 }
68 if (strcmp(ut.ut_name, "date") == 0) {
69 if (ut.ut_line[0] == '|') ut.ut_type = OLD_TIME;
70 if (ut.ut_line[0] == '{') ut.ut_type = NEW_TIME;
71 }
72 }
73
74 if (ut.ut_type!=USER_PROCESS) {
75 switch (ut.ut_type) {
76 case OLD_TIME:
77 case NEW_TIME:
78 case RUN_LVL:
79 case SHUTDOWN_TIME:
80 continue;
81 case BOOT_TIME:
82 strcpy(ut.ut_line, "system boot");
83 break;
84 }
85 }
"Vladimir N. Oleynik"dd14ca02006-01-31 09:35:45 +000086 t_tmp = (time_t)ut.ut_tv.tv_sec;
Eric Andersen2e9c2572003-08-08 22:26:06 +000087 printf("%-10s %-14s %-18s %-12.12s\n", ut.ut_user, ut.ut_line, ut.ut_host,
"Vladimir N. Oleynik"dd14ca02006-01-31 09:35:45 +000088 ctime(&t_tmp) + 4);
Eric Andersen2e9c2572003-08-08 22:26:06 +000089 }
90
Denis Vlasenkof0ed3762006-10-26 23:21:47 +000091 fflush_stdout_and_exit(EXIT_SUCCESS);
Eric Andersen2e9c2572003-08-08 22:26:06 +000092}