Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
openeuler
Kernel
提交
d1000edd
K
Kernel
项目概览
openeuler
/
Kernel
1 年多 前同步成功
通知
8
Star
0
Fork
0
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
0
列表
看板
标记
里程碑
合并请求
0
DevOps
流水线
流水线任务
计划
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
K
Kernel
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
0
Issue
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
DevOps
DevOps
流水线
流水线任务
计划
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
流水线任务
提交
Issue看板
提交
d1000edd
编写于
3月 28, 2012
作者:
P
Paul Mundt
浏览文件
操作
浏览文件
下载
差异文件
Merge branch 'sh/cpufreq' into sh-latest
上级
344dd2f4
3cbb08ae
变更
1
隐藏空白更改
内联
并排
Showing
1 changed file
with
82 addition
and
39 deletion
+82
-39
arch/sh/kernel/cpufreq.c
arch/sh/kernel/cpufreq.c
+82
-39
未找到文件。
arch/sh/kernel/cpufreq.c
浏览文件 @
d1000edd
...
@@ -3,7 +3,7 @@
...
@@ -3,7 +3,7 @@
*
*
* cpufreq driver for the SuperH processors.
* cpufreq driver for the SuperH processors.
*
*
* Copyright (C) 2002 - 20
07
Paul Mundt
* Copyright (C) 2002 - 20
12
Paul Mundt
* Copyright (C) 2002 M. R. Brown
* Copyright (C) 2002 M. R. Brown
*
*
* Clock framework bits from arch/avr32/mach-at32ap/cpufreq.c
* Clock framework bits from arch/avr32/mach-at32ap/cpufreq.c
...
@@ -14,6 +14,8 @@
...
@@ -14,6 +14,8 @@
* License. See the file "COPYING" in the main directory of this archive
* License. See the file "COPYING" in the main directory of this archive
* for more details.
* for more details.
*/
*/
#define pr_fmt(fmt) "cpufreq: " fmt
#include <linux/types.h>
#include <linux/types.h>
#include <linux/cpufreq.h>
#include <linux/cpufreq.h>
#include <linux/kernel.h>
#include <linux/kernel.h>
...
@@ -21,15 +23,18 @@
...
@@ -21,15 +23,18 @@
#include <linux/init.h>
#include <linux/init.h>
#include <linux/err.h>
#include <linux/err.h>
#include <linux/cpumask.h>
#include <linux/cpumask.h>
#include <linux/cpu.h>
#include <linux/smp.h>
#include <linux/smp.h>
#include <linux/sched.h>
/* set_cpus_allowed() */
#include <linux/sched.h>
/* set_cpus_allowed() */
#include <linux/clk.h>
#include <linux/clk.h>
#include <linux/percpu.h>
#include <linux/sh_clk.h>
static
struct
clk
*
cpuclk
;
static
DEFINE_PER_CPU
(
struct
clk
,
sh_cpuclk
)
;
static
unsigned
int
sh_cpufreq_get
(
unsigned
int
cpu
)
static
unsigned
int
sh_cpufreq_get
(
unsigned
int
cpu
)
{
{
return
(
clk_get_rate
(
cpuclk
)
+
500
)
/
1000
;
return
(
clk_get_rate
(
&
per_cpu
(
sh_cpuclk
,
cpu
)
)
+
500
)
/
1000
;
}
}
/*
/*
...
@@ -40,8 +45,10 @@ static int sh_cpufreq_target(struct cpufreq_policy *policy,
...
@@ -40,8 +45,10 @@ static int sh_cpufreq_target(struct cpufreq_policy *policy,
unsigned
int
relation
)
unsigned
int
relation
)
{
{
unsigned
int
cpu
=
policy
->
cpu
;
unsigned
int
cpu
=
policy
->
cpu
;
struct
clk
*
cpuclk
=
&
per_cpu
(
sh_cpuclk
,
cpu
);
cpumask_t
cpus_allowed
;
cpumask_t
cpus_allowed
;
struct
cpufreq_freqs
freqs
;
struct
cpufreq_freqs
freqs
;
struct
device
*
dev
;
long
freq
;
long
freq
;
if
(
!
cpu_online
(
cpu
))
if
(
!
cpu_online
(
cpu
))
...
@@ -52,13 +59,15 @@ static int sh_cpufreq_target(struct cpufreq_policy *policy,
...
@@ -52,13 +59,15 @@ static int sh_cpufreq_target(struct cpufreq_policy *policy,
BUG_ON
(
smp_processor_id
()
!=
cpu
);
BUG_ON
(
smp_processor_id
()
!=
cpu
);
dev
=
get_cpu_device
(
cpu
);
/* Convert target_freq from kHz to Hz */
/* Convert target_freq from kHz to Hz */
freq
=
clk_round_rate
(
cpuclk
,
target_freq
*
1000
);
freq
=
clk_round_rate
(
cpuclk
,
target_freq
*
1000
);
if
(
freq
<
(
policy
->
min
*
1000
)
||
freq
>
(
policy
->
max
*
1000
))
if
(
freq
<
(
policy
->
min
*
1000
)
||
freq
>
(
policy
->
max
*
1000
))
return
-
EINVAL
;
return
-
EINVAL
;
pr_debug
(
"cpufreq:
requested frequency %u Hz
\n
"
,
target_freq
*
1000
);
dev_dbg
(
dev
,
"
requested frequency %u Hz
\n
"
,
target_freq
*
1000
);
freqs
.
cpu
=
cpu
;
freqs
.
cpu
=
cpu
;
freqs
.
old
=
sh_cpufreq_get
(
cpu
);
freqs
.
old
=
sh_cpufreq_get
(
cpu
);
...
@@ -70,78 +79,112 @@ static int sh_cpufreq_target(struct cpufreq_policy *policy,
...
@@ -70,78 +79,112 @@ static int sh_cpufreq_target(struct cpufreq_policy *policy,
clk_set_rate
(
cpuclk
,
freq
);
clk_set_rate
(
cpuclk
,
freq
);
cpufreq_notify_transition
(
&
freqs
,
CPUFREQ_POSTCHANGE
);
cpufreq_notify_transition
(
&
freqs
,
CPUFREQ_POSTCHANGE
);
pr_debug
(
"cpufreq: set frequency %lu Hz
\n
"
,
freq
);
dev_dbg
(
dev
,
"set frequency %lu Hz
\n
"
,
freq
);
return
0
;
}
static
int
sh_cpufreq_verify
(
struct
cpufreq_policy
*
policy
)
{
struct
clk
*
cpuclk
=
&
per_cpu
(
sh_cpuclk
,
policy
->
cpu
);
struct
cpufreq_frequency_table
*
freq_table
;
freq_table
=
cpuclk
->
nr_freqs
?
cpuclk
->
freq_table
:
NULL
;
if
(
freq_table
)
return
cpufreq_frequency_table_verify
(
policy
,
freq_table
);
cpufreq_verify_within_limits
(
policy
,
policy
->
cpuinfo
.
min_freq
,
policy
->
cpuinfo
.
max_freq
);
policy
->
min
=
(
clk_round_rate
(
cpuclk
,
1
)
+
500
)
/
1000
;
policy
->
max
=
(
clk_round_rate
(
cpuclk
,
~
0UL
)
+
500
)
/
1000
;
cpufreq_verify_within_limits
(
policy
,
policy
->
cpuinfo
.
min_freq
,
policy
->
cpuinfo
.
max_freq
);
return
0
;
return
0
;
}
}
static
int
sh_cpufreq_cpu_init
(
struct
cpufreq_policy
*
policy
)
static
int
sh_cpufreq_cpu_init
(
struct
cpufreq_policy
*
policy
)
{
{
if
(
!
cpu_online
(
policy
->
cpu
))
unsigned
int
cpu
=
policy
->
cpu
;
struct
clk
*
cpuclk
=
&
per_cpu
(
sh_cpuclk
,
cpu
);
struct
cpufreq_frequency_table
*
freq_table
;
struct
device
*
dev
;
if
(
!
cpu_online
(
cpu
))
return
-
ENODEV
;
return
-
ENODEV
;
cpuclk
=
clk_get
(
NULL
,
"cpu_clk"
);
dev
=
get_cpu_device
(
cpu
);
cpuclk
=
clk_get
(
dev
,
"cpu_clk"
);
if
(
IS_ERR
(
cpuclk
))
{
if
(
IS_ERR
(
cpuclk
))
{
printk
(
KERN_ERR
"cpufreq: couldn't get CPU#%d clk
\n
"
,
dev_err
(
dev
,
"couldn't get CPU clk
\n
"
);
policy
->
cpu
);
return
PTR_ERR
(
cpuclk
);
return
PTR_ERR
(
cpuclk
);
}
}
/* cpuinfo and default policy values */
policy
->
cur
=
policy
->
min
=
policy
->
max
=
sh_cpufreq_get
(
cpu
);
policy
->
cpuinfo
.
min_freq
=
(
clk_round_rate
(
cpuclk
,
1
)
+
500
)
/
1000
;
policy
->
cpuinfo
.
max_freq
=
(
clk_round_rate
(
cpuclk
,
~
0UL
)
+
500
)
/
1000
;
policy
->
cpuinfo
.
transition_latency
=
CPUFREQ_ETERNAL
;
policy
->
cur
=
sh_cpufreq_get
(
policy
->
cpu
)
;
freq_table
=
cpuclk
->
nr_freqs
?
cpuclk
->
freq_table
:
NULL
;
policy
->
min
=
policy
->
cpuinfo
.
min_freq
;
if
(
freq_table
)
{
policy
->
max
=
policy
->
cpuinfo
.
max_freq
;
int
result
;
/*
result
=
cpufreq_frequency_table_cpuinfo
(
policy
,
freq_table
);
* Catch the cases where the clock framework hasn't been wired up
if
(
!
result
)
* properly to support scaling.
cpufreq_frequency_table_get_attr
(
freq_table
,
cpu
);
*/
}
else
{
if
(
unlikely
(
policy
->
min
==
policy
->
max
))
{
dev_notice
(
dev
,
"no frequency table found, falling back "
printk
(
KERN_ERR
"cpufreq: clock framework rate rounding "
"to rate rounding.
\n
"
);
"not supported on CPU#%d.
\n
"
,
policy
->
cpu
);
clk_put
(
cpuclk
);
policy
->
cpuinfo
.
min_freq
=
return
-
EINVAL
;
(
clk_round_rate
(
cpuclk
,
1
)
+
500
)
/
1000
;
policy
->
cpuinfo
.
max_freq
=
(
clk_round_rate
(
cpuclk
,
~
0UL
)
+
500
)
/
1000
;
}
}
printk
(
KERN_INFO
"cpufreq: CPU#%d Frequencies - Minimum %u.%03u MHz, "
policy
->
min
=
policy
->
cpuinfo
.
min_freq
;
policy
->
max
=
policy
->
cpuinfo
.
max_freq
;
policy
->
cpuinfo
.
transition_latency
=
CPUFREQ_ETERNAL
;
dev_info
(
dev
,
"CPU Frequencies - Minimum %u.%03u MHz, "
"Maximum %u.%03u MHz.
\n
"
,
"Maximum %u.%03u MHz.
\n
"
,
policy
->
cpu
,
policy
->
min
/
1000
,
policy
->
min
%
1000
,
policy
->
min
/
1000
,
policy
->
min
%
1000
,
policy
->
max
/
1000
,
policy
->
max
%
1000
);
policy
->
max
/
1000
,
policy
->
max
%
1000
);
return
0
;
return
0
;
}
}
static
int
sh_cpufreq_
verify
(
struct
cpufreq_policy
*
policy
)
static
int
sh_cpufreq_
cpu_exit
(
struct
cpufreq_policy
*
policy
)
{
{
cpufreq_verify_within_limits
(
policy
,
policy
->
cpuinfo
.
min_freq
,
unsigned
int
cpu
=
policy
->
cpu
;
policy
->
cpuinfo
.
max_freq
);
struct
clk
*
cpuclk
=
&
per_cpu
(
sh_cpuclk
,
cpu
);
return
0
;
}
static
int
sh_cpufreq_exit
(
struct
cpufreq_policy
*
policy
)
cpufreq_frequency_table_put_attr
(
cpu
);
{
clk_put
(
cpuclk
);
clk_put
(
cpuclk
);
return
0
;
return
0
;
}
}
static
struct
freq_attr
*
sh_freq_attr
[]
=
{
&
cpufreq_freq_attr_scaling_available_freqs
,
NULL
,
};
static
struct
cpufreq_driver
sh_cpufreq_driver
=
{
static
struct
cpufreq_driver
sh_cpufreq_driver
=
{
.
owner
=
THIS_MODULE
,
.
owner
=
THIS_MODULE
,
.
name
=
"sh"
,
.
name
=
"sh"
,
.
init
=
sh_cpufreq_cpu_init
,
.
verify
=
sh_cpufreq_verify
,
.
target
=
sh_cpufreq_target
,
.
get
=
sh_cpufreq_get
,
.
get
=
sh_cpufreq_get
,
.
exit
=
sh_cpufreq_exit
,
.
target
=
sh_cpufreq_target
,
.
verify
=
sh_cpufreq_verify
,
.
init
=
sh_cpufreq_cpu_init
,
.
exit
=
sh_cpufreq_cpu_exit
,
.
attr
=
sh_freq_attr
,
};
};
static
int
__init
sh_cpufreq_module_init
(
void
)
static
int
__init
sh_cpufreq_module_init
(
void
)
{
{
pr
intk
(
KERN_INFO
"cpufreq:
SuperH CPU frequency driver.
\n
"
);
pr
_notice
(
"
SuperH CPU frequency driver.
\n
"
);
return
cpufreq_register_driver
(
&
sh_cpufreq_driver
);
return
cpufreq_register_driver
(
&
sh_cpufreq_driver
);
}
}
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录