blob: 0ee7dc756ba6701a461b1422ac55c1bb297a182d [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Nobuhiro Iwamatsu5b6918e2008-08-31 22:48:33 +09002/*
3 * Copyright (C) 2007,2008 Nobobuhiro Iwamatsu <iwamatsu@nigauri.org>
4 * Copyright (C) 2008 Renesas Solutions Corp.
5 *
6 * (C) Copyright 2003
7 * Wolfgang Denk, DENX Software Engineering, wd@denx.de.
Nobuhiro Iwamatsu5b6918e2008-08-31 22:48:33 +09008 */
9
Simon Glass97589732020-05-10 11:40:02 -060010#include <init.h>
Simon Glass495a5dc2019-11-14 12:57:30 -070011#include <time.h>
Nobuhiro Iwamatsu5b6918e2008-08-31 22:48:33 +090012#include <asm/io.h>
13#include <asm/processor.h>
Simon Glassdbd79542020-05-10 11:40:11 -060014#include <linux/delay.h>
Nobuhiro Iwamatsu5b6918e2008-08-31 22:48:33 +090015
Jean-Christophe PLAGNIOL-VILLARD9f37e8f2008-12-20 15:27:45 +010016#define CMT_CMCSR_INIT 0x0001 /* PCLK/32 */
Nobuhiro Iwamatsu5b6918e2008-08-31 22:48:33 +090017#define CMT_CMCSR_CALIB 0x0000
18#define CMT_MAX_COUNTER (0xFFFFFFFF)
19#define CMT_TIMER_RESET (0xFFFF)
20
21static vu_long cmt0_timer;
22
23static void cmt_timer_start(unsigned int timer)
24{
25 writew(readw(CMSTR) | 0x01, CMSTR);
26}
27
28static void cmt_timer_stop(unsigned int timer)
29{
30 writew(readw(CMSTR) & ~0x01, CMSTR);
31}
32
33int timer_init(void)
34{
35 cmt0_timer = 0;
36 /* Divide clock by 32 */
37 readw(CMCSR_0);
38 writew(CMT_CMCSR_INIT, CMCSR_0);
39
40 /* User Device 0 only */
41 cmt_timer_stop(0);
Graeme Russ5e669ff2011-07-15 02:18:12 +000042 writew(CMT_TIMER_RESET, CMCOR_0);
Nobuhiro Iwamatsu5b6918e2008-08-31 22:48:33 +090043 cmt_timer_start(0);
44
45 return 0;
46}
47
48unsigned long long get_ticks(void)
49{
50 return cmt0_timer;
51}
52
Nobuhiro Iwamatsu67438082008-12-09 11:32:46 +090053static vu_long cmcnt = 0;
54static unsigned long get_usec (void)
Nobuhiro Iwamatsu5b6918e2008-08-31 22:48:33 +090055{
56 ulong data = readw(CMCNT_0);
57
58 if (data >= cmcnt)
59 cmcnt = data - cmcnt;
60 else
61 cmcnt = (CMT_TIMER_RESET - cmcnt) + data;
62
63 if ((cmt0_timer + cmcnt) > CMT_MAX_COUNTER)
64 cmt0_timer = ((cmt0_timer + cmcnt) - CMT_MAX_COUNTER);
65 else
66 cmt0_timer += cmcnt;
67
68 cmcnt = data;
Nobuhiro Iwamatsu67438082008-12-09 11:32:46 +090069 return cmt0_timer;
70}
71
72/* return msec */
73ulong get_timer(ulong base)
74{
Jean-Christophe PLAGNIOL-VILLARD9f37e8f2008-12-20 15:27:45 +010075 return (get_usec() / 1000) - base;
Nobuhiro Iwamatsu5b6918e2008-08-31 22:48:33 +090076}
77
Ingo van Lilf0f778a2009-11-24 14:09:21 +010078void __udelay(unsigned long usec)
Nobuhiro Iwamatsu5b6918e2008-08-31 22:48:33 +090079{
Nobuhiro Iwamatsu67438082008-12-09 11:32:46 +090080 unsigned long end = get_usec() + usec;
Nobuhiro Iwamatsu5b6918e2008-08-31 22:48:33 +090081
Nobuhiro Iwamatsu67438082008-12-09 11:32:46 +090082 while (get_usec() < end)
Nobuhiro Iwamatsu5b6918e2008-08-31 22:48:33 +090083 continue;
84}
85
86unsigned long get_tbclk(void)
87{
Nobuhiro Iwamatsubefb5cc2014-01-08 14:57:30 +090088 return CONFIG_SH_CMT_CLK_FREQ;
Nobuhiro Iwamatsu5b6918e2008-08-31 22:48:33 +090089}