123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144 |
- /*
- * (C) Copyright 2012, ASTRI
- * Ho Ka Leung <kalho@astri.org>
- *
- * See file CREDITS for list of people who contributed to this
- * project.
- *
- * This program is free software; you can redistribute it and/or
- * modify it under the terms of the GNU General Public License as
- * published by the Free Software Foundation; either version 2 of
- * the License, or (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program; if not, write to the Free Software
- * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
- * MA 02111-1307 USA
- */
- #include <common.h>
- #include <asm/io.h>
- #include <asm/arch/clock.h>
- #include <div64.h>
- #include <linux/math64.h>
- #define APBTMR_N_LOAD_COUNT 0x00
- #define APBTMR_N_CURRENT_VALUE 0x04
- #define APBTMR_N_CONTROL 0x08
- #define APBTMR_N_EOI 0x0c
- #define APBTMR_N_INT_STATUS 0x10
- #define APBTMRS_INT_STATUS 0xa0
- #define APBTMRS_EOI 0xa4
- #define APBTMRS_RAW_INT_STATUS 0xa8
- #define APBTMRS_COMP_VERSION 0xac
- #define APBTMR_CONTROL_ENABLE (1 << 0)
- /* 1: periodic, 0:free running. */
- #define APBTMR_CONTROL_MODE_PERIODIC (1 << 1)
- #define APBTMR_CONTROL_INT (1 << 2)
- static void __iomem *timer_reg = (void*)CONFIG_SYS_TIMERBASE;
- static unsigned long long timestamp;
- static unsigned long lastdec;
- #define TIMER_PERIOD 1000000 /* 1MHz counter plus 1 = 1us */
- #define TIMER_LOAD_VAL 0xffffffff
- void reset_timer_masked (void)
- {
- /* reset time */
- lastdec = readl(timer_reg + APBTMR_N_CURRENT_VALUE); /* capure current decrementer value time */
- timestamp = 0; /* start "advancing" time stamp from 0 */
- }
- ulong get_timer_masked (void)
- {
- unsigned long now = readl(timer_reg + APBTMR_N_CURRENT_VALUE); /* current tick value */
- if (lastdec >= now) { /* normal mode (non roll) */
- /* normal mode */
- timestamp += lastdec - now; /* move stamp fordward with absoulte diff ticks */
- } else { /* we have overflow of the count down timer */
- /* nts = ts + ld + (TLV - now)
- * ts=old stamp, ld=time that passed before passing through -1
- * (TLV-now) amount of time after passing though -1
- * nts = new "advancing time stamp"...it could also roll and cause problems.
- */
- timestamp += lastdec + (TIMER_LOAD_VAL - now);
- }
- lastdec = now;
- return div_u64(div_u64(timestamp, DIV_ROUND_UP(ark_get_timer_clock(), TIMER_PERIOD)), 1000);
- }
- int timer_init(void)
- {
- writel(0, timer_reg + APBTMR_N_CONTROL);
- writel(TIMER_LOAD_VAL, timer_reg + APBTMR_N_LOAD_COUNT);
- /* No timer interrupt */
- writel(APBTMR_CONTROL_MODE_PERIODIC | APBTMR_CONTROL_ENABLE, timer_reg + APBTMR_N_CONTROL);
- return 0;
- }
- unsigned long long get_ticks(void)
- {
- return 0;
- }
- ulong get_timer(ulong base)
- {
- return get_timer_masked () - base;
- }
- void __udelay(unsigned long usec)
- {
- long tmo = usec * DIV_ROUND_UP(ark_get_timer_clock(), TIMER_PERIOD);
- unsigned long last = readl(timer_reg + APBTMR_N_CURRENT_VALUE);
- unsigned long now;
- while (tmo > 0) {
- now = readl(timer_reg + APBTMR_N_CURRENT_VALUE);
- if (last >= now)
- tmo -= last - now;
- else
- tmo -= last + (TIMER_LOAD_VAL - now);
- last = now;
- }
- }
- ulong get_tbclk(void)
- {
- return CONFIG_SYS_HZ;
- }
- #ifdef CONFIG_SPL_BUILD
- void timer_init_24M(void)
- {
- /* writel(0, timer_reg + TIMER_CTL0);
- writel(23, timer_reg + TIMER_PRS0);
- writel(TIMER_LOAD_VAL, timer_reg + APBTMR_N_LOAD_COUNT);
- writel(3, timer_reg + TIMER_CTL0); */
- }
- void timer_uninit(void)
- {
- /* writel(0, timer_reg + TIMER_CTL0); */
- }
- void timer_delay_us(unsigned int us)
- {
- __udelay(us);
- }
- #endif
|