virtual.c 8.9 KB
Newer Older
M
Mark Brown 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
/*
 * reg-virtual-consumer.c
 *
 * Copyright 2008 Wolfson Microelectronics PLC.
 *
 * Author: Mark Brown <broonie@opensource.wolfsonmicro.com>
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License as
 * published by the Free Software Foundation; either version 2 of the
 * License, or (at your option) any later version.
 */

#include <linux/err.h>
#include <linux/mutex.h>
#include <linux/platform_device.h>
#include <linux/regulator/consumer.h>
18
#include <linux/slab.h>
M
Mark Brown 已提交
19 20 21 22

struct virtual_consumer_data {
	struct mutex lock;
	struct regulator *regulator;
23
	bool enabled;
M
Mark Brown 已提交
24 25 26 27 28 29 30
	int min_uV;
	int max_uV;
	int min_uA;
	int max_uA;
	unsigned int mode;
};

31 32
static void update_voltage_constraints(struct device *dev,
				       struct virtual_consumer_data *data)
M
Mark Brown 已提交
33 34 35 36 37
{
	int ret;

	if (data->min_uV && data->max_uV
	    && data->min_uV <= data->max_uV) {
38 39
		dev_dbg(dev, "Requesting %d-%duV\n",
			data->min_uV, data->max_uV);
M
Mark Brown 已提交
40
		ret = regulator_set_voltage(data->regulator,
41
					data->min_uV, data->max_uV);
M
Mark Brown 已提交
42
		if (ret != 0) {
43 44
			dev_err(dev,
				"regulator_set_voltage() failed: %d\n", ret);
M
Mark Brown 已提交
45 46 47 48 49
			return;
		}
	}

	if (data->min_uV && data->max_uV && !data->enabled) {
50
		dev_dbg(dev, "Enabling regulator\n");
M
Mark Brown 已提交
51 52
		ret = regulator_enable(data->regulator);
		if (ret == 0)
53
			data->enabled = true;
M
Mark Brown 已提交
54
		else
55
			dev_err(dev, "regulator_enable() failed: %d\n",
M
Mark Brown 已提交
56 57 58 59
				ret);
	}

	if (!(data->min_uV && data->max_uV) && data->enabled) {
60
		dev_dbg(dev, "Disabling regulator\n");
M
Mark Brown 已提交
61 62
		ret = regulator_disable(data->regulator);
		if (ret == 0)
63
			data->enabled = false;
M
Mark Brown 已提交
64
		else
65
			dev_err(dev, "regulator_disable() failed: %d\n",
M
Mark Brown 已提交
66 67 68 69
				ret);
	}
}

70 71
static void update_current_limit_constraints(struct device *dev,
					  struct virtual_consumer_data *data)
M
Mark Brown 已提交
72 73 74 75 76
{
	int ret;

	if (data->max_uA
	    && data->min_uA <= data->max_uA) {
77 78
		dev_dbg(dev, "Requesting %d-%duA\n",
			data->min_uA, data->max_uA);
M
Mark Brown 已提交
79 80 81
		ret = regulator_set_current_limit(data->regulator,
					data->min_uA, data->max_uA);
		if (ret != 0) {
82 83 84
			dev_err(dev,
				"regulator_set_current_limit() failed: %d\n",
				ret);
M
Mark Brown 已提交
85 86 87 88 89
			return;
		}
	}

	if (data->max_uA && !data->enabled) {
90
		dev_dbg(dev, "Enabling regulator\n");
M
Mark Brown 已提交
91 92
		ret = regulator_enable(data->regulator);
		if (ret == 0)
93
			data->enabled = true;
M
Mark Brown 已提交
94
		else
95
			dev_err(dev, "regulator_enable() failed: %d\n",
M
Mark Brown 已提交
96 97 98 99
				ret);
	}

	if (!(data->min_uA && data->max_uA) && data->enabled) {
100
		dev_dbg(dev, "Disabling regulator\n");
M
Mark Brown 已提交
101 102
		ret = regulator_disable(data->regulator);
		if (ret == 0)
103
			data->enabled = false;
M
Mark Brown 已提交
104
		else
105
			dev_err(dev, "regulator_disable() failed: %d\n",
M
Mark Brown 已提交
106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128
				ret);
	}
}

static ssize_t show_min_uV(struct device *dev,
			   struct device_attribute *attr, char *buf)
{
	struct virtual_consumer_data *data = dev_get_drvdata(dev);
	return sprintf(buf, "%d\n", data->min_uV);
}

static ssize_t set_min_uV(struct device *dev, struct device_attribute *attr,
			  const char *buf, size_t count)
{
	struct virtual_consumer_data *data = dev_get_drvdata(dev);
	long val;

	if (strict_strtol(buf, 10, &val) != 0)
		return count;

	mutex_lock(&data->lock);

	data->min_uV = val;
129
	update_voltage_constraints(dev, data);
M
Mark Brown 已提交
130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154

	mutex_unlock(&data->lock);

	return count;
}

static ssize_t show_max_uV(struct device *dev,
			   struct device_attribute *attr, char *buf)
{
	struct virtual_consumer_data *data = dev_get_drvdata(dev);
	return sprintf(buf, "%d\n", data->max_uV);
}

static ssize_t set_max_uV(struct device *dev, struct device_attribute *attr,
			  const char *buf, size_t count)
{
	struct virtual_consumer_data *data = dev_get_drvdata(dev);
	long val;

	if (strict_strtol(buf, 10, &val) != 0)
		return count;

	mutex_lock(&data->lock);

	data->max_uV = val;
155
	update_voltage_constraints(dev, data);
M
Mark Brown 已提交
156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180

	mutex_unlock(&data->lock);

	return count;
}

static ssize_t show_min_uA(struct device *dev,
			   struct device_attribute *attr, char *buf)
{
	struct virtual_consumer_data *data = dev_get_drvdata(dev);
	return sprintf(buf, "%d\n", data->min_uA);
}

static ssize_t set_min_uA(struct device *dev, struct device_attribute *attr,
			  const char *buf, size_t count)
{
	struct virtual_consumer_data *data = dev_get_drvdata(dev);
	long val;

	if (strict_strtol(buf, 10, &val) != 0)
		return count;

	mutex_lock(&data->lock);

	data->min_uA = val;
181
	update_current_limit_constraints(dev, data);
M
Mark Brown 已提交
182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206

	mutex_unlock(&data->lock);

	return count;
}

static ssize_t show_max_uA(struct device *dev,
			   struct device_attribute *attr, char *buf)
{
	struct virtual_consumer_data *data = dev_get_drvdata(dev);
	return sprintf(buf, "%d\n", data->max_uA);
}

static ssize_t set_max_uA(struct device *dev, struct device_attribute *attr,
			  const char *buf, size_t count)
{
	struct virtual_consumer_data *data = dev_get_drvdata(dev);
	long val;

	if (strict_strtol(buf, 10, &val) != 0)
		return count;

	mutex_lock(&data->lock);

	data->max_uA = val;
207
	update_current_limit_constraints(dev, data);
M
Mark Brown 已提交
208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239

	mutex_unlock(&data->lock);

	return count;
}

static ssize_t show_mode(struct device *dev,
			 struct device_attribute *attr, char *buf)
{
	struct virtual_consumer_data *data = dev_get_drvdata(dev);

	switch (data->mode) {
	case REGULATOR_MODE_FAST:
		return sprintf(buf, "fast\n");
	case REGULATOR_MODE_NORMAL:
		return sprintf(buf, "normal\n");
	case REGULATOR_MODE_IDLE:
		return sprintf(buf, "idle\n");
	case REGULATOR_MODE_STANDBY:
		return sprintf(buf, "standby\n");
	default:
		return sprintf(buf, "unknown\n");
	}
}

static ssize_t set_mode(struct device *dev, struct device_attribute *attr,
			const char *buf, size_t count)
{
	struct virtual_consumer_data *data = dev_get_drvdata(dev);
	unsigned int mode;
	int ret;

240 241 242 243
	/*
	 * sysfs_streq() doesn't need the \n's, but we add them so the strings
	 * will be shared with show_mode(), above.
	 */
244
	if (sysfs_streq(buf, "fast\n"))
M
Mark Brown 已提交
245
		mode = REGULATOR_MODE_FAST;
246
	else if (sysfs_streq(buf, "normal\n"))
M
Mark Brown 已提交
247
		mode = REGULATOR_MODE_NORMAL;
248
	else if (sysfs_streq(buf, "idle\n"))
M
Mark Brown 已提交
249
		mode = REGULATOR_MODE_IDLE;
250
	else if (sysfs_streq(buf, "standby\n"))
M
Mark Brown 已提交
251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273
		mode = REGULATOR_MODE_STANDBY;
	else {
		dev_err(dev, "Configuring invalid mode\n");
		return count;
	}

	mutex_lock(&data->lock);
	ret = regulator_set_mode(data->regulator, mode);
	if (ret == 0)
		data->mode = mode;
	else
		dev_err(dev, "Failed to configure mode: %d\n", ret);
	mutex_unlock(&data->lock);

	return count;
}

static DEVICE_ATTR(min_microvolts, 0666, show_min_uV, set_min_uV);
static DEVICE_ATTR(max_microvolts, 0666, show_max_uV, set_max_uV);
static DEVICE_ATTR(min_microamps, 0666, show_min_uA, set_min_uA);
static DEVICE_ATTR(max_microamps, 0666, show_max_uA, set_max_uA);
static DEVICE_ATTR(mode, 0666, show_mode, set_mode);

274 275 276 277 278 279 280
static struct attribute *regulator_virtual_attributes[] = {
	&dev_attr_min_microvolts.attr,
	&dev_attr_max_microvolts.attr,
	&dev_attr_min_microamps.attr,
	&dev_attr_max_microamps.attr,
	&dev_attr_mode.attr,
	NULL
M
Mark Brown 已提交
281 282
};

283 284 285 286 287
static const struct attribute_group regulator_virtual_attr_group = {
	.attrs	= regulator_virtual_attributes,
};

static int __devinit regulator_virtual_probe(struct platform_device *pdev)
M
Mark Brown 已提交
288 289 290
{
	char *reg_id = pdev->dev.platform_data;
	struct virtual_consumer_data *drvdata;
291
	int ret;
M
Mark Brown 已提交
292 293

	drvdata = kzalloc(sizeof(struct virtual_consumer_data), GFP_KERNEL);
294
	if (drvdata == NULL)
295
		return -ENOMEM;
M
Mark Brown 已提交
296 297 298 299 300 301

	mutex_init(&drvdata->lock);

	drvdata->regulator = regulator_get(&pdev->dev, reg_id);
	if (IS_ERR(drvdata->regulator)) {
		ret = PTR_ERR(drvdata->regulator);
302 303
		dev_err(&pdev->dev, "Failed to obtain supply '%s': %d\n",
			reg_id, ret);
M
Mark Brown 已提交
304 305 306
		goto err;
	}

307 308 309 310 311 312
	ret = sysfs_create_group(&pdev->dev.kobj,
				 &regulator_virtual_attr_group);
	if (ret != 0) {
		dev_err(&pdev->dev,
			"Failed to create attribute group: %d\n", ret);
		goto err_regulator;
M
Mark Brown 已提交
313 314 315 316 317 318 319 320
	}

	drvdata->mode = regulator_get_mode(drvdata->regulator);

	platform_set_drvdata(pdev, drvdata);

	return 0;

321 322
err_regulator:
	regulator_put(drvdata->regulator);
M
Mark Brown 已提交
323 324 325 326 327
err:
	kfree(drvdata);
	return ret;
}

328
static int __devexit regulator_virtual_remove(struct platform_device *pdev)
M
Mark Brown 已提交
329 330 331
{
	struct virtual_consumer_data *drvdata = platform_get_drvdata(pdev);

332 333
	sysfs_remove_group(&pdev->dev.kobj, &regulator_virtual_attr_group);

M
Mark Brown 已提交
334 335 336 337 338 339
	if (drvdata->enabled)
		regulator_disable(drvdata->regulator);
	regulator_put(drvdata->regulator);

	kfree(drvdata);

340 341
	platform_set_drvdata(pdev, NULL);

M
Mark Brown 已提交
342 343 344 345
	return 0;
}

static struct platform_driver regulator_virtual_consumer_driver = {
346 347
	.probe		= regulator_virtual_probe,
	.remove		= __devexit_p(regulator_virtual_remove),
M
Mark Brown 已提交
348 349
	.driver		= {
		.name		= "reg-virt-consumer",
350
		.owner		= THIS_MODULE,
M
Mark Brown 已提交
351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368
	},
};

static int __init regulator_virtual_consumer_init(void)
{
	return platform_driver_register(&regulator_virtual_consumer_driver);
}
module_init(regulator_virtual_consumer_init);

static void __exit regulator_virtual_consumer_exit(void)
{
	platform_driver_unregister(&regulator_virtual_consumer_driver);
}
module_exit(regulator_virtual_consumer_exit);

MODULE_AUTHOR("Mark Brown <broonie@opensource.wolfsonmicro.com>");
MODULE_DESCRIPTION("Virtual regulator consumer");
MODULE_LICENSE("GPL");
369
MODULE_ALIAS("platform:reg-virt-consumer");