提交 a6ccdcd9 编写于 作者: H Haojian Zhuang 提交者: Samuel Ortiz

mfd: 88pm860x: Use REG resource for backlight

Now resource of 88pm860x backlight is changed from IORESOURCE_IO
to IORESOURCE_REG. In original driver, the resource is using
self-defined IORESOURCE_IO. So change the resource to register
offset to match the definition of IORESOURCE_REG.
Signed-off-by: NHaojian Zhuang <haojian.zhuang@gmail.com>
Signed-off-by: NSamuel Ortiz <sameo@linux.intel.com>
上级 015625a2
...@@ -21,10 +21,20 @@ ...@@ -21,10 +21,20 @@
#define INT_STATUS_NUM 3 #define INT_STATUS_NUM 3
static struct resource bk_resources[] __devinitdata = { static struct resource bk0_resources[] __devinitdata = {
{PM8606_BACKLIGHT1, PM8606_BACKLIGHT1, "backlight-0", IORESOURCE_REG,}, {2, 2, "duty cycle", IORESOURCE_REG, },
{PM8606_BACKLIGHT2, PM8606_BACKLIGHT2, "backlight-1", IORESOURCE_REG,}, {3, 3, "always on", IORESOURCE_REG, },
{PM8606_BACKLIGHT3, PM8606_BACKLIGHT3, "backlight-2", IORESOURCE_REG,}, {3, 3, "current", IORESOURCE_REG, },
};
static struct resource bk1_resources[] __devinitdata = {
{4, 4, "duty cycle", IORESOURCE_REG, },
{5, 5, "always on", IORESOURCE_REG, },
{5, 5, "current", IORESOURCE_REG, },
};
static struct resource bk2_resources[] __devinitdata = {
{6, 6, "duty cycle", IORESOURCE_REG, },
{7, 7, "always on", IORESOURCE_REG, },
{5, 5, "current", IORESOURCE_REG, },
}; };
static struct resource led_resources[] __devinitdata = { static struct resource led_resources[] __devinitdata = {
...@@ -99,9 +109,22 @@ static struct resource rtc_resources[] __devinitdata = { ...@@ -99,9 +109,22 @@ static struct resource rtc_resources[] __devinitdata = {
}; };
static struct mfd_cell bk_devs[] = { static struct mfd_cell bk_devs[] = {
{"88pm860x-backlight", 0,}, {
{"88pm860x-backlight", 1,}, .name = "88pm860x-backlight",
{"88pm860x-backlight", 2,}, .id = 0,
.num_resources = ARRAY_SIZE(bk0_resources),
.resources = bk0_resources,
}, {
.name = "88pm860x-backlight",
.id = 1,
.num_resources = ARRAY_SIZE(bk1_resources),
.resources = bk1_resources,
}, {
.name = "88pm860x-backlight",
.id = 2,
.num_resources = ARRAY_SIZE(bk2_resources),
.resources = bk2_resources,
},
}; };
static struct mfd_cell led_devs[] = { static struct mfd_cell led_devs[] = {
...@@ -615,36 +638,21 @@ static void __devinit device_osc_init(struct i2c_client *i2c) ...@@ -615,36 +638,21 @@ static void __devinit device_osc_init(struct i2c_client *i2c)
static void __devinit device_bk_init(struct pm860x_chip *chip, static void __devinit device_bk_init(struct pm860x_chip *chip,
struct pm860x_platform_data *pdata) struct pm860x_platform_data *pdata)
{ {
int ret; int ret, i;
int i, j, id;
if (pdata && pdata->backlight) {
if ((pdata == NULL) || (pdata->backlight == NULL)) if (pdata->num_backlights > ARRAY_SIZE(bk_devs))
return; pdata->num_backlights = ARRAY_SIZE(bk_devs);
for (i = 0; i < pdata->num_backlights; i++) {
if (pdata->num_backlights > ARRAY_SIZE(bk_devs)) bk_devs[i].platform_data = &pdata->backlight[i];
pdata->num_backlights = ARRAY_SIZE(bk_devs); bk_devs[i].pdata_size =
sizeof(struct pm860x_backlight_pdata);
for (i = 0; i < pdata->num_backlights; i++) {
bk_devs[i].platform_data = &pdata->backlight[i];
bk_devs[i].pdata_size = sizeof(struct pm860x_backlight_pdata);
for (j = 0; j < ARRAY_SIZE(bk_devs); j++) {
id = bk_resources[j].start;
if (pdata->backlight[i].flags != id)
continue;
bk_devs[i].num_resources = 1;
bk_devs[i].resources = &bk_resources[j];
ret = mfd_add_devices(chip->dev, 0,
&bk_devs[i], 1,
&bk_resources[j], 0);
if (ret < 0) {
dev_err(chip->dev, "Failed to add "
"backlight subdev\n");
return;
}
} }
} }
ret = mfd_add_devices(chip->dev, 0, bk_devs,
ARRAY_SIZE(bk_devs), NULL, 0);
if (ret < 0)
dev_err(chip->dev, "Failed to add backlight subdev\n");
} }
static void __devinit device_led_init(struct pm860x_chip *chip, static void __devinit device_led_init(struct pm860x_chip *chip,
......
...@@ -31,57 +31,26 @@ struct pm860x_backlight_data { ...@@ -31,57 +31,26 @@ struct pm860x_backlight_data {
int port; int port;
int pwm; int pwm;
int iset; int iset;
int reg_duty_cycle;
int reg_always_on;
int reg_current;
}; };
static inline int wled_a(int port)
{
int ret;
ret = ((port - PM8606_BACKLIGHT1) << 1) + 2;
return ret;
}
static inline int wled_b(int port)
{
int ret;
ret = ((port - PM8606_BACKLIGHT1) << 1) + 3;
return ret;
}
/* WLED2 & WLED3 share the same IDC */
static inline int wled_idc(int port)
{
int ret;
switch (port) {
case PM8606_BACKLIGHT1:
case PM8606_BACKLIGHT2:
ret = ((port - PM8606_BACKLIGHT1) << 1) + 3;
break;
case PM8606_BACKLIGHT3:
default:
ret = ((port - PM8606_BACKLIGHT2) << 1) + 3;
break;
}
return ret;
}
static int backlight_power_set(struct pm860x_chip *chip, int port, static int backlight_power_set(struct pm860x_chip *chip, int port,
int on) int on)
{ {
int ret = -EINVAL; int ret = -EINVAL;
switch (port) { switch (port) {
case PM8606_BACKLIGHT1: case 0:
ret = on ? pm8606_osc_enable(chip, WLED1_DUTY) : ret = on ? pm8606_osc_enable(chip, WLED1_DUTY) :
pm8606_osc_disable(chip, WLED1_DUTY); pm8606_osc_disable(chip, WLED1_DUTY);
break; break;
case PM8606_BACKLIGHT2: case 1:
ret = on ? pm8606_osc_enable(chip, WLED2_DUTY) : ret = on ? pm8606_osc_enable(chip, WLED2_DUTY) :
pm8606_osc_disable(chip, WLED2_DUTY); pm8606_osc_disable(chip, WLED2_DUTY);
break; break;
case PM8606_BACKLIGHT3: case 2:
ret = on ? pm8606_osc_enable(chip, WLED3_DUTY) : ret = on ? pm8606_osc_enable(chip, WLED3_DUTY) :
pm8606_osc_disable(chip, WLED3_DUTY); pm8606_osc_disable(chip, WLED3_DUTY);
break; break;
...@@ -104,13 +73,13 @@ static int pm860x_backlight_set(struct backlight_device *bl, int brightness) ...@@ -104,13 +73,13 @@ static int pm860x_backlight_set(struct backlight_device *bl, int brightness)
if (brightness) if (brightness)
backlight_power_set(chip, data->port, 1); backlight_power_set(chip, data->port, 1);
ret = pm860x_reg_write(data->i2c, wled_a(data->port), value); ret = pm860x_reg_write(data->i2c, data->reg_duty_cycle, value);
if (ret < 0) if (ret < 0)
goto out; goto out;
if ((data->current_brightness == 0) && brightness) { if ((data->current_brightness == 0) && brightness) {
if (data->iset) { if (data->iset) {
ret = pm860x_set_bits(data->i2c, wled_idc(data->port), ret = pm860x_set_bits(data->i2c, data->reg_current,
CURRENT_BITMASK, data->iset); CURRENT_BITMASK, data->iset);
if (ret < 0) if (ret < 0)
goto out; goto out;
...@@ -123,17 +92,17 @@ static int pm860x_backlight_set(struct backlight_device *bl, int brightness) ...@@ -123,17 +92,17 @@ static int pm860x_backlight_set(struct backlight_device *bl, int brightness)
} }
if (brightness == MAX_BRIGHTNESS) { if (brightness == MAX_BRIGHTNESS) {
/* set WLED_ON bit as 100% */ /* set WLED_ON bit as 100% */
ret = pm860x_set_bits(data->i2c, wled_b(data->port), ret = pm860x_set_bits(data->i2c, data->reg_always_on,
PM8606_WLED_ON, PM8606_WLED_ON); PM8606_WLED_ON, PM8606_WLED_ON);
} }
} else { } else {
if (brightness == MAX_BRIGHTNESS) { if (brightness == MAX_BRIGHTNESS) {
/* set WLED_ON bit as 100% */ /* set WLED_ON bit as 100% */
ret = pm860x_set_bits(data->i2c, wled_b(data->port), ret = pm860x_set_bits(data->i2c, data->reg_always_on,
PM8606_WLED_ON, PM8606_WLED_ON); PM8606_WLED_ON, PM8606_WLED_ON);
} else { } else {
/* clear WLED_ON bit since it's not 100% */ /* clear WLED_ON bit since it's not 100% */
ret = pm860x_set_bits(data->i2c, wled_b(data->port), ret = pm860x_set_bits(data->i2c, data->reg_always_on,
PM8606_WLED_ON, 0); PM8606_WLED_ON, 0);
} }
} }
...@@ -174,7 +143,7 @@ static int pm860x_backlight_get_brightness(struct backlight_device *bl) ...@@ -174,7 +143,7 @@ static int pm860x_backlight_get_brightness(struct backlight_device *bl)
struct pm860x_chip *chip = data->chip; struct pm860x_chip *chip = data->chip;
int ret; int ret;
ret = pm860x_reg_read(data->i2c, wled_a(data->port)); ret = pm860x_reg_read(data->i2c, data->reg_duty_cycle);
if (ret < 0) if (ret < 0)
goto out; goto out;
data->current_brightness = ret; data->current_brightness = ret;
...@@ -193,43 +162,50 @@ static const struct backlight_ops pm860x_backlight_ops = { ...@@ -193,43 +162,50 @@ static const struct backlight_ops pm860x_backlight_ops = {
static int pm860x_backlight_probe(struct platform_device *pdev) static int pm860x_backlight_probe(struct platform_device *pdev)
{ {
struct pm860x_chip *chip = dev_get_drvdata(pdev->dev.parent); struct pm860x_chip *chip = dev_get_drvdata(pdev->dev.parent);
struct pm860x_backlight_pdata *pdata = NULL; struct pm860x_backlight_pdata *pdata = pdev->dev.platform_data;
struct pm860x_backlight_data *data; struct pm860x_backlight_data *data;
struct backlight_device *bl; struct backlight_device *bl;
struct resource *res; struct resource *res;
struct backlight_properties props; struct backlight_properties props;
char name[MFD_NAME_SIZE]; char name[MFD_NAME_SIZE];
int ret; int ret = 0;
res = platform_get_resource(pdev, IORESOURCE_REG, 0);
if (res == NULL) {
dev_err(&pdev->dev, "No I/O resource!\n");
return -EINVAL;
}
pdata = pdev->dev.platform_data;
if (pdata == NULL) {
dev_err(&pdev->dev, "platform data isn't assigned to "
"backlight\n");
return -EINVAL;
}
data = devm_kzalloc(&pdev->dev, sizeof(struct pm860x_backlight_data), data = devm_kzalloc(&pdev->dev, sizeof(struct pm860x_backlight_data),
GFP_KERNEL); GFP_KERNEL);
if (data == NULL) if (data == NULL)
return -ENOMEM; return -ENOMEM;
strncpy(name, res->name, MFD_NAME_SIZE); res = platform_get_resource_byname(pdev, IORESOURCE_REG, "duty cycle");
if (!res) {
dev_err(&pdev->dev, "No REG resource for duty cycle\n");
ret = -ENXIO;
goto out;
}
data->reg_duty_cycle = res->start;
res = platform_get_resource_byname(pdev, IORESOURCE_REG, "always on");
if (!res) {
dev_err(&pdev->dev, "No REG resorce for always on\n");
ret = -ENXIO;
goto out;
}
data->reg_always_on = res->start;
res = platform_get_resource_byname(pdev, IORESOURCE_REG, "current");
if (!res) {
dev_err(&pdev->dev, "No REG resource for current\n");
ret = -ENXIO;
goto out;
}
data->reg_current = res->start;
memset(name, 0, MFD_NAME_SIZE);
sprintf(name, "backlight-%d", pdev->id);
data->port = pdev->id;
data->chip = chip; data->chip = chip;
data->i2c = (chip->id == CHIP_PM8606) ? chip->client \ data->i2c = (chip->id == CHIP_PM8606) ? chip->client \
: chip->companion; : chip->companion;
data->current_brightness = MAX_BRIGHTNESS; data->current_brightness = MAX_BRIGHTNESS;
data->pwm = pdata->pwm; if (pdata) {
data->iset = pdata->iset; data->pwm = pdata->pwm;
data->port = pdata->flags; data->iset = pdata->iset;
if (data->port < 0) {
dev_err(&pdev->dev, "wrong platform data is assigned");
kfree(data);
return -EINVAL;
} }
memset(&props, 0, sizeof(struct backlight_properties)); memset(&props, 0, sizeof(struct backlight_properties));
...@@ -248,12 +224,14 @@ static int pm860x_backlight_probe(struct platform_device *pdev) ...@@ -248,12 +224,14 @@ static int pm860x_backlight_probe(struct platform_device *pdev)
/* read current backlight */ /* read current backlight */
ret = pm860x_backlight_get_brightness(bl); ret = pm860x_backlight_get_brightness(bl);
if (ret < 0) if (ret < 0)
goto out; goto out_brt;
backlight_update_status(bl); backlight_update_status(bl);
return 0; return 0;
out: out_brt:
backlight_device_unregister(bl); backlight_device_unregister(bl);
out:
devm_kfree(&pdev->dev, data);
return ret; return ret;
} }
......
...@@ -34,12 +34,6 @@ enum { ...@@ -34,12 +34,6 @@ enum {
PM8606_ID_MAX, PM8606_ID_MAX,
}; };
enum {
PM8606_BACKLIGHT1 = 0,
PM8606_BACKLIGHT2,
PM8606_BACKLIGHT3,
};
enum { enum {
PM8606_LED1_RED = 0, PM8606_LED1_RED = 0,
PM8606_LED1_GREEN, PM8606_LED1_GREEN,
...@@ -340,10 +334,8 @@ enum { ...@@ -340,10 +334,8 @@ enum {
}; };
struct pm860x_backlight_pdata { struct pm860x_backlight_pdata {
int id;
int pwm; int pwm;
int iset; int iset;
unsigned long flags;
}; };
struct pm860x_led_pdata { struct pm860x_led_pdata {
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册