Skip to content

Commit

Permalink
---
Browse files Browse the repository at this point in the history
yaml
---
r: 146839
b: refs/heads/master
c: c94a857
h: refs/heads/master
i:
  146837: da83f69
  146835: 2ffea54
  146831: 397b70c
v: v3
  • Loading branch information
Magnus Damm authored and Paul Mundt committed May 26, 2009
1 parent e27b74b commit c0c0a14
Show file tree
Hide file tree
Showing 3 changed files with 94 additions and 1 deletion.
2 changes: 1 addition & 1 deletion [refs]
Original file line number Diff line number Diff line change
@@ -1,2 +1,2 @@
---
refs/heads/master: c9904dd15922f349b5f06839e34b1723d4a75940
refs/heads/master: c94a85746f7bdc13035acdf88c130d7b6fa41bde
18 changes: 18 additions & 0 deletions trunk/arch/sh/include/asm/clock.h
Original file line number Diff line number Diff line change
Expand Up @@ -100,4 +100,22 @@ enum clk_sh_algo_id {
IP_N1,
};

struct clk_div_mult_table {
unsigned int *divisors;
unsigned int nr_divisors;
unsigned int *multipliers;
unsigned int nr_multipliers;
};

struct cpufreq_frequency_table;
void clk_rate_table_build(struct clk *clk,
struct cpufreq_frequency_table *freq_table,
int nr_freqs,
struct clk_div_mult_table *src_table,
unsigned long *bitmap);

long clk_rate_table_round(struct clk *clk,
struct cpufreq_frequency_table *freq_table,
unsigned long rate);

#endif /* __ASM_SH_CLOCK_H */
75 changes: 75 additions & 0 deletions trunk/arch/sh/kernel/cpu/clock.c
Original file line number Diff line number Diff line change
Expand Up @@ -29,13 +29,88 @@
#include <linux/err.h>
#include <linux/platform_device.h>
#include <linux/debugfs.h>
#include <linux/cpufreq.h>
#include <asm/clock.h>
#include <asm/machvec.h>

static LIST_HEAD(clock_list);
static DEFINE_SPINLOCK(clock_lock);
static DEFINE_MUTEX(clock_list_sem);

void clk_rate_table_build(struct clk *clk,
struct cpufreq_frequency_table *freq_table,
int nr_freqs,
struct clk_div_mult_table *src_table,
unsigned long *bitmap)
{
unsigned long mult, div;
unsigned long freq;
int i;

for (i = 0; i < nr_freqs; i++) {
div = 1;
mult = 1;

if (src_table->divisors && i < src_table->nr_divisors)
div = src_table->divisors[i];

if (src_table->multipliers && i < src_table->nr_multipliers)
mult = src_table->multipliers[i];

if (!div || !mult || (bitmap && !test_bit(i, bitmap)))
freq = CPUFREQ_ENTRY_INVALID;
else
freq = clk->parent->rate * mult / div;

freq_table[i].index = i;
freq_table[i].frequency = freq;
}

/* Termination entry */
freq_table[i].index = i;
freq_table[i].frequency = CPUFREQ_TABLE_END;
}

long clk_rate_table_round(struct clk *clk,
struct cpufreq_frequency_table *freq_table,
unsigned long rate)
{
unsigned long rate_error, rate_error_prev = ~0UL;
unsigned long rate_best_fit = rate;
unsigned long highest, lowest;
int i;

highest = lowest = 0;

for (i = 0; freq_table[i].frequency != CPUFREQ_TABLE_END; i++) {
unsigned long freq = freq_table[i].frequency;

if (freq == CPUFREQ_ENTRY_INVALID)
continue;

if (freq > highest)
highest = freq;
if (freq < lowest)
lowest = freq;

rate_error = abs(freq - rate);
if (rate_error < rate_error_prev) {
rate_best_fit = freq;
rate_error_prev = rate_error;
}

if (rate_error == 0)
break;
}

if (rate >= highest)
rate_best_fit = highest;
if (rate <= lowest)
rate_best_fit = lowest;

return rate_best_fit;
}

/* Used for clocks that always have same value as the parent clock */
unsigned long followparent_recalc(struct clk *clk)
{
Expand Down

0 comments on commit c0c0a14

Please sign in to comment.