Release 4.14 arch/mips/mti-malta/malta-time.c
/*
* Carsten Langgaard, carstenl@mips.com
* Copyright (C) 1999,2000 MIPS Technologies, Inc. All rights reserved.
*
* This program is free software; you can distribute it and/or modify it
* under the terms of the GNU General Public License (Version 2) as
* published by the Free Software Foundation.
*
* This program is distributed in the hope 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.
*
* Setting up the clock on the MIPS boards.
*/
#include <linux/types.h>
#include <linux/i8253.h>
#include <linux/init.h>
#include <linux/kernel_stat.h>
#include <linux/libfdt.h>
#include <linux/math64.h>
#include <linux/sched.h>
#include <linux/spinlock.h>
#include <linux/interrupt.h>
#include <linux/timex.h>
#include <linux/mc146818rtc.h>
#include <asm/cpu.h>
#include <asm/mipsregs.h>
#include <asm/mipsmtregs.h>
#include <asm/hardirq.h>
#include <asm/irq.h>
#include <asm/div64.h>
#include <asm/setup.h>
#include <asm/time.h>
#include <asm/mc146818-time.h>
#include <asm/msc01_ic.h>
#include <asm/mips-cps.h>
#include <asm/mips-boards/generic.h>
#include <asm/mips-boards/maltaint.h>
static int mips_cpu_timer_irq;
static int mips_cpu_perf_irq;
extern int cp0_perfcount_irq;
static unsigned int gic_frequency;
static void mips_timer_dispatch(void)
{
do_IRQ(mips_cpu_timer_irq);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Ralf Bächle | 13 | 100.00% | 3 | 100.00% |
Total | 13 | 100.00% | 3 | 100.00% |
static void mips_perf_dispatch(void)
{
do_IRQ(mips_cpu_perf_irq);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Chris Dearman | 12 | 92.31% | 1 | 50.00% |
Ralf Bächle | 1 | 7.69% | 1 | 50.00% |
Total | 13 | 100.00% | 2 | 100.00% |
static unsigned int freqround(unsigned int freq, unsigned int amount)
{
freq += amount;
freq -= freq % (amount*2);
return freq;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Steven J. Hill | 32 | 100.00% | 1 | 100.00% |
Total | 32 | 100.00% | 1 | 100.00% |
/*
* Estimate CPU and GIC frequencies.
*/
static void __init estimate_frequencies(void)
{
unsigned long flags;
unsigned int count, start;
unsigned char secs1, secs2, ctrl;
int secs;
u64 giccount = 0, gicstart = 0;
#if defined(CONFIG_KVM_GUEST) && CONFIG_KVM_GUEST_TIMER_FREQ
mips_hpt_frequency = CONFIG_KVM_GUEST_TIMER_FREQ * 1000000;
return;
#endif
local_irq_save(flags);
if (mips_gic_present())
clear_gic_config(GIC_CONFIG_COUNTSTOP);
/*
* Read counters exactly on rising edge of update flag.
* This helps get an accurate reading under virtualisation.
*/
while (CMOS_READ(RTC_REG_A) & RTC_UIP);
while (!(CMOS_READ(RTC_REG_A) & RTC_UIP));
start = read_c0_count();
if (mips_gic_present())
gicstart = read_gic_counter();
/* Wait for falling edge before reading RTC. */
while (CMOS_READ(RTC_REG_A) & RTC_UIP);
secs1 = CMOS_READ(RTC_SECONDS);
/* Read counters again exactly on rising edge of update flag. */
while (!(CMOS_READ(RTC_REG_A) & RTC_UIP));
count = read_c0_count();
if (mips_gic_present())
giccount = read_gic_counter();
/* Wait for falling edge before reading RTC again. */
while (CMOS_READ(RTC_REG_A) & RTC_UIP);
secs2 = CMOS_READ(RTC_SECONDS);
ctrl = CMOS_READ(RTC_CONTROL);
local_irq_restore(flags);
if (!(ctrl & RTC_DM_BINARY) || RTC_ALWAYS_BCD) {
secs1 = bcd2bin(secs1);
secs2 = bcd2bin(secs2);
}
secs = secs2 - secs1;
if (secs < 1)
secs += 60;
count -= start;
count /= secs;
mips_hpt_frequency = count;
if (mips_gic_present()) {
giccount = div_u64(giccount - gicstart, secs);
gic_frequency = giccount;
}
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
James Hogan | 108 | 40.91% | 3 | 21.43% |
Ralf Bächle | 70 | 26.52% | 2 | 14.29% |
Steven J. Hill | 38 | 14.39% | 2 | 14.29% |
Paul Burton | 14 | 5.30% | 2 | 14.29% |
Sanjay Lal | 14 | 5.30% | 1 | 7.14% |
Andrew Morton | 10 | 3.79% | 1 | 7.14% |
Linus Torvalds | 5 | 1.89% | 1 | 7.14% |
Andrew Bresticker | 4 | 1.52% | 1 | 7.14% |
Thomas Gleixner | 1 | 0.38% | 1 | 7.14% |
Total | 264 | 100.00% | 14 | 100.00% |
void read_persistent_clock(struct timespec *ts)
{
ts->tv_sec = mc146818_get_cmos_time();
ts->tv_nsec = 0;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Martin Schwidefsky | 16 | 69.57% | 1 | 20.00% |
Ralf Bächle | 4 | 17.39% | 2 | 40.00% |
Andrew Morton | 2 | 8.70% | 1 | 20.00% |
Linus Torvalds | 1 | 4.35% | 1 | 20.00% |
Total | 23 | 100.00% | 5 | 100.00% |
int get_c0_fdc_int(void)
{
/*
* Some cores claim the FDC is routable through the GIC, but it doesn't
* actually seem to be connected for those Malta bitstreams.
*/
switch (current_cpu_type()) {
case CPU_INTERAPTIV:
case CPU_PROAPTIV:
return -1;
};
if (cpu_has_veic)
return -1;
else if (mips_gic_present())
return gic_get_c0_fdc_int();
else if (cp0_fdc_irq >= 0)
return MIPS_CPU_IRQ_BASE + cp0_fdc_irq;
else
return -1;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
James Hogan | 59 | 96.72% | 2 | 66.67% |
Paul Burton | 2 | 3.28% | 1 | 33.33% |
Total | 61 | 100.00% | 3 | 100.00% |
int get_c0_perfcount_int(void)
{
if (cpu_has_veic) {
set_vi_handler(MSC01E_INT_PERFCTR, mips_perf_dispatch);
mips_cpu_perf_irq = MSC01E_INT_BASE + MSC01E_INT_PERFCTR;
} else if (mips_gic_present()) {
mips_cpu_perf_irq = gic_get_c0_perfcount_int();
} else if (cp0_perfcount_irq >= 0) {
mips_cpu_perf_irq = MIPS_CPU_IRQ_BASE + cp0_perfcount_irq;
} else {
mips_cpu_perf_irq = -1;
}
return mips_cpu_perf_irq;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Andrew Bresticker | 22 | 33.85% | 2 | 22.22% |
Chris Dearman | 21 | 32.31% | 1 | 11.11% |
Ralf Bächle | 19 | 29.23% | 4 | 44.44% |
Paul Burton | 2 | 3.08% | 1 | 11.11% |
Linus Torvalds | 1 | 1.54% | 1 | 11.11% |
Total | 65 | 100.00% | 9 | 100.00% |
EXPORT_SYMBOL_GPL(get_c0_perfcount_int);
unsigned int get_c0_compare_int(void)
{
if (cpu_has_veic) {
set_vi_handler(MSC01E_INT_CPUCTR, mips_timer_dispatch);
mips_cpu_timer_irq = MSC01E_INT_BASE + MSC01E_INT_CPUCTR;
} else if (mips_gic_present()) {
mips_cpu_timer_irq = gic_get_c0_compare_int();
} else {
mips_cpu_timer_irq = MIPS_CPU_IRQ_BASE + cp0_compare_irq;
}
return mips_cpu_timer_irq;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Ralf Bächle | 31 | 59.62% | 4 | 57.14% |
Chris Dearman | 10 | 19.23% | 1 | 14.29% |
Andrew Bresticker | 9 | 17.31% | 1 | 14.29% |
Paul Burton | 2 | 3.85% | 1 | 14.29% |
Total | 52 | 100.00% | 7 | 100.00% |
static void __init init_rtc(void)
{
unsigned char freq, ctrl;
/* Set 32KHz time base if not already set */
freq = CMOS_READ(RTC_FREQ_SELECT);
if ((freq & RTC_DIV_CTL) != RTC_REF_CLCK_32KHZ)
CMOS_WRITE(RTC_REF_CLCK_32KHZ, RTC_FREQ_SELECT);
/* Ensure SET bit is clear so RTC can run */
ctrl = CMOS_READ(RTC_CONTROL);
if (ctrl & RTC_SET)
CMOS_WRITE(ctrl & ~RTC_SET, RTC_CONTROL);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
James Hogan | 39 | 60.94% | 1 | 50.00% |
Paul Burton | 25 | 39.06% | 1 | 50.00% |
Total | 64 | 100.00% | 2 | 100.00% |
#ifdef CONFIG_CLKSRC_MIPS_GIC
static u32 gic_frequency_dt;
static struct property gic_frequency_prop = {
.name = "clock-frequency",
.length = sizeof(u32),
.value = &gic_frequency_dt,
};
static void update_gic_frequency_dt(void)
{
struct device_node *node;
gic_frequency_dt = cpu_to_be32(gic_frequency);
node = of_find_compatible_node(NULL, NULL, "mti,gic-timer");
if (!node) {
pr_err("mti,gic-timer device node not found\n");
return;
}
if (of_update_property(node, &gic_frequency_prop) < 0)
pr_err("error updating gic frequency property\n");
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Matt Redfearn | 61 | 100.00% | 1 | 100.00% |
Total | 61 | 100.00% | 1 | 100.00% |
#endif
void __init plat_time_init(void)
{
unsigned int prid = read_c0_prid() & (PRID_COMP_MASK | PRID_IMP_MASK);
unsigned int freq;
init_rtc();
estimate_frequencies();
freq = mips_hpt_frequency;
if ((prid != (PRID_COMP_MIPS | PRID_IMP_20KC)) &&
(prid != (PRID_COMP_MIPS | PRID_IMP_25KF)))
freq *= 2;
freq = freqround(freq, 5000);
printk("CPU frequency %d.%02d MHz\n", freq/1000000,
(freq%1000000)*100/1000000);
mips_scroll_message();
#ifdef CONFIG_I8253
/* Only Malta has a PIT. */
setup_pit_timer();
#endif
if (mips_gic_present()) {
freq = freqround(gic_frequency, 5000);
printk("GIC frequency %d.%02d MHz\n", freq/1000000,
(freq%1000000)*100/1000000);
#ifdef CONFIG_CLKSRC_MIPS_GIC
update_gic_frequency_dt();
timer_probe();
#endif
}
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Steven J. Hill | 100 | 68.03% | 2 | 18.18% |
Ralf Bächle | 26 | 17.69% | 2 | 18.18% |
Maciej W. Rozycki | 5 | 3.40% | 1 | 9.09% |
Paul Burton | 5 | 3.40% | 2 | 18.18% |
Linus Torvalds | 5 | 3.40% | 1 | 9.09% |
Matt Redfearn | 4 | 2.72% | 1 | 9.09% |
Daniel Lezcano | 1 | 0.68% | 1 | 9.09% |
Andrew Bresticker | 1 | 0.68% | 1 | 9.09% |
Total | 147 | 100.00% | 11 | 100.00% |
Overall Contributors
Person | Tokens | Prop | Commits | CommitProp |
James Hogan | 209 | 22.57% | 6 | 15.00% |
Ralf Bächle | 201 | 21.71% | 11 | 27.50% |
Steven J. Hill | 175 | 18.90% | 2 | 5.00% |
Matt Redfearn | 103 | 11.12% | 1 | 2.50% |
Paul Burton | 53 | 5.72% | 3 | 7.50% |
Chris Dearman | 46 | 4.97% | 1 | 2.50% |
Andrew Bresticker | 41 | 4.43% | 5 | 12.50% |
Linus Torvalds | 33 | 3.56% | 1 | 2.50% |
Martin Schwidefsky | 16 | 1.73% | 1 | 2.50% |
Andrew Morton | 15 | 1.62% | 1 | 2.50% |
Sanjay Lal | 14 | 1.51% | 1 | 2.50% |
Maciej W. Rozycki | 9 | 0.97% | 2 | 5.00% |
Felix Fietkau | 5 | 0.54% | 1 | 2.50% |
David Howells | 2 | 0.22% | 1 | 2.50% |
Stephen Hemminger | 2 | 0.22% | 1 | 2.50% |
Daniel Lezcano | 1 | 0.11% | 1 | 2.50% |
Thomas Gleixner | 1 | 0.11% | 1 | 2.50% |
Total | 926 | 100.00% | 40 | 100.00% |
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.