Kevin Hilman had written, on 11/19/2009 07:31 PM, the following:
Nishanth Menon <nm@xxxxxx> writes:
Menon, Nishanth had written, on 11/15/2009 08:54 AM, the following:
[...]
b) Improvements of the omap accessor functions, I will send out few of
the proposals I can think of later when I get some free time.. we can
discuss that too.
Here is my initial proposal, do feel free to comment/add/modify etc:
Thanks for this. Some comments on your proposal, and a proposed
header from me below.
Thanks for a thorough review and the new ideas. I will do a proposal
later tomorrow to aggregate and provide a common solution I hope. some
views follow.
In the accessor functions listed here, I have followed:
a) The function will return 0 if success else appropriate error
values, all values are passed by pointer params.
b) Overall, the idea is to transition from exposing omap_opp to users
currently to a system where opp handling logic will be blackboxed from
the rest of the system - giving us options to optimize and scale
internal logic and storage mechanism without impacting the rest of the
system.
completely agree here.
c) resultant code should be readable - this is one of the issues I see
with my previous implementation.
Files: Introduce opp.c and opp.h
A) Data Structures:
Interim:
opp.h
struct omap_opp {
bool enabled;
unsigned long rate; /* in hz */
u8 opp_id; /* int */
u16 vsel; /* T2 voltage value */
};
typedef struct omap_opp omap_opp_def;
Final, once resource34xx.c and all direct accesses are cleaned up:
opp.h:
struct omap_opp;
/* initial definitions of OPP, will be used by all */
struct omap_opp_def {
bool enabled; /* true/false */
unsigned long rate; /* in hz */
u16 vsel; /* in millivolts */
};
opp.c:
struct omap_opp {
bool enabled;
unsigned long rate;
u16 vsel;
};
or what ever implementation it needs. can be array, list, hash
implementation or what ever we choose it to be.
After SRF is gone, the header can simply have a 'struct omap_opp' and
the actual definition can be in opp.c
The the functions will need the handle for it to know which opp table is
referenced to. so we need the prototype in header, though the definition
has to be in .c file
B) Accessor functions to be used:
B.1) These functions should be removed once SRF is replaced/cleaned up:
int opp_id_to_freq(unsigned long *freq, const struct omap_opp *opps,
u8 opp_id);
int opp_freq_to_id(u8 *opp_id, const struct omap_opp *opps, unsigned
long freq);
agreed, could flag these as __deprecated to be clear they are going away,
or define them local to SRF.
Ack, was actually wondering if I can add __must_check instead to make it
noisy as hell and trouble all of us into fixing it eventually.
B.2) initialization functions:
/* Register the OPP list for the silicon */
int opp_register(struct omap_opp *opp_list,
const struct omap_opp_def *opp_reg);
ok, I was thinking opp_add(), but had the same idea.
Rather than having another omap_opp_def, could also add with just a
'u32 freq' and 'u32 voltage'.
The idea was ability to register an initial table with certain opps
disabled. init code will be simpler that way.
/* Enable a specific frequency */
int opp_enable(struct omap_opp *opp_list,
unsigned long freq, bool enable);
I was thinking opp_enable() would take a 'struct omap_opp' pointer
rather than a frequency, but same idea.
as I point below, yep, that might actually be a improvement in
performance doing that. I think I can generate another rule to my
guiding principles: "initializers, validators and search basics should
use freq, rest should use omap_opp pointers".
Users:
opp_register will be called by pm34xx.c based on silicon type.
opp_enable will be called by board file/ has_feature based logic
etc.. on a need basis
B.3) verification functions:
/* Check if we have a frequency in the list(enabled/disabled)
* this is needed by users who would like to enable a disabled
* frequency
*/
int opp_has_freq(struct omap_opp *opp_list, unsigned long freq);
/* Check if we have a enabled frequency - this is what most users
* will need
*/
int opp_is_freq_valid(struct omap_opp *opp_list, unsigned long freq);
Users:
files who'd want to use opp_enable,
validation paths.
B.4) Limit functions:
/* To get a number of frequencies enabled */
int opp_get_freq_count(const struct omap_opp *opp_list, u8 *num);
ok
/* Get highest enabled frequency */
int opp_get_highest_freq(struct omap_opp *opp_list,
unsigned long *freq);
/* Get lowest enabled frequency */
int opp_get_lowest_freq(struct omap_opp *opp_list, unsigned long *freq);
Users:
Obvious user is the current cpufreq
I assume you mean the CPUfreq table init. For that, I'd prefer to use
a generic iterator for walking the list to initialize CPUfreq.
you will still need a start point as cpufreq does not even know which
processor it is running on and whether some freq is disabled or not.
B.5) Search functions -> these will check only enabled
frequencies. This will not check if the start freq provided in *freq
is an enabled frequency or not. Assumption I am making is: if you are
searching for a frequency that is disabled in the opp list, you are
not trying something generic -hence wont support.
/* Get higher enabled frequency than the provided one */
int opp_get_higher_freq(struct omap_opp *opp_list, unsigned long *freq);
/* Get lower enabled frequency than the provided one */
int opp_get_lower_freq(struct omap_opp *opp_list, unsigned long *freq);
What about finding an exact match?
My proposal is to have a single 'find by freq' function with rounding
options, but whether there is one function w/options or 3 dedicated
functions, it doesn't matter to me.
opp_is_freq_valid already does the exact match, so a generic function is
redundant and makes code a mite difficult to read + allowing scope for
misuse in terms of extending the definitions. we can do it if there is a
need for the same in the future.
Users:
current srf, future scale logic
B.6) voltage functions - opp layer will not trigger voltage
transition, it is upto other users of opp layer to make a decision.
/* get voltage corresponding to a frequency */
int opp_freq_to_voltage(u16 *voltage, struct omap_opp *opp_list,
unsigned long *freq);
Users:
current SR/SRF, future voltage framework.
Agree on the need for this, but had a slightly different design in
mind. I think this should be done by something like:
opp = opp_find_by_freq(opps, <freq>)
voltage = opp_get_voltage(opp);
yes, this is one way to do it, this assumes that opp is the handle to
specific opp -> might actually be a better approach than mine in terms
of sticking with the rule that freq is only used, causing a research
overheads every time.
I dont however like the idea of function returning value -> if the opp
functions were always to return error/success, it is easier to review
and maintain further developments on it.
Anyways, below is the API header that I worked on last week after
looking at your initial proposal. It's still rough around the edges,
but shows that your proposal above and mine are heading in basically
the same direction.
Kevin
/*
* OMAP OPP interface
*
* Author: Kevin Hilman, Deep Root Systems, LLC
*
* 2009 (c) Deep Root Systems, LLC. This file is licensed under
* the terms of the GNU General Public License version 2. This program
* is licensed "as is" without any warranty of any kind, whether express
* or implied.
*/
/**
* struct omap_opp - clock frequency-to-OPP ID table for DSP, MPU, L3
* @enabled: enabled if true, disabled if false
* @rate: target clock rate
* @min_vdd: minimum VDD1 voltage (in millivolts) for this OPP
*
* Operating performance point data. Can vary by OMAP chip and board.
*/
struct omap_opp {
bool enabled;
u32 freq;
u32 voltage;
};
#define OMAP_OPP_ROUND_NONE 0x0
#define OMAP_OPP_ROUND_UP BIT(1)
#define OMAP_OPP_ROUND_DOWN BIT(2)
/**
* opp_find_by_freq()
* @opps: Pointer to OPP table
* @freq: Frequency to look for in OPP table
* @rounding: Rounding option: NONE, UP, DOWN
*
* Look for an OPP with a frequency value matching @freq. If
* @rounding != ROUND_NONE, find closest match using rounding.
*
* Can be used to find exact OPP, or closest match using given rounding.
*
* Returns pointer to OPP entry if match is found, or NULL if no match
* found (only possible when no rounding is used)
*/
struct omap_opp *opp_find_by_freq(struct omap_opp *opps,
u32 freq, u32 rounding);
/**
* opp_add()
* @opp: pointer to list of OPPs to be added to
* @freq: frequency value of new OPP
* @voltage: voltage value of new OPP
*
* Add new OPP to list. New OPP is enabled by default
*
* Returns pointer to new OPP, or NULL upon failure.
*
* Implementation (option): list_add()
*/
struct omap_opp *opp_add(struct omap_opp *opps, u32 freq, u32 voltage);
/**
* opp_get_freq()
* @opp: pointer to OPP
*
* Returns frequency of given OPP, or -EINVAL if OPP not valid
*
* Implementation: return opp->freq
*/
int opp_get_freq(struct omap_opp *opp);
/**
* opp_get_voltage()
* @opp: pointer to OPP
*
* Returns voltage of given OPP, or -EINVAL if OPP not valid
*
* Implementation: return opp->freq
*/
int opp_get_voltage(struct omap_opp *opp);
/**
* opp_get_opp_count()
* @opps: Pointer to OPP table
*
* Returns number of (enabled) OPPs, or -EINVAL if invalid OPP list
*/
int opp_get_opp_count(struct omap_opp *opps);
/* Generalized iterator: list_for_each() */
int pwrdm_for_each(int (*fn)(struct omap_opp *opp, void *user),
void *user);
--
Regards,
Nishanth Menon
--
To unsubscribe from this list: send the line "unsubscribe linux-omap" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html