|  | #include <linux/init.h> | 
|  | #include <linux/proc_fs.h> | 
|  | #include <linux/sched.h> | 
|  | #include <linux/time.h> | 
|  | #include <asm/cputime.h> | 
|  |  | 
|  | static int proc_calc_metrics(char *page, char **start, off_t off, | 
|  | int count, int *eof, int len) | 
|  | { | 
|  | if (len <= off + count) | 
|  | *eof = 1; | 
|  | *start = page + off; | 
|  | len -= off; | 
|  | if (len > count) | 
|  | len = count; | 
|  | if (len < 0) | 
|  | len = 0; | 
|  | return len; | 
|  | } | 
|  |  | 
|  | static int uptime_read_proc(char *page, char **start, off_t off, int count, | 
|  | int *eof, void *data) | 
|  | { | 
|  | struct timespec uptime; | 
|  | struct timespec idle; | 
|  | int len; | 
|  | cputime_t idletime = cputime_add(init_task.utime, init_task.stime); | 
|  |  | 
|  | do_posix_clock_monotonic_gettime(&uptime); | 
|  | monotonic_to_bootbased(&uptime); | 
|  | cputime_to_timespec(idletime, &idle); | 
|  | len = sprintf(page, "%lu.%02lu %lu.%02lu\n", | 
|  | (unsigned long) uptime.tv_sec, | 
|  | (uptime.tv_nsec / (NSEC_PER_SEC / 100)), | 
|  | (unsigned long) idle.tv_sec, | 
|  | (idle.tv_nsec / (NSEC_PER_SEC / 100))); | 
|  | return proc_calc_metrics(page, start, off, count, eof, len); | 
|  | } | 
|  |  | 
|  | static int __init proc_uptime_init(void) | 
|  | { | 
|  | create_proc_read_entry("uptime", 0, NULL, uptime_read_proc, NULL); | 
|  | return 0; | 
|  | } | 
|  | module_init(proc_uptime_init); |