| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446 | 
							- /*
 
-  * OMAP3/OMAP4 Voltage Management Routines
 
-  *
 
-  * Author: Thara Gopinath	<thara@ti.com>
 
-  *
 
-  * Copyright (C) 2007 Texas Instruments, Inc.
 
-  * Rajendra Nayak <rnayak@ti.com>
 
-  * Lesly A M <x0080970@ti.com>
 
-  *
 
-  * Copyright (C) 2008, 2011 Nokia Corporation
 
-  * Kalle Jokiniemi
 
-  * Paul Walmsley
 
-  *
 
-  * Copyright (C) 2010 Texas Instruments, Inc.
 
-  * Thara Gopinath <thara@ti.com>
 
-  *
 
-  * 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.
 
-  */
 
- #include <linux/delay.h>
 
- #include <linux/io.h>
 
- #include <linux/err.h>
 
- #include <linux/export.h>
 
- #include <linux/debugfs.h>
 
- #include <linux/slab.h>
 
- #include <linux/clk.h>
 
- #include "common.h"
 
- #include "prm-regbits-34xx.h"
 
- #include "prm-regbits-44xx.h"
 
- #include "prm44xx.h"
 
- #include "prcm44xx.h"
 
- #include "prminst44xx.h"
 
- #include "control.h"
 
- #include "voltage.h"
 
- #include "powerdomain.h"
 
- #include "vc.h"
 
- #include "vp.h"
 
- static LIST_HEAD(voltdm_list);
 
- /* Public functions */
 
- /**
 
-  * voltdm_get_voltage() - Gets the current non-auto-compensated voltage
 
-  * @voltdm:	pointer to the voltdm for which current voltage info is needed
 
-  *
 
-  * API to get the current non-auto-compensated voltage for a voltage domain.
 
-  * Returns 0 in case of error else returns the current voltage.
 
-  */
 
- unsigned long voltdm_get_voltage(struct voltagedomain *voltdm)
 
- {
 
- 	if (!voltdm || IS_ERR(voltdm)) {
 
- 		pr_warning("%s: VDD specified does not exist!\n", __func__);
 
- 		return 0;
 
- 	}
 
- 	return voltdm->nominal_volt;
 
- }
 
- /**
 
-  * voltdm_scale() - API to scale voltage of a particular voltage domain.
 
-  * @voltdm: pointer to the voltage domain which is to be scaled.
 
-  * @target_volt: The target voltage of the voltage domain
 
-  *
 
-  * This API should be called by the kernel to do the voltage scaling
 
-  * for a particular voltage domain during DVFS.
 
-  */
 
- int voltdm_scale(struct voltagedomain *voltdm,
 
- 		 unsigned long target_volt)
 
- {
 
- 	int ret, i;
 
- 	unsigned long volt = 0;
 
- 	if (!voltdm || IS_ERR(voltdm)) {
 
- 		pr_warning("%s: VDD specified does not exist!\n", __func__);
 
- 		return -EINVAL;
 
- 	}
 
- 	if (!voltdm->scale) {
 
- 		pr_err("%s: No voltage scale API registered for vdd_%s\n",
 
- 			__func__, voltdm->name);
 
- 		return -ENODATA;
 
- 	}
 
- 	/* Adjust voltage to the exact voltage from the OPP table */
 
- 	for (i = 0; voltdm->volt_data[i].volt_nominal != 0; i++) {
 
- 		if (voltdm->volt_data[i].volt_nominal >= target_volt) {
 
- 			volt = voltdm->volt_data[i].volt_nominal;
 
- 			break;
 
- 		}
 
- 	}
 
- 	if (!volt) {
 
- 		pr_warning("%s: not scaling. OPP voltage for %lu, not found.\n",
 
- 			   __func__, target_volt);
 
- 		return -EINVAL;
 
- 	}
 
- 	ret = voltdm->scale(voltdm, volt);
 
- 	if (!ret)
 
- 		voltdm->nominal_volt = volt;
 
- 	return ret;
 
- }
 
- /**
 
-  * voltdm_reset() - Resets the voltage of a particular voltage domain
 
-  *		    to that of the current OPP.
 
-  * @voltdm: pointer to the voltage domain whose voltage is to be reset.
 
-  *
 
-  * This API finds out the correct voltage the voltage domain is supposed
 
-  * to be at and resets the voltage to that level. Should be used especially
 
-  * while disabling any voltage compensation modules.
 
-  */
 
- void voltdm_reset(struct voltagedomain *voltdm)
 
- {
 
- 	unsigned long target_volt;
 
- 	if (!voltdm || IS_ERR(voltdm)) {
 
- 		pr_warning("%s: VDD specified does not exist!\n", __func__);
 
- 		return;
 
- 	}
 
- 	target_volt = voltdm_get_voltage(voltdm);
 
- 	if (!target_volt) {
 
- 		pr_err("%s: unable to find current voltage for vdd_%s\n",
 
- 			__func__, voltdm->name);
 
- 		return;
 
- 	}
 
- 	voltdm_scale(voltdm, target_volt);
 
- }
 
- /**
 
-  * omap_voltage_get_volttable() - API to get the voltage table associated with a
 
-  *				particular voltage domain.
 
-  * @voltdm:	pointer to the VDD for which the voltage table is required
 
-  * @volt_data:	the voltage table for the particular vdd which is to be
 
-  *		populated by this API
 
-  *
 
-  * This API populates the voltage table associated with a VDD into the
 
-  * passed parameter pointer. Returns the count of distinct voltages
 
-  * supported by this vdd.
 
-  *
 
-  */
 
- void omap_voltage_get_volttable(struct voltagedomain *voltdm,
 
- 				struct omap_volt_data **volt_data)
 
- {
 
- 	if (!voltdm || IS_ERR(voltdm)) {
 
- 		pr_warning("%s: VDD specified does not exist!\n", __func__);
 
- 		return;
 
- 	}
 
- 	*volt_data = voltdm->volt_data;
 
- }
 
- /**
 
-  * omap_voltage_get_voltdata() - API to get the voltage table entry for a
 
-  *				particular voltage
 
-  * @voltdm:	pointer to the VDD whose voltage table has to be searched
 
-  * @volt:	the voltage to be searched in the voltage table
 
-  *
 
-  * This API searches through the voltage table for the required voltage
 
-  * domain and tries to find a matching entry for the passed voltage volt.
 
-  * If a matching entry is found volt_data is populated with that entry.
 
-  * This API searches only through the non-compensated voltages int the
 
-  * voltage table.
 
-  * Returns pointer to the voltage table entry corresponding to volt on
 
-  * success. Returns -ENODATA if no voltage table exisits for the passed voltage
 
-  * domain or if there is no matching entry.
 
-  */
 
- struct omap_volt_data *omap_voltage_get_voltdata(struct voltagedomain *voltdm,
 
- 						 unsigned long volt)
 
- {
 
- 	int i;
 
- 	if (!voltdm || IS_ERR(voltdm)) {
 
- 		pr_warning("%s: VDD specified does not exist!\n", __func__);
 
- 		return ERR_PTR(-EINVAL);
 
- 	}
 
- 	if (!voltdm->volt_data) {
 
- 		pr_warning("%s: voltage table does not exist for vdd_%s\n",
 
- 			__func__, voltdm->name);
 
- 		return ERR_PTR(-ENODATA);
 
- 	}
 
- 	for (i = 0; voltdm->volt_data[i].volt_nominal != 0; i++) {
 
- 		if (voltdm->volt_data[i].volt_nominal == volt)
 
- 			return &voltdm->volt_data[i];
 
- 	}
 
- 	pr_notice("%s: Unable to match the current voltage with the voltage table for vdd_%s\n",
 
- 		  __func__, voltdm->name);
 
- 	return ERR_PTR(-ENODATA);
 
- }
 
- /**
 
-  * omap_voltage_register_pmic() - API to register PMIC specific data
 
-  * @voltdm:	pointer to the VDD for which the PMIC specific data is
 
-  *		to be registered
 
-  * @pmic:	the structure containing pmic info
 
-  *
 
-  * This API is to be called by the SOC/PMIC file to specify the
 
-  * pmic specific info as present in omap_voltdm_pmic structure.
 
-  */
 
- int omap_voltage_register_pmic(struct voltagedomain *voltdm,
 
- 			       struct omap_voltdm_pmic *pmic)
 
- {
 
- 	if (!voltdm || IS_ERR(voltdm)) {
 
- 		pr_warning("%s: VDD specified does not exist!\n", __func__);
 
- 		return -EINVAL;
 
- 	}
 
- 	voltdm->pmic = pmic;
 
- 	return 0;
 
- }
 
- /**
 
-  * omap_change_voltscale_method() - API to change the voltage scaling method.
 
-  * @voltdm:	pointer to the VDD whose voltage scaling method
 
-  *		has to be changed.
 
-  * @voltscale_method:	the method to be used for voltage scaling.
 
-  *
 
-  * This API can be used by the board files to change the method of voltage
 
-  * scaling between vpforceupdate and vcbypass. The parameter values are
 
-  * defined in voltage.h
 
-  */
 
- void omap_change_voltscale_method(struct voltagedomain *voltdm,
 
- 				  int voltscale_method)
 
- {
 
- 	if (!voltdm || IS_ERR(voltdm)) {
 
- 		pr_warning("%s: VDD specified does not exist!\n", __func__);
 
- 		return;
 
- 	}
 
- 	switch (voltscale_method) {
 
- 	case VOLTSCALE_VPFORCEUPDATE:
 
- 		voltdm->scale = omap_vp_forceupdate_scale;
 
- 		return;
 
- 	case VOLTSCALE_VCBYPASS:
 
- 		voltdm->scale = omap_vc_bypass_scale;
 
- 		return;
 
- 	default:
 
- 		pr_warn("%s: Trying to change the method of voltage scaling to an unsupported one!\n",
 
- 			__func__);
 
- 	}
 
- }
 
- /**
 
-  * omap_voltage_late_init() - Init the various voltage parameters
 
-  *
 
-  * This API is to be called in the later stages of the
 
-  * system boot to init the voltage controller and
 
-  * voltage processors.
 
-  */
 
- int __init omap_voltage_late_init(void)
 
- {
 
- 	struct voltagedomain *voltdm;
 
- 	if (list_empty(&voltdm_list)) {
 
- 		pr_err("%s: Voltage driver support not added\n",
 
- 			__func__);
 
- 		return -EINVAL;
 
- 	}
 
- 	list_for_each_entry(voltdm, &voltdm_list, node) {
 
- 		struct clk *sys_ck;
 
- 		if (!voltdm->scalable)
 
- 			continue;
 
- 		sys_ck = clk_get(NULL, voltdm->sys_clk.name);
 
- 		if (IS_ERR(sys_ck)) {
 
- 			pr_warning("%s: Could not get sys clk.\n", __func__);
 
- 			return -EINVAL;
 
- 		}
 
- 		voltdm->sys_clk.rate = clk_get_rate(sys_ck);
 
- 		WARN_ON(!voltdm->sys_clk.rate);
 
- 		clk_put(sys_ck);
 
- 		if (voltdm->vc) {
 
- 			voltdm->scale = omap_vc_bypass_scale;
 
- 			omap_vc_init_channel(voltdm);
 
- 		}
 
- 		if (voltdm->vp) {
 
- 			voltdm->scale = omap_vp_forceupdate_scale;
 
- 			omap_vp_init(voltdm);
 
- 		}
 
- 	}
 
- 	return 0;
 
- }
 
- static struct voltagedomain *_voltdm_lookup(const char *name)
 
- {
 
- 	struct voltagedomain *voltdm, *temp_voltdm;
 
- 	voltdm = NULL;
 
- 	list_for_each_entry(temp_voltdm, &voltdm_list, node) {
 
- 		if (!strcmp(name, temp_voltdm->name)) {
 
- 			voltdm = temp_voltdm;
 
- 			break;
 
- 		}
 
- 	}
 
- 	return voltdm;
 
- }
 
- /**
 
-  * voltdm_add_pwrdm - add a powerdomain to a voltagedomain
 
-  * @voltdm: struct voltagedomain * to add the powerdomain to
 
-  * @pwrdm: struct powerdomain * to associate with a voltagedomain
 
-  *
 
-  * Associate the powerdomain @pwrdm with a voltagedomain @voltdm.  This
 
-  * enables the use of voltdm_for_each_pwrdm().  Returns -EINVAL if
 
-  * presented with invalid pointers; -ENOMEM if memory could not be allocated;
 
-  * or 0 upon success.
 
-  */
 
- int voltdm_add_pwrdm(struct voltagedomain *voltdm, struct powerdomain *pwrdm)
 
- {
 
- 	if (!voltdm || !pwrdm)
 
- 		return -EINVAL;
 
- 	pr_debug("voltagedomain: %s: associating powerdomain %s\n",
 
- 		 voltdm->name, pwrdm->name);
 
- 	list_add(&pwrdm->voltdm_node, &voltdm->pwrdm_list);
 
- 	return 0;
 
- }
 
- /**
 
-  * voltdm_for_each_pwrdm - call function for each pwrdm in a voltdm
 
-  * @voltdm: struct voltagedomain * to iterate over
 
-  * @fn: callback function *
 
-  *
 
-  * Call the supplied function @fn for each powerdomain in the
 
-  * voltagedomain @voltdm.  Returns -EINVAL if presented with invalid
 
-  * pointers; or passes along the last return value of the callback
 
-  * function, which should be 0 for success or anything else to
 
-  * indicate failure.
 
-  */
 
- int voltdm_for_each_pwrdm(struct voltagedomain *voltdm,
 
- 			  int (*fn)(struct voltagedomain *voltdm,
 
- 				    struct powerdomain *pwrdm))
 
- {
 
- 	struct powerdomain *pwrdm;
 
- 	int ret = 0;
 
- 	if (!fn)
 
- 		return -EINVAL;
 
- 	list_for_each_entry(pwrdm, &voltdm->pwrdm_list, voltdm_node)
 
- 		ret = (*fn)(voltdm, pwrdm);
 
- 	return ret;
 
- }
 
- /**
 
-  * voltdm_for_each - call function on each registered voltagedomain
 
-  * @fn: callback function *
 
-  *
 
-  * Call the supplied function @fn for each registered voltagedomain.
 
-  * The callback function @fn can return anything but 0 to bail out
 
-  * early from the iterator.  Returns the last return value of the
 
-  * callback function, which should be 0 for success or anything else
 
-  * to indicate failure; or -EINVAL if the function pointer is null.
 
-  */
 
- int voltdm_for_each(int (*fn)(struct voltagedomain *voltdm, void *user),
 
- 		    void *user)
 
- {
 
- 	struct voltagedomain *temp_voltdm;
 
- 	int ret = 0;
 
- 	if (!fn)
 
- 		return -EINVAL;
 
- 	list_for_each_entry(temp_voltdm, &voltdm_list, node) {
 
- 		ret = (*fn)(temp_voltdm, user);
 
- 		if (ret)
 
- 			break;
 
- 	}
 
- 	return ret;
 
- }
 
- static int _voltdm_register(struct voltagedomain *voltdm)
 
- {
 
- 	if (!voltdm || !voltdm->name)
 
- 		return -EINVAL;
 
- 	INIT_LIST_HEAD(&voltdm->pwrdm_list);
 
- 	list_add(&voltdm->node, &voltdm_list);
 
- 	pr_debug("voltagedomain: registered %s\n", voltdm->name);
 
- 	return 0;
 
- }
 
- /**
 
-  * voltdm_lookup - look up a voltagedomain by name, return a pointer
 
-  * @name: name of voltagedomain
 
-  *
 
-  * Find a registered voltagedomain by its name @name.  Returns a pointer
 
-  * to the struct voltagedomain if found, or NULL otherwise.
 
-  */
 
- struct voltagedomain *voltdm_lookup(const char *name)
 
- {
 
- 	struct voltagedomain *voltdm ;
 
- 	if (!name)
 
- 		return NULL;
 
- 	voltdm = _voltdm_lookup(name);
 
- 	return voltdm;
 
- }
 
- /**
 
-  * voltdm_init - set up the voltagedomain layer
 
-  * @voltdm_list: array of struct voltagedomain pointers to register
 
-  *
 
-  * Loop through the array of voltagedomains @voltdm_list, registering all
 
-  * that are available on the current CPU. If voltdm_list is supplied
 
-  * and not null, all of the referenced voltagedomains will be
 
-  * registered.  No return value.
 
-  */
 
- void voltdm_init(struct voltagedomain **voltdms)
 
- {
 
- 	struct voltagedomain **v;
 
- 	if (voltdms) {
 
- 		for (v = voltdms; *v; v++)
 
- 			_voltdm_register(*v);
 
- 	}
 
- }
 
 
  |