rk3188 : add dvfs && cpufreq
author陈亮 <cl@rock-chips.com>
Sat, 8 Feb 2014 03:32:55 +0000 (19:32 -0800)
committer陈亮 <cl@rock-chips.com>
Sat, 8 Feb 2014 03:33:27 +0000 (19:33 -0800)
arch/arm/boot/dts/rk3188-tb.dts
arch/arm/boot/dts/rk3188.dtsi
arch/arm/mach-rockchip/Kconfig
arch/arm/mach-rockchip/Makefile
arch/arm/mach-rockchip/dvfs.c [new file with mode: 0644]
arch/arm/mach-rockchip/dvfs.h [new file with mode: 0644]
arch/arm/mach-rockchip/rk3188-cpufreq.c [new file with mode: 0644]
arch/arm/mach-rockchip/rk3188.c
include/linux/clk-private.h

index 99ac451e2fb45561a87d3d9d538abe91575d31b9..6c5f22e4ba00337b41683b1d3f6233c2af1414a8 100644 (file)
 &pwm3 {
         status = "okay";
 };
+
+&clk_core_dvfs_table {
+       operating-points = <
+               /* KHz    uV */
+               312000 1100000
+               504000 1100000
+               816000 1100000
+               1008000 1100000
+               1200000 1200000
+               1416000 1300000
+               1608000 1350000
+               >;
+};
+
+&clk_gpu_dvfs_table {
+       operating-points = <
+               /* KHz    uV */
+               200000 1200000
+               300000 1200000
+               400000 1300000
+               >;
+};
+
+&clk_ddr_dvfs_table {
+       operating-points = <
+               /* KHz    uV */
+               200000 1200000
+               300000 1200000
+               400000 1300000
+               >;
+};
+
 /include/ "act8846.dtsi"
 &act8846 {
        gpios =<&gpio3 GPIO_D3 GPIO_ACTIVE_LOW>; 
index f37778b38e73cd92d570f64266c9d46fd67f26e6..5e2b9d5d37e0baa6d9145ba6b2e03f140b73601f 100755 (executable)
                 status = "disabled";
 
         };
+       dvfs {
+               vd_cpu:
+               vd_cpu {
+                       regulator_name="vdd_arm";
+                       suspend_volt=<1000>; //mV
+                       pd_a9 {
+                               clk_core_dvfs_table:
+                               clk_core {
+                                       operating-points = <
+                                               /* KHz    uV */
+                                               312000 900000
+                                               504000 950000
+                                               816000 1000000
+                                               1008000 1100000
+                                               1200000 1200000
+                                               1416000 1300000
+                                               1608000 1350000
+                                               >;
+                               };
+                       };
+               };
+
+               vd_core:
+               vd_core {
+                       regulator_name="vdd_logic";
+                       suspend_volt=<1000>; //mV
+
+                       pd_gpu {
+                               clk_gpu_dvfs_table:
+                               clk_gpu {
+                                       operating-points = <
+                                               /* KHz    uV */
+                                               200000 1200000
+                                               300000 1200000
+                                               400000 1200000
+                                               >;
+                               };
+                       };
+
+                       pd_ddr {
+                               clk_ddr_dvfs_table:
+                               clk_ddr {
+                                       operating-points = <
+                                               /* KHz    uV */
+                                               200000 1200000
+                                               300000 1200000
+                                               400000 1200000
+                                               >;
+                               };
+                       };
+               };
+       };
 };
index 10ac34ae441990e0d54cdd7d989eb98485130c4d..38a28e81921a5de838b83632b6a689c886984efa 100644 (file)
@@ -14,6 +14,8 @@ config ARCH_ROCKCHIP
        select COMMON_CLK
        select GENERIC_CLOCKEVENTS
        select CLKSRC_OF if OF
+       select ARCH_HAS_CPUFREQ
+       select ARCH_HAS_OPP
 
 if ARCH_ROCKCHIP
 
@@ -26,4 +28,8 @@ config RK_DEBUG_UART
 config RK_FPGA
        bool "FPGA Board"
 
+config DVFS
+        bool "Enable dvfs"
+       default y
+       select PM_OPP
 endif
index eaa07ec4981714743b125bc0db7688e545419b48..e0eb29e72cd9ddee7ab3667445aff99e1a1b7e4e 100644 (file)
@@ -5,3 +5,5 @@ obj-$(CONFIG_RK_FPGA) += fpga.o
 obj-$(CONFIG_CPU_IDLE) += cpuidle.o
 obj-$(CONFIG_SMP) += platsmp.o
 obj-$(CONFIG_FIQ_DEBUGGER) += rk_fiq_debugger.o
+obj-$(CONFIG_CPU_FREQ) += rk3188-cpufreq.o
+obj-$(CONFIG_DVFS) += dvfs.o
diff --git a/arch/arm/mach-rockchip/dvfs.c b/arch/arm/mach-rockchip/dvfs.c
new file mode 100644 (file)
index 0000000..83b959e
--- /dev/null
@@ -0,0 +1,1693 @@
+/* arch/arm/mach-rk30/rk30_dvfs.c
+ *
+ * Copyright (C) 2012 ROCKCHIP, Inc.
+ *
+ * This software is licensed under the terms of the GNU General Public
+ * License version 2, as published by the Free Software Foundation, and
+ * may be copied, distributed, and modified under those terms.
+ *
+ * This program is distributed in the hope that 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.
+ *
+ */
+#include <linux/clk-provider.h>
+#include <linux/kernel.h>
+#include <linux/err.h>
+#include <linux/spinlock.h>
+#include <linux/list.h>
+#include <linux/slab.h>
+#include <linux/clk.h>
+#include <linux/cpufreq.h>
+#include <linux/regulator/consumer.h>
+#include <linux/delay.h>
+#include <linux/io.h>
+#include <linux/hrtimer.h>
+#include <linux/stat.h>
+#include <linux/of.h>
+#include <linux/clk.h>
+#include <linux/clk-private.h>
+#include <linux/opp.h>
+
+#include "dvfs.h"
+
+#define MHz    (1000 * 1000)
+static LIST_HEAD(rk_dvfs_tree);
+static DEFINE_MUTEX(mutex);
+static DEFINE_MUTEX(rk_dvfs_mutex);
+
+int dump_dbg_map(char *buf);
+
+#define PD_ON  1
+#define PD_OFF 0
+#define DVFS_STR_DISABLE(on) ((on)?"enable":"disable")
+
+#define get_volt_up_delay(new_volt, old_volt)  \
+       ((new_volt) > (old_volt) ? (((new_volt) - (old_volt)) >> 9) : 0)
+/**************************************vd regulator functions***************************************/
+static void dvfs_volt_up_delay(struct vd_node *vd,int new_volt, int old_volt)
+{
+       int u_time;
+       if(new_volt<=old_volt)
+               return;
+       if(vd->volt_time_flag>0)        
+               u_time=regulator_set_voltage_time(vd->regulator,old_volt,new_volt);
+       else
+               u_time=-1;              
+       if(u_time<0)// regulator is not suported time,useing default time
+       {
+               DVFS_DBG("%s:vd %s is not suported getting delay time,so we use default\n",
+                               __FUNCTION__,vd->name);
+               u_time=((new_volt) - (old_volt)) >> 9;
+       }
+       DVFS_DBG("%s:vd %s volt %d to %d delay %d us\n",__FUNCTION__,vd->name,
+               old_volt,new_volt,u_time);
+       if (u_time >= 1000) {
+               mdelay(u_time / 1000);
+               udelay(u_time % 1000);
+               DVFS_ERR("regulator set vol delay is larger 1ms,old is %d,new is %d\n",old_volt,new_volt);
+       } else if (u_time) {
+               udelay(u_time);
+       }                       
+}
+
+int dvfs_regulator_set_voltage_readback(struct regulator *regulator, int min_uV, int max_uV)
+{
+       int ret = 0, read_back = 0;
+       ret = dvfs_regulator_set_voltage(regulator, max_uV, max_uV);
+       if (ret < 0) {
+               DVFS_ERR("%s now read back to check voltage\n", __func__);
+
+               /* read back to judge if it is already effect */
+               mdelay(2);
+               read_back = dvfs_regulator_get_voltage(regulator);
+               if (read_back == max_uV) {
+                       DVFS_ERR("%s set ERROR but already effected, volt=%d\n", __func__, read_back);
+                       ret = 0;
+               } else {
+                       DVFS_ERR("%s set ERROR AND NOT effected, volt=%d\n", __func__, read_back);
+               }
+       }
+       return ret;
+}
+// for clk enable case to get vd regulator info
+void clk_enable_dvfs_regulator_check(struct vd_node *vd)
+{
+       vd->cur_volt = dvfs_regulator_get_voltage(vd->regulator);
+       if(vd->cur_volt<=0)
+       {
+               vd->volt_set_flag = DVFS_SET_VOLT_FAILURE;
+       }
+       vd->volt_set_flag = DVFS_SET_VOLT_SUCCESS;
+}
+
+static void dvfs_get_vd_regulator_volt_list(struct vd_node *vd)
+{
+       unsigned i,selector=dvfs_regulator_count_voltages(vd->regulator);
+       int sel_volt=0;
+       
+       if(selector>VD_VOL_LIST_CNT)
+               selector=VD_VOL_LIST_CNT;
+       
+       mutex_lock(&mutex);
+       for (i = 0; i<selector; i++) {
+               sel_volt=dvfs_regulator_list_voltage(vd->regulator,i);
+               if(sel_volt<=0)
+               {       
+                       DVFS_WARNING("%s : selector=%u,but volt <=0\n",vd->name,i);
+                       break;
+               }
+               vd->volt_list[i]=sel_volt;      
+               DVFS_DBG("%s:selector=%u,volt %d\n",vd->name,i,sel_volt);
+       }
+       vd->n_voltages=selector;
+       mutex_unlock(&mutex);
+}
+
+// >= volt
+static int vd_regulator_round_volt_max(struct vd_node *vd, int volt)
+{
+       int sel_volt;
+       unsigned i;
+       
+       for (i = 0; i<vd->n_voltages; i++) {
+               sel_volt=vd->volt_list[i];
+               if(sel_volt<=0)
+               {       
+                       DVFS_WARNING("%s:list_volt : selector=%u,but volt <=0\n",__FUNCTION__,i);
+                       return -1;
+               }
+               if(sel_volt>=volt)
+                return sel_volt;       
+       }
+       return -1;
+}
+// >=volt
+static int vd_regulator_round_volt_min(struct vd_node *vd, int volt)
+{
+       int sel_volt;
+       unsigned i;
+       
+       for (i = 0; i<vd->n_voltages; i++) {
+               sel_volt=vd->volt_list[i];
+               if(sel_volt<=0)
+               {       
+                       DVFS_WARNING("%s:list_volt : selector=%u,but volt <=0\n",__FUNCTION__,i);
+                       return -1;
+               }
+               if(sel_volt>volt)
+               {
+                       if(i>0)
+                               return vd->volt_list[i-1];
+                       else
+                               return -1;
+               }       
+       }
+       return -1;
+}
+
+// >=volt
+int vd_regulator_round_volt(struct vd_node *vd, int volt,int flags)
+{
+       if(!vd->n_voltages)
+               return -1;
+       if(flags==VD_LIST_RELATION_L)
+               return vd_regulator_round_volt_min(vd,volt);
+       else
+               return vd_regulator_round_volt_max(vd,volt);    
+}
+EXPORT_SYMBOL(vd_regulator_round_volt);
+
+
+static void dvfs_table_round_volt(struct dvfs_node  *clk_dvfs_node)
+{
+       int i,test_volt;
+
+       if(!clk_dvfs_node->dvfs_table||!clk_dvfs_node->vd||IS_ERR_OR_NULL(clk_dvfs_node->vd->regulator))
+               return;
+       mutex_lock(&mutex);
+       for (i = 0; (clk_dvfs_node->dvfs_table[i].frequency != CPUFREQ_TABLE_END); i++) {
+
+               test_volt=vd_regulator_round_volt(clk_dvfs_node->vd,clk_dvfs_node->dvfs_table[i].index,VD_LIST_RELATION_H);
+               if(test_volt<=0)
+               {       
+                       DVFS_WARNING("clk %s:round_volt : is %d,but list <=0\n",clk_dvfs_node->name,clk_dvfs_node->dvfs_table[i].index);
+                       break;
+               }
+               DVFS_DBG("clk %s:round_volt %d to %d\n",clk_dvfs_node->name,clk_dvfs_node->dvfs_table[i].index,test_volt);
+               clk_dvfs_node->dvfs_table[i].index=test_volt;           
+       }
+       mutex_unlock(&mutex);
+}
+void dvfs_vd_get_regulator_volt_time_info(struct vd_node *vd)
+{
+       if(vd->volt_time_flag<=0)// check regulator support get uping vol timer
+       {
+               vd->volt_time_flag=dvfs_regulator_set_voltage_time(vd->regulator,vd->cur_volt,vd->cur_volt+200*1000);
+               if(vd->volt_time_flag<0)
+               {
+                       DVFS_DBG("%s,vd %s volt_time is no support\n",__FUNCTION__,vd->name);
+               }
+               else
+               {
+                       DVFS_DBG("%s,vd %s volt_time is support,up 200mv need delay %d us\n",__FUNCTION__,vd->name,vd->volt_time_flag);
+
+               }       
+       }
+}
+
+void dvfs_vd_get_regulator_mode_info(struct vd_node *vd)
+{
+       //REGULATOR_MODE_FAST
+       if(vd->mode_flag<=0)// check regulator support get uping vol timer
+       {
+               vd->mode_flag=dvfs_regulator_get_mode(vd->regulator);
+               if(vd->mode_flag==REGULATOR_MODE_FAST||vd->mode_flag==REGULATOR_MODE_NORMAL
+                       ||vd->mode_flag==REGULATOR_MODE_IDLE||vd->mode_flag==REGULATOR_MODE_STANDBY)
+               {
+                       if(dvfs_regulator_set_mode(vd->regulator,vd->mode_flag)<0)
+                       {
+                               vd->mode_flag=0;// check again
+                       }
+                       
+               }
+               if(vd->mode_flag>0)
+               {
+                       DVFS_DBG("%s,vd %s mode(now is %d) support\n",__FUNCTION__,vd->name,vd->mode_flag);
+               }
+               else
+               {
+                       DVFS_DBG("%s,vd %s mode is not support now check\n",__FUNCTION__,vd->name);
+
+               }
+               
+       }
+}
+struct regulator *dvfs_get_regulator1(char *regulator_name) 
+{
+       struct vd_node *vd;
+       list_for_each_entry(vd, &rk_dvfs_tree, node) {
+               if (strcmp(regulator_name, vd->regulator_name) == 0) {
+                       return vd->regulator;
+               }
+       }
+       return NULL;
+}
+
+int dvfs_get_rate_range(struct clk *clk)
+{
+       struct dvfs_node *clk_dvfs_node = clk_get_dvfs_info(clk);
+       struct cpufreq_frequency_table *table;
+       int i = 0;
+
+       if (!clk_dvfs_node)
+               return -1;
+
+       clk_dvfs_node->min_rate = 0;
+       clk_dvfs_node->max_rate = 0;
+
+       table = clk_dvfs_node->dvfs_table;
+       for (i = 0; table[i].frequency != CPUFREQ_TABLE_END; i++) {
+               clk_dvfs_node->max_rate = table[i].frequency / 1000 * 1000 * 1000;
+               if (i == 0)
+                       clk_dvfs_node->min_rate = table[i].frequency / 1000 * 1000 * 1000;
+       }
+
+       DVFS_DBG("%s: clk %s, limit rate [min, max] = [%u, %u]\n",
+                       __func__, clk_dvfs_node->name, clk_dvfs_node->min_rate, clk_dvfs_node->max_rate);
+
+       return 0;
+}
+
+/**************************************dvfs clocks functions***************************************/
+int clk_dvfs_enable_limit(struct clk *clk, unsigned int min_rate, unsigned max_rate)
+{
+       struct dvfs_node *clk_dvfs_node;
+       u32 rate = 0, ret = 0;
+       clk_dvfs_node = clk_get_dvfs_info(clk);
+       
+       if (IS_ERR_OR_NULL(clk_dvfs_node)) {
+               DVFS_ERR("%s: can not get dvfs clk(%s)\n", __func__, __clk_get_name(clk));
+               return -1;
+
+       }
+
+       if (clk_dvfs_node->vd && clk_dvfs_node->vd->vd_dvfs_target){
+               mutex_lock(&rk_dvfs_mutex);
+               
+               dvfs_get_rate_range(clk);
+               clk_dvfs_node->freq_limit_en = 1;
+               clk_dvfs_node->min_rate = min_rate > clk_dvfs_node->min_rate ? min_rate : clk_dvfs_node->min_rate;
+               clk_dvfs_node->max_rate = max_rate < clk_dvfs_node->max_rate ? max_rate : clk_dvfs_node->max_rate;
+               if (clk_dvfs_node->last_set_rate == 0)
+                       rate = clk_get_rate(clk);
+               else
+                       rate = clk_dvfs_node->last_set_rate;
+               ret = clk_dvfs_node->vd->vd_dvfs_target(clk->hw, rate, rate);
+               clk_dvfs_node->last_set_rate = rate;
+
+               mutex_unlock(&rk_dvfs_mutex);
+
+       }
+
+       DVFS_DBG("%s: clk(%s) last_set_rate=%u; [min_rate, max_rate]=[%u, %u]\n",
+                       __func__, __clk_get_name(clk), clk_dvfs_node->last_set_rate, clk_dvfs_node->min_rate, clk_dvfs_node->max_rate);
+
+       return 0;
+}
+
+int clk_dvfs_disable_limit(struct clk *clk)
+{
+       struct dvfs_node *clk_dvfs_node;
+       u32 ret = 0;
+       clk_dvfs_node = clk_get_dvfs_info(clk);
+       if (IS_ERR_OR_NULL(clk_dvfs_node)) {
+               DVFS_ERR("%s: can not get dvfs clk(%s)\n", __func__, __clk_get_name(clk));
+               return -1;
+
+       }
+
+       if (clk_dvfs_node->vd && clk_dvfs_node->vd->vd_dvfs_target){
+               mutex_lock(&rk_dvfs_mutex);
+               /* To reset clk_dvfs_node->min_rate/max_rate */
+               dvfs_get_rate_range(clk);
+
+               clk_dvfs_node->freq_limit_en = 0;
+               ret = clk_dvfs_node->vd->vd_dvfs_target(clk->hw, clk_dvfs_node->last_set_rate, clk_dvfs_node->last_set_rate);
+
+               mutex_unlock(&rk_dvfs_mutex);
+       }
+
+       DVFS_DBG("%s: clk(%s) last_set_rate=%u; [min_rate, max_rate]=[%u, %u]\n",
+                       __func__, __clk_get_name(clk), clk_dvfs_node->last_set_rate, clk_dvfs_node->min_rate, clk_dvfs_node->max_rate);
+       return 0;
+}
+
+int dvfs_vd_clk_set_rate(struct clk_hw *hw, unsigned long rate,
+                           unsigned long parent_rate)
+{
+       int ret = -1;
+       struct dvfs_node *dvfs_info = clk_get_dvfs_info(hw->clk);
+       
+       DVFS_DBG("%s(%s(%lu))\n", __func__, dvfs_info->name, rate);
+       
+       #if 0 // judge by reference func in rk
+       if (dvfs_support_clk_set_rate(dvfs_info)==false) {
+               DVFS_ERR("dvfs func:%s is not support!\n", __func__);
+               return ret;
+       }
+       #endif
+       
+       if (dvfs_info->vd && dvfs_info->vd->vd_dvfs_target) {
+               // mutex_lock(&vd->dvfs_mutex);
+               mutex_lock(&rk_dvfs_mutex);
+               ret = dvfs_info->vd->vd_dvfs_target(hw, rate, rate);
+               dvfs_info->last_set_rate = rate;
+               mutex_unlock(&rk_dvfs_mutex);
+               // mutex_unlock(&vd->dvfs_mutex);
+       } else {
+               //DVFS_WARNING("%s(%s),vd is no target callback\n", __func__, __clk_get_name(clk));     
+               return -1;
+       }
+               
+       //DVFS_DBG("%s(%s(%lu)),is end\n", __func__, clk->name, rate);
+       return ret;
+}
+EXPORT_SYMBOL(dvfs_vd_clk_set_rate);
+
+static void dvfs_table_round_clk_rate(struct dvfs_node  *clk_dvfs_node)
+{
+       int i;
+       unsigned long temp_rate;
+       int rate;
+       int flags;
+       
+       if(!clk_dvfs_node->dvfs_table||clk_dvfs_node->clk==NULL)//||is_suport_round_rate(clk_dvfs_node->clk)<0)
+               return;
+       
+       mutex_lock(&mutex);
+       for (i = 0; (clk_dvfs_node->dvfs_table[i].frequency != CPUFREQ_TABLE_END); i++) {
+               //ddr rate = real rate+flags
+               flags=clk_dvfs_node->dvfs_table[i].frequency%1000;
+               rate=(clk_dvfs_node->dvfs_table[i].frequency/1000)*1000;
+               temp_rate=clk_round_rate(clk_dvfs_node->clk,rate*1000);
+               if(temp_rate<=0)
+               {       
+                       DVFS_WARNING("clk %s:round_clk_rate : is %d,but round <=0",clk_dvfs_node->name,clk_dvfs_node->dvfs_table[i].frequency);
+                       break;
+               }
+               
+               /* Set rate unit as MHZ */
+               if (temp_rate % MHz != 0)
+                       temp_rate = (temp_rate / MHz + 1) * MHz;
+
+               temp_rate = (temp_rate / 1000) + flags;
+               
+               DVFS_DBG("clk %s round_clk_rate %d to %d\n",
+                       clk_dvfs_node->name,clk_dvfs_node->dvfs_table[i].frequency,(int)(temp_rate));
+               
+               clk_dvfs_node->dvfs_table[i].frequency=temp_rate;               
+       }
+       mutex_unlock(&mutex);
+}
+
+int clk_dvfs_node_get_ref_volt(struct dvfs_node *clk_dvfs_node, int rate_khz,
+               struct cpufreq_frequency_table *clk_fv)
+{
+       int i = 0;
+       
+       if (rate_khz == 0 || !clk_dvfs_node || !clk_dvfs_node->dvfs_table) {
+               /* since no need */
+               return -EINVAL;
+       }
+       clk_fv->frequency = rate_khz;
+       clk_fv->index = 0;
+
+       for (i = 0; (clk_dvfs_node->dvfs_table[i].frequency != CPUFREQ_TABLE_END); i++) {
+               if (clk_dvfs_node->dvfs_table[i].frequency >= rate_khz) {
+                       clk_fv->frequency = clk_dvfs_node->dvfs_table[i].frequency;
+                       clk_fv->index = clk_dvfs_node->dvfs_table[i].index;
+                        //printk("%s,%s rate=%ukhz(vol=%d)\n",__func__,clk_dvfs_node->name,
+                        //clk_fv->frequency, clk_fv->index);
+                       return 0;
+               }
+       }
+       clk_fv->frequency = 0;
+       clk_fv->index = 0;
+       //DVFS_DBG("%s get corresponding voltage error! out of bound\n", clk_dvfs_node->name);
+       return -1;
+}
+
+static int dvfs_pd_get_newvolt_byclk(struct pd_node *pd, struct dvfs_node *clk_dvfs_node)
+{
+       struct clk_list *child;
+       int volt_max = 0;
+
+       if (!pd || !clk_dvfs_node)
+               return 0;
+
+       if (clk_dvfs_node->set_volt >= pd->cur_volt) {
+               return clk_dvfs_node->set_volt;
+       }
+
+       list_for_each_entry(child, &pd->clk_list, node) {
+               // DVFS_DBG("%s ,pd(%s),dvfs(%s),volt(%u)\n",__func__,pd->name,
+               // clk_dvfs_node->name,clk_dvfs_node->set_volt);
+               volt_max = max(volt_max, child->clk_dvfs_node->set_volt);
+       }
+       return volt_max;
+}
+
+void dvfs_update_clk_pds_volt(struct dvfs_node *clk_dvfs_node)
+{
+       struct pd_node *pd;
+       
+       if (!clk_dvfs_node)
+               return;
+       
+       pd = clk_dvfs_node->pd;
+       if (!pd)
+               return ;
+       
+       pd->cur_volt = dvfs_pd_get_newvolt_byclk(pd, clk_dvfs_node);
+       /*for (i = 0; (clk_dvfs_node->pds[i].pd != NULL); i++) {
+               pd = clk_dvfs_node->pds[i].pd;
+               // DVFS_DBG("%s dvfs(%s),pd(%s)\n",__func__,clk_dvfs_node->name,pd->name);
+               pd->cur_volt = dvfs_pd_get_newvolt_byclk(pd, clk_dvfs_node);
+       }*/
+}
+
+int dvfs_vd_get_newvolt_bypd(struct vd_node *vd)
+{
+       int volt_max_vd = 0;
+       struct pd_node *pd;
+       //struct depend_list    *depend;
+
+       if (!vd)
+               return -EINVAL;
+       
+       list_for_each_entry(pd, &vd->pd_list, node) {
+               // DVFS_DBG("%s pd(%s,%u)\n",__func__,pd->name,pd->cur_volt);
+               volt_max_vd = max(volt_max_vd, pd->cur_volt);
+       }
+
+       /* some clks depend on this voltage domain */
+/*     if (!list_empty(&vd->req_volt_list)) {
+               list_for_each_entry(depend, &vd->req_volt_list, node2vd) {
+                       volt_max_vd = max(volt_max_vd, depend->req_volt);
+               }
+       }*/
+       return volt_max_vd;
+}
+
+int dvfs_vd_get_newvolt_byclk(struct dvfs_node *clk_dvfs_node)
+{
+       if (!clk_dvfs_node)
+               return -1;
+       dvfs_update_clk_pds_volt(clk_dvfs_node);
+       return  dvfs_vd_get_newvolt_bypd(clk_dvfs_node->vd);
+}
+
+void clk_dvfs_register_set_rate_callback(struct clk *clk, clk_dvfs_target_callback clk_dvfs_target)
+{
+       struct dvfs_node *clk_dvfs_node = clk_get_dvfs_info(clk);
+       if (IS_ERR_OR_NULL(clk_dvfs_node)) {
+               DVFS_ERR("%s %s get clk_dvfs_node err\n", __func__, clk->name);
+               return ;
+       }
+       clk_dvfs_node->clk_dvfs_target = clk_dvfs_target;
+}
+
+/************************************************ freq volt table************************************/
+struct cpufreq_frequency_table *dvfs_get_freq_volt_table(struct clk *clk) 
+{
+       struct dvfs_node *info = clk_get_dvfs_info(clk);
+       struct cpufreq_frequency_table *table;
+       if (!info || !info->dvfs_table) {
+               return NULL;
+       }
+       mutex_lock(&mutex);
+       table = info->dvfs_table;
+       mutex_unlock(&mutex);
+       return table;
+}
+EXPORT_SYMBOL(dvfs_get_freq_volt_table);
+
+int dvfs_set_freq_volt_table(struct clk *clk, struct cpufreq_frequency_table *table)
+{
+       struct dvfs_node *info = clk_get_dvfs_info(clk);
+
+       if (!info)
+               return -1;      
+       if (!table)
+       {               
+               info->min_rate=0;       
+               info->max_rate=0;       
+               return -1;
+       }
+
+       mutex_lock(&mutex);
+       info->dvfs_table = table;
+       dvfs_get_rate_range(clk);
+       mutex_unlock(&mutex);
+
+       dvfs_table_round_clk_rate(info);
+       dvfs_table_round_volt(info);
+       return 0;
+}
+EXPORT_SYMBOL(dvfs_set_freq_volt_table);
+
+int clk_enable_dvfs(struct clk *clk)
+{
+       struct dvfs_node *clk_dvfs_node;
+       struct cpufreq_frequency_table clk_fv;
+       
+       if (!clk) {
+               DVFS_ERR("clk enable dvfs error\n");
+               return -EINVAL;
+       }
+       
+       clk_dvfs_node = clk_get_dvfs_info(clk);
+       if (!clk_dvfs_node || !clk_dvfs_node->vd) {
+               DVFS_ERR("%s clk(%s) not support dvfs!\n", __func__, __clk_get_name(clk));
+               return -EINVAL;
+       }
+       if (clk_dvfs_node->enable_dvfs == 0) {
+               if (IS_ERR_OR_NULL(clk_dvfs_node->vd->regulator)) {
+                       //regulator = NULL;
+                       if (clk_dvfs_node->vd->regulator_name)
+                               clk_dvfs_node->vd->regulator = dvfs_regulator_get(NULL, clk_dvfs_node->vd->regulator_name);
+                       if (!IS_ERR_OR_NULL(clk_dvfs_node->vd->regulator)) {
+                               // DVFS_DBG("dvfs_regulator_get(%s)\n",clk_dvfs_node->vd->regulator_name);
+                               clk_enable_dvfs_regulator_check(clk_dvfs_node->vd);
+                               dvfs_get_vd_regulator_volt_list(clk_dvfs_node->vd);
+                               dvfs_vd_get_regulator_volt_time_info(clk_dvfs_node->vd);
+                               //dvfs_vd_get_regulator_mode_info(clk_dvfs_node->vd);
+                       } else {
+                               //clk_dvfs_node->vd->regulator = NULL;
+                               clk_dvfs_node->enable_dvfs = 0;
+                               DVFS_ERR("%s can't get regulator in %s\n", clk_dvfs_node->name, __func__);
+                               return -ENXIO;
+                       }
+               } else {
+                            clk_enable_dvfs_regulator_check(clk_dvfs_node->vd);
+                       // DVFS_DBG("%s(%s) vd volt=%u\n",__func__,clk_dvfs_node->name,clk_dvfs_node->vd->cur_volt);
+               }
+
+               dvfs_table_round_clk_rate(clk_dvfs_node);
+
+               mutex_lock(&mutex);
+               dvfs_get_rate_range(clk);
+               mutex_unlock(&mutex);
+               
+               dvfs_table_round_volt(clk_dvfs_node);
+               clk_dvfs_node->set_freq = clk_dvfs_node_get_rate_kz(clk);
+               DVFS_DBG("%s ,%s get freq %u!\n", __func__, clk_dvfs_node->name, clk_dvfs_node->set_freq);
+
+               if (clk_dvfs_node_get_ref_volt(clk_dvfs_node, clk_dvfs_node->set_freq, &clk_fv)) {
+                       if (clk_dvfs_node->dvfs_table[0].frequency == CPUFREQ_TABLE_END) {
+                               DVFS_ERR("%s table empty\n", __func__);
+                               clk_dvfs_node->enable_dvfs = 0;
+                               return -1;
+                       } else {
+                               DVFS_WARNING("%s table all value are smaller than default, use default, just enable dvfs\n", __func__);
+                               clk_dvfs_node->enable_dvfs++;
+                               return 0;
+                       }
+               }
+
+               clk_dvfs_node->set_volt = clk_fv.index;
+               dvfs_vd_get_newvolt_byclk(clk_dvfs_node);
+               DVFS_DBG("%s, %s, freq %u(ref vol %u)\n",__func__, clk_dvfs_node->name,
+                        clk_dvfs_node->set_freq, clk_dvfs_node->set_volt);
+#if 0
+               if (clk_dvfs_node->dvfs_nb) {
+                       // must unregister when clk disable
+                       clk_notifier_register(clk, clk_dvfs_node->dvfs_nb);
+               }
+#endif
+
+#if 0
+               if(clk_dvfs_node->vd->cur_volt < clk_dvfs_node->set_volt) {
+                       int ret;
+                       mutex_lock(&rk_dvfs_mutex);
+                       ret = dvfs_regulator_set_voltage_readback(clk_dvfs_node->vd->regulator, clk_dvfs_node->set_volt, clk_dvfs_node->set_volt);
+                       if (ret < 0) {
+                               clk_dvfs_node->vd->volt_set_flag = DVFS_SET_VOLT_FAILURE;
+                               clk_dvfs_node->enable_dvfs = 0;
+                               DVFS_ERR("dvfs enable clk %s,set volt error \n", clk_dvfs_node->name);
+                               mutex_unlock(&rk_dvfs_mutex);
+                               return -1;
+                       }
+                       clk_dvfs_node->vd->volt_set_flag = DVFS_SET_VOLT_SUCCESS;
+                       mutex_unlock(&rk_dvfs_mutex);
+               }
+#endif
+               clk_dvfs_node->enable_dvfs++;
+       } else {
+               DVFS_ERR("dvfs already enable clk enable = %d!\n", clk_dvfs_node->enable_dvfs);
+               clk_dvfs_node->enable_dvfs++;
+       }
+       return 0;
+}
+
+int clk_disable_dvfs(struct clk *clk)
+{
+       struct dvfs_node *clk_dvfs_node;
+       clk_dvfs_node = clk_get_dvfs_info(clk);
+       if (!clk_dvfs_node->enable_dvfs) {
+               DVFS_DBG("clk is already closed!\n");
+               return -1;
+       } else {
+               clk_dvfs_node->enable_dvfs--;
+               if (0 == clk_dvfs_node->enable_dvfs) {
+                       DVFS_ERR("clk closed!\n");
+#if 0
+                       clk_notifier_unregister(clk, clk_dvfs_node->dvfs_nb);
+                       DVFS_DBG("clk unregister nb!\n");
+#endif
+               }
+       }
+       return 0;
+}
+struct dvfs_node *dvfs_get_clk_dvfs_node_byname(char *name) 
+{
+       struct vd_node *vd;
+       struct pd_node *pd;
+       struct clk_list *child;
+       list_for_each_entry(vd, &rk_dvfs_tree, node) {
+               list_for_each_entry(pd, &vd->pd_list, node) {
+                       list_for_each_entry(child, &pd->clk_list, node) {
+                               if (0 == strcmp(child->clk_dvfs_node->name, name)) {
+                                       return child->clk_dvfs_node;
+                               }
+                       }
+               }
+       }
+       return NULL;
+}
+int rk_regist_vd(struct vd_node *vd)
+{
+       if (!vd)
+               return -EINVAL;
+       mutex_lock(&mutex);
+       //mutex_init(&vd->dvfs_mutex);
+       list_add(&vd->node, &rk_dvfs_tree);
+       INIT_LIST_HEAD(&vd->pd_list);
+       INIT_LIST_HEAD(&vd->req_volt_list);
+       vd->mode_flag=0;
+       vd->volt_time_flag=0;
+       vd->n_voltages=0;
+       mutex_unlock(&mutex);
+       return 0;
+}
+
+int rk_regist_pd(struct pd_node *pd)
+{
+       struct vd_node  *vd;
+
+       vd = pd->vd;
+       if (!vd)
+               return -EINVAL;
+       
+       mutex_lock(&mutex);
+       
+       list_add(&pd->node, &vd->pd_list);
+       INIT_LIST_HEAD(&pd->clk_list);  
+
+       mutex_unlock(&mutex);
+
+       return 0;
+}
+
+int rk_regist_clk(struct dvfs_node *clk_dvfs_node)
+{
+       struct pd_node  *pd;
+       struct clk_list *child;
+
+       pd = clk_dvfs_node->pd;
+       if (!pd)
+               return -EINVAL;
+       
+       mutex_lock(&mutex);
+
+       child = &clk_dvfs_node->clk_list;
+       child->clk_dvfs_node = clk_dvfs_node;
+       list_add(&child->node, &pd->clk_list);
+       
+       mutex_unlock(&mutex);
+       
+       return 0;
+}
+
+int correct_volt(int *volt_clk, int *volt_dep, int clk_biger_than_dep, int dep_biger_than_clk)
+{
+       int up_boundary = 0, low_boundary = 0;
+
+       up_boundary = *volt_clk + dep_biger_than_clk;
+       low_boundary = *volt_clk - clk_biger_than_dep;
+
+       if (*volt_dep < low_boundary || *volt_dep > up_boundary) {
+
+               if (*volt_dep < low_boundary) {
+                       *volt_dep = low_boundary;
+
+               } else if (*volt_dep > up_boundary) {
+                       *volt_clk = *volt_dep - dep_biger_than_clk;
+               }
+       }
+
+       return 0;
+}
+
+int dvfs_scale_volt(struct vd_node *vd_clk, struct vd_node *vd_dep,
+               int volt_old, int volt_new, int volt_dep_old, int volt_dep_new, int clk_biger_than_dep, int dep_biger_than_clk)
+{
+       struct regulator *regulator, *regulator_dep;
+       int volt = 0, volt_dep = 0, step = 0, step_dep = 0;
+       int volt_tmp = 0, volt_dep_tmp = 0;
+       int volt_pre = 0, volt_dep_pre = 0;
+       int ret = 0;
+
+       DVFS_DBG("ENTER %s, volt=%d(old=%d), volt_dep=%d(dep_old=%d)\n", __func__, volt_new, volt_old, volt_dep_new, volt_dep_old);
+       regulator = vd_clk->regulator;
+       regulator_dep = vd_dep->regulator;
+
+       if (IS_ERR_OR_NULL(regulator) || IS_ERR(regulator_dep)) {
+               DVFS_ERR("%s clk_dvfs_node->vd->regulator or depend->dep_vd->regulator == NULL\n", __func__);
+               return -1;
+       }
+
+       volt = volt_old;
+       volt_dep = volt_dep_old;
+
+       step = volt_new - volt_old > 0 ? 1 : (-1);
+       step_dep = volt_dep_new - volt_dep_old > 0 ? 1 : (-1);
+
+       DVFS_DBG("step=%d step_dep=%d %d\n", step, step_dep, step * step_dep);
+
+       DVFS_DBG("Volt_new=%d(old=%d), volt_dep_new=%d(dep_old=%d)\n",
+                       volt_new, volt_old, volt_dep_new, volt_dep_old);
+       do {
+               volt_pre = volt;
+               volt_dep_pre = volt_dep;
+               if (step * step_dep < 0) {
+                       // target is between volt_old and volt_dep_old, just
+                       // need one step
+                       DVFS_DBG("step * step_dep < 0\n");
+                       volt = volt_new;
+                       volt_dep = volt_dep_new;
+
+               } else if (step > 0) {
+                       // up voltage
+                       DVFS_DBG("step > 0\n");
+                       volt_tmp = volt_dep + clk_biger_than_dep;
+                       volt_dep_tmp = volt + dep_biger_than_clk;
+
+                       volt = volt_tmp > volt_new ? volt_new : volt_tmp;
+                       volt_dep = volt_dep_tmp > volt_dep_new ? volt_dep_new : volt_dep_tmp;
+
+               } else if (step < 0) {
+                       // down voltage
+                       DVFS_DBG("step < 0\n");
+
+                       volt_tmp = volt_dep - dep_biger_than_clk;
+                       volt_dep_tmp = volt - clk_biger_than_dep;
+
+                       volt = volt_tmp < volt_new ? volt_new : volt_tmp;
+                       volt_dep = volt_dep_tmp < volt_dep_new ? volt_dep_new : volt_dep_tmp;
+
+               } else {
+                       DVFS_ERR("Oops, some bugs here:Volt_new=%d(old=%d), volt_dep_new=%d(dep_old=%d)\n",
+                                       volt_new, volt_old, volt_dep_new, volt_dep_old);
+                       goto fail;
+               }
+
+               if (vd_clk->cur_volt != volt) {
+                       DVFS_DBG("\t\t%s:%d->%d\n", vd_clk->name, vd_clk->cur_volt, volt);
+                       ret = dvfs_regulator_set_voltage_readback(regulator, volt, volt);
+                       //udelay(get_volt_up_delay(volt, volt_pre));
+                       dvfs_volt_up_delay(vd_clk,volt, volt_pre);
+                       if (ret < 0) {
+                               DVFS_ERR("%s %s set voltage up err ret = %d, Vnew = %d(was %d)mV\n",
+                                               __func__, vd_clk->name, ret, volt_new, volt_old);
+                               goto fail;
+                       }
+                       vd_clk->cur_volt = volt;
+               }
+               if (vd_dep->cur_volt != volt_dep) {
+                       DVFS_DBG("\t\t%s:%d->%d\n", vd_dep->name, vd_dep->cur_volt, volt_dep);
+                       ret = dvfs_regulator_set_voltage_readback(regulator_dep, volt_dep, volt_dep);
+                       //udelay(get_volt_up_delay(volt_dep, volt_dep_pre));
+                       dvfs_volt_up_delay(vd_dep,volt_dep, volt_dep_pre);
+                       if (ret < 0) {
+                               DVFS_ERR("depend %s %s set voltage up err ret = %d, Vnew = %d(was %d)mV\n",
+                                               __func__, vd_dep->name, ret, volt_dep_new, volt_dep_old);
+                               goto fail;
+                       }
+                       vd_dep->cur_volt = volt_dep;
+               }
+
+               DVFS_DBG("\t\tNOW:Volt=%d, volt_dep=%d\n", volt, volt_dep);
+
+       } while (volt != volt_new || volt_dep != volt_dep_new);
+
+       vd_clk->volt_set_flag = DVFS_SET_VOLT_SUCCESS;
+       vd_clk->cur_volt = volt_new;
+
+       return 0;
+fail:
+       DVFS_ERR("+++++++++++++++++FAIL AREA\n");
+       vd_clk->cur_volt = volt_old;
+       vd_dep->cur_volt = volt_dep_old;
+       vd_clk->volt_set_flag = DVFS_SET_VOLT_FAILURE;
+       ret = dvfs_regulator_set_voltage_readback(regulator, volt_old, volt_old);
+       if (ret < 0) {
+               vd_clk->volt_set_flag = DVFS_SET_VOLT_FAILURE;
+               DVFS_ERR("%s %s set callback voltage err ret = %d, Vnew = %d(was %d)mV\n",
+                               __func__, vd_clk->name, ret, volt_new, volt_old);
+       }
+
+       ret = dvfs_regulator_set_voltage_readback(regulator_dep, volt_dep_old, volt_dep_old);
+       if (ret < 0) {
+               vd_dep->volt_set_flag = DVFS_SET_VOLT_FAILURE;
+               DVFS_ERR("%s %s set callback voltage err ret = %d, Vnew = %d(was %d)mV\n",
+                               __func__, vd_dep->name, ret, volt_dep_new, volt_dep_old);
+       }
+
+       return -1;
+}
+
+int dvfs_scale_volt_direct(struct vd_node *vd_clk, int volt_new)
+{
+       int ret = 0;
+       
+       DVFS_DBG("ENTER %s, volt=%d(old=%d)\n", __func__, volt_new, vd_clk->cur_volt);
+       
+       if (IS_ERR_OR_NULL(vd_clk)) {
+               DVFS_ERR("%s vd_node error\n", __func__);
+               return -EINVAL;
+       }
+
+       if (!IS_ERR_OR_NULL(vd_clk->regulator)) {
+               ret = dvfs_regulator_set_voltage_readback(vd_clk->regulator, volt_new, volt_new);
+               //udelay(get_volt_up_delay(volt_new, vd_clk->cur_volt));
+               dvfs_volt_up_delay(vd_clk,volt_new, vd_clk->cur_volt);
+               if (ret < 0) {
+                       vd_clk->volt_set_flag = DVFS_SET_VOLT_FAILURE;
+                       DVFS_ERR("%s %s set voltage up err ret = %d, Vnew = %d(was %d)mV\n",
+                                       __func__, vd_clk->name, ret, volt_new, vd_clk->cur_volt);
+                       return -1;
+               }
+
+       } else {
+               DVFS_ERR("%s up volt clk_dvfs_node->vd->regulator == NULL\n", __func__);
+               return -1;
+       }
+
+       vd_clk->volt_set_flag = DVFS_SET_VOLT_SUCCESS;
+       vd_clk->cur_volt = volt_new;
+
+       return 0;
+
+}
+
+int dvfs_scale_volt_bystep(struct vd_node *vd_clk, struct vd_node *vd_dep, int volt_new, int volt_dep_new,
+               int cur_clk_biger_than_dep, int cur_dep_biger_than_clk, int new_clk_biger_than_dep, int new_dep_biger_than_clk)
+{
+
+       struct regulator *regulator, *regulator_dep;
+       int volt_new_corrected = 0, volt_dep_new_corrected = 0;
+       int volt_old = 0, volt_dep_old = 0;
+       int ret = 0;
+
+       volt_old = vd_clk->cur_volt;
+       volt_dep_old = vd_dep->cur_volt;
+
+       DVFS_DBG("ENTER %s, volt=%d(old=%d) vd_dep=%d(dep_old=%d)\n", __func__,
+                       volt_new, volt_old, volt_dep_new, volt_dep_old);
+       DVFS_DBG("ENTER %s, VOLT_DIFF: clk_cur=%d(clk_new=%d) dep_cur=%d(dep_new=%d)\n", __func__,
+                       cur_clk_biger_than_dep, new_clk_biger_than_dep, 
+                       cur_dep_biger_than_clk, new_dep_biger_than_clk);
+
+       volt_new_corrected = volt_new;
+       volt_dep_new_corrected = volt_dep_new;
+       correct_volt(&volt_new_corrected, &volt_dep_new_corrected, cur_clk_biger_than_dep, cur_dep_biger_than_clk);
+       ret = dvfs_scale_volt(vd_clk, vd_dep, volt_old, volt_new_corrected, volt_dep_old, volt_dep_new_corrected,
+                       cur_clk_biger_than_dep, cur_dep_biger_than_clk);
+       if (ret < 0) {
+               vd_clk->volt_set_flag = DVFS_SET_VOLT_FAILURE;
+               DVFS_ERR("set volt error\n");
+               return -1;
+       }
+
+       if (cur_clk_biger_than_dep != new_clk_biger_than_dep || cur_dep_biger_than_clk != new_dep_biger_than_clk) {
+               regulator = vd_clk->regulator;
+               regulator_dep = vd_dep->regulator;
+
+               volt_new_corrected = volt_new;
+               volt_dep_new_corrected = volt_dep_new;
+               correct_volt(&volt_new_corrected, &volt_dep_new_corrected, new_clk_biger_than_dep, new_dep_biger_than_clk);
+
+               if (vd_clk->cur_volt != volt_new_corrected) {
+                       DVFS_DBG("%s:%d->%d\n", vd_clk->name, vd_clk->cur_volt, volt_new_corrected);
+                       ret = dvfs_regulator_set_voltage_readback(regulator, volt_new_corrected, volt_new_corrected);
+                       //udelay(get_volt_up_delay(volt_new_corrected, vd_clk->cur_volt));
+                       dvfs_volt_up_delay(vd_clk,volt_new_corrected, vd_clk->cur_volt);
+                       if (ret < 0) {
+                               DVFS_ERR("%s %s set voltage up err ret = %d, Vnew = %d(was %d)mV\n",
+                                               __func__, vd_clk->name, ret, volt_new_corrected, vd_clk->cur_volt);
+                               return -1;
+                       }
+                       vd_clk->cur_volt = volt_new_corrected;
+               }
+               if (vd_dep->cur_volt != volt_dep_new_corrected) {
+                       DVFS_DBG("%s:%d->%d\n", vd_clk->name, vd_clk->cur_volt, volt_dep_new_corrected);
+                       ret = dvfs_regulator_set_voltage_readback(regulator_dep, volt_dep_new_corrected, volt_dep_new_corrected);
+                       //udelay(get_volt_up_delay(volt_dep_new_corrected, vd_dep->cur_volt));
+                       dvfs_volt_up_delay(vd_dep,volt_dep_new_corrected, vd_dep->cur_volt);
+                       if (ret < 0) {
+                               DVFS_ERR("depend %s %s set voltage up err ret = %d, Vnew = %d(was %d)mV\n",
+                                               __func__, vd_dep->name, ret, volt_dep_new_corrected, vd_dep->cur_volt);
+                               return -1;
+                       }
+                       vd_dep->cur_volt = volt_dep_new_corrected;
+               }
+       }
+
+       return 0;
+}
+
+int dvfs_reset_volt(struct vd_node *dvfs_vd)
+{
+       int flag_set_volt_correct = 0;
+       if (!IS_ERR_OR_NULL(dvfs_vd->regulator))
+               flag_set_volt_correct = dvfs_regulator_get_voltage(dvfs_vd->regulator);
+       else {
+               DVFS_ERR("dvfs regulator is ERROR\n");
+               return -1;
+       }
+       if (flag_set_volt_correct <= 0) {
+               DVFS_ERR("%s (vd:%s), try to reload volt ,by it is error again(%d)!!! stop scaling\n",
+                               __func__, dvfs_vd->name, flag_set_volt_correct);
+               return -1;
+       }
+       dvfs_vd->volt_set_flag = DVFS_SET_VOLT_SUCCESS;
+       DVFS_WARNING("%s (vd:%s), try to reload volt = %d\n",
+                       __func__, dvfs_vd->name, flag_set_volt_correct);
+
+       /* Reset vd's voltage */
+       dvfs_vd->cur_volt = flag_set_volt_correct;
+
+       return dvfs_vd->cur_volt;
+}
+
+
+/*********************************************************************************/
+int dvfs_target(struct clk_hw *hw, unsigned long rate,
+                               unsigned long parent_rate)
+{
+       struct cpufreq_frequency_table clk_fv;
+       const struct clk_ops *origin_clk_ops;
+       struct clk *clk = hw->clk;
+       struct dvfs_node *clk_dvfs_node;
+       unsigned long old_rate = 0, volt_new = 0, clk_volt_store = 0;
+       int ret = 0;
+
+       if (!clk) {
+               DVFS_ERR("%s is not a clk\n", __func__);
+               return -EINVAL;
+       }
+
+       old_rate = clk_get_rate(clk);
+       if (rate == old_rate)
+               return 0;
+       
+       clk_dvfs_node = clk_get_dvfs_info(clk);
+       if (IS_ERR_OR_NULL(clk_dvfs_node))
+               return PTR_ERR(clk_dvfs_node);
+       
+       DVFS_DBG("enter %s: clk(%s) new_rate = %lu Hz, old_rate =  %lu Hz\n", 
+               __func__, clk_dvfs_node->name, rate, old_rate);
+
+       if (!clk_dvfs_node->enable_dvfs){
+               DVFS_WARNING("dvfs(%s) is disable\n", clk_dvfs_node->name);
+               return 0;
+       }
+
+       origin_clk_ops = clk_dvfs_node->origin_clk_ops;
+       
+       if (clk_dvfs_node->vd->volt_set_flag == DVFS_SET_VOLT_FAILURE) {
+               /* It means the last time set voltage error */
+               ret = dvfs_reset_volt(clk_dvfs_node->vd);
+               if (ret < 0) {
+                       return -EAGAIN;
+               }
+       }
+
+       /* find the clk corresponding voltage */
+       ret = clk_dvfs_node_get_ref_volt(clk_dvfs_node, rate / 1000, &clk_fv);
+       if (ret) 
+               return ret;
+       clk_volt_store = clk_dvfs_node->set_volt;
+       clk_dvfs_node->set_volt = clk_fv.index;
+       volt_new = dvfs_vd_get_newvolt_byclk(clk_dvfs_node);
+       DVFS_DBG("%s %s new rate=%lu(was=%lu),new volt=%lu,(was=%d)\n",
+               __func__, clk_dvfs_node->name, rate, old_rate, volt_new,clk_dvfs_node->vd->cur_volt);
+
+       /* if up the rate */
+       if (rate > old_rate) {
+               ret = dvfs_scale_volt_direct(clk_dvfs_node->vd, volt_new);
+               if (ret)
+                       goto fail_roll_back;
+       }
+
+       /* scale rate */
+       if (clk_dvfs_node->clk_dvfs_target) {
+               ret = clk_dvfs_node->clk_dvfs_target(hw, rate, origin_clk_ops->set_rate);
+       } else {
+               ret = origin_clk_ops->set_rate(hw, rate, clk->parent->rate);
+       }
+
+       if (ret) {
+               DVFS_ERR("%s set rate err\n", __func__);
+               goto fail_roll_back;
+       }
+       clk_dvfs_node->set_freq = rate / 1000;
+
+       DVFS_DBG("dvfs %s set rate %lu ok\n", clk_dvfs_node->name, clk_get_rate(clk));
+
+       /* if down the rate */
+       if (rate < old_rate) {
+               ret = dvfs_scale_volt_direct(clk_dvfs_node->vd, volt_new);
+               if (ret)
+                       goto out;
+       }
+
+       return 0;
+fail_roll_back:
+       clk_dvfs_node->set_volt = clk_volt_store;
+out:
+       return ret;
+}
+
+
+static long _clk_dvfs_node_round_rate(struct clk_hw *hw, unsigned long rate,
+                               unsigned long *prate)
+{
+       struct clk *clk = hw->clk;
+       struct dvfs_node *dvfs_node = clk_get_dvfs_info(clk);
+
+       if (rate < dvfs_node->min_rate) {
+               rate = dvfs_node->min_rate;
+       } else if (rate > dvfs_node->max_rate) {
+               rate = dvfs_node->max_rate;
+       }
+
+       return dvfs_node->origin_clk_ops->round_rate(hw, rate, prate);
+}
+
+static int _clk_dvfs_node_enable(struct clk_hw *hw)
+{
+       struct clk *clk = hw->clk;
+       struct dvfs_node *dvfs_node = clk_get_dvfs_info(clk);   
+
+       return dvfs_node->origin_clk_ops->enable(hw);
+}
+
+
+static void _clk_dvfs_node_disable(struct clk_hw *hw)
+{
+       struct clk *clk = hw->clk;
+       struct dvfs_node *dvfs_node = clk_get_dvfs_info(clk);   
+
+       dvfs_node->origin_clk_ops->disable(hw); 
+}
+
+
+
+static int _clk_register_dvfs(struct clk *clk, struct dvfs_node *dvfs_node)
+{
+       struct clk_ops  *ops;
+
+       if (IS_ERR(clk) || IS_ERR(dvfs_node))
+               return -EINVAL;
+
+       //mutex_lock(&mutex);
+
+       dvfs_node->clk = clk;
+       dvfs_node->origin_clk_ops = clk->ops;
+
+       ops = kzalloc(sizeof(struct clk_ops), GFP_KERNEL);
+       if (!ops)
+               return -ENOMEM;
+       *ops = *(clk->ops);
+       ops->set_rate = dvfs_vd_clk_set_rate;
+       ops->round_rate = _clk_dvfs_node_round_rate;
+       ops->enable = _clk_dvfs_node_enable;
+       ops->disable = _clk_dvfs_node_disable;
+       
+       clk->ops = ops;
+       clk->private_data = dvfs_node;
+
+       //mutex_unlock(&mutex);
+       
+       return 0;
+}
+
+static int _rk_convert_cpufreq_table(struct dvfs_node *dvfs_node)
+{
+       struct opp *opp;
+       struct device *dev;
+       struct cpufreq_frequency_table *table;
+       int i;
+
+       table = dvfs_node->dvfs_table;
+       dev = &dvfs_node->dev;
+
+       for (i = 0; table[i].frequency!= CPUFREQ_TABLE_END; i++){
+               opp = opp_find_freq_exact(dev, table[i].frequency * 1000, true);
+               if (IS_ERR(opp))
+                       return PTR_ERR(opp);
+               table[i].index = opp_get_voltage(opp);
+       }
+       return 0;
+}
+
+
+
+int of_dvfs_init(void)
+{
+       struct vd_node *vd;
+       struct pd_node *pd;
+       struct device_node *dvfs_dev_node, *clk_dev_node, *vd_dev_node, *pd_dev_node;
+       struct dvfs_node *dvfs_node;
+       struct clk *clk;
+       int ret;
+
+       printk("%s\n", __func__);
+
+       dvfs_dev_node = of_find_node_by_name(NULL, "dvfs");
+       if (IS_ERR_OR_NULL(dvfs_dev_node)) {
+               DVFS_ERR("%s get dvfs dev node err\n", __func__);
+               return PTR_ERR(dvfs_dev_node);
+       }
+
+       for_each_available_child_of_node(dvfs_dev_node, vd_dev_node) {
+               vd = kzalloc(sizeof(struct vd_node), GFP_KERNEL);
+               if (!vd)
+                       return -ENOMEM;
+
+               vd->name = vd_dev_node->name;
+               ret = of_property_read_string(vd_dev_node, "regulator_name", &vd->regulator_name);
+               if (ret) {
+                       DVFS_ERR("%s vd(%s) get regulator_name err, ret:%d\n", 
+                               __func__, vd_dev_node->name, ret);
+                       kfree(vd);
+                       continue;
+               }
+               
+               /*vd->regulator = regulator_get(NULL, vd->regulator_name);
+               if (IS_ERR(vd->regulator)){
+                       DVFS_ERR("%s vd(%s) get regulator(%s) failed!\n", __func__, vd->name, vd->regulator_name);
+                       kfree(vd);
+                       continue;
+               }*/
+
+               vd->suspend_volt = 0;
+               
+               vd->volt_set_flag = DVFS_SET_VOLT_FAILURE;
+               vd->vd_dvfs_target = dvfs_target;
+               ret = rk_regist_vd(vd);
+               if (ret){
+                       DVFS_ERR("%s vd(%s) register err:%d\n", __func__, vd->name, ret);
+                       kfree(vd);
+                       continue;
+               }
+
+               DVFS_DBG("%s vd(%s) register ok, regulator name:%s,suspend volt:%d\n", 
+                       __func__, vd->name, vd->regulator_name, vd->suspend_volt);
+               
+               for_each_available_child_of_node(vd_dev_node, pd_dev_node) {            
+                       pd = kzalloc(sizeof(struct pd_node), GFP_KERNEL);
+                       if (!pd)
+                               return -ENOMEM;
+
+                       pd->vd = vd;
+                       pd->name = pd_dev_node->name;
+                       
+                       ret = rk_regist_pd(pd);
+                       if (ret){
+                               DVFS_ERR("%s pd(%s) register err:%d\n", __func__, pd->name, ret);
+                               kfree(pd);
+                               continue;
+                       }
+                       DVFS_DBG("%s pd(%s) register ok, parent vd:%s\n", 
+                               __func__, pd->name, vd->name);                  
+                       for_each_available_child_of_node(pd_dev_node, clk_dev_node) {
+                               dvfs_node = kzalloc(sizeof(struct dvfs_node), GFP_KERNEL);
+                               if (!dvfs_node)
+                                       return -ENOMEM;
+                               
+                               dvfs_node->name = clk_dev_node->name;
+                               dvfs_node->pd = pd;
+                               dvfs_node->vd = vd;
+                               dvfs_node->dev.of_node = clk_dev_node;
+                               ret = of_init_opp_table(&dvfs_node->dev);
+                               if (ret) {
+                                       DVFS_ERR("%s clk(%s) get opp table err:%d\n", __func__, dvfs_node->name, ret);
+                                       kfree(dvfs_node);
+                                       continue;
+                               }
+                               
+                               ret = opp_init_cpufreq_table(&dvfs_node->dev, &dvfs_node->dvfs_table);
+                               if (ret) {
+                                       DVFS_ERR("%s clk(%s) get cpufreq table err:%d\n", __func__, dvfs_node->name, ret);
+                                       kfree(dvfs_node);
+                                       continue;
+                               }
+                               ret = _rk_convert_cpufreq_table(dvfs_node);
+                               if (ret) {
+                                       kfree(dvfs_node);
+                                       continue;
+                               }
+                               
+                               clk = clk_get(NULL, clk_dev_node->name);
+                               if (IS_ERR(clk)){
+                                       DVFS_ERR("%s get clk(%s) err:%ld\n", __func__, dvfs_node->name, PTR_ERR(clk));
+                                       kfree(dvfs_node);
+                                       continue;
+                                       
+                               }
+
+                               _clk_register_dvfs(clk, dvfs_node);
+                               clk_put(clk);
+                               
+                               ret = rk_regist_clk(dvfs_node);
+                               if (ret){
+                                       DVFS_ERR("%s dvfs_node(%s) register err:%d\n", __func__, dvfs_node->name, ret);
+                                       return ret;
+                               }
+
+                               DVFS_DBG("%s dvfs_node(%s) register ok, parent pd:%s\n", 
+                                       __func__, clk_dev_node->name, pd->name);        
+
+                       }
+               }       
+       }
+       return 0;
+}
+
+/*********************************************************************************/
+
+/**
+ * dump_dbg_map() : Draw all informations of dvfs while debug
+ */
+int dump_dbg_map(char *buf)
+{
+       int i;
+       struct vd_node  *vd;
+       struct pd_node  *pd; //*clkparent;
+       struct clk_list *child;
+       struct dvfs_node        *clk_dvfs_node;
+       //struct depend_list *depend;
+       char *s = buf;
+       mutex_lock(&rk_dvfs_mutex);
+
+       printk( "-------------DVFS TREE-----------\n\n\n");
+       printk( "DVFS TREE:\n");
+       list_for_each_entry(vd, &rk_dvfs_tree, node) {
+               printk( "|\n|- voltage domain:%s\n", vd->name);
+               printk( "|- current voltage:%d\n", vd->cur_volt);
+               //list_for_each_entry(depend, &vd->req_volt_list, node2vd) {
+               //      printk( "|- request voltage:%d, clk:%s\n", depend->req_volt, depend->clk_dvfs_node->name);
+               //}
+
+               list_for_each_entry(pd, &vd->pd_list, node) {
+                       printk( "|  |\n|  |- power domain:%s, status = %s, current volt = %d\n",
+                                       pd->name, (pd->pd_status == PD_ON) ? "ON" : "OFF", pd->cur_volt);
+
+                       list_for_each_entry(child, &pd->clk_list, node) {
+                               clk_dvfs_node = child->clk_dvfs_node;
+                               printk( "|  |  |\n|  |  |- clock: %s current: rate %d, volt = %d,"
+                                               " enable_dvfs = %s\n",
+                                               clk_dvfs_node->name, clk_dvfs_node->set_freq, clk_dvfs_node->set_volt,
+                                               clk_dvfs_node->enable_dvfs == 0 ? "DISABLE" : "ENABLE");
+                               printk( "|  |  |- clk limit:[%u, %u]; last set rate = %u\n",
+                                               clk_dvfs_node->min_rate, clk_dvfs_node->max_rate,
+                                               clk_dvfs_node->last_set_rate);
+
+                               /*for (i = 0; clk_dvfs_node->pds[i].pd != NULL; i++) {
+                                       clkparent = clk_dvfs_node->pds[i].pd;
+                                       printk( "|  |  |  |- clock parents: %s, vd_parent = %s\n",
+                                                       clkparent->name, clkparent->vd->name);
+                               }*/
+
+                               for (i = 0; (clk_dvfs_node->dvfs_table[i].frequency != CPUFREQ_TABLE_END); i++) {
+                                       printk( "|  |  |  |- freq = %d, volt = %d\n",
+                                                       clk_dvfs_node->dvfs_table[i].frequency,
+                                                       clk_dvfs_node->dvfs_table[i].index);
+
+                               }
+                       }
+               }
+       }
+       printk( "-------------DVFS TREE END------------\n");
+       
+       mutex_unlock(&rk_dvfs_mutex);
+       return s - buf;
+}
+/*******************************AVS AREA****************************************/
+/*
+ * To use AVS function, you must call avs_init in machine_rk30_board_init(void)(board-rk30-sdk.c)
+ * And then call(vdd_log):
+ *     regulator_set_voltage(dcdc, 1100000, 1100000);
+ *     avs_init_val_get(1,1100000,"wm8326 init");
+ *     udelay(600);
+ *     avs_set_scal_val(AVS_BASE);
+ * in wm831x_post_init(board-rk30-sdk-wm8326.c)
+ * AVS_BASE can use 172
+ */
+
+static struct avs_ctr_st *avs_ctr_data=NULL;
+#define init_avs_times 10
+#define init_avs_st_num 5
+
+struct init_avs_st {
+       int is_set;
+       u8 paramet[init_avs_times];
+       int vol;
+       char *s;
+};
+
+static struct init_avs_st init_avs_paramet[init_avs_st_num];
+
+void avs_board_init(struct avs_ctr_st *data)
+{
+       
+       avs_ctr_data=data;
+}
+void avs_init(void)
+{
+       memset(&init_avs_paramet[0].is_set, 0, sizeof(init_avs_paramet));
+       if(avs_ctr_data&&avs_ctr_data->avs_init)
+               avs_ctr_data->avs_init();
+       avs_init_val_get(0, 1200000,"board_init");
+}
+static u8 rk_get_avs_val(void)
+{
+       
+       if(avs_ctr_data&&avs_ctr_data->avs_get_val)
+       {       
+               return avs_ctr_data->avs_get_val();
+       }
+       return 0;
+
+}
+/******************************int val get**************************************/
+void avs_init_val_get(int index, int vol, char *s)
+{
+       int i;
+       if(index >= init_avs_times)
+               return;
+       init_avs_paramet[index].vol = vol;
+       init_avs_paramet[index].s = s;
+       init_avs_paramet[index].is_set++;
+       printk("DVFS MSG:\tAVS Value(index=%d): ", index);
+       for(i = 0; i < init_avs_times; i++) {
+               init_avs_paramet[index].paramet[i] = rk_get_avs_val();
+               mdelay(1);
+               printk("%d ", init_avs_paramet[index].paramet[i]);
+       }
+       printk("\n");
+}
+int avs_set_scal_val(u8 avs_base)
+{
+       return 0;
+}
+
+/*************************interface to get avs value and dvfs tree*************************/
+#define USE_NORMAL_TIME
+#ifdef USE_NORMAL_TIME
+static struct timer_list avs_timer;
+#else
+static struct hrtimer dvfs_hrtimer;
+#endif
+
+static u32 avs_dyn_start = 0;
+static u32 avs_dyn_data_cnt;
+static u8 *avs_dyn_data = NULL;
+static u32 show_line_cnt = 0;
+static u8 dly_min;
+static u8 dly_max;
+
+#define val_per_line (30)
+#define line_pre_show (30)
+#define avs_dyn_data_num (3*1000*1000)
+
+static u32 print_avs_init(char *buf)
+{
+       char *s = buf;
+       int i, j;
+
+       for(j = 0; j < init_avs_st_num; j++) {
+               if(init_avs_paramet[j].vol <= 0)
+                       continue;
+               s += sprintf(s, "%s ,vol=%d,paramet following\n",
+                               init_avs_paramet[j].s, init_avs_paramet[j].vol);
+               for(i = 0; i < init_avs_times; i++) {
+                       s += sprintf(s, "%d ", init_avs_paramet[j].paramet[i]);
+               }
+
+               s += sprintf(s, "\n");
+       }
+       return (s - buf);
+}
+
+static ssize_t avs_init_show(struct kobject *kobj, struct kobj_attribute *attr,
+               char *buf)
+{
+       return print_avs_init(buf);
+}
+
+static ssize_t avs_init_store(struct kobject *kobj, struct kobj_attribute *attr,
+               const char *buf, size_t n)
+{
+
+       return n;
+}
+static ssize_t avs_now_show(struct kobject *kobj, struct kobj_attribute *attr,
+               char *buf)
+{
+       return sprintf(buf, "%d\n", rk_get_avs_val());
+}
+
+static ssize_t avs_now_store(struct kobject *kobj, struct kobj_attribute *attr,
+               const char *buf, size_t n)
+{
+       return n;
+}
+static ssize_t avs_dyn_show(struct kobject *kobj, struct kobj_attribute *attr,
+               char *buf)
+{
+       char *s = buf;
+       u32 i;
+
+       if(avs_dyn_data == NULL)
+               return (s - buf);
+
+       if(avs_dyn_start) {
+               int start_cnt;
+               int end_cnt;
+               end_cnt = (avs_dyn_data_cnt ? (avs_dyn_data_cnt - 1) : 0);
+               if(end_cnt > (line_pre_show * val_per_line))
+                       start_cnt = end_cnt - (line_pre_show * val_per_line);
+               else
+                       start_cnt = 0;
+
+               dly_min = avs_dyn_data[start_cnt];
+               dly_max = avs_dyn_data[start_cnt];
+
+               //s += sprintf(s,"data start=%d\n",i);
+               for(i = start_cnt; i <= end_cnt;) {
+                       s += sprintf(s, "%d", avs_dyn_data[i]);
+                       dly_min = min(dly_min, avs_dyn_data[i]);
+                       dly_max = max(dly_max, avs_dyn_data[i]);
+                       i++;
+                       if(!(i % val_per_line)) {
+                               s += sprintf(s, "\n");
+                       } else
+                               s += sprintf(s, " ");
+               }
+
+               s += sprintf(s, "\n");
+
+               s += sprintf(s, "new data is from=%d to %d\n", start_cnt, end_cnt);
+               //s += sprintf(s,"\n max=%d,min=%d,totolcnt=%d,line=%d\n",dly_max,dly_min,avs_dyn_data_cnt,show_line_cnt);
+
+
+       } else {
+               if(show_line_cnt == 0) {
+                       dly_min = avs_dyn_data[0];
+                       dly_max = avs_dyn_data[0];
+               }
+
+
+               for(i = show_line_cnt * (line_pre_show * val_per_line); i < avs_dyn_data_cnt;) {
+                       s += sprintf(s, "%d", avs_dyn_data[i]);
+                       dly_min = min(dly_min, avs_dyn_data[i]);
+                       dly_max = max(dly_max, avs_dyn_data[i]);
+                       i++;
+                       if(!(i % val_per_line)) {
+                               s += sprintf(s, "\n");
+                       } else
+                               s += sprintf(s, " ");
+                       if(i >= ((show_line_cnt + 1)*line_pre_show * val_per_line))
+                               break;
+               }
+
+               s += sprintf(s, "\n");
+
+               s += sprintf(s, "max=%d,min=%d,totolcnt=%d,line=%d\n",
+                               dly_max, dly_min, avs_dyn_data_cnt, show_line_cnt);
+               show_line_cnt++;
+               if(((show_line_cnt * line_pre_show)*val_per_line) >= avs_dyn_data_cnt) {
+
+                       show_line_cnt = 0;
+
+                       s += sprintf(s, "data is over\n");
+               }
+       }
+       return (s - buf);
+}
+
+static ssize_t avs_dyn_store(struct kobject *kobj, struct kobj_attribute *attr,
+               const char *buf, size_t n)
+{
+       const char *pbuf;
+
+       if((strncmp(buf, "start", strlen("start")) == 0)) {
+               if(avs_dyn_data == NULL)
+                       avs_dyn_data = kmalloc(avs_dyn_data_num, GFP_KERNEL);
+               if(avs_dyn_data == NULL)
+                       return n;
+
+               pbuf = &buf[strlen("start")];
+               avs_dyn_data_cnt = 0;
+               show_line_cnt = 0;
+               if(avs_dyn_data) {
+#ifdef USE_NORMAL_TIME
+                       mod_timer(&avs_timer, jiffies + msecs_to_jiffies(5));
+#else
+                       hrtimer_start(&dvfs_hrtimer, ktime_set(0, 5 * 1000 * 1000), HRTIMER_MODE_REL);
+#endif
+                       avs_dyn_start = 1;
+               }
+               //sscanf(pbuf, "%d %d", &number, &voltage);
+               //DVFS_DBG("---------ldo %d %d\n", number, voltage);
+
+       } else if((strncmp(buf, "stop", strlen("stop")) == 0)) {
+               pbuf = &buf[strlen("stop")];
+               avs_dyn_start = 0;
+               show_line_cnt = 0;
+               //sscanf(pbuf, "%d %d", &number, &voltage);
+               //DVFS_DBG("---------dcdc %d %d\n", number, voltage);
+       }
+
+
+
+       return n;
+}
+
+static ssize_t dvfs_tree_store(struct kobject *kobj, struct kobj_attribute *attr,
+               const char *buf, size_t n)
+{
+       return n;
+}
+static ssize_t dvfs_tree_show(struct kobject *kobj, struct kobj_attribute *attr,
+               char *buf)
+{
+       return dump_dbg_map(buf);
+
+}
+
+static void avs_timer_fn(unsigned long data)
+{
+       int i;
+       for(i = 0; i < 1; i++) {
+               if(avs_dyn_data_cnt >= avs_dyn_data_num)
+                       return;
+               avs_dyn_data[avs_dyn_data_cnt] = rk_get_avs_val();
+               avs_dyn_data_cnt++;
+       }
+       if(avs_dyn_start)
+               mod_timer(&avs_timer, jiffies + msecs_to_jiffies(10));
+}
+#if 0
+struct hrtimer dvfs_hrtimer;
+static enum hrtimer_restart dvfs_hrtimer_timer_func(struct hrtimer *timer)
+{
+       int i;
+       for(i = 0; i < 1; i++) {
+               if(avs_dyn_data_cnt >= avs_dyn_data_num)
+                       return HRTIMER_NORESTART;
+               avs_dyn_data[avs_dyn_data_cnt] = rk_get_avs_val();
+               avs_dyn_data_cnt++;
+       }
+       if(avs_dyn_start)
+               hrtimer_start(timer, ktime_set(0, 1 * 1000 * 1000), HRTIMER_MODE_REL);
+
+}
+#endif
+
+/*********************************************************************************/
+static struct kobject *dvfs_kobj;
+struct dvfs_attribute {
+       struct attribute        attr;
+       ssize_t (*show)(struct kobject *kobj, struct kobj_attribute *attr,
+                       char *buf);
+       ssize_t (*store)(struct kobject *kobj, struct kobj_attribute *attr,
+                       const char *buf, size_t n);
+};
+
+static struct dvfs_attribute dvfs_attrs[] = {
+       /*     node_name        permision               show_func       store_func */
+//#ifdef CONFIG_RK_CLOCK_PROC
+       __ATTR(dvfs_tree,       S_IRUSR | S_IRGRP | S_IWUSR,    dvfs_tree_show, dvfs_tree_store),
+       __ATTR(avs_init,        S_IRUSR | S_IRGRP | S_IWUSR,    avs_init_show,  avs_init_store),
+       __ATTR(avs_dyn,         S_IRUSR | S_IRGRP | S_IWUSR,    avs_dyn_show,   avs_dyn_store),
+       __ATTR(avs_now,         S_IRUSR | S_IRGRP | S_IWUSR,    avs_now_show,   avs_now_store),
+//#endif
+};
+
+
+static int __init dvfs_init(void)
+{
+       int i, ret = 0;
+#ifdef USE_NORMAL_TIME
+       init_timer(&avs_timer);
+       //avs_timer.expires = jiffies+msecs_to_jiffies(1);
+       avs_timer.function = avs_timer_fn;
+       //mod_timer(&avs_timer,jiffies+msecs_to_jiffies(1));
+#else
+       hrtimer_init(&dvfs_hrtimer, CLOCK_MONOTONIC, HRTIMER_MODE_REL);
+       dvfs_hrtimer.function = dvfs_hrtimer_timer_func;
+       //hrtimer_start(&dvfs_hrtimer,ktime_set(0, 5*1000*1000),HRTIMER_MODE_REL);
+#endif
+
+       dvfs_kobj = kobject_create_and_add("dvfs", NULL);
+       if (!dvfs_kobj)
+               return -ENOMEM;
+       for (i = 0; i < ARRAY_SIZE(dvfs_attrs); i++) {
+               ret = sysfs_create_file(dvfs_kobj, &dvfs_attrs[i].attr);
+               if (ret != 0) {
+                       DVFS_ERR("create index %d error\n", i);
+                       return ret;
+               }
+       }
+
+       return ret;
+}
+
+subsys_initcall(dvfs_init);
diff --git a/arch/arm/mach-rockchip/dvfs.h b/arch/arm/mach-rockchip/dvfs.h
new file mode 100644 (file)
index 0000000..0af28cc
--- /dev/null
@@ -0,0 +1,231 @@
+/* arch/arm/mach-rk30/rk30_dvfs.h
+ *
+ * Copyright (C) 2012 ROCKCHIP, Inc.
+ *
+ * This software is licensed under the terms of the GNU General Public
+ * License version 2, as published by the Free Software Foundation, and
+ * may be copied, distributed, and modified under those terms.
+ *
+ * This program is distributed in the hope that 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.
+ *
+ */
+#ifndef _RK30_DVFS_H_
+#define _RK30_DVFS_H_
+
+#include <linux/device.h>
+#include <linux/clk-private.h>
+
+typedef int (*dvfs_set_rate_callback)(struct clk_hw *hw, unsigned long rate,
+                               unsigned long parent_rate);
+typedef int (*clk_dvfs_target_callback)(struct clk_hw *hw, unsigned long rate,
+                                        dvfs_set_rate_callback set_rate);
+typedef int (*clk_dvfs_node_disable_callback)(struct clk *clk,int on);
+
+/**
+ * struct vd_node:     To Store All Voltage Domains' info
+ * @name:              Voltage Domain's Name
+ * @regulator_name:    Voltage domain's regulator name
+ * @cur_volt:          Voltage Domain's Current Voltage
+ * @regulator:         Voltage Domain's regulator point
+ * @node:              Point of he Voltage Domain List Node
+ * @pd_list:           Head of Power Domain List Belongs to This Voltage Domain
+ * @req_volt_list:     The list of clocks requests
+ * @dvfs_mutex:                Lock
+ * @vd_dvfs_target:    Callback function       
+ */
+ #define VD_VOL_LIST_CNT (200)
+ #define VD_LIST_RELATION_L 0
+ #define VD_LIST_RELATION_H 1
+
+struct vd_node {
+       const char              *name;
+       const char              *regulator_name;
+       int                     volt_time_flag;// =0 ,is no initing checking ,>0 ,support,<0 not support
+       int                     mode_flag;// =0 ,is no initing checking ,>0 ,support,<0 not support
+       int                     cur_volt;
+       int                     volt_set_flag;
+       int                     suspend_volt;
+       struct regulator        *regulator;
+       struct list_head        node;
+       struct list_head        pd_list;
+       struct list_head        req_volt_list;
+       //struct mutex          dvfs_mutex;
+       clk_dvfs_node_disable_callback  vd_clk_disable_target;
+       dvfs_set_rate_callback      vd_dvfs_target;
+       unsigned n_voltages;
+       int volt_list[VD_VOL_LIST_CNT];
+};
+
+/**
+ * struct pd_node:     To Store All Power Domains' info
+ * @name:              Power Domain's Name
+ * @cur_volt:          Power Domain's Current Voltage
+ * @pd_status:         Power Domain's status
+ * @vd:                        Voltage Domain the power domain belongs to
+ * @pd_clk:            Look power domain as a clock
+ * @node:              List node to Voltage Domain
+ * @clk_list:          Head of Power Domain's Clocks List
+ */
+struct pd_node {
+       const char              *name;
+       int                     cur_volt;
+       unsigned char           pd_status;
+       struct vd_node          *vd;
+       struct list_head        node;
+       struct list_head        clk_list;
+};
+
+struct clk_list{
+       struct dvfs_node                *clk_dvfs_node;
+       struct list_head        node;
+};
+/**
+ * struct dvfs_node:   To Store All dvfs clocks' info
+ * @name:              Dvfs clock's Name
+ * @set_freq:          Dvfs clock's Current Frequency
+ * @set_volt:          Dvfs clock's Current Voltage
+ * @enable_dvfs:       Sign if DVFS clock enable
+ * @clk:               System clk's point
+ * @pd:                        Power Domains dvfs clock belongs to
+ * @vd:                        Voltage Domains dvfs clock belongs to
+ * @dvfs_nb:           Notify list
+ * @dvfs_table:                Frequency and voltage table for dvfs
+ * @clk_dvfs_target:   Callback function
+ */
+struct dvfs_node {
+       struct device           dev;            //for opp
+       const char              *name;
+       int                     set_freq;       //KHZ
+       int                     set_volt;       //MV
+       int                     enable_dvfs;
+       int                     freq_limit_en;  //sign if use limit frequency
+       unsigned int            min_rate;       //limit min frequency
+       unsigned int            max_rate;       //limit max frequency
+       unsigned int            last_set_rate;
+       struct clk              *clk;
+       struct pd_node          *pd;
+       struct vd_node          *vd;
+       struct clk_list         clk_list;
+       struct notifier_block   *dvfs_nb;
+       struct cpufreq_frequency_table  *dvfs_table;
+       struct clk_disable_ctr          *disable_ctr;
+       const struct clk_ops            *origin_clk_ops;
+       clk_dvfs_target_callback        clk_dvfs_target;
+};
+
+#define DVFS_MHZ (1000*1000)
+#define DVFS_KHZ (1000)
+
+#define DVFS_V (1000*1000)
+#define DVFS_MV (1000)
+#if 0
+#define DVFS_DBG(fmt, args...) printk(KERN_DEBUG "DVFS DBG:\t"fmt, ##args)
+#else
+#define DVFS_DBG(fmt, args...) {while(0);}
+#endif
+
+#define DVFS_ERR(fmt, args...) printk(KERN_ERR "DVFS ERR:\t"fmt, ##args)
+#define DVFS_LOG(fmt, args...) printk(KERN_DEBUG "DVFS LOG:\t"fmt, ##args)
+#define DVFS_WARNING(fmt, args...) printk(KERN_WARNING "DVFS WARNING:\t"fmt, ##args)
+
+
+
+#define DVFS_SET_VOLT_FAILURE  1
+#define DVFS_SET_VOLT_SUCCESS  0
+
+
+#define dvfs_regulator_get(dev,id) regulator_get((dev),(id))
+#define dvfs_regulator_put(regu) regulator_put((regu))
+#define dvfs_regulator_set_voltage(regu,min_uV,max_uV) regulator_set_voltage((regu),(min_uV),(max_uV))
+#define dvfs_regulator_get_voltage(regu) regulator_get_voltage((regu))
+#define dvfs_regulator_set_voltage_time(regu, old_uV, new_uV) regulator_set_voltage_time((regu), (old_uV), (new_uV))
+#define dvfs_regulator_set_mode(regu, mode) regulator_set_mode((regu), (mode))
+#define dvfs_regulator_get_mode(regu) regulator_get_mode((regu))
+#define dvfs_regulator_list_voltage(regu,selector) regulator_list_voltage((regu),(selector))
+#define dvfs_regulator_count_voltages(regu) regulator_count_voltages((regu))
+
+#define clk_dvfs_node_get(a,b) clk_get((a),(b))
+#define clk_dvfs_node_get_rate_kz(a) (clk_get_rate((a))/1000)
+#define clk_dvfs_node_set_rate(a,b) clk_set_rate((a),(b))
+
+
+
+
+typedef void (*avs_init_fn)(void);
+typedef u8 (*avs_get_val_fn)(void);
+struct avs_ctr_st {
+       avs_init_fn             avs_init;
+       avs_get_val_fn          avs_get_val;
+};
+
+#ifdef CONFIG_DVFS
+int of_dvfs_init(void);
+int clk_dvfs_node_get_ref_volt(struct dvfs_node *clk_dvfs_node, int rate_khz,
+               struct cpufreq_frequency_table *clk_fv);
+int dvfs_reset_volt(struct vd_node *dvfs_vd);
+int dvfs_vd_get_newvolt_byclk(struct dvfs_node *clk_dvfs_node);
+int dvfs_vd_get_newvolt_bypd(struct vd_node *vd);
+int dvfs_scale_volt_bystep(struct vd_node *vd_clk, struct vd_node *vd_dep, int volt_new, int volt_dep_new,
+               int cur_clk_biger_than_dep, int cur_dep_biger_than_clk, int new_clk_biger_than_dep, int new_dep_biger_than_clk);
+int rk_regist_vd(struct vd_node *vd);
+int rk_regist_pd(struct pd_node *pd);
+int rk_regist_clk(struct dvfs_node *clk_dvfs_node);
+struct dvfs_node *dvfs_get_clk_dvfs_node_byname(char *name);
+int vd_regulator_round_volt(struct vd_node *vd, int volt,int flags);
+/*********************************if not define dvfs ,the following function is need defined func{}******************************/
+int dvfs_vd_clk_set_rate(struct clk_hw *hw, unsigned long rate,
+                           unsigned long parent_rate);
+int clk_enable_dvfs(struct clk *clk);
+int clk_disable_dvfs(struct clk *clk);
+void clk_dvfs_register_set_rate_callback(struct clk *clk, clk_dvfs_target_callback clk_dvfs_target);
+struct cpufreq_frequency_table *dvfs_get_freq_volt_table(struct clk *clk);
+int dvfs_set_freq_volt_table(struct clk *clk, struct cpufreq_frequency_table *table);
+struct regulator* dvfs_get_regulator(char *regulator_name);
+int clk_dvfs_enable_limit(struct clk *clk, unsigned int min_rate, unsigned max_rate);
+int clk_dvfs_disable_limit(struct clk *clk);
+int dvfs_scale_volt_direct(struct vd_node *vd_clk, int volt_new);
+/******************************** inline *******************************/
+static inline struct dvfs_node *clk_get_dvfs_info(struct clk *clk)
+{
+    return clk->private_data;
+}
+static inline bool dvfs_support_clk_set_rate(struct dvfs_node *dvfs_info)
+{
+       return (dvfs_info&&dvfs_info->enable_dvfs);
+}
+static inline bool dvfs_support_clk_disable(struct dvfs_node *dvfs_info)
+{
+       return (dvfs_info&&dvfs_info->disable_ctr&&dvfs_info->enable_dvfs);
+}
+/********************************avs*******************************/
+void avs_init(void);
+void avs_init_val_get(int index,int vol,char *s);
+int avs_set_scal_val(u8 avs_base);
+void avs_board_init(struct avs_ctr_st *data);
+
+#else
+static inline int of_dvfs_init(void){ return 0; };
+static inline bool dvfs_support_clk_set_rate(struct dvfs_node *dvfs_info) { return 0; }
+static inline bool dvfs_support_clk_disable(struct dvfs_node *dvfs_info) { return 0; }
+static inline int dvfs_vd_clk_set_rate(struct clk *clk, unsigned long rate) { return 0; }
+static inline int dvfs_vd_clk_disable(struct clk *clk, int on) { return 0; }
+static inline int clk_enable_dvfs(struct clk *clk) { return 0; }
+static inline int clk_disable_dvfs(struct clk *clk) { return 0; }
+static inline void clk_dvfs_register_set_rate_callback(struct clk *clk, clk_dvfs_target_callback clk_dvfs_target) {}
+static inline struct cpufreq_frequency_table *dvfs_get_freq_volt_table(struct clk *clk) { return NULL; }
+static inline int dvfs_set_freq_volt_table(struct clk *clk, struct cpufreq_frequency_table *table) { return 0; }
+static inline struct regulator* dvfs_get_regulator(char *regulator_name){ return NULL; }
+static inline int clk_dvfs_enable_limit(struct clk *clk, unsigned int min_rate, unsigned max_rate){ return 0; }
+static inline int clk_dvfs_disable_limit(struct clk *clk){ return 0; };
+static inline int dvfs_scale_volt_direct(struct vd_node *vd_clk, int volt_new){ return 0; };
+
+static inline void avs_init(void){};
+static inline void avs_init_val_get(int index, int vol, char *s){};
+static inline int avs_set_scal_val(u8 avs_base){ return 0; };
+static inline void avs_board_init(struct avs_ctr_st *data){ return 0; };
+#endif
+
+#endif
diff --git a/arch/arm/mach-rockchip/rk3188-cpufreq.c b/arch/arm/mach-rockchip/rk3188-cpufreq.c
new file mode 100644 (file)
index 0000000..a32c543
--- /dev/null
@@ -0,0 +1,723 @@
+/*
+ * Copyright (C) 2013 ROCKCHIP, Inc.
+ *
+ * This software is licensed under the terms of the GNU General Public
+ * License version 2, as published by the Free Software Foundation, and
+ * may be copied, distributed, and modified under those terms.
+ *
+ * This program is distributed in the hope that 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.
+ *
+ */
+
+#define pr_fmt(fmt) "cpufreq: " fmt
+#include <linux/clk.h>
+#include <linux/cpufreq.h>
+#include <linux/err.h>
+#include <linux/kernel_stat.h>
+#include <linux/init.h>
+#include <linux/reboot.h>
+#include <linux/suspend.h>
+#include <linux/tick.h>
+#include <linux/workqueue.h>
+#include <linux/delay.h>
+#include <linux/regulator/consumer.h>
+#include <linux/fs.h>
+#include <linux/miscdevice.h>
+#include <linux/string.h>
+#include <asm/smp_plat.h>
+#include <asm/cpu.h>
+#include <asm/unistd.h>
+#include <asm/uaccess.h>
+
+#include "dvfs.h"
+#include "cpu.h"
+
+#define VERSION "2.2"
+
+#ifdef DEBUG
+#define FREQ_DBG(fmt, args...) pr_debug(fmt, ## args)
+#define FREQ_LOG(fmt, args...) pr_debug(fmt, ## args)
+#else
+#define FREQ_DBG(fmt, args...) do {} while(0)
+#define FREQ_LOG(fmt, args...) do {} while(0)
+#endif
+#define FREQ_ERR(fmt, args...) pr_err(fmt, ## args)
+
+/* Frequency table index must be sequential starting at 0 */
+static struct cpufreq_frequency_table default_freq_table[] = {
+       {.frequency = 312 * 1000,       .index = 875 * 1000},
+       {.frequency = 504 * 1000,       .index = 925 * 1000},
+       {.frequency = 816 * 1000,       .index = 975 * 1000},
+       {.frequency = 1008 * 1000,      .index = 1075 * 1000},
+       {.frequency = 1200 * 1000,      .index = 1150 * 1000},
+       {.frequency = 1416 * 1000,      .index = 1250 * 1000},
+       {.frequency = 1608 * 1000,      .index = 1350 * 1000},
+       {.frequency = CPUFREQ_TABLE_END},
+};
+
+static struct cpufreq_frequency_table *freq_table = default_freq_table;
+
+/*********************************************************/
+
+/* additional symantics for "relation" in cpufreq with pm */
+#define DISABLE_FURTHER_CPUFREQ         0x10
+#define ENABLE_FURTHER_CPUFREQ          0x20
+#define MASK_FURTHER_CPUFREQ            0x30
+/* With 0x00(NOCHANGE), it depends on the previous "further" status */
+#define CPUFREQ_PRIVATE                 0x100
+static int no_cpufreq_access;
+static unsigned int suspend_freq = 816 * 1000;
+#if defined(CONFIG_ARCH_RK3026)
+static unsigned int suspend_volt = 1100000; // 1.1V
+#else
+static unsigned int suspend_volt = 1000000; // 1V
+#endif
+static unsigned int low_battery_freq = 600 * 1000;
+//static unsigned int low_battery_capacity = 5; // 5%
+static bool is_booting = true;
+
+static struct workqueue_struct *freq_wq;
+static struct clk *cpu_clk;
+
+static DEFINE_MUTEX(cpufreq_mutex);
+
+static struct clk *gpu_clk;
+static bool gpu_is_mali400;
+static struct clk *ddr_clk;
+
+static int cpufreq_scale_rate_for_dvfs(struct clk_hw *hw, unsigned long rate, dvfs_set_rate_callback set_rate);
+
+/*******************************************************/
+static unsigned int rk3188_cpufreq_get(unsigned int cpu)
+{
+       return clk_get_rate(cpu_clk) / 1000;
+}
+
+static bool cpufreq_is_ondemand(struct cpufreq_policy *policy)
+{
+       char c = 0;
+       if (policy && policy->governor)
+               c = policy->governor->name[0];
+       return (c == 'o' || c == 'i' || c == 'c' || c == 'h');
+}
+
+static unsigned int get_freq_from_table(unsigned int max_freq)
+{
+       unsigned int i;
+       unsigned int target_freq = 0;
+       for (i = 0; freq_table[i].frequency != CPUFREQ_TABLE_END; i++) {
+               unsigned int freq = freq_table[i].frequency;
+               if (freq <= max_freq && target_freq < freq) {
+                       target_freq = freq;
+               }
+       }
+       if (!target_freq)
+               target_freq = max_freq;
+       return target_freq;
+}
+
+/**********************thermal limit**************************/
+//#define CONFIG_RK30_CPU_FREQ_LIMIT_BY_TEMP
+
+#ifdef CONFIG_RK30_CPU_FREQ_LIMIT_BY_TEMP
+static unsigned int temp_limit_freq = -1;
+module_param(temp_limit_freq, uint, 0444);
+
+static struct cpufreq_frequency_table temp_limits[4][4] = {
+       {       // 1 CPU busy
+               {.frequency =          -1, .index = 50},
+               {.frequency =          -1, .index = 55},
+               {.frequency =          -1, .index = 60},
+               {.frequency = 1608 * 1000, .index = 75},
+       }, {    // 2 CPUs busy
+               {.frequency = 1800 * 1000, .index = 50},
+               {.frequency = 1608 * 1000, .index = 55},
+               {.frequency = 1416 * 1000, .index = 60},
+               {.frequency = 1200 * 1000, .index = 75},
+       }, {    // 3 CPUs busy
+               {.frequency = 1608 * 1000, .index = 50},
+               {.frequency = 1416 * 1000, .index = 55},
+               {.frequency = 1200 * 1000, .index = 60},
+               {.frequency = 1008 * 1000, .index = 75},
+       }, {    // 4 CPUs busy
+               {.frequency = 1416 * 1000, .index = 50},
+               {.frequency = 1200 * 1000, .index = 55},
+               {.frequency = 1008 * 1000, .index = 60},
+               {.frequency =  816 * 1000, .index = 75},
+       }
+};
+
+static struct cpufreq_frequency_table temp_limits_cpu_perf[] = {
+       {.frequency = 1008 * 1000, .index = 100},
+};
+
+static struct cpufreq_frequency_table temp_limits_gpu_perf[] = {
+       {.frequency = 1008 * 1000, .index = 0},
+};
+
+static int rk3188_get_temp(void)
+{
+       return 60;
+}
+
+static char sys_state;
+static ssize_t sys_state_write(struct file *file, const char __user *buffer, size_t count, loff_t *ppos)
+{
+       char state;
+
+       if (count < 1)
+               return count;
+       if (copy_from_user(&state, buffer, 1)) {
+               return -EFAULT;
+       }
+
+       sys_state = state;
+       return count;
+}
+
+static const struct file_operations sys_state_fops = {
+       .owner  = THIS_MODULE,
+       .write  = sys_state_write,
+};
+
+static struct miscdevice sys_state_dev = {
+       .fops   = &sys_state_fops,
+       .name   = "sys_state",
+       .minor  = MISC_DYNAMIC_MINOR,
+};
+
+static void rk3188_cpufreq_temp_limit_work_func(struct work_struct *work)
+{
+       static bool in_perf = false;
+       struct cpufreq_policy *policy;
+       int temp, i;
+       unsigned int new_freq = -1;
+       unsigned long delay = HZ / 10; // 100ms
+       unsigned int nr_cpus = num_online_cpus();
+       const struct cpufreq_frequency_table *limits_table = temp_limits[nr_cpus - 1];
+       size_t limits_size = ARRAY_SIZE(temp_limits[nr_cpus - 1]);
+
+       temp = rk3188_get_temp();
+
+       if (sys_state == '1') {
+               in_perf = true;
+               if (gpu_is_mali400) {
+                       unsigned int gpu_irqs[2];
+                       gpu_irqs[0] = kstat_irqs(IRQ_GPU_GP);
+                       msleep(40);
+                       gpu_irqs[1] = kstat_irqs(IRQ_GPU_GP);
+                       delay = 0;
+                       if ((gpu_irqs[1] - gpu_irqs[0]) < 8) {
+                               limits_table = temp_limits_cpu_perf;
+                               limits_size = ARRAY_SIZE(temp_limits_cpu_perf);
+                       } else {
+                               limits_table = temp_limits_gpu_perf;
+                               limits_size = ARRAY_SIZE(temp_limits_gpu_perf);
+                       }
+               } else {
+                       delay = HZ; // 1s
+                       limits_table = temp_limits_cpu_perf;
+                       limits_size = ARRAY_SIZE(temp_limits_cpu_perf);
+               }
+       } else if (in_perf) {
+               in_perf = false;
+       } else {
+               static u64 last_time_in_idle = 0;
+               static u64 last_time_in_idle_timestamp = 0;
+               u64 time_in_idle = 0, now;
+               u32 delta_idle;
+               u32 delta_time;
+               unsigned cpu;
+
+               for_each_online_cpu(cpu) {
+                       time_in_idle += get_cpu_idle_time_us(cpu, &now);
+               }
+               delta_time = now - last_time_in_idle_timestamp;
+               delta_idle = time_in_idle - last_time_in_idle;
+               last_time_in_idle = time_in_idle;
+               last_time_in_idle_timestamp = now;
+               delta_idle += delta_time >> 4; // +6.25%
+               if (delta_idle > (nr_cpus - 1) * delta_time && delta_idle < (nr_cpus + 1) * delta_time)
+                       limits_table = temp_limits[0];
+               else if (delta_idle > (nr_cpus - 2) * delta_time)
+                       limits_table = temp_limits[1];
+               else if (delta_idle > (nr_cpus - 3) * delta_time)
+                       limits_table = temp_limits[2];
+               FREQ_DBG("delta time %6u us idle %6u us %u cpus select table %d\n", delta_time, delta_idle, nr_cpus, (limits_table - temp_limits[0]) / ARRAY_SIZE(temp_limits[0]));
+       }
+
+       for (i = 0; i < limits_size; i++) {
+               if (temp >= limits_table[i].index) {
+                       new_freq = limits_table[i].frequency;
+               }
+       }
+
+       if (temp_limit_freq != new_freq) {
+               unsigned int cur_freq;
+               temp_limit_freq = new_freq;
+               cur_freq = rk3188_cpufreq_get(0);
+               FREQ_DBG("temp limit %7d KHz cur %7d KHz\n", temp_limit_freq, cur_freq);
+               if (cur_freq > temp_limit_freq) {
+                       policy = cpufreq_cpu_get(0);
+                       cpufreq_driver_target(policy, policy->cur, CPUFREQ_RELATION_L | CPUFREQ_PRIVATE);
+                       cpufreq_cpu_put(policy);
+               }
+       }
+
+       queue_delayed_work_on(0, freq_wq, to_delayed_work(work), delay);
+}
+
+static DECLARE_DELAYED_WORK(rk3188_cpufreq_temp_limit_work, rk3188_cpufreq_temp_limit_work_func);
+
+static int rk3188_cpufreq_notifier_policy(struct notifier_block *nb, unsigned long val, void *data)
+{
+       struct cpufreq_policy *policy = data;
+
+       if (val != CPUFREQ_NOTIFY)
+               return 0;
+
+       if (cpufreq_is_ondemand(policy)) {
+               FREQ_DBG("queue work\n");
+               queue_delayed_work_on(0, freq_wq, &rk3188_cpufreq_temp_limit_work, 0);
+       } else {
+               FREQ_DBG("cancel work\n");
+               cancel_delayed_work_sync(&rk3188_cpufreq_temp_limit_work);
+       }
+
+       return 0;
+}
+
+static struct notifier_block notifier_policy_block = {
+       .notifier_call = rk3188_cpufreq_notifier_policy
+};
+
+static void rk3188_cpufreq_temp_limit_init(struct cpufreq_policy *policy)
+{
+       unsigned int i;
+       struct cpufreq_frequency_table *table;
+
+       table = temp_limits[0];
+       for (i = 0; i < sizeof(temp_limits) / sizeof(struct cpufreq_frequency_table); i++) {
+               table[i].frequency = get_freq_from_table(table[i].frequency);
+       }
+       table = temp_limits_cpu_perf;
+       for (i = 0; i < sizeof(temp_limits_cpu_perf) / sizeof(struct cpufreq_frequency_table); i++) {
+               table[i].frequency = get_freq_from_table(table[i].frequency);
+       }
+       table = temp_limits_gpu_perf;
+       for (i = 0; i < sizeof(temp_limits_gpu_perf) / sizeof(struct cpufreq_frequency_table); i++) {
+               table[i].frequency = get_freq_from_table(table[i].frequency);
+       }
+       misc_register(&sys_state_dev);
+       if (cpufreq_is_ondemand(policy)) {
+               queue_delayed_work_on(0, freq_wq, &rk3188_cpufreq_temp_limit_work, 0*HZ);
+       }
+       cpufreq_register_notifier(&notifier_policy_block, CPUFREQ_POLICY_NOTIFIER);
+}
+
+static void rk3188_cpufreq_temp_limit_exit(void)
+{
+       cpufreq_unregister_notifier(&notifier_policy_block, CPUFREQ_POLICY_NOTIFIER);
+       if (freq_wq)
+               cancel_delayed_work(&rk3188_cpufreq_temp_limit_work);
+}
+#else
+static inline void rk3188_cpufreq_temp_limit_init(struct cpufreq_policy *policy) {}
+static inline void rk3188_cpufreq_temp_limit_exit(void) {}
+#endif
+
+/************************************dvfs tst************************************/
+//#define CPU_FREQ_DVFS_TST
+#ifdef CPU_FREQ_DVFS_TST
+static unsigned int freq_dvfs_tst_rate;
+static int test_count;
+#define TEST_FRE_NUM 11
+static int test_tlb_rate[TEST_FRE_NUM] = { 504, 1008, 504, 1200, 252, 816, 1416, 252, 1512, 252, 816 };
+//static int test_tlb_rate[TEST_FRE_NUM]={504,1008,504,1200,252,816,1416,126,1512,126,816};
+
+#define TEST_GPU_NUM 3
+
+static int test_tlb_gpu[TEST_GPU_NUM] = { 360, 400, 180 };
+static int test_tlb_ddr[TEST_GPU_NUM] = { 401, 200, 500 };
+
+static int gpu_ddr = 0;
+
+static void rk3188_cpufreq_dvsf_tst_work_func(struct work_struct *work)
+{
+       struct cpufreq_policy *policy = cpufreq_cpu_get(0);
+
+       gpu_ddr++;
+#if 0
+       FREQ_LOG("cpufreq_dvsf_tst,ddr%u,gpu%u\n",
+               test_tlb_ddr[gpu_ddr % TEST_GPU_NUM],
+               test_tlb_gpu[gpu_ddr % TEST_GPU_NUM]);
+       clk_set_rate(ddr_clk, test_tlb_ddr[gpu_ddr % TEST_GPU_NUM] * 1000 * 1000);
+       clk_set_rate(gpu_clk, test_tlb_gpu[gpu_ddr % TEST_GPU_NUM] * 1000 * 1000);
+#endif
+
+       test_count++;
+       freq_dvfs_tst_rate = test_tlb_rate[test_count % TEST_FRE_NUM] * 1000;
+       printk("cpufreq_dvsf_tst,cpu set rate %d\n", freq_dvfs_tst_rate);
+       cpufreq_driver_target(policy, policy->cur, CPUFREQ_RELATION_L);
+       cpufreq_cpu_put(policy);
+
+       queue_delayed_work_on(0, freq_wq, to_delayed_work(work), msecs_to_jiffies(1000));
+}
+
+static DECLARE_DELAYED_WORK(rk3188_cpufreq_dvsf_tst_work, rk3188_cpufreq_dvsf_tst_work_func);
+#endif /* CPU_FREQ_DVFS_TST */
+
+/***********************************************************************/
+static int rk3188_cpufreq_verify(struct cpufreq_policy *policy)
+{
+       if (!freq_table)
+               return -EINVAL;
+       return cpufreq_frequency_table_verify(policy, freq_table);
+}
+
+static int rk3188_cpufreq_init_cpu0(struct cpufreq_policy *policy)
+{
+       unsigned int i;
+       //struct cpufreq_frequency_table *table_adjust;
+
+       gpu_is_mali400 = cpu_is_rk3188();
+       gpu_clk = clk_get(NULL, "clk_gpu");
+       if (IS_ERR(gpu_clk))
+               return PTR_ERR(gpu_clk);
+
+       ddr_clk = clk_get(NULL, "clk_ddr");
+       if (IS_ERR(ddr_clk))
+               return PTR_ERR(ddr_clk);
+
+       cpu_clk = clk_get(NULL, "clk_core");
+       if (IS_ERR(cpu_clk))
+               return PTR_ERR(cpu_clk);
+
+       //table_adjust = dvfs_get_freq_volt_table(cpu_clk);
+       //dvfs_adjust_table_lmtvolt(cpu_clk, table_adjust);
+       //table_adjust = dvfs_get_freq_volt_table(gpu_clk);
+       //dvfs_adjust_table_lmtvolt(gpu_clk, table_adjust);
+
+       clk_enable_dvfs(gpu_clk);
+       if (gpu_is_mali400)
+               clk_dvfs_enable_limit(gpu_clk, 133000000, 600000000);
+
+       clk_enable_dvfs(ddr_clk);
+
+       clk_dvfs_register_set_rate_callback(cpu_clk, cpufreq_scale_rate_for_dvfs);
+       freq_table = dvfs_get_freq_volt_table(cpu_clk);
+       if (freq_table == NULL) {
+               freq_table = default_freq_table;
+       } else {
+               int v = INT_MAX;
+               for (i = 0; freq_table[i].frequency != CPUFREQ_TABLE_END; i++) {
+                       if (freq_table[i].index >= suspend_volt && v > freq_table[i].index) {
+                               suspend_freq = freq_table[i].frequency;
+                               v = freq_table[i].index;
+                       }
+               }
+       }
+       low_battery_freq = get_freq_from_table(low_battery_freq);
+       clk_enable_dvfs(cpu_clk);
+       /*if(rk_tflag()){
+#define RK3188_T_LIMIT_FREQ    (1416 * 1000)
+               dvfs_clk_enable_limit(cpu_clk, 0, RK3188_T_LIMIT_FREQ * 1000);
+               for (i = 0; freq_table[i].frequency != CPUFREQ_TABLE_END; i++) {
+                       if (freq_table[i].frequency > RK3188_T_LIMIT_FREQ) {
+                               printk("cpufreq: delete arm freq(%u)\n", freq_table[i].frequency);
+                               freq_table[i].frequency = CPUFREQ_TABLE_END;
+                       }
+               }
+       }*/
+       freq_wq = alloc_workqueue("rk3188_cpufreqd", WQ_NON_REENTRANT | WQ_MEM_RECLAIM | WQ_HIGHPRI | WQ_FREEZABLE, 1);
+       rk3188_cpufreq_temp_limit_init(policy);
+#ifdef CPU_FREQ_DVFS_TST
+       queue_delayed_work(freq_wq, &rk3188_cpufreq_dvsf_tst_work, msecs_to_jiffies(20 * 1000));
+#endif
+
+       printk("rk3188 cpufreq version " VERSION ", suspend freq %d MHz\n", suspend_freq / 1000);
+       return 0;
+       
+}
+
+static int rk3188_cpufreq_init(struct cpufreq_policy *policy)
+{
+       if (policy->cpu == 0) {
+               int err = rk3188_cpufreq_init_cpu0(policy);
+               if (err)
+                       return err;
+       }
+       
+       //set freq min max
+       cpufreq_frequency_table_cpuinfo(policy, freq_table);
+       //sys nod
+       cpufreq_frequency_table_get_attr(freq_table, policy->cpu);
+
+
+       policy->cur = rk3188_cpufreq_get(0);
+
+       policy->cpuinfo.transition_latency = 40 * NSEC_PER_USEC;        // make ondemand default sampling_rate to 40000
+
+       /*
+        * On SMP configuartion, both processors share the voltage
+        * and clock. So both CPUs needs to be scaled together and hence
+        * needs software co-ordination. Use cpufreq affected_cpus
+        * interface to handle this scenario. Additional is_smp() check
+        * is to keep SMP_ON_UP build working.
+        */
+       if (is_smp())
+               cpumask_setall(policy->cpus);
+
+       return 0;
+
+}
+
+static int rk3188_cpufreq_exit(struct cpufreq_policy *policy)
+{
+       if (policy->cpu != 0)
+               return 0;
+
+       cpufreq_frequency_table_cpuinfo(policy, freq_table);
+       clk_put(cpu_clk);
+       rk3188_cpufreq_temp_limit_exit();
+       if (freq_wq) {
+               flush_workqueue(freq_wq);
+               destroy_workqueue(freq_wq);
+               freq_wq = NULL;
+       }
+
+       return 0;
+}
+
+static struct freq_attr *rk3188_cpufreq_attr[] = {
+       &cpufreq_freq_attr_scaling_available_freqs,
+       NULL,
+};
+
+#ifdef CONFIG_POWER_SUPPLY
+extern int rk_get_system_battery_capacity(void);
+#else
+static int rk_get_system_battery_capacity(void) { return 100; }
+#endif
+
+static unsigned int cpufreq_scale_limit(unsigned int target_freq, struct cpufreq_policy *policy, bool is_private)
+{
+       bool is_ondemand = cpufreq_is_ondemand(policy);
+
+#ifdef CPU_FREQ_DVFS_TST
+       if (freq_dvfs_tst_rate) {
+               target_freq = freq_dvfs_tst_rate;
+               freq_dvfs_tst_rate = 0;
+               return target_freq;
+       }
+#endif
+
+       if (!is_ondemand)
+               return target_freq;
+
+       if (is_booting) {
+               s64 boottime_ms = ktime_to_ms(ktime_get_boottime());
+               if (boottime_ms > 60 * MSEC_PER_SEC) {
+                       is_booting = false;
+               }/* else if (target_freq > low_battery_freq &&
+                          rk_get_system_battery_capacity() <= low_battery_capacity) {
+                       target_freq = low_battery_freq;
+               }*/
+       }
+
+#ifdef CONFIG_RK30_CPU_FREQ_LIMIT_BY_TEMP
+       {
+               static unsigned int ondemand_target = 816 * 1000;
+               if (is_private)
+                       target_freq = ondemand_target;
+               else
+                       ondemand_target = target_freq;
+       }
+
+       /*
+        * If the new frequency is more than the thermal max allowed
+        * frequency, go ahead and scale the mpu device to proper frequency.
+        */
+       target_freq = min(target_freq, temp_limit_freq);
+#endif
+
+       return target_freq;
+}
+
+static int cpufreq_scale_rate_for_dvfs(struct clk_hw *hw, unsigned long rate, dvfs_set_rate_callback set_rate)
+{
+
+       unsigned int i;
+       int ret;
+       struct clk *clk = hw->clk;
+       struct cpufreq_freqs freqs;
+       struct cpufreq_policy *policy;
+       
+       freqs.new = rate / 1000;
+       freqs.old = clk_get_rate(clk) / 1000;
+       
+       for_each_online_cpu(freqs.cpu) {
+               policy = cpufreq_cpu_get(freqs.cpu);
+               cpufreq_notify_transition(policy, &freqs, CPUFREQ_PRECHANGE);
+               cpufreq_cpu_put(policy);
+       }
+       
+       FREQ_DBG("cpufreq_scale_rate_for_dvfs(%lu)\n", rate);
+       
+       ret = set_rate(hw, rate, rate);
+
+#ifdef CONFIG_SMP
+       /*
+        * Note that loops_per_jiffy is not updated on SMP systems in
+        * cpufreq driver. So, update the per-CPU loops_per_jiffy value
+        * on frequency transition. We need to update all dependent CPUs.
+        */
+       for_each_possible_cpu(i) {
+               per_cpu(cpu_data, i).loops_per_jiffy = loops_per_jiffy;
+       }
+#endif
+
+       freqs.new = clk_get_rate(clk) / 1000;
+       /* notifiers */
+       for_each_online_cpu(freqs.cpu) {
+               policy = cpufreq_cpu_get(freqs.cpu);
+               cpufreq_notify_transition(policy, &freqs, CPUFREQ_POSTCHANGE);
+               cpufreq_cpu_put(policy);
+       }
+
+       return ret;
+       
+}
+
+static int rk3188_cpufreq_target(struct cpufreq_policy *policy, unsigned int target_freq, unsigned int relation)
+{
+
+       unsigned int i, new_freq = target_freq, new_rate, cur_rate;
+       int ret = 0;
+       bool is_private;
+
+       if (!freq_table) {
+               FREQ_ERR("no freq table!\n");
+               return -EINVAL;
+       }
+
+       mutex_lock(&cpufreq_mutex);
+
+       is_private = relation & CPUFREQ_PRIVATE;
+       relation &= ~CPUFREQ_PRIVATE;
+
+       if (relation & ENABLE_FURTHER_CPUFREQ)
+               no_cpufreq_access--;
+       if (no_cpufreq_access) {
+               FREQ_LOG("denied access to %s as it is disabled temporarily\n", __func__);
+               ret = -EINVAL;
+               goto out;
+       }
+       if (relation & DISABLE_FURTHER_CPUFREQ)
+               no_cpufreq_access++;
+       relation &= ~MASK_FURTHER_CPUFREQ;
+
+       ret = cpufreq_frequency_table_target(policy, freq_table, target_freq, relation, &i);
+       if (ret) {
+               FREQ_ERR("no freq match for %d(ret=%d)\n", target_freq, ret);
+               goto out;
+       }
+       new_freq = freq_table[i].frequency;
+       if (!no_cpufreq_access)
+               new_freq = cpufreq_scale_limit(new_freq, policy, is_private);
+
+       new_rate = new_freq * 1000;
+       cur_rate = clk_get_rate(cpu_clk);
+       FREQ_LOG("req = %7u new = %7u (was = %7u)\n", target_freq, new_freq, cur_rate / 1000);
+       if (new_rate == cur_rate)
+               goto out;
+       ret = clk_set_rate(cpu_clk, new_rate);
+
+out:
+       FREQ_DBG("set freq (%7u) end, ret %d\n", new_freq, ret);
+       mutex_unlock(&cpufreq_mutex);
+       return ret;
+
+}
+
+static int rk3188_cpufreq_pm_notifier_event(struct notifier_block *this, unsigned long event, void *ptr)
+{
+       int ret = NOTIFY_DONE;
+       struct cpufreq_policy *policy = cpufreq_cpu_get(0);
+
+       if (!policy)
+               return ret;
+
+       if (!cpufreq_is_ondemand(policy))
+               goto out;
+
+       switch (event) {
+       case PM_SUSPEND_PREPARE:
+               ret = cpufreq_driver_target(policy, suspend_freq, DISABLE_FURTHER_CPUFREQ | CPUFREQ_RELATION_H);
+               if (ret < 0) {
+                       ret = NOTIFY_BAD;
+                       goto out;
+               }
+               ret = NOTIFY_OK;
+               break;
+       case PM_POST_RESTORE:
+       case PM_POST_SUSPEND:
+               cpufreq_driver_target(policy, suspend_freq, ENABLE_FURTHER_CPUFREQ | CPUFREQ_RELATION_H);
+               ret = NOTIFY_OK;
+               break;
+       }
+out:
+       cpufreq_cpu_put(policy);
+       return ret;
+}
+
+static struct notifier_block rk3188_cpufreq_pm_notifier = {
+       .notifier_call = rk3188_cpufreq_pm_notifier_event,
+};
+
+static int rk3188_cpufreq_reboot_notifier_event(struct notifier_block *this, unsigned long event, void *ptr)
+{
+       struct cpufreq_policy *policy = cpufreq_cpu_get(0);
+
+       if (policy) {
+               is_booting = false;
+               cpufreq_driver_target(policy, suspend_freq, DISABLE_FURTHER_CPUFREQ | CPUFREQ_RELATION_H);
+               cpufreq_cpu_put(policy);
+       }
+
+       return NOTIFY_OK;
+}
+
+static struct notifier_block rk3188_cpufreq_reboot_notifier = {
+       .notifier_call = rk3188_cpufreq_reboot_notifier_event,
+};
+
+static struct cpufreq_driver rk3188_cpufreq_driver = {
+       .flags = CPUFREQ_CONST_LOOPS,
+       .verify = rk3188_cpufreq_verify,
+       .target = rk3188_cpufreq_target,
+       .get = rk3188_cpufreq_get,
+       .init = rk3188_cpufreq_init,
+       .exit = rk3188_cpufreq_exit,
+       .name = "rk3188",
+       .attr = rk3188_cpufreq_attr,
+};
+
+static int __init rk3188_cpufreq_driver_init(void)
+{      
+       register_pm_notifier(&rk3188_cpufreq_pm_notifier);
+       register_reboot_notifier(&rk3188_cpufreq_reboot_notifier);
+       return cpufreq_register_driver(&rk3188_cpufreq_driver);
+}
+
+static void __exit rk3188_cpufreq_driver_exit(void)
+{
+       cpufreq_unregister_driver(&rk3188_cpufreq_driver);
+}
+
+device_initcall(rk3188_cpufreq_driver_init);
+module_exit(rk3188_cpufreq_driver_exit);
+
index c86293e9037261c5682a80df9e9641a3a834e2fa..d6e5ffd02d2bb0895f11cf078cfc15078bdf0f8b 100644 (file)
@@ -32,6 +32,7 @@
 #include "loader.h"
 #include "pmu.h"
 #include "sram.h"
+#include "dvfs.h"
 
 #define RK3188_DEVICE(name) \
        { \
@@ -293,6 +294,7 @@ static void __init rk3188_dt_init_timer(void)
        rockchip_pmu_ops.set_idle_request = rk3188_pmu_set_idle_request;
        of_clk_init(NULL);
        clocksource_of_init();
+       of_dvfs_init();
 }
 
 static const char * const rk3188_dt_compat[] __initconst = {
index 8138c94409f3f3d1253c2b83e62c3f131ff70d84..f8206c3c08dcf698d68729a1ae7103d72ddae376 100644 (file)
@@ -44,6 +44,8 @@ struct clk {
        struct hlist_head       children;
        struct hlist_node       child_node;
        unsigned int            notifier_count;
+
+       void                    *private_data;
 #ifdef CONFIG_COMMON_CLK_DEBUG
        struct dentry           *dentry;
 #endif