2019-05-27 08:55:21 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2015-05-13 17:58:38 +03:00
/*
* Utility functions for parsing Tegra CVB voltage tables
*
2019-01-04 11:06:48 +08:00
* Copyright ( C ) 2012 - 2019 NVIDIA Corporation . All rights reserved .
2015-05-13 17:58:38 +03:00
*/
# include <linux/err.h>
# include <linux/kernel.h>
# include <linux/pm_opp.h>
# include "cvb.h"
/* cvb_mv = ((c2 * speedo / s_scale + c1) * speedo / s_scale + c0) */
static inline int get_cvb_voltage ( int speedo , int s_scale ,
const struct cvb_coefficients * cvb )
{
int mv ;
/* apply only speedo scale: output mv = cvb_mv * v_scale */
mv = DIV_ROUND_CLOSEST ( cvb - > c2 * speedo , s_scale ) ;
mv = DIV_ROUND_CLOSEST ( ( mv + cvb - > c1 ) * speedo , s_scale ) + cvb - > c0 ;
return mv ;
}
static int round_cvb_voltage ( int mv , int v_scale ,
const struct rail_alignment * align )
{
/* combined: apply voltage scale and round to cvb alignment step */
int uv ;
int step = ( align - > step_uv ? : 1000 ) * v_scale ;
int offset = align - > offset_uv * v_scale ;
uv = max ( mv * 1000 , offset ) - offset ;
uv = DIV_ROUND_UP ( uv , step ) * align - > step_uv + align - > offset_uv ;
return uv / 1000 ;
}
enum {
DOWN ,
UP
} ;
static int round_voltage ( int mv , const struct rail_alignment * align , int up )
{
if ( align - > step_uv ) {
int uv ;
uv = max ( mv * 1000 , align - > offset_uv ) - align - > offset_uv ;
uv = ( uv + ( up ? align - > step_uv - 1 : 0 ) ) / align - > step_uv ;
return ( uv * align - > step_uv + align - > offset_uv ) / 1000 ;
}
return mv ;
}
2016-04-08 15:09:56 +02:00
static int build_opp_table ( struct device * dev , const struct cvb_table * table ,
2019-01-04 11:06:48 +08:00
struct rail_alignment * align ,
2016-04-08 15:09:56 +02:00
int speedo_value , unsigned long max_freq )
2015-05-13 17:58:38 +03:00
{
int i , ret , dfll_mv , min_mv , max_mv ;
2016-04-08 15:09:56 +02:00
min_mv = round_voltage ( table - > min_millivolts , align , UP ) ;
max_mv = round_voltage ( table - > max_millivolts , align , DOWN ) ;
2015-05-13 17:58:38 +03:00
for ( i = 0 ; i < MAX_DVFS_FREQS ; i + + ) {
2016-04-08 15:09:56 +02:00
const struct cvb_table_freq_entry * entry = & table - > entries [ i ] ;
if ( ! entry - > freq | | ( entry - > freq > max_freq ) )
2015-05-13 17:58:38 +03:00
break ;
2016-04-08 15:09:56 +02:00
dfll_mv = get_cvb_voltage ( speedo_value , table - > speedo_scale ,
& entry - > coefficients ) ;
dfll_mv = round_cvb_voltage ( dfll_mv , table - > voltage_scale ,
align ) ;
2015-05-13 17:58:38 +03:00
dfll_mv = clamp ( dfll_mv , min_mv , max_mv ) ;
2016-04-08 15:09:56 +02:00
ret = dev_pm_opp_add ( dev , entry - > freq , dfll_mv * 1000 ) ;
2015-05-13 17:58:38 +03:00
if ( ret )
return ret ;
}
return 0 ;
}
/**
2016-04-08 15:09:56 +02:00
* tegra_cvb_add_opp_table - build OPP table from Tegra CVB tables
2016-10-01 21:46:29 +02:00
* @ dev : the struct device * for which the OPP table is built
* @ tables : array of CVB tables
* @ count : size of the previously mentioned array
2015-05-13 17:58:38 +03:00
* @ process_id : process id of the HW module
* @ speedo_id : speedo id of the HW module
* @ speedo_value : speedo value of the HW module
2016-10-01 21:46:29 +02:00
* @ max_freq : highest safe clock rate
2015-05-13 17:58:38 +03:00
*
* On Tegra , a CVB table encodes the relationship between operating voltage
* and safe maximal frequency for a given module ( e . g . GPU or CPU ) . This
* function calculates the optimal voltage - frequency operating points
* for the given arguments and exports them via the OPP library for the
2016-10-01 21:46:29 +02:00
* given @ dev . Returns a pointer to the struct cvb_table that matched
2015-05-13 17:58:38 +03:00
* or an ERR_PTR on failure .
*/
2016-04-08 15:09:56 +02:00
const struct cvb_table *
tegra_cvb_add_opp_table ( struct device * dev , const struct cvb_table * tables ,
2019-01-04 11:06:48 +08:00
size_t count , struct rail_alignment * align ,
int process_id , int speedo_id , int speedo_value ,
unsigned long max_freq )
2015-05-13 17:58:38 +03:00
{
2016-04-08 15:09:56 +02:00
size_t i ;
int ret ;
2015-05-13 17:58:38 +03:00
2016-04-08 15:09:56 +02:00
for ( i = 0 ; i < count ; i + + ) {
const struct cvb_table * table = & tables [ i ] ;
2015-05-13 17:58:38 +03:00
2016-04-08 15:09:56 +02:00
if ( table - > speedo_id ! = - 1 & & table - > speedo_id ! = speedo_id )
2015-05-13 17:58:38 +03:00
continue ;
2016-04-08 15:09:56 +02:00
if ( table - > process_id ! = - 1 & & table - > process_id ! = process_id )
2015-05-13 17:58:38 +03:00
continue ;
2019-01-04 11:06:48 +08:00
ret = build_opp_table ( dev , table , align , speedo_value ,
max_freq ) ;
2016-04-08 15:09:56 +02:00
return ret ? ERR_PTR ( ret ) : table ;
2015-05-13 17:58:38 +03:00
}
return ERR_PTR ( - EINVAL ) ;
}
2016-04-08 15:16:28 +02:00
void tegra_cvb_remove_opp_table ( struct device * dev ,
const struct cvb_table * table ,
unsigned long max_freq )
{
unsigned int i ;
for ( i = 0 ; i < MAX_DVFS_FREQS ; i + + ) {
const struct cvb_table_freq_entry * entry = & table - > entries [ i ] ;
if ( ! entry - > freq | | ( entry - > freq > max_freq ) )
break ;
dev_pm_opp_remove ( dev , entry - > freq ) ;
}
}