Merge branches 'pm-cpuidle', 'pm-devfreq' and 'pm-clk'
* pm-cpuidle: cpuidle/coupled: Remove redundant 'dev' argument of cpuidle_state_is_coupled() cpuidle/coupled: Remove cpuidle_device::safe_state_index intel_idle: Skylake Client Support intel_idle: allow idle states to be freeze-mode specific * pm-devfreq: PM / devfreq: exynos-ppmu: Update documentation to support PPMUv2 PM / devfreq: exynos-ppmu: Add the support of PPMUv2 for Exynos5433 PM / devfreq: event: Remove incorrect property in exynos-ppmu DT binding * pm-clk: PM / clk: don't return int on __pm_clk_enable()
This commit is contained in:
commit
5cf896b3c7
9 changed files with 353 additions and 26 deletions
|
@ -11,15 +11,14 @@ to various devfreq devices. The devfreq devices would use the event data when
|
|||
derterming the current state of each IP.
|
||||
|
||||
Required properties:
|
||||
- compatible: Should be "samsung,exynos-ppmu".
|
||||
- compatible: Should be "samsung,exynos-ppmu" or "samsung,exynos-ppmu-v2.
|
||||
- reg: physical base address of each PPMU and length of memory mapped region.
|
||||
|
||||
Optional properties:
|
||||
- clock-names : the name of clock used by the PPMU, "ppmu"
|
||||
- clocks : phandles for clock specified in "clock-names" property
|
||||
- #clock-cells: should be 1.
|
||||
|
||||
Example1 : PPMU nodes in exynos3250.dtsi are listed below.
|
||||
Example1 : PPMUv1 nodes in exynos3250.dtsi are listed below.
|
||||
|
||||
ppmu_dmc0: ppmu_dmc0@106a0000 {
|
||||
compatible = "samsung,exynos-ppmu";
|
||||
|
@ -108,3 +107,41 @@ Example2 : Events of each PPMU node in exynos3250-rinato.dts are listed below.
|
|||
};
|
||||
};
|
||||
};
|
||||
|
||||
Example3 : PPMUv2 nodes in exynos5433.dtsi are listed below.
|
||||
|
||||
ppmu_d0_cpu: ppmu_d0_cpu@10480000 {
|
||||
compatible = "samsung,exynos-ppmu-v2";
|
||||
reg = <0x10480000 0x2000>;
|
||||
status = "disabled";
|
||||
};
|
||||
|
||||
ppmu_d0_general: ppmu_d0_general@10490000 {
|
||||
compatible = "samsung,exynos-ppmu-v2";
|
||||
reg = <0x10490000 0x2000>;
|
||||
status = "disabled";
|
||||
};
|
||||
|
||||
ppmu_d0_rt: ppmu_d0_rt@104a0000 {
|
||||
compatible = "samsung,exynos-ppmu-v2";
|
||||
reg = <0x104a0000 0x2000>;
|
||||
status = "disabled";
|
||||
};
|
||||
|
||||
ppmu_d1_cpu: ppmu_d1_cpu@104b0000 {
|
||||
compatible = "samsung,exynos-ppmu-v2";
|
||||
reg = <0x104b0000 0x2000>;
|
||||
status = "disabled";
|
||||
};
|
||||
|
||||
ppmu_d1_general: ppmu_d1_general@104c0000 {
|
||||
compatible = "samsung,exynos-ppmu-v2";
|
||||
reg = <0x104c0000 0x2000>;
|
||||
status = "disabled";
|
||||
};
|
||||
|
||||
ppmu_d1_rt: ppmu_d1_rt@104d0000 {
|
||||
compatible = "samsung,exynos-ppmu-v2";
|
||||
reg = <0x104d0000 0x2000>;
|
||||
status = "disabled";
|
||||
};
|
||||
|
|
|
@ -38,7 +38,7 @@ struct pm_clock_entry {
|
|||
* @dev: The device for the given clock
|
||||
* @ce: PM clock entry corresponding to the clock.
|
||||
*/
|
||||
static inline int __pm_clk_enable(struct device *dev, struct pm_clock_entry *ce)
|
||||
static inline void __pm_clk_enable(struct device *dev, struct pm_clock_entry *ce)
|
||||
{
|
||||
int ret;
|
||||
|
||||
|
@ -50,8 +50,6 @@ static inline int __pm_clk_enable(struct device *dev, struct pm_clock_entry *ce)
|
|||
dev_err(dev, "%s: failed to enable clk %p, error %d\n",
|
||||
__func__, ce->clk, ret);
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
|
@ -176,14 +176,12 @@ void cpuidle_coupled_parallel_barrier(struct cpuidle_device *dev, atomic_t *a)
|
|||
|
||||
/**
|
||||
* cpuidle_state_is_coupled - check if a state is part of a coupled set
|
||||
* @dev: struct cpuidle_device for the current cpu
|
||||
* @drv: struct cpuidle_driver for the platform
|
||||
* @state: index of the target state in drv->states
|
||||
*
|
||||
* Returns true if the target state is coupled with cpus besides this one
|
||||
*/
|
||||
bool cpuidle_state_is_coupled(struct cpuidle_device *dev,
|
||||
struct cpuidle_driver *drv, int state)
|
||||
bool cpuidle_state_is_coupled(struct cpuidle_driver *drv, int state)
|
||||
{
|
||||
return drv->states[state].flags & CPUIDLE_FLAG_COUPLED;
|
||||
}
|
||||
|
@ -473,7 +471,7 @@ int cpuidle_enter_state_coupled(struct cpuidle_device *dev,
|
|||
return entered_state;
|
||||
}
|
||||
entered_state = cpuidle_enter_state(dev, drv,
|
||||
dev->safe_state_index);
|
||||
drv->safe_state_index);
|
||||
local_irq_disable();
|
||||
}
|
||||
|
||||
|
@ -521,7 +519,7 @@ int cpuidle_enter_state_coupled(struct cpuidle_device *dev,
|
|||
}
|
||||
|
||||
entered_state = cpuidle_enter_state(dev, drv,
|
||||
dev->safe_state_index);
|
||||
drv->safe_state_index);
|
||||
local_irq_disable();
|
||||
}
|
||||
|
||||
|
|
|
@ -210,7 +210,7 @@ int cpuidle_enter_state(struct cpuidle_device *dev, struct cpuidle_driver *drv,
|
|||
tick_broadcast_exit();
|
||||
}
|
||||
|
||||
if (!cpuidle_state_is_coupled(dev, drv, entered_state))
|
||||
if (!cpuidle_state_is_coupled(drv, entered_state))
|
||||
local_irq_enable();
|
||||
|
||||
diff = ktime_to_us(ktime_sub(time_end, time_start));
|
||||
|
@ -259,7 +259,7 @@ int cpuidle_select(struct cpuidle_driver *drv, struct cpuidle_device *dev)
|
|||
int cpuidle_enter(struct cpuidle_driver *drv, struct cpuidle_device *dev,
|
||||
int index)
|
||||
{
|
||||
if (cpuidle_state_is_coupled(dev, drv, index))
|
||||
if (cpuidle_state_is_coupled(drv, index))
|
||||
return cpuidle_enter_state_coupled(dev, drv, index);
|
||||
return cpuidle_enter_state(dev, drv, index);
|
||||
}
|
||||
|
|
|
@ -34,15 +34,14 @@ extern int cpuidle_add_sysfs(struct cpuidle_device *dev);
|
|||
extern void cpuidle_remove_sysfs(struct cpuidle_device *dev);
|
||||
|
||||
#ifdef CONFIG_ARCH_NEEDS_CPU_IDLE_COUPLED
|
||||
bool cpuidle_state_is_coupled(struct cpuidle_device *dev,
|
||||
struct cpuidle_driver *drv, int state);
|
||||
bool cpuidle_state_is_coupled(struct cpuidle_driver *drv, int state);
|
||||
int cpuidle_enter_state_coupled(struct cpuidle_device *dev,
|
||||
struct cpuidle_driver *drv, int next_state);
|
||||
int cpuidle_coupled_register_device(struct cpuidle_device *dev);
|
||||
void cpuidle_coupled_unregister_device(struct cpuidle_device *dev);
|
||||
#else
|
||||
static inline bool cpuidle_state_is_coupled(struct cpuidle_device *dev,
|
||||
struct cpuidle_driver *drv, int state)
|
||||
static inline
|
||||
bool cpuidle_state_is_coupled(struct cpuidle_driver *drv, int state)
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
/*
|
||||
* exynos_ppmu.c - EXYNOS PPMU (Platform Performance Monitoring Unit) support
|
||||
*
|
||||
* Copyright (c) 2014 Samsung Electronics Co., Ltd.
|
||||
* Copyright (c) 2014-2015 Samsung Electronics Co., Ltd.
|
||||
* Author : Chanwoo Choi <cw00.choi@samsung.com>
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
|
@ -82,6 +82,15 @@ struct __exynos_ppmu_events {
|
|||
PPMU_EVENT(mscl),
|
||||
PPMU_EVENT(fimd0x),
|
||||
PPMU_EVENT(fimd1x),
|
||||
|
||||
/* Only for Exynos5433 SoCs */
|
||||
PPMU_EVENT(d0-cpu),
|
||||
PPMU_EVENT(d0-general),
|
||||
PPMU_EVENT(d0-rt),
|
||||
PPMU_EVENT(d1-cpu),
|
||||
PPMU_EVENT(d1-general),
|
||||
PPMU_EVENT(d1-rt),
|
||||
|
||||
{ /* sentinel */ },
|
||||
};
|
||||
|
||||
|
@ -96,6 +105,9 @@ static int exynos_ppmu_find_ppmu_id(struct devfreq_event_dev *edev)
|
|||
return -EINVAL;
|
||||
}
|
||||
|
||||
/*
|
||||
* The devfreq-event ops structure for PPMU v1.1
|
||||
*/
|
||||
static int exynos_ppmu_disable(struct devfreq_event_dev *edev)
|
||||
{
|
||||
struct exynos_ppmu *info = devfreq_event_get_drvdata(edev);
|
||||
|
@ -200,10 +212,158 @@ static const struct devfreq_event_ops exynos_ppmu_ops = {
|
|||
.get_event = exynos_ppmu_get_event,
|
||||
};
|
||||
|
||||
/*
|
||||
* The devfreq-event ops structure for PPMU v2.0
|
||||
*/
|
||||
static int exynos_ppmu_v2_disable(struct devfreq_event_dev *edev)
|
||||
{
|
||||
struct exynos_ppmu *info = devfreq_event_get_drvdata(edev);
|
||||
u32 pmnc, clear;
|
||||
|
||||
/* Disable all counters */
|
||||
clear = (PPMU_CCNT_MASK | PPMU_PMCNT0_MASK | PPMU_PMCNT1_MASK
|
||||
| PPMU_PMCNT2_MASK | PPMU_PMCNT3_MASK);
|
||||
|
||||
__raw_writel(clear, info->ppmu.base + PPMU_V2_FLAG);
|
||||
__raw_writel(clear, info->ppmu.base + PPMU_V2_INTENC);
|
||||
__raw_writel(clear, info->ppmu.base + PPMU_V2_CNTENC);
|
||||
__raw_writel(clear, info->ppmu.base + PPMU_V2_CNT_RESET);
|
||||
|
||||
__raw_writel(0x0, info->ppmu.base + PPMU_V2_CIG_CFG0);
|
||||
__raw_writel(0x0, info->ppmu.base + PPMU_V2_CIG_CFG1);
|
||||
__raw_writel(0x0, info->ppmu.base + PPMU_V2_CIG_CFG2);
|
||||
__raw_writel(0x0, info->ppmu.base + PPMU_V2_CIG_RESULT);
|
||||
__raw_writel(0x0, info->ppmu.base + PPMU_V2_CNT_AUTO);
|
||||
__raw_writel(0x0, info->ppmu.base + PPMU_V2_CH_EV0_TYPE);
|
||||
__raw_writel(0x0, info->ppmu.base + PPMU_V2_CH_EV1_TYPE);
|
||||
__raw_writel(0x0, info->ppmu.base + PPMU_V2_CH_EV2_TYPE);
|
||||
__raw_writel(0x0, info->ppmu.base + PPMU_V2_CH_EV3_TYPE);
|
||||
__raw_writel(0x0, info->ppmu.base + PPMU_V2_SM_ID_V);
|
||||
__raw_writel(0x0, info->ppmu.base + PPMU_V2_SM_ID_A);
|
||||
__raw_writel(0x0, info->ppmu.base + PPMU_V2_SM_OTHERS_V);
|
||||
__raw_writel(0x0, info->ppmu.base + PPMU_V2_SM_OTHERS_A);
|
||||
__raw_writel(0x0, info->ppmu.base + PPMU_V2_INTERRUPT_RESET);
|
||||
|
||||
/* Disable PPMU */
|
||||
pmnc = __raw_readl(info->ppmu.base + PPMU_V2_PMNC);
|
||||
pmnc &= ~PPMU_PMNC_ENABLE_MASK;
|
||||
__raw_writel(pmnc, info->ppmu.base + PPMU_V2_PMNC);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int exynos_ppmu_v2_set_event(struct devfreq_event_dev *edev)
|
||||
{
|
||||
struct exynos_ppmu *info = devfreq_event_get_drvdata(edev);
|
||||
int id = exynos_ppmu_find_ppmu_id(edev);
|
||||
u32 pmnc, cntens;
|
||||
|
||||
/* Enable all counters */
|
||||
cntens = __raw_readl(info->ppmu.base + PPMU_V2_CNTENS);
|
||||
cntens |= (PPMU_CCNT_MASK | (PPMU_ENABLE << id));
|
||||
__raw_writel(cntens, info->ppmu.base + PPMU_V2_CNTENS);
|
||||
|
||||
/* Set the event of Read/Write data count */
|
||||
switch (id) {
|
||||
case PPMU_PMNCNT0:
|
||||
case PPMU_PMNCNT1:
|
||||
case PPMU_PMNCNT2:
|
||||
__raw_writel(PPMU_V2_RO_DATA_CNT | PPMU_V2_WO_DATA_CNT,
|
||||
info->ppmu.base + PPMU_V2_CH_EVx_TYPE(id));
|
||||
break;
|
||||
case PPMU_PMNCNT3:
|
||||
__raw_writel(PPMU_V2_EVT3_RW_DATA_CNT,
|
||||
info->ppmu.base + PPMU_V2_CH_EVx_TYPE(id));
|
||||
break;
|
||||
}
|
||||
|
||||
/* Reset cycle counter/performance counter and enable PPMU */
|
||||
pmnc = __raw_readl(info->ppmu.base + PPMU_V2_PMNC);
|
||||
pmnc &= ~(PPMU_PMNC_ENABLE_MASK
|
||||
| PPMU_PMNC_COUNTER_RESET_MASK
|
||||
| PPMU_PMNC_CC_RESET_MASK
|
||||
| PPMU_PMNC_CC_DIVIDER_MASK
|
||||
| PPMU_V2_PMNC_START_MODE_MASK);
|
||||
pmnc |= (PPMU_ENABLE << PPMU_PMNC_ENABLE_SHIFT);
|
||||
pmnc |= (PPMU_ENABLE << PPMU_PMNC_COUNTER_RESET_SHIFT);
|
||||
pmnc |= (PPMU_ENABLE << PPMU_PMNC_CC_RESET_SHIFT);
|
||||
pmnc |= (PPMU_V2_MODE_MANUAL << PPMU_V2_PMNC_START_MODE_SHIFT);
|
||||
__raw_writel(pmnc, info->ppmu.base + PPMU_V2_PMNC);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int exynos_ppmu_v2_get_event(struct devfreq_event_dev *edev,
|
||||
struct devfreq_event_data *edata)
|
||||
{
|
||||
struct exynos_ppmu *info = devfreq_event_get_drvdata(edev);
|
||||
int id = exynos_ppmu_find_ppmu_id(edev);
|
||||
u32 pmnc, cntenc;
|
||||
u32 pmcnt_high, pmcnt_low;
|
||||
u64 load_count = 0;
|
||||
|
||||
/* Disable PPMU */
|
||||
pmnc = __raw_readl(info->ppmu.base + PPMU_V2_PMNC);
|
||||
pmnc &= ~PPMU_PMNC_ENABLE_MASK;
|
||||
__raw_writel(pmnc, info->ppmu.base + PPMU_V2_PMNC);
|
||||
|
||||
/* Read cycle count and performance count */
|
||||
edata->total_count = __raw_readl(info->ppmu.base + PPMU_V2_CCNT);
|
||||
|
||||
switch (id) {
|
||||
case PPMU_PMNCNT0:
|
||||
case PPMU_PMNCNT1:
|
||||
case PPMU_PMNCNT2:
|
||||
load_count = __raw_readl(info->ppmu.base + PPMU_V2_PMNCT(id));
|
||||
break;
|
||||
case PPMU_PMNCNT3:
|
||||
pmcnt_high = __raw_readl(info->ppmu.base + PPMU_V2_PMCNT3_HIGH);
|
||||
pmcnt_low = __raw_readl(info->ppmu.base + PPMU_V2_PMCNT3_LOW);
|
||||
load_count = (u64)((pmcnt_high & 0xff) << 32) + (u64)pmcnt_low;
|
||||
break;
|
||||
}
|
||||
edata->load_count = load_count;
|
||||
|
||||
/* Disable all counters */
|
||||
cntenc = __raw_readl(info->ppmu.base + PPMU_V2_CNTENC);
|
||||
cntenc |= (PPMU_CCNT_MASK | (PPMU_ENABLE << id));
|
||||
__raw_writel(cntenc, info->ppmu.base + PPMU_V2_CNTENC);
|
||||
|
||||
dev_dbg(&edev->dev, "%25s (load: %ld / %ld)\n", edev->desc->name,
|
||||
edata->load_count, edata->total_count);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const struct devfreq_event_ops exynos_ppmu_v2_ops = {
|
||||
.disable = exynos_ppmu_v2_disable,
|
||||
.set_event = exynos_ppmu_v2_set_event,
|
||||
.get_event = exynos_ppmu_v2_get_event,
|
||||
};
|
||||
|
||||
static const struct of_device_id exynos_ppmu_id_match[] = {
|
||||
{
|
||||
.compatible = "samsung,exynos-ppmu",
|
||||
.data = (void *)&exynos_ppmu_ops,
|
||||
}, {
|
||||
.compatible = "samsung,exynos-ppmu-v2",
|
||||
.data = (void *)&exynos_ppmu_v2_ops,
|
||||
},
|
||||
{ /* sentinel */ },
|
||||
};
|
||||
|
||||
static struct devfreq_event_ops *exynos_bus_get_ops(struct device_node *np)
|
||||
{
|
||||
const struct of_device_id *match;
|
||||
|
||||
match = of_match_node(exynos_ppmu_id_match, np);
|
||||
return (struct devfreq_event_ops *)match->data;
|
||||
}
|
||||
|
||||
static int of_get_devfreq_events(struct device_node *np,
|
||||
struct exynos_ppmu *info)
|
||||
{
|
||||
struct devfreq_event_desc *desc;
|
||||
struct devfreq_event_ops *event_ops;
|
||||
struct device *dev = info->dev;
|
||||
struct device_node *events_np, *node;
|
||||
int i, j, count;
|
||||
|
@ -214,6 +374,7 @@ static int of_get_devfreq_events(struct device_node *np,
|
|||
"failed to get child node of devfreq-event devices\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
event_ops = exynos_bus_get_ops(np);
|
||||
|
||||
count = of_get_child_count(events_np);
|
||||
desc = devm_kzalloc(dev, sizeof(*desc) * count, GFP_KERNEL);
|
||||
|
@ -238,7 +399,7 @@ static int of_get_devfreq_events(struct device_node *np,
|
|||
continue;
|
||||
}
|
||||
|
||||
desc[j].ops = &exynos_ppmu_ops;
|
||||
desc[j].ops = event_ops;
|
||||
desc[j].driver_data = info;
|
||||
|
||||
of_property_read_string(node, "event-name", &desc[j].name);
|
||||
|
@ -354,11 +515,6 @@ static int exynos_ppmu_remove(struct platform_device *pdev)
|
|||
return 0;
|
||||
}
|
||||
|
||||
static struct of_device_id exynos_ppmu_id_match[] = {
|
||||
{ .compatible = "samsung,exynos-ppmu", },
|
||||
{ /* sentinel */ },
|
||||
};
|
||||
|
||||
static struct platform_driver exynos_ppmu_driver = {
|
||||
.probe = exynos_ppmu_probe,
|
||||
.remove = exynos_ppmu_remove,
|
||||
|
|
|
@ -26,6 +26,9 @@ enum ppmu_counter {
|
|||
PPMU_PMNCNT_MAX,
|
||||
};
|
||||
|
||||
/***
|
||||
* PPMUv1.1 Definitions
|
||||
*/
|
||||
enum ppmu_event_type {
|
||||
PPMU_RO_BUSY_CYCLE_CNT = 0x0,
|
||||
PPMU_WO_BUSY_CYCLE_CNT = 0x1,
|
||||
|
@ -90,4 +93,71 @@ enum ppmu_reg {
|
|||
#define PPMU_PMNCT(x) (PPMU_PMCNT0 + (0x10 * x))
|
||||
#define PPMU_BEVTxSEL(x) (PPMU_BEVT0SEL + (0x100 * x))
|
||||
|
||||
/***
|
||||
* PPMU_V2.0 definitions
|
||||
*/
|
||||
enum ppmu_v2_mode {
|
||||
PPMU_V2_MODE_MANUAL = 0,
|
||||
PPMU_V2_MODE_AUTO = 1,
|
||||
PPMU_V2_MODE_CIG = 2, /* CIG (Conditional Interrupt Generation) */
|
||||
};
|
||||
|
||||
enum ppmu_v2_event_type {
|
||||
PPMU_V2_RO_DATA_CNT = 0x4,
|
||||
PPMU_V2_WO_DATA_CNT = 0x5,
|
||||
|
||||
PPMU_V2_EVT3_RW_DATA_CNT = 0x22, /* Only for Event3 */
|
||||
};
|
||||
|
||||
enum ppmu_V2_reg {
|
||||
/* PPC control register */
|
||||
PPMU_V2_PMNC = 0x04,
|
||||
PPMU_V2_CNTENS = 0x08,
|
||||
PPMU_V2_CNTENC = 0x0c,
|
||||
PPMU_V2_INTENS = 0x10,
|
||||
PPMU_V2_INTENC = 0x14,
|
||||
PPMU_V2_FLAG = 0x18,
|
||||
|
||||
/* Cycle Counter and Performance Event Counter Register */
|
||||
PPMU_V2_CCNT = 0x48,
|
||||
PPMU_V2_PMCNT0 = 0x34,
|
||||
PPMU_V2_PMCNT1 = 0x38,
|
||||
PPMU_V2_PMCNT2 = 0x3c,
|
||||
PPMU_V2_PMCNT3_LOW = 0x40,
|
||||
PPMU_V2_PMCNT3_HIGH = 0x44,
|
||||
|
||||
/* Bus Event Generator */
|
||||
PPMU_V2_CIG_CFG0 = 0x1c,
|
||||
PPMU_V2_CIG_CFG1 = 0x20,
|
||||
PPMU_V2_CIG_CFG2 = 0x24,
|
||||
PPMU_V2_CIG_RESULT = 0x28,
|
||||
PPMU_V2_CNT_RESET = 0x2c,
|
||||
PPMU_V2_CNT_AUTO = 0x30,
|
||||
PPMU_V2_CH_EV0_TYPE = 0x200,
|
||||
PPMU_V2_CH_EV1_TYPE = 0x204,
|
||||
PPMU_V2_CH_EV2_TYPE = 0x208,
|
||||
PPMU_V2_CH_EV3_TYPE = 0x20c,
|
||||
PPMU_V2_SM_ID_V = 0x220,
|
||||
PPMU_V2_SM_ID_A = 0x224,
|
||||
PPMU_V2_SM_OTHERS_V = 0x228,
|
||||
PPMU_V2_SM_OTHERS_A = 0x22c,
|
||||
PPMU_V2_INTERRUPT_RESET = 0x260,
|
||||
};
|
||||
|
||||
/* PMNC register */
|
||||
#define PPMU_V2_PMNC_START_MODE_SHIFT 20
|
||||
#define PPMU_V2_PMNC_START_MODE_MASK (0x3 << PPMU_V2_PMNC_START_MODE_SHIFT)
|
||||
|
||||
#define PPMU_PMNC_CC_RESET_SHIFT 2
|
||||
#define PPMU_PMNC_COUNTER_RESET_SHIFT 1
|
||||
#define PPMU_PMNC_ENABLE_SHIFT 0
|
||||
#define PPMU_PMNC_START_MODE_MASK BIT(16)
|
||||
#define PPMU_PMNC_CC_DIVIDER_MASK BIT(3)
|
||||
#define PPMU_PMNC_CC_RESET_MASK BIT(2)
|
||||
#define PPMU_PMNC_COUNTER_RESET_MASK BIT(1)
|
||||
#define PPMU_PMNC_ENABLE_MASK BIT(0)
|
||||
|
||||
#define PPMU_V2_PMNCT(x) (PPMU_V2_PMCNT0 + (0x4 * x))
|
||||
#define PPMU_V2_CH_EVx_TYPE(x) (PPMU_V2_CH_EV0_TYPE + (0x4 * x))
|
||||
|
||||
#endif /* __EXYNOS_PPMU_H__ */
|
||||
|
|
|
@ -591,6 +591,67 @@ static struct cpuidle_state bdw_cstates[] = {
|
|||
.enter = NULL }
|
||||
};
|
||||
|
||||
static struct cpuidle_state skl_cstates[] = {
|
||||
{
|
||||
.name = "C1-SKL",
|
||||
.desc = "MWAIT 0x00",
|
||||
.flags = MWAIT2flg(0x00),
|
||||
.exit_latency = 2,
|
||||
.target_residency = 2,
|
||||
.enter = &intel_idle,
|
||||
.enter_freeze = intel_idle_freeze, },
|
||||
{
|
||||
.name = "C1E-SKL",
|
||||
.desc = "MWAIT 0x01",
|
||||
.flags = MWAIT2flg(0x01),
|
||||
.exit_latency = 10,
|
||||
.target_residency = 20,
|
||||
.enter = &intel_idle,
|
||||
.enter_freeze = intel_idle_freeze, },
|
||||
{
|
||||
.name = "C3-SKL",
|
||||
.desc = "MWAIT 0x10",
|
||||
.flags = MWAIT2flg(0x10) | CPUIDLE_FLAG_TLB_FLUSHED,
|
||||
.exit_latency = 70,
|
||||
.target_residency = 100,
|
||||
.enter = &intel_idle,
|
||||
.enter_freeze = intel_idle_freeze, },
|
||||
{
|
||||
.name = "C6-SKL",
|
||||
.desc = "MWAIT 0x20",
|
||||
.flags = MWAIT2flg(0x20) | CPUIDLE_FLAG_TLB_FLUSHED,
|
||||
.exit_latency = 75,
|
||||
.target_residency = 200,
|
||||
.enter = &intel_idle,
|
||||
.enter_freeze = intel_idle_freeze, },
|
||||
{
|
||||
.name = "C7s-SKL",
|
||||
.desc = "MWAIT 0x33",
|
||||
.flags = MWAIT2flg(0x33) | CPUIDLE_FLAG_TLB_FLUSHED,
|
||||
.exit_latency = 124,
|
||||
.target_residency = 800,
|
||||
.enter = &intel_idle,
|
||||
.enter_freeze = intel_idle_freeze, },
|
||||
{
|
||||
.name = "C8-SKL",
|
||||
.desc = "MWAIT 0x40",
|
||||
.flags = MWAIT2flg(0x40) | CPUIDLE_FLAG_TLB_FLUSHED,
|
||||
.exit_latency = 174,
|
||||
.target_residency = 800,
|
||||
.enter = &intel_idle,
|
||||
.enter_freeze = intel_idle_freeze, },
|
||||
{
|
||||
.name = "C10-SKL",
|
||||
.desc = "MWAIT 0x60",
|
||||
.flags = MWAIT2flg(0x60) | CPUIDLE_FLAG_TLB_FLUSHED,
|
||||
.exit_latency = 890,
|
||||
.target_residency = 5000,
|
||||
.enter = &intel_idle,
|
||||
.enter_freeze = intel_idle_freeze, },
|
||||
{
|
||||
.enter = NULL }
|
||||
};
|
||||
|
||||
static struct cpuidle_state atom_cstates[] = {
|
||||
{
|
||||
.name = "C1E-ATM",
|
||||
|
@ -810,6 +871,12 @@ static const struct idle_cpu idle_cpu_bdw = {
|
|||
.disable_promotion_to_c1e = true,
|
||||
};
|
||||
|
||||
static const struct idle_cpu idle_cpu_skl = {
|
||||
.state_table = skl_cstates,
|
||||
.disable_promotion_to_c1e = true,
|
||||
};
|
||||
|
||||
|
||||
static const struct idle_cpu idle_cpu_avn = {
|
||||
.state_table = avn_cstates,
|
||||
.disable_promotion_to_c1e = true,
|
||||
|
@ -844,6 +911,8 @@ static const struct x86_cpu_id intel_idle_ids[] __initconst = {
|
|||
ICPU(0x47, idle_cpu_bdw),
|
||||
ICPU(0x4f, idle_cpu_bdw),
|
||||
ICPU(0x56, idle_cpu_bdw),
|
||||
ICPU(0x4e, idle_cpu_skl),
|
||||
ICPU(0x5e, idle_cpu_skl),
|
||||
{}
|
||||
};
|
||||
MODULE_DEVICE_TABLE(x86cpu, intel_idle_ids);
|
||||
|
@ -965,7 +1034,8 @@ static int __init intel_idle_cpuidle_driver_init(void)
|
|||
for (cstate = 0; cstate < CPUIDLE_STATE_MAX; ++cstate) {
|
||||
int num_substates, mwait_hint, mwait_cstate;
|
||||
|
||||
if (cpuidle_state_table[cstate].enter == NULL)
|
||||
if ((cpuidle_state_table[cstate].enter == NULL) &&
|
||||
(cpuidle_state_table[cstate].enter_freeze == NULL))
|
||||
break;
|
||||
|
||||
if (cstate + 1 > max_cstate) {
|
||||
|
|
|
@ -84,7 +84,6 @@ struct cpuidle_device {
|
|||
struct list_head device_list;
|
||||
|
||||
#ifdef CONFIG_ARCH_NEEDS_CPU_IDLE_COUPLED
|
||||
int safe_state_index;
|
||||
cpumask_t coupled_cpus;
|
||||
struct cpuidle_coupled *coupled;
|
||||
#endif
|
||||
|
|
Loading…
Reference in a new issue