Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
linux
Commits
c8541203
Commit
c8541203
authored
May 06, 2016
by
Rafael J. Wysocki
Browse files
Options
Browse Files
Download
Plain Diff
Merge back new material for v4.7.
parents
21f8a99c
f47b72a1
Changes
6
Expand all
Show whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
763 additions
and
544 deletions
+763
-544
drivers/base/power/opp/Makefile
drivers/base/power/opp/Makefile
+1
-0
drivers/base/power/opp/core.c
drivers/base/power/opp/core.c
+21
-419
drivers/base/power/opp/cpu.c
drivers/base/power/opp/cpu.c
+97
-102
drivers/base/power/opp/of.c
drivers/base/power/opp/of.c
+591
-0
drivers/base/power/opp/opp.h
drivers/base/power/opp/opp.h
+14
-0
include/linux/pm_opp.h
include/linux/pm_opp.h
+39
-23
No files found.
drivers/base/power/opp/Makefile
View file @
c8541203
ccflags-$(CONFIG_DEBUG_DRIVER)
:=
-DDEBUG
obj-y
+=
core.o cpu.o
obj-$(CONFIG_OF)
+=
of.o
obj-$(CONFIG_DEBUG_FS)
+=
debugfs.o
drivers/base/power/opp/core.c
View file @
c8541203
This diff is collapsed.
Click to expand it.
drivers/base/power/opp/cpu.c
View file @
c8541203
...
...
@@ -18,7 +18,6 @@
#include <linux/err.h>
#include <linux/errno.h>
#include <linux/export.h>
#include <linux/of.h>
#include <linux/slab.h>
#include "opp.h"
...
...
@@ -119,8 +118,66 @@ void dev_pm_opp_free_cpufreq_table(struct device *dev,
EXPORT_SYMBOL_GPL
(
dev_pm_opp_free_cpufreq_table
);
#endif
/* CONFIG_CPU_FREQ */
/* Required only for V1 bindings, as v2 can manage it from DT itself */
int
dev_pm_opp_set_sharing_cpus
(
struct
device
*
cpu_dev
,
cpumask_var_t
cpumask
)
void
_dev_pm_opp_cpumask_remove_table
(
const
struct
cpumask
*
cpumask
,
bool
of
)
{
struct
device
*
cpu_dev
;
int
cpu
;
WARN_ON
(
cpumask_empty
(
cpumask
));
for_each_cpu
(
cpu
,
cpumask
)
{
cpu_dev
=
get_cpu_device
(
cpu
);
if
(
!
cpu_dev
)
{
pr_err
(
"%s: failed to get cpu%d device
\n
"
,
__func__
,
cpu
);
continue
;
}
if
(
of
)
dev_pm_opp_of_remove_table
(
cpu_dev
);
else
dev_pm_opp_remove_table
(
cpu_dev
);
}
}
/**
* dev_pm_opp_cpumask_remove_table() - Removes OPP table for @cpumask
* @cpumask: cpumask for which OPP table needs to be removed
*
* This removes the OPP tables for CPUs present in the @cpumask.
* This should be used to remove all the OPPs entries associated with
* the cpus in @cpumask.
*
* Locking: The internal opp_table and opp structures are RCU protected.
* Hence this function internally uses RCU updater strategy with mutex locks
* to keep the integrity of the internal data structures. Callers should ensure
* that this function is *NOT* called under RCU protection or in contexts where
* mutex cannot be locked.
*/
void
dev_pm_opp_cpumask_remove_table
(
const
struct
cpumask
*
cpumask
)
{
_dev_pm_opp_cpumask_remove_table
(
cpumask
,
false
);
}
EXPORT_SYMBOL_GPL
(
dev_pm_opp_cpumask_remove_table
);
/**
* dev_pm_opp_set_sharing_cpus() - Mark OPP table as shared by few CPUs
* @cpu_dev: CPU device for which we do this operation
* @cpumask: cpumask of the CPUs which share the OPP table with @cpu_dev
*
* This marks OPP table of the @cpu_dev as shared by the CPUs present in
* @cpumask.
*
* Returns -ENODEV if OPP table isn't already present.
*
* Locking: The internal opp_table and opp structures are RCU protected.
* Hence this function internally uses RCU updater strategy with mutex locks
* to keep the integrity of the internal data structures. Callers should ensure
* that this function is *NOT* called under RCU protection or in contexts where
* mutex cannot be locked.
*/
int
dev_pm_opp_set_sharing_cpus
(
struct
device
*
cpu_dev
,
const
struct
cpumask
*
cpumask
)
{
struct
opp_device
*
opp_dev
;
struct
opp_table
*
opp_table
;
...
...
@@ -131,7 +188,7 @@ int dev_pm_opp_set_sharing_cpus(struct device *cpu_dev, cpumask_var_t cpumask)
opp_table
=
_find_opp_table
(
cpu_dev
);
if
(
IS_ERR
(
opp_table
))
{
ret
=
-
EINVAL
;
ret
=
PTR_ERR
(
opp_table
)
;
goto
unlock
;
}
...
...
@@ -152,6 +209,9 @@ int dev_pm_opp_set_sharing_cpus(struct device *cpu_dev, cpumask_var_t cpumask)
__func__
,
cpu
);
continue
;
}
/* Mark opp-table as multiple CPUs are sharing it now */
opp_table
->
shared_opp
=
true
;
}
unlock:
mutex_unlock
(
&
opp_table_lock
);
...
...
@@ -160,112 +220,47 @@ int dev_pm_opp_set_sharing_cpus(struct device *cpu_dev, cpumask_var_t cpumask)
}
EXPORT_SYMBOL_GPL
(
dev_pm_opp_set_sharing_cpus
);
#ifdef CONFIG_OF
void
dev_pm_opp_of_cpumask_remove_table
(
cpumask_var_t
cpumask
)
{
struct
device
*
cpu_dev
;
int
cpu
;
WARN_ON
(
cpumask_empty
(
cpumask
));
for_each_cpu
(
cpu
,
cpumask
)
{
cpu_dev
=
get_cpu_device
(
cpu
);
if
(
!
cpu_dev
)
{
pr_err
(
"%s: failed to get cpu%d device
\n
"
,
__func__
,
cpu
);
continue
;
}
dev_pm_opp_of_remove_table
(
cpu_dev
);
}
}
EXPORT_SYMBOL_GPL
(
dev_pm_opp_of_cpumask_remove_table
);
int
dev_pm_opp_of_cpumask_add_table
(
cpumask_var_t
cpumask
)
{
struct
device
*
cpu_dev
;
int
cpu
,
ret
=
0
;
WARN_ON
(
cpumask_empty
(
cpumask
));
for_each_cpu
(
cpu
,
cpumask
)
{
cpu_dev
=
get_cpu_device
(
cpu
);
if
(
!
cpu_dev
)
{
pr_err
(
"%s: failed to get cpu%d device
\n
"
,
__func__
,
cpu
);
continue
;
}
ret
=
dev_pm_opp_of_add_table
(
cpu_dev
);
if
(
ret
)
{
pr_err
(
"%s: couldn't find opp table for cpu:%d, %d
\n
"
,
__func__
,
cpu
,
ret
);
/* Free all other OPPs */
dev_pm_opp_of_cpumask_remove_table
(
cpumask
);
break
;
}
}
return
ret
;
}
EXPORT_SYMBOL_GPL
(
dev_pm_opp_of_cpumask_add_table
);
/*
* Works only for OPP v2 bindings.
/**
* dev_pm_opp_get_sharing_cpus() - Get cpumask of CPUs sharing OPPs with @cpu_dev
* @cpu_dev: CPU device for which we do this operation
* @cpumask: cpumask to update with information of sharing CPUs
*
* This updates the @cpumask with CPUs that are sharing OPPs with @cpu_dev.
*
* Returns -ENOENT if operating-points-v2 bindings aren't supported.
* Returns -ENODEV if OPP table isn't already present.
*
* Locking: The internal opp_table and opp structures are RCU protected.
* Hence this function internally uses RCU updater strategy with mutex locks
* to keep the integrity of the internal data structures. Callers should ensure
* that this function is *NOT* called under RCU protection or in contexts where
* mutex cannot be locked.
*/
int
dev_pm_opp_
of_get_sharing_cpus
(
struct
device
*
cpu_dev
,
cpumask_var_t
cpumask
)
int
dev_pm_opp_
get_sharing_cpus
(
struct
device
*
cpu_dev
,
struct
cpumask
*
cpumask
)
{
struct
device_node
*
np
,
*
tmp_np
;
struct
device
*
tcpu_dev
;
int
cpu
,
ret
=
0
;
/* Get OPP descriptor node */
np
=
_of_get_opp_desc_node
(
cpu_dev
);
if
(
!
np
)
{
dev_dbg
(
cpu_dev
,
"%s: Couldn't find cpu_dev node.
\n
"
,
__func__
);
return
-
ENOENT
;
}
cpumask_set_cpu
(
cpu_dev
->
id
,
cpumask
);
/* OPPs are shared ? */
if
(
!
of_property_read_bool
(
np
,
"opp-shared"
))
goto
put_cpu_node
;
for_each_possible_cpu
(
cpu
)
{
if
(
cpu
==
cpu_dev
->
id
)
continue
;
struct
opp_device
*
opp_dev
;
struct
opp_table
*
opp_table
;
int
ret
=
0
;
tcpu_dev
=
get_cpu_device
(
cpu
);
if
(
!
tcpu_dev
)
{
dev_err
(
cpu_dev
,
"%s: failed to get cpu%d device
\n
"
,
__func__
,
cpu
);
ret
=
-
ENODEV
;
goto
put_cpu_node
;
}
mutex_lock
(
&
opp_table_lock
);
/* Get OPP descriptor node */
tmp_np
=
_of_get_opp_desc_node
(
tcpu_dev
);
if
(
!
tmp_np
)
{
dev_err
(
tcpu_dev
,
"%s: Couldn't find tcpu_dev node.
\n
"
,
__func__
);
ret
=
-
ENOENT
;
goto
put_cpu_node
;
opp_table
=
_find_opp_table
(
cpu_dev
);
if
(
IS_ERR
(
opp_table
))
{
ret
=
PTR_ERR
(
opp_table
);
goto
unlock
;
}
/* CPUs are sharing opp node */
if
(
np
==
tmp_np
)
cpumask_set_cpu
(
cpu
,
cpumask
);
cpumask_clear
(
cpumask
);
of_node_put
(
tmp_np
);
if
(
opp_table
->
shared_opp
)
{
list_for_each_entry
(
opp_dev
,
&
opp_table
->
dev_list
,
node
)
cpumask_set_cpu
(
opp_dev
->
dev
->
id
,
cpumask
);
}
else
{
cpumask_set_cpu
(
cpu_dev
->
id
,
cpumask
);
}
put_cpu_node:
of_node_put
(
np
);
unlock:
mutex_unlock
(
&
opp_table_lock
);
return
ret
;
}
EXPORT_SYMBOL_GPL
(
dev_pm_opp_of_get_sharing_cpus
);
#endif
EXPORT_SYMBOL_GPL
(
dev_pm_opp_get_sharing_cpus
);
drivers/base/power/opp/of.c
0 → 100644
View file @
c8541203
This diff is collapsed.
Click to expand it.
drivers/base/power/opp/opp.h
View file @
c8541203
...
...
@@ -28,6 +28,8 @@ struct regulator;
/* Lock to allow exclusive modification to the device and opp lists */
extern
struct
mutex
opp_table_lock
;
extern
struct
list_head
opp_tables
;
/*
* Internal data structure organization with the OPP layer library is as
* follows:
...
...
@@ -183,6 +185,18 @@ struct opp_table {
struct
opp_table
*
_find_opp_table
(
struct
device
*
dev
);
struct
opp_device
*
_add_opp_dev
(
const
struct
device
*
dev
,
struct
opp_table
*
opp_table
);
struct
device_node
*
_of_get_opp_desc_node
(
struct
device
*
dev
);
void
_dev_pm_opp_remove_table
(
struct
device
*
dev
,
bool
remove_all
);
struct
dev_pm_opp
*
_allocate_opp
(
struct
device
*
dev
,
struct
opp_table
**
opp_table
);
int
_opp_add
(
struct
device
*
dev
,
struct
dev_pm_opp
*
new_opp
,
struct
opp_table
*
opp_table
);
void
_opp_remove
(
struct
opp_table
*
opp_table
,
struct
dev_pm_opp
*
opp
,
bool
notify
);
int
_opp_add_v1
(
struct
device
*
dev
,
unsigned
long
freq
,
long
u_volt
,
bool
dynamic
);
void
_dev_pm_opp_cpumask_remove_table
(
const
struct
cpumask
*
cpumask
,
bool
of
);
#ifdef CONFIG_OF
void
_of_init_opp_table
(
struct
opp_table
*
opp_table
,
struct
device
*
dev
);
#else
static
inline
void
_of_init_opp_table
(
struct
opp_table
*
opp_table
,
struct
device
*
dev
)
{}
#endif
#ifdef CONFIG_DEBUG_FS
void
opp_debug_remove_one
(
struct
dev_pm_opp
*
opp
);
...
...
include/linux/pm_opp.h
View file @
c8541203
...
...
@@ -65,6 +65,10 @@ void dev_pm_opp_put_prop_name(struct device *dev);
int
dev_pm_opp_set_regulator
(
struct
device
*
dev
,
const
char
*
name
);
void
dev_pm_opp_put_regulator
(
struct
device
*
dev
);
int
dev_pm_opp_set_rate
(
struct
device
*
dev
,
unsigned
long
target_freq
);
int
dev_pm_opp_set_sharing_cpus
(
struct
device
*
cpu_dev
,
const
struct
cpumask
*
cpumask
);
int
dev_pm_opp_get_sharing_cpus
(
struct
device
*
cpu_dev
,
struct
cpumask
*
cpumask
);
void
dev_pm_opp_remove_table
(
struct
device
*
dev
);
void
dev_pm_opp_cpumask_remove_table
(
const
struct
cpumask
*
cpumask
);
#else
static
inline
unsigned
long
dev_pm_opp_get_voltage
(
struct
dev_pm_opp
*
opp
)
{
...
...
@@ -109,25 +113,25 @@ static inline struct dev_pm_opp *dev_pm_opp_get_suspend_opp(struct device *dev)
static
inline
struct
dev_pm_opp
*
dev_pm_opp_find_freq_exact
(
struct
device
*
dev
,
unsigned
long
freq
,
bool
available
)
{
return
ERR_PTR
(
-
E
INVAL
);
return
ERR_PTR
(
-
E
NOTSUPP
);
}
static
inline
struct
dev_pm_opp
*
dev_pm_opp_find_freq_floor
(
struct
device
*
dev
,
unsigned
long
*
freq
)
{
return
ERR_PTR
(
-
E
INVAL
);
return
ERR_PTR
(
-
E
NOTSUPP
);
}
static
inline
struct
dev_pm_opp
*
dev_pm_opp_find_freq_ceil
(
struct
device
*
dev
,
unsigned
long
*
freq
)
{
return
ERR_PTR
(
-
E
INVAL
);
return
ERR_PTR
(
-
E
NOTSUPP
);
}
static
inline
int
dev_pm_opp_add
(
struct
device
*
dev
,
unsigned
long
freq
,
unsigned
long
u_volt
)
{
return
-
E
INVAL
;
return
-
E
NOTSUPP
;
}
static
inline
void
dev_pm_opp_remove
(
struct
device
*
dev
,
unsigned
long
freq
)
...
...
@@ -147,73 +151,85 @@ static inline int dev_pm_opp_disable(struct device *dev, unsigned long freq)
static
inline
struct
srcu_notifier_head
*
dev_pm_opp_get_notifier
(
struct
device
*
dev
)
{
return
ERR_PTR
(
-
E
INVAL
);
return
ERR_PTR
(
-
E
NOTSUPP
);
}
static
inline
int
dev_pm_opp_set_supported_hw
(
struct
device
*
dev
,
const
u32
*
versions
,
unsigned
int
count
)
{
return
-
E
INVAL
;
return
-
E
NOTSUPP
;
}
static
inline
void
dev_pm_opp_put_supported_hw
(
struct
device
*
dev
)
{}
static
inline
int
dev_pm_opp_set_prop_name
(
struct
device
*
dev
,
const
char
*
name
)
{
return
-
E
INVAL
;
return
-
E
NOTSUPP
;
}
static
inline
void
dev_pm_opp_put_prop_name
(
struct
device
*
dev
)
{}
static
inline
int
dev_pm_opp_set_regulator
(
struct
device
*
dev
,
const
char
*
name
)
{
return
-
E
INVAL
;
return
-
E
NOTSUPP
;
}
static
inline
void
dev_pm_opp_put_regulator
(
struct
device
*
dev
)
{}
static
inline
int
dev_pm_opp_set_rate
(
struct
device
*
dev
,
unsigned
long
target_freq
)
{
return
-
ENOTSUPP
;
}
static
inline
int
dev_pm_opp_set_sharing_cpus
(
struct
device
*
cpu_dev
,
const
struct
cpumask
*
cpumask
)
{
return
-
ENOTSUPP
;
}
static
inline
int
dev_pm_opp_get_sharing_cpus
(
struct
device
*
cpu_dev
,
struct
cpumask
*
cpumask
)
{
return
-
EINVAL
;
}
static
inline
void
dev_pm_opp_remove_table
(
struct
device
*
dev
)
{
}
static
inline
void
dev_pm_opp_cpumask_remove_table
(
const
struct
cpumask
*
cpumask
)
{
}
#endif
/* CONFIG_PM_OPP */
#if defined(CONFIG_PM_OPP) && defined(CONFIG_OF)
int
dev_pm_opp_of_add_table
(
struct
device
*
dev
);
void
dev_pm_opp_of_remove_table
(
struct
device
*
dev
);
int
dev_pm_opp_of_cpumask_add_table
(
cpumask_var_t
cpumask
);
void
dev_pm_opp_of_cpumask_remove_table
(
cpumask_var_t
cpumask
);
int
dev_pm_opp_of_get_sharing_cpus
(
struct
device
*
cpu_dev
,
cpumask_var_t
cpumask
);
int
dev_pm_opp_set_sharing_cpus
(
struct
device
*
cpu_dev
,
cpumask_var_t
cpumask
);
int
dev_pm_opp_of_cpumask_add_table
(
const
struct
cpumask
*
cpumask
);
void
dev_pm_opp_of_cpumask_remove_table
(
const
struct
cpumask
*
cpumask
);
int
dev_pm_opp_of_get_sharing_cpus
(
struct
device
*
cpu_dev
,
struct
cpumask
*
cpumask
);
#else
static
inline
int
dev_pm_opp_of_add_table
(
struct
device
*
dev
)
{
return
-
E
INVAL
;
return
-
E
NOTSUPP
;
}
static
inline
void
dev_pm_opp_of_remove_table
(
struct
device
*
dev
)
{
}
static
inline
int
dev_pm_opp_of_cpumask_add_table
(
cpumask_var_t
cpumask
)
{
return
-
ENOSYS
;
}
static
inline
void
dev_pm_opp_of_cpumask_remove_table
(
cpumask_var_t
cpumask
)
static
inline
int
dev_pm_opp_of_cpumask_add_table
(
const
struct
cpumask
*
cpumask
)
{
return
-
ENOTSUPP
;
}
static
inline
int
dev_pm_opp_of_get_sharing_cpus
(
struct
device
*
cpu_dev
,
cpumask_var_t
cpumask
)
static
inline
void
dev_pm_opp_of_cpumask_remove_table
(
const
struct
cpumask
*
cpumask
)
{
return
-
ENOSYS
;
}
static
inline
int
dev_pm_opp_
set_sharing_cpus
(
struct
device
*
cpu_dev
,
cpumask_var_t
cpumask
)
static
inline
int
dev_pm_opp_
of_get_sharing_cpus
(
struct
device
*
cpu_dev
,
struct
cpumask
*
cpumask
)
{
return
-
ENO
SYS
;
return
-
ENO
TSUPP
;
}
#endif
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment