| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
 100
 101
 102
 103
 104
 105
 106
 107
 108
 109
 
 | 
/*  timer_test
    Victor Yodaiken (c) 1999, Released under the GPL.
    */
#include <linux/module.h>
#include <linux/kernel.h>
#include <asm/ptrace.h>
#include <linux/rtl.h>
#include <asm/io.h>
#include <linux/cons.h>
#include <rtl_sync.h>
#include <rtl_fifo.h>
#include <rtl_time.h>
#include <rtl_core.h>
#include "common.h"
clockid_t clock;
void shutdown(void);
static int shutdown_flag = 0;
static  unsigned long max_diff = 0;
static  unsigned long min_diff = -1;
static  unsigned long last_time = 0;
static  unsigned long count = 0;
#define FIFO_NUM 0
struct timespec spec;
void handler(struct pt_regs *p) {
	unsigned long x,diff;
	struct sample samp;
	switch(shutdown_flag){
		case 0:
			rdtscl(x);
			if(last_time){
				diff = (last_time< x? x- last_time : 0); //ignore overflows
				if(diff > max_diff)max_diff=diff;
				if(diff & (diff < min_diff)) min_diff = diff;
			}
			if(count++ > 1000){
				count = 0;
				samp.min = timespec_from_ns(min_diff);
				samp.max = timespec_from_ns(max_diff);
				rtf_put(FIFO_NUM, &samp, sizeof(samp));
				max_diff = 0;
				min_diff = -1;
			}
			rdtscl(last_time);
			break;
		case 1: 
			shutdown_flag = 2;
		default: /* do nothing */
	}
	return;
}
int fifo_size=4000;
int init_module(void)
{
	int old_irq_state;
	int fifo_status; 
	rtf_destroy(FIFO_NUM); /* just in case someone else is using!*/
	fifo_status = rtf_create(FIFO_NUM, fifo_size);
	if (fifo_status) {
		printk("RTL measurement test fail. fifo_status=%d\n",fifo_status);
		return -1;
	}
        printk("Starting Timer measurement module: numbers in TSC ticks!\n");
	rtl_no_interrupts(old_irq_state);
	clock = rtl_getbestclock(rtl_getcpuid());
	if (!clock || rtl_setclockhandler(clock, handler)) {
		printk("Can't get clock\n");
		rtf_destroy(FIFO_NUM); 
		return -1;
	}
	
	printk("Requested timer and got %x\n",(unsigned int)clock);
	clock->init(clock);
	clock->settimermode(clock, RTL_CLOCK_MODE_PERIODIC);
	clock->settimer(clock, 1000000);
	rtl_restore_interrupts(old_irq_state);
	return 0;
}
void shutdown(){
	int old_irq_state;
	rtl_no_interrupts(old_irq_state);
	clock->uninit(clock);
	rtf_destroy(FIFO_NUM);
	rtl_restore_interrupts(old_irq_state);
}
void cleanup_module(void)
{
	int timeout = 100000;
	shutdown_flag = 1;
	while((shutdown_flag==1) && timeout--);
	/* so it's kinda sloppy, wait for timeout or for 
	   interrupt routine to ack shutdown, whatever comes
	   first */
	shutdown();
}
 |