⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 time.c

📁 linux-2.4.29操作系统的源码
💻 C
📖 第 1 页 / 共 2 页
字号:
void local_timer_interrupt(int irq, void *dev_id, struct pt_regs *regs){	if (!user_mode(regs)) {		if (prof_buffer && current->pid) {			extern int _stext;			unsigned long pc = regs->cp0_epc;			pc -= (unsigned long) &_stext;			pc >>= prof_shift;			/*			 * Dont ignore out-of-bounds pc values silently,			 * put them into the last histogram slot, so if			 * present, they will show up as a sharp peak.			 */			if (pc > prof_len - 1)				pc = prof_len - 1;			atomic_inc((atomic_t *)&prof_buffer[pc]);		}	}#ifdef CONFIG_SMP	/* in UP mode, update_process_times() is invoked by do_timer() */	update_process_times(user_mode(regs));#endif}/* * High-level timer interrupt service routines.  This function * is set as irqaction->handler and is invoked through do_IRQ. */void timer_interrupt(int irq, void *dev_id, struct pt_regs *regs){	unsigned long j;	unsigned int count;	count = mips_hpt_read();	mips_timer_ack();	/* Update timerhi/timerlo for intra-jiffy calibration. */	timerhi += count < timerlo;			/* Wrap around */	timerlo = count;	/*	 * call the generic timer interrupt handling	 */	do_timer(regs);	/*	 * If we have an externally synchronized Linux clock, then update	 * CMOS clock accordingly every ~11 minutes. rtc_set_time() has to be	 * called as close as possible to 500 ms before the new second starts.	 */	read_lock(&xtime_lock);	if ((time_status & STA_UNSYNC) == 0 &&	    xtime.tv_sec > last_rtc_update + 660 &&	    xtime.tv_usec >= 500000 - ((unsigned) tick) / 2 &&	    xtime.tv_usec <= 500000 + ((unsigned) tick) / 2) {		if (rtc_set_mmss(xtime.tv_sec) == 0) {			last_rtc_update = xtime.tv_sec;		} else {			/* do it again in 60 s */			last_rtc_update = xtime.tv_sec - 600;		}	}	read_unlock(&xtime_lock);	/*	 * If jiffies has overflown in this timer_interrupt, we must	 * update the timer[hi]/[lo] to make fast gettimeoffset funcs	 * quotient calc still valid. -arca	 *	 * The first timer interrupt comes late as interrupts are	 * enabled long after timers are initialized.  Therefore the	 * high precision timer is fast, leading to wrong gettimeoffset()	 * calculations.  We deal with it by setting it based on the	 * number of its ticks between the second and the third interrupt.	 * That is still somewhat imprecise, but it's a good estimate.	 * --macro	 */	j = jiffies;	if (j < 4) {		static unsigned int prev_count;		static int hpt_initialized;		switch (j) {		case 0:			timerhi = timerlo = 0;			mips_hpt_init(count);			break;		case 2:			prev_count = count;			break;		case 3:			if (!hpt_initialized) {				unsigned int c3 = 3 * (count - prev_count);				timerhi = 0;				timerlo = c3;				mips_hpt_init(count - c3);				hpt_initialized = 1;			}			break;		default:			break;		}	}#if !defined(CONFIG_SMP)	/*	 * In UP mode, we call local_timer_interrupt() to do profiling	 * and process accouting.	 *	 * In SMP mode, local_timer_interrupt() is invoked by appropriate	 * low-level local timer interrupt handler.	 */	local_timer_interrupt(irq, dev_id, regs);#else	/* CONFIG_SMP */	if (emulate_local_timer_interrupt) {		/*		 * this is the place where we send out inter-process		 * interrupts and let each CPU do its own profiling		 * and process accouting.		 *		 * Obviously we need to call local_timer_interrupt() for		 * the current CPU too.		 */		panic("Not implemented yet!!!");	}#endif	/* CONFIG_SMP */}asmlinkage void ll_timer_interrupt(int irq, struct pt_regs *regs){	int cpu = smp_processor_id();	irq_enter(cpu, irq);	kstat.irqs[cpu][irq]++;	/* we keep interrupt disabled all the time */	timer_interrupt(irq, NULL, regs);	irq_exit(cpu, irq);	if (softirq_pending(cpu))		do_softirq();}asmlinkage void ll_local_timer_interrupt(int irq, struct pt_regs *regs){	int cpu = smp_processor_id();	irq_enter(cpu, irq);	if (cpu != 0)		kstat.irqs[cpu][irq]++;	/* we keep interrupt disabled all the time */	local_timer_interrupt(irq, NULL, regs);	irq_exit(cpu, irq);	if (softirq_pending(cpu))		do_softirq();}/* * time_init() - it does the following things. * * 1) board_time_init() - * 	a) (optional) set up RTC routines, *      b) (optional) calibrate and set the mips_hpt_frequency *	    (only needed if you intended to use fixed_rate_gettimeoffset *	     or use cpu counter as timer interrupt source) * 2) setup xtime based on rtc_get_time(). * 3) choose a appropriate gettimeoffset routine. * 4) calculate a couple of cached variables for later usage * 5) board_timer_setup() - *	a) (optional) over-write any choices made above by time_init(). *	b) machine specific code should setup the timer irqaction. *	c) enable the timer interrupt */void (*board_time_init)(void);void (*board_timer_setup)(struct irqaction *irq);unsigned int mips_hpt_frequency;static struct irqaction timer_irqaction = {	.handler = timer_interrupt,	.flags = SA_INTERRUPT,	.name = "timer",};static unsigned int __init calibrate_hpt(void){	u64 frequency;	u32 hpt_start, hpt_end, hpt_count, hz;	const int loops = HZ / 10;	int log_2_loops = 0;	int i;	/*	 * We want to calibrate for 0.1s, but to avoid a 64-bit	 * division we round the number of loops up to the nearest	 * power of 2.	 */	while (loops > 1 << log_2_loops)		log_2_loops++;	i = 1 << log_2_loops;	/*	 * Wait for a rising edge of the timer interrupt.	 */	while (mips_timer_state());	while (!mips_timer_state());	/*	 * Now see how many high precision timer ticks happen	 * during the calculated number of periods between timer	 * interrupts.	 */	hpt_start = mips_hpt_read();	do {		while (mips_timer_state());		while (!mips_timer_state());	} while (--i);	hpt_end = mips_hpt_read();	hpt_count = hpt_end - hpt_start;	hz = HZ;	frequency = (u64)hpt_count * (u64)hz;	return frequency >> log_2_loops;}void __init time_init(void){	if (board_time_init)		board_time_init();	if (!rtc_set_mmss)		rtc_set_mmss = rtc_set_time;	xtime.tv_sec = rtc_get_time();	xtime.tv_usec = 0;	/* Choose appropriate high precision timer routines.  */	if (!cpu_has_counter && !mips_hpt_read) {		/* No high precision timer -- sorry.  */		mips_hpt_read = null_hpt_read;		mips_hpt_init = null_hpt_init;	} else if (!mips_hpt_frequency && !mips_timer_state) {		/* A high precision timer of unknown frequency.  */		if (!mips_hpt_read) {			/* No external high precision timer -- use R4k.  */			mips_hpt_read = c0_hpt_read;			mips_hpt_init = c0_hpt_init;		}		if ((current_cpu_data.isa_level == MIPS_CPU_ISA_M32) ||			 (current_cpu_data.isa_level == MIPS_CPU_ISA_I) ||			 (current_cpu_data.isa_level == MIPS_CPU_ISA_II))			/*			 * We need to calibrate the counter but we don't have			 * 64-bit division.			 */			do_gettimeoffset = calibrate_div32_gettimeoffset;		else			/*			 * We need to calibrate the counter but we *do* have			 * 64-bit division.			 */			do_gettimeoffset = calibrate_div64_gettimeoffset;	} else {		/* We know counter frequency.  Or we can get it.  */		if (!mips_hpt_read) {			/* No external high precision timer -- use R4k.  */			mips_hpt_read = c0_hpt_read;			if (mips_timer_state)				mips_hpt_init = c0_hpt_init;			else {				/* No external timer interrupt -- use R4k.  */				mips_hpt_init = c0_hpt_timer_init;				mips_timer_ack = c0_timer_ack;			}		}		if (!mips_hpt_frequency)			mips_hpt_frequency = calibrate_hpt();		do_gettimeoffset = fixed_rate_gettimeoffset;		/* Calculate cache parameters.  */		cycles_per_jiffy = (mips_hpt_frequency + HZ / 2) / HZ;		/* sll32_usecs_per_cycle = 10^6 * 2^32 / mips_counter_freq  */		do_div64_32(sll32_usecs_per_cycle,			    1000000, mips_hpt_frequency / 2,			    mips_hpt_frequency);		/* Report the high precision timer rate for a reference.  */		printk("Using %u.%03u MHz high precision timer.\n",		       ((mips_hpt_frequency + 500) / 1000) / 1000,		       ((mips_hpt_frequency + 500) / 1000) % 1000);	}	if (!mips_timer_ack)		/* No timer interrupt ack (e.g. i8254).  */		mips_timer_ack = null_timer_ack;	/* This sets up the high precision timer for the first interrupt.  */	mips_hpt_init(mips_hpt_read());	/*	 * Call board specific timer interrupt setup.	 *	 * this pointer must be setup in machine setup routine.	 *	 * Even if a machine chooses to use a low-level timer interrupt,	 * it still needs to setup the timer_irqaction.	 * In that case, it might be better to set timer_irqaction.handler	 * to be NULL function so that we are sure the high-level code	 * is not invoked accidentally.	 */	board_timer_setup(&timer_irqaction);}#define FEBRUARY		2#define STARTOFTIME		1970#define SECDAY			86400L#define SECYR			(SECDAY * 365)#define leapyear(y)		((!((y) % 4) && ((y) % 100)) || !((y) % 400))#define days_in_year(y)		(leapyear(y) ? 366 : 365)#define days_in_month(m)	(month_days[(m) - 1])static int month_days[12] = {	31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};void to_tm(unsigned long tim, struct rtc_time *tm){	long hms, day, gday;	int i;	gday = day = tim / SECDAY;	hms = tim % SECDAY;	/* Hours, minutes, seconds are easy */	tm->tm_hour = hms / 3600;	tm->tm_min = (hms % 3600) / 60;	tm->tm_sec = (hms % 3600) % 60;	/* Number of years in days */	for (i = STARTOFTIME; day >= days_in_year(i); i++)		day -= days_in_year(i);	tm->tm_year = i;	/* Number of months in days left */	if (leapyear(tm->tm_year))		days_in_month(FEBRUARY) = 29;	for (i = 1; day >= days_in_month(i); i++)		day -= days_in_month(i);	days_in_month(FEBRUARY) = 28;	tm->tm_mon = i - 1;		/* tm_mon starts from 0 to 11 */	/* Days are what is left over (+1) from all that. */	tm->tm_mday = day + 1;	/*	 * Determine the day of week	 */	tm->tm_wday = (gday + 4) % 7;	/* 1970/1/1 was Thursday */}EXPORT_SYMBOL(rtc_lock);EXPORT_SYMBOL(to_tm);EXPORT_SYMBOL(rtc_set_time);EXPORT_SYMBOL(rtc_get_time);

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -