| /* | 
 |  * OMAP SoC specific OPP wrapper function | 
 |  * | 
 |  * Copyright (C) 2009-2010 Texas Instruments Incorporated - http://www.ti.com/ | 
 |  *	Nishanth Menon | 
 |  *	Kevin Hilman | 
 |  * Copyright (C) 2010 Nokia Corporation. | 
 |  *      Eduardo Valentin | 
 |  * | 
 |  * This program is free software; you can redistribute it and/or modify | 
 |  * it under the terms of the GNU General Public License version 2 as | 
 |  * published by the Free Software Foundation. | 
 |  * | 
 |  * This program is distributed "as is" WITHOUT ANY WARRANTY of any | 
 |  * kind, whether express or implied; without even the implied warranty | 
 |  * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
 |  * GNU General Public License for more details. | 
 |  */ | 
 | #include <linux/module.h> | 
 | #include <linux/opp.h> | 
 |  | 
 | #include <plat/omap_device.h> | 
 |  | 
 | #include "omap_opp_data.h" | 
 |  | 
 | /* Temp variable to allow multiple calls */ | 
 | static u8 __initdata omap_table_init; | 
 |  | 
 | /** | 
 |  * omap_init_opp_table() - Initialize opp table as per the CPU type | 
 |  * @opp_def:		opp default list for this silicon | 
 |  * @opp_def_size:	number of opp entries for this silicon | 
 |  * | 
 |  * Register the initial OPP table with the OPP library based on the CPU | 
 |  * type. This is meant to be used only by SoC specific registration. | 
 |  */ | 
 | int __init omap_init_opp_table(struct omap_opp_def *opp_def, | 
 | 		u32 opp_def_size) | 
 | { | 
 | 	int i, r; | 
 |  | 
 | 	if (!opp_def || !opp_def_size) { | 
 | 		pr_err("%s: invalid params!\n", __func__); | 
 | 		return -EINVAL; | 
 | 	} | 
 |  | 
 | 	/* | 
 | 	 * Initialize only if not already initialized even if the previous | 
 | 	 * call failed, because, no reason we'd succeed again. | 
 | 	 */ | 
 | 	if (omap_table_init) | 
 | 		return -EEXIST; | 
 | 	omap_table_init = 1; | 
 |  | 
 | 	/* Lets now register with OPP library */ | 
 | 	for (i = 0; i < opp_def_size; i++) { | 
 | 		struct omap_hwmod *oh; | 
 | 		struct device *dev; | 
 |  | 
 | 		if (!opp_def->hwmod_name) { | 
 | 			pr_err("%s: NULL name of omap_hwmod, failing [%d].\n", | 
 | 				__func__, i); | 
 | 			return -EINVAL; | 
 | 		} | 
 | 		oh = omap_hwmod_lookup(opp_def->hwmod_name); | 
 | 		if (!oh || !oh->od) { | 
 | 			pr_warn("%s: no hwmod or odev for %s, [%d] " | 
 | 				"cannot add OPPs.\n", __func__, | 
 | 				opp_def->hwmod_name, i); | 
 | 			return -EINVAL; | 
 | 		} | 
 | 		dev = &oh->od->pdev->dev; | 
 |  | 
 | 		r = opp_add(dev, opp_def->freq, opp_def->u_volt); | 
 | 		if (r) { | 
 | 			dev_err(dev, "%s: add OPP %ld failed for %s [%d] " | 
 | 				"result=%d\n", | 
 | 			       __func__, opp_def->freq, | 
 | 			       opp_def->hwmod_name, i, r); | 
 | 		} else { | 
 | 			if (!opp_def->default_available) | 
 | 				r = opp_disable(dev, opp_def->freq); | 
 | 			if (r) | 
 | 				dev_err(dev, "%s: disable %ld failed for %s " | 
 | 					"[%d] result=%d\n", | 
 | 					__func__, opp_def->freq, | 
 | 					opp_def->hwmod_name, i, r); | 
 | 		} | 
 | 		opp_def++; | 
 | 	} | 
 |  | 
 | 	return 0; | 
 | } |