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

📄 time-tc.c

📁 linux 内核源代码
💻 C
字号:
/* * Copyright (C) 2004-2007 Atmel Corporation * * Based on MIPS implementation arch/mips/kernel/time.c *   Copyright 2001 MontaVista Software Inc. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as * published by the Free Software Foundation. */#include <linux/clk.h>#include <linux/clocksource.h>#include <linux/time.h>#include <linux/module.h>#include <linux/interrupt.h>#include <linux/irq.h>#include <linux/kernel_stat.h>#include <linux/errno.h>#include <linux/init.h>#include <linux/profile.h>#include <linux/sysdev.h>#include <linux/err.h>#include <asm/div64.h>#include <asm/sysreg.h>#include <asm/io.h>#include <asm/sections.h>#include <asm/arch/time.h>/* how many counter cycles in a jiffy? */static u32 cycles_per_jiffy;/* the count value for the next timer interrupt */static u32 expirelo;/* the I/O registers of the TC module */static void __iomem *ioregs;cycle_t read_cycle_count(void){	return (cycle_t)timer_read(ioregs, 0, CV);}struct clocksource clocksource_avr32 = {	.name		= "avr32",	.rating		= 342,	.read		= read_cycle_count,	.mask		= CLOCKSOURCE_MASK(16),	.shift		= 16,	.flags		= CLOCK_SOURCE_IS_CONTINUOUS,};static void avr32_timer_ack(void){	u16 count = expirelo;	/* Ack this timer interrupt and set the next one, use a u16	 * variable so it will wrap around correctly */	count += cycles_per_jiffy;	expirelo = count;	timer_write(ioregs, 0, RC, expirelo);	/* Check to see if we have missed any timer interrupts */	count = timer_read(ioregs, 0, CV);	if ((count - expirelo) < 0x7fff) {		expirelo = count + cycles_per_jiffy;		timer_write(ioregs, 0, RC, expirelo);	}}u32 avr32_hpt_read(void){	return timer_read(ioregs, 0, CV);}static int avr32_timer_calc_div_and_set_jiffies(struct clk *pclk){	unsigned int cycles_max = (clocksource_avr32.mask + 1) / 2;	unsigned int divs[] = { 4, 8, 16, 32 };	int divs_size = ARRAY_SIZE(divs);	int i = 0;	unsigned long count_hz;	unsigned long shift;	unsigned long mult;	int clock_div = -1;	u64 tmp;	shift = clocksource_avr32.shift;	do {		count_hz = clk_get_rate(pclk) / divs[i];		mult = clocksource_hz2mult(count_hz, shift);		clocksource_avr32.mult = mult;		tmp = TICK_NSEC;		tmp <<= shift;		tmp += mult / 2;		do_div(tmp, mult);		cycles_per_jiffy = tmp;	} while (cycles_per_jiffy > cycles_max && ++i < divs_size);	clock_div = i + 1;	if (clock_div > divs_size) {		pr_debug("timer: could not calculate clock divider\n");		return -EFAULT;	}	/* Set the clock divider */	timer_write(ioregs, 0, CMR, TIMER_BF(CMR_TCCLKS, clock_div));	return 0;}int avr32_hpt_init(unsigned int count){	struct resource *regs;	struct clk *pclk;	int irq = -1;	int ret = 0;	ret = -ENXIO;	irq = platform_get_irq(&at32_systc0_device, 0);	if (irq < 0) {		pr_debug("timer: could not get irq\n");		goto out_error;	}	pclk = clk_get(&at32_systc0_device.dev, "pclk");	if (IS_ERR(pclk)) {		pr_debug("timer: could not get clk: %ld\n", PTR_ERR(pclk));		goto out_error;	}	clk_enable(pclk);	regs = platform_get_resource(&at32_systc0_device, IORESOURCE_MEM, 0);	if (!regs) {		pr_debug("timer: could not get resource\n");		goto out_error_clk;	}	ioregs = ioremap(regs->start, regs->end - regs->start + 1);	if (!ioregs) {		pr_debug("timer: could not get ioregs\n");		goto out_error_clk;	}	ret = avr32_timer_calc_div_and_set_jiffies(pclk);	if (ret)		goto out_error_io;	ret = setup_irq(irq, &timer_irqaction);	if (ret) {		pr_debug("timer: could not request irq %d: %d\n",				irq, ret);		goto out_error_io;	}	expirelo = (timer_read(ioregs, 0, CV) / cycles_per_jiffy + 1)		* cycles_per_jiffy;	/* Enable clock and interrupts on RC compare */	timer_write(ioregs, 0, CCR, TIMER_BIT(CCR_CLKEN));	timer_write(ioregs, 0, IER, TIMER_BIT(IER_CPCS));	/* Set cycles to first interrupt */	timer_write(ioregs, 0,  RC, expirelo);	printk(KERN_INFO "timer: AT32AP system timer/counter at 0x%p irq %d\n",			ioregs, irq);	return 0;out_error_io:	iounmap(ioregs);out_error_clk:	clk_put(pclk);out_error:	return ret;}int avr32_hpt_start(void){	timer_write(ioregs, 0, CCR, TIMER_BIT(CCR_SWTRG));	return 0;}irqreturn_t timer_interrupt(int irq, void *dev_id){	unsigned int sr = timer_read(ioregs, 0, SR);	if (sr & TIMER_BIT(SR_CPCS)) {		/* ack timer interrupt and try to set next interrupt */		avr32_timer_ack();		/*		 * Call the generic timer interrupt handler		 */		write_seqlock(&xtime_lock);		do_timer(1);		write_sequnlock(&xtime_lock);		/*		 * In UP mode, we call local_timer_interrupt() to do profiling		 * and process accounting.		 *		 * SMP is not supported yet.		 */		local_timer_interrupt(irq, dev_id);		return IRQ_HANDLED;	}	return IRQ_NONE;}

⌨️ 快捷键说明

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