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: Haojian Zhuang <haojian.zhuang@gmail.com> Signed-off-by: Samuel Ortiz <sameo@linux.intel.com>
This commit is contained in:
committed by
Samuel Ortiz
parent
015625a20f
commit
a6ccdcd98c
@@ -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 = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
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)
|
|||||||
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 == NULL) || (pdata->backlight == NULL))
|
if (pdata && pdata->backlight) {
|
||||||
return;
|
if (pdata->num_backlights > ARRAY_SIZE(bk_devs))
|
||||||
|
pdata->num_backlights = ARRAY_SIZE(bk_devs);
|
||||||
if (pdata->num_backlights > ARRAY_SIZE(bk_devs))
|
for (i = 0; i < pdata->num_backlights; i++) {
|
||||||
pdata->num_backlights = ARRAY_SIZE(bk_devs);
|
bk_devs[i].platform_data = &pdata->backlight[i];
|
||||||
|
bk_devs[i].pdata_size =
|
||||||
for (i = 0; i < pdata->num_backlights; i++) {
|
sizeof(struct pm860x_backlight_pdata);
|
||||||
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 {
|
|||||||
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)
|
|||||||
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)
|
|||||||
}
|
}
|
||||||
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)
|
|||||||
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 = {
|
|||||||
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)
|
|||||||
/* 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 {
|
|||||||
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 {
|
|||||||
};
|
};
|
||||||
|
|
||||||
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 {
|
||||||
|
Reference in New Issue
Block a user